This is kind of old stuff, but I ran across the issue of Byte that had the Apple II system description by Steve Wozniak:

| 00     | 1 | Return to 6502 mode                          |
|--------|---|----------------------------------------------|
| 01     | 2 | Branch Always                                |
| 02     | 2 | Branch no Carry                              |
| 03     | 2 | Branch on Carry                              |
| 04     | 2 | Branch on Positive                           |
| 05     | 2 | Branch on Negative                           |
| 06     | 2 | Branch if equal                              |
| 07     | 2 | Branch not equal                             |
| 08     | 2 | Branch on negative 1                         |
| 09     | 2 | Branch not negative 1                        |
| OA     | 1 | Break to Monitor                             |
| 0B- 0F | 1 | No operation                                 |
| 1R     | 3 | R<-2 byte constant (load register immediate) |
| 2R     | 1 | ACC<-R                                       |
| 3R     | 1 | ACC->R                                       |
| 4R     | 1 | ACC<-@R, R<-R+1                              |
| 5R     | 1 | ACC->@R, R<-R+1                              |
| 6R     | 1 | ACC<-@R double                               |
| 7R     | 1 |                                              |
| 8R     | 1 | R<-R-1, ACC<-@R (pop)                        |
| 9R     | 1 | R < -R - 1, $ACC - > @R$                     |
| AR     | 1 | ACC<-@R(pop) double                          |
| BR     | 1 | compare ACC to R                             |
| CR     | 1 | ACC<- ACC+R                                  |
| DR     | 1 | ACC<- ACC- R                                 |
| ER     | 1 | R<- R+1                                      |
| FR     | 1 | R<- R- 1                                     |
|        |   |                                              |

notes

1. All braches are followed by a 1 byte relative displacement. Works identically to 6502 branches.

2. Only ADD, SUB, and COMPARE can set carry

3. Notation:

 $R = a \ 16 \ bit$  "Register" operand designation, one of 16 labelled 0 to 15 (decimal), 0 to F (hexidecimal)

ACC = register operand RO

@R = indicrect reference, using the register R as the pointer

<-, -> = assignment of values

4. Length of instructions: Branches are always two bytes: opcodes followed by relative displacement. Load register immediate (1R) is three bytes: the hexadecimal opcode 10 to 1F followed by the 2 byte literal value of a 16 bit number. All other instructions are one byte in length.

And from that issue of Byte (Apr 1977, I think)...some words from the Woz himself. Retyped without permission.

### The Story of Sweet Sixteen

While writing Apple BASIC, I ran into the problem of manipulating the 16 bit pointer data and its arithmetic in an 8 bit machine.

My solution to this problem of handling 16 bit data, notably pointers, with an 8 bit microprocessor was to implement a non-existent 16 bit processor in software, interpreter fashion, which I refer to as SWEET16.

SWEET16 contains sixteen internal 16 bit registers, actually the first 32 bytes in main memory, labelled RO through R15. RO is defined as the accumulator, R15 as the program counter, and R14 as a status register. R13 stores the result of all COMPARE operations for branch testing. The user acceses SWEET16 with a subroutine call to hexadecimal address F689. Bytes stored after the subroutine call are thereafter interpreted and executed by SWEET16. One of SWEET16's commands returns the user back to 6502 mode, even restoring the original register contents.

Implemented in only 300 bytes of code, SWEET16 has a very simple instruction set tailored to operations such as memory moves and stack manipulation. ost opcodes are only one byte long, but since she runs approximately ten times slower than equivalent 6502 code, SWEET16 should be employed only when code is at a premium or execution is not. As an example of her usefulness, I have estimated that about 1K byte could be weeded out of my 5K byte Apple-II BASIC interpreter with no observable performance degradation by selectively applying SWEET16. []

Article 1684 of comp.sys.apple2.programmer: Newsgroups: comp.sys.apple2.programmer Path: amdahl.uts.amdahl.com!amdahl!amd!decwrl!sdd.hp.com!elroy.jpl.nasa.gov!swrinde!ih np4.ucsd.edu!library.ucla.edu!csulb.edu!csus.edu!netcom.com!sheldon From: sheldon@netcom.com (Sheldon Simms) Subject: Source code (Sweet 16) Message-ID: <sheldonCLz3qM.85r@netcom.com> Organization: NETCOM On-line Communication Services (408 241-9760 guest) Date: Tue, 1 Mar 1994 06:37:33 GMT Lines: 264

Well maybe this should go to comp.sources.apple2 but it's not too long and in my mind it's part of the thread on learning assembly language programming. It's an opportunity to learn from Woz himself.

- Shel don

| **********                      |  |
|---------------------------------|--|
| * *                             |  |
| * APPLE-II PSEUDO MACHINE *     |  |
| * I NTERPRETER *                |  |
| * *                             |  |
| * COPYRI GHT (C) 1977 *         |  |
| * APPLE COMPUTER, INC *         |  |
| * *                             |  |
| * ALL RIGHTS RESERVED *         |  |
| * *                             |  |
| * S. WOZNI AK *                 |  |
| * *                             |  |
| **********                      |  |
| * *                             |  |
| * TITLE: SWEET 16 INTERPRETER * |  |
| * *                             |  |
| **********                      |  |
|                                 |  |

| ROL<br>ROH<br>R14H<br>R15L<br>R15H<br>SAVE<br>RESTORE | EQU<br>EQU<br>EQU<br>EQU<br>EQU | SO<br>S1<br>S1D<br>S1E<br>S1F<br>SFF4A<br>SFF3F |                                                 |
|-------------------------------------------------------|---------------------------------|-------------------------------------------------|-------------------------------------------------|
|                                                       | ORG                             | \$F689                                          |                                                 |
|                                                       | AST                             | 32                                              |                                                 |
|                                                       | JSR<br>PLA                      | SAVE                                            | ; PRESERVE 6502 REG CONTENTS                    |
|                                                       | STA<br>PLA                      | R15L                                            | ;INIT SWEET16 PC<br>;FROM RETURN                |
|                                                       | STA                             | R15H                                            | ; ADDRESS                                       |
| SW16B                                                 | JSR                             |                                                 |                                                 |
|                                                       |                                 | SW16B                                           | ONE SWEET16 INSTR.                              |
| SW16C                                                 | I NC                            | R15L                                            |                                                 |
|                                                       | BNE                             | SW16D                                           | ; INCR SWEET16 PC FOR FETCH                     |
|                                                       | I NC                            | R15H                                            |                                                 |
| SW16D                                                 | LDA                             | >SET                                            | ; COMMON HIGH BYTE FOR ALL ROUTINES             |
|                                                       | PHA                             |                                                 | ; PUSH ON STACK FOR RTS                         |
|                                                       | LDY                             | \$0                                             |                                                 |
|                                                       | LDA                             | (R15L), Y                                       | ; FETCH INSTR                                   |
|                                                       | AND                             | \$F                                             | ; MASK REG SPECI FI CATI ON                     |
|                                                       | ASL                             |                                                 | ; DOUBLE FOR TWO BYTE REGISTERS                 |
|                                                       | TAX                             |                                                 | ; TO X REG FOR INDEXING                         |
|                                                       | LSR                             |                                                 |                                                 |
|                                                       | EOR                             | (R15L), Y                                       | ; NOW HAVE OPCODE<br>: I F ZERO THEN NON-REG OP |
|                                                       | BEQ                             |                                                 | ,                                               |
|                                                       | STX                             | R14H                                            | ; I NDI CATE "PRI OR RESULT REG"                |
|                                                       | LSR<br>LSR                      |                                                 | ; OPCODE*2 TO LSB'S                             |
|                                                       | LSR                             |                                                 | , OFCODE 2 TO LSD 5                             |
|                                                       | TAY                             |                                                 | ; TO Y REG FOR INDEXING                         |
|                                                       | LDA                             | OPTBL-2, Y                                      | ; LOW ORDER ADR BYTE                            |
|                                                       | PHA                             |                                                 | ; ONTO STACK                                    |
|                                                       | RTS                             |                                                 | ; GOTO REG-OP ROUTI NE                          |
| TOBR                                                  | I NC                            | R15L                                            |                                                 |
|                                                       | BNE                             | TOBR2                                           | ; I NCR PC                                      |
|                                                       | I NC                            | R15H                                            |                                                 |
| TOBR2                                                 | LDA                             | BRTBL, X                                        | ; LOW ORDER ADR BYTE                            |
|                                                       | PHA                             |                                                 | ; ONTO STACK FOR NON-REG OP                     |
|                                                       | LDA                             | R14H                                            | ; "PRI OR RESULT REG" INDEX                     |
|                                                       | LSR                             |                                                 | ; PREPARE CARRY FOR BC, BNC.                    |
| סתיים                                                 | RTS                             |                                                 | ; GOTO NON-REG OP ROUTI NE                      |
| RTNZ                                                  | PLA                             |                                                 | ; POP RETURN ADDRESS                            |
|                                                       | PLA<br>JSR                      | RESTORE                                         | ; RESTORE 6502 REG CONTENTS                     |
|                                                       | JMP                             |                                                 | ; RETURN TO 6502 CODE VIA PC                    |
| SETZ                                                  | J MP<br>LDA                     | (R15L)<br>(R15L), Y                             | ; HI GH ORDER BYTE OF CONSTANT                  |
| JEIL                                                  | STA                             | ROH, X                                          | , HI OH OLDER DITE OF CONSTANT                  |
|                                                       | DEY                             | 10011, A                                        |                                                 |
|                                                       | LDA                             | (R15L), Y                                       | ; LOW ORDER BYTE OF CONSTANT                    |
|                                                       | STA                             |                                                 | , Low OWDER DITE OF CONSTRUCT                   |
|                                                       | ~                               |                                                 |                                                 |

|          | TYA         |                   | ; Y REG CONTAI NS 1     |
|----------|-------------|-------------------|-------------------------|
|          | SEC         |                   |                         |
|          | ADC<br>STA  | R15L<br>R15L      | ; ADD 2 TO PC           |
|          | BCC         | SET2              |                         |
| SET2     | I NC<br>RTS | R15H              |                         |
| OPTBL    | DFB         |                   | ; 1X                    |
| BRTBL    | DFB<br>DFB  | LD-1              | ; 0<br>; 2X             |
|          | DFB         |                   | ; 1                     |
|          | DFB<br>DFB  | ST- 1<br>BNC- 1   | ; 3X<br>; 2             |
|          | DFB         | LDAT-1            | ; 4X                    |
|          | DFB<br>DFB  | BC- 1<br>STAT- 1  | ; 3<br>; 5X             |
|          | DFB         | BP-1              | ; 4                     |
|          |             | LDDAT- 1<br>BM- 1 | ; 6X<br>; 5             |
|          | DFB         | STDAT-1           | ; 7X                    |
|          |             | BZ- 1<br>POP- 1   | ; 6<br>; 8X             |
|          |             | BNZ-1             | ; 7                     |
|          | DFB<br>DFB  |                   | ; 9X<br>; 8             |
|          | DFB         |                   | , 8<br>; AX             |
|          |             | BNM1-1            | ; 9<br>. DV             |
|          | DFB<br>DFB  |                   | ; BX<br>; A             |
|          | DFB         |                   | ; CX                    |
|          | DFB<br>DFB  |                   | ; B<br>; DX             |
|          | DFB         |                   | ; C                     |
|          | DFB<br>DFB  | I NR- 1<br>NUL- 1 | ; EX<br>; D             |
|          | DFB         | DCR-1             | ; FX                    |
|          | DFB<br>DFB  |                   | ; E<br>; UNUSED         |
|          | DFB         | NUL- 1            | ; F                     |
| * FOLLO  | WI NG       | CODE MUST BE      |                         |
| * CONTA  | I NED       | ON A SINGLE PAG   | E!                      |
| SET      | BPL         | SETZ              | ; ALWAYS TAKEN          |
| LD<br>BK | LDA         | ROL, X<br>*- 1    |                         |
| DK       | EQU<br>STA  | ROL               |                         |
|          |             | ROH, X            | ; MOVE RX TO RO         |
|          | STA<br>RTS  | ROH               |                         |
| ST       |             | ROL<br>ROL, X     |                         |
|          | STA<br>LDA  | ROL, X<br>ROH     | ; MOVE RO TO RX         |
|          | STA         | ROH, X            |                         |
| STAT     | RTS<br>LDA  | ROL               |                         |
| STAT2    |             | (ROL, X)          | ; STORE BYTE I NDI RECT |
|          |             |                   |                         |

|       | LDY  |             |                                                                                                                                            |
|-------|------|-------------|--------------------------------------------------------------------------------------------------------------------------------------------|
| STAT3 |      | R14H        | ; INDICATE RO IS RESULT NEG                                                                                                                |
| I NR  | I NC | ROL, X      |                                                                                                                                            |
|       | BNE  | I NR2       | ; I NCR RX                                                                                                                                 |
|       | I NC | ROH, X      |                                                                                                                                            |
| I NR2 | RTS  |             |                                                                                                                                            |
| LDAT  | LDA  | (ROL, X)    | ; LOAD INDIRECT (RX)<br>; TO RO                                                                                                            |
|       | STA  | ROL         | ; TO RO                                                                                                                                    |
|       | LDY  | \$0         |                                                                                                                                            |
|       | STY  | ROH         | ; ZERO HI GH ORDER RO BYTE<br>; ALWAYS TAKEN                                                                                               |
|       | BEQ  | STAT3       | ; ALWAYS TAKEN                                                                                                                             |
| POP   | LDÝ  | \$0         | ; HI GH ORDER BYTE = $0$                                                                                                                   |
|       | BEQ  | POP2        | : ALWAYS TAKEN                                                                                                                             |
| POPD  | JSR  | DCR         | ; DECR RX<br>; POP HI GH ORDER BYTE @RX<br>; SAVE IN Y REG<br>; DECR RX                                                                    |
|       | LDA  | (ROL, X)    | ; POP HI GH ORDER BYTE @RX                                                                                                                 |
|       | TAY  |             | ; SAVE IN Y REG                                                                                                                            |
| POP2  | JSR  | DCR         | ; DECR RX                                                                                                                                  |
|       | LDA  | (ROL, X)    | ; LOW ORDER BYTE<br>; TO RO                                                                                                                |
|       | STA  | ROL         | ; TO RO                                                                                                                                    |
|       | STY  | ROH         |                                                                                                                                            |
| POP3  | LDY  | \$0         | ; INDICATE RO AS LAST RESULT REG                                                                                                           |
|       | STY  | R14H        |                                                                                                                                            |
|       | RTS  |             |                                                                                                                                            |
| LDDAT | JSR  | LDAT        | ; LOW ORDER BYTE TO RO, INCR RX<br>; HI GH ORDER BYTE TO RO                                                                                |
|       | LDA  | (ROL, X)    | ; HI GH ORDER BYTE TO RO                                                                                                                   |
|       | STV  | RUH         |                                                                                                                                            |
|       | JMP  | I NR        | ; I NCR RX<br>; STORE I NDI RECT LOW ORDER<br>; BYTE AND I NCR RX. THEN<br>; STORE HI GH ORDER BYTE.<br>; I NCR RX AND RETURN<br>; DECR RX |
| STDAT | JSR  | STAT        | STORE INDIRECT LOW ORDER                                                                                                                   |
|       | LDA  | ROH         | ; BYTE AND INCR RX. THEN                                                                                                                   |
|       | STA  | (ROL, X)    | STORE HIGH ORDER BYTE.                                                                                                                     |
|       | JMP  | INR         | INCR RX AND RETURN                                                                                                                         |
| STPAT | JSR  | I NR<br>DCR | ; DECR RX                                                                                                                                  |
|       | LDA  | ROL         |                                                                                                                                            |
|       | STA  | (ROL, X)    | ; STORE RO LOW BYTE @RX                                                                                                                    |
|       | JMP  | POP3        | ; STORE RO LOW BYTE @RX<br>; I NDI CATE RO AS LAST RESULT REG                                                                              |
| DCR   | LDA  | ROL, X      |                                                                                                                                            |
|       | BNE  | DCR2        | ; DECR RX                                                                                                                                  |
|       | DEC  | ROH, X      |                                                                                                                                            |
| DCR2  | DEC  | ROL, X      |                                                                                                                                            |
|       | RTS  |             |                                                                                                                                            |
| SUB   | LDY  | \$0         | ; RESULT TO RO                                                                                                                             |
|       | CPR  | SEC         | ; NOTE Y REG = $13*2$ FOR CPR                                                                                                              |
|       | LDA  | ROL         |                                                                                                                                            |
|       | SBC  | ROL, X      |                                                                                                                                            |
|       | STA  | ROL, Y      | ; RO- RX TO RY                                                                                                                             |
|       | LDA  | ROH         |                                                                                                                                            |
|       | SBC  | ROH, X      |                                                                                                                                            |
| SUB2  | STA  | ROH, Y      |                                                                                                                                            |
|       | TYA  |             | ; LAST RESULT REG*2                                                                                                                        |
|       | ADC  | \$0         | ; CARRY TO LSB                                                                                                                             |
|       | STA  | R14H        |                                                                                                                                            |
|       | RTS  |             |                                                                                                                                            |
| ADD   | LDA  | ROL         |                                                                                                                                            |
|       | ADC  | ROL, X      |                                                                                                                                            |
|       | STA  | ROL         | ; RO+RX TO RO                                                                                                                              |
|       | LDA  | ROH         |                                                                                                                                            |
|       | ADC  | ROH, X      |                                                                                                                                            |
|       |      |             |                                                                                                                                            |

| BS               | LDĂ<br>JSR<br>LDA        | SUB2<br>R15L<br>STAT2<br>R15H     | ; RO FOR RESULT<br>; FINISH ADD<br>; NOTE X REG IS 12*2!<br>; PUSH LOW PC BYTE VIA R12    |
|------------------|--------------------------|-----------------------------------|-------------------------------------------------------------------------------------------|
| BR<br>BNC<br>BR1 | LDA<br>BPL               | STAT2<br>BNC2<br>(R15L), Y<br>BR2 | ; PUSH HI GH ORDER PC BYTE<br>; NO CARRY TEST<br>; DI SPLACEMENT BYTE                     |
| BR2              | STA<br>TYA               | R15L<br>R15L<br>R15H              | ; ADD TO PC                                                                               |
| BNC2<br>BC       | STA<br>RTS<br>BCS<br>RTS | R15H                              |                                                                                           |
| BP               | ASL<br>TAX<br>LDA<br>BPL | ROH, X<br>BR1                     | ; DOUBLE RESULT-REG INDEX<br>; TO X REG FOR INDEXING<br>; TEST FOR PLUS<br>; BRANCH IF SO |
| BM               | RTS<br>ASL               |                                   | ; DOUBLE RESULT-REG INDEX                                                                 |
|                  | TAX<br>LDA<br>BMI<br>RTS | ROH, X<br>BR1                     | ; TEST FOR MINUS                                                                          |
| BZ               | ASL<br>TAX               |                                   | ; DOUBLE RESULT-REG INDEX                                                                 |
|                  | LDA<br>ORA<br>BEQ<br>RTS | ROL, X<br>ROH, X<br>BR1           | ; TEST FOR ZERO<br>; (BOTH BYTES)<br>; BRANCH I F SO                                      |
| BNZ              | ASL<br>TAX               |                                   | ; DOUBLE RESULT-REG INDEX                                                                 |
|                  | LDA<br>ORA<br>BNE<br>RTS | ROL, X<br>ROH, X<br>BR1           | ; TEST FOR NON-ZERO<br>; (BOTH BYTES)<br>; BRANCH IF SO                                   |
| BM1              | ASL<br>TAX               |                                   | ; DOUBLE RESULT-REG INDEX                                                                 |
|                  | LDA<br>AND<br>EOR        | ROL, X<br>ROH, X<br>SFF           | ; CHECK BOTH BYTES<br>; FOR \$FF (MINUS 1)                                                |
|                  | BEQ<br>RTS               | BR1                               | ; BRANCH I F SO                                                                           |
| BNM1             | ASL<br>TAX               |                                   | ; DOUBLE RESULT-REG INDEX                                                                 |
|                  | LDA<br>AND<br>EOR        | ROL, X<br>ROH, X<br>\$FF          | ; CHECK BOTH BYTES FOR NO \$FF                                                            |
| NUL              | BNE<br>RTS               | BR1                               | ; BRANCH IF NOT MINUS 1                                                                   |
| RS               | LDX<br>JSR               | \$18<br>DCR                       | ; 12*2 FOR R12 AS STACK POINTER<br>; DECR STACK POINTER                                   |

; POP HIGH RETURN ADDRESS TO PC LDA (ROL, X) STA R15H JSR DCR ; SAME FOR LOW ORDER BYTE LDA (ROL, X) STA R15L RTS JMP RTNZ RTN - -W. Sheldon Simms | Newt's Friend / Jack Kemp for President sheldon@netcom.com | Freedom implies responsibility \_\_\_\_\_ Article 1685 of comp. sys. apple2. programmer: Newsgroups: comp. sys. appl e2. programmer Path: amdahl.uts.amdahl.com!amdahl!amd!decwrl!sdd.hp.com!elroy.jpl.nasa.gov!swrinde!ih np4. ucsd. edu!library. ucla. edu!csulb. edu!csus. edu!net.com. com!sheldon From: sheldon@netcom.com (Sheldon Simms) Subject: Sweet 16 description Message-ID: <sheldonCLz3tB.8C4@netcom.com> Organization: NETCOM On-line Communication Services (408 241-9760 guest) Date: Tue, 1 Mar 1994 06:39:11 GMT Lines: 992 Well since I posted the source, here's what it does for anyone who might not know... - Shel don \_\_\_\_\_ SWEET 16 - INTRODUCTION by Dick Sedgewick Sweet 16 is probably the least used and least understood seed in the Apple ][. In exactly the same sense that Integer and Applesoft Basics are languages, SWEET 16 is a language. Compared to the Basics, however, it would be classed as low level with a strong likeness to conventional 6502 Assembly language. To use SWEET 16, you must learn the language - and to quote "WOZ", "The opcode list is short and uncomplicated". "WOZ" (Steve Wozniak), of course is Mr. Apple, and the creator of SWEET 16. SWEET 16 is ROM based in every Apple ][ from \$F689 to \$F7FC. It has it's own set of opcodes and instruction sets, and uses the SAVE and RESTORE routines from the Apple Monitor to preserve the 6502 registers when in use, allowing SWEET 16 to be used as a subroutine.

It uses the first 32 locations on zero page to set up its 16 double byte registers, and is therefore not compatible with Applesoft Basic without some additional efforts.

The original article, "SWEET 16: The 6502 Dream Machine", first appeared in Byte Magazine, November 1977 and later in the original "WOZ PAK". The article is included here and again as test material to help understand the use and implementation of SWEET 16.

Examples of the use of SWEET 16 are found in the Programmer's Aid #1, in the Renumber, Append, and Relocate programs. The Programmer's Aid Operating Manual contains complete source assembly listings, indexed on page 65.

The demonstration program is written to be introductory and simple, consisting of three parts:

- 1. Integer Basic Program
- 2. Machine Language Subroutine
- 3. SWEET 16 Subroutine

The task of the program will be to move data. Parameters of the move will be entered in the Integer Basic Program.

The "CALL 768" (\$300) at line 120, enters a 6502 machine language subroutine having the single purpose of entering SWEET 16 and subsequently returning to BASIC (addresses \$300, \$301, \$302, and \$312 respectively). The SWEET 16 subroutine of course performs the move, and is entered at Hex locations \$303 to \$311 (see listing Number 3).

After the move, the screen will display three lines of data, each 8 bytes long, and await entry of a new set of parameters. The three lines of data displayed on the screen are as follows:

- Line 1: The first 8 bytes of data starting at \$800, which is the fixed source data to be moved (in this case, the string A\$).
- Line 2: The first 8 bytes of data starting at the hex address entered as the destination of the move (high order byte only).
- Line 3: The first 8 bytes of data starting at \$0000 (the first four SWEET 16 registers).

The display of 8 bytes of data was chosen to simplify the illustration of what goes on.

Integer Basic has its own way of recording the string AS. Because the name chosen for the string "AS" is stored in 2 bytes, a total of five housekeeping bytes precede the data entered as AS, leaving only three additional bytes available for display. Integer Basic also adds a housekeeping byte at the end of a string, known as the "string terminator".

Consequently, for convenience purposes of the display, and to see the string terminator as the 8th byte, the string data entered via the keyboard should be limited to two characters, and will appear as the 6th and 7th bytes. Additionally, parameters to be entered include the number of bytes to be moved. A useful range for this demonstration would be 1-8 inclusive, but of course 1-255 will work.

Finally, the starting address of the destination of the move must be entered. Again, for simplicity, only the high-order byte is entered, and the program allows a choice between Decimal 9 and high-order byte of program pointer 1, to avoid unnecessary problems (in this demonstration enter a decimal number between 9 and 144 for a 48K APPLE).

The 8 bytes of data displayed starting at \$00 will enable one to observe the condition of the SWEET 16 registers after a move has been accomplished, and thereby understand how the SWEET 16 program works.

From the article "SWEET 16: A 6502 Dream Machine", remember that SWEET 16 can establish 16 double byte registers starting at \$00. This means that SWEET 16 can use the first 32 addresses on zero page.

The "events" occurring in this demonstration program can be studied in the first four SWEET 16 registers. Therefore, the 8 byte display starting at \$0000 is large enough for this purpose.

These four registers are established as RO, R1, R2, R3:

| RO<br>R1<br>R2<br>R3 | \$0000<br>\$0002<br>\$0004<br>\$0006 | &<br>&<br>&<br>& | 0001<br>0003<br>0005<br>0007 | -SWEET 16 accumulator<br>-Source address<br>-Destination address<br>-Number of bytes to move |
|----------------------|--------------------------------------|------------------|------------------------------|----------------------------------------------------------------------------------------------|
|                      |                                      |                  |                              |                                                                                              |
| R14<br>R15           | \$001C<br>\$001E                     | &<br>&           | 001D<br>001F                 | -Prior result register<br>-SWEET 16 Program counter                                          |

Additionally, an examination of registers R14 and R15 will extend and understanding of SWEET 16, as fully explained in the "WOZ" text. Notice that the high order byte of R14, (located at \$1D) contains \$06, and is the doubled register specification (3X2=\$06). R15, the SWEET 16 program counter contains the address of the next operation as it did for each step during execution of the program, which was \$0312 when execution ended and the 6502 code resumed.

To try a sample run, enter the Integer Basic program as shown in Listing #1. Of course, REM statements can be omitted, and line 10 is only helpful if the machine code is to be stored on disk. Listing #2 must also be entered starting at \$300. NOTE: A 6502 disassembly does not look like listing #3, but the SOURCEROR disassembler would create a correct disassembly.

Enter "RUN" and hit RETURN Enter "12" and hit RETURN (A\$ - A\$ string data) Enter "18" and hit RETURN (high-order byte of destination)

The display should appear as follows:

\$0800-C1 40 00 10 08 B1 B2 1E (SOURCE) \$0A00-C1 40 00 10 08 B1 B2 1E (Dest.) \$0000-1E 00 08 08 08 0A 00 00 (SWEET 16)

NOTE: The 8 bytes stored at \$0A00 are identical to the 8 bytes starting at \$0800, indicating that an accurate move of 8 bytes length has been made. They are moved one byte at a time starting with token C1 and ending with token 1E. If moving less than 8 bytes, the data following the moved data would be whatever existed at those locations before the move.

The bytes have the following significance:

A Token\$

| C1 | 40 | 00  | 10  | O 08 B1 B2 |      | 1E   |            |
|----|----|-----|-----|------------|------|------|------------|
|    |    |     |     |            |      |      |            |
|    |    |     |     |            |      |      | String     |
| VŇ |    | DSP | NÝ. | Ą          | DATA | DATA | Terminator |

The SWEET 16 registers are as shown:

| low       | hi gh | low  | hi gh | low  | hi gh | low | hi gh   |
|-----------|-------|------|-------|------|-------|-----|---------|
| \$0000 1E | 0Ō    | 08   | 08    | 08   | 0Ā    | 00  | 0Ō      |
|           |       |      |       |      |       |     |         |
|           |       |      |       |      |       |     |         |
| reg       | ister | regi | ster  | regi | ster  | reg | ji ster |
| ]         | RO    | I    | 21    | F    | 22    |     | R3      |
| (a        | cc)   | (sou | ırce) | (de  | est)  | (#b | ytes)   |

The low order byte of RO, the SWEET 16 accumulator, has 1E in it, the last byte moved (the 8th).

The low order byte of the source register R1 started as 00 and was incremented eight times, once for each byte of moved data.

The high order byte of the destination register R2 contains \$0A, which was entered at 10 (the variable) and poked into the SWEET 16 code. The low-order byte of R2 was incremented exactly like R1.

Finally, register R3, the register that stores the number of bytes to be moved, has been poked to 8 (the variable B) and decremented eight times as each byte got moved, ending up \$0000.

By entering character strings and varying the number of bytes to be moved, the SWEET 16 registers can be observed and the contents predicted.

Working with this demonstration program, and study of the text material will enable you to write SWEET 16 programs that perform additional 16 bit manipulations. The unassigned opcodes mentioned in the "WOZ Dream Machine" article should present a most interesting opportunity to "play".

SWEET 16 as a language - or tool - opens a new direction to Apple ][ owners without spending a dime, and it's been there all the time.

"Apple-ites" who desire to learn machine language programming, can use SWEET 16 as a starting point. With this text material to use, and less opcodes to learn, a user can quickly be effective.

# Listing #1

#### >Li st

| 10  | PRINT "[D]BLOAD SWEET": REM CTRL D        |
|-----|-------------------------------------------|
| 20  | CALL - 936: DIM A \$ (10)                 |
| 30  | INPUT "ENTER STRING A \$ ", A \$          |
| 40  | INPUT "ENTER # BYTES " , B                |
| 50  | IF NOT B THEN 40 : REM AT LEAST 1         |
| 60  | POKE 778 , B : REM POKE LENGTH            |
| 70  | INPUT "ENTER DESTINATION ", A             |
| 80  | IF A > PEEK (203) - 1 THEN 70             |
| 90  | IF A < PEEK (205) + 1 THEN 70             |
| 100 | POKE 776 , A : REM POKE DESTINATION       |
| 110 | M = 8 : GOSUB 160 : REM DI SPLAY          |
| 120 | CALL 768 : REM GOTO \$0300                |
| 130 | M = A : GOSUB 160 : REM DI SPLAY          |
| 140 | M = 0 : GOSUB 160 : REM DI SPLAY          |
| 150 | PRINT : PRINT : GOTO 30                   |
| 160 | POKE 60 , 0 : POKE 61 , M                 |
| 170 | CALL - 605 : RETURN : REM XAM8 IN MONITOR |
|     |                                           |

### Listing #2

300: 20 89 F6 11 00 08 12 00 00 13 00 00 41 52 F3 07 FB 00 60

# Listing #3

### SWEET 16

| +     | 11 | 00 | 08 | JSR<br>SET<br>SET<br>A |    | 89<br>source address<br>destination address |
|-------|----|----|----|------------------------|----|---------------------------------------------|
| \$309 | 13 | 00 | 00 | SET                    | R3 | length                                      |

|       |    | В   |       |
|-------|----|-----|-------|
| \$30C | 41 | LD  | @R1   |
| \$30D | 52 | ST  | @R2   |
| \$30E | F3 | DCR | R3    |
| \$30F | 07 | BNZ | \$30C |
| \$311 | 00 | RTN |       |
| \$312 | 60 | RTS |       |

Data will be poked from the Integer Basic program:

р

| " A" | from | Li ne | 100 |
|------|------|-------|-----|
| " B" | from | Li ne | 60  |

\_\_\_\_\_

SWEET 16: A Pseudo 16 Bit Microprocessor

by Steve Wozniak

Description:

While writing APPLE BASIC for a 6502 microprocessor, I repeatedly encountered a variant of MURPHY'S LAW. Briefly stated, any routine operating on 16-bit data will require at least twice the code that it should. Programs making extensive use of 16-bit pointers (such as compilers, editors, and assemblers) are included in this category. In my case, even the addition of a few double-byte instructions to the 6502 would have only slightly alleviated the problem. What I really needed was a 6502/RCA 1800 hybrid - an abundance of 16-bit registers and excellent pointer capability. My solution was to implement a non-existant (meta) 16-bit processor in software, interpreter style, which I call SWEET 16.

SWEET 16 is based on sixteen 16-bit registers (RO-15), which are actually 32 memory locations. RO doubles as the SWEET 16 accumulator (ACC), R15 as the program counter (PC), and R14 as the status register. R13 holds compare instruction results and R12 is the subroutine return stack pointer if SWEET 16 subroutines are used. All other SWEET 16 registers are at the user's unrestricted disposal.

SWEET 16 instructions fall into register and non-register categories. The register ops specify one of the sixteen registers to be used as either a data element or a pointer to data in memory, depending on the specific instruction. For example INR R5 uses R5 as data and ST @R7 uses R7 as a pointer to data in memory. Except for the SET instruction, register ops take one byte of code each. The non-register ops are primarily 6502 style branches with the second byte specifying a +/-127 byte displacement relative to the address of the following instruction. Providing that the prior register op result meets a specified branch condition, the displacement is added to the SWEET 16 PC, effecting a branch.

SWEET 16 is intended as a 6502 enhancement package, not a stand alone processor. A 6502 program switches to SWEET 16 mode with a subroutine call and subsequent code is interpreted as SWEET 16 instructions. The nonregister op RTN returns the user program to 6502 mode after restoring the internal register contents (A, X, Y, P, and S). The following example illustrates how to use SWEET 16.

| 305 | B9 00 02<br>C9 CD<br>D0 09<br>20 89 F6 |        | LDA<br>CMP<br>BNE<br>JSR | I N, Y<br>#"M"<br>NOMOVE<br>SW16 | ;get a char<br>;"M" for move<br>;No. Skip move |
|-----|----------------------------------------|--------|--------------------------|----------------------------------|------------------------------------------------|
|     | 20 89 F0<br>41                         | MLOOP  | LD                       | @R1                              | ;Yes, call SWEET 16<br>;R1 holds source        |
| 30B | 52                                     |        | ST                       | @R2                              | ;R2 holds dest. addr.                          |
| 30C | F3                                     |        | DCR                      | R3                               | ;Decr. length                                  |
| 30D | 07 FB                                  |        | BNZ                      | MLOOP                            | ;Loop until done                               |
| 30F | 00                                     |        | RTN                      |                                  | ;Return to 6502 mode.                          |
| 310 | C9 C5                                  | NOMOVE | CMP                      | #"E"                             | ;"E" char?                                     |
| 312 | DO 13                                  |        | BEQ                      | EXI T                            | ;Yes, exit                                     |
| 314 | C8                                     |        | I NY                     |                                  | ;No, cont.                                     |

NOTE: Registers A, X, Y, P, and S are not disturbed by SWEET 16.

Instruction Descriptions:

The SWEET 16 opcode listing is short and uncomplicated. Excepting relative branch displacements, hand assembly is trivial. All register opcodes are formed by combining two Hex digits, one for the opcode and one to specify a register. For example, opcodes 15 and 45 both specify register R5 while codes 23, 27, and 29 are all ST ops. Most register ops are assigned in complementary pairs to facilitate remembering them. Therefore, LD ans ST are opcodes 2N and 3N respectively, while LD @ and ST @ are codes 4N and 5N.

Opcodes 0 to C (Hex) are assigned to the thirteen non-register ops. Except for RTN (opcode 0), BK (OA), and RS (OB), the non register ops are 6502 style branches. The second byte of a branch instruction contains a +/-127 byte displacement value (in two's complement form) relative to the address of the instruction immediately following the branch.

If a specified branch condition is met by the prior register op result, the displacement is added to the PC effecting a branch. Except for the BR (Branch always) and BS (Branch to a Subroutine), the branch opcodes are assigned in complementary pairs, rendering them easily remembered for hand coding. For example, Branch if Plus and Branch if Minus are opcodes 4 and 5 while Branch if Zero and Branch if NonZero are opcodes 6 and 7.

SWEET 16 Opcode Summary:

Register OPS-

| 1n<br>2n<br>3n<br>4n<br>5n<br>6n<br>7n<br>8n<br>9n<br>An<br>Bn<br>Cn<br>Dn<br>En | SET<br>LD<br>ST<br>LD<br>ST<br>LDD<br>STD<br>POP<br>STP<br>ADD<br>SUB<br>POPD<br>CPR<br>LNR | Rn<br>Rn<br>@Rn<br>@Rn<br>@Rn<br>@Rn<br>Rn<br>Rn<br>Rn<br>Rn<br>Rn<br>Rn | Constant (Set)<br>(Load)<br>(Store)<br>(Load Indirect)<br>(Store Indirect)<br>(Load Double Indirect)<br>(Store Double Indirect)<br>(Pop Indirect)<br>(Store POP Indirect)<br>(Add)<br>(Sub)<br>(Pop Double Indirect)<br>(Compare)<br>(Increment) |
|----------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------|--------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| En                                                                               | I NR                                                                                        | Rn                                                                       | (Increment)                                                                                                                                                                                                                                      |
| Fn                                                                               | DCR                                                                                         | Rn                                                                       | (Decrement)                                                                                                                                                                                                                                      |

Non-register OPS-

| RTN<br>BR<br>BNC | ea<br>ea                                                            | (Return to 6502 mode)<br>(Branch always)<br>(Branch if No Carry)                                        |
|------------------|---------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------|
| BC               | ea                                                                  | (Branch if Carry)                                                                                       |
| BP               | ea                                                                  | (Branch if Plus)                                                                                        |
| BM               | ea                                                                  | (Branch if Minus)                                                                                       |
| BZ               | ea                                                                  | (Branch if Zero)                                                                                        |
| BNZ              | ea                                                                  | (Branch if NonZero)                                                                                     |
| BM1              | ea                                                                  | (Branch if Minus 1)                                                                                     |
| BNM1             | ea                                                                  | (Branch if Not Minus 1)                                                                                 |
| BK               |                                                                     | (Break)                                                                                                 |
| RS               |                                                                     | (Return from Subroutine)                                                                                |
| BS               | ea                                                                  | (Branch to Subroutine)                                                                                  |
|                  |                                                                     | (Unassi gned)                                                                                           |
|                  |                                                                     | (Unassi gned)                                                                                           |
|                  |                                                                     | (Unassi gned)                                                                                           |
|                  | BR<br>BNC<br>BC<br>BP<br>BM<br>BZ<br>BNZ<br>BM1<br>BNM1<br>BK<br>RS | BR ea<br>BNC ea<br>BC ea<br>BP ea<br>BM ea<br>BZ ea<br>BNZ ea<br>BM1 ea<br>BM1 ea<br>BM1 ea<br>BK<br>RS |

Register Instructions:

SET:

SET Rn, Constant [ 1n Low High ]

The 2-byte constant is loaded into Rn (n=0 to F, Hex) and branch conditions set accordingly. The carry is cleared.

EXAMPLE:

15 34 A0 SET R5 \$A034 ; R5 now contains \$A034

# LOAD:

LD Rn [ 2n ]

The ACC (RO) is loaded from Rn and branch conditions set according to the data transferred. The carry is cleared and

contents of Rn are not disturbed.

EXAMPLE:

15 34 AO SET R5 \$AO34 25 LD R5 ;ACC now contains \$AO34

# STORE:

ST Rn [3n]

The ACC is stored into Rn and branch conditions set according to the data transferred. The carry is cleared and the ACC contents are not disturbed.

### EXAMPLE:

| 25 | LD | R5 | ;Copy the  | contents |
|----|----|----|------------|----------|
| 36 | ST | R6 | ; of R5 to | R6       |

# LOAD INDIRECT:

LD @Rn [4n]

The low-order ACC byte is loaded from the memory location whose address resides in Rn and the high-order ACC byte is cleared. Branch conditions reflect the final ACC contents which will always be positive and never minus 1. The carry is cleared. After the transfer, Rn is incremented by 1.

### EXAMPLE

| 15 34 AO | SET | R5  | \$A034 |                             |
|----------|-----|-----|--------|-----------------------------|
| 45       | LD  | @R5 |        | ; ACC is loaded from memory |
|          |     |     |        | ;location \$A034            |
|          |     |     |        | ;R5 is incr to \$A035       |
|          |     |     |        | -                           |

### STORE INDIRECT:

ST @Rn [ 5n ]

The low-order ACC byte is stored into the memory location whose address resides in Rn. Branch conditions reflect the 2-byte ACC contents. The carry is cleared. After the transfer Rn is incremented by 1.

### EXAMPLE:

| 15 34 AO | SET | R5  | \$A034 | ;Load pointers R5, R6 with        |
|----------|-----|-----|--------|-----------------------------------|
| 16 22 90 | SET | R6  | \$9022 | ; \$A034 and \$9022               |
| 45       | LD  | @R5 |        | ; Move byte from \$A034 to \$9022 |
| 56       | ST  | @R6 |        | ;Both ptrs are incremented        |

## LOAD DOUBLE-BYTE INDIRECT:

LDD @Rn [6n]

The low order ACC byte is loaded from memory location whose address resides in Rn, and Rn is then incremented by 1. The high order ACC byte is loaded from the memory location whose address resides in the incremented Rn, and Rn is again incremented by 1. Branch conditions reflect the final ACC contents. The carry is cleared.

EXAMPLE:

| 15 34 AO | SET | R5  | \$A034 | ;The low-order ACC byte is loaded |
|----------|-----|-----|--------|-----------------------------------|
| 65       | LDD | @R6 |        | ;from \$A034, high-order from     |
|          |     |     |        | ;\$A035, R5 is incr to \$A036     |

### STORE DOUBLE-BYTE INDIRECT:

STD @Rn [7n]

The low-order ACC byte is stored into memory location whose address resides in Rn, and Rn is the incremented by 1. The high-order ACC byte is stored into the memory location whose address resides in the incremented Rn, and Rn is again incremented by 1. Branch conditions reflect the ACC contents which are not disturbed. The carry is cleared.

EXAMPLE:

| 15 34 AO | SET R5  | \$A034 | ;Load pointers R5, R6            |
|----------|---------|--------|----------------------------------|
| 16 22 90 | SET R6  | \$9022 | ;with \$A034 and \$9022          |
| 65       | LDD @R5 |        | ;Move double byte from           |
| 76       | STD @R6 |        | ;\$A034-35 to \$9022-23.         |
|          |         |        | ;Both pointers incremented by 2. |

#### POP INDIRECT:

POP @Rn [8n]

The low-order ACC byte is loaded from the memory location whose address resides in Rn after Rn is decremented by 1, and the high order ACC byte is cleared. Branch conditions reflect the final 2-byte ACC contents which will always be positive and never minus one. The carry is cleared. Because Rn is decremented prior to loading the ACC, single byte stacks may be implemented with the ST @Rn and POP @Rn ops (Rn is the stack pointer).

#### EXAMPLE:

| 15 34 AO | SET | R5  | \$A034 | ;Init stack pointer |
|----------|-----|-----|--------|---------------------|
| 10 04 00 | SET | RO  | 4      | ;Load 4 into ACC    |
| 55       | ST  | @R5 |        | ;Push 4 onto stack  |
| 10 05 00 | SET | RO  | 5      | ;Load 5 into ACC    |

| 55       | ST @R5  |   | ;Push 5 onto stack        |
|----------|---------|---|---------------------------|
| 10 06 00 | SET RO  | 6 | ;Load 6 into ACC          |
| 55       | ST @R5  |   | ;Push 6 onto stack        |
| 85       | POP @R5 |   | ;Pop 6 off stack into ACC |
| 85       | POP @R5 |   | ;Pop 5 off stack          |
| 85       | POP @R5 |   | ;Pop 4 off stack          |

# STORE POP INDIRECT:

STP @Rn

[ 9n ]

The low-order ACC byte is stored into the memory location whose address resides in Rn after Rn is decremented by 1. Branch conditions will reflect the 2-byte ACC contents which are not modified. STP @Rn and POP @Rn are used together to move data blocks beginning at the greatest address and working down. Additionally, single-byte stacks may be implemented with the STP @Rn ops.

### EXAMPLE:

| 14 | 34 | AO | SET | R4  | \$A034 | ;Init pointers    |
|----|----|----|-----|-----|--------|-------------------|
| 15 | 22 | 90 | SET | R5  | \$9022 | -                 |
| 84 |    |    | POP | @R4 |        | ;Move byte from   |
| 95 |    |    | STP | @R5 |        | ;\$A033 to \$9021 |
| 84 |    |    | POP | @R4 |        | ;Move byte from   |
| 95 |    |    | STP | @R5 |        | ;\$A032 to \$9020 |

### ADD:

ADD Rn [An]

The contents of Rn are added to the contents of ACC (RO), and the low-order 16 bits of the sum restored in ACC. the 17th sum bit becomes the carry and the other branch conditions reflect the final ACC contents.

#### EXAMPLE:

| 10 34 76       | SET RO | \$7634 | ;Init RO (ACC) and R1       |
|----------------|--------|--------|-----------------------------|
| $11 \ 27 \ 42$ | SET R1 | \$4227 |                             |
| A1             | ADD R1 |        | ;Add R1 (sum=B85B, C clear) |
| AO             | ADD RO |        | ;Double ACC (RO) to \$70B6  |
|                |        |        | ;with carry set.            |

# SUBTRACT:

SUB Rn [Bn]

The contents of Rn are subtracted from the ACC contents by performing a two's complement addition:

ACC = ACC + Rn + 1

The low order 16 bits of the subtraction are restored in the ACC, the 17th sum bit becomes the carry and other branch conditions reflect the final ACC contents. If the 16-bit unsigned ACC contents are greater than or equal to the 16-bit unsigned Rn contents, then the carry is set, otherwise it is cleared. Rn is not disturbed.

### EXAMPLE:

| 10 34 76 | SET | RO | \$7634 | ;Init RO (ACC)            |
|----------|-----|----|--------|---------------------------|
| 11 27 42 | SET | R1 | \$4227 | ; and R1                  |
| B1       | SUB | R1 |        | ;subtract R1              |
|          |     |    |        | ;(diff=\$340D with c set) |
| BO       | SUB | RO |        | ;clears ACC. (RO)         |

#### POP DOUBLE-BYTE INDIRECT:

#### POPD @Rn [ Cn ]

Rn is decremented by 1 and the high-order ACC byte is loaded from the memory location whose address now resides in Rn. Rn is again decremented by 1 and the low-order ACC byte is loaded from the corresponding memory location. Branch conditions reflect the final ACC contents. The carry is cleared. Because Rn is decremented prior to loading each of the ACC halves, double-byte stacks may be implemented with the STD @Rn and POPD @Rn ops (Rn is the stack pointer).

EXAMPLE:

| 15 | 34 | AO | SET  | R5  | \$A034 | ;Init stack pointer     |
|----|----|----|------|-----|--------|-------------------------|
| 10 | 12 | AA | SET  | RO  | \$AA12 | ;Load \$AA12 into ACC   |
| 75 |    |    | STD  | @R5 |        | ;Push \$AA12 onto stack |
| 10 | 34 | BB | SET  | RO  | \$BB34 | ;Load \$BB34 into ACC   |
| 75 |    |    | STD  | @R5 |        | ;Push \$BB34 onto stack |
| C5 |    |    | POPD | @R5 |        | ;Pop \$BB34 off stack   |
| C5 |    |    | POPD | @R5 |        | ;Pop \$AA12 off stack   |

### COMPARE:

# CPR Rn [Dn]

The ACC (RO) contents are compared to Rn by performing the 16 bit binary subtraction ACC-Rn and storing the low order 16 difference bits in R13 for subsequent branch tests. If the 16 bit unsigned ACC contents are greater than or equal to the 16 bit unsigned Rn contents, then the carry is set, otherwise it is cleared. No other registers, including ACC and Rn, are disturbed.

### EXAMPLE:

| 15 34 AO |       | SET | R5 | \$A034 | ;Pointer to memory |
|----------|-------|-----|----|--------|--------------------|
| 16 BF AO |       | SET | R6 | \$A0BF | ;Limit address     |
| BO       | L00P1 | SUB | RO |        | ;Zero data         |

| 75    | STD | @R5   | ;clear 2 locations   |
|-------|-----|-------|----------------------|
|       |     |       | ;increment R5 by 2   |
| 25    | LD  | R5    | ; Compare pointer R5 |
| D6    | CPR | R6    | ;to l̃imit R6        |
| 02 FA | BNC | LOOP1 | ;loop if C clear     |
|       |     |       |                      |

# I NCREMENT:

INR Rn [En]

The contents of Rn are incremented by 1. The carry is cleared and other branch conditions reflect the incremented value.

#### EXAMPLE:

| 15 34 AO | SET  | R5  | \$A034 | ;(Pointer)            |
|----------|------|-----|--------|-----------------------|
| BO       | SUB  | RO  |        | ;Zero to RO           |
| 55       | ST   | @R5 |        | ;Clr Location \$A034  |
| E5       | I NR | R5  |        | ;Incr R5 to \$A036    |
| 55       | ST   | @R5 |        | ;Clrs location \$A036 |
|          |      |     |        | ;(not \$A035)         |

# DECREMENT:

DCR Rn [Fn]

The contents of Rn are decremented by 1. The carry is cleared and other branch conditions reflect the decremented value.

EXAMPLE: (Clear 9 bytes beginning at location A034)

| SET    | R5                          | \$A034                                   | ;Init pointer        |
|--------|-----------------------------|------------------------------------------|----------------------|
| SET    | R4                          | 9                                        | ;Init counter        |
| SUB    | RO                          |                                          | ;Zero ACC            |
| OP2 ST | @R5                         |                                          | ;Clear a mem byte    |
| DCR    | R4                          |                                          | ;Decrement count     |
| BNZ    | LOOP                        | 2                                        | ;Loop until Zero     |
|        | SET<br>SUB<br>OP2 ST<br>DCR | SET R4<br>SUB R0<br>OP2 ST @R5<br>DCR R4 | SUB RO<br>OP2 ST @R5 |

Non-Register Instructions:

RETURN TO 6502 MODE:

RTN 00

Control is returned to the 6502 and program execution continues at the location immediately following the RTN instruction. the 6502 registers and status conditions are restored to their original contents (prior to entering SWEET 16 mode).

BRANCH ALWAYS:

BR ea [ 01 d ]

An effective address (ea) is calculated by adding the signed displacement byte (d) to the PC. The PC contains the address of the instruction immediately following the BR, or the address of the BR op plus 2. The displacement is a signed two's complement value from -128 to +127. Branch conditions are not changed.

NOTE: The effective address calculation is identical to that for 6502 relative branches. The Hex add & Subtract features of the APPLE ][ monitor may be used to calculate displacements.

\$300: 01 50 BR \$352

BRANCH IF NO CARRY:

BNC ea [ 02 d ]

A branch to the effective address is taken only is the carry is clear, otherwise execution resumes as normal with the next instruction. Branch conditions are not changed.

BRANCH IF CARRY SET:

BC ea [ 03 d ]

A branch is effected only if the carry is set. Branch conditions are not changed.

BRANCH IF PLUS:

BP ea [ 04 d ]

A branch is effected only if the prior 'result' (or most recently transferred dat) was positive. Branch conditions are not changed.

EXAMPLE: (Clear mem from AO34 to AO3F)

| 15 | 34 AO | SET | R5 | \$A034 | ;Init pointer |
|----|-------|-----|----|--------|---------------|
| 14 | 3F A0 | SET | R4 | \$A03F | ;Init limit   |

| BO    | LOOP3 | SUB | RO    |                  |
|-------|-------|-----|-------|------------------|
| 55    |       | ST  | @R5   | ;Clear mem byte  |
|       |       |     |       | ;Increment R5    |
| 24    |       | LD  | R4    | ;Compare limit   |
| D5    |       | CPR | R5    | ; to pointer     |
| 04 FA |       | BP  | LOOP3 | ;Loop until done |

BRANCH IF MINUS:

BM ea [ 05 d ]

A branch is effected only if prior 'result' was minus (negative, MSB = 1). Branch conditions are not changed.

BRANCH IF ZERO:

BZ ea [ 06 d ]

A Branch is effected only if the prior 'result' was zero. Branch conditions are not changed.

# BRANCH IF NONZERO

BNZ ea [ 07 d ]

A branch is effected only if the priot 'result' was non-zero Branch conditions are not changed.

BRANCH IF MINUS ONE

BM1 ea [ 08 d ]

A branch is effected only if the prior 'result' was minus one (\$FFFF Hex). Branch conditions are not changed.

BRANCH IF NOT MINUS ONE

BNM1 ea [ 09 d ]

A branch effected only if the prior 'result' was not minus 1. Branch conditions are not changed.

# BREAK:

BK [ OA ]

A 6502 BRK (break) instruction is executed. SWEET 16 may be re-entered non destructively at SW16d after correcting the stack pointer to its value prior to executing the BRK.

# RETURN FROM SWEET 16 SUBROUTINE:

RS [ OB ]

RS terminates execution of a SWEET 16 subroutine and returns to the SWEET 16 calling program which resumes execution (in SWEET 16 mode). R12, which is the SWEET 16 subroutine return stack pointer, is decremented twice. Branch conditions are not changed.

BRANCH TO SWEET 16 SUBROUTINE:

BS ea [ 0c d ]

A branch to the effective address (PC + 2 + d) is taken and execution is resumed in SWEET 16 mode. The current PC is pushed onto a SWEET 16 subroutine return address stack whose pointer is R12, and R12 is incremented by 2. The carry is cleared and branch conditions set to indicate the current ACC contents.

EXAMPLE: (Calling a 'memory move' subroutine to move A034-A03B to 3000-3007)

| 15 34 A0<br>14 3B A0<br>16 00 30<br>0C 15<br>45<br>56<br>24<br>D5<br>04 E4 | MOVE | ST<br>LD<br>CPR | R5<br>R4<br>R6<br>MOVE<br>@R5<br>@R6<br>R4<br>R5 | \$A034<br>\$A03B<br>\$3000 | ;Init pointer 1<br>;Init limit 1<br>;Init pointer 2<br>;Call move subroutine<br>;Move one<br>;byte<br>;Test if done |
|----------------------------------------------------------------------------|------|-----------------|--------------------------------------------------|----------------------------|---------------------------------------------------------------------------------------------------------------------|
| O4 FA<br>OB                                                                |      | BP<br>RS        | MOVE                                             |                            | ;Return                                                                                                             |

Theory of Operation:

SWEET 16 execution mode begins with a subroutine call to SW16. All 6502 registers are saved at this time, to be restored when a SWEET 16 RTN instruction returns control to the 6502. If you can tolerate indefinate 6502 register contents upon exit, approximately 30 usec may be saved by entering at SW16 + 3. Because this might cause an inadvertant switch from Hex to Decimal mode, it is advisable to enter at SW16 the first time through.

After saving the 6502 registers, SWEET 16 initializes its PC (R15) with the subroutine return address off the 6502 stack. SWEET 16's PC points to the location preceding the next instruction to be executed. Following the subroutine call are 1-, 2-, and 3-byte SWEET 16 instructions, stored in ascending memory like 6502 instructions. the main loop at SW16B repeatedly calls the 'execute instruction' routine to execute it.

Subroutine SW16C increments the PC (R15) and fetches the next opcode, which is either a register op of the form OP REG with OP between 1

and 15 or a non-register op of the form 0 OP with OP between 0 and 13. Assuming a register op, the register specification is doubled to account for the 3 byte SWEET 16 registers and placed in the X-reg for indexing. Then the instruction type is determined. Register ops place the doubled register specification in the high order byte of R14 indicating the 'prior result register' to subsequent branch instructions. Non-register ops treat the register specification (right-hand half-byte) as their opcode, increment the SWEET 16 PC to point at the displacement byte of branch instructions, load the A-reg with the 'prior result register' index for branch condition testing, and clear the Y-reg.

When is an RTS really a JSR?

Each instruction type has a corresponding subroutine. The subroutine entry points are stored in a table which is directly indexed into by the opcode. By assigning all the entries to a common page, only a single byte to address need be stored per routine. The 6502 indirect jump might have been used as follows to transfer control to the appropriate subroutine.

| LDA | #ADRH    | ;High-order byte. |
|-----|----------|-------------------|
| STA | I ND+1   |                   |
| LDA | OPTBL, X | ;Low-order byte.  |
| STA | I ND     | <sup>0</sup>      |
| JMP | (IND)    |                   |

To save code, the subroutine entry address (minus 1) is pushed onto the stack, high-order byte first. A 6502 RTS (return from subroutine) is used to pop the address off the stack and into the 6502 PC (after incrementing by 1). The net result is that the desired subroutine is reached by executing a subroutine return instruction!

Opcode Subroutines:

The register op routines make use of the 6502 'zero page indexed by X' and 'indexed by X direct' addressing modes to access the specified registers and indirect data. The 'result' of most register ops is left in the specified register and can be sensed by subsequent branch instructions, since the register specification is saved in the high-order byte of R14. This specification is changed to indicate R0 (ACC) for ADD and SUB instructions and R13 for the CPR (compare) instruction.

Normally the high-order R14 byte holds the 'prior result register' index times 2 to account for the 2-byte SWEET 16 registers and the LSB is zero. If ADD, SUB, or CPR instructions generate carries, then this index is incremented, setting the LSB.

The SET instruction increments the PC twice, picking up data bytes in the specified register. In accordance with 6502 convention, the low-order data byte precedes the high-order byte.

Most SWEET 16 non-register ops are relative branches. The corresponding

subroutines determine whether or not the 'prior result' meets the specified branch condition and if so, update the SWEET 16 PC by adding the displacement value (-128 to +127 bytes).

The RTN op restores the 6502 register contents, pops the subroutine return stack and jumps indirect through the SWEET 16 PC. This transfers control to the 6502 at the instruction immediately following the RTN instruction.

The BK op actually executes a 6502 break instruction (BRK), transferring control to the interrupt handler.

Any number of subroutine levels may be implemented within SWEET 16 code via the BS (Branch to Subroutine) and RS (Return from Subroutine) instructions. The user must initialize and otherwise not disturb R12 if the SWEET 16 subroutine capability is used since it is utilized as the automatic return stack pointer.

Memory Allocation:

The only storage that must be allocated for SWEET 16 variables are 32 consecutive locations in page zero for the SWEET 16 registers, four locations to save the 6502 register contents, and a few levels of the 6502 subroutine return address stack. if you don't need to preserve the 6502 register contents, delete the SAVE and RESTORE subroutines and the corresponding subroutine calls. This will free the four page zero locations ASAV, XSAV, YSAV, and PSAV.

User Modifications:

You may wish to add some of your own instructions to this implementation of SWEET 16. If you use the unassigned opcodes SOE and SOF, remember that SWEET 16 treats these as 2-byte instructions. You may wish to handle the break instruction as a SWEET 16 call, saving two bytes of code each time you transfer into SWEET 16 mode. Or you may wish to use the SWEET 16 BK (break) op as a 'CHAROUT' call in the interrupt handler. You can perform absolute jumps within SWEET 16 by loading the ACC (RO) with the address you wish to jump to (minus 1) and executing a ST R15 instruction.

- -

| W. Sheldon Simms   | Newt's Friend / Jack Kemp for President |
|--------------------|-----------------------------------------|
| sheldon@netcom.com | Freedom implies responsibility          |
|                    |                                         |