# **NICK HAMPSHIRE** WITH RICHARD FRANKLIN AND CARL GRAHAM # THE COMMODORE # The Commodore 64 Kernal and Hardware Revealed #### Also by Nick Hampshire The Commodore 64 ROMs Revealed 0 00 383087 X Advanced Commodore 64 Graphics and Sound 0 00 383089 6 Advanced Commodore 64 BASIC Revealed 0 00 383088 8 The Commodore 64 Disk Drive Revealed 0 00 383091 8 # The Commodore 64 Kernal and Hardware Revealed ## Nick Hampshire with Richard Franklin and Carl Graham Collins Professional and Technical Books William Collins Sons & Co. Ltd 8 Grafton Street, London WIX 3LA First published in Great Britain by Collins Professional and Technical Books 1985 Copyright © Nick Hampshire 1985 British Library Cataloguing in Publication Data Hampshire, Nick The Commodore 64 kernal and hardware revealed. 1. Commodore 64 (Computer) I. Title II. Franklin, Richard III. Graham, Carl 001.64'04 QA76.8.C64 ISBN 0-00-383090-X Typeset by V & M Graphics Ltd, Aylesbury, Bucks Printed and bound in Great Britain by Mackays of Chatham, Kent All rights reserved. No part of this publication may be reproduced, stored in a retrieval system or transmitted, in any form, or by any means, electronic, mechanical, photocopying, recording or otherwise, without the prior permission of the publishers. #### Also by Nick Hampshire The Commodore 64 ROMs Revealed 0 00 383087 X Advanced Commodore 64 Graphics and Sound 0 00 383089 6 Advanced Commodore 64 BASIC Revealed 0 00 383088 8 The Commodore 64 Disk Drive Revealed 0 00 383091 8 # Contents | Pi | reface | vii | |----|------------------------------------|-----| | 1 | Inside the Commodore 64 | 1 | | 2 | The Keyboard, Joysticks and Screen | 15 | | 3 | Serial Communications | 32 | | 4 | The Cassette Units | 78 | | 5 | The User Port | 127 | | 6 | Interrupts and Their Use | 184 | | In | dex | 193 | ## Preface Whether you program the CBM 64 in Basic or machine code, an understanding of how the kernal software works and how the system hardware functions is essential before writing many programs, particularly those involving connecting the CBM 64 to external devices. This book looks at the way the operating system and system hardware work and should be used in conjunction with Volume 1 of this series, *The Commodore 64 ROMs Revealed*, which gives the entire operating system kernal software source code. A knowledge of how the operating system and hardware work enables one to perform many interesting functions. Notable amongst these is the high speed tape load and save routines, which allow the tape deck to operate at speeds equivalent to that on a 1541 disk drive. The whole area of program security is also covered. Without an understanding of the system software and hardware, the operation and use of the serial and RS232 ports is often quite mysterious. This book is the product of many year's work on Commodore machines, and I am confident that it provides the most complete and useful set of information available from any one source. All serious programmers should find this an invaluable and constant reference book. Nick Hampshire ## Chapter One # Inside the Commodore 64 Fig. 1.1. Commodore 64 memory architecture map. #### 1.1 Commodore 64 design concept The division of memory space into blocks in the Commodore 64 is shown in Fig.1.1. The Commodore 64 is built around the VIC chip and the 6510 microprocessor. As with all microcomputers the Commodore 64 is designed to use the minimum number of chips. This is to reduce component and assembly costs. Fig. 1.2. Commodore 64 schematics (Reproduced by courtesy of Commodore Business Machines (UK) Ltd). The main computer circuit (see Fig. 1.2) consists of the 6510, the VIC chip with clock circuit and the eight 64K bit RAM chips. These chips are enough to make a computer circuit that can perform machine code instructions. Note that the VIC chip is required as it supplies most of the computer's RAM control signals and timing. However, to make a usable personal computer the 64 needs a few chips for I/O and sound, also a Basic interpreter and operating system in ROM. #### 4 The Commodore 64 Kernal and Hardware Revealed The required input/output is supplied by two 6526 CIA chips. Together these two chips supply 4 timers used for IRQ timing, the tape system and serial. Their I/O ports are used for keyboard scanning, the user port, serial ports and VIC chip bank select. These chips also have serial ports and time of day clocks but these are not used in the CBM 64. Lastly sound is generated by the SID chip. This has 3 voices each with 4 waveforms an envelope control and filters. The 2 channel analog to digital converter on this chip is connected to the joystick ports. #### 1.2 Chips in the Commodore 64 The Commodore 64 consists of a plastic case holding a double sided printed Fig. 1.3. Printed circuit board assembly for the Commodore 64 (Reproduced by courtesy of Commodore Business Machines (UK) Ltd). circuit board, a 66 key keyboard and assorted plugs and sockets, not forgetting the power switch and LED. #### 1.2.1 The PC board The printed circuit board (see Fig. 1.3) has soldered onto one side of it, a few hundred assorted resistors, capacitors, ferrite beads, diodes, coils, 2 voltage regulators, a fuse and 31 integrated circuits (chips). The power supply box supplies the board with 5 volt dc regulated, 9 volt ac and a shielded ground line. The 5 V line is used to power most of the main chips but not the VIC chip or the clock circuit. The 5 V and 12 V for these chips is generated on the board using diodes and voltage regulators to convert the 9 volt ac supplied. This is done to limit interference from the VIC chip and clock circuit which are in a shielded can. (Do not run the computer with this can open as the lid of this provides the heat sink for the VIC chip.) #### 1.3 The main chips #### 1.3.1 651 microprocessor (MPU) The microprocessor (Fig. 1.4) is a version of the very common 6502. The main difference is that the 6510 has a 6 pin I/O port. In the CBM 64 this port is used for controlling memory configurations through the PLA chip, controlling output lines to the tape deck and sensing keys being pressed on the tape deck. Fig. 1.4. 6510 microprocessor (MPU). | , | • | • | , | u | v | v | , | • | , | , | , | |---|---|---|---|---|---|---|---|---|---|---|---| | | | | | | | | | | | | | | | | | | | | | | | | | | 651Ø Signals and Lines | Pin 1 | φ <b>Ø</b> in | Clock in (from VIC chip) | |------------|------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | Pin 2 | RDY | Ready. Processor waits in current state while this line is low. If this line is low and interrupts are enabled at the end of the current instruction cycle then an interrupt will be initiated | | Pin 4 | NMI | Non maskable interrupt (negative edge sensitive input). When this line goes from high to low an interrupt will be initiated at the end of the current instruction | | Pin 5 | AEC | Address bus enable control. When this line goes low the processor frees the address bus for use by other chips (VIC in the 64) | | Pin 6 | Vcc | Supply voltage (+5 V) | | Pins 7-20 | | | | & 22,23 | AØ-A15 | Address bus (enabled by AEC) | | Pin 21 | GND | Ground (Ø V) | | Pins 24-29 | P5-PØ | Processor I/O port | | Pins 30-37 | D7-DØ | Data bus | | Pin 38 | $R/\overline{W}$ | Read/write. (Output: low flags for processor write.) | | Pin 39 | $\phi$ 2in | Phase 2 clock input | | Pin 4Ø | RES | Reset (Active low) | | | | | #### 1.3.2 6526 complex interface adapter (CIA) There are two of these chips in the Commodore 64. The first CIA#1 is used for Fig. 1.5. 6526 complex interface adapter (CIA). scanning the keyboard and inputting from the cassette and serial port. The second CIA#2 supports the user port and most of the other serial port lines. It also is used as a latch for the VIC chip bank select value. CIA chip 2 has its IRQ output connected to the NMI line so that its timers can generate NMIs instead of IRQs. The chip select lines for the two chips are decoded from the processor address bus by the PLA chip to addresses \$DC00 & \$DD00 (Fig. 1.5). | 6526 Signals and L | ines | |--------------------|------| |--------------------|------| | Pin 1 | GND | Ground (Ø V) | |------------|------------------|---------------------------------------------------------| | Pins 2-9 | PAØ-PA7 | Data port A (Bi-directional data port) | | Pins 10-17 | PBØ-PB7 | Data port B (Bi-directional data port) | | Pin 18 | $\overline{PC}$ | Handshaking output for port B | | Pin 19 | TOD | Clock input for TOD clock | | Pin 2Ø | Vcc | Supply voltage (5 V) | | Pin 21 | ĪRQ | Interrupt request output | | Pin 22 | $R/\overline{W}$ | Read/write. Input from processor (low for processor | | | | write) | | Pin 23 | CS | Chip select. Low indicates a processor read or write to | | | | the CIA | | Pin 24 | FLAG | Negative edge sensing input | | Pin 25 | $\phi 2$ | Phase 2 clock input | | Pins 26-33 | D7-DØ | Data bus (Bi-directional depending on R/W) | | Pin 34 | RES | Reset (Active is low) | | Pins 35-38 | RS3-RSØ | Register select. Connected to the lower order address | | | | lines to one of the 16 registers | | Pin 39 | SP | Serial input. Not used for Commodore serial bus | | Pin 4Ø | CNT | Pulse counter, serial clock. Not used for serial bus | | | | | Fig. 1.6. 6581 sound interface device (SID). #### 1.3.3 6581 sound interface device (SID) SID is a music/sound effects generator for computer games. Its output goes to the modulator and audio/video socket, and has a sound input from a pin on this socket (see Fig. 1.6). | SID Signals and | d Lines | | |-----------------|------------------|-------------------------------------------------| | Pins 1 & 2 | CAPla-b | Filter capacitor | | Pins 3 & 4 | | Second filter capacitor | | Pin 7 | $R/\overline{W}$ | Read/write | | Pin 8 | CS | Chip select. Decoded from address bus by PLA to | | | | \$D400 | | Pins 9-13 | AØ-A4 | Register select | | Pin 14 | GND | Ground (Ø V) | | Pins 15-22 | DØ-D7 | Data bus | | Pin 23 | POT Y | Analog input for A/D converter | | Pin 24 | POT X | Analog input for second A/D | | Pin 25 | Vcc | Supply voltage 5 V | | Pin 26 | EXT in | Audio input | | Pin 27 | AUDIO | Audio output | | Pin 28 | Vdd | Supply voltage 12 V | #### 1.3.4 6567-9 video interface chip (VIC) This video display generator chip (Fig. 1.7) also produces most of the internal timing and control signals for the CBM 64, including the processor clock. VIC generates its own address bus like the 651 $\emptyset$ . This is used to fetch display data from RAM and character ROM, but since the computer cannot have two completely separate address and data bus systems, VIC and the processor have to share them. 65xx series processors use the system buses only during phase 2 of the clock cycle. The VIC chip takes advantage of this and uses phase 1 of the clock ( $\phi$ 2 low $\phi$ $\emptyset$ high). This chip has been given a higher internal bus priority than the 651 $\emptyset$ processor. VIC can disable the 651 $\emptyset$ and free the address bus for its own use during phase 2 by sending the lines AEC & BA low. The AEC line disables the 651 $\emptyset$ address drivers so that its own can drive the address bus. The VIC chip can send the AEC line low during phase 1 and use the address bus without interfering with the processor's operation. The line BA is connected to the 651 $\emptyset$ 's RDY (ready) pin. This can be set low during phase 1 and then held low causing the 651 $\emptyset$ to pause at the end of its next read cycle. (This is ignored during 651 $\emptyset$ write operations. VIC accessing memory with R/ $\overline{W}$ low would not be desirable anyway!). BA will go low three cycles before AEC is used in phase 2. This ensures that all write operations have finished and avoids conflict with DMA (direct memory access) from any cartridge port device (Z8 $\emptyset$ card). VIC also refreshes the dynamic RAM chips using its RAS line and its lower order address bus during phase 1. , Fig. 1.7. Video interface chip (VIC II). #### VIC Signals and Lines | Pins 1-6 | D6-DØ & | Data bus | |------------|------------------|------------------------------------------------------| | & 35-39 | D11-D7 | DØ-D7 are bi-directional and are used for | | | | register access and VIC memory fetches | | | | D8-D11 are used for reading colour RAM | | Pin 8 | ĪRQ | Interrupt request output | | Pin 9 | LP | Light pen input | | Pin 1Ø | <del>cs</del> _ | Chip select | | Pin 11 | $R/\overline{W}$ | Read/write | | Pin 12 | BA | Bus available | | Pin 13 | Vdd | Supply voltage +12 V | | Pin 14 | Colour | Colour output | | Pin 15 | S/LUM | Sync/luminance | | Pin 16 | AEC | Address bus enable | | Pin 17 | $\phi \emptyset$ | Phase one clock out | | Pin 18 | RAS | Row address select. Dynamic RAM control signal, | | | | used for low order of multiplexed address and for | | | | refreshing | | Pin 19 | CAS | Column address select. Dynamic RAM control signal | | | | for high order address | | Pin 2Ø | Vss | Ground (Ø V) | | Pin 21 | $\phi$ colour | Colour clock in 14–18 MHz | | Pin 22 | $\phi$ in | Clock in 8 MHz | | Pins 23 | A11 & | | | & 3Ø-34 | A6-A1Ø | High order address output | | Pins 24-29 | AØ/A8- | | | | A5/A13 | Address lines AØ-A13 multiplexed together. Gives | | | | address for VIC for memory fetches in output mode or | | | | register select in input mode | | Pin 4Ø | Vcc | Supply voltage 5 V | #### 12 The Commodore 64 Kernal and Hardware Revealed #### 1.3.5 Programmable logic array (PLA) This is an array of logic gates programmed together at the time of manufacture to give most of the required logic circuits of the 64 (see Fig. 1.8). The chip has 16 inputs and 8 outputs. A very complicated logic table relates the outputs to the inputs. The pin names of this chip are 10-115 and 10-115 and 10-115 and 10-115 are 10-115 and 10-115 are 10-115 and 10-115 are 10-115 and 10-115 are 10-115 are 10-115 are 10-115 are 10-115 and 10-115 are 10-115 and 10-115 are 10-115 are 10-115 are 10-115 are 10-115 are 10-115 and 10-115 are 10-115 and 10-115 are and 10-115 are and 10-115 are and 10-115 are 10-11 Fig. 1.8. Programmable logic array (PLA). The other main chips in the 64 are the RAM and ROM chips. The RAM comprises eight 4164 dynamic RAM chips. One chip gives 1 by 64K bits of memory, so the eight chips give 8 by 64K bits or 64K bytes. The 4164 has 8 address lines. The 16 bit address is multiplexed in (low byte first) and timed with the RAS and CAS lines. The main improvement with this chip over older dynamic RAMs is that it requires only a single 5 V supply instead of -5,+5 and +12 V for a 4116. The colour memory RAM is a 2114 chip; this is a 4 by 1024 bit static RAM. The Basic and kernal ROMs are 8 bit by 8K ROMs. The character ROM is an 8 by 4K. These chips are Commodore's own manufacture and type. The 6510, 6526 CIA, VIC and SID chips are all manufactured by Commodore's chip manufacturing subsidiary, MOS Technology Inc. It is unfortunate that none of these devices appears to be second sourced and consequently replacements are either very difficult or impossible to obtain. #### 1.4 System logic and timing Like all computers the Commodore 64 is a group of chips linked together by address and data buses. The main chips which are connected to the data and address buses are instructed to send to, take from or ignore the data bus by the control system lines. There are, in addition, lines controlling the use of the address bus. These control lines are defined as follows: #### 1.4.1 Clock lines #### φ colour clock This is the colour clock used by the VIC chip for generating colour signals. It is divided by part of the clock circuit and used as a reference for producing the VIC chip's dot clock. #### Dot clock This signal produced by the clock circuit is the clock input to the VIC chip. VIC uses this as the timing for producing pixels on the sceen. Also VIC divides this signal by 8 and supplies it as the processor phase zero clock. #### Phase 2 processor clock $\phi$ 2 This clock line controls all 6510 read and write operations. It is produced by the 6510 from the VIC chip's $\phi$ 0 line. The 65xx series processors require only the system buses while this line is high (5 V). #### 1.4.2 Main system control signals #### Read/Write R/W If this line is low when a byte of memory or I/O device register is selected by the address bus, then the contents of the data bus will be transferred to the selected byte or register. If the line is high then the contents of the selected address are transferred onto the data bus. #### Reset RES This line is connected to all the main chips including the processor. On machine power up this line is held low for a few clock cycles to ensure the supply voltages have stabilised. This holds all chips in their reset state until they are ready. #### Ready RDY RDY is a processor input which, if low, causes the 6510 to pause at the end of the read cycle. It is used with the AEC line to disable the processor during phase 2 clock cycles for direct memory access. #### Interrupt request IRQ When this line is low it signals that one or more of the CIAs or VIC is requesting an interrupt service. #### Non maskable interrupt NMI When this line goes from high to low the processor will be interrupted at the end of the current instruction cycle. Only a change from high to low will cause an interrupt, so if this line is held low after an NMI it will disable future NMIs. #### Bus available BA When this line is low it flags that the VIC chip needs the system buses during phase 2. It disables the processor via the RDY line. ### RAM control signals CAS RAS & CASRAM The 4164 dynamic RAM chips have their 16 bit addresses fed to in two lots of 8 #### **14** The Commodore 64 Kernal and Hardware Revealed bits. This is because the 16 pin chip has only an 8 bit address bus. RAS, the row address and CAS, the column address are used to strobe in the low and high bytes. In the 64, CAS and RAM chip select are combined into CASRAM, so when this line is low it flags the high byte of address on the chip's address pins and the chip is selected for read or write. ### Chapter Two # The Keyboard, Joysticks and Screen #### 2.1 Keyboard #### 2.1.1 Keyboard hardware and software operation The CBM 64 keyboard has a total of 66 keys, the layout of which is shown in Fig. 2.1. These 66 keys can be divided as follows: Fig. 2.1. The keyboard layout. - 1. RESTORE key; this is connected directly to the NMI line. - 2. The left shift key and the shift lock are connected together. - 3. All other 63 keys. The main section of the keyboard thus has a total of 64 keys. These are organised electrically as an 8 by 8 matrix. The keyboard scanning is performed by the operating system software. The matrix is organised such that the columns are set as outputs by the scanning routine and the rows return a value if a key is pressed. These 8 row inputs and 8 column outputs from the keyboard matrix are connected to the computer via the CIA#1 I/O chip, where the output is via address \$DCØ0 and the input is via address \$DCØ1. The scanning routine loops through 8 times, and each time sends a different line on the output to a low state. It then reads the input port connected to the matrix row lines, which will return values for 8 keys (each key takes up one bit, Ø=down, l=up). Therefore, looping 8 times through the scanning routine will look at each of the columns and return all keys which are pressed. The keyboard is laid out as follows: | Row # | | Column # (output) | | | | | | | | |---------|-----|-------------------|---|---|---|---|------|------------|--| | (input) | Ø | 1 | 2 | 3 | 4 | 5 | 6 | 7 | | | Ø | DEL | 3 | 5 | 7 | 9 | + | £ | 1 | | | 1 | RET | W | R | Y | I | P | * | - | | | 2 | - | Α | D | G | J | L | ; | CTRL | | | 3 | F7 | 4 | 6 | 8 | Ø | _ | CLR | 2 | | | 4 | Fl | Z | C | В | M | | R.SH | SPACE | | | 5 | F3 | S | F | Н | K | : | = | <b>G</b> × | | | 6 | F5 | E | T | U | Ο | @ | † | Q | | | 7 | 1 | L.SH | X | V | N | , | / | STOP | | The scanning of the keyboard matrix, and the testing for depression of the RESTORE key, are all under software control. The entire processor time cannot be devoted to keyboard scanning, therefore scanning is initiated by a regular 1/60 second interrupt. Keyboard scanning is one of the functions of the IRQ interrupt servicing routine. The 1/60 second regular interrupt is generated by Timer A of CIA#1. The interrupt service routine starts at location \$EA31 and the keyboard scanning portion at \$EA87. The keyboard scanning routine goes through a sequence of operations, the result of which is to place each input character into a special section of memory; the keyboard buffer. The sequence is as follows: - 1. Check if key pressed; if not then exit from routine. - 2. Initialise I/O ports of CIA#1 for keyboard scan and set pointers into keyboard character table 1. Set character counter to $\emptyset$ . - 3. Set one line of port A low and test for character input on port B by performing eight right shifts of the contents of port B register; if carry is clear then key present. Each shift increments key count; store key count in .Y. - 4. Go back to step 3 and repeat for next column; if key found then continue. - 5. Use key count value as index pointer into keyboard character table to get ASCII code corresponding to depressed key. - 6. See if it is SHIFT or STOP key. - 7. Evaluate shift function - If SHIFT key then use table 2 - If CBM key then use table 3 - If CONTROL key then use table 4 - 8. Use key count value as index pointer into keyboard character table designated in step 7. - 9. Check for repeat key operation. - 10. Do repeat if required. - 11. Put ASCII character obtained from keyboard character tables into the keyboard buffer; increment the pointer into the keyboard buffer. The contents of the 10 character keyboard buffer are accessed on a first in first out basis by the INPUT and GET character routines. These routines take the first character in the keyboard buffer, decrement the buffer pointer and close up the buffer by moving the contents down one byte thereby leaving space for new input characters. The characters put into the keyboard buffer are removed by either the INPUT or GET kernal routines. Both these routines call a subroutine at \$E5D4 which removes the first character and puts it in register. Y then moves the whole buffer down by one byte. This routine is only called if at least one character is in the buffer. Warning: Do not call the routine at \$E5D4 when there are no characters in the keyboard buffer as this will crash the computer. The GET character routine which is accessed by the kernal jumpblock at location \$FFE4 (vectored at \$\000932A) will return the Commodore ASCII code of the next character in the keyboard buffer in register .A. If no character was present, the value of zero is returned. The INPUT routine, when called, will set the cursor flashing and will input characters from the keyboard buffer until a carriage return is found. Each character received is printed to the screen using the routine at \$E716 and when a carriage return is found, the routine inputs the first character on the line from the screen and returns it in register .A. Subsequent calls to this routine will return one character at a time until they have all been returned. At this point, if the ASCII value of SHIFT/STOP is found, the LOAD/RUN combination is stored to the buffer replacing all characters following it. The routine is accessed via the kernal jumpblock at location \$FFCF (vectored at \$\0324). A Basic program to emulate the keyboard scanning routine is given in Program 1. ``` 1000 REM KEYBOARD SCAN SIMULATION PROGRAM 1020 REM 1030 REM THIS BASIC PROGRAM SIMULATES 1040 REM THE IRQ SCANNING ROUTINE WITH 1050 REM A FULL SCREEN DISPLAY OF WHAT 1060 REM IS HAPPENING. THE ROUTINE FIRST 1070 REM WAITS FOR A KEY TO BE PRESSED 1080 REM AND THEN SCANS THROUGH TO PICK UP THE KEY(S). ANY KEY PRESSED 1090 REM 1100 REM WILL BE DISPLAYED AS A REVERSE 1110 REM KEY IN THE BOX LABELLED 'KEY'. 1120 REM 1130 REM YOU MUST HOLD DOWN A KEY UNTIL 1140 REM IT IS RECOGNISED. 1150 REM 1160 REM A KEYBOARD BUFFER IS KEPT AND 1170 REM THE ROUTINE WILL EXIT WHEN 1180 REM EITHER THE RETURN KEY IS FOUND 1190 REM OR WHEN THERE ARE TEN CHARACTERS 1200 REM IN THE BUFFER. 1210 REM 1230 DIM K$(71):FORI=0T071:READK$(]):NEXT 1240 DIM K(3,64):FORJ=0T02:FORI=0T064:K(J,I)=PEEK(60289+J*65+I):NEXTI,J 1250 FORI=0T064:K(3,I)=PEEK(60536+I):NEXTI 1260 PRINT"四 KEYBOARD SCAN SIMULATION" 1270 PRINT"M OUTPUT: $DC00 INPUT $DC01" 1280 PRINT"X BIT: 76543210 76543210" BIT : 1290 PRINT" 1 14 1300 PRINT" 1310 PRINT" 110 | 10 1320 PRINT" KEY : I 1330 FORI=1T04:PRINTP$;:FORJ=1T0I:PRINT")#";:NEXT ``` ``` 1360 PRINT"SMANIAMANACURRENT SHIFT : 0" 1370 PRINT"CURRENT KEY : 64" 1380 PRINT"% KEY BUFFER" 1390 GOSUB 1990 1400 POKE 56333,1:FORI=0T07:P2(I)=2fI:NEXT 1410 KT=0:C3=0:CK=64:VL=8:A=255:GOSUB1700 1420 Y=0:POKE56320, V:GOSUB1660: A=PEEK(56321) 1430 IF R=255 THEN 1420 1440 GOSUB 1700 1450 V=254:FORVL=0T07 1460 POKE 56320, V: GOSUB 1660 DISPLAY OUTNEW 1470 A=PEEK(56321):GOSUB 1700 DISPLAY INPUT 1480 GOSUB 1860 CHECK FOR ANY NEWS DOWN 1490 V=((V*2)AND255)+1:NEXT 1500 IF CS<>3 THEN 1550 1510 IF (PEEK (53248+24) AND 2)=2 THEN 1540 1520 POKE 53248+24, PEEK (53248+24) OR2 1530 GOT01560 1540 POKE 53248+24, PEEK (53248+24) AND 253 1550 KT=CS: IFKT>3THENKT=3 1560 CC=K(KT,CK) 1570 IF CC=255 THEN 1650 1580 KB(BP)=CC:BP=BP+1 1590 IF(BP<>10)AND(CC<>13)THEN 1640 1600 FOR I=0TOBP-1:POKE 631+I,KB(I):NEXT 1610 POKE198, BP: POKE 56333,129 1620 PRINT" SUMMERS AND ADDRESS OF THE STATE 1630 END 1640 GOSUB 1990 1650 GOTO1360 1660 REM 1670 REM DISPLAY OUTPUT VALUE BITWISE 1680 REM 1700 REM 1710 REM DISPLAY INPUT VALUE BITWISE 1720 REM 1740 FORI=7TO0STEP-1:PRINTP$; 1750 FORJ=7T0ISTEP-1:PRINT"腳;:NEXTJ 1760 C=VL*8+I:FORJ=1T05 ",J,1> 1770 C$=MID$(K$(C)+" 1780 PRINT" P";: IF(A AND P2(I))=0 THEN PRINT" B"; 1790 PRINTC$" 週間;:NEXT:NEXT:PRINT"9";:RETURN 1300 REM 1810 REM DISPLAY VALUE IN A BITWISE 1820 REM 1830 FOR I=7 TO 0 STEP -1 1840 C$="0": IF A AND P2(I) THEN C$="1" 1850 PRINT" : C * " " ; : NEXT : RETURN 1860 REM 1870 REM CHECK FOR A KEY FROM THIS COLUMN 1880 REM 1890 FOR X=0 TO 7 1900 IF(A AND P2(X))=0 THEN 1950 1910 NEXT 1920 PRINT"到现现现现现现现现现现的原因原因原因原因原因原因的"; CS 1930 PRINT TRUMBULAR TO PRINT TO THE TRUE TRUE TO THE TRUE TRUE TR 1940 RETURN 1950 IN=K(KT,X+VL*8) 1960 IF IN>4 THEN CK=X+VL*8:GOT01910 1970 IF IN=3 THEN CK=X+VL*8:GOT01910 1980 CS=CSORIN: GOTO1910 1990 REM 2000 REM DISPLAY KEYBOARD BUFFER 2010 REM 2030 FOR I=0 TO 9 2040 PRINT"p即即回回回回回回回回回回";KB(I):NEXT ``` ``` 2060 IF BP=0 THEN 2080 2090 RETURN 2100 REM REM MNEMONICS FOR KEY PRESS 2110 2120 REM 2130 DATA DEL, RET, C.RT, F7, F1, F3, F5, C. DN 2140 DATA 3,W,A,4,Z,S,E,L.SH 2150 DATA 5.R.D.6.C.F.T.X 2160 DATA 7, Y, G, 8, B, H, U, Y 2170 DATA 9.I.J.0.M.K.O.N 2180 DATA +,P,L,-,.,":",@,"," 2190 DATA \,*,";",CLR,R.SH,=,↑,/ 2200 DATA 1,+,CTRL,2,SPC,CBM,Q,STOP 2210 DATA Program 1. ``` #### 2.1.2 Modification of keyboard operation Program 2 shows how a wedge can be made into the INPUT routine to give the function keys text definition. This follows the same principle as the expansion of the SHIFT/STOP character, with the exception that subsequent characters are not lost. Each function key definition can be up to 255 characters long and the definitions are stored behind the Basic ROM. ``` 033C ! FUNCTION KEYS FOR THE 64. 033C 033C EACH KEY CAN HAVE A MAXIMUM DEFINITION OF 255 CHARACTERS LONG (MAXIMUM BASIC STRING LENGTH) 033C 033C 033C ! DEFINE A FUNCTION KEY USING: 933C 033C !SYS 49163,N,DEFINITION$ 033C ! INITIALISE FUNCTION KEYS WITH: 033C Ø33C !SYS 49152. Ø33C C000 *=$C000 C000 A963 LDA #<FUNCTION C002 8D2403 STA $9324 C005 A9C0 LDA #>FUNCTION C007 8D2503 STA $0325 C00A 60 RTS C00B C00B 20F1B7 DEFINE JSR $B7F1 !GET KEY# C00E E009 CPX #$09 !LESS THAN 8? C010 9005 BCC DEF1 !YES C012 A20E ERROR LDX #$9E !ILLEGAL QUANTITY C014 6C0003 JMP ($0300) !SEND ERROR CØ17 C017 E000 DEF1 CPX #$00 !IS IT ZERO? C019 F0F7 BEQ ERROR !YES C01B CA C01C 8A DEX TXA C01D 48 PHA C01E 20FDRE JSR SAEFD C021 209EAD JSR $AD9E !GET STRING JSR $B6A3 !DISCARD STRING C024 20A3B6 C027 8D51C0 STR STLEN C02A 68 PLA ASL A C02B 0A CØ2C AA TAX C02D BD53C0 LDA POINT X C030 8524 STR $24 LDA POINT+1,X C032 BD54C0 ``` ``` C035 8525 C037 A000 C039 B122 DEF2 STA $25 LDY #$00 LDA ($22),Y C03B 9124 C03D C8 STA ($24), Y INY CPY STLEN BEQ DEF4 CPY #$00 BNE DEF2 !END OF STRING? !YES !END OF ROOM? !NOT YET C03E CC51C0 C041 F005 C043 C000 C045 D0F2 C047 60 DEF3 C048 C000 DEF4 RTS !STRING LENGTH=256? CPY #$00 CPY #$00 !STRING LENGTH=25 BEQ DEF3 !YES LDA #$00 !ZERO TERMINATOR STA ($24),Y !STORE IT C04A F0FB C04C A900 C04E 9124 C050 60 RTS C051 ! C051 00 STLEN BYT 0 C052 00 FUNCFLG BYT 0 CØ53 C063 C063 A599 FUNCTION LDA $99 C065 F003 BEQ FUNC02 C067 4C57F1 JMP $F157 !FROM KEYBOARD? C065 F003 C067 4C57F1 C06A TYES !DO NORMAL !DISPLAY CHAR TO SCREEN !SWITCH OFF BLINK !RESTORE CHAR !REMOVE CHAR !RUN/STOP? !NO !COPY TEXT INTO RI C0A6 A209 !COPY TEXT INTO BUFFER LDX #$09 SEI STX $C6 LDA $ECE6.X STA $0276.X CØR8 78 C0A9 86C6 COAB BDESEC FUNCOS COAE 9D7602 DEX CØB1 CA !REPEAT UNTIL DONE !DONE !CARRIAGE RETURN? !NO !END INPUT C0B2 D0F7 BNE FUNC06 C0B4 F0CA BEQ FUNC04 C0B6 C90D FUNC07 CMP #$0D C0B8 D003 BNE FUNC08 C0BB 4C02E6 FUNCEXT JMP $E602 ! CØBD CØBD A6D4 FUNCØ8 LDX $D4 !QUOTES? CØBF DØBC BNE FUNCØ3 !YES ``` ``` CMP #133 CØC1 C985 !LESS THAN F1? C0C3 90B8 BCC FUNCØ3 CMP #141 CØC5 C98D !GREATER THAN F8? C0C7 B0B4 BCS FUNCØ3 CØC9 38 SEC COCA E985 SBC #133 !CHANGE VALUE TO 0-7 CØCC 18 CLC COCD 69B8 ADC #$B8 !SET HIGH BYTE OF COCF 85FE STA #FE ! DEFINITION POINTER C0D1 A900 LDA #$00 C0D3 85FD STA $FD C0D5 A000 C0D7 2003C1 LDY #$00 JSR GETVAL !GET FIRST BYTE CODA C8 INY CODB 8C52C0 STY FUNCFLG !FLAG FUNCTION CODE COOD CMP #$0D !CARRIAGE RETURN? C0E0 F0D8 BEQ FUNCEXT !YES C0E2 C900 CMP #$00 BNE FUNCØ3 C0E4 D097 C0E6 8D52C0 STA FUNCFLG C0E9 F095 BEQ FUNCØ4 CØEB COEB AC52CO FUNCOS LDY FUNCFLG C0EE 2003C1 C0F1 D005 JSR GETVAL !GET CHARACTER BNE FUNC10 C0F3 8D52C0 STA FUNCFLG C0F6 F088 BEQ FUNC04 CØF8 C8 FUNC10 INY C0F9 8C52C0 C0FC C90D C0FE F0BA STY FUNCFLG CMP #$0D !CARRIAGE RETURN? BEQ FUNCEXT !YES, EXIT C100 4C7DC0 JMP FUNCØ3 C103 C103 A501 GETVAL LDA $01 !BASIC ROM OUT C105 29FE C107 8501 C109 B1FD AND #$FE STA $91 LDA ($FD),Y !GET CHARACTER C10B 48 PHA C10C A501 LDA $01 !BASIC ROM IN ORA #$01 C10E 0901 C110 8501 STA $01 C112 68 PLA RTS C113 60 ``` Program 2. To enable the function keys enter: SYS 49152 To define a function key use: ``` SYS 49163,k#,def$ ``` where k# is the number on the function key (without the 'f') and def\$ is any string expression. The text on the function keys will appear only if the function key character is removed by the input routine. This means that when using the GET command, the ASCII character for the function key is returned rather than a character from the text. The function key is not expanded if it is within quotes. The following is an example definition of a function key: ``` A$="":FOR I=\emptyset TO 79:A$=A$+CHR$(32)+CHR$(2\emptyset):NEXT SYS 49163,7,A$ ``` This will set up a function on key 7 which deletes from the cursor position to the end of the line, leaving the cursor at the same position (space-delete 80 times). #### 2.2 Joysticks There are two different types of joystick which can be connected to the CBM 64; a simple paddle switch joystick and a potentiometer or analog joystick. The switch joystick is widely used in games programs to move a cursor about the screen or to move an object. A switch joystick is primarily capable of only very simple directional input. It is, however, a very low cost device. The analog joystick is fairly expensive but is capable of far greater positional control. An interesting version of the analog joystick has started to appear in the form of low cost digitising pads which, when combined with the appropriate software, can produce some excellent computer art on the CBM 64. #### 2.2.1 The switch joystick These joysticks are not part of the keyboard hardware but they are connected to the same lines on the CIA#1 chip: port 1 to the read line and port 2 to the write line: ``` Bits 7–5 Not used $DCØØ: 4 JOY2 fire button 3 JOY2 east 2 JOY2 west 1 JOY2 south JOY2 north Ø Bits 7-5 Not used $DC01: 4 JOY1 fire button 3 JOY1 east 2 JOY1 west 1 JOY1 south JOY1 north ``` As with the keyboard, both joysticks must be read assuming that when the bit is zero, the contact is made. Because port 1 is connected to the same line as the keyboard read, any switches on joystick 1 will affect the character read in. Program 3 demonstrates the operation of the switch joystick. #### 2.2.2 Potentiometer joystick A potentiometer joystick consists of two potentiometers mounted at right angles to each other in a mechanism which allows the joystick when moved to change the wiper position of either one or both of the potentiometers. One potentiometer registers the potentiometer movement in the X axis and the other in the Y axis. The rotational movement of either potentiometer is divided by the computer into 255 divisions. With the joystick centered vertically the X and Y potentiometers will both have a value of 128. The position of the joystick can thus be mapped in terms of 2D graph coordinates. The two potentiometers are connected to the SID chip. SID has two analog inputs, and the two analog lines from each joystick port are multiplexed onto each input using a 4066 quad analog switch. The 4066 switching is controlled by lines PA6 and PA7 on CIA#1. Program 4 allows the input of values from two joysticks using the USR command. ``` Сиии *=$0000 0999 !ROUTINE TO READ PADDLE PORTS 0000 ! USES BASIC USR COMMAND 0000 ! TO INITIALISE USR COMMAND C000 ! POKE 785,0 0000 0999 ! POKE 786,192 0999 ! P= USR( PADDLE NUMBER) 0000 ! NOTE PADDLE NUMBER = 0 TO 3 0000 0999 0000 20BFB1 USR JSR $B1BF !FLOAT TO FIXED C003 A565 LDA $65 !LOW BYTE 0005 2903 0007 A8 AND #3 TRY C008 A240 LDX #$40 C00A 2902 AND #2 !PORTS 0,1 OR 2,3 BEQ POOR1 C00C F002 C00E A280 LDX #$80 !KEYBOARD SCAN C010 78 P90R1 SEI STX $DC00 C011 8E00DC !USES $DC00 C014 A280 LDX #$80 C016 CA LOOP DEX !DELAY FOR A/D C017 10FD BPL LOOP ! CONVERTER C019 98 C01A 2901 TYR AND #1 C01C AA TAX C01D BC19D4 LDY $D419,X !READ PORT C020 58 CLI C021 A900 LDA #0 !ZERO HIGH BYTE C023 4C91B3 JMP $B391 !FIXED TO FLOAT & EXIT Program 4. ``` #### 2.3 The screen #### 2.3.1 The hardware The screen display on the Commodore 64 is created and controlled by one chip; the VIC II (video interface controller 6567/9). A detailed description of the VIC II hardware can be found in Chapter 1. #### 2.3.2 The screen display operating system software None of the wide range of potential features of the VIC II chip are implemented by the software of the 64 with the exception that on power-up the default screen and border colours are set up, and the case bit is toggled. The kernal software to control the text screen is split into two sections; print a character to the screen, #### 24 The Commodore 64 Kernal and Hardware Revealed and scroll the screen. The routine to print a character to the screen is located at \$E716. This routine prints the character in register .A to the screen taking into account colour control codes, etc. This routine does several tasks before the character is printed; these tasks are shown in the flow chart in Fig. 2.2. The flow chart in Fig. 2.3 shows how the screen scrolls. This routine can be called from Basic with SYS 59626. Readers interested in the addition of extra commands which utilise the capabilities of the VIC II chip should consult the companion volume in this series, Advanced Commodore 64 Graphics and Sound. \$ET16 PUSH OFF ALL REGISTERS. DISABLE INPUT FROM SCREEN. YES NO 15 CHAR # <128? Display non control char Fig. 2.2. Character output flowchart. Fig. 2.2. (contd.) Fig. 2.2. (contd.) Fig. 2.2. (contd.) Fig. 2.2. (contd.) Fig. 2.2. (contd.) Fig. 2.3. Screen scrolling flowchart. ## Chapter Three # **Serial Communications** The CBM 64 has two different forms of serial communications capability; these are the Commodore serial interface and the RS232 interface. The Commodore serial interface is designed to allow the CBM 64 to be connected to other Commodore peripherals, in particular the 1541 disk drive. The RS232 serial interface is a simplified version of an industry standard communications interface widely used when connecting computers to printers and modems. Unfortunately the RS232 interface does not conform to the correct industry standards and therefore requires a small additional circuit to make it function properly. #### 3.1 Commodore serial bus The Commodore serial bus connects the Commodore 64 to its peripherals such as a disk drive and printers. This serial system has an effective speed of 3000 baud. This is not a true baud rate but is given just for comparison with the 300 baud normal cassette or 3600 baud for the high speed tape system in Chapter 4. A speed of 3000 baud is adequate for communicating with printers but makes the 1541 disk drive a little slow. The serial bus uses 5 lines including the ground line. #### 3.1.1 Commodore serial bus lines ### Serial service request Input: This enables a serial device to generate an IRQ in the 64. (No CBM 64 firmware support is available for this feature.) #### Signal ground This is a common ground line for serial devices. It is for signal reference and shielding the cable. #### Serial attention Input and output: Normally the CBM 64 can use this line only as an output. The 64 pulls this line low when sending command bytes to serial devices. It instructs all serial devices to listen for a command. #### Serial clock and serial data lines These two lines are both inputs and outputs. The current talking device uses these lines to send data and clock signals. Together these lines carry all data and perform the required handshaking. The serial bus signals are produced in the CBM 64 by port A of CIA chip 2. The following table shows the line connections. | Serial Data | in on PA7 | |--------------|---------------------------| | | out on PA5 | | Serial Clock | in on PA6 | | | out on PA4 | | Serial Atn | in to user port pin 9 | | | out on PA3 | | SRQ | in to CIA chip 1 FLAG pin | | | | Fig. 3.1 shows the serial port line driver. The serial port lines are driven by a Fig. 3.1. Serial port line driver. 7406 inverting buffer/driver chip with its outputs tied to 5 V with 1K resistors (Fig. 3.2). The 7406 was chosen for its open collector outputs. An open collector Fig. 3.2. 74\$\phi6\$ schematics (one gate). output can only drain current, not source it. So when the output of a 7406 gate is low it can pull the serial line low but when the output is high the serial line has to be pulled to 5 V by the 1K resistor. The 64 uses this by having the 5 V state as the release state i.e. available for use by other devices. When a line is in the released state the open collector outputs on another serial device can pull the line low. The 7406 is an inverting buffer/driver, so all clock, data and attention signals sent are transmitted inverted. Therefore release or line high is sent as a zero but received as a logic one. Only one device on the serial bus can talk at any one time but any number can listen. The Commodore 64 controls which device talks and which listens by commands sent with the attention line low (true). A timing diagram for serial operation is shown in Fig. 3.3. #### 1. Command byte sent under attention. Fig. 3.3. Serial bus timing. Fig. 3.3. (contd.) Fig. 3.3. (contd.) #### 3.1.2 Serial commands LISTEN Command device to listen TALK Command device to talk UNLSN Commands all listening devices to unlisten UNTLK Commands talking device to stop talking (UNTALK) ### 3.2 Serial ROM routines ``` LOC CODE LINE 0000 **************** 0000 ; SERIAL SYSTEM *************** 0000 0000 .LIB KDECLARE 0000 × =$90 0090 ; 0090 0090 KERNAL VARIABLES 0090 0090 STATUS *=*+1 : I/O OPERATION STATUS BYTE 0091 STKEY *=*+1 STOP KEY FLAG 0092 SVXT *=*+1 ; TEMPORARY 0093 VERCK *=*+1 ;LOAD OR VERIFY FLAG 0094 C3F0 *=*+1 ; IEEE BUFFERED CHAR FLAG 0095 *=*+1 BSOUR ;CHAR BUFFER FOR IEEE 0096 *=*+1 SYNO ; CASSETTE SYNC # 0097 XSAU #=#+1 TEMP FOR BASIN 0098 LDTND *=*+1 :INDEX TO LOGICAL FILE 0079 DFLIN *=*+1 ;DEFAULT INFUT DEVICE # 009A DEFAULT OUTFUT DEVICE # DFLTO *=*+1 *=*+1 009B PRTY ;CASSETTE PARITY 009C *=*+1 DESM ; CASSETTE DIFOLE SWITCH 009D MSGFLG *=*+1 :OS MESSAGE FLAG 009E FTR1 ; CASSETTE ERROR FASS1 009E T1 *=*+1 :TEMPORARY 1 009F TMF'C 009F PTR2 :CASSETTE ERROR PASS2 009F 12 *=*+1 ; TEMPORARY 2 00A0 TIME *=*+3 ;24 HOUR CLOCK IN 1/60TH SECONDS 00A3 R2D2 SERIAL BUS USAGE 00A3 PCNTR *=*+1 ; CASSETTE STUFF 00A4 BSOUR1 :TEMP USED BY SERIAL ROUTINE 00A4 FIRT *=*+1 00A5 COUNT ;TEMP USED BY SERIAL ROUTINE 00A5 ; CASSETTE SYNC COUNTDOWN CNTDN *=#+1 00A6 RHEPT *=*+1 CASSETTE BUFFER FOINTER 00A7 INBIT ;RS-232 RCVR INPUT BIT STORAGE 00A7 SHCNL *=*+1 :CASSETTE SHORT COUNT 00A8 BITCI :RS-232 RCVR BIT COUNT IN 00A8 RER *=*+1 ; CASSETTE READ ERROR 00A9 RINONE :RS-232 RCVR FLAG FOR START BIT CHECK 00A9 *=*+1 RF7 ;CASSETTE READING ZEROS ;RS-232 RCVR BYTE BUFFER 00AA RIDATA 00AA RDFLG *=*+1 CASSETTE READ MODE RIPRTY 00AB :RS-232 RCVR PARITY STORAGE OOAP. SHCNH *=*+1 ;CASSETTE SHORT CNT 00AC SAL *=*+1 00AD SAH *=*+1 00AE EAL *=*+1 00AF FAH *=*+1 00B0 CMF<sub>0</sub> *=*+1 99B1 TEMP *=*+1 00B2 TAPE1 *=*+2 ; ADDRESS OF TAPE BUFFER 00B4 BITTS :RS-232 TRNS BIT COUNT 00P4 SNSW1 *=*+1 ``` ``` LOC CODE LINE 00B5 NXTRIT ;RS-232 TRNS NEXT BIT TO BE SENT 00B5 DIFF *=×+1 00E6 RODATA :RS-232 TRNS BYTE BUFFER 00P.6 PRP *=*+1 00B7 FNLEN *=*+1 :LENGTH CURRENT FILE N STR 00BB LA *=*+1 CURRENT FILE LOGICAL ADDR 00B9 SA #=#+1 CURRENT FILE 2ND ADDR 00PA FA ∺=±+1 CURRENT FILE PRIMARY ADDR 00BB FNADR *=*+2 :ADDR CURRENT FILE NAME STR 00BD ROPRIY :RS-232 TRNS PARITY BUFFER ØØBD OCHAR *=*+1 00RF FSBLK *=*+1 *CASSETTE READ BLOCK COUNT 00P.F MYCH *=*+1 99C9 CAS1 *=*+1 :CASSETTE MANUAL/CONTROLLED SWITCH 00C1 TMPO 00C1 STAL *=*+1 00C2 STAH *=*+1 00C3 MEMUSS : CASSETTE LOAD TEMPS (2 BYTES) 00C3 TMF<sub>2</sub> *=*+2 00C5 00C5 VARIABLES FOR SCREEN EDITOR 00C5 00C5 LSTX *=*+1 :KEY SCAN INDEX 00C6 #=¥+1 NDX INDEX TO KEYBOARD Q 00C7 RVS *=*+1 RVS FIELD ON FLAG 00C8 INDX *=*+1 0009 LSXP *=*+1 :X FOS AT START LSTP 00CA *=*+1 66CB SEDX *=*+1 SHIFT MODE ON FRINT 09CC CURSUR BLINK ENAB BLNSW *=*+1 aacn COUNT TO TOGGLE CUR BLNCT *=*+1 *=*+1 00CE GOBLN ;CHAR BEFORE CURSOR : UN/OFF BLINK FLAG 00CF PLNON *=*+1 0000 CRSW *=*+1 :INPUT VS GET FLAG 00D1 *=*+2 FOINTER TO ROW F·NT POINTER TO COLUMN 0003 PNTR *=*+1 0004 QTSW *=*+1 , QUOTE SWITCH 0005 LNMX *=*+1 :40/80 MAX PUSITION 00D6 TEL X *=*+1 0007 DATA *=*+1 0008 INSKT *=*+1 :INSERT MODE FLAG 8009 *=*+25 :40/80 LINE FLAGS LDTR1 LINTMP *=*+1 00F2 :TEMPORARY FOR LINE INDEX 00F3 USER 8=8+2 SCREEN EDITOR COLOUR IP 00F5 KEYTAB *=*+2 :KEYSCAN TABLE INDIRECT 09F7 ;RS-232 Z-PAGE 00F7 00F7 66F7 *=*+2 RIBUF :RS-232 INPUT BUFFER POINTER 00F9 ROBUF *=*+2 ;RS-232 OUTPUT BUFFER POINTER FREKZP *=*+4 GOF P. FREE KERNAL ZERO PAGE OOFF BASZPT *=*+1 ;LOCATION ($00FF) USED BY BASIC 0100 0100 *=$100 0100 BAD *=*+1 0101 *=$200 0200 PHF *=*+89 :BASIC/MONITOR BUFFER 0259 :TABLES FOR OPEN FILES 0259 0259 ĹAT 0259 *=*+10 ;LOGICAL FILE NUMBERS 0263 FAT *=*+10 PRIMARY DEVICE NUMBERS Ø26D SAT *=*+10 ;SECONDARY ADDRESSES 0277 SYSTEM STORAGE 0277 0277 0277 KEYD *=*+10 :IRQ KEYBOARD BUFFER 0281 MEMSTR *=*+2 START OF MEMORY ``` Ľ ``` CODE 1.0C LINE :TOP OF MEMORY 0283 MEMSIZ *=*+2 0285 TIMOUT *=*+1 ; IEEE TIMEOUT FLAG 0286 SCREEN EDITOR STORAGE 0286 0286 0286 COLOR *=*+1 ACTIVE COLOUR NIBBLE 0287 GUCOL *=*+1 ORIGINAL COLOUR BEFORE CURSOR 0288 HIBASE *=*+1 BASE LOCATION OF SCREEN (TOP) 0289 XMAX *=*+1 028A RETELG *=*+1 KEY REFEAT FLAG 0288 KOUNT *=*+1 Ø28C DELAY *=*+1 028D SHFLAG *=*+1 SHIFT FLAG BYTE LAST SHIFT FATTERN 028E LSISHF *=*+1 KEYLOG *=*+2 ;INDIRECT FOR KEYBOARD TABLE SETUP 022F MODE *=*+1 0291 ;0-FET MODE, 1-CATTACANNA 0292 AUTODN *=*+1 :AUTO SCROLL DOWN FLAG (=0 ON <>0 OFF) 0293 0293 :RS-232 STORAGE 0293 0293 M26CTR *=*+1 :6526 CONTROL REGISTER 0294 M26CDR *=*+1 :6526 COMMAND REGISTER 0295 M26AJB *=*+2 :NON STANDARD (BITTIME/2-100) 0297 RSSIAT *=*+1 ;RS-232 STATUS REGISTER NUMBER OF BITS TO SEND (FAST RESPONSE) 0298 PITNUM *=*+1 BAUDOF *=*+2 0299 :BAUD RATE FULL BIT TIME Ø29B 029B :RECIEVER STORAGE 029B 029B RIDBE *=*+1 ; INFUT BUFFER INDEX TO END 029C RIDBS *=*+1 :INPUT BUFFER POINTER TO START 0290 ;TRANSMITTER STORAGE 0290 029D 029D RODES *=*+1 COUTPUT BUFFER INDEX TO START 029E RODGE *=*+1 COUTPUT BUFFER INDEX TO END 029F 029F IROTME *=*+2 HOLDS IRQ DURING TAPE OFS 02A1 ľ 02A1 : 02A1 *=$0300 :FROGRAM INDIRECTS(10) 0300 *=$0300+20 :KERNAL/OS INDIRECTS (20) 0314 CINV *=*+2 ; IRQ RAM VECTOR ; BRK INSTR RAM VECTOR ; NMI RAM VECTOR 0316 CBINA *=*+2 0318 VMINV *=*+2 *=*+2 031A IOPEN ; INDIRECTS FOR CODE 031C ICLOSE *=*+2 :CONFORMS TO KERNAL SPEC 8/19/80 031E ICHKIN *=*+2 0320 ICKOUT *=*+2 0322 ICLRCH *=*+2 IBASIN *=*+2 0324 0326 IBSOUT *=*+2 0328 ISTOR *=*42 IGETIN *=*+2 032A Ø32C ICLALL *=*+2 032E USRCMD *=*+2 0330 TLOAD *=*+2 0332 ISAVE *=*+2 SAVESP 0334 0334 *=$0300+60 033C TBUFFR *=*+192 CASSETTE DATA BUFFER 03FC 03FC *=$0400 0400 CBMSCN *=*+999 :64 SCREEN 07E7 97E7 *=$0800 0800 RAMLUC 0800 ``` ``` LOC CODE LINE 9889 *=$D000 D000 VICREG *=*+47 :VIC REGISTERS D02F DØ2F *=$D400 D400 SIDREG *=*+29 SID REGISTERS D410 D41D *=$D800 D800 CBMCOL *=*+999 :64 COLOUR NIBBLES DBE 7 :I/O DEVICES DBE7 DPE 7 DBE7 * =$DC00 :6526 (IRQ) ;KEYBUARD MATRIX DC00 COLM DC00 D1DFA *=*+1 DC01 ROWS *KEYBOARD MATRIX D10FB *=*+1 DC01 D1DDRA *=*+1 DC02 0100RB *=*+1 DC03 DC 94 D17AL *=*+1 *=*+1 DC05 D1TAH *=*+1 DITEL DC66 DC07 D1TBH *=*+1 DC08 D1TOD1 *=*+1 D1TOD2 *=*+1 DC 09 D110D3 *=*+1 DC0A DC0B D1TOD4 *=*+1 D110DB *=*+1 DC0C DCØD D11CR *=*+1 D1CRA *=*+1 DC0E DC0F D1CRB *=*+1 DC10 DC10 *= $0000 :6526 (NMI) DD00 D2DFA *=*+1 0001 D2DFB *=*+1 0002 D2DDRA *=*+1 0003 D2DDRB *=*+1 DD04 D2TAL *=*+1 0095 *=*+1 D2TAH DD06 D2TEL *=*+1 0007 D2TBH *=*+1 8900 D2T001 *=*+1 DD09 D2T0D2 *=*+1 DDØA D2T0D3 *=*+1 DDØB 02TOD4 *=*+1 DDØC D21008 *=*+1 DDØD D21CR *=*+1 DDØE D2CRA *=*+1 DDØF D2CRB *=*+1 DD10 DD10 TAPE BLOCK TYPES DD10 ÉOT DD10 =5 END OF TAPE DD10 BLF =1 ; BASIC LOAD FILE DD10 BUF ≖2 BASIC DATA FILE DD10 FLF =3 FIXED PROGRAM TYPE DD10 BDFH BASIC DATA FILE HEADER =4 DD10 BUFSZ =192 BUFFER SIZE DD10 TAPE ERROR TYPES DD10 0010 DD10 SPERR =16 DD10 CKERR =32 DU10 SBERR =4 DD10 LBERR =8 0010 DD10 SCREEN EDITOR CONSTANTS ``` DD16 ``` LOC CODE LINE ;SINGLE LINE 40 COLUMNS ;DOUBLE LINE 80 COLUMNS LLEN =40 DD10 LLEN2 =80 UU10 :25 ROWS ON SCREEN NLINES =25 DD10 , BLUE SCREEN COLOUR BLUE =6 0010 ;LT BLUE CHAR COLOUR LTBLUE =14 DD10 CARRIAGE RETURN DD10 CR =$D 0010 MAXCHR =80 NURAP =2 DD10 0010 .END .LIB KSER1 0010 ``` ### **TALK** Entry point: \$FFB4 Function: Command serial device to talk (transmit data) Input parameters: .A device number Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$90) – attention not acknowledged by data low within 1 ms Frame error (in ST) – no data accepted response (data low) within 1 ms of last bit of byte being sent. Description: This routine ORs the device number in the .A register with \$40. Before the command is sent the single character serial buffer is checked for being empty. If it is not the character in it is sent (with end message marker (EOI)). After this the attention line is set low (bit 3 set in chip 2 port A (assembler label D2DPA in listing). Then the command byte is sent with the attention line held low. ``` LOC CODE LINE DD10 *=$ED09 ED09 ************** ED09 **COMMAND SERIAL BUS TO TALK. ED09 **THE ACCUMULATOR MUST BE LOADED WITH THE ED09 **DEVICE NUMBER THAT YOU WISH TO TALK. ED09 09 40 ED09 L836 ORA #$46 #MAKE ADDR TALK SKIP NEXT COMMAND EDØB 20 .BYT $2C ``` #### LISTEN Entry point: \$FFB1 Function: Command serial device to listen Input parameters: .A device number Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$90) – attention not acknowledged by data low within 1 ms Frame error (in ST) - no data accepted response (data low) within 1 ms of last bit of byte being sent Description: This routine ORs the device number in the .A register with \$20. Before the command is sent the single character serial buffer is checked for being empty. If it is not the character in it is sent with an EOI handshake to mark it as the last byte of its message. After this the attention line is set low. Then the command byte is sent with the attention line held low. This routine includes the main routine to send a byte to the serial bus. This is done as follows: L842 - Set clock low Set data high Delay 1 ms L859 - Set data high (released) Set clock line high If EOI no handshake required then L850 ; Wait with clock high for End Or Identify handshake Wait for data high Wait for data low ; That is end of hold, until serial device is ready L850 - Wait for data high Set clock low Put 8 in counter L848 - If data not high framing error Get next bit of byte to send : Low bit first If bit is zero then set data low Set clock high; flag bit Sort pause Set data high and clock low Decrease bit counter Go to L848 if not all sent Set timer for 1 ms L855 - Has timer expired? If so then L847 (framing error) If data not low go back to L855 Exit LOC CODE LINE ``` EDØC ED90 :*COMMAND SERIAL BUS TO LISTEN. ED0C ;*TO USE THIS ROUTINE, THE ACCUMULATOR MUST ;*FIRST BE LOADED WITH THE DEVICE NUMBER THAT ED90 ED0C *YOU WISH TO LISTEN (RECEIVE DATA). ED9C EDØC 09 20 URA #$20 L966 :MAKE ADDR LISTEN FDØF 20 A4 F0 JSR $FOA4 PROTECT FROM RS323 NMI ED11 48 L980 PHA ED12 24 94 BIT C3PO CHAR IN BUFFER? ED14 10 0A BFL L864 : NO ED16 ED16 ; SEND BUFFERED CHAR E016 ED16 38 SEC SET EOI FLAG ED17 66 A3 ROR R202 ED19 JSR L859 SEND LAST CHAR 20 40 ED 46 94 ED1C LSR C3PO BUFFER CLEAR ED1E 46 A3 LSR R2D2 CLEAR EOI FLAG PLA E020 88 L864 ITALK/LISTEN ADDR ED21 85 95 STA BSOUR ED23 78 SEI ED24 20 97 EE JSR LB44 E027 C9 3F CMP #$3F CLKHI ONLY ON UNLISTEN ED29 DØ Ø3 PNE L839 ED28 20 85 EE JSR L875 ED2E AD 00 DD L839 LDA D2DFA ASSERT ATTENTION E031 09 08 ORA #$08 ED33 8D 00 DD STA D2DFA ED36 Ĺ842 ED36 78 SEI E037 20 8E EE JSR L843 SET CLOCK LINE LOW 20 97 EE ED3A JSR L844 ED30 JSR L846 20 B3 EE ; DELAY 1 MS ED40 78 L859 SEI DISABLE IRO ED41 20 97 EE JSR L844 :MAKE SURE DATA IS RELEASED ED44 20 A9 EE JSR L854 ;DATA SHOULD BE LOW BCS L856 ED47 BØ 64 ED49 20 85 EE JSR L875 :CLOCK LINE HI ED4C 24 A3 BIT R2D2 ;EUI FLAG TEST ED4E 10 0A BPL L850 ED50 ;00 E01 ED50 20 A9 EE L846 JSR L854 ;WAIT FOR DATA HI EU53 90 FB BCC L840 ED55 20 A9 EE L849 JSR L854 :WAIT FOR DATA LO E058 BØ FB BCS L849 ED5A 20 A9 EE L850 JSR L854 :WAIT FOR DATA HI EDSD 90 FB BCC L850 ED5F 20 BE EE JSR L843 :SET CLOCK LO ED62 ED62 SET TO SEND DATA ED62 ED62 A9 08 LDA #$08 :COUNT 8 BITS 85 A5 ED64 STA COUNT LDA D2DFA ED66 AD 00 DD L848 :DEBOUNCE BUS ED69 CD 00 DD CMP D20PA ED6C DØ F8 BNE L848 FD9E ØA ASL A ED6F 90 3F BCC L847 :DATA MUST BE HI ED71 66 95 RUR BSOUR ;NEXT BIT INTO CARRY ED73 PCS L851 80 05 ED75 20 A0 EE JSR L841 BNE L853 ED78 DØ Ø3 EU/A 20 97 EE L851 JSR L844 ED7D 20 85 EE JSR L875 LB53 ;CLUCK HI ED89 EA NOP ED81 NOF EA ED82 EA NOF ``` | LOC | CODE | LINE | | | |------|----------|------|------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ED83 | EA | | NOP | | | ED84 | AU 99 DU | | LDA D2UFA | | | ED87 | 29 DF | | AND #SDF | ;DATA HI | | E089 | 09 10 | | URA #\$10 | CLOCK LO | | ED86 | 8D 00 DD | | STA D2DFA | , | | FD8E | C6 A5 | | DEC COUNT | | | ED90 | DØ D4 | | PNE L848 | | | ED92 | AY 04 | | LDA #\$04 | SET TIMER FOR 1 MS | | ED94 | 8D 07 DC | | STA DITEH | | | ED97 | A9 19 | | LDA #819 | | | ED99 | 8D OF DC | | STA DICRE | | | ED9C | AD ØD DC | | LDA DIICR | | | ED9F | AD ØD DC | L855 | LDA DIICR | | | EDA2 | 29 02 | | AND #\$02 | | | EDA4 | D0 0A | | PHE LB47 | | | EDA6 | 20 A9 EE | | JSR L854 | | | EDA9 | ₽0 F4 | | BCS L855 | | | EDAB | 58 | | CLI | ;ENAPLE IRG | | EDAC | 60 | | RTS | | | EDAD | A9 80 | L856 | LDA #880 | DEVICE NOT PRESENT | | EDAF | 20 | 1017 | .BYT \$2C | PURSA AND SAME AND ADDRESS OF THE PARTY T | | EDB0 | A9 03 | L847 | LDA #\$03 | FRAMING ERROR | | EDB2 | 20 10 FE | L852 | JSR \$FE1C | ;SEND MESSAGE | | ED85 | 58 | | CLI | ;ENABLE IRQ | | ED89 | 18 | | CLC | | | ED87 | 90 4A | | BCC L1004 | ;ALWAYS | ### SECOND Entry point: \$FF93 Function: Send secondary address after listen Input parameters: Secondary address in .A register ORed with \$60 Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$90) - attention not acknowledged by data low within 1 ms Frame error (in ST) - no data accepted response (data low) within 1 ms of last bit of byte being sent Description: The secondary address is stored in the serial buffer and then sent to listening devices. Next the attention line is released (set high). | LOC | CODE | LINE | |------|-------|-----------------------------------------------| | EDB9 | | ************ | | ED89 | | *SEND SECONDARY ADDRESS AFTER LISTEN. | | EDB9 | | *THIS ROUTINE IS USED TO SEND A SECONDARY | | EDRA | | **ADDRESS AFTER A CALL TO THE LISTEN COMMAND. | | EDBA | | *********** | | EDBA | 85 95 | L871 STA BSOUR ;BUFFER CHAR | ``` 20 36 FD JSR 1 842 FDRR SEND IT EDBE RELEASE ATTENTION EDBE EDBE Ĺ983 EDBE AD 00 DD LDA D2DFA 29 F7 AND #$F7 EDC1 EDC3 8D 00 DD STA D2DFA . RELEASE EDC4 60 RTS ``` ### TKSA Entry point: \$FF96 Function: Send secondary address after talk Input parameters: Secondary address in .A register Output parameters: None Registers used: .A ### Error messages: CODE LOC Device not present (returned in STatus var. \$90) – attention not acknowledged by data low within 1 ms Frame error (in ST) – no data accepted response (data low) within 1 ms of last bit of byte being sent Description: The secondary address is loaded into the serial buffer and then sent to the serial bus. This routine then waits for the new talking device to acknowledge it is the new talker by changing the clock line. This is done as follows: ``` Hold data low Set attention high (release) Set clock high Then wait for clock to go low ``` LINE ``` EDC7 ************** EDC<sub>2</sub> *SEND TALK SA. EDC7 :*THIS ROUTINE IS USED TO SEND A SECONDARY EDU7 ;*ADDRESS TO A DEVICE THAT HAS ALREADY BEEN EDC7 **COMMANDED TO TALK. EDC7 * *********************************** 85 95 EUC7 L860 STA BSOUR BUFFER CHAR EDC9 20 36 ED JSR L842 SEND SA EDCC SHIFT OVER TO LISTENER EDCC EDCC L970 ;DISABLE IRQ EUCC 78 SEI EDCD 20 A0 EE JSR L841 DATA LINE LO EDDØ 20 BE ED JSR L983 EDD3 20 85 EE JSR L875 ICLOCK LINE HI 20 A9 EE EDD6 L968 JSR L854 WAIT FOR CLOCK LO EDD9 30 FB PMI L968 EDDB 58 CLI ; DONE EDUC RTS ``` ### CIOUT Entry point: \$FFA8 Function: Send byte to serial bus Input parameters: Byte to send in .A Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$9\(\theta\)) – attention not acknowledged by data low within 1 ms Frame error (in ST) - no data accepted response (data low) within 1 ms of last bit of byte being sent Description: Any character in the serial buffer is sent to the serial port. Then the current character is stored in the buffer. | LOC | CODE | LINE | | | |-------------|--------------|-------------------------------|---------|------------------------| | EDDD | | .****** | ******* | :***** | | EDDD | | BUFFERED OUTPUT TO SERIAL BUS | | | | EDDD | | ******* | | ************ | | EUUU | 24 94 | Ĺ861 BIT | C3PO | ;BUFFERED CHAR? | | EDDF | 30 05 | P.M.I | L949 | YES, SEND LAST | | EUE 1 | 38 | SEC | | ; NO | | EDE2 | 66 94 | ROR | C3FO | SET BUFFERED CHAR FLAG | | EDE 4 | DØ Ø5 | BNE | L862 | ;ALWAYS | | EDE 6 | | ş | | | | EDE 6 | 48 | L949 PHA | | ; SAVE CURRENT CHAR | | <b>EUE7</b> | 20 40 ED | JSR | L859 | ;SEND LAST CHAR | | EDEA | 68 | PLA | | RESTURE CURRENT | | EDEB | 85 <b>95</b> | L862 STA | BSOUR | ;BUFFER II | | EDED | 18 | CLC | | ;6000 EXIT | | EDEE | 60 | RTS | | | #### UNTLK Entry point: \$FFAB Function: Send command UNTALK Input parameters: None Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$90) - attention not acknowledged by data low within 1 ms Frame error (in ST) – no data accepted response (data low) within 1 ms of last bit of byte being sent Description: This routine sends the \$5F under attention to serial bus. This tells the current talking to stop. After a delay this routine ends by releasing clock and data lines. ``` CODE LOC ****************** EDEF **SEND UNTALK. EDEF **THIS ROUTINE SENDS AN 'UNTALK' TO THE SERIAL EDEF ;*BUS. IT WILL TELL ALL DEVICES IN TALK EDEF **MODE TO STOP TALKING (SENDING DATA). EDEF **************** EDEF L863 SE I EDEF 78 20 8E EE JSR L843 EUF 0 :FULL ATN LDA D2DFA EDF3 AD 00 DD URA #$08 EUF 6 09 08 8D 00 DD STA D2DFA EDF8 ; UNTALK LDA #$5F EDFP. A9 5F .BYT $2C ISKIP NEXT COMMAND EDFD 2C ``` ### UNLSN Entry point: \$FFAE Function: Send command UNLISTEN Input parameters: None Output parameters: None Registers used: .A Error messages: Device not present (returned in STatus var. \$90) - attention not acknowledged by data low within 1 ms Frame error (in ST) - no data accepted response (data low) within 1 ms of last bit of byte being sent Description: This routine sends the \$3F under attention to serial bus. This tells the current listening devices to stop. After a delay this routine ends by releasing clock and data lines. ``` LOC CODE LINE EDFE ************** ; *SEND UNLISTEN. EDFE ;*THIS ROUTINE SENDS AN 'UNLISTEN' TO EDFE EDFE **THE SERIAL BUS. IT WILL TELL ALL DEVICES **IN LISTEN MODE TO STOP LISTENING. EDFE EDFE ********************* A9 3F JUNLISTEN COMMAND EDF E L1006 LDA #$3F EE00 20 11 ED JSR L980 SEND EE03 RELEASE ALL LINES EE03 EE03 EE03 20 BE ED L1004 JSR L983 RELEASE ATN EE06 ``` | EE06 | | ; DELAY | THEN RELEASE | CLOCK AND | DATA | | |------|----------|---------|--------------|-----------|----------------------|---| | EE06 | | 3 | | | | | | EE#6 | 8A | L858 | TXA | ; DELAY | APPROX 60 MICRO SECS | 3 | | EE07 | A2 ØA | | LDX #\$0A | • | | | | EE09 | CA | L876 | DEX | | | | | EEØA | DØ FD | | BNE L876 | | | | | EE0C | AA | | TAX | | | | | EE0D | 20 85 EE | | JSR L875 | | | | | EE10 | 40 97 EE | | JMF L844 | | | | ### ACPTR Entry point: \$FFA5 Function: Input byte from serial port Input parameters: None Output parameters: Character in .A Registers used: .A Error messages: Read timeout (in ST) – no clock low response within $\emptyset.2$ ms of data being released Description: This routine gets a byte from serial bus and returns it in the .A register. It does this as follows: L865 - Zero COUNT Release clock line Wait for clock to go high L866 - Set timer to 256 ms Release data L872 - If timer expired go to L868 If clock still high go back to L872 Otherwise go to L870, to read byte L868 - If COUNT non zero flag read timeout in ST and exit via a routine to release lines Otherwise assume EOI : Handshake EOI Set data low Pause and release data Flag EOI in ST Increase COUNT Go back to L866, to wait for clock ; Get a byte L870 - Set COUNT for 8 bits L869 - Wait for clock to go high Get next bit of byte from data line Wait for clock to go low Decrease COUNT If COUNT not zero go back to L869 to get next bit Acknowledge byte by sending data low Check EOI flag in ST ; EOI flags end of message If set then delay and release data Exit this byte read in .A ``` LOC CODE LINE EE13 EE13 ; INPUT A BYTE FROM SERIAL BUS EE13 EE13 78 L865 SEI ;DISABLE IRQ EE14 A9 00 LDA #$00 ;SET EDI/ERROR FLAG EE16 85 AS SIA COUNT EE18 20 85 EE JSR L875 ; RELEASE CLOCK LINE EE1B 20 A9 EE L943 JSR L854 WAIT FOR CLOCK HI EE1E 10 FB BFL L943 A9 01 EE20 L866 LDA #$01 SET TIMER B FOR 256 US EE22 8D 67 DC STA DITEH EE25 A9 19 LDA #$19 EE27 8D ØF DC STA DICRB 20 97 EE JSR L844 EE2A AD ØD DC EE2D LDA DIICR AD 0D DC EE30 L872 LDA DIICR EE33 29 02 AND #$02 ; CHECK THE TIMER EE35 DØ 07 ;RAN OUT EE37 20 A9 EE ; CHECK THE CLOCK LINE JSR L854 EE3A 30 F4 BMI L872 ; NUT YET EE3C 10 18 BFL L870 1YES EE3E EE3E EE3E 45 A5 L868 LDA COUNT ; CHECK FOR ERROR EE 40 FØ Ø5 BEQ L867 A9 02 EE42 LDA #802 JMF L852 EE44 4C B2 ED ;ST=2, READ TIME OUT EE47 EE47 TIMER RAN OUT, DO AN EOI EE47 EE47 20 A0 EE Ĺ867 JSR L841 IDATA LINE LO EE4A 20 85 EE JSR L875 DELAY, SET DATA HI EE4D A9 40 LDA #$40 EE4F 20 1C FE JSR $FE1C ; OR AN EOI BIT INTO ST EE52 E6 A5 INC COUNT AND AGAIN FOR ERROR CHECK EE54 DØ CA PNE L866 EE56 , BYTE TRANSFER EE56 EE56 EE56 A9 08 L870 LDA #$68 ;SET UP COUNTER EE38 85 A5 STA COUNT EE5A LDA D2DPA AD 00 DD L869 ; WAIT FOR CLOCK HI EE5D CD 00 DD CMF D2UFA ; DEBOUNCE EE60 DØ F8 PNE LB69 ASL A EE62 ØΑ 10 F5 EE63 BFL L869 ROR BSDUR1 EE 65 66 A4 ROTATE DATA IN EE67 AD 00 DD L873 LDA D2DPA ; WAIT FOR CLOCK LO EE6A CD 00 DD CMP D2DPA :DEBOUNCE EE6D DØ F8 BNE L873 EE6F ØA ASL A EE76 30 F5 PMI L873 EE72 C6 A5 DEC COUNT EE74 DØ E4 PNE L869 , MORE BITS EE76 20 A0 EE JSR L841 ;DATA LO ``` | EE79 | 24 90 | | BIT STATUS | ;CHECK FOR EOI | |------|----------|------|-----------------|-------------------| | EE78 | 50 03 | | <b>BVC L874</b> | NONE | | EE7D | 20 06 EE | | JSR L858 | DELAY AND DATA HI | | EE80 | AS A4 | L874 | LDA BSOUR1 | • | | EE82 | 58 | | CLI | :ENABLE IRQ | | EE83 | 18 | | CLC | GOOD EXIT | | EE84 | 60 | | RTS | , | #### 3.3 General routines All routines change only the .A register. Set clock high, set clock low, set data high & set data low all just set or unset a bit in port A of CIA chip 2. Note that the bit is set to send a line low. Debounce CIA routine first loops until a consistent value is read from port A of the CIA. It then sets the carry flag to the state of the data line, and the sign flag to the state of the clock line. The 1 millisecond delay is a software delay loop lasting approx 1 ms. ``` LOC CODE LINE EE85 ; :SET CLOCK LINE HI (INVERTED) EE85 EE85 Ĺ875 AD 60 DD EE85 LDA D2DPA EE88 29 EF AND #SEF EE8A BD 00 DD STA D2DPA 60 EE8D RTS EE8E SET CLOCK LINE LO (INVERTED) EE8E EE8E EE8E AD 00 DD Ĺ843 LDA D2DFA EE91 09 10 ORA #$10 EE93 8D 00 DD STA D2DPA EE96 60 RTS EE97 SET DATA LINE HI (INVERTED) EE97 EE97 Ĺ844 EE97 AD 00 DD LDA D2DFA EE9A AND #BDF 29 DF EE90 8D 00 DD STA D2DFA EE9F RTS 60 EEA0 :SET DATA LINE LO (INVERTED) EEA0 EEA0 Ľ841 AD 00 DD LUA DZUPA EEA0 EEA3 09 20 ORA #$20 EEA5 8D 00 DD STA D2DFA EEA8 60 RTS EEA9 EEA9 DEBOUNCE THE PIA EEA9 Ĺ854 EEA9 AD 00 DD LDA D2DFA CD 00 DD EEAC CMP D2DFA EEAF DØ F8 PNE L854 ASL A EEB1 ØA EEB2 60 RTS EEB3 DELAY 1 MS EEB3 EEB3 Ĺ846 EEB3 RA TXA ``` | <b>LLU4</b> | A2 88 | | LDX #\$88 | |-------------|-------|------|------------| | EEB6 | CA | L845 | DEX | | EEP7 | DØ FD | | BNE L845 | | EEB9 | AA | | TAX | | EEBA | 60 | | RTS | | EEBB | | .END | | | EEBB | | | .LIB KSER2 | ### **GETIN** Entry point: \$FFE4 Function: Get a character from the current input device Input parameters: None Output parameters: .A holds character, CARRY clear Registers used: .A Error messages: None Description: For serial devices, GETIN is redirected to BASIN. ``` LOC CODE LINE *=$F13E EEBB F13E F13E ;* GETIN -- GET CHARACTER FROM CHANNEL. F13E ; * F13E CHANNEL IS DETERMINED BY DFLTN. ;* IF DEVICE IS 0, KEYBOARD QUEUE IS ;* EXAMINED AND A CHARACTER REMOVED IF F13E F13E ;* AVAILABLE. IF QUEUE IS EMPTY, Z F13E ;* FLAG IS RETURNED SET. F13E DEVICES 1-31 F13E ;* ADVANCE TO BASIN. THE CHARACTER IS F13E ;* RETURNED IN .A. IF ZERO, NULL CHAR. F13E ************************** F13E ;CHECK DEVICE F13E A5 99 NGETIN LDA DFLTN ;NOT KEYBOARD F140 D0 08 BNE L924 , QUEUE INDEX F142 A5 C6 LDA NDX F144 F0 0F BEQ L944 ,NOTHING THERE, EXIT F146 78 SEI , REMOVE A CHAR 4C B4 E5 F147 JMP $E584 ;RS-232? L924 CMF #$62 F14A C9 02 DØ 18 84 97 F14C BNE L927 ,NO, USE BASIN , SAVE .Y, USED IN RS-232 F14E L926 STY XSAV F150 20 86 F0 JSR $F086 LDY XSAV ; RESTORE .Y F153 A4 97 L944 F155 18 CLC ; GOOD RETURN F156 60 RTS ``` **BASIN** Entry point: \$FFCF Function: Get a character from the current input device Input parameters: None Output parameters: .A holds character, CARRY clear Registers used: .A Error messages: None Description: If the status from the last character read was $\neq \emptyset$ (EOF), the character 13 (carriage return) is returned with CARRY clear. Otherwise one byte is read using the ACPTR routine. ``` LOC CODE LINE F157 *************** F157 F157 ;* BASIN-- INFUT CHARACTER FROM CHANNEL. F157 ; * BASIN DIFFERS FROM GETIN ON KEYBOARD F157 ;* AND RS-232 ONLY. THE SCREEN EDITOR F157 ;* MAKES READY AN ENTIRE LINE WHICH IS F157 ;* PASSED CHARACTER BY CHARACTER UP F157 ;* TO THE CARRIAGE RETURN. THE CHARACTER F157 ;* IS RETURNED IN .A. ZERO FOR NULL CHAR F157 ;* OTHER DEVICES ARE: ; * F157 0 --- KEYBOARD ; * F157 1 --- CASSETTE ; * F157 2 --- RS-232 ; * F157 3 --- SCREEN F157 4-31 --- SERIAL BUS ı× F157 3. 我实现就实现实现实现的现在分词的现在分词的现在分词的现在分词 F157 A5 99 F157 NBASIN LDA DELTN ; CHECK DEVICE F159 DØ ØB BNE L927 :NOT KEYBOARD F158 INPUT FROM KEYBOARD F15B F15B ļ F15B A5 D3 LDA FNTR ; SAVE CURRENT: STA LSTP F150 85 CA ; CURSOR COLUMN, F15F A5 D6 LDA TELX F161 85 C9 STA LSXP :LINE NUMBER F163 4C 32 E6 JMF $E632 BLINK CURSOR UNTIL RETURN F166 Ĺ927 F166 C9 03 CMF #$03 :SCREEN? F168 00 09 BNE L928 : NO F16A 85 DØ STA CRSW ;FAKE CARRIAGE RETURN ;ENDED: F16C A5 D5 LDA LNMX F16E 85 C8 ON THIS LINE STA INDX F170 4C 32 E6 JMP $E632 FICK UP CHARACTERS F173 L928 F173 80 38 BCS L939 :DEVICES>3 F175 CMP #$62 C9 02 ;RS-232? F0 3F ;YES F177 BEQ $F188 F179 F179 *=$F1AD F1AD F1AD INPUT FROM SERIAL BUS F1AD F1AD A5 90 Ĺ939 LDA STATUS :STATUS FROM LAST F1AF F0 04 BEQ L941 ;0.K. A9 ØD F1B1 L.932 LDA NSOD ; BAD, ALL DONE F183 18 L946 CLC ; VALID DATA F124 L945 60 RTS F185 F185 4C 13 EE L941 JMP L865 :GOOD, HANDSHAKE F188 ``` ### **BSOUT** Entry point: \$FFD2 Function: Output the character stored in .A to the current output device. Input parameters: .A holds character Output parameters: .A holds same character, CARRY clear Registers used: None Error messages: None Description: This routine just jumps to the send buffered character to serial routine. outine. ``` LOC CODE LINE F188 *=$F1CA F1CA *************** F1CA ** BSOUT -- OUTPUT CHAR STORED IN .A TO F1CA ;* CHANNEL DETERMINED BY VARIABLE DFLTO: F1CA ; * F1CA 0 --- INVALID ; * F1CA 1 --- CASSETTE ;× F1CA 2 --- RS-232 F1CA ;× 3 --- SCREEN 4-31 --- SERIAL BUS F1CA F1CA ************ F1CA NESOUT PHA F1CA 48 ; PRESERVE .A ; CHECK DEVICE A5 9A F1CB LDA DFLTO C9 03 F1CD CMF #$03 ;SCREEN? F1CF BNE L933 ; NO DØ 04 YES, RESTORE .A F1D1 68 F'LA F102 4C 16 E7 JMP $E716 PRINT TO SCREEN F1D5 Ĺ933 F105 90 04 BCC $F1DB ; DEVICE 1 OR 2 F107 FRINT TO SERIAL BUS F107 F107 F107 FLA 68 4C DD ED JMP L861 F108 F1DB ; F1DB ţ F1DP .LIB KSER3 ``` #### CHKIN Entry point: \$FFC6 Function: Set a previously OPENed file for input. Input parameters: .X holds the logical file number of the OPENed file. Output parameters: CARRY clear - OK CARRY set - error, error number in .A Registers used: .A, .X #### Error messages: File not open – if the logical file number in .X is not in the LFN table Device not present – if bit 7 of ST is set, the device did not respond to the TALK command Description: This routine first checks that the LFN in .X has a reference in the LFN table. If not, the message File not open is sent. The device referenced by the LFN is told to TALK and a secondary address is sent (if present). After sending the TALK secondary address, the device is shifted over to listener. If bit 7 of the STATUS byte (ST) is set, the message Device not present is sent. ``` LOC CODE LINE *=$F20E FIUB F20E F26E * CHKIN -- OPEN CHANNEL FOR INPUT. F20E THE NUMBER OF THE LOGICAL FILE TO F20E ;* BE OPENED FOR INPUT IS PASSED IN .X. F20F ;* CHKIN SEARCHES THE LOGICAL FILE TO ;* LOOK UP DEVICE AND COMMAND INFO. F20E F20E * ERRURS ARE REPORTED IF THE DEVICE WAS ;* NOT OPENED FOR INPUT, (E.G. CASSETTE F20E ;* WRITE FILE), OR THE LOGICAL FILE HAS ;* NO REFERENCE IN THE TABLES. DEVICE 0, F20E F20E F20E * (KEYBOARD), AND DEVICE 3 (SCREEN), ;* REQUIRE NO TABLE ENTRIES AND ARE F20E * HANDLED SEPARATELY. F20E ************************************ F20E F20E F20E 20 0F F3 NCHKIN JSR L1000 :FILE OPENED? ;YES F211 F0 03 BEQ L950 ;NO, FILE NOT OPEN F213 4C 01 F7 JMF L1009 F216 20 1F F3 L950 JSR L1002 :GET FILE INFO F219 A5 BA LDA FA F21B FØ 16 BEQ L963 :KEYBOARD F21D COULD BE SCREEN, RS-232, UR SERIAL F210 F21D CMP #$03 C9 03 ;SCREEN? F21D FØ 12 BEQ L963 F21F ; YES, DONE F221 BØ 14 BCS L961 SERIAL F223 C9 02 CMF #$02 ;RS-232? ;NO, MUST BE TAPE ;RS-232 BNE L958 F225 DØ 03 4C 4D F0 F227 JMF $F04D F22A CHECK FOR INPUT FILE ON TAPE F22A F22A ; L958 A6 B9 LDX SA CHECK SECONDARY AD F22A ; INFUT? F220 EØ 60 CFX #$60 ; YES F22E FØ Ø3 BEO L963 ;NO, NOT INPUT FILE ;SET INPUT F230 4C 0A F7 JMP L971 85 99 L963 STA DFLTN F233 F235 18 CLC :GOOD RETURN F236 RIS 60 F237 ÷ ``` ``` LGC CODE LINE F237 ; A SERIAL DEVICE MUST TALK F237 Ĺ961 ;SAVE DEVICE # F237 AA TAX F238 20 09 ED JSR L836 ;TALK ;SECOND? F23B A5 B9 LDA SA F23D ;YES, SEND IT 10 06 BPL L962 20 CC ED ;NO, LET GO F23F JSR L970 F242 4C 48 F2 JMP L967 F245 20 C7 ED L962 JSR L860 :SEND TALK SA F248 88 L967 TXA 24 90 10 E6 SUTATE TIE F249 ;DID IT LISTEN? F24B YES BPL L963 F24D 4C 07 F7 JMF L1026 DEVICE NOT PRESENT ``` ### **CHKOUT** Entry point: \$FFC9 Function: Set a previously OPENed file for output. Input parameters: .X holds the logical file number of the OPENed file. ### Output parameters: CARRY clear - OK CARRY set - error, error number in .A Registers used: .A, .X #### Error messages: File not open – if the logical file number in .X is not in the LFN table Device not present – if bit 7 of ST is set, the device did not respond to the LISTEN command Description: This routine first checks that the LFN in .X has a reference in the LFN table. If not, the message File not open is sent. The device referenced by the LFN is told to LISTEN and a secondary address is sent (if present). If bit 7 of the STATUS byte (ST) is set, the message Device not present is sent. ``` LOC CODE LINE F250 * CHKOUT -- OFEN CHANNEL FOR OUTFUT. F250 F250 THE NUMBER OF THE LOGICAL FILE TO ;* BE OPENED FOR OUTPUT IS PASSED IN .X. F250 ;* CHKOUT SEARCHES THE LOGICAL FILE TO F250 * LOOK UP DEVICE AND COMMAND INFO. F250 * ERRORS ARE REPORTED IF THE DEVICE WAS F250 ;* NOT OPENED FOR INFUT, (E.G. KEYBOARD) F250 ;* OR THE LOGICAL FILE HAS NO REFERENCE F250 ;* IN THE TABLES. DEVICE 3 (SCREEN) F250 * REQUIRES NO TABLE ENTRY AND IS F250 ; * HANDLED SEPARATELY. F250 ***************** F250 ``` ``` LUC CODE LINE F250 NCKOUT JSR L1000 F250 20 OF F3 ;FILE IN TABLE; F253 FØ Ø3 BEQ L969 , YES JMF L1009 F255 4C 01 F7 , NO, FILE NOT OPEN F258 20 1F F3 L969 JSR L1002 GET TABLE INFO F258 A5 BA LDA FA F250 DØ 93 ENE L979 NOT KEYPOARD F25F 4C ØD F7 L972 JMF L965 KEYBOARD, NOT OUTFUT FILE F262 F262 ; COULD BE SCREEN, SERIAL, ; CASSETTE, OR RS-232 F262 F262 F262 C9 03 1.979 CMF #$03 ;SCREEN? ;YES, DONE ;NO, SERIAL ;RS-232? F264 FØ ØF PER L977 F266 BØ 11 BCS L975 F268 C9 02 CMF #$62 ;NO, MUST BE CASSETTE ;SET UP FOR RS-232 F26A DØ Ø3 BNE L973 4C E1 EF F26C JAF SEFE1 F26F CHECK FOR CASSETTE FILE TYPE F26F F26F Ĺ973 F26F A6 B9 LDX SA ; INPUT FILE? F271 EØ 60 CFX #$60 F273 BEO L972 ;YES, ERROR ;SET OUTFUT FØ EA F275 85 9A L977 STA DFLTO F277 CLC 18 GOOD RETURN F278 60 RTS F279 F279 SERIAL DEVICES F279 F279 Ĺ975 AA TAX ; SAVE DEVICE # F27A 20 0C ED JSR L966 ;LISTEN ; AND SECOND? F27D A5 B9 LDA SA F27F 10 05 BFL L976 ;YES ,NO, RELEASE LINES F281 20 BE ED JSR L983 ENE L981 ; ALWAYS F284 DØ Ø3 L976 :SEND LISTEN SA F286 20 B9 ED JSR L871 F289 88 L981 TXA 24 90 BIT STATUS ;DID IT LISTEN? F28A F28C 10 E7 BFL L977 ;YES, FINISH F28E 4C 07 F7 JMF L1026 ,NO, DEVICE NOT PRESENT F291 .END F291 .LIB KSER4 ``` #### **CLOSE** Entry point: \$FFC3 Function: Close a logical file. Input parameters: . A holds the logical file number to close. Output parameters: CARRY clear. Registers used: .A, .Y, .X Error messages: None. Description: The LFN table is checked for the file to be closed. If the file is not open the routine exits, otherwise the device is told to listen and then unlisten and the file entry is removed from the table. ``` LUC CODE LINE F291 ************** F291 ;* CLOSE -- CLOSE LOGICAL FILE. F291 F291 THE LOGICAL FILE NUMBER OF THE * FILE TO BE CLOSED IS PASSED IN .A. F291 * KEYBOARD, SCREEN, AND FILES NOT OFEN F291 * PASS STRAIGHT THROUGH. TAPE FILES F291 ;* OFEN FOR WRITE ARE CLOSED BY DUMFING F291 F291 * THE LAST BUFFER AND CONDITIONALLY * WRITING AN END OF TAPE BLOCK. SERIAL F291 ;* FILES ARE CLOSED BY SENDING A CLOSE F291 F291 ;* FILE COMMAND IF A SECUNDARY ADDRESS F291 * WAS SPECIFIED IN ITS UPEN CUMMAND. F291 F291 F291 20 14 F3 NCLOSE JSR L957 ;LOOK UP FILE F294 FØ 02 ; FOUND BEQ L982 F296 :ELSE RETURN 18 CLC F297 60 RTS F298 20 1F F3 L982 JSR L1002 GET FILE DATA F29B 8A TXA SAVE TABLE INDEX F290 48 F'HA AS BA F290 LDA FA CHECK DEVICE # F29F FØ 50 EEQ L987 ; KEYBOARD, DONE ;SCREEN? F2A1 C9 03 CMF #$03 F2A3 FØ 4C BEQ L987 ; YES, DUNE SERIAL F2A5 BØ 47 BCS L997 F2A7 C9 02 CMF #$02 ;RS-232? BNE $F2C8 F2A9 NO, MUST BE TAPE DØ 1D F2AB F2A8 *=$F2EE F2EE CLOSE A SERIAL FILE F2EE 20 42 F6 L997 F2EE JSR L1681 F2F1 F2F1 REMOVE FILE ENTRIES FROM TABLES F2F1 F2F1 68 L987 FLA :GET TABLE INDEX F2F2 AA L986 TAX F2F3 C6 98 DEC LOTNO F2F5 E4 98 CPX LDTND ; IS IT AT END? F2F7 FØ 14 BEQ L989 ;YES, DONE ;NO, SHIFT LAST ENTRY LDY LDTND F2F9 A4 98 F2FB B9 59 02 LDA LAT,Y ; INTO DELETED ENTRY'S F2FE 90 59 02 STA LAT,X ; POSITION F301 B9 63 02 LDA FAT,Y F304 9D 63 02 STA FAT, X LDA SAT,Y F307 B9 6D 02 F30A 9D 6D 02 STA SAT, X F300 18 L989 CLC :GOOD EXIT F30E 60 RTS ;FIND FILE ENTRY F30F F30F A9 00 L1000 LDA #$00 F36F STA STATUS F311 85 90 TXA F313 8A F314 A6 98 L957 LDX LDTND F316 L984 CA DEX BMI L969 F317 30 15 F319 DD 59 02 CMF LAT,X F31C D0 F8 BNE L984 F31E RTS 60 F31F FETCH TABLE ENTRIES F31F F31F LDA LAT,X F31F BD 59 02 L1002 ``` STA LA F322 85 B8 | LOC | CODE | LINE | |------------------------------|-------------------|--------------------------------------------------------| | F329<br>F320<br>F32E<br>F32F | 85 BA<br>BD 6D 02 | LDA FAT,X<br>STA FA<br>LDA SAT,X<br>STA SA<br>L960 RTS | | F32F | | .LIB KSER5 | ### CLALL Entry point: \$FFE7 Function: Close all logical files. Input parameters: None Output parameters: None Registers used: .A, .X Error messages: None Description: The number of files open is zeroed and the CLRCH routine is entered. ``` CODE LINE LOC F32F F32F ;* CLALL -- CLOSE ALL LOGICAL FILES. F32F F32F DELETES ALL TABLE ENTRIES AND ;* RESTORES DEFAULT I/O F32F CHANNELS AND * CLEARS SERIAL FORT DEVICES F32F , ********************************* F32F F32F F32F A9 00 NCLALL LDA #$00 F331 85 98 STA LDTND *FORGET ALL FILES ``` #### CLRCH Entry point: \$FFCC Function: Abort any serial I/O files and reset default I/O. Input parameters: None Output parameters: None Registers used: .A, .X Error messages: None Description: The output device is checked and if it is serial, the command UNLISTEN is sent to it. The input device is then checked and if that is serial the command UNTALK is sent to it. The input device is then set to \( \emptyset \) (keyboard) and the output device is set to 3 (screen). ``` LINE LOC CODE F333 ************** F333 * CLRCH -- CLEAR CHANNELS. F333 UNLISTEN OR UNTALK SERIAL F333 * DEVICES, BUT LEAVE OTHERS ALONE. F333 F333 ** DEFAULT CHANNELS ARE RESTORED. F333 F333 NCLRCH LDX #$03 F333 A2 03 ; DUTPUT CHANNEL SERIAL? F335 E4 9A CFX DFLTO BCS L1001 F337 80 03 ;NO F339 20 FE ED JSR L1006 YES, UNLISTEN E4 99 :INFUT CHANNEL SERIAL? L1001 F33C CPX DFLTN , NO F33E P.0 03 PCS L1603 ;YES, UNTALK 20 EF ED F340 JSR LB63 F343 86 9A L1003 STX DFLTO OUTPUT CHANNEL=3 F345 A9 00 LDA #800 :1NPUT CHANNEL=0 F347 85 99 STA DELTN F349 RTS 60 .END F34A F34A .LIB KSER6 ``` ### OPEN Entry point: \$FFCØ Function: Open a logical file. #### Input parameters: **\$B7** Length of text string to send with OPEN command (filename) - Logical file number **\$B8** \$B9 - Secondary address \$BA - Device number \$BB/\$BC - Pointer to filename ### Output parameters: CARRY clear - OK CARRY set - error, error number in .A Registers used: .A, .X, .Y #### Error messages: File open - if the file number in \$B8 is equal to any entry in the LFN table Too many files - if the LFN table already has 10 entries Device not present - if the device in \$BA did not respond to the LISTEN command Description: The LFN table is checked to see if the file number already exists (file open), and if so exits with error. The number of files open is then checked for ten (too many files), and if so exits with error. Otherwise, the file entry is submitted to the file tables (with the secondary address ORed with \$60) and the number of files open incremented. If there is no secondary address (>127), OPEN then exits. If there is no filename, OPEN exits. Otherwise, the device to be opened is told to LISTEN and the secondary address is sent. If bit 7 of ST is set, the routine exits with a device not present error. The filename is then sent one byte at a time and an UNLISTEN command is sent to the device. ``` LOC CODE LINE F34A F34A ************* ; * OPEN -- OPEN A FILE. F34A F34A CREATES AN ENTRY IN THE FILE ;* FILE TABLES CONSISTING OF LOGICAL F34A F34A * FILE NUMBER, DEVICE NUMBER, AND SEC ;* ADDRESS NUMBER. F34A F34A ;× ROUTINES SETLES & SETNAM SHOULD F34A ;* BE USED FIRST. F34A ************** F34A NOF EN ; CHECK FILE # F34A A6 B8 LDX LA F34C DØ 03 PNE L1005 ;NOT KEYBOARD ;NOT INPUT FILE 4C 0A F7 JMP L971 F34E 20 0F F3 L1005 ; ALREADY OFEN? F351 JSR L1000 F354 DØ 63 PNE L1007 ; NO F356 4C FE F6 JMF L1011 ; YES, FILE OPEN F359 A6 98 L1007 LDX LDTND :END OF TABLE? F358 E0 0A CPX #$0A F35D 90 93 PCC L1008 , NO F35F 4C F8 F6 JMP L1097 YES, TOO MANY FILES E6 98 L1008 F362 INC LOTNO , NEW FILE F364 A5 B8 LDA LA F366 9D 59 02 STA LAT, X STORE FILE # F369 A5 B9 LDA SA F368 69 60 DRA #$60 : MAKE SA SERIAL F360 85 B9 STA SA F36F 9D 6D 62 STA SAT,X ;STURE SA F372 AS BA LDA FA F374 9D 63 02 STA FAT, X :STURE DEVICE # F377 FERFORM DEVICE SPECIFIC OPEN TASKS F377 F377 FØ 5A ;KEYBOARD, DONE F377 BEO L1030 F379 C9 03 CMF #$03 ;SCREEN? ; YES, DONE F378 FØ 56 PEQ L1030 F370 90 05 BCC $F384 ;CASSETTE UR RS-232 ; OPEN SERIAL F37F 20 D5 F3 JSR L1021 ;ALWAYS, DONE F382 90 4F BCC L1939 F384 *=$F3D3 F384 F3D3 18 L.1030 CLC F3D4 60 L1012 RTS ;EXIT F3D5 OPEN SERIAL F305 F3D5 F305 A5 B9 L1021 LDA SA F3D7 30 FA BMI L1036 :NO SA. DONE F3D9 A4 87 LDY FNLEN F3DB F0 F6 BEO L1030 ;NO FILENAME, DONE ``` ``` LOC CODE LINE F300 A9 00 LDA #800 F3DF 85 90 STA STATUS LDA FA F3E1 A5 BA F3E3 20 0C ED JSR L966 IDEVICE LA TO LISTEN F3E6 A5 B9 LDA SA F3E8 09 F0 DRA #$FØ 20 B9 ED F3EA JSR L871 :DEVICE THERE? F3ED A5 90 LDA STATUS F3EF 10 05 BPL L1014 ;YES F3F1 68 FLA 3 NO F3F2 68 PLA F3F3 4C 07 F7 JMF L1026 DEVICE NOT PRESENT F3F6 A5 B7 L1014 LUA FNLEN :NO NAME, DONE F3F8 F0 0C BEQ L1033 F3FA SEND FILE NAME F3FA F3FA F3FA A0 00 LDY #$60 LDA (FNADR), Y F3FC B1 BB L1031 F3FE 20 DD ED JSR LB61 F401 C8 INY C4 B7 F402 CPY FNLEN F404 DØ F6 BNE L1031 F406 4C 54 F6 L1033 JMF L999 :UNLISTEN AND RETURN F409 .END F409 F409 .LIB KSER7 ``` #### LOAD/VERIFY Entry point: \$FFD5 Function: Load or verify a file from serial to RAM. #### Input parameters: \$B7 - Length of text string to send with OPEN command (filename) \$B8 - Logical file number \$B9 - Secondary address \$BA - Device number \$BB/\$BC - Pointer to filename .A - Load (∅)/verify (≠∅) flag .Y/.X - Alternative load address (only if \$B9=∅) #### Output parameters: OK - CARRY clear .Y/.X end address Error - CARRY set .A error number Registers used: .A, .X, .Y #### Error messages: Missing filename - if length of filename is zero File not found - if attempting to read the first byte gives a framing error Break - if the stop key was pressed Verify - if on verifying, the file does not match the memory contents Description: The alternative load address is stored away. If the filename length is zero, a missing filename error is produced. Otherwise, the message 'searching for ...' is printed to the screen and the file is opened (with SA=\$6\(\rho\)). The device is commanded to TALK and the first byte is read in and stored to the load address low. If bit 1 of ST is set (file not found), the file is closed and LOAD exits with error in .A. Another byte is read and stored in the load address high. If the original secondary address was zero, the load address is replaced by the alternative load address. The message 'loading' is printed to the screen and each byte is read in until an end of file (bit 6 of ST) is encountered or the stop key is pressed (break). With each byte, it is either stored to memory or compared with memory and if different, bit 4 of ST is set. The address is bumped by 1 and the next byte handled. When EOF has been found, the .X and .Y registers are loaded with the end address, CARRY is cleared and LOAD exits. ``` LOC CODE LINE F409 *=$F49E F49E ************ F49E F49E ;* LOAD RAM FUNCTION. ; <del>*</del> F49E LOADS FROM CASSETTE OR SERIAL BUS F 49F ;* DEVICES >=4 TO 31 AS DETERMINED BY F49E ;* CONTENTS OF VARIABLE FA. VERIFY FLAG F49E :* IN .A. F49E ; * ALT LOAD IF SA=0. NORMAL SA=1 .X, .Y LOAD ADDRESS IF SA=0 ; × F49E F49E .A=0 PERFORMS LOAD, <>0 IS VERIFY. F49E ;* HIGH LOAD RETURN IN .X..Y F49E ;* USE SETLES & SETNAM REFORE THIS ROUTINE F49E *************** F49E L990 F49E 86 C3 STX MEMUSS :LO ALT START F4A0 84 C4 STY MEMUSS+1 ;HI ALT START 6C 30 03 85 93 F4A2 JMP (ILOAD) F4A5 STA VERCK NLOAD STORE VERIFY FLAG LDA #$00 F4A7 A9 00 F4A9 85 90 STA STATUS F4AB A5 BA LDA FA :CHECK DEVICE # F'4AD D0 03 BNE L1046 4C 13 F7 F4AF L1241 JMP L1049 ;KEYBOARD, BAD DEVICE CMF #$03 F4B2 C9 Ø3 L1046 :SCREEN? F484 FØ F9 BEQ L1241 ; YES F486 90 7B BCC $F533 : TAFE F488 F488 ;LOAD FROM SERIAL BUS DEVICES F488 F448 A4 P.7 LDY FNLEN ; MUST HAVE FILENAME F4BA D0 03 BNE L1045 ;0.K. F4BC 4C 10 F7 JMP L974 ; MISSING FILENAME F4BF A6 89 L1045 LDX SA F4C1 20 AF F5 JSR L1062 ;'SEARCHING' F4C4 A9 60 LDA #$60 SPECIAL LOAD COMMAND F4C6 85 B9 STA SA F4C8 20 D5 F3 JSR L1021 OPEN FILE ``` ``` LOC CODE LINE F 4CB AS BA LDA FA F4CD 20 09 ED JSR L836 ; TALK, ESTABLISH CHANNEL F400 A5 B9 LDA SA F4D2 20 C7 ED JSR L860 ;TELL IT TO LOAD F405 20 13 E.E. JSR L865 GET FIRST BYTE F408 85 AE STA EAL F4DA A5 90 LDA STATUS :ERROR? F4DC 4A LSR A F4DD 4A LSR A F 4DE 80 50 BCS L1058 FILE NOT FOUND F4E0 20 13 EE JSR L865 F4E3 85 AF STA EAH F4E5 BA TXA ; ORIG SA=0? F4E6 DØ 08 BNE L1048 , NO F4E8 A5 C3 LDA MEMUSS YES, SET ALT F4EA 85 AE STA EAL ; LOAD ADDRESS F4EC A5 C4 LDA MEMUSS+1 F4EE 85 AF STA EAH ;'LOADING' F4F0 20 D2 F5 L1048 JSR L1070 F4F3 A9 FD L1051 LDA #$FD :MASK OFF TIMEOUT F4F5 25 90 AND STATUS F4F7 85 90 STA STATUS F4F9 20 E1 FF STOP KEY? JSR $FFE1 F4FC DØ 03 BNE L1055 ; NO F4FE ,'BREAK' 4C 33 F6 JNF L1084 20 13 EE F501 L1055 JSR L865 GET BYTE F504 AA TAX F505 A5 90 LDA STATUS :TIMEOUT? F507 4A LSR A F508 4A LSR A F509 60 E8 PCS L1051 :YES. TRY AGAIN F50B 88 TXA A4 93 F50C ; VERIFY? LDY VERCK F50E F0 0C BEQ L1053 ;NO, LOAD IT F510 A0 00 LDY #$00 CMF (EAL),Y F512 D1 AE ; VERIFY IT F514 FØ 08 BEQ L1056 ;0.K. A9 10 F516 LDA #SPERR ;NO, VERIFY ERROR F518 20 1C FE JSR $FE1C ;UPDATE STATUS F518 20 .BYT $2C SKIP STORE F51C 91 AE L1053 STA (EAL),Y F51E E6 AE L1056 INC EAL ; INCREMENT STORE ADDR F520 DØ 02 ENE L1057 F522 E6 AF INC EAH F524 24 90 L1057 BIT STATUS ;END OF INFUT? F526 50 CB ;NO, CARRY UN BVC L1051 F528 20 EF ED JSR L863 ; CLOSE CHANNEL 20 42 F6 F 52B JSR L1081 ;CLOSE FILE F52E 90 79 PCC L1067 ; ALWAYS F530 4C 04 F7 L1058 JMP L959 FILE NOT FOUND F533 • F533 *=$F5A9 F5A9 18 L1067 CLC :GOOD EXIT F5AA F5AA ; SET UP END ADDRESS F5AA F5AA A6 AE LDX EAL F5AC A4 AF LDY EAH F5AE 60 L1659 RTS F5AF F5AF FRINT 'SEARCHING EFOR NAMEJ' F5AF F5AF A5 9D LDA MSGFLG L1062 ;FRINT IT? , NO F581 10 1E BPL L1071 FSB3 AO OC LDY #$6C ; 'SEARCHING' 20 2F F1 F585 JSR $F12F F588 A5 B7 LDA FNLEN ``` ``` LOC CODE LINE F5BA FØ 15 BEQ L1071 F5BC AØ 17 LDY #$17 :'FOR' 20 2F F1 F5BE JSR $F12F F5C1 F5C1 PRINT FILE NAME F5C1 Ĺ1022 ; NAME? F5C1 A4 B7 LDY FNLEN , NO, DONE F5C3 FØ ØC PEQ L1671 F505 A0 00 LDY #900 B1 BB F5C7 L1091 LDA (FNADR),Y 20 D2 FF F5C9 JSR $FFD2 F5CC C8 INY FSCD C4 B7 CPY FNLEN F5CF DØ F6 BNE L1091 L1071 F501 60 RTS F5D2 PRINT LOADING/VERIFYING F302 F5D2 Ĺ1979 ; ASSUME 'LOADING' F502 A0 49 LDY #$49 F5D4 A5 93 LDA VERCK :CHECK FLAG BEQ L1052 ; YES, LOADING F506 FØ 02 F508 AØ 59 LDY #$59 .'VERIFYING' 4C 2B F1 L1052 JMP $F12B F5DA F5DD .END F500 .LIB KSER8 ``` ## SAVE Entry point: \$FFD8 Function: SAVE a section of memory to serial device. ## Input parameters: **\$B7** - Length of text string to send with open command (filename) - Logical file number **\$B8** \$B9 - Secondary address - Device number \$BA \$BB/\$BC - Pointer to filename - Pointer to zero page save address .A .Y/.X- End of save address (.A) - Page zero indirect start of save ## Output parameters: CARRY clear - OK CARRY set - error, error number in .A Registers used: .A, .X, .Y #### Error messages: Missing filename – if the length of the filename is zero Break - if the stop key was pressed during SAVE Description: The length of the filename is checked and if zero (missing filename), exits with error. The file is opened, the message 'saving ...' is printed to the screen and the device is told to LISTEN. The save address (low followed by high) is sent to the device. The start address is compared with the end address and if reached the file is closed and the routine exits with CARRY clear. One byte of the file is sent to the device and the stop key is tested. If the stop key was not pressed, the start address is bumped by 1 and the routine loops back to the address comparison. If the stop key was pressed, the file is closed and the routine exits with CARRY set. ``` LOC CODE LINE FSDD ************* F5DD ;* SAVE MEMORY FUNCTION. F5DD ; * F'5DD SAVES TO CASSETTE OR SERIAL F500 :* DEVICES >=4 TO 31 AS SELECTED BY ;* VARIABLE FA. F500 FSDD , × START OF SAVE IS INDIRECT AT .A ;* END OF SAVE IS .X, .Y F500 F500 * USE SETLES & SETNAM BEFORE THIS ROUTINE F5DD *************** F500 L1072 STX EAL F5DD 86 AE F5DF 84 AF STY EAH SET UP START F5E1 AA TAX FSE2 B3 00 LDA $00.X F5E4 85 C1 STA STAL F5E6 B5 01 LDA $01,X F5E8 85 C2 STA STAH FSEA 6C 32 03 JMP (ISAVE) F5ED NSAVE LDA FA A5 BA F5EF DØ 03 BNE L1075 F5F1 4C 13 F7 L1242 JMF L1049 ;BAD DEVICE F5F4 C9 03 CMP #803 L1075 SERIAL? F5F6 FØ F9 ; SCREEN. BEQ L1242 BAD DEVICE 90 SF ;NO, TAPE F5F8 BCC $F659 F5FA A9 61 ; YES LDA #$61 F5FC 85 B9 STA SA FSFE A4 B7 LDY FNLEN BNE L1074 F600 DØ Ø3 ; MISSING FILE NAME 4C 10 F7 F602 JMP L974 L1074 JSR L1021 F605 20 U5 F3 ;OPEN F608 20 BF F6 JSR L1087 'SAVING' F60B A5 BA LDA FA F60D 20 0C ED JSR L966 ;LISTEN F610 A5 B9 LDA SA F612 20 B9 ED JSR L871 ;LISTEN SA F615 A0 00 LUY #$00 F617 20 8E FB JSR $FBBE F61A AU AC LDA SAL JSR L861 F61C 20 DD ED F61F AS AD LDA SAH F621 20 DD ED JSR L861 F624 20 D1 FC L1077 :COMPARE START TO END JSR $FCD1 F627 BØ 16 BCS L1082 HAVE REACHED END F629 B1 AC LUA (SAL),Y F62B 20 DD ED JSR L861 ;STOP KEY? F62E 20 E1 FF JSR $FFE1 DØ 07 ;NO F631 BNE L1054 20 42 F6 L1084 F633 JSR L1081 ;YES, CLOSE F636 A9 00 LDA #$00 F638 38 SEC F639 60 RTS F63A L1054 F63A 20 DB FC JSR $FCDB :INCREMENT CURRENT ADDR BNE L1077 F630 DØ E5 ``` ``` LOC CODE LINE F63F 20 FE ED L1082 JSR L1006 :UNLISTEN F642 24 89 L1081 BIT SA 30 11 F644 BMI L1034 F646 AS BA LDA FA F648 20 0C ED JSR L966 :LISTEN F64B A5 B9 LDA SA F64D 29 EF AND #$EF F64F 09 E0 URA #$EØ F651 20 B9 ED JSR L871 ;LISTEN SA L999 F654 20 FE ED JSR L1006 UNLISTEN F657 18 L1034 CLC ; GOOD EXIT F658 RIS F659 *=$F68E F 659 F68E 60 L1096 RTS F 68F FRINT 'SAVING CFILE NAMEJ' 1486 f F 68F F68F L1087 A5 9D LDA MSGFLG :PRINT IT? 10 FB F691 BFL L1090 ; NO F693 AØ 51 LDY #$51 ; 'SAVING' F695 20 2F F1 JSR $F12F F698 4C C1 F5 JMP L1022 ; SEND FILENAME F698 .END F 69B .LIB KSER9 ``` ## Error handler Entry point: \$F6FB - (1) too many files F6FE - (2) file open F701 - (3) file not open F704 - (4) file not found F707 - (5) device not present F70A - (6) not input file F70D - (7) not output file F710 - (8) missing filename \$F713 - (9) bad device Function: To flag an error and print it if output is enabled. Input parameters: None Output parameters: CARRY set, error number in .A Registers used: .A, .X, .Y Error message: I/O error #(number) - if bit 6 of MSGFLG is set Description: At each entry point, the .A register is loaded with the value in brackets. The routine CLRCH is called and if bit 6 of MSGFLG (output enable) is clear, CARRY is set and .A holds the error number upon exit. If bit 6 is set, the message 'I/O error #' is printed to the screen and the number is converted to ASCII and printed. CARRY is set and .A is reloaded with the error number. LINE ``` F698 *=$F6F8 F 6FB ****************************** F 6FB ;* ERROR HANDLER. F6FB ** FRINTS KERNAL ERROR MESSAGE IF BIT 6 F6FB ;* OF MSGFLG IS SET. RETURNS WITH ERROR F6FB ;* # IN .A AND CARRY SET. F6FB F6FB F6FB L1097 LDA #$01 AY 01 :TOO MANY FILES F6FD 2C .8YT $2C F6FE A9 02 L1011 LDA #$02 ;FILE OPEN F700 2C .BYT $2C F701 A9 03 L1009 LDA #$03 FILE NOT OPEN F703 20 .BYT $2C F704 A9 04 L959 LDA #$04 FILE NOT FOUND F706 2C .BYT $20 F707 A9 05 L1026 LDA #$05 DEVICE NOT PRESENT F709 20 .BYT $2C F70A A9 06 L971 LDA #$06 ;NOT INFUT FILE F70C 20 .BYT $2C F70D A9 07 L965 LDA #$07 ;NOT OUTPUT FILE F70F 20 .BYT $20 F710 A9 08 L974 LDA #$08 MISSING FILE NAME F712 20 .BYT $20 F713 A9 09 L1049 LDA #$09 ;BAD DEVICE # F715 48 F'HA ;ERRUR # UN STACK F716 20 CC FF JSR $FFCC RESTORE I/O LDY #$00 F719 A0 00 F718 24 9D BIT MSGFLG FRINT ERROR? F710 50 0A ; NO BVC L1018 F71F 20 2F F1 JSR $F12F :FRINT 'I/O ERROR #' F722 68 FLA F723 48 PHA F724 09 30 ORA #$30 ;MAKE ERROR # ASCII F726 20 D2 FF JSR $FFD2 FRINT IT F729 68 L1018 FLA F72A 38 SEC F72P 60 RTS F720 F720 .END F720 .END ``` #### Symbol table LOC CODE | SYMBOL | VALUE | | | | | | | |--------|--------------|--------|--------------|--------|-------------|--------------|--------------| | AUTODN | 9292 | BAD | 0100 | BASZFT | 00FF | PAUDOF | <b>0299</b> | | BDF | 0002 | BDFH | 0004 | BITCI | 8A00 | BITNUM | 0298 | | BITTS | 0084 | BLF | 0001 | BLNCT | 00CD | <b>BLNON</b> | 00CF | | BLNSW | 99CC | BLUE | 0006 | BSOUR | 0095 | BSOUR1 | 00A4 | | BUF. | 0200 | BUFFT | 00A6 | BUFSZ | 90C9 | C3F0 | 0094 | | CAS1 | 90C <b>9</b> | CRINV | 0316 | CBMCOL | D800 | CRMSCN | 0466 | | CINV | 0314 | CKERR | 0020 | CMPO | 9989 | CNTDN | 00A5 | | COLM | DC00 | COLOR | 0286 | COUNT | 00A5 | CR | 000D | | CRSW | 0000 | D1CRA | DCØE | DICRE | DC0F. | D1DDRA | DC02 | | DIDDRE | DC03 | D1DFA | DC00 | DIDFE | DCØ1 | DIICR | DCØD | | 011008 | DC0C | DITAH | DC05 | DITAL | DC04 | D1TBH | DC07 | | DITBL | DC06 | D1T0D1 | DC08 | D1T0D2 | DC09 | D1TOD3 | DC0A | | 011004 | DCOR | D2CRA | DDØE | D2CRB | DD0F | D2DDRA | 0002 | | D2DDRB | DD03 | D2DFA | DD00 | D2DF® | DDØ1 | D21CR | DD&D | | 021008 | DDOC | D2TAH | DD05 | D2TAL | DD04 | D2TBH | DD07 | | DZTBL | DD@6 | D2T0D1 | DD68 | D2T0D2 | DD&A | D2T0D3 | DDØA | | 021004 | | DATA | <b>めめ</b> Dフ | DELAY | 028C | DFLTN | 0099 | | DFLTO | 009A | DIFF | 66B2 | DPSW | 669C | EAH | 00AF | | EAL | OUAE | EOT | 0005 | FA | 60BA | FAT | 0263 | | FIRT | 60A4 | FNADR | 00BB | FNLEN | 60B7 | FREKZP | 00F B | | FSBLK | OORE | ODELN | 00CE | GUCUL | <b>0287</b> | HIBASE | <b>0</b> 289 | | SYMBOL V | AL UF | | | | | | | |-----------------|--------------|------------------|--------------|-----------------|--------------|------------------|--------------| | 1BASIN | 0324 | IBSOUT | 0326 | 1CHKIN | <b>031E</b> | ICKOUT | 0320 | | ICLALL | 032C | ICLUSE | 031C | ICLRCH | 0322 | IGELIN | 032A | | ILUAD | 0330 | INBIT | 00A7 | INDX | 00C8<br>0332 | INSET<br>ISTOP | 0008<br>0328 | | IOPEN<br>KEYD | 031A<br>0277 | IRQYMP<br>KEYLOG | 029F<br>028F | ISAVE<br>KEYTAB | 0332<br>06F5 | KOUNT | 028B | | L1000 | F30F | L1001 | F33C | L1002 | F31F | L1003 | F343 | | L1004 | EE03 | L1005 | F351 | L1006 | EDFE | L1607 | F359 | | L1008 | F362 | L1009 | F701 | L1011 | F6FE | L1012 | F3D4 | | L1014 | F3F6 | L1018 | F729 | L1021 | F 305 | L1022 | F5C1 | | L1026 | F707 | L1030 | F3D3 | L1031 | F3FC | L1033 | F406 | | L1034<br>L1049 | F657<br>F713 | L1045<br>L1051 | F48F<br>F4F3 | L1046<br>L1052 | F4B2<br>F5DA | L1048<br>L1053 | F4F0<br>F51C | | L1054 | F63A | L1055 | F501 | L1056 | F51E | L1057 | F524 | | L1058 | F530 | L1059 | FSAE | L1062 | F5AF | L1067 | F5A9 | | L1070 | F 5D2 | L1071 | F5D1 | L1072 | F5DD | L1074 | F 605 | | L1075 | F5F4 | L1077 | F624 | L1081 | F642 | L1082 | F63F | | L1084 | F633 | L1087 | F 68F | L1090 | F 68E | L1691 | F507 | | L1097 | F 6FB | L1241 | F 4AF | L1242 | F3F1 | L836 | ED09 | | L839 | ED2E | L846 | ED50 | L841 | EEA0 | L842 | ED36 | | L843<br>L847 | EDB0 | L844<br>L848 | EE97<br>ED66 | L845<br>L849 | EEB6<br>ED55 | L846 | EEB3 | | L851 | ED2A | L852 | EDB2 | L853 | ED70 | 1.850<br>1.854 | ED5A<br>EEA9 | | L855 | ED9F | L856 | EDAD | L858 | EE06 | L859 | ED40 | | L860 | EDC7 | L861 | EDDD | L862 | EDEB | L863 | EDEF | | L864 | ED20 | 1.865 | EE13 | L866 | EE20 | L867 | EE47 | | L868 | EE3E | L869 | EE5A | L870 | EES6 | L.871 | EDB9 | | l.872 | E.E.30 | L873 | EE67 | L874 | EE80 | L875 | EE85 | | L876 | EE.09 | L924 | F14A | L926 | F14E | L927 | F166 | | L928 | F173 | L932 | F1B1 | 1.933 | F1D5 | L939 | F1AD | | L941<br>L946 | F185 | L943 | EE18 | L944 | F155 | L945 | F184 | | L958 | F1B3<br>F22A | L949<br>L959 | EDE6<br>F704 | L950<br>L960 | F216 | L957 | F314 | | L962 | F 245 | L963 | F 233 | L965 | F32E<br>F70D | L961<br>L966 | F237<br>ED60 | | L967 | F248 | L968 | EDD6 | L969 | F258 | L970 | EDCC | | L971 | F70A | L972 | F 25F | L973 | F 26F | L974 | F710 | | L975 | F279 | L976 | F286 | L977 | F275 | L979 | F262 | | L980 | ED11 | L981 | F289 | L982 | F298 | L983 | EDBE | | L984 | F316 | L986 | F2F2 | L987 | F2F1 | L989 | F30D | | L990 | F49E | L997 | F2EE | L999 | F654 | LA | 00B8 | | LA!<br>Linimp | 0259 | LBERR | 8999 | LDTB1 | 00D9 | LDTND | 0098 | | LSTF | 00F2<br>00CA | LLEN<br>LSTSHF | 0028<br>028E | LLEN2 | 0050 | LNMX | 00D5 | | LTBLUE | 900E | M26AJB | 0295 | LSTX<br>M2&CDR | 0005<br>0294 | LSXP | 0009 | | MAXCHR | 0050 | MEMS1Z | 0283 | MEMSTR | 0274 | M26CTR<br>MEMUSS | 0293<br>0003 | | MODE | 0291 | MSGFLG | 0090 | MYCH | 00PF | NEASIN | F157 | | NESOUT | F1CA | NCHKIN | F20E | NCKOUT | F 250 | NCLALL | F32F | | NCLOSE | F291 | NCLRCH | F333 | NDX | 99C4 | NGETIN | F13E | | NLINES | 0019 | NLOAD | F 4A5 | VMINV | 0318 | NOFEN | F34A | | NSAVE | F5ED | NWRAP | 0002 | NXTBIT | 00B5 | OCHAR | 0080 | | PONTR<br>PRP | 0083<br>0086 | PLF<br>PRTY | 0003 | PNT | 00D1 | PNTR | 00D3 | | arsw | 00D4 | R2D2 | 009B | PTR1 | 009E | PTR2 | 009F | | REZ | 0007<br>00A9 | RIBUF | 00A3<br>00F7 | RDFLG<br>RIDATA | 00AA<br>00AA | RER<br>RIDBE | 00A8<br>029B | | KIDBS | 029C | RINONE | 00A9 | RIFRTY | 00AB | ROBUE | 00F9 | | RODATA | 9888 | RODBE | 029E | RODES | 029D | ROPRTY | 00BD | | ROWS | DC01 | RFTFLG | 028A | RSSTAT | 6297 | RVS | 00C7 | | SA | 00B9 | SAH | OOAD | SAL | 00AC | SAT | 026D | | SBERR<br>SHFLAG | 0004 | SFDX | 69CB | SHCNH | 00AB | SHCNL | 00A7 | | STAH | 028D<br>0002 | SIDREG<br>STAL | D400 | SNSW1 | 9984 | SPERR | 0010 | | SVXT | 0092 | SYNO | 0001<br>0096 | STATUS<br>11 | 009E | STKEY | 9991<br>0095 | | TAPE1 | 00B2 | TBLX | 0006 | TBUFFR | 933C | T2<br>Temp | 009F<br>00B1 | | TIME | 00A0 | TUONIT | 0285 | TMP2 | 99C3 | 1MPC | 009F | | TMFO | 00C1 | USER | 00F3 | USRCMD | 032E | VERCK | 0093 | | VICREG | D000 | XMAX | 0289 | XSAV | 0097 | | | | | | | | | | | | #### 3.4 RS232 serial communications The CBM 64 is able to communicate with peripheral devices, known as an RS232 I/O port. The name RS232 simply refers to an industry standard form of serial communication for computing devices. A serial I/O port can consist of as few as three lines, an output or transmit line, an input or receive line and a common ground line. The data is transmitted or received as a stream of pulses; a single byte becomes a string of eight pulses. Although a serial port can have just three lines, other lines are frequently used to transfer control information. The 64 is able to receive and generate such control signals to implement a full 'X line' interface as well as the simple '3 line' interface. Whichever implementation is used all the lines are connected to I/O port B of CIA#2 (user port). The RS232 routines inside the 64 also use two other lines; PA2 on port A and FLAG which is connected to the NMI line. Normally an RS232 interface card will be used to connect between the user port and a standard RS232 connector. The card will also provide buffering and a higher drive voltage. For communications using the simple 3 line mode an interface card can easily be constructed using a couple of buffer/driver ICs. The RS232 line normally transmits data using a 12 volt signal, however, and providing cables are kept short it will work with a 5 volt signal. The standard RS232 connector is shown in Fig. 3.4. The function and pin assignment of each of these lines is as follows: | CIA<br>line # | RS232<br>pin # | CIA<br>pin # | Abv | EIA | In/<br>Out | Modes | Function | |---------------|----------------|--------------|------|-----|------------|-------|----------------------| | GND | 1 | Α | GND | AA | | 1,2 | Protective ground | | FLAG | 3 | В | SIN | BB | In | 1,2 | Received data | | PBØ | 3 | C | SIN | BB | In | 1,2 | Connected to FLAG | | PB1 | 4 | D | RTS | CA | Out | 2 | Request to send | | PB2 | 2Ø | E | DTR | CD | Out | 2 | Data terminal ready | | PB3 | 18 | F | RI | CE | In | 3 | Ring indicator | | PB4 | 8 | Н | DCD | CF | In | 2 | Received line signal | | PB5 | Not ass | signed | | | | | _ | | PB6 | 5 | K | CTS | CB | In | 2 | Clear to send | | PB7 | 6 | L | DSR | CC | In | 2 | Data set ready | | PA2 | 2 | M | SOUT | BA | Out | 1,2 | Transmitted data | | GND | 7 | N | GND | AB | | 2,3 | Signal ground | | | | | | | | | | #### Modes: - 1 Three line interface (note RTS and DTR are both held high during this mode) - 2 X line interface - 3 User only, not implemented in the CBM 64 code | PIN | TYPE | RS232 FUNCTION | |-----|--------------|----------------| | Α | | | | В | FLAG | ارا | | С | P <b>B</b> Ø | | | D | PB1 | - RTS | | Ε | PB2 | — DTR | | F | PB3 | Ri | | н | PB4 | - DCD | | J | | | | ĸ | PB6 | - cts | | L | PB7 | DSR | | м | PA2 | - Sout | | N | GND | GND | PIN | | | 1 | Pratective Ground | AA | |-----|-----|-----|---------------------|----| | | | 2 | Transmitted Data | BA | | | | 3 | Received Data | 88 | | | | 4 | Request To Send | CA | | | _ | . 5 | Clear To Send | СВ | | | 10 | 6 | Data Set Ready | CC | | 014 | 20 | 7 | Signal Ground | AB | | O15 | | 8 | Carrier Detect | CF | | 016 | 30 | 9 | (not used) | | | 1 | 40 | 10 | <i>,,</i> | | | 017 | 50 | 11 | •• | | | 018 | | 12 | " | | | 019 | 60 | 13 | " | | | 020 | 70 | 14 | " | | | 1 | 80 | 15 | " | | | 021 | 90 | 16 | " | | | 022 | - 1 | 17 | •• | | | 023 | 100 | 18 | " | | | 1 | 110 | 19 | " | | | 024 | 120 | 20 | Data Terminal Ready | CD | | 025 | - 1 | 21 | (not used) | | | _ | 130 | 22 | " | | | | _ | 23 | " | | | | | 24 | •• | | | | | 25 | " | | | | | | | | Fig. 3.4. CBM 64 RS232 connector, pin allocations and E/A line coding. The implementation of the RS232 port on the 64 is very interesting since it involves the use of software (originally used on the VIC 200 with very few modifications for the 64) to emulate a hardware device (that was never used). This device was called the 6551 universal asynchronous receiver and transmitter or UART. Like the other I/O chips, it was intended that the 6551 functions were to be controlled by registers at specific memory locations. The software uses the same principle because when it was written for the VIC 20, Commodore intended to replace the software with the 6551 when it became available, so there would be complete compatibility. The pseudo registers are located in various parts of the variable storage area at the bottom of CBM 64 memory. Besides the registers, the RS232 routines require two 256 byte buffers; one for received data and one for data to be transmitted. The 512 bytes of memory occupied by these buffers are located at the top of available RAM memory, and the starting address of the two buffers is stored in four register bytes. The two most important registers are the control and command registers. These determine the exact operation of the RS232 port. They can be summarised as follows: ## 3.4.1 RS232 control register – Hex \$\$923 Decimal 659 The function of the control register (Fig. 3.5) is to set the speed of data transmission and reception and set the number of bits needed to transmit each character. The speed at which data is input or output is called the baud rate, and Fig. 3.5. Function of bits in the CBM 64 RS232 control register. the value assigned to this is the number of bits per second. If the baud rate is set to 300 baud, and each character is transmitted as the eight character bits plus one stop bit and one parity bit – a total of ten bits – then 30 characters will be transmitted every second. The selected baud rate depends on the specifications of the device communicating with the 64 via the RS232 port – check the manual of the device before setting this value. Bits 5, 6 and 7 control the number of bits needed to transmit or receive data between the 64 and a peripheral. The number of bits per character plus the number of stop bits depends on the peripheral. # 3.4.2 RS232 command register - Hex \$\phi\$294 Decimal 66\psi The command register (Fig. 3.6) controls the mode for data transmission and reception. Bit Ø sets the mode; a 3 line mode or an X line mode. Bit 4 sets the duplex mode as follows: Full duplex - simultaneous transmission and reception of data Half duplex - alternate transmission and reception of data Fig. 3.6. Function of bits in the CBM 64 RS232 command register. Bits 5, 6 and 7 determine the nature of the parity bit and whether the mark or space is transmitted. The parity bit is transmitted after the data bits and has an error checking function. The choice of whether the parity is disabled or set to odd or even depends on the communicating peripheral. The mark/space setting determines whether a logic '1' is transmitted as a zero voltage or a positive voltage. # 3.4.3 RS232 status register - Hex \$\$\phi\$297 Decimal 663 The function of each bit of the status register is shown in Fig. 3.7. The memory locations and pseudo registers of the RS232 routines are as follows: #### 74 The Commodore 64 Kernal and Hardware Revealed RS-232 STATUS REGISTER - \$02A1 Fig. 3.7. Function of bits in the CBM 64 RS232 status register. \$A7 - receiver bit storage \$A8 - receiver bit count \$A9 - receiver flag start bit check \$AA - receiver byte buffer \$AB - receiver parity storage - transmitter bit count \$B4 - transmitter next bit **\$B5** - transmitter byte buffer \$B6 \$BD - transmitter parity storage \$F7-\$F8 - input buffer pointer \$F9-\$FA - output buffer pointer \$0293 RS232 control register \$0294 - RS232 command register - RS232 status register \$Ø297 **\$**Ø298 - number of bits to send/receive \$Ø299-- baud rate \$Ø29A \$Ø29B - input buffer index to end \$Ø29C - input buffer index to start - output buffer index to start \$Ø29D - output buffer index to end \$Ø29E ## 3.4.4 RS232 system routine entry points \$EEB1 – entry for NMI continue routine \$EED7 - calculate parity \$EFØØ - count stop bits \$EFØ6 - entry to start of byte transmission **\$EF13** - set up to send next byte \$EF2E set errors SEF4A - calculate number of bits to be sent **\$EF59** - NMI routine to collect data into bytes \$EF63 - calculate parity **\$EF69** - shift data bit in SEF6E - have stop bit so store in buffer \$EF7E - enable to receive a byte - receiver start bit check \$EF90 \$EF97 - put data in buffer (at parity time) SEFB3 - parity checking - check calculated parity **SEFBC** \$EFC7 - errors reported - output a file over RS232 SEFE1 \$EFE9 - check for DSR and RTS \$EFF2 - check for active input \$EFF9 - wait for CTS to be off - turn on RTS **SEFFE** \$FØØ6 - wait for CTS to go on \$FØ14 - buffer handler to output a character \$FØ28 - set up if necessary to output - input a file over RS232 \$FØ4D - check for DSR and not RTS \$FØ59 \$FØ62 - wait for active output to be done \$FØ68 turn off RTS - wait for DCD to go high \$FØ7Ø - enable FLAG for RS232 input \$FØ77 \$FØ7D - if not 3 line half then see if we need to turn on FLAG - input a character buffer handler \$FØ86 - receiver always runs ## 3.5 Using the RS232 port \$FØ9B \$FØA4 # 3.5.1 Opening an RS232 channel Basic syntax: OPEN If,2,0,"(control register) (command register)" The syntax coding is as follows: - protect serial/cassette from RS232 NMIs If - normal logical file ID (1-255). If If>127 then line feed follows carriage return. (control register) - an ASCII character equivalent to the required bit setting of the control register. Example: to set baud rate to 300 and transmit 7 bit code use CHR\$(6+32) – this sets bits 1, 2 and 5 to '1' and leaves the rest at ' $\emptyset$ '. (command register) - an ASCII character equivalent to the required bit setting of the command register. Example: to set the output to mark parity and full duplex use CHR\$(32+128) – this sets bits 5 and 7 to '1' and leaves the rest at '\( \textit{0}'\). Entry point: \$FFCØ #### 76 The Commodore 64 Kernal and Hardware Revealed Notes on usage: Only one RS232 channel should be open at any time. Since the OPEN statement resets the buffer pointers, a second OPEN will destroy any data in the buffers set up in the first OPEN. The OPEN RS232 channel command should be used before any variable DIM statements; failure to do this will cause wiping of data. This is because the OPEN RS232 channel command performs an automatic CLR before allocating the 512 bytes at the top of memory used for the two RS232 buffers. ## 3.5.2 Receiving data from an RS232 channel Basic syntax: GET #lf,(string variable) lf - logical file ID used in OPEN RS232 channel command. #### Entry points: \$FFC6 – set channel for input. Handles full X line implementation according to EIA standard RS232C interfaces. The RTS, CTS, and DCD lines are implemented when the CBM 64 is designated as a data terminal device. \$FFE4 – get character from buffer. Notes on usage: Received data is put into the 64's 256 byte internal receiver buffer set up during the OPEN routine. Data input is under control of the 6526 timers and NMIs, and is performed in the background during the running of a Basic program. This is done by having the RS232 data input line connected to the FLAG handshake line, and input on FLAG will generate an NMI system interrupt. The use of NMI interrupts is the reason why the cassette and serial bus should not be used during RS232 data communications. The NMI will call the serial data input routines whenever data is present on the RS232 input. These routines will place the received data into the 256 byte receiver buffer located at the top of RAM memory. If the input data has a word width less than eight bits then all unused bits will be filled with zero. The receiver buffer is organised as a first in first out – FIFO – buffer. The buffer removes the necessity for Basic to wait for data input before processing each byte of data. Instead the Basic program can take data from the buffer when it needs it rather than when it is presented. Basic accesses the buffer using the GET# command to transfer a single byte of data into a Basic variable. If there is no data in the buffer then the GET# command will return with a null character. If the buffer should overflow then all characters are lost. An overflow condition is indicated by bit 2 in the RS232 status register being set. An overflow condition will frequently result if an attempt is made to input data at fairly high data rates using Basic. This is because Basic is normally slow and the use of the GET# command with string concatenation will give rise to frequent garbage collects. Machine language routines are best used for data rates above the normal 300 baud. ## 3.5.3 Transmitting data to an RS232 channel Basic syntax: CMD If PRINT#If,(variable list) lf - logical file ID set up in the OPEN command. #### Entry points: \$FFC9 - set channel for output. This handles X line handshaking for the implementation of an EIA standard RS232 interface. The RTS, CTS, and DCD lines are implemented with the 64 as a data terminal. \$FFD2 - output character to channel. Notes on usage: When either one of the two Basic commands is used data is first transferred from the assigned string or memory block to the 256 byte transmitter buffer. From here it is output to the RS232 channel using the format and baud rate assigned in the OPEN command. Data output is transparent to the operation of Basic since the timing is done by the 6526 timers and output of each byte initiated by an NMI system interrupt. As with data input on the RS232, the cassette or serial port should not be used during data transmission otherwise interrupt conflicts will occur. There is no carriage return delay implemented by the output channel, therefore a normal RS232 printer cannot correctly output the data unless some form of internal buffering or other holdoff is implemented by the printer. If a CTS handshake is implemented (in the X line mode) then the 64 buffer will fill, and output will not occur until transmission is allowed by an input on CTS. ## 3.5.4 Closing an RS232 data channel Basic syntax: CLOSE If lf - logical file ID set up in the OPEN channel command. Entry point: \$FFC3 - close logical file Notes on usage: Closing the RS232 file causes all the data in the buffers to be discarded, stops data transmitting or receiving, sets the RTS and SOUT lines high, and deallocates the memory area used for the RS232 buffers. Closing an RS232 file will also allow the cassette or serial ports to be used. Before closing the channel care should be taken to ensure that all data in the buffer is transmitted. This can be done by checking the status (ST variable) is=Ø and that bit Ø of the RS232 enable register at location 673 (\$\psi 2A1) is set to logic 1. If both are true then there is still data in the buffer. # Chapter Four # The Cassette Units #### 4.1 The cassette hardware The CBM 64 has a single external cassette unit which is used for program and data storage. The cassette deck is connected to the CBM 64 by six lines – write, read, motor, sense and two power lines; ground and +5 volts. The connections are shown in Fig. 4.1. The cassette is controlled by I/O lines from the the CIA chip and the processor I/O register. The source of each of the cassette control lines is shown in Fig. 4.2. The cassette motor power supply lines are connected to the processor chip via a three transistor driver, used to boost the power and voltage, allowing the motor to be driven directly. The output to the motor is an | PIN# | TYPE | |------|-----------------| | A-1 | GND | | B-2 | + 5V | | C-3 | CASSETTE MOTOR | | D-4 | CASSETTE READ | | E-5 | CASSETTE WRITE | | F-6 | CASSETTE SWITCH | Fig. 4.1. The allocation and function of pins on the cassette connector. Fig. 4.2. The cassette circuit and its connection to the 6522 chips. unregulated +9 volts at a power rating of up to 100 mA. The cassette deck motor can be turned on and off by toggling line 5 of the processor I/O register: POKE 1, PEEK(1) AND 191 turns the motor on POKE 1, PEEK(1) OR 64 turns it off Great care should be taken not to alter the status of bits 1 and 2 of location 1 when using this command, since these control the memory configuration of the machine. The sense line input, line 4 of the processor I/O register, is connected to a switch on the cassette deck which senses when the play, rewind or fast forward buttons have been pressed. The switch is only required to sense the pushing of the play button during a read or write to tape routine; this is done by a subroutine at \$F82E. If either the rewind or fast forward button is pressed accidentally instead of the play button, the system will be unable to tell the difference and will act as if the play button has been pressed. For a similar reason during a record routine the record button must be pressed before the play button, since recording will start as soon as the sense switch is closed by pressing the play button. The cassette read line is connected to the negative edge sensitive serial input line of CIA#1 and the cassette write line to line 3 of the processor I/O register. During a read operation the operating system uses the setting of the CIA#1 interrupt flag to detect transitions on the cassette read line. The functioning of the read and write lines is controlled entirely by the operating system, the only hardware required being signal amplification and pulse shaping circuitry. These circuits are contained on a small PC board within the cassette deck, their function being to give correct voltage and current to the record head and amplify the input from the read head to give a 5 volt square wave output, able to produce an interrupt on the FLAG or CB1 lines. #### 4.2 Cassette operation In normal usage the cassette deck is assigned an I/O device number. The cassette is device number 1, and the number of the device currently being used is stored in location 186. The device number together with the logical file number and the secondary address is used when saving or retrieving data files from the cassette deck. The logical file number can be any number from 1 to 255 and is used to allow multiple files to be kept on the same device. It is of little use with cassette tape and is intended primarily for use with floppy disk units. It is usual to have the logical file number the same as the device number; the logical file number of the current file is stored in location 184. The secondary address is important since it determines the operational mode of the cassette; the current secondary address is stored in location 185, the normal default value being zero. If the secondary address is zero then the tape is opened for a 'read' operation, if it is set to 1 then it is opened for a 'write' operation, and if 2 then it is opened for a 'write' with an end of tape header being forced when the file is closed. The CBM 64 operating system is configured to allow two different types of file to be stored on cassette: program files and data files. These names are rather misleading, however, since a program can be stored as a data file and data can be stored as a program file. The difference between these two file types is not in their application but in the way the contents of the machine's memory are recorded. Instead of program and data files we must look upon them as binary and ASCII files. #### 4.3 Binary files A binary file is usually used to store programs, since a binary file is created by the operating system to store the contents of memory between a starting location and an end location. It is called a binary file because it stores on tape the binary value in each memory location within the assigned memory area. Basic statements are stored in memory using tokens. The use of tokens means that Basic commands are not stored in the same manner as they are listed on the display or were entered on the keyboard. They are instead stored in memory in a partly encoded form. Being partly encoded, a binary file is a quicker and more efficient way of storing programs. Binary files are essential when saving and loading machine code programs. The starting address from which a binary file will be saved is stored in locations 172 and 173. These locations are loaded by the SAVE routine with the memory location at which the SAVE will begin. Normally they will be set to $\emptyset$ 1 and $\emptyset$ 8, thereby pointing to the start of the Basic text area at $2\emptyset$ 49. They can be altered by the SAVE routine to point to any location in memory. The end address of the area of memory to be saved is stored in locations 174 and 175. Normally when saving a Basic program these are set to the address of the double zero byte terminating link address. The end address can be altered to any desired location. To change either of these addresses one cannot use the normal SAVE routine since this automatically initialises these locations. Instead one must write a small machine code initialisation routine incorporating the desired operating system subroutines. By default a SAVE command will write a binary file and a LOAD command will read a binary file. #### 4.4 ASCII files An ASCII file is normally used to store data but it can be used to store programs (see the MERGE procedure). The format is the same as that displayed on the screen or entered on the keyboard. ASCII files are created or read almost exclusively by instructions from within a Basic program. A binary file is created or read mostly by direct instructions, though the LOAD and SAVE instructions can be used within a program. An ASCII file must first be opened with an OPEN statement. This specifies the logical file, device number, secondary address and filename. The operating system interprets these parameters and allows the user to read or write the file to the specified device. Data is written to an ASCII file on a particular device with a command to PRINT to the specified logical file number, and data is read by a READ from logical file command. Whereas a binary file is loaded with the contents of successive memory locations, an ASCII file is loaded with a string of variables. Storing these would require the tape to be turned on and off repeatedly, storing a few bytes of data at a time. The CBM 64 overcomes this by having a 192 byte tape buffer into which all data to be written to or read from tape is loaded. Only when this buffer is full is the tape motor turned on. Data is stored on tape in blocks of 192 bytes, and since the motor is turned on and off between blocks a two second interval is left between blocks to allow the motor to accelerate and decelerate. The beginning of the 192 character buffer starts at address 828. The pointer to the start of the buffer is located at addresses 178 and 179. The number of characters in a buffer is stored in location 166. These locations can be used by the programmer to control the amount of space left in a data file. If, having opened a file on cassette, the command POKE 166,191 is executed, then the contents of the tape buffer even if empty are loaded onto the tape. If records are kept in multiples of 191 bytes we can very easily keep null or partially filled records allowing future data expansion. ## 4.5 Recording method Whether the file being stored is binary or ASCII the recording method used is the same, involving an encoding method unique to Commodore and designed to ensure maximum reliability of recording and playback. Each byte of data or program is encoded by the operating system using pulses of three distinct audio frequencies. These are: long pulses with a frequency of 1488 Hz, medium pulses at 1953 Hz and short pulses at 2840 Hz. All these pulses are square waves with a mark space ratio of 1:1. One cycle of a medium frequency is: 256 microseconds in the high state and 256 microseconds in the low state. The operating system takes about 9 milliseconds to record a byte of data consisting of the eight data bits, a word marker bit and an odd parity bit. The data bits are either ones or zeros and are encoded by a sequence of medium and short pulses: a '1' is one cycle of a medium length pulse followed by one cycle of a short length pulse, and 'Ø' is one cycle of a short length pulse followed by one cycle of a medium length pulse. Each bit consists of two square wave pulse cycles, one short and one medium, with a total duration of 864 microseconds. The waveform timing is shown in the diagram in Fig. 4.3. The 'odd parity' bit is required for error checking and is encoded like the eight data bits using a long and short pulse. Its state is determined by the contents of the eight data bits. The word marker separates each byte of data and signals to the operating system the beginning of each byte. The word marker is encoded as one cycle of a long pulse followed by one cycle of a medium pulse (see Fig. 4.3). Since a byte of data is recorded in just 8.96 milliseconds, a 192 byte block of data in an ASCII file should be recorded in just over 1.7 seconds. However, on timing such a recording we find it takes 5.7 seconds. There are two causes for this discrepancy in timing. Firstly, to reduce the possibility of audio dropouts the data is recorded twice. Secondly, a two second interrecord gap is left between each record of 192 bytes. Fig. 4.3. Output waveforms to the cassette recorder. The extensive use of error checking techniques is one reason why the tape system on the CBM 64 is slow but also quite reliable compared with that available on many other popular computers. There are two levels of error checking. The first divides the data into blocks of eight bytes and then computes a ninth byte, the checksum digit. The checksum is obtained by adding the eight data bytes together; the checksum is the least significant byte of the result. On reading the tape, if one bit in the eight bytes is dropped and a zero becomes a one or vice versa, the checksum can be used to detect this error. To do this the same procedure to calculate the check digit is performed, but the result will be different from that stored in byte nine - the check digit of that block computed when the tape was recorded. The second level of error checking involves recording each block of data twice. This allows errors detected by the check digit to be corrected during the second reading of the 192 byte data block. By recording the data twice a verification can be performed by comparing the contents of the two blocks; this will highlight the few errors not detected by the checksum. The use of pulse sequences rather than two frequencies as in a standard FSK recording has a great advantage since it allows the operating system to compensate easily for variations in recording speed. Normally a hardware phase locked loop circuit would be used to lock the system onto the correct frequencies coming from the tape head. The CBM 64, however, uses software to perform this process. A ten second leader is written on the tape before recording of the data or program commences. This leader has two functions; first it allows the tape motor to reach the correct speed, and secondly the sequence of short pulses written on the leader is used to synchronise the read routine timing to the timing on the tape. The operating system can thus produce a correction factor which allows a very wide variation in tape speed without affecting reading. The system timing used to perform both reading and writing is very accurate, based as it is on the crystal controlled system clock and Timer 1 and Timer 2 of CIA#2. Interrecord gaps are used only in ASCII files and their function is to allow the tape motor time to decelerate after being turned off and accelerate to the correct speed when turned on prior to a block read or write. Each interrecord gap is approximately two seconds long and is recorded as a sequence of short pulses in the same manner as the ten second leader. There is also a gap between blocks. When the first block of 192 bytes is recorded it is followed by a block end marker which consists of one single long pulse followed by 500+ cycles of short pulses. Then the second recording of the 192 block starts, which is identical to the first block. The first record written on the tape after the ten second leader in both ASCII and binary files is a 192 character file header block. The file header contains the name of the file, the starting memory location, and the end location. In an ASCII file these addresses are the beginning and end of the tape buffer; in a binary file they point to the area of memory in which the program is to be stored. The filename can be up to 187 bytes long. The length of the filename is stored in location 183. When read it is compared with the requested filename in the LOAD or OPEN command; if the name is the same the operating system will read the file, if different then it will search for the next ten second interfile gap and another header block. The filename is stored during a read or write operation in a block memory, the starting address of which is stored in locations 187 and 188. On completion of the operation these are reset to point to a location in the operating system. The starting location is normally set to the beginning of the user memory area, address 2049, however it can be changed to point to any location - a method employed when recording programs in machine code using the monitor. The starting address is pointed to by the contents of locations 172 and 173. The end address is stored in locations 174 and 175. Normally this is the highest byte of memory occupied by the program. however it can be altered to point to any address providing it is greater than the start address. ## 4.6 Cassette operating system routines The CBM 64 kernal contains a whole series of routines for handling data transfer between the processor and the cassette unit. The following sections describe these routines and how they can be used. These descriptions are accompanied by annotated source code listings of each routine (consult volume 1 of this series, The Commodore 64 ROMs Revealed, for the full kernal source code listing). The variable declaration file for these routines is to be found in Chapter 5. # Protect cassette/serial from RS232 NMI interrupts Entry point: \$FØA4 Function: This routine checks location \$\mathref{9}2A1\$ to see if RS232 communications are enabled. If they are not then this location contains a zero and the routine exits to allow serial or cassette operation to commence. If RS232 communications are enabled then the routine goes into a loop waiting for the RS232 NMI interrupt to reset location \$\mathre{9}2A1\$ to \$\mathre{9}3\$, thereby signalling its completion. As soon as this happens the Timer A interrupt is disabled, the flag at \$\mathre{9}2A1\$ is set to zero and the routine exits. The reason for this routine is to prevent an NMI interrupt from occurring during cassette or serial operation thereby causing data loss. Input parameters: \$\psi 2A1 - if non zero then RS232 enabled Output parameters: None Registers used: A is used but is pushed to the stack by the instruction at \$FØA4 and then retrieved at the end of the routine by \$FØBB. Routine source code: ``` LOC CODE LINE *=$F0A4 DD10 F9A4 PROTECT SERIAL/CASSETTE FROM FOA4 FØA4 :RS-232 NMI'S FØA4 ;SAVE A PHA FOA4 48 RS232 ENABLES ? LDA $02A1 F 0A5 AD A1 02 8EQ L923 ;NO FOAR FO LI AD A1 02 L838 LDA $02A1 FGAA AND #$03 FOAD 29 93 BNE L838 DØ F9 FOAF ;DISABLE FLAG A9 10 LDA #$10 F981 8D 0D DD STA DZICK F0B3 LDA #$99 A9 00 F086 STA $02A1 8D A1 02 F088 :ALL DONE L923 PLA FOBB 68 RTS FØBC F 9BD ``` #### Cassette error message output Entry point: \$F12B - tests direct mode flag first \$F12F - displays message to screen Function: This routine outputs a message to the screen concerning cassette operation. The first entry point tests the contents of location \$9D to see if the output is in direct or run mode. The second entry point performs the actual message output, the choice of message being determined by the value in the .Y index register. The messages used by this routine are stored in the area of memory immediately above this routine starting at \$FØBD. ## Input parameters: .Y index register contains message number \$9D - direct mode flag; if the high bit of .A is set and the contents of location \$9D are non zero, the required message is printed. #### Output parameters: None ``` LOC CODE LINE ; ERROR MESSAGES FØBD F080 MS1 .BYT $0,'I/O ERROR ', $A3 FØBD ЗD 49 2F FOBE. F0C8 A3 FØC9 MS5 .BYT $D.'SEARCHING', $A0 6D F9CA 33 43 F0D3 AØ 46 4F 52 .BYT 'FOR', $A0 FØD4 MS6 F007 A0 .BYT $D. 'PRESS PLAY ON TAP', $C5 MS7 FØD8 ØĐ 59 52 F009 F0EA C5 .BYT 'PRESS RECORD & PLAY ON TAP', $C5 AS8 FØEB 50 52 C_{2} F195 .BYT $D,'LOADIN',$C7 F106 0D MS10 4C 4F F197 F10D C7 .BYT $D, 'SAVING', $A0 F10E ØD MS11 FLOF 53 41 F115 A6 MS21 .BYT $0. 'VERIFYIN', $C7 F116 ØD 56 45 F117 F11F C7 MS17 _BYT $D, 'FUUND', $A0 F120 вD 46 4F F121 F126 AØ MS18 .BYT $0,'UK',$8D F127 0D 4F 4B F128 F12A 8D F128 PRINT MESSAGE TO SCREEN ONLY IF F128 , OUTPUT ENABLED F12B F128 PRINTING MESSAGES? Ĺ922 BIT MSGFLG F12B 24 9D BPL L925 :NO F120 10 0D B9 BD F0 L1073 LDA #S1,Y F12F F132 PHP 29 7F AND #$7F F133 29 D2 FF JSR $FF02 F135 F138 C8 INY PLP F139 28 F13A BPL L1973 10 F3 F13C L923 CLC 18 RTS F13D 66 .END F13E .LIB KTAPE1 F13E ``` #### Load RAM function #### Entry point: \$F49E Function: This routine loads from cassette or a serial bus device (with a device number between 4 and 31 where this device number is stored in location BA) into the memory starting at the LOAD address in the file if the secondary address is greater than $\emptyset$ , or at the specified address if the secondary address is $\emptyset$ . ### Input parameters: \$BA – device number \$B9 - secondary address .X - LOAD address lo if secondary address is zero .Y - LOAD address hi if secondary address is zero .A = if $= \emptyset$ then load, $\neq \emptyset$ then verify #### Output parameters: .X - return high LOAD address hi .Y - return high LOAD address lo ``` LOC CODE LINE *=$F49E F13E F49E F49E ;* LOAD RAM FUNCTION. F49E LOADS FROM CASSETTE OR SERIAL BUS F49E F49E :* DEVICES >=4 TO 31 AS DETERMINED BY ;* CONTENTS OF VARIABLE FA. VERIFY FLAG F49E ;* IN .A. E49E ; * ALT LOAD IF SA-6, NORMAL SA-1 F49E ;* .X, .Y LOAD ADDRESS IF SA=0 F49E ; <del>*</del> .A=0 PERFORMS LOAD, <>0 IS VERIFY. F49E * HIGH LOAD RETURN IN .X, .Y F49E ** USE SETLES & SETNAM BEFORE THIS ROUTINE F49E F49E F49E Ĺ990 86 C3 STX MEMUSS F49E ;LO ALT START 84 C4 6C 30 03 F4A0 STY MEMUSS+1 HI ALT START F4A2 JMP (ILDAD) 85 93 NLOAD STA VERCK :STORE VERIFY FLAG F4A5 F4A7 A9 00 LDA #$00 F4A9 85 99 STA STATUS F4AB A5 BA LDA FA :CHECK DEVICE # BNE L1046 F4AD D0 03 F4AF 4C 13 F7 ;KEYBOARD, BAD DEVICE L1241 JMP L1649 :SCREEN? F482 C9 93 L1046 CMF #$03 F484 F0 F9 ;YES BEQ L1241 ; TAPE F486 99 7B BCC L1050 F488 F4B8 F488 *:-$F530 L1038 JMF L959 :FILE NOT FOUND 4C 04 F7 F530 F533 ;**** LOAD FROM TAPE F533 F533 ``` ``` LOC CUDE LINE ;TAPE? F533 L1050 LSR A 44 ;YES F534 B6 03 BCS L1647 ;NO, BAD DEVICE F536 4C 13 F7 JMP L1949 SET TAPE POINTERS F539 20 D0 F7 L1047 JSR L1164 BCS L1060 F530 89 93 4C 13 F7 JAP L1049 :DEALLOCATED F53E ; 'PRESS PLAY ON TAPE' 20 17 F8 L1060 JSR L938 F541 ;STOP KEY? F544 86 68 BCS L1059 ; 'SEARCHING' 29 AF F5 JSR L1062 F346 :NAME? F549 A5 B7 L1661 LDA FNLEN ;NO, LOAD FIRST PROG ;YES, FIND A FILE F548 F0 99 BEQ L1966 F54D 20 EA F7 JSR L1108 90 0B BCC L1963 FOUND F350 F0 5A BEQ L1059 :STOP KEY F552 BCS L1038 ;NO, END OF TAPE F554 BØ DA FIND ANY HEADER JSR L1098 F556 20 2C F7 L1066 F559 F0 33 BEQ L1959 STOP KEY F55B B0 D3 BCS L1658 ;NO HEADER LDA STATUS F53D A5 90 L1963 29 10 AND #SPERR F55F ; MUST HAVE GOT HEADER RIGHT F561 38 SEC ; IS BAD F562 DØ 4A BNE L1659 CPX #BLF ; MOVEABLE? F564 E.9 91 ; YES F566 FØ 11 BEQ L1068 ;PROGRAM E9 93 CPX #PLF F568 BNE L1061 D6 DD ;NO, TRY FOR NEXT F56A F36C FIXED LOAD A9 91 L1964 LDY #$01 ; ADDRESS IN BUFFER F56E B1 B2 LDA (TAPE1),Y 85 C3 STA MEMUSS F579 ; IS LOAD ADDRESS F572 C8 INY F573 B1 B2 LDA (TAPE1),Y F575 85 C4 STA MEMUSS+1 F577 89 94 BCS L1965 F579 A5 89 ; MONITOR LOAD? L1668 LDA SA ;YES, FIXED TYPE F378 DØ EF BNE L1064 F57D A0 03 L1665 LDY #$03 :TAPEA-TAPESTA 81 82 LDA (TAPE1) Y F57F A0 61 F581 LDY #$91 F583 F1 82 SBC (TAPE1),Y F 585 :LO TO .X AA TAX F386 A9 94 LDY #$84 F598 B1 B2 LDA (TAPE1),Y A0 02 LDY #$02 F38A F580 F1 B2 SBC (TAPE1),Y ;HI TO .Y F58E 8A TAY F58F CLC 18 ;EA=STA+(TAPEA-TAPESTA) F570 84 TXA F591 65 C3 ADC MEMUSS 85 AE STA EAL F593 98 F595 TYA F596 45 C4 ADC MEMUSS+1 F598 85 AF STA EAH FS9A A5 C3 LDA MEMUSS :SET UP START ADDRESS F590 85 C1 STA STAL F59E A5 C4 LDA MEMUSS+1 F5A6 85 C2 STA STAH ; 'LOADING' E5A2 20 D2 F5 JSR L1070 FSAS 20 4A F8 JSR L940 ;LOAD TAPE BLOCK F5A8 24 .BYT $24 ;SKIP NEXT COMMAND F5A9 L1067 CLC :GOOD EXIT 18 FSAA SET UP END ADDRESS F5AA F5AA F5AA A6 AE LDX EAL LDY EAH F5AC A4 AF F5AE 60 L1659 RTS FSAF ``` #### 00 ## Print tape loading messages ## Entry points: \$F5AF - print 'searching [for filename]' \$F5C1 - print filename \$F5D2 - print loading/verifying Function: The function of these three routines is simply to display the appropriate messages on the screen when loading a program or file from tape. #### Input parameters: \$9D - flag to indicate whether 'searching [for filename]' is printed; if high bit is set then message is printed \$B7 - filename length \$BB - filename address \$93 – loading/verifying flag; if $= \emptyset$ then loading, otherwise verifying ## Output parameters: None ``` LOC CODE LINE F5AF :PRINT 'SEARCHING FOR ENAMEJ' F5AF A5 9D FSAF L1062 LDA MSGFLG :PRINT IT? F5B1 19 IE. BPL L1071 ;NO ; 'SEARCHING' FSB3 A0 0C LDY WASS-MS1 F585 20 2F F1 JSR L1073 A5 B7 F5B8 LDA FNLEN F5BA F9 15 BEQ L1071 F5BC A@ 17 LDY #AS6-AS1 :'FOR' 20 2F F1 F5BE JSR L10/3 F5C1 PRINT FILENAME FSC1 F5C1 ; NAME LENGTH F5C1 A4 87 L1922 LDY FALEN F5C3 F6 0C BEQ L1671 :NO NAME A0 99 F505 LDY #$90 F5C7 B1 BB L1091 LDA (FNADR),Y 20 D2 FF F5C9 JSR $FFD2 C8 F5CC INY F5CD C4 87 CPY FALEN F5CF DØ F6 BNE L1091 F501 69 L1971 RTS F5D2 FRINT LOADING/VERIFYING F502 F5D2 ;ASSUME 'LOADING' F5D2 AØ 49 L1070 LDY ##S10-#S1 CHECK FLAG F504 A5 93 LDA VERCK F5D6 FØ 02 BEQ L1652 ;YES, LOADING F508 A0 59 LDY #MS21-MS1 ; 'VERIFYING' F5DA 4C 2B F1 L1052 JMP L922 .END F500 F5DD .LIB KTAPE2 ``` ## Save memory function # Entry point: \$F5DD Function: A specified block of memory is saved by this routine onto cassette or a serial device with a device number between 4 and 31. This routine must be preceded by the routine at \$FFBA which sets logical first and secondary addresses and at \$FFBD which sets up the filename. #### Input parameters: - indirect pointer to start of memory area to be saved $\mathbf{X}$ - end of SAVE lo . Y - end of SAVE hi \$BA - device number Output parameters: None ``` CODE LOC LINE F5DD · ; ********************************* F 5DD * SAVE MEMORY FUNCTION. F 500 SAVES TO CASSETTE OR SERIAL F5DD ;* DEVICES >=4 TO 31 AS SELECTED BY F500 F5DD * VARIABLE FA. ;* START OF SAVE IS INDIRECT AT .A F5DD * END OF SAVE IS .X, .Y FODD * USE SETLES & SETNAM BEFORE THIS ROUTINE F500 F 50D FSDD F 5DD 86 AE L1072 STX EAL STORE END ADDRESS F5DF 84 AF STY EAH :SET UP START TAX F5E1 AA B5 00 LDA $99,X F5E2 STA STAL F5E4 85 C1 F 5E.6 85 91 LDA $91,X 85 C2 6C 32 93 F5E8 STA STAH F5EA JMP (ISAVE) F5ED NSAVE AS BA LDA FA BNE L1075 F5EF D9 93 4C 13 F7 ;BAD DEVICE F5F1 L1242 JMP L1049 F5F4 C9 93 L1975 CMP #$03 :SERIAL? ;SCREEN, BAD DEVICE F5F6 F3 F9 BEQ L1242 ,NO, TAPE F5F8 99 3F BCC L1085 F5FA F5FA *=$F659 F659 ; ***** TAPE SAVE F659 F659 ;RS-232? F659 44 L1085 LSR A F65A 80 63 BCS L1076 ;NO, MUST BE TAPE 4C 13 F7 ;BAD DEVICE F6SC JAP L1949 F65F 20 DØ F7 L1676 JSR L1164 GET BUFFER ADDR F662 99 80 BCC L1242 :NOT ALLOCATED JSR L1114 26 38 F8 F664 F667 89 25 :STOP KEY BCS L1090 F669 20 8F F6 JSR L1687 ; 'SAVING' ``` #### 90 The Commodore 64 Kernal and Hardware Revealed | LOC | CODE | LINE | | | |----------------------------------------------|----------------------------------------|----------------|-----------------------------------------------------------------|-------------------------------------------------------------------------| | F56C<br>F66E<br>F679<br>F672<br>F674<br>F676 | 45 B9<br>29 91<br>D0 02 | L1686 | LDX #PLF<br>LDA SA<br>AND #\$@1<br>BNE L1086<br>LDX #BLF<br>TXA | ;DECIDE TYPE TO SAVE<br>;1-PLF, 0-BLF | | F677<br>F67A<br>F67C<br>F67F | 20 6A F7<br>B0 12<br>20 67 F8<br>B0 0D | | JSR L1099<br>BCS L1090<br>JSR L952<br>BCS L1090 | ;WRITE HEADER BLOCKS<br>;STOP KEY<br>;WRITE PROGRAM BLOCKS<br>;STOP KEY | | F681<br>F683<br>F685<br>F687<br>F689 | 29 02<br>F0 06<br>A9 05 | | LDA SA<br>AND #\$62<br>BEQ L1088<br>LDA #\$05<br>JSR L1099 | ;WRITE END OF TAPE?<br>;NO<br>:WRITE END TABLE BLOCKS | | F68C<br>F68D<br>F68E | 24<br>18<br>60 | L1988<br>L1090 | _BYT \$24<br>CLC<br>RTS | ;SKIP COMMAND | ## Print 'saving' Entry point: \$F68F Function: Prints the message Saving [filename] on the screen. Note that this message can be output to another device such as a printer, but the following SAVE will give an error. Input parameters: \$9D - flag to indicate if message is to be printed; if high bit is not set then message is not printed. Output parameters: None Routine source code: ``` LOC CUDE LINE F68F :PRINT 'SAVING [FILENAME]' F68F F68F ;PRINT IT? L1087 LDA ASGFLG F68F 45 9D ; NO BPL L1990 F691 10 FB LDY ##S11-#S1 ; 'SAVING' F693 AØ 51 F695 20 2F F1 F698 4C C1 F5 JSR L1073 JMP L1022 ; SEND FILENAME F698 .END .LIB KTAPE3 F 69B ``` #### Stop key servicing Entry point: \$F6ED Function: This routine is included in this section because it is called by so many of the other routines. The function of this routine is to check the stop key flag and if set then close any active I/O channels, flush the keyboard queue and return the machine to direct mode. Input parameters: \$91 - value of last keyboard row - contains 'stop' key ## Output parameters: Z flag - set if stop key depressed - keys depressed from last keyboard row ## Routine source code: | LOC | CODE | LINE | |----------------------------------------------------------------------|-------------------------------|-------------------------------------------------------------------| | F69B<br>F6ED<br>F6ED<br>F6ED<br>F6ED<br>F6ED<br>F6ED<br>F6ED<br>F6ED | A5 91<br>C9 7F | *=\$F6ED ; ;********************************** | | F6F1<br>F6F3 | D0 07<br>08 | BNE L1243 ;NUT DOWN PHP | | F6F4<br>F6F7<br>F6F9<br>F6FA | 20 CC FF<br>85 C6<br>28<br>60 | JSR \$FFCC ;CLEAR CHANNELS STA NDX ;CLEAR KEY QUEUE PLP L1243 RTS | #### Error handler #### Entry points: \$F6FB - too many files \$F6FE - file open \$F7Ø1 – file not open \$F7Ø4 – file not found \$F7\07 - device not present \$F7ØA - not input file \$F7ØD - not output file \$F710 - missing filename \$F713 - illegal device number Function: This will display a designated error message from a list of nine cassette and serial I/O related messages. The table of actual error message texts is stored in locations \$A19E to \$A225. #### 92 The Commodore 64 Kernal and Hardware Revealed Input parameters: None Output parameters: None Routine source code: ``` LOC CODE TIME F6FB * ERROR HANDLER. F6FB ** PRINTS KERNAL ERROR MESSAGE IF BIT 6 F6FB ;* OF MSGFLG IS SET. RETURNS WITH ERROR F6FB ;* # IN .A AND CARRY SET. F6FB F6FB F 6FB :TOO MANY FILES L1097 LDA #$01 69 01 F 6FB .BYT $2C F6F0 20 LDA #$02 A9 02 L1011 ;FILE OPEN F6FE BYT $2C F799 20 :FILE NOT OPEN LDA #$03 L1009 F701 A9 63 .BYT $2C F793 20 :FILE NOT FOUND LDA #$64 F704 49 64 L959 F796 20 .BYT $2C DEVICE NOT PRESENT F767 05 L1026 LDA #$65 Α9 .BYT $2C F799 20 :NOT INPUT FILE F70A A9 06 L971 LDA #$06 BYT $2C F79C 20 LDA #$07 :NOT OUTPUT FILE A9 07 L965 F70D .BYT $2C F79F 20 :MISSING FILENAME F710 49 98 L974 LDA #$08 .BYT $2C F712 2C ; BAD DEVICE # L1049 LDA #$09 F713 A9 09 :ERROR # ON STACK F715 48 PHA 20 CC FF JSR $FFCC :RESTORE 1/0 F716 F719 LDY #MS1-MS1 A9 99 :PRINT ERROR? BIT MSGFLG F71B 24 9D ;NO F71D 39 9A BVC L1018 JSR L1673 :PRINT 'I/O ERROR #' 20 2F F1 F71F F722 PLA 68 F723 PHA 48 ;MAKE ERROR # ASCII F724 09 39 URA #$30 JSR $FFD2 :FRINT IT 20 D2 FF F726 L1918 PLA F729 68 SEC F72A 38 F72B RTS 60 F720 .END .LIB KTAPE4 F720 ``` ## Find any tape header Entry point: \$F72C Function: This routine reads the tape device until one of the following two block types is found: 'basic data file header' or 'basic load file'. The state of the carry flag indicates whether a header was found or not. Having found the header the message Found is displayed, followed by the filename from the header. A pause of 8.5 seconds is then generated before the routine exits to perform the rest of the load. This delay can be eliminated by pressing the CBM key. # Input parameters: None # Output parameters: $A - \emptyset$ if stop key pressed Carry flag - clear = header found; set = header not found ## Routine source code: | LOC | CODE | LINE | | | |--------------------------------------------------------------|----------|-------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------| | F72C<br>F72C<br>F72C<br>F72C<br>F72C<br>F72C<br>F72C<br>F72C | | ;* FINI<br>;* REAI<br>;* FOLI<br>;* BAS<br>;* LOAI<br>;* CLE,<br>;* IS !<br>;* 0 II | D ANY TAPE HEADER DS TAPE DEVICE UN LOWING BLOCK TYPE IC DATA FILE HEAD OFILE. FOR SUCCE AR ON RETURN. FOR SET ON RETURN. IN F STOP KEY WAS PI | NTIL ONE OF THE ES IS FOUND: BOFH DER, BLFBASIC ESS, CARRY IS K FAILURE, CARRY N ADDITION, .A IS | | F72C<br>F72E | | ;<br>L1098 | PHA | ;SAVE OLD VERIFY | | F72F<br>F732 | 68 | | JSR L1029<br>PLA | ;READ TAPE BLOCK | | F733<br>F735<br>F737 | 80 32 | | STA VERCK<br>BCS L1191<br>LDY #\$00 | RESTORE VERIFY READ TERMINATED | | F739<br>F738 | B1 B2 | | LDA (TAPE1),Y | ;GET HEADER TYPE<br>;END OF TAPE? | | F730 | | | BEQ L1101 | :YES | | F 73F | | | CMP MRIF | :BASIC LOAD FILE? | | F741 | | | CMP #BLF<br>BEQ L1027 | :YES | | F743 | | | CMP #PLF | :FIXED LOAD FILE? | | F745 | | | BEQ L1927 | :YES | | F747 | C9 04 | | CMP #BDFH | BASIC DATA FILE? | | F749 | D0 E1 | | BNE L1098 | ;NO, TRY AGAIN | | F74B | | L1027 | | ;FILE TYPE IN .X | | F74C | | | BIT MSGFLG | ;PRINT MESSAGE? | | F74E | | | BFL L1102 | ;NO | | F759<br>F <i>7</i> 52 | | | LDY #MS17-MS1<br>JSR L1073 | ;'FOUND' | | F755 | | | LDY #\$05 | | | F757 | | L1160 | | ;OUTPUT COMPLETE | | F759 | | | JSR \$FFD2 | FILENAME | | F750 | C8 | | INY | , | | F730 | C9 15 | | CPY #\$15 | | | F 75F | D0 F6 | | BNE L1160 | | | F761 | | | | ; WAIT FOR 8.5 SECONDS | | F763 | 20 E3 E4 | | JSR \$E4E0 | ; OR FOR THE CRM KEY | | F766 | | | NOP | ougorio <b>o</b> | | F767 | | L1102 | | ;SUCCESS | | F768<br>F769 | 88<br>60 | L1101 | DEY<br>RTS | | | F 7 0 7 | 0.0 | C1101 | KIU | | ## Write tape header Entry point: \$F76A Function: This routine first pushes the program start and end addresses onto the #### 94 The Commodore 64 Kernal and Hardware Revealed stack and then blanks the tape buffer memory area and sets up a tape header with all the requisite information being stored in the correct position in the tape buffer. The tape buffer contents are then written to tape and the start and end addresses restored off the stack. Input parameters: All tape header variables and filename Output parameters: .A - tape SAVE error flag ``` LOC CODE LINE F76A F76A ;* WRITE TAPE HEADER F76A F764 ERROR IF TAPE BUFFER DE-ALLOCATED * CARRY CLEAR IF U.K. F76A ; ****************************** F764 F76A 85 9E L1099 STA T1 F76A ;GET BUFFER ADDRESS ;NOT ALLOCATED ;FRESERVE START AND END 20 D0 F7 JSR L1104 F760 F76F 96 5E BCC L1166 F771 A5 C2 LDA STAH F773 PHA 48 ; ADDRESSES A5 C1 F774 LDA STAL F776 48 PHA F777 AS AF LDA EAH F779 PHA 48 F77A A5 AE LDA EAL F77C 48 PHA A9 BF F770 LDY #BUFSZ-1 BLANK TAPE BUFFER F77F LDA #$20 ;SPACE CHARS A9 20 L998 STA (TAPE1),Y F781 91 82 F783 88 DEY DØ FB BNE L998 F784 :BLOCK TYPE IN HEADER F786 AS 9E LDA T1 STA (TAPE1).Y F738 91 B2 F78A cs INY F788 A5 C1 LDA STAL START ADDRESS IN HEADER 91 B2 STA (TAPE1),Y F78D £78F 08 YKI F790 A5 C2 LDA STAH F792 91 B2 STA (TAPE1),Y F794 :END ADDRESS IN HEADER C8 YMI F795 A5 AE LDA EAL F797 91 B2 STA (TAPE1),Y F799 08 INY F79A A5 AF LDA EAH 91 82 F79C STA (TAPE1),Y F79E :FILENAME IN HEADER C8 INY 84 9F F79F STY T2 40 00 LDY #$06 F7A1 F7A3 84 9E SIY T1 F7A5 A4 9E L1105 LDY T1 F7A7 C4 B7 CPY FNLEN F0 6C F7A9 BEQ L1107 F7AB B1 BB LDA (FNADR), Y F7AD A4 9F LDY T2 FZAF 91 B2 STA (TAPE1),Y F781 E6 9E INC T1 E6 9F F783 INC T2 F785 DØ E.E. BNE L1165 SET UP START & END L1107 JSR L995 F787 20 D7 F7 ;ADDR OF HEADER & SET F7BA 49 69 LDA #$69 ; TIME FOR LEADER F7BC 83 AB STA SHCNH ``` | LOC | CODE | LINE | | | |----------------------|----------------------|-------------------|--------------|-----------------------------------------------| | F7BE<br>F7C1 | 20 6B F8<br>A8 | JSR<br>TAY | L1089 | :WRITE FILE TO TAPE<br>;SAVE ERROR CODE IN .Y | | F702<br>F703<br>F705 | 68<br>85 AE<br>68 | PLA<br>Sta<br>Pla | EAL | ;RESTORE START & END<br>; ADDRESSES | | F7C6<br>F7C8 | 85 AF<br>68 | PLA | | | | F709<br>F708<br>F700 | 85 C1<br>68<br>85 C2 | PLA | STAL<br>STAH | | | F7CE<br>F7CF | 98<br>60 | TYA<br>LI106 RTS | 1 | RESTORE ERROR CODE | #### Return buffer address Entry point: \$F7D7 Function: This routine is in two parts; the first tests if the tape buffer is allocated and the second calculates the start and end address pointers which are required by the SAVE routines. Input parameters: None Output parameters: \$C1 - start address lo \$C2 - start address hi \$AE - end address lo \$AF - end address hi Routine source code: | LOC | CODE | LINE | |------------------------------|----------------------|------------------------------------------------------------------| | F7D0<br>F7D0<br>F7D0<br>F7D0 | A6 B2 | ;<br>;RETURN BUFFER ADDRESS<br>;<br>L1194 LDX TAPE1 | | F7D2<br>F7D4<br>F7D6 | A4 B3<br>C0 92<br>60 | LDY TAPE1+1 CPY #\$02 ;ALLOCATED? RTS :CARRY CLEAR, DE-ALLOCATED | | F707<br>F70A | | L995 JSR L1104 ;GET PTR TO CASSETTE TXA | | F70B<br>F7DD | 85 Cl<br>18 | STA STAL ;SAVE START LOW<br>CLC | | F7DE | 69 CØ | ADC #BUFSZ ;COMPUTE POINTER TO END | | F7E0<br>F7E2 | 85 AE<br>98 | STA EAL ;SAVE END LOW<br>Tya | | F7E3 | 85 C2 | STA STAH ;SAVE START HI | | F 7E5 | 69 00 | ADC #\$00 ;COMPUTE POINTER TO END | | F7E7<br>F7E9 | 85 AF<br>60 | STA EAH ;SAVE END HIGH<br>RTS | Find correct file on tape Entry point: \$F7EA Function: This routine searches for a program header on tape. Having found a header the filename is compared with that specified (if the contents of location \$B7 are zero then the first program encountered is loaded). If the program name in the header is not the same as that specified then the routine searches for the next header. It should be noted that this routine only compares the header filename for the number of characters in the filename specified in the LOAD/VERIFY command, thus if the filename Test is specified in the LOAD command but the header contains the filename Testing, then the routine will take this as a positive match and load Testing. Input parameters: \$B7 - length of current filename string Output parameters: None Routine source code: ``` L.OC CODE LINE F7EA ** FIND CORRECT FILE ON TAPE F7EA ;* IF FMLEN = 0 THEN USE F7EA F7EA :* FIRST HEADER FOUND : ******************* F7EA F7EA 26 2C F7 FIND ANY HEADER F7EA L1108 JSR L1698 F7ED 89 10 BCS LIIII ;FAILED A6 05 LDY #$05 :CHECK NAME F7EF STY T2 LDY #$00 F7F1 84 9F OFFSET TO HEADER F7F3 A6 00 34 9E STY TI OFFSET TO NAME F7F5 L1024 CFY FNLEN ; COMPARE THIS MANY F7F7 C4 B7 F9 19 F7F9 BEQ L1112 :DONE F7FB B1 BB LDA (FNADR),Y A4 9F F7FD LDY T2 D1 B2 CMP (TAPE1)_Y F7FF BNE LI108 ; WRONG FILENAME F891 DØ F7 F803 E6 9E INC T1 E6 9F F805 INC T2 A4 9E F807 LDY T1 BNE L1024 :ALWAYS F899 00 EC L1112 ;SUCCESS F868 18 CL.C LIIII RTS FROC 50 F80D .END FROD .LIB KTAPES ``` ## Miscellaneous cassette support routines #### Entry points: \$F8ØD - increase pointer in tape buffer \$F817 - wait for play switch \$F82E - test cassette switch \$F838 - check for record and play \$F841 - read header block ## \$F84A - read LOAD block entry #### Routine source code: ``` LOC CODE LINE F80D ; INCREASE POINTER IN TAPE BUFFER F890 F80D 20 D0 F7 L1110 JSR L1104 :GET BUFFER ADDRESS F810 E6 A6 INÇ BUFPT F812 LDY BUFPT A4 A6 F814 C9 C9 CPY #BUFSZ :CHECK END BUFFER F816 RTS 68 F817 F817 :WAIT FOR PLAY SWITCH F817 F817 26 2E F8 Ĺ938 JSR L1116 F0 1A F81A BEQ L1113 LDY #MS7-MS1 F81C AØ 12 ; 'PRESS PLAY...' 20 2F F1 L1929 JSR L1973 F81E F821 20 DØ F8 JSR L1125 : TEST STOP KEY L1117 F824 20 2E F8 JSR L1116 : TEST CASSETTE SWITCHES F827 D0 F8 BNE L1117 F829 LDY #MS18-MS1 A0 6A ;'OK' 4C 2F F1 F828 JAP L1673 F82E :TEST CASSETTE SWITCH F82E F82E A9 16 F82E L1116 LDA #$10 ; CHECK PORT F830 24 91 BIT $91 ;CLOSED? F832 DØ 02 BNE L1113 ;NO F834 24 01 BIT $91 ;DEBOUNCE F836 18 L1113 CLC ;GOOD EXIT F837 60 RTS F838 CHECK FOR RECORD & PLAY F838 F838 F838 20 2E F8 L1114 JSR L1116 F6 F9 F83B BEQ L1113 LDY #MS8-MS1 F83D A0 2E ;'PRESS RECORD...' F83F DØ DD BNE L1020 F841 READ HEADER BLOCK F841 F841 LDA #$00 A9 00 L1029 F841 STA STATUS F843 85 90 STA VERCK 85 93 F845 JSR L995 F847 29 D7 F7 F844 READ LOAD BLOCK ENTRY F84A F844 Ĺ946 JSR L938 ; 'PRESS PLAY ... .' 20 17 F8 F84A BCS L1109 STOP KEY F840 80 1F SEI F84F 78 LDA #$99 :CLEAR FLAGS F850 A9 00 STA RDFLG F852 85 AA 85 B4 STA SNSW1 F854 STA CAPO F856 85 B0 STA PIRI 85 YE F858 STA PTR2 85 9F F85A 85 90 STA DPSW F850 A9 90 LDA #590 :ENABLE FOR TAPE 1RG F85E POINT ING VECTOR TO READ LDX #$0E F860 A2 9E BNE L1118 ; ALWAYS F862 DØ 11 ``` Write memory Entry point: \$F864 - write tape buffer \$F867 - write memory between start address and end address Function: The first entry point sets up the addresses to SAVE the tape buffer. The second entry point is to the main tape write routine. This routine writes the contents of memory between the previously determined start and end addresses onto tape. This routine calls up several small routines located at \$FBA6. ## Input parameters: \$C1 - start address lo \$C2 - start address hi \$AE – end address lo \$AF - end address hi ## Output parameters: None ``` LOC CODE LINE F864 ·*********************** F864 ** WRITE TAPE BUFFER F864 F864 F364 SET UP TO SAVE TAPE BUFFER F864 F864 F864 20 D7 F7 L1669 JSR L995 F867 F867 ;WRITE MEMORY BETWEEN STAL, STAH F867 ; AND EAL, EAH AS A BLOCK F867 F867 A9 14 L952 LDA #$14 BETWEEN BLOCK SHORTS 85 AB F869 STA SHCNH ;'PRESS RECORD...' F868 20 38 F8 F86E 80 6C L.1989 JSR LII14 L1109 BCS L1115 :STOP KEY 28 F870 SEI ;ENABLE T2 1R0 F871 A9 82 LDA #$82 LDX #$08 F873 A2 98 ;POINT IRQ VECTOR TO WRITE F875 START TAPE OPERATION ENTRY POINT F875 F875 F875 AØ 7F L1118 LDY #$7F :KILL UŃWANTED 1R0 F877 8C 9D DC STY DIICR 8D 0D DC F87A :ENABLE WANTED STA D11CR F87D AD ØE DC LDA DICRA F890 69 19 ORA #$19 8D 9F DC F882 STA DICRB F885 29 91 AND #$91 STA $02A2 F887 8D A2 02 JSR L921 ;WAIT FOR RS232 LDA VICREG+17 ;BLANK SCREEN F884 20 A4 F0 F880 AD 11 D0 F896 29 EF AND #$EF F892 8D 11 D9 STA VICREG+17 F895 AD 14 63 LDA CINV ; MOVE 1RQ TO 1RQ TEMP 8D 9F 02 F398 STA IRQTMP FOR CASSETTE OFS F898 AD 15 63 LDA CINV+1 F89E 8D A0 02 STA IRQTMP+1 F8A1 20 BD FC JSR L1195 :CHANGE IRQ VECTOR A9 02 F844 LDA #$02 FSBLK STARTS AT 2 F846 85 BE STA FSBLK F8A8 20 97 FB *PREPARE LOCAL COUNTERS JSR L1079 ``` ``` LÜC CODE LINE F8AB A5 01 LDA $61 ; TURN CASSETTE MOTOR ON F840 29 1F AND #$1F F8AF 85 61 STA $61 F8B1 85 C0 STA CASI FLAG INTERNAL CONTROL A2 FF F883 LDX #$FF ; DELAY BETWEEN BLOCKS F885 AO FF L1119 LDY ##FF F887 88 L1124 DEY F888 DØ FD BNE L1124 F8BA CA DEX F888 DØ F8 BNE L1119 F8BD ; ENABLE TAPE IRR ROUTINES TO F880 F88D START WRITE OPERATION F880 58 CLI FBBE AD A6 62 L1123 LDA IRQTMP+1 :CHECK FOR IRQ VECTOR FSCI CD 15 03 CMP CINV+1 FOINTING AT WRITE ROUTINE F8C4 18 CLC F8C5 F9 15 ; YES. RETURN BEQ L1115 F8C7 20 D0 F8 JSR L1125 :NO CHECK STOP F8C<sub>A</sub> 20 BC F6 JSR $F6BC ;UPDATE TIME F8CD 40 BE F8 JMP L1123 ;STAY IN LOOP F800 29 E1 FF L1125 JSR $FFE1 TOP KEY DOWN? F8D3 18 CLC ; ASSUME NOT F804 D9 98 BNE L1120 :CORRECT ASSUMPTION F8D6 20 93 FC JSR L1192 :STOP DOWN STOP TAPE F809 38 SEC ;FAILED F8DA 68 PLA BACK ON RTS FBDB 68 PLA F8DC A9 00 L1115 LDA #$00 ;DISABLE IRQTMP F80E 8D A0 92 STA IRQTMP+1 F8E1 L1126 RIS ``` ## Set up time out watch for next dipole Entry point: \$F8E2 Function: This routine is used to detect read errors by checking the timing of each pulse pair (dipole); if the pulses are too long then a time out error is assumed. Input parameters: .X - time out constant for particular dipole Output parameters: None Routine source code: ``` LOC CODE LINE F8E2 SET UP TIMEOUT WATCH FOR NEXT DIPOLE F8E2 F8E2 F8E2 86 81 L1126 STX TEMP :TIMEOUT CONSTANT F8E4 A5 80 LDA CMPO :CMPO*5 F8E6 ASL A 6A F8E7 94 ASL A F8E8 18 CLC F8E9 55 B9 ADC CMPO F 8EB 18 CLC ``` | LOC | CODE | LINE | | | |--------------|-------------------|-------|-----------------------|-----------------------------| | FBEC | 65 B1 | | ADC TEMP | ADJUST LONG BYTE COUNT | | FBEE | 85 B1 | | STA TEMP | • | | F8F0 | A9 00 | | LDA #\$00 | | | F8F2 | 24 BØ | | BIT CMPO | ;CHECK CMPO | | F3F4 | 39 01 | | BMI Ll146 | TRULDA ON , RUNIÑ ; | | F8F6 | 2A | | ROL A | ; FLUS, ADJUST POS | | F8F7 | 96 B1 | L1146 | ASL TEMP | ;MULTIPLY CORRECTED | | F8F9 | 2A | | ROL A | ; VALUE BY 4 | | F8FA | 06 B1 | | ASL TEMP | | | F8FC | 2A | | ROL A | | | F8FD | AA | | TAX | | | F8FE<br>F991 | AD 06 DC<br>C9 16 | L1128 | LDA DITEL | ;WATCH OUT FOR ROLLOVER | | F 9 6 3 | 90 F9 | | CMP #\$16 | ;TIME FOR ROUTINE? | | F993 | 65 B1 | | BCC L1128<br>ADC TEMP | ;TOO CLOSE SO WAIT | | F907 | 8D 04 DC | | STA DITAL | CALCULATE AND | | F90A | 8A | | TXA | ; STORE ADJUSTED TIME COUNT | | F96B | 6D 07 DC | | ADC D118H | :ADJUST FOR HI TIME COUNT | | F90E | 8D 05 DC | | STA DITAH | HOSOSI FOR HI TIME COURT | | F911 | AD A2 02 | | LDA \$02A2 | | | F914 | 8D ØE DC | | STA DICRA | | | F917 | 8D A4 62 | | STA \$0244 | | | F91A | AD 90 DC | | LDA DIICR | | | F91D | 29 10 | | ANI) #\$16 | | | F91F | F0 09 | | BEQ L1129 | | | F921 | A9 F9 | | LDA #\$F9 | | | F923 | 48 | | PHA | | | F924 | A9 2A | | LDA #\$2A | | | F926 | 48 | | PHA | | | F927 | 4C 43 FF | | JMP \$FF43 | | | F92A | 58 | L1129 | CLI | | | F 92B | 60 | | RTS | | | F920 | | .END | | | | F920 | | | .L1B KYAPE6 | | ## Cassette read subroutines Entry point: \$F92C Function: This is the main routine which reads data from the tape. The bulk of the routine performs the timing of the incoming pulses in order to decode the pulse type and to give a software servo loop, which adjusts the timing of the pulses to the speed of the cassette deck. To understand the timing of the pulses see the waveform diagrams in Fig. 4.3 plus the documentation accompanying the source code listing. Input parameters: None Output parameters: \$B6 - tape read error Routine source code: | | | | | | The casselle offits 10 | |---------------|-------------------|------------|------|--------------------|-------------------------------------------------| | LOC | CODE | LINE | | | | | | | | | | | | F920 | | ; | | | | | F92C | | ; TAPE | READ | IRQ ROUTINE | | | F920 | 6E 47 00 | ; | | 13.1.7.011 | CET TIME CLUCE LACT TOO | | F92C<br>F92F | AE 07 DC<br>A0 FF | L1130 | | D1TBH<br>#\$FF | GET TIME SINCE LAST IRQ | | F931 | 98 | | IYA | # ЪГ.Г | COMPONE COUNTER DIFF | | F932 | ÉD 06 DC | | | D1TBL | | | | EC 07 DC | | | | ;TIMER HIGH ROLLOVER? | | F938 | D0 F2 | | BNE | L1130 | YES, RECOMPUTE | | F 93A | 86 B1 | | | TEMP | | | F930 | AA | | TAX | | DE 1 04 B 73 m 5 0 | | F93D | 8C 06 DC | | | | ;RE-LOAD TIMER B | | F943 | 8C 07 DC<br>A9 19 | | | D11BH<br>#\$19 | | | F945 | | | | DICRB | | | F948 | AD ØD DC | | | D11CR | | | F94B | 8D A3 02 | | STA | \$02A3 | | | F94E | 98 | | TYA | | | | F94F | E2 B1 | | | | ;CALCULATE HIGH | | F951 | 86 B1 | | | TEMP | MOUT O DITC COOM | | F953<br>F954 | 4A<br>66 B1 | | LSR | A<br>TEMP | ;MOVE 2 BITS FROM<br>: HIGH TO TEMP | | F956 | 4A | | LSR | | ; nion to tene | | F 957 | 66 B1 | | | TEMP | | | F959 | A5 89 | | | CMPO | ;CALC MIN PULSE VALUE | | F 95B | 18 | | CLC | | | | F950 | 69 3C | | | #\$3C | | | F95E | C5 B1 | | | TEMP | ;PULSE LESS THAN MIN? | | F960 | 80 4A | | | LI141 | ;YES, NOISE | | F962<br>F964 | A6 9C<br>F0 03 | | | DPSW<br>L1132 | ;NO, LAST B1T?<br>;NO, CONTINUE | | F966 | 4C 60 FA | | | L1154 | :YES. FINISH BYTE | | F969 | | : | | | ,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, | | F969 | A6 A3 | Ĺ1132 | LDX | PCNTR | ;9 BITS READ? | | F96B | 39 1B | | | L1134 | YES, GOTO ENDING | | F96D | A2 00 | | | #\$66 | ;SET BIT VAL TO ZERO | | F96F | 69 30 | | | #\$30 | ; ADD UP TO HALF WAY BETWEEN | | F971<br>F973 | C2 B1 | | | CMPO<br>TEMP | : SHIRT PULSE AND SYNC PULSE :SHORT? | | F975 | BØ 1C | | | L1139 | :YES | | F977 | E.8 | | INX | 22207 | SET BIT VAL TO L | | F978 | 69 26 | | ADC | #\$26 | MOVE TO MIDDLE OF HIGH | | F97A | <b>გე 80</b> | | | CAPO | , | | F97C | C5 P1 | | | TEMP | ;1? | | F97E | 89 17 | | | L1137 | TES LONGLONG | | F980<br>F982 | 69 2C<br>63 B0 | | | #\$2C<br>CMPO | ; MOVE TO LONGLONG | | F984 | C5 B1 | | | TEMP | ;LONGLONG? | | F985 | 99 03 | | | L1136 | GREATER THAN, ERROR | | F988 | 4C 10 FA | L1134 | JMF | L1145 | YES | | F98B | | : | | | • | | F98B | A5 84 | L1136 | | SNSW1 | ;NUT SYNCRUNISED? | | F980 | F0 10 | | | L1141 | ;NO, ERROR | | F98F | 85 A8 | | | RER | ;YES, FLAG RER | | F991<br>F993 | D0 19 | | BME | L1141 | ;ALWAYS | | F993 | E6 A9 | ;<br>L1139 | INC: | REZ | COUNT REZ UP ON ZEROS | | F993 | Bø 92 | | | L1138 | :ALWAYS | | F997 | | ; | | | , | | F9 <b>9</b> 7 | C6 A9 | L1137 | DEC | REZ | COUNT REZ DOWN ON ONES | | F999 | 38 | L1138 | SEC | | ;CALC ACTUAL VAL FOR COMPARE | | F994 | E9 13 | | | #\$13 | CHR PRACT CARREST | | F99C<br>F99E | E5 B1<br>65 92 | | | TEMP | ;SUBTRACT INPUT VAL<br>: ADD DIFF TO TEMP STORE | | F9A0 | 85 92 | | | SVXT | : USED TO ADJUST SOFT SERVO | | F9A2 | A5 A4 | | | FIRT | ;FLIP DIPOLE FLAG | | F9A4 | 49 91 | | | #\$ <del>6</del> 1 | , | | | | | | | | LOC CODE LINE F9A6 85 A4 STA FIRT F0 2B :SECOND HALF OF DIPOLE F9A8 **BEQ L1143** 86 D7 F944 STX DATA :FIRST HALF SO STORE VAL F9AC F9AC A5 B4 Ĺ1141 LDA SNS₩1 :NO BYTE START? BEQ LI150 F9AE FØ 22 ;YES, RETURN LDA \$02A3 F986 AD A3 02 ;TIMER A 1RQ'D? F9B3 29 91 AND #\$91 **BNE L1133** F985 DØ 65 :YES F987 LDA \$92A4 AD A4 92 F9BA DØ 16 **BNE L1150** ;NO, EXIT F9BC A9 99 L1133 LDA #\$00 SET DIPOLE FLAG FOR FIRST HALF F9BE STA FIRT 85 A4 F9C0 STA \$02A4 8D A4 92 ; WHERE IN BYTE F9C3 A5 A3 LDA PONTR ; STILL DOING DATA F9C3 10 30 **BPL L1148** F9C7 ; PROCESS PARITY 30 BF BMI L1134 F9C9 F909 A2 A6 L1144 LDX #\$A6 :SETUP FOR LONGLONG F9CB 20 E2 F8 JSR L1126 ;EVEN PARITY? F9CE A5 9B LDA PRTY F9D9 D9 B9 **BNE L1136** ,NO, SET ERROR F902 4C BC FE L1156 JAP \$FEBC RESTORE REGS AND RTI F905 F905 A5 92 ;ADJUST SOFT SERVO? L1143 LDA SVXT ;NO F907 F9 97 **BEQ L1149** ;YES, MORE BASE TIME F9D9 30 03 **BMI L1142** ;YES, LESS BASE TIME ;SKIP NEXT F9DB C6 B9 DEC CMPO F9DD 20 .BYT \$2C F90E E6 89 L1142 INC CMPO F9E0 A9 00 L1149 LDA #\$60 ;CLEAR DIFF FLAG F9E2 85 92 STA SVXT ; CONSEC. LIKE VALS IN DIFOLE? F9E4 E4 D7 CPX DATA F9E6 D0 0F **BNE L1148** NO, PROCESS INFO F9E8 8A TXA ;YES, CHECK VALS F9E9 **BNE L1136** DO AO ;ONES, ERROR F9EB AS A9 LDA REZ ;HOW MANY ZEROS? ; TOO MANY F9ED 39 BD BMI L1141 F9EF C9 1.0 CMF #\$16 : 16? F9F1 90 89 BCC L1141 ;NO, CONTINUE STA SYNO F9F3 85 96 ;YES, FLAG SYNO F9F\$ 80 B5 BCS L1141 ;ALWAYS F9F7 L1148 TXA F9F7 8A ; MOVE READ DATA TO .A F9F8 45 9B EOR PRTY :CALC PARITY F9FA 85 9B STA PRTY A5 84 F9FC LDA SNSW1 :REAL DATA? F9FE FØ D2 **BEQ L1150** ;NO, FORGET FA00 C6 A3 DEC PONTR DEC BIT COUNT FA02 30 C5 ;NEG, TIME FOR PARITY ;SHIFT BIT FROM DATA BM1 L1144 FA04 46 D7 LSR DATA FA66 66 BF ROR MYCH ; INTO BYTE STOKE FA08 A2 DA LDX #\$DA SETUP FOR NEXT DIPOLE FAGA 20 E2 F8 JSR L1126 4C BC FE FAOD JMP \$FEBC \*RESTORE REGS AND RT1 FA10 **FA10** LONGLONG HANDLER FA16 A5 96 ;GOT BLOCK SYNC? FA10 L1145 LDA SYNO FA12 F9 94 **BEQ L1140** ,NO FA14 A5 84 LDA SNSW1 :HAD REAL BYTE? ; NO FA16 F9 97 **BEQ L1151** A5 A3 FA18 L1140 LDA PCNTR ;END OF BYTE? FA1A 30 03 BMI L1131 ;YES 4C 97 F9 FAIC ;NO, TREAT AS LONG JMP L1137 FAIF L1151 LSR TEMP FA1F 46 B1 :ADJUST TIME OUT FOR ``` LOC CODE LINE FA21 A9 93 LDA #$93 : LONGLONG PULSE VAL FA23 38 SEC FA24 £5 B1 SBC TEMP FA26 65 B0 ADC CMPO ASL A FA28 9A FA29 TAX AA :SET TIME OUT FOR LAST BIT FA2A 29 E2 F8 JSR L1126 FA2D INC DPSW E6 90 SET BIT THROW AWAY FLAG FA2F A5 84 LDA SNSW1 BYTE SYNCRONISED? BNE L1152 LDA SYNO BEQ L1155 ;YES, SKIP TO PASS CHAR ;THROW OUT DATA UNTIL SYNC FA31 DØ 11 A5 96 FA33 FØ 26 FA35 :NO SYNC FA37 85 A8 STA RER ;FLAG DATA AS ERROR ;KILL 16 SYNC FLAG FA39 A9 00 LDA #$60 85 96 FA3B STA SYNO FA3D A9 81 LDA #$81 :SETUP FOR TIMER 8 1RQ 80 90 DC FA3F STA DIICR 85 B4 FA42 STA SNSW1 :FLAG WE HAVE BYTE SYNC FA44 L1152 LDA SYNO FA44 AS 96 :SAVE SYNO STATUS FA46 85 BS STA DIFF F0 09 A9 00 FA48 BEQ L1153 :NO BLOCK SYNC FA4A LDA #$90 ;TURN OFF BYTE SYNC SWITCH FA4C 85 B4 STA SNSW1 FA4E A9 01 LDA #$61 :DISABLE TIMER B IRQ FA50 8D 0D DC STA D11CR FA53 AS BF L1153 LDA AYCH :PASS CHAR TO BYTE ROUTINE FA55 85 BD STA OCHAR COMBINE ERROR VALS BA CA EA57 LDA RER 05 A9 ORA REZ FA59 ; AND SAVE IN PRP FA5B 85 B6 STA PRP 4C BC FE L1155 JMP $FEBC GET LAST BYTE FA5D FA60 FA60 20 97 FB L1154 JSR L1079 ;FINISH BYTE, CLR FLAGS GET BIT THROW AWAY FLAG FA63 85 90 STA DESW FA65 A2 DA LDX #$DA :1NIT FOR NEXT DIPOLE FA67 20 E2 F8 JSR L1126 FAAA AS BE LDA FSBLK :CHECK FOR LAST VAL BEQ L1135 FA6C F0 02 STA SHCNL 85 A7 FAGE FA70 FA70 * BYTE HANDLER OF CASSETTE READ. FA79 RER IS SET IF THE BYTE IS IN FA70 ; <del>*</del> FA70 ** ERROR. REZ IS SET IF THE INTERRUPT ** PROGRAM IS READING ZEROS. RDFLG TELLS F470 * US WHAT WE ARE DOING. BIT 7 SAYS TO FA79 ;* IGNORE BYTES UNTIL REZ 18 SET, BIT 6 F478 ;* SAYS TO LOAD THE BYTE. OTHERWISE FA79 ;* RDFLG 1S A COUNTDOWN AFTER SYNC. 1F FA70 ** VERCK IS SET WE DO A COMPARE INSTEAD FA70 ;* OF A STOKE AND SET STATUS. FSBLK ;* COUNTS THE TWO BLOCKS. PTR1 IS THE FA70 FA70 FA76 ** INDEX TO THE ERROR TABLE FOR PASSI. * PTR2 IS THE INDEX TO THE CORRECTION FA70 * TABLE FOR PASS2. FA70 FAZO FA70 L1135 LDA #$0F FA70 A9 0F :TEST FUNCTION MODE FA72 24 AA BIT RDFLG NOT WAITING FOR ZEROS 10 17 BFL L1159 FA74 ; ZEROS YET? A5 85 FA76 LDA DIFF YES, WAIT FOR SYNC BNE L1156 FA78 A6 BE LDX FSBLK FA7A DEX :ZERO, NO ERROR FA7C CA BNE L1158 :NO FA7D D0 0B FA7F A9 08 LDA #LBERR ``` LOC CODE LINE ; YES, LONG BLOCK ERROR 20 1C FE FA81 JSR \$FE1C FA84 D9 94 BNE L1158 :ALWAYS A9 00 LDA #\$00 FA86 L1156 :NEW MODE, WAIT FOR SYNC FA88 85 AA STA RDFLG ;EXIT, DONE FA8A 4C BC FE L1158 JMP SFEBC ,LOADING L1159 BVS L1163 FA8D 79 31 ;SYNCING DØ 18 AS 85 BNE L1162 FA8F HAVE BLOCK SYNC? LDA DIFF FA91 ;YES, EXLT FA93 DØ F5 BNE L1158 FIRST BYTE IN ERROR? FA93 A5 86 LDA PRP BNE L1158 ;YES, EXIT FA97 DØ F1 MOVE FSBLK TO CARRY FA99 A5 A7 LDA SHCNL LSR A FA9B 4A LDA OCHAR ; SHOULD BE A HEADER COUNT CHAR AS BD FA90 ;NEG, FIRST BLOCK DATA FA9E 30 03 BM1 L1157 99 18 :EXPECTING FIRST BLOCK DATA FAA0 BCC LI161 FAA2 18 CLC ;EXPECTING 2ND BLOCK FAA3 80 15 L1157 BCS L1161 29 ØF :MASK OFF HIGH STORE HEADER FAAS AHD #\$0F ; COUNT IN MODE FLAG FAA7 85 AA STA RDFLG WALT FOR KEAL DATA L1162 DEC RDFLG FAA9 C6 AA ; REAL FAAB DØ DD BNE L1158 FAAD A9 40 LDA #\$40 NEXT UP 15 REAL DATA ; SET DATA MODE FAAF 85 AA STA RDFLG JSR L1174 SETUP ADDR POINTERS FAR1 20 BE FB A9 00 L.DA #\$00 FAB4 FAB6 85 AB STA SHCNH FABS F9 D9 BEQ L1158 ;ALWAYS, EXII L1161 LDA #\$80 ; IGNORE BYTES MODE FARA A9 80 FABC 85 AA STA RDFLG BNE L1158 FARE DØ CA ;ALWAYS A5 85 LI163 LDA DIFF ;END OF BLOCK? FAC0 FØ 64 ;YES FAC2 BEQ L1160 A9 94 LDA #SBERR ;SHORT BLOCK ERROR FAC4 FAC6 20 1C FE JSR %FE1C A9 99 4C 4A FB FAC9 LDA #\$00 :FORCE RDFLG FOR ERROR JMP L1167 FACE 20 D1 FC ;END OF STORE AREA? FACE L1160 JSR L1193 FAD1 90 03 BCC L1164 ; NOT YET ;YES 40 48 FB JMP L11/2 FAD3 FAD6 A6 A7 L1164 LDX SHCNL :WHICH PASS? FAD8 CA DEX FAD9 F6 2D BEQ L1169 : SECOND FADS A5 93 LDA VERCK :LOAD OR VERIFY? FADD F0 6C **BEQ L1166** ;LOADING ;VERIFYING FADE A0 99 LDY #\$99 FAE1 A5 BD LDA OCHAR CMP (SAL),Y ;COMPARE FAE3 D1 AC ; GOOD, CONTINUE FAES FO 04 BEQ L1166 FAEZ A9 91 LDA #\$91 ;BAD, FLAG FAE9 STA PRP 85 B6 ; AS ERROK FAEB STORE BAD LOCATIONS FOR 2ND PASS RE-IRY FAER FAEB ;CHK FOR ERRORS FAEB A5 86 L1166 LDA PRP BEQ L1171 NONE FAED FO 48 :MAX OF 30 A2 3D LDX #\$3D FAEF ; REACHED MAX? CPX PTR1 FAFI E4 9E :YES, FLAG 2ND PASS BCC L1173 FAF3 90 3E ; INDEX INTO BAD FAF5 A6 9E LDX PTR1 ; AND STORE BAD LOC LDA SAH FAF7 A5 AD STA BAD+1,X : IN BAD TABLE 9D 01 01 FAF9 LDA SAL FAFC A5 AC 90 00 01 STA BAD, X FAFE :ADVANCE TO NEXT XNI E.8 FB01 XW1 STX PTR1 JMP L1171 ;STORE CHAR FB92 FB03 E8 FB95 4C 3A FB 86 9E ``` FB08 CHECK BAD TABLE FOR RE-TRY F808 FB08 FB68 A6 9F L1169 LDX PTR2 ;DONE ALL IN TABLE? FB9A E4 9E CPX PIRI FB0C F0 35 BEQ L1176 ;YES FBOE A5 AC LDA SAL :NEXT IN TABLE? FB10 CMP BAD, X DD 00 01 BNE L1170 FB13 D0 2E : 40 FB15 A5 AD LDA SAH DD 01 01 CMP BAD+1,X FB17 BNE L1170 FB1A DØ 27 ; NÚ INC PTR2 INC PTR2 FB1C FOUND NEXT ONE, ADVANCE E6 9F FP1E E6 9F FB20 A5 93 LDA VERCK ;LOAD OR VERIFY? FB22 F0 0B BEQ L1168 ;LOAD1NG FB24 A5 80 LDA OCHAR :VERIFYING FB26 A0 00 FB28 D1 AC FB2A F0 17 LDY #$00 CMP (SAL).7 BEQ L1176 ;O.K. FB2C C8 ; . Y = 1 INY ;FLAG IT AS AN ERROR FB2D 84 B6 STY PRP FB2F AS B6 FB31 F0 07 L1168 LDA PRF SECOND PASS ERROR? BEQ L1171 :NO FB33 A9 10 L1173 LDA #SPERR FB35 20 1C FE JSR $FE1C ;ALWAYS FB38 D0 09 BNE L1170 ;LOAD OR VERIFY? FB3A A5 93 L1171 LDA VERCK ;VERIFY FB3C D9 95 BNE L1170 FB3Ê 88 TAY A5 B0 FB3F LDA OCHAR STORE CHARACTER STA (SAL),Y FB41 91 AC L1179 JSR L1080 ; NEXT ADDRESS FB43 29 DB FC FB46 DØ 43 BRE L1177 : ALWAYS FR48 Ĺ1172 :SET SKIP NEXT DATA LDA #$86 FB48 A9 80 FB4A 85 AA L1167 STA RDFLG 78 A2 91 FB4C SEI LDX #$01 FB4D STX DIICR 8E 0D DC FB4F FB52 AE 0D DC LDX D11CR DEC FSBLK FOR NEXT PASS F855 A6 BE LDX FSBLK DEX FB57 ŬA ;DONE, FSBLK=0 F858 30 02 PM1 L1165 ; ELSE, NEXT ;DEC PASS CALC FB5A 86 8E FB5C C6 A7 STX FSBLK L1165 DEC SHCNL FB5E F0 98 BEQ L1175 :ALL DONE :F1RST PASS ERRORS? FB60 A5 9E LDA PIRI ; YES, CONTINUE FB62 00 27 BNE L1177 CLEAR FSBLK IF NO ERRORS FB64 85 BE FB66 F0 23 STA FSBLK BEQ L1177 ;ALWAYS, EXII FRAS FB68 20 93 FC L1175 JSR L1192 ;READ IT ALL, EXIT JSR L1174 RESTORE SAL & SAH FB6B 20 8E FB FB6E A0 00 FB70 84 AB ;SHCNH=€ LDY #$60 ; USED TO CALC PARITY BYTE STY SHONH FB72 COMPUTE PARITY BYTE FB72 FB72 FB72 B1 AC L1176 LDA (SAL),Y :CALC BLOCK BCC EOR SHONH FB74 45 AB FB76 85 AB STA SHCNH 20 DB FC ;BUMP ADDRESS FB78 JSR L1080 ;A1 END? JSR L1193 20 D1 FC FB7B 90 F2 A5 AB BCC L1176 :NOT YET FB7E LDA SHCNH :BCC CHAR MATCH? FB86 FB82 45 BD EOR OCHAR ``` ``` LOC CODE LINE FB84 FØ 05 BEQ L1177 :YES. EXIT A9 20 FB85 LDA: #CKERR :CHKSUM ERROR FP.88 26 1C FE JSR $FE1C FB8B 40 BC FE L1177 JMF SFERC FB8E FB8E A5 C2 Ĺ1174 LDA STAH :RESTORE START ADDR FB99 85 AD STA SAH : TO POINTER SAH & SAL FB92 A5 C1 LDA STAL FB94 85 AC STA SAL FB96 60 RTS FB97 ; L1079 FB97 A9 68 LDA #$08 :SETUP FOR 8 BITS+PARITY FB99 85 A3 STA PONTR A9 00 FB9B LDA #$00 :INITIALISE ; DIPOLE COUNTER FB9D 85 A4 STA FIRT FB9F 85 A8 STA RER : ERROR FLAG FBA1 85 9B STA PRTY ; PARITY BIT ; ZERO COUNT FBA3 85 A9 STA REZ FBAS 60 RIS ;.A=0 ON RETURN FBA6 .END FBA6 .LIB KTAPE7 ``` ## Cassette write subroutines Entry point: \$FBA6 Function: These five routines are all required by the main write to tape routine at \$F867. #### Routine source code: ``` LOC CODE LINE FRAA FBA6 :* CASSETTE WRITE SUBROUTINES. FBA6 ; * FSBLK IS BLOCK COUNTER FOR RECORD FBA6 ;* = 0 SECOND DATA FRAA ; <del>*</del> FBA6 = 1 FIRST DATA = 2 FIRST HEADER FBA6 FBA6 FBA6 TOGGLE WRITE BIT ACCORDING TO LSB FBA6 ; IN OCHAR FRAG FPA6 A5 BD Ĺ1122 BIT TO WRITE INTO CARRY FBA6 LUA OCHAR LSR A FBA8 4A LDA #$60 ;ASSUME CARRY CLEAR (SHORT) A9 60 FBA9 ; CORRECT BCC LI184 FBAB 90 92 LDA #$80 L1185 FRAD A9 80 ;SET LONG SET AND STORE TIME FBAF A2 99 L1184 LDX #500 8D 06 DC L1178 STA DITBL ;LO BYTE FBB1 ;HI BYTE FBB4 8E 97 DC STX D1TBH AD 0D DC LDA D11CR ;CLEAR IRQ FBB7 A9 19 LDA #$19 FBBA ; FORCE LOAD & START TIMER 8D OF DC STA DICRE FBBC LDA $01 : TOGGLE WRITE BIT A5 01 FB8F 49 08 EOR #$08 FBC1 FBC3 85 91 STA $91 ``` | LOC | CODE | LINE | | | |----------------------|-----------------------------|---------------------|-------------------------|--------------------------------------------------| | FBC5<br>FBC7<br>FBC8 | 29 08<br>60<br>38 | L1181 | AND #\$08<br>RIS<br>SEC | ; LEAVE JUST WRITE BIT | | FBC9 | <b>66</b> 86 | L1101 | ROR PRP | ;FLAG PRP FOR END OF BLOCK | | FBCB<br>FBCD | 30 3C | : | BM1 L1183 | ; ALWAYS | | FBCD<br>FBCD<br>FBCD | | ;TAPE<br>; | WRITE IRQ ENTRY | | | FBCD | A5 A8 | ;<br>WRTN | LDA REK | ;CHECK FOR ONE LONG | | FBCF<br>FBD1 | DØ 12<br>A9 10 | | BNE L1191<br>LDA #\$10 | ;WRITE LONG BIT | | F803<br>F805 | A2 01<br>20 B1 FB | | LDX #\$01<br>JSR L1178 | | | FBD8 | 09 2F<br>E6 A8 | | BNE L1183<br>INC RER | | | FBDC | AS BS | | LDA PRP | ;END OF BLOCK? | | FBDE<br>FBE3 | 10 29<br>40 <b>5</b> 7 FC | _ | BPL L1183<br>JMP L1194 | ;NO, CONTINUE<br>;YES, FINISH OFF | | FBE3 | A5 A9 | L1191 | LDA REZ | ;CHECK FOR A ONE BIT | | FBE5 | D0 09<br>20 AD FB | | BNE L1180<br>JSR L1185 | | | FBEA<br>FBEC | D0 1D<br><b>E</b> 6 A9 | | BNE L1183<br>INC REZ | | | FBEE | D9 19 | | BNE L1183 | | | FBF0 | 20 A6 FB | ;<br>L1180 | JSR L1122 | ;WRITE | | FBF3<br>FBF5 | DØ 14<br>A5 A4 | | BHE L1183<br>LDA FIRI | ;ON BIT LOW, EXIT<br>:FIRST OF DIPULE? | | FBF7 | 49 01 | | EOR #401 | , | | FBF9<br>FBFB | 85 A4<br>F0 0F | | STA FIRT<br>BEQ L1179 | ;DIPOLE DONE | | FBFF | A5 BD<br>49 01 | | LDA OCHAR<br>EOR #\$01 | FLIPS BIT FOR COMPLEMENTARY | | FC01<br>FC03 | 85 BD<br>29 <b>0</b> 1 | | STA OCHAR<br>AND #\$01 | :TOGGLE PARITY | | FC05 | 45 9B | | EOR PRTY- | , robott raktri | | FC07<br>FC09 | 85 9B<br>4C BC FE | L1183 | STA PRTY<br>JMP #FEBC | RESTURE REGS AND RTI | | FC0C<br>FC0C | 46 BD | ;<br>L11 <i>7</i> 9 | LSR OCHAR | TI8 TX3M: | | FC0E | C6 A3 | LII// | DEC FONTR | DEC COUNTER FOR # BITS | | FC10<br>FC12 | A3 A3<br>F0 3A | | LDA PCNTR<br>BEQ L1190 | ;q BITS SENT?<br>;YES, DO PARITY | | FC14<br>FC16 | 10 F3 | • | BPL L1183 | ,NO, SEND REST | | FC16 | 20 97 FB | Ĺ1186 | JSR L1079 | CLEAN UP COUNTERS ALLOW INTERRUPTS TO NEST | | FC19<br>FC1A | 58<br>A <b>5</b> A <b>5</b> | | CLI<br>LDA CNTDN | ; WRITING HEADER COUNTER? | | FC1C<br>FC1E | F0 12<br>A2 00 | | BEQ L1189<br>LDX #\$00 | ;NO<br>;WRITE HEADER COUNTERS | | FC29 | 86 D7 | | STX DATA<br>DEC CNTDN | CLEAR BCC | | FC22<br>FC24 | C6 A5<br>A6 BE | | LDX FSBLK | ;FIRST BLOCK HEADER? | | FC26<br>FC28 | E0 02<br>D0 02 | | CPX #\$62<br>BNE L1196 | ;NO | | FC2A<br>FC2C | 09 8 <b>0</b><br>85 BD | L1196 | ORA #\$80<br>STA OCHAR | YES, MARK 1ST BLOCK HEADER WRITE CHARS IN HEADER | | FC2E | DØ D9<br>20 D1 FC | L1189 | BNE L1183<br>JSR L1193 | ;ADDR≖END? | | FC39<br>FC33 | 90 04 | L1107 | BCC L1188 | NOT YET | | FC35<br>FC37 | D0 91<br>E6 AD | | BNE L1181<br>INC SAH | ; MARK END | | FC39<br>FC3B | A5 D7<br>85 BD | | LDA DATA<br>STA OCHAR | ;WRITE BCC | | FC3D | BO CA | | 8CS L1183 | ;ALHAYS | ``` LOC CODE LINE FC3F FC3F A9 99 LDY #$09 :NEXT CHAR L1188 LDA (SAL),Y FC41 B1 AC FC43 85 80 STA OCHAR STORE IN OUTPUT CHAR FC45 45 D7 EOR DATA :UPDATE BCC FC47 STA DATA 85 07 20 DB FC : BUMP ADDRESS FC49 JSR L1080 D9 BB BNE L1183 :ALWAYS FC4C FC4E FC4E A5 9B L1190 LDA PRTY :PARITY INTO OCHAR FC50 49 01 EOR #$01 ; FOR NEXT BIT FC52 85 BD STA OCHAR FC54 40 BC FE L1187 JMP $FEBC RESTORE REGS AND RTI FC57 DEC FSBLK C6 BE L1194 ;END? FC57 ; BLOCK ONLY FC59 DØ 63 BNE L1182 ; WRITE SO TURN OFF MOTOR FC5B 20 CA FC JSR L1121 ;PUT 80 CASSETTE L1182 LDA #$50 FC5E A9 50 STA SHCNL ; SYNCS AT END FC30 95 A7 FC62 A2 08 LDX #$08 SEI 78 FC64 20 BD FC ;SET VECTOR TO WRITE ZEROS FC65 JSR L1195 BNE L1187 ;ALWAYS FC68 D9 EA ``` ## Tape IRQ ## Entry point: \$FC6A - tape IRQ entry \$FCBD - change IRQ vectors Function: The first of these two routines performs the main IRQ loop for both tape LOAD and SAVE. It should be noted that all SAVE and LOAD operations are performed under IRQ as a background program. The second routine is used to change the IRQ vectors for different tape read and write operations. #### Routine source code: ``` LOC. CUDE LINE FC6A FC3A TAPE IRQ ENTRY FOR FC6A FC6A A9 78 WKTZ LDA #$78 ; WRITING LEADING ZEROS 20 AF FB FC6C JSR L1184 ; FOR SYNC D0 E3 BNE L1187 FC6F DEC SHONL FC71 C6 A7 ; DONE WITH LOW SYNC? FC73 DØ DE BNE L1187 :40 YES, CLEAN UP COUNTERS FC25 20 97 FB JSR L1979 FC78 C6 AB DEC SHONH ;DONE WITH SYNC? FC7A 10 D8 BPL L1187 ;NO FC7C A2 0A LDX #$0A :YES, VECTOR FOR DATA FC/E JSR L1195 29 BD FC FC81 58 CLI FC82 E6 AB INC SHONH ; ZERO SHCNH FC84 AS BE LDA FSBLK ; DONE? FC86 FØ 39 BEQ L1198 :YES. SYSTEM RESTORE ``` ``` CODE LOC LINE F.C88 26 BE FB JSR L1174 FC8B A2 09 LDX #$09 :SETUP FOR HEADER COUNT FC8D STX CNTDN 86 A5 86 86 STX PRP FC8F FC91 DØ 83 BNE L1186 :ALWAYS FC93 . 11192 PHP ;CLEAN UP IRQ AND ; RESTORE PIA'S FC93 98 FC94 73 SEL FC95 AD 11 DO LDA VICREG+17 RESTORE SCREEN FC98 09 10 ORA #$10 FC9A 8D 11 D0 STA VICREG+17 ; TURN OFF MOTOR JSR L1121 FC9D 20 CA FC FCA0 49 7F LDA #$7F :CLEAR INTERRUPTS 8D 0D DC STA DIICR FCA2 FCA5 20 DD FD JSK $FDDD ;RESTORE KEYBOARD 1RG FCA8 AD A0 92 LDA IRUTMP+1 *RESTORE KEYBOARD INTERRUPT VECTOR FCAR F6 09 BEQ L1127 ;NO 1RQ FCAD 8D 15 93 STA CINU+1 FCB0 AD 9F 02 LDA IRRTAP FC83 80 14 93 STA CINV FCB6 28 L1127 PLP FCB7 60 RIS FCB8 FCB8 20 93 FC L1198 JSR L1192 :RESTORE SYSTEM IRQ JSR Li192 ; RESTURE SYSTEM ING SEQ Li187 ; CAME FOR TAPE IRQ SO RTI FCBB F9 97 FCBD ** SUBROUTINE TO CHANGE IRQ VECTORS. FCBD FCBD :* ON ENTRY, .X = 8 WRITE ZEROS TO TAPE :* FCBD = 10 WRITE DATA TO TAPE ; * = 12 RESTORE TO KEYSCAN FCBD ;* FCBD = 14 READ DATA FROM TAPE FCBD FCBD FCBD BD 93 FD L1195 LDA $FD93,X :MOVE IRQ VECTORS FCC0 8D 14 03 STA CINV ; TO VECTOR TABLE FCC3 BD 94 FD LDA $FD94.X 8D 15 93 FCC3 STA CINV+1 FCC9 60 RTS FCCA FCCA A5 01 L1121 LDA $01 :TURN OFF CASSETTE MOTOR FCCC 99 20 ORA #$20 FCCE 85 01 FCD0 60 STA $01 RIS ; xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx FCD1 * COMPARE START AND END OF LOAD/SAVE FCD1 F CD1 :* ADDRESSES. SUBROUTINE CALLED BY * TAPE READ, SAVE. TAPE WRITE FCD1 FCD1 FCD1 FCD1 38 L1193 SEC FCD2 A5 AC FCD4 E5 AE FCD6 A5 AD LDA SAL SBC EAL LDA SAH FCD8 E5 AF SBC EAH FCDA RTS FCDB BUMP ADDRESS POINTER SAL FCDB FCDB L1980 INC SAL FCD8 E6 AC F CDD DØ 02 BHE L1083 FCDF E6 AD INC SAH FCE1 66 L1083 RTS FCE2 FCE2 *=$FD9B FCE2 FD9B : TAPE IRQ VECTORS F098 ``` ``` LUC CODE LINE FD9B ţ ; WRITE ZEROS TAPE FD9B 6A FC .WOR WRTZ WRITE NORMAL TAPE .WOR WRIN FD9D CD FB .WOR $EA31 ; NORMAL IRU 31 E.A FD9F FDA1 2C F9 .WOR L1130 READ TAPE FDA3 .END FDA3 FDA3 .END ``` ## 4.7 High speed tape operation Virtually all Commodore 64 software currently being marketed uses some form of fast loader. These fast loaders are given names like: Turbo (this was the first fast loader available), Pavload, Flash Load, etc. The origin of these fast loader routines is rather obscure since many of the software houses use the same loader routines. In this section we give the source code for two fast loaders and their associated SAVE routine; these have been used on several software products of Zifra Software Ltd. under the name of ZITload and ZIFRAload. A fast loader is a routine which replaces the existing LOAD and allows a program or data to be loaded from tape at about ten times the speed of a normal LOAD. This means that a tape can be as fast as a disk drive. A fast loader is achieved by simply changing the format of the pulse sequence which is stored onto the tape in order to allow a far greater density of information storage per inch of tape. In order to create a fast loader two programs are needed; a fast loader program which is a fairly short machine code routine loaded at the beginning of a LOAD operation and then auto run to LOAD the rest of the program and/or data which is stored in fast loader format. The second program which is required is a routine to SAVE a program in fast loader format, the fast SAVE. The first major problem to be overcome in designing a fast loader is how each bit is stored on the tape. Each bit is stored on tape as a pulse which goes through a high-low transition (see Fig. 4.4). The length of the total pulse decides whether the bit is a 1 or $\emptyset$ . A short pulse is a zero and a long pulse is a one. The bit is flagged in the interrupt register on the falling edge of the pulse. The loader is a machine code program which runs with the interrupts disabled, sets a timer to between the two lengths, and when the timer runs out the interrupt register is checked to see if the pulse came in or not. If the falling edge of the pulse generates an interrupt before the timer runs out then the pulse was a zero, otherwise it was a one. The bits are then rotated into a byte storage until 8 bits have been read, thereby loading a full byte. Before any bytes can be read and stored, the loader must set itself to be in sync with the bits on the tape. This is done by writing a string of zero bits with a single one bit at every byte interval. The routine then tries to align itself by recognising the value of the byte. An example of a header byte for aligning would be the value 64, hex \$4\psi\$ or in binary: \$1\psi 1\psi 90\psi 90\psi\$. A series of these bytes is written as the header; only when this byte has been read in and recognised can the actual program can be read without risk of alignment errors. Fig. 4.4. High speed tape waveform. The program is stored in different ways depending on how much protection it is desired to put in the program. The simplest way of formatting the file is to first SAVE the two byte load address followed by the two byte end address and then the actual file. The final byte following the end of the file is a checksum that was calculated by the SAVE routine and is also calculated during loading. If the two values are the same, the LOAD was successful. The routine for this form of fast loader is given in Program 5. ``` ! FAST TAPE SAVE FOR THE 64. Ø33C 033C 033C !THIS ROUTINE WILL SAVE A PROGRAM 033C !TO TAPE SO THAT WHEN LOADED BACK !IT WILL LOAD FASTER THAN THE 0330 Ø33C !1541 DISK DRIVE. 033C 0330 !AN OPTION FOR AUTO-RUN IS Ø33C !INCLUDED. 033C C000 *=$C000 LDA #<SAVVEC !CHANGE SAVE VECTOR C000 A90B C002 8D3203 STA $0332 ! TO GO TO NEW LDA #>SAYVEC ! SAVE ROUTINE C005 A9C0 C007 8D3303 STA $0333 C00A 60 RTS C00B SAVVEC PHA !SAVE OFF .A C00B 48 COOC ASBA !GET DEVICE # LDA $BA C00E C907 CMP #$07 !NUMBER 7? BEQ TSAVE C010 F004 !YES C012 68 PLA JMP $F5ED !DO NORMAL SAVE C013 4CEDF5 CØ16 C016 A5B9 TSAVE LDA $B9 !GET SEC. ADDR. !FLAG FOR AUTO-RUN STA RUNFLG C018 8D29C2 C01B A00F LDY #$0F LDA #$20 C01D A920 !BLANK FILENAME C01F 998CC1 LOOP1 STA FLNAME, Y C022 88 DEY C023 10FA BPL LOOP1 C025 A4B7 LDY $B7 !GET FILENAME LENGTH CPY #$11 BCC LOOP2 !GREATER THAN 16? C027 C011 ! NO C029 9002 LDY #$10 !ONLY 1ST 16 CHARS C02B A010 LOOP2 DEY C02D 88 BMI TSAVE1 C02E 3008 !GET FILENAME C030 B1BB LDA ($BB),Y C032 998CC1 C035 4C2DC0 ISTORE IT IDO NEXT CHAR STA FLNAME, Y JMP LOOP2 0038 C038 R047 TSAVE1 LDY #$47 !GET LOADER BYTE C03A B944C1 TSAVE2 LDA LOADER, Y C03D 99BC02 ISTORE IT TO SAVE STA $02BC,Y C040 88 DEY BPL TSAVE2 !FOR ALL BYTES C041 10F7 C043 A901 LDA #$@1 C045 AA TAX C046 A8 C047 20BAFF TAY !SET FILE DETAILS JSR $FFBA !LENGTH OF FILENAME C04A A99E LDA #$9E LDX #<FLNAME !FILENAME LO C04C A28C !FILENAME HI LDY #>FLNAME C04E A0C1 JSR $FFBB ISET NAME DETAILS ISTOP NAME FROM BEING C050 20BDFF C053 A900 LDA #$00 C055 859D !PRINTED ON SAVE STA $9D LDA #$02 C057 A902 ``` ``` !WRITE ALIGNMENT BYTES IGET PAGE OFFSET !CHECK END OF SAVE !WRITE IT !CLOSE OFF LAST BIT !UNBLANK SCREEN !EXIT TO 'READY.' !BASIC ROM OUT & !BLANK SCREEN !PAUSE FOR TAPE ! TO GET TO FULL ! VALUE FOR DELAY ``` ``` COE2 SDO5DD STA $DD05 ``` ``` C174 F00A BEQ EXIT C176 208EA6 JSR $A68E C179 A900 LDA #$00 C17B 859D STA $9D C17D 4CAEA7 JMP $A7AE INO RUN !SET CHARGET POINTER !FLAG RUN MODE RUN C180 C180 6C0203 EXIT JMP ($0302) C180 600200 L... C183 ! C183 A21D LODERR LDX #$1D C185 4C37A4 JMP $A437 !WARM START !'?LOAD ERROR' ISEND ERROR C188 8BE3 WOR $E38B WOR $02BC !ERROR LINK C18A BC02 !WARM START LINK C18C 202020 FLNAME TXT " ! !16 SPACES CALCULATE CHECKSUM !CHECK END OF LOAD C1C8 E52E C1CR 90EA C1CR 20BA03 C1CF 85FB SBC $2E BCC TLOAD1 JSR $03BA STA $FB RTS !NOT YET !READ CHECKSUM C1D1 60 C1D2 ! C1D2 !*=$0387 C1D2 ! C1D2 A907 LDA #$07 STA $01 LDA #$0B C1D6 A908 C1D8 8D11D0 ISTART TAPE !BLANK SCREEN !PAUSE FOR TAPE TO STA $D011 ! REACH FULL SPEED !DISABLE IRQ !ZERO CHECKSUM ISET TIMER HI !SET TIMER LO !READ A BIT !INTO BYTE !ALIGNED? !NOT YET !READ A BYTE !IS IT 64? !YES ``` | C200<br>C202<br>C204<br>C205<br>C205<br>C205 | DØEA | !<br>!*=\$03BA<br>! | | #\$5R<br>RHEAD2 | !ALIGNMENT CHECK? | |----------------------------------------------|----------------|---------------------|-----|-----------------|---------------------| | C205 | | • | | #\$01 | | | C207 | | | STA | | | | | 200803 | GBYTE1 | | \$93C8 | !READ A BIT | | C20C | | | ROL | | INTO BYTE | | C20E | ASBD | | | GBYTE1<br>\$BD | !COMPLETE BYTE | | C212 | | | RTS | ₩DU Ud\$ | | | C213 | 00 | 1 | | | | | C213 | | ! *=\$03C8 | | | | | C213 | | ! | | | | | C213 | A910 | | LDA | #\$19 | !WAIT FOR BIT | | | 2CØDDC | GBIT1 | | \$DC0D | | | | F0FB | | | GBIT1 | | | | ADODDD | | | \$DD0D | !GET BIT | | C21D | | | PHA | 4040 | | | | A919<br>SDØEDD | | | #\$19<br>\$DD0E | START TIMER | | C223 | | | PLA | <b>∌₽₽₽₽</b> | SIRI IIIER | | | EE20D0 | | | \$D020 | SHOW IT IS WORKING | | C227 | | | LSR | | MOVE BIT INTO CARRY | | C228 | | | RTS | •• | | | C229 | | ! | | | | | C229 | | !*=\$03DE | | | | | C229 | | ! | | _ | | | C229 | 00 | RUNFLG | BYT | 0 | | Program 5. Another type of LOAD, which uses the same saver but is slower, is the interrupt loader. This method has the advantage of being able to LOAD with the screen on and a foreground program running whilst the main program is loaded. Loaders of this type are: Novaload and Micro Load. The difference with this type of LOAD is that an interrupt is created when a pulse is read by the tape recorder, and the timer is checked to find out whether the pulse was a zero or a one. The whole LOAD is done in the background allowing a foreground program to play music, run a clock, etc. The foreground program must check at regular intervals to see if the loader has flagged for the end of load. The example of a background LOAD in Program 6 has only a foreground program that is waiting for the end of LOAD flag to be set. ``` FAST TAPE SAVE FOR THE 64. 033C 033C Ø33C !THIS ROUTINE WILL SAVE A PROGRAM 033C ITO TAPE SO THAT WHEN LOADED BACK !IT WILL LOAD WITH THE SCREEN ON. 033C 033C C000 #=$C000 LDA #KSAVVEC !CHANGE SAVE VECTOR C000 A90B STA $0332 ! TO GO TO NEW C002 8D3203 LDA #>SAVVEC ! SAYE ROUTINE C005 A9C0 STA $0333 C007 8D3303 C00A 60 RTS C00B C00B 48 C00C A5BA SAVYEC !SAYE OFF .A !GET DEVICE # LDA $BA !NUMBER 7? C00E C907 CMP #$07 ``` ``` C075 | STRRTS HERE. C075 | C076 | C078 C07 !WRITE ALIGNMENT BYTES !ZERO CHECKSUM IGET PAGE OFFSET ``` | C098 20F5C0<br>C09B C8<br>C09C D002<br>C09F F62C | | JSR<br>INY<br>BNE | WRTBYT TSAVE3 | !WRITE IT | |----------------------------------------------------|-------------|--------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------| | C0A0 C42D<br>C0A2 A52C<br>C0A4 E52E | TSAVE3 | CPY<br>LDA<br>SBC | \$2D<br>\$2C<br>\$2E<br>\$2E | !CHECK END OF SAVE | | COAS ASFB<br>COAR 20F5CO<br>COAD 2009C1 | | LDA<br>JSR<br>JSR | \$FB WRTBYT WRTBIT ##19 | INDITE I<br>IGET CHECKSUM<br>IWRITE IT<br>ICLOSE OFF LAST BIT | | COB2 8D11D0<br>COB5 A937 | | STA | \$D011<br>#\$37 | !UNBLANK SCREEN | | COBP 58 COBA 68 | | CLI | **** | RESTART IRQ | | COBE 852E<br>COBE 852B | | PLA | \$2B | STURE IT<br>SET START LO<br>STORE IT | | C0C0 2084FF<br>C0C3 4C74R4<br>C0C6 | ! | JMP | \$R474 | !RESET I/O<br>!EXIT TO 'READY.' | | C0C6 H906<br>C0C8 8501<br>C0CA A90B | WRTHDR | STA<br>LDA | #\$06<br>\$01<br># <b>\$</b> 0B | !BASIC ROM OUT &<br>! START TAPE | | COCC SD11DO<br>COCF CA<br>CODO DOFD | HEADR1 | STA<br>DEX<br>BNE | \$D011<br>HEADR1 | !BLANK SCREEN<br>!PAUSE FOR TAPE<br>! TO GET TO FULL | | C0D2 88<br>C0D3 D0FA<br>C0D5 78 | | DEY<br>BNE<br>SEI | HEADR1 | ! SPEED<br>!DISABLE IRQ | | CODE ASAO<br>CODE SDO4DD<br>CODE ASOO | | LDA<br>STA<br>LDA | #\$80<br>\$DD94<br>#\$90 | !INITIAL TIMER<br>! VALUE FOR DELAY | | C0E0 A919<br>C0E2 SD0EDD | | LDA | \$11065<br>#\$19<br>\$1106E | START TIMER | | C0E3 H040<br>C0E7 A940<br>C0E9 20F5C0 | HEADR2 | LDA | #\$40<br>#RTBYT | !01000000 FOR<br>!ALIGNMENT | | COED DOFS<br>COEF A95A<br>COF1 20F5CO<br>COF4 60 | | BNE<br>LDA<br>JSR<br>RTS | WRTBYT TSAVE3 \$2C \$2D \$2C \$2D \$2E TSAVLOOP \$FB WRTBYT WRTBIT #\$18 \$D011 #\$37 \$01 \$2C \$2B \$FF84 \$A474 #\$06 \$01 #\$08 \$D011 HEADR1 HEADR1 HEADR1 HEADR1 HEADR1 HEADR1 #\$70 \$D005 #\$19 \$D005 #\$19 \$D005 #\$19 \$D005 #\$19 \$D006 #\$1005 #\$19 \$D006 #\$1005 #\$19 \$D006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 #\$1006 | !WRITE 64 OF THEM<br>!CHECK ALIGNMENT<br>!WRITE IT | | COF5<br>COF5 85BD<br>COF7 45FB<br>COF9 85FB | !<br>WRTBYT | STA<br>EOR<br>STA | \$BD<br>\$FB<br>\$ER | STORE BYTE | | COFB A908<br>COFD 85A3<br>COFF 26BD | LIBUTE | LDA | #\$08<br>\$A3<br>\$PD | !LOOP FOR 8 BITS | | C101 2009C1<br>C104 C6A3<br>C106 D0F7 | ADTICE | JSR<br>DEC | \$BD<br>WRTBIT<br>\$A3<br>WBYTE1 | !WRITE THE BIT | | C108 60<br>C109<br>C109 A270 | 1 | KIS | #\$70 | !ASSUME ZERO BIT | | C10B 9002<br>C10D A2FF<br>C10F 8E04DD<br>C112 A900 | | BCC<br>LDX<br>STX | #\$FF<br>\$DD04<br>#\$60 | !CORRECT ASSUMPTION<br>!SET FOR ONE BIT<br>!SET TIMER | | C114 8D05DD<br>C117 A901<br>C119 2C0DDD | WBIT2 | STA<br>LDA<br>BIT | \$DD05<br>#\$01<br>\$DD0D | !WAIT FOR TIMER | | C11C F0FB<br>C11E A501<br>C120 4908<br>C122 8501 | | BEQ<br>LDA<br>EOR | WBIT2 | !TOGGLE WRITE BIT<br>! IN 6510 REGISTER | ``` C124 EE20D0 INC $D020 !SHOW IT IS V C127 A919 LDA #$19 C129 8D0EDD STA $DD0E !START TIMER C12C A901 LDA #$01 !WAIT FOR TIMEN C12E 2C0DDD WBIT3 BIT $DD0D C13I F0FB BEQ WBIT3 C133 A501 LDA $01 !TOGGLE WRITE C135 4908 EOR #$08 ! IN 6510 REC C137 8501 STA $01 C139 A919 LDA #$19 C13B 8D0EDD STA $DD0E !START TIMER C13F !THE LOADER STARTS HERE C13F !THE LOADER LDY #$05 !SHOW IT IS WORKING !WAIT FOR TIMER !TOGGLE WRITE BIT ! IN 6510 REGISTER !WAIT FOR END OF LOAD C198 932R20 FLNAME TXT "D# C1R8 48 PHA C1R9 98 TYR C1RA 48 PHA C1R ! IRQ ENTRY POINT C1A9 98 TVA C1A9 48 PHA C1AB AD05DC LDA $DC05 !GET TIMER HI BY C1AE A019 LDY #$19 !RESTART TIMER C1B0 8C0EDC STY $DC0E C1B3 4902 EOR #$02 !FLIP BIT 1 C1B5 4A LSR A ! AND SHIFT TO C1B6 4A LSR A ! CARRY C1B7 26A9 ROL $A9 !MOVE BIT INTO C1B9 A5A9 !BYTE READ C1BB 9002 BCC BITGOT !WHEN BYTE READ GET TIMER HI BYTE!RESTART TIMER ``` | 120 1110 00 | Jiiiiiodore o- | r Korriar aria riaraware rieve | are a | |------------------------------------------------------------------------------------------------------------------------------------------|----------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------| | C1BD B00D<br>C1BF C940<br>C1C1 D009<br>C1C3 A916<br>C1C5 8D6503 | BITGOT | BCS EXIT<br>CMP #\$40<br>BNE EXIT<br>LDA #\$16<br>STB #\$265 | !NOT COMPLETE BYTE<br>!ALIGNMENT?<br>!NO<br>!SET NEW ADDRESS | | C1C8 A9FE<br>C1CA 85A9<br>C1CC ADODDC | EXIT | LDA #\$FE<br>STA \$A9<br>LDA \$DCOD | !GET READY FOR<br>! A NEW BYTE<br>!CLEAR IRQ | | C1D0 A8<br>C1D1 68 | | TAY<br>PLA | IEIT TO | | C1D2 40 | 1 | RII | !EXII IRQ | | C1D3 C940<br>C1D5 F0F1<br>C1D7 C95A<br>C1D9 F007<br>C1DB A902<br>C1DD 8D6503 | | BCS EXIT CMP #\$40 BNE EXIT LDA #\$16 STA \$0365 LDA #\$FE STA \$A9 LDA \$DC0D PLA RTI CMP #\$40 BEQ EX1 CMP #\$5A BEQ BITGT2 LDA #\$02 STA \$0365 BNE EX1 LDA #\$30 STA \$0365 LDA #\$40 STA \$0365 LDA #\$40 STA \$60365 LDA #\$40 STA \$60365 LDA #\$40 STA \$60365 LDA #\$40 STA \$60365 LDA #\$60 STA \$61 LDA #\$60 STA \$61 LDA #\$60 STA \$61 LDA #\$65 SNE EX1 LDA #\$65 SNE EX1 LDA #\$65 SNE EX1 LDA #\$65 SNE BITGT5 INC \$FB BNE BITGT5 INC \$FB STA \$61 LDA #\$7 LDA \$FB CMP \$FD LDA \$FB CMP \$FD LDA \$FB STA \$6365 SNE EX1 | !MORE ALIGNMENT?<br>!YES<br>!FINAL CHECK?<br>!YES<br>!GO BACK TO<br>! ALIGNMENT ROUTINE | | C1E2 A930<br>C1E4 8D6503<br>C1E7 A900<br>C1E9 85C1<br>C1EB F0DB | BITGT2 | LDA #\$30<br>STA \$0365<br>LDA #\$00<br>STA \$C1<br>BEQ EX1 | !SET NEW ADDRESS<br>! TO READ IN LOAD<br>! ADDRESSES. CLEAR<br>! CHECKSUM | | C1ED 85FB<br>C1EF EE9703<br>C1F2 AD9703<br>C1F5 C9FF | · | STA \$FB INC \$0397 LDA \$0397 CMP #\$FF | STORE LOAD ADDRESS INCREASE STORE UNTIL 4 BYTES | | C1F7 D0CF<br>C1F9 A943<br>C1FB 8D6503<br>C1FE D0C8<br>C200 | 1 | BNE EX1<br>LDA #\$43<br>STA \$0365<br>BNE EX1 | INOT YET<br>ISTORE NEW ADDRESS<br>IFOR READING FILE | | C200 A000<br>C202 91FB<br>C204 45C1<br>C206 85C1 | | LDY #\$00<br>STA (\$FB),Y<br>EOR \$C1<br>STA \$C1 | STORE A BYTE CHECKSUM | | C208 EE00D8<br>C20B E6FB<br>C20D D002<br>C20F E6FC | | INC \$D800<br>INC \$FB<br>BNE BITGT5<br>INC \$FC | SHOW IT IS WORKING INCREASE ADDRESS | | C211 A5FB<br>C213 C5FD<br>C215 A5FC<br>C217 E5FE | BITGT5 | LDA \$FB<br>CMP \$FD<br>LDA \$FC<br>SBC \$FE | !END OF LOAD? | | C219 90AD<br>C21B A965<br>C21D 8D6503<br>C220 D0A6<br>C222 | ! | BCC EX1<br>LDA ##63<br>STA \$0365<br>BNE EX1 | !NOT YET<br>!NEW ADDRESS FOR<br>!CHECKSUM | | C222 85C2<br>C224 A9FF<br>C226 8502<br>C228 A902 | | STA \$C2<br>LDA #\$FF<br>STA \$02<br>LDA #\$02<br>STA \$0365 | | | C222<br>C222 85C2<br>C224 A9FF<br>C226 8502<br>C228 A902<br>C22A 8D6503<br>C22D A9FB<br>C22F 8D9703<br>C232 D094<br>C234<br>C234<br>C234 | ! | STA \$0365<br>LDA #\$FB<br>STA \$0397<br>BNE EX1 | !RESET BRANCH TO<br>! ALIGNMENT | | 0234<br>0234 | !*=\$93DD | | | | C234 A502<br>C236 F0FC<br>C238 A900<br>C23A 8502<br>C23C A907<br>C23E 8501<br>C240 20F302 | PAUSE | LDA \$02<br>BEQ PAUSE<br>LDA #\$00<br>STA \$02<br>LDA #\$07 | !WAIT FOR FILE<br>! TO BACKGROUND LOAD | | C23E 8501<br>C240 20F302 | | STA \$01<br>JSR \$02F3 | !RESET KERNAL ROM | | C243<br>C246<br>C248 | | | LDA | \$R663<br>\$C1<br>\$C2 | !CLR<br>!COMPARE CHECKSUMS | |-----------------------|----------------|--------|-----|-------------------------------|------------------------------------| | C24R<br>C24C<br>C24F | DØØ3<br>4C74A4 | ! | BNE | LODERR<br>\$R474 | !DIFFERENTERROR<br>!GO TO 'READY.' | | C24F<br>C2 <b>5</b> 1 | A21D<br>4C37A4 | LODERR | | #\$1D<br>\$R437<br>Program 6. | !'?LOAD ERROR'<br>!SEND ERROR | ## 4.7.1 Fast tape routines Putting the theory into practice to create the fast loader routines is not difficult. The actual timing for the SAVE routine was not calculated from any theoretical formula but was obtained just by trial and error. The only guidelines were that the short pulse should be slightly shorter than half the long pulse, as the waveform of the pulse is evened out by the cassette hardware. The timing value used by the loader is just shorter than the time required before the long pulse reaches its falling edge. There are two program listings in this section, one for each of the two types of LOAD. Each program will SAVE a Basic program to tape in its fast format and automatically put the fast loader routine into the filename where it is stored and, when loaded, will automatically start on the warm start vector. The routines are initialised by SYS(49152). A Basic program can be fast saved by using the SAVE command as normal but with a device number of 7, thus: ## SAVE"PROGRAM".7 In addition the first kind of fast LOAD also makes use of the secondary address to auto run the program, thus: #### SAVE"PROGRAM",7,1 will cause the program to auto run when loaded back. With both routines, when a program has been saved using one of these fast loader SAVE routines it is unnecessary to load anything before loading the program; it will load directly from the LOAD command. An example of how fast these routines can be is shown by the following timing table. This was based on the time taken to load a 26.3K byte Basic program: :1 minute Method 1 :1 minute 10 seconds Disk Method 2 :1 minute 25 seconds 40 seconds Normal tape :8 minutes It should also be noted that the SAVE routines for the fast tape operation are considerably shorter than the normal tape routines which were analysed at the beginning of this chapter. ## 4.8 Causing programs to auto run from tape The facility to have a program run automatically after completing its LOAD is a nice feature to include, particularly if the program is intended for commercial sale. Adding this auto run feature to tape loaded programs is not difficult and considerably enhances a program's professionalism. Before saving a Basic program to tape, the secondary address is used to indicate whether the LOAD routine starts loading into the memory area from which it was saved, or starts loading at an address stored in the pointer to the start of Basic program storage variables. Thus if a program is saved with: SAVE"PROGRAM" it will commence loading wherever the pointer stored at \$2B,\$2C (decimal 43,44) indicates, regardless of where it was saved from. If, however, SAVE "PROGRAM",1,1 is used the LOAD routine will load the program into the same locations from which it was saved. The use of the secondary address is thus the main principle required for auto running. To auto run a program a short machine code loader is required; this is loaded first, and on loading will then take control of the computer. The only way to make this happen is to write over one of the operating system vectors in page 3 of memory, the top end of the stack, or the 'Tape load IRQ' vector temporary storage at location \$\$\000929F/\$\00092A\00096\$. #### 4.8.1 Page three vectors There are plenty of vectors which can be used. The most commonly used is the Basic warm start vector (as in the fast load routines in the previous section). This is the easiest one to use since it can be set to point to the auto run routine in the vectors saved with the program, and then reset afterwards. In addition, use of this vector allows code to access the sprite 11 block. Other vectors which can be used are: Input vector at \$\0324 Output vector at \$\0326 Abort I/O vector at \$\032C These three vectors will also cause control to be transferred to the routine after loading, but use of the sprite 11 block for code is impossible so the code must therefore be located in the filename. Problems can arise in using these vectors when saving, for example: the output vector is set up, as soon as the SAVE routine is called, and the computer will crash when it tries to print the message 'PRESS RECORD & PLAY ON TAPE'. The way to overcome this is to add a bit of code into the SAVE routine which is called before the vector is set up: LDA #\$00 STA \$9D !disable the message 'saving' JSR \$F838 !wait for record and play You can then set up the vector and save it. #### 4.8.2 The stack A machine code program can be made to auto run by using the top 8 bytes of the stack. These locations are all set to a value of 2, and the machine code starts at location \$\psi 2\psi 3\$. This method is not widely used, since it will only work on the majority of occasions when the machine is freshly powered up. There is one advantage, however; if it does not auto run, there is less chance of the machine code being intact for prying eyes. ## 4.8.3 Tape IRQ save With this vector you must have a SAVE routine which saves a program from one area of memory which will be loaded into another (see beginning of this section). All the auto run code must be located in the filename. Having decided which vector to auto run and where to place the machine code loader, it is necessary to decide what the loader will do. The first function of any loader should be to get the kernal LOAD routine to stop printing messages. This will prevent the pause for CBM key when the next file is found. It may also be necessary to disable the RUN/STOP key (see next section). Other security methods that you can add into your loader are detailed in the next section of this chapter. Whichever vector is used to auto run, it must be reset to normal before running the main program. If page 3 vectors are used (IRQ save included), there are two ROM subroutines to use: \$E453 for vectors from \$0300 to \$030B, or \$FF8A for vectors between \$0314 and \$0333 (IRQ save changes \$0314). The program can then be loaded. Depending on whether the program is in machine code or Basic, the autor unroutine can either jump straight into the main program or cause the Basic program to run. Running a machine code program is straightforward, however there are several ways to initiate the running of a Basic program from a machine code routine: ## a) Keyboard buffer By storing the characters R, shift U, and carriage return into the keyboard buffer (\$\psi 277 - \$\psi 28\phi) and setting the number of characters to 3 (\$C6), the Basic program will then run by: JMP (\$\psi 3\psi 2). The problem with this is that, to be on the safe side, the screen should first be cleared or there is a possibility of a syntax error occurring. #### b) Basic ROM routines The second, and best, way of running a Basic program is to use the routines in the ROM. The code to run a program this way is shorter than that for the keyboard buffer method. In both types, the end address from the LOAD must be stored into locations \$2D and \$2E. The code for running a Basic program using the ROM routines is as follows: JSR \$A65C !perform 'CLR' JSR \$A68E !reset charget pointers !execute the next statement JMP \$A7AE There is no need to store anything into the keyboard buffer or to clear the screen. ## 4.9 Tape security and anti piracy techniques The greatest problem for anyone writing and/or selling commercial software is illegal copying. This can lose the author a substantial proportion of the expected royalty. It has been estimated that often as many as two out of every three copies of a program in circulation are illegal pirate copies. The SAVE command makes pirating of unprotected programs so easy that it is essential to put some protection onto any commercial program. There is no absolutely secure way of protecting a piece of software on the 64. If someone has enough patience they can break any protection method and copy the program. Therefore, the main thing to concentrate on is making the job of breaking the protection as difficult or laborious as possible. The initial methods include disabling the RUN/STOP key, encoding the program before saving and decoding it on loading, etc. To disable the RUN/STOP key is very simple: Another method of disabling the STOP key is by altering the low byte, but the above method is the only totally reliable way. This disables both the normal stop from a Basic program and the STOP/RESTORE combination. If the program is a machine code game, then it is better to just disable the NMI vector. This can be done by changing the vector at \$\mathref{9}318\$ to point to an RTI instruction (\$FEC1). The NMI vector does not have to be disabled; it could be of use in the actual program (see Chapter 6). Encryption of the program is useful as it will stop a pirate from loading the main file without the auto run part. Encryption means that a special SAVE routine is used to encode a program which is then totally indecipherable. The loading then decodes it so that it can run properly. The best way of encoding and decoding is to use one of the arithmetic commands in the 6510 instruction set. The most common and easiest to use is the EOR command. To do this take a key value, a number between 0 and 255, EOR it with a byte of the original code and store the result; this is the encrypted code. To restore the original code, simply take the stored encrypted code byte, EOR it with the key value and the original code is restored: LDA STORE !GET THE VALUE FROM MEMORY EOR #\$A1 !ENCODE IT STA STORE !STORE IT This routine, when called the first time, will encode the byte. Call it a second time and the original value will be restored. Use the following routine to encode (or decode) a complete program where (\$2B) is the start and (\$2D) is the end plus 1: | !SET START ADDRESS | |--------------------| | !GET A BYTE | | !ENCODE/DECODE IT | | !STORE IT | | !INCREMENT POINTER | | | | | | !CHECK END OF | | ! PROGRAM | | | | | | !NOT YET | | | It is not necessary to use location \$FB for the EOR code, but whatever value is used it must be the same on encoding as it is on decoding. With the advent of fast tape formats, the need for this EOR encoding/decoding is nullified due to the fact that the high speed loader must be present to be able to load the main program. Final security checks should be made on running the program to check certain locations for the presence of known values. Obvious locations are the cassette buffer (filename), device number (to check the last device used), etc. #### 4.9.1 Undocumented codes All the previously mentioned methods can be displayed with the use of a monitor, and so with a little detective work they can be understood by someone intent on breaking the security. The use of some of the undocumented codes of the 65\( \textit{g} 2 \) within the program and its security makes the use of a monitor much harder. On all 65\( \textit{g} 2 \) microcomputers there are some instructions that don't appear in most documentation. These codes are therefore not included in any of the monitors available. The most useful of these codes are the multi-byte NOP instructions. These instructions have the same effect as the normal NOP with the exception that one or two bytes following are ignored. Using a two byte NOP before a three byte instruction with the byte to be ignored as, for example, \$2\( \textit{g} \) (JSR) or \$4C (JMP) will result in the code looking like garbage upon disassembly. 2 byte NOPs: \$\text{94.}\$14.\$34.\$44.\$54.\$64.\$74, and \$F4 3 byte NOPs: \$\textit{ØC,\$1C,\$3C,\$5C,\$7C,\$DC, and \$FC} Three byte NOPs are useful since the next two bytes could contain a 2 byte instruction which will read well with the rest of the code but is in fact ignored. For example, on assembly: | BYT \$44,\$4C | !2 BYTE NOP | |---------------|----------------------| | JSR \$FFD5 | !LOAD FILE | | BYT \$3C | !3 BYTE NOP | | LDX #\$ØØ | !IGNORED | | STX \$2D | !STORE END LO | | BYT \$7C | !3 BYTE NOP | | LDY #\$ØØ | !IGNORED | | STY \$2E | <b>!STORE END HI</b> | | BYT \$74,\$2Ø | !2 BYTE NOP | | JMP \$A474 | !GOTO 'READY.' | ## actually does: JSR \$FFD5 STX \$2D STY \$2E JMP \$A474 ## but on disassembly it gives: ``` Ø33C 44 ??? ., Ø33D 4C 2Ø D5 JMP $D520 Ø34Ø FF ??? Ø341 3C ??? Ø342 A2 ØØ LDX #$00 Ø344 86 2D STX $2D Ø346 7C ??? Ø347 AØ ØØ LDY #$ØØ Ø349 84 2E STY $2E Ø34B 74 ??? Ø34C 2Ø 4C 74 JSR $744C Ø34F A4 xx LDY $xx ``` The byte xx has nothing to do with the code. The subject of program protection and security methods is one which can be gone into in great depth but unfortunately it would be inadvisable to give more information than has been included since a knowledge of how to protect a program will also tell the intending pirate how to break that protection. Readers interested in adding protection and security to their programs should write to Zifra Software Ltd., 40 Bowling Green Lane, London EC1. Zifra have considerable experience and expertise in security and protection methods for both tape and disk which have been used on Zifra products. # Chapter Five # The User Port ## 5.1 The I/O ports and the 6526 The CBM 64 communicates with peripheral devices via five integrated circuits. The most important of the five is the 6510 microprocessor. This has a single eight line I/O port which is used principally to control memory bank switching but also some of the tape operations. The 6566 VIC chip controls the video display and has a light pen input. The sound output is generated by a 6581 SID chip, which also has four analog joystick inputs (see Fig. 5.1 for I/O connections on the CBM 64). The other two integrated circuits are 6526 complex interface adapters or CIAs, which are used to perform all the other I/O functions of the CBM 64. We can summarise the function of these two chips as follows: Keyboard input User port Cassette deck Serial I/O – used by the disk drive and printer RS232 I/O – for printers, modems etc. Joystick – simple switch type IRQ timing for real time clock and keyboard The two CIA chips which are used to control all these functions have between them just 32 programmable I/O lines and 8 handshake lines; many of these lines are thus used by more than one of the above functions. - 2) Memory expansion - 3) Audio and video - 4) Serial I/O - 6) User port - 7) Modulated TV output - Fig. 5.1. The position of the different CBM 64 I/O outputs. # SERIAL I/O | PIN# | TYPE | |------|--------------------| | 1 | SERIAL SRQ IN | | 2 | GND | | 3 | SERIAL ATN IN/OUT | | 4 | SERIAL CLK IN/OUT | | 5 | SERIAL DATA IN/OUT | | 6 | NC | # **AUDIO/VIDEO** | PIN# | TYPE | |------|------------| | 1 | LUMINANCE | | 2 | GND | | 3 | AUDIO OUT | | 4 | COMP VIDEO | | 5 | AUDIO IN | ## GAME I/O Port #1 | PIN# | TYPE | NOTE | |------|------------|---------------------| | 1 | JOYØ | | | 2 | JOY1 | | | 3 | JOY2 | | | 4 | JOY3 | | | 5 | POT Y | | | 6 | LIGHT PEN/ | | | | BUTTON A | | | 7 | + 5V | MAX. 1 <b>00</b> mA | | 8 | GND | | | 9 | POT X | | Fig. 5.1. (contd.) #### Port #2 | PIN# | TYPE | NOTE | |------|----------|------------| | 1 | JOYØ | | | 2 | JOY1 | | | 3 | JOY2 | | | 4 | JOY3 | | | 5 | POT Y | | | 6 | BUTTON B | | | 7 | + 5V | MAX. 166mA | | 8 | GND | | | 9 | POT X | | Fig. 5.1. (contd.) An understanding of the two 6526 CIA interface chips is essential if all the features of the CBM 64 are to be used to the full, and a knowledge of these chips helps to explain some of the quirks of the system. The functioning of the chips is controlled by internal programmable registers, and there are 16 registers in each chip. These 32 registers (16 from each chip) are located in addressable memory space and are located at hex \$DCØØ to \$DDFF (decimal 5632Ø to 56831). They can thus be accessed from Basic using PEEK and POKE statements and from machine code using LDA and STA commands. Of the 40 I/O lines output from the two CIA chips the user can directly connect equipment to, and control the functioning of or input from, 21 lines; the other 18 lines are used by the keyboard or memory bank select and are not therefore particularly usable; one line is not connected. All but two of the I/O lines on CIA#2 can be used, but only three of the lines on CIA#1, CIA#2 is thus used in all the examples in this section. The functions of each I/O line from the two CIA chips are shown in Fig. 5.2 and the electrical connections which allow the user to utilise some of the lines are shown in Fig. 5.3. Though these lines are all assigned particular functions the user is not confined to using a particular | PIN# | TYPE | NOTE | PIN# | TYPE | NOTE | |------|---------------|---------------------|------|------|------| | 1 | GND | | Α | GND | | | 2 | +5V | 1 <b>φ</b> φmΑ ΜΑΧ. | В | FLAG | | | 3 | RESET | | С | PBØ | 1 | | 4 | CNT1 | | D | PB1 | | | 5 | SP1 | | Ε | PB2 | | | 6 | CNT2 | | F | PB3 | | | 7 | SP2 | | Н | PB4 | | | 8 | PC2 | | J | PB5 | | | 9 | SERIAL ATN IN | | K | PB6 | | | 1 Ø | +9V AC | 1 <b>00</b> mA MAX. | L | PB7 | | | 11 | +9V AC | | M | PA2 | | | 12 | GND | | N | GND | | Fig. 5.2. The allocation and function of pins on the user port connector. Fig. 5.3. User port edge connector line definition. I/O line for the function designated for that line. This is because all the I/O lines are under software control and it is not until the routines within the operating system which utilise that line for a particular function are called that that line is used. This flexibility allows the redefinition of I/O line function and is one of the most useful features of the CBM 64. The 6526 is a very complex chip with sixteen different addressable registers. Each bit within these registers has a specific function, either as an input or an output or to control the operation of the 6526. The registers are of six basic types; $I/\emptyset$ , data direction, peripheral control, shift register, timers and timer control registers. The 6526 can be functionally divided into two component parts. On one side are the connections to the processor; the processor interface. On the other side are the input output lines; the peripheral interface. The main components of the processor interface are the eight bi-directional data lines. These are connected directly to the processor data bus and are used to transfer data between the CIA and the processor. As with any memory, the processor treats the 6526 as a sixteen byte block of memory. The direction of data transfer is controlled by the R/W line, the exact timing of a transfer being controlled by the $\phi$ 2 clock line. The individual registers are addressed by the register select lines connected to the bottom address lines A@-A3. The exact location of the 6526 within memory space is determined by decoding some of the address lines and connecting these to the chip select inputs. The registers of the 6526 will be accessed only if chip select CS is low. As with all the I/O chips the 6526 can generate a processor interrupt by pulling the IRQ line low. This occurs whenever an internal interrupt flag is set as a result of an input on one of the peripheral control lines. The processor interface lines have seven basic functions which can be summarised as follows: - 1) Phase two clock $(\phi 2)$ data transfers between the 6526 and the processor take place only when the $\phi 2$ clock is high. This clock also acts as a time base for the internal 6526 timers and shift register. On the CBM 64 the $\phi 2$ clock is derived from the 6510 microprocessor chip which in turn is derived from the $\phi 0$ clock produced by the VIC chip. The $\phi 2$ clock has a frequency of 0.98 MHz on a PAL machine (UK version 64) and 1.02 MHz on an NTSC (US version 64). - 2) Chip select line (CS) the chip select input is connected to the decoding circuitry connected to the PLA chip. - 3) Register select lines (RSØ, RS1, RS2, RS3) the four register select lines are connected to the processor address bus lines AØ A3. This allows the register to select one of the sixteen registers in the 6526. - 4) Read/write line (R/W) the direction of data transfer between the 6526 and the processor is controlled by the R/W line. If R/W is high then a 'read' operation is performed and data is transferred from the 6526 onto the data bus. If R/W is low then a 'write' operation is performed and data currently on the data bus is loaded into the addressed register of the 6526. - 5) Data bus (DBØ to DB7) data is transferred between the processor and the 6526 via the eight bi-directional lines of the data bus. The internal data bus of the 6526 will only be connected to the processor data bus when the two chip select lines are enabled and the $\phi 2$ clock is high. The direction of data transfer will depend on the state of the R/W line and the register addressed on lines RSØ to RS3. - 6) Reset (RES) the reset line clears all the internal registers of the 6526 (except the timers and shift register) and sets them all at logic zero. The result is that all the interface lines are put in the input state, and timers, shift register and interrupts are all disabled. This is connected to the processor power up circuitry and is used only when the system is switched on (this line is accessible externally and since the system software can be changed its function could be modified). 7) Interrupt request (IRQ) the interrupt request output from the 6526 is very important in the CBM 64. The IRQ line goes low whenever an internal interrupt flag is set and the corresponding interrupt enable flag is high. On CIA#2 the IRQ line is connected to the processor NMI interrupt line; the NMI line is also used to test the RESTORE key. On CIA#1 the IRQ line is connected to the processor IRQ line. The function of this line is to generate a regular 60 Hz interrupt which is used by the clock, I/O and keyboard routines; this interrupt is provided by Timer A in the CIA. # 5.2 The peripheral interface lines The peripheral interface lines on each 6526 CIA chip are divided into two I/O ports, each port having eight bi-directional I/O lines. The two ports share two handshaking and two serial control lines. The following is a brief description of the I/O buses and control lines of a 6526. - 1) Peripheral ports A and B (PAØ PA7) and (PBØ PB7) these ports consist of eight bi-directional lines each of which can be independently programmed under control of the data direction register to act as either an input or an output. The polarity of the lines defined as outputs is controlled by the contents of the output register. The internal control registers are used by the processor to control the modes of operation of the 6526. All lines represent a load of one standard TTL gate in the input mode and will drive two standard TTL loads in the output mode. - 2) Handshaking lines (FLAG, PC) the FLAG peripheral control line acts as interrupt input, and PC as handshake output for peripheral port B. Each line controls an internal interrupt flag with a corresponding interrupt enable bit. The various modes of operation are controlled by the processor via the internal control registers of the 6526. ## 5.3 Operation of the I/O ports Three registers are required to access each of the eight line peripheral ports; they are a data direction register, an output register and an input register. Each port has a data direction register for specifying whether each of the eight lines acts as either an input or an output. A zero in a bit of the data direction register causes the corresponding peripheral line to act as an input. A one causes the line to act as an output. Example: Set lines \( \theta \) to 3 as inputs and 4 to 7 as outputs on port B of CIA#2. | I/O line<br>number | Data<br>direction | DDR contents if line = in | DDR contents for example | |--------------------|-------------------|---------------------------|--------------------------| | ø | in | 1 | ø | | 1 | in | 2 | Ø | | 2 | in | 4 | Ø | | 3 | in | 8 | Ø | | 4 | out | 16 | 16 | | 5 | out | 32 | 32 | | 6 | out | 64 | 64 | | 7 | out | 128 | 128 | | | | Total for example - | 24Ø | Each peripheral line is connected to an input register and an output register. When a line is programmed to act as an output the voltage on that line is controlled by the corresponding bit in the output register. A '1' in the output register causes the corresponding line to go high, and a 'Ø' causes it to go low. Example: Output to port B of CIA#2 using the data direction set out in the previous example. Lines 4 and 7 are high and lines 5 and 6 are low. | I/O line<br>number | Data direction of line | High or<br>low | Value of line<br>in I/O reg | |--------------------|------------------------|-------------------|-----------------------------| | Ø | in | _ | _ | | 1 | in | - | - | | 2 | in | - | _ | | 3 | in | - | - | | 4 | out | high | 16 | | 5 | out | low | Ø | | 6 | out | low | Ø | | 7 | out | high | 128 | | | | Total for example | 144 | Reading one of the peripheral port registers causes the contents of the input register to be transferred onto the data bus. With input latching disabled the contents of the input registers will always reflect the data currently on all the peripheral port lines. Example: Read the contents of the input lines of port B of CIA#2 set up in the example on data direction, and store the contents as variable A. ``` A = PEEK (56577) AND 15 ``` The AND 15 masks off the lines used as outputs; they must be removed since the current state of the output lines is stored in the input register. AND commands can then be used to determine which lines are high and which are low. ## 5.3.1 Registers used in the operation of the I/O ports Register 1 Parallel port A I/O register CIA#1 - Hex \$DCØØ decimal 5632Ø CIA#2 - Hex \$DDØØ decimal 56576 This register contains the contents of the input and output lines of port A of the 6526. Register 2 Parallel port B I/O register with handshake control CIA#1 - Hex \$DCØ1 decimal 56321 CIA#2 - Hex \$DDØ1 decimal 56577 This register contains the contents of the input and output lines of port B. It is identical to that of port A, except that this register has control over the handshake line PC. The PC line goes low for one clock cycle following either a read or write to the port B peripheral I/O register. Register 3 Data direction register for port A CIA#1 - Hex \$DCØ2 decimal 56322 CIA#2 - Hex \$DDØ2 decimal 56578 This register controls each of the eight lines on port A and determines whether they are acting as inputs or outputs. A one in any of the eight bits of this register sets the corresponding line into the output mode, and a zero puts it into the input mode. Register 4 Data direction register for port B CIA#1 - Hex \$DC03 decimal 56323 CIA#2 - Hex \$DDØ3 decimal 56579 This register controls each of the eight lines on port B and determines whether they are acting as inputs or as outputs. A one in any of the eight bits of this register sets the corresponding line into the output mode, and a zero puts it into the input mode. ## 5.3.2 Handshaking The term handshaking is used to refer to signals which control or synchronise the transfer of data between the computer and another device. The 6526 has two handshaking lines for use on data transfers; the $\overline{FLAG}$ input and the PC output. The PC line is used to indicate that data is ready to be transmitted or received on port B. This is indicated by the PC line going low for one clock cycle following a read or write of the port B data register. The FLAG line is an input which, on receiving a negative transition, will set the FLAG bit in the interrupt control register. The FLAG line can be used to detect the PC output from another 6526. #### 5.4 The interval timers and counters of the 6526 The 6526 has three internal interval timers (Timer A, Timer B, and TOD). One of these, TOD, is a 24 hour time of day clock. Timer A will also function as a counter of pulses input on one of the I/O lines. These timers are not only useful but are of vital importance to the operation of the CBM 64. It is these timers which are used to control the generation of the 6 $\emptyset$ Hz interrupt used to update the real time clock and scan the keyboard. They are also used to control the timing of I/O on the serial port, the RS232 port and the cassette. Since the CBM 64 interface uses two 6526 chips there are a total of six timers available for use by the system software. The timers are used in conjunction with the processor interrupts. The following table shows some of the functions of each timer plus the interrupt line affected: ## CIA#1 IRQ interrupt Timer A - System 60 Hz interrupt Real time clock updating Keyboard scanning Cassette read/write timing User programmable functions Timer B - Cassette read/write timing Serial port timing User programmable functions TOD - User programmable functions Note that Timer A is used for both updating the kernal software's real time clock and cassette timing; for this reason the real time clock loses whenever the cassette is used. #### CIA#2 NMI interrupt Timer A - RS232 port I/O timing User programmable functions Timer B - RS232 port I/O timing User programmable functions TOD - User programmable #### 5.4.1 Timer A and Timer B Each interval timer consists of two eight bit latches and a sixteen bit counter. Each timer occupies two of the 6526 registers; register numbers 5 to 8. Their locations in the CBM 64 are as follows: | Register 5 | <ul> <li>Timer A low order byte</li> </ul> | | |------------|---------------------------------------------|---------------| | | CIA#1 – Hex \$DCØ4 | decimal 56324 | | | CIA#2 – Hex \$DDØ4 | decimal 5658Ø | | Register 6 | <ul> <li>Timer A high order byte</li> </ul> | | | | CIA#1 – Hex \$DCØ5 | decimal 56325 | | | CIA#2 – Hex \$DDØ5 | decimal 56581 | | Register 7 | <ul> <li>Timer B low order byte</li> </ul> | | | | CIA#1 – Hex \$DCØ6 | decimal 56326 | | | CIA#2 – Hex \$DDØ6 | decimal 56582 | | Register 8 | <ul> <li>Timer B high order byte</li> </ul> | | | | CIA#1 – Hex \$DCØ7 | decimal 56327 | | | CIA#2 – Hex \$DDØ7 | decimal 56583 | These registers are used to load values into the counters. After loading, the counter decrements at the system clock rate (0.98 MHz on PAL machines and 1.02 MHz on NTSC). Thus if the counter is loaded with its maximum value (all sixteen bits = 1 or decimal 65535) it will be decremented to zero in .0669 seconds. Upon reaching zero, an interrupt flag is set and one of the two interrupt lines will go low and generate a processor interrupt. The timer will thus disable any further interrups until the interrupt servicing routine reads the interrupt control register of the 6526. In addition, the timer can be instructed to invert the output level on one of the peripheral I/O lines each time it 'times out'. The modes of operation are controlled by reading or writing to the four timer registers plus the two control registers and the interrupt register. The two timers A and B can be linked to create a single 32 bit counter which is capable of creating long delays. Program 7 demonstrates the operation of a 32 bit timer. ``` 10 POKE56583,0:POKE56582,0 20 POKE56581,255:POKE56580,255 30 POKE56590,16+1:POKE56591,64+16+8+1 ``` 30 PURE36330,16+1.PURE36331,64+16+6+1 35 T1=TI 40 IF (PEEK (56589) AND 2) < 2THEN 40 50 PRINT(TI-T1)/60 Program 7. # 5.4.2 Time of day clock - TOD This is a general purpose 24 hr am/pm timer with a 1 10 second resolution. This timer occupies four registers within the 6526. They are located and organised as follows in the CBM 64: | Reg # | CIA#1 location | CIA#2 location | Function | |-------|----------------|----------------|---------------| | 8 | \$DCØ8 - 56328 | \$DDØ8 - 56584 | TOD 1/10 secs | | 9 | \$DCØ9 - 56329 | \$DDØ9 - 56585 | TOD secs | | 1Ø | \$DCØA - 5633Ø | \$DDØA - 56586 | TOD mins | | 11 | \$DCØB - 56331 | \$DDØB - 56587 | TOD hours | 136 The Commodore 64 Kernal and Hardware Revealed Each of these registers stores its data as shown in the following table. It should be noted that the values are stored in BCD form rather than straight binary: | Register | Byte | 7 | 6 | 5 | 4 | 3 | 2 | 1 | Ø | |---------------------------------------------------------------------------|-------|--------------------|------------------------|------------------------|---------------------------|--------------|------|------|---| | 8 - TOD 1/10 sec<br>9 - TOD seconds<br>10 - TOD minutes<br>11 - TOD hours | PM fl | Ø<br>Ø<br>Ø<br>lag | Ø<br>shi4<br>mhi4<br>Ø | ø<br>shi2<br>mhi2<br>ø | Ø<br>shil<br>mhil<br>hhil | slo8<br>mlo8 | mlo4 | slo2 | | shi, mhi and hhi are the decade portion of the respective second, minute or hour and slo, mlo and hlo are the unit portion. The TOD clock is timed by a 5 $\emptyset$ or 6 $\emptyset$ Hz external clock pulse provided for the system interrupt timing (5 $\emptyset$ Hz is found on US machines and 6 $\emptyset$ Hz on UK machines). This external clock frequency must always be set to the correct value by setting the required bit in control register, .A bit 7, to $\emptyset$ =6 $\emptyset$ Hz and 1=5 $\emptyset$ Hz. The TOD time registers can be preset by writing to them the required time yalues. This must, however, be done in the correct sequence. The TOD clock is stopped when a write is performed to the hours register. The clock will not start again until a write is performed to the 1/1 $\emptyset$ seconds register, thereby allowing accurate time setting. When reading the contents of the TOD registers there is the problem of the register values changing whilst they are being read. This is overcome by latching all the register values on a read of the hours register. Whilst the values are latched the TOD clock will continue to count but will not affect the register values until the 1/10 seconds register is read; this disables the latches. Any of the registers (apart from hours) can be read without latching providing the problem of carry between registers is not important. The TOD clock incorporates an alarm feature which causes an interrupt to be generated whenever the time reaches a preset value. The alarm is set by first setting bit 7 of control register .B to 1 and then writing the desired alarm time value into the four TOD registers. Having written the desired values into these registers, control register .B bit 7 should be reset to zero. Program 8 demonstrates the use of the TOD clock registers and the alarm feature. ``` 100 POKE1031/(SAND15)0R48 110 POKE1033, $100R48 120 B8=16: IF(HAND128)<128 THEN B8=1 130 POKE1035, B8 140 IF PEEK(CIA+13)AND4=4THENPRINT"如如紹和LARM":END 150 GOTO 10 997 REM 998 REM SET CLOCK OR TIME 999 REM 1000 IF MO=0 THEN PRINT"=C%SET TIME:":POKECIA+15,PEEK(CIA+15)AND127:GOTO 1020 1010 PRINT"=CISSET ALARM: ": POKECIA+15, PEEK(CIA+15) OR128 1020 INPUT"HOURS"; H: H=INT(H) 1030 IF HC1 OR H>12 THEN PRINT":T7":GOTO 1020 1040 PRINT"AM OR PM ?"; 1050 GETA$: IFA$<>"A"ANDA$<>"P"THEN1050 1060 PRINTAS 1061 IF MO=1 THEN 1070 1065 IF H=12 AND A$="R" THEN A$="P":GOTO1070 1066 IF H=12 AND A$="P" THEN A$="A" 1070 B=128: IFA$="A"THENB=0 1080 POKECIA+11, B+INT(H/10)*16+H-INT(H/10)*10 1090 INPUT"MINUTES";M:M=INT(M) 1100 IF M<0 OR M>59 THEN PRINT"TT":GOTO1090 1110 POKECIA+10, INT(M/10)*16+M-INT(M/10)*10 1120 INPUT"SECONDS";S:S=INT(S) 1130 IF S<0 OR S>59 THEN PRINT"TT":GOT01120 1140 POKECIA+9, INT(S/10)*16+S-INT(S/10)*10 1145 IF MO=1 THEN 1170 1150 PRINT MORPRESS ANY KEY TO START TIMER" 1160 GETA$: IFA$=""THEN1160 1170 POKECIA+8,0:RETURN ``` Program 8. ## 5.5 Serial data register (SDR) One of the registers of the 6526, register 12, functions as a serial in/parallel out or parallel in/serial out shift register. The serial input or output from this register is connected to the SP pin on the chip and is designed to be used in conjunction with the CNT line to allow serial communications between 6526 chips. Data is clocked in or out of the shift register using either Timer A or the CNT pulses. In the input mode data is clocked in off the SP line on the rising edge of a clock pulse applied to the CNT line. Each pulse on the CNT line clocks in one bit of data, represented by the state of the SP line. After 8 CNT pulses the data in the shift register is transferred to the serial data register and an interrupt to the processor is generated. In the output mode, data is loaded into the serial data register. Timer A is used to generate the timing rate at which data is clocked out. Timer A is first set into continuous running mode and data will be shifted out at half the timer underflow rate, the highest possible data rate being one quarter of the $\phi 2$ clock (245 KHz on a UK version of the 64 and 255 KHz on a US version). As soon as the data is written into the SDR transmission will commence, assuming that Timer A has already been set into continuous operation. Every time an underflow is generated by Timer A a CNT pulse is generated. A bit from the SDR data is shifted onto the SP line, becomes valid on the falling edge of the CNT pulse and remains valid until the next falling edge of a CNT pulse. After eight CNT pulses the entire contents of the SDR will have been transmitted on the SP line and an interrupt is then generated to indicate that data transmission has been completed. On completion of transmission the CNT line will go high and the SP line will stay at the level of the last data bit transmitted. If a new byte of data is loaded into the SDR before the transmission of the previous byte has been completed, the 6526 will complete transmission of the current byte, and then instead of generating an interrupt, will carry on and transmit the second byte. In this manner continuous transmission can be achieved. An important potential use for the serial line is in serial communications between computers and/or peripheral devices which also use the 6526. Program 9 shows how two or more CBM 64 computers can be connected via the CNT and SP lines, plus ground, to enable the transmission of programs and data between the two machines. This could easily be expanded to work with more machines. ``` 0 REM RUN TO SEND FIRST 1 REM OR RUN70 TO RECEIVE FIRST 2 REM 5 INPUTAs: As=As+CHRs(13) 10 POKE56580,2:POKE56581,0 20 POKE56590,64+16+1 25 FORI=1TOLEN(A$) 26 J=ASC(MID$(A$,[,1)) 30 POKE56588,J:PRINTCHR$(J); 50 IF (PEEK(56589)AND8) < 8 THEN 50 60 NEXT: POKE56588, 0 65 IF (PEEK(56589)AND8) < 8 THEN 65 70 POKE 56590, PEEK (56590) AND (128+63) 30 POKE56588,0 90 IF (PEEK(56589)AND8)(8 THEN 90 100 I=PEEK(56588): IFI<>0THENPRINTCHR$(I);:GOTO90 110 GOTO5 10 POKE56579,1 20 POKE56326,255:POKE56327,255 30 POKE56335,32+16+1 40 PRINT"D" 50 PRINT"#"65535-(PEEK(56326)+PEEK(56327)#256) 60 FORI=0T0100:NEXT 70 GETA$: IFA$=""THEN70 80 POKE56577,1:POKE56577,0 90 GOTO 50 10 POKE56590, PEEK (56590) AND (128+63) 20 POKE56579,1 30 GETA$: IFA$=""THEN30 40 POKE56577,0:POKE56577,1 50 IF(PEEK(56589)AND8)<8THEN30 60 PRINTPEEK (56588): GOTO30 ``` Program 9. ## 5.6 The interrupt control register (ICR) The 6526 can generate interrupts in several different ways. There are altogether five sources of interrupts. The source of the interrupt is identified by examining which bit is set in the interrupt control register; this is register 13 of the 6526. The functions of the individual bits of this register are: | ICR bit # | Interrupt source | |-----------|---------------------------------| | Ø | Underflow from Timer A | | 1 | Underflow from Timer B | | 2 | TOD clock alarm | | 3 | Serial data register full/empty | | 4 | Flag line input | | | | Bits 5 and 6 are not used, while bit 7 is used to set or clear the ICR mask register. The ICR, in fact, consists of two separate registers; the interrupt flag register which is read only and the interrupt mask register which is write only. When an interrupt occurs the corresponding bit in the interrupt flag register is set. providing it has previously been enabled by the mask register; an interrupt to the processor is also generated. The mask register is used to control which function or functions can create an interrupt. An interrupt will occur only when the corresponding mask register bit is set. When enabling an interrupt the mask register, with bit 7 = 1, is set by writing the appropriate bit pattern to register 13. the ICR. If, when writing to the ICR, bit 7 is cleared then all mask bits set to one will be cleared whilst all mask bits set to zero will remain in their previous state. Any interrupt which is enabled by the mask register will set bit 7 of the ICR flag register and thereby cause the IRQ pin to go low and generate an interrupt. The interrupt is cleared by reading the interrupt flag register. The interrupts on the CBM 64 are examined in greater detail in the Chapter 6. # 5.7 The 6526 control registers (CRA and CRB) The two control registers of the 6526 are used, as their names imply, to control the actual functioning and modes of the timers and the serial port. Each bit in the two registers has a separate control function; they can be summarised as follows: | ontrol | register A | | |--------|------------|---------------------------------------------------------------| | Bit | State | Function | | Ø | Ø | start Timer A | | | 1 | stop Timer A; in one shot mode this bit is reset on underflow | | 1 | Ø | line PB6 functions normally as an I/O line | | | 1 | output from Timer A appears on line PB6 | | 2 | Ø | pulse output on PB6 (only if bit 1 set) | | | 1 | toggle output on PB6 (only if bit 1 set) | | 3 | Ø | Timer A in continuous running mode | | | 1 | Timer A in one shot mode | | Bit | State | Function | |-----|-------|----------------------------------------------------------------| | 4 | Ø | no effect | | | 1 | forces the Timer A counter to be loaded from the Timer A latch | | 5 | Ø | Timer A counts $\phi$ 2 clock pulses | | | 1 | Timer A counts positive transitions on the CNT line | | 6 | Ø | SP line in input mode using external shift pulses on CNT line | | | 1 | SP line in output mode. CNT sources shift pulses | | 7 | Ø | TOD clock timing pulse at 60 Hz (use on US machines) | | | 1 | TOD clock timing pulse at 50 Hz (use on UK machines) | | Control<br>Bit | Register B<br>State | Function | | | |----------------|---------------------|------------|----------|--------------------------------------------------------------------| | Ø | Ø | stop Time | er B | | | | 1 | start Time | er B; ii | n one shot mode this bit is reset on underflow | | 1 | Ø | normal I/ | O ope | ration on the PB7 line | | | 1 | output fro | om Tir | ner B appears on line PB7 | | 2 | Ø | pulse outp | out on | PB7 (only if bit 1 of CRB set) | | | 1 | toggled or | utput ( | on PB7 (only if bit 1 of CRB set) | | 3 | Ø | Timer B i | n cont | inuous running mode | | | 1 | Timer B i | n one | shot mode | | 3 | Ø | no effect | | | | | 1 | forces the | Timer | B counter to be loaded from the Timer B latch | | 5,6 | | these two | bits se | elect one of four input modes for Timer B: | | | | Bit 6 | Bit 5 | Mode | | | | Ø | Ø | Timer B counts $\phi$ 2 clock pulses | | | | Ø | 1 | Timer B counts positive CNT transitions | | | | 1 | Ø | Timer B counts Timer A underflow pulses | | | | 1 | 1 | Timer B counts Timer A underflow pulses while the CNT line is high | | 7 | Ø | writing to | TOD | registers will set TOD clock | | | 1 | writing to | TOD | registers will set TOD alarm | ## 5.8 Parallel interfacing In some applications it is simply not possible to connect devices directly to the eight user port I/O lines. This is generally for one of two reasons. The first is that the user port lines do not output enough power to drive the device; directly connecting the device to the user port could result in damage to the CIA chip. The second reason is that there are insufficient I/O lines for the application. The first type of problem can be overcome by using relays and opto-isolators; the second type by using multiplexing techniques to expand the available number of I/O lines. The eight I/O lines from the CIA chip are in output mode only, capable of each driving the input of a single TTL chip. The simplest method of improving this drive capability is to put a buffer onto each output line; this will expand the drive capability of each line to 10 TTL chip inputs. This is adequate if the I/O lines are used just to control some TTL circuitry, but it is inadequate for controlling power devices such as motors, relays and lamps; for these a power driver circuit is needed. The simplest power driver circuit consists of a single transistor whose base is connected to the output of a buffer and whose outputs are connected on one side to the power supply and on the other to the device to be driven. It is, however, much easier to use one of the peripheral driver ICs like the SN75446. This chip is used in the circuit diagram in Fig. 5.4. The SN75446 is capable of driving devices requiring up to 50 volts and 400 mA. The circuit shows it driving two relays. Fig. 5.4. Dual relay control circuit. If there is any risk of a high voltage accidentally being present on one of the I/O lines it is advisable to protect the computer by using opto-isolators on lines. An opto-isolator simply consists of an optically coupled LED and photo transistor. An example of such a device which provides four separate opto-isolators which work on 5 volt lines is the ILQ74. The most commonly needed I/O expansion is the requirement to test the state of a large number of input lines. This kind of I/O expansion is used when adding a special keyboard to the machine or testing switches in a security alarm system. Fig. 5.5. I/O expansion circuit. A circuit of this kind is shown in Fig. 5.5. It uses four 16 line to 4 line demultiplexers which convert the state of the 16 lines into a 4 bit binary value. This value is fed into four lines of the user port. The chip select line of each of the four demultiplexers is controlled by four output lines of the user port. Using this circuit the computer can scan the state of up to 64 input lines. # 5.8.1 Parallel port application example Program 10 is an example of how the parallel port can be used. This program is designed to allow disabled people to use a computer, both to write and use programs and to control various devices such as TV, radio, lamps etc. The program requires line $\emptyset$ of the user port to be connected to a switch, the other side of which is connected to ground. The switch can be a simple microswitch usable with minimum finger pressure, or a suck/blow switch controllable by the user blowing into it. The other I/O lines are connected to devices which are to be controlled, and each line will require connection via a power driver circuit and relay, as in Fig. 5.4. A selection of possible applications is shown in the program. ``` 6 PR=5:MC=6 ","SENT ","DELC ","DELW ", ","PARA ","SEND ","ESC ", 10 DATA" LC 20 DATA" UC 30 DATA"TEXT ","PROG ","USER ","HELP ","-VAR " 100 DATA" SP ",A,T,L,G,V," THE " , "DEL 110 DATAE, I, S, U, Z, Q, " 120 DATAN, O.R.P.J.#," 130 DATAC. D. H. M. X. @. " 140 DATAF, B, W, Y, K, 7, " 150 DATA0, 2, 4, 6, 8, " // IN 160 DATA1,3,5,7,9,!,"THAT 170 DATA.,+,-,*,/,←,$ 180 DATA", ",?,=,%,(,),< 190 DATA&, ↑,";",":",[,],) 200 DATA" ",A,T,L,G,V,THE 210 DATAE, I, S, U, Z, Q, AND ``` ``` 220 DATAN, 0, R, P, J, #, 0F 230 DATAC,D,H,M,X,@,IS 240 DATAF, B, W, Y, K, T, TO 250 DATA0,2,4,6,8, ,IN 260 DATA1,3,5,7,9,!,THAT 270 DATA.,+,-,*,/,←,$ 280 DATA",",?,=,%,(,),( 280 DATA",",?,=,%,(,),<br/> 290 DATA&,↑,";",":",[,],><br/> 400 DATA" SP ",1,7,=," <> ","GOTO "," ON "<br/> 410 DATA0,2,6,",",>,"INPUT"," VAL "<br/> 420 DATA3,4,5,#,<,"POKE ","CHR$ "<br/> 430 DATA8,9,.," '< "," <= ","PEEK ","STR$ "<br/> 440 DATA),(,$,↑," >= ","LEN ","LEFT$"<br/> 450 DATA+,-,*,/," ","ASC ","RGHT$"<br/> 460 DATA"LIST ","DATA ","IF ","GOSUB"," DIM ","TRON ","MID$ "<br/> 470 DATA"RUN ","CONT ","THEN ","RTRN ","STOP ","TROFF"," LOG "<br/> 480 DATA"READ ","INT ","ABS ","SGN ","RND ","SQR ","EXP "<br/> 490 DATA"PRINT"," FOR ","NEXT ","SIN ","COS ","TAN "," " 500 DATA" ",1,7,=,<>,GOTO,ON 510 DATA0,2,6,",",>,INPUT,VAL( 520 DATA3,4,5,#,<,POKE(,CHR$( 530 DATA8,9,,," ",<=,PEEK(,STR$( 540 DATA),(,$,↑,>=,LEN(,LEFT$( 550 DATA+,-,*,/,"",ASC(,RIGHT$( 560 DATALIST, DATA, IF, GOSUB, DIM, TRACEON, MID&C 570 DATARUN, CONT, THEN, RETURN, STOP, TRACEOFF, LOG( 580 DATAREAD, INTC. ABSC. SGNC, RNDC, SQRC, EXPC 590 DATAPRINT, FOR, NEXT, SINC, COSC, TANC, "" 1000 RESTORE 1010 DIMDF$(2,6),DA$(1,9,6),TA$(1,9,6),DV$(20),TV$(20) 1020 FORI=0T02:FORJ=0T06 1030 READDF $(I, J) 1040 NEXTJ:NEXTI 1050 FORI=0T01 1060 FORJ=0T09 1070 FORK=0T06 1080 READDA$(Î,J,K) 1090 IFLEN(DA$(I,J,K))=1THENDA$(I,J,K)=" "+DA$(I,J,K)+" " 1100 NEXTK: NEXT,I 1110 FORJ=0T09 1120 FORK=0T06 1130 READTA$(I,J,K) 1140 NEXTK:NEXTJ 1150 NEXTI 1170 TA$(0,5,5)=CHR$(34) 1190 TA$(1,3,3)=CHR$(34) 1200 FORI=0T020 ":TV$(I)="" 1210 DV$(I)=" 1220 NEXTI 1700 UP=56577:UX=1 1710 VN=0:M=0:A$="":UC=1:ED=0:HP=0:TV=0:NY=0 1720 PD=70:PQ=50:FM=0 1730 GOSUB 20000 2000 PRINT" TRANSMANADVICE"; 2010 PRINT" SOURCE WHOME"; 2020 PRINT" SEMENDER :: 2030 FORI=0T09:PRINT" 2040 FORJ=0TO6:PRINTDA$(M,I,J); 2050 NEXTJ:NEXTI 2055 PRINT 2060 PRINT"SONDO CONTROL : "NO CONTROL : "*VARIABLES"; 2070 PRINT" SUBDED DESCRIPTION PROPERTY; 2080 FORI=0T02:PRINT" "; 2090 FORJ=0T06:PRINTDV$(I*7+J); 2100 NEXTJ:NEXTI 2110 PRINT" SAMMANIAN PRINTE PRINT" SAMMANIAN PRINTE P 2120 PRINT "SECONDED DE LE SERVE DE LE PRINT "SECONDE SE LE SECONDE L 2130 FORI=0T02:PRINT" 2140 FORJ=0TO6:PRINTDF$(I,J); 2150 NEXTJ:NEXTI ``` ``` 2160 PRINT"3"; A$; 3000 PRINT"%": PRINT 3005 IFHP=0G0T03040 3010 PRINT"海现现现株ADVICE SSELECT ANY BOX FOR HELP室"; 3020 HP=HP+1 3030 GOT03200 3040 IFLEN(A$)+LEN(N$)+LEN(Y$)+LEN(Z$)<160G0T03070 3050 PRINT"周期现成体WARNING #PRINT STRING -- NOW! !!;:GOTO3200 3070 FORI=1TOLEN(N$) 3080 NB$=MID$(N$,1,1) 3090 IFUC>0G0T03130 3110 IFNB$<"A"ORNB$>"Z"GOTO3130 3120 NB$=CHR$(ASC(NB$)+128) 3130 A$=A$+NB$ 3141 IFUC=2THENGOSUB20100 3142 IFUC=2THENUC=0 3143 NEXTI 3145 N$="" 3146 IFUC=3THENUC=2 3150 PRINT"%";A$; 3160 IFED=0G0T03180 3170 PRINT"3"; Y$; "9"; Z$; 3180 PRINTFL $: FL $= " " 3200 PRINT"到":PRINT:PRINT"或吸吸吸激淋HOME "; 3205 IF(PEEK(UP)ANDUX)=0THEN3205 3210 IF(PEEK(UP)ANDUX)GOT03210 3212 IFHP>OTHEN3220 3213 IFED>0THEN3220 3214 IFNV>0THEN3220 3215 PRINT" STREET RADVICE n ; 3220 PRINT" STRUMBENHOME"; 3230 PRINT "MINICEPHEN"; 3240 FORI=0T015 3250 PRINT"# > 團"; 3260 FORJ=1TOPD:IF(PEEK(UP)ANDUX)GOT03400 3265 NEXTJ 3270 PRINT"個圈圈圈 9期周周保好: 3280 PRINT"M";: IFI=90RI=12THENPRINT"M"; 3290 NEXTI 3310 GOTO3200 3400 IF(PEEK(UP)ANDUX)GOT03400 3430 IFI>9G0T03600 3440 FORJ=0T06 3450 PRINT" #"; DA$(M, I, J); "@"; 3460 FORK=1TOPQ: IF(PEEK(UP)ANDUX)G0T03490 3465 NEXTK 3470 PRINT"日間園園園"; DA$(M, I, J); 3480 NEXTJ 3485 PRINT"即回回回回"; DA$(M, I, J-1); : GOTO3200 3490 PRINT"9團團團團"; DA本(M, I, J); 3500 IFHP=2G0T040000 3505 IF NV>0 G0T07720 3510 N$=TA$(M,I,J):GOTO3000 3600 IFI>12G0T03800 3610 II=I-10 3620 FORJ=0T06:PRINT"8";DV$(II*7+J);"9"; 3630 FORK=1TOPQ:IF(PEEK(UP)ANDUX)G0T03670 3635 NEXTK 3650 PRINT"9團團團團"; DV$(II*7+J); 3660 NEXTJ 3665 PRINT"颶圓圓圓門; DV$(II*7+J-1); : GOTO3200 3670 PRINT"個個團團"; DV$(II*7+J); 3680 IFHP=2G0T042000 3685 IF NV>0 G0T07740 3690 N$=TV$(II*7+J):GOT03000 3800 II=I-13 3810 FORJ=0T06:PRINT"#";DF$(II,J);"9" 3830 FORK=1TOPQ:IF(PEEK(UP)ANDUX)GOT03860 3835 NEXTK ``` ``` 3840 PRINT"調圖圖圖J"; DF$(II, J); 3850 NEXTJ 3855 PRINT"網圈團團團"; DF$(II, J-1); : GOT03200 3860 PRINT"随圈圈圈"; DF$(II, J); 3870 SS=(II*7+J)+1 3875 IF NY>0 GOTO7720 3880 IFHP=2G0T044000 4000 IFSS>7G0T04020 4010 ONSSGOTO4100,4300,4500,4700,4900,5100,5300 4020 IFSS>14G0T04040 4030 ONSS-7GOTO5500,5700,5900,6100,6300,6500,6700 4040 ONSS-14GOT06900,7100,7300,7500,7700,7900,8100 4100 REM SET TO LOWER CASE 4110 UC=0:GOSUB20100:GOT03200 4300 REM BEGIN A SENTENCE 4310 UC=3:GOSUB20000:A$=A$+". ":GOT03000 4500 REM DELETE CHARACTER 4505 IFLEN(A$)=0G0T03200 4510 IFLEN(A$)>1G0T04530 4520 A$="":FL$=" ":GOTO3000 4530 A$=LEFT$(A$,LEN(A$)-1):FL$=" ":GOTO3000 4700 REM DELETE WORD 4701 IFLEN(A$)=0G0T03000 4702 IFLEN(A$)=1G0T04745 4705 FORI=LEN(A$)T02STEP-1 4710 NB$=MID$(A$,I,1) 4720 IFNB$=" "GOT04750 4730 A$=LEFT$(A$,I-1):FL$=FL$+" " 4740 NEXTI 4745 IFLEN(A$)=1ANDLEFT$(A$,1)<>" "THENGUSUB4800 4750 GOT03000 4800 A$="":FL$=FL$+" ":RETURN 4900 REM NOT DEFINED 4910 GOTO3200 5100 REM NOT DEFINED 5110 GOTO3200 5300 REM CR-SENDCR/LF TO PRINTER 5310 OPENPR, PR 5320 PRINT#PR 5330 CLOSE PR 5340 GOTO3000 5500 REM SET TO UPPER CASE 5510 UC=1:GOSUB20000:GOTQ3200 5700 REM PARAGRAPH 5710 OPENPR, PR 5720 GOSUB 21000:GOSUB 20900 5725 GOSUB 20000 5730 PRINT#PR:PRINT#PR:PRINT#PR 5740 A$=" 5750 UC=3 5760 CLOSE PR 5770 GOTO3000 5900 REM PRINT CONTENTS OF A$ 5910 OPEN PR,PR 5920 GOSUB21000:GOSUB20900 5930 A$="" 5940 CLOSEPR 5950 GOT03000 6100 REM SEND LINE TO MICRO 6110 OPEN MC.MC 6120 PRINT#MC,A$ 6125 GOSUB20900 6130 A$="" 6140 CLOSEMC 6150 GOTO3000 6300 REM ADD A MAKE A$ THE NEXT VARIABLE 6310 IF VN<21 THEN 6340 6320 PRINT SOMOWHADVICE - NO ROOM FOR NEW VARIABLE"; 6330 GOTO3000 6335 IF A$=""THEN3000 ``` ``` 6340 TV$(VN)=A$ 6350 DV$(VN)=LEFT$(TV$(VN)+" ",5) 6360 DV$(VN)=LEFT$(DV$(VN),4) 6365 DV$(VN)=DV$(VN)+CHR$(32) 6370 GOSUB22000 6380 YN=VN+1 6390 GOTO3000 6500 REM SEND ASCII 3 TO MICRO (BREAK-IN) 6510 OPEN MC, MC 6520 PRINT#MC, CHR$(3); CHR$(3); 6530 CLOSE MC 6540 G0T03000 6700 REM UNDEFINED 6710 GOTO 3200 6900 REM TEXT MODE 6910 M=0:GOTO2000 7100 REM PROGRAM MODE 7110 M=1:UC=1:GOSUB20000:GOTO2000 7300 REM USER MODE 7310 GOTO30000 7500 REM HELP SUB-SYSTEM 7510 HP=1:GOT03000 7700 REM DELETE VARIABLE 7701 IF VN>0 G0T07710 7703 PRINT # # PRINT 7705 G0T03000 7710 PRINT" BOROGONADVICE - SELECT VARIABLE TO BE REMOVED"; 7713 NV=1 7715 GOTO3000 7720 REM UNKNOWN VARIABLE 7725 PRINT"SMUMBHADVICE - NO VARIABLE SELECTED! 7730 NV=0 7735 GOTO3000 7740 REM DELETE SELECTED VARIABLE 7745 PRINT" DOMONDOWADVICE - O.K. 7750 FOR K=II*7+J TO 19 7755 DV$(K)=DV$(K+1) 7760 TV$(K)=TV$(K+1) 7762 NEXT K 7765 DV$(20)=" 7770 TV$(20)="" 7780 GOSUB 22000 7785 NV=0 7790 VN=VN-1 7795 GOTO3000 7900 REM UNDEFINED 7910 GOTO3200 8100 REM DELETE BUFFER 8110 GOSUB20900:A$="":GOTO3000 10000 PRINT" 知识的淋ADVICE NO HELP IN THIS VERSION"; 10005 HP=0 10010 GOTO3000 20000 PRINT SERREFERENCE DE LE CONTROL 20010 PRINT"、阿國國國國 UC米 "; 20020 DF$(1,0)=" UC* ":DF$(0,0)=" LC 20030 RETURN 20100 PRINT" SEPREMENTAL AND AND LOCK "; 20110 PRINT" A B B B B UC "; 20120 DF$(1,0)=" UC ":DF$(0,0)=" LC* " 20130 RETURN 20900 PRINT"#"; 20910 FORL=1T010 20920 PRINT" 20930 NEXT 20940 RETURN 21000 PRINT#PR,A$ 21010 RETURN 22010 FORI=0T02:PRINT" 22020 FORJ=0T06:PRINTDV$(I*7+J); ``` ``` 22030 NEXTJ:NEXTI 22049 RETURN 30000 REM USER FRAME 30010 PRINT" 海本本 SELECT FUNCTION 本本本" 30020 PRINT 30030 PRINT" T.V. ON" 30040 PRINT" T.V. OFF" 30050 PRINT" B.B.C. 1" 30060 PRINT" B.B.C. I.T.V." 30070 PRINT" 30080 PRINT" RADIO ON" 30090 PRINT" RADIO OFF" 30100 PRINT" LAMP ON" 30110 PRINT" LAMP OFF" 30120 PRINT" DOWN FASTER" 30130 PRINT" DOWN SLOWER" 30140 PRINT" ACROSS FASTER" 30150 PRINT" ACROSS SLOWER" 30160 PRINT" RETURN" 30170 PRINT" 30 *" 30180 IF(PEEK(UP)AND UX)=0 GOTO 30180 30190 IF(PEEK(UP)AND UX) GOTO 30190 30200 PRINT" ## ##"; 30210 FOR I=1 TO 14 30220 PRINT"XX"; 30230 FOR J=1 TO PD 30240 IF(PEEK(UP)AND UX) GOTO 30300 30250 NEXT J 30260 PRINT"8 01; 30270 NEXT I 30280 PRINT"0 "; 30290 GOTO 30170 30300 PRINT"H "; 30310 IF I>7 GOTO 30330 30320 ON I GOTO 30400,30500,30600,30700,30800,30900,31000 30330 ON I-7 GOTO 31100,31200,31300,31400,31500,31600,31700 30400 REM TURN TV ON 30410 POKE UP/PEEK(UP)AND 251 30420 GOTO 30600 30500 REM TURN TV OFF 30510 POKE UP, PEEK (UP) OR 4 30520 GOTO 30170 30600 REM BBC1 30610 POKE UP PEEK(UP)OR 3 30620 GOTO30170 30700 REM BBC2 30710 POKE UP/(PEEK(UP)OR3)AND254 30720 GOTO30170 30800 REM ITV 30810 POKE UP/(PEEK(UP)OR3)AND253 30820 GOTO30170 30900 REM RADIO ON 30910 POKE UP,PEEK(UP)AND247 30920 GOTO30170 31000 REM RADIO OFF 31010 POKE UP, PEEK (UP) OR8 31020 GOTO30170 31100 REM LAMP ON 31110 POKE UP, PEEK (UP) AND 239 31120 GOT030170 31200 REM LAMP OFF 31210 POKE UP, PEEK (UP) OR16 31220 GOT030170 31300 REM DOWN FASTER 31310 PD=PD-10:IF PD<30 THEN PD=30 31320 GOTO30170 31400 REM DOWN SLOWER 31410 PD=PD+10 31420 GOTO30170 31500 REM ACROSS FASTER ``` ``` 31510 PQ=PQ-10: IF PQK30 THEN PQ=30 31520 GOT030170 31600 REM ACROSS SLOWER 31610 PQ=PQ+10 31620 GOT030170 31700 REM RETURN 31710 GOTO2000 40000 REM HELP WITH LETTERS 40010 GOSUB 20900 40020 PRINT"≅BY SELECTING (":DA$(M,I,J); 40030 IF LEN(TA$(M,I,J))=0THEN PRINT" NOTHING" 40040 IF LEN(TA$(M,I,J))=1THEN PRINT" THE CHARACTER (";TA$(M,I,J);"" 40050 IF LEN(TA$(M,I,J))>1THEN PRINT"/ THE STRING /";TA$(M,I,J);"/" 40060 PRINT"TIS ADDED TO THE BUFFER." 40070 G0T049000 42000 REM HELP WITH VARIABLES 42010 GOSUB 20900: FRINT"3"; 42020 IF LEN(TV$(II*7+J))>0 THEN 42060 42030 PRINT"THIS VARIABLE (";II*7+J;") IS EMPTY;" 42040 PRINT" TSEE +VAR AND -VAR." 42050 GOTO49000 42060 PRINT"BY SELECTING VARIABLE <";DV$(II*7+J);"< THEN" 42070 PRINT"C";TV$(II*7+J);"< IS ADDED" 42080 PRINT"TO THE BUFFER." 42090 GOTO49000 44000 REM HELP WITH FUNCTIONS 44005 GOSUB 20900:PRINT"類"; 44010 IF SS>7 GOTO 44030 44020 ON SS GOT045000,45100,45200,45300,45400,45500,45600 44030 IF SS>14 GOTO 44050 44040 ON SS-7 GOTO 45700,45800,45900,46000,46100,46200,46300 44050 ON SS-14 GOTO 46400,46500,46600,46700,46800,46900,47000 45000 PRINT"$ LC ■ FOLLOWING LETTERS WILL BE LOWER" 45010 PRINT"TCASE. ALSO SEE / UC 45020 GOTO49000 45100 PRINT" #SENT ■ END OF SENTENCE, ADDS A FULL" 45110 PRINT" TSTOP AND THREE SPACES TO THE BUFFER" 45120 PRINT"AND MAKES THE NEXT LETTER A CAPITAL." 45130 GOT049000 45200 PRINT" STOELC @ DELETES THE MOST RECENT" 45210 PRINT" TCHARACTER IN BUFFER." 45220 GOTO49000 45300 PRINT" #DELW ■ DELETES THE MOST RECENT WORD" 45310 PRINT"TIN THE BUFFER, NOT INCLUDING SPACES." 45320 G0T049000 45400 PRINT"NOT DEFINED - CODE AT 4900" 45410 G0T049000 45500 PRINT"NOT DEFINED - CODE AT 5100" 45510 GOTO49000 45600 PRINT"# NL 型 NEWLINE ON PRINTER, DOES NOT" 45610 PRINT" TAFFECT BUFFER. 45620 GOTO49000 45700 PRINT" # UC ■ FOLLOWING LETTERS WILL BE" 45710 PRINT" TUPPER CASE - ALSO SEE / LC 45720 G0T049000 45800 PRINT" SPARA E END OF PARAGRAPH, PRINTS" 45810 PRINT" DBUFFER, THREE NEW LINES, AND SETS" 45820 PRINT"BUFFER TO THREE SPACES." 45830 G0T049000 45900 PRINT SSEND PRINTS CONTENTS OF BUFFER." 45910 PRINT" TERASES BUFFER AND TAKES A NEWLINE." 45920 G0T049000 46000 PRINT" NESC ■ SENDS BUFFER TO 2ND. MICRO," 46010 PRINT" TAND ESCAPE CHARACTER." 46020 GOTO49000 46100 PRINT"3+VAR ■ SAVES THE CONTENTS OF THE BUFFER" 46110 PRINT"7IN THE NEXT FREE VARIABLE LOCATION." 46120 PRINT"SHOWS FIRST FOUR CHARACTERS." 46140 GOTO49000 46200 PRINT"의 1C · B SENDS CONTROL C (ASCII(3)) TO" ``` ``` 46210 PRINT"TTHE 2ND. MICRO, ACTS AS A BREAK IN." 46220 GOTO49000 46300 PRINT"NOT DEFINED - CODE AT 6700" 46310 GOTO49000 46400 PRINT"MTEXT ■ GOTO TEXT MODE, DISPLAYS ASCII" 46410 PRINT"DCHARACTER SET AND A SELECTION OF" 46420 PRINT"FREQUENTLY USED WORDS." 46430 GOTO49000 46500 PRINT"%PROG @ PROGRAMMING MODE, DISPLAYS CHAR-" 46510 PRINT" CACTERS AND WORDS USED IN BASIC. SETS" 46520 PRINT"UC, VARIABLES MAY BE DEFINED IN TEXT." 46540 GOTO49000 46600 PRINT" SUSER ■ USE TO CONTROL EXTERNAL" 46610 PRINT" TEQUIPMENT (T.V., RADIO ETC.) AND" 46620 PRINT"ALTER CURSOR SCAN SPEED." 46630 GOTO49000 46700 PRINT" MHELP ■ A HELP SYSTEM. USE AT FIRST FOR" 46710 PRINT" DENERAL INFORMATION, THEN TO CHECK" 46720 PRINT" CONTENTS OF VARIABLES - HELP/VARIABLE." 46730 GOT049000 46800 PRINT" S-VAR B REMOVES SELECTED VARIABLE AND" 46810 PRINT" TSHIFTS REMAINING ONES TO FILL SPACE." 46820 GOTO49000 46900 PRINT"NOT DEFINED - CODE AT 7900" 46910 GOTO49000 47000 PRINT" NDEL ■ CLEARS BUFFER - NO OTHER EFFECT." 49000 PRINT SAMMANDVICE - ON/OFF TO RETURN и: 49010 IF(PEEK(UP)AND UX)=0 GOTO 49010 49020 IF (PEEK (UP) AND UX) GOT049020 49030 GOSUB 20900 49040 PRINT" SRINGWADVICE - O.K. 49050 HP=0 49060 GOTO 3000 ``` Program 10. ## 5.9 Voice synthesis Adding a voice synthesiser to the CBM 64 is both simple and cheap, and probably one of the easiest ways is to use the General Instrument SP0256 speech processor chip. This IC is connected directly to the user port and its output is simply fed via an amplifier to the audio input line on the SID chip and thence to the monitor or TV speaker. This circuit is shown in Fig. 5.6. The SPØ256 is an allophone speech generator which can be used to synthesise any English word by concatenating the individual speech sounds (phonemes) which comprise the word. Within this chip is a table of 64 different allophones and pauses; a full list of these is given in Table 5.1. These are accessed via the chip's six address lines. By having the user port connected directly to these six address lines we can generate any required allophone simply by outputting the correct address to these six lines. Normal speech contains between ten and twelve allophones per second, and consequently allophone synthesis is a very compact way of storing speech. The major advantage of allophone synthesis is that it can provide an unlimited vocabulary with fairly low storage requirements. When using this circuit to generate speech it must be realised that the allophones do not necessarily correspond directly to the written letters and therefore a word must first be converted to its phonetic form. Thus because of Table 5.1. Allophone address (Reproduced by courtesy of General Instruments). | Octal | | Sample | | Octal | | Sample | | |---------|-----------|----------|----------|---------|-----------|----------|----------| | Address | Allophone | • | Duration | Address | Allophone | • | Duration | | 000 | PA1 | PAUSE | 10ms | 040 | /AW/ | Out | 250ms | | 001 | PA2 | PAUSE | 30ms | 041 | /DD2/ | Do | 80ms | | 002 | PA3 | PAUSE | 50ms | 042 | /GG3 | Wig | 120ms | | 003 | PA4 | PAUSE | 100ms | 043 | /VV/ - | Vest | 130ms | | 004 | PA5 | PAUSE | 200ms | 044 | /GG1/ | Guest | 80ms | | 005 | /OY/ | Boy | 290ms | 045 | /SH/ | Ship | 120ms | | 006 | /AY/ | Sky | 170ms | 046 | /ZH/ | Azure | 130ms | | 007 | /EH/ | End | 50ms | 047 | /RR2/ | Brain | 80ms | | 010 | /KK3/ | Comb | 80ms | 050 | /FF/ | Food | 110ms | | 011 | /PP/ | Pow | 150ms | 051 | /KK2/ | Sky | 140ms | | 012 | /JH/ | Dodge | 100ms | 052 | /KK1/ | Can't | 120ms | | 013 | /NN1/ | Thin | 170ms | 053 | /ZZ/ | Zoo | 150ms | | 014 | /IH/ | Sit | 50ms | 054 | /NG | Anchor | 200ms | | 015 | /TT2/ | То | 100ms | 055 | /LL/ | Lake | 80ms | | 016 | /RR1/ | Rural | 130ms | 056 | /WW/ | Wool | 140ms | | 017 | /AX/ | Succeed | 50ms | 057 | /XR/ | Repair | 250ms | | 020 | /MM/ | Milk | 180ms | 060 | /WH/ | Whig | 150ms | | 021 | /TT1/ | Part | 80ms | 061 | /YY1/ | Yes | 90ms | | 022 | /DH1/ | They | 140ms | 062 | /CH/ | Church | 150ms | | 023 | /IY/ | See | 170ms | 063 | /ER1/ | Fir | 110ms | | 024 | /EY/ | Beige | 200ms | 064 | /ER2/ | Fir | 210ms | | 025 | /DD1/ | Could | 50ms | 065 | /OW/ | Beau | 170ms | | 026 | /UW1/ | То | 60ms | 066 | /DH2/ | They | 180ms | | 027 | /AO/ | Aught | 70ms | 067 | /SS/ | Vest | 60ms | | 030 | /AA/ | Hot | 60ms | 070 | /NN2/ | No | 140ms | | 031 | /YY2/ | Yes | 130ms | 071 | /HH2/ | Hoe | 130ms | | 032 | /AE/ | Hat | 80ms | 072 | /OR/ | Store | 240ms | | 033 | /HH1/ | He | 90ms | 073 | /AR/ | Alarm | 200ms | | 034 | /BB1/ | Business | 40ms | 074 | /YR/ | Clear | 250ms | | 035 | /TH/ | Thin | 130ms | 075 | /GG2/ | Got | 80ms | | 036 | /UH/ | Book | 70ms | 076 | /EL/ | Saddle | 140ms | | 037 | /UW2/ | Food | 170ms | 077 | /BB2/ | Business | 60ms | irregularities in spelling it is necessary to use the sounds of the word rather than the letters when dealing with speech allophones. A second problem is the segmentation of speech. This means that although we think of a spoken word as consisting of a sequence of separate sounds which correspond to a letter name, in fact speech sound is a continuously varying signal which cannot easily be broken into discrete units. This accounts for the occasional problems of intelligibility in allophone synthesised speech. A third problem is that the ear will perceive the same acoustic signal differently depending on the sounds which precede or follow it. Thus the initial p in 'pop' is different from the p in 'spy'. An attempt is made to overcome some of these problems by the design of the allophone sounds and by careful selection of allophones to describe a particular word. The individual sounds of language are called phonemes. In each language these are slightly different. The SPØ256 is designed to give English phonemes. The phonemes can be divided into three different catagories; consonants, vowels, and speech sounds such as aspirants and pauses. Tables 5.2 to 5.6 show the consonant and vowel phonemes and how allophones can be used. Program 11 shows how to use the voice synthesis circuit and allophones from Basic. ### 152 The Commodore 64 Kernal and Hardware Revealed ``` 1 REM 埃米米米米米米米米米米米米米米米米米米米米米米米米米米米米 2 REM * SPEECH DATA TO USER PORT * 4 REM 5 POKE54272+24,15 10 DD=56576 20 POKEDD+3,127 35 READD: IFD=-1THENEND 40 POKEDD+1,D OR64 50 POKEDD+1,D AND63 51 IFPEEK(DD+1)<128THEN51 60 GOTO 35 70 DATA42,59,45,3 80 DATA 3,39,12,50,59,21,3 85 DATA 3,26,11,33,3 90 DATA 3,11,12,41,3 91 DATA 3,46,52,41,3 93 DATA 40,58,3 100 DATA 43,12,40,39,26,3 110 DATA 12,11,40,58,16,20,37,15,11,3 120 DATA 13,7,42,56,53,45,53,10,19,3,-1 ``` ### Program 11. *Table 5.2.* Examples of spelling irregularities. (Reproduced by courtesy of General Instruments). | | Same sound represented by different letters | Different sounds represented by the same letter(s) | |------------|---------------------------------------------|----------------------------------------------------| | Vowels | meat | vein | | | feet | foreign | | | Pete | deism | | | p <b>eo</b> ple | deicer | | | penn <b>y</b> | g <b>ei</b> sha | | Consonants | <b>sh</b> ip | althou <b>gh</b> | | | tension | <b>gh</b> astly | | | precious | cou <b>gh</b> | | | nation | | Table 5.3. Consonant phonemes of English (Reproduced by courtesy of General Instruments). | | | Labial <sup>1</sup> | Labio-<br>Dental <sup>2</sup> | Inter-<br>Dental <sup>3</sup> | Alveo-<br>lar <sup>4</sup> | Palatal <sup>5</sup> | Velar <sup>6</sup> | Glottal <sup>7</sup> | |-------------|---------------------|---------------------|-------------------------------|-------------------------------|----------------------------|----------------------|--------------------|----------------------| | Stops: | Voiceless<br>Voiced | PP `<br>BB | | | TT<br>DD | | KK<br>GG | | | Fricatives: | Voiceless<br>Voiced | WH | FF<br>VV | TH<br>DH | SS<br>ZZ | SH<br>ZH* | | нн | | Affricates: | Voiceless<br>Voiced | | | | | CH<br>JH | | | | Nasals: | Voiced | MM | | | NN | | NG* | | | Resonants: | Voiced | ww | | | RR, LL | YY | | | <sup>\*</sup>These do not occur in word-initial position in English. <sup>1.</sup> Upper and Lower Lips Touch or Approximate <sup>2.</sup> Upper Teeth and Lower Lip Touch <sup>3.</sup> Tongue Between Teeth <sup>4.</sup> Tip of Tongue Touches or Approximates Alveolar Ridge (just behind upper teeth) <sup>5.</sup> Body of Tongue Approximates Palate (roof of mouth) <sup>6.</sup> Body of Tongue Touches Velum (posterior portion of roof of mouth) <sup>7.</sup> Glottis (opening between vocal cords) Table 5.4. Vowel phonemes of English (Reproduced by courtesy of General Instruments). | | FRONT | CENTRAL | BACK | |------|-----------------------------|------------------------|--------------------------| | HIGH | YR<br>IY<br>IH* | | UW#<br>UH <sup>*</sup> # | | MID | EY<br>EH <sup>*</sup><br>XR | ER<br>AX* | OW#<br>OY# | | LOW | AE <sup>*</sup> | AW#<br>AY<br>AR<br>AA* | AO*#<br>OR# | <sup>\*</sup> SHORT VOWELS # ROUNDED VOWELS *Table 5.5.* Examples of words made from allophones (Reproduced by courtesy of General Instruments). | DD2-AO-TT2-ER1 | "daughter" | |------------------------------|------------| | KK3-AX-LL-AY-DD1 | "collide" | | SS-SS-IH-SS-TT2-ER1 | "sister" | | KK1-LL-AW-NN1 | "clown" | | KK3-UH-KK1-IY | "cookie" | | LL-EH-TT2-ER | "letter" | | LL-IH-TT2-EL | "little" | | AX-NG-KK3-EL | "uncle" | | KK1-AX-MM-PP1-YY1-UW1-TT2-ER | "computer" | | EH-KK1-SS-TT2-EH-EH-NN1-TT2 | "extent" | | TT2-UW2 | "two" | | AX-LL-AR-MM | "alarm" | | SS-KK3-OR | "score" | | FF-ER2 | "fir" | *Table 5.6.* Guidelines for using the allophones (Reproduced by courtesy of General Instruments). | Silence | | |--------------|----------------------------------------------| | PA1 (10ms) | before BB, DD, GG, and JH | | PA2 (30ms) | before BB, DD, GG, and JH | | PA3 (50ms) | before PP, TT, KK, and CH, and between words | | PA4 (100ms) | between clauses and sentences | | PA5 (200ms) | between clauses and sentences | | Short Vowels | | | */IH/ | sitting, stranded | | */EH/ | extent, gentlemen | | */AE/ | extract, acting | | */UH/ | cookie, full | | */AO/ | talking, song | | */AX/ | lapel, instruct | | */AA/ | pottery, cotton | | Long Vowels | | | /IY/ | treat, people, penny | | /EY/ | great, statement, tray | | /AY/ | kite, sk <b>y</b> , mighty | | /OY/ | noise, toy, voice | | /UW1/ | after clusters with YY: computer | | /UW2 | in monosyllabic words: two, food | | /OW/ | zone, close, snow | | /AW/ | sound, mouse, down | # 154 The Commodore 64 Kernal and Hardware Revealed Table 5.6. Continued. | R-Colored Vo | | |-------------------|------------------------------------------------------------------------| | /ER1/ | | | | letter, furniture, interrupt | | /ER2/ | monosyllables: bird, fern, burn | | /OR/ | fortune, adorn, store | | /AR/ | farm, alarm, garment | | /YR/ | hear, earring, irresponsible | | /XR/ | hair, declare, stare | | Resonants | | | /WW/ | we, warrant, linguist | | /RR1/ | initial position: read, write, x-ray | | /RR2/ | initial clusters: brown, crane, grease | | /LL/ | like, hello, steel | | /EL/ | little, angle, gentlemen | | /YY1/ | clusters: cute, beauty, computer | | /YY2/ | initial position: yes, yarn, yo-yo | | Voiced Fricativ | | | /VV/ | vest, prove, even | | /DH1/ | word-initial position: this, then, they | | /DH2/ | word-final and between vowels: bathe, bathing | | /ZZ/ | zoo, phase | | /ZH/ | beige, pleasure | | Voiceless Frica | | | */FF/ | ) There was be do to be a significant and | | */TH/ | These may be doubled for initial position | | */SS/ | and used singly in final position | | /SH/ | shirt, leash, nation | | /HH1/ | before front vowels: YR, IY, IH, EY, EH, XR, AE | | /HH2/ | before back vowels: UW, UH, OW, OY, AO, OR, AR | | /WH/ | white, whim, twenty | | Voiced Stops | | | /BB1/ | final position: rib; between vowels: fibber; in clusters; bleed, brown | | /BB2/ | initial position before a vowel: beast | | /DD1/ | final position: played, end | | /DD2/ | initial position: down; clusters: drain | | /GG1/ | before high front vowels: YR, IY, IH, EY, EH, XR | | /GG2/ | before high back vowels: UW, UH, OW, OY, AX; and clusters: green, glue | | /GG3/ | before low vowels: AE, AW, AY, AR, AA, AO, OR, ER; in medial clusters: | | 7 4 4 4 5 7 | anger; and final position: peg | | Voiceless Stor | | | /PP/ | pleasure, ample, trip | | /TT1/ | final clusters before SS: tests, its | | /TT2/ | all other positions: test, street | | / 1 1 2/<br>/KK1/ | | | / ININ 1/ | before front vowels: YR, IY, IH, EY, EH, XR, AY, AE, ER, AX; | | /KK2/ | initial clusters: cute, clown, scream | | /KK2/ | final position: speak; final clusters: task | | /KK3/ | before back vowels: UW, UH, OW, OY, OR, AR, AO; | | 0.66-1 | initial clusters: crane, quick, clown, scream | | Affricates | ahurah faatura | | /CH/ | church, feature | | /JH/ | judge, injure | | Nasal | milk plarm ample | | /MM/<br>/NN1/ | milk, alarm, ample | | /NN1/ | before front and central vowels: YR, IY, IH, EY, EH, XR, AE, ER, | | /NINIO/ | AX, AW, AY, UW; final clusters: earn | | /NN2/ | before back vowels: UH, OW, OY, OR, AR, AA | | /NG/ | string, anger | <sup>\*</sup>These allophones can be doubled. ## 5.10 Analog interfacing In digital systems two voltage levels are used to represent data, with the binary digit Ø represented by a low voltage and binary 1 by a high voltage. Digital data is therefore represented by a series of pulses. Analog signals have an infinite range of voltage levels and are therefore continuous rather than having discrete units of information. The difference between the two types of waveform is shown in Fig. 5.7. The digital equivalent of an analog signal is generated by regularly sampling the waveform, and on each sample converting the measured voltage into a binary value using an analog to digital conversion circuit. The sampling of a waveform is shown in Fig. 5.8. For a computer to be able to create a smooth analog waveform it must at regular intervals output a binary value, which is converted to an analog voltage by a digital to analog conversion circuit. The digital synthesis of an analog waveform is shown in Fig. 5.9. Fig. 5.7. Comparison of digital and analog waveforms (Reproduced by courtesy of Practical Electronics). Fig. 5.8. Digital sampling of an analog waveform (Reproduced by courtesy of Practical Electronics). Fig. 5.9. Digital synthesis of an analog waveform (Reproduced by courtesy of *Practical Electronics*). ### 5.11 Digital to analog conversion The circuit used to convert a digital signal to an analog signal is quite simple involving the use of a weighted resistor network. This most commonly takes the form of an R, 2R, 4R, 8R etc. ladder, such as the one shown in Fig. 5.10. In operation each successively lower weighted input produces an output voltage which is exactly half that produced by the preceding input. The voltages from each input are summed at the combined output and a voltage developed which is the analog representation of the binary digital value. Although a digital to analog conversion circuit can be constructed easily from resistors it is simpler and often more accurate to use one of the many D to A converter ICs. An example of such a D to A IC is the ZN425E. This chip is an 8 bit dual mode A/D and D/A converter, incorporating a voltage reference, resistor network, input switches and an 8 bit binary counter. A block diagram and pin out for this IC is shown in Fig. 5.11. The 8 bit counter is used with an external comparator IC to facilitate analog to digital conversion. The logic Fig. 5.10. Digital to analog converter using R-2R register network (Reproduced by courtesy of *Practical Electronics*). ### PIN CONNECTIONS (TOP VIEW) Fig. 5.11. Pin connections and block diagram for the A/D and D/A converter (Reproduced by courtesy of RS Components Limited). select pin determines which function, A/D or D/A, is being used. The on-chip 2.5 volt reference signal ensures that the conversion is very accurate (0.2%) accuracy). This chip is very easy to connect to the processor; the circuit diagram for connecting it is shown in Fig. 5.12. It should be noted that the output from this chip is a voltage ranging between $\emptyset$ volts and 2.5 volts (an increase of 1 in the output to the D to A therefore gives a corresponding increase of approximately 0.01 volts). A buffer amplifier is necessary to change this voltage range, particularly if a positive to negative ranging voltage is desired, and calibrate the output. The ability to generate an analog signal from the computer has many applications, probably the most obvious is in the generation of simple or complex analog waveforms. The technique for outputting a repeating waveform of frequency is quite simple. The fundamental requirement is a waveform table, usually one page (256 bytes) long. This table contains a sequence of precalculated values which define the waveform shape. By repeatedly outputting all the values in the table, in sequence, to the digital to analog converter a regular waveform is generated. The frequency of the waveform can be varied either by varying the delay between outputting each Fig. 5.12. Analog to digital and digital to analog conversion circuit. ``` LOC CODE LINE C291 FUNC.SIMPLE .LIB 0291 ROUTINE TO OUTPUT A FUNCTION C291 C291 TABLE OVER THE USER PORT TO A DIGITAL TO ANALOG CONVERTER. C291 0291 THE FREQUENCY IS CONTROLLED BY EITHER THE TIMER VALUE OR C291 C291 C291 ; INCREASING/DECREASING THE SAMPLING VALUE. C291 C291 FUNCTH JSR $AEFD GET VALUE FOR 20 FD AE 0294 20 8A AD JSR $AD8A ; TIMER A C297 20 F7 B7 JSR $B7F7 LDA #<NMI STA $0318 C29A A9 D1 SET NMI VECTOR 8D 18 03 0290 C29F C2A2 8D FA FF STA #FFFA 89 C2 LDA #>NMI C284 8D 19 03 STA $0319 C287 8D FB FF STA $FFFB C2AA C2AC C2AF C2B1 A5 14 LDA $14 STORE TIMER VALUE STA $DD04 8D 04 DD LDA $15 A5 15 8D 05 DD STA $DD05 C2B4 LDA #$FF A9 FF C2B6 STA $DD03 JUSER PORT TO OUTPUT SD 03 DD C2B9 A9 0F LDA #$0F STA $D400+24 C2BB 8D 18 D4 MAX SID VOLUME C2BE LDA #$00 A9 00 0200 8D 01 C3 STA FLAG POINTER TO TABLE 0203 AD ØD DD LDA $DDOD CLEAR NMIS 0206 A9 81 LDA #$81 ;SET TIMER A NMI 0208 SD ØD DD STA $DD0D C2CB LDA #$11 A9 11 ;START TIMER A C2CD SD ØE DD STA $DD0E czne RTS 60 C2D1 C2D1 C2D2 STORE REGISTERS IMM PHA 48 88 TXA C2D3 PHA 48 C2D4 TYA 98 C2D5 48 PHA C2D6 LDA #$01 CAUSED BY TIMER A? A9 01 C2D8 C2DB C2DD C2DF BIT $DDOD BNE SEND 20 0D DD ; YES DØ 11 A9 7F LDA #$7F CLEAR ENABLED MMIS SD ØD DD STA $DD@D C2E2 20 84 FF JSR $FF84 RESET I/O C2E5 C2E8 20 8A FF FRESET KERNAL JSR $FF8A 68 NMEXIT PLA ; PULL REGISTERS C2E9 C2EA C2EB 88 TAY PLA 68 AA TAX C2EC PLA 68 C2ED RTI 40 C2EE GET POINTER C2EE SEND LDX FLAG AE 01 C3 GET BYTE C2F1 BD 13 C3 LDA TABLE,X SEND TO USER PORT STA $DD01 C2F4 8D 01 DD C2F7 TXA 88 CLC 18 ; ADD SAMPLING VALUE ADC #$@1 STORE C2F9 69 01 C2FB STA FLAG 8D 01 C3 JEXIT NMI JMP NMEXIT C2FE 4C E8 C2 C301 C301 .BYT 0 FLAG 00 ``` ### 160 The Commodore 64 Kernal and Hardware Revealed | LOC | CODE | LINE | | |----------------------------------------------------------------------|----------------------------------------------------------------------|---------------|---------------------------------------------------------------------------------------| | C302<br>C302<br>C305<br>C308<br>C30A<br>C30D<br>C30F<br>C312<br>C313 | 20 3A FF<br>20 84 FF<br>A9 FE<br>8D FB FF<br>A9 43<br>8D FA FF<br>60 | ;<br>DISAB | JSR \$FF8A<br>JSR \$FF84<br>LDA #\$FE<br>STA \$FFFB<br>LDA #\$43<br>STA \$FFFA<br>RTS | | C313<br>C413 | | TABLE<br>.END | *= <b>*</b> +256 | ### Symbol table | | LUE<br>CØA3 | ATOD | CØ9C | CLRMEM | C203 | CLRSCN | C26A | |------------------|--------------|---------------|--------------|-----------------|--------------|-----------------|--------------| | ALOOP1<br>DISAB | C302 | DISPLY | C999 | DIAS | C125 | DONE | C1F0 | | DOT | CØD1 | ERROR | C013<br>C1E1 | EXIT<br>GXY | C0CD<br>C198 | FLAG<br>KERIN | C301<br>C1D1 | | FUNCTN<br>KEROUT | C291<br>C1C9 | GRAPH<br>LOOP | C206 | LOOP1 | C26E | LOOP2 | C0C3 | | MUL320 | C147 | MUL8 | C12E | NMEXIT | C2E8 | NMI | C2D1<br>C181 | | NORM<br>READ10 | C27E<br>C060 | OK<br>RLOOP1 | C018<br>C06E | PLOT1<br>RLOOP2 | C18A<br>C070 | PLOTIT<br>SEND | C2EE | | SETUP | 0092 | STORE | C2F9 | TABLE | C313 | TOP2X | C179 | | TX | C1C7 | XER | COEI | XOK<br>ZLOOP2 | C0E3<br>C042 | YND70K<br>ZPLOT | C123<br>CØ4F | | ZLOOP | 0022 | ZLOOP1 | CØ3A | 2LUUP 2 | UU42 | ا تا ہے | - N | Program 12. value or for higher frequencies by sampling the table rather than using every value, with the frequency being determined by the sampling step. The machine code routine in Program 12 can be used to output a waveform using a predefined waveform table over a range of different frequencies, the frequency being controlled by the value stored in Timer A. This routine is designed such that the analog output is connected to the sound input of the SID chip so that the sound generated by the waveform can be heard. The Basic program in Program 13 is used in conjunction with Program 12 to create a waveform table, the waveform being created either from a mathematical function or drawn directly using either a joystick or the cursor keys. The resulting waveform is displayed on the screen using high resolution graphics. It should be noted that in order to obtain the graphics, the graphics routines in Program 14 should be present. These are integrated with the frequency generation routine so that they can both reside in memory together. ``` 10 PRINT"DUSE:" 20 PRINT"N 1 - JOYSTICK" 30 PRINT"N 2 - KEYBOARD" 40 PRINT"N 3 - FUNCTION" 50 GETA$::IFA$=""THEN50 60 A=VAL(A$>:IFA<50RA>3THEN50 90 T=49939:SYS49627 100 DIMT(255):FORI=0T0255:T(I)=127:POKET+I,127:SYS49537,I,T(I)-27:NEXT 110 ONAGOSUB1000,2000,3000 120 GETA$:IFA$=""THEN120 ``` ``` 130 SYS49790 140 END 1000 REM 1010 REM INPUT WAVEFORM USING JOYSTICK 1020 REM 1030 X=0:Y=T(X)-27 1040 A=PEEK(56320) E=0:W=0:S=0:N=0 1050 IF (AAND16)=0 THEN RETURN 1060 IF (AAND8)=0 THEN E=1:GOTO1110 1070 IF (AAND4)=0 THEN W=-1:GOTO1110 1080 IF (AAND2)=0 THEN S=-1:GOTO1110 1090 IF (AAND1)=0 THEN N=1:GOTO1110 1100 GOTO 1040 1110 GOSUB 11100:GOTO1040 2000 REM 2010 REM INPUT WAYEFORM USING KEYBOARD 2020 REM 2030 X=0:Y=T(X)-27 2040 GETA$:E=0:W=0:S=0:N=0 2050 IF A$=CHR$(13) THEN RETURN 2060 IF A$="N" THEN E=1:GOTO 2110 2070 IF A$="0" THEN W=-1:GOTO 2110 2080 IF A$="%" THEN S=-1:GOTO 2110 2090 IF A$="∏" THEN N=1:GOTO 2110 2100 GOTO 2040 2110 GOSUB 11100:GOTO2040 3000 REM 3010 REM PUT UP A WAVEFORM DEFINED IN 3020 REM FNF 3030 REM 3040 DEF FNF(Z)=128+(31-.122*Z)*SIN((Z*#/128)12) 3050 FORI=0T0255 3060 V=FNF(I):SYS49537,I,T(I)-27 3070 POKET+1,V:T(I)=V:SYS49537,I,T(I)-27 3080 NEXTI:RETURN 11100 IF E OR W THEN 11120 11110 IF(Y+S+N) COTHENRETURN 11111 IF (Y+S+N)>199 THEN RETURN 11112 SYS49537,X,Y:Y=Y+S+N 11115 T(X)=(Y+27):POKET+X,(Y+27):SYS49537,X,Y:RETURN 11120 X=(X+E+W)AND255:Y1=T(X)-27:SYS49537,X,Y1:SYS49537,X,Y 11130 T(X)=Y+27:POKET+X,Y+27:RETURN ``` #### Program 13. | LOC | CODE | LINE | J | | |-------------------------------------------------------------|----------------------------------------------------------------------------------------|----------------------------|--------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------| | 0000<br>0000<br>0000<br>0000<br>0006<br>0006<br>0006<br>000 | 20 FD AE<br>20 8A AD<br>20 87 B7<br>A5 15<br>C9 78<br>90 04<br>C9 A0<br>90 05<br>A9 01 | *=\$C006<br>.LIB<br>DISPLY | DIS.SIMPLE JSR \$AEFD JSR \$AEFD JSR \$B7F7 LDA \$15 CMP #\$78 BCC ERROR CMP #\$40 BCC OK LDA #\$01 STA \$02 | GET MEMORY ADDRESS; OF DISPLAY LESS THAN BOTTOM? YES IN RANGE? YES FLAG ERROR AND EXIT | | C017<br>C018<br>C01A | 60<br>A9 00<br>85 02 | 0K | RTS<br>LDA #\$00<br>STA \$02 | ;FLAG O.K. | | C01C<br>C01E<br>C020<br>C022 | A9 00<br>85 FD<br>85 FE<br>A0 00 | ZL00P | LDA #\$00<br>STA \$FD<br>STA \$FE<br>LDY #\$00 | ;SET X COUNTER | | 0024<br>0026<br>0029<br>0028<br>0020 | B1 14<br>20 4F C0<br>A5 14<br>18<br>69 01 | | LDA (\$14),Y<br>JSR ZPLOT<br>LDA \$14<br>CLC<br>ADC #\$01 | GET A VALUE PLOT IT INCREASE TABLE POINTER BY SAMPLING RATE | | LOC | CODE. | LINE | | |-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------| | C044<br>C046<br>C048<br>C04A<br>C04C<br>C04E | A5 FE<br>C9 01<br>D0 DA | STA \$14<br>LDA \$15<br>ADC #\$00<br>CMP #\$A0<br>BNE ZLOOP1<br>LDA #\$78<br>ZLOOP1 STA \$15<br>INC \$FD<br>BNE ZLOOP2<br>INC \$FE<br>ZLOOP2 LDA \$FE<br>CMP #\$01<br>BNE ZLOOP<br>LDA \$FD<br>CMP #\$40<br>BNE ZLOOP<br>RTS | ;INCREASE X | | 094F<br>094F<br>0951<br>0953<br>0955<br>0957<br>095B<br>095D<br>0969 | 85 5B<br>A9 00<br>85 5C<br>A5 FD<br>85 59<br>A5 FE<br>85 5A<br>4C 84 C1 | | STORE THE Y COORDINATE STORE THE X COORDINATE PLOT THE POINT | | C0600<br>C0601<br>C0663<br>C0666<br>C0666<br>C0666<br>C0666<br>C0677<br>C0777<br>C0777<br>C0887<br>C0887<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888<br>C0888 | 85 FD<br>A9 78<br>85 FE | .END .LIB READ.SIMPLE READ10 SEI LDA #\$08 STA \$D011 LDA #\$00 STA \$FD LDA #\$78 STA \$FE RLOOP1 LDA #\$01 RLOOP2 JSR SETUP DEX BNE RLOOP2 LDY #\$01 STA \$FD LDA #\$01 CLC ADC \$FD STA \$FD STA \$FD LDA #\$01 CLC ADC \$FD STA \$FE CMP #\$40 \$50011 CLI | :IGNORE SAMPLE | | C091<br>C092<br>C092<br>C094<br>C097<br>C099<br>C099<br>C096<br>C0A1<br>C0A8<br>C0AB<br>C0AB | A9. FF<br>8D 03 DD<br>A9 FB<br>8D 02 DD<br>A9 80<br>BD 01 DD<br>85 61<br>AD 01 DD<br>05 61<br>8D 01 DD<br>EA<br>EA | RTS; ; SETUP LDA #\$FF STA \$DD03 LDA #\$FB STA \$DD02 ATOD LDA #\$80 STA \$DD01 STA \$61 AL00P1 LDA \$DD01 ORA \$61 STA \$DD01 NOP NOP LDA \$DD00 | ;SET USER PORT ; TO OUTPUT ;SET ONE INPUT LINE ; (PA2) ;SET A TO D ;START VALUE ;START MAIN LOOP ;INPUT FROM | ``` LOC CODE LINE 6A ROR A CORO ; COMPARATOR INTO CØB1 6A ROR A ; CARRY 00B2 ROR A 6A ; VALUE TOO SMALL ; VALUE TOO LARGE ; TRY HALF VALUE COBS B0 0E BCS LOOP2 COR5 A5 61 LDA $61 EOR $DD01 CØB7 4D 01 DD CØBA SD 01 DD STA $DD01 COBD 46 61 LSR $61 DECREASE SEARCH STEP COBE F0 0C BEQ EXIT COMPLETE 0901 10 E0 BPL ALCOPI JTRY AGAIN 0003 46 61 LOOP2 LSR $61 DECREASE SEARCH STEP EΑ NOP JADJUST TIMING 0006 EΑ NOP 0907 EΑ NOP cacs EA NOP 0009 A5 61 LDA $61 COCB DØ D6 BNE ALOOP1 JTRY AGAIN AD 01 DD COCD EXIT LDA $DD01 : VALUE RETURNED IN RTS CODO 60 COD1 . END .LIB CØD1 DOT. SIMPLE CØD1 ROUTINE TO CALCULATE LOCATION AND BIT(S) FROM THE X AND Y COD1 CØD1 COORDINATES. CØD1 CØD1 CØD1 A5 5A DOT LDA $5A ; CHECK THAT X AND Y CMP #$00 CØD3 09 00 ARE WITHIN BOUNDS 00D5 F0 0C BEQ XOK CODZ C9 01 CMP #$91 DØ 06 BNE XER COD9 A5 59 C9 40 CODE LDA $59 CODD CMP #$40 90 02 BCC XOK CODE 38 XER C0E1 SEC ; TOO LARGE EXIT CØE2 60 RTS CØEG A5 5C XOK LDA $50 CØE5 DØ FA BNE XER C0E7 A5 5B LDA $5B C0E9 09 08 CMP #$C8 CØEB BØ F4 BCS XER A9 C7 LDA #199 COED COEF 38 SEC CØFØ E5 SBC $5B 5B 35 CØF2 5B STA $5B A5 59 LDA $59 CALCULATE THE BIT TO CØF4 CØF6 AND $$07 STA $5E ; BE PLOTTED AS 29 07 85 CØF8 5E ; 7-(X AND Y) CØFA A9 07 LDA #$97 COFC 38 SEC SBC $5E CØFD E5 5E COFF TAX ĤĤ CALCULATE 21$5E C100 C103 79 C1 BD LDA TOP2X/X 85 5E STA $5E C105 C105 CALCULATE INT(Y/8)*320 C105 AND STORE IN $57 C105 C105 A5 5B LDA $5B C107 LSR A 48 LSR A 0108 4A C109 4Ĥ LSR A C10A C10B 9A ASL A AA TAX C10C BD 47 C1 LDA MUL320,X C10F 85 57 STA $57 48 C1 LDA MUL320+1,X C111 ``` | 164 | The Commod | dore 64 Kernal and Hardware Revealed | |------------------------------------------------------|-------------------------------------------------|------------------------------------------------------------------| | LOC | CODE | LINE | | C114<br>C116<br>C116 | 85 58 | STA \$58<br>;ADD Y AND 7 TO \$57 | | C116<br>C116<br>C118<br>C118 | A5 5B<br>29 07<br>18 | ;<br>LDA \$5B<br>AND #\$07<br>CLC | | C11B<br>C11D<br>C11F<br>C121 | 65 57<br>85 57<br>90 02<br>E6 58 | ADC \$57<br>STA \$57<br>BCC YND7OK<br>INC \$58 | | C123<br>C123<br>C123 | | CALCULATE INT(X/8) | | C123<br>C125<br>C127<br>C129<br>C12A<br>C12C | A0 03<br>46 5A<br>66 59<br>88<br>D0 F9 | YND7OK LDY #\$03<br>DIV8 LSR \$5A<br>ROR \$59<br>DEY<br>BNE DIV8 | | C12C<br>C12C | | CALCULATE INT(X/8)*8 | | C12C<br>C12E<br>C130<br>C132<br>C133<br>C135 | A0 03<br>06 59<br>26 5A<br>88<br>D0 F9 | LDY #\$03 MUL8 ASL \$59 ROL \$5A DEY BNE MUL8 ; | | C135<br>C135 | 05.53 | ;ADD INT(X/8)*8 INTO \$57 | | C135<br>C137<br>C138<br>C13A<br>C13C<br>C13E<br>C140 | A5 57<br>18<br>65 59<br>85 57<br>A5 58<br>65 5A | LDA \$57<br>CLC<br>ADC \$59<br>STA \$57<br>LDA \$58<br>ADC \$5A | | C140<br>C140 | | ;ADD \$E000 INTO \$57<br>; | | C140<br>C141<br>C143<br>C145<br>C146 | 18<br>69 EØ<br>85 58<br>18<br>60 | CLC<br>ADC #\$E0<br>STA \$58<br>CLC<br>RTS | | C147<br>C149<br>C14B<br>C14D | 00 00<br>40 01<br>80 02<br>C0 03 | MUL320 .WOR 0,320,640,960,1280 | | C14F<br>C151<br>C153<br>C155<br>C157 | 00 05<br>40 06<br>80 07<br>C0 08<br>00 0A | .WOR 1600,1920,2240,2560,2880 | | C159<br>C15B<br>C15D<br>C15F<br>C161 | 40 0B<br>80 0C<br>C0 0D<br>00 0F<br>40 10 | .WOR 3200,3520,3840,4160,4480 | | C163<br>C165<br>C167<br>C169<br>C16B | 80 11<br>C0 12<br>00 14<br>40 15<br>80 16 | .WOR 4800,5120,5440,5760,6080 | | C16D<br>C16F<br>C171<br>C173 | C0 17<br>00 19<br>40 1A<br>30 1B | .WOR 6400,6720,7040,7360,7680 | ``` LOC CODE LINE C175 CØ 1C C177 00 1E C179 91 TOP2X .BYT 1,2,4,8,16,32,64,128 C17A 02 C17B C17C 04 98 C17D 10 C17E 20 C17F 40 C180 80 .END C181 C181 .LIB PLOT. SIMPLE C181 C181 ; ROUTINE TO PLOT A POINT C181 C181 20 98 C1 PLOTIT JSR GXY ; GET X AND Y C184 20 D1 C0 JSR DOT 90 01 BCC PLOTI C187 0189 RTS 60 C18A 20 09 01 PLOT1 JSR KEROUT JDISABLE IRQ CISD A0 00 LDY #$90 C18F B1 57 LDA ($57), Y ; OTHERWISE PLOT POINT C191 C193 EOR $5E 45 5E 57 91 STA ($57),Y C195 4C D1 C1 JMP KERIN C198 C198 GET X AND Y VALUE 0198 ; INTO $59 AND $5B C198 C198 GXY 20 FD AE JSR $AEFD C19B 20 SA AD JSR $ADSA JGET X C19E 20 BF B1 JSR $B1BF FIX IT CIAI A6 65 LDX $65 CIAG A4 64 LDY $64 STX TX STY TX+1 JSR $AEFD 8E C7 C1 8C C8 C1 C1A5 C1A8 CHECK 1/1 CIAB 20 FD AE CIRE 20 8A AD JSR $AD8A JGET Y C1B1 20 BF B1 JSR $B1BF FIX IT C1B4 A6 65 LDX $65 LDY $64 STX $5B C1B6 A4 64 86 5B C1B8 CIBA 84 50 STY $50 AD C7 C1 C1BC LDA TX CIBF 85 59 STA $59 C1C1 AD C8 C1 LDA TX+1 C1C4 85 5A STA $5A 0106 60 RTS 0107 00 00 TX . WOR 0 C1C9 C1C9 C1C9 C1C9 ; DISABLE KERNAL AND IRQ 78 KEROUT SEI CICA A5 01 LDA $01 0100 29 FD AND #$FD SWITCH OUT 85 01 CICE STA $01 C1D0 RTS 60 C1D1 C1D1 ; ENABLE KERNAL AND IRQ C1D1 48 KERIN PHA C1D1 C1D2 A5 01 LDA $01 09 02 C1D4 ORA #$02 ;SWITCH IN CIDS 85 01 STA $01 C1D8 58 CLI C1D9 PLA 68 CIDA 60 RTS ``` C1DB .END ``` LOC CODE LINE .LIB CIDE MODE. SIMPLE CIDB CIDB ROUTINE TO SET UP HIRES SCREEN CIDB CIDB 20 03 02 JSR CLRMEM ; CLEAR HIRES SCREEN CIDE 20 6A C2 JSR CLRSCN ; CLEAR VIDEO SCREEN C1E1 GRAPH COMMAND ENTRY C1E1 C1E1 C1E1 C1E3 A9 3B 8D 11 D0 LDA #$3B GRAPH STA $D011 ; SELECT BIT MAP MODE C1E6 A9 3D LDA #$3D C1E8 3D 18 STA $D018 CHOOSE HIRES SCREEN CIEB A9 C8 LDA #$C8 ; ELSE SET HIRES MODE ; SELECT BANK 2 FOR HIRES 8D 16 CIED DØ STA $D016 C1FØ AD 02 DD DONE LDA $DD02 SCREEN C1F3 09 03 ORA #$03 C1F5 3D 02 DD STA $DD02 C1F3 AD 00 DD LDA $DD00 C1FB 29 FC AND #$FC CIFD 99 99 ORA #$00 CIFF SD 00 DD STA $DD00 C202 ŘTS 60 0203 C203 C203 C203 C205 CLG COMMAND ENTRY A0 00 CLRMEM LDY #$90 ; LOOP TO CLEAR HIRES 92 TYA C206 99 00 E0 LOOP STA $E000, Y 0209 99 00 E1 STA $E100, Y C20C C20F C212 C215 99 00 E2 STA $E200,Y 99 00 E3 STA $E300,Y 99 00 E4 STA $E400, Y 99 00 E5 STA $E500,Y C218 99 00 E6 STA $E600.Y C21B 99 00 E7 STA $E700,Y C21E C221 99 00 E8 STA $E800,Y STA $E900,Y 99 00 E9 C224 C227 C22A 99 00 EA STA $EA00, Y 99 00 EB STA $EB00,Y STA $EC00,Y 99 00 EC C230 99 00 ED STA $ED00,Y 99 00 EE STA $EE00,Y C233 99 00 EF STA $EF00,Y C236 C239 C23C 99 00 F0 STA $F000,Y 99 00 F1 STA #F100, Y 99 00 F2 STA $F200, Y C23F 99 00 F3 STA $F300, Y 99 00 F4 99 00 F5 C242 C245 STA $F400, Y STA $F500, Y C248 C24B 99 00 F6 STA $F600,Y 99 00 F7 STA $F700, Y C24E 99 00 F8 STA $F800,Y C251 C254 99 00 F9 STR $F900,Y 99 00 FA STA $FA00, Y C257 C25A 99 00 FB STR $FB00,Y 99 00 FC STR $FC00,Y C25D 99 00 FD STA $FD00,Y 99 00 FE C260 STA $FE00,Y 0263 99 F9 FE STA $FEF9,Y C266 88 DEY 0267 DØ 9D BNE LOOP C269 60 RTS C26A A0 00 CLRSCH LDY #$00 ; LOOP TO CLEAR 0260 A9 1B LDA #$1B 99 00 CC L00P1 C26E STA $CC00,Y 99 00 CD C271 STA $CD00,Y ``` | LOC | CODE | LINE | |----------------------------------------------------------------------|-------------------------------------------|-------------------------------------------------------------------| | C274<br>C277<br>C278<br>C278<br>C270<br>C27E<br>C27E<br>C27E<br>C27E | 99 00 CE<br>99 00 CF<br>88<br>D0 F1<br>60 | STR \$CE00,Y STR \$CF00,Y DEY BNE LOOP1 RTS .END .LIB NORM.SIMPLE | | C27E<br>C27E | | ; ROUTINE TO RETURN TO NORMAL SCREEN | | C27E | AD 02 DD | NORM LDA \$DD02 | | C281 | 29 FC | AND #\$FC | | 0283<br>0286 | 8D 02 DD<br>89 1B | STA \$DD02 ; BACK TO BANK 0<br>LDA #\$1B | | C288<br>C28B | 8D 11 D0<br>A9 15 | STA \$DØ11 ; BIT MAP MODE OFF | | C28D | SD 18 D0 | STA \$D018 ; NORMAL SCREEN | | 0290 | 60 | RTS | | C291 | | .END | Program 14. Another application for D to A converters involves using two converters, the output of each being connected to the X and Y inputs on an oscilloscope. This configuration can then be used to generate true vector graphics displays; the two D/A converters are switched by using the PA2 line. One of the D/A converters uses only 7 bits with the eighth bit used to control the Z axis or intensity control input on the scope. Alternatively the two D/A converters could be used to control two rotating mirrors for a laser display. An extension of the waveform generation routine is a music generator. Such a routine is given in Program 15. This is a four voice sophisticated music synthesiser. The program uses four waveform tables, one for each voice. These are shown in Program 16, and their respective waveforms in Fig. 5.13. In addition to the waveform tables it also requires a score table; a sample score table is given in Program 17. It is divided into two sections; the main control table and the music table. It is in two sections for several reasons, the principal one being to allow it to be stored more compactly. The score is compacted by having sections of the score which are identical stored only once and then repeatedly called by the main control table. The control loop also allows the tempo and waveforms of each voice to be changed. This system may seem fairly complex but examination of Program 17 will show that it is fairly straightforward. This program is based on an original idea by Hal Chamberlin in his book *Musical Applications of Microprocessors* published by Hayden Books. These are the command codes used in the control and music tables: ### Main control table commands If byte is FF this specifies that the following byte contains a control code If followed by 01 then the next byte contains tempo If followed by $\emptyset 2$ then the next four bytes specify the waveform for each voice, each byte being the msb of the start of the specified waveform table If byte is not FF then each pair of bytes specify the hi, lo address of a pointer into the start of a section of the music table. ``` LOC CODE LINE JOUTPUT PORT 0000 USRPRT =$DD01 DATA DIRECTION FOUR VOICE WAVEFORM 0000 =$DD03 DDR 0000 V1PT =$40 0000 V2PT =$45 ; POINTERS VSPT 0000 =$48 V4PT 0000 =$4B 0000 INCPT =$4E POINTER TO MUSIC 0000 NOTES =$50 =$52 FOUR VOICE INCREMENT V1IN 0000 ; POINTERS 0000 V2IN =$54 0000 V3IN =$56 V4IN 0000 =$58 DUR =$5A ; DURATION COUNTER 0000 ; INITIAL INCPT 0000 INCA =$5D =$5F JIEMPO VALUE 0000 TEMPO =$0801 0000 4 ØC Ø8 .WOR END 0801 ; NEXT LINE POINTER 0803 9A 99 .WOR 10 ;LINE NUMBER 10 0805 9E .BYT $9E, '02062', 0 ; SYS02062 0806 30 32 080B 99 .WOR 0 ; END OF BASIC 080C 00 00 END 080E ; ENTRY 080E 080E SEI DISABLE IRQ 080E 78 A9 0B DISABLE SCREEN DMA 080F LDA #$0B 0811 8D 11 DØ STA $D011 0814 DISABLE DECIMAL D8 CLD 0815 89 ØF LDA #$0F SET SID VOLUME 0817 8D 18 D4 STR $D418 TO MAX JISET USER PORT 081A 89 FF LDA #$FF TO OUTPUT 8D 03 DD STA DDR 081C 081F A2 00 LDX #$00 B5 00 STORE SAVE OFF ZERO 0821 LDA $0000/X 0823 9D 00 C0 STA $C000,X ; PAGE 0826 E8 INX BNE STORE DØ F8 0827 LDX #$00 SET UP INCA TO 0829 A2 00 082B AØ 52 LDY #$52 ; POINT TO VIIN 082D 36 5E STX INCA+1 STY INCA LDY #$0F 84 5D 082F 0831 A0 0F CONTROL TABLE STARTS STX $FB 0833 86 FB 0835 84 FC STY $FC 86 41 STX V1PT+1 0837 ZERO WAVEFORM 0839 86 46 STX V2PT+1 ; HIGH BYTES 083B 86 49 STX V3PT+1 STX V4PT+1 083D 86 40 86 4F STX INCPT+1 083F 0841 A0 00 LDY #$00 0843 LOOP GET CONTROL CODE B1 FB LDA ($FB),Y 0845 C9 FF /PLAY CONTROL? CMP #$FF 0847 FØ 25 BEQ CNTROL ; YES STORE AS HIGH BYTE 0849 AA TAX 084A 68 INY FOR MUSIC 084B 98 TYA 084C 48 PHR 084D B1 FB LDA ($FB),Y GET LOW BYTE 084F TAY 88 0850 20 98 08 JSR MUSIC FLAY THE PIECE 0853 PLA 68 0854 TAY A8 9855 CS NEXT INY FREPEAT UNTIL MUSIC 0856 ; COMPLETE DØ EB BNE LOOP 0858 A2 00 EXIT LDX #$00 ``` RESTRE LDA \$C000,X COPY BACK TO 985A BD 00 C0 ``` LOC CODE LINE 085D 95 00 STA $0000,X ; ZERO PAGE 985F E8 INX 0860 DØ F8 BNE RESTRE A9 00 0862 LDA #$00 :NO SID VOLUME STA $D418 0864 8D 18 D4 A9 1B 8D 11 DØ 0867 LDA #$1B RESTORE SCREEN 0869 STA $D011 086C 58 CLI ;START IRQ 086D 60 RTS FINISHED 086E CHTROL INY 086E C8 JGET CONTROL NUMBER B1 FB 086F LDA ($FB), Y 30 E5 BMI EXIT ; END OF MUSIC 0871 0873 CMP #$91 C9 01 JIS VALUE 1? 0875 DØ 07 BNE CONTRI ;NO 0877 CS INY 0878 B1 FB 85 5F GET TEMPO LDA ($FB), Y STA TEMPO 087A 0870 DØ D7 BHE NEXT ;TRY AGAIN 087E 09 02 CONTR1 CMP #$02 JIS CONTROL 2? BNE EXIT 0880 DØ D6 ;NO EXIT PROG 0882 INY CS. 0883 B1 FB LDA ($FB),Y GET HI BYTE FOR 0885 85 42 STA VIPT+2 ; THE WAVEFORM ; POINTERS OF THE 0887 08 INY B1 FB LDA ($FB),Y 0388 FOUR VOICES 85 47 STA V2PT+2 088A 088C C8 INY 088D B1 FB LDA ($FB),Y 85 4A STA V3PT+2 083F 0891 CS INY 0892 B1 FB LDA ($FB),Y 0894 85 4D STA V4PT+2 0896 DØ BD BNE NEXT 0898 STORE MUSIC TABLE 0898 86 51 MUSIC STX NOTES+1 84 59 089A STY NOTES ; POINTERS MUSIC1 LDY #$00 0890 A0 00 SET UP TO 089E A5 5D LDA INCA ; TRANSLATE FOUR ; VOICES INTO INCREMENTS 08A0 85 4E STA INCPT SET TO READ CONTROL LDA #$7F 08A2 A9 7F ; KEY STA $DC00 08A4 8D 00 DC JOET ANY KEYS 08A7 AD 01 DC LDA $DC01 STOP KEY? 08AA C9 7F CMP #$7F 08AC DØ 06 BNE MUSIC9 ;NO PLA CLEAN UP STACK 08AE 68 PLA 98AF 68 08E0 68 PLA 58 08 JMP EXIT JEXIT ROUTINE 08B1 ·40 08B4 08B4 B1 50 MUSIC9 LDA (NOTES), Y GET DURATION ; IF ZERO EXIT PHRASE ; IF 1 GET NEXT SEGMENT 08B6 F0 3E BEQ ENDSNG C9 01 9838 CMP #$91 ; OF PHRASE 08BA F0 2B BEQ NXTSEG 85 5A IS DURATION 08BC STA DUR E6 50 INCREMENT MUSIC 08BE MUSIC2 INC NOTES BNE MUSICS ; POINTER 98C9 D0 02 08C2 E6 51 INC NOTES+1 08C4 B1 50 MUSIC3 LDA (NOTES),Y FREAD IN FOUR ; VOICES AND STORE TAX 0806 AA 0807 BD 01 0A LDA FRQTAB,X ; IN VOICE INCREMENT 08CA STA (INCPT), Y 91 4E ; LOCATIONS 08CC E6 4E INC INCPT 08CE BD 00 ØA. LDA FRQTAB-1,X STA (INCPT),Y 91 4E 08D1 08D3 E6 50 INC NOTES BNE MUSIC4 08D5 DØ 02 ``` ``` LOC CODE LINE INC NOTES+1 MUSIC4 INC INCPT 08D7 E6 4E FREPEAT FOR 08D9 98DB A5 4E LDA INCPT ; OTHER VOICES 08DD C9 5A CMP #V4IN+2 BNE MUSIC3 08DF DØ E3 08E1 20 F7 08 JSR PLAY JPLAY THE NOTES 4C 9C 08 JMP MUSIC1 08E4 ; DO NEXT LINE 08E7 NXTSEG INY GET POINTER TO 08E7 08 B1 50 LDA (NOTES),Y ; NEW SEGMENT OF 08E8 ; MUSIC PHA 08EA 48 08EB C8 INY B1 50 08EC LDA (NOTES), Y 08EE 85 51 STA NOTES+1 08F0 68 PLA 98F1 85 50 STA NOTES JMP MUSIC1 08F3 4C 9C 08 08F6 08F6 60 ENDSNG RTS FRETURN TO CONTROL LOOP 08F7 08F7 A0 00 PLAY LDY #$00 FLAY THE NOTES 08F9 86 5F LDX TEMPO 08FB PLAY1 08FB 18 CLC SUM WAVEFORMS OF 08FC LDA (VIPT+1),Y B1 41 ADC (V2PT+1),Y ADC (V3PT+1),Y ; FOUR VOICES FOR ; OUTPUT 08FE 71 46 0900 71 49 ADC (V4PT+1),Y 0902 71 40 0904 STA USRPRT JOUTPUT VALUE 8D 01 DD 0907 LDA VIPT JADD INCREMENTS A5 40 0909 65 52 ; TO THE FOUR WAVE- ADC VIIN 090B 85 40 STA VIPT ; FORM TRBLE POINTERS LDA VIPT+1 ADC VIIN+1 090D A5 41 VOICE 1 090F 65 53 85 41 STA VIPT+1 0911 0913 A5 45 LDA V2PT ; 2 0915 65 54 ADC V2IN 0917 85 45 STA V2PT LDA V2PT+1 ADC V2IN+1 0919 A5 46 091B 65 55 091D 85 46 STA V2PT+1 091F A5 48 LDA V3PT ; 3 0921 65 56 ADC V3IN 85 48 0923 STA V3PT 0925 A5 49 LDA V3PT+1 ADC V3IN+1 0927 65 57 0929 85 49 STR V3PT+1 092B A5 4B LDA V4PT 4 ; 092D 65 58 ADC V4IN 092F 85 4B STA V4PT 0931 A5 4C LDA V4PT+1 ADC V4IN+1 65 59 0933 STA V4PT+1 0935 35 4C 0937 CH DEX 0938 DØ 08 BNE TIMMAS ; WASTE TIME DEC DUR 093A C6 5A ; DECREMENT DURATION 0930 FØ ØC BEQ ENDNOT ; IF DUR=0 THEN DO NEXT LINE 093E A6 5F LDX TEMPO 0940 DØ B9 BNE PLAY1 0942 TIMWAS BNE *+2 DØ 00 WASTE A BIT OF 0944 DØ 00 BNE *+2 ; TIME 0946 DØ 00 BNE *+2 0948 DØ B1 BNE PLAY1 094A ENDNOT RTS 094B 094B FRQTAB =$0A01 094B . END ``` ## Symbol table | SYMBOL VE | ALUE | | | | | | | |-----------|------|--------|------|--------|------|--------|-------| | CHTROL | 086E | CONTR1 | 087E | DDR | DD93 | DUR | 005A | | END | 989C | ENDNOT | 094A | ENDSNG | 08F6 | EXIT | Ø858 | | FRQTAB | 0801 | INCA | 005D | INCPT | 094E | LOOP | 0843 | | MUSIC | 0898 | MUSIC1 | 089C | MUSIC2 | 98BE | MUSIC3 | Ø8C4 | | MUSIC4 | Ø8D9 | MUSIC9 | 08B4 | NEXT | 0855 | NOTES | 0050 | | NXTSEG | 08E7 | PLAY | 08F7 | PLAY1 | 08FB | RESTRE | 085A | | STORE | 0821 | TEMPO | 005F | TIMWAS | 0942 | USRPRT | DDØ 1 | | VIIN | 0052 | V1PT | 0040 | V2IN | 0054 | V2PT | 0045 | | VSIN | 9956 | V3PT | 0048 | V4IN | 9958 | Y4PT | 004B | #### Program 15. ## Music data frequency tables Bass frequency table Hi-lo ØA7C to ØA93 ``` .:0A00 00 00 01 E9 02 06 02 25'......% .:0A08 02 45 02 68 02 8C 02 B3'.E....4 .:0A10 02 DC 03 08 03 36 03 67'.*..6.. .:0A18 03 9A 03 D1 04 0B 04 49'.......I . 0A20 04 8A 04 CF 05 19 05 66'...... . 0A28 05 B8 06 0F 06 6C 06 CD'. 35 07 A3 15 09 9F 08 921.5. .:0A30 07 08 17 .:0A38 09 ØA 31 1F .:0A40 0B 71 90 0C D7 6A 0F 45 10 2E 11 .:0A48 0E .:0A50 12 29 13 3E 14 62 15 99'.).>... 19 AF 18 36'...>...6 20 5C 22 48'....\"H .:0A58 16 E2 18 3Ē 8B 20 .:0A60 10 D4 1E 2B 31/$R&.(T+1 36 6C/-T0.316. 28 05 .:0A68 24 52 26 7B .:0A70 2D Ċ3 30 70 33 5E 0319**=.... .:0A78 39 A8 3D 15 00 F4 01 .:0A80 01 12 01 23 01 34 01 46'...#.4.F .:0A88 01 5A 01 6D 01 84 01 9B'.Z..... .:0A90 01 B3 01 CD 00 00 00 00'.+.\... 5A 01 ``` ## Wraparound waveform table no 1 ``` 35 36 3A 3B 3C 3C 36 37 38 39/34566789 3B 3B 3C 3C/9::///CC .:0800 33 34 .:0808 39 3A .:0810 3C 3C 3019: 11117 301000000 30 3C 30 .:0818 3C 3C 3C 3B 3B 3B 3B 3B*(<<;;;;; .:0820 3A 3A 3A 3A 3A 3A 3A 3A 39 39*:::::99 .:0828 39 39 39 39 39 39 39*99999999 3A 3A 3A 3A 3B 3C 3C 3C 3E 3E 3E 3E 3A 3B 3B 3B':::::;;; 3D 3D 3D 3D';<<<==== .:0B30 3D 3D1;<<<==== 3F 3F1>>>>???? .:0338 3F 3F 3F .:0B40 3F . 0B48 3F 3F 3F 3F 3F 3F 3F1????????? 3D 3C 3C 38 37 36 3B1>>>==<<; 3E 3E ЗE .:0B50 ЗD 351): 988765 .:0B58 3B 3A 39 38 34 33 20 2B 32 31 2A 29 22 21 30 2F 2E 28 27 26 21 20 1F 201432107. - 251,+*)(1&% .:0B60 .:0B68 23 1F'$#"!! .. .:0B70 24 1F 20 20 21 25 29 .:0B90 23 28 23 28 24 24 29 29 26 26 27/##$$%&&/ 28 28/(()))**+ .:0398 2B 2B 2B 2B 29/++++++* .:0BA0 2B 2B 2B 29 29 28 27 27 26(**))(((& .:0BA8 28 2A 25 24 23 22 .:0BB0 21 20 1F 1D1%##"! .. 18 19 18 17 15 14 13'..... 10 0F 0D 0C 0B 09 08'..... 06 05 04 03 03 02 01'..... .:0BB8 10 .:0BC0 11 .:0BC8 07 .: 0BD0 01 00 00 00 00 00 00 00 00′...... .:0BD8 00 00 01 01 01 02 03 04′ 0D/..... 1A/....#2/ 27/..."#2/ 95 96 09 0B 0C .:0BE0 97 08 12 13 1F 20 15 22 .:0BE8 0F 19 16 13 .:0BF0 1B 1D 23 25 .:0BF8 28 2A 2B 2C ``` ## Wraparound waveform table no 2 ## Wraparound waveform table no 3 ## Wraparound waveform table no 4 ``` 20 20 20 20 20 21 22 22 22 23 25 26 26 27 28 28 28 20 2D 2E .:0E00 20 20 21 25 28 23/!!!""## :0E08 21 :0E10 24 281$%%&4(( :0E18 29 30 31 32 36 36 37 38 38 38 36 35 34 :0E20 2F 32 33 34 341/0122344 38 38 38 38 32 31 37 38 33 35 :0E28 38156677388 38 36 38 37 :0E30 :0E38 2F176543217 2B 29 221 :0E40 2E 2D 27 26 24 18 :0E48 20 1E 10 18 16 15 131 ØF ØE 12 ed ec :0E50 10 9C 0B1 :0E58 ØB 0B 0C 90 0D 0E ØF 11 16 18 28 28 211 :0E60 1D 2D 1F 13 14 1 B 24 26 20 2F 39/18(*,-/9 :0E68 :0E70 30 31 31 31 30 2F 2E 2D/01110/. :0E70 30 0. :0E78 2B 29 :0E80 19 17 27 24 22 20 1D 1B(+) 15 13 12 11 11 11 12 16 27 13 :0E88 11 14 18 18 23 25 20 2B 1E 10 1E 2C 23 21 20 21 :0E90 29 28 2B1 2A 29 1A 18 251 :0E98 27 17 15 # :0EA0 :0EA8 15 15 15 16 17 18 18 20 28 22 27 24 26 25 25 27 23 1E 28 21 287 287 :0EB0 :0EB8 28 ((1&2#! :0EC0 1E 10 1B 1A 19 18 191 18 21 24 1 D :0EC8 1A 1B 10 1F 231 22 25 25 23 22/$ :0ED0 24 25 25 1F 1C :0ED8 20 1 D 1E 10 1 B 1B 1B1 :0EE0 10 1 D 1E 1F 20 21 22 22 22 22 22 22 21 21 291 :0EE8 1F :0EF0 1F 1Ε 1E 1E 1E 1E / 1E :0EF8 1F 1F 20 20 20 201 ... ``` Fig. 5.13. Sample voice waveforms for music synthesis program. ## Main control table FF specifies control code Followed by 1 = tempo as next byte Followed by 2 = next 4 bytes are waveform table pointers If not FF then hi-lo of pointer to following score table #### Music table In groups of 5 Duration and 4 notes, 1 per voice $\phi\phi$ = return to main control loop $\emptyset$ 1 = read next 2 bytes as pointer into this table lo-hi ``` .:1000 09 30 30 80 06 09 30 80′...... .:1020 34 48 30 22 10 0C 00 00′4H0″.... .:1028 00 00 14 4A 32 2C 14 0C1...J2... .:1030 00 00 00 00 00 34 4E 36 301...4N60 .:1038 18 0C 00 00 00 00 14 521.....R .:1040 3A 32 1A 0C 00 00 00 00 001:2..... 54 3C 36 1E 0C 00 00/4T<6.... .:1048 34 .:1070 34 48 30 28 10 3C 00 00/4H0(.C.. .:1088 8A 20 00 00 00 00 00 201. .:1090 40 28 00 00 20 44 20 0010 001@<.. D .:1098 00 20 48 30 00 00 40 4A/. H0..@J .:1080 00 32 02 20 4A 40 3A 02/.2. J@:. .:10A8 40 40 00 00 0A 20 40 3A/00... 0: .:1080 32 10 20 3A 00 22 1A 20/2. :.". .:1088 3A 00 22 00 20 3A 32 28/:.". :2( .:1000 00 40 36 20 26 08 20 32/.@6,&. 2 .:1008 20 26 08 40 30 24 00 06/.&.@0$.. .:1000 00 .:10D0 20 44 00 00 00 40 40 3A' D...@@: .:10D8 2E 04 20 40 3C 30 06 201.. @K0. .:10E0 40 00 10 8A 20 40 28 00'@... @( .:10E8 00 20 44 2C 00 00 20 48'. D... H .:10F0 30 00 00 20 4A 32 00 00′0.. J2.. 34 00 00 40 4E 36" L4..@N6 .:10F8 20 4C .:1100 00 10 20 4E 3C 40 10 40'.. NC@.@ .:1108 48 00 00 1E 20 48 30 40'H... HK@ 40 40 28 00 18 20 401.00(.. @ .:1110 1E .:1118 30 36 18 40 30 30 28 10/(6.000). .:1120 20 36 30 28 10 40 32 001 60(.@2. ``` ``` .:1128 00 1A 20 32 28 22 1A 40',. 2(".@ .:1130 32 00 00 14 20 32 28 22'2... 2(" 14 20 32 00 00 10 20 40°. 2...@ 28 00 10 20 44 20 00 10°(...D)... .:1138 14 .:1140 .:1148 20 48 30 00 0A 20 4A 32′ H0.. J2 .:1150 00 0A 20 4E 36 00 0A 40′.. N6..@ .:1158 52 00 00 02 20 52 4A 401R... RJ@ 40 4A 00 00 0A 20 4A/.@J... J .:1160 02 .:1168 40 3A 10 40 40 00 00 16'@:.@@... .:1170 20 40 3A 32 16 40 3A 32/ @:2.@:2 20 32 32 32 24 14 28 16 00′(. 22(.. 20 40 00′0<2$. 0: . 1178 28 16 30 .:1180 40 .:1188 28 14 40 44 00 2C 0C 20°(.@D.,. .:1190 48 00 30 0A 00 40 4A 007H.0..@J. .:1198 32 06 20 4A 44 3E 06 4012. JD).@ .:11A0 48 00 30 06 20 44 3E 2C/H.O. D>, 00 40 40 28 10 8A 201.00(.. .:11A8 06 36 30 28 40 40 28 00/@60(@@(. .:11B0 40 .:11B8 06 20 48 36 00 00 40 4E'. H6..@N .:11C0 3E 14 8E 20 4E 3E 00 00'>.. N>.. .:11C0 3E 14 8E 20 4E 3E 00 00/).. N).. .:11C8 40 4A 3E 36 06 20 44 3E @J>6. D> .:11D0 36 00 60 4E 40 36 18 60/6..Ne6.. .:11D8 4E 3E 32 14 40 4E 3C 30/ND2.@NC6 301ND2.@NK0 .:11E0 10 20 00 00 00 00 00 40/. ....@ .:11E8 3C 00 24 14 20 3C 32 2C/(.$. <2, .:11F0 14 40 44 00 2C 0C 20 444.@D.,. D : 0A 00 40 40 00 28/2,..@.( 40 3C 30 28 40 40/. @<0(@ 06 20 48 00 30 00/.(. H.0. 3C 36 10 20 4E 3C/@N(6. NK .:11F8 32 2C 0A .:1200 10 20 40 .:1208 00 28 06 .:1210 40 4E 3C 32 10 2016.@J<2. .:1218 36 00 40 4A 3C .:1220 48 3C 30 00 20 4A 40 3A/HK0. J@: .:1228 1A 20 4A 40 3A 14 20 4A . J@:, J .:1230 40 3A 10 20 4A 40 3A 0C'@:, J@:, .:1238 20 4A 40 3A 0A 20 4A 40 J@:, J@ .:1240 3A 06 40 4A 40 3A 02 20':.@J@:, 00 00 1A 40 44 2C 24/2...@D.$ 20 44 2C 34 1A 40 42'. D.$.@B 22 02 20 44 2C 24 1A'*". D.$. 4E 2E 1E 0C 20 4E 3E'@N... N .:1260 0C .:1268 2A 22 02 .:1270 40 4E 2E 1E 02 20'6.@J,.. .:1278 36 1E 40 4A 2C 1E 02 20^6.@J,... :1280 44 2C 20 0E 40 46 2E 22^1D, .@F." .:1288 10 20 44 2C 20 00 40 46^7. D, .@F. :1290 2E 22 02 20 46 2E 22 00^7.". F.". .:1298 20 46 2E 22 10 20 46 2E 27 F.". F. .:12A0 24 10 20 46 2E 28 00 20^$. F.(... :12A0 24 10 20 46 2E 28 00 20^$. F.(... :12B0 02 20 46 2E 1A 00 40 46^7. F...@F. .:12B8 28 22 1A 20 44 2C 26 02^7(". D,&... :12C0 60 46 28 1A 02 40 40 2A^7. F.(...@H** :12C0 22 1A 20 46 2E 28 02 60^*. F.(...@H** :12C0 48 2A 1A 02 40 4A 2C 24^7H*..@J,$ .:1278 36 1E 40 4A 2C .:12D0 48 2A 1A 02 40 4A 2C 24'H*..@J,$ .:12D8 0C 20 4A 32 00 1A 40 44'. J2..@D .:12E0 2C 00 14 20 3C 00 24 0C',... (.$. .:12E8 40 32 00 1A 02 20 00 00/02..... .:12E8 40 32 00 1n 02 20 00 00 02... .:12F0 00 00 40 40 3A 32 02 20'..@@:2. .:12F8 00 00 00 00 40 44 24 1A'...@D$. .:1300 0C 20 44 24 2C 1A 40 42'. D$..@B .:1308 2A 22 02 20 44 2C 24 1A'*". D.$. 2E'@N... N. .:1310 40 4E 2E 1A 0C 20 4E .:1318 36 1A 00 40 4A 2C 1A 02/6..0J,.. 1A 40 40 28′ D$,.@@< .:1320 20 44 24 20 .:1328 22 1A 20 40 28 22 1A 40/". @(".@ .:1348 40 4A 00 00 02 20 4A 28/@J... J( .:1350 22 02 40 40 28 24 1E 204".@@($. ``` ``` .:1358 40 28 24 1E 40 4E 36 00′@($.@N6. 10 20 4E1. N($. N 28 00 1A1(.. N(.. 20 4E 28 .: 1360 10 24 10 .:1368 28 00 20 4E 18 30' N($. H0 .:1370 20 4E 28 24 1E 20 48 20 44 2C .:1378 24 10 24 14 20'$, D,$. .:1380 40 28 24 18 60 4A 28 22/@(≴..J(" 30 24 10 60 521..NO$..R 00 40 48 2C 201:(..@J, .:1388 1A 60 4E .:1390 3A 28 02 00 40 4A 2C 44 24 20 08 40 46′. D$ .@F .:1398 08 20 2E 24 00'$.. F.$. .:13A0 24 1E 06 20 46 .:13A8 40 44 24 16 20 46 2E/@D,$. F. 20 24 00 40 52 3A 2A 18 20/$.@R:*. .:13B0 .:13B8 52 3A 2A 00 40 4E 36 281R: #.@N6# .:1358 52 36 26 00 40 40 40 467. H0*.@J .:1368 32 00 06 20 46 32 26 24/2.. J2,$ 32/@D... J2 20/..@F.". 20 4A 22 10 .:13D0 40 44 00 2C 02 .:13D8 2C 00 40 46 2E 22/J2".@N6" 54/. R:". T 1E/K2, T2,. 32/ T2,. T2 22 .:13E0 4A 32 00 40 4E 36 .:13E8 02 20 52 .:13F0 3C 32 2C 3A 22 20 54 00 20 32 20 .:13F8 20 54 32 2C 1A 20 54 žē/,. т2,. .:1400 2C 16 20 54 32 20 14 10 20 00 00 00'T2,. ... .:1408 54 32 20 .:1408 54 32 2C 10 20 00 00 00 00 12..... .:1410 0C 40 00 00 00 00 14 544.@....T .:1418 4A 44 00 0C 00 00 00 00 07.JD.... .:1420 14 54 4A 44 00 0C 00 007.TJD... .:1428 00 00 14 54 4A 44 00 0C7...TJD... .:1430 00 00 00 00 20 54 4A 444....TJD... .:1438 00 20 54 4E 44 24 20 547.TND$ T .:1448 0C 00 00 20 08 14 54 4A/....TJ .:1450 44 20 0C 00 00 20 08 20/p ... .:1458 56 4A 44 1E 20 56 4A 44/VJD. VJD 00 00 00 00'.(..... .:1490 00 28 10 0C 00/...(.... 40/...(.@ .:1498 14 00 00 28 10 0C 00 .:14A0 00 00 20 00 00 28 10 .:14A8 00 00 00 00 00 40 40 284.....@@( .:14B0 10 8A 20 40 36 30 28 404.. @60(@ .:14B8 48 00 30 06 20 48 40 307H.0. H@0 .:14C0 00 00 20 38 34 2E 08 207.. 84.. .:14C8 38 34 2E 0C 20 38 34 2E784.. 84 2E184.. 84. .:14D0 10 40 00 00 00 12 20 38/.e... 8 .:14D8 32 28 12 40 00 00 00 08/2*.e... 2A 08 40 00 00' 82*.@.. .:14E0 20 38 32 .:14E8 00 02 20 38 32 2A 02 40′.. 82*.@ 00 90 20 38 32 00 00 00 SA 20 2A′....82* 38′.@...8 .:14F0 01 00 00 90 20 .:14F8 8C 40 40 00 00 00 08/4(.@.... .:1500 34 28 00 341 (4(, 84 .:1508 20 30 34 28 08 20 38 .:1510 28 08 14 00 00 00 08 0C/(..... .:1518 00 00 00 00 14 00 00 00′...... .:1540 00 00 00 00 40 00 00 00 ....e... 001.84..0. .:1548 16 20 38 34 2E 16 40 .:1550 00 00 10 20 38 34 2E 34 @.... 84 .:1558 40 00 20 00 00 08 38 40 00 00 00 04 201.... .:1560 2E 08 00 00 00'84..@... .:1568 38 34 2E 90 40 20 .:1570 SC 38 32 2A 8C 14 3C1. 82*..C 0C 00 00 00 3C12*..... .:1578 32 2A 8C .:1580 14 00 32 2A 86 0C 00 001..2*... ``` ``` .:1588 00 SC 14 3C 00′...(2*.. 2A′.... 82* 32 2A 80 99 99 SC 20 38 32 .:1590 00 32 2A 08 0C 384..82*..8 :1598 80 14 38 2A 00 .:15A0 32 . 15A8 0C 38 32 10 OC .:15B0 00 .:15B8 00 00 00 .:15C0 00 00 00 14 99 .:15C8 .:15D0 00 00 00 20 38 32 .:15D8 00 12 20 .:15E0 00 00 00 .:15E8 08 40 00 00 00 02 20 .:15F0 32 2A 8C 40 00 00 00 04/2*.@.... 28 04 40 00 00 40 < 44*.0. 30 34 28 00 40'. < 4*.0 12 20 30 34 28'... < 4* 00 00 10 20 30'.0... ğč 34 .:15F8 20 .:1600 00 .:1608 00 99 99 .:1610 12 40 00 00 00 10 40 00 00 00 08/4*.@.... .:1618 34 2A 0C 38 34 08 20 2E 08 40 00 001 84..@.. .:1620 20 38 34 2E 10 40/.. 84..@ 16 20 36 2E 2A/... 6.* 00 00 12 20 36/.@... 6 60 40 38 2E 16/.*..@8. .:1628 00 .:1630 00 00 00 .:1638 16 40 00 00 00 12 20 .:1640 2E 2A 0C 38 08 0C 00 00′.F@8.... .:1648 14 46 40 .:1650 00 00 14 .:1658 00 99 99 ØC 00 00 00 00 40 .:1660 08 29 99 99 99 99'@<. .... .: 1668 40 30 10 20 00 00 00 00 00 00. 00 00 10 20 00'.@@... 40 4A 00 00 1A'...@J... 00 00 40 40 00'....@@. 1 00 00 00 00 40'....@ 1 1E 20 00 00 00'N.... .:1670 00 40 40 .:1678 00 00 00 .:1680 20 00 00 .:1688 00 10 20 .:1690 4E 00 00 48 00 30/060(0H.0 30 40 4E/. H0600N 30 00 00/66. Nc.. 36 .:16B8 40 30 28 40 .:1600 06 20 48 40 36 .:1608 30 36 10 20 4E 20 48 40′@J<2. H@ .:16D0 40 4A 3C 32 10 60 4A 40 3A 1A′<0..J@:. .:16D8 30 30 00 40 48 40'.D>2 @H@ 44 32 .:16E0 60 3E 20 00 00 00 00 2010.... 10 20 44 30 301000. DO .:16E8 3C 1E 20 30 .:16F0 40 30 30 10 00 141. HK0... .:16F8 10 20 48 30 .:1700 4A 40 1A 02 0C 40 4A 00′J@...@J. 14 40 4A 2C 14 0C 40/..@J...@ 00 00 14 40 4A 28 10/J...@J(. 40 4A 00 00 14 40 4A/.@J...@J 0C 0C 40 4A 00 20 14/$..@J... .:1708 00 .:1710 4A 00 00 .:1718 0C .:1720 24 0C 0C .:1728 40 4A 22 40 4A 00'@J"..@J. 0A 0C 4A 1E 06 0C 407..@J...@ 14 4E 46 1A 02/J...NF. 00 00 14 46 4E/.FN...FN 46 4E 00 00 14/...FN... .:1730 00 14 40 48 1E :1738 4A 00 00 14 4E .:1740 0C .:1748 2C 46 4E 14 ØC .:1750 46 46 4E 00'FN(..FN. 4E 10 OC 28 .:1758 00 46′..FN$..F 14 46.4E 24 0C 0C :1760 4E @A'N...FN" 00 00 14 46 4E 22 4E FN...FN 14'...FN... 00'JR...JR. 4A'..JR...J .:1768 0C .:1770 1E 46 4E 00 00 14 46 06 OC 46 4E 00 00 .:1778 4A 52 1A 02 0C 4A 52 .:1780 00 4A 52 20 14 ØC .:1788 10'R...JR(. 52 28 52 00 00 14 4A 14 4A 52′.JR...JR 00 00 14′$..JR... 4A 52 00′JR"..JR. .:1790 0C 4A 52 00 00 .:1798 24 ØC. 9C 4A 52 .:17A0 4A 52 22 OA OC 06 0C 4A'..JR...J .:17A8 00 14 4A 52 1E 54 18 02 R. . . NT. . .:1780 52 00 00 14 4E ``` ``` .:17B8 0C 4E 54 00 00 14 4E 547.NT...NT .:1700 2C 14 0C 4E 54 00 00 144,...NT... .:17C8 4E 54 28 10 0C 4E 54 00'NT(..NT. .:17D0 00 14 4E 54 24 0C 0C 4E'..NT$..N .:17D8 54 00 00 14 4E 54 22 0A/T...NT". .:17E0 0C 4E 54 00 00 14 4E 54/.NT...NT .:17E8 1E 06 0C 4E 54 00 00 204 58 1A 02 20 3A 00 00′RX.. .:17F0 52 :17F8 0A 20 40 00 00 10 20 4A/. @., .:1800 00 00 1A 20 40 00 00 10/... @. 18 201." T .:1808 20 4A 00 00 18 20 .:1810 00 22 20 4A 00 00 .:1818 52 00 00 22 20 58 00 00'R.." X. .:1820 28 20 52 00 00 22 .:1828 00 00 28 80 62 58 20 5844 R.. " X 52 32/..(..XR2 .:1830 20 00 00 00 00 20 00 004 .:1838 10 SA 40 00 00 00 02 80′..@.... .:1840 00 00 00 00 00 00 00 00/...... ``` Program 17. #### Music table commands The bytes in this table are stored in groups of 5 bytes; these are a duration value and a note value for each of the four voices. If the duration byte contains a $\emptyset\emptyset$ then this specifies the end of the score segment and the program returns to the main control table. If the duration byte contains a $\emptyset$ 1 then this specifies that the next two bytes contain a pointer to another section of the music table. This address is stored in lo, hi form. ## 5.12 Analog to digital conversion The circuit used to convert an analog signal to a digital value is very simple. It involves the use of a voltage comparator IC and a digital to analog converter. The comparator has two inputs; one is the voltage to be measured and the other is a variable reference voltage. The comparator output will go high when the reference voltage is equal to or greater than the voltage being measured. If the reference voltage is generated by a D to A converter then it is a fairly simple matter to vary the converter output until it matches the input voltage. This point is detected by a change in the comparator output. Fig. 5.12 shows such a circuit. Analog to digital conversion using the circuit in Fig. 5.12 relies heavily on software to find the correct D to A output value. This could be done simply by ramping up the output voltage from zero (using a simple increment loop) until the desired voltage is reached. This, however, would be very slow and could take up to 255 steps to find the match. A quicker technique, known as successive approximation, requires just eight loops. The successive approximation technique starts by setting the most significant bit (bit 8) to 1 and all other bits to zero. It then tests to see if the voltage resulting from this value is greater or smaller than the voltage to be measured. If it is larger then the msb is left set and if smaller then the msb is cleared. The routine then sets bit 7 to 1 leaving bit 8 in the state defined in the previous loop and all less significant bits set to zero. The same test is then performed to discover whether the resulting voltage value is ``` 10 REM PROGRAM TO INPUT 10K VALUES 20 REM 30 REM FROM AN A TO D CONVERTER AND THEN 40 REM DISPLAY 320 VALUES AT A TIME TO 50 REM A GRAPHICS SCREEN 60 REM 70 REM PROTECT MEMORY 80 REM 90 POKE52,120:POKE54,120:POKE56,120:CLR 100 REM 110 REM READ THE VALUES 120 REM 130 POKE 49263,1:SYS49248:REM 49263 IS SAMPLING VALUE 140 REM 150 REM GO INTO GRAPHICS MODE 160 REM 170 SYS49627 180 REM 190 REM LOOP TO DISPLAY 32 SCREENS FULL 200 REM 210 POKE 49197,1:FORI=0T031:REM 49197 IS SAMPLING VALUE 220 SYS49152. I*320+30720 230 GETA$: IFA$=""THEN230 240 SYS49667: NEXT: SYS49790 ``` Program 18. greater or less than the input voltage to be measured. Depending on the result bit 7 is left set or cleared. This procedure is then repeated for all the other bit positions in the byte, with the result that only eight operations need be performed to obtain the required value. A successive approximation technique is shown in the first part of Program 13. Program 18 is an example of one of the many applications to which an analog to digital conversion circuit can be applied. This program performs the function of a simple storage oscilloscope. This storage oscilloscope program is very short and written in Basic. It does, however, require that the machine code program in Program 13 is already loaded into memory. The program samples 10240 values with a maximum sampling rate of approximately 2500 samples per second. This sampling rate can be varied to less than this by changing the contents of location 49263. The input waveform is then displayed in high resolution as 32 screens of information. ## 5.13 Expansion port The expansion port is a 44 pin edge connector on the rear of the CBM 64. It gives access to most of the Commodore 64's internal signals. The port is designed to take two main types of device. The first are simple memory mapped devices such as ROMs or I/O chips like a 6526 (if you can get one) or a 6522. The second type of device is more interesting. These are less passive in that they can read or write direct to memory without going through the processor. The most common of these devices is the Commodore Z80 card. Using the DMA (direct memory access) line totally disables the 6510 processor while the card is active. #### 5.13.1 Pin descriptions #### **Expansion** port 44 pin double sided .1 edge connector socket. Labelled 1-22 (top) and A-Z (G,I,O and Q skipped) ## Power connection (power out to boards) Pins 1,22,A,Z Ground Ø V 2,3 +5 volts ## Timing signals Pins I/O 6 Out Dot clock 8 MHz approx (varies with TV standard (PAL NTSC ..) $\phi$ 2 Out $\phi$ 2 Phase two clock ## Bus control signals Pins I/O Out BA system buses available from VIC chip 13 Input DMA Direct memory access (gives expansion card control of system buses) 5 Input $R/\overline{W}$ Read/write ## **Interrupts** Pins I/O 4 Input IRO Interrupt request D Input NMI Non maskable interrupt ## Memory mapping Pins I/O 7 Out I/O1 Address decoded \$DEØØ-\$DEFF 10 Out 1/O2 Address decoded \$DF00-\$DFFF 11 Out ROML Addr decoded \$8000-\$A000 B Out ROMH Addr decoded \$E000-\$FFFF 8 Input GAME Expansion ROM at \$A000-\$C000 (no Basic ROM) 9 Input EXROM Exp ROM at \$8000 #### Reset line Pins I/O C Both RES Reset everything ## System buses Pins I/O 14-21 Both Data bus consisting of eight unbuffered lines with a maximum load of 1 TTL device. Line 14 is D7 and line 21 is $D\emptyset$ . F-Y Both Address lines; these sixteen lines are unbuffered and have a maximum load of 1 TTL device. Line F is A15 and line Y is A0. ## 5.13.2 ROM cartridge The expansion port is set up to make ROM cartridges a simple direct connection. An expansion ROM for address \$8000 using a 2764 (8K by 8) is not too hard if you can obtain or make a board to fit the expansion port edge connector. Just connect the 13 address lines and 8 data lines. Then connect chip select to LROM and connect the 64's EXROM to ground. The 7464 pins Vpp, Vcc and PGM go to 5 V and Vss goes to ground. ## 5.13.3 I/O chips on the expansion port Wiring up a 6526 or 6522 is similar but clock, interrupt and reset also have to be implemented. It is important to connect this type of chip to I/O1 or I/O2 and not to LROM or HROM. Figure 5.14 shows 2764 and 6522 pin outs and appropriate expansion port connections. #### **MEMORY EXPANSION** | PIN# | TYPE | |------|----------------| | 1 | GND | | 2 | +5V | | 3 | +5V | | 4 | ĪRQ | | 5 | R∕W | | 6 | DOT CLOCK | | 7 | <u>1∕</u> Ø1 | | 8 | GAME | | 9 | EXROM | | 1Ø | 1 <b>∕ Ø</b> 2 | | 11 | ROML | | PIN# | TYPE | |------|------| | 12 | ВА | | 13 | DMA | | 14 | D7 | | 15 | D6 | | 16 | D5 | | 17 | D4 | | 18 | D3 | | 19 | D2 | | 2Ø | D1 | | 21 | DØ | | 22 | GND | | PIN# | TYPE | |------|----------| | Α | GND | | В | ROMH | | С | RESET | | D | N.MI | | E | $\phi_2$ | | F | A15 | | Н | A14 | | J | A13 | | K | A12 | | L | A11 | | M | A1Ø | | PIN # | TYPE | |-------|------| | N | A9 | | P | A8 | | R | A7 | | S | A6 | | T | A5 | | U | A4 | | V | A3 | | W | A2 | | X | A1 | | Y | ΑØ | | Z | GND | Fig. 5.14. The allocation and function of pins on the memory expansion connector. Connection of 6522 via I/O expansion connections. Fig. 5.14. (contd.) ## Chapter Six # **Interrupts and Their Use** Interrupts are the signals used by peripheral devices, such as the CIA chips, to signal to the processor that they require servicing. This IRQ signal will then cause the processor to halt its current operation temporarily in order to service the interrupt generating device. Having completed this servicing the processor returns to the interrupted program. ## 6.1 Interrupt requests (IRQ) The major implementation of IRQs in the Commodore 64's operating system is to scan and receive key presses from the keyboard. This IRQ runs on Timer A of CIA#1. The timer value is set up so that the keyboard is scanned every 1/60th of a second. IRQ interrupts can be disabled by setting bit 2 of the processor status register or by the use of the command SEI. To re-enable IRQ, reset bit 2 or use the CLI command. The SEI command is used by the disk operating system to prevent timing errors when accessing the disk. The only other standard use of IRQs in the operating system of the Commodore 64 is in the tape I/O routines. Rather than just disabling IRQs, the tape system uses IRQs for reading from or writing to the tape. The tape system uses both Timer A and Timer B on CIA#1 for reading and writing. For more information on the tape routines, see Chapter 4. ## 6.2. Interrupt generating devices ## 6.2.1 The CIA chips ## CIA#1 Register 14 (\$DCØD) - Bit 7 Enable/disable (write), occurred (read) - 6 Not used - 5 Not used - 4 FLAG 1 line (cassette read) - 3 Serial data register - 2 TOD clock alarm - 1 Timer B - Ø Timer A When reading bit 7 is used to determine whether an enabled IRQ on this chip occurred (if more than one device is connected to the IRQ line) i.e. if this bit was not set when the IRQ routine was caused, it must have been either the VIC chip or the expansion port. If bit 7 is set, bits Ø-4 will tell what caused the IRQ. It should be noted that when using IRQs, it is advisable to keep a separate record of the IRQs that are enabled, since their respective bits may be set but not necessarily enabled. When writing, bit 7 is used to tell the CIA whether the lower bits are for disabling or enabling. If bit 7 is set, any other bits set are to enable an IRQ. If bit 7 is not set, any other bits set are to disable an IRQ. #### 1. Cassette read FLAG 1 line This line is used by the cassette read routines and creates an IRQ when it is enabled. The tape flags an IRQ on this line when the pulse on the tape goes from high to low. An example of the use of this IRQ is shown in Chapter 4 (fast tape operation). ## 2. Serial data register (SDR) The SDR is a serial input/output device of the 6526 CIA chip. When IRQ is enabled on this register, the IRQ will be caused either when the full byte has been read in (input) or when it has been sent out (output). When the IRQ occurs, either a new value to send must be put into the SDR or the byte contained in the SDR will be read and the SDR left to input the next byte. The SDR uses 2 lines on the user port. These lines are SP1 and CNT1, which together are used to send/receive data. When sending, each bit is set on SP and the CNT line is used to clock the bit using Timer A. An example use of the SDR can be found in Chapter 5. #### 3. TOD clock alarm When the TOD clock alarm IRQ has been enabled (after setting TOD and the alarm) an IRQ occurs when the value in TOD becomes equal to the value set in the alarm. An example of how to use the TOD clock can be found in Chapter 5. #### 4. Timer B Timer B can run in three different modes; as a straight timer, a count down on pulses from the CNT line of the user port, and a count down on Timer A running out. These three methods are outlined in Chapter 5. An IRQ will occur on Timer B in any of the three modes of operation when the value in Timer B clocks past zero. #### 5. Timer A Timer A has only one mode of operation; as a straight timer. An IRQ on Timer A will occur when the value in Timer A clocks past zero. Note that with Timers A and B, the timer always decreases until it clocks past zero. Therefore, to time something, the timer should be set to the period and when it runs out the time is up. With CIA IRQs, the IRQ is cleared by reading register 14. ## 6.2.2 The VIC chip The VIC chip is also connected to the IRQ line and VIC chip IRQs are controlled by registers 25 and 26 on the VIC chip. ``` VIC register 25 ($DØ19) (Interrupt flag register) ``` Bit Set on any enabled VIC IRQ occurring - 6-4 Not used - 3 Light pen (1=occurred) - 2 Sprite to sprite collision (1=occurred) - Sprite to background collision (1=occurred) 1 - Ø Raster compare (1=occurred) VIC register 26 (\$DØ1A) (Interrupt enable mask) Bit 7–4 Not used - Light pen (1=enabled) - Sprite to sprite collision (1=enabled) 2 - Sprite to background collision (1=enabled) 1 - Raster compare (l=enabled) Ø To enable IRQ, register 26 should be read and the bit to enable set and then written back to register 26. When the IRQ occurs, reading register 25 will tell you which VIC IRQ has occurred. To clear the IRQ, the corresponding bit to clear is written to register 25. ## 1. Light pen The light pen IRO occurs when the raster scan reaches the position of the light pen and the light pen values can then be read from registers 19 and 20. ## 2. Sprite to sprite collision Sprite to sprite collision IRQ occurs when any bit in the sprite to sprite collision register $(3\emptyset - D\emptyset 1E)$ is set. ## 3. Sprite to background collision Sprite to background collision IRQ occurs when any bit in the sprite to background collision register (31 - \$DØ1F) is set. #### 4. Raster compare Raster compare IRQ occurs when the raster position being displayed becomes equal to the compare value written to registers 17 (\$DØ11 high bit) and 18 (\$DØ12). #### 6.2.3 The expansion port IRQ can be caused by any I/O device connected to the Commodore 64 via the expansion port. There are two 'spare' areas for such I/O devices; they can either be addressed at \$DEØØ or \$DFØØ. See Chapter 5 for an example of adding a 6522 VIA chip to the Commodore 64 via the expansion port. ## 6.3 Non maskable interrupts (NMI) NMIs are so named because they cannot be disabled by the SEI command. Normally the NMI routine is not called regularly like the IRQ routine. This is because NMI is only caused by 2 devices: - a) RS232 (user port FLAG sent low) - b) RESTORE key There are five other ways of causing an NMI on the 64 that are not implemented in the software. These are Timers A and B, internal shift register, expansion port, and time of day clock on CIA#2. All NMIs except the RESTORE key and the expansion port are controlled by register 14 (\$DD\ODD) on CIA#2. This register is used as a dual purpose write (enable/disable NMI) and read (to determine the source of NMI). ## CIA#2 Register 14 (\$DDØD) Bit 7 Enable/disable (write), occurred (read) - 6 Not used - 5 Not used - 4 User port FLAG line RS232 data received) - 3 Shift register - 2 TOD clock alarm - 1 Timer B - Ø Timer A When reading bit 7 is used to determine whether an enabled NMI on this chip has occurred (if more than one CIA chip is connected to the NMI line) i.e. if this bit was not set when the NMI routine was caused, the NMI must have been either the RESTORE key or expansion port. If bit 7 is set, bits $\emptyset$ -4 will tell what caused the NMI. It should be noted that when using NMIs, it is advisable to keep a separate record of the NMIs that are enabled as their respective bits could be set but not enabled. When writing, bit 7 is used to tell the CIA whether the lower bits are for disabling or enabling. If bit 7 is set, any other bits set are to enable an NMI. If bit 7 is not set, any other bits set are to disable an NMI. #### 6.4 Devices that cause NMI ## 1 User port FLAG line This line is the one used by the RS232 routines and causes an NMI when it is enabled. The method of flagging an NMI on this line is to set the line to +5 V and then to ØV. This method is outlined in Program 19 which uses 1Ø lines on the user port to transfer a block of memory from one CBM 64 to another (8 data lines and 2 lines to flag the NMI on the other 64). When initialised, the NMI ``` 2000 *=$C000 C000 A91F C000 07 !DO NORMAL STOP/RESTORE C06A 811/0C0 C06A 68 EXIT C06B A8 C06C 68 C06D AA C06E 68 C06F 40 PLA !RESTORE REGISTERS TAY ! AND EXIT HMI PLA TAX PLA RTI C070 C070 04 FLAG BYT 4 C071 ! C071 !ROUTINE TO SEND A FILE OVER THE C071 ! USER PORT C071 C071 AD70C0 SAVER LDA FLAG C074 C904 CMP #$04 C076 D0F9 BNE SAVER C078 A204 LDX #$04 !IF RECEIVING. ! DON'T SEND LDX #$@4 !POINT TO SAVE ``` ``` CO7A B5AB LOOP LDA $AB,X !GET ADDRESS BYTE CO7C 2002C0 JSR SBYTE !SEND THE BYTE CO7F CA DEX !DO NEXT? C080 D0F8 BNE LOOP !YES C082 A000 LDY #$00 C084 B1AC LOOP1 LDA (*AC),Y !GET A FILE BYTE CO86 2002C0 JSR SBYTE !SEND IT CO89 2090C0 JSR SBYTE !SEND IT CO89 2090C0 JSR BUMP !INCREMENT AND TEST END CO8C 90F6 BCC LOOP1 !NOT YET CO8E 18 CLC !SAVED OK CO8F 60 RTS !DONE C090 ! C090 E6AC BUMP INC *AC !INCREMENT LO BYTE CO90 E6AC BUMP INC *AD !INCREMENT LO BYTE CO96 ASAC BUMP1 LDA *AC !COMPARE SAVE CO96 ASAC BUMP1 LDA *AC !COMPARE SAVE CO96 CSAF COMP *AFE ! ADDRESS TO END CO96 E5AF SBC *AFF CO95 E6A BUMP2 TO CO96 E5AF BUMP2 TO CO96 E5AF SBC *AFF CO96 E6A BUMP2 TO CO96 E5AF C09F ! C09F E6FB BUMP2 INC $FB C081 D002 BNE BUMP3 C083 E6FC INC $FC C085 A5FB BUMP3 LDA $FB C087 C5FD CMP $FD C089 A5FC LDA $FC C088 E5FE SBC $FE C081 60 RTS C086 !INCREMENT LO BYTE !INCREMEN' !INCREMEN' !COMPARE L ! ADDRESS ! ADDRESS !INCREMENT HI BYTE COMPARE LOAD ! ADDRESS WITH END C089 85AC STA $AC C08B A5C2 LDA $C2 C08B A5C2 LDA $C2 C08B 85AD STA $AD C08F 4C71C0 JMP SAVER C0C2 !ROUTINE TO SEND 1 BYTE ACROSS C0C2 ! THE USER PORT C0C2 !C0C2 ! ``` Program 19. vector is set to point to the receive routine and the SAVE vector is set to the send routine. When the user of one computer SAVEs a block of memory with device 7, the file is passed through to the other computer by setting a full byte onto the data lines and causing an NMI by setting the PA2 line hi then lo (PA2 is connected to FLAG both ways). The NMI routine then reads the byte from the port and either stores it as a load address or as part of the file. To send a file, use SAVE", 7. Files are automatically received. ## 2. Serial data register The NMI SDR has exactly the same operation as the IRQ SDR except that instead of lines CNT1 and SP1, lines CNT2 and SP2 are used. SDR use can be seen in Chapter 5. #### 3. TOD clock alarm The NMI TOD clock alarm has exactly the same operation as the IRO TOD clock alarm. An example of how to use the TOD clock can be found in Chapter 5. #### 4. Timer B The NMI Timer B has exactly the same operation as the IRQ Timer B. #### 5. Timer A The NMI Timer A has the same operation as the IRQ Timer A. ## 6. RESTORE key The RESTORE key on the keyboard is connected directly to the NMI line and is not a true NMI. When RESTORE is pressed, the NMI routine is called and if the STOP key is also down, NMI will cause a restart of the computer. This is done by jumping to a routine pointed to by an indirection at \$A\pma\_2: JMP (\$A002). If a cartridge ROM is in place (with the power-up bytes), JMP (\$8002) is used instead. #### 7. Expansion port Expansion port NMI has the same operation as expansion port IRQ except that IRO occurs if the line is low, whereas NMI occurs when the line goes low. #### 6.5 The kernal vectors There are a group of vectors in page three memory that are used for indirect jumps into some of the most useful kernal routines. These have been provided so that the machine code programmer can patch into them to change the operation of the computer. Each vector is a two byte low-high vector to the main machine code kernal routine and by changing its value, you may point it to your own routine. The vectors are as follows: | Address | Default | Use | |--------------------------|---------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | \$Ø314 | \$EA31 | Vector to the IRQ routine. This vector can be changed to point to your own IRQ routine for things such as screen scrolling etc. | | \$Ø316 | \$FE66 | Vector for BRK instruction is changed by all monitors so that when a BRK is encountered, the computer jumps to the monitor. | | \$Ø318 | \$FE47 | Vector to the NMI routine. Its major use is for the detection of the RESTORE key. Other methods are outlined in Chapter 5. | | \$Ø31A | \$F34A | Vector to open file routine. | | \$Ø31C | \$F291 | Vector to close file routine. | | \$Ø31E | \$F2ØE | Vector to set input device. | | \$Ø32Ø | \$F25Ø | Vector to set output device. | | \$Ø322 | \$F333 | Vector to restore I/O. | | \$Ø324 | \$F157 | Vector to input. This routine is used in all peripheral input. It could be used for function keys etc. | | \$Ø326 | \$F1CA | Vector to output. This routine controls all output to the same devices as input (except keyboard). | | <b>\$Ø</b> 328 | \$F6ED | Vector to test STOP routine. The most widely used patch is for disabling the STOP key. | | \$Ø32A | \$F13E | Vector to get. This routine is used to get a single key from the keyboard buffer. The character received is not displayed but is just returned in register .A. The get key has the same operation as input from all devices except the keyboard where input inputs a line until carriage return is pressed. | | \$Ø32C | \$F32F | Vector to abort I/O. | | \$Ø32E | \$FE66 | Unused vector. This vector can be used by your own routines. | | \$\textit{0}33\textit{0} | \$F4A5 | Vector to load routine. This vector is jumped to after the load parameters have been set up. | | \$Ø332 | \$F5ED | Vector to save routine. An example of a patch into this vector can be seen in Chapter 4 and in Program 19 in this chapter. | # Index abort serial I/O files, 59 ACPTR, 49 address bus, 8, 10 allophones, 151 analog interfacing, 155 analog music synthesis, 159 analog to digital converters, 5, 10, 155 anti piracy techniques, 123 ASCII files, 80 auto run, 121 Basic interpreter, 3 BASIN, 52 binary files, 80 BSOUT, 54 cartridge port, 10, 179 cassette buffer, 79 cassette error messages, 84 cassette hardware, 78 cassette operating system routines, 83 cassette operation, 79 character output, 25 CHKIN, 54 CHKOUT, 56 CIA 6526, 4, 8, 15, 33, 70, 78, 79, 127 ff CIA signals and lines, 9 CIOUT, 47 CLALL, 59 clock signals, 8, 10, 13, 130, 137 CLOSE, 57 close all logical files, 59 CLRCH, 59 colour clock, 13 Commodore 64 design concept, 2 computer control for the disabled, 142 data bus, 8, 10 digital to analog conversion, 156 digitising pads, 22 DMA, 10, 13 dot clock, 13 error handler, 67 expansion port, 179, 185 find any tape header, 92 find correct file on tape, 95 FLAG, 131, 186 function key definition, 19 general function serial routines, 51 get character from current input device, 52, 53 GETIN, 52 handshaking, 131, 133 high speed data transfer, 138, 186 high speed tape operation, 110 I/O, 3, 4, 7, 15, 127, 128, 132 input byte from serial port, 49 interrupt control register, 138 IRQ, 4, 9, 13, 108, 138, 183 ff joystick, 5, 10, 22, 127 kernal vectors, 189 keyboard, 15, 127, keyboard buffer, 16 keyboard matrix scanning, 4, 15, 16 keyboard operation modification, 19 keyboard scan simulation program, 17 light pen, 11, 185 listen, 42 load RAM function, 86 LOAD/VERIFY, 62 microprocessor 6510, 2, 7 miscellaneous cassette routines, 96 MPU signal lines, 8, 130 NMI, 9, 13, 84, 186 OPEN, 60 operating system, 3 output character, 54 parallel port expansion, 142, 181 peripheral interface lines, 131 phase 2 clock, 10, 13 PLA, 7, 9, 12 potentiometer joystick, 22 power supply, 7 print 'saving', 90 print tape loading messages, 88 printed circuit board, 6, 7 protect cassette from RS232 NMI, 84 RAM, 3, 10, 12, 13 read cassette, 100 read/write, 13 ready, 13 recording method, 81 relay control circuit, 141 return buffer address, 95 ROM cartridge, 180 RS232 close channel, 77 RS232 command register, 73 RS232 connections, 71 RS232 control register, 72 RS232 open channel, 75 RS232 receive from channel, 76 RS232 serial communications, 70 RS232 status register, 73 RS232 transmit to channel, 76 save memory function, 89 screen, 23 screen display software, 23 screen scrolling, 31 secondary address, 45 send byte to serial bus, 47 send secondary address after talk, 46 serial bus lines, 32 serial bus timings, 34 serial communications, 32, 70 serial data register, 137, 184 serial system routines, 42 serial system variable declare file, 38 set opened file for input, 54 set opened file for output, 56 set up time out for next dipole, 99 SID 6581, 4, 9, 10 SID signals and lines, 10 sound generation, 4, 10 stop key servicing, 90 switch joystick, 22 system control signals, 13 system logic and timing, 12 TALK, 42 tape error handler, 91 tape IRQ, 108, 123 tape security, 123 time of day (TOD) clock, 4, 9, 135, 184 timers, 4, 9, 84, 134, 184 TKSA, 46 UNLISTEN, 48 UNTALK, 47 user port, 128 user port connections, 127, 128, 129 VIC 6567 Chip, 2, 10, 11, 187 VIC signals and lines, 11 voice synthesis, 149 write cassette, 106 write memory, 97 write tape header, 93 Z80 card, 10 A knowledge of the Commodore 64 kernal software and the hardware with which it interacts is essential for programmers wishing to make full use of the machine's capabilities. The kernal software provides the interface between the user, the BASIC interpreter and the electronics – and a thorough knowledge of its functioning gives the programmer a wealth of ideas and methods for interesting programming techniques. This book gives the programmer a unique insight into the operation of the Commodore 64 plus a wide variety of very useful hints on subjects as diverse as reconfiguring the keyboard and anti tape-copying security. The book also covers the user port and the addition of external circuitry to it. ## The Authors Nick Hampshire is a well-known author and microcomputer expert who has specialised in Commodore computer equipment. He started the first hobby microcomputer magazine, later absorbed into *Practical Computing*, of which he was technical editor for several years. He was the co-founder of *Popular Computing Weekly* and founder and managing editor of *Commodore Computing International* magazine. He is also the author of over a dozen books on popular computing, including the very successful and widely acclaimed *PET Revealed* and *VIC Revealed*. Richard Franklin and Carl Graham are programmers with Zifra Software Ltd and together with Nick Hampshire have written some of the software included in this book. Also by Nick Hampshire THE COMMODORE 64 ROMs REVEALED 0 00 383087 X **ADVANCED COMMODORE 64 BASIC REVEALED** 0 00 383088 8 ADVANCED COMMODORE 64 GRAPHICS AND SOUND 0 00 383089 6 THE COMMODORE 64 DISK DRIVE REVEALED 0 00 383091 8 X-060E9E-00-0 NASI COLLINS Printed in Great Britain £10.95 net