Apple Assembly Line
Volume 2 -- Issue 10July 1982

In This Issue...

A New Software Tool: ES-CAPE

ES-CAPE stands for Extended S-C Applesoft Program Editor. You are somewhat familiar with it already as AED II from Linn Software. Bill has added more features, and I am in the process this month of re-doing the reference manual. I am shooting for it to be all packaged by the middle of July. The price will hold at $40 at least until September 1st.

If you are using Applesoft and feel the need for advanced editing tools to use on your programs-under-development, ES-CAPE ought to be in your tool-box. Like any tool, it doesn't do everything and it won't replace all your other tools. (You wouldn't try to tighten a screw with a hammer, or assemble a Heathkit with a SkilSaw....) But neither does it use all your money or memory!

Current Advertising Rates

For the August 1982 issue the price will be $60 for a full page, $35 for a half page. To be included, I must receive your camera-ready copy by July 20th.


Run-Anywhere Subroutine CallsBob Sander-Cederlof

Bob Nacon (author of Amper-Magic) called yesterday and told me about his new way to call subroutines in programs that will be loaded anywhere in memory without relocation or reassembly. He does this a lot inside Amper-Magic, and you might want to do it yourself sometime.

Instead of JSR <subroutine name>, put the following three lines whenever you call a subroutine:

     CLV
     JSR $FF58
     BVC <subroutine name>

The byte at $FF58 in the monitor ROM is always $60, an RTS instruction. Since this is used by most Apple interface boards, Apple has guaranteed that it will always be $60. The JSR to a guaranteed RTS instruction seems silly, doesn't it? Not quite, because it does put two bytes on the stack, and then pop them off again. But we can get them back later, inside the called subroutine, like this:

     TSX    GET STACK POINTER
     DEX
     DEX
     TXS    REVISED STACK POINTER

Now the subroutine we called has a return address to go to, just as though we had used JSR <subroutine name>! The only problem is that if we execute an RTS, we will re-execute the BVC <subroutine name> and be in a loop. Unless....

Unless we set overflow, so the BVC falls through. But there is no SEV opcode in the 6502, so what do we do? $FF58 to the rescue again! Here is how we end the subroutine:

     BIT $FF58    SET OVERFLOW
     RTS

The BIT instruction copies bit 7 of $FF58 into the Carry Status bit, and bit 6 into the Overflow Status bit. This, in other words, (since $FF58 has $60 in it) clears carry and sets overflow. If you want carry to be set as a return flag, you can insert SEC between the BIT and RTS lines.

I thank Bob Nacon for this technique, and he thanks Roger Wagner for putting him on the trail to its discovery. Roger writes the monthly column in Softalk Magazine called "Assembly Lines"; the December, 1981, issue covered writing run-anywhere programs. If you haven't got Roger's book yet, called "Assembly Lines: The Book", it is currently the best book for beginners that I know of. The regular price is $19.95+$2 shipping, but I sell them for $18+$2 shipping.


Still More About "The Other Epson Reference Manual"Bob Sander-Cederlof

No sooner did I print my cutting comments about Cut The Bull Software last month, than I received a copy of the new edition of "The Other Epson Manual" in the mail. Bill Parker, author and publisher, has done an excellent job. By now all of you who ordered the booklet should have received your copy.

Bill has now quit his previous job to devote full time to the software company. The nature of that previous job prevented him from publishing his telephone number. Now you can reach him at (714) 223-3576. He says that in the future should a back order situation develop he will hold customer checks until ready to ship.


Who are "we" and what are "we" doing?Mike Laumer

Some of you may wonder about the people whose articles you see in the AAL on a fairly regular basis and who you may have talked to on the phone at one time or another.

Bob Sander-Cederlof is the president of the S-C Software Corporation and the author of the S-C Assemblers and Double Precision Floating Point package. Bob has been working with computers since 1957, at such places as Control Data Corporation and Texas Instruments. He is developing a new text editor somewhat compatible with Apple Writer. Believe it or not the editor is half the size of Apple Writer. Both the editor and printer sections are in memory at once and it has more capabilities than Apple Writer. He also edits this newsletter every month, with the aid of Bill Morgan.

Bill Morgan is Bob's first full-time employee and helps in all areas: programming, shipping, accounting, phone sales, and writing articles for the AAL. He helps author the reference manuals as well, and tries to make our products fail before we start shipping them (so we can fix 'em before you see 'em!).

Bobby Deen is a part-time employee still in high school. He is currently helping Bob S-C develop a line of compatible Macro Cross Assemblers for 6800, 6809 and Z-80 processors to round out Bob's assembler product line. (The 6800 and 6809 versions are ready now.) He has helped develop an 18-digit decimal math package compatible with Applesoft soon to be a new product. He has also assisted in the CPR project with Mike Laumer.

Mike Laumer (that's me!) is owner of Laumer Research and author of FLASH! the Integer BASIC compiler, and of the upcoming MIKE'S MAGIC MATRIX hires graphics editor and animation design tool. As a sub-contractor to S-C Software for the last year, I have been working on an incredible application using Apples and video disks. You can read all about it in the June 1982 issue of BYTE magazine, pages 108-138. The American Heart Association sponsors the project, which will teach Cardiopulmonary Resuscitation (CPR). The Apple is supported by a video disk player, light pen, two CPR manikins, a random-access audio unit, and two monitors.

If you have called, you may have talked with Bob's daughter Patricia (oldest of five children). She is a Junior in High School, and works part-time at shipping, phone sales, mailing list maintenance, word processing, Visicalc-ing, program entry, paste-up and folding, and whatever comes up. She is assisted by Lisa MacCorkle, another high school friend.

We enjoy talking with all of you, so if you have a problem, need a book, or whatever, give us a call!


Giant Macro for Writing MessagesRobert B. Steiner

Every time I turn around I seem to need a quick and dirty routine to print out a message. I must have written them a dozen different ways, to fill various requirements. Sometimes they are only different because of a silly mistake...a difference usually called a bug. I could keep a handful of them on a subroutine library, but then I might get mixed up as to which one was which.

S-C Macro Assembler to the rescue! By writing one of largest macros I have ever seen, I can get all the message-printer-variants into one neat little package. Then by choosing the correct parameters, the kind of printing routine I want is generated on the spot.

I call the macro CRT, and you call it with up to five parameters. The call line will look like one of these:

     >CRT L,N,"your message"
     >CRT L,I,"your message"
     >CRT A,N,address of your message
     >CRT A,I,address of your message

The first parameter, which may be "L" or "A", indicates whether you will give an actual message in quotation marks, or the address of the message.

The second parameter, which may be "N" or "I", stands for Normal or Inverse video display.

The third parameter is either the message itself in quotes, or the address of the message (a label, of course).

An optional fourth parameter may be "I", "Y", or "R". "I" will generate code to read an immediate one byte reply, which is returned in the A-register. "Y" will generate the one byte reply code, followed by additional code to check for a yes/no response. It will loop until you type "Y" or "N"; then it will echo the letter, print a RETURN, and return with the character in the A-register.

If the fourth parameter is "R",an entire line of reply is expected. If there is no fifth parameter, the line will be at $200 for your program to analyze. If a fifth parameter is used, it is the name of a buffer for the reply message.

If I counted correctly, there are twenty different possible ways the macro can be generated!

Here is the macro definition, and some sample call lines. Try it out; you'll find it fun and educational, whether its useful to you or not. Then you can apply some of the techniques in your own work.

 1000  *--------------------------------
 1010  * MACRO: >CRT SRC,DSPMODE,MSG
 1020  *        >CRT SRC,DSPMODE,MSG,REPMODE
 1030  * MACRO: >CRT SRC,DSPMODE,MSG,REPMODE,REPADDR
 1040  *--------------------------------
 1050         .MA CRT
 1060         LDY #0       INITIALIZE INDEX
 1070         .DO ']1='L   *** LITERAL MESSAGE ***
 1080  :1     LDA :9,Y     GET MESSAGE CHARACTER
 1090         .ELSE        *** ADDRESSED MESSAGE ***
 1100  :1     LDA ]3,Y     GET MESSAGE CHARACTER
 1110         .FIN
 1120         PHA          SAVE CHARACTER ON STACK
 1130  *--------------------------------
 1140         .DO ']2='N   *** NORMAL DISPLAY ***
 1150         ORA #$80     SET TOP BIT OF CHARACTER
 1160         .ELSE        *** INVERSE DISPLAY ***
 1170         AND #$3F
 1180         .FIN
 1190  *--------------------------------
 1200         JSR $FDF0    DISPLAY CHARACTER
 1210         INY          POINT TO NEXT CHARACTER
 1220         PLA          GET ORIGINAL CHARACTER
 1230         BMI :1       MORE IF TOP BIT = 1
 1240  *--------------------------------
 1250         .DO ']1='L   *** LITERAL ***
 1260         BPL :2       ...ALWAYS
 1270  :9     .AT -']3'    MESSAGE ITSELF
 1280  :2
 1290         .FIN
 1300  *--------------------------------
 1310         .DO ]#=3     *** DISPLAY ONLY ***
 1320         LDA #$8D     CARRIAGE RETURN
 1330         JSR $FDF0
 1340         .ELSE
 1350         .DO ']4='R   *** STRING REPLY EXPECTED ***
 1360         LDA #$8D     CARRIAGE RETURN
 1370         JSR $FDF0
 1380         JSR $FD6F    READ REPLY
 1390         .DO ]#=5     *** SPECIFY REPLY LOCATION ***
 1400         LDY #0
 1410  :3     LDA $200,Y   MOVE REPLY TO CALLER'S BUFFER
 1420         STA ]5,Y
 1430         INY
 1440         CMP #$8D     WAS IT END OF LINE?
 1450         BNE :3       MORE TO MOVE
 1460         .FIN
 1470         .ELSE
 1480         LDA #$A0     ADD ONE BLANK TO MESSAGE
 1490         JSR $FDF0
 1500  :5     JSR $FD0C    GET REPLY CHARACTER
 1510         .DO ']4='Y   *** Y/N REPLY ***
 1520         CMP #'Y+$80
 1530         BEQ :6
 1540         CMP #'N+$80
 1550         BNE :5       NEITHER Y NOR N
 1560  :6
 1570         .FIN
 1580         PHA          SAVE REPLY
 1590         JSR $FDF0    DISPLAY THE CHARACTER
 1600         LDA #$8D
 1610         JSR $FDF0    CARRIAGE RETURN
 1620         PLA          GET REPLY CHARACTER
 1630         .DO ']4='Y   *** Y/N REPLY ***
 1640         CMP #'Y+$80  .EQ. IF "Y", .NE. IF "N"
 1650         .FIN
 1660         .FIN
 1670         .FIN
 1680         .EM
 1690  *--------------------------------
 1700         >CRT L,N,"ABCDEFG"
 1710         >CRT L,I,"ABCDEFG"
 1720         >CRT A,N,MSG
 1730         >CRT A,I,MSG
 1740  *--------------------------------
 1750         >CRT L,N,"ABCDEFG",Y
 1760         >CRT L,I,"ABCDEFG",I
 1770         >CRT A,N,MSG,R
 1780         >CRT A,I,MSG,R,BUFFER
 1790         RTS
 1800  MSG    .AT -/MESSAGE/
 1810  BUFFER .BS 256

Sorting Out Zero-Page ReferencesTracy L. Shafer

The search for page-zero references program in last month's AAL turned out to be (almost) the very thing I've been needing.

I have a clock card capable of generating NMI and IRQ interrupts. Up to now, I haven't been able to do any deep research on the IRQ due to the DOS and monitor conflict mentioned in the January issue of AAL. (They both use location $48.) I can't modify the monitor because I don't have access to a PROM burner, and the thought of searching through DOS really put a damper on the IRQ project until now.

Since I didn't need to know every page-zero reference used by DOS, I modified the program to search for a specific page-zero reference. That worked fine, but I didn't want to have to type in a separate search value for every group of references I might need later, so I further changed the program to print out all the references in numerical order of page-zero location.

To make the changes to the program as published last month, just remove the ".3" from line 1580 and add the following lines:

     1285 PAGE.REF .HS 00           VARIABLE TO HOLD THE
                                    CURRENT ZERO-PAGE
                                    LOCATION

     1571 .3        INY             NEW PLACE FOR ".3" LABEL
     1572           LDA (MON.PCL),Y GET PAGE REFERENCE
     1573           DEY             RESTORE VALUE OF Y
     1574           CMP PAGE.REF    ONE WE ARE SEARCHING FOR?
     1575           BNE .6          NO, IGNORE THIS ONE

     1861           LDX #1          RESTORE X-VALUE FOR
                                    MON.A1PC ABOVE
     1862           INC PAGE.REF    NEXT ZERO-PAGE ADDRESS
     1863           BNE CTRL.Y      NOT FINISHED

The program now searches through the memory range 256 times instead of just once, so it doesn't run nearly as fast, but it's easier to find all the references to specific locations.


AXLON RAMDISK 320: A ReviewBill Morgan

AXLON's RAMDISK 320 is a system designed to add 320K of memory to an Apple, configured to look to the Apple like two very fast disk drives. The speed improvement ranges from half the time for a large assembly to one-twelfth the time for directly dumping 192 pages of memory.

Hardware

The RAMDISK is a cabinet just the size of an Apple disk drive, containing the memory, its own power supply, and a backup battery. There is also a large interface card, which includes 2K of static RAM for the operating system.

The backup battery is said to provide up to three hours of protection against power outage. It did maintain power when we moved the system into another room (about 5 minutes), but you should certainly make floppy disk backups of the RAMDISK data before leaving the system unplugged overnight. As long as it is plugged into the wall, the battery is kept charged and the memory is maintained.

Software

There are several programs supplied with the RAMDISK. These fall into the general categories of system software, utilities, and demonstrations.

RAMDSK1 is the operating system, which is stored in static RAM on the interface card, addressed in the $C800-CFFF space. BRUNning this program hooks it into DOS and copies one or two mechanical drives into the RAMDISK.

RDCOPY copies between the mechanical and RAM disks, to load or back up the RAMDISK. SELECT creates modified versions of RAMDSK1 for different slot/drive configurations.

The EXTRA40K utility allows you to access "tracks" 36-40 on the RAMDISK, but only on a level comparable to using RWTS directly. That is, you must work in terms of addresses and track/sectors rather than variables and filenames. The manual has a complete assembler source listing of this program.

SECTOR CHECKER and BYTE-BY-BYTE are self-test utilities to verify correct operation of the RAMDISK.

The demonstrations are The Directory and the Mini-Base Phone Book. The Directory is a large, disk-based, data-base program, in machine language, which uses the speed of the RAMDISK to its full advantage. The problem with this program is that it is strictly fixed-format, with no provision for modifying the record structure. The fields built into a record are last name, first name, dept #, mail stop, phone, special interest 1, special interest 2, and comments. If you are a large company needing an on-line, internal phone directory, then The Directory is outstanding. Otherwise, it's just an interesting demonstration of the system's capabilities.

The Mini-Base Phone Book is a memory-based data base, somewhat similar to File Cabinet. The Mini-Base is also set up as an internal phone directory, but since it is written in Applesoft, it can be modified to suit your needs. The documentation includes instructions for changing the record structure. The manual also contains instructions for calling special machine-language routines for keyboard input, fast loading of text files (in a specified format), and fast sorting of a string array.

Documentation

The manual is in three sections: 63 pages on the system, 34 pages on The Directory program, and 43 pages on the Mini-Base Phone Book program. It all comes in a large (8 1/2 by 11) 3-ring binder. The system section has chapters on setting up the RAMDISK, using the included software, calling it from DOS 3.3, attaching and using it in Pascal, technical information, and accessing the system from assembly language.

The setup and software chapters are quite good; the DOS chapter just says that everything is standard. I don't have Pascal, so I can't evaluate that section. The technical and assembly language chapters have all the information about memory usage, addressing, and programming techniques needed to use the RAMDISK without all of DOS's overhead.

Using the RAMDISK

To use the RAMDISK with your programs, you need to copy the RAMDSK1 program onto your disk and set up the HELLO program to BRUN RAMDSK1. This will load the operating system into the interface card, then fast-copy your disk into drive one of the RAMDISK. Once your information is loaded into the RAMDISK, you can use all the normal DOS techniques to read and write files; the only difference is speed.

You can avoid the DOS overhead either by calling RWTS in the usual manner, or by directly using the RAMDISK registers and memory window. To do that, you just store track, sector, and drive information into two bytes, then read the data from $C800-C8FF. This approach is fastest, but you must then take on all memory management chores. Appendices to the manual list assembler source code for routines using both techniques.

The Negative Side

We discovered one apparent bug in the RAMDISK's operating system. The program does not properly update the slot and drive found parameters in the I/O Control Block used by RWTS. If a program tries to use those locations to determine which drive it was run from, it will get the wrong data.

Mechanical disk drives are known to be error-prone, so DOS has some built-in protection against errors. Each sector is recorded with a checksum; when a sector is read the checksum should match. This is very poor protection, but it does catch most errors. The RAMDISK has no such protection. The RAMDISK is much less likely to have any errors than the mechanical drives, yet it still would be nice to have at least a sector checksum. Parity on each byte would be even better, but it would be expensive.

Timing Comparisons
OperationDisk II timeRAMDISK time
Assemble 102 sectors of source code.89 sec.41 sec.
BLOAD Hi-res screen.11 sec.3 sec.
LOAD Applesoft program.14 sec.4 sec.
Dump RAM (192 sectors) calling RWTS.9 sec..8 sec.
Dump 192 sectors directn/a.7 sec.

Summary

The RAMDISK is a well-made and well-documented unit; it performs as advertised. The RAMDISK gives a terrific speed improvement over mechanical disk drives, especially if you do your own reading and writing and avoid DOS.

Two standard Apple drives with controller at normal retail prices would cost $1180; RAMDISK goes for $1395, and you get the equivalent of 10 extra tracks thrown in. (On the other hand, several non-Apple drives are available with 40 to 80 tracks, at competitive prices. And the 5- and 10-megabyte Winchesters are rapidly falling in price.)

I have seen RAMDISK advertised for as low as $1170 in Byte Magazine.

The RAMDISK 320 is available from AXLON, Inc., 170 N. Wolfe Rd., Sunnyvale, CA 94086, (408) 730-0216. RAMDISK 320, The Directory, and Mini-Base Phone Book are trademarks of AXLON INC.


Simple Hires AnimationMike Laumer

One thing that I have been working with in my next product (MIKE'S MAGIC MATRIX) is animation using hires graphics. I have been developing a hires graphics editor using the FLASH! Integer BASIC Compiler. I may not be the first one to bring a commercial product to market using the FLASH! compiler since there are at least six other programmers who are striving to beat me.

There are several methods used to achieve animation in the popular game programs. The one presented in this program is possibly the simplest. This program will animate an image in one place on the screen (in-place animation) from a series of frames of data.

The technique used to display the frame data on the screen is simply moving the data with 'LDA' and 'STA' instructions. A more powerful method of animation is to use the 'EOR' instruction to merge one frame of animation into the next. This is accomplished by using the frame data obtained by 'EOR'ing two successive frames of data. Then using that new data to 'EOR' to the image data. The 'EOR' istruction is very useful since it can add and delete data to and from the screen without disturbing any background that may be on the screen already.

A frame of data for the animation is written to the screen and then a delay loop entered to delay before the next data frame is written to the screen. If the delay is smaller the animation will speed up. If the delay is larger the animation will slow down. The delay could be read from the game paddle.

The method I used in the routine to compute the hires graphics screen addresses is to use two tables (one for lo-byte, one for hi-byte) with 192 entries to convert the Y-coordinate into a hires address. Otherwise, the Y-addresses would have to be computed by using a complicated formula:

    A = Y MOD 8
    B = (Y / 8) MOD 8
    C = Y / 64
    YADRS = 8192 + A*1024 + B*128 + C*40
    (add another 8192 if hires page2 adress needed)

So you see that even with an efficiently coded machine language routine to compute a screen address it will take a bit of time to perform. It is much more effecient to simply look up the address of the first byte of the Y-row in a table. Since the Y-coordinate never exceeds 191 (which is less than 256) the Y-register can be used easily to index the table. The table in the program only provides the offset from the beginning of a hires page. The program uses an 'ORA' instruction to put $20 or $40 into the hi-byte to specify hires page 1 or 2.

The data for the animations were built with MIKE'S MAGIC MATRIX and the first frame looks like this:

    ............................   00000000    0   0   0   0
    ............................   00000000    0   0   0   0
    ............****............   00600300    0  96   3   0
    ...........******...........   00700700    0 112   7   0 
    ..........**.**.**..........   00580D00    0  88  13   0
    ..........********..........   00780F00    0 120  15   0
    ..........***  ***..........   00380E00    0  56  14   0
    ...........******...........   00700700    0 112   7   0
    ............****............   00600300    0  96   3   0
    .............**.............   00400100    0  64   1   0
    .............**.............   00400100    0  64   1   0
    ..........********..........   00780F00    0 120  15   0
    .........**********.........   OO7C1FOO    0 124  31   0
    ........**..****..**........   00663300    0 102  51   0
    .......**....**....**.......   00436100    0  67  97   0
    .......**...****...**.......   00636300    0  99  99   0
    .......**..******..**.......   00736700    0 115 103   0
    ......**...******...**......   40714701   64 113  71   1
    ......**..***..***..**......   40394E01   64  57  78   1
    ..........**....**..........   00180C00    0  24  12   0
    ..........**....**..........   00180C00    0  24  12   0
    ..........**....**..........   00180C00    0  24  12   0
    ..........**....**..........   00180C00    0  24  12   0
    ..........**....**..........   00180C00    0  24  12   0

The data was written to a text file from within the editor and run through an Applesoft program to create an EXEC file for the S-C Macro Assembler to insert the data tables into the program.

You can make your own frames of animation by a hand process of drawing the animation dots on graph paper and reducing the data into hexadecimal data. To do this you must take each row of dots (on or off) on the graph paper and take them 7 dots at a time. The 7 dots must then be flipped into reverse order before converting into hex. Here is an example of 14 pixels width:

bit re-arrangement

As you can see the process is a pain in the neck. If the animation has a flaw in it you have to repeat the process for every frame of data that is wrong. That is where a hires graphics editor and animation design tool like MIKE'S MAGIC MATRIX really shines, because you can perfect your animation and test it in the editor without ever leaving. MIKE'S MAGIC MATRIX is not yet ready for sale lacking a manual and a little more work. I expect to have the first version ready in about two more months. Preliminary showings to the Dallas Apple Corps indicated an enormous popularity.

Since hexadecimal strings take up a lot of listing space when they are assembled, I decided to print the tables here using just the LIST command, without the assembled object code listing. The program part is shown in the normal assembled format.

Here is what you will see if you get it all typed correctly: little men here

Of course, they will all appear one after the other in the same screen position, not side-by-side.

 1000  *SAVE S.SIMPLE ANIMATION
 1010  *--------------------------------
 1020  * SIMPLE ANIMATION
 1030  *--------------------------------
 1040  MON.WAIT    .EQ $FCA8    MONITOR DELAY ROUTINE
 1050  *--------------------------------
 1060  T1          .EQ $0,1
 1070  T2          .EQ $2,3
 1080  T3          .EQ $4,5
 1090  Y.INDEX     .EQ $6,7
 1100  *--------------------------------
 1110  * ANIMATION PLAYBACK LOCATIONS
 1120  *--------------------------------
 1130  HIRES.PAGE  .EQ $20      $20 = PAGE 1, $40 = PAGE 2
 1140  Y.COORD     .EQ 100      WHERE TO PUT ANIMATION
 1150  X.COORD     .EQ 20       WHERE TO PUT ANIMATION
 1160  *--------------------------------
 1170         .OR $803
 1180         .TF B.ANIMATE
 1190  *--------------------------------
 1200  START  JSR HIRES.INIT    INITIALIZE HIRES SCREEN
 1210  .1     JSR PLAY.FRAMES   PLAY 1 SET OF FRAMES
 1220         JMP .1            GO DO IT AGAIN
 1230  *--------------------------------
 1240  PLAY.FRAMES LDA #0       INIT FRAME INDEX
 1250         STA FRAME.INDEX
 1260  .1     LDA FRAME.INDEX   GET FRAME INDEX POINTER
 1270         CMP #NUM.FRAMES   HAVE ALL FRMES BEEN DONE
 1280         BEQ .3            YES, SO RETURN
 1290         LDY $C000    HAS A KEY BEEN PRESSED
 1300         BPL .2       NO, SO KEY PLAYING THE FRAMES
 1310         LDA $C051    RESTORE TEXT SCREEN
 1320         LDA $C054    PRIMARY PAGE
 1330         JMP $3D0     EXIT ON ANY KEY
 1340  .2     ASL          DOUBLE INDEX
 1350         TAY
 1360         LDA FRAME.TABLE,Y  GET TABLE ADDRESS
 1370         STA T1       SAVE ADRS IN T1
 1380         INY          NEXT BYTE OF ADRS
 1390         LDA FRAME.TABLE,Y
 1400         STA T1+1
 1410         JSR ANIMATE  MOVE FRAME DATA TO SCREEN
 1420         INC FRAME.INDEX NEXT FRAME
 1430         BNE .1       ...ALWAYS
 1440  .3     RTS
 1450  *--------------------------------
 1460  HIRES.INIT LDA #HIRES.PAGE
 1470         STA T1+1
 1480         LDY #0
 1490         STY T1
 1500  .0     TYA          ZERO A REG
 1510  .1     STA (T1),Y   CLEAR SCREEN PAGE
 1520         INY
 1530         BNE .1
 1540         INC T1+1     NEXT PAGE
 1550         LDA T1+1     CHECK FOR
 1560         AND #$1F     END OF HIRES PAGE
 1570         BNE .0       NO, CLEAR MORE
 1580         LDA $C050    ENABLE GRAPHICS
 1590         LDA $C057    ENABLE HIRES
 1600         LDA $C054    ENABLE PAGE 1 (C055 IS PAGE 2)
 1610         LDA $C052    NOMIX
 1620         RTS
 1630  *--------------------------------
 1640  INTER.FRAME.DELAY    .DA #20
 1650  XSIZE  .DA #4        X FRAME SIZE IN BYTES
 1660  YSIZE  .DA #24       Y FRAME SIZE IN BYTES
 1670  FRAME.TABLE
 1680         .DA FRAME1
 1690         .DA FRAME2
 1700         .DA FRAME3
 1710         .DA FRAME4
 1720         .DA FRAME5
 1730         .DA FRAME6
 1740         .DA FRAME7
 1750         .DA FRAME8
 1760         .DA FRAME9
 1770  NUM.FRAMES .EQ 9
 1780  FRAME.INDEX .DA #0
 1790  *--------------------------------
 1800  ANIMATE LDA #Y.COORD     THIS IS THE STARTING ROW
 1810         STA Y.INDEX       FOR THE ANIMATION
 1820         LDY YSIZE    NUMBER OF ROWS TO PUT ON SCREEN
 1830         STY T2
 1840  .1     LDY Y.INDEX
 1850         LDA YTBL.LO,Y     COMPUTE THE ROW ADRS
 1860         CLC
 1870         ADC #X.COORD      ADD THE X OFFSET
 1880         STA T3
 1890         LDA YTBL.HI,Y
 1900         ADC #HIRES.PAGE   ADD THE HIRES PAGE BITS
 1910         STA T3+1          T3 POINTS TO ROW POSITION
 1920         LDY XSIZE         NUMBER OF BYTES TO PUT INTO ROW
 1930         DEY               INDEX BEGINS AT ZERO TO XSIZE-1
 1940  .3     LDA (T1),Y        GET FRAME DATA
 1950         STA (T3),Y        PUT ONTO SCREEN
 1960         DEY               FOR ALL BYTES IN THE ROW
 1970         BPL .3
 1980  .4     INC Y.INDEX       NEXT ROW INDEX
 1990         LDA T1
 2000         CLC
 2010         ADC XSIZE         STEP FRAME ADRS AHEAD
 2020         STA T1            TO NEXT ROW OF DATA
 2030         LDA T1+1
 2040         ADC #0
 2050         STA T1+1
 2060         DEC T2            COUNT DOW THE ROWS
 2070         BNE .1            GO MOVE REST OF FRAME ROWS
 2080         LDY INTER.FRAME.DELAY
 2090         BEQ .6            NO DELAY BETWEEN FRAMES
 2100         STY T2            SAVE DELAY
 2110  .5     LDA #30           REPEAT THIS SMALL DELAY
 2120         JSR MON.WAIT
 2130         DEC T2            FOR COUNT IN 'T2'
 2140         BNE .5            MORE DELAY
 2150  .6     RTS               FRAME IS ALL DONE
 2160  *--------------------------------
 2170  * HIRES Y OFFSET TABLES
 2180  * OFFSET FROM $2000 OR $4000
 2190  * HIRES PAGE DISPLAYS
 2200  * USING THESE TABLES SPEEDS UP
 2210  * HIRES SCREEN ADRS COMPUTATION
 2220  * A GREAT DEAL!
 2230  *
 2240  * FOR EVERY Y VALUE FROM 0-191
 2250  * THERE IS AN ENTRY IN THIS TABLE
 2260  * TO COMPUTE THE ADRS OF FIRST
 2270  * BYTE IN THE ROW.
 2280  *--------------------------------
 2290  YTBL.LO   .EQ *
 2300   .HS 00000000000000008080808080808080
 2310   .HS 00000000000000008080808080808080
 2320   .HS 00000000000000008080808080808080
 2330   .HS 00000000000000008080808080808080
 2340   .HS 2828282828282828A8A8A8A8A8A8A8A8
 2350   .HS 2828282828282828A8A8A8A8A8A8A8A8
 2360   .HS 2828282828282828A8A8A8A8A8A8A8A8
 2370   .HS 2828282828282828A8A8A8A8A8A8A8A8
 2380   .HS 5050505050505050D0D0D0D0D0D0D0D0
 2390   .HS 5050505050505050D0D0D0D0D0D0D0D0
 2400   .HS 5050505050505050D0D0D0D0D0D0D0D0
 2410   .HS 5050505050505050D0D0D0D0D0D0D0D0
 2420  YTBL.HI   .EQ *
 2430   .HS 0004080C1014181C0004080C1014181C
 2440   .HS 0105090D1115191D0105090D1115191D
 2450   .HS 02060A0E12161A1E02060A0E12161A1E
 2460   .HS 03070B0F13171B1F03070B0F13171B1F
 2470   .HS 0004080C1014181C0004080C1014181C
 2480   .HS 0105090D1115191D0105090D1115191D
 2490   .HS 02060A0E12161A1E02060A0E12161A1E
 2500   .HS 03070B0F13171B1F03070B0F13171B1F
 2510   .HS 0004080C1014181C0004080C1014181C
 2520   .HS 0105090D1115191D0105090D1115191D
 2530   .HS 02060A0E12161A1E02060A0E12161A1E
 2540   .HS 03070B0F13171B1F03070B0F13171B1F
 2550  *--------------------------------
 2560  * ANIMATION DATA
 2570  *--------------------------------
 2580  FRAME1
 2590    .HS 000000000000000000600300
 2600    .HS 0070070000580D0000780F00
 2610    .HS 00380E000070070000600300
 2620    .HS 004001000040010000780F00
 2630    .HS 007C1F000066330000436100
 2640    .HS 006363000073670040714701
 2650    .HS 40394E0100180C0000180C00
 2660    .HS 00180C0000180C0000180C00
 2670  FRAME2
 2680    .HS 000000000060030000700700
 2690    .HS 00580D0000780F0000380E00
 2700    .HS 007007000060030000400100
 2710    .HS 0040010000780F00007C1F00
 2720    .HS 00663300404141016C60031B
 2730    .HS 3C70071E0070070000380E00
 2740    .HS 00180C00000C1800000C1800
 2750    .HS 000C1800000C180000000000
 2760  FRAME3
 2770    .HS 006003000070070000580D00
 2780    .HS 00780F0000380E0000700700
 2790    .HS 06600330064001300C400118
 2800    .HS 78780F0E607F7F0740677301
 2810    .HS 004001000060030000700700
 2820    .HS 00700700001C1C00000C1800
 2830    .HS 000C1800000C180000063000
 2840    .HS 000630000000000000000000
 2850  FRAME4
 2860    .HS 000000004061430140714701
 2870    .HS 60580D0320780F0220380E02
 2880    .HS 607007034061430100414100
 2890    .HS 00463100007E3F0000780F00
 2900    .HS 006003000040010000600300
 2910    .HS 0070070000700700001C1C00
 2920    .HS 000630000003600000036000
 2930    .HS 400140014001400100000000
 2940  FRAME5
 2950    .HS 000000000000000040610303
 2960    .HS 4071070320580D0220780F02
 2970    .HS 60380E036070070340614301
 2980    .HS 0043610000463100007E3F00
 2990    .HS 00780F000060030000400100
 3000    .HS 006003000070070000700700
 3010    .HS 001C1C000006300000036000
 3020    .HS 000360004001400140014001
 3030  FRAME6
 3040    .HS 000000004061430140714701
 3050    .HS 60580D0320780F0260380E03
 3060    .HS 607007034061430100436100
 3070    .HS 00463100007E3F0000780F00
 3080    .HS 006003000040010000600300
 3090    .HS 0070070000700700001C1C00
 3100    .HS 000630000003600000036000
 3110    .HS 400140014001400100000000
 3120  FRAME7
 3130    .HS 406143014071470160580D03
 3140    .HS 20780F0220380E0260700703
 3150    .HS 406143010043610000463100
 3160    .HS 007E3F0000780F0000600300
 3170    .HS 004001000060030000700700
 3180    .HS 00700700001C1C0000063000
 3190    .HS 000360000003600040014001
 3200    .HS 400140010000000000000000
 3210  FRAME8
 3220    .HS 006003000070070000580D00
 3230    .HS 00780F0000380E0000700700
 3240    .HS 06600330064001300C400118
 3250    .HS 78780F0E607F7F0740677301
 3260    .HS 004001000060030000700700
 3270    .HS 00700700001C1C00000C1800
 3280    .HS 000C1800000C180000063000
 3290    .HS 000630000000000000000000
 3300  FRAME9
 3310    .HS 000000000060030000700700
 3320    .HS 00580D0000780F0000380E00
 3330    .HS 007007000060030000400100
 3340    .HS 0040010000780F00007C1F00
 3350    .HS 00663300404141016C60031B
 3360    .HS 3C70071E0070070000380E00
 3370    .HS 00180C00000C1800000C1800
 3380    .HS 000C1800000C180000000000

A Text File Display Command for DOSBob Sander-Cederlof

How many times have you wished that you could see what was in a TEXT file? You end up loading a word processor (if you are lucky enough to have one that can read normal DOS TEXT files), or EXECing it into the S-C Macro Assembler, or writing a special Applesoft program.... Why not a DOS command for this very common need?

The June 1982 issue of Call A.P.P.L.E. has an article by Lee Reynolds describing the addition of a FILEDUMP command to DOS. Lee gives a 20-byte program which fits nicely in an unused space in DOS. He replaced the MAXFILES command with "FILEDUMP". In case you want to try it, here are the patches for Lee's method.

     ]CALL -151
     *BCDF:20
     *BCE0:8E FD 20 A3 A2 20 8C A6 F0 05 20 F0 FD D0 F6 20
     *BCF0:FC A2 60
     *A8E7:46 49 4C 45 44 55 4D D0
     *9D48:DE BC
     *A933:20 30
     *3D0G
     ]

$BCDF-BCF2 is the FILEDUMP command processor. $A8E7-$A8EE is the string "FILEDUMP", the command name. The two bytes at $9D48,9D49 are the address (minus 1) of the command processor. The two bytes at $A933,A934 are flags indicating that the FILEDUMP command requires a filename, and can optionally have S and D parameters.

My first reaction to the program, being a programmer, was to try to modify it. The first change I made saved one byte. The last two instructions are a JSR and an RTS. By ending with a JMP to the final subroutine, the RTS at BCF2 is not needed. Then I tried modifying the order of the loop, and saved another three bytes. Here is my revised listing:

 1000  *--------------------------------
 1010  *      "FILEDUMP" COMMAND
 1020  *--------------------------------
 1030  DOS.OPEN.TEXT.FILE   .EQ $A2A3
 1040  DOS.CLOSE.FILE       .EQ $A2FC
 1050  DOS.READ.ONE.BYTE    .EQ $A68C
 1060  MON.COUT1            .EQ $FDF0
 1070  *--------------------------------
 1080         .OR $BCDF
 1090         .TA $8DF
 1100  FILEDUMP
 1110         JSR DOS.OPEN.TEXT.FILE
 1120         LDA #$8D
 1130  .1     JSR MON.COUT1
 1140         JSR DOS.READ.ONE.BYTE
 1150         BNE .1       PRINT IT
 1160         JMP DOS.CLOSE.FILE
 1170  *--------------------------------
 1180         .OR $A8E7
 1190         .TA $8E7
 1200         .AT /FILEDUMP/   NAME OF FILEDUMP COMMAND
 1210  *--------------------------------
 1220         .OR $9D48
 1230         .TA $848
 1240         .DA FILEDUMP-1   BRANCH FOR FILEDUMP COMMAND
 1250  *--------------------------------
 1260         .OR $A933
 1270         .TA $833
 1280         .HS 2030     FILENAME REQUIRED, SLOT & DRIVE
 1290  *                   ARE OPTIONAL

After playing with the new command a little, I thought of several more changes. I wanted to be able to stop the file listing, to restart it, and to abort it. The first article I ever wrote about Apples described just such an addition, at that time for Integer BASIC. (See MICRO, June/July, 1978.) With this addition, the program would not fit in the unused space at $BCDF, so I decided to put it in the place of the INIT command instead. I changed the name to "SHOW".

Not all of the code would fit in the spot where the INIT command processor is, at $A54F. Therefore I broke out the routine to check for the pause/abort keys as a separate subroutine, and placed in over the top of some of the INIT code inside the File Manager of DOS. If you install this patch, you could call on the PAUSE.CHECK subroutine from your own programs.

 1000  *--------------------------------
 1010  *      "SHOW" COMMAND
 1020  *--------------------------------
 1030  DOS.OPEN.TEXT.FILE  .EQ $A2A3
 1040  DOS.CLOSE.FILE      .EQ $A2FC
 1050  DOS.READ.ONE.BYTE   .EQ $A68C
 1060  KEYBOARD            .EQ $C000
 1070  STROBE              .EQ $C010
 1080  MON.COUT1           .EQ $FDF0
 1090  *--------------------------------
 1100         .OR $A54F
 1110         .TA $84F
 1120  SHOW
 1130         JSR DOS.OPEN.TEXT.FILE
 1140         LDA #$8D
 1150  .1     JSR MON.COUT1
 1160         JSR PAUSE.CHECK
 1170         BEQ .2
 1180         JSR DOS.READ.ONE.BYTE
 1190         BNE .1       PRINT IT
 1200  .2     JMP DOS.CLOSE.FILE
 1210  *--------------------------------
 1220  *      RETURN .EQ. IF ABORT
 1230  *             .NE. IF CONTINUE
 1240  *--------------------------------
 1250         .OR $AE8E         OVER "INIT" CODE
 1260         .TA $88E
 1270  PAUSE.CHECK
 1280         LDA KEYBOARD      ANY KEY PRESSED?
 1290         BPL .2            NO, CONTINUE
 1300         STA STROBE        YES, CLEAR STROBE
 1310         CMP #$8D          ABORT?
 1320         BEQ .2            YES, RETURN .EQ. STATUS
 1330  .1     LDA KEYBOARD      NO, PAUSE TILL KEYPRESS
 1340         BPL .1            NONE PRESSED YET
 1350         STA STROBE        CLEAR STROBE
 1360         CMP #$8D          ABORT?
 1370  .2     RTS               .EQ. IF ABORT
 1380  *--------------------------------
 1390         .OR $A884
 1400         .TA $884
 1410         .AT /SHOW/   SHOW COMMAND NAME
 1420  *--------------------------------
 1430         .OR $A909
 1440         .TA $809
 1450         .HS 2030     FLAGS FOR SHOW COMMAND
 1460  *--------------------------------

After assembling the program above, the various pieces are in memory in page 8 and 9, instead of inside DOS. I did it this way because DOS is protected during assembly. You can install the patches by hex input commands, or by some memory moves. I did it this way:

     :$A54F<84F.863M
     :$AE8E<88E.8A4M
     :$A884:53 48 4F D7
     :$A909:20 30

Then try typing "SHOW filename", where "filename" is a text file, and see the action.

You may want to put some POKEs in your HELLO file on some disks to install the SHOW command. If so, this is what they might look like:

     100 DATA 21,42319,32,163,162,169,141,32,240,253,32,142,
              174,240,5,32,140,166,208,243,76,252,162
     110 DATA 23,44686,173,0,192,16,17,141,16,192,201,141,
              240,10,173,0,192,16,251,141,16,192,201,141,96
     120 DATA 4,43140,83,72,79,215
     130 DATA 2,43273,32,48
     140 DATA 0
     150 READ N : IF N THEN READ A : FOR I = 1 TO N : READ D
         : POKE A+I-1,D : NEXT : GO TO 150

I tried several other versions, with features like clearing the screen, filling it up, and waiting; a stand-alone program, rather than a DOS command; and so on. You will probably want to try your own experiments.


Hierographic Transport (review)Mike Laumer

Hierographic Tranport is a Hi-Res printer dump program for the Epson series of printers (MX-70, MX-80 and MX-100). The program is a very easy to use, menu driven system. The user manual is only 12 pages long, but most functions are self-apparent. I used the program for over an hour before I felt the need to refer to the manual. The program allows very complete control over the dot graphics mode of the Epson printers.

From the menus you can load a Hi-Res picture into either page 1 or page 2. Selections are provided for normal/inverse picture, normal/rotated pictures, normal/compressed print mode and a setable left margin to allow centering a picture on the page.

You can control magnifying or scaling the picture from 1 to 99 times normal size in the X or Y directions. This magnification is performed by repeatedly printing each screen dot, in the X and Y directions. The magnification only affects the printed image and not the screen image.

There is also the ability to select a "window" from the Hi-Res Screen that will be printed on the printer. That way you can print the rectangular section of the screen that you are interested in.

The "window" is controlled with two sets of cursor control keys. The "WASZ" keys control the top and left sides of the cursor. While the familiar "IJKM" keys control the right and bottom sides of the cursor. This is adequate for controlling the "window" but I would have prefered one set to control inward movement of the cursor sides and the other set to control outward movement of the cursor sides.

The cursor is presented as a set of blinking lines overlayed on the picture image. This technique uses the HXPLOT function described in the June issue of AAL. This function allows non-destructive lines to be drawn and erased over the top of an image on the Hi-Res screen.

The cursor lines are automatically stepped by an amount from 1-9, selectable by the number keys. The space bar or any other valid command key will stop the cursor from advancing. If "0" is selected for the step distance, the cursor lines will step by 1 whenever a cursor control key is pressed. This allows a fine positioning mechanism.

Once a "window" is selected the user can have it printed on his printer. When this is selected, the program automatically checks up on the parameters you have selected and computes the size of the image as it should be on the printer. If you have scaled the image too big, an error message will result.

The overall design of the program is good. There are however, a few minor problems in operation of the program. When the "window" is very large the automatic steps in advancing the "window" occur slowly. As the size of the "window" gets smaller, the speed of the automatic advance gets very fast making it hard to stop on the exact point you want. The cursor routine needs a delay which varies by the size of the "window" to help even out the speed of the automatic cursor advance.

There is a record of data kept at the bottom of the screen when you are selecting a "window". This data provides you with the cursor locations and a unique display of the computed size of the picture to be printed. As the cursor is moved, the data is updated to the new recomputed picture size. The size display often flickers because blanks are written to the screen and then the data is written. If the data were written then the line cleared to the end of line, the flicker would be less noticeable.

The size display had the only bug in the whole program that I could find. The bug is rather trivial and does not affect the quality of the program. A bug, however, is a bug! [ I am sure they will fix it, once they read this review. ] When a very large scale factor (99 x 99) is used, the routine to print out the size goes bananas and displays some garbage characters on the screen. When compressed printing is selected (where the dot spacing on the Epson goes from 1/60 of an inch to 1/120 of an inch on the horizontal direction), the size display goes one character too far and scrolls the data up the screen. As the cursor window is moved arround the scrolling eventually scrolls the title lines off the main menu.

Unless you plan to print a wall mural for the side of your barn, you should never encounter the problem. A 99 x 99 scaling factor will give a pixel size of 1.5 inches square! A full screen print would be 38 feet by 21 feet in size!!! That's way beyond the carriage width of even the MX-100. The program could handle it though as long as you print it in narrow window strips. (A nice future enhancement would be for the program to automatically print an oversize picture in strips sized for your particular printer.)

The program has a built in configuration routine and can easily be configured for the following interfaces:

The Epson printers are very popular, but many more brands of printers are now on the market which have comparable capabilities. For example, the NEC PC-8023, the MPI-88G, and the Okidata series. I hope that the GSR folks come out with equivalent "Transports" for these other printers. All of them on the same disk would be especially nice!

Conclusion: A fine program for graphics printer dumping. I rate this program a "B+". A little attention to its few problems would raise the grade to "A".

This program is sold for $39.00 and is available from GSR Associates, P.O. Box 401462, Garland, Texas 75040. (Don't be afraid of the P. O. Box, they are real people.)


Christmas in July?Bob Sander-Cederlof

Mike Laumer has decided to offer a special price to readers of Apple Assembly Line on his FLASH! Integer BASIC Compiler. For a limited time, AAL readers can buy FLASH! for only $49, a savings of almost 40% from the normal $79 price. The offer expires September 1, 1982, and is limited to one per customer. To qualify you must mention that you read about it in AAL, and call or write directly to Laumer Research. Mike's phone is (214) 245-3927; write to 1832 School Road, Carrollton, TX 75006.

What a bargain! The FLASH! compiler is an incredible software design tool which can translate Integer BASIC programs into extremely fast machine language programs. It is the only full feature compiler on the market that can provide assembly language listings and source files compatible with my S-C Assemblers.

Synergistic Software is now selling the Galfo Integer BASIC Compiler for $149; it is copy protected, has no assembly language output, fewer extensions to the language, an undocumented run-time package, and no option to buy the run-time package source code. I have heard that it is a good compiler, but I think the price is too high.

FLASH!, on the other hand, is NOT copy protected. You can make as many copies for your own use as you need. FLASH! adds features for hi-res graphics and system programming to the Integer BASIC language. The FLASH! run-time package is fully documented, and owners of FLASH! can get the source code of the run-time package on disk for only $39. FLASH! allows easy relocation of the object code for any requirements. Used in combination with the S-C Assembler, you can further optimize the object code for even greater memory and time savings. And at this special price, it truly is a bargain. Christmas in July!


Apple Assembly Line is published monthly by S-C SOFTWARE, P. O. Box 280300, Dallas, TX 75228. Phone (214) 324-2050 Subscription rate is $15 per year, in the USA, sent Second Class Mail; $18 per year sent First Class Mail in USA, Canada, and Mexico; $28 per year sent Air Mail to other countries. Back issues are available for $1.50 each (other countries add $1 per back issue for postage). All material herein is copyrighted by S-C SOFTWARE, all rights reserved. Unless otherwise indicated, all material herein is authored by Bob Sander-Cederlof. (Apple is a registered trademark of Apple Computer, Inc.)