## University of Wollongong Research Online Department of Computing Science Working Paper Series Faculty of Engineering and Information Sciences 1984 # An EPROM programmer for the ET3400 expansion system Michael J. Milway University of Wollongong, mjm@uow.edu.au #### Recommended Citation Milway, Michael J., An EPROM programmer for the ET3400 expansion system, Department of Computing Science, University of Wollongong, Working Paper 84-7, 1984, 25p. http://ro.uow.edu.au/compsciwp/80 #### THE UNIVERSITY OF WOLLONGONG #### DEPARTMENT OF COMPUTING SCIENCE ### An EPROM programmer for the ET3400 expansion system #### **Michael Milway** Department of Computing Science University of Wollongong #### **Abstract** An EPROM programmer has been constructed that can program 2716 type 2K by 8 bit and 2532 type 4K by 8 bit EPROMs. Expansion to 8K by 8 bits EPROMs is possible. The EPROM programmer is used as a peripheral device to the ET3400 Microcomputer Trainer Expansion system. Data to be programmed may be down line loaded from a host system or read in from another ROM or EPROM. The EPROM programmer software can program an EPROM, verify that programming was correct, read a ROM or EPROM, check for correct erasure and fill an area of memory with OxFF. **Preprint No 84-7** May 16, 1984 ## An EPROM programmer for the ET3400 expansion system Michael Milway Department of Computing Science University of Wollongong #### ABSTRACT An EPROM programmer has been constructed that can program 2716 type 2K by 8 bit and 2532 type 4K by 8 bit EPROMs. Expansion to 8K by 8 bits EPROMs is possible. The EPROM programmer is used as a peripheral device to the ET3400 Microcomputer Trainer Expansion system. Data to be programmed may be down line loaded from a host system or read in from another ROM or EPROM. The EPROM programmer software can program an EPROM, verify that programming was correct, read a ROM or EPROM, check for correct erasure and fill an area of memory with 0xFF. #### 1. Introduction In many small microcomputers there is a small amount of ROM that contains a monitor program. The rest of the memory is made up of RAM which does not retain its data after power is turned off. Programs must be reloaded after power is reapplied. This is very inconvenient, especially where frequently used software is involved. The solution to this problem is to write the programs into EPROM which will retain its data after power down. EPROMs are programmed by setting up the address and data and applying a high voltage programming pulse for a set time. Once the data has been programmed in it can only be erased by exposing the EPROM to high intensity ultraviolet light. To facilitate the programming of EPROMs and EPROM programmer has been constructed that is capable of programming the most common types of EPROMs. It was made as a peripheral device for the existing ET3400 microcomputer trainer system, as this is much more flexible than a stand alone device. The ET3400 system is a small teaching system based on the Heathkit ET3400 microcomputer trainer. An expansion bus and card cage has been added enabling various experiment cards to be plugged in. #### 2. Design Considerations #### 2.1. Pin Compatibility Fortunately, as new EPROM types have been developed they have remained fairly pin compatible with the older types. With the exception of a few pins the pinouts of these devices have remained constant. Even the newer 28 pin devices have been designed so that they are compatible with the 24 pin devices. Mask programmed ROMs also have the same pinouts as EPROMs. This is done so that systems can be developed using easily programmed EPROMs and then mass produced using the much cheaper (in very large quantities) ROMs without any hardware modifications. Thus if the EPROM programmer can read the data from an EPROM it can also read it from a compatible ROM. It was decided to concentrate solely on the single supply 24 pin EPROM types rather than the older triple supply ones. Single supply EPROMs require a single 5V supply rather than the 5V and +/-12V supplies required by the triple supply ones. Table 1 gives the pinouts of single supply 2K by 8 and 4K by 8 24 pin EPROMs and 8K by 8 and 16K by 8 28 pin EPROMs. Note that the 24 pin devices only differ in two pins and the 28 pin devices fit neatly over the top of the 24 pin devices, with a change in only two more pins. | pin #<br>24 pins | pin # 2716<br>28 pins 2K * 8 | | 2532<br>4K * 8 | 2764<br>8K * 8 | 27128<br>16K * 8 | | | |------------------|------------------------------|-----------------|-----------------|-------------------------------|-------------------------------|--|--| | | | | | | | | | | 22<br>23<br>24* | 24<br>25<br>26<br>27<br>28 | A9<br>A8<br>Vcc | A9<br>A8<br>Vcc | A9<br>A8<br>N/C<br>Pgm<br>Vcc | A9<br>A8<br>A13<br>Pgm<br>Vcc | | | Signals marked '~' are active low. Pins marked \* have different signals on different EPROM types. Table 1 EPROM pinouts #### 2.2. Software Compatibility All the EPROMs mentioned in table 1 are programmed in the same manner. The programming voltage Vpp is applied and the correct address and data asserted. The control pins $\sim$ CE, $\sim$ OE, $\sim$ G and $\sim$ E/prog are set to the correct level, depending on the EPROM type, and left for 50 milliseconds. After this time the data will have been programmed into the specified location. By having a set of variable control bytes that are initialised when the EPROM type is determined the same routine may be used to program any of the above mentioned EPROM types. #### 2.3. Other Design Considerations No chip should be inserted or removed from a socket while power or logic levels are present. It was decided to include a dual pole switch to control both the supply and programming voltages. The address, data and control buffers are also enabled by this switch. The programming supply voltage, Vpp, of 25 volts is not available on the ET3400 system. This voltage could either be supplied by an external power supply or by an on board DC to DC converter. The latter option, although mare complex, was chosen to make the operation of the EPROM programmer easier. #### 3. Implementation Figure 1 shows the circuit diagram for the EPROM programmer. It consists of: - (i) An interface to the ET3400 expansion bus. - (ii) A set of buffers for the address, data and control lines. - (iii) The power supply for Vpp. - (iv) Electronic switching to allow Vpp to be applied under program control. - (v) A rotary switch to determine EPROM type. - (vi) A Zero Insertion Force socket to hold the EPROM. The pinouts of the signals on the expansion bus are shown in brackets beside the signal. A 74LS42 is used to decode a board select and three address lines to give a single select line for the EPROM programmer. Address line A2 is used as one of the select lines on PIA2 and inverted for use as a select line on PIA1. The outputs of the PIAs are buffered by the tristate buffers. These buffers are enabled when the EPROM power switch is on. The buffer on the data lines is also bidirectional. The power supply for Vpp consists of an oscillator (NE555), a pair of current pumps (C2, D1, D2 and C4, D3, D4) and a voltage regulator (Q1). The output of the oscillator has a swing of 12V which is applied to C2 and C4. The other side of C2 follows this voltage, but is constrained by D1 to always be greater than 11.4V. Thus this side swings between 11.4V and 23.4V. D2 will conduct whenever it is forward biased by more than 0.6V, charging C3 to 22.8V. The second current pump adds this voltage to the input to C4, charging C5 to 33.6V. This voltage will drop under load. Q1 will regulate the on C5 down to 25V for use as Vpp. R2 allows for the adjustment of Vpp between 12V and 33.6V. The programming voltage may be switched to the EPROM under program control. When PBO of PIA2 is low and the power switch is turned on, Q2 is turned on allowing Vpp into the EPROM. When Q2 is off the Vpp pin of the EPROM is at 5V. The three position rotary switch is a simple means of changing the address and control lines to suit different EPROM types. The switch has been wired for 2716 and 2532 type EPROMs. The third position is currently unused. The switch is also connected to three inputs on PIA2 to allow the software to determine the EPROM type. The EPROM power switch switches Vcc and Vpp to the EPROM. It is also used to enable the tristate buffers. Vcc is buffered and fed into PB4 of PIA2 to tell the software if the power is switched on. Finally three LEDs are connected to PB5 - 7 of PIA2. These LEDs are used to tell whether programming, verifying or reading operations are in progress. #### 3.1. Addressing When a PIA is normally wired up its internal registers are addressed as follows: | Register | |--------------------| | Data Register A | | Control Register A | | Data Register B | | Control Register B | | | This is fine for eight bit data transfers. The addressing of the PIA registers may be modified to enable sixteen bit transfers (such as outputting EPROM addresses) by swapping A0 and A1 on the register select pins. The address map for the PIA then becomes: | Address | Register | | | | | | |---------|--------------------|--|--|--|--|--| | 0 | Data Register A | | | | | | | 1 | Data Register B | | | | | | | 2 | Control Register A | | | | | | | 3 | Control Register B | | | | | | Sixteen bit loads and stores, such as ldx and stx, may now be made. Data Register A will be the high order register and data Register B the low order one. A PIA may have all the bits in a data register independently configured as input or output. Reading from the input pins and writing to the output pins may be done without affecting the other pins even though they all share the same address. PIA1 is used to output the EPROM address. It has been connected to allow sixteen bit transfers as described above. Data Register A is used to output the high order address bits and Data Register B outputs the low order bits. Only thirteen bits out of the possible sixteen bits are used to output the address. The three high order bits of Data Register A are used as inputs to signify the EPROM type. Data Register A of PIA2 is used to transfer data to the EPROM when programming and to transfer data from the EPROM when reading, verifying or checking erasure. Data Register B of PIA2 is used to control the function of the EPROM programmer. Bit 0 controls the programming voltage Vpp. Bits 1 and 2 control the Enable and Gate inputs to the EPROM respectively. Bit 3 controls the direction of transfers through the data buffer. Bit 4 is used to sense whether the EPROM power switch is on. Bits 5, 6 and 7 control the Read Write and Verify LEDs respectively. Figure 2 gives an address map of the PIAs. #### 4. Software A program has been written in 6800 assembly language to program 2K or 4K EPROMs. The program currently resides in ``` /usr/hardware/eprom/4keprom.s ``` on system B. A copy of this program is included as an appendix. The following commands are available: ``` Program p<ramstart>[,<count>[,<romstart>]] Verify v<ramstart>[,<count>[,<romstart>]] Read r<ramstart>[,<count>[,<romstart>]] Erased? e[<romstart>[,<count>] Fill f<ramstart>[,<count>] Exit x[<address>] ``` Arguments in angle brackets, <>, are hexadecimal values and arguments in square brackets, [], are optional. The default values for optional arguments are 0x0800 or 0x1000 for the count when using 2K or 4K EPROMs respectively, 0x0000 for the EPROM start address and the reset vector (0xf800 for the Heathkit trainers) for the exit address. The program has been written in structured code. The top level is a command interpreter which takes an input line, extracts the command and argument values. The commands are called as separate subroutines. The command subroutines in turn call other subroutines such as 'ready' which tests for EPROM type and power on and 'putline' which writes a line of text to the screen. #### 4.1. Command Interpreter On cold start or errors the command interpreter prints out a welcoming message which includes a list of available commands and required arguments. It then reads a line of text from the keyboard into a buffer. The command character is stripped off and saved. The arguments are then extracted and stored as sixteen bit numbers along with an argument count. The arguments must be valid hexadecimal numbers, separated by commas and terminated by a newline character. The next step is to load the default values for 2K EPROMs. If 4K EPROMs are being used the count will later be overwritten by the default value for 4K EPROMs. If the optional arguments have been supplied they will overwrite the default values and thus be used instead. A jump table is used to identify the command and call the appropriate subroutine. When the subroutine returns the program branches back to a warm start and repeats the command interpreter loop. If the command is not found the program prints an error message and branches back to a cold start. #### 4.2. Program This command programs either a 2K or 4K EPROM. The user may specify the address in RAM of the data to be programmed, a count of how many bytes to program and where to put the data in the EPROM. The count and the EPROM start address are optional. The 'ready' subroutine is called first. It checks that the EPROM power switch is on and determines the type of EPROM from the position of the rotary switch. If a 4K EPROM is to be programmed it changes the default count value. It also initialies two variables, 'pinit' and 'pprog', with information necessary to program the required type. The 'ports' subroutine is then called. It programs the PIAs for correct data direction and sets some initial values. As this subroutine returns with the data port configured for reading the EPROM, the direction of this port is then reversed so that data may be written to the EPROM. The arguments required to determine the start of the RAM buffer, the count and the EPROM start address are read in, overwriting the default values. If these arguments are not supplied then the default value will be used. The actual programming of the EPROM now follows. The initial control value is written out to the control port. The current EPROM address is written out to the Heathkit display to show the progress of the programmer. The Current EPROM address and data are then written out to the EPROM. The control value is changed to enable programming. After a fifty millisecond delay the control value is changed back to the initial value and the location has been programmed. The RAM and EPROM addresses are incremented and the count decremented. The programming loop is repeated until all bytes have been programmed. The control value is reset to 0xFF and a farewell message is printed out before returning to the main program. Note that all the command subroutines follow the same general pattern of calling 'ready' and 'ports', and loading the address and count arguments before performing the required task. The subroutines then reset the control value, print a farewell message and return to the main program. #### 4.3. Verify This command verifies that the data has been correctly programmed into the EPROM. 'Ports' and 'ready' are called and the addresses and count loaded if supplied. The control port is set to enable reading of the EPROM. The control value is the same for 2K or 4K EPROMs. The data is read one byte at a time and compared with the corresponding byte in RAM. If an error is found, the EPROM address, data and the correct data are printed in an error message. The subroutine then waits for any key to be pressed before continuing. If an 'x' has been pressed the command is aborted. 'Verify' exits in the same manner as 'program' when it successfully completes or is aborted. #### 4.4. Erased This command checks that all the required locations in the EPROM have been erased, i. e. set to 0xFF. It works in the same manner as verify but checks each location for 0xFF rather than comparing it with a location in RAM. #### 4.5. Read This command reads the contents of an EPROM or ROM into a specified area of RAM. 'Ports' and 'ready' are called and the appropriate arguments loaded in. The control value is set to enable the EPROM or ROM to be read. The control value is the same for both 2K and 4K devices. Each location is read in turn and dumped into the specified area of RAM. The control register is then set to 0xFF, a farewell message printed and the subroutine returns to the main program. #### 4.6. Fill This command fills a specified area of RAM with 0xFF. This is useful when programming an EPROM with several data segments. The RAM buffer is filled with 0xFF and the data segments loaded into the appropriate positions in the buffer. The whole EPROM is then programmed. Unused areas of the EPROM will remain set to the erased value of 0xFF and thus may be programmed at a later date. Since this command does not use the EPROM programmer hardware 'ports' and 'ready' are not called. The required RAM address and count, if supplied, are loaded and a loop executed filling the required area with 0xFF. The default count is 0x800 so that if 4K EPROMs are to be used the command must either specify the count or the command used twice. The command exits by printing a farewell message and returning to the main program. #### 4.7. Exit This command allows the user to exit the EPROM programmer. If the exit address is supplied the command prints a message and jumps to that address. The default exit address is the reset vector for the microcomputer being used. It is read out of locations 0xfffe and 0xffff, which is the reset vector for both the 6800 and 6809 microprocessor. #### 4.8. Other Routines 'Ports' is used to set up both PIAs prior to a command being executed. The two address ports are set up as outputs, with the top three bits of the high order address port being set up as inputs. The data port is set up as input. The initial value of the control port is set to 0xFF before it is set to be an output port. This ensures that there is no glitch as the port changes from being an input (default setting on reset) to an output port. 'Ready' checks that the EPROM power switch is on and determines what type of EPROM, 2K or 4K, is being used. It starts by assuming that a 2K EPROM has been selected and sets 'pinit' and 'pprog' to suit. It then sits in a loop waiting until the power switch is on and either a 2K or 4K EPROM has been selected. The third setting for an 8K EPROM is ignored by this program. If a 4K EPROM has been selected 'pinit' and 'pprog' are reinitialised and the default count set to 0x1000. The terminal I/O subroutines used by this program are already available in the EPROM on the ET3400 expansion system master card (called 6800lib) and the subroutines needed to use the ET3400 display are in the monitor ROM. #### 5. Operation The following items of hardware are needed to use the EPROM programmer. FT3400 trainer Expansion system with separate + 5V, + 12V and -12V power supply Master card with at least 4K of RAM (6K for 4K EPROMs) EPROM programmer card Downline load the EPROM programmer software from /usr/hardware/eprom/4keprom.out on system B. If the data is to be downline loaded then do so at this stage. Use the B(ias) option on dll to load the data in available RAM. Start the program running at 0x2000. Turn the EPROM power switch off. Select either 2K (2716) or 4K (2532) type using the rotary switch. Insert the EPROM into the Zero Insertion Force Socket with pin 1 to the upper left. Turn the EPROM power switch on. Enter the required command on the terminal keyboard and press return. When an EPROM is to be programmed it should first be checked for erasure. When it has been programmed it should be verified. The erasure command can also be used to single step through the EPROM data to view the contents. It will skip over locations containing 0xFF. #### Sample commands: | p2800,100,400 | Program 100 hex bytes starting from location 2800 hex in RAM into the EPROM starting at location 400 hex. | |---------------|------------------------------------------------------------------------------------------------------------------------------| | p2800 | Program all of the EPROM with data starting at location 2800 hex in RAM. The default count and EPROM start address are used. | | p2800,400 | Program the first 400 hex bytes of the EPROM with data starting at location 2800 hex in RAM. | | v2800 | Verify the whole EPROM against data starting at location 2800 hex in RAM. | | v2800,100,400 | Verify that the first programming example worked correctly. | | r2800 | Read the whole of the EPROM into RAM starting at location 2800 hex. | | e | Check the whole EPROM for erasure. | | f2800 | Fill 2k (default count) of RAM with 0xFF starting at location 2800 hex. | | f2800,1000 | Fill 4K or RAM with 0xFF starting at location 2800 hex. | | x | Exit to the reset vector (i. e. the Heathkit monitor). | #### 6. Future Enhancements xa000 The EPROM programmer in its current state can only handle 24 pin devices. If the 24 pin ZIF socket was replaced by a 28 pin ZIF socket and the third position of the rotary switch used, its capacity would be increased to include the 2764 8K by 8 bit EPROM type. 24 pin devices could still be used by inserting them in the 28 pin socket so that their pins corresponded to the equivalent 28 pin device types, see table 1. The third position of the rotary switch would be used to connect the enable signal to pin 20, output enable to pin 22 and A11 to pin 23. The program voltage, Vpp, would be connected to pin 1, A12 to pin 2, ~Vppon to pin 27 and the supply voltage, Vcc, to pin 28. Since pin 26 is no connect, Vcc for 24 pin devices may remain connected. Exit to ET3400 transparent link. The only software changes required are to check for the 8K position of the rotary switch and reprogram pinit, pprog and the default count to suit the 2764. It is impractical to extend the design to handle 27128 or 27256 type devices as these require additional address lines and more switching around of control signals. It would be better to design a new EPROM programmer to cater for these types. Figure 1b ET-3400 EPROM PROGRAMMER SHEET 2 OF 2 DRAWN M.J.M. 24/4/1984 Figure 2, Addresses of PIAs, bit usage and data directions. | address | port | bit | direction | use | DDR | initial data | |---------|-------------------------------------------|---------------------------------|------------------------------------|-------------------------------------------------|---------------------------------|---------------------------------| | 9380 | data reg<br>(1A)<br>high order<br>address | 7<br>6<br>5<br>4<br>3<br>2<br>1 | < < <>>>> | 8K<br>4K<br>2K<br>A12<br>A11<br>A10<br>A9<br>A8 | 0<br>0<br>0<br>1<br>1<br>1<br>1 | X<br>x<br>x<br>0<br>0<br>0<br>0 | | 9381 | data reg<br>(2A)<br>low order<br>address | 7<br>6<br>5<br>4<br>3<br>2<br>1 | >>>>>>> | A7<br>A6<br>A5<br>A4<br>A3<br>A2<br>A1<br>A0 | 1<br>1<br>1<br>1<br>1<br>1 | 0<br>0<br>0<br>0<br>0<br>0 | | 9382 | control reg (1A) | 7<br>6<br>5<br>4<br>3<br>2<br>1 | 00H to load DDR<br>04H to load add | | | | | 9383 | control reg (2A) | 7<br>6<br>5<br>4<br>3<br>2<br>1 | 00H to load DDR<br>04H to transfer | | | | | 9384 | data reg<br>(1B)<br>EPROM data | 7<br>6<br>5<br>4<br>3<br>2<br>1 | <> <> <> <> <> <> <> <> | D7 D6 D5 D4 D3 D2 D1 D0 | 0<br>0<br>0<br>0<br>0<br>0 | 1<br>1<br>1<br>1<br>1<br>1 | Figure 2 continued. | 9385 | control reg | 7<br>6<br>5<br>4<br>3<br>2<br>1 | DDR = FFH for data<br>DDR = OOH for data<br>OOH to load DDR<br>O4H to transfer dat | read | | | |------|-------------|---------------------------------|------------------------------------------------------------------------------------|---------|---------|---| | 9386 | data reg | 7 | > ~ve | erify 1 | l | 1 | | | (2B) | 6 | > ~wr | ite 1 | l | 1 | | | command/ | 5 | | ead ] | L | 1 | | | status | 4 | < ~Er | ı ( | 0 | x | | | | 3 | | rn ] | 1 | 1 | | | | 2 | > ~G | | 1 | 1 | | | | 1 | > ~E | 1 | 1 | 1 | | | | 0 | > ~v <sub>I</sub> | ppon 1 | 1 | 1 | | 9387 | control reg | 7 | | | | | | | (2B) | 6 | | | | | | | | 5 | | | | | | | | 4 | OOH to load DDR | | | | | | | 3 | 04H to transfer con | nmands | /status | | | | | 2 | | | | | | | | 1 | | | | | | | | 0 | . , | | | | #### note: Initial data for the command/status register should be loaded before the I/O bits are set to output. --> means into EPROM <-- means out of EPROM <--> means bidirectional data flow #### Appendix 1, Software source listing ``` ; Terminal driven Eprom programmer for use with Heathkit trainers. Michael Milway ; 15/8/1983 ; ammended 26/4/1984 ; Commands are: ; p<ramstart>[,<count>[,<romstart>]] Program Verify v<ramstart>[,<count>[,<romstart>]] ; r<ramstart>[,<count>[,<romstart>]] Read Erased? e[<romstart>[,<count>]] ; F111 f<ramstart>[,<count>] ; exit x[<address>] ; ; equates dromaddr equ 0x0000 :default start address for rom 0x0800 d2count equ :default count 0x1000 d4count :default count for 4K equ 0x9380 addrhd equ 0x9381 addrld ;Eprom address registers equ 0x9382 addrhc equ 0x9383 addrlc equ ;eprom address control registers datad 0x9384 ;Eprom data register equ datac 0x9385 equ ;Eprom data control register cntrld 0x9386 ;Programmer control data register equ 0x9387 cntrle equ ;Programmer control control register reset equ 0xfffe ;address of reset vector 0xfcbc redis ;reset display routine equ outbyt 0xfe20 ;display byte routine equ putline equ 0xa009 getline equ 0xa00c wr4 0xa00f equ 0xa012 wr2 equ putchar 0xa015 equ getchar 0xa018 equ 0xa02d getaddr equ 0xa030 tohex equ toascii 0xa033 equ tolower 0xa036 equ 0xa039 ishex equ 1s16 0xa03c egu 0xa03f wait equ ;**************************** ; Command get command line ; collect addresses and check for some command errors ; 0x2000 org command #blurb 1dx putline jsr ;print out welcoming message cmd.warm 1dx #cmd.prmt jsr putline ;the prompt 1dx #buffer jsr getline ;get command line ``` ``` 1dx #buffer clr noargs ; counter for number of arguments ;get command character 1da a,0(x) inx sta a,cmmd isr getaddr get first argument tst ;valid address? я bne cmd.1 ;yes ;no, get next character 1da a,0(x) jmp cmd.99 cmd.1 stx temp1 addr 1dx stx addr1 ;save first address 1dx temp1 inc noargs number of arguments ++ 1da a,0(x) ;next char inx a,#'," ;legal separator cmp cmd.2 beq cmd.99 jmp cmd.2 jsr getaddr ;yes, next address tst ;valid address? cmd.3 bne ;yes cmd.err jmp cmd.3 stx temp1 1dx addr addr2 stx ;second address 1dx temp1 inc noargs 1da a,0(x) inx ;legal separator? cmp cmd.4 beq ;yes cmd.99 jmp cmd.4 jsr getaddr tst ;valid address cmd.5 bne ;yes cmd.err jmp cmd.5 temp1 stx addr 1dx addr3 stx 1dx temp1 inc noargs 1da a,0(x) inx cmd.99 cmp a,#~\n~ ;end of line? beq interp ; yes, have valid command so interpret it cmd.err #errmsg 1dx jsr putline ;output error message ;start again command jmp dc 'arguments error\n\0" errmsg "eprom.2-> \0" cmd.prmt dc Now have a command and arguments, so interpret it interp ``` ``` 1dx reset :default exit stx xaddr 1dx #d2count stx count 1dx #dromaddr stx romaddr ;load default values for 2K a,cmmd 1da get command isr tolower :make lower case cmp a,#~p~ ;program? bne intr.1 ;no isr program bra intr.9 intr.1 cmp a,#~v~ ;verify? bne intr.2 isr verify bra intr.9 a,#'r' intr.2 cmp ;read? intr.3 bne jsr read bra intr.9 a.#'e' intr.3 cmp ;erased? intr.4 bne erased jsr bra intr.9 a.#f intr.4 cmp ;fil1? intr.5 bne jsr fill bra intr.9 a,#'x' intr.5 cmp ;exit? bne intr.6 jmp exit ;exit does not return #cmd.111 intr.6 1dx must be illegal command ;illegal command putline jsr jmp command ;cold restart intr.9 jmp cmd.warm next command cmd.ill dc "illegal command\n\0" dc "\nEPROM programmer, version 2\n" rb dc "Program 2K or 4K EPROMs\n" "Arguments in angle brackets" dc " are hex numbers\n" dc dc "Arguments in square" " brackets are optional\n" dc dc "Commands are:\n" "Program dc p<ramstart>[,<" "count>[,<romstart>]]\n" dc "Verify dc v<ramstart>[,<" dc "count>[,<romstart>]]\n" dс "Read r<ramstart>|,<" dc "count>[,<romstart>]]\n" "Erased? e[<romstart>[" dc ",<count>}]\n" dc "Fill FF dc f<ramstart>[," dc "<count>]\n" "exit dc x[\langle addr \rangle] \n" dc "type 'x' to abort verify " "and erased\n" dc ``` ``` dc "type 'return' to continue " dc "verify and erased\n" dc "Default values are 0800H for " "count, 0000 for romstart\n" dc "and processor reset vector" dc " for x \in 0" dc ; Main Eprom subroutines here "Program" title ;program an eprom ;inputs: noargs number of arguments ;addrl..addr3 initial pointers ;outputs: ;calls: ports, ready, putline A,B,CC,X,memory pointers ;destroys: jsr Ida program ports ;set up ports, data = input. a,#00 sta a,datac lda a,#0xff sta a,datad ;data = output. 1da a,#0x04 sta a,datac ;point to data reg ; check that Eprom is selected and power is on jsr ready 1da a, noargs ; how many arguemants bne psome 1dx #toofew putline jsr rts psome 1dx addrl ;ramstart stx ramaddr dec beq pbegin 1dx addr2 ;count stx count dec beq pbegin 1dx addr3 ;romaddr stx romaddr pbegin 1da a,pinit ;set up for programming eprom sta a,cntrld prpt1 jsr redis 1da a,romaddr jsr outbyt 1da a, romaddr+1 outbyt jsr 1dx romaddr ;get eprom pointer stx addrhd ;address eprom inx stx romaddr ;next location 1dx ramaddr 1da a,0x00(x) ;get data ``` ``` sta a,datad ;output to EPROM inx stx ramaddr ;next memory location ;program byte 1da a,pprog a,cntr1d #6250 sta ;count for 50msec 1dx isr wait 1da a,pinit sta a, cntrld 1dx count dex ;one less to go stx count bne prpt1 ;loop until all positions done 1da a,#0xff sta a, cntrld ;reset program control word #pdone 1dx putline jsr ;say that its done rts "Program completed\n\0" pdone dc "Verify" title ;verify eprom contents ;inputs addrl..addr3, noargs ;outputs none ;calls: putline,wr4,ready, ports, wr2, putchar ;destroys: all verify ;set up ports, data = input jsr ports ; check selected and power on jsr ready 1da a, noargs bne vsome 1dx #toofew jsr putline rts 1dx addr1 vsome ramaddr stx dec bea vbegin 1dx addr2 stx count dec vbegin beq addr3 1dx romaddr ;load required values stx vbegin a, #0x79 1da ;verify led on ;input data ;output enable on ; chip select on ; Vpp off sta a, cntrld romaddr vrpt1 1dx ;get eprom pointer stx addrhd ;address eprom lda a,datad ;get eprom data ramaddr 1dx ``` ``` a,00(x) compare eprom contents with cmp ;memory contents beq pood ;no error #vbad1 1dx jsr putline 1dx romaddr ;which location? jsr wr4 ;print it 1dx #vbad2 putline jsr 1da a,datad ;get bad data ;print it jsr wr2 1dx #vbad3 isr putline 1dx ramaddr 1da a,0(x) ;get proper data jsr wr2 ;print it a,#~\n~ 1da jsr putchar jsr getchar ;wait for any character to resume cmp a,#'x' ;if x then exit beq vend 1dx ramaddr vgood inx ramaddr stx romaddr 1dx inx stx romaddr :next location 1dx count dex stx count bne ;leave loop if all locations vrpt1 ;inspected 1dx #vdone vend putline jsr ;print all well message 1da a,#0xff sta ;reset programmer control byte a,cntrld rts "Verify completed\n\0" vdone dc vbad1 đс "Location \0" " contains \0" vbad2 dc " instead of \0" vbad3 dc title "Check 2K EPROM erased" ;read a 2K EPROM from the EPROM programmer ; check that each location = FFH addr1, addr2, noargs ;inputs: ;outputs: none ;calls: ports, ready, putline, wr4, wr2, putchar, getc ;destroys: erased ports ;set up ports ; check that Eprom is selected and power is on jsr ready 1da b, noargs ;how many arguments? beq ebgn ;none 1dx addr1 ;start address stx romaddr ``` ``` dec b ebgn beq addr2 1dx stx count ;how many ebgn 1da a,#0x79 ;verify led on ;input data ;output enable on ;chip select on ;Vpp off sta a,cntrld ewh2 1dx romaddr stx addrhd ;address eprom ;get data lda a,datad a,#0xff ;should be FF if erased cmp egood beq 1dx #ebad1 putline jsr 1dx romaddr jsr wr4 1dx #ebad2 putline jsr lda a,datad jsr wr2 ;write out error message lda a,#'\n' jsr putchar getchar jsr a,#'x' x to exit cmp ebye beq romaddr egood ldx inx romaddr stx ;next please 1dx count dex stx count bne ewh2 ;more? ldx #ebye1 ebye jsr putline lda a,#0xff sta a, cntrld ;turn off read led rts ebad1 dc "Location \0" " contains \0" đс ebad2 dc "Erasure check completed\n\0" ebye1 title "Read 2K Eprom" ;read a 2K EPROM from the EPROM programmer ;inputs: addrl..addr2, noargs ;outputs: none ;calls: ready, ports, putline ;destroys: all ;set up ports read jsr ports ;check that Eprom is selected and power is on jsr ready 1da a, noargs ;how many arguments? bne rsome ;at least one 1dx #toofew ``` ``` jsr putline rts ;error not enough 1dx addr1 rsome ramaddr stx dec beq rbegin 1dx addr2 count stx dec beg rbegin 1dx addr3 romaddr stx rbegin a,#0xd9 1da ;read led on ;input data ;output enable on ; chip select on ; Vpp off sta a, cntrld 1dx romaddr rwh1 ;get address pointer stx addrhd ;output to Eprom inx romaddr ;next stx lda a, datad ;get Eprom data ramaddr 1dx sta a,0(x) ;dump to ram inx ;next address stx ramaddr ;next dump address 1dx count dex count stx rwh1 ;any more? bne 1da a,#0xff a, cntrld ;clear control register sta 1dx #rdone jsr putline ;say read done rts rdone "Read completed\n\0" dc title "Fill with FF" ;Fill ram with OxFF ;inputs: addrl, addr2, noargs ;outputs: none ;calls: putline ;destroys: A, X, CC fi11 1da a, noargs ;how many arguments? bne fsome 1dx #toofew jsr putline rts fsome addr1 1dx stx ramaddr dec beq fbegin ``` ``` 1dx addr2 stx count fbegin a,#0xff 1da fwh1 ramaddr 1dx sta a,0(x) inx stx ramaddr 1dx count dex stx count bne fwh1 #fdone 1dx isr putline ;say fill done rts fdone "Fill completed\n\0" dc. exit ; exit to monitor or user program addr - exit address ;inputs: ;outputs: ;calls: putline, wr4, putchar, wait :destroys: al1 1da a, noargs beq xnone ;use default exit addrl 1dx xaddr stx ;use user exit address xnone #exitmsg 1dx jsr putline 1dx xaddr isr wr4 ;write out message 1da a,#~\n~ putchar jsr 1dx #6520 ;count for 50 msec wait jsr 1dx xaddr ;finally exit jmp 0(x) "branching to address \0" exitmsg dc "misc programmer routines" title ports ;set up ports ;inputs: none ;outputs: none ;calls: nothing ;destroys: A, X, CC a,#0x04 1da sta a, cntrlc ;point to data part of programmer ;conrol register 1da a,#0xff sta a, cntrld ;initial control values all 1's 1da a,#00 sta a,addrlc sta a,addrhc sta a,datac sta a, cntrlc ;point to ddr of all ports #0x1fff ;ddr for address port 1dx addrhd stx ;data port all input sta a,datad ``` ``` 1da a,#0xef ;1 bit in, 7 bits out sta a, cntrld 1da a, #0x04 a,addrhc sta a,addrlc sta a,datac sta ;point to data register of all ports sta a, cntrlc rts ;Ready ;loop until both 2K or 4K selected and power on ;inputs: none ;outputs: none ;calls: nothing ;destroys: A, CC a,#0xa5 ready ;assume have 2K eprom 1da ;set up initial value to be used by ;program routine ;write led on, output data, G=1, E=0, ;Vpp off a,pinit sta a,#0xa6 1da ; value to be used for programming ;E <-- 1, Vpp <-- on sta a,pprog ready1 1da a,cntrld a,#0x10 and ;check power bne readyl 1da a,addrhd ;get 2k, 4k, 8k switch settings a, #0x20 ;2K? and ready2 beq ;yes lda a,addrhd a,#0x40 and ;4K? bne ready1 ;neither so wait ;4K so replace pinit and pprog with 4K values ;init value for 4K eprom 1da a,#0xa3 ; write led on, data output, G=x, E=1 ;Vpp off sta a,pinit 1da a,#0xa0 ;program value ;E <-- 0, Vpp <-- on sta a,pprog 1dx #d4count ;default count for 4K count stx 1dx #r4kmsg putline jsr ready2 rts r4kmsg "4k eprom selected\n\0" dc toofew "not enough arguments\n\0" đс seg variable storage area ; addr 0x102 ;return value from getaddr equ 0x108 temp1 equ addrl equ 0x10a addr2 0x10c equ addr3 0x10e equ ``` | ramaddr | | • | 0x110 | | | | | | | | | | |--------------|----------|----------|-------|------------------|------|----|-------|------|-------|------|------|------| | count | | equ | 0x112 | | | | | | | | | | | xaddr | | equ | 0x114 | | | | | | | | | | | romaddr | | equ | 0x116 | | | | | | | | | | | cmmd | | equ | 0x118 | | | | | | | | | | | noargs | | | 0x119 | | | | | | | | | | | pinit | | | 0x11a | | | | | | | | | | | pprog | | - | 0x11b | | | | | | | | | | | buffer | | equ | !+1 | | | | | | | | | | | | ing for | EPROM Pr | | • | | | | | | | | | | ;<br>address | port | | bit | directio | n | | use | | DDR | ini | tial | data | | , | 1.4 | | ~ | , | | | Orr | | ^ | ** | | | | - | data reg | 3 | 7 | < | | | 8K | | 0 | X | | | | ; | (A1) | _ | 6 | < | | | 4K | | 0 | x | | | | - | high ord | ler | 5 | < | | | 2K | | 0 | x | | | | ; | address | | 4 | > | | | A12 | | 1 | 0 | | | | ; | | | 3 | > | | | A11 | | 1 | 0 | | | | ; | | | 2 | > | | | A10 | | 1 | 0 | | | | ; | | | 1 | > | | | A9 | | 1 | 0 | | • | | ; | | | 0 | > | | | A8 | | 1 | 0 | | | | | data reg | ξ | 7 | > | | | A7 | | 1 | 0 | | | | | (B1) | | 6 | > | | | A6 | | 1 | 0 | | | | ; | low orde | r | 5 | ><br>><br>><br>> | | | A5 | | ĩ | ŏ | | | | | address | | 4 | > | | | A4 | | 1 | Ō | | | | ; | address | | 3 | | | | A3 | | î | Ö | | | | ; | | | | / | | | | | | | | | | ; | | | 2 | / | | | A2 | | 1 | 0 | | | | ; | | | 1 | | | | A1 | | 1 | 0 | | | | ; | | | 0 | > | | | A0 | | 1 | 0 | | | | ;9382 | control | reg | 7 | | | | | | | | | | | ; | (A1) | | 6 | | | | | | | | | | | ; | | | 5 | | | | | | | | | | | ; | | • | 4 | | 00H | to | load | DDR | | | | | | ; | | | 3 | | 04H | to | 1oad | addr | esses | | | | | ; | | | 2 | | | | | | | | | | | ; | | | 1 | | | | | | | | | | | | | | 0 | | | | | | | | | | | ;<br>;9383 | control | rec | 7 | | | | | | | | | | | | (B1) | reg | 6 | | | | | | | | | | | ; | (21) | | | | | | | | | | | | | ; | | | 5 | | ΛΩττ | +- | load | מממ | | | | | | • | | | 4 | | | | | | 4.4. | | | | | ; | | | 3 | | 04H | τo | trans | rer | aata | | | | | ; | | | 2 | | | | | | | | | | | ; | | | 1 | | | | | | | | | | | <b>;</b> | | | 0 | | | | | | | | | | | ;9384 | data reg | 3 | 7 | <> | | | D7 | | 0 | 1 | | | | ; | (A2) | - | 6 | <> | | | D6 | | 0 | 1 | | | | : | EPROM da | ata | 5 | <> | | | D5 | | 0 | 1 | | | | : | | | 4 | <> | | | D4 | | 0 | 1 | | | | • | | | 3 | <> | | | D3 | | 0 | 1 | | | | • | | | 2 | <> | | | D2 | | Ö | 1 | | | | , | | | | | | • | | | 0 | | | | | į | | | 1 | <> | | | D1 | | | 1 | | | | ; | | | 0 | <> | | | DO | | 0 | 1 | | | | ;9385 | control | reg | 7 | | | | | | | | | | | ; | (A2) | | 6 | | | | | | | _ | | | | ; | | | 5 | • | | | DDR | ≈ FI | H for | data | writ | e | | | | | | | | | | | | | | | ``` 4 DDR = 00H for data read 3 00H to load DDR ; 2 04H to transfer data 1 0 ;9386 7 data reg ~verify l 1 6 ~write 1 (B) 1 5 command/ 1 ~read 1 4 status ~En 0 х 3 ~drn 1 1 2 ~G 1 1 ~Ē 1 0 ~Vppon 1 1 ;9387 7 control reg 6 (B2) 5 4 00H to load DDR 3 04H to transfer commands/status 2 1 ;note: Initial data for the command/status register should be loaded ;before the I/O bits are set to output. --> means into EPROM <-- means out of EPROM <--> means bidirectional data flow ```