home *** CD-ROM | disk | FTP | other *** search
/ CP/M / CPM_CDROM.iso / jsage / znode3 / tcj / tcj34.wz / TCJ34.WS
Encoding:
Text File  |  1993-06-07  |  23.8 KB  |  506 lines

  1. For this issue I am going to live up to a long-standing tradition: onceì
  2. again I am not going to cover the material that I said I was.  Last time Iì
  3. presented half the new material on ARUNZ and said I would cover the secondì
  4. half this time.  Well, I am not going to.  ARUNZ is now up to version 'N'ì
  5. (it was 'J' last time).  Until it settles down a bit, it is probably futileì
  6. to try to describe it.  Besides that, I am getting a little bored with theì
  7. subject (though obviously not with the program), and perhaps you are, too.
  8.  
  9. Though living up to tradition, I am going to reverse a trend.  For some timeì
  10. now my columns have been getting longer and longer.  This time I really amì
  11. going to write a short one.  Besides the fact that because of me Art Carlsonì
  12. is apparently running low on printer's ink, I am just about written out,ì
  13. having just completed the manuals for NZ-COM and Z3PLUS.
  14.  
  15.  
  16.                              NZ-COM and Z3PLUS
  17.  
  18. Those manuals started out being simple affairs, but I just don't seem to beì
  19. able to get my obsession with thoroughness and completeness under control,ì
  20. and they soon turned into full-fledged books about the respective productì
  21. and Z-System in general.  I've been burning the midnight (actually, 2 am)ì
  22. oil for the past two or three months.  Each manual now runs about 80 pages! ì
  23. No wonder I don't have many words left in my system at this point.
  24.  
  25. Though somewhat reluctant to indulge in self-praise, I have to say that theì
  26. manuals are really quite good, and the products (NZ-COM and Z3PLUS) areì
  27. absolutely fantastic.  Joe Wright (NZ-COM), Bridger Mitchell (Z3PLUS), and Iì
  28. have had a very enjoyable and highly productive partnership in this effort. ì
  29. I sincerely urge you all to buy the automatic, universal, dynamic Z-Systemì
  30. appropriate for your computer: NZ-COM for CP/M-2.2 computers and Z3PLUS forì
  31. CP/M-Plus computers.  Both are $69.95 from Sage Microsystems East,ì
  32. Plu*Perfect Systems, or Alpha Systems (see ads in TCJ).
  33.  
  34. After the experience bringing these products to market, I will no longerì
  35. laugh so heartily when I hear stories about Borland or Lotus or Ashton-Tateì
  36. not delivering their products on the promised dates.  Hopefully you haveì
  37. lost your TCJ issue #32 and have forgotten that I wrote there, and I quote:ì
  38. "By the time you read this, they will definitely be available."  In issueì
  39. #33 I said, "the two new dynamic Z-Systems that will have been released byì
  40. the time you read this."  That almost made a double liar out of me. ì
  41. Luckily, issue #33 was delivered just enough behind schedule to let thatì
  42. statement squeak through -- barely.
  43.  
  44. With respect to NZ-COM, Joe Wright and I have agreed to publicly blame eachì
  45. other for the delay.  Actually, following common practice, we originallyì
  46. both agreed to blame Bridger Mitchell, though he, of course, had nothing toì
  47. do with the NZ-COM delay (Z3PLUS is another story).  However, now thatì
  48. Bridger has a TCJ column, too, we worried that such a slander might not goì
  49. unanswered.  Anyway, Joe can blame me for not getting the manual done onì
  50. time, and I can blame Joe for not writing the code to conform to myì
  51. description of it in the manual!  You can readily see that no one shouldì
  52. ever get involved in a product development all by himself.  Always make sureì
  53. there is someone else to blame.
  54.  
  55. The truth of the matter is that we really thought the coding was complete byì
  56. early April and that a simple manual could be knocked off in a weekì
  57. (naive!!).  In fact, as I alluded to above, the scope of the manual keptì
  58. expanding.  At the same time, as we attempted to describe the programs veryì
  59. precisely, we discovered a number of deficiencies in the code.  Some codingì
  60. limitations that we thought we would accept in the current version of NZ-COMì
  61. and would upgrade later just didn't seem acceptable any more once we wroteì
  62. them down on paper.  As a result, we have really skipped version 1 of NZ-COMì
  63. and have gone directly to version 2.  There are quite a few exciting newì
  64. features beyond what I described in issue #32; many will appeal especiallyì
  65. to those with a penchant for the unconventional (but I won't say any moreì
  66. about them now).
  67.  
  68.  
  69.                        PRL Files and Type-4 Programs
  70.  
  71. One of the new features introduced with ZCPR34 is the type-4 program.  Aì
  72. number of questions have been appearing in messages on Z-Nodes, so I thoughtì
  73. I would say a few words on this subject.
  74.  
  75. Just to refresh your memory, ordinary CP/M program files are loadedì
  76. beginning at an address of 100H.  This was also true of Z-System programs. ì
  77. They differ from standard CP/M programs in that the code starts with aì
  78. special header.  One item in the header is the text string 'Z3ENV'ì
  79. beginning with the fourth byte of the program.  This string is used toì
  80. identify the program as a Z-System program.
  81.  
  82. After the text string comes a number, now called the program type identifier. ì
  83. If the number is 2, then the so-called environment descriptor is includedì
  84. in the first page of the program file.  These type-2 programs are rarelyì
  85. seen today.  If the number is 1, then the program was designed to run in aì
  86. ZCPR3 system with a permanent operating system memory buffer containing theì
  87. environment descriptor.  The program only has to store the address of thatì
  88. descriptor, and it can then adapt to any system in which it is run.
  89.  
  90. The environment or ENV address is stored in the two bytes immediatelyì
  91. following the program type byte.  Prior to ZCPR version 3.3, the address hadì
  92. to be installed into programs using a utility like Z3INS before they couldì
  93. be used.  Starting with ZCPR33, the command processor installs the value asì
  94. part of the process of loading the file from disk.
  95.  
  96. With ZCPR33, the type-3 program was introduced.  These programs are notì
  97. limited to execution at 100H, as all previous programs had been.  The twoì
  98. bytes after the ENV address contain the address at which the code isì
  99. designed to run.  The Z33 command processor examines the program type byte,ì
  100. and if it is 3, it reads the load address from the header and proceeds toì
  101. load the program to the designated address and execute it there.
  102.  
  103. The type-3 program made it possible to load and run programs at addressesì
  104. other than 100H, but the address at which any given program file would runì
  105. was still fixed.  In his column in the last issue, Bridger Mitchellì
  106. described a fascinating and remarkable program structure that allows aì
  107. program to run at whatever address it is loaded to.  That same idea is theì
  108. basis for the new type-4 program.  Bridger's ANYWHERE program could beì
  109. loaded manually to any address and then executed.  Type-4 programs areì
  110. loaded automatically by the command processor to the highest address inì
  111. memory at which they can run without overwriting the command processor orì
  112. any resident system extension (RSX) that is present.  I would like toì
  113. provide some additional details on how type-4 programs work and how they areì
  114. constructed.
  115.  
  116. As with ANYWHERE, type-4 programs are derived from so-called (and, asì
  117. Bridger pointed out, mis-named) page-relocatable or PRL files.  Bridgerì
  118. defined and described those files in his column in the last issue, butì
  119. another shot at it probably won't hurt.  I will approach the subjectì
  120. somewhat differently -- with a concrete example.  Consider the short andì
  121. simple program in Listing 1.  It is set up for a starting address of 100H. ì
  122. Fig. 1 shows the binary image of the sort one would see if the program wereì
  123. loaded with a debugger (e.g., DDT) and displayed.
  124.  
  125. If we change the argument of the ORG directive from 100H to 200H andì
  126. assemble again, we get the results shown in Listing 2 and Fig. 2.  Youì
  127. should examine those results and note the things that have stayed the sameì
  128. and the things that have changed.  Note in particular that only three bytesì
  129. in the code have actually changed.  One is the high order byte of theì
  130. address of the initial jump instruction.  The destination of that jump is inì
  131. the program, and, since the program has moved up by 100H, the jump addressì
  132. has increased by an identical amount.  The second change is in the data wordì
  133. containing the entry point address.  Obviously that address changes when theì
  134. program origin is changed.  The third change is in the value loaded into theì
  135. DE register pair.  It is the address of the message string, which isì
  136. likewise a part of the program.
  137.  
  138. Note that the argument of the jump to DOS has not changed.  It is anì
  139. absolute address outside the program.  Therefore, it does not change.
  140.  
  141. Now let's look at a PRL file for the same program.  I am not aware of anyì
  142. assemblers that can produce a PRL file directly.  The usual procedure is toì
  143. remove the ORG statement from the source code, assemble the program to a RELì
  144. (normal relocatable format), and then use a linker to generate the PRL fileì
  145. from the REL file.  Fig. 3 shows the binary image of a PRL file producedì
  146. using the SLR virtual-memory linker SLRNK+.  Unfortunately, inexpensiveì
  147. linkers, such as SLRNK and ZLINK, are not able to produce PRL files.  Laterì
  148. we will show you a method, though somewhat tedious, that allows you toì
  149. construct a reasonable approximation to a PRL file using an ordinaryì
  150. assembler (no linker at all).
  151.  
  152. SLRNK+ actually cannot produce a PRL directly using the source code asì
  153. listed.  The SLR manual discusses in a somewhat opaque way the technique forì
  154. generating a correct PRL file.  The problem is that the one-page nearlyì
  155. empty header at the beginning of the program is not generated.  Joe Wrightì
  156. invented the trick of linking in the file SLR.REL derived by assemblingì
  157. source code with the sole statement
  158.  
  159.     DS    256
  160.  
  161. This allocates one page of memory.  The PRL file is produced by the linkerì
  162. command
  163.  
  164.     SLRNK+ TEST/K,SLR,TEST,/E
  165.  
  166. The term "TEST/K" defines the output file, the term "SLR" allocates theì
  167. empty header, and the term "TEST" links in the actual program code.
  168.  
  169. You should notice in Fig. 3 the following things.  First, the PRL fileì
  170. begins with a one-page header, which is entirely zero except for a word atì
  171. address 101H (you can't tell from this example that it is a word, but itì
  172. is).  This word is the length of the code, 001BH or 27 decimal in thisì
  173. example.  The program code itself begins on the next page (200H) and is theì
  174. same as the code in Fig. 1.
  175.  
  176. The other new bytes in the PRL file are those that follow the last byte ofì
  177. the program code.  These bytes comprise the relocation bitmap that Bridgerì
  178. Mitchell described in his column in the previous issue of TCJ.  The firstì
  179. byte is 20H, which expanded to binary is 00100000.  This means that theì
  180. third byte in the program code is the high byte of an address that must beì
  181. relocated to make the program code execute at an address other than 100H. ì
  182. Indeed, the third byte is the address to which the JP instruction jumps. ì
  183. The second byte in the bitmap is 08H or 00001000 binary.  This tells us thatì
  184. the 13th byte in the program code is an address that has to be relocatedì
  185. when the program is relocated.  Indeed, this is the address of the start ofì
  186. the program in the Z-header.  The third byte in the bitmap is 01H orì
  187. 00000001 binary.  It tells us that byte 23 is an address.  If we lookì
  188. carefully, this is the address part of the "LD DE,MSG" instruction.
  189.  
  190.  
  191. How Does ZCPR34 Load and Execute a Type-4
  192.  
  193. Bridger Mitchell explained last time in some detail how a PRL file can beì
  194. relocated to run at any address.  It really is not necessary to understandì
  195. all the details.  The basic idea is that the bitmap tells a loader whichì
  196. bytes in the code to adjust.
  197.  
  198. The Z34 command processor has a special mechanism for processing type-4ì
  199. programs.  After the command processor has located a transient program, itì
  200. loads the first record of the file into the default buffer at 80h.  Here itì
  201. can examine it to see what kind of program it is.  If it is a standard CP/Mì
  202. program or a type-1 or type-2 Z-System program, it sets up the load addressì
  203. as 100H and proceeds to load the entire file into memory, starting over withì
  204. the first record.  If it is a type-3 program, the same procedure is followedì
  205. except that the load address is extracted from the program header.
  206.  
  207. With the type-4 program things are not so simple, because the load addressì
  208. has to be calculated and the code has to be relocated.  Z34 gets these tasksì
  209. accomplished in a very clever and tricky way.  It could have done all theì
  210. work itself, but that would have added a lot of code to the commandì
  211. processor.  Instead, we took advantage of the fact that a PRL file has thatì
  212. two-record header with almost nothing in it.  To make a type-4 program, weì
  213. overlay onto this header a special loader program.  Z34 executes the codeì
  214. there to calculate the load address and then to perform the code relocation.
  215.  
  216. The loader is available in HEX format (TYP4LDR.HEX) and can be grafted ontoì
  217. the PRL file using the command
  218.  
  219.     MLOAD file=file.PRL,TYP4LDR
  220.  
  221. where 'file' is the name of the PRL file that you want to convert to a type-4 executable file.
  222.  
  223. By putting the loader code in the program rather than in the commandì
  224. processor, we provide additional flexibility.  TYP4LDR calculates theì
  225. highest address in the TPA to which a program can be loaded, but otherì
  226. loaders could return the address of the RCP or FCP and make possible self-installing modules.  Clever users will undoubtedly come up with some otherì
  227. interesting applications that use special header code.
  228.  
  229.  
  230. How Do We Make a PRL File
  231.  
  232. The easiest way to make a PRL file and from that a type-4 program is with aì
  233. capable linker like LINK from Digital Research or SLRNK+ from SLR Systems. ì
  234. LINK came with my CP/M-Plus computer; I do not know how much it costs or howì
  235. to obtain it otherwise.  SLRNK+, which offers many very useful and powerfulì
  236. features besides the ability to make PRL files, costs $195.  For someone whoì
  237. wants to experiment casually with type-4 programming, this is probably tooì
  238. much money to spend.  If you are not going to do it very often and don'tì
  239. mind a little work, you can hand craft a PRL file using a debugger like DDT. ì
  240. I will take you through the procedure using our sample program from Listingì
  241. 1.
  242.  
  243. Making the bitmap is the hard part of the procedure.  You should key in theì
  244. program called MAKEPRL.Z80 in Listing 3 and assemble it to a HEX file.  Weì
  245. will use that code in the debugger first to make a "byte-map" and then toì
  246. convert the byte-map into a bitmap.  We assume that we have alreadyì
  247. assembled versions of the program with ORGs of 100H and 200H.
  248.  
  249. To construct the PRL file, we invoke the debugger (assumed to be DDT) andì
  250. issue the commands shown in Fig. 4.  The first pair of commands loads theì
  251. utility program MAKEPRL.  The next two lines load the version of our programì
  252. that was assembled to run at 100H.  At this point we have to note the "nextì
  253. load address" reported by the debugger (I suggest you write it down).  Nowì
  254. we load the version of the program assembled to run at 200H so that itì
  255. follows right on the end of the 100H version.  To do this, we use an offsetì
  256. value in the "R" command that is 100H lower than the "next address" that wasì
  257. reported a moment ago.
  258.  
  259. There is one other very important step we need to perform at this point. ì
  260. MAKEPRL has to be told the address at which the second program image wasì
  261. loaded.  The value is patched in at address 10EH using the commands shown inì
  262. Fig. 4 starting with "S10E".  For our example program, the next address isì
  263. reported as 0280.  Therefore, low-nextaddr is 80 and the high-nextaddris 02.
  264.  
  265. Now we let MAKEPRL do the hard part by running it with the "G" command. ì
  266. When it is finished, we need to examine the value in the HL register, sinceì
  267. it tells us the next address after the bitmap.  After leaving DDT we have toì
  268. save the code image from 100H up to but not including that address.  For theì
  269. example program, the value in HL is reported to be 290H.  Since we areì
  270. presumably running Z34 and have the type-4 SAVE program, we save the resultì
  271. using the command
  272.  
  273.     SAVE 100-28F PRLTEST.COM
  274.  
  275. If you do not have the type-4 SAVE,you will have to calculate the number ofì
  276. sectors to save.
  277.  
  278. Fig. 4 lists one DDT command that we did not discuss.  The "F103,1FF,0"
  279. command fills the part of the header after the code size word with zeros.
  280. This makes the file look prettier, but it is not absolutely necessary,
  281. especially if you are later going to overlay the type-4 loader as described
  282. below.
  283.  
  284. The PRL files made this way can be used to make type-4 programs, and theyì
  285. can be used in Bridger Mitchell's ANYWHERE program.  However, we shouldì
  286. point out that these PRL files are not as efficient as those produced by aì
  287. linker.  We assumed that the code in the COM files extended to the end ofì
  288. the last record in the file.
  289.  
  290. Perhaps you can build on my simple method and figure out how to extend it toì
  291. produce an optimal PRL file just like the one from SLRNK+.  It would alsoì
  292. not be too difficult to write a program using routines in SYSLIB to read inì
  293. the pair of COM files and generate a PRL file from them completelyì
  294. automatically.  The most elegant method for doing this would use random-record writes.  I invite readers to send me such a program.
  295.  
  296. =============================================================================
  297.  
  298.     00 01 02 03    04 05 06 07    08 09 0A 0B    0C 0D 0E 0F
  299. -------------------------------------------------------------------
  300. 0100    C3 13 01 5A    33 45 4E 56    03 00 00 00    01 48 65 6C
  301. 0110    6C 6F 24 0E    09 11 0D 01    C3 05 00
  302.  
  303. Fig. 1.  Binary image of the sample program in Listing 1.
  304.  
  305. =============================================================================
  306.  
  307.     00 01 02 03    04 05 06 07    08 09 0A 0B    0C 0D 0E 0F
  308. -------------------------------------------------------------------
  309. 0100    C3 13 02 5A    33 45 4E 56    03 00 00 00    02 48 65 6C
  310. 0110    6C 6F 24 0E    09 11 0D 02    C3 05 00
  311.  
  312. Fig. 2.  Binary image of the sample program in Listing 1 when linked to runì
  313. at a starting address of 200H and loaded at 100H.
  314.  
  315. =============================================================================
  316.  
  317.     00 01 02 03    04 05 06 07    08 09 0A 0B    0C 0D 0E 0F
  318. -------------------------------------------------------------------
  319. 0100    00 1B 00 00    00 00 00 00    00 00 00 00    00 00 00 00
  320. 0110    00 00 00 00    00 00 00 00    00 00 00 00    00 00 00 00
  321.  
  322. 0200    C3 13 01 5A    33 45 4E 56    03 00 00 00    01 48 65 6C
  323. 0210    6C 6F 24 0E    09 11 0D 01    C3 05 00 20    08 01 00
  324.  
  325. Fig. 3.  Binary image of PRL file produced for the same test program andì
  326. loaded at address 100H.  Some memory regions containing bytes of 00 haveì
  327. been omitted from the display here.
  328.  
  329. =============================================================================
  330.  
  331.     IMAKEPRL.HEX<cr>    ; Ready to load PRL maker routine
  332.     R,<cr>            ; Load it to address 100h
  333.     ITEST100.COM<cr>    ; Ready to load program ORGed for 100h
  334.     R100<cr>        ; Load it to address 200h (offset 100)
  335.     ITEST200.COM<cr>    ; Ready to load program ORGed for 200h
  336.     R<nextaddr-100><cr>    ; Load it to proper offset
  337.     s10E<cr>        ; Ready to patch in code size
  338.     low-nextaddr<cr>    ; Low byte of code size
  339.     high-nextaddr<cr>    ; High byte of code size
  340.     .<cr>            ; End patch with period
  341.     G<cr>            ; Run MAKEPRL code at 100h
  342.     X<cr>            ; Display registers -- note value of HL
  343.     F103,1FF,0        ; Clean up the header area
  344.     G0<cr>            ; Exit from DDT
  345.  
  346. Figure 4.  Commands issued to DDT to produce a PRL file from two COM filesì
  347. assembled to run at addresses of 100h and 200h.
  348.  
  349. =============================================================================
  350.  
  351. Z80ASM SuperFast Relocating Macro Assembler     Z80ASM 1.31 Page   1
  352. PRLTEST Z80
  353.  
  354.     1         0100          org    100h
  355.     2                   
  356.     3 0100              entry:
  357.     4 0100  C3 0113         jp    start
  358.     5                   
  359.     6 0103  5A 33 45 4E     db    'Z3ENV'
  360.     7 0108  03              db    3
  361.     8 0109  0000            dw    0
  362.     9 010B  0100            dw    entry
  363.    10                   
  364.    11 010D  48 65 6C 6C msg:    db    'Hello','$'
  365.    12                   
  366.    13 0113              start:
  367.    14 0113  0E 09           ld    c,9
  368.    15 0115  11 010D         ld    de,msg
  369.    16 0118  C3 0005         jp    0005h
  370.    17                   
  371.    18                       end
  372.  0 Error(s) Detected.
  373.  27 Absolute Bytes. 3 Symbols Detected.
  374.  
  375. Listing 1.  Simple example program assembled for a load address of 100H.
  376.  
  377. =============================================================================
  378.  
  379. Z80ASM SuperFast Relocating Macro Assembler     Z80ASM 1.31 Page   1
  380. PRLTEST Z80
  381.  
  382.     1         0200          org    200h
  383.     2                   
  384.     3 0200              entry:
  385.     4 0200  C3 0213         jp    start
  386.     5                   
  387.     6 0203  5A 33 45 4E     db    'Z3ENV'
  388.     7 0208  03              db    3
  389.     8 0209  0000            dw    0
  390.     9 020B  0200            dw    entry
  391.    10                   
  392.    11 020D  48 65 6C 6C msg:    db    'Hello','$'
  393.    12                   
  394.    13 0213              start:
  395.    14 0213  0E 09           ld    c,9
  396.    15 0215  11 020D         ld    de,msg
  397.    16 0218  C3 0005         jp    0005h
  398.    17                   
  399.    18                       end
  400.  0 Error(s) Detected.
  401.  27 Absolute Bytes. 3 Symbols Detected.
  402.  
  403. Listing 2.  Simple example program assembled for a load address of 200H.
  404.  
  405. =============================================================================
  406.  
  407. ; This code, which assists in the generation of a PRL file from a pair of COM
  408. ; files assembled for execution at 100H and 200H, is by no means optimized for
  409. ; speed or size.  I have tried to optimize it for clarity!
  410.  
  411.     org    100h
  412.  
  413.     db    0        ; Standard PRL header (and NOP)
  414. size:    dw    0        ; PRL file size (filled in by code)
  415.  
  416.     jp    start
  417.  
  418.     db    'CODE200:'    ; Identification string
  419. c200:
  420.     dw    0        ; Patch to address of code linked to 200h
  421.  
  422. start:
  423.  
  424. ; The first step is to compute the size of the code and store the value at
  425. ; address 101h as required for a PRL file.  We also put this value in BC.
  426. ; We set up DE to point to the code assembled for 200H and HL to point to
  427. ; the code assembled for 100H.
  428.  
  429.     ld    hl,(c200)    ; Start of code for 200h
  430.     ld    de,200h        ; Start of code for 100h
  431.     xor    a
  432.     sbc    hl,de        ; Difference is assumed size of code
  433.     ld    (size),hl    ; Store in proper place for PRL file
  434.     ld    b,h        ; ..and in BC
  435.     ld    c,l
  436.  
  437.     ld    hl,(c200)
  438.     ex    de,hl        ; DE -> code for 200h, HL -> code for 100h
  439.  
  440. ; Now we subtract the code for 100h from the code for 200h to generate the map
  441. ; of bytes that are addresses that have to be relocated.  There will be a byte
  442. ; of 01 corresponding to each byte in the code that is the high order byte of
  443. ; an address that must be relocated.  There will be bytes of 00 everywhere
  444. ; else.
  445.  
  446. bytemap:
  447.     ld    a,(de)        ; Get byte from 200h version
  448.     sub    (hl)        ; Subtract byte from 100h version
  449.     ld    (de),a        ; Replace 200h code with byte map
  450.     inc    hl        ; Point to next bytes
  451.     inc    de
  452.     dec    bc        ; Any more to do?
  453.     ld    a,b
  454.     or    c
  455.     jr    nz,bytemap    ; Loop until done
  456.  
  457. ; Now we have to compress the byte map into a bit map, taking each 8 bytes of
  458. ; the byte map and packing the values into a single byte in the bit map.  The
  459. ; result is written immediately following the code (i.e., at the location of
  460. ; the code linked to 200h).
  461.  
  462.     ld    hl,(size)    ; Get number of bytes in byte map
  463.     ld    b,3        ; Divide by 8 (2 to the 3rd power)
  464. divide:
  465.     xor    a        ; Clear carry
  466.     rr    h        ; Rotate H right, low bit to carry
  467.     rr    l        ; Rotate L right, carry into high bit
  468.     djnz    divide        ; Repeat 3 times
  469.     ld    (mapsize),hl    ; Save the value
  470.  
  471.     ld    de,(c200)    ; Point to byte map
  472.     ld    hl,(c200)    ; Point to bit map (same place!)
  473.     
  474. makemap:
  475.     ld    b,8        ; Process block of 8 bytes into 1 byte
  476.     ld    c,0        ; Initial value for byte in bit map
  477.  
  478. makebyte:
  479.     ld    a,(de)        ; Get byte from byte map
  480.     inc    de        ; Advance pointer
  481.     rr    a        ; Move relocation bit into carry flag
  482.     rl    c        ; Move carry flag into byte in C
  483.     djnz    makebyte    ; Repeat 8 times
  484.  
  485.     ld    (hl),c        ; Put result into bit map
  486.     inc    hl        ; ..and advance its pointer
  487.  
  488.     push    hl
  489.     ld    hl,(mapsize)    ; See if we are done
  490.     dec    hl
  491.     ld    (mapsize),hl
  492.     ld    a,h
  493.     or    l
  494.     pop    hl
  495.     jr    nz,makemap
  496.  
  497.     rst    38h        ; Breakpoint to end program
  498.  
  499. mapsize:
  500.     ds    2        ; Scratch area
  501.  
  502.     end
  503.  
  504. Listing 3.  Utility program to perform the hard part of making a PRL file
  505. using a debugger.
  506.