|
Are you guys getting tired of OMI yet? (Don't let the spin-doctors in Rochester fool you--it's _Old_ MI, not Original MI :-) Do you yearn to see some NMI? Do you remember wistfully the GENOPT(*LIST) parm on CRTRPGPGM that we all used back in our youth to learn MI? Well, here's a little trick that will print out an NMI listing when you compile an ILE C module with CRTCMOD. GENOPT(*LIST) isn't actually gone--you just have to twist a few bits to make it work. Unfortunately, I haven't been able to get this to work with CRTRPGMOD--it gets a "pointer not set" error. YARTER (yet another reason to eschew RPG :-) I'm including a file containing the source for a simple C pgm and the two listings generated, for those of you who don't have ILE C. The listings don't look much like source code (too verbose). I think they're more in the nature of a dis-assembly from NMI object code. (In the following, I heartily invite Bruce or anyone else to jump in and correct anything I've got wrong :-) As far as I can tell, what happens when you compile a module is that the compilers generate _object_ W-code (in binary form, as opposed to human-readable source), which they pass to a pgm called QWXCRTMD, which translates to object NMI and calls the translator (which is in SLIC) to create the PowerPC machine code. This is in contrast to the OPM compilers. They generated _source_ MI and passed it to the MI "assembler", QPRROOTP. That's why we can have the QPRCRTPG API--it's just a front-end to the same sourcecode "assembler" that the compilers use. However, it appears that a sourcecode assembler for NMI doesn't exist on the system. To have one, we'd have to pay IBM the $1M or write it ourselves. Anyone want to dust off YACC or Bison? Some Caveats for what follows: 1. This method involves changing storage with SST. If you screw up you could cause "unpredictable results" machine-wide! The experts on this mailing list will be comfortable with this, having done similar stuff thousands of times. The beginners (those for whom Phil Hall's tutorials were intended) should probably content themselves with perusing the listings I've included :-) Caveat pgmr! 2. This is for V4R4--the offsets may be different for other releases. If you're still game, here's the method for getting the NMI listings: 1. strdbg qsys/qtexcrts updprod(*yes) 2. addbkp '/1' 3. Compile a C module (in the same job) with crtcmod, specifying dbgview(*all) 4. When the breakpoint occurs, use STRSST Display/Alter/Dump in another job to dump the process (job) that you're compiling in. Use "dump in hexadecimal (logical blocks)" 5. In the process dump listing, scan for QWXCRTMD. There'll be several ISF's for QWXCRTMD--go down to the one for ENTRY MAIN (Notice that the one just above it is for "TRANSLATE_WCODE" :-) 6. Get the ISF address (3 lines above the ENTRY MAIN line). E.g. "ISF ADDRESS DF70773E39 FF69C0" 7. Scan for the offset part of the ISF address (e.g. FF69C0). You'll find several hits. Continue to the one that's followed by the registers GPR 0 thru GPR 31. 8. Find GPR 24, e.g. "GPR 24 EDDB0C91DC 0051B0" 9. Add hex 1E8 to that address. E.g. EDDB0C91DC 0051B0 + 1E8 = EDDB0C91DC 005398 10. Use STRSST Display/Alter/Dump to display storage at that calculated starting address. 11. Notice that if the address ends in "8" as it does in this example, that address refers to the 3rd of the 4 columns of hex data. In this example, the data at that address is "DC093BD6 7F002248" (in the 3rd & 4th columns). 12. Go to that address by keying "p" over the first digit of data ("D" in this case) and pressing F10. SST will display the data at DC093BD67F 002248. Again, since the address ends in 8, the data at that address is in the 3rd column. 13. The data should be 00000000. Key a 1 in the last digit, e.g. 0000001. Press F11 to alter the data. The heading of the screen will change to "Alter Storage" and the bottom line will say "The data has been altered..." Press F11 again to change back to "Display Storage". 14. Press F9 to go back to the previous address. 15. Add 8 to the calculated address from step 9. Find the data at that address. In other words, the next 8 bytes after the data in step 11. In this case, it starts on the second line (at offset 53A0) and is "DC093BD6 7F007010" 16. Again, go to that address by keying "p" over the first digit (the "D") and pressing F10. In this example, the data at that address is "20AA4AF0" 17. Change the leftmost bit of the first byte to 1. E.g. change '20' to 'A0' (20=00100000, A0=10100000) Press F11 twice to alter the data and get back to "Display Storage" 18. Go back to the compile job and press Enter on the breakpoint to continue. If you get further breakpoints, just press Enter. 19. When the compile finishes, you should have the two extra listings, QCBNDLST & QCNMILST. 20. enddbg. BTW, when you are at the breakpoint in step 4, if you dump QTEMP by "dmpobj qtemp *lib" you will see 4 space objects named WBODY, WHEADER, WINIT & WSTRING. I'm guessing these contain the W-code. You can use SST to dump them by address. This method uses debug on the pgm QTEXCRTS (which is an OPM pgm) as a means of stopping the job after QWXCRTMD has been called but before it checks whether to print the NMI listing. Ideally, we would have put QWXCRTMD itself in debug, but I don't know how to add a breakpoint to an ILE pgm that doesn't have debug data. Does anybody know of a way? Of course, IBM must have provided a better way of switching on these listings, but I currently don't know what it is. However, I've found some clues: 1. The 00000000 we changed to 00000001 in step 13 seems to be a switch that is related to DST Debug mode, which is described in the Licensed Internal Code Diagnostic Aids (Vol 1) manual. 2. The CRTCMOD cmd has 3 hidden parms which you can see using RTVCMDSRC from http://www.news400.com/sharewarefiles/rtvcmd.zip They are defined as type(*zeroelem), which makes them pass empty lists to the CPP. They are called: SERVICEFE - Service front-end? SERVICEC3 - Service Cube-3? SERVICEWC - Service W-Code? But I haven't found what values the CPP is expecting from these parms, if they were defined as something other than *zeroelem :-(
As an Amazon Associate we earn from qualifying purchases.
This mailing list archive is Copyright 1997-2025 by midrange.com and David Gibbs as a compilation work. Use of the archive is restricted to research of a business or technical nature. Any other uses are prohibited. Full details are available on our policy page. If you have questions about this, please contact [javascript protected email address].
Operating expenses for this site are earned using the Amazon Associate program and Google Adsense.