

# MEGA 65

## USER'S GUIDE



MEGA 65

MUSEUM OF ELECTRONIC GAMES & ART



# **REGULATORY INFORMATION**

The MEGA65 home computer and portable computer have not been subject to FCC, EC or other regulatory approvals as of the time of writing.

# MEGA65 TEAM

## **Dr. Paul Gardner-Stephen**

(highlander)

Founder

Software and Virtual Hardware Architect

Spokesman and Lead Scientist

## **Detlef Hastik**

(deft)

Co-Founder

General Manager

Marketing & Sales

## **Martin Streit**

(seriously)

Video and Photo Production

Tax and Organization

Social Media

## **Anton Schneider-Michallek**

(adtbm)

Hardware Pool Management

Soft-, Hard- and V-Hardware Testing

Forum Administration

## **Falk Rehwagen**

(bluewaysw)

Jenkins Build Automation

GEOS, Hardware Quality Management

## **Antti Lukats**

(antti-brain)

Host Hardware Design and Production

## **Dieter Penner**

(doubleflash)

Host Hardware Review and Testing

File Hosting

## **Dr. Edilbert Kirk**

(Bit Shifter)

Manual and Tools

ROM Enhancements

## **Gábor Lénárt**

(LGB)

Emulator

## **Mirko H.**

(sy2002)

Additional Hardware and Platforms

## **Farai Aschwanden**

(Tayger)

File Base, Tools

Financial Advisory

## **Thomas Hertzler**

(grumpyninja)

USA Spokesman

Artist Relations

## **Andrew Owen**

(Cheveron)

Keyboard Advisory, Sinclair Support

## **Daniel England**

(Mew Pokémon)

Additional Code and Tools

## **Roman Standzikowski**

(FeralChild)

Open Roms

## **Hernán di Pietro**

(indiocolifa)

Additional Emulation

# Reporting Errors and Omissions

This book is a work-in-progress produced by and for the MEGA65 community. The version of this edition is:

```
commit 4f7464a5fe12d9f8657718cb0544cec4d9f86674
Date: Thu Aug 5 20:13:58 2021 +0930
```

We want this book to be the best that it possibly can. So if you see any errors, or find anything that is missing, or that you would like more information on, please report them using the MEGA65 User Guide issue tracker:

<https://github.com/mega65/mega65-user-guide/issues>

You can also check there to see if anyone else has reported a similar problem, while you wait for this book to be updated.

Finally, you can always download the latest version of this book from:

<https://github.com/mega65/mega65-user-guide>



# MEGA65 REFERENCE GUIDE

Published by  
the MEGA Museum of Electronic Games & Art e.V., Germany.

## WORK IN PROGRESS

Copyright ©2019 – 2021 by Paul Gardner-Stephen, the MEGA Museum of Electronic Games & Art e.V., and contributors.

This reference guide is made available under the GNU Free Documentation License v1.3, or later, if desired. This means that you are free to modify, reproduce and redistribute this user guide, subject to certain conditions. The full text of the GNU Free Documentation License v1.3 can be found at <https://www.gnu.org/licenses/fdl-1.3.en.html>.

Implicit in this copyright license, is the permission to duplicate and/or redistribute this document in whole or in part for use in education environments. We want to support the education of future generations, so if you have any worries or concerns, please contact us.

August 8, 2021

# Contents

|          |                                                              |               |
|----------|--------------------------------------------------------------|---------------|
| <b>1</b> | <b>Introduction</b>                                          | <b>xxxiii</b> |
|          | Other Books in this series . . . . .                         | xxxvi         |
| <br>     |                                                              |               |
| <b>I</b> | <b>GETTING TO KNOW YOUR MEGA65</b>                           | <b>1-1</b>    |
| <b>2</b> | <b>SETUP</b>                                                 | <b>2-1</b>    |
|          | Unpacking and connecting the MEGA65 . . . . .                | 2-3           |
|          | Rear Connections . . . . .                                   | 2-4           |
|          | Side Connections . . . . .                                   | 2-5           |
|          | Installation . . . . .                                       | 2-6           |
|          | Connecting your MEGA65 to a screen and peripherals . . . . . | 2-6           |
|          | Optional Connections . . . . .                               | 2-7           |
|          | Operation . . . . .                                          | 2-7           |
|          | Using the MEGA65 . . . . .                                   | 2-7           |
|          | THE CURSOR . . . . .                                         | 2-8           |
| <b>3</b> | <b>GETTING STARTED</b>                                       | <b>3-1</b>    |
|          | Keyboard . . . . .                                           | 3-3           |
|          | Command Keys . . . . .                                       | 3-3           |
|          | RETURN . . . . .                                             | 3-3           |

|                                |     |
|--------------------------------|-----|
| SHIFT . . . . .                | 3-3 |
| SHIFT LOCK . . . . .           | 3-3 |
| CTRL . . . . .                 | 3-4 |
| RUN/STOP . . . . .             | 3-4 |
| RESTORE . . . . .              | 3-4 |
| THE CURSOR KEYS . . . . .      | 3-5 |
| INSerT/DElete . . . . .        | 3-5 |
| CLeaR/HOME . . . . .           | 3-5 |
| MEGA KEY . . . . .             | 3-5 |
| NO SCROLL . . . . .            | 3-6 |
| Function Keys . . . . .        | 3-6 |
| HELP . . . . .                 | 3-6 |
| ALT . . . . .                  | 3-6 |
| CAPS LOCK . . . . .            | 3-6 |
| The Screen Editor . . . . .    | 3-7 |
| Editor Functionality . . . . . | 3-9 |

|                                                        |            |
|--------------------------------------------------------|------------|
| <b>4 Configuring your MEGA65</b>                       | <b>4-1</b> |
| Preparing for First Use: Formatting SD Cards . . . . . | 4-3        |
| Installing ROM and Other Support Files . . . . .       | 4-7        |
| ROM File . . . . .                                     | 4-8        |
| Support Files . . . . .                                | 4-9        |
| On-boarding . . . . .                                  | 4-10       |
| Configuration Utility . . . . .                        | 4-13       |
| Input Devices . . . . .                                | 4-18       |
| Chipset . . . . .                                      | 4-19       |
| Video . . . . .                                        | 4-20       |
| Audio . . . . .                                        | 4-21       |
| Network . . . . .                                      | 4-22       |

|                                                     |             |
|-----------------------------------------------------|-------------|
| <b>5 Cores and Flashing</b>                         | <b>5-1</b>  |
| What are cores, and why does it matter? . . . . .   | 5-5         |
| Bitstream files . . . . .                           | 5-5         |
| File types . . . . .                                | 5-6         |
| Where to download . . . . .                         | 5-6         |
| Selecting a core . . . . .                          | 5-7         |
| Installing an upgrade core for the MEGA65 . . . . . | 5-8         |
| Installing other cores . . . . .                    | 5-11        |
| Creating cores for the MEGA65 . . . . .             | 5-12        |
| Replacing the factory core in slot 0 . . . . .      | 5-12        |
| Understanding The Core Booting Process . . . . .    | 5-12        |
| <b>II FIRST STEPS IN CODING</b>                     | <b>5-15</b> |
| <b>6 How Computers Work</b>                         | <b>6-1</b>  |
| Computers are stupid. Really stupid . . . . .       | 6-3         |
| Making an Egg Cup Computer . . . . .                | 6-3         |
| <b>7 Getting Started in BASIC</b>                   | <b>7-1</b>  |
| Your first BASIC programmes . . . . .               | 7-3         |
| Exercises to try . . . . .                          | 7-12        |
| First steps with text and numbers . . . . .         | 7-13        |
| Exercises to try . . . . .                          | 7-29        |
| Making simple decisions . . . . .                   | 7-30        |
| Exercises to try . . . . .                          | 7-43        |
| Random numbers and chance . . . . .                 | 7-44        |
| Exercises to try . . . . .                          | 7-48        |

|                                                                         |            |
|-------------------------------------------------------------------------|------------|
| <b>8 Text Processing</b>                                                | <b>8-1</b> |
| Characters and Strings . . . . .                                        | 8-3        |
| String Literals . . . . .                                               | 8-5        |
| String Variables . . . . .                                              | 8-5        |
| String Statements . . . . .                                             | 8-7        |
| Simple Formatting . . . . .                                             | 8-8        |
| Suppressing New Lines . . . . .                                         | 8-8        |
| Automatic Tab Stops . . . . .                                           | 8-8        |
| Tabs Stops and Spacing . . . . .                                        | 8-8        |
| Sample Programs . . . . .                                               | 8-9        |
| Palindromes . . . . .                                                   | 8-9        |
| Simple Ciphers . . . . .                                                | 8-9        |
| <b>9 C64, C65 and MEGA65 Modes</b>                                      | <b>9-1</b> |
| Switching Modes from BASIC . . . . .                                    | 9-3        |
| From C65 to C64 Mode . . . . .                                          | 9-3        |
| From C64 to C65 Mode . . . . .                                          | 9-3        |
| Entering Machine Code Monitor Mode . . . . .                            | 9-4        |
| The KEY Register . . . . .                                              | 9-4        |
| Un-hiding C65 Extra Registers . . . . .                                 | 9-4        |
| Re-hiding C65/MEGA65 Extra Registers . . . . .                          | 9-5        |
| Un-hiding MEGA65 Extra Registers . . . . .                              | 9-6        |
| Traps to watch out for . . . . .                                        | 9-7        |
| Accessing the MEGA65's Extra Memory from BASIC 65 in C65 Mode . . . . . | 9-7        |
| The MAP Instruction . . . . .                                           | 9-8        |

|                                                        |              |
|--------------------------------------------------------|--------------|
| <b>III SOUND AND GRAPHICS</b>                          | <b>9-9</b>   |
| <b>IV HARDWARE</b>                                     | <b>9-11</b>  |
| <b>10 Using Nexys4 boards as a MEGA65</b>              | <b>10-1</b>  |
| Building your own MEGA65 Compatible Computer . . . . . | 10-3         |
| Working Nexys4 Boards . . . . .                        | 10-4         |
| The Nexys4 board . . . . .                             | 10-4         |
| The Nexys4DDR board . . . . .                          | 10-4         |
| The Nexys A7 . . . . .                                 | 10-5         |
| Power, Jumpers, Switches and Buttons . . . . .         | 10-6         |
| Micro-USB Power . . . . .                              | 10-7         |
| External Power Supply . . . . .                        | 10-7         |
| Other Jumpers and Switches . . . . .                   | 10-8         |
| Connections and Peripherals . . . . .                  | 10-9         |
| Communicating with your PC . . . . .                   | 10-9         |
| Onboard buttons . . . . .                              | 10-10        |
| Keyboard . . . . .                                     | 10-10        |
| Some key mappings with a USB keyboard . . . . .        | 10-11        |
| Preparing microSDHC card . . . . .                     | 10-12        |
| Preparation Steps . . . . .                            | 10-12        |
| Loading the bitstream from QSPI . . . . .              | 10-14        |
| Preparation Steps . . . . .                            | 10-14        |
| Useful Tips . . . . .                                  | 10-14        |
| <b>V CROSS-PLATFORM DEVELOPMENT TOOLS</b>              | <b>10-17</b> |
| <b>11 Emulators</b>                                    | <b>11-1</b>  |
| Using The Xmega65 Emulator . . . . .                   | 11-3         |

|                                                                |             |
|----------------------------------------------------------------|-------------|
| Using the Live ISO image . . . . .                             | 11-3        |
| Creating a Bootable USB stick or DVD . . . . .                 | 11-3        |
| Getting Started . . . . .                                      | 11-4        |
| Other Features of the Live ISO . . . . .                       | 11-5        |
| <b>12 Data Transfer and Debugging Tools</b>                    | <b>12-1</b> |
| m65 command line tool . . . . .                                | 12-3        |
| Screenshots using m65 tool . . . . .                           | 12-3        |
| Load and run a programme on the MEGA65 . . . . .               | 12-4        |
| Reconfigure the FPGA to run a different bitstream . . . . .    | 12-4        |
| Remote keyboard entry . . . . .                                | 12-4        |
| Unit testing and logging support . . . . .                     | 12-5        |
| Using unit tests with C . . . . .                              | 12-6        |
| Using unit tests with BASIC65 . . . . .                        | 12-6        |
| BASIC65 example . . . . .                                      | 12-8        |
| M65Connect . . . . .                                           | 12-8        |
| mega65_ftp . . . . .                                           | 12-9        |
| TFTP Server . . . . .                                          | 12-10       |
| Converting a BASIC text file listing into a PRG file . . . . . | 12-10       |
| <b>13 Assemblers</b>                                           | <b>13-1</b> |
| <b>14 C and C-Like Compilers</b>                               | <b>14-1</b> |
| MEGA65 libc . . . . .                                          | 14-3        |
| <b>15 MEGA65 Standard C Library</b>                            | <b>15-1</b> |
| Structure and Usage . . . . .                                  | 15-3        |
| conio.h . . . . .                                              | 15-3        |
| cononxit . . . . .                                             | 15-3        |
| setscreenaddr . . . . .                                        | 15-4        |

|                            |       |
|----------------------------|-------|
| getscreenaddr . . . . .    | 15-4  |
| setcolramoffset . . . . .  | 15-4  |
| getcolramoffset . . . . .  | 15-4  |
| setcharsetaddr . . . . .   | 15-5  |
| getcharsetaddr . . . . .   | 15-5  |
| clrscr . . . . .           | 15-5  |
| getscreensize . . . . .    | 15-5  |
| setscreensize . . . . .    | 15-6  |
| set16bitcharmode . . . . . | 15-6  |
| sethotregs . . . . .       | 15-6  |
| setextendedattr . . . . .  | 15-7  |
| togglecase . . . . .       | 15-7  |
| bordercolor . . . . .      | 15-7  |
| bgcolor . . . . .          | 15-7  |
| textcolor . . . . .        | 15-8  |
| revers . . . . .           | 15-8  |
| highlight . . . . .        | 15-8  |
| blink . . . . .            | 15-9  |
| underline . . . . .        | 15-9  |
| cellcolor . . . . .        | 15-9  |
| fillrect . . . . .         | 15-10 |
| box . . . . .              | 15-10 |
| hline . . . . .            | 15-11 |
| vline . . . . .            | 15-11 |
| gohome . . . . .           | 15-12 |
| gotoxy . . . . .           | 15-12 |
| gotox . . . . .            | 15-12 |
| gotoy . . . . .            | 15-12 |

|                |       |
|----------------|-------|
| moveup         | 15-13 |
| movedown       | 15-13 |
| moveleft       | 15-13 |
| moveright      | 15-14 |
| wherex         | 15-14 |
| wherey         | 15-14 |
| cputc          | 15-14 |
| cputnc         | 15-15 |
| cputhex        | 15-15 |
| cputdec        | 15-15 |
| cputs          | 15-16 |
| cputsxxy       | 15-16 |
| cputexy        | 15-16 |
| cputncxy       | 15-17 |
| cprintf        | 15-17 |
| cgetc          | 15-18 |
| kbhit          | 15-18 |
| getkeymodstate | 15-18 |
| flushkeybuf    | 15-19 |
| cinput         | 15-19 |
| VIC_BASE       | 15-20 |

## 16 BASIC Tokenisers

16-1

## VI APPENDICES

16-5

### A ACCESSORIES

A-1

**B BASIC 65 Command Reference**

B-1

|                                                       |      |
|-------------------------------------------------------|------|
| Format of Commands, Functions and Operators . . . . . | B-3  |
| Commands, Functions and Operators . . . . .           | B-5  |
| BASIC 65 constants . . . . .                          | B-6  |
| BASIC 65 variables . . . . .                          | B-6  |
| BASIC 65 arrays . . . . .                             | B-6  |
| BASIC command reference . . . . .                     | B-8  |
| ABS . . . . .                                         | B-9  |
| AND . . . . .                                         | B-10 |
| APPEND . . . . .                                      | B-11 |
| ASC . . . . .                                         | B-12 |
| ATN . . . . .                                         | B-13 |
| AUTO . . . . .                                        | B-14 |
| BACKGROUND . . . . .                                  | B-15 |
| BACKUP . . . . .                                      | B-16 |
| BANK . . . . .                                        | B-17 |
| BEGIN . . . . .                                       | B-18 |
| BEND . . . . .                                        | B-19 |
| BLOAD . . . . .                                       | B-20 |
| BOOT . . . . .                                        | B-22 |
| BORDER . . . . .                                      | B-23 |
| BOX . . . . .                                         | B-24 |
| BSAVE . . . . .                                       | B-25 |
| BUMP . . . . .                                        | B-26 |
| BVERIFY . . . . .                                     | B-27 |
| CATALOG . . . . .                                     | B-28 |
| CHANGE . . . . .                                      | B-30 |
| CHAR . . . . .                                        | B-31 |

|           |      |
|-----------|------|
| CHR\$     | B-33 |
| CIRCLE    | B-34 |
| CLOSE     | B-36 |
| CLR       | B-37 |
| CMD       | B-38 |
| COLLECT   | B-39 |
| COLLISION | B-40 |
| COLOR     | B-41 |
| CONCAT    | B-42 |
| CONT      | B-43 |
| COPY      | B-44 |
| COS       | B-46 |
| CURSOR    | B-47 |
| DATA      | B-48 |
| DCLEAR    | B-49 |
| DCLOSE    | B-50 |
| DEC       | B-51 |
| DEF FN    | B-52 |
| DELETE    | B-53 |
| DIM       | B-54 |
| DIR       | B-55 |
| DISK      | B-57 |
| DLOAD     | B-58 |
| DMA       | B-59 |
| DMODE     | B-60 |
| DO        | B-61 |
| DOPEN     | B-62 |
| DPAT      | B-64 |

|            |      |
|------------|------|
| DS         | B-65 |
| DS\$       | B-66 |
| DSAVE      | B-67 |
| DT\$       | B-68 |
| DVERIFY    | B-69 |
| EDIT       | B-70 |
| EDMA       | B-72 |
| EL         | B-73 |
| ELLIPSE    | B-74 |
| ELSE       | B-76 |
| END        | B-77 |
| ENVELOPE   | B-78 |
| ERASE      | B-79 |
| ER         | B-80 |
| ERR\$      | B-81 |
| EXIT       | B-82 |
| EXP        | B-83 |
| FAST       | B-84 |
| FILTER     | B-85 |
| FIND       | B-86 |
| FN         | B-87 |
| FONT       | B-88 |
| FOR        | B-89 |
| FOREGROUND | B-90 |
| FRE        | B-91 |
| FREAD      | B-92 |
| FWRITE     | B-93 |
| GET        | B-94 |

|             |       |
|-------------|-------|
| GET#        | B-95  |
| GETKEY      | B-96  |
| GO64        | B-97  |
| GOSUB       | B-98  |
| GOTO        | B-99  |
| GRAPHIC     | B-100 |
| HEADER      | B-101 |
| HELP        | B-102 |
| HEX\$       | B-103 |
| HIGHLIGHT   | B-104 |
| IF          | B-105 |
| INPUT       | B-106 |
| INPUT#      | B-107 |
| INSTR       | B-109 |
| INT         | B-110 |
| JOY         | B-111 |
| KEY         | B-112 |
| LEFT\$      | B-114 |
| LEN         | B-115 |
| LET         | B-116 |
| LINE        | B-117 |
| LINE INPUT# | B-118 |
| LIST        | B-119 |
| LOAD        | B-120 |
| LOADIFF     | B-121 |
| LOG         | B-122 |
| LOG10       | B-123 |
| LOOP        | B-124 |

|                   |       |
|-------------------|-------|
| LPEN . . . . .    | B-125 |
| MERGE . . . . .   | B-126 |
| MID\$ . . . . .   | B-127 |
| MOD . . . . .     | B-128 |
| MONITOR . . . . . | B-129 |
| MOUSE . . . . .   | B-130 |
| MOVSPR . . . . .  | B-131 |
| NEW . . . . .     | B-133 |
| NEXT . . . . .    | B-134 |
| NOT . . . . .     | B-135 |
| OFF . . . . .     | B-136 |
| ON . . . . .      | B-137 |
| OPEN . . . . .    | B-139 |
| OR . . . . .      | B-140 |
| PAINT . . . . .   | B-141 |
| PALETTE . . . . . | B-142 |
| PEEK . . . . .    | B-143 |
| PEEKW . . . . .   | B-144 |
| PEN . . . . .     | B-145 |
| PIXEL . . . . .   | B-146 |
| PLAY . . . . .    | B-147 |
| POINTER . . . . . | B-149 |
| POKE . . . . .    | B-150 |
| POKEW . . . . .   | B-151 |
| POLYGON . . . . . | B-152 |
| POS . . . . .     | B-153 |
| POT . . . . .     | B-154 |
| PRINT . . . . .   | B-155 |

|                       |       |
|-----------------------|-------|
| PRINT# . . . . .      | B-156 |
| PRINT USING . . . . . | B-157 |
| RCOLOR . . . . .      | B-159 |
| RCURSOR . . . . .     | B-160 |
| READ . . . . .        | B-161 |
| RECORD . . . . .      | B-162 |
| REM . . . . .         | B-163 |
| RENAME . . . . .      | B-164 |
| RENUMBER . . . . .    | B-165 |
| RESTORE . . . . .     | B-166 |
| RESUME . . . . .      | B-167 |
| RETURN . . . . .      | B-168 |
| RGRAPHIC . . . . .    | B-169 |
| RIGHT\$ . . . . .     | B-170 |
| RMOUSE . . . . .      | B-171 |
| RND . . . . .         | B-172 |
| RPEN . . . . .        | B-173 |
| RREG . . . . .        | B-174 |
| RSPCOLOR . . . . .    | B-175 |
| RSPPOS . . . . .      | B-176 |
| RSprite . . . . .     | B-177 |
| RUN . . . . .         | B-178 |
| RWINDOW . . . . .     | B-179 |
| SAVE . . . . .        | B-180 |
| SCNCLR . . . . .      | B-181 |
| SCRATCH . . . . .     | B-182 |
| SCREEN . . . . .      | B-183 |
| SET . . . . .         | B-186 |

|                    |       |
|--------------------|-------|
| SGN . . . . .      | B-187 |
| SIN . . . . .      | B-188 |
| SLEEP . . . . .    | B-189 |
| SLOW . . . . .     | B-190 |
| SOUND . . . . .    | B-191 |
| SPC . . . . .      | B-192 |
| SPRCOLOR . . . . . | B-193 |
| SPRITE . . . . .   | B-194 |
| SPRSAV . . . . .   | B-195 |
| SQR . . . . .      | B-196 |
| ST . . . . .       | B-197 |
| STEP . . . . .     | B-198 |
| STOP . . . . .     | B-199 |
| STR\$ . . . . .    | B-200 |
| SYS . . . . .      | B-201 |
| TAB . . . . .      | B-203 |
| TAN . . . . .      | B-204 |
| TEMPO . . . . .    | B-205 |
| THEN . . . . .     | B-206 |
| TI . . . . .       | B-207 |
| TI\$ . . . . .     | B-208 |
| TO . . . . .       | B-209 |
| TRAP . . . . .     | B-210 |
| TROFF . . . . .    | B-211 |
| TRON . . . . .     | B-212 |
| TYPE . . . . .     | B-213 |
| UNTIL . . . . .    | B-214 |
| USING . . . . .    | B-215 |

|                                                      |       |
|------------------------------------------------------|-------|
| USR . . . . .                                        | B-217 |
| VAL . . . . .                                        | B-218 |
| VERIFY . . . . .                                     | B-219 |
| VIEWPORT . . . . .                                   | B-220 |
| VOL . . . . .                                        | B-221 |
| WAIT . . . . .                                       | B-222 |
| WHILE . . . . .                                      | B-223 |
| WINDOW . . . . .                                     | B-224 |
| XOR . . . . .                                        | B-225 |
| <br><b>C Special Keyboard Controls and Sequences</b> |       |
| PETSCII Codes and CHR\$ . . . . .                    | C-3   |
| Control codes . . . . .                              | C-5   |
| Shifted codes . . . . .                              | C-7   |
| Escape Sequences . . . . .                           | C-8   |
| <br><b>D The MEGA65 Keyboard</b>                     |       |
| Hardware Accelerated Keyboard Scanning . . . . .     | D-3   |
| Latin-1 Keyboard Map . . . . .                       | D-4   |
| Keyboard Theory of Operation . . . . .               | D-4   |
| C65 Keyboard Matrix . . . . .                        | D-4   |
| Synthetic Key Events . . . . .                       | D-5   |
| Keyboard LED Control . . . . .                       | D-6   |
| Native Keyboard Matrix . . . . .                     | D-7   |
| <br><b>E Decimal, Binary and Hexadecimal</b>         |       |
| Numbers . . . . .                                    | E-3   |
| Notations and Bases . . . . .                        | E-4   |
| Decimal . . . . .                                    | E-6   |
| Binary . . . . .                                     | E-7   |

|                                                           |             |
|-----------------------------------------------------------|-------------|
| Hexadecimal . . . . .                                     | E-9         |
| <b>Operations . . . . .</b>                               | <b>E-11</b> |
| Counting . . . . .                                        | E-11        |
| Arithmetic . . . . .                                      | E-13        |
| Logic Gates . . . . .                                     | E-15        |
| Signed and Unsigned Numbers . . . . .                     | E-17        |
| Bit-wise Logical Operators . . . . .                      | E-18        |
| Converting Numbers . . . . .                              | E-20        |
| <b>F System Memory Map . . . . .</b>                      | <b>F-1</b>  |
| Introduction . . . . .                                    | F-3         |
| MEGA65 Native Memory Map . . . . .                        | F-4         |
| The First Sixteen 64KB Banks . . . . .                    | F-4         |
| Colour RAM . . . . .                                      | F-5         |
| 28-bit Address Space . . . . .                            | F-6         |
| \$D000 - \$FFFF IO Personalities . . . . .                | F-8         |
| CPU Memory Banking . . . . .                              | F-10        |
| C64/C65 ROM Emulation . . . . .                           | F-11        |
| C65 Compatibility ROM Layout . . . . .                    | F-12        |
| <b>G 45GS02 Microprocessor . . . . .</b>                  | <b>G-1</b>  |
| Introduction . . . . .                                    | G-3         |
| Differences to the 6502 . . . . .                         | G-3         |
| Supervisor/Hypervisor Privileged Mode . . . . .           | G-3         |
| 6502 Illegal Opcodes . . . . .                            | G-4         |
| Read-Modify-Write Instruction Bug Compatibility . . . . . | G-4         |
| Variable CPU Speed . . . . .                              | G-5         |
| Slow (1MHz - 3.5MHz) Operation . . . . .                  | G-5         |
| Full Speed (40MHz) Instruction Timing . . . . .           | G-5         |

|                                                            |      |
|------------------------------------------------------------|------|
| CPU Speed Fine-Tuning . . . . .                            | G-6  |
| Direct Memory Access (DMA) . . . . .                       | G-6  |
| Accessing memory between the 64KB and 1MB points . . . . . | G-6  |
| C64-Style Memory Banking . . . . .                         | G-6  |
| VIC-III "ROM" Banking . . . . .                            | G-7  |
| VIC-III Display Address Translator . . . . .               | G-7  |
| The MAP instruction . . . . .                              | G-8  |
| Direct Memory Access (DMA) Controller . . . . .            | G-10 |
| Flat Memory Access . . . . .                               | G-10 |
| Accessing memory beyond the 1MB point . . . . .            | G-11 |
| Using the MAP instruction to access >1MB . . . . .         | G-11 |
| Flat-Memory Access . . . . .                               | G-13 |
| Virtual 32-bit Register . . . . .                          | G-13 |
| C64 CPU Memory Mapped Registers . . . . .                  | G-16 |
| New CPU Memory Mapped Registers . . . . .                  | G-16 |
| MEGA65 CPU Maths Acceleration Registers . . . . .          | G-21 |
| MEGA65 Hypervisor Mode . . . . .                           | G-29 |
| Reset . . . . .                                            | G-29 |
| Entering / Exiting Hypervisor Mode . . . . .               | G-29 |
| Hypervisor Memory Layout . . . . .                         | G-30 |
| Hypervisor Virtualisation Control Registers . . . . .      | G-33 |
| Programming for Hypervisor Mode . . . . .                  | G-36 |

## H 45GS02 & 6502 Instruction Sets

H-1

|                             |     |
|-----------------------------|-----|
| Addressing Modes . . . . .  | H-3 |
| Implied . . . . .           | H-3 |
| Accumulator . . . . .       | H-3 |
| Q Pseudo Register . . . . . | H-4 |
| Immediate Mode . . . . .    | H-4 |

|                                                         |      |
|---------------------------------------------------------|------|
| Immediate Word Mode . . . . .                           | H-4  |
| Base Page (Zero-Page) Mode . . . . .                    | H-5  |
| Base Page (Zero-Page) Quad Mode . . . . .               | H-5  |
| Base Page (Zero-Page) X Indexed Mode . . . . .          | H-5  |
| Base Page (Zero-Page) Quad X Indexed Mode . . . . .     | H-6  |
| Base Page (Zero-Page) Y Indexed Mode . . . . .          | H-6  |
| Base Page (Zero-Page) Base Y Indexed Mode . . . . .     | H-6  |
| Base Page (Zero-Page) Z Indexed Mode . . . . .          | H-6  |
| Base Page (Zero-Page) Quad Z Indexed Mode . . . . .     | H-7  |
| Absolute Mode . . . . .                                 | H-7  |
| Absolute Quad Mode . . . . .                            | H-7  |
| Absolute X Indexed Mode . . . . .                       | H-7  |
| Absolute Quad X Indexed Mode . . . . .                  | H-7  |
| Absolute Y Indexed Mode . . . . .                       | H-8  |
| Absolute Quad Y Indexed Mode . . . . .                  | H-8  |
| Absolute Z Indexed Mode . . . . .                       | H-8  |
| Absolute Quad Z Indexed Mode . . . . .                  | H-8  |
| Absolute Indirect Mode . . . . .                        | H-9  |
| Absolute Indirect X-Indexed Mode . . . . .              | H-9  |
| Base Page Indirect X-Indexed Mode . . . . .             | H-9  |
| Base Page Quad Indirect X-Indexed Mode . . . . .        | H-9  |
| Base Page Indirect Y-Indexed Mode . . . . .             | H-9  |
| Base Page Quad Indirect Y-Indexed Mode . . . . .        | H-10 |
| Base Page Indirect Z-Indexed Mode . . . . .             | H-10 |
| Base Page Quad Indirect Z-Indexed Mode . . . . .        | H-10 |
| 32-bit Base Page Indirect Z-Indexed Mode . . . . .      | H-10 |
| 32-bit Base Page Indirect Quad Z-Indexed Mode . . . . . | H-11 |
| 32-bit Base Page Indirect Mode . . . . .                | H-11 |

|                                                 |       |
|-------------------------------------------------|-------|
| 32-bit Base Page Indirect Mode . . . . .        | H-12  |
| Stack Relative Indirect, Indexed by Y . . . . . | H-12  |
| Relative Addressing Mode . . . . .              | H-12  |
| Relative Word Addressing Mode . . . . .         | H-12  |
| 6502 Instruction Set . . . . .                  | H-13  |
| Opcode Map . . . . .                            | H-14  |
| Instruction Timing . . . . .                    | H-15  |
| Addressing Mode Table . . . . .                 | H-16  |
| Official And Unintended Instructions . . . . .  | H-17  |
| 4510 Instruction Set . . . . .                  | H-57  |
| Opcode Map . . . . .                            | H-58  |
| Instruction Timing . . . . .                    | H-59  |
| Addressing Mode Table . . . . .                 | H-61  |
| 45GS02 Compound Instructions . . . . .          | H-116 |

|                                              |            |
|----------------------------------------------|------------|
| <b>I Developing System Programmes</b>        | <b>I-1</b> |
| Introduction . . . . .                       | I-5        |
| Flash Menu . . . . .                         | I-5        |
| Format/FDISK Utility . . . . .               | I-6        |
| Keyboard Test Utility . . . . .              | I-7        |
| MEGA65 Configuration Utility . . . . .       | I-7        |
| Freeze Menu . . . . .                        | I-7        |
| Freeze Menu Helper Programmes . . . . .      | I-7        |
| Hypervisor . . . . .                         | I-7        |
| OpenROM . . . . .                            | I-8        |
| <b>J MEGA65 Hypervisor Services</b>          | <b>J-1</b> |
| General Services . . . . .                   | J-3        |
| \$00:\$00 - Get Hypervisor Version . . . . . | J-3        |

|                                                                                |     |
|--------------------------------------------------------------------------------|-----|
| \$00:\$38 - Get Current Error Code (geterrorcode)                              | J-3 |
| \$00:\$3A - Setup Transfer Area for Other Calls (setup_transfer_area)          | J-5 |
| Disk/Storage Hypervisor Calls                                                  | J-5 |
| \$00:\$02 - Get Default Drive (SD Card Partition)                              | J-5 |
| \$00:\$04 - Get Current Drive (SD Card Partition)                              | J-6 |
| \$00:\$06 - Select Drive (SD Card Partition)                                   | J-6 |
| \$00:\$08 - <i>NOT IMPLEMENTED</i> Get Disk Size                               | J-6 |
| \$00:\$0A - <i>NOT IMPLEMENTED</i> Get Current Working Directory               | J-6 |
| \$00:\$0C - Change Working Directory                                           | J-6 |
| \$00:\$0E - <i>NOT IMPLEMENTED</i> Create Directory                            | J-7 |
| \$00:\$10 - <i>NOT IMPLEMENTED</i> Remove Directory                            | J-7 |
| \$00:\$12 - Open Directory (opendir)                                           | J-7 |
| \$00:\$14 - Read Next Directory Entry (readdir)                                | J-8 |
| \$00:\$16 - Close Directory (closedir)                                         | J-8 |
| \$00:\$18 - Open File (openfile)                                               | J-8 |
| \$00:\$1A - Read From a File (readfile)                                        | J-8 |
| \$00:\$1C - <i>NOT IMPLEMENTED</i> Write to a File (writefile)                 | J-8 |
| \$00:\$1E - <i>NOT IMPLEMENTED</i> Create File (mkfile)                        | J-8 |
| \$00:\$20 - Close a File (closefile)                                           | J-8 |
| \$00:\$22 - Close All Open Files (closeall)                                    | J-8 |
| \$00:\$24 - <i>NOT IMPLEMENTED</i> Seek to a Given Offset in a File (seekfile) | J-8 |
| \$00:\$26 - <i>NOT IMPLEMENTED</i> Delete a File (rmfile)                      | J-8 |
| \$00:\$28 - <i>NOT IMPLEMENTED</i> Get Information About a File (fstat)        | J-8 |
| \$00:\$2A - <i>NOT IMPLEMENTED</i> Rename a File (rename)                      | J-9 |
| \$00:\$2C - <i>NOT IMPLEMENTED</i> Set time stamp of a file (filedate)         | J-9 |
| \$00:\$2E - Set the current filename (setname)                                 | J-9 |
| \$00:\$30 - Find first matching file (findfirst)                               | J-9 |
| \$00:\$32 - Find subsequent matching file (findnext)                           | J-9 |

|                                                                              |      |
|------------------------------------------------------------------------------|------|
| \$00:\$34 - Find matching file (one only) (findfile)                         | J-9  |
| \$00:\$36 - Load a File into Memory (loadfile)                               | J-9  |
| \$00:\$3C - Change Working Directory to Root Directory of Selected Partition | J-9  |
| Disk Image Management                                                        | J-9  |
| \$00:\$40 - Attach a D81 Disk Image to Drive 0                               | J-10 |
| \$00:\$42 - Detach All D81 Disk Images                                       | J-10 |
| \$00:\$44 - Write Enable All Currently Attached D81 Disk Images              | J-10 |
| \$00:\$46 - Attach a D81 Disk Image to Drive 1                               | J-10 |
| Task and Process Management                                                  | J-10 |
| \$00:\$50 - <i>NOT IMPLEMENTED</i> Get Task List                             | J-10 |
| \$00:\$52 - <i>NOT IMPLEMENTED</i> Send Message to Another Task              | J-10 |
| \$00:\$54 - <i>NOT IMPLEMENTED</i> Receive Messages From Other Tasks         | J-10 |
| \$00:\$56 - <i>NOT IMPLEMENTED</i> Write Into Memory of Another Task         | J-10 |
| \$00:\$58 - <i>NOT IMPLEMENTED</i> Read From Memory of Another Task          | J-10 |
| \$00:\$60 - <i>NOT IMPLEMENTED</i> Terminate Another Task                    | J-11 |
| \$00:\$62 - <i>NOT IMPLEMENTED</i> Create a Native MEGA65 Task               | J-11 |
| \$00:\$64 - <i>NOT IMPLEMENTED</i> Load File Into Task                       | J-11 |
| \$00:\$66 - <i>NOT IMPLEMENTED</i> Create a C64-Mode Task                    | J-11 |
| \$00:\$68 - <i>NOT IMPLEMENTED</i> Create a C65-Mode Task                    | J-11 |
| \$00:\$6A - <i>NOT IMPLEMENTED</i> Exit and Switch to Another Task           | J-11 |
| \$00:\$6C - <i>NOT IMPLEMENTED</i> Context-Switch to Another Task            | J-11 |
| \$00:\$6E - <i>NOT IMPLEMENTED</i> Exit This Task                            | J-11 |
| \$00:\$70 - Toggle Write Protection of ROM Area                              | J-11 |
| \$00:\$72 - Toggle 4510 vs 6502 Processor Mode                               | J-12 |
| \$00:\$74 - Get current 4510 memory MAPping                                  | J-12 |
| \$00:\$76 - Set 4510 memory MAPping                                          | J-12 |
| \$00:\$7C - Write Character to Serial Monitor/Matrix Mode Interface          | J-12 |
| \$00:\$7E - Reset MEGA65                                                     | J-12 |

|                                                                               |      |
|-------------------------------------------------------------------------------|------|
| \$01:\$00 - Enable Write Protection of ROM Area . . . . .                     | J-12 |
| \$01:\$02 - Disable Write Protection of ROM Area . . . . .                    | J-13 |
| System Partition & Freezing . . . . .                                         | J-13 |
| \$02:\$00 - Read System Config Sector Into Memory . . . . .                   | J-13 |
| \$02:\$02 - Write System Config Sector From Memory . . . . .                  | J-13 |
| \$02:\$04 - Apply System Config Sector Current Loaded Into Memory . . . . .   | J-13 |
| \$02:\$06 - Set DMAgic Revision Based On Loaded ROM . . . . .                 | J-13 |
| \$02:\$10 - Locate First Sector of Freeze Slot . . . . .                      | J-13 |
| \$02:\$12 - Unfreeze From Freeze Slot . . . . .                               | J-13 |
| \$02:\$14 - Read Freeze Region List . . . . .                                 | J-13 |
| \$02:\$16 - Get Number of Freeze Slots . . . . .                              | J-13 |
| \$03:\$XX - Write Character to Serial Monitor/Matrix Mode Interface . . . . . | J-14 |
| Secure Mode . . . . .                                                         | J-14 |
| \$11:\$XX - Request Enter Secure Mode . . . . .                               | J-14 |
| \$12:\$XX - Request Exit Secure Mode . . . . .                                | J-14 |
| \$32:\$XX - <i>DEPRECATED</i> Set Protected Hardware Configuration . . . . .  | J-14 |
| \$3F:\$XX - Freeze Self . . . . .                                             | J-14 |

|                                                      |            |
|------------------------------------------------------|------------|
| <b>K Machine Language Monitor</b>                    | <b>K-1</b> |
| Introduction . . . . .                               | K-3        |
| C65 ROM Standard Machine Language Monitor . . . . .  | K-3        |
| Table of C65 ROM Standard Monitor Commands . . . . . | K-4        |
| Calling the Monitor . . . . .                        | K-4        |
| addresses and numbers . . . . .                      | K-5        |
| D : DISASSEMBLE . . . . .                            | K-5        |
| M : MEMORY . . . . .                                 | K-6        |
| Enhanced Machine Language Monitor . . . . .          | K-7        |
| Table of MEGA65 Enhanced Monitor Commands . . . . .  | K-8        |
| Calling the Monitor . . . . .                        | K-9        |

|                                                 |      |
|-------------------------------------------------|------|
| addresses and numbers . . . . .                 | K-9  |
| Assembler . . . . .                             | K-9  |
| A : ASSEMBLE . . . . .                          | K-10 |
| B : BITMAPS . . . . .                           | K-11 |
| MEGA65 Matrix Mode Monitor Interface . . . . .  | K-12 |
| Table of Matrix Mode Monitor Commands . . . . . | K-14 |
| Calling the Monitor . . . . .                   | K-14 |
| # : Hypervisor trap enable/disable . . . . .    | K-14 |
| + : Set Serial Interface UART Divisor . . . . . | K-15 |
| @ : CPUMEMORY . . . . .                         | K-15 |
| ? or H : HELP . . . . .                         | K-16 |
| B : BREAKPOINT . . . . .                        | K-16 |
| D : DISASSEMBLE . . . . .                       | K-16 |
| E : FLAGWATCH . . . . .                         | K-16 |
| F : FILL . . . . .                              | K-17 |
| G : SETPC . . . . .                             | K-17 |
| I : INTERRUPTS . . . . .                        | K-17 |
| J : DEBUGMON . . . . .                          | K-18 |
| L : LOADMEMORY . . . . .                        | K-18 |
| M : MEMORY . . . . .                            | K-18 |
| R : REGISTERS . . . . .                         | K-18 |
| S : SETMEMORY . . . . .                         | K-19 |
| T : TRACE . . . . .                             | K-19 |
| W : WATCHPOINT . . . . .                        | K-19 |
| Z : CPUHISTORY . . . . .                        | K-20 |

|                                                                             |            |
|-----------------------------------------------------------------------------|------------|
| <b>L F018-Compatible Direct Memory Access (DMA) Controller</b>              | <b>L-1</b> |
| F018A/B DMA Jobs . . . . .                                                  | L-3        |
| F018 DMA Job List Format . . . . .                                          | L-4        |
| F018 11-byte DMA List Structure . . . . .                                   | L-5        |
| F018B 12-byte DMA List Structure . . . . .                                  | L-5        |
| Performing Simple DMA Operations . . . . .                                  | L-7        |
| MEGA65 Enhanced DMA Jobs . . . . .                                          | L-12       |
| Texture Scaling and Line Drawing . . . . .                                  | L-15       |
| Audio DMA . . . . .                                                         | L-17       |
| Sample Address Management . . . . .                                         | L-18       |
| Sample Playback frequency and Volume . . . . .                              | L-18       |
| Pure Sine Wave . . . . .                                                    | L-19       |
| Sample playback control . . . . .                                           | L-19       |
| F018 "DMAgic" DMA Controller . . . . .                                      | L-19       |
| MEGA65 DMA Controller Extensions . . . . .                                  | L-20       |
| Unimplemented Functionality . . . . .                                       | L-24       |
| <b>M VIC-IV Video Interface Controller</b>                                  | <b>M-1</b> |
| Features . . . . .                                                          | M-5        |
| VIC-II/III/IV Register Access Control . . . . .                             | M-6        |
| Detecting VIC-II/III/IV . . . . .                                           | M-7        |
| Video Output Formats, Timing and Compatibility . . . . .                    | M-8        |
| Integrated Marvellous Digital Hookup™(IMDH™) Digital Video Output . . . . . | M-9        |
| Connecting to Naughty Proprietary Digital Video Standards . . . . .         | M-9        |
| Frame Timing . . . . .                                                      | M-10       |
| Physical and Logical Rasters . . . . .                                      | M-13       |
| Bad Lines . . . . .                                                         | M-13       |
| Memory Interface . . . . .                                                  | M-14       |
| Relocating Screen Memory . . . . .                                          | M-14       |

|                                                                                          |      |
|------------------------------------------------------------------------------------------|------|
| Relocating Character Generator Data . . . . .                                            | M-14 |
| Relocating Colour / Attribute RAM . . . . .                                              | M-15 |
| Relocating Sprite Pointers and Images . . . . .                                          | M-15 |
| Hot Registers . . . . .                                                                  | M-16 |
| New Modes . . . . .                                                                      | M-17 |
| Why the new VIC-IV modes are Character and Bitmap modes, not Bitplane modes . . . . .    | M-17 |
| Displaying more than 256 unique characters via "Super-Extended Attribute Mode" . . . . . | M-18 |
| Using Super-Extended Attribute Mode . . . . .                                            | M-21 |
| Full-Colour (256 colours per character) Text Mode (FCM) . . . . .                        | M-25 |
| Nybl-colour (16 colours per character) Text Mode (NCM) . . . . .                         | M-25 |
| Alpha-Blending / Anti-Aliasing . . . . .                                                 | M-26 |
| Flipping Characters . . . . .                                                            | M-26 |
| Variable Width Fonts . . . . .                                                           | M-26 |
| Raster Re-write Buffer . . . . .                                                         | M-27 |
| Sprites . . . . .                                                                        | M-27 |
| VIC-II/III Sprite Control . . . . .                                                      | M-27 |
| Extended Sprite Image Sets . . . . .                                                     | M-28 |
| Variable Sprite Size . . . . .                                                           | M-28 |
| Variable Sprite Resolution . . . . .                                                     | M-29 |
| Sprite Palette Bank . . . . .                                                            | M-29 |
| Full-Colour Sprite Mode . . . . .                                                        | M-30 |
| VIC-II / C64 Registers . . . . .                                                         | M-32 |
| VIC-III / C65 Registers . . . . .                                                        | M-35 |
| VIC-IV / MEGA65 Specific Registers . . . . .                                             | M-38 |

## **N 6526 Complex Interface Adaptor (CIA) Registers**

**N-1**

|                                         |     |
|-----------------------------------------|-----|
| CIA 6526 Registers . . . . .            | N-3 |
| CIA 6526 Hypervisor Registers . . . . . | N-6 |

|                                                                       |            |
|-----------------------------------------------------------------------|------------|
| <b>O 4551 UART, GPIO and Utility Controller</b>                       | <b>O-1</b> |
| C65 6551 UART Registers . . . . .                                     | O-3        |
| 4551 General Purpose IO & Miscellaneous Interface Registers . . . . . | O-4        |
| <b>P 45E100 Fast Ethernet Controller</b>                              | <b>P-1</b> |
| Overview . . . . .                                                    | P-3        |
| Differences to the RR-NET and similar solutions . . . . .             | P-3        |
| Theory of Operation: Receiving Frames . . . . .                       | P-4        |
| Accessing the Ethernet Frame Buffers . . . . .                        | P-6        |
| Theory of Operation: Sending Frames . . . . .                         | P-7        |
| Advanced Features . . . . .                                           | P-7        |
| Broadcast and Multicast Traffic and Promiscuous Mode . . . . .        | P-7        |
| Debugging and Diagnosis Features . . . . .                            | P-8        |
| Memory Mapped Registers . . . . .                                     | P-9        |
| COMMAND register values . . . . .                                     | P-10       |
| Example Programs . . . . .                                            | P-11       |
| <b>Q 45IO27 Multi-Function IO Controller</b>                          | <b>Q-1</b> |
| Overview . . . . .                                                    | Q-3        |
| F011-compatible Floppy Controller . . . . .                           | Q-3        |
| Multiple Drive Support . . . . .                                      | Q-3        |
| Buffered Sector Operations . . . . .                                  | Q-4        |
| Reading Sectors from a Disk . . . . .                                 | Q-4        |
| Track Auto-Tune Function Deprecated . . . . .                         | Q-5        |
| Sector Skew and Target Any Mode . . . . .                             | Q-6        |
| Disk Layout and 1581 Logical Sectors . . . . .                        | Q-6        |
| FD2000 Disks . . . . .                                                | Q-7        |
| Formatting Disks . . . . .                                            | Q-7        |
| Buffered Sector Writing . . . . .                                     | Q-8        |

|                                                                |      |
|----------------------------------------------------------------|------|
| Using HD, ED Drives and/or variable Data Rates . . . . .       | Q-8  |
| F011 Floppy Controller Registers . . . . .                     | Q-9  |
| SD Card Controller and F011 Virtualisation Functions . . . . . | Q-11 |
| SD-Card Based Disk Image Access . . . . .                      | Q-12 |
| F011 Virtualisation . . . . .                                  | Q-14 |
| Dual-Bus SD Card Controller . . . . .                          | Q-14 |
| Write Gate . . . . .                                           | Q-15 |
| Fill Mode . . . . .                                            | Q-15 |
| Selecting Among Multiple SD Cards . . . . .                    | Q-16 |
| SD Controller Command Table . . . . .                          | Q-16 |
| Touch Panel Interface . . . . .                                | Q-18 |
| Audio Support Functions . . . . .                              | Q-20 |
| Miscellaneous IO Functions . . . . .                           | Q-23 |

|                            |            |
|----------------------------|------------|
| <b>R Reference Tables</b>  | <b>R-1</b> |
| Units of Storage . . . . . | R-3        |
| Base Conversion . . . . .  | R-4        |

|                                                                           |            |
|---------------------------------------------------------------------------|------------|
| <b>S Flashing the FPGAs and CPLDs in the MEGA65</b>                       | <b>S-1</b> |
| Suggested PC specifications . . . . .                                     | S-5        |
| Warning . . . . .                                                         | S-6        |
| Installing Vivado . . . . .                                               | S-6        |
| Installing the FTDI drivers . . . . .                                     | S-21       |
| Linux drivers . . . . .                                                   | S-22       |
| Windows drivers . . . . .                                                 | S-22       |
| Flashing the main FPGA using Vivado . . . . .                             | S-26       |
| Flashing the CPLD in the MEGA65's Keyboard with Lattice Diamond . . . . . | S-39       |
| Flashing the MAX10 FPGA on the MEGA65's Mainboard with INTEL QUARTUS .    | S-46       |

|                                                       |                                |                |
|-------------------------------------------------------|--------------------------------|----------------|
| <b>T</b>                                              | <b>Trouble shooting</b>        | <b>T-1</b>     |
| Vivado . . . . .                                      | . . . . .                      | T-3            |
| RAM requirements . . . . .                            | . . . . .                      | T-3            |
| <b>U</b>                                              | <b>Model Specific Features</b> | <b>U-1</b>     |
| Detecting MEGA65 Models . . . . .                     | . . . . .                      | U-3            |
| MEGA65 Desktop Computer, Revision 3 onwards . . . . . | . . . . .                      | U-3            |
| MEGA65 Desktop Computer, Revision 2 . . . . .         | . . . . .                      | U-4            |
| MEGApone Handheld, Revisions 1 and 2 . . . . .        | . . . . .                      | U-5            |
| Nexys4 DDR FPGA Board . . . . .                       | . . . . .                      | U-6            |
| <b>V</b>                                              | <b>Schematics</b>              | <b>V-1</b>     |
| MEGA65 R3 Schematics . . . . .                        | . . . . .                      | V-3            |
| MEGA65 R2 Schematics . . . . .                        | . . . . .                      | V-28           |
| <b>W</b>                                              | <b>Supporters &amp; Donors</b> | <b>W-1</b>     |
| Organisations . . . . .                               | . . . . .                      | W-3            |
| Contributors . . . . .                                | . . . . .                      | W-4            |
| Supporters . . . . .                                  | . . . . .                      | W-5            |
| <b>INDEX</b>                                          |                                | <b>Index-3</b> |



# 1

## CHAPTER

### Introduction

- Other Books in this series



Congratulations on your purchase of one of the most long-awaited computers in the history of computing. The MEGA65 is a community designed computer, based on the never-released Commodore® 65<sup>1</sup> computer; a computer designed in 1989 and intended for public release in 1990. Decades have passed, and the MEGA 65 invokes an earlier time when computers were simple and friendly. They were not only simple to operate and understand how they work, but friendly and approachable for new users.

These 1980s computers inspired an entire generation of professionals to choose the exciting and rewarding technology careers they have today. Just imagine the joy of these individuals as they learned they could use their new computer to solve problems, write a letter, prepare taxes, invent new things, or even discover how the universe works. We want to recreate that level of excitement not found in modern computing, so we made the **MEGA65**.

The MEGA65 team believes that owning a computer is like owning a home; you don't just use a home; you change things big and small to make it your own custom living space. After a while, when you settle in, you may decide to renovate or expand your home to make it more comfortable or provide more utility. Think of the MEGA65 as a "computing home."

This guide will teach you how to do more than just hang pictures on a wall, it will ask you to build your dream home. While you read this user's guide, you will learn how to operate the MEGA65, code programs, add additional software, and extend hardware capabilities. What won't be immediately obvious is that along the journey, you will also learn about the history of computing as you explore Commodore BASIC version 10 and operating system commands.

Computer graphics and music make computing more fun; and we designed the MEGA65 for fun! In this user's guide, you will learn to code using the MEGA65's built-in **graphics** and **sound** capabilities. But you don't need to be a coder to have fun with the MEGA65. Because the MEGA65 includes a complete Commodore® 64™<sup>2</sup>, it can also run thousands of games, utilities, and business software from the past and new programs being written today by Commodore enthusiasts. Excitement for the MEGA65 will grow as we discover what programmers as they learn about the power and features of this modern Commodore computer recreation. Together, we will create a whole new "home-brew" community to do things that even we didn't think were possible when creating the MEGA65.

We welcome you on this journey! Thank you for becoming a part of the **MEGA65** community of users, coders, and enthusiasts! Get involved, learn more about your MEGA65, and join us online at: <https://mega65.net>

---

<sup>1</sup>Commodore is a trademark of C= Holdings

<sup>2</sup>Commodore 64 is a trademark of C= Holdings,

# OTHER BOOKS IN THIS SERIES

This is one of several MEGA65 documentation volumes. The series include:

- The MEGA65 User Guide – Provides an introduction to the MEGA65, and a condensed BASIC 10 command reference
- The MEGA65 BASIC 10 Reference Guide – Comprehensive documentation of all BASIC 10 commands, functions and operators
- The MEGA65 Chipset Reference – Detailed documentation about the MEGA65 and C65's custom chips
- The MEGA65 Developer Guide – Information for developers wishing to write programmes for the MEGA65
- The MEGA65 Book – All volumes in a single huge PDF for easy online searching. 850 pages and growing!

# PART I

## GETTING TO KNOW YOUR MEGA65



# CHAPTER **2**

## **SETUP**

- **Unpacking and connecting the MEGA65**
- **Rear Connections**
- **Side Connections**
- **Installation**
- **Optional Connections**
- **Operation**



# UNPACKING AND CONNECTING THE MEGA65

Time to set up your MEGA65 home computer. The box contains the following:

- MEGA65 computer.
- Power supply (black box with socket for mains supply).
- This book, the MEGA65 User's Guide.

In addition, to be able to use your MEGA65 computer you may need:

- A television or computer monitor with a VGA or digital video input, that is capable of displaying an image with 480p or 576p (720x480 or 720x576 pixel resolution at 50Hz or 60Hz).
- A VGA video cable, or;
- A digital video cable.

These items are not included with the MEGA65.

You may also want to use the following to get the most out of your MEGA65:

- 3.5mm mini-jack audio cable and suitable speakers or hifi system, so that you can enjoy the sound capabilities of your MEGA65.
- RJ45 Ethernet cable (regular network cable) and a network router or switch. This allows the usage of the high-speed networking capabilities of your MEGA65.

# REAR CONNECTIONS



- |   |                            |
|---|----------------------------|
| 1 | 3.5mm Audio Mini-Jack      |
| 2 | SDCard                     |
| 3 | Network LAN Port           |
| 4 | Digital Video Connector    |
| 5 | VGA Video Connector        |
| 6 | External Floppy Disk Drive |
| 7 | Cartridge Expansion Port   |
| 8 | DC Power-In Socket         |

# SIDE CONNECTIONS



Various peripherals can be connected to Controller Ports 1 and 2 such as joysticks or paddles.

# INSTALLATION

Connecting your MEGA65 to a screen and peripherals



1. Connect the power supply to the Power Supply socket of the MEGA65.
2. If you have a VGA monitor and a VGA cable, connect one end to the VGA port of the MEGA65 and the other end into your VGA monitor.
3. If you have a TV or monitor with a compatible Digital Video connector, connect one end of your cable to the Digital Video port of the MEGA65, and the other into the Digital Video port of your monitor. If you own a monitor with a DVI socket, you can purchase a DVI to Digital Video adaptor.

## OPTIONAL CONNECTIONS

1. The MEGA65 houses an internal 3.5" floppy disk drive. You can also connect older Commodore® IEC serial floppy drives to the MEGA65: the Commodore® 1541, 1571 or 1581. Connect one end of your IEC cable to the Commodore® floppy disk drive and the other end to the Disk Drive socket of the MEGA65. You can also connect SD2IEC devices and PI1541's. It is possible to daisy-chain additional floppy disk drives or Commodore® compatible printers.
2. You can connect your MEGA65 to a network using a standard Ethernet cable.
3. For enjoying audio from your MEGA65, you can connect a 3.5mm stereo mini-jack cable to an audio amplifier or speaker system. If your system has RCA connectors you will need to purchase a 3.5mm mini-jack to twin RCA adaptor cable. The MEGA65 also has a built in amplifier to allow connecting headphones.
4. A Secure Digital Card or SDCard (SDHC and SDXC) can be inserted into the rear of the MEGA65 as a drive.

## OPERATION

### Using the MEGA65

1. Turn on the computer by using the switch on the left hand side of the MEGA65.
2. After a moment, the following will be displayed on your TV or monitor:

THE COMMODORE C65 DEVELOPMENT SYSTEM  
COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.  
BASIC 10.0 V0.9.910111 ALL RIGHTS RESERVED

ENGLISH KEYBOARD  
EXPANSION RAM

READY.

The flashing cursor  
indicates the MEGA65  
is ready for input.

## THE CURSOR

The flashing square underneath the READY prompt is called the cursor. The cursor indicates that the computer is ready to accept input. Pressing keys on the keyboard will print that character onto the screen. The character will be printed in the current cursor position, and then the cursor advances to the next position.

You can type commands, for example: telling the computer to load a program. You can even start entering program code.

# CHAPTER

# 3

## GETTING STARTED

- **Keyboard**
- **The Screen Editor**
- **Editor Functionality**



# KEYBOARD

Now that you have everything connected, it's time to get familiar with the MEGA65 keyboard.

You may notice that the keyboard is a little different from the standard used on computers today. While most keys will be in familiar positions, there are some specialised keys, and some with special graphic symbols marked on the front.

Here's a brief description of how some of these special keys function.

## Command Keys

The Command Keys are: **RETURN**, **SHIFT**, **Ctrl**, **!<** and **RESTORE**.

### RETURN

Pressing the **RETURN** key enters the information you have typed into the MEGA65's memory. The computer will either act on a command, store some information, or return you an error if you made a mistake.

### SHIFT

The two **SHIFT** keys are located on the left and the right. They work very much like Shift on a regular keyboard, however they also perform some special functions too.

In upper case mode, holding down **SHIFT** and pressing any key with a graphic symbol on the front produces the right hand symbol on that key. For example, **SHIFT** and **J** prints the **☒** character.

In lower case mode, pressing **SHIFT** and a letter key prints the upper case letter on that key.

Finally, holding down the **SHIFT** key and pressing a Function key accesses the function shown on the front of that key. For example: **SHIFT** and **F1** activates **F2**.

### SHIFT LOCK

In addition to the Shift key is **SHIFT LOCK**. Press this key to lock down the Shift function. Now any key you press prints the character to the screen as if you were holding down **SHIFT**. That includes special graphic characters.

## **CTRL**

**CTRL** is the Control key. Holding down **CTRL** and pressing another key allows you to perform Control Functions. For example, holding down **CTRL** and one of the number keys allows you to change text colours.

There are some examples of this in Chapter/Appendix 3 on page 3-7, and all the Control Functions are listed in Chapter/Appendix C on page C-5.

If a program is being listed to the screen, holding down **CTRL** slows down the display of each line on the screen.

Holding **CTRL** and pressing **\*** enters the Matrix Mode Debugger.

## **RUN/STOP**

Normally, pressing the **RUN STOP** key stops execution of a program. Holding **SHIFT** while pressing **RUN STOP** loads the first program from disk.

Programs are able to disable the **RUN STOP** key.

You can boot your machine into the machine code monitor by holding down **RUN STOP** and pressing reset on the MEGA65.

## **RESTORE**

The computer screen can be restored to a clean state without clearing the memory by holding down the **RUN STOP** key and tapping **RESTORE**. This combination also resets operating system vectors and re-initialises the screen editor, which makes it a handy combination if the computer has become a little confused.

Programs are able to disable this key combination.

Enter the Freeze Menu by holding down **RESTORE** for more than one second. You can access the machine code monitor via the Freeze menu.

## THE CURSOR KEYS

At the bottom right hand of the keyboard are the cursor keys. These four directional keys allow you move the cursor to any position for onscreen editing.

The cursor moves in the direction indicated on the keys:

However, it is also possible to move the cursor up using and . In the same way you can move the cursor left using and .

You don't have to keep pressing a cursor key over and over. When moving the cursor a long way, you can keep the key pressed in. When you are finished, release the key.

## INSerT/DELeTe

This is the INSERT / DELETE key. When pressing , the character to the left is deleted, and all characters to the right are shifted one position to the left.

To insert a character, hold the key and press . All the characters are shifted to the right. This allows you to type a letter, number or any other character into the newly inserted space.

## CLeAr/HOME

Pressing the key returns the cursor into the top left-most position of the screen.

Holding down and pressing clears the entire screen and places the cursor into the top left-most position of the screen.

## MEGA KEY

The key or the MEGA key provides a number of different functions and special utilities.

Holding the key and pressing switches between lower and upper case character modes.

Holding and pressing any key with graphic symbols on the front prints the left-most graphic symbol to the screen.

Holding and pressing any key that shows a single graphic symbol on the front prints that graphic symbol to the screen.

Holding and pressing a number key switches to one of the colours in the second range.

Holding  and pressing  enters the Matrix Mode Debugger.

When switching on the MEGA65 or pressing the reset button on the side, while holding down  switches the MEGA65 into C64 mode.

## NO SCROLL

If a program is being listed to the screen, pressing  freezes the screen output. Not available in C64 mode.

## Function Keys

There are seven Function keys available for use by software applications,       and  to perform functions with a single press.

Hold  to access  through to  as shown on the front of each Function key.

Only Function keys  to  are available in C64 mode.

## HELP

The  key can be used by software and acts as an  /  key.

## ALT

Holding  down while pressing other keys can be used by software to perform functions. Not available in C64 mode.

Holding  down when switching the MEGA65 on activates the Utility Menu. You can format the SD card or enter the MEGA65 Configuration Utility to select the default video mode and other settings, or test your keyboard.

## CAPS LOCK

The  works like  in C65 and MEGA65 modes, but only modifies the alphabet keys. Also, holding the  down forces the processor to run at the maximum speed. This can be used, for example, to speed up loading from the internal disk drive or SD card, or to greatly speed up the de-packing process after a program is run. This can reduce the loading and de-packing time from many seconds to as little as a 10th of a second.

# THE SCREEN EDITOR

When you turn on your MEGA65, or reset it, the editor screen will appear.



The colour bars in the top left hand of the screen can be used as a guide to help calibrate the colours of your display. The screen also displays the name of the system, the copyright notice and what version and revision of BASIC is contained in the Read-only Memory.

Also displayed is the type of keyboard and whether or not there is additional hardware present, such as a RAM expansion.

Finally, you will see the READY prompt and the flashing cursor.

You can begin typing keys on the keyboard and the characters will be printed under the cursor. The cursor itself advances after each key press.

You can also produce reverse text or colour bars by holding down the **CTRL** key and pressing the **9** key or the **R** key. This enters reverse text mode.

Try holding down the **SPACE BAR**. A white bar will be drawn across the screen.

You can even change the current colour by holding down the **CTRL** key and pressing a number key. Try key **8** and then hold down the **SPACE BAR** again. A yellow bar will be drawn.

Change the bar to a number of other colours.

You will get an effect something like:



You can turn off the reverse text mode by holding **CTRL** and pressing the **0** key.

By pressing any keys, the characters will be typed out in the chosen colour.

There are a further eight colours available via the **M** key. Hold the **M** key and press a key from **1** to **8** to change to one of the secondary colours. For even more colours, see Chapter/Appendix C on page C-8.

You can create fun pictures just using these colours and the letters. Here's an example of what a year four student drew:



What will you draw?

## Functions

Functions using the **CTRL** key are called **Control Functions**. Functions using the **M** key are called **Mega Functions**. There are also functions called by using the **SHIFT** key. These are (not surprisingly) called **Shift Functions**.

Lastly, using the **ESC** key are **Escape Sequences**.

## **ESC Sequences**

Escape sequences are performed a little differently than a Control function or a Shifted function. Instead of holding the modifier key down, an Escape sequence is performed by pressing the **ESC** key and releasing it, followed by pressing the desired key code.

For example: to switch between 40/80 column mode, press and release the **ESC** key. Then press the **X** key.

You can see all the available Chapter/Appendix C on page C-8. We will cover some examples of these shortly.

There are more modes available. You can create flashing text by holding the **CTRL** key and pressing the **O** key. Any characters you press will flash. Turn flash mode off by pressing **ESC** then **O**.

# **EDITOR FUNCTIONALITY**

The MEGA65 screen can allow you to do advanced tabbing, and moving quickly around the screen in many ways to help you to be more productive.

Press the **HOME** key to go to the home position on the screen. Hold the **CTRL** key down and press the **W** key several times. This is the **Word Advance function** which jumps your cursor to the next word, or printable character.

You can set custom tab positions on the screen for your convenience. Press **HOME** and then **→** to the fourth column. Hold down **CTRL** and press the **X** key to set a tab. Move another 20 positions to the right again, and do **CTRL** and **X** again to set a second tab.

Press the **HOME** key to go back to the home position. Hold the **CTRL** key and press the **I** key. This is the **Forward Tab function**. Your cursor will tab to the fourth position.

Press **CTRL** and **I** again. Your cursor will move to position 8. Why? By default, every 8th position is already set as a tabbed position. So the 4th and 20th positions have been added to the existing tab positions. You can continue to press the **CTRL** and **I** keys to the 16th and 20th positions.

To find the complete set of Control codes, see Chapter/Appendix C on page C-5.

## **Creating a Window**

You can set a window on the MEGA65 working screen. Move your cursor to the beginning of the "BASIC 65" text. Press **ESC**, then press **T**. Move the cursor 10 lines down and 15 to the right.

Press the **ESC** key, then **B**. Anything you type will be contained within this window.

To escape from the window back to the full screen, press the **HOME** key twice.

## **Extras**

Long press on **RESTORE** to go into the Freeze Menu. Then press **J** to switch joystick ports without having to physically swap the joystick to the other port.

Go to **Fast mode** with poke 0, 65 or go to the freeze menu.

**MEGA** and **SHIFT** switches between text uppercase and lowercase for the entire display.

# CHAPTER 4

## Configuring your MEGA65

- Preparing for First Use: Formatting SD Cards
- Installing ROM and Other Support Files
- On-boarding
- Configuration Utility



# PREPARING FOR FIRST USE: FORMATTING SD CARDS

The MEGA65 has two SD card slots: A full-size SD card slot inside, next to the trap-door, and a microSD size slot on the rear. The current version of the MEGA65 firmware only supports the use of one SD card at a time. If you have cards in both slots, the MEGA65 will use the external microSD slot in preference. The exception to this, is that the MEGA65's FDISK/FORMAT utility can access both, allowing you to select which you wish to format or repair.

Depending on the model, your MEGA65 may or may not come with a pre-configured SD card. If it doesn't, or if you wish to use a different SD card, e.g., with a larger capacity, you must first format it for use in the MEGA65.

*This must be done in the MEGA65, not in a PC or other computer.*

*Only SDHC cards should be used. Older SD cards (typically with a capacity of <4GB) will not work. Newer SDXC cards with capacities greater than 32GB may or may not work. We would appreciate hearing your experience with such cards. It is unimportant what file-system is currently on the card, as the MEGA65 FDISK/FORMAT utility will completely reformat the card.*

There are several reasons for this: First, in order to fit the most features into the MEGA65's small operating system, it is particular about the FAT32 file system it uses. Second, only the MEGA65 FDISK/FORMAT utility can create a MEGA65 System Partition. The MEGA65 System Partition holds non-volatile configuration settings for your MEGA65, and also contains the freeze slots, that make it easy to switch which programme or game you are running on your MEGA65.

Fortunately, formatting an SD card on the MEGA65 is very easy.

First, power the MEGA65 on while holding down the **ALT** key. This will present the MEGA65 Utility Menu, which contains a selection of built-in utilities, similar to the following:

# MEGA<sup>65</sup>

MEGAG65 MEGROS HYPERVISOR V00.15

```
GIT: 138-HD.3EC382B+DIRTY.20200512.20
NO SCROLL=FLASH, ALT=UTILS, CTRL=HOLD
GIT: 138-HD.3EC382B+DIRTY.20200512.20
SELECT UTILITY TO LAUNCH
1. CONFIGURE MEGAG65
2. SDCARD FDISK+FORMAT UTILITY
3. KEYBOARD TEST
```

Note that Utility Menu is always accessible, even if no SD card is present in both internal and external slots.

The exact set of utilities depends on the model of your MEGA65 and the version of the MEGA65 factory core which it is running. However, all versions include both the MEGA65 FDISK/FORMAT utility, and the MEGA65 Configure utility. Most models also include a keyboard test utility, that you can use to test that your keyboard is functioning correctly. This is the same utility used in the factory when testing brand new keyboards.

Select the number that corresponds to the FDISK/FORMAT utility. This will typically be 2. The FDISK utility will start, and attempt to detect the size of all SD cards you have installed. If you have both an internal and external SD card installed, it will allow you to choose which one you wish to format. The internal SD card is bus 1, and the external card is bus 0. Note that the MEGA65 will always attempt to boot using an external microSD card, if one is installed.

For safety when formatting we *strongly* recommend that you remove any SD card or microSD card that you do not intend to format, so that you do not accidentally destroy any data. This is because formatting an SD card in the MEGA65 cannot be undone, and all data currently on the SD card *will be lost*. If you have files or data on the SD card that you wish to retain, you should first back this up. The contents of the FAT32 partition can be easily backed up by inserting the SD card into another computer. The contents of

the MEGA65 System Partition, including the contents of freeze slots requires the use of specialised software.

You should aim to backup valuable data from your MEGA65 on a regular basis, especially while the computer remains under development. While we take every care to avoid data corruption or other mishaps, we cannot guarantee that the MEGA65 is free of bugs in this regard.

If you have only an internal SD card, you might see a display similar to the following:

```
Detecting SD card(s) (can take a while)
SD Card 0 (External MicroSD slot):
Maximum readable sector is $00EEB7FE
7638 MiB SD CARD FOUND.
SD Card read speed = 1152 KB/sec

Current partition table:
0C : Start=0 /0 /0      or 00000800 / End=0 /0 /0      or 00AEAFFE
41 : Start=0 /0 /0      or 00AEB7FE / End=0 /0 /0      or 00400000
00 : Start=0 /0 /0      or 00000000 / End=0 /0 /0      or 00000000
00 : Start=0 /0 /0      or 00000000 / End=0 /0 /0      or 00000000
```

```
SD Card 1 (internal SD slot):
No card detected on bus 1
```

```
Please select SD card to modify: 0/1
```

```
MEGA65 FDISK+FORMAT V00.08 : (C) COPYRIGHT 2017-2019 PAUL GARDNER-STEPHEN ETC.
```

Once you have selected the bus, the FDISK/FORMAT utility asks you to confirm that you wish to delete everything:

```
Please select SD card to modify: 0/1
Maximum readable sector is $00EEB7FE
7638 MiB SD CARD FOUND,
SD Card read speed = 1154 KB/sec

Current partition table:
0C : Start=0 /0 /0      or 00000000 / End=0 /0 /0      or 00AEAFFE
41 : Start=0 /0 /0      or 00AEB7FE / End=0 /0 /0      or 00400000
00 : Start=0 /0 /0      or 00000000 / End=0 /0 /0      or 00000000
00 : Start=0 /0 /0      or 00000000 / End=0 /0 /0      or 00000000

$00400000 Sectors available for MEGA65 System partition.
2046 Freeze and OS Service slots.
$00AEAFFE Sectors available for VFAT32 partition.

Format Card with new partition table and FAT32 file system?
5589 MiB VFAT32 Data Partition C 00000000:
$0015CAD5 Clusters, 11158 Sectors/FAT, 568 Reserved Sectors.
2048 MiB MEGA65 System Partition C 00AEB7FE:

Type DELETE EVERYTHING to continue:
Or type FIX MBR to re-write MBR

MEGA65 FDISK+FORMAT V00.08 : (C) COPYRIGHT 2017-2019 PAUL GARDNER-STEPHEN ETC.
```

To avoid accidentally loss of data, you must type “DELETE EVERYTHING” in capitals and press **RETURN**. Alternatively, turn the MEGA65 off and on to abort this process without causing damage to your data.

It is also possible to attempt to recover from a lost Master Boot Record (“Boot Sector”) by instead typing “FIX MBR,” should the need arise.

As an end result, we want to have a correctly formatted SD card with the essential files stored on it for MEGA65 to boot from.

This is how it works: When powering on, MEGA65 will search for- and boot these files:

- FREEZER.M65
- AUDIOMIX.M65
- C64THUMB.M65
- C65THUMB.M65
- MEGA65.ROM
- MEGA65.D81 (default disk image, automatically mounted during start)

Straight out-of-the-box, MEGA65 will only have one SD card installed, located inside behind the trap-door. This SD card contains the essential files to properly boot from. When an external microSD card is inserted, MEGA65 consider the external one as higher priority and will try to boot from it. That means that the microSD card needs to have the essential files on it, otherwise MEGA65 cannot boot properly and will fall back into loading the OpenROM which does not support all the MEGA65 features. In general, if MEGA65 cannot boot properly and fall back to OpenROM, it will be shown in the power-up screen, similar to this:



## INSTALLING ROM AND OTHER SUPPORT FILES

The MEGA65 FDISK/FORMAT utility will install a version of the open-source OpenROM project's C64-compatible ROMs as part of the formatting process. However, you may have other ROMs that you wish to use on the MEGA65. The 911001 version of the C65 ROM in particular is known to work well with the MEGA65. You can copy as many of these as you wish onto the SD card. Make sure that they have the .ROM extension. The default ROM should be called MEGA65.ROM. These files should be 128KB in size, and use the

same internal format as ROMs intended for the C65. This means that the C64-mode KERNEL should be placed at offset \$E000, a C65-mode BASIC at \$A000, and a suitable character set at \$D000.

Other important support files include FREEZER.M65 and AUDIOMIX.M65, which allow you to use the MEGA65's integrated freezer. More details provided below.

## ROM File

### Original C65 ROMs

You may want to source your own separate C65 ROM via other means. There were many different versions created during the development of the Commodore 65, and the MEGA65 can use any of them. However, out of this set, we recommend you use 911001.bin, as this has the most complete BASIC and DOS implementations.

### MEGA65 ROMs

There are newer versions of the **MEGA65 Closed ROM** actively under development also. These ROMs improve upon the original C65 ROMs and make better use of the extra hardware capabilities that the MEGA65 has over the original C65 hardware. These ROMs are available via the filehost also, but only to owners of the MEGA65, who will need to log into the filehost with their credentials in order to download it. It can be located by visiting the "**Files**" tab and searching for "**kernel rom**":

The screenshot shows a web browser interface for a filehost. At the top, there is a navigation bar with 'filehost' on the left and 'MEGA65' on the right. Below the navigation bar, there are tabs for 'News', 'Files' (which is highlighted with a pink box), and 'Knowledge Base'. A search bar with the placeholder 'Search...' is positioned to the right of the tabs. Below the search bar, there is a search input field containing the text 'kernel rom' with a pink box around it. To the right of the search input is a magnifying glass icon. The main content area displays a table of search results. The table has columns: Title, Category, Type, OS, Filename, Size, Published, and Author. One row in the table is highlighted with a pink box and shows the following information: 'C65/MEGA65 Kernel ROM', 'Firmware', 'ROM File', 'MEGA65, C65', '920186.BIN', '131 KB', '2021/06/25', and 'adtbm'. To the right of the last column, there is a download icon with a pink box around it.

| Title                 | Category | Type     | OS          | Filename   | Size   | Published  | Author |
|-----------------------|----------|----------|-------------|------------|--------|------------|--------|
| C65/MEGA65 Kernel ROM | Firmware | ROM File | MEGA65, C65 | 920186.BIN | 131 KB | 2021/06/25 | adtbm  |

### MEGA65 ROM diff files

If you have sourced your own 911001.bin C65 ROM and would like to patch it to the latest MEGA65 ROM, we do provide patches, as the additional improvements we have made to the closed rom are open source. Those diff files are available here:

<https://files.mega65.org?id=fd2c40b9-f337-41f7-8a81-0254b1e09fb5>

### MEGA65 OpenROMs

Another available option is to make use of **MEGA65 OpenROMs**. The latest version of this is always downloadable from either of the following urls:

- <https://files.mega65.org?id=8aec2fba-3b0a-4677-80ae-7a7f5f4f0cb8>
- <https://github.com/MEGA65/open-roms/raw/master/bin/mega65.rom>

## Support Files

For official owners of the MEGA65 (both devkit and final product), visit the following url and log in with the user credentials you have been provided. This will take you to the MEGA65 Filehost location where the "**MEGA65 SD card essentials**" download page is located. Then click the "**Download**" link to retrieve the latest "**SD essentials.rar**" file.

<https://files.mega65.org?id=a809e0ae-30ac-42f5-ab9c-766d72fd6331>



Note that this link is only available to official owners of the MEGA65 product, as the fileset also contains the licensed closed-source MEGA65.ROM file.

For Nexus board owners in search of a similar fileset (without the ROM), visit the following url instead:

<https://files.mega65.org/?id=0fb941fe-5c5f-4608-b0f1-32849d4a8dff>

This will take you to the MEGA65 Filehost location where the "**MEGA65 SD card essentials - No ROM**" download page is located. Click the "**Download**" link to retrieve the latest "**SD essentialsNoROM.rar**" file.

Note that while this fileset does not contain a ROM, there are future plans for it to include the freely available OpenROM.

**MEGA65 SD card essentials - No ROM**

Description

These are the MEGA65 SD card essential files.  
You should always use the latest version of these files to avoid issues.  
Due to copyright issues the MEGA65.ROM is not included in this package. You have to organize it yourself  
(remember to rename the .bin file of the Kernel ROM to MEGA65.ROM)  
(you also have to add a default D81 image, that you would like to be mounted on power up. This D81 has to be renamed to MEGA65.D81, or  
rename the Default boot image in the CONFIGURE menu of the MEGA65)

| Category | Type          | OS     | Size   | Version | Downloads | Author | Published   |
|----------|---------------|--------|--------|---------|-----------|--------|-------------|
| Firmware | SD Card Files | MEGA65 | 266 KB | 1.0     | 113       | adtbm  | 02 Jan 2021 |

**Download**

## ON-BOARDING

On first launch of your MEGA65, you will see the on-boarding screen.



Here you can select and test your screen configuration.

For example, type **TAB** to switch to PAL 50HZ

Welcome to the MEGA65!

Before you go further, there are couple of things you need to do.

Press F3 - F13 to set the time and date.

Press RETURN when done.

Video: **DVI** (no sound) → **PAL 50Hz**

**FAB** = cycle through modes

**SPRCE** = apply and test mode.

Test Audio (set video mode first):

**A** = play a tune

CRT Emulation: **Enabled**

**C** = toggle

Time: **00:00:00 00/00/1900**

**F3 F5 F9 F11 F13**

Press RETURN when done to continue.

Then press **RETURN**, followed by **Y** to test the new video mode.

Welcome to the MEGA65!

Before you go further, there are couple of things you need to do.

Press F3 - F13 to set the time and date.

Press

Try video mode:  
**PAL, Pure DVI**

(Will revert on fail after  
15 seconds)

**(Y)es or (N)o?**

Video

Test

CRT Emulation: **Enabled**

**C** = toggle

Time: **00:00:00 00/00/1900**

**F3 F5 F9 F11 F13**

Press RETURN when done to continue.

Press **K** to keep the new video mode.

Welcome to the MEGAS6!

Before you go further, there are couple of things you need to do.

Press F3 - F13 to set the time and date.

Press **K** to keep video mode.

Video Timeout in 10 sec.

Test

CRT Emulation: Enabled  
C = toggle

Time: 00:00:00 00/00/1900  
F3 F5 F7 F9 F11 F13

Press **RETURN** when done to continue.

Press **RETURN** to complete the configuration.

Welcome to the MEGAS6!

Before you go further, there are couple of things you need to do.

Press F3 - F13 to set the time and date.

Press RETURN when done.

Video: DVD (no sound), PAL, 50Hz  
TAB = cycle through modes  
SPACE = apply and test mode.

Test Audio (set video mode first):  
A = Play a tune

CRT Emulation: Enabled  
C = toggle

Time: 00:00:00 00/00/1900  
F3 F5 F7 F9 F11 F13

Press **RETURN** when done to continue.

Note for Nexys4 board users:

At this very specific step, the board is supposed to reboot and display the main

MEGA65 screen. If the board does not reboot and the screen remains black, then switch power to the board off then on again.

After the reboot you will get the main MEGA65 screen:



## CONFIGURATION UTILITY

The configuration utility for the MEGA65 fills a similar purpose to the BIOS on a PC, and allows you to control certain default behaviours of your MEGA65. However, rather than storing the configuration data in a battery-backed RAM, it stores them on sector 1 of the SD card. This means that if you switch SD cards, you will change the configuration data that you are using.

To enter the configuration utility, turn the MEGA65 on while holding the **ALT** key. This will show the utility menu, similar to the following:



Now press the number corresponding to the Utility Menu. The MEGA65 Configuration Utility will launch, showing a display similar to the following:



If your MEGA65's System Partition has become corrupt, you may be prompted to press **F14** to correct this, i.e., hold **SHIFT** and tap the **F13** key, with a display like the following:



If you do, you will need to use **F7** to save the reset configuration, as otherwise the reset data will not be saved to the MEGA65 System Partition.

Once you have dismissed that display, or if your MEGA65 System Partition was not corrupted, you can begin exploring and adjusting various settings. The programme can be controlled using the keyboard, or optionally, an Amiga(tm) or C1351 mouse.

You can advance screens by pressing **F1**, or use **F2** to navigate in the opposite direction through the screens. You can also use the **←** and **→** keys to navigate between screens.

The **↑** and **↓** keys can be used to select an item.

Press **RETURN** or **SPACE** to toggle a setting, or to allow changing a text or numeric value. The black circle next to an option indicates that it is the selected setting.

When finished, you can press **F7** to receive the option to save the changes. This will give you four options:



- *Exit Without Saving* allows you to abandon any changes made in the MEGA65 Configure utility, without saving them.
- *Apply and Test Settings Now* makes the settings take immediate effect. This can be helpful for testing compatibility of your TV or monitor with PAL or NTSC video modes. If you still see your display after applying such a change, it is safe to save those settings.
- *Restore Factory Defaults* allows you to reset the MEGA65 configuration settings to the factory defaults. It also randomly selects a new MAC address for models that include an internal Ethernet adaptor. If you wish to commit these changes, you must still save them.
- *Save as Default and Exit* commits any changes you have made to the SD card storage, so that they will be used whenever your MEGA65 is turned on.

## Input Devices



PAGE 1/1

- Joystick 1 Amiga Mouse Mode allows either **normal** operation, where software will see it as an Amiga mouse, or **1351 emulation** mode, where the MEGA65 translates the Amiga mouse's movements into 1351 compatible signals. This allows you to use an Amiga mouse with existing C64/C65 software that expect a 1351 mouse.
- Joystick 1 Amiga Mouse Detection can be set to conservative or aggressive. If you use an Amiga mouse, and it fails to move smoothly in all directions, you may set it to **aggressive**. Conversely, if you regularly use joysticks in the port, and have difficulties with the joystick input mis-behaving, you may select the **conservative** option.
- Joystick 2 Amiga Mouse Mode is the same as the first option, but for the second joystick port. This allows you to have different policies for each port.
- Joystick 2 Amiga Mouse Detection similarly provides the ability to separately control the Amiga mouse detection algorithm for the second joystick port.

## Chipset



- *Real-Time Clock* allows setting the MEGA65's Real-Time Clock for those models that include one. To set the clock or calendar, simply edit the field and press the **RETURN** key. The display does not change while viewing this page, but if you use the cursor left and right keys to select another page and return to this page, the values will update if a Real-Time Clock is fitted and functioning.
- *DMAgic Revision* allows selecting the default mode of operation for the C65 DMAgic DMA controller. This option is only required for ROMs not detected by the MEGA65's HYppo Hypervisor. If you see screen corruption in BASIC, try toggling this option.
- *F011 Disk Controller* This option allows you to select whether the internal 3.5" floppy drive functions using real diskettes, or whether it simply makes noises to add atmosphere when using D81 disk images from the SD card. This merely sets the default option, and you can change this setting, or select a different disk image for use as either or both of the C65 3.5" DOS based drives.
- *Default Disk Image* allows you to choose the D81 disk image used with the internal drive, if the F011 Disk Controller option above is set to use an SD card disk image.

## Video



- *Video Mode* selects whether the MEGA65 starts in PAL or NTSC. The MEGA65 supports true 480p NTSC and 576p PAL double-scan modes, with exact 60Hz / 50Hz frame-rates. This setting sets the default value, and the system can be switched between PAL and NTSC via the Freeze Menu, or under software control by MEGA65-enabled programmes.

## Audio



- *Audio Output* selects whether the SIDs and digital audio channels are combined to provide a mono-aural signal, or whether the left and right tagged audio sources are separated to provide a stereo signal. Again, this setting can be varied from in the Audio Mixer of the Freeze Menu, or under the control of MEGA65-enabled software.
- *Swap Stereo Channels* allows switching the left and right sides of the stereo audio output. This is primarily useful for software that expects left and right SIDs to be at swapped addresses compared with the MEGA65.
- *DAC Algorithm* allows selecting between two different digital to analog conversion algorithms. Both are very good, but you may have a preference for one or the other.
- *Audio Amplifier* allows enabling or disabling the audio amplifier contained in some models of the MEGA65 for certain audio outputs, e.g., internal speaker or loud speaker.

## Network



- *MAC Address* allows you to set the default MAC address of your MEGA65. This can be changed at run-time by MEGA65-enabled software

# CHAPTER

# 5

## Cores and Flashing

- What are cores, and why does it matter?
- Bitstream files
- Selecting a core
- Installing an upgrade core for the MEGA65
- Installing other cores
- Creating cores for the MEGA65
- Replacing the factory core in slot 0

- **Understanding The Core Booting Process**





# WHAT ARE CORES, AND WHY DOES IT MATTER?

The MEGA65 computer uses a versatile chip called an FPGA as its heart. FPGAs are “Field Programmable Gate Arrays”. This is a fancy way of saying that FPGAs are chips that can be programmed to behave impersonate other chips. They do this by configuring their arrays of logic gates to reproduce the circuits of other chips. In this way, FPGAs are not emulation but re-creation of other chips.

FPGAs forget what chip they were pretending to be whenever the power is turned off, or when they are “reconfigured”. This might sound annoying, but it’s actually very powerful. It means that we can tell the FPGA in the MEGA65 to impersonate not just the MEGA65 design as it currently stands, but to impersonate any improvements we make to the design. In other words, we can upgrade the MEGA65 hardware just by providing a new set of instructions to the FPGA. These sets of instructions are called “cores” or “bitstreams”. For the purpose of the MEGA65, these two terms can usually be considered to be interchangeable.

FPGAs are so flexible, that not only is it possible to teach the MEGA65 to be a better MEGA65, but it is also possible to teach the MEGA65’s FPGA to be other interesting home computers. We believe that the FPGA is powerful enough that it could pretend to be a VIC-20 (tm), Commodore PET (tm), Apple II (tm), Spectrum (tm), BBC Micro (tm), or even an Amiga (tm) or one of the 16-bit era game consoles. Unlike some previous FPGA-based retro-computers, the MEGA65, its FPGA instructions, board layout and other information is all available for free under open-source licenses. This means that anyone is free to create other cores for the MEGA65 hardware.

To top it all off, the MEGA65 has enough storage for 7 different sets of FPGA instructions, so that you can easily switch the MEGA65’s “personality” from being a MEGA65 to another of these systems (once the cores are available) and back again.

The remainder of this chapter describes how to select which core to run on the MEGA65, and how to store a core into one of the three (or for MEGA65R2, seven) slots in the flash memory storage.

## BITSTREAM FILES

Firstly, there are a variety of files related to the project’s bitstreams/cores that we need to get familiar with, in order to decide what file-types are needed for what occasion.

## File types

| File-type | Purpose                                                                                                                                                                                                                                                                                                                            |
|-----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| .cor      | The MEGA65 project's custom bitstream file format, containing extra header information to help identify the bitstream and the specific MEGA65 target device it is intended for. The MEGA65's flashing utility makes use of this additional information to assure you don't accidentally flash the bitstream of a different device. |
| .mcs      | The bitstream file in a format needed when flashing it to your device's QSPI flash memory chip via Vivado                                                                                                                                                                                                                          |
| .prm      | This (optional) file contains checksum information that can be used by Vivado to verify the .mcs file you have tried to flash                                                                                                                                                                                                      |
| .bit      | A plain bitstream file that can be copied onto your sd-card                                                                                                                                                                                                                                                                        |

## Where to download

Visit the following url:

<https://mega.scryptos.com/sharefolder-link/MEGA/MEGA65+filehost/Bitstreams/Jenkins-Out/mega65-core/development/>

Sort the list of build folders by 'Date' and click on the latest:

The screenshot shows a file manager interface with a sidebar on the left and a main content area. The sidebar includes icons for Home, MEGA, ShareFolder, Bitstreams, Jenkins-Out, mega65-core, and development. The main content area displays a list of build folders. The columns are Name, Version, Size, Date (which is highlighted with a pink box), and Created by. Below the table are navigation links for Previous and Next. The folder '28-73a2cab' is highlighted with a pink box and has a pink arrow pointing to the 'Date' column header. The other entries are '77-c755111' and '76-hcf0765'. The 'Created by' column shows UploadBot for the first two and UnloadBot for the third.

| Name       | Version | Size   | Date              | Created by |
|------------|---------|--------|-------------------|------------|
| ← Previous |         |        |                   |            |
| 28-73a2cab |         | 132 MB | 18.06.2021, 15:11 | UploadBot  |
| 77-c755111 |         | 134 MB | 15.06.2021, 05:51 | UploadBot  |
| 76-hcf0765 |         | 132 MB | 07.06.2021, 18:40 | UnloadBot  |

For the purposes of this chapter on core-flashing, download the desired .cor file that suits your target device:

- **nexys4.cor** (for Nexys4 PSRAM boards)
- **nexys4ddr-widget.cor** (for Nexys4 DDR boards)
- **mega65r2.cor** (for MEGA65R2 boards)
- **mega65r3.cor** (for MEGA65R3 boards, aka DevKits)

- You can also find .bit, .mcs and .prm files located here too.

Alternatively, if you intend to flash the QSPI chip via Vivado, you'd instead download the .mcs file for your target device (and optionally, the .prm files too).

Or another alternative Nexys4 board users is to download .bit files and put them on their sd-cards, you can grab such files from here also.

But once again, to confirm, for the purposes of this chapter on core-flashing, you will only be interested in the .cor files.

## SELECTING A CORE

To operate the MEGA65 using an alternative core, turn off the power to the MEGA65, and then hold the **NO SCROLL** key down while turning the power back on. This instructs the MEGA65 to enter the flash and core menu, instead of booting normally. You should see a display like the following:



To select a core and start it, use the cursor keys to highlight the desired core, and then press the **RETURN** key. If you select a flash slot that does not contain a valid core, it will highlight in red to indicate that it cannot be booted from:



Alternatively, you can press the number corresponding to the core you would like to use. The MEGA65 immediately reconfigures the FPGA, and launches the core. If for some reason the core is faulty, the MEGA65 may instead restart normally after a few seconds, and depending on the circumstances, take you automatically back into the menu.

The MEGA65 will keep running the new core until you physically power it off. Pressing the reset button will not reset which core is being run.

## INSTALLING AN UPGRADE CORE FOR THE MEGA65

To install an upgrade core for the MEGA65, there are few easy steps.

First, copy the core file onto the MEGA65's SD card. You can do this by removing the SD card and inserting it into another computer that has internet access, and downloading the core from that computer. Alternatively you can insert an SD card that already contains the upgrade core. Finally, you can use the MEGA65 TFTP Server program and the MEGA65's Ethernet port to copy the core upgrade file onto the SD card from another computer on your local network.

The flash menu will preferentially use the external microSD slot over the internal SD card: If you have both a microSD card and SD card inserted in your MEGA65, the flash menu will currently ignore the internal SD card. Simply copy the core file(s) from the internal SD card to the external microSD card, or temporarily remove the external microSD card from the rear of your MEGA65, so that the flash menu will be able to find the core files. Also note that the flash menu currently supports only DOS-style 8.3 character filenames. If your core files have a longer name, you will need to rename them when copying them onto your microSD or SD card.

Second, once you have the upgrade core on the MEGA65's SD card, enter the flash and core menu as above, i.e., turn off the power, hold the **NO SCROLL** key down while turning the power on. When the flash and core menu appears, hold the **CTRL** key down and press the **1** key (or **CTRL** and a different number if you wish to replace the contents of a different flash slot). The MEGA65 will present you a list of core files that are on the SD card, similar to this:



Select the upgrade core file you wish to install using the cursor keys, and then pressing the **RETURN** key. The MEGA65 will then erase the flash slot, before writing the upgraded core. You will see a progress bar while the MEGA65 erases the flash slot, similar to this:



```
Erasing flash slot...
sector at $00436000 erased.
```

The progress bar will then reset, and the MEGA65 will write the new core into the slot. This process can take up to 15 minutes, depending on the size of the core file. If you simply wish to erase a flash slot, you can select the “- erase slot -” option instead of a file name. This will perform only the erasure part of the process.

It is important to not turn the power off during this process. If you do, the core file will be only partially installed, and the MEGA65 may not start properly. While inconvenient, if this happens, it won’t damage your MEGA65 or leave it in an unusable state: It will just fall back to using the factory supplied core. If this happens, enter the flash and core menu as described above, and follow the instructions again.

When the flashing process completes, you will see a message like this, indicating that the process is complete:

```
Erasing flash slot...
&Flash slot successfully written.
Press any key to return to menu.
```

When this happens, simply turn off the power to the MEGA65 and turn it back on for it to start using the upgraded core. This is because the MEGA65 will always try to automatically start the core in slot 1 when it is turned on.

## INSTALLING OTHER CORES

Installing other cores works very similarly to installing upgrade cores. The only difference is that you press **CTRL** and **2** to **7** from the flash and core menu, so that the core gets installed in another slot.

Of course, there is nothing stopping you installing a different core in slot 1, so that the MEGA65 behaves as a different type of computer when you turn it on. If you do this, you can always choose to run the MEGA65 core by entering the flash and core menu, and selecting the MEGA65 core.

# CREATING CORES FOR THE MEGA65

If you would like to create your own cores for the MEGA65, or help contribute to the MEGA65 core, then you may also wish to take a look at Chapter/Appendix S on page S-5 which explains how to use the FPGA development tools to flash the MEGA65.

## REPLACING THE FACTORY CORE IN SLOT 0

Replacing the core in slot 0 is not recommended, because if you mess it up, it will brick the machine. This may require that you have to purchase a TE-0790 JTAG programmer, open your MEGA65 case, install the module, go through some rather convoluted software preparation steps, similar to if you were creating your own bitstream/core, and then restore a working bitstream into the slot.

The MEGA65 is an open system though. Therefore we have not made it impossible for you to do this, just very hard: There is a secret key press in the flash menu that will then challenge you with a series of questions of increasing difficulty, to ensure that you know what you are doing. Only after you have correctly answered these questions, will you be given the option to erase and/or replace the contents of slot 0. We are purposely not documenting the method for doing this, nor further details of the questions you will be asked.

However, there really should be no reason for using this method to replace the contents of slot 0: If you want to make your own bitstreams/cores, you can either put them in other slots, and use the flash menu to activate them, or you should simply use a TE-0790 JTAG adaptor, and then use the Vivado or other FPGA development tools to write to the flash directly. That method is also somewhat faster than flashing through the flash menu.

You have been warned!

## UNDERSTANDING THE CORE BOOTING PROCESS

This section summarises how the MEGA65 selects which core to start when it is powered on. The process is as shown in the following figure: When the MEGA65 is powered on, it always starts the bitstream stored in slot 0 of the flash. If that is the MEGA65 Factory Core, the MEGA65 HYPPO Hypervisor starts. If it is the first boot since power-on, HYPPO

starts the Flash Menu programme – but note that the Flash Menu in this mode may not show anything on the screen to indicate that it is running!

The Flash Menu checks if the system is booting from Flash Slot 0. If it is, it checks if the NO SCROLL key is being held. If it is, the Flash Menu programme shows its display, allowing the user to select or re-flash a core. If the NO SCROLL key is not being pressed, then the Flash Menu programme checks if Flash Slot 1 contains a valid core. If it does, then the Flash menu programme attempts to start that core. If it succeeds, then the system reconfigures to that core, after which the behaviour of the system is according to that core. If it fails, the keyboard will go into “ambulance mode” showing flashing blue lights to indicate that some first-aid is required. Note that in “ambulance mode” the reset button has no effect: You must turn the computer off and on again.

If the user selected a different core in the Flash Menu, the process is similar, except that the ambulance lights will appear for only a limited time, as the FPGA will automatically search through the flash memory until it finds a valid core. If it gets to the end of the flash memory, it will start the MEGA65 Factory Core from slot 0 again.



# PART

## FIRST STEPS IN CODING



# CHAPTER

# 6

## How Computers Work

- Computers are stupid. Really stupid



Did you know that many computer experts and programmers learned how to use computers when they were still small children? Home computers only became common in the early 1980s. They were so new, that people would often write programmes to do what they wanted to do, because no software existed to do the job for them.

It was also quite common for people working in all sorts of office jobs to learn how to program the computers they used for their jobs. For example, the people processing payroll for a company would often learn how to program the computer to calculate the everyone's pay!

Things have changed a lot since then, though. Now most people choose existing programmes or apps to do what they need, and think that programming is a specialised skill that only some people have the ability to learn. But this isn't true. Of course, like every other field of pursuit everyone will be better at some things than others, whether it be sports, knitting, maths or writing. But almost everyone is able to learn enough to help them in their life.

We created the MEGA65, because we believe that YOU can learn to programme, so that computers can be more useful to you, and as with learning any new skill, that you can have the satisfaction and enjoyment and new adventures that this brings!

## COMPUTERS ARE STUPID. REALLY STUPID

How can this be so? Computers are able to do so many different things, often thousands of times faster than a person can. So how can we say that computers are stupid? The answer is that no computer can do anything that it hasn't been instructed by a person to do. Even the latest Artificial Intelligence systems were instructed how to learn (or how to learn, how to learn). To understand why this is so, it is helpful to understand how computers really work.

### Making an Egg Cup Computer

The heart of a computer is its Central Processing Unit, or CPU for short. Many modern computers have more than one CPU, but let's keep things simple to begin with. The CPU has a set of simple instructions that it understands, like, "get the thing from cup #21," "put this thing into cup #403," "add these things together," or "do the following instruction, but only if the thing in cup #712 is the number 3."

But what do we mean with all of these "things" and "cups"? Let's start by thinking about how we could pretend to be a computer using just an empty egg carton, some small pieces of paper and a pencil or pen. Start by writing numbers, beginning with one, in

each of the little egg cups in the egg carton. Then write the number zero on a little scrap of paper and put it in the first cup. Do the same for the other cups. You should now have an egg carton with numbered cups, and with every cup having a scrap of paper with the number zero written on it. Now we just need to decide on a few simple rules that will explain how our egg-cup computer will work:

- First, each cup is allowed to hold exactly one thing at a time. Never more. Never less. This so that when we ask the question “what is in box such-and-such,” that there is a single clear answer. It’s also how computer memory works: Each piece of memory can hold only one thing at a time.
- Second, we need a way for the computer to know what to do next. On most computers this is called the Programme Counter, or PC, for short (not to be confused with PC when people are talking about a Personal Computer). The PC is just the number of the next of the next memory location (or in our case, egg-cup), that the computer will examine, when deciding what to do next. You might like to have another piece of paper that you can use to write the PC number on as you go along.
- Third, we need to have a list of things that the egg-cup computer will do, based on what number is in the egg-cup indicated by the PC.

So let’s come up with the set of things that the computer can do, based on the number in the egg-cup indicated by the PC. We’ll keep things simple with just the following:

| Number in the egg-cup | Action                                                                                                                                                                                                                                                                                                                                   |
|-----------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0                     | i) Add one to the PC, and do nothing else.                                                                                                                                                                                                                                                                                               |
| 1                     | i) Add one to the PC.<br>ii) Set the PC to be the number stored in that egg-cup.                                                                                                                                                                                                                                                         |
| 2                     | i) Add one to the PC.<br>ii) Add the number in the egg-cup indicated by the PC to the number in the egg-cup indicated by the number in the egg-cup following that.<br>iii) Put the answer in the egg-cup indicated by the egg-cup following that.<br>iv) Finally, add two more to the PC, to skip over the egg-cups that we made use of. |

Don’t worry if that sounds a bit confusing for now, specially that last one – we will go through it in detail very soon! The best way to explain it, is to go through some examples.

# CHAPTER 7

## Getting Started in BASIC

- Your first BASIC programmes
- First steps with text and numbers
- Making simple decisions
- Random numbers and chance



It is possible to code on the MEGA65 in many languages, however most people start with BASIC. That makes sense, because BASIC stands for Beginner's All-purpose Symbolic Instruction Code: It was made for people like you to get started with in the world of coding!

A few short words before we dive in: BASIC is a programming language, and like spoken language it has conventions, grammar and vocabulary. Fortunately, it is much quicker and easier to learn than our complex human languages. But if you pay attention, you might notice some of these structures, and that can help you along your path in the world of coding.

If you haven't already read Chapter/Appendix 3 on page 3-3, it might be a good idea to do so. This will help you be able to more confidently interact with the MEGA65 computer.

It's also great to remember that if you really confuse the MEGA65, you can always get back to the READY. prompt by just pressing the reset button on the left-hand side of the keyboard, or if that doesn't help, then by turning it off and on again using the power switch on the left-hand side of the keyboard. You don't have to worry about shutting the computer down properly or any of that nonsense. The only thing to remember is that if you had any unsaved work, it will be lost when you turn the computer off and on again or press the reset button.

Finally, if you don't understand all of the descriptions and information with an example – don't worry! We have provided as much information as we can, so that it is there in case you have questions, encounter problems are just curious to discover more. Feel free to skip ahead to the examples and try things out, and then you can go back and re-read it when you are motivated to find something out, or help you work though a problem. And if you don't find the answer to your problem, send us a message! There are support forums for the MEGA65 at <https://mega65.net>, and you can report problems with this guide at:

<https://github.com/mega65/mega65-user-guide>

We hope you have as much fun learning to programme the MEGA65 as we have had making it!

## YOUR FIRST BASIC PROGRAMMES

The MEGA65 was designed to be programmed! When you turn it on, it takes a couple of seconds to get its house in order, and then it quickly shows you a "READY." prompt and flashing block called the cursor. When the cursor is blinking, it tells you that the computer is waiting for input. The "READY." message tells you that the BASIC programming language is running and ready for you to start programming. You don't even need to load any programmes – you can just get started.

Try typing the following into the computer and see what happens:

HELLO COMPUTER

To do this, just type the letters as you see them above. The computer will already be in upper-case mode, so you don't need to hold the **SHIFT** or **CAPS LOCK** key down. When you have typed "HELLO COMPUTER" press the **RETURN** key. This tells the computer you want it to accept the line of input you have typed. When you do this, you should see a message something like the following:



If you saw a **SYNTAX ERROR** message something like that one, then congratulations: You have succeeded in communicating with the computer! Error messages sound much nastier than they are. The MEGA65 uses them, especially the syntax error to tell you when it is having trouble understanding what you have typed, or what you have put in a programme. They are nothing to be afraid of, and experienced programmers get them all the time.

In this case, the computer was confused because it doesn't understand the word "hello" or the word "computer". That is, it didn't know what you wanted it to do. In this regard, computers are quite stupid. They know only a few words, and aren't particularly imaginative about how they interpret them.

So let's try that again in a way that the computer will understand. Try typing the following in. You can just type it right away. It doesn't matter that the syntax error message can still be seen on the screen. The computer has already forgotten about that by the time it told you **READY.** again.

```
PRINT "HELLO COMPUTER"
```

Again, make sure you don't use shift or shift-lock while typing it in. The symbols around the words **HELLO COMPUTER** are double-quotes. If you are used to an Australian or American keyboard, you might discover that they double-quote key is in a rather different place to where you are used to: Double-quotes can be typed on the MEGA65 by holding down the **SHIFT** key, and then pressing 2. Don't forget to press the **RETURN** key when you are done, so that the computer knows you want it to do something with your input.

If you make a mistake while typing, you can use the **INST DEL** key to rub out the mistake and fix it up. You can also use the cursor keys to move back and forth on the line while you edit the line you are typing, but there is a bit of a trick if you have already typed a double-quote: If you try to use the cursor keys, it will print a funny reversed symbol instead of moving the cursor. This is because the computer thinks you want to record moving the cursor in the text itself, which can be really useful and fun, and which you can read more about in Chapter/Appendix 3 on page 3-3. But for now, if you make a mistake just press the **RETURN** key and type the messed up line again.

Hopefully now you will see something like the following:

The screenshot shows a Commodore C65 Development System interface. At the top, it displays "THE COMMODORE C65 DEVELOPMENT SYSTEM", "COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.", and "COPYRIGHT 1977 MICROSOFT". Below that, it says "BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED". The main area of the screen shows the following text:  
ENGLISH KEYBOARD  
NO EXPANSION RAM  
C1565 DRIVE  
ROM CHECKSUM \$4BCF  
READY.  
HELLO COMPUTER  
?SYNTAX ERROR  
READY.  
PRINT"HELLO COMPUTER"  
HELLO COMPUTER  
READY.  
A small portion of the Commodore logo is visible on the left side of the screen.

This time no new **SYNTAX ERROR** message should appear. But if some kind of error message has appeared, just try typing in the command again, after taking a close look to work out where the mistake might be.

Instead of an error, we should see **HELLO COMPUTER** repeated underneath the line you typed in. The reason this happened is that the computer does understand the word **PRINT**. It knows that whatever comes after the word **PRINT** should be printed to the screen. We had to put **HELLO COMPUTER** inside double-quotes to tell the computer that we want it to be printed literally.

If we hadn't put the double-quotes in, the computer would have thought that **HELLO COMPUTER** was the name of a stored piece of information. But because we haven't stored any piece of information in such a place, the computer will have zero there, so the computer will print the number zero. If the computer prints zero or some other number when you expected a message of some sort, this can be the reason.

You can try it, if you like, and you should see something like the following:

```
COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.  
COPYRIGHT 1977 MICROSOFT  
BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED  
  
ENGLISH KEYBOARD  
NO EXPANSION RAM  
C1565 DRIVE  
ROM CHECKSUM $4BCF  
  
READY.  
HELLO COMPUTER  
  
?SYNTAX ERROR  
READY.  
PRINT "HELLO COMPUTER"  
HELLO COMPUTER  
  
READY.  
PRINT HELLO COMPUTER  
0  
  
READY.
```

In the above examples we typed commands in directly, and the computer executed them immediately after you pressed the **RETURN** key. This is why typing commands in this way is often called *direct mode* or *immediate mode*.

But we can also tell the computer to remember a list of commands to execute one after the other. This is done using the rather unimaginatively named *non-direct mode*. To use non-direct mode, we just put a number between 0 and 63999 at the start of the command. The computer will then remember that command. Unlike when we executed a direct-mode command, the computer doesn't print **READY.** again. Instead the cursor just reappears on the next line, ready for us to type in more commands.

Let's try that out with a simple little programme. Type in the following three lines of input:

```
1 FOR I = 1 TO 10 STEP 1  
2 PRINT I  
3 NEXT I
```

When you have done this, the screen should show something like this:



If it doesn't work, you can try again. Don't forget, if you feel that the computer is getting all muddled up, you can just press the reset button or flip the power switch off and on on the left side of the computer to reboot it. This only takes a couple of seconds, and doesn't hurt the MEGA65 in anyway.

We have told the computer to remember three commands, that is, **FOR I = 1 TO 10**, **STEP 1**, **PRINT I** and **NEXT I**. We have also told the computer which order we would like to run them in: The computer will start with the command with the lowest number, and execute each command that has the next higher number in turn, until it reaches the end of the list. So it's a bit like a reminder list for the computer. This is what we call a programme, a bit like the programme at a concert or the theatre, it tells us what is coming up, and in what order. So let's tell the computer to execute this programme.

But first, let's try to guess what will happen. Let's start with the middle command, **PRINT I**. We've seen the **PRINT** command, and we know it tells the computer to print things to the screen. The thing it will try to print is **I**. Just like before, because there are no double-quotes around the **I**, it will try to print a piece of stored information. The piece of information it will try to print will be the piece associated with the thing **I**.

When we give a piece of information like this a name, we call it a *variable*. They are called variables because they can vary. That is, we can replace the piece of information associated with the variable called **I** with another piece of information. The old piece will be forgotten as a result. So if we gave a command like **LET I = 3**, this would replace whatever was stored in the variable called **I** with the number 3.

Back to our programme, we now know that the 2<sup>nd</sup> command will try to print the piece of information stored in the variable **I**. So lets look at the first command: **FOR I = 1 TO 10 STEP 1**. Although we haven't seen the **FOR** command before, we can take a bit of a guess at how it works. It looks like it is going to put something into the variable **I**. That something seems to have something to do with the range of number 1 through 10, and a step or interval of 1. What do you think it will do?

If you guessed that it will put the values 1, 2, 3, 4, 5, 6, 7, 8, 9 and then 10 into the variable **I**, then you can give yourself a pat on the back, because that's exactly what it does. It also helps us to understand the 3<sup>rd</sup> command, **NEXT I**: That command tells the computer to put the next value into the variable **I**. And here is a little bit of magic: When the computer does that, it goes back up the list of commands, and continues again from the command after the **FOR** command.

So lets pull that together: When the computer executes the first command, it discovers that it has to put 10 different values into the variable **I**. It starts by putting the first value in there, which in this case will be the number 1. The computer then continues to the second command, which tells the computer to print the piece of information that is currently stored in the variable called **I**. That will be the number 1, since that was the last thing the computer was told to put there. Then the computer proceeds to the third command, which tells it that it is time to put the next value into the variable **I**. So the computer will throw away the number 1 that is currently in the variable **I**, and put the number 2 in there, since that is the next number in the list. It will then continue from the 2<sup>nd</sup> command, which will cause the computer to print out the contents of the variable **I** again. Except that this time **I** has had the number 2 stored in it most recently, so the computer will print the number 2. This process will repeat, until the computer has printed all ten values that the **FOR** command indicated it to do.

To see this in action, we need to tell the computer to execute the programme of commands we typed in. We do this by using the **RUN** command. Because we want it to run the programme immediately, we should use immediate mode (remember, this is another name for direct mode). So just type in the word **RUN** and press the **RETURN** key. You should then see a display that looks something like the following:

The screenshot shows a terminal window for BASIC 10.0 version 0.98.911001. The title bar reads "BASIC 10.0 V0.98.911001 ALL RIGHTS RESERVED". The window displays the following text:  
ENGLISH KEYBOARD  
NO EXPANSION RAM  
C1565 DRIVE  
ROM CHECKSUM \$4BCF  
  
READY.  
1 FOR I = 1 TO 10 STEP 1  
2 PRINT I  
3 NEXT I  
RUN  
1  
2  
3  
4  
5  
6  
7  
8  
9  
10  
  
READY.

You might notice a couple of things here:

First, the computer has told us it is **READY**. again as soon as it finished running the programme. This just makes it easier for us to know when we can start giving commands to the computer again.

Second, when the computer got to the bottom of the screen it automatically scrolled the display up to make space. This is quite normal. What is important to remember, is that the computer forgets everything that scrolls off the top. The only exception is if you have told the computer to remember a command by putting a number in front of it. So our programme is quite safe for now. We can see that this is the case by typing the **RUN** command a couple more times: The programme listing will have scrolled off the top of the screen, but we can still RUN the programme, because the computer has remembered it. Give it a try! Did it work?

If you wish to see the programme of remembered commands, you can use the **LIST** command. This command causes the computer to display the remembered programme of commands to the screen, like in the display here. If you would like to replace any of the commands in the programme, you can type a new line that has the same number as the one you wish to change.

```
9  
10  
  
READY.  
RUN  
1  
2  
3  
4  
5  
6  
7  
8  
9  
10  
  
READY.  
LIST  
  
1 FOR I = 1 TO 10 STEP 1  
2 PRINT I  
3 NEXT I  
  
READY.
```

For example, to print the results all on one line, we could modify the second line of the programme to **PRINT I;** by typing the following line of input and pressing the **RETURN** key:

```
2 PRINT I;
```

You can make sure that the change has been remembered by running the **LIST** command again, as we can see here. You can then use the **RUN** command to run the modified programme, like this:

```
1  
2  
3  
4  
5  
6  
7  
8  
9  
10  
  
READY.  
LIST  
  
1 FOR I = 1 TO 10 STEP 1  
2 PRINT I  
3 NEXT I  
  
READY.  
2 PRINT I;  
LIST  
  
1 FOR I = 1 TO 10 STEP 1  
2 PRINT I;  
3 NEXT I  
  
READY.  
RUN  
1 2 3 4 5 6 7 8 9 10  
READY.
```

It is quite easy to modify your programmes in this way. As you become more comfortable with the process, there are two additional helpful tricks:

First, you can give the **LIST** command the number of a command, or line as they are referred to, and it will display only that line of the programme. Alternatively, you can give a range separated by a minus sign to display only a section of the programme, e.g., **LIST 1 - 2** to list the first two lines of our programme.

Second, you can use the cursor keys to move the cursor to a line which has already been remembered and is displayed on the screen. If you modify what you see on the screen, and then press the **RETURN** key while the cursor is on that line, the BASIC interpreter will read in the modified line and replace the old version of it. It is important to note that if you modify multiple lines of the programme at the same time, you must press the **RETURN** key on each line that has been modified. It is good practice to check that the programme has been correctly modified. Use the **LIST** command again to achieve this.

## Exercises to try

### 1. Can you make it count to a higher or lower number?

At the moment it counts from 1 to 10. Can you change it to count to 20 instead? Or to count from 3 to 17? Or how about from 14.5 to 21.5? What do you think you would need to reverse the order in which it counts?

*Clue:* You will need to modify the **FOR** command.

## **2. Can you change the counting step?**

At the moment it counts by ones, i.e., each number is one more than the last. Can you change it to count by twos instead? Or by halves, so that it counts 1, 1.5, 2, 2.5, 3, ...?

*Clue:* You will need to modify the **STEP** clause of the **FOR** command.

## **3. Can you make it print out one of the times tables?**

At the moment it prints the answers to the 1 times tables, because it counts by ones. Can you make it count by threes, and show the three times tables?

*Clue:* You will need to modify the **FOR** command.

## **4. Can you make it print out the times tables from $1 \times 1$ to $10 \times 10$ ?**

*Clue:* You might like to use ; on the end of **PRINT** command, so that you can have more than one entry per line on the screen.

*Clue:* The **PRINT** command without any argument will just advance to the start of the next line.

*Clue:* You might need to have multiple **FOR** loops, one inside the other.

# FIRST STEPS WITH TEXT AND NUMBERS

In the last section we started to use both numbers and text. Text on computers is made by stringing individual letters and other symbols together. For this reason they are called *strings*. We also call the individual letters and symbols *characters*. The name character comes from the printing industry where each of the symbols that can be printed on a page. For computers, it has much the same meaning, and the set of characters that a computer can display is rather unimaginatively called a *character set*.

When the MEGA65 expects some for of input, it is typically looking for one of four things:

1. a keyword like **PRINT** or **STEP**, which are words that have a special meaning to the computer;
2. a variable name like **I** or **A\$** that it will then use to either store or retrieve a piece of information;
3. a number like **42** or **-30.3137**; or
4. a string like **"HELLO COMPUTER"** or **"23 KILOMETRES"**.

Sometimes you have a choice of which sort of thing you can provide, while other times you have less choice. What sort of thing the computer will accept depends on what you are doing at the time. For example, in the previous section we discovered that when the computer tells us that it is **READY**, that we can give it a keyword or a number. Do you think that the computer will accept all four kinds of thing when it says **READY**.? We already know that keywords and numbers and keywords can be entered, but what about variable names or strings? Let's try typing in a variable name, say **N**, and pressing the  key, and see what happens. And then lets try with a string, say "**"THIS IS A STRING"**".



THE COMMODORE C65 DEVELOPMENT SYSTEM  
COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.  
COPYRIGHT 1977 MICROSOFT  
BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED

ENGLISH KEYBOARD  
NO EXPANSION RAM  
C1565 DRIVE  
ROM CHECKSUM \$4BCF

READY.  
**N**  
2SYNTAX ERROR  
READY.  
**"THIS IS A STRING"**  
2SYNTAX ERROR  
READY.

You should get a syntax error each time, telling you that the computer doesn't understand the input you have given it. Let's start with when you typed the variable: If you just tell the computer the name of a stored piece of information, it doesn't have the foggiest idea what you are wanting it to do. It's the same when you give it a piece of information, like a string, without telling the computer what to do with it.

But as we discovered in the last section, we can tell the computer that we want to see the piece of information that is stored in a variable using the **PRINT** command. So we could instead type in **PRINT N**, and the computer would know what to do, and will print the piece of information stored in the variable called **N**.

In fact, using the **PRINT** command is so common, that programmers got annoyed having to type in the **PRINT** command all the time, that they made a short cut: If you type a question mark character, i.e., a ?, the computer knows that you mean **PRINT**. So for example if you type **? N**, it will do the same as typing **PRINT N**. Of course, you have to press the  key after each command to tell the computer you want it to process what you typed. From here on, we will assume that you can remember to do that, without being reminded.

The **?** shortcut also works if you are telling the computer to remember a command as part of a programme. So if you type **I ? N**, and then **LIST**, you will see **I PRINT N**, as we can see in the following screen-shot:

```
READY.  
N  
?SYNTAX ERROR  
READY.  
"THIS IS A STRING"  
?SYNTAX ERROR  
READY.  
PRINT N  
0  
READY.  
? N  
0  
READY.  
I ? N  
LIST  
I PRINT N  
READY.
```

Like we saw in the last section, the variable **N** has not had a value stored in it, so when the computer looks for what is there, it finds nothing. Because **N** is a *numeric variable*, when there is nothing there, this means zero. If it was a *string variable*, then it would have found literally nothing. We can try that, but first we have to explain how we tell the computer we are talking about a string variable. We do that by putting a dollar sign character, i.e., a **\$**, on the end of the variable name. So if we put a **\$** on the end of the variable name **N**, it will refer to a string variable called **N\$**.

We can experiment with these variables by using the hopefully now familiar **PRINT** command (or the **?** shortcut) to see what is in the variables. But we need a convenient way to put values into them. Fortunately we aren't the first people to want to put values into variables, and so the **LET** exists. The **LET** command is used to put a value into a variable. For example, we can tell the computer:

```
LET N = 5.3
```

This tells the computer to put the value 5.3 into the variable **N**. We can then use the **PRINT** command to check that it worked. Similarly, we can put a value into the variable **N\$** with something like:

```
LET N$ = "THE KING OF THE POTATO PEOPLE"
```

If we try those, we will see something like the following:

```
? N  
5.3  
  
READY.  
1 ? N  
LIST  
  
1 PRINT N  
  
READY.  
LET N = 5.3  
  
READY.  
? N  
5.3  
  
READY.  
LET N$ = "THE KING OF THE POTATO PEOPLE"  
  
READY.  
? N$  
THE KING OF THE POTATO PEOPLE  
  
READY.
```

We mentioned just before that **N** is a numeric variable and that **N\$** is a string variable. This means that we can only put numbers into **N** and strings into **N\$**. If we try to put the wrong kind of information into a variable, the computer will tell us that we have mis-matched the kind of information with the place we are trying to put it by giving us a **TYPE MISMATCH ERROR** like this:

```
READY.  
LET N = 5.3  
  
READY.  
? N  
5.3  
  
READY.  
LET N$ = "THE KING OF THE POTATO PEOPLE"  
  
READY.  
? N$  
THE KING OF THE POTATO PEOPLE  
  
READY.  
LET N = "MR FLIBBLE"  
  
?TYPE MISMATCH ERROR  
READY.  
LET N$ = 42  
  
?TYPE MISMATCH ERROR  
READY.
```

This leads us to a rather important point: **N** and **N\$** are separate variables, even though they have similar names. This applies to all possible variable names: If the variable name has a **\$** character on the end, it means it is a string variable quite separate from the similarly named numeric variable. To use a bit of jargon, this means that each *type* of variable has their own separate *name spaces*.

(There are also four other variable name spaces that we haven't talked about yet: integer variables, identified by having a **%** character at the end of their name, e.g., **N%**, and arrays of numeric, string or integer variables. But don't worry about those for now. We'll talk about those a bit later on.)

So far we have only given values to variables in direct mode, or by using constructions like **FOR** loops. But we haven't seen how we can get information from the user when a programme is running. One way that we can do this, is with the **INPUT** command.

**INPUT** is quite easy to use: We just have to say which variable we would like the input to go into. For example, to tell the computer to ask for the user to provide something to put into the variable **A\$**, we could use something like **INPUT A\$**. The only trick with the **INPUT** command is that it cannot be used in direct mode. If you try it, the computer will tell you **ILLEGAL DIRECT ERROR**. Try it, and you should see something like the following



This means that the **INPUT** command can only be used as part of a programme. So we can instead do something like the following:

```
1 INPUT A$  
2 PRINT "YOU TYPED "; A$  
RUN
```

What do you think that this will do? The first line will ask the computer for something to put into the variable **A\$**, and the second line will print the string "**"YOU TYPED"**", followed by what the **INPUT** command read from the user. Let's try it out:

The screenshot shows the Commodore C64 Development System interface. At the top, it displays "THE COMMODORE C64 DEVELOPMENT SYSTEM", "COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.", and "COPYRIGHT 1977 MICROSOFT". Below that, it says "BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED". The system configuration listed is "ENGLISH KEYBOARD", "NO EXPANSION RAM", "C1565 DRIVE", and "ROM CHECKSUM \$4BCF". The screen shows the following BASIC code:

```
READY.  
INPUT A$  
?ILLEGAL DIRECT ERROR  
READY.  
1 INPUT A$  
2 PRINT "YOU TYPED "; A$  
RUN  
? ■
```

Did you expect that to happen? What is this question mark doing there? The **?** here is the computer's way of telling you that a programme is waiting for some input from you. This means that the computer uses the same symbol, **?**, to mean two different things: If you type it as part of a programme or in direct mode, then it is a short-cut for the **PRINT** command. That's when you type it. But if the computer shows it to you, it has this other meaning, that the computer is waiting for you to type something in. There is also a third way that the computer uses the **?** character. Have you noticed what it is? It is to indicate the start of an error message. For example, a Syntax Error is indicated by **?SYNTAX ERROR**. When a character or something has different meanings in different situations or contexts, we say that it is *context dependent*.

But returning to our example, if we now type something in, and press the **RETURN** key to tell the computer that you are done, the programme will continue, like this:

The screenshot shows the Commodore C64 Development System interface. At the top, it displays "THE COMMODORE C64 DEVELOPMENT SYSTEM", "COPYRIGHT 1991 COMMODORE ELECTRONICS, LTD.", and "COPYRIGHT 1977 MICROSOFT". Below that, it says "BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED". The screen then shows system configuration: "ENGLISH KEYBOARD", "NO EXPANSION RAM", "C1565 DRIVE", and "ROM CHECKSUM \$4BCF". It then displays a series of user inputs and responses:  
READY.  
INPUT A\$  
?ILLEGAL DIRECT ERROR  
READY.  
1 INPUT A\$  
2 PRINT "YOU TYPED "; A\$  
RUN  
? LIGHT SABRE  
YOU TYPED LIGHT SABRE  
READY.

Of course, we didn't really know what to type in, because the programme didn't give any hints to the user as to what the programmer wanted them to do. So we should try to provide some instructions. For example, if we wanted the user to type their name, we could print a message asking them to type their name, like this:

```
1 PRINT "WHAT IS YOUR NAME"  
2 INPUT A$  
3 PRINT "HELLO "; A$
```

Now if we run this programme, the user will get a clue as to what we expect them to do, and the whole experience will make a lot more sense for them:

```
ROM CHECKSUM $4BCF
READY.
INPUT A$  

?ILLEGAL DIRECT ERROR
READY.
1 INPUT A$  

2 PRINT "YOU TYPED "; A$  

RUN  

? LIGHT SABRE  

YOU TYPED LIGHT SABRE  

READY.  

1 PRINT "WHAT IS YOUR NAME"  

2 INPUT A$  

3 PRINT "HELLO "; A$  

RUN  

WHAT IS YOUR NAME  

? LISTER  

HELLO LISTER  

READY.
```

When we run the programme, we first see the **WHAT IS YOUR NAME** message from line 1. The computer doesn't print the double-quote symbols, because they only told the computer that the piece of information between them is a string. The string itself is only the part in between.

After this we see the **?** character again and the blinking cursor telling us that the computer is waiting for some input from us. The rest of the programmed is *blocked* from continuing until it we type the piece of information. Once we type the piece of input, the computer stores it into the variable **A\$**, and can continue. Thus when it reaches line 3 of the programme, it has everything it needs, and prints out both the **HELLO** message, as well as the information stored in the variable called **A\$**.

Notice that the word **LISTER** doesn't appear anywhere in the programme. It exists only in the variable. This ability to process information that is not part of a programme is one of the things that makes computer programmes so powerful and able to be used for so many purposes. All we have to do is to change the input, and we can get different output.

For example, with our programme we run it again and again, and give it different input each time, and the programme will adapt its output to what we type. Pretty nifty, right? Let's have the rest of the crew try it out:

```
WHAT IS YOUR NAME  
? LISTER  
HELLO LISTER  
  
READY.  
RUN  
WHAT IS YOUR NAME  
? HOLLY  
HELLO HOLLY  
  
READY.  
RUN  
WHAT IS YOUR NAME  
? KRYPTON  
HELLO KRYPTON  
  
READY.  
RUN  
WHAT IS YOUR NAME  
? RIMMER, BSC  
?EXTRA IGNORED  
HELLO RIMMER  
  
READY.
```

We can see that each time the programme prints out the message customised with the input that you typed in...Until we get to **RIMMER, BSC**. As always, Mr. Rimmer is causing trouble. In this case, he couldn't resist putting his Bronze Swimming Certificate qualification on the end of his name.

We see that the computer has given us a kind of error message, **?EXTRA IGNORED**. The error is not written in red, and doesn't have the word **ERROR** on the end. This means that it is a warning, rather than an error. Because it is only a warning, the programme continues. But something has happened: The computer has ignored Mr. Rimmer's **BSC**, that is, it has ignored the extra input. This is because the **INPUT** command doesn't really read a whole line of input. Rather, it reads *one piece of information*. The **INPUT** command thinks that a piece of information ends at the end of a line of input, or when it encounters a comma (,) or colon (:) character.

If you want to include one of those symbols, you need to surround the whole piece of information in double-quotes. So, if Mr. Rimmer had read this guide instead of obsessing over the Space Core Directives, he would have known to type "**"RIMMER, BSC"**" (complete with the double-quotes), to have the programme run correctly. It is important that the quotes go around the whole piece of information, as otherwise the computer will think that the first quote marks the start of a new piece of information. We can see the difference it makes below:

```
LIST
1 PRINT "WHAT IS YOUR NAME"
2 INPUT A$
3 PRINT "HELLO "; A$

READY.
RUN
WHAT IS YOUR NAME
? "RIMMER, BSC"
HELLO RIMMER, BSC

READY.
RUN
WHAT IS YOUR NAME
? "RIMMER" " BSC
?EXTRA IGNORED
HELLO RIMMER"

READY.
```

While this can all be a bit annoying at times, it has a purpose: The **INPUT** command can be used to read more than one piece of information. We do this by putting more than one variable after the **INPUT** command, each separated by a comma. The **INPUT** command will then expect multiple pieces of information. For example, we could ask for someone's name and age, with a programme like this:

```
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT A$, A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
```

If we run this programme, we can provide the two pieces of information on the one line when the computer presents us with the ? prompt, for example **LISTER, 3000000**. Note the comma that separates the two pieces of information, **LISTER** and **3000000**. It's also worth noticing that we haven't put any thousands separators into the number 3,000,000. If we did, the computer would think we meant three separate pieces of information, **3**, **000** and **000**, which is not what we meant. So let's see what it looks like when we give **LISTER, 3000000** as input to the programme:

```
READY.  
RUN  
WHAT IS YOUR NAME  
? RIMMER" " BSC  
?EXTRA IGNORED  
HELLO RIMMER"  
  
READY.  
LIST  
  
1 PRINT "WHAT IS YOUR NAME AND AGE"  
2 INPUT A$; A  
3 PRINT "HELLO "; A$  
4 PRINT "YOU ARE"; A; " YEARS OLD."  
READY.  
  
RUN  
WHAT IS YOUR NAME AND AGE  
? LISTER, 3000000  
HELLO LISTER  
YOU ARE 3000000 YEARS OLD.  
READY.
```

In this case, the **INPUT** command reads the two pieces of information, and places the first into the variable **A\$**, and the second into the variable **A**. When the programme reaches line 3 it prints **HELLO** followed by the first piece of information. Then when it gets to line 4, it prints the string **YOU ARE**, followed by the contents of the variable **A**, which is the number 3,000,000, and finally the string **YEARS OLD**.

It's also possible to just give one piece of information at a time. In that case, the **INPUT** command will ask for the second piece of information with a double question-mark prompt, i.e., **??**. Once it has the second piece of information. (If we had more than two variables on the **INPUT** command, it will still present the same **??** prompt, rather than printing more and more question-marks.)

So if we try this with our programme, we can see this **?** and **??** prompts, and how the first piece of information ends up in **A\$** because it is the first variable in the **INPUT** command. The second piece of information ends up in **A** because **A** is the second variable after the **INPUT** command. Here's how it looks if we give this input to our programme:

```
READY.  
LIST  
1 PRINT "WHAT IS YOUR NAME AND AGE"  
2 INPUT A$,A  
3 PRINT "HELLO "; A$  
4 PRINT "YOU ARE"; A; " YEARS OLD."  
READY.  
  
RUN  
WHAT IS YOUR NAME AND AGE  
? LISTER, 3000000  
HELLO LISTER  
YOU ARE 3000000 YEARS OLD.  
  
READY.  
RUN  
WHAT IS YOUR NAME AND AGE  
? LISTER  
?? 3000000  
HELLO LISTER  
YOU ARE 3000000 YEARS OLD.  
READY.
```

Until now we have been asking the user to input information by using a **PRINT** command to display the message, and then an **INPUT** command to tell the computer which variables we would like to have some information input into. But, like with the **PRINT** command, this is something that happens often enough, that there is a shortcut for it. It also has the advantage that it looks nicer when running, and makes the programme a little shorter. The short cut is to put the message to show after the **INPUT** command, but before the first variable.

We can change our programme to use this approach. First, we can change line 3 to include the prompt after the **INPUT** command. We can do this one of two ways: First, we could just type in a new line 3. The computer will automatically replace the old line 3 with the new one.

But, as we have mentioned a few times now, programmers are lazy beasts, and so there is a short-cut: If you can see the line on the screen that you want to change, you can use the cursor keys to navigate to that line, edit it on the screen, and then press the **RETURN** key to tell the computer to accept the new version of the line.

Either way, you can check that the changes succeeded by typing the **LIST** command on any line of the screen that is blank. This will show the revised version of the programme. For example:

```
LIST
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT A$,A,B
3 PRINT "HELLO ", A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
LIST
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
3 PRINT "HELLO ", A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
```

We still have a little problem, though: Line 1 will print the message **WHAT IS YOUR NAME AND AGE**, and then Line 2 will print it again! We only want the message to appear once. Thus we would like to change line 1 so that it doesn't do this any more. Because there is no other command on line 1 that we want to keep, that line can just become empty. So we can type in something like this:

```
1
```

We can confirm that the contents of the line have been deleted by running the **LIST** command again, like this:

```
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT A$,A,B
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
LIST
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
1
LIST
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
```

Did you notice something interesting? When we told the computer to make line 1 of the programme empty, it deleted it completely! That's because the computer thinks that an empty line is of no use. It also makes sure that your programmes don't get all cluttered up with empty lines if you make lots of changes to your programmes.

It is also possible to **DELETE** a range of lines. For example (but don't do this now), you could delete lines 3-4 with:

```
DELETE 3-4
```

You can read more about the **DELETE** command in the BASIC 65 Command Reference.

With that out the way, let's run our programme and see what happens. As usual, just type in the **RUN** command and hit the **RETURN** key. You should see something like this:

```
1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT A$,B
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; B; " YEARS OLD."
READY.
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
LIST

1 PRINT "WHAT IS YOUR NAME AND AGE"
2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; B; " YEARS OLD."
READY.
1
LIST

2 INPUT "WHAT IS YOUR NAME AND AGE",A$,A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; B; " YEARS OLD."
READY.
RUN
WHAT IS YOUR NAME AND AGE?
```

We can see our prompt of **WHAT IS YOUR NAME AND AGE** there, but now the cursor is appearing without any ? character. This is because we put a comma (,) after the message in the **INPUT** command. To get the question mark, we have to instead put a semi-colon (;) after the message, like this:

```
INPUT "WHAT IS YOU NAME AND AGE"; A$, A
```

Now if we run the programme, we should see what we are looking for:

```
LIST
2 INPUT "WHAT IS YOUR NAME AND AGE";A$,A
3 PRINT "HELLO "; A$
4 PRINT "YOU ARE"; A; " YEARS OLD."
READY.
RUN
WHAT IS YOUR NAME AND AGE? LISTER,3000000
HELLO LISTER
YOU ARE 3000000 YEARS OLD.

READY.
```

### Exercises to try

#### 1. Can you make the programme ask someone for their name, and then for their favourite colour?

At the moment it asks for their name and age. Can you change the programme so that it reports on their favourite colour instead of their age?

*Clue:* What type of information is age? Is it numeric or a string? Is it the same type of information as the name of a colour?

#### 2. Can you write a programme that asks someone for their name, prints the hello message, and then asks for their age and prints out that response?

At the moment, the programme expects both pieces of information at the same time. This means the programme can't print a message about the first message until after it has both pieces of information. Change the programme so that you can have an interaction like the following instead:

```
WHAT IS YOUR NAME? DEEP THOUGHT
HELLO DEEP THOUGHT
WHAT IS THE ANSWER? 42
YOU SAID THE ANSWER IS 42
```

*Clue:* You will need more lines in your programme, so that you can have more than one **INPUT** and **PRINT** command.

### **3. Can you write a programme that asks several questions, and then prints out the list of answers given?**

Think of several questions you would like to be able to ask someone, and then write a programme that asks them, and remembers the answers and prints them out with an appropriate message. For example, running your programme could look like this:

```
WHAT IS YOUR NAME? FRODO
HOW OLD ARE YOU? 33
WHAT IS YOUR FAVOURITE FOOD? EVERYTHING!
THANK YOU FOR ANSWERING.
YOUR NAME IS FRODO
YOU ARE 33 YEARS OLD
YOU FAVOURITE FOOD IS EVERYTHING!
```

*Clue:* You will need more lines in your programme, to have the various **INPUT** and **PRINT** commands.

*Clue:* You will need to think carefully about which variable names you will use.

## **MAKING SIMPLE DECISIONS**

In the previous section we have learnt how to input text and numeric data, and how to display it. However, the programmes have just followed the lines of instruction in order, without any way to decide what to do, based on what has been input.

In this section we will see how we can take simple decisions using the **IF** and **THEN** commands. The **IF** command checks if something is true or false, and if it is true, causes the computer to execute the command that comes after the **THEN** command.

The way the computer decides whether something is true or false is that it operates on the supplied information using one of several symbols. These symbols are thus called *operators*. Also, because they compare two things, they depend on the relationship of the things. For this reason they are called *relational operators*. They include the following:

- Equals (=). For example, **3 = 3** would be true, while **3 = 2** would be false.
- Less than (<). For example, **1 < 3** would be true, while both **3 < 3** and **1 < 3** would be false.
- Greater than (>). For example, **3 > 1** would be true, while both **3 > 3** and **1 > 3** would be false.

As it is common to want to consider when something might be equal or greater than, or equal or less than, there are short cuts for this. Similarly, if you wish to test if something is not equal to something else, there is a relational operator for this, too:

- Unequal, which we normally say as *not equal* ( $\neq$ ). This is different to the mathematical symbol for not equal,  $\neq$ , because the MEGA65's character set does not include a character that looks like that. So the programmers who created BASIC for the MEGA65 used the greater than and less than signs together to mean either less than or greater than, that is, not equal to. For example,  $1 \neq 3$  would be true, while  $3 \neq 3$  would be false.
- Less than or equal to ( $\leq$ ). For example,  $1 \leq 3$  and  $3 \leq 3$  would be true, while both  $4 \leq 3$  would be false.
- Greater than or equal to ( $\geq$ ). For example,  $3 \geq 1$  and  $3 \geq 3$  would be true, while both  $1 \geq 3$  would be false.

A good trick if you have trouble remembering which way the ( $<$ ) and ( $>$ ) signs go, the side with more ends of lines is the one that needs to have more. For example, the ( $<$ ) symbol has one point on the left, but two ends of lines on the right hand side. So for something to be true with ( $<$ ), the number on the left side needs to be less than the number on the right side. This trick even works for the equals sign, ( $=$ ), because it has the same number of ends on both sides, so you can remember that the numbers on both sides need to be equal. It also works when you have two symbols together, like ( $\geq$ ), it is true if the condition is true for any of the symbols in it. So in this case the ( $>$ ) symbol has more ends on the left than the right, so if the number on the left is bigger than the number on the right, it will be true. But also because the ( $=$ ) symbol has two ends on each side, it will be true if the two numbers are the same.

Using these relational operators, we can write a line that will do something, but only if something is true or false. Let's try this out, with a few examples:

```
IF -2 < 0 THEN PRINT "-2 IS A NEGATIVE NUMBER"
IF 2 < 0 THEN PRINT "2 IS A NEGATIVE NUMBER"
IF 0 < -2 THEN PRINT "-2 IS A POSITIVE NUMBER"
IF 0 < 2 THEN PRINT "2 IS A POSITIVE NUMBER"
```

These commands work fine in direct mode, so you can just type them directly into the computer to see what they will do. This can be handy for testing whether you have the logic correct when planning an **IF - THEN** command. If you type in those commands, you should see something like the following:

```
COPYRIGHT 1977 MICROSOFT
BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED

ENGLISH KEYBOARD
NO EXPANSION RAM
C1565 DRIVE
ROM CHECKSUM $4BCF

READY.
IF -2 < 0 THEN PRINT "-2 IS A NEGATIVE NUMBER"
-2 IS A NEGATIVE NUMBER

READY.
IF 2 < 0 THEN PRINT "2 IS A NEGATIVE NUMBER"

READY.
IF 0 < -2 THEN PRINT "-2 IS A POSITIVE NUMBER"

READY.
IF 0 < 2 THEN PRINT "2 IS A POSITIVE NUMBER"
2 IS A POSITIVE NUMBER

READY.
```

We can see that only the **PRINT** commands that followed an **IF** command that has a true value were executed. The rest were silently ignored by the computer. But we can of course include these into a programme. So lets make a little programme that will ask for two numbers, and say whether they are equal, or if one is greater or less than the other. Before you have a look at the programme, have a think about how you might do it, and see if you can figure it out. The clue I will give you, is that the **IF** command also accepts the name of a variables, not just numbers. So you can do something like **IF A > B THEN PRINT "SOMETHING"**. The programme will be on the next page, to stop you peeking before you have a think about it!

Did you have a go? There are lots of different ways it could be done, but here is what I came up with:

```
1 INPUT "WHAT IS THE FIRST NUMBER"; A  
2 INPUT "WHAT IS THE SECOND NUMBER"; B  
3 IF A = B THEN PRINT "THE NUMBERS ARE EQUAL"  
4 IF A > B THEN PRINT "THE FIRST NUMBER IS BIGGER"  
5 IF B > A THEN PRINT "THE SECOND NUMBER IS BIGGER"
```

We can then run the programme as often as we like, and the computer can tell us which of the two numbers we give it is biggest, or if they are equal:

```
1 INPUT "WHAT IS THE FIRST NUMBER"; A  
2 INPUT "WHAT IS THE SECOND NUMBER"; B  
3 IF A = B THEN PRINT "THE NUMBERS ARE EQUAL"  
4 IF A > B THEN PRINT "THE FIRST NUMBER IS BIGGER"  
5 IF B > A THEN PRINT "THE SECOND NUMBER IS BIGGER"  
  
RUN  
WHAT IS THE FIRST NUMBER? 2  
WHAT IS THE SECOND NUMBER? 2  
THE NUMBERS ARE EQUAL  
  
READY.  
RUN  
WHAT IS THE FIRST NUMBER? 3  
WHAT IS THE SECOND NUMBER? 4  
THE SECOND NUMBER IS BIGGER  
  
READY.  
RUN  
WHAT IS THE FIRST NUMBER? 10  
WHAT IS THE SECOND NUMBER? 2  
THE FIRST NUMBER IS BIGGER  
  
READY.
```

Notice how in this programme, we didn't use fixed numbers in the **IF** command, but instead gave variable names instead. This is one of the very powerful things in computer programming, together with being able to make decision based on data. By being able to refer to data by name, regardless of its current value or how it got there, lets the programmer create very flexible programmes.

Let's think about a bit of a more interesting example: a "guess the number" game. For this, we need to have a number that someone has to guess, and then we need to accept guesses, and indicate whether the guess was correct or not. If the guess is incorrect, we should tell the user if the correct number is higher or lower.

We have already learned most the ingredients to make such a program: We can use **LET** to set a variable to the secret number, **INPUT** to prompt the user for their guess, and then **IF**, **THEN** and **PRINT** to tell the user whether their guess was correct or not. So let's make

something. Again, if you like, stop and think and experiment for a few minutes to see if you can make such a programme yourself.

Here is how I have done it. But don't worry if you have done it in a quite different way: There are often many ways to write a programme to perform a particular task.

```
1 SN=23
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
3 INPUT "WHAT IS YOUR GUESS"; G
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUESSED MY NUMBER!"
```

The first line puts our secret number into the variable **SN**. The second line prints a message telling the user what they are supposed to do. The third line asks the user for their guess, and puts it into the variable **G**. The fourth, fifth and sixth lines then check whether the guess is correct or not, and if not, which message it should print. This is done by using the **IF** command and an appropriate relative operator to make each decision. This works well, to a point. For example:

```
LIST
1 SN=23
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
3 INPUT "WHAT IS YOUR GUESS"; G
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUESSED MY NUMBER!"
```

```
READY.
RUN
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 10
MY NUMBER IS BIGGER

READY.
RUN
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 23
CONGRATULATIONS! YOU GUESSED MY NUMBER!
```

```
READY.
```

We can see that it prints the message, and it asks for a guess, and responds appropriately. But if we want to guess again, we have to use the **RUN** command again for each extra guess. That's a bit poor from the user's perspective. However that is unlikely to be a problem for long, because the user can see the secret number in the listing on the screen!

So we would like to fix these problems. Let's start with hiding the listing. We previously mentioned that when the screen scrolls, anything that was at the top of the screen dis-

appears. So we could just make sure the screen scrolls enough, that any listing that was visible is no longer visible. We could do this using **PRINT** and a **FOR** loop. The screen is 25 lines, so we could do something like:

```
FOR I = 1 to 25  
PRINT  
NEXT I
```

But there are better ways. If you hold down the **SHIFT** key, and then press the **CLR HOME** key, it clears the screen. This is much simpler and more convenient. But how can we do something like that in our programme? It turns out to be very simple: You can type it while entering a string! This is because the keyboard works differently based on whether you are in *quote mode*.

Quote mode is just a fancy way of describing what happens when you type a double-quote character into the computer: Until you type another double-quote or press the **RETURN**. You might remember we mentioned the problem of funny symbols coming up when using the cursor keys. I didn't want to distract you at the time, but that is a symptom of being in quote mode: In quote mode many special keys show a symbol that represents them, rather than taking their normal action. For example, if you press the cursor left key while in quote mode, a **↔** symbol appears. If you press the cursor right key, a **↙**, up **↑**, down **↓** and the **CLR HOME** a **⇞**, and if you are holding down **SHIFT** and press **CLR HOME** a **⇟**.

So let's use this to make the second line clear the screen when it prints the **GUESS THE NUMBER BETWEEN 1 AND 100** message. The first time you try it is a bit confusing, but once you get the hang of it, it is quite easy. What we want in the end is a line that looks like this:

```
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
```

To do this, start by typing **2 PRINT "**. Then hold the **SHIFT** key down, and tap the **CLR HOME** key. Your line should now look like **2 PRINT"█**. If so, you have succeeded! You can now finish typing the line as normal. When you have done that, you can use the **LIST** command as usual, to make sure that you have successfully modified the programme. You should see your modified line with the █ symbol in it.

```
LIST
1 SN=23
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
3 INPUT "WHAT IS YOUR GUESS";G
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUESSED MY NUMBER!"

READY.
2 PRINT"█GUESS THE NUMBER BETWEEN 1 AND 100"
LIST

1 SN=23
2 PRINT"█GUESS THE NUMBER BETWEEN 1 AND 100"
3 INPUT "WHAT IS YOUR GUESS";G
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUESSED MY NUMBER!"

READY.
```

If you now run the programme by typing in **RUN** and pressing the **RETURN** key as usual, the 2<sup>nd</sup> line tells the compute to clear the screen before printing the rest of the message, like this:



```
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? ■
```

This hides the listing from the user, so that they can't immediately see what our secret number is. We can type our guess in, just like before, but just like before, after one guess, it returns to the **READY.** prompt. We really would like people to be able to make more than one guess, without needing to know that they need to run the programme again.

There are a few ways we could do this. We already saw the **FOR - NEXT** pattern. With that, we could make the programme give the user a certain number of guesses. If we followed the **NEXT** command with another programme line, we could even tell the user when they have taken too many guesses. So lets have a look at our programme and see how we might do that. Here is our current listing again:

```
1 SN=23
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
3 INPUT"WHAT IS YOUR GUESS"; G
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUESSED MY NUMBER!"
```

If we want the user to have multiple guesses, we need to have lines 2 through 6 run multiple times. This makes our life a bit tricky, because it means we need to insert a line between line 1 and 2. But unless you are a mathemagician, there are no whole numbers between 1 and 2, and the MEGA65 doesn't understand line numbers like 1.5.

Fortunately, the MEGA65 has the **RENUMBER** command. This command can be typed only in direct mode. When executed, it changes the line numbers in the programme, while keeping them in the same order. The new numbers are normally multiples of 10, so that you have lots of spare numbers in between to add extra lines. For example, if we use it on our programme, it will renumber the lines to 10, 20, ..., 60. We can see that this has happened by using the **LIST** command:

```
READY.  
LIST  
  
1 SN=23  
2 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"  
3 INPUT"What is your guess";G  
4 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"  
5 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"  
6 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"  
  
READY.  
RENUMBER  
  
READY.  
LIST  
  
10 SN=23  
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"  
30 INPUT"What is your guess";G  
40 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"  
50 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"  
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"  
  
READY.
```

Now our life is much easier: We can choose any number that is between 10 and 20 to put our **FOR** command into. It's a common choice to use the middle number, so that if you think of other things you want to add in later, you have the space to do it. So let's add a **FOR** command to give the user 10 chances to guess the number. We can use any variable name we like for this, except for **G** and **SN**, because we are using those. It would be very confusing if we mixed those up! So lets add a line like this:

```
15 FOR I = 1 TO 10 STEP 1
```

Now we need a matching **NEXT I** after line 60. Let's keep the nice pattern of adding 10 to work out the next line number, and put it as line 70:

```
70 NEXT I
```

We can type those lines in, and then use **LIST** command to make sure the result is correct:

```
LIST
10 SN=23
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
30 INPUT"WHAT IS YOUR GUESS";G
40 IF G>SN THEN PRINT "MY NUMBER IS BIGGER"
50 IF G<SN THEN PRINT "MY NUMBER IS SMALLER"
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"

READY.
15 FOR I = 1 TO 10 STEP 1
70 NEXT I
LIST

10 SN=23
15 FOR I = 1 TO 10 STEP 1
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
30 INPUT"WHAT IS YOUR GUESS";G
40 IF G>SN THEN PRINT "MY NUMBER IS BIGGER"
50 IF G<SN THEN PRINT "MY NUMBER IS SMALLER"
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"
70 NEXT I

READY.
```

That's looking pretty good. But there are a couple of little problems still. Can you work out what they might be? What will happen now after the user makes a guess? What will happen if they run out of guesses?

If you worked out that making a guess that the screen will be immediately cleared, you can give yourself a pat on the back! The user will hardly have time to see the message. Worse, if they guess the number correctly, they won't know, and the programme will keep going. We'd really like the programme to stop or end, once the user makes a correct guess.

We can do this using either the **STOP** or **END** commands. These two commands are quite similar. The main difference is that if you **STOP** a programme, the computer tells you where it has stopped, and you have the chance to continue the programme using the **CONT** command. The **END** command, on the other hand, tells the computer that the programme has reached its end, and it should go back to being **READY**. The **END** command makes more sense for our programme, because after the user has guessed the number, there isn't any reason to continue.

Now we need a way to be able tell the computer to do two different things when the user makes a correct guess. We could just add an extra **IF** command after line 60 which prints the congratulations message, e.g., **65 IF G=SN THEN END.**

But we can be a bit more elegant than that: There is a way to have multiple commands on a single line. If you remember back to when we were learning about the **INPUT** command, you might remember that there were two different characters that separate pieces of information: , and :. The second one, :, is called a colon, and can also be used to

separate BASIC commands on a single line. So if we want to change line 60 to **PRINT** the message of congratulations and then **END** the programme, we can just add : **END** to the end of the line. The line should look like this:

```
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!": END
```

That solves that problem. But it would also be nice to not clear the screen after every guess, so that the user can see what their last guess was, and whether it was bigger or smaller than the number. To do this, we can remove the clear-screen code from line 20, and add a new print command to a lower line number, so that it clears the screen once at the start of the programme, before the user gets to start guessing.

For example, we could put in line 5, so that it happens as the absolute first action of the programme. As we mentioned earlier, the line numbers themselves aren't important: All that is important is to remember that the computer starts at the lowest line number, and runs the lines in order. Anyway, let's make those changes to our programme:

```
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"  
5 PRINT "W"
```

If you type those lines in, and **LIST** the programme again, you should see something like the following:

```
40 IF G>SN THEN PRINT "MY NUMBER IS BIGGER"  
50 IF G<SN THEN PRINT "MY NUMBER IS SMALLER"  
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"  
70 NEXT I
```

READY.

```
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"  
5 PRINT "W"  
LIST  
  
5 PRINT "W"  
10 SN=23  
15 FOR I = 1 TO 10 STEP 1  
20 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"  
30 INPUT "WHAT IS YOUR GUESS";G  
40 IF G>SN THEN PRINT "MY NUMBER IS BIGGER"  
50 IF G<SN THEN PRINT "MY NUMBER IS SMALLER"  
60 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!"  
70 NEXT I
```

READY.

We can now **RUN** the programme, and see whether it worked. Let's try it!



```
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? █
```

The screen still clears, which is good. Can you notice one little difference already, though? There is a blank line above the first message. This is because our **PRINT** command in line 5 goes to the next row on the screen after it has printed the clear-screen character. We can fix this by putting a ; (semi-colon) character at the end of the **PRINT** command. This tells the **PRINT** command that it shouldn't go to the start of the next row on the screen when it has done everything. So if we change line 5 to **5 PRINT "█";** this will make the empty space at the top of the screen disappear.

But back to our programme, we can now make guesses, and the programme will tell us whether each guess is more or less than the correct number. And after 10 guesses, it stops asking for guesses, and goes back to the **READY.** prompt, like this:

```
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 60
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 50
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 40
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 30
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 20
MY NUMBER IS BIGGER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 28
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 27
MY NUMBER IS SMALLER

READY.
```

It would be nice to tell the user if they have run out of guesses. We need to add this message after the **NEXT** command. We should also be nice and tell them what the secret number was, instead of leaving them wondering. So let's add the line to the end of our programme as line 80:

```
80 PRINT "SORRY! YOU RAN OUT OF GUESSES, MY NUMBER WAS"; SN
```

Now if the user doesn't guess the number, they will get a useful message, like this:

```
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 97
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 96
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 95
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 94
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 93
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 92
MY NUMBER IS SMALLER
GUESS THE NUMBER BETWEEN 1 AND 100
WHAT IS YOUR GUESS? 91
MY NUMBER IS SMALLER
SORRY! YOU RAN OUT OF GUESSES. MY NUMBER WAS 23
READY.
```

## Exercises to try

### 1. Can you make the programme ask at the start for the secret number?

At the moment the programme sets the secret number to 23 every time. To make the game more interesting it would be great to ask the first user for the secret number, and then start the rest of the game, so that someone else can try to guess the number.

*Clue:* You will need change the line that sets the **SN** variable so that it can be read from the first user. You might find the **INPUT** statement useful.

### 2. Can you make the programme ask for the user's name and give personalised responses?

At the moment, the programme displays very simple messages. It would be nice to ask the user their name, and then use their name to produce personalised messages, like **SORRY DAVE, BUT THAT NUMBER IS TOO SMALL.**

*Clue:* You will need to add a line early in the programme to ask the user their name.

*Clue:* You might like to review how we used the **PRINT** command, including with ; to print more than one thing on a line.

### 3. Can you improve the appearance of the messages with colours and better spacing?

We haven't really made the programme particularly pretty. It would be great to use colours.

*Clue: You might like to add more **PRINT** commands to improve the spacing and layout of the messages.*

*Clue: You might like to use either the colour codes in the messages you **PRINT***

*Clue: You might also like to use the **FOREGROUND**, **BACKGROUND** and **BORDER** commands to set the colour of the text, screen background and border.*

#### **4. Can you make the programme say if a guess is “warmer” or “colder” than the previous guess?**

At the moment the programme just tells you if the guess is higher or lower than the secret number. It would be great if it could tell you if a guess is getting closer or further away with each guess: When they get closer, it should tell the user that they are getting “warmer”, and “colder” when they get further away.

This is quite a bit more involved than the previous exercises, and requires you to work out some new things.

*Clue: You will need to remember the previous guess in a different variable, and then compare it with the last one: Is it nearer or further away. You might need to have **IF** commands that have another **IF** after the first one, or to learn how to use the **AND** operator.*

## RANDOM NUMBERS AND CHANCE

We'll come back to the Guess The Number game shortly, but let's take a detour first. Through a maze. Let's hope we can get back out before the end of the lesson! Let's look at a simple way to make a maze. This programme has been known for a long time. It works by choosing at random whether to display a  or a  symbol. These symbols are obtained by holding down the  key and tapping either the N or M keys. You can see the symbols on the front of those keys. While they are shown on the keys with a box around them, the box does not appear, only the diagonal line. It turns out that printing either of these two characters at random draws a decent looking maze.

Let's give it a try. To be able to do this, we need a way to generate randomness. The MEGA65 has the **RND(1)** function to do this. This function works like a variable, but each time you try to use it, it gives a different result. Let's see how that works. Type in the following:

```
PRINT RND(1)
```

Each time you type this, it will give a different answer, as you can see here:

```
BASIC 10.0 V0.9B.911001 ALL RIGHTS RESERVED

ENGLISH KEYBOARD
NO EXPANSION RAM
C1565 DRIVE
ROM CHECKSUM $4BCF

READY.
PRINT RND(1)
1.07870447E-03

READY.
PRINT RND(1)
.793262171

READY.
PRINT RND(1)
.44889513

READY.
PRINT RND(1)
.697215893

READY.
```

We can see that this gives us several different results: **1.07870447E-03**, **.793262171**, **.44889513**, **.697215893**. Each of these is a number between 0 and 1, even the first one. The first one is written in *scientific notation*. The **E-03** means that the value is  $1.07870447 \times 10^{-3} = 0.000107870447$ . That is, the **E-03** means to move the decimal place three places to the left. If there is a **+** after **E**, then it means to move the decimal place to the right. For example, **1.23456E+3** represents the number 1234.56.

Now, I promised a maze, so I better give you one. We can use this **RND(1)** to pick between these two symbols. The first one has a character code of 205, and the second one conveniently 206. This means that if we add the result of **RND(1)** to 205.5, we will get a number between 205.5 and 206.5. Half the time it will be 205.something, and the other half of the time it will be 206.something. We can use this to print one or the other characters by using the **CHR\$(C)** function that returns the character corresponding to the number we put between the brackets. This means we can do something like:

```
LET C = 205.5+RND(1)
PRINT CHR$(C);
```

This will print one or the other of these symbols each time. We could use this already to print the maze by doing this over and over, making a loop. We could use **FOR** and **NEXT**. But in this case, we want it to go forever, that is, each time the programme gets to the end, we want it to go to the start again. The people who created BASIC really weren't very creative, so the command to do this is called **GOTO**. You put the number of the line that you want to be executed next after it, e.g., **GOTO 1**. We can use this to write our little maze programme so that it will run continuously:

```
10 LET C = 205.5+RND(1)
20 PRINT CHR$(C);
30 GOTO 10
```

If you **RUN** this programme, it will start drawing a maze forever, that looks like the screen shot below. You can stop it at any time by pressing the **RUN STOP** key, or you can pause it by pressing the **NO SCROLL** key, and unpause it by pressing the **NO SCROLL** key again. If you press the **RUN STOP** key, the computer will tell you where it was up to at the time. In the case of the screenshot below, it was working on line 10:

```
10 LET C=205.5+RND(1)
20 PRINT CHR$(C);
30 GOTO 10
```

READY.

RUN



PBREAK IN 10  
READY.

That works nicely, and draws a very famous maze [1]. We can, however, make the programme smaller. We don't need to put the result of the calculation of which symbol to display on a separate line. We can put the calculation directly into brackets for the **CHR\$(C)** function:

```
10 PRINT CHR$(205.5+RND(1));
20 GOTO 10
```

And we can use what we learnt about the : (colon) symbol, and put the **GOTO** command onto the same line as the **PRINT** command:

```
10 PRINT CHR$(205.5+RND(1));: GOTO 10
```

Can you see how there are often many ways to get the same effect from a programme? This is quite normal. For complex programmes, there are many, many ways to get the same function. This is one of the areas in computer programming where you can be very creative.

But back to the topic of randomness. It's all well and good using these random numbers between 0 and 1 for drawing a maze, but it's a bit tricky to ask people to get a really long decimal. If we want a number in the range 1 to 100, we can multiply what we get from **RND(1)** by 100. If we do that, it gets a bit better, but we will still get numbers like **55.0304651**, **30.3140154**, **60.2505497** and **.759229916**.

That's closer, but we really want to get rid of those fractional parts. That is, we want whole numbers or *integers*. BASIC has the **INT()** function that works like the **RND(1)** function, except that whatever number you put in the brackets, it will return just the whole part of that. So for example **INT(2.18787)** will return the value **2**. As I said just now, it chops off the fractional part, that is, it always rounds down. So even if we do **INT(2.9999999)** the result will still be **2**, not **3**. This means that if we multiply the result of **RND(1)** by 100, we will get a number in the range of 0 – 99, not 1 – 100. This is nice and easy to fix: We can just add 1 to the result. So to generate an integer, that is a whole number, that is between 1 and 100 inclusive, we can do something like:

```
PRINT INT(RND(1)*100) + 1
```

That looks much better. So let's type in our "guess the number" programme again. But this time, let's replace the place where we set our secret number to the number 23, to instead set it to a random integer between 1 and 100. Don't peek at the solution just yet. Have a think about how we can use the above to set **SN** to a random integer between 1 and 100. Once you have your guess ready, have a look what I came up with below. You might have made a different programme that can do the same job. That's quite fine, too!

```
10 SN=INT(RND(1)*100)+1
20 PRINT " "
30 FOR I = 1 TO 10 STEP 1
40 PRINT "GUESS THE NUMBER BETWEEN 1 AND 100"
50 INPUT "WHAT IS YOUR GUESS"; G
60 IF G<SN THEN PRINT "MY NUMBER IS BIGGER"
70 IF G>SN THEN PRINT "MY NUMBER IS SMALLER"
80 IF G=SN THEN PRINT "CONGRATULATIONS! YOU GUessed MY NUMBER!": END
90 NEXT I
100 PRINT "SORRY, YOU HAVE RUN OUT OF GUESSES"
```

Now we don't have to worry about someone guessing the number, and we don't need someone else to pick the number for us. This makes the programme much more fun to play. Can you beat it?

### Exercises to try

#### 1. Can you make the maze programme make different mazes?

The maze programme currently displays equal numbers of  $\square$  and  $\blacksquare$ . Can you change the programme to print twice as many of one than the other? How does the maze look then?

Clue: We used **205.5** so that when we add a random number between 0 and 1, we end up with **205.something** half the time and **206.something** the other half of the time. If you reduce **205.5** towards **205**, or increase it towards **206** you will change the relative proportion of each character that appears.

#### 2. Can you modify the "guess my number" programme to choose a number between 1 and 10?

At the moment, the programme picks a number between 1 and 100. Modify the programme so that it picks a number from a different range. Don't forget to update the message printed to the user. Do they still need 10 guesses? Change the maximum number of guesses they get before losing to a more suitable amount.

Clue: You will need to modify the line that sets **SN**, as well as the **PRINT** message that gives instruction to the user.

#### 3. Set the screen, border and text colour to random colours

Modify either the maze or “guess my number” programme to use random colours. How might you make sure that the text is always visible?

*Clue: Use the **BACKGROUND**, **BACKGROUND** and **BORDER** commands to set the colours. Use colour numbers between 0 and 15, inclusive. You can put a calculation at the end of these commands in place of a simple number.*

*Clue: To make sure you don't set the text colour to the same as the background, you might like to calculate which background colour you wish to use and keep it in one variable, and then calculate the text colour to use and store it in a different variable. If the two variables have the same number, then you need to change one of them.*

#### **4. Make the “guess my number” programme randomly choose between two different greeting messages when it starts.**

The “guess my number” programme currently always prints the same message every time it starts. Modify it so that it prints one of two possible messages each time.

*Clue: Use **RND(1)** to obtain a random number. If that number is less than some threshold, print the first message, else print the second message.*

*Clue: It might be easier, if you store the random number in a variable, so that you can use two **IF** statements to decide whether to print each message.*

*Clue: If you use < (less than) as the relational operator in one of the **IF** statements, you will need to use the opposite in the other one. The opposite of less than is greater than or equal to.*



# 8

## CHAPTER

### Text Processing

- **Characters and Strings**
- **String Literals**
- **String Variables**
- **String Statements**
- **Simple Formatting**
- **Sample Programs**



# CHARACTERS AND STRINGS

Representing textual information in the form of printable letters, numbers and symbols is a common requirement of many computer programs. The need for text arises in word processing applications and word games. It is also required in natural language processing and text-based adventure games, both of which need to understand the input. Understanding text input is called *parsing*. In short, text processing is used everywhere. In order to input, output and manipulate such information, we must introduce two key concepts: characters and strings.

Characters can be printable or non-printable. A character most often represents a single, primitive element of printable text which may be displayed on the screen via the statement **PRINT**. It is most common and most natural to think of a character as representing a letter of an alphabet. A character might, for example, be any of the uppercase letters 'A' to 'Z', or any of the lowercase letters 'a' to 'z'. However, characters can also represent commonly used symbols such as punctuation marks or currency symbols. Indeed, characters can also represent the decimal digits, '0' to '9'. It is worth noting that this refers to the text-based representation of the numerals 0 to 9 as printable symbols as opposed to their numeric counterparts. In addition, the MEGA65 provides an extensive range of special symbols that can be used together for games, for drawing fancy borders or art. Besides displaying information, such symbols can create simple yet intriguing visual patterns. For convenience, these special symbols appear on the front sides of the MEGA65's keys.

A character can also be non-printable. Using such characters (in a **PRINT** statement) can activate certain behaviors or cause certain modes to become active, such as the switching of all text on the screen to lowercase or setting the foreground color to orange. Other non-printable characters might represent a carriage return or clear the screen.

For a complete catalog of available characters, refer to Chapter/Appendix C on page C-3. The table lists the characters that correspond to a given code number. The code number must be supplied as an argument to the statement **CHR\$** which, when combined with the **PRINT** statement, outputs the respective characters to the screen.

Here's an example of printing the exclamation mark using a character code:

```
PRINT CHR$(33)  
!
```

Note that the '!' is actually visible on the display because it is a printable character.

Here's an example of changing the foreground color to white using character codes:

```
PRINT CHR$(5)
```

Although no character is output, all subsequent printable characters displayed will be colored white.

Sometimes it can be useful to do the conversion in reverse: from a character to its code number. To do this, a single character must be supplied as an argument to the statement **ASC** within quotation marks which, when combined with the **PRINT** statement, outputs the respective code number to the screen in decimal.

Here's an example of obtaining the code number for the exclamation mark.

```
PRINT ASC("!")
33
```

And here's an example of obtaining the code number for the exclamation mark and storing it in an integer variable:

```
A% = ASC("!")
```

Although we could output individual characters repeatedly by using **CHR\$** it would be tedious to do this all the time.

The concept of a string is needed because it embodies the idea of a contiguous block of text. Thus, a string can contain multiple printable and/or multiple non-printable characters in any combination. A string can potentially be empty and contain no characters at all. To write a string we enclose the characters inside quotation marks. So "HELLO WORLD!" is an example of a string literal.

```
PRINT "HELLO WORLD!"
HELLO WORLD!
```

All strings have a property called length which is how many printable and non-printable characters there are present in that string. The length can be as low as 0 (the empty string) or as high as 255. Attempting to create a string with a length in excess of 255 characters results in a **?STRING TOO LONG ERROR**.

```
PRINT LEN("HELLO WORLD!")
12
```

```
PRINT LEN("")
0
```

It is possible to create variables specifically for strings. All such string variables have names that begin with a leading alphabetic character, have an optional second character

that is alphanumeric, and end with a \$ sign. Once given a value, they can be used with **PRINT**.

```
AB$ = "HELLO WORLD!": PRINT AB$  
HELLO WORLD!
```

```
A1$ = "HELLO WORLD!": PRINT LEN(A1$)  
12
```

## STRING LITERALS

String literals can be joined with one or more other such string literals to form a compound string. This process is called *concatenation*. To concatenate two or more string literals, use the + operator to chain them together.

Here are some examples:

```
PRINT ("SECOND" + "HAND")  
SECONDHAND
```

```
PRINT ("COUNTER" + "CLOCK" + "WISE")  
COUNTERCLOCKWISE
```

Sometimes punctuation or spaces may be required to make the final output appear correctly formatted, as in the following example.

```
PRINT ("FRUIT: " + "APPLE, " + "PEAR AND " + "RASPBERRY.")  
FRUIT: APPLE, PEAR AND RASPBERRY.
```

## STRING VARIABLES

Concatenation is more commonly used with string variables combined with string literals. For example, in a text-based adventure game you might want to list some exits such as north or south. Because these exits will vary depending on the location you are currently at it would make sense to use variables for the exits themselves and use concatenation with literals such as commas, spaces and full stops to format the output appropriately.

```
A$ = "PEA": B$ = "NUT": PRINT (A$ + B$ + "BUTTER")
PEANUTBUTTER
```

It is also possible to use strings as the parameters of **DATA** statements, to be read later, using the **READ** statement. The following example also demonstrates that arrays can hold strings too.

```
10 DIM A$(6)
20 PRINT "RAINBOW COLORS: ";
30 FOR I = 0 TO 5
40 : READ A$(I): PRINT (A$(I) + ", ")
50 NEXT I
60 READ A$(I): PRINT ("AND " + A$(I) + ".")
70 DATA "RED", "ORANGE", "YELLOW", "GREEN", "BLUE", "INDIGO", "VIOLET"
```

It is common for string data or single-character data to come directly from user input. When the user types some text, that text will often need to be parsed or printed back to the screen. In general, there are three main ways that this can be done: via the **GET** statement, via the **GETKEY** statement or via the **INPUT** statement.

All three statements have different behaviours, and it's important to understand how each one operates by contrasting and comparing them.

The **GET** statement is useful for storing the current keypress in a variable. The program does not wait for a keypress: it continues executing the next statement immediately. For this reason it is sometimes important to place the **GET** statement inside some kind of loop—the loop is to be exited only when a valid keypress is detected. If the variable to **GET** is a string variable and no keypress is detected, then that string variable is set to equal an empty string.

```
10 GET A$: REM DO NOT WAIT FOR A KEYPRESS--READ ANY KEYPRESS INTO THE VARIABLE
20 PRINT A$: IF (A$ = "Y" OR A$ = "N") THEN END
30 GOTO 10
```

The **GETKEY** statement is also useful for storing the current keypress in a variable. In contrast to the **GET** statement, the **GETKEY** statement, when executed, does wait for a single keypress before it continues executing the next statement.

```
10 GETKEY A$: REM WAIT FOR A KEYPRESS--PAUSE AND READ IT INTO THE VARIABLE
20 PRINT A$: IF (A$ = "Y" OR A$ = "N") THEN END
30 GOTO 10
```

While **GET** and **GETKEY** are fine for reading single characters, the **INPUT** statement is useful for reading in entire strings—that is, zero or more characters at a time.

When the **INPUT** statement is used with a comma and a variable, the prompt string is displayed normally with a cursor that permits the user to type in some text. When the **INPUT** statement is used with a semicolon and a variable, the prompt string is displayed with a question mark appended and a cursor that permits the user to type in some text.

```
10 INPUT "ENTER YOUR NAME", A$: REM NOT A QUESTION  
20 PRINT ("HELLO " + A$)
```

```
10 INPUT "WHAT IS YOUR NAME"; A$: REM A QUESTION  
20 PRINT ("HELLO " + A$)
```

In either case, pressing **RETURN** will complete the text entry—the text entered will be stored in the given variable. Note that if the string variable is already equal to some string and **RETURN** is pressed without entering in new data, then the old string value currently stored in the variable is retained.

## STRING STATEMENTS

There are three commonly-used string manipulation commands: **MID\$**, **LEFT\$** and **RIGHT\$**. These are good for isolating substrings, including individual characters.

The following program asks for an input string and then prints all left substrings.

```
10 INPUT "ENTER A WORD:", A$  
20 PRINT "ALL LEFT SUBSTRINGS ARE:"  
30 FOR I = 0 TO LEN(A$)  
40 : PRINT LEFT$(A$, I)  
50 NEXT I
```

The following program asks for an input string and then prints all right substrings.

```
10 INPUT "ENTER A WORD:", A$  
20 PRINT "ALL RIGHT SUBSTRINGS ARE:"  
30 FOR I = 0 TO LEN(A$)  
40 : PRINT RIGHT$(A$, I)  
50 NEXT I
```

The following program ask for an input string consisting of a first name following by a space followed by a last name. It then outputs the initial letters of both names.

```
10 INPUT "ENTER A FIRST NAME, A SPACE AND A LAST NAME:", A$  
20 N = -1  
30 FOR I = 1 TO LEN(A$)  
40 : IF (MID$(A$, I, 1) = " ") THEN N = I: GOTO 60  
50 NEXT I  
60 IF (N = -1) THEN GOTO 10  
70 PRINT "INITIALS ARE: "; MID$(A$, 1, 1)+"."+MID$(A$, N + 1, 1)+".",
```

## SIMPLE FORMATTING

### Suppressing New Lines

When using the **PRINT** statement in a program, the default behaviour is to output the string and then move to the next line. To stop the behaviour of automatically moving to the next line, simply append a ; (semicolon) after the end of the string. Contrast lines 10, 20 and 30 in the following program.

```
10 PRINT "THIS A SINGLE LINE OF TEXT": REM A NEW LINE IS ADDED AT THE END  
20 PRINT "THE SECOND LINE"; : REM A NEW LINE IS SUPPRESSED  
30 PRINT " USES A SEMICOLON" : REM A NEW LINE IS ADDED AT THE END
```

### Automatic Tab Stops

Sometimes it can be convenient to use the **PRINT** statement to output information neatly into columns. This can be done by appending a , (comma) after the end of the string. Consider the following example program.

```
10 PRINT "TEXT 1", "TEXT 2", "TEXT 3", "TEXT 4"
```

Note that each tab stop is 10 characters apart. So TEXT 1 begins at column 0, TEXT 2 begins at column 10, TEXT 3 begins at column 20, and TEXT 4 begins at column 30.

### Tabs Stops and Spacing

When printing text on the screen, it is often necessary to format text by using spaces and tabs. Two commands come in handy here: **SPC** and **TAB**.

The command **SPC(5)**, for example, moves five characters to the right. Any intervening characters that lie between the current cursor position and the position five characters to the right are left unchanged.

The command **TAB(20)**, for example, moves to column 20 by subtracting the cursor's current position away from twenty and then moving that number of characters to the right. If the cursor's initial position is to the right of column 20 then the command does nothing. This command can often be used to make text line up neatly into columns.

## SAMPLE PROGRAMS

We conclude with some examples.

### Palindromes

A *palindrome* is a word or phrase or number that reads the same forwards as it does backwards. Some examples are: CIVIC, LEVEL, RADAR, MADAM and 1234321. The following program reverses the input text and then determines whether the original phrase is equal to the reversed phrase.

```
10 REM *** PALINDROMES ***
20 INPUT "ENTER SOME TEXT: ", A$
30 B$ = ""
40 FOR I = 1 TO LEN(A$)
50 : B$ = MID$(A$, I, 1) + B$
60 NEXT I
70 IF (A$ = B$) THEN PRINT (A$ + " IS A PALINDROME"): ELSE PRINT (A$ + " IS NOT A PALINDROME")
80 GOTO 20
```

### Simple Ciphers

We now look at three simple examples of scrambling and unscrambling English language text messages. This scrambling and unscrambling process is the study of *cryptography* and is used to keep information secure so that it can't be read by others except for those privileged to know the cipher's method and secret key.

The process of scrambling a given message is called *encryption*. The ordinary, readable unscrambled text is called *plaintext*. Encrypting plaintext results in a scrambled message. This scrambled text is called *ciphertext*. The process of unscrambling the ciphertext is called *decryption*. Decrypting the ciphertext results in an unscrambled message—the plaintext.

Suppose that we were to encrypt some plaintext and then send the resulting ciphertext to a friend. Provided that the friend knows the method and secret key used to scramble the message, they could then decrypt the ciphertext and would be able to recover and read our original plaintext message.

If someone else attempts to read the ciphertext using the wrong method and/or the wrong secret key, the resulting text will be unintelligible.

The cryptographic systems we describe here are very simple. Obviously, they shouldn't be used today because they are easily broken by techniques of cryptanalysis. Nevertheless, they illustrate some basic techniques and show how we might structure a sample program.

We investigate three ciphers. These are the ROT13 cipher, the Caesar Cipher and the Atbash Cipher. These are part of a group of ciphers known as *affine ciphers*.

Mathematically, it is useful to think of the letters of the English alphabet as numbered. A is 0, B is 1 and so, with Z being equal to 25.

| Letter | A | B | C | D | E | F | G | H | I | J | K  | L  | M  | N  | O  | P  | Q  | R  | S  | T  | U  | V  | W  | X  | Y  | Z  |
|--------|---|---|---|---|---|---|---|---|---|---|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|
| Value  | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 |

A key mathematical component of a cryptographic system is *modular arithmetic*, sometimes casually referred to as "clock arithmetic" because the numbers begin at zero and increase until they reach an upper limit, at which point they wrap around back to zero again, much like a circle. In our case, since there are 26 letters in the English alphabet, we use modulo 26 arithmetic—our letters are numbered from 0 to 25.



To reduce a given number using modulo 26 we can use the following function:

$$f(x) = x - \left\lfloor \frac{x}{26} \right\rfloor \times 26$$

This says that to obtain the value of a number  $x$  using modulo 26 we first divide  $x$  by 26 and round down, which gives us the number of times we went around the circle. We then multiply this result by 26 again and subtract this from  $x$ . The final result is the remainder left over and will always be a value between 0 and 25.

As an example, the number 28 in modulo 26 is equal to 2:

$$f(28) = 28 - \left\lfloor \frac{28}{26} \right\rfloor \times 26 = 28 - 1 \times 26 = 2$$

The program at the end of this chapter makes use of this formula by defining a corresponding function at line 30:

```
DEF FN F(X)=X-INT(X/26)*26
```

**ROT13:** When we encrypt each plaintext letter we move forward 13 places. So the plaintext letter A becomes the ciphertext letter N, B becomes O, with latter letters "wrapping around" back to the beginning of the alphabet. Thus, the plaintext letter Z becomes the ciphertext letter M. This covers encryption. To decrypt each ciphertext letter we simply repeat the process by moving forward 13 places again, which brings us full circle, back to where we started. Thus, a ciphertext letter N becomes the plaintext letter A.

We can see this visually as a mapping in the form of a table:

|                   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|-------------------|---|---|---|---|---|---|---|---|---|---|---|---|---|
| English Plaintext | A | B | C | D | E | F | G | H | I | J | K | L | M |
| ROT13 Ciphertext  | N | O | P | Q | R | S | T | U | V | W | X | Y | Z |

  

|                   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|-------------------|---|---|---|---|---|---|---|---|---|---|---|---|---|
| English Plaintext | N | O | P | Q | R | S | T | U | V | W | X | Y | Z |
| ROT13 Ciphertext  | A | B | C | D | E | F | G | H | I | J | K | L | M |

To encrypt using ROT13, find the plaintext letter in the top row and move down to the bottom row to find the corresponding ciphertext letter. To decrypt using ROT13, find the ciphertext letter in the bottom row and move up to the top row to find the corresponding plaintext letter.

If we consider the ROT13 cipher from a mathematical standpoint, we can see that to both encrypt and decrypt we simply add 13 to the numerical value of a plaintext or ciphertext letter and reduce it using modulo 26. This gives us a new number between 0 and 25 which corresponds to the encrypted or decrypted letter. Function  $E_{ROT13}$  is the encryption function. It accepts the value of a plaintext letter  $x$  as an argument and returns the value of the ciphertext letter as a result. Function  $D_{ROT13}$  is the decryption function. It accepts the value of a ciphertext letter  $x$  as an argument and returns the value of the plaintext letter as a result.

$$E_{ROT13}(x) = (x + 13) \bmod 26$$

$$D_{ROT13}(x) = (x + 13) \bmod 26$$

Notice that the definitions of both the encryption and decryption functions are, in this case, exactly the same.

**Atbash:** Atbash is an ancient technique used to encrypt the 22-letter Hebrew alphabet, but we can apply the same logic to encrypt the 26-letter English alphabet. To encrypt a letter using Atbash we need to consider the English alphabet written backwards. So encrypting the plaintext letter A becomes the ciphertext letter Z, B becomes Y, C becomes X and so on. Decrypting the ciphertext works the same way: the ciphertext letter A becomes the plaintext letter Z, B becomes Y, C becomes X and so on.

We can see this visually as a mapping in the form of a table:

|                   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|-------------------|---|---|---|---|---|---|---|---|---|---|---|---|---|
| English Plaintext | A | B | C | D | E | F | G | H | I | J | K | L | M |
| Atbash Ciphertext | Z | Y | X | W | V | U | T | S | R | Q | P | O | N |
| English Plaintext | N | O | P | Q | R | S | T | U | V | W | X | Y | Z |
| Atbash Ciphertext | M | L | K | J | I | H | G | F | E | D | C | B | A |

To encrypt using Atbash, find the plaintext letter in the top row and move down to the bottom row to find the corresponding ciphertext letter. To decrypt using Atbash, find the ciphertext letter in the bottom row and move up to the top row to find the corresponding plaintext letter.

If we consider the Atbash cipher from a mathematical standpoint, we can see that to encrypt and decrypt, we need to multiply by 25 and then add 25 to the numerical value of the plaintext or ciphertext and reduce it using modulo 26. This gives us a new number between 0 and 25 which corresponds to the encrypted or decrypted letter. Function  $E_{Atbash}$  is the encryption function. It accepts the value of a plaintext letter  $x$  as an argument and returns the value of the ciphertext letter as a result. Function  $D_{Atbash}$  is the decryption function. It accepts the value of a ciphertext letter  $x$  as an argument and returns the value of the plaintext letter as a result.

$$E_{Atbash}(x) = (25 \times x + 25) \bmod 26$$

$$D_{Atbash}(x) = (25 \times x + 25) \bmod 26$$

Notice that the definitions of both the encryption and decryption functions are, in this case, exactly the same.

**Caesar:** The Caesar cipher is also an ancient technique used encrypt and decrypt messages. To encrypt a letter using the Caesar cipher we move three positions forward. So encrypting the plaintext letter A becomes the ciphertext letter D, B becomes E, C becomes

F and so on. Decrypting the ciphertext works the opposite way. Instead of moving forward, we move three positions backward. The ciphertext letter A becomes the plaintext letter X, B becomes Y, C becomes Z and so on.

We can see this visually as a mapping in the form of a table:

|                   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|-------------------|---|---|---|---|---|---|---|---|---|---|---|---|---|
| English Plaintext | A | B | C | D | E | F | G | H | I | J | K | L | M |
| Caesar Ciphertext | D | E | F | G | H | I | J | K | L | M | N | O | P |
| English Plaintext | N | O | P | Q | R | S | T | U | V | W | X | Y | Z |
| Caesar Ciphertext | Q | R | S | T | U | V | W | X | Y | Z | A | B | C |

To encrypt using the Caesar cipher, find the plaintext letter in the top row and move down to the bottom row to find the corresponding ciphertext letter. To decrypt using the Caesar cipher, find the ciphertext letter in the bottom row and move up to the top row to find the corresponding plaintext letter.

If we consider the Caesar cipher from a mathematical standpoint, we can see that to encrypt, we need to add 3 to the numerical value of the plaintext and reduce it using modulo 26. This gives us a new number between 0 and 25 which corresponds to the encrypted letter. To decrypt, we need to subtract 3 from the numerical value of the ciphertext and reduce it modulo 26. This gives us a new number between 0 and 25 which corresponds to the decrypted letter.

Function  $E_{Caesar}$  is the encryption function. It accepts the value of a plaintext letter  $x$  as an argument and returns the value of the ciphertext letter as a result. Function  $D_{Caesar}$  is the decryption function. It accepts the value of a ciphertext letter  $x$  as an argument and returns the value of the plaintext letter as a result.

$$E_{Caesar}(x) = (x + 3) \bmod 26$$

$$D_{Caesar}(x) = (x - 3) \bmod 26$$

Notice that the definitions of both the encryption and decryption functions are, in this case, different.

We can generalise all three of the above methods by stating that they use the following encryption and decryption functions:

$$E(x) = (A_1x + B_1) \bmod 26$$

$$D(x) = (A_2x + B_2) \bmod 26$$

Here,  $A_1$ ,  $A_2$ ,  $B_1$  and  $B_2$  are constants and put together they comprise the *encryption key* for an affine cipher.

Running the following program displays a text menu. The user can choose to encrypt or decrypt a string, or quit the program. You can practice typing in a plaintext phrase to encrypt and then decrypt the ciphertext phrase to retrieve the original plaintext.

A good sample text string for testing a cipher is:

### **THE QUICK BROWN FOX JUMPS OVER THE LAZY DOG**

This text string, which is 43 characters long, contains 8 spaces and 35 alphabetic characters. Every character of the alphabet occurs at least once in this string, so encrypting and decrypting with it checks that every letter is transformed as expected.

Encrypting the above text string using the ROT13 cipher yields:

**GUR DHVPX OEBJA SBK WHZCF BIRE GUR YNML QBT**

Encrypting the above text string using the Atbash cipher yields:

**GSV JFRXP YILDM ULC QFNKH LEVI GSV OZAB WLT**

Encrypting the above text string using the Caesar cipher yields:

**WKH TXLFN EURZQ IRA MXPSV RYHU WKH ODCB GRJ**

```
10 REM *** CRYPTOGRAPHY ***
20 POKE 0,65: PRINT CHR$(142): PRINT CHR$(147)
30 DEF FN F(X)=X-INT(X/26)*26
40 C$="" P$=""
50 PRINT "SELECT AN OPTION (E, D OR Q)": PRINT
60 PRINT "(SPACE*3)[E] ENCRYPT PLAINTEXT": PRINT
70 PRINT "(SPACE*3)[D] DECRYPT CIPHERTEXT": PRINT
80 PRINT "(SPACE*3)[Q] QUIT": PRINT
90 GET $$
100 IF ($$=="Q") THEN END
110 IF ($$=="E") THEN GOSUB 150: GOTO 40
120 IF ($$=="D") THEN GOSUB 270: GOTO 40
130 GOTO 90
140 REM ENCRYPT
150 INPUT "ENTER PLAINTEXT MESSAGE TO ENCRYPT: ", P$
160 IF P$="" THEN GOTO 150
170 M$=P$: GOSUB 390
180 IF (V=0) THEN GOSUB 460: GOTO 150
190 A=1:B=3
200 FOR I=1 TO LEN(P$)
210 : L$ = MID$(P$,I,1)
220 : IF (L$=" ") THEN C$=C$+" ": ELSE C$=C$+CHR$(65+(FN F(A*(ASC(L$)-65)+B)))
230 NEXT I
240 PRINT: PRINT "(REVERSE ON)ENCRYPTED CIPHERTEXT:(REVERSE OFF)", C$: PRINT
250 RETURN
260 REM DECRYPT
270 INPUT "ENTER CIPHERTEXT MESSAGE TO DECRVPT: ", C$
280 IF C$="" THEN GOTO 270
290 M$=C$: GOSUB 390
300 IF (V=0) THEN GOSUB 460: GOTO 270
310 A=1: B=-3
320 FOR I=1 TO LEN(C$)
330 : L$ = MID$(C$,I,1)
340 : IF (L$=" ") THEN P$=P$+" ": ELSE P$=P$+CHR$(65+(FN F(A*(ASC(L$)-65)+B)))
350 NEXT I
360 PRINT: PRINT "(REVERSE ON)DECRYPTED PLAINTEXT:(REVERSE OFF)", P$: PRINT
370 RETURN
380 REM VALIDATE
390 V = 1
400 FOR I=1 TO LEN(M$)
410 : L$ = MID$(M$,I,1)
420 : IF NOT (((L$ >= "A") AND (L$ <= "Z")) OR (L$=" ")) THEN V = 0
430 NEXT I
440 RETURN
450 REM ERROR MESSAGE
460 PRINT: PRINT "USE LETTERS AND SPACES ONLY": PRINT
470 RETURN
```

If you wish to use the ROT13 cipher ensure that the following lines are changed:

```
190 A=1: B=13  
310 A=1: B=13
```

If you wish to use the Atbash cipher ensure that the following lines are changed:

```
190 A=25: B=25  
310 A=25: B=25
```

If you wish to use the Caesar cipher ensure that the following lines are changed:

```
190 A=1: B=3  
310 A=1: B=-3
```

The program listing, as written, uses the Caesar cipher by default.

# 9

## CHAPTER

# C64, C65 and MEGA65 Modes

- **Switching Modes from BASIC**
- **The KEY Register**
- **Accessing the MEGA65's Extra  
Memory from BASIC 65 in  
C65 Mode**
- **The MAP Instruction**



The MEGA65, like the C65 and the C128 has multiple operating modes, however there are important differences between the MEGA65 and both of these earlier computers. For people familiar with the C128, the most important difference is that all of the MEGA65's new features can be accessed from every mode, and you can even switch back and forth between the different modes, or create hybrid modes that combine different features from the different modes – all you need is the MAP and KEY.

In this chapter we explain the different modes, the MAP instruction and the KEY register that allows you to change the mode of operation of the computer, as well being able to use BASIC commands that can be used to completely switch from one mode to another.

## SWITCHING MODES FROM BASIC

At the time of writing, there is no MEGA65 Mode BASIC. The computer is used either in C64 mode, running BASIC 2, or C65 mode, running BASIC 65. However, various MEGA65 features can be accessed from both C64 and C65 mode. All MEGA65 features are available to programmes written in assembly language / machine code. The information required to write such programmes can be found in the various appendices.

### From C65 to C64 Mode

To switch from C65 to C64 Mode, use the familiar GO 64 command, identically to switching to C64 mode on a C128:

```
GO 64  
ARE YOU SURE? Y
```

Note that, just like on the C128, any programmes in memory will be lost in the process of switching modes.

Alternatively, you can hold down the MEGA key when pressing the reset button or turning the computer on. Again, this is the same as on the C128.

### From C64 to C65 Mode

To switch from C64 to C65 Mode, there is no official method. However the following command switches from C64 mode to C65 mode. Note that this command does not ask you for confirmation!

```
SYS 58552
```

Alternatively, you can switch back to C65 mode by pressing the reset button on the left side of the computer, or simply turning the computer off and on again.

Another option is to long-press the RESTORE key, and then choose F5 from the freeze menu. This simulates pressing the reset button.

Note that, just like on the C128, any programmes in memory will be lost in the process of switching modes.

## Entering Machine Code Monitor Mode

The machine code monitor can be entered by typing either the MONITOR command from BASIC 65 in C65 mode, or by holding the RUN/STOP key down, and then pressing the reset button on the left side of the computer.

# THE KEY REGISTER

The MEGA65 has a VIC-IV video controller chip instead of the C64's VIC-II or the C65's VIC-III. Just as the VIC-III has extra registers compared to the VIC-II, the VIC-IV has even more registers. If these were visible all of the time, software that was made for the C64 and its VIC-II might accidentally use the new registers, resulting in unexpected or unhelpful results. Therefore the creators of the C65 invented a way to hide the extra VIC-III registers from old C64 programs. This is called the KEY register. For more information about which registers are hidden and visible in each of the VIC-II, VIC-III and VIC-IV IO modes refer to Chapter/Appendix F on page F-8.

The KEY register 53295 (hex \$D02F) is just an unused register of the VIC-II, that you can POKE and PEEK like the other registers. But the KEY register has a special function: If you write two special values to it in quick succession, you can tell the VIC-IV to stop hiding the VIC-III or VIC-IV registers from the rest of the computer.

## Un-hiding C65 Extra Registers

For example, to un-hide the VIC-III's new registers when in C64 mode, you must POKE the values 165 and 150 into the KEY register. Make sure you are in C64 mode before trying the following. The easiest way to do this is to turn your MEGA65 off and on again, and type GO 64 and answer YES to enter C64 mode.

(If you do it from C65 mode, the computer will get rather confused, because in between the first and second POKE commands running, none of the C65-mode extra features will be visible to the computer, and BASIC 65 will probably crash or freeze as a result. But don't worry, if you accidentally do this, just turn the computer off and on again, or press and release the reset button the left side of the computer.)

Once you are in C64 mode, try typing the following commands:

```
POKE 53295,165: POKE 53295,150
```

When you type these commands, the computer just returns with a **READY.** prompt, and seemingly nothing else has happened. This is expected, because all we have done is un-hide the VIC-III's new registers (and some other C65 mode features) from the computer. However, the C64 BASIC and KERNAL are well behaved, and don't try to do anything strange, and so we don't immediately notice anything is different... But things are different.

As an example, we will now do something that the C64 and its VIC-II can't do: smoothly change one colour into another. The VIC-III has registers that let you change the red, green and blue components of the colours. So now that we have un-hidden those registers, we can change the colour of the background progressively from blue to purple, by increasing the red component of the colour that is normally blue on the C64. The red component value registers are at 53504 - 53759 (hex \$D100 - \$D1FF). Blue is colour 6, so we want to change register  $53504 + 6 = 53510$  (hex \$D106). We can do a nice FOR loop to change the colour for us:

```
FOR I = 0 TO 15 STEP 0.2 : POKE 53510,I : NEXT
```

You should hopefully have seen the background of the screen fade from blue to purple. If you would like to make the effect go faster, increase the 0.2 to a bigger number, like 0.5, or to make it slower, make it a smaller number, like 0.02. You can also change the red component you are changing by adding a different number to 53504. Or you might like to change the green component (53760 - 54015, hex \$D200 - \$D2FF) or blue component (54016 - 54271) - or any combination. For example, to make the border and text (since they are both normally "light blue") fade from blue to green, you could do:

```
POKE 53518,0 : FOR I = 0 TO 15 STEP 0.1 : POKE 53774,I : POKE 54030,15-I : NEXT
```

## Re-hiding C65/MEGA65 Extra Registers

You can also hide the VIC-III registers again by POKEing any number you like into the KEY register, e.g.:

```
POKE 53295,0
```

If you try the examples from above, the colours won't change this time because the registers are hidden again. Instead, writing to those addresses changes some of the VIC-II's registers because on a C64 they appear several times over. Fortunately, we chose an example where the registers don't have any ill-effect in our case (for the curious, it is the sprite positions that would be messed up, but since there are no sprites on the screen, we don't see any problems).

## Un-hiding MEGA65 Extra Registers

The MEGA65 has even more registers than the C65. To un-hide those from C64 mode, we write two different values into the KEY register:

```
POKE 53295,71: POKE 53295,83
```

(Don't forget you have to be in C64 mode, as BASIC 65 will probably crash or freeze when the C65 / VIC-III registers get briefly hidden after the first POKE has been performed, but the second one hasn't yet.)

Again, you won't see any immediate difference, just like when un-hiding C65 / VIC-III registers. However, now the computer can access not only the C64 / VIC-II and C65 / VIC-III registers, but also the MEGA65 / VIC-IV registers. If you like, you can try the examples from earlier in this chapter, to assure yourself that the C65 / VIC-III registers are accessible again. But we can also do MEGA65 specific things. For example, if we wanted to move the start of the top border higher up the screen, we could type something like:

```
POKE 53320,60
```

Or again, we could have some fun, and animate the screen borders moving closer and further apart:

```
FOR I = 255 TO 0 STEP -1 : POKE 53320,I : POKE 53322, 255 - I : NEXT
```

(We made this loop go backwards, so that you wouldn't end up with only a tiny sliver of the screen visible. But you can make it go forwards if you like. If you do get stuck with a sliver of the screen, you can just press RUN/STOP and RESTORE. You might be wondering why RUN/STOP and RESTORE works, when these are MEGA65 / VIC-IV registers that the C64-mode BASIC and KERNAL don't know about. The reason it works is because the VIC-IV has a feature called "hot registers," where certain C64 and C65 registers cause some of the MEGA65 registers to be reset to the C64 or C65 mode defaults. In this particular case, it is the KERNAL resetting the VIC-II screen using 53265 (hex \$D011), which adjusts

the vertical border size in C64/C65 mode, and is thus a “hot register” for the MEGA65’s vertical border position registers.)

See if you can instead make the screen shake around by changing the TEXTXPOS and TEXTYPOS registers of the VIC-IV. You can find out the POKE codes for those and lots of other interesting new registers by looking through Chapter/Appendix M on page M-5.

## Traps to watch out for

In both C64 and C65 mode, the DOS for the internal 3.5" disk drive (including when you use D81 disk images from an SD card) resets the KEY register to C64 / VIC-II mode whenever it is accessed. This means if you check the drive status, LOAD or SAVE a file, for example, the KEY register will be reset, and only the C64 / VIC-II registers will be visible. You can of course make the C65 or MEGA65 extra registers visible again by POKEing the correct values to the KEY register again.

# ACCESSING THE MEGA65'S EXTRA MEMORY FROM BASIC 65 IN C65 MODE

The C65's BASIC 65 contains powerful memory banking and Direct Memory Access (DMA) commands that can be used to read, fill, copy and write areas of memory beyond the C65's 128KB of RAM. The MEGA65 has 384KB of main memory. Of this, the first 128KB (BANK 0 and BANK 1) acts as the C65's normal 128KB RAM. The second 128KB (BANK 2 and BANK 3) is normally write-protected, and is used to hold the C65's ROM image. The last 128KB (BANK 4 and BANK 5) is however, completely free.

Using the BANK and PEEK and POKE commands, this region of memory can be easily accessed, for example:

```
BANK 4: POKE8,123: REM PUT 123 IN LOCATION $40000  
BANK 4: PRINT PEEK(8): REM SHOW CONTENTS OF LOCATION $40000
```

Or using the DMA command, you could copy the current contents of the screen and colour RAM into BANK 4 with something like this:

```
DMA 0, 2000, 2048, 0, 0, 4 : REM SCREEN TEXT TO BANK 4  
DMA 0, 2000, DEC("F800"), 1, 2000, 4 : REM COPY COLOUR RAM TO BANK 4
```

You could then put something else on the screen, and then copy it back with something like:

```
DMA 0, 2000, 0, 4, 2048, 0, : REM SCREEN TEXT FROM BANK 4  
DMA 0, 2000, 2000, 4, DEC("F800"), 1 : REM COPY COLOUR RAM FROM BANK 4
```

Note that there is currently no way to tell BASIC 65 to put graphics screen, variables, arrays or program text in these extra banks of RAM.

## THE MAP INSTRUCTION

The above methods can be used from BASIC. In contrast, the MAP instruction is an assembly language instruction that can be used to rearrange the memory that the MEGA65 sees. It is used by the C65 ROM and BASIC 65 to manage what memory it can see at any particular point in time. For further explanation of the MAP instruction, refer to the relevant section of Chapter/Appendix G on page G-8.

# PART III

## SOUND AND GRAPHICS



# PART IV

# HARDWARE



# 10

## CHAPTER

### Using Nexys4 boards as a MEGA65

- Building your own MEGA65 Compatible Computer
- Working Nexys4 Boards
- Power, Jumpers, Switches and Buttons
- Keyboard
- Preparing microSDHC card
- Loading the bitstream from QSPI
- Useful Tips



# BUILDING YOUR OWN MEGA65 COMPATIBLE COMPUTER

You can build your own MEGA65-compatible computer by using either a Nexys4DDR (aka. Nexys A7) or the older Nexys4 (Non-DDR) FPGA development boards. This appendix describes the process to set up a Nexys4DDR (Nexys A7) board for this purpose (which is the newer, preferred board). The older non-DDR Nexys4 board is also supported, and the instructions are the same, except that you must use a bitstream designed for that board. Using a Nexys4DDR bitstream on a non-DDR Nexys4 board, or vice versa, may cause irreparable damage to your board, so make sure you have the correct bitstream to suit your board.

DISCLAIMER: M.E.G.A cannot take any responsibility for any damage that may occur to your Nexys4DDR/NexysA7 /Nexys4 boards.

# WORKING NEXYS4 BOARDS

There are currently 3 Nexys FPGA boards which can be setup as a MEGA65:

## The Nexys4 board

No longer manufactured but still available for sale on some websites with old stock.



Documentation:

- <https://reference.digilentinc.com/reference/programmable-logic/nexys-4/reference-manual>
- [https://reference.digilentinc.com/\\_media/reference/programmable-logic/nexys-4/nexys4\\_rm.pdf](https://reference.digilentinc.com/_media/reference/programmable-logic/nexys-4/nexys4_rm.pdf)

## The Nexys4DDR board

No longer manufactured but still available for sale on some websites with old stock.



Documentation:

- <https://reference.digilentinc.com/reference/programmable-logic/nexys-4-ddr/reference-manual>
- [https://reference.digilentinc.com/\\_media/reference/programmable-logic/nexys-4-ddr/nexys4ddr\\_rm.pdf](https://reference.digilentinc.com/_media/reference/programmable-logic/nexys-4-ddr/nexys4ddr_rm.pdf)

## The Nexys A7

This is the re-branded version of the above Nexys4 DDR board:



Documentation:

- <https://reference.digilentinc.com/reference/programmable-logic/nexys-a7/reference-manual>
- [https://reference.digilentinc.com/\\_media/reference/programmable-logic/nexys-a7/nexys-a7\\_rm.pdf](https://reference.digilentinc.com/_media/reference/programmable-logic/nexys-a7/nexys-a7_rm.pdf)

# POWER, JUMPERS, SWITCHES AND BUTTONS

This top-down picture highlights the key jumper positions of interest on the Nexys4 board:



The Nexys4 boards can be powered in two ways: using an external power supply, or from a standard USB port.

## Micro-USB Power



Connect your micro-usb cable to a USB port on a USB charger or PC to provide power. Connect the other end to the Nexys4's micro-usb connector. Place the JP3 jumper on pins 1 and 2 to select USB power. Use the switch to turn on the Nexys4.

## External Power Supply



The MEGA65 core can consume a lot of power, and a standard USB port could potentially be too little for the Nexys4 board. In particular, writing to the SD card might hang or perform odd behaviour. Therefore you should consider a 5V power supply.

Digilent sell a power supply for the Nexys4 board, and we recommend you use this to ensure you avoid the risk of damage to your Nexys4 board. The chosen power supply should be center positive, 2.1mm internal diameter plug, and should deliver 4.5VDC to 5.5VDC rated at least 1 Amp.

Connect the power supply cable to the supply plug of the Nexys4. Place the JP3 jumper on pins 2 and 3 to select WALL power. Use the switch to turn on the Nexys4.

## Other Jumpers and Switches

For your initial set up, we'd suggest you set the following jumpers on your Nexys4 board to these positions:

- JP1 - USB/SD
- JP2 - SD



This will assure that the bitstream files will get loaded from your sd-card on start-up.

At some later stage, you may prefer to load the bitstream from the on-board QSPI flash, and at that point, you can revisit your JP1 jumper setting and adjust it to the QSPI position.

All 16 switches on the lower edge of the board must be set to the off position.

## Connections and Peripherals



A USB keyboard can be connected to the USB port. Only a keyboard that lacks a USB hub will work with the Nexys4 board. Generally, extremely cheap keyboards will work, while more expensive keyboards tend to have a USB hub integrated, and will not work. You may need to try several keyboards before you find one that works.

You can connect a VGA monitor to the VGA port.

The mono audio-out jack can be connected to the line-in of an amplifier.

## Communicating with your PC

There may be occasions where you wish to communicate with your Nexys4 board from your PC, in order to perform activities such as:

- Flash your QSPI flash chip via Vivado
- Upload bitstream files directly from your PC (via m65 tool)

- Make use of support tools such as M65Connect, m65, mega65\_ftp, m65dbg, etc

On such occasions, you will need to connect your micro-usb cable up to your PC.



## Onboard buttons



The “CPU RESET” button will reset the MEGA65 when pressed, while the “PROG” button will cause the FPGA itself to reload the MEGA65 core. The main difference between the two is that CPU RESET is faster, and does not clear the contents of memory, while the FPGA button is slower, and does reset the contents of memory.



Two of the five buttons in the cross arrangement can also be used: BTND acts as though you have pressed the **RESTORE** key, while BTNC will trigger an IRQ, as though the IRQ line had been pulled to ground.

## KEYBOARD

The keyboard layout is positional rather than logical. This means that keys in similar positions to the keys on a C65 keyboard will have similar function. This relationship assumes that your USB keyboard uses a US keyboard layout.

To help you locate what the various MEGA65 keys are mapped to, the MEGA65 has a built-in virtual keyboard test feature. This can be accessed in two ways.

The easiest way is to keep the **ALT** key held in while turning on the Nexys4, or resetting the Nexys4 with the "PROG" button. The configure menu will be presented and by pressing 3, the virtual keyboard will be presented on a black background.

| RUN<br>STOP | ESC          | ALT    | ASC<br>DIN | NO<br>SCRL | F1<br>F2 | F3<br>F4 | F5<br>F6 | F7<br>F8 | F9<br>F10 | F11<br>F12 | F13<br>F14      | HELP     |             |             |      |
|-------------|--------------|--------|------------|------------|----------|----------|----------|----------|-----------|------------|-----------------|----------|-------------|-------------|------|
| ~<br>—      | !<br>1       | "<br>2 | #<br>3     | \$<br>4    | %<br>5   | &<br>6   | '<br>7   | (<br>8   | )<br>9    | `<br>0     | +               | -<br>GBP | CLR<br>HOME | INST<br>DEL |      |
| TAB         | Q            | W      | E          | R          | T        | Y        | U        | I        | O         | P          | @               | *        | PI<br>~^    | RESTORE     |      |
| CTRL        | SHFT<br>LOCK | A      | S          | D          | F        | G        | H        | J        | K         | L          | {:<br>};        | =        | RETURN      |             |      |
| M=          | SHIFT        | Z      | X          | C          | V        | B        | N        | M        | <<br>,    | ><br>.:    | ?<br>/<br>SHIFT | UP       |             |             |      |
| SPACE       |              |        |            |            |          |          |          |          |           |            |                 |          | LEFT        | DOWN        | RGHT |

Pressing a key on the USB keyboard will show the highlighted key on the virtual keyboard to help you identify the key mapping.

The other way to access the virtual keyboard is from within the MEGA65. Hold **M** and press **TAB** to access the Matrix Mode Debugger. From here, enter the following:

```
s ffd3615 ff
```

This will open a semi-transparent virtual keyboard at the top of the screen. Alternatively:

```
s ffd3615 ff ff
```

This will open a semi-transparent virtual keyboard in the centre of the screen.

Hold **M** and press **TAB** to exit Matrix Mode Debugger and return to the MEGA65.

## Some key mappings with a USB keyboard

The **RESTORE** key is mapped to the PAGE UP key.

The **RUN  
STOP** key is mapped to the ESC key.

# PREPARING MICROSDHC CARD

The MEGA65 requires an SDHC card of between 4GB and 64GB capacity. Some SDXC cards may work, however, this is not officially supported.

Preparation steps for the Nexys4 board's sd-card share much in common with the steps needed for real MEGA65 hardware, and as such, it is worth having a look over the Configuring your MEGA65 chapter if you ever need details.

So in this section, we'll provide more details on the distinctive steps, and be more brief on the common steps.

One point of distinction between the Nexys board and the real MEGA65 hardware is that the latter already has a default bitstream/core provided, which permits you to format your sd-card in the specific style required by the MEGA65.

For Nexys4 board owners however, you have no such default bitstream, so see Bitstream files for more details on where the appropriate "nexys4.bit" or "nexys4ddr-widget.bit" files for your device can be downloaded from.

## Preparation Steps

The steps are:

- Format the SD card in a convenient computer using the FAT32 file-system. The MEGA65 and Nexys4 boards do not understand other file systems, especially the exFAT file system.
- Copy your bitstream file (with name ending in ".bit") onto the SD card.
- Insert the SD card into the SD card slot on the under-side of the Nexys4 board.
- Turn on the Nexys4 board.
- Enter the Utility Menu by holding the **ALT** key down on the USB keyboard you have connected to the Nexys4 board.
- Enter the FDISK/FORMAT tool by pressing 2 when the option appears on the MEGA65 boot screen.
- Follow the prompts in the FDISK/FORMAT program to again format the SD card for use by the MEGA65.

The FDISK tool will partition your SD card into two partitions and format them.

- One is type \$41 = MEGA65 System Partition, where the save slots, configuration data and other files live.  
(This partition is invisible in i.e. Win PCs).

- The other partition with type \$0C = VFAT32, where KERNEL, support files, games, and so on, will be copied to later.  
(This partition is visible on i.e. Win PCs).
- Once formatting is complete, switch off the Nexys4 board and remove the microSD card from the Nexys board and put it back into your PC
- This time, copy the following items onto the SD card:
  - The bitstream file
  - The extracted files from within either the "**SD essentials.rar**" or "**SD essentialsNoROM.rar**" file that you downloaded from the MEGA65 filehost. (See Installing ROM and Other Support Files for more details).
  - If you have sourced your own preferred ROM file (e.g. "**911001.BIN**"), copy it onto the SD card also, and rename it to "**MEGA65.ROM**" (uppercase is essential).
  - Any .D81 disk image files you wish to make use of.
    - \* Note that if a file named MEGA65.D81 is added to the SD card, it will be mounted automatically on startup.
    - \* Make sure that all .D81 files have names that fit the old DOS 8.3 character limit, and are upper case. This restriction will be removed in a future release.
- Remove the SD card and reinsert it into your Nexys4 board.
- Power the Nexys4 board back on. The MEGA65 should boot within 15 seconds.
- On first start up, you will find yourself at the on-boarding screen, of which more details can be found in the Configuring your MEGA65 chapter.

Congratulations. Your MEGA65 has been set up and is ready to use.

Please note that the above method of copying the bitstream file to the SD card means that the bitstream is loaded into the Nexys FPGA each time on boot - which takes around 13 seconds for the system to start. The bitstream can also be flashed using Vivado software into the QSPI flash to deliver a boot up time of 0.3 seconds.

For more detailed information on preparing and configuring your MEGA65, please refer to the Configuring your MEGA65 chapter.

# LOADING THE BITSTREAM FROM QSPI

While loading the bitstream from the SD-card is the suggested (and well-trodden) path this document has chosen, of late, more nexys4 users have been exploring the alternative pathway of loading the bitstream from the QSPI flash. Some potential reasons they have chosen this pathway are:

- Faster loading times (0.3 seconds versus 13 seconds)
- Some people were interested in the possibility of flashing multiple cores onto their QSPI (via steps described in the Cores and Flashing Chapter)
- Some people have experienced niggling issues with the sd-card pathway, such as:
  - System unable to reboot from on-boarding screen
  - System unable to reboot from freeze-menu after switching between PAL/NTSC

In time, if this proves to be a more popular pathway, we can revise our documentation here to suit it. Here are some steps in brief.

## Preparation Steps

For users that want to try this pathway, you will need to adjust the JP1 jumper setting to use QSPI and then follow the steps in the Flashing the FPGAs and CPLDs in the MEGA65 chapter in relation to Installing Vivado and Flashing the main FPGA using Vivado.

Be forewarned that the installation of Vivado is a lengthy process (both in terms of download time, and installation time).

Once you have flashed Slot0 of your QSPI chip via Vivado, you can then follow the steps described in Configuring your MEGA65 to perform the custom sd-card formatting, installing of ROM and support files and on-boarding.

## USEFUL TIPS

The following are some useful tips for getting familiar with the MEGA65:

- Press & hold  (or the Commodore key if using a Commodore 64 or 65 keyboard) during boot to start up in C64 mode instead of C65 mode
- Press & hold  during boot to enter the machine language monitor, instead of starting BASIC.
- Press the **RESTORE** key for approximately 1/2 - 1 second to enter the MEGA65 Freeze Menu. From this menu you have convenient tools to change the CPU speed, switch between PAL & NTSC video mode, change Audio settings, manage freeze-states, select D81 disk images, examine and modify memory of the frozen program, among other features. This is in many ways the heart of the MEGA65, so it is well worth exploring and getting familiar with.
- Type **POKE0,65** in C64 mode to switch the CPU to full speed (40MHz). Some software may behave incorrectly in this mode, while other software will work very well, and run many times faster than on a C64.
- Type **POKE0,64** in C64 mode to switch the CPU to 1MHz.
- Type **\$Y958552** in C64 mode to switch to C65 mode.
- Type **6064** in C65 mode and confirm, by pressing **Y**, to switch to C64 mode, just like on a C128.
- The C65 ROM makes device 8 the default, so you can normally leave off the **,8** from the end of LOAD and SAVE commands.
- Pressing **SHIFT** +  from either C64 or C65 mode will attempt to boot from disk.

Have fun! The MEGA65 has been lovingly crafted over many years for your enjoyment. We hope you have as much fun using it as we have had creating it!

The MEGA Museum of Electronic Games & Art welcomes your feedback, suggestions and contributions to this open-source digital heritage preservation project.



# PART V

## CROSS-PLATFORM DEVELOPMENT TOOLS



# 11

## CHAPTER

## Emulators

- Using The Xmega65 Emulator
- Using the Live ISO image



At the time of writing, there is only one emulator for the MEGA65, **xmega65**, from LGB's Xemu emulator suite. The developers of this emulator are working hard to keep up with the development of the MEGA65. Thus some aspects of the MEGA65 may not be perfectly emulated. However, it is typically sufficient for developing software for the MEGA65. To be safest, you should always test software regularly on real hardware, whether that be on a MEGA65 computer, or on an FPGA board which is capable of running a MEGA65 core.

The source code for the MEGA65 emulator can be downloaded from <https://github.com/lgb1gb1gb/xemu>. Pre-compiled version can be downloaded from <https://github.lgb.hu/xemu/>. There is also a live ISO image containing the emulator, documentation and other tools. The link to this ISO image can be found in on Forum64.de at <https://www.forum64.de/index.php?thread/104698-xemu-live-system-iso-file/&postID=1549927#post1549936>.

## USING THE XMEGA65 EMULATOR

### USING THE LIVE ISO IMAGE

The Live ISO image is the product of volunteers in the community. The MEGA65 team are not responsible for it, but we document it here for your convenience.

#### Creating a Bootable USB stick or DVD

The first step is to create the live ISO image. The method for doing this will depend on your operating system, and whether you wish to install it on a USB stick, or burn it to a DVD. Burning to a DVD is straightforward, assuming you own a computer that has a DVD writer. If you don't, or if you wish to in any case make a bootable USB stick, e.g., because it will boot faster than a DVD, you will need to use an appropriate tool to make a bootable USB stick from an ISO image. If you are using Windows, you might consider a tool like <http://www.isotousb.com/>. On Linux, you can use the instructions at <https://fossbytes.com/create-bootable-usb-media-from-iso-ubuntu/>. For Apple Macs, you might consider the information at <https://ubuntu.com/tutorials/create-a-usb-stick-on-macos#1-overview>. Similar instructions are available for other popular computers, such as Amigas (<https://forum.hyperion-entertainment.com/viewtopic.php?t=3857>), or Sun UltraSPARC workstations (<https://forums.servethehome.com/index.php?threads/how-to-create-a-bootable-solaris-11-usb.1998/>).

## Getting Started

To avoid any potential problems with copyright, the bootable ISO image does not include any proprietary ROMs for the MEGA65, such as the legacy C65 ROMs. It does include an open-source replacement ROM from our OpenROMs project. This is sufficient to boot into a BASIC 2 like environment, that can be used to load and execute many C64 programmes. It will result in a display like the following:



However, if you wish to use a C65 ROM and its included BASIC 65, you will need to download the appropriate ROM file, and place it on another USB stick, and named **MEGA65.ROM**. The Live ISO will prompt you on start-up to ask if you have such a file already downloaded:



If the Live ISO cannot find this file, it will ask you if you would like for it to automatically download such a ROM for you, as can be seen below:



Naturally you will need to have an internet connection available for this to work.

## Other Features of the Live ISO

As the previous screen-shots show, the Live ISO provides various desktop shortcuts for your convenience. On the left-hand side, we see shortcuts for launching the MEGA65 emulator, and also the C64 emulator, if you wish to check if your programmes can run on a standard C64 computer. As previously mentioned, both emulators are works in progress, and thus may not be 100% faithful in all aspects of their emulation.

Next we have a link to the MEGA65 Book, which is the all-in-one volume containing the almost 800 pages of all official MEGA65 documentation. The majority of this developer's guide is also present in the MEGA65 Book.

Below this, there is documentation for the C64 Notepad, a programme for the C64 and MEGA65 written by Snoopy, who also prepared the Live ISO image. A "read me" file is also provided, that contains further information about the Live ISO.

Finally, there is a link to access the contents of the Live ISO image via the file explorer, should you wish to do so.

Then on the right-hand side, there are links to download a C64 ROM, and to update the MEGA65 Book to the latest version, so that you don't need to create a new bootable image each time the MEGA65 Book is updated. This is very helpful, as the MEGA65 Book continues to grow and evolve.



# 12

## CHAPTER

### Data Transfer and Debugging Tools

- **m65 command line tool**
- **M65Connect**
- **mega65\_ftp**
- **TFTP Server**
- **Converting a BASIC text file listing into a PRG file**



The key to effective cross-platform development is having quick and easy means to deploy and test software on the MEGA65. This is especially true while the MEGA65 emulator continues to be developed. In fact, even once the MEGA65 emulator is complete, it is unlikely that it will be able to offer full compatibility at full speed, because the MEGA65 is much more demanding to emulate than the C64.

There are a variety of tools that can be used for data transfer and debugging. These typically function using either the MEGA65's serial monitor interface, or via the MEGA65's fast ethernet adapter. The serial monitor interface is available via the UART lines on the JB1 header.

If you do not have access to the serial monitor interface, there are tools being developed for the fast ethernet port that provide some, but not all, of the capabilities of the serial monitor interface. These will be documented as they become available. The remainder of this chapter focusses on methods that access the serial monitor interface.

You can either connect a 3.3V UART adaptor to the appropriate lines, or more conveniently, connect a TE-0790-03 JTAG debug module onto this connector. This gives you a USB connection that can be used for injecting software, remote debugging and memory inspection, as well as activating or flashing bitstreams. With this connection, there are the following tools:

## M65 COMMAND LINE TOOL

The <https://github.com/mega65/mega65-tools> repository contains a number of tools, utilities and example programmes. These tools are mainly for Linux but can be used on Windows with Cygwin. One of those is the **m65** command line tool. This is rather a swiss-army knife collection of utilities in one. Common useful functions include:

### Screenshots using m65 tool

To take a screenshot of the MEGA65 use:

```
m65 -s
```

This will create a file called **mega65-screeen-000000.png**, or if that file already exists, the first non-used number will be used in place of **000000**.

Note that this screenshot function works by having **m65** emulate the function of the VIC-IV. Thus while it produces excellent looking digital screenshots, it may

not exactly match the real display of the MEGA65. At the time of writing it does not render sprites or bitplanes, only text and bitmap-based video modes.

## Load and run a programme on the MEGA65

To load and run a programme on the MEGA65, you can use a command like:

```
m65 -F -4 -r foo.prg
```

The **-F** option tells **m65** to reset the MEGA65 before loading the programme.

The **-4** option tells **m65** to switch the MEGA65 to C64 mode before loading the programme. If this is left off, then it will attempt to load the programme in C65 mode.

The **-r** option tells **m65** to run the programme immediately after loading.

Note that this command works using the normal BASIC LOAD command, and is thus limited to loading programmes into the lower 64KB of RAM

## Reconfigure the FPGA to run a different bitstream

To try out a different bitstream, a command like the following can be used:

```
m65 -b bitstream.bit
```

This will cause the named bitstream to be sent to the FPGA. As the FPGA will be reconfigured by this action, and programme currently running will not merely be stopped, but also main memory will be cleared. For models of the MEGA65 that are fitted with 8MB or 16MB of expansion memory, those expansion memories are implemented in external chips, and so the contents of them will not be erased.

## Remote keyboard entry

The MEGA65's keyboard interface logic supports the injection of synthetic key events using the registers \$D615 – \$D617. The **m65** utility uses this to allow remote typing on the MEGA65 in a way that is transparent to software. There are three ways to use this:

```
m65 -t sometext
```

This form types the supplied text, in this case *sometext*, but does not simulate pressing the  key. If you wish to simulate the pressing of the  key, use **-T** instead of **-t**, e.g.:

```
m65 -T list
```

This would cause the LIST command to be typed and executed.

Finally, it is possible to begin general remote keyboard control via:

```
m65 -t -
```

In this mode, any key pressed on the keyboard of the computer where **m65** is running will be relayed to the MEGA65. Note that not all special keys are supported, and that there is some latency, so using key repeat can cause unexpected results. But for general remote control, it is a very helpful facility.

## Unit testing and logging support

The **m65** tool includes support to facilitate remote unit testing directly on MEGA65 hardware. When *unit testing mode* is active, **m65** waits for the MEGA65 to send certain byte sequences over the serial interface which signal the current state (started, passed, failed) of a given test. Additionally, it is possible to send log messages from the MEGA65 to the host computer.

Unit testing mode is entered by calling **m65** with the **-u** flag. To run a remote BASIC program in C65 mode and simultaneously put **m65** into *unit testing mode*, the following command can be used:

```
m65 -Fur attic-ram.prg -w tests.log
```

The **-F** and **-r** options tell **m65** to reset the MEGA65 before loading the program "attic-ram.prg" and then automatically run it. The **-u** option then tells **m65** go into *unit testing mode* instead of exiting after launching the program. The optional **-w** option makes **m65** append the test results to the file "test.log" (creating the file if it doesn't exist).

Please note that **m65** automatically exits from *unit testing mode* if no test state signals were received for over 10 seconds.

Support is provided for sending unit test signals to the host computer from C and BASIC65 programs:

## Using unit tests with C

The MEGA65 libc contains support for unit testing via functions defined in `tests.h` and `tests.c`.

To signal the start of a test, include `tests.h` and use

```
unit_test_setup("testName", issueNumber);
```

where "`testName`" is a human-readable name of the test (e.g. "VIC-II") and `issueNumber` a reference to the corresponding bug issue (for example, the issue number from github).

After starting a test, it's possible to signal passed tests with the `unit_test_ok()` function:

```
unit_test_ok();
```

A failed test is signalled with `unit_test_fail()`:

```
unit_test_fail("fail message");
```

Each time the `unit_test_ok()` or `unit_test_fail()` functions are called, the *sub issue* of the test (reported on the host computer) is incremented. This makes it easier to combine and identify multiple tests in one file.

You can send arbitrary log messages via `unit_test_log()`:

```
unit_test_log("hello world from mega65!");
```

...and finally, when all is done, the end of unit testing is signalled by the use of

```
unit_test_done();
```

## Using unit tests with BASIC65

`b65support.bin` is a machine language module providing support for unit testing from BASIC65, available in the `bin65` folder of the mega65-tools repository. This module works by redirecting the USR vector to perform the functions needed to communicate with the testing host.

In an automated test scenario, you may want to inject the `b65support.bin` binary into MEGA65 RAM by using `m65`:

```
m65 -C mega65-tools/bin65/b65support.bin@15fe
```

Of course it's also possible to load `b65support.bin` directly from the MEGA65 by mounting the `M65UTILS.D81` image from the freezer and issuing

```
BL0AD "B65SUPPORT.BIN"
```

After loading, **b65support.bin** is initialized with

```
SVS $1600
```

Once initialized, the following functions are provided by **b65support.bin**:

```
A=USR(<issueNum>)
```

prepares a new test with number <issueNum> and resets subissue number to 0

```
A=USR("=<testName>")
```

sets test name and sends test start signal; for example: **A=USR("=VIC-III")** sets the test name to 'VIC-III' and signals the host computer that the test has started.

```
A=USR("/<logMessage>")
```

sends a log message to the host computer

```
A=USR("P")
```

sends the 'passed' signal to the host computer and increases the sub issue number

```
A=USR("F")
```

sends the 'test failed' signal to the host computer and increases the sub issue number

```
A=USR("D")
```

sends the 'test done' signal to the host computer

All calls return the current sub issue number or **ILLEGAL QUANTITY ERROR** in case of calling an invalid command.

## BASIC65 example

The following is a complete BASIC65 example showing how to use **m65**'s unit testing features:

```
100 rem attic raw cache test
110 poke $bfffff2,$e0          : rem enable attic raw cache
120 sys $1600                  : rem init test module
130 a=usr(379)                 : rem set issue number
140 a=usr("=attic-raw-cache")  : rem set test name
150 bank128:poke0,65          : rem just to be sure
160 b0=$80000000 : b1=$8000100 : rem attic raw areas to be tested
170 for r=0 to $ff
180   poke b0+r,0              : rem fill area 1 with 0
190   poke b1+r,$ff            : rem fill area 2 with $ff
200 next r
210 for t=1 to 10             : rem 10 tries
220   poke b0,32               : rem write to b0
230   for x=0 to $ff:ti=bit+x
240     a=peek(b0)             : rem read from b0
250     b=peek(ti):b=peek(ti) : rem read twice from ti
260     ifb<>255 then f=t:t=11:x=256 : rem this shouldn't happen
270   next x
280 next t
290 if f=0 then begin
300   print "no faults detected after";t;"tries."
310   a=usr("p")                : rem signal 'test passed' to host
320 bend : else begin
330   a=usr("f")                : rem signal 'test failed' to host
340   print "hyper raw fault detected after";f;"tries."
350   print "peek($";hex$(ti);") [ti] is";b;"but should be 255"
360 bend
370 a=usr("d")                  : rem test done
```

## M65CONNECT

This is a cross-platform graphical tool available for Windows, Linux and Mac OSX, which allows access to most of the functions of the **m65** command-line tool, without needing to use a command line, or being able to compile the tool for your preferred operating system.

The repository for M65Connect is: <https://github.com/MEGA65/m65connect>

The latest binary version is available from <https://files.mega65.org>.

With the MEGA65 or Nexys FPGA switched off, connect a USB cable from your computer to the MEGA65 or Nexys FPGA board. Run the *M65Connect* executable and follow the prompts to connect. The program will help you identify which USB Serial Port to communicate over.

With this tool you can easily transfer PRG programs and a variety of other files. M65Connect can handle the transfer, switching to C64 mode, and execution of programs.

## MEGA65\_FTP

The **mega65\_ftp** utility from the <https://github.com/MEGA65/mega65-tools> repository is a little misleadingly named: While it is a File Transfer Programme, it does not use the File Transfer Protocol (FTP). Rather, it uses the serial monitor interface to take remote control of a MEGA65, and directly access its SD card to enable copying of files between the MEGA65 and the host computer.

Note that it does not perfectly restore the MEGA65's state on exit, and thus should only be used when the MEGA65 is at the READY prompt, so that any running software doesn't go haywire. In particular, you should avoid using it when a sensitive programme is running, such as the Freeze Menu, MEGA65 Configuration Utility, or the MEGA65 Format/FDISK utility. (This problem could be solved with a little effort, if someone has the time and interest to fix it).

When run, it provides an FTP-like interface that supports the **get**, **put**, **rename** and **dir** commands. Note that when putting a file, you should make sure that it is given a name that is all capitals and has a DOS-compatible 8.3 character file name. This is due to limitations in both **mega65\_ftp** and the MEGA65's Hypervisor's VFAT32 file system code. Again, these problems could be fixed with a modest amount of effort on the part of a motivated member of the community.

Finally, the **mega65\_ftp** programme is very slow to push new files to the MEGA65, typically yielding speeds of around 5KB/sec. This is partly because the serial monitor interface is capable of transferring data at only 40KB/sec (when set to 4,000,000 bits per second), and partly because the remote control process results in a lot of round-trips where helper routines are executed on the MEGA65 to read, write and verify sectors on the SD card. It would

be quite feasible to improve this to reach close to 40KB/sec, and potentially faster using either some combination of data compression, de-duplication of identical sectors (especially when uploading disk images) and other techniques. Again, this would be a very welcome contribution that someone in the community could contribute to everyone's benefit.

## TFTP SERVER

Work on a true TFTP server for the MEGA65 that supports fast TFTP transfers over the 100mbit ethernet has begun, and can be used to very quickly read files from the MEGA65. Speeds of close to 1MB/sec are possible, depending on SD card performance. Rather than using DHCP, this utility will respond to *any* IP address that ends in .65. It always uses the MAC address 40:40:40:40:40:40. True DHCP support as well as using the MEGA65's configured ethernet MAC address may be added in the future.

More importantly, support for writing files to the SD card is not yet complete, and is blocked by the need for the implementation of the necessary functions in the MEGA65's Hypervisor for creating and growing files. A particular challenge is enabling the creation of files with contiguous clusters as is required for D81 disk images: If a D81 file is fragmented, then it cannot be mounted, because the mounting mechanism requires a pointer to the contiguous block of the SD card containing the disk image. In the interim, **mega65\_ftp** can be used as a substitute.

## CONVERTING A BASIC TEXT FILE LISTING INTO A PRG FILE

If you have a untokenised BASIC program in plain text format sourced from somewhere like an internet post, and you wish to try it on the MEGA65 without typing it in, it is possible to convert it to a PRG.

*C64List* is a Windows-based command-line tool that will allow you to make the conversion. Once you have a .PRG file, you can use a tool like M65Connect to upload it to the MEGA65 or Nexys FPGA.

*C64List* is available for download from <http://www.commodoreserver.com/Downloads.asp>

Ensure you have a program listing saved to a file on your local computer (for example, *program.txt*) encoded as ANSI or UTF8.

Use C64List to convert the file to a PRG file using:

```
C64List program.txt -prg
```

Now you can upload your newly converted program to the MEGA65 with M65Connect or one of the other tools described previously.

It is worth noting that this method will not be 100% effective on listings with special PETSCII characters. Programs with PETSCII will require some editing on the MEGA65 itself before saving to disk.



# 13

## CHAPTER

# Assemblers



The table below shows an overview of assemblers known to work with MEGA65. For general use we recommend **ACME** as it has good support for the 45GS02 instruction set; is open source; and finally written in C. The latter means that it may be ported to run natively on the MEGA65 in the future.

| Name    | 45GS02          | Source | Reference                                                                                                   |
|---------|-----------------|--------|-------------------------------------------------------------------------------------------------------------|
| ACME    | yes             | C      | <a href="https://sourceforge.net/projects/acme-crossass">https://sourceforge.net/projects/acme-crossass</a> |
| KickAss | yes             | Java   |                                                                                                             |
| Ophis   | yes             | Python | <a href="https://github.com/michaelcmartin/Ophis">https://github.com/michaelcmartin/Ophis</a>               |
| BSA     | yes             | C      | <a href="https://github.com/Edilbert/BSA">https://github.com/Edilbert/BSA</a>                               |
| CA65    | no <sup>1</sup> | C      | <a href="https://github.com/mega65/cc65">https://github.com/mega65/cc65</a>                                 |

The **BSA** assembler is currently used to build the **MEGA65.ROM**. Most of this source code is written in the syntax of the ancient **BSO** assembler (Boston Systems Office), which was used in the years 1989 - 1991 by software developers, working on the C65. The **BSA** Assembler has a compatibility mode, which makes it possible to assemble these old source codes with minor or none modifications. The **BSA** Assembler has currently only a description of commands embedded in the C-source of the assembler.

---

<sup>1</sup> Our fork of CA65 (part of CC65) correctly detects the MEGA65's CPU, but has no explicit support for the processor's features



# 14

## CHAPTER

### C and C-Like Compilers

- MEGA65 libc



Short answer: CC65 and KickC both work on the MEGA65.

Both CC65 and KickC are known to work on the MEGA65. However, both by default have only a C64 memory model, and use only 6502 opcodes. It would be super for someone to create a C65 memory configuration for CC65, and should not be too hard to do.

CC65 supports overlays, which could be powerfully used with the MEGA65's extra memory to allow programmes larger than 64KB. However, this would require writing a suitable loader for such programmes, which also does not yet exist.

Similarly, modifying the code generator of CC65 to use 45GS02 features would not be particularly difficult to do, and would help to overcome the otherwise horribly slow and bloated code that CC65 produces. Also adding first-class support for the 45GS02 CPU features in CA65 (or perhaps even better, making CC65 produce ACME compatible assembly output) would be of tremendous advantage, and not particularly hard to do. These would all be great tasks to tackle while you wait for your MEGA65 DevKit to arrive!

An example template for a C programme that can be compiled using CC65 and executed on the MEGA65 can be found in the repository <https://github.com/MEGA65/hello-world>. This repository will even download and compile CC65, if you don't already have it installed on your system. This repository should work on Linux and Mac, and on Windows under the Windows Subsystem for Linux (WSL).

## MEGA65 LIBC

A C library is being developed for the MEGA65, and which already includes a number of useful features. This library is available from <http://github.com/mega65/mega65-libc>. The procedures, functions and definitions it provides are documented in a separate chapter.

The MEGA65 libc is currently available only for CC65, although we would welcome someone maintaining a KickC port of it.



# 15

## CHAPTER

### MEGA65 Standard C Library

- Structure and Usage
- `conio.h`



A C library is being developed for the MEGA65, and which already includes a number of useful features. This library is available from <http://github.com/mega65/mega65-libc>. The procedures, functions and definitions it provides are documented in a separate chapter.

The MEGA65 libc is currently available only for CC65, although we would welcome someone maintaining a KickC port of it.

## STRUCTURE AND USAGE

The MEGA65 libc is purposely provided in source-form only, and with groups of functions in separate files, and with separate header files for including. The idea is that you include only the header files that you require, and add only the source files required to the list of source files of the programme you are compiling. This avoids the risk of the compiler including functions in your compiled programme that are never used, and thus wasting precious memory space.

Note that some library source files are written in C, and thus are present as files with a `.c` extension, while others are written in assembly language either for efficiency or out of necessity, and have a `.s` extension.

Typical usage is to either have the `mega65-libc` source code checked out in an adjacent directory, or within the source directory of your own project. In the latter case, this can be done using the git submodule facility.

The following sections document each of the header files and the corresponding functions that they provide.

## CONIO.H

### conionit

**Description:**

Initialises the library internal state

**Syntax:**    `void conioinit(void)`

**Notes:**    This must be called before using any conio library function.

## setscreenaddr

**Description:**

Sets the screen RAM start address

**Syntax:** `void setscreenaddr(long addr);`**Parameters:**

**addr:** The address to set as start of screen RAM

**Notes:** No bounds check is performed on the selected address

## getscreenaddr

**Description:**

Returns the screen RAM start address

**Syntax:** `long getscreenaddr(void);`**Return Value:**

The current screen RAM address start address.

## setcolramoffset

**Description:**

Sets the color RAM start offset value

**Syntax:** `void setcolramoffset(long offset);`**Parameters:**

**addr:** The offset from the beginning of the color RAM address (\$FF80000)

**Notes:** No bounds check is performed on the resulting address. Do not exceed the available Color RAM size

## getcolramoffset

**Description:**

Returns the color RAM start offset value

**Syntax:** `long getcolramoffset(void);`

**Return Value:**

The current color RAM start offset value.

## setcharsetaddr

**Description:**

Sets the character set start address

**Syntax:** `void setcharsetaddr(long addr);`**Parameters:**

**addr:** The address to set as start of character set

**Notes:** No bounds check is performed on the selected address

## getcharsetaddr

**Description:**

Returns the current character set start address

**Syntax:** `long getscreenaddr(void);`**Return Value:**

The current character set start address.

## clrscr

**Description:**

Clear the text screen.

**Syntax:** `void clrscr(void)`

**Notes:** Color RAM will be cleared with current text color

## getscreensize

**Description:**

Returns the dimensions of the text screen

**Syntax:** `void getscreensize(unsigned char* width, unsigned char* height)`

**Parameters:**

**width:** Pointer to location where width will be returned

**height:** Pointer to location where height will be returned

## setscreensize

**Description:**

Sets the dimensions of the text screen

**Syntax:** `void setscreensize(unsigned char width, unsigned char height)`

**Parameters:**

**width:** The width in columns (40 or 80)

**height:** The height in rows (25 or 50)

**Notes:** Currently only 40/80 and 25/50 are accepted. Other values are ignored.

## set16bitcharmode

**Description:**

Sets or clear the 16-bit character mode

**Syntax:** `void set16bitcharmode(unsigned char f)`

**Parameters:**

**f:** Set true to set the 16-bit character mode

**Notes:** This will trigger a video parameter reset if HOTREG is ENABLED. See sethotregs function.

## sethotregs

**Description:**

Sets or clear the hot-register behavior of the VIC-IV chip.

**Syntax:** `void set16bitcharmode(unsigned char f)`

**Parameters:**

**f:** Set true to enable the hotreg behavior

**Notes:** When this mode is ENABLED a video mode reset will be triggered when touching \$D011, \$D016, \$D018, \$D031 or the VIC-II bank bits of \$DD00.

## setextendedattrib

### Description:

Sets or clear the VIC-III extended attributes mode to support blink, underline, bold and highlight.

**Syntax:** `void setextendedattrib(unsigned char f)`

### Parameters:

**f:** Set true to set the extended attributes mode

## togglecase

### Description:

Toggle the current character set case

**Syntax:** `void togglecase(void)`

## bordercolor

### Description:

Sets the current border color

**Syntax:** `void bordercolor(unsigned char c)`

### Parameters:

**c:** The color to set

## bgcolor

### Description:

Sets the current screen (background) color

**Syntax:** `void bgcolor(unsigned char c)`

**Parameters:**

**c:** The color to set

## textcolor

**Description:**

Sets the current text color

**Syntax:** `void textcolor(unsigned char c)`**Parameters:**

**c:** The color to set

## revers

**Description:**

Enable the reverse attribute

**Syntax:** `void revers(unsigned char c)`**Parameters:**

**enable:** 0 to disable, 1 to enable

**Notes:** Extended attributes mode must be active. See setextendedatrib.

## highlight

**Description:**

Enable the highlight attribute

**Syntax:** `void highlight(unsigned char c)`**Parameters:**

**enable:** 0 to disable, 1 to enable

**Notes:** Extended attributes mode must be active. See setextendedatrib.

# blink

**Description:**

Enable the blink attribute

**Syntax:** `void blink(unsigned char c)`**Parameters:**

**enable:** 0 to disable, 1 to enable

**Notes:** Extended attributes mode must be active. See setextendedattrib.

# underline

**Description:**

Enable the underline attribute

**Syntax:** `void underline(unsigned char c)`**Parameters:**

**enable:** 0 to disable, 1 to enable

**Notes:** Extended attributes mode must be active. See setextendedattrib.

# cellcolor

**Description:**

Sets the color of a character cell

**Syntax:** `void cellcolor(unsigned char x, unsigned char y, unsigned char c)`**Parameters:**

**x:** The cell X-coordinate

**y:** The cell Y-coordinate

**c:** The color to set

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# fillrect

## Description:

Fill a rectangular area with character and color value

**Syntax:** `void fillrect(const RECT *rc, unsigned char ch,  
unsigned char col)`

## Parameters:

**rc:** A RECT structure specifying the box coordinates

**ch:** A char code to fill the rectangle

**col:** The color to fill

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# box

## Description:

Draws a box with graphic characters

**Syntax:** `void box(const RECT *rc, unsigned char color, un-  
signed char style, unsigned char clear, unsigned  
char shadow)`

## Parameters:

**rc:** A RECT structure specifying the box coordinates

**color:** The color to use for the graphic characters

**style:** The style for the box borders. Can be set to BOX\_STYLE\_NONE, BOX\_STYLE\_ROUNDED, BOX\_STYLE\_INNER, BOX\_STYLE\_OUTER, BOX\_STYLE\_MID

**clear:** Set to 1 to clear the box interior with the selected color

**shadow:** Set to 1 to draw a drop shadow

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# **hline**

## **Description:**

Draws an horizontal line.

**Syntax:**    `void hline(unsigned char x, unsigned char y, unsigned char len, unsigned char style)`

## **Parameters:**

**x:** The line start X-coordinate

**y:** The line start Y-coordinate

**len:** The line length

**style:** The style for the line. See HLINE\_ constants for available styles.

**Notes:**    No screen bounds checks are performed; out of screen behavior is undefined

# **vline**

## **Description:**

Draws a vertical line.

**Syntax:**    `void vline(unsigned char x, unsigned char y, unsigned char len, unsigned char style)`

## **Parameters:**

**x:** The line start X-coordinate

**y:** The line start Y-coordinate

**len:** The line length

**style:** The style for the line. See VLINE\_ constants for available styles.

**Notes:**    No screen bounds checks are performed; out of screen behavior is undefined

# gohome

**Description:**

Set the current position at home (0,0 coordinate)

**Syntax:** `void gohome(void)`

# gotoxy

**Description:**

Set the current position at X,Y coordinates

**Syntax:** `void gotoxy(unsigned char x, unsigned char y)`**Parameters:**

**x:** The new X-coordinate

**y:** The new Y-coordinate

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# gotox

**Description:**

Set the current position X-coordinate

**Syntax:** `void gotox(unsigned char x)`**Parameters:**

**x:** The new X-coordinate

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# gotoy

**Description:**

Set the current position Y-coordinate

**Syntax:** `void gotoy(unsigned char y)`**Parameters:**

**y:** The new Y-coordinate

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

## moveup

**Description:**

Move current position up

**Syntax:** `void moveup(unsigned char count)`

**Parameters:**

**count:** The number of positions to move

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

## movedown

**Description:**

Move current position down

**Syntax:** `void movedown(unsigned char count)`

**Parameters:**

**count:** The number of positions to move

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

## moveleft

**Description:**

Move current position left

**Syntax:** `void moveleft(unsigned char count)`

**Parameters:**

**count:** The number of positions to move

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# **moveright**

**Description:**

Move current position right

**Syntax:** `void moveright(unsigned char count)`**Parameters:**

**count:** The number of positions to move

**Notes:** No screen bounds checks are performed; out of screen behavior is undefined

# **wherex**

**Description:**

Return the current position X coordinate

**Syntax:** `unsigned char wherex(void)`**Return Value:**

The current position X coordinate

# **wherey**

**Description:**

Return the current position Y coordinate

**Syntax:** `unsigned char wherey(void)`**Return Value:**

The current position Y coordinate

# **cputc**

**Description:**

Output a single character to screen at current position

**Syntax:** `void cputc(unsigned char c)`**Parameters:**

**c:** The character to output

## cputnc

### Description:

Output N copies of a character at current position

Syntax: `void cputnc(unsigned char count, unsigned char c)`

### Parameters:

**c:** The character to output

**count:** The count of characters to print

## cputhex

### Description:

Output an hex-formatted number at current position

Syntax: `void cputhex(long n, unsigned char prec)`

### Parameters:

**n:** The number to write

**prec:** The precision of the hex number, in digits. Leading zeros will be printed accordingly

**Notes:** The \$ symbol will be automatically added at beginning of string

## cputdec

### Description:

Output a decimal number at current position

Syntax: `void cputdec(long n, unsigned char padding, unsigned char leadingZ)`

### Parameters:

**n:** The number to write

**padding:** The padding space to add before number

**leadingZ:** The leading zeros to print

## cputs

**Description:**

Output a string at current position

**Syntax:** `void cputs(const unsigned char* s)`**Parameters:**

**s:** The string to print

**Notes:** No pointer check is performed. If s is null or invalid, behavior is undefined

## cputssxy

**Description:**

Output a string at X,Y coordinates

**Syntax:** `void cputssxy (unsigned char x, unsigned char y,  
const unsigned char* s)`**Parameters:**

**x:** The X coordinate where string will be printed

**y:** The Y coordinate where string will be printed

**s:** The string to print

**Notes:** No pointer check is performed. If s is null or invalid, behavior is undefined

## cputcxy

**Description:**

Output a single character at X,Y coordinates

**Syntax:** `void cputcxy (unsigned char x, unsigned char y,  
unsigned char c)`**Parameters:**

**x:** The X coordinate where character will be printed

**y:** The Y coordinate where character will be printed

**c:** The character to print

## cputncxy

### Description:

Output N copies of a single character at X,Y coordinates

**Syntax:** `void cputncxy (unsigned char x, unsigned char y,  
unsigned char count, unsigned char c)`

### Parameters:

**x:** The X coordinate where character will be printed

**y:** The Y coordinate where character will be printed

**count:** The number of characters to output

**c:** The character to print

## cprintf

### Description:

Prints formatted output.

Escape strings can be used to modify attributes, move cursor, etc similar to PRINT in CBM BASIC.

**Syntax:** `unsigned char cprintf (const unsigned char* format, ...)`

### Parameters:

**format:** The string to output. The available escape codes are:

#### Cursor positioning

\t Go to next tab position (multiple of 8s)

\r Carriage Return

\n New line

{clr} Clear screen      {home} Move cursor to home (top-left)

{d} Move cursor down      {u} Move cursor up

{r} Move cursor right      {l} Move cursor left

**Attributes**

|         |                        |          |                         |
|---------|------------------------|----------|-------------------------|
| {rvson} | Reverse attribute ON   | {rvsoff} | Reverse attribute OFF   |
| {blon}  | Blink attribute ON     | {bloff}  | Blink attribute OFF     |
| {ulon}  | Underline attribute ON | {uloff}  | Underline attribute OFF |

**Colors (default palette)**

|         |        |        |         |
|---------|--------|--------|---------|
| {blk}   | {wht}  | {red}  | {cyan}  |
| {pur}   | {grn}  | {blu}  | {yel}   |
| {ora}   | {brn}  | {pink} | {gray1} |
| {gray2} | {lblu} | {lgrn} | {gray3} |

Currently no argument replacement is done with the variable arguments.

**cgetc****Description:**

Waits until a character is in the keyboard buffer and returns it

**Syntax:**    `unsigned char cgetc (void);`

**Return Value:**

The last character in the keyboard buffer

**Notes:**    Returned values are ASCII character codes

**kbhit****Description:**

Returns the character in the keyboard buffer

**Syntax:**    `unsigned char kbhit (void);`

**Return Value:**

The character code in the keyboard buffer, 0 otherwise.

**Notes:**    Returned values are ASCII character codes

**getkeymodstate****Description:**

Return the key modifiers state.

**Syntax:**    `unsigned char getkeymodstate(void)`

**Return Value:**

A byte with the key modifier state bits, where bits:

| <b>Bit</b> | <b>Meaning</b>    | <b>Constant</b> |
|------------|-------------------|-----------------|
| 0          | Right SHIFT State | KEYMOD_RSHIFT   |
| 1          | Left SHIFT state  | KEYMOD_LSHIFT   |
| 2          | CTRL state        | KEYMOD_CTRL     |
| 3          | MEGA state        | KEYMOD_MEGA     |
| 4          | ALT state         | KEYMOD_ALT      |
| 5          | NOSCRL state      | KEYMOD_NOSCRL   |
| 6          | CAPSLOCK state    | KEYMOD_CAPSLOCK |
| 7          | Reserved          | -               |

## **flushkeybuf**

### **Description:**

Flush the keyboard buffer

**Syntax:** `void flushkeybuf(void)`

## **cinput**

### **Description:**

Get input from keyboard, printing incoming characters at current position.

**Syntax:** `unsigned char cinput(char* buffer, unsigned char buflen, unsigned char flags)`

### **Parameters:**

**buffer:** Target character buffer preallocated by caller

**buflen:** Target buffer length in characters, including the null character terminator

**flags:** Flags for input: (default is accept all printable characters)

`CINPUT_ACCEPT_NUMERIC`

Accepts numeric characters.

`CINPUT_ACCEPT_LETTER`

Accepts letters.

`CINPUT_ACCEPT_SYM`

Accepts symbols.

`CINPUT_ACCEPT_ALL`

Accepts all. Equals to `CINPUT_ACCEPT_NUMERIC | CINPUT_ACCEPT_LETTER | CINPUT_ACCEPT_SYM`

### CINPUT\_ACCEPT\_ALPHA

Accepts alphanumeric characters. Equals to CINPUT\_ACCEPT\_NUMERIC  
|CINPUT\_ACCEPT LETTER

### CINPUT\_NO\_AUTOTRANSLATE

Disables the feature that makes cinput to autodisplay uppercase characters when standard lowercase character set is selected and the user enters letters without the SHIFT key, that would display graphic characters instead of alphabetic ones.

#### **Return Value:**

Count of successfully read characters in buffer

## **VIC\_BASE**

*VIC\_BASE* is a pre-processor macro that provides the base address of the VIC-IV chip, i.e., \$D000.

*IS\_H640* is a pre-processor macro that returns 0 if the current VIC-III/IV video mode is set to 320 pixels accross (40 column mode), and non-zero if it is set to 640 pixels across (80 column mode).

# 16

## CHAPTER

## BASIC Tokenisers



Various tokenisers for C64 BASIC exist, e.g., <https://github.com/catseye/hatoucan>, <https://www.c64-wiki.com/wiki/C64list>, or the **petcat** utility that is part of VICE. If you are using Ubuntu Linux, you can install **petcat** by using the following command:

```
sudo apt-get install vice
```

We recommend **petcat**, because it supports both C64 BASIC 2 and C65 BASIC 10.



# PART VI

## APPENDICES



# **APPENDICES**



# APPENDIX A

## ACCESSORIES



# B

## APPENDIX

### **BASIC 65 Command Reference**

- **Format of Commands, Functions and Operators**
- **Commands, Functions and Operators**
- **BASIC 65 constants**
- **BASIC 65 variables**
- **BASIC 65 arrays**
- **BASIC command reference**



# FORMAT OF COMMANDS, FUNCTIONS AND OPERATORS

This appendix describes each of the commands, functions and other callable elements of BASIC 65, an enhanced version of BASIC 65. Some of these can take one or more arguments, that is, pieces of input that you provide as part of the command or function call. Some also require that you use special keywords. Here is an example of how commands, functions and operators will be described in this appendix:

## **KEY <numeric expression>,<string expression>**

In this case, KEY is what we call a **keyword**. That just means a special word that BASIC understands. Keywords are always written in CAPITALS, so that you can easily recognise them.

The < and > signs mean that whatever is between them must be there for the command, function or operator to work. In this case, it tells us that we need to have a **numeric expression** in one place, and a **string expression** in another place. We'll explain what there are a bit more in a few moments.

You might also see square brackets around something, for example, [**numeric expression**]. This means that whatever appears between the square brackets is optional, that is, you can include it if you need to, but that the command, function or operator will work just fine without it. For example, the CIRCLE command has an optional numeric argument to indicate if the circle should be filled when being drawn.

The comma, and some other symbols and punctuation marks just represent themselves. In this case, it means that there must be a comma between the **numeric expression** and the **string expression**. This is what we call syntax: If you miss something out, or put the wrong thing in the wrong place, it is called a syntax error, and the computer will tell you if you have a syntax error by giving a ?SYNTAX ERROR message.

There is nothing to worry about getting an error from the computer. Instead, it is just the computer's way of telling you that something isn't quite right, so that you can more easily find and fix the problem. Error messages like this can't hurt the computer or damage your program, so there is nothing to worry about. For example, if we accidentally left the comma out, or replaced it with a full-stop, the computer will respond with a syntax error, like this:

```
KEY 8"FISH"
```

```
?SYNTAX ERROR
```

```
KEY 8."FISH"
```

```
?SYNTAX ERROR
```

It is very common for commands, functions and operators to use one or more “**expression**”. An expression is just a fancy name for something that has a value. This could be a string, such as “HELLO”, or a number, like 23.7, or it could be a calculation, that might include one or more functions or operators, such as `LEN("HELLO") * (3 XOR 7)`. Generally speaking, expressions can result in either a string or numeric result. In this case we call the expressions either string expressions or numeric expressions. For example, “HELLO” is a **string expression**, while 23.7 is a **numeric expression**.

It is important to use the correct type of expression when writing your programs. If you accidentally use the wrong type, the computer will give you a `?TYPE MISMATCH ERROR`, to say that the type of expression you gave doesn’t match what it expected, that is, there is a mismatch between the type of expression it expected, and the one you gave. For example, we will get a `?TYPE MISMATCH ERROR` if we type the following command, because “POTATO” is a string expression instead of a numeric expression:

```
KEY "POTATO","SOUP"
```

You can try typing this into the computer yourself now, if you like.

# COMMANDS, FUNCTIONS AND OPERATORS

Commands are statements that you can use directly from the **READY.** prompt, or from within a program, for example:

```
PRINT "HELLO"  
HELLO
```

```
10 PRINT "HELLO"  
RUN  
HELLO
```

# BASIC 65 CONSTANTS

| type                   | example | example   |
|------------------------|---------|-----------|
| decimal integer        | 32000   | -55       |
| decimal fixed point    | 3.14    | -7654.321 |
| decimal floating point | 1.5E03  | 7.7E-02   |
| hex                    | \$D020  | \$FF      |
| string                 | "X"     | "TEXT"    |

# BASIC 65 VARIABLES

Each scalar variable consumes 8 bytes of storage in memory. The reserved area in bank 0 from \$F700 → \$FEFF can store 256 variables. Variables need not to be declared, the type is determined by the appendix character. All variables with no appendix are regarded as REAL per default. The storage is claimed at their first usage and they are initialised to zero, string variables are initialised as NULL string "".

| type    | appendix | range             | example       |
|---------|----------|-------------------|---------------|
| byte    | &        | 0 .. 255          | BY& = 23      |
| integer | %        | -32768 .. 32767   | I% = 5        |
| real    | none     | -1E37 .. 1E37     | XY = 1/3      |
| string  | \$       | length = 0 .. 255 | AB\$ = "TEXT" |

# BASIC 65 ARRAYS

Each array consumes the number of elements times the item size plus the size of the header ( $6 + 2 * \text{dimensions}$ ) in memory. Arrays are stored in bank 1 starting at address \$2000 and expands upwards. They share the available memory (\$2000 .. \$F6FF) with the string area, which starts in bank 1 on address \$F6FF and expands downwards. Each of the above scalar variable types can be used as an array by declaring them with a DIM statement. The arrays are initialised to zero for all elements on declaration. If an undeclared array element is used, an automatic implicit declaration is done, which sets the upper boundary for each dimension to 10. For example the usage of an undeclared element AB(3,5) would automatically perform a "DIM AB(10,10)". The lower boundary for each dimension is always 0 (zero). So an array DIM AB(10) consists of 11 elements and accepts indexes from 0 to 10.

String arrays are, more precisely expressed, arrays of string descriptors. Each item consists of three bytes, which hold the values: length of the string and the address (low/high byte) of the assigned string in string memory. The usage of the BASIC function POINTER with a

string or string array element as argument, returns the address of the descriptor, not the string.

| type & item size | appendix | range | example          |
|------------------|----------|-------|------------------|
| byte array       | 1        | &     | BY&(5,6) = 23    |
| integer array    | 2        | %     | I%(0,10) = 5     |
| real array       | 5        | none  | XY(I%) = 1/3     |
| string array     | 3        | \$    | AB\$(X) = "TEXT" |

# BASIC COMMAND REFERENCE

# **ABS**

**Token:** \$B6

**Format:** **ABS(x)**

**Usage:** The numeric function **ABS(x)** returns the absolute value of the numeric argument **x**.

**x** = numeric argument (integer or real expression).

**Remarks:** The result is of real type.

**Example:** Using **ABS**

```
PRINT ABS(-123)
123
PRINT ABS(4.5)
4.5
PRINT ABS(-4.5)
4.5
```

# AND

**Token:** \$AF

**Format:** operand **AND** operand

**Usage:** The Boolean **AND** operator performs a bit-wise logical AND operation on two 16-bit values. Integer operands are used as they are. Real operands are converted to a signed 16 bit integer. Logical operands are converted to 16 bit integer using \$FFFF, decimal -1 for TRUE and \$0000, decimal 0, for FALSE.

```
0 AND 0  ->  0
0 AND 1  ->  0
1 AND 0  ->  0
1 AND 1  ->  1
```

**Remarks:** The result is of integer type. If the result is used in a logical context, the value of 0 is regarded as FALSE, all other, nonzero values are regarded as TRUE.

**Example:** Using **AND**

```
PRINT 1 AND 3
1
PRINT 128 AND 64
0
```

In most cases **AND** is used in **IF** statements.

```
IF (C) = 0 AND C < 256 THEN PRINT "BYTE VALUE"
```

# APPEND

**Token:** \$FE \$0E

**Format:** APPEND# **Ifn**, **filename** [**D drive**] [**U unit**]

**Usage:** Opens an existing sequential file of type SEQ or USR for writing and positions the write pointer at the end of the file.

**Ifn** = logical file number

1 <= Ifn <= 127: line terminator is CR

128 <= Ifn <= 255: line terminator is CR LF

**filename** is either a quoted string, e.g. "data" or a string expression in parentheses, e.g. (FI\$)

**drive** = drive # in dual drive disk units.

The drive # defaults to 0 and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to 8.

**Remarks:** APPEND# functions similar to the OPEN# command, except that if the file already exists, the existing content of the file will be retained, and any PRINT# commands made to the open file will cause the file to grow longer.

**Example:** Open file in append mode:

```
APPEND#5,"DATA",U9  
APPEND#130,(DD$),U(UN%)  
APPEND#3,"USER FILE,U"  
APPEND#2,"DATA BASE"
```

# ASC

**Token:** \$C6

**Format:** **ASC(string)**

**Usage:** Takes the first character of the string argument and returns its numeric code value. The name is apparently chosen to be a mnemonic to ASCII, but the returned value is in fact the so called PETSCII code.

**Remarks:** **ASC** returns a zero for an empty string, which behaviour is different to BASIC 2, where ASC("") gave an error. The inverse function to **ASC** is **CHR\$**.

**Example:** Using **ASC**

```
PRINT ASC("MEGA")
77
PRINT ASC("")
0
```

# ATN

**Token:** \$C1

**Format:** **ATN(numeric expression)**

**Usage:** Returns the arc tangent of the argument. The result is in the range  $(-\pi/2$  to  $\pi/2)$

**Remarks:** A multiplication of the result with  $180/\pi$  converts the value to the unit "degrees". **ATN** is the inverse function to **TAN**.

**Example:** Using **ATN**

```
PRINT ATN(0.5)
.463647609
PRINT ATN(0.5) * 180 / pi
26.5650512
```

# AUTO

**Token:** \$DC

**Format:** **AUTO [step]**

**Usage:** Enables faster typing of BASIC programs. After submitting a new program line to the BASIC editor with the RETURN key, the AUTO function generates a new BASIC line number for the entry of the next line. The new number is computed by adding **step** to the current line number.

**step** = line number increment

Typing **AUTO** with no argument switches this function off.

**Example:** Using **AUTO**

```
AUTO 10 : USE AUTO WITH INCREMENT 10
AUTO    : SWITCH AUTO OFF
```

# BACKGROUND

**Token:** \$FE \$3B

**Format:** BACKGROUND colour

**Usage:** Sets the background colour of the screen to the argument, which must be in the range 0 to 15. (See colour table).

**Colours:** Index and RGB values of colour palette

| index | red | green | blue | colour      |
|-------|-----|-------|------|-------------|
| 0     | 0   | 0     | 0    | black       |
| 1     | 15  | 15    | 15   | white       |
| 2     | 15  | 0     | 0    | red         |
| 3     | 0   | 15    | 15   | cyan        |
| 4     | 15  | 0     | 15   | magenta     |
| 5     | 0   | 15    | 0    | green       |
| 6     | 0   | 0     | 15   | blue        |
| 7     | 15  | 15    | 0    | yellow      |
| 8     | 15  | 6     | 0    | orange      |
| 9     | 10  | 4     | 0    | brown       |
| 10    | 15  | 7     | 7    | pink        |
| 11    | 5   | 5     | 5    | dark grey   |
| 12    | 8   | 8     | 8    | medium grey |
| 13    | 9   | 15    | 9    | light green |
| 14    | 9   | 9     | 15   | light blue  |
| 15    | 11  | 11    | 11   | light grey  |

**Example:** Using BACKGROUND

```
BACKGROUND 3 : REM SELECT BACKGROUND COLOUR CYAN
```

# BACKUP

**Token:** \$F6

**Format:** **BACKUP U source TO U target**  
**BACKUP D source TO D target [,U unit]**

**Usage:** The first form of the **BACKUP** command, specifying units for source and target can only be used for the drives connected to the internal FDC (Floppy Disk Controller). Units 8 and 9 are reserved for this controller. These can be either the internal floppy drive (unit 8) and another floppy drive (unit 9), attached to the same ribbon cable or mounted D81 disk images. So this command can be used to copy from floppy to floppy, floppy to image, image to floppy and image to image, dependent on image mounts and existence of a second physical floppy drive.

The second form of the **BACKUP** command, specifying drives for source and target is meant to be used for dual drives units, connected to the IEC bus. For example: CBM 4040, 8050, 8250 via IEEE-488 to IEC adapter. Then the backup is done by the disk unit internally.

**source** = unit or drive # of source disk.

**target** = unit or drive # of target disk.

**Remarks:** The target disk will be formatted and an identical copy of the source disk is written.

The **BACKUP** command cannot be used to backup from internal devices to IEC devices or vice versa.

**Example:** Using **BACKUP**

```
BACKUP U8 TO U9      : REM BACKUP INTERNAL DRIVE 8 TO DRIVE 9  
BACKUP U9 TO U8      : REM BACKUP DRIVE 9 TO INTERNAL DRIVE 8  
BACKUP D0 TO D1, U10 : REM BACKUP ON DUAL DRIVE CONNECTED VIA IEC
```

# BANK

**Token:** \$FE \$02

**Format:** **BANK bank-number**

**Usage:** Selects the memory configuration for BASIC commands, that use 16-bit addresses. These are LOAD, SAVE, PEEK, POKE, WAIT and SYS. See system memory map in Chapter/Appendix F on page F-3 for details.

**Remarks:** A value > 127 selects memory mapped I/O. The default value for the bank number is 128. This configuration has RAM from \$0000 to \$1FFF and BASIC ROM's, KERNAL ROM's and I/O from \$2000 to \$FFFF.

**Example:** Using **BANK**

```
BANK 1 :REM SELECT MEMORY CONFIGURATION 1
```

# BEGIN

**Token:** \$FE \$18

**Format:** **BEGIN ... BEND**

**Usage:** The **BEGIN** and **BEND** keywords act like a pair of brackets around a compound statement to be executed after a **THEN** or **ELSE** keyword. This overcomes the single line limitation of the standard **IF ... THEN ... ELSE** clause.

**Remarks:** Do not jump with **GOTO** or **GOSUB** into a compound statement. It may lead to unexpected results.

**Example:** Using **BEGIN** and **BEND**

```
10 GET A$  
20 IF A$>"A" AND A$<"Z" THEN BEGIN  
30 PW$=PW$+A$  
40 IF LEN(PW$)>7 THEN 90  
50 BEND :REM IGNORE ALL EXCEPT (A-Z)  
60 IF A$>CHR$(13) GOTO 10  
90 PRINT "PW=",PW$
```

# BEND

**Token:** \$FE \$19

**Format:** BEGIN ... BEND

**Usage:** The **BEGIN** and **BEND** keywords act like a pair of brackets around a compound statement to be executed after a **THEN** or **ELSE** keyword. This overcomes the single line limitation of the standard **IF ... THEN ... ELSE** clause.

**Remarks:** The example below shows a quirk in the implementation of the compound statement. If the condition evaluates to **FALSE**, execution does not resume right after **BEND** as it should, but at the beginning of next line. Test this behaviour with the following program:

**Example:** Using **BEGIN** and **BEND**

```
10 IF Z > 1 THEN BEGIN:A$="ONE"  
20 B$="TWO"  
30 PRINT A$;" ";B$;:BEND:PRINT " QUIRK"  
40 REM EXECUTION RESUMES HERE FOR Z <= 1
```

# BLOAD

**Token:** \$FE \$11

**Format:** **BLOAD filename [,B bank] [,P address] [,R] [,D drive] [,U unit]**

**Usage:** "Binary LOAD" loads a file of type PRG into RAM at address P.

The **BLOAD** command has two modes: The flat memory address mode can be used to load a program to any address in the 28 bit (256 MB) address range where RAM is installed. This includes the standard RAM banks 0 to 5, but also the 8MB so called "attic RAM" at address \$8000000.

This mode is triggered by specifying an address at the P parameter, that is larger than \$FFFF. The bank parameter is ignored in this mode.

For compatibility reasons with BASIC-10 and BASIC-7, the **BLOAD** accepts the syntax with a 16bit address at P and a bank number at B too. The attic RAM is out of range for this compatibility mode.

The optional parameter **R** (RAW MODE) does not interpret or use the first two bytes of the program file as load address, which is otherwise the default behaviour. In RAW MODE every byte is read as data.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

**bank** specifies the RAM bank to be used. If not specified the current bank, as set with the last **BANK** statement, will be used.

**address** can be used to overrule the load address, that is stored in the first two bytes of the PRG file.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The **BLOAD** cannot cross bank boundaries.

**BLOAD** uses the load address from the file, if no P parameter is given.

**Example:** Using **BLOAD**

```
BLOAD "ML DATA", B0, U9
BLOAD "SPRITES"
BLOAD "ML ROUTINES", B1, P32768
BLOAD (FI$), B(BA%), P(PA), U(UN%)
BLOAD "CHUNK",P($8000000)      :REM LOAD TO ATTIC RAM
```

# BOOT

- Token:** \$FE \$1B
- Format:** **BOOT filename [,B bank] [,P address] [,D drive] [,U unit]**  
**BOOT SYS**  
**BOOT**
- Usage:** **BOOT filename** loads a file of type PRG into RAM at address P and bank B and starts executing the code at the load address.  
**BOOT SYS** loads the boot sector from sector 0, track 1 and unit 8 to address \$0400 on bank 0 and performs a JSR \$0400 afterwards (Jump To Subroutine).  
The **BOOT** command with no parameter tries to load and execute a file named AUTOBOOT.C65 from the default unit 8. It's short for **RUN "AUTO-BOOT.C65"**.  
**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. **(FI\$)**  
**bank** specifies the RAM bank to be used. If not specified the current bank, as set with the last **BANK** statement, will be used.  
**address** can be used to overrule the load address, that is stored in the first two bytes of the PRG file.  
**drive** = drive # in dual drive disk units.  
The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.  
**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.  
**Remarks:** **BOOT SYS** copies the contents of one physical sector (two logical sectors) = 512 bytes from disc to RAM, filling RAM from \$0400 to \$05ff.
- Example:** Using **BOOT**

```
BOOT SYS  
BOOT (FI$), B(B%), P(P%), U(UN%)  
BOOT
```

# BORDER

**Token:** \$FE \$3C

**Format:** BORDER colour

**Usage:** Sets the border colour of the screen to the argument, which must be in the range 0 to 15. (See colour table).

**Colours:** Index and RGB values of colour palette

| index | red | green | blue | colour      |
|-------|-----|-------|------|-------------|
| 0     | 0   | 0     | 0    | black       |
| 1     | 15  | 15    | 15   | white       |
| 2     | 15  | 0     | 0    | red         |
| 3     | 0   | 15    | 15   | cyan        |
| 4     | 15  | 0     | 15   | magenta     |
| 5     | 0   | 15    | 0    | green       |
| 6     | 0   | 0     | 15   | blue        |
| 7     | 15  | 15    | 0    | yellow      |
| 8     | 15  | 6     | 0    | orange      |
| 9     | 10  | 4     | 0    | brown       |
| 10    | 15  | 7     | 7    | pink        |
| 11    | 5   | 5     | 5    | dark grey   |
| 12    | 8   | 8     | 8    | medium grey |
| 13    | 9   | 15    | 9    | light green |
| 14    | 9   | 9     | 15   | light blue  |
| 15    | 11  | 11    | 11   | light grey  |

**Example:** Using BORDER

```
10 BORDER 4 : REM SELECT BORDER COLOUR MAGENTA
```

# BOX

**Token:** \$E1

**Format:** **BOX X0,Y0, X1,Y1, X2,Y2, X3,Y3, SOLID**

**Usage:** Draws a quadrangle by connecting the coordinate pairs 0 -> 1 -> 2 -> 3 -> 0. The quadrangle is drawn using the current drawing context set with SCREEN, PALETTE and PEN. The quadrangle is filled, if the parameter SOLID is 1.

**Remarks:** A quadrangle is a geometric figure with four sides and four angles. A box is a special form of a quadrangle, with all four angles at 90 degrees. Rhomboids, kites and parallelograms are special forms too. So the name of this command is misleading, because it can be used to draw all kind of quadrangles, not only boxes.

It is possible to draw bow-tie shapes.

**Example:** Using **BOX**

```
BOX 0,0, 160,0, 160,80, 0,80
```



```
BOX 0,0, 160,80, 160,0, 0,80
```



```
BOX 0,0, 160,0, 140,80, 20,80
```



# BSAVE

**Token:** \$FE \$10

**Format:** **BSAVE filename ,P start TO end [,B bank] [,D drive] [,U unit]**

**Usage:** "Binary SAVE" saves a memory range to a file of type PRG.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. **(FI\$)** If the first character of the filename is an at-sign '@' it is interpreted as a "save and replace" operation. It is dangerous to use this replace option on drives 1541 and 1571, because they contain the notorious "save and replace bug" in their DOS.

**bank** specifies the RAM bank to be used. If not specified the current bank, as set with the last **BANK** statement, will be used.

**start** is the first address, where the saving begins. It becomes also the load address, that is stored in the first two bytes of the PRG file.

**end** is the address, where the saving stops. **end-1** is the last address to be used for saving.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The length of the file is **end - start + 2**.

**Example:** Using **BSAVE**

```
BSAVE "ML DATA", P 32768 TO 33792, B0, U9  
BSAVE "SPRITES", P 1536 TO 2058  
BSAVE "ML ROUTINES", B1, P(DEC("9000")) TO (DEC("A000"))  
BSAVE (FI$), B(BA%), P(PA) TO (PE), U(UN%)
```

# BUMP

**Token:** \$CE \$03

**Format:** **b = BUMP(type)**

**Usage:** Used to detect sprite-sprite (type=1) or sprite-data (type=2) collisions. the return value **b** is a 8-bit mask with one bit per sprite. The bit position corresponds with the sprite number. Each bit set in the return value indicates, that the sprite for this position was involved in a collision since the last call of **BUMP**. Calling **BUMP** resets the collision mask, so you get always a summary of collisions encountered since the last call of **BUMP**.

**Remarks:** It's possible to detect multiple collisions, but you need to evaluate sprite coordinates then to detect which sprite collided with which one.

**Example:** Using **BUMP**

```
10 $% = BUMP(1) : REM SPRITE-SPRITE COLLISION
20 IF ($% AND 6) = 6 THEN PRINT "SPRITE 1 & 2 COLLISION"
30 REM ---
40 $% = BUMP(2) : REM SPRITE-DATA COLLISION
50 IF ($% <> 0) THEN PRINT "SOME SPRITE HIT DATA REGION"
```

| sprite | return | mask      |
|--------|--------|-----------|
| 0      | 1      | 0000 0001 |
| 1      | 2      | 0000 0010 |
| 2      | 4      | 0000 0100 |
| 3      | 8      | 0000 1000 |
| 4      | 16     | 0001 0000 |
| 5      | 32     | 0010 0000 |
| 6      | 64     | 0100 0000 |
| 7      | 128    | 1000 0000 |

# BVERIFY

**Token:** \$FE \$28

**Format:** **BVERIFY filename [,P address] [,B bank] [,D drive] [,U unit]**

**Usage:** "Binary VERIFY" compares a memory range to a file of type PRG.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. **(F1\$)**

**bank** specifies the RAM bank to be used. If not specified the current bank, as set with the last **BANK** statement, will be used.

**address** is the address, where the comparison begins. If the parameter P is omitted, it is the load address, that is stored in the first two bytes of the PRG file.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** **BVERIFY** can only test for equality. It gives no information about the number or position of different valued bytes. In direct mode the command exits either with the message **OK** or with **VERIFY ERROR**. In program mode a **VERIFY ERROR** either stops execution or enters the **TRAP** error handler, if active.

**Example:** Using **BVERIFY**

```
BVERIFY "ML DATA", P 32768, B0, U9  
BVERIFY "SPRITES", P 1536  
BVERIFY "ML ROUTINES", B1, P(DEC("9000"))  
BVERIFY (F1$), B(BA%), P(PA), U(UN%)
```

# CATALOG

**Token:** \$FE \$0C

**Format:** CATALOG [filepattern] [,W] [,R] [,D drive] [,U unit]

**Format:** \$ [filepattern] [,W] [,R] [,D drive] [,U unit]

**Usage:** Prints a file catalog/directory of the specified disk.

The shortcut symbol **\$** can be used in direct mode only.

The **W** (Wide) parameter lists the directory three columns wide on the screen and pauses after a the screen is full (63 directory entries). Pressing any key displays the next page.

The **R** (Recoverable) parameter includes files in the directory, which are flagged as deleted but are still recoverable.

**filepattern** is either a quoted string, for example: "**da\***" or a string expression in parentheses, e.g. **(DI\$)**

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The command **CATALOG** is a synonym for **DIRECTORY** or **DIR** and produces the same listing. The **filepattern** can be used to filter the listing. The wild-card characters \* and ? may be used. Adding a ,T= to the pattern string, with **T** specifying a filetype of **P**, **S**, **U** or **R** (for **PRG**, **SEQ**, **USR**, **REL**) filters the output to that filetype.

**Example:** Using **CATALOG**

```
CATALOG
0 "BLACK SMURF" BS 2A
508 "STORY PHOBOS" SEQ
27 "C8096" PRG
25 "C128" PRG
104 BLOCKS FREE.
```

CATALOG "\*,T-S"  
 0 "BLACK SMURF" BS 2A  
 508 "STORY PHOBOS" SEQ  
 104 BLOCKS FREE.

| MEGA65 - Xemu [100% 9%] running 3.5MHz |   |                   |   |                       |   |   |
|----------------------------------------|---|-------------------|---|-----------------------|---|---|
|                                        |   |                   |   |                       |   |   |
| 0 "MEGA65 DEUKIT"                      | P | 128 "PULP.IFF"    | P | 23 "RASTER65 2020"    | P | P |
| 1 "IFF"                                | P | 85 "TROOPER.IFF"  | P | 1 "INDEX"             | P | P |
| 1 "RSW-TEST"                           | P | 72 "RIPLEY.IFF"   | P | 77 "BMPVIEW"          | P | P |
| 1 "SPRITES"                            | P | 0 "-----"         | D | 2 "BENCH"             | P | P |
| 1 "SPRITE<3"                           | P | 50 "C65NOTEPAID"  | P | 48 "C64.5 DEMO"       | P | P |
| 2 "ELLIPSE"                            | P | 3 "README"        | P | 0 "___ GAMES ___"     | D | P |
| 1 "SPRITE<4"                           | P | 33 "NPAGES"       | P | 1 "MATH"              | P | P |
| 1 "PARROT"                             | P | 8 "BPAGE1"        | P | 2 "CIRCLE"            | P | P |
| 1 "SPRITE<5"                           | P | 8 "BPAGE2"        | P | 68 "C64ANABALT"       | P | P |
| 1 "SPRITE<6"                           | P | 8 "BPAGE3"        | P | 15 "SPIKES"           | P | P |
| 1 "SPRITE<7"                           | P | 8 "BPAGE4"        | P | 1 "CIRCLEBUG"         | P | P |
| 3 "DEMOsprites1"                       | P | 8 "BPAGE5"        | S | 1 "BEN"               | P | P |
| 1 "DEMOsprites2"                       | P | 8 "BTXTIIPETSCII" | S | 1 "FOR"               | P | P |
| 27 "DEMOGRAPH"                         | P | 8 "BTXTIUTF8"     | S | 0 "___ DEMOS ___"     | D | P |
| 129 "REGINA.0"                         | P | 0 "               | D | 38 "DRAGON II/ELC"    | P | P |
| 129 "REGINA.1"                         | P | 0 " C64 MODE "    | D | 53 "TIME 4 DOLE /MEC" | P | P |
| 4 "REGINA.P"                           | P | 0 "               | D | 48 "-GREENWARS/TLS!-" | P | P |
| 2 "IFF SLIDE SHOW"                     | P | 61 "MENU"         | P | 81 "ELECTRIC CAFE"    | P | P |
| 182 "ALIEN.IFF"                        | P | 3 "M.MEGA65"      | P | 26 "D.Y.C.P./JEWELS"  | P | P |
| 70 "BEAKER.IFF"                        | P | 12 "MEGA65INFO"   | P | 1 "SPRITE<1"          | P | P |
| 42 "JOKER.IFF"                         | P | 198 "EE"          | P | 1 "SPRITE<2"          | P | P |
| 74 "PICARD.IFF"                        | P | 1 "SPRITE<0"      | P |                       |   |   |
| 560 BLOCKS FREE                        |   |                   |   |                       |   |   |
| READY.                                 |   |                   |   |                       |   |   |

# CHANGE

**Token:** \$FE \$2C

**Format:** **CHANGE "find" TO "replace" [,from-to]**

**Usage:** Used in direct mode only. It searches the line range if specified or the whole BASIC program else. At each occurrence of the "find string" the line is listed and the user prompted for an action:

'Y' <RETURN> do the change and find next string

'N' <RETURN> do **not** change and find next string

'\*' <RETURN> change this and all following matches

<RETURN> exit command, don't change.

**Remarks:** Instead of the quote ("") each other character may be used as delimiter for the findstring and replacestring. Using the quote as delimiter finds text strings, that are not tokenised and therefore not part of a keyword.

**CHANGE "LOOP" TO "OOPS"** will not find the BASIC keyword **LOOP**, because the keyword is stored as token and not as text. However **CHANGE &LOOP& TO &OOPS&** will find and replace it (probably spoiling the program).

**Example:** Using **CHANGE**

```
CHANGE "XX$" TO "UU$", 2000-2700
CHANGE &IN& TO &OUT&
```

# CHAR

**Token:** \$E0

**Format:** **CHAR column, row, height, width, direction, string [, address of character set]**

**Usage:** Displays text on a graphic screen. It can be used for all resolutions.

**column** is the start position of the output in horizontal direction. One column is 8 pixels wide, so a screen width of 320 has a column range 0 -> 39, while a width of 640 has a range of 0 -> 79.

**row** is the start position of the output in vertical direction. Other than column, its unit is pixel with top row having the value 0.

**height** is a factor applied to the vertical size of the characters. 1 is normal size (8 pixels) 2 is double size (16 pixels), and so on.

**width** is a factor applied to the horizontal size of the characters. 1 is normal size (8 pixels) 2 is double size (16 pixels), and so on.

**direction** controls the printing direction:

1: up

2: right

4: down

8: left

The optional **address of character set** can be used to select a character set different from the default character set at \$29800, which is the set with upper/lower characters.

**string** is a string constant or expression which will be printed. This string may optionally contain one or more of the following control characters:

|               |               |                |
|---------------|---------------|----------------|
| CHR\$(6)      | CTRL+F        | flip character |
| CHR\$(18)     | RVSON         | reverse        |
| CHR\$(146)    | RVSOFF        | reverse off    |
| CHR\$(21)     | CTRL+U        | underline      |
| CHR\$(25)+"-" | CTRL+Y + "--" | rotate left    |
| CHR\$(25)+"+" | CTRL+Y + "+"  | rotate right   |
| CHR\$(26)     | CTRL+Z        | mirror         |

**Remarks:** Regular text mode control characters, for example: cursor movement codes, will be ignored (neither printed nor interpreted).

**Example:** Using **CHAR**

```
CHAR 304,196, 1,1,2, "MEGA65"
```

will print the text "MEGA65" on the centre of a 640 x 400 graphic screen.

# **CHR\$**

**Token:** \$C1

**Format:** **CHR\$(numeric expression)**

**Usage:** Returns a string of length one character using the argument to insert the character having this value as PETSCII code.

**Remarks:** The argument range is 0 -> 255, so this function may also be used to insert control codes into strings. Even the NULL character, with code 0, is allowed.  
**CHR\$** is the inverse function to **ASC**.

**Example:** Using **CHR\$**

```
10 QUOTES = CHR$(34)
20 ESCAPE$ = CHR$(27)
30 PRINT QUOTES;"MEGAS5";QUOTES : REM PRINT "MEGAS5"
40 PRINT ESCAPE$;"Q";      : REM CLEAR TO END OF LINE
```

# CIRCLE

**Token:** \$E2

**Format:** **CIRCLE** **xcentre, ycentre, radius, [,solid]**

**Usage:** A special case of the **ELLIPSE** command using the same value for horizontal and vertical radius.

**xcentre** x coordinate of centre in pixels.

**ycentre** y coordinate of centre in pixels.

**radius** radius of the circle in pixels.

**solid** will fill the circle if not zero.

**Remarks:** The **CIRCLE** command is used to draw circles on screens with an aspect ratio 1:1 (for example: 320 x 200 or 640 x 400). On other resolutions (like: 640 x 200) the shape will degrade to an ellipse.



**Example:** Using **CIRCLE**

```
100 REM CIRCLE (AFTER F.BOWEN)
110 H=0:W=0 :REM 320 X 200
120 BORDER 0 :REM BLACK
130 GRAPHIC CLR :REM INIT
140 SCREEN DEF 1,W,H,4 :REM SCREEN SETUP
150 SCREEN OPEN 1 :REM OPEN
160 PALETTE 1,0,0,0,0 :REM BLACK
170 PALETTE 1,1,RND(.)*16,RND(.)*16,15 :REM RANDOM COLOURS
180 PALETTE 1,2,RND(.)*16,15,RND(.)*16
190 PALETTE 1,3,15,RND(.)*16,RND(.)*16
200 PALETTE 1,4,RND(.)*16,RND(.)*16,15
210 PALETTE 1,5,RND(.)*16,15,RND(.)*16
220 PALETTE 1,6,15,RND(.)*16,RND(.)*16
230 SCREEN SET 1,1 :REM DRAW & VIEW
240 SCNCLR 0 :REM CLEAR
250 FOR I=0 TO 32 :REM CIRCLE LOOP
260 PEN 0,RND(.)*6+1 :REM RANDOM PEN
270 R=RND(.)*36+1 :REM RADIUS
280 XC=R+RND(.)*320:IF(XC>319)THEN280 :REM X CENTRE
290 YC=R+RND(.)*200:IF(YC>199)THEN290 :REM Y CENTRE
300 XC=XC+WT*320:YC=YC+HT*200
310 CIRCLE XC,YC,R,. :REM DRAW
320 NEXT
330 GETKEY A$:REM WAIT FOR KEY
340 SCREENCLOSE 1:BORDER 6:PALETTE RESTORE
```

# CLOSE

**Token:** \$A0

**Format:** **CLOSE channel**

**Usage:** Closes an input or output channel, that was established before by an **OPEN** command.

**channel** is a value in the range 0 -> 255.

**Remarks:** Closing open files before the program stops is very important, especially for output files. This command flushes output buffers and updates directory information on disks. Failing to **CLOSE** can corrupt files and disks. BASIC does NOT automatically close channels or files when the program stops.

**Example:** Using **CLOSE**

```
10 OPEN 2,8,2,"TEST$,W"  
20 PRINTH2,"TESTSTRING"  
30 CLOSE 2 : REM OMITTING CLOSE GENERATES A SPLAT FILE
```

# CLR

**Token:** \$9C

**Format:** **CLR**

**CLR DS\$**

**CLR ERR\$**

**Usage:** **CLR** with no parameters Resets all pointers, that are used for management of BASIC variables, arrays and strings. The run-time stack pointers are reset and the table of open channels is reset. A **RUN** command performs **CLR** automatically.

**CLR DS\$** clears the currently buffered disk status. Any reference to **DS** or **DS\$** after clearing will read the status from the disk device.

**CLR ERR\$** clears the program error status. This command is typically used in a **TRAP** handler, which has resolved an error and calls **RESUME**.

**Remarks:** **CLR** should not be used inside loops or subroutines because it destroys the return address. After a **CLR** all variables are unknown and will be initialised at the next usage.

**Example:** Using **CLR**

```
10 A=5: P$="MEGA65"
20 CLR
30 PRINT A;P$

0
READY.
CLR DS$
PRINT DS$
00,OK,00,00
```

# CMD

**Token:** \$9D

**Format:** **CMD channel [,string]**

**Usage:** Redirects the standard output from screen to the channel. This enables to print listings and directories or other screen outputs. It is also possible to redirect this output to a disk file or a modem.

**channel** must be opened by the **OPEN** command.

The optional **string** is sent to the channel before the redirection begins and can be used, for example, for printer setup escape sequences.

**Remarks:** The **CMD** mode is stopped by a **PRINT# channel** or by closing the channel with **CLOSE channel**. It is recommended to use a **PRINT# channel** before closing, to make sure, that the output buffer is flushed.

**Example:** Using **CMD** to print a program listing:

```
OPEN 1,4    :REM OPEN CHANNEL #1 TO PRINTER AT UNIT 4
CMD 1
LIST
PRINT#1
CLOSE 1
```

# COLLECT

**Token:** \$F3

**Format:** **COLLECT [,D drive] [,U unit]**

**Usage:** Rebuilds the **BAM** (Block Availability Map) deleting splat files and marking unused blocks as free.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** While this command is useful for cleaning the disk from splat files (for example: write files, that weren't properly closed) it is dangerous for disks with boot blocks or random access files. These blocks are not associated with standard disk files and will therefore be marked as free too and may be overwritten by further disk write operations.

**Example:** Using **COLLECT**

```
COLLECT  
COLLECT U9  
COLLECT D0, U9
```

# COLLISION

**Token:** \$FE \$17

**Format:** **COLLISION type [,linenumber]**

**Usage:** Enables or disables an user programmed interrupt handler. A call without linenumber disables the handler, while a call with linenumber enables it. After the execution of **COLLISION** with linenumber a sprite collision of the same type, as specified in the **COLLISION** call, interrupts the BASIC program and perform a **GOSUB** to **linenumber** which is expected to contain the user code for handling sprite collisions. This handler must give control back with a **RETURN**.

**type** specifies the collision type for this interrupt handler:

1 = sprite - sprite collision

2 = sprite - data - collision

3 = light pen

**linenumber** must point to a subroutine which holds code for handling sprite collision and ends with a **RETURN**.

**Remarks:** It is possible to enable interrupt handler for all types, but only one can execute at any time. A interrupt handler cannot be interrupted by another interrupt handler. Functions like **BUMP**, **RSPPOS** and **LPEN** may be used for evaluation of the sprites which are involved and their positions.

**Example:** Using **COLLISION**

```
10 COLLISION 1,70 : REM ENABLE
20 SPRITE 1,1 : MOVSPPR 1,120, 0 : MOVSPPR 1,0#5
30 SPRITE 2,1 : MOVSPPR 2,120,100 : MOVSPPR 2,180#5
40 FOR I=1 TO 50000:NEXT
50 COLLISION 1 : REM DISABLE
50 END
70 REM SPRITE <-> SPRITE INTERRUPT HANDLER
80 PRINT "BUMP RETURNS";BUMP(1)
90 RETURN: REM RETURN FROM INTERRUPT
```

# COLOR

**Token:** \$E7

**Format:** COLOR <ON|OFF>

**Usage:** Enables or disables handling of the character attributes on the screen. If **COLOR** is **ON**, the screen routines take care for both character RAM and attribute RAM. E.g. if the screen is scrolled for text, the attributes are scrolled too, so each character keeps his attribute or colour. If **COLOR** is **OFF**, the attribute or colour RAM is fixed and character movement is only done for screen characters. This speeds up screen handling, if moving characters with different colours is not intended.

**Example:** COLOR ON - with colour/attribute handling

COLOR OFF - no colour/attribute handling

# CONCAT

**Token:** \$FE \$13

**Format:** **CONCAT appendfile [,D drive] TO targetfile [,D drive] [,U unit]**

**Usage:** The **CONCAT** (concatenation) appends the contents of **appendfile** to the **targetfile**. Afterwards **targetfile** contains the contents of both files, while **appendfile** remains unchanged.

**appendfile** is either a quoted string, for example: "**data**" or a string expression in parentheses, for example: (**FI\$**)

**targetfile** is either a quoted string, for example: "**safe**" or a string expression in parentheses, for example: (**FS\$**)

If the disk unit has dual drives, it is possible to apply the **CONCAT** command to files, which are stored on different disks. In this case, it is necessary to specify the drive# for both files in the command. This is necessary too, if both files are stored on drive#1.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The **CONCAT** commands is executed in the DOS of the disk drive. Both files must exist and no pattern matching is allowed. Only sequential files of type **SEQ** may be concatenated.

**Example:** Using **CONCAT**

```
CONCAT "NEW DATA" TO "ARCHIVE" ,U9  
CONCAT "ADDRESS",D8 TO "ADDRESS BOOK",D1
```

# CONT

**Token:** \$9A

**Format:** **CONT**

**Usage:** Used to resume program execution after a break or stop caused by an **END** or **STOP** statement or by pressing the **STOP KEY**. This is a useful debug tool. The BASIC program may be stopped and variables can be examined and even changed. The **CONT** statement then resumes execution.

**Remarks:** **CONT** cannot be used, if the program stops due to errors. Also any editing of the program inhibits continuation. Stopping and continuation can spoil the screen output or interfere with input/output operations.

**Example:** Using **CONT**

```
10 I=I+1:GOTO 10
RUN

BREAK IN 10
READY.
PRINT I
947
CONT
```

# COPY

**Token:** \$F4

**Format:** **COPY source [,D drive] [,U unit] TO target [,D drive] [,U unit]**

**Usage:** Copies the contents of **source** to the **target**. It is used to copy either single files or, by using wildcard characters, multiple files.

**source** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**).

**target** is either a quoted string, e.g. "**backup**" or a string expression in parentheses, e.g. (**FS\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

If none or one unit number is given or the unit numbers before and after the TO token are equal, the COPY command is executed inside the disk unit and the source and target files are on the same disk.

If the source unit (before TO) is different to the target unit (after TO), the COPY command is executed in the MEGA65 BASIC by reading the source files into a RAM buffer and writing to the target unit. In this case the target file name cannot be chosen, but will be the same as the destination filename. The extended unit to unit copy mode allows to copy single files, pattern matching files or all files of a disk. Any combination of units is allowed, internal floppy, SD-card images, IEC floppy drives like 1541, 1571, 1581 or CMD floppy and hard drives.

**Remarks:** The file types PRG, SEQ and USR can be copied. If source and target are on the same disk, the target filename must be different from the source file name.

The COPY command cannot copy **DEL** files, that are commonly used as title or separators in disk directories. These do not conform to Commodore DOS rules and cannot be accessed by standard OPEN routines.

**REL** files cannot be copied from unit to unit.

**Example:** Using **COPY**

```
COPY U8 TO U9      :REM COPY ALL FILES  
COPY "CODES" TO "BACKUP" :REM COPY SINGLE FILE  
COPY "*.TXT",U8 TO U9   :REM PATTERN COPY  
COPY "M*U",U9 TO U11    :REM PATTERN COPY
```

# COS

**Token:** \$BE

**Format:** **COS(numeric expression)**

**Usage:** The **COS** function returns the cosine of the argument. The argument is expected in units of **[radians]**. The result is in the range (-1.0 to +1.0)

**Remarks:** An argument in units of **[degrees]** can be converted to **[radians]** by multiplication with  $\pi/180$ .

**Example:** Using **COS**

```
PRINT COS(0.7)
```

```
.764842187
```

```
X=60:PRINT COS(X * pi / 180)
```

```
.500000001
```

# CURSOR

**Format:** **CURSOR [<ON/OFF>] [,column] [,row] [,style]**

**Usage:** Moves the text cursor to the specified position on the current text screen.

**ON** or **OFF** displays or hides the cursor.

**column** and **row** specify the new position.

**style** defines a solid (1) or flashing (0) cursor.

**Example:** Using **CURSOR**

```
10 CURSOR ON,1,2,1 :REM SET SOLID CURSOR AT COLUMN 1, ROW 2
```

# DATA

**Token:** \$83

**Format:** **DATA [list of constants]**

**Usage:** Used to define constants which can be read by **READ** statements somewhere in the program. Numbers and strings are allowed, but no expressions. Items are separated by commas. Strings containing commas, colons or spaces must be put in quotes.

A **RUN** command initialises the data pointer to the first item of the first **DATA** statement and advances it for every read item. It is in the responsibility of the programmer, that the type of the constant and the variable in the **READ** statement match. Empty items with no constant between commas are allowed and will be interpreted as zero for numeric variables and an empty string for string variables.

The **RESTORE** command may be used to set the data pointer to a specific line for subsequent readings.

**Remarks:** It is good programming style to put large amount of **DATA** statements at the end of the program. Otherwise they slow down the search for line numbers after **GOTO** and other statements with targets.

**Example:** Using **DATA**

```
1 REM DATA
10 READ N$, VE
20 READ NX : FOR I=2 TO NX : READ GL(I) : NEXT I
30 PRINT "PROGRAM:";N$;" VERSION:";VE
40 PRINT "N-POINT GAUSSLEGENDRE FACTORS E1":
50 FOR I=2 TO NX:PRINT I,GL(I):NEXT I
60 END
80 DATA "MEGAB5",1.1
90 DATA 5,0.5120,0.3573,0.2760,0.2252

RUN
PROGRAM:MEGAB5 VERSION: 1.1
N-POINT GAUSSLEGENDRE FACTORS E1
2 0.512
3 0.3573
4 0.276
5 0.2252
```

# DCLEAR

**Token:** \$FE \$15

**Format:** **DCLEAR [,D drive] [,U unit]**

**Usage:** Sends an initialise command to the specified unit and drive.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The DOS inside the disk unit will close all open files, clear all channels, free buffers and reread the BAM. This command should be used together with a **DCLOSE** to make sure, that the computer and the drive agree on the status, otherwise strange side effects may occur.

**Example:** Using **DCLEAR**

```
DCLOSE :DCLEAR  
DCLOSE U9:DCLEAR U9  
DCLOSE U9:DCLEAR D0, U9
```

# DCLOSE

**Token:** \$FE \$OF

**Format:** DCLOSE [#channel] [,U unit]

**Usage:** Closes a single file or all files for the specified unit.

**channel** = channel # assigned with the DOPEN statement.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

The **DCLOSE** command is used either with a channel argument or a unit number, but never both.

**Remarks:** It is important to close all open files before the program ends. Otherwise buffers will not be freed and even worse, open write files will be incomplete (splat files) and no more usable.

**Example:** Using **DCLOSE**

```
DCLOSE#2 :REM CLOSE FILE ASSIGNED TO CHANNEL 2  
DCLOSE U9:REM CLOSE ALL FILES OPEN ON UNIT 9
```

# **DEC**

**Token:** \$D1

**Format:** **DEC(string expression)**

**Usage:** Returns the decimal value of the argument, that is written as a hex string. The argument range is "0000" to "FFFF" or 0 to 65535 respectively. The argument must have 1-4 hex digits.

**Remarks:** Allowed digits in uppercase/graphics mode are:  
0123456789ABCDEF and in lowercase/uppercase mode:  
0123456789abcdef.

**Example:** Using **DEC**

```
PRINT DEC("D000")
53248
POKE DEC("600"),255
```

# DEF FN

**Token:** \$96

**Format:** **DEF FN name(real variable)**

**Usage:** Defines a single statement user function with one argument of real type returning a real value. The definition must be executed before the function can be used in expressions. The argument is a dummy variable, which will be replaced by the argument in the function usage.

**Remarks:** The value of the dummy variable will not be changed and the variable may be used in other context without side effects.

**Example:** Using **DEF FN**

```
10 PD =  $\pi$  / 180
20 DEF FN CD(X)= COS(X*PD): REM COS FOR DEGREES
30 DEF FN SD(X)= SIN(X*PD): REM SIN FOR DEGREES
40 FOR D=0 TO 360 STEP 90
50 PRINT USING "####";D;
60 PRINT USING " ##.##";FNCD(D);
70 PRINT USING " ##.##";FNSD(D)
80 NEXT D
RUN
0 1.00 0.00
90 0.00 1.00
180 -1.00 0.00
270 0.00 -1.00
360 1.00 0.00
```

# DELETE

**Token:** \$F7

**Format:** **DELETE [line range]**  
**DELETE filename [,D drive] [,U unit] [,R]**

**Usage:** Used either to delete a range of lines from the BASIC program or to delete a disk file.

**line range** consist of the first and the last line to delete or a single line number. If the first number is omitted, the first BASIC line is assumed. The second number in the range specifier defaults to the last BASIC line.

**filename** is either a quoted string, for example: "**safe**" or a string expression in parentheses, for example: (**FS\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**R** = Recover a previously deleted file. This will only work, if there were no write operations between deletion and recovery, which may have altered the contents of the file.

**Remarks:** The **DELETE filename** command works like the **SCRATCH filename** command.

**Example:** Using **DELETE**

```
DELETE 100      :REM DELETE LINE 100
DELETE 240-350  :REM DELETE ALL LINES FROM 240 TO 350
DELETE 500-      :REM DELETE FROM 500 TO END
DELETE -70      :REM DELETE FROM START TO 70

DELETE "DRM",U9 :REM DELETE FILE DRM ON UNIT 9
```

# DIM

**Token:** \$86

**Format:** **DIM name(limits) [,name(limits)]...**

**Usage:** Declares the shape, the bounds and the type of a BASIC array. As a declaration statement it must be executed only once and before any usage of the declared arrays. An array can have one or more dimensions. One dimensional arrays are often called vectors while two or more dimensions define a matrix. The lower bound of a dimension is always zero, while the upper bound is declared. The rules for variable names apply for array names too. There are integer arrays, real arrays and string arrays. It is legal to use the same identifier for scalar variables and array variables. The left parenthesis after the name identifies array names.

**Remarks:** Integer arrays consume two bytes per element, real arrays five bytes and string arrays three bytes for the string descriptor plus the length of the string. If an array identifier is used without previous declaration, an implicit declaration of an one dimensional array with limit 10 is performed.

**Example:** Using **DIM**

```
10 DIM AX(8) :REM ARRAY OF 9 ELEMENTS
20 DIM XX(2,3) :REM ARRAY OF 3x4 = 12 ELEMENTS
30 FOR I=0 TO 8:AX(I)=PEEK(256+I):NEXT
40 FOR I=0 TO 2:FOR J=0 TO 3:READ XX(I,J):NEXT J,I
50 END
60 DATA 1,-2,3,-4,5,-6,7,-8,9,-10,11,-12
```

# DIR

- Token:** \$EE (DIR) \$FE \$29 (ECTORY)
- Format:** DIRECTORY [filepattern] [,W] [,R] [,D drive] [,U unit]
- Format:** DIR [filepattern] [,W] [,R] [,D drive] [,U unit]
- Format:** \$ [filepattern] [,W] [,R] [,D drive] [,U unit]
- Usage:** Prints a file directory/catalog of the specified disk.
- The shortcut symbol \$ can be used in direct mode only.
- The **W** (Wide) parameter lists the directory three columns wide on the screen and pauses after a the screen is full (63 directory entries). Pressing any key displays the next page.
- The **R** (Recoverable) parameter includes files in the directory, which are flagged as deleted but are still recoverable.
- filepattern** is either a quoted string, for example: "**da\*\***" or a string expression in parentheses, e.g. (**DI\$**)
- drive** = drive # in dual drive disk units.  
The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.
- unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.
- Remarks:** The command **DIR** is a synonym for **CATALOG** or **DIRECTORY** and produces the same listing. The **filepattern** can be used to filter the listing. The wild-card characters \* and ? may be used. Adding a ,T= to the pattern string, with **T** specifying a filetype of **P**, **S**, **U** or **R** (for **PRG**, **SEQ**, **USR**, **REL**) filters the output to that filetype.
- Example:** Using **DIRECTORY**

```
DIRECTORY
0 "BLACK SMURF      " BS 2A
508 "STORY PHOBOS"   SEQ
27  "C8096"          PRG
25  "C128"           PRG
104 BLOCKS FREE.
```

| MEGA65 - Xemu [100% 9%] running 3.5MHz |   |                    |   |                       |   |
|----------------------------------------|---|--------------------|---|-----------------------|---|
| <b>0 "MEGA65 DEVKIT"</b>               |   |                    |   |                       |   |
| 1 "IFF"                                | P | 128 "PULP.IFF"     | P | 23 "RASTER65 2020"    | P |
| 2 "ASH-TEST"                           | P | 85 "TROOPER.IFF"   | P | 1 "INDEX"             | P |
| 1 "SPRITES"                            | P | 72 "RIPLEY.IFF"    | P | 77 "BMPVIEW"          | P |
| 1 "SPRITE+3"                           | P | 8 "-----"          | D | 2 "BENCH"             | P |
| 2 "ELLIPSE"                            | P | 50 "C65NOTE PAD"   | P | 48 "C64.5 DEMO"       | P |
| 1 "SPRITE+4"                           | P | 3 "README"         | P | 8 "____ GAMES ____"   | D |
| 1 "PARROT"                             | P | 33 "NPAGES"        | P | 1 "MATH"              | P |
| 1 "SPRITE+5"                           | P | 8 "BPAGE1"         | P | 2 "CIRCLE"            | P |
| 1 "SPRITE+6"                           | P | 8 "BPAGE2"         | P | 68 "C64NABALT"        | P |
| 1 "SPRITE+7"                           | P | 8 "BPAGE3"         | P | 15 "SPIKES"           | P |
| 3 "DEMOsprites1"                       | P | 8 "BPAGE4"         | P | 1 "CIRCLEBUG"         | P |
| 1 "DEMOsprites2"                       | P | 8 "BTEXT1 PETSCII" | S | 1 "BEN"               | P |
| 27 "DEMOGRAPH"                         | P | 8 "BTEXT1 UTF8"    | S | 1 "FOR"               | P |
| 129 "REGINA.0"                         | P | 8 "-----"          | D | 8 "____ DEMOS ____"   | D |
| 129 "REGINA.1"                         | P | 8 "-----"          | D | 38 "DRAGON II/ELC"    | P |
| 4 "REGINA.P"                           | P | 8 "-----"          | D | 53 "TIME 4 DOLE /NEC" | P |
| 2 "IFF SLIDE SHOW"                     | P | 61 "MENU"          | P | 48 "-GREENWARS/TLS!"  | P |
| 182 "ALIEN.IFF"                        | P | 3 "W.MEGA65"       | P | 81 "ELECTRIC CAFE"    | P |
| 70 "BEAKER.IFF"                        | P | 12 "MEGA65INFO"    | P | 26 "D.Y.C.P./JEWELS"  | P |
| 42 "JOKER.IFF"                         | P | 198 "EE"           | P | 1 "SPRITE+1"          | P |
| 74 "PICARD.IFF"                        | P | 1 "SPRITE+0"       | P | 1 "SPRITE+2"          | P |
| 568 BLOCKS FREE                        |   |                    |   |                       |   |
| READY.                                 |   |                    |   |                       |   |

# DISK

**Token:** \$FE \$40

**Format:** **DISK command [,U unit]**

**Format:** **@ command [,U unit]**

**Usage:** The shortcut symbol @ can be used in direct mode only.

Sends a command string to the specified disk unit.

Using the command with no parameters, prints the disk status.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**command** is a string expression.

**Remarks:** The command string is interpreted by the disk unit and must be compatible to the used DOS version. Read the disk drive manual for possible commands.

**Example:** Using **DISK**

```
DISK "I8" :REM INITIALISE DISK IN DRIVE 0  
DISK "U8>9" :REM CHANGE UNIT# TO 9
```

# DLOAD

**Token:** \$F0

**Format:** **DLOAD filename [,D drive] [,U unit]**

**Usage:** "Disk LOAD" loads a file of type PRG into memory reserved for BASIC program source.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The load address, stored in the first two bytes of the file is ignored. The program is loaded into the BASIC memory. This enables loading of BASIC programs, that were saved on other computers with different memory configurations. After loading the program is re-linked and ready to run or edit. It is possible to use DLOAD in a running program (Called overlay or chaining). Then the new loaded program replaces the current one and the execution starts automatically on the first line of the new program. Variables, arrays and strings from the current run are preserved and can be used by the new loaded program.

**Example:** Using **DLOAD**

```
DLOAD "APOCALYPSE"  
DLOAD "MEGA TOOLS",U9  
DLOAD (FI$),U(UNX)
```

# DMA

**Token:** \$FE \$1F

**Format:** **DMA command [,length, source address, source bank, target address, target bank, sub]**

**Usage:** The **DMA** ("Direct Memory Access") command is obsolete and replaced by the command **EDMA**.

**command** 0 = copy, 1 = mix, 2 = swap, 3 = fill

**length** = number of bytes

**source address** = 16 bit address of read area or fill byte

**source bank** = bank number for source (ignored for fill mode)

**target** = 16 bit address of write area

**target bank** = bank number for target

**sub** = sub command

**Remarks:** The **DMA** command has access to the lower 1 MB address range organised in 16 banks of 64 K. To avoid this limitation, use the command **EDMA**, which has access to the full 256 MB address range.

**Example:** Using **DMA**

```
DMA 0, 80*25, 2048, 0, 0, 4 :REM SAVE SCREEN TO $00000 BANK 4  
DMA 3, 80*25, 32, 0, 2048, 0 :REM FILL SCREEN WITH BLANKS  
DMA 0, 80*25, 0, 4, 2048, 0 :REM RESTORE SCREEN FROM $00000 BANK 4  
DMA 2, 80, 2048, 0, 2048+80, 0 :REM SWAP CONTENTS OF LINE 1 & 2 OF SCREEN
```

# **DMODE**

**Token:** \$FE \$35

**Format:** **DMODE jam,complement,inverse,stencil,style,thick**

**Usage:** "Display MODE" sets several parameter of the graphical context for drawing commands.

|                   |       |
|-------------------|-------|
| <b>jam</b>        | 0 - 1 |
| <b>complement</b> | 0 - 1 |
| <b>inverse</b>    | 0 - 1 |
| <b>stencil</b>    | 0 - 1 |
| <b>style</b>      | 0 - 3 |
| <b>thick</b>      | 1 - 8 |

# DO

**Token:** \$EB

**Format:** **DO ... LOOP**

**DO** [ <UNTIL | WHILE> <logical expr.>]  
. . . statements [**EXIT**]  
**LOOP** [ <UNTIL | WHILE> <logical expr.>]

**Usage:** The **DO** and **LOOP** keywords define the start and end of the most versatile BASIC loop. Using **DO** and **LOOP** alone, without any modifiers creates an infinite loop, that can be left by the **EXIT** statement only. The loop can be controlled by adding an **UNTIL** or a **WHILE** statement after the **DO** or **LOOP**.

**Remarks:** **DO** loops may be nested. An **EXIT** statement exits the current loop only.

**Example:** Using **DO** and **LOOP**

```
10 PW$="" : DO  
20 GET A$: PW$=PW$+A$  
30 LOOP UNTIL LEN(PW$)>? OR A$=CHR$(13)  
  
10 DO : REM WAIT FOR USER DECISION  
20 GET A$  
30 LOOP UNTIL A$='Y' OR A$='N' OR A$='y' OR A$='n'  
  
10 DO WHILE ABS(EPS) > 0.001  
20 GOSUB 2000 : REM ITERATION SUBROUTINE  
30 LOOP  
  
10 IX=0 : REM INTEGER LOOP 1 -> 100  
20 DO IX=IX+1  
30 LOOP WHILE IX < 101
```

# DOPEN

**Token:** \$FE \$0D

**Format:** DOPEN# Ifn, filename [,L[reclen]] [,W] [,D drive] [,U unit]

**Usage:** Opens a file for reading, writing or modifying.

**Ifn** = logical file number

1 <= Ifn <= 127: line terminator is CR

128 <= Ifn <= 255: line terminator is CR LF

**L** indicates, that the file is a relative file, which is opened for read/write and random access. The reclength is mandatory for creating relative files. For existing relative files, the reclen is used as a safety check, if given.

**W** opens a file for write access. The file must not exist.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** DOPEN## may be used to open all file types. The sequential file type **SEQ** is default. The relative file type **REL** is chosen by using the **L** parameter. Other file types must be specified in the filename, e.g. by adding ",P" to the filename for program files or ",U" for USR files.

The usage of the "save-and-replace" character 'e' at the beginning of the filename is not recommended, because many Commodore disk drives have a bug, that can cause data loss when using this feature.

**Example:** Using **DOPEN**

```
DOPEN#5,"DATA",U9
DOPEN#130,(DD$),U(UNK)
DOPEN#3,"USER FILE,U"
DOPEN#2,"DATA BASE",L240
DOPEN#4,"MYPROG,P" : REM OPEN PRG FILE
```

# DPAT

**Token:** \$FE \$36

**Format:** DPAT **type [,number, pattern, ...]**

**Usage:** "Drawing PATtern" sets pattern of the graphical context for drawing commands.

|                |         |
|----------------|---------|
| <b>type</b>    | 0 - 63  |
| <b>number</b>  | 1 - 4   |
| <b>pattern</b> | 0 - 255 |

# DS

**Format:** **DS** is a reserved system variable

**Usage:** **DS** holds the status of the last disk operation. It is a volatile variable. Each use triggers the reading of the disk status from the current disk device in usage. **DS** is coupled to the string variable **DS\$** which is updated at the same time. Reading the disk status from a disk device automatically clears any error status on that device, so subsequent reads will return 0, if no other activity was in between.

**Example:** Using **DS**

```
100 DOPEN#1,"DATA"  
110 IF DS<>0 THEN PRINT"COULD NOT OPEN FILE DATA":STOP
```

# **DS\$**

**Format:** **DS\$** is a reserved system variable

**Usage:** **DS\$** holds the status of the last disk operation in text form of the format: Code,Message,Track,Sector.

**DS\$** is coupled to the numeric variable **DS**. It is updated, when **DS** is used. **DS\$** is set to "00,OK,00,00", if there was no error, otherwise it is set to a DOS error message (listed in the disk manuals).

**Example:** Using **DS\$**

```
100 DOPEN#1,"DATA"  
110 IF DS<>0 THEN PRINT DS$:STOP
```

# DSAVE

**Token:** \$EF

**Format:** DSAVE filename [,D drive] [,U unit]

**Usage:** "Disk SAVE" saves a BASIC program to a file of type PRG.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**) The maximum length of the filename is 16 characters. If the first character of the filename is an at-sign '@' it is interpreted as a "save and replace" operation. It is dangerous to use this replace option on drives 1541 and 1571, because they contain the notorious "save and replace bug" in their DOS.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The **DVERIFY** can be used after **DSAVE** to check, if the saved program on disk is identical to the program in memory.

**Example:** Using **DSAVE**

```
DSAVE "ADVENTURE"  
DSAVE "ZORK-I",U9  
DSAVE "DUNGEON",D1,U10
```

# **DT\$**

**Format:** **DT\$** is a reserved system variable

**Usage:** **DT\$** holds the current date and is updated before each usage from the RTC (Real Time Clock). The RTC can be set in the CONFIGURE menu. The string **DT\$** is formatted as: "DD-MON-YYYY", for example: "04-APR-2021".

**Example:** Using **DT\$**

```
100 PRINT "TODAY IS: ",DT$
```

# DVERIFY

**Token:** \$FE \$14

**Format:** **DVERIFY filename [,D drive] [,U unit]**

**Usage:** "Disk VERIFY" compares a BASIC program in memory with a disk file of type PRG.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** **DVERIFY** can only test for equality. It gives no information about the number or position of different valued bytes. The command exits either with the message **OK** or with **VERIFY ERROR**.

**Example:** Using **DVERIFY**

```
DVERIFY "ADVENTURE"  
DVERIFY "ZORK-I",U9  
DVERIFY "DUNGEON",D1,U10
```

# EDIT

**Format:** **EDIT <ON | OFF>**

**Usage:** **EDIT** switches the builtin editor either to text mode **EDIT ON** or BASIC program editor **EDIT OFF**.

After power up or reset, the editor is initialised as BASIC program editor.

After setting the editor to text mode with **EDIT ON**, the differences to program mode are:

The editor does no tokenising. All text entered after a linenumbers remains pure text, BASIC keywords like **FOR** or **GOTO** are not converted to BASIC tokens, like in program mode.

The line numbers are only used for text organisation sorting, deleting, listing etc. When the text is saved to file with **DSAVE**, a sequential file (type SEQ) is written, not a program (PRG) file, like in program mode. Line numbers are not written to the file.

**DLOAD** in text mode can load only sequential files. Linenumbers are automatically generated for editing purposes.

The mode of the editor can be recognised by looking at the prompt: In program mode, the prompt is: **READY.**, while in text mode the prompt is: **OK**.

The text mode affects entered lines with leading number only, lines with no linenumbers are executed as BASIC commands, as usual.

Sequential files, created with the text editor, can be displayed (without loading them) on the screen by using the **TYPE <filename>** command.

**Example:** Using **EDIT**

```
ready.  
edit on  
  
ok.  
100 This is a simple text editor.  
dsave "example"  
  
ok.  
new  
  
ok.  
catalog  
  
0 "demoempty"    " 00 3d  
1  "example"      seq  
3159 blocks free  
  
ok.  
type "example"  
This is a simple text editor.  
  
ok.  
dload "example"  
  
loading  
  
ok.  
list  
  
1000 This is a simple text editor.  
  
ok.
```

# EDMA

**Token:** \$FE \$21

**Format:** **EDMA command ,length, source, target [, sub , mod]**

**Usage:** The **EDMA** ("Extended Direct Memory Access") command is the fastest method to manipulate memory areas using the DMA controller.

**command** 0 = copy, 1 = mix, 2 = swap, 3 = fill

**length** = number of bytes (maximum = 65535)

**source** = 28 bit address of read area or fill byte

**target** = 28 bit address of write area

**sub** = sub command (see chapter on DMA controller))

**mod** = modifier (see chapter on DMA controller)

**Remarks:** The **EDMA** command can access to the whole 256 MB address range using up to 28 bit for the addresses of source and target.

**Example:** Using **EDMA**

```
EDMA 0, $800, $F700, $8000000 :REM COPY SCALAR VARIABLES TO ATTIC RAM  
EDMA 3, 80*25, 32, 2048      :REM FILL SCREEN WITH BLANKS  
EDMA 0, 80*25, 2048, $8000000 :REM COPY SCREEN TO ATTIC RAM
```

# **EL**

**Format:** **EL** is a reserved system variable

**Usage:** **EL** has the value of the line, where the latest BASIC error occurred or the value -1 if there was no error.

This variable is typically used in a TRAP routine, where the error line is taken from **EL**.

**Example:** Using **EL**

```
10 TRAP 100  
  
100 IF ER>0 AND ER<42 THEN PRINT ERR$(ER); " ERROR"  
110 PRINT " IN LINE";EL  
120 RESUME
```

# ELLIPSE

**Token:** \$FE \$30

**Format:** ELLIPSE **xcentre, ycentre, xradius, yradius, [,solid]**

**Usage:** As the name says, it draws an ellipse.

**xcentre** x coordinate of centre in pixels.

**ycentre** y coordinate of centre in pixels.

**xradius** x radius of the ellipse in pixels.

**yradius** y radius of the ellipse in pixels.

**solid** will fill the ellipse if not zero.

**Remarks:** The **ELLIPSE** command is used to draw ellipses on screens with various resolutions. It can also be used to draw circles.

**Example:** Using **ELLIPSE**



```
100 REM ELLIPSE
110 S% = 0 :REM SCREEN
120 W% = 0 :REM WIDTH 320
130 H% = 0 :REM HEIGHT 200
140 B% = 2 :REM BITPLANES
150 POKE 0, 65 :REM 40 MEGAHERTZ
160 W=(W%+1)*320:H=(H%+1)*200 :REM WIDTH & HEIGHT IN PIXELS
170 X0=W/2:Y0=H/2:XD=W/4:YD=H/4 :REM CENTRE AND HALF AXIS
180 BORDER 0 :REM BLACK
190 BACKGROUND 0 :REM BLACK
200 FOREGROUND 5 :REM GREEN
210 GRAPHIC CLR :REM INIT
220 SCREEN DEF S%,W%,H%,B% :REM SET PARAMETERS
230 SCREEN OPEN S% :REM OPEN
240 SCREEN SET S%,S% :REM SET VIEW AND DRAW SCREEN
250 SCNCLR S% :REM CLEAR SCREEN
260 PALETTE S%,0, 0, 0, 0 :REM BLACK
270 PALETTE S%,1,15,15,15 :REM WHITE
280 PALETTE S%,2,15, 0, 0 :REM RED
290 PALETTE S%,3, 0, 0,15 :REM BLUE
300 PEN 0,2 :REM DRAW PEN
310 ELLIPSE X0,Y0,XD,YD,1
320 PEN 0,3 :REM DRAW PEN
330 ELLIPSE X0,Y0,XD+8,YD+8,0
340 A$=STR$(W)+" X"+STR$(H)+" X"+STR$(B%)
350 PEN 0,1 :REM DRAW PEN
360 CHAR 12,10,1,1,2,A$
370 GETKEY A$
380 SCREEN CLOSE S%
390 PALETTE RESTORE
```

# ELSE

**Token:** \$D5

**Format:** **IF expression THEN true clause ELSE false clause**

**Usage:** The **ELSE** keyword is part of an **IF** statement.

**expression** is a logical or numeric expression. A numerical expression is evaluated as **FALSE** if the value is zero and **TRUE** for any non zero value.

**true clause** are one or more statements starting directly after **THEN** on the same line. A linenumber after **THEN** performs a **GOTO** to that line.

**false clause** are one or more statements starting directly after **ELSE** on the same line. A linenumber after **ELSE** performs a **GOTO** to that line.

**Remarks:** The standard **IF ... THEN ... ELSE** structure is restricted to a single line. But the **true clause** or **false clause** may be expanded to several lines using a compound statement bracketed with the keywords **BEGIN** and **BEND**.

**Example:** Using **ELSE**

```
100 REM ELSE
110 RED$=CHR$(28):BLACK$=CHR$(144):WHITE$=CHR$(5)
120 INPUT "ENTER A NUMBER";V
130 IF V<0 THENPRINT RED$;:ELSEPRINT BLACK$;
140 PRINT V : REM PRINT NEGATIVE NUMBERS IN RED
150 PRINT WHITE$
160 INPUT "END PROGRAM:(Y/N)";A$
170 IF A$="Y" THENEND
180 IF A$="N" THEN120:ELSE160
```

# **END**

**Token:** \$80

**Format:** **END**

**Usage:** Ends the execution of the BASIC program. The **READY.** prompt appears and the computer goes into direct mode waiting for keyboard input.

**Remarks:** **END** does **not** clear channels or close files. Also variable definitions are still valid after **END**. The program may be continued with the **CONT** statement. After executing the very last line of the program **END** is executed automatically.

**Example:** Using **END**

```
10 IF V < 0 THEN END : REM NEGATIVE NUMBERS END THE PROGRAM  
20 PRINT V
```

# ENVELOPE

**Token:** \$FE \$0A

**Format:** ENVELOPE n, [attack,decay,sustain,release, waveform,pw]

**Usage:** Used to define the parameters for the synthesis of a musical instrument.

**n** = envelope slot (0 -> 9)

**attack** = attack rate (0 -> 15)

**decay** = decay rate (0 -> 15)

**sustain** = sustain rate (0 -> 15)

**release** = release rate (0 -> 15)

**waveform** = (0:triangle, 1:sawtooth, 2:square/pulse, 3:noise, 4:ring modulation)

**pw** = pulse width (0 -> 4095) for waveform = pulse.

There are 10 slots for storing tunes, preset with following values:

| n | A  | D | S  | R | WF | PW   | Instrument  |
|---|----|---|----|---|----|------|-------------|
| 0 | 0  | 9 | 0  | 0 | 2  | 1536 | piano       |
| 1 | 12 | 0 | 12 | 0 | 1  |      | accordion   |
| 2 | 0  | 0 | 15 | 0 | 0  |      | calliope    |
| 3 | 0  | 5 | 5  | 0 | 3  |      | drum        |
| 4 | 9  | 4 | 4  | 0 | 0  |      | flute       |
| 5 | 0  | 9 | 2  | 1 | 1  |      | guitar      |
| 6 | 0  | 9 | 0  | 0 | 2  | 512  | harpsichord |
| 7 | 0  | 9 | 9  | 0 | 2  | 2048 | organ       |
| 8 | 8  | 9 | 4  | 1 | 2  | 512  | trumpet     |
| 9 | 0  | 9 | 0  | 0 | 0  |      | xylophone   |

**Example:** Using ENVELOPE

```
10 ENVELOPE 9,10,5,10,5,2,4000:PLAY "T9"  
20 VOL 8  
30 TEMPO 100  
40 PLAY "C D E F G A B"  
50 PLAY "U5 V1 C D E F G A B"
```

# ERASE

**Token:** \$FE \$2A

**Format:** ERASE **filename** [,D **drive**] [,U **unit**] [,R]

**Usage:** Used to erase a disk file.

**filename** is either a quoted string, e.g. "data" or a string expression in parentheses, e.g. (F1\$)

**drive** = drive # in dual drive disk units.

The drive # defaults to 0 and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to 8.

**R** = Recover a previously erased file. This will only work, if there were no write operations between erasing and recovery, which may have altered the contents of the file.

**Remarks:** The ERASE **filename** command works like the SCRATCH **filename** command.

The success and the number of erased files can be examined by printing or using the system variable DS\$. The second last number, which reports the track number in case of an disk error, now reports the number of successfully erased files.

**Example:** Using ERASE

```
SCRATCH "DRM",09 :REM SCRATCH FILE DRM ON UNIT 9
PRINT DS$
01, FILES SCRATCHED,01,00
SCRATCH "OLD*"   :REM SCRATCH ALL FILES BEGINNING WITH "OLD"
PRINT DS$
01, FILES SCRATCHED,04,00
```

# **ER**

**Format:** **ER** is a reserved system variable

**Usage:** **ER** has the value of the latest BASIC error occurred or the value -1 if there was no error.

This variable is typically used in a TRAP routine, where the error number is taken from **ER**.

**Example:** Using **ER**

```
10 TRAP 100  
  
100 IF ER>0 AND ER<42 THEN PRINT ERR$(ER); " ERROR"  
110 RESUME
```

# ERR\$

**Token:** \$D3

**Format:** **ERR\$(number)**

**Usage:** Used to convert an error number to an error string.

**number** is a BASIC error number (1 -> 41).

This function is typically used in a TRAP routine, where the error number is taken from the reserved variable **ER**.

**Remarks:** Arguments out of range (1 -> 41) will produce an 'ILLEGAL QUANTITY' error.

**Example:** Using **ERR\$**

```
10 TRAP 100  
  
100 IF ER>0 AND ER<42 THEN PRINT ERR$(ER); " ERROR"  
110 RESUME
```

# EXIT

**Token:** \$FD

**Format:** EXIT

**Usage:** Exits the current **DO .. LOOP** and continues execution at the first statement after the next **LOOP** statement.

**Remarks:** In nested loops **EXIT** exits only one loop continuing executing in the next outer loop if there is one.

**Example:** Using **EXIT**

```
1 REM EXIT
10 OPEN 2,8,0,"$"      : REM OPEN CATALOG
15 IF DS THEN PRINT DS$: STOP: REM CANT READ
20 GETH#2,D$,D$        : REM DISCARD LOAD ADDRESS
25 DO                  : REM LINE LOOP
30 GETH#2,D$,D$        : REM DISCARD LINE LINK
35 IF ST THEN EXIT     : REM END-OF-FILE
40 GETH#2,LO,HI        : REM FILE SIZE BYTES
45 S=LO + 256 * HI    : REM FILE SIZE
50 LINE INPUT#2, F$    : REM FILE NAME
55 PRINT S;F$          : REM PRINT FILE ENTRY
60 LOOP
65 CLOSE 2
```

# **EXP**

**Token:** \$BD

**Format:** **EXP(numeric expression)**

**Usage:** The **EXP** (EXponential function) computes the value of the mathematical constant Euler's number **e = 2.71828183** raised to the power of the argument.

**Remarks:** An argument greater than 88 produces an OVERFLOW ERROR:

**Example:** Using **EXP**

```
PRINT EXP(1)  
2.71828183
```

```
PRINT EXP(0)  
1
```

```
PRINT EXP(LOG(2))  
2
```

# FAST

**Token:** \$FE \$25

**Format:** **FAST**

**Usage:** Sets the system speed to 3.5 MHz. The system default is **FAST**. However after using **SLOW**, **FAST** can be used to return to fast mode.

**Remarks:** Switching the MEGA65 to the fastest mode at 40 MHz is done with the command **POKE 0,65**.

**Example:** Using **FAST**

```
10 SLOW  
20 GOSUB 1000:REM DO SOME SLOW I/O  
30 FAST
```

# FILTER

**Token:** \$FE \$03

**Format:** **FILTER [freq, lp, bp, hp, res]**

**Usage:** Sets the parameters for sound filter.

**freq** = filter cut off frequency (0 -> 2047)

**lp** = low pass filter (0:off, 1:on)

**bp** = band pass filter (0:off, 1:on)

**hp** = high pass filter (0:off, 1:on)

**resonance** = resonance (0 -> 15)

**Remarks:** Missing parameter keep their current value. The effective filter is the sum of of all filter settings. This enables band reject and notch effects.

**Example:** Using **FILTER**

```
FILTER 1023,1,0,0,10 :REM LOW PASS  
FILTER 1023,0,1,0,10 :REM BAND PASS  
FILTER 1023,0,0,1,10 :REM HIGH PASS
```

# FIND

**Token:** \$FE \$2B

**Format:** **FIND delimiter string delimiter [,from-to]**

**Usage:** **FIND** is an editor command and can be used in direct mode only. It searches the line range (if specified) or the whole BASIC program else. At each occurrence of the "find string" the line is listed with the string highlighted. The <NO-SCROLL> key can be used to pause the output.

**Remarks:** Basically any unshifted character, that is not part of the string, can be used as delimiter.

But using quotes " as delimiter has a special effect: In this case the search text is not tokenised: FIND "FOR" will search the three letters F, O, R, not the BSASIC keyword **FOR**. So it can find the word **FOR** in string constants or REM statements, but not in program code.

On the other hand FIND /FOR/ will find all occurrences of the BASIC keyword, but not the text "FOR" in strings.

Also notice, that you cannot search for partial keywords. **FIND /LOO/** will not find the keyword **LOOP**,

**Example:** Using **FIND**

The screenshot shows the MEGA65 - Xemu interface. The title bar reads "MEGA65 - Xemu [100% 2%] running 3.5MHz". The main window displays the following text:

```
READY.  
LIST  
10 REM PARROT COLOUR SCHEME  
20 FONT 8 :REM SERIF  
30 FOREGROUND 5 :REM GREEN  
40 BACKGROUND 0 :REM BLACK  
50 HIGHLIGHT 4,0 :REM SYSTEM PURPLE  
60 HIGHLIGHT 14,1 :REM REM BLUE  
70 HIGHLIGHT 7,2 :REM KEYWORD YELLOW  
  
READY.  
FIND /O/0/  
10 REM PARROT COLOUR SCHEME  
  
READY.  
FIND /HIGHLIGHT/  
  
50 HIGHLIGHT 4,0 :REM SYSTEM PURPLE  
60 HIGHLIGHT 14,1 :REM REM BLUE  
70 HIGHLIGHT 7,2 :REM KEYWORD YELLOW  
  
READY.
```

# **FN**

**Token:** \$A5

**Format:** **FN name(numeric expression)**

**Usage:** The **FN** functions are user defined functions, that accept a numeric expression as argument and return a real value. They must be defined with **DEF FN** before the first usage.

**Example:** Using **FN**

```
10 PD = π / 180
20 DEF FN CD(X)= COS(X*PD): REM COS FOR DEGREES
30 DEF FN SD(X)= SIN(X*PD): REM SIN FOR DEGREES
40 FOR D=0 TO 360 STEP 90
50 PRINT USING "###";D
60 PRINT USING " ##.##";FNCD(D);
70 PRINT USING " ##.##";FNSD(D)
80 NEXT D
RUN
    0  1.00  0.00
    90  0.00  1.00
    180 -1.00  0.00
    270  0.00 -1.00
    360  1.00  0.00
```

# FONT

**Token:** \$FE \$46

**Format:** **FONT [A|B|C]**

**Usage:** The **FONT** command is used to switch between fonts and the code pages PETSCII and enhanced PETSCII. The enhanced PETSCII includes all ASCII symbols, that are missing in the PETSCII code page, though the order is still PETSCII. The ASCII symbols are typed by holding the  together with the desired key. The codes for uppercase and lowercase are swapped compared to ASCII. The uppercase/graphics code page is not changed.

| code | key        | PETSCII | ASCII |
|------|------------|---------|-------|
| \$5C | pound      | £       | \     |
| \$5E | up arrow   | ↑       | ^     |
| \$5F | back arrow | ↑       | -     |
| \$7B | colon      | †       | {     |
| \$7C | dot        | ‡       |       |
| \$7D | semicolon  |         | }     |
| \$7E | comma      | ¶       | ~     |

**Example:** Using **FONT**

```
FONT A :REM ASCII - ENABLE {}_~^
FONT B :REM LIKE A, WITH A SERIF FONT
FONT C :REM COMMODORE FONT (DEFAULT)
```

# FOR

**Token:** \$81

**Format:** **FOR index=start TO end [STEP step] ... NEXT [index]**

**Usage:** The **FOR** statement starts the definition of a BASIC loop with an index variable.

The **index** variable may be incremented or decremented by a constant value on each iteration. The default is to increment the variable by 1. The index variable must be a real variable.

The **start** value is used to initialise the index.

The **end** value is used at the end of the loop and controls, whether the next iteration will be started or the loop exited.

The **step** value defines the change applied to the index variable at the end of the loop. Positive step values increment it, while negative values decrement it. It defaults to 1.0 if not specified.

**Remarks:** For positive increments **end** must be greater or equal than **start**, for negative increments **end** must be less or equal than **start**.

It is bad programming style to change the value of the index variable inside the loop or to jump into or out of the loop body with **GOTO**.

**Example:** Using **FOR**

```
10 FOR D=0 TO 360 STEP 30
20 R = D * π / 180
30 PRINT D;R;SIN(R);COS(R);TAN(R)
40 NEXT D

10 DIM M(20,20)
20 FOR I=0 TO 20
30 FOR J=I TO 20
40 M(I,J) = I + 100 * J
50 NEXT J,I
```

# FOREGROUND

**Token:** \$FE \$39

**Format:** FOREGROUND colour

**Usage:** Sets the foreground colour (text colour) of the screen to the argument, which must be in the range 0 to 15. (See colour table).

**Example:** FOREGROUND 7 - select foreground colour yellow.

**Colours:** Index and RGB values of colour palette

| index | red | green | blue | colour      |
|-------|-----|-------|------|-------------|
| 0     | 0   | 0     | 0    | black       |
| 1     | 15  | 15    | 15   | white       |
| 2     | 15  | 0     | 0    | red         |
| 3     | 0   | 15    | 15   | cyan        |
| 4     | 15  | 0     | 15   | magenta     |
| 5     | 0   | 15    | 0    | green       |
| 6     | 0   | 0     | 15   | blue        |
| 7     | 15  | 15    | 0    | yellow      |
| 8     | 15  | 6     | 0    | orange      |
| 9     | 10  | 4     | 0    | brown       |
| 10    | 15  | 7     | 7    | pink        |
| 11    | 5   | 5     | 5    | dark grey   |
| 12    | 8   | 8     | 8    | medium grey |
| 13    | 9   | 15    | 9    | light green |
| 14    | 9   | 9     | 15   | light blue  |
| 15    | 11  | 11    | 11   | light grey  |

# FRE

**Token:** \$B8

**Format:** **FRE(bank)**

**Usage:** Returns the number of free bytes for banks 0 or 1, or the ROM version, if the argument is negative.

**FRE(0)** returns the number of free bytes in bank 0, which is used for BASIC program source.

**FRE(1)** returns the number of free bytes in bank 1, which is the bank for BASIC variables, arrays and strings. A usage of **FRE(1)** also triggers the "garbage collection", a process, that collects used strings at the top of the bank, thereby defragmenting string memory.

**FRE(-1)** returns the ROM version, a six-digit number of the form 92xxxx.

**Example:** Using **FRE**:

```
10 PM = FRE(0)
20 VM = FRE(1)
30 RV = FRE(-1)
40 PRINT PM;" FREE FOR PROGRAM"
50 PRINT VM;" FREE FOR VARIABLES"
60 PRINT RV;" ROM VERSION"
```

# FREAD

**Token:** \$FE \$1C

**Format:** **FREAD#channel, pointer, size**

**Usage:** Reads **size** bytes from **channel** to memory starting at the 32 bit address **pointer**.

Care must be taken, not to overwrite memory, that is used by the system or the interpreter.

It is recommended to use the **POINTER** statement for the pointer argument and compute the size parameter by multiplying the number of elements with the item size.

| type          | item size |
|---------------|-----------|
| byte array    | 1         |
| integer array | 2         |
| real array    | 5         |

Keep in mind, that the **POINTER** function with a string argument does NOT return the string address, but the string descriptor. It is recommend, not to use **FREAD** for strings or string arrays unless you are fully aware, how to handle the string storage internals.

Also take care, that you always specify an index, if you use an array. The start address of array XY() is **POINTER(XY(0))**. **POINTER(XY)** returns the address of the scalar variable XY.

**Example:** Using **FREAD**:

```
100 N=23
110 DIM B&(N),C&(N)
120 DOPEN#2,"TEXT"
130 FREAD#2,POINTER(B&(0)),N
140 DCLOSE#2
150 FOR I=0 TO N-1:PRINT CHR$(B&(I));:NEXT
160 FOR I=0 TO N-1:C&(I)=B&(N-1-I):NEXT
170 DOPEN#2,"REVERS",N
180 FWRITE#2,POINTER(C&(0)),N
190 DCLOSE#2
```

# FWRITE

**Token:** \$FE \$1E

**Format:** FWRITE#channel, pointer, size

**Usage:** Writes **size** bytes from **channel** to memory starting at the 32 bit address **pointer**.

It is recommended to use the **POINTER** statement for the pointer argument and compute the size parameter by multiplying the number of elements with the item size.

| type          | item size |
|---------------|-----------|
| byte array    | 1         |
| integer array | 2         |
| real array    | 5         |

Keep in mind, that the **POINTER** function with a string argument does NOT return the string address, but the string descriptor. It is recommend, not to use **FWRITE** for strings or string arrays unless you are fully aware, how to handle the string storage internals.

Also take care, that you always specify an index, if you use an array. The start address of array XY() is **POINTER(XY(0))**. **POINTER(XY)** returns the address of the scalar variable XY.

**Example:** Using **FWRITE**:

```
100 N=23
110 DIM B&(N),C&(N)
120 DOPEN#2,"TEXT"
130 FREAD#2,POINTER(B&(0)),N
140 DCLOSE#2
150 FOR I=0 TO N-1:PRINTCHR$(B&(I));:NEXT
160 FOR I=0 TO N-1:C&(I)=B&(N-1-I):NEXT
170 DOPEN#2,"REVERS",N
180 FWRITE#2,POINTER(C&(0)),N
190 DCLOSE#2
```

# GET

**Token:** \$A1

**Format:** **GET variable**

**Usage:** Gets the next character or byte value of that character from the keyboard queue. If the variables is of type string and the queue is empty an empty string is assigned to the variable, otherwise a one character string is created and assigned to the string variable. If the variable is of numeric type, the character code of the key is assigned to it, or zero if there was no keypress. This command does not wait for keyboard input, so it's useful to check for key presses in regular intervals or loops.

**Remarks:** The command **GETKEY** is similar, but waits until a key was hit.

**Example:** Using **GET**:

```
10 DO: GET A$: LOOP UNTIL A$ <> ""
40 IF A$ = "W" THEN 1000 :REM GO NORTH
50 IF A$ = "A" THEN 2000 :REM GO WEST
60 IF A$ = "S" THEN 3000 :REM GO EAST
70 IF A$ = "Z" THEN 4000 :REM GO SOUTH
80 IF A$ = CHR$(13) THEN 5000 :REM RETURN
90 GOTO 10
```

# GET#

**Token:** \$A1 '#'

**Format:** GET# channel, list of variables

**Usage:** Reads as many bytes as necessary from the channel argument and assigns strings of length one to string variables or the read 8-bit binary value to numeric variables. This is useful to read characters or bytes from an input stream one by one.

**Remarks:** All values from 0 to 255 are valid, so this command can also be used to read binary data.

**Example:** Using GET# to read a disk directory:

```
1 REM GET#
10 OPEN 2,8,0,"$"           : REM OPEN CATALOG
15 IF DS THEN PRINT DS$: STOP: REM CANT READ
20 GETH#2,D$,D$             : REM DISCARD LOAD ADDRESS
25 DO                      : REM LINE LOOP
30   GETH#2,D$,D$           : REM DISCARD LINE LINK
35   IF ST THEN EXIT        : REM END-OF-FILE
40   GETH#2,LO,HI            : REM FILE SIZE BYTES
45   S=LO + 256 * HI         : REM FILE SIZE
50   LINE INPUT#2, F$         : REM FILE NAME
55   PRINT S;F$              : REM PRINT FILE ENTRY
60 LOOP
65 CLOSE 2
```

# GETKEY

**Token:** \$A1 \$F9 (GET token and KEY token)

**Format:** **GETKEY variable**

**Usage:** Gets the next character or code value from the keyboard queue. If the queue is empty the program waits until a key is hit. Then a one character string is created and assigned to the string variable. If the variable is numeric, the character code is assigned as value.

**Example:** Using **GETKEY**:

```
10 GETKEY A$ :REM WAIT AND GET CHARACTER
40 IF A$ = "W" THEN 1000 :REM GO NORTH
50 IF A$ = "A" THEN 2000 :REM GO WEST
60 IF A$ = "S" THEN 3000 :REM GO EAST
70 IF A$ = "Z" THEN 4000 :REM GO SOUTH
80 IF A$ = CHR$(13) THEN 5000 :REM RETURN
90 GOTO 10
```

# **GO64**

**Token:** \$CB \$36 \$34 (GO token and 64 )

**Format:** **GO64**

**Usage:** Switches the computer to the C64 compatible mode. In direct mode a security prompt **ARE YOU SURE?** is printed, which must be responded with 'Y' to continue. Use **\$VS58552** to switch back to C65 mode.

**Example:** Using **GO64**:

```
GO64  
ARE YOU SURE?
```

# GOSUB

**Token:** \$8D

**Format:** **GOSUB** line

**Usage:** The **GOSUB** (GOto SUBroutine) command continues program execution at the given BASIC line number, saving the current BASIC program counter and line number on the run-time stack. This enables the resume of the execution after the **GOSUB** statement, once a **RETURN** statement in the called subroutine was executed. Calls to subroutines via **GOSUB** may be nested but the end of the subroutine code must always be a **RETURN**. Otherwise a stack overflow may occur.

**Remarks:** Unlike other programming languages, this BASIC version does not support arguments or local variables for subroutines.

Programs can be optimised by grouping subroutines at the beginning of the program source. The **GOSUB** calls will then have low line numbers with only few digits to decode. Also the subroutines will be found faster, because the search for subroutines starts very often at the start of the program.

**Example:** Using **GOSUB**:

```
10 GOTO 100 :REM TO MAIN PROGRAM
20 REM *** SUBROUTINE DISK STATUS CHECK ***
30 DD=DS:IF DD THEN PRINT "DISK ERROR",DS$
40 RETURN
50 REM *** SUBROUTINE PROMPT Y/N ***
60 DO:INPUT "CONTINUE (Y/N)",A$
70 LOOP UNTIL A$="Y" OR A$="N"
80 RETURN
90 REM *** MAIN PROGRAM ***
100 DOPENH2,"BIG DATA"
110 GOSUB 30: IF DD THEN DCLOSEH2:GOSUB 60:REM ASK
120 IF A$="N" THEN STOP
130 GOTO 100: REM RETRY
```

# GOTO

**Token:** \$89 (GOTO) or \$CB \$A4 (GO TO)

**Format:** **GOTO line**  
**GO TO line**

**Usage:** Continues program execution at the given BASIC line number.

The **GOTO** command written as a single word executes faster than the **GO TO** command.

**Remarks:** The new line number will be searched by scanning the BASIC source linearly upwards. If the target line number is higher than the current one, the search starts from the current line upwards. If the target line number is lower, the search starts from the start of the program. Knowing this mechanism it is possible to optimise the run-time by grouping often used targets at the start of the program.

**Example:** Using **GOTO**:

```
10 GOTO 100 :REM TO MAIN PROGRAM
20 REM *** SUBROUTINE DISK STATUS CHECK ***
30 DD=DS:IF DD THEN PRINT "DISK ERROR";DS$
40 RETURN
50 REM *** SUBROUTINE PROMPT Y/N ***
60 DO:INPUT "CONTINUE (Y/N)",A$
70 LOOP UNTIL A$="Y" OR A$="N"
80 RETURN
90 *** MAIN PROGRAM ***
100 DOPEN#2,"BIG DATA"
110 GOTO 30: IF DD THEN DCLOSE#2:GOTO 60:REM ASK
120 IF A$="N" THEN STOP
130 GOTO 100: REM RETRY
```

# GRAPHIC

**Token:** \$DE

**Format:** **GRAPHIC CLR**

**Usage:** Initialises the BASIC graphic system. It clears the graphics memory and screen and sets all parameters of the graphics context to the default values.

**Remarks:** A second form of the **GRAPHIC** command, which serves as an interface to internal subroutines may be added later.

**Example:** Using **GRAPHIC**:

```
100 REM GRAPHIC
110 GRAPHIC CLR      : REM INITIALISE
120 SCREEN DEF 1,1,1,2 : REM 640 X 400 X 2
130 SCREEN OPEN 1      : REM OPEN IT
140 SCREEN SET 1,1      : REM VIEW IT
150 PALETTE 1,0,0, 0,0 : REM BLACK
160 PALETTE 1,1,0,15,0 : REM GREEN
170 SCNCLR 0          : REM FILL SCREEN WITH BLACK (0)
180 PEN 0,1            : REM SELECT PEN
190 LINE 50,50,590,350 : REM DRAW LINE
200 GETKEY A$          : REM WAIT FOR KEYPRESS
210 SCREEN CLOSE 1     : REM CLOSE GRAPHIC SCREEN
220 PALETTE RESTORE    : REM RESTORE DEFAULT COLOURS
```

# HEADER

**Token:** \$F1

**Format:** HEADER **diskname [,lid] [,D drive] [,U unit]**

**Usage:** Used to format or clear a diskette or disk.

**diskname** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**DNS\$**) The maximum length of the diskname is 16 characters.

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** For new diskettes or disks, which are not already formatted it is absolutely necessary to specify the disk ID with the parameter **lid**. This switches the format command to the full format, which writes sector IDs and erases all contents. This will need some time, because every block on the disk will be written.

If the **lid** parameter is omitted, a quick format will be performed. This is only possible, if the disk is formatted already. A quick format writes a new disk name and clears the block allocation map, marking all blocks as free. The disk ID is not changed, the blocks are not overwritten, so contents may be recovered with the **ERASE R** command.

**Example:** Using **HEADER**

```
HEADER "ADVENTURE",IBS  
HEADER "ZORK-I",U9  
HEADER "DUNGEON",D1,U10
```

# HELP

**Token:** \$EA

**Format:** **HELP**

**Usage:** When the BASIC program stops due to an error, type **HELP** for further information. The interpreted line is listed, with the erroneous statement highlighted or underlined.

**Remarks:** Displays BASIC errors. For errors in disk I/O one should print the disk status variable **DS** or the disk status string **DS\$**.

**Example:** Using **HELP**

```
10 A=1.E20
20 B=A+A:C=EXP(A):PRINT A,B,C
RUN

?OVERFLOW ERROR IN 20
READY.
HELP

20 B=A+A:C=EXP(A):PRINT A,B,C
```

# **HEX\$**

**Token:** \$D2

**Format:** **HEX\$(numeric expression)**

**Usage:** Returns a four character string in hexadecimal notation converted from the argument. The argument must be in the range 0 -> 65535 corresponding to the hex numbers 0000 -> FFFF.

**Remarks:** If real numbers are used as arguments, the fractional part will be cut off, not rounded.

**Example:** Using **HEX\$**:

```
PRINT HEX$(10),HEX$(100),HEX$(1000.9)  
000A      0064      03E8
```

# HIGHLIGHT

**Token:** \$FE \$3D

**Format:** HIGHLIGHT colour [,mode]

**Usage:** Sets the colour to be used for the "highlight" text attribute. The colour index must be in the range 0 to 15. (See colour table).

The optional parameter **mode** defines, how BASIC listings to the screen use highlighting:

|                 |                          |
|-----------------|--------------------------|
| <b>mode = 0</b> | no syntax highlighting   |
| <b>mode = 1</b> | highlight REM statements |
| <b>mode = 2</b> | highlight BASIC keywords |

**Remarks:** The highlight text attribute is used to mark text in listings generated by the **HELP FIND CHANGE** commands.

**Example:** HIGHLIGHT 8,2 – select highlight colour orange for keywords.

**Colours:** Index and RGB values of colour palette

| index | red | green | blue | colour      |
|-------|-----|-------|------|-------------|
| 0     | 0   | 0     | 0    | black       |
| 1     | 15  | 15    | 15   | white       |
| 2     | 15  | 0     | 0    | red         |
| 3     | 0   | 15    | 15   | cyan        |
| 4     | 15  | 0     | 15   | magenta     |
| 5     | 0   | 15    | 0    | green       |
| 6     | 0   | 0     | 15   | blue        |
| 7     | 15  | 15    | 0    | yellow      |
| 8     | 15  | 6     | 0    | orange      |
| 9     | 10  | 4     | 0    | brown       |
| 10    | 15  | 7     | 7    | pink        |
| 11    | 5   | 5     | 5    | dark grey   |
| 12    | 8   | 8     | 8    | medium grey |
| 13    | 9   | 15    | 9    | light green |
| 14    | 9   | 9     | 15   | light blue  |
| 15    | 11  | 11    | 11   | light grey  |

# IF

**Token:** \$8B

**Format:** **IF expression THEN true clause ELSE false clause**

**Usage:** Starts a conditional execution statement.

**expression** is a logical or numeric expression. A numerical expression is evaluated as **FALSE** if the value is zero and **TRUE** for any non zero value.

**true clause** are one or more statements starting directly after **THEN** on the same line. A linenumber after **THEN** performs a **GOTO** to that line.

**false clause** are one or more statements starting directly after **ELSE** on the same line. A linenumber after **ELSE** performs a **GOTO** to that line.

**Remarks:** The standard **IF ... THEN ... ELSE** structure is restricted to a single line. But the **true clause** or **false clause** may be expanded to several lines using a compound statement bracketed with the keywords **BEGIN** and **BEND**.

**Example:** Using **IF**

```
10 IF V < 0 THEN PRINT RED$;:ELSE PRINT BLACK$;
20 PRINT V : REM PRINT NEGATIVE NUMBERS IN RED
30 INPUT "END PROGRAM:(Y/N)";A$
40 IF A$="Y" THEN END
50 IF A$="N" THEN 10:ELSE 30
```

# INPUT

**Token:** \$85

**Format:** **INPUT [prompt <,|;>] variable list**

**Usage:** Prints an optional prompt string and question mark to the screen, flashes the cursor and waits for user input from the keyboard.

**prompt** = string expression to be printed as prompt. It may be omitted.  
If the separator between prompt and variable list is a comma, the cursor is placed directly after the prompt. If the separator is a semicolon, a question mark and a space is added to the prompt.

**variable list** = list of one or more variables, that receive the input.

The input will be processed after the user hits RETURN.

**Remarks:** The user must take care to enter the correct type of input matching variable types. Also the number of input items must match the number of variables. Entering non numeric characters for integer or real variables will produce a TYPE MISMATCH ERROR. Strings for string variables have to be put in quotes if they contain spaces or commas.

Many programs, that need a safe input routine use **LINE INPUT** and use an own parser, in order to avoid program breaks by wrong user input.

**Example:** Using **INPUT**:

```
10 DIM N$(100),A%(100),S$(100):
20 DO
30 INPUT "NAME, AGE, SEX";N$,A%,SE$
40 IF N$="" THEN 30
50 IF N$="END" THEN EXIT
60 IF A% < 18 OR A% > 100 THEN PRINT "AGE?":GOTO 30
70 IF SE$ <> "M" AND SE$ <> "F" THEN PRINT "SEX?":GOTO 30
80 REM CHECK OK: ENTER INTO ARRAY
90 N$(N)=N$:A%(N)=A%:S$(N)=SE$:N=N+1
100 LOOP UNTIL N=100
110 PRINT "RECEIVED";N;" NAMES"
```

# **INPUT#**

**Token:** \$84

**Format:** **INPUT# channel, variable list**

**Usage:** Reads a record from an input device, e.g. a disk file and assigns the data to the variables in the list.

**channel** = channel number assigned by a **DOPEN** or **OPEN** command.

**variable list** = list of one or more variables, that receive the input.

The input record must be terminated by a RETURN character and must be not longer than the input buffer (160 characters).

**Remarks:** The type and number of data in a record must match the variable list. Reading non numeric characters for integer or real variables will produce a FILE DATA ERROR. Strings for string variables have to be put in quotes if they contain spaces or commas.  
The command **LINE INPUT#** may be used to read a whole record into a single string variable.

**Example:** Using **INPUT#**:

```
10 DIM N$(100),B%(100),S$(100):
20 DOPENH2,"CBM-PEOPLE":REM OPEN SEQ FILE
25 IF DS THEN PRINT DS$:STOP:REM OPEN ERROR
30 FOR I=0 TO 100
40 INPUTH2,N$(I),B%(I),S$(I)
50 IF ST AND 64 THEN 80:REM END OF FILE
60 IF DS THEN PRINT DS$:GOTO 80:REM DISK ERROR
70 NEXT I
80 DCLOSEH2
110 PRINT "READ";I;" RECORDS"
120 FOR J=0 TO I:PRINT N$(I):NEXT J
```

RUN

CHUCK PEDDLE

JACK TRAMIEL

BILL MENSCH

TYPE "CBM-PEOPLE"

"CHUCK PEDDLE",1937,"ENGINEER OF THE 6502"

"JACK TRAMIEL",1928,"FOUNDER OF CBM"

"BILL MENSCH",1945,"HARDWARE

# INSTR

**Token:** \$D4

**Format:** **INSTR(haystack, needle [,start])**

**Usage:** Locates the position of the string expression "needle" in the string expression "haystack" and returns the index of the first occurrence or zero, if there is no match.

The string expression **haystack** is searched for the occurrence of the string expression **needle**.

An enhanced version of string search using pattern matching is triggered, if the first character of the search string is a pound sign '#'. The pound sign is not part of the search but enables the use of the '.' (dot) as wildcard character, which matches any other character. The second special pattern character is the '\*' (asterisk). The asterisk in the search string indicates, that the character preceding the asterisk may occur never, once or many times, in order to be counted as matching.

The optional argument **start** is an integer expression, which defines the starting position for the search in **haystack**. If not present it defaults to one.

**Remarks:** If either string is empty or there is no match the function returns zero.

**Example:** Using **INSTR**:

```
I = INSTR("ABCDEF","CD")      : REM I = 3
I = INSTR("ABCDEF","XY")      : REM I = 0
I = INSTR("RAIIIN","*A*IN")   : REM I = 2
I = INSTR("ABCDEF","E.C.E")   : REM I = 3
I = INSTR(A$+B$,C$)
```

# INT

**Token:** \$B5

**Format:** **INT(numeric expression)**

**Usage:** Searches the greatest integer value, that is less or equal to the argument and returns this value as a real number. This function is **NOT** limited to the typical 16-bit integer range (-32768 -> 32767), because it uses real arithmetic. The allowed range is therefore determined by the size of the real mantissas (32-bit) : (-2147483648 -> 2147483647).

**Remarks:** It is not necessary to use the **INT** function for assigning real values to integer variables, because this conversion will be done implicitly, but then for the 16-bit range.

**Example:** Using **INT**:

```
X = INT(1.9)      :REM X = 1
X = INT(-3.1)     :REM X = -4
X = INT(100000.5) :REM X = 100000
N% = INT(100000.5) :REM ?ILLEGAL QUANTITY ERROR
```

# JOY

**Token:** \$CF

**Format:** JOY(port)

**Usage:** Returns the state of the joystick for the selected port (1 or 2). Bit 7 contains the state of the fire button. The stick can be moved in eight directions, which are numbered clockwise starting at the upper position.

|        | left | centre | right |
|--------|------|--------|-------|
| up     | 8    | 1      | 2     |
| centre | 7    | 0      | 3     |
| down   | 6    | 5      | 4     |

**Example:** Using JOY:

```
10 N = JOY(1)
20 IF N AND 128 THEN PRINT "FIRE! ";
30 REM           N  NE  E   SE  S   SW  W  NW
40 ON N AND 15 GOSUB 100,200,300,400,500,600,700,800
50 GOTO 10
100 PRINT "GO NORTH" :RETURN
200 PRINT "GO NORTHEAST":RETURN
300 PRINT "GO EAST"   :RETURN
400 PRINT "GO SOUTHEAST":RETURN
500 PRINT "GO SOUTH"  :RETURN
600 PRINT "GO SOUTHWEST":RETURN
700 PRINT "GO WEST"   :RETURN
800 PRINT "GO NORTHWEST":RETURN
```

# KEY

**Token:** \$F9

**Format:** **KEY [ ON | OFF | LOAD | SAVE | number, string]**

**Usage:** The function keys can either send their key code when pressed, or a string assigned to this key. After power up or reset this feature is activated and the keys have default assignments.

**KEY OFF:** switch off function key strings. The keys will send their character code if pressed.

**KEY ON:** switch on function key strings. The keys will send assigned strings if pressed.

**KEY LOAD:** loads key definitions from file.

**KEY SAVE:** saves key definitions to file.

**KEY:** list current assignments.

**KEY number, string** assigns the string to the key with that number.

Default assignments:

```
KEY
KEY 1,CHR$(27)+"X"
KEY 2,CHR$(27)+"@"
KEY 3,"DIR"+CHR$(13)
KEY 4,"DIR "+CHR$(34)+"*=PRG"+CHR$(34)+CHR$(13)
KEY 5,"@"
KEY 6,"KEY6"+CHR$(141)
KEY 7,"@"
KEY 8,"MONITOR"+CHR$(13)
KEY 9,"@"
KEY 10,"KEY10"+CHR$(141)
KEY 11,"@"
KEY 12,"KEY12"+CHR$(141)
KEY 13,CHR$(27)+"0"
KEY 14,"@"+CHR$(27)+"0"
KEY 15,"HELP"+CHR$(13)
KEY 16,"RUN "+CHR$(34)+"*"+CHR$(34)+CHR$(13)
```

**Remarks:** The sum of the lengths of all assigned strings must not exceed 240 characters. Special characters like RETURN or QUOTE are entered using their codes with the CHR\$(code) function.

**Example:** Using **KEY**:

```
KEY ON           :REM ENABLE FUNCTION KEYS
KEY OFF          :REM DISABLE FUNCTION KEYS
KEY              :REM LIST ASSIGNMENTS
KEY 2,"PRINT #"+CHR$(14) :REM ASSIGN PRINT PI TO F2
KEY SAVE "MY KEY SET"   :REM SAVE CURRENT DEFINITIONS TO FILE
KEY LOAD "ELEVEN-SET"    :REM LOAD DEFINITIONS FROM FILE
```

# LEFT\$

**Token:** \$C8

**Format:** **LEFT\$(string, n)**

**Usage:** Returns a string containing the first **n** characters from the argument **string**. If the length of **string** is equal or less than **n**, the result string will be identical to the argument string.

**string** = a string expression

**n** = a numeric expression (0 -> 255)

**Remarks:** Empty strings and zero lengths are legal values.

**Example:** Using **LEFT\$**:

```
PRINT LEFT$("MEGA-65",4)
```

```
MEGA
```

# LEN

**Token:** \$C3

**Format:** **LEN(string)**

**Usage:** Returns the length of the string.

**string** = a string expression

**Remarks:** There is no terminating character, like the NULL character in C programs. The length of the string is internally stored in an extra byte of the string descriptor.

**Example:** Using **LEN**:

```
PRINT LEN("MEGA-65"+CHR$(13))  
8
```

# **LET**

**Token:** \$88

**Format:** **LET variable = expression**

**Usage:** The **LET** statement is obsolete and not needed. Assignment to variables can be done without using **LET**.

**Example:** Using **LET**:

```
LET A=5 :REM LONGER AND SLOWER  
A=5      :REM SHORTER AND FASTER
```

# LINE

**Token:** \$E5

**Format:** LINE **xbeg,ybeg** [[,xnext1,ynext1], [...]]

**Usage:** Draws a pixel at (xbeg/ybeg), if only one coordinate pair is given.

If more than one pair is defined, a line is drawn on the current graphics screen from the coordinate (xbeg/ybeg) to the next coordinate pair(s).

All currently defined modes and values of the graphic context are used.

**Example:** Using LINE:

```
10 GRAPHIC CLR :REM INITIALISE
20 SCREEN DEF 1,1,1,2 :REM 640 X 400 X 2
30 SCREEN SET 1,1 :REM VIEW IT
40 SCNCLR 0 :REM CLEAR SCREEN
50 LINE 320,200 :REM PLOT PIXEL
60 LINE 50,50,590,350 :REM DRAW LINE
70 LINE 0,0,639,0,639,399,0,399,0,0 :REM DRAW RECT
```

# LINE INPUT#

**Token:** \$E5 \$84

**Format:** LINE INPUT# channel, variable list

**Usage:** Reads one record per variable from an input device, e.g. a disk file and assigns the read data to the variable. The records must be terminated by a **RETURN** character, which will not be copied to the string variable. An empty line consisting of the **RETURN** character only will therefore assign an empty string to the variable.

**channel** = channel number assigned by a **DOPEN** or **OPEN** command.

**variable list** = list of one or more variables, that receive the input.

The input record must be terminated by a RETURN character and must be not longer than the input buffer (160 characters).

**Remarks:** Only string variables or string array elements can be used in the variable list. Unlike other INPUT commands, LINE INPUT# does not interpret or remove quote characters in the input. They are accepted as data, as all other characters.

**Example:** Using **LINE INPUT#**:

```
10 DIM N$(100)
20 DOPEN#2,"DATA"
30 FOR I=0 TO 100
40 LINE INPUT#2,N$(I)
50 IF ST=64 THEN 80:REM END OF FILE
60 IF DS THEN PRINT DS$:GOTO 80:REM DISK ERROR
70 NEXT I
80 DCLOSE#2
110 PRINT "READ";I;" RECORDS"
```

# LIST

**Token:** \$9B

**Format:** (1) **LIST [line range]**

(2) **LIST filename [,U unit]**

**Usage:** (1) Used to list a range of lines from the BASIC program.

**line range** consist of the first and the last line to list or a single line number. If the first number is omitted, the first BASIC line is assumed. The second number in the range specifier defaults to the last BASIC line.

(2) Used to list a BASIC program directly from **unit**. **unit** defaults to 8.

**Remarks:** The **LIST** command's output can be redirected to other devices via the **CMD** command.

The keys **F9** and **F11** or **Ctrl P** and **Ctrl V** scroll a BASIC listing on screen up or down.

**Example:** Using **LIST**

```
LIST 100    :REM LIST LINE 100
LIST 240-350 :REM LIST ALL LINES FROM 240 TO 350
LIST 500-    :REM LIST FROM 500 TO END
LIST -70     :REM LIST FROM START TO 70
LIST "DEMO"   :REM LIST FILE "DEMO"
```

# LOAD

**Token:** \$93

**Format:** **LOAD filename [unit [,flag]]**

**Format:** **/ filename [ unit [,flag]]**

**Usage:** The shortcut symbol **/** can be used in direct mode only. A common use of the shortcut symbol **/** is to print a listing to screen, move the cursor to the desired line and type a **/** into the first column. After hitting **RETURN**, the listed file will be loaded. Characters before the file name and after the filename (PRG) will be ignored.

The **LOAD** loads a file of type PRG into RAM bank 0, which is also used for BASIC program source.

**filename** is either a quoted string, e.g. "**prog**" or a string expression.

The unit number ist optional. If not present, the default disk device is assumed.

If **flag** has a non zero value, the file is loaded to the address, which is read from the first two bytes of the file. Otherwise it is loaded to the start of BASIC memory and the load address in the file is ignored.

**Remarks:** This command is implemented in BASIC-10 to keep it backward compatible to BASIC-2.

**Example:** Using **LOAD**

```
LOAD "APOCALYPSE"  
LOAD "MEGA TOOLS",9  
LOAD "*",8,1
```

# LOADIFF

**Token:** \$FE \$43

**Format:** LOADIFF filename [unit ]

**Usage:** The **LOADIFF** loads an IFF file into graphics memory. The IFF (Interchange File Format) is supported by many different applications and operating systems. The **LOADIFF** command assumes a file that contains bitplane graphics, which fit into the MEGA65 graphics memory. Supported resolutions are:

| Width | Height | Bitplanes | Colours  | Memory     |
|-------|--------|-----------|----------|------------|
| 320   | 200    | max. 8    | max. 256 | max. 64 K  |
| 640   | 200    | max. 8    | max. 256 | max. 128 K |
| 320   | 400    | max. 8    | max. 256 | max. 128 K |
| 640   | 400    | max. 4    | max. 16  | max. 128 K |

**filename** is either a quoted string, e.g. "**picture.iff**" or a string expression.

The unit number ist optional. If not present, the default disk device is assumed.

**Remarks:** IFF files can be created or converted from other formats on several operating systems, like AMIGA OS, MAC OS, Linux and Windows. The tool **convert** is part of the free graphics package **ImageMagick**.

Example for converting a JPG file to an IFF file on Linux:

```
convert <myImage.jpg> <myImage.ppm>
ppmtoilbm -aga <myImage.pbm> > <myImage.iff>
```

**Example:** Using **LOADIFF**

```
100 BANK128:SCRCLR
110 REM DISPLAY PICTURES IN 320 X 200 X 7 RESOLUTION
120 GRAPHIC CLR:SCREEN DEF 0,0,0,7:SCREEN OPEN 0:SCREEN SET 0,0
130 FORT=1TO?: READF$ 
140 LOADIFF(F$+".IFF"):SLEEP 4:NEXT
150 DATA ALIEN,BEAKER,JOKER,PICARD,PULP,TROOPER,RIPLEY
160 SCREEN CLOSE 0
170 PALETTE RESTORE
```

# LOG

**Token:** \$BC

**Format:** **LOG(numeric expression)**

**Usage:** Computes the value of the natural logarithm of the argument. The natural logarithm uses Euler's number **e = 2.71828183** as base, not the number 10 which is typically used in log functions on a pocket calculator.

**Remarks:** The log function with base 10 can be computed by dividing the result by  $\log(10)$ .

**Example:** Using **LOG**

```
PRINT LOG(1)
0

PRINT LOG(0)
?ILLEGAL QUANTITY ERROR

PRINT LOG(4)
1.38629436

PRINT LOG(100) / LOG(10)
2
```

# LOG10

**Token:** \$CE \$08

**Format:** LOG10(numeric expression)

**Usage:** Computes the value of the decadal logarithm of the argument. The decadal logarithm uses 10 as base.

**Example:** Using LOG10

```
PRINT LOG10(1)
0

PRINT LOG10(0)
?ILLEGAL QUANTITY ERROR

PRINT LOG10(5)
0.69897

PRINT LOG10(100);LOG(10);LOG(0.1);LOG(0.01)
2 1 -1 -2
```

# LOOP

**Token:** \$EC

**Format:** **DO** ... **LOOP**

**DO** [ <**UNTIL** | **WHILE**> <logical expr.>]  
... statements [**EXIT**]  
**LOOP** [ <**UNTIL** | **WHILE**> <logical expr.>]

**Usage:** The **DO** and **LOOP** keywords define the start and end of the most versatile BASIC loop. Using **DO** and **LOOP** alone, without any modifiers creates an infinite loop, that can be left by the **EXIT** statement only. The loop can be controlled by adding an **UNTIL** or a **WHILE** statement after the **DO** or **LOOP**.

**Remarks:** **DO** loops may be nested. An **EXIT** statement exits the current loop only.

**Example:** Using **DO** and **LOOP**

```
10 PW$="":DO  
20 GET A$:PW$=PW$+A$  
30 LOOP UNTIL LEN(PW$)>? OR A$=CHR$(13)  
  
10 DO : REM WAIT FOR USER DECISION  
20 GET A$  
30 LOOP UNTIL A$='Y' OR A$='N' OR A$='y' OR A$='n'  
  
10 DO WHILE ABS(EPS) > 0.001  
20 GOSUB 2000 : REM ITERATION SUBROUTINE  
30 LOOP  
  
10 IX=0 : REM INTEGER LOOP 1 -> 100  
20 DO IX=IX+1  
30 LOOP WHILE IX < 101
```

# LPEN

**Token:** \$CE \$04

**Format:** LPEN(**coordinate**)

**Usage:** This function requires the use of a CRT monitor or TV and a light pen. It will not work with a LCD or LED screen. The light pen must be connected to port 1.

**LPEN(0)** returns the X position of the light pen, the range is 60 -> 320.

**LPEN(1)** returns the Y position of the light pen, the range is 50 -> 250.

**Remarks:** The X resolution is two pixels, **LPEN(0)** returns therefore only even numbers. A bright background colour is needed to trigger the light pen. The **COLLISION** statement may be used to install an interrupt handler.

**Example:** Using **LPEN**

```
PRINT LPEN(0),LPEN(1) :REM PRINT LIGHT PEN COORDINATES
```

# MERGE

**Token:** \$E6

**Format:** **MERGE filename [,D drive] [,U unit]**

**Usage:** **MERGE** loads a BASIC program file from disk and appends it to the program in memory.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. **(FI\$)**

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The load address, stored in the first two bytes of the file is ignored. The loaded program does not replace a program in memory, like **DLOAD** does, but is appended to a program in memory. After loading the program is re-linked and ready to run or edit.

It is in the user's responsibility, to ensure, that there is no line number conflict among the program in memory and the merged program. The first line number of the merged program must be greater, than the last line number of the program in memory.

**Example:** Using **MERGE**

```
DLOAD "MAIN PROGRAM"  
MERGE "LIBRARY"
```

# MID\$

**Token:** \$CA

**Format:** **variable\$ = MID\$(string, index, n)**  
**MID\$(string, index, n) = string expression**

**Usage:** **MID\$** can be used either as a function, which returns a string or as a statement for inserting sub-strings into an existing string.

**string** = a string expression

**index** = start index (0 -> 255)

**n** = length of sub-string (0 -> 255)

**Remarks:** Empty strings and zero lengths are legal values.

**Example:** Using **MID\$**:

```
10 A$ = "MEGA-65"
20 PRINT MID$(A$,3,4)
30 MID$(A$,5,1) = "+"
40 PRINT A$
RUN
GA-6
MEGA+65
```

# MOD

**Token:** \$NN

**Format:** **MOD(dividend,divisor)**

**Usage:** The **MOD** function returns the remainder of the division.

**Remarks:** In other programming languages, like C, this function is implemented as an operator. Here it is used as function.

**Example:** Using **MOD**:

```
FOR I = 0 TO 8: PRINT MOD(I,4);: NEXT I  
0 1 2 3 0 1 2 3 0
```

# MONITOR

**Token:** \$FA

**Format:** MONITOR

**Usage:** Calls the machine language monitor program, which is mainly used for debugging.

**Remarks:** Using the **MONITOR** requires knowledge of the CSG4510 / 6502 / 6510 CPU, the assembler language and internals.

See monitor chapter in Enhanced Machine Language Monitor for details.

To exit the monitor type **X**.

Display help text with **?**.

**Example:** Using **MONITOR**:

```
MONITOR
BS MONITOR COMMANDS:ABCDEFHJMRTUXC.,;?:+&%'LSU
PC  SR  AC  XR  YR  ZR  BP   SP  NVEBDIZC
; 00FFA2 00 00 00 00 00 00 01F8 -----
ASSEMBLE - A ADDRESS MNEMONIC OPERAND
BLIMPS  - B [FROM]
COMPARE - C FROM TO WITH
DISASSEMBLE - D [FROM [TO]]
FILL    - E FROM TO FILLBYTE
GO     - G [ADDRESS]
HUNT   - H FROM TO (STRING OR BYTES)
JSR    - J ADDRESS
LOAD   - L FILENAME [UNIT [ADDRESS]]
MEMORY - M [FROM [TO]]
REGISTERS - R
SAVE   - S FILENAME UNIT FROM TO
TRANSFER - T FROM TO TARGET
VERIFY  - U FILENAME [UNIT [ADDRESS]]
EXIT   - X
.DOT   - . ADDRESS MNEMONIC OPERAND
.GREATER> - > ADDRESS BYTE SEQUENCE
;SEMICOLON> - ; REGISTER CONTENTS
.DOS   - @ EDOS COMMAND
?HELP  - ?
```

# MOUSE

**Token:** \$FE \$3E

**Format:** **MOUSE ON [,port [,sprite [,pos]]]**  
**MOUSE OFF**

**Usage:** Enables the mouse driver and connects the mouse at the specified port with the mouse pointer sprite.

**port** = mouse port 1, 2 (default) or 3 (both).

**sprite** = sprite number for mouse pointer (default 0).

**pos** = initial mouse position (x,y).

The **MOUSE OFF** command disables the mouse driver and frees the associated sprite.

**Remarks:** The "hot spot" of the mouse pointer is the upper left pixel of the sprite.

**Example:** Using **MOUSE**:

```
REM LOAD DATA INTO SPRITE #0 BEFORE USING IT
MOUSE ON, 1      :REM ENABLE MOUSE WITH SPRITE #0
MOUSE OFF        :REM DISABLE MOUSE
```

# MOVSPR

**Token:** \$FE \$06

**Format:** **MOVSPR number, position**

**MOVSPR number, start-position TO end-position, speed**

**Usage:** Each position argument consists of two 16 bit values, which specify either an absolute coordinate, a relative coordinate, an angle or a speed. The value type is determined by a prefix:

+value = relative coordinate: positive offset  
-value = relative coordinate: negative offset  
#value = speed  
no prefix = absolute coordinate or angle

So the position argument can be used to set the sprite to an absolute position on screen, to specify a displacement from the current position or describe a movement with angle and speed.

The first format **MOVSPR number, position** is used to set the sprite immediately to the position or, in case of an angle#speed argument, describe its further movement.

The second format **MOVSPR number, start-position TO end-position, speed** puts the sprite into the start position and defines the destination position and the speed of movement. The sprite is put immediately to its start position and will move on a straight line to the destination at the given speed. This second format can use absolute or relative coordinates only. The angle#speed coordinate is restricted to the first format. The movement is controlled by the BASIC interrupt handler and happens concurrently with the program execution.

**number** = sprite number (0-7)

**position** = x,y | xrel,y | x,yrel | xrel,yrel | angle#speed

**x** = absolute screen coordinate [pixel].

**y** = absolute screen coordinate [pixel].

**xrel** = relative screen coordinate [pixel].

**yrel** = relative screen coordinate [pixel].

**angle** = compass direction for sprite movement [degrees]. 0 = up, 90 = right, 180 = down, 270 = left, 45 upper right, etc.

**speed** = speed of movement, a floating point number in the range (0.0 to 127.0) uses the the unit (pixel/frame). PAL mode has 50 frames per second and NTSC 60 frames per second. A speed value of 1.0 will move the sprite 50 pixels per second in PAL mode.

**Remarks:** The "hot spot" is the upper left pixel of the sprite.

**Example:** Using **MOVSPR**:

```
100 CLR:SCRCLR:SPRITECLR  
110 BLOAD "DEMOsprites1",B0,P1536  
130 FORI=0T07: C=I+1:SP=0.07*(I+1)  
140 MOVSPRI, 160,120  
145 MOVSPRI,45*I#SP  
150 SPRITEI,1,C,,0,0  
160 NEXT  
170 SLEEP 3  
180 FORI=0T07:MOVSPR I,0#0:NEXT
```



# **NEW**

**Token:** \$A2

**Format:** **NEW**

**NEW RESTORE**

**Usage:** Resets all BASIC parameters to their default values. After **NEW** the maximum RAM is available for program and data storage.

Because **NEW** resets parameters and pointers, but does not physically overwrite the address range of a BASIC program, that was in memory before **NEW**, it is possible to recover the program. If there were no **LOAD** operations or editing after the **NEW** command, the program can be restored with the command

**NEW RESTORE.**

**Example:** Using **NEW**:

```
NEW      :REM RESET BASIC
NEW RESTORE :REM TRY TO RECOVER NEW'ED PROGRAM
```

# NEXT

**Token:** \$82

**Format:** **FOR index=start TO end [STEP step] ... NEXT [index]**

**Usage:** Terminates the definition of a BASIC loop with an index variable.

The **index** variable may be incremented or decremented by a constant value **step** on each iteration. The default is to increment the variable by 1. The index variable must be a real variable.

The **start** value is used to initialise the index.

The **end** value is used at the end of the loop and controls, whether the next iteration will be started or the loop exited.

The **step** value defines the change applied to the index variable at the end of the loop. Positive step values increment it, while negative values decrement it. It defaults to 1.0 if not specified.

**Remarks:** The **index** variable after **NEXT** is optional. If it is missing, the variable for the current loop is assumed. Several consecutive **NEXT** statements may be combined by specifying the indexes in a comma separated list. The statements **NEXT I:NEXT J:NEXT K** and **NEXT I,J,K** are equivalent.

**Example:** Using **NEXT**

```
10 FOR D=0 TO 360 STEP 30
20 R = D * π / 180
30 PRINT D;R;SIN(R);COS(R);TAN(R)
40 NEXT D

10 DIM M(20,20)
20 FOR I=0 TO 20
30 FOR J=I TO 20
40 M(I,J) = I + 100 * J
50 NEXT J,I
```

# NOT

**Token:** \$A8

**Format:** NOT operand

**Usage:** Performs a bit-wise logical NOT operation on a 16 bit value. Integer operands are used as they are. Real operands are converted to a signed 16 bit integer. Logical operands are converted to 16 bit integer using \$FFFF, decimal -1 for TRUE and \$0000, decimal 0, for FALSE.

```
NOT 0  ->  1  
NOT 1  ->  0
```

**Remarks:** The result is of integer type. If the result is used in a logical context, the value of 0 is regarded as FALSE, all other, nonzero values are regarded as TRUE.

**Example:** Using NOT

```
PRINT NOT 3  
-4  
PRINT NOT 64  
-65
```

In most cases the **NOT** will be used in **IF** statements.

```
OK = C < 256 AND C >= 0  
IF (NOT OK) THEN PRINT "NOT A BYTE VALUE"
```

# **OFF**

**Token:** \$FE \$24

**Format:** keyword **OFF**

**Usage:** **OFF** is a secondary keyword used in combination with primary keywords like **COLOR, KEY, MOUSE**.

**Remarks:** The keyword **OFF** cannot be used on its own.

**Example:** Using **OFF**

```
COLOR OFF :REM DISABLE SCREEN COLOUR  
KEY OFF   :REM DISABLE FUNCTION KEY STRINGS  
MOUSE OFF :REM DISABLE MOUSE DRIVER
```

# ON

**Token:** \$91

**Format:** **ON expression GOSUB line list**  
**ON expression GOTO line list**  
keyword **ON**

**Usage:** The **ON** keyword starts either a computed **GOSUB** or **GOTO** statement. Dependent on the value of the expression, the target for the **GOSUB** or **GOTO** is chosen from the table of line addresses at the end of the statement.

As a secondary keyword, **ON** is used in combination with primary keywords like **COLOR**, **KEY**, **MOUSE**.

**expression** is a positive numeric value. Real values are cut to integer.

**line list** is a comma separated list of valid line numbers.

**Remarks:** Negative values for **expression** will stop the program with an error message. The **line list** specifies the targets for values of 1,2,3,... An expression value of zero or a value, that is greater than the number of target lines will do nothing and continue program execution with the next statement.

**Example:** Using **ON**

```
10 COLOR ON :REM ENABLE SCREEN COLOUR
20 KEY ON :REM ENABLE FUNCTION KEY STRINGS
30 MOUSE ON :REM ENABLE MOUSE DRIVER
40 N = JOY(1):IF N AND 128 THEN PRINT "FIRE! ";
60 REM           N  NE  E  SE  S  SW  W  NW
70 ON N AND 15 GOSUB 100,200,300,400,500,600,700,800
80 GOTO 40
100 PRINT "GO NORTH"   :RETURN
200 PRINT "GO NORTHEAST":RETURN
300 PRINT "GO EAST"    :RETURN
400 PRINT "GO SOUTHEAST":RETURN
500 PRINT "GO SOUTH"   :RETURN
600 PRINT "GO SOUTHWEST":RETURN
700 PRINT "GO WEST"    :RETURN
800 PRINT "GO NORTHWEST":RETURN
```

# OPEN

**Token:** \$9F

**Format:** **OPEN lfn, first address [,secondary address [,filename]]**

**Usage:** Opens an input/output channel for a device.

**lfn** = logical file number

1 <= lfn <= 127: line terminator is CR

128 <= lfn <= 255: line terminator is CR LF

**first address** = device number. For IEC devices the unit number is the primary address. Following primary address values are possible:

| unit | device                  |
|------|-------------------------|
| 0    | Keyboard                |
| 1    | System default          |
| 2    | RS232 serial connection |
| 3    | Screen                  |
| 4-7  | IEC printer and plotter |
| 8-31 | IEC disk drives         |

The **secondary address** has some special values for IEC disk units, 0:load, 1:save, 15:command channel. The values 2 -> 14 may be used for disk files.

**filename** is either a quoted string, e.g. "**data**" or a string expression. The syntax is different to the **DOPEN#** command. The **filename** for **OPEN** includes all file attributes, e.g.: "0:data,s,w".

**Remarks:** For IEC disk units the usage of **DOPEN#** is recommended.

The usage of the "save-and-replace" character '@' at the beginning of the filename is not recommended, because many Commodore disk drives have a bug, that can cause data loss when using this feature.

**Example:** Using **OPEN**

```
OPEN 4,4 :REM OPEN PRINTER
CMD 4 :REM REDIRECT STANDARD OUTPUT TO 4
LIST :REM PRINT LISTING ON PRINTER DEVICE 4
OPEN 3,8,3,"0:USER FILE,U"
OPEN 2,9,2,"0:DATA,S,W"
```

# OR

**Token:** \$B0

**Format:** operand **OR** operand

**Usage:** Performs a bit-wise logical OR operation on two 16-bit values. Integer operands are used as they are. Real operands are converted to a signed 16-bit integer. Logical operands are converted to 16-bit integer using \$FFFF, decimal -1 for TRUE and \$0000, decimal 0, for FALSE.

```
0 OR 0  -> 0
0 OR 1  -> 1
1 OR 0  -> 1
1 OR 1  -> 1
```

**Remarks:** The result is of integer type. If the result is used in a logical context, the value of 0 is regarded as FALSE, all other, nonzero values are regarded as TRUE.

**Example:** Using **OR**

```
PRINT 1 OR 3
3
PRINT 128 OR 64
192
```

In most cases the **OR** will be used in **IF** statements.

```
IF (C < 0 OR C > 255) THEN PRINT "NOT A BYTE VALUE"
```

# PAINT

**Token:** \$DF

**Format:** PAINT x, y, mode [,colour]

**Usage:** Performs a flood fill of an enclosed graphics area.

**x, y** is a coordinate pair, which must lie inside the area to be filled.

**mode** specifies the fill mode.

0: use the **colour** to fill the area.

1: use the colour of pixel (x,y) to fill the area.

**Example:** Using PAINT

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 1,0,0,2 :REM 320 X 200
30 SCREEN OPEN 1     :REM OPEN
40 SCREEN SET 1,1    :REM MAKE SCREEN ACTIVE
50 LINE 160,0,240,100 :REM 1ST. LINE
60 LINE 240,100,80,100 :REM 2ND. LINE
70 LINE 80,100,160,0  :REM 3RD. LINE
80 PAINT 160,10,0,1   :REM FILL TRIANGLE WITH COLOUR 1
90 GETKEY K$         :REM WAIT FOR KEY
100 SCREEN CLOSE 1    :REM END GRAPHICS
```

# PALETTE

**Token:** \$FE \$34

**Format:** PALETTE [screen]COLOR], colour, red, green, blue  
PALETTE RESTORE

**Usage:** The **PALETTE** command can be used to change an entry of the system colour palette or the palette of a screen.

**PALETTE RESTORE** resets the system palette to the default values.

**screen** = screen number 0 -> 3.

**COLOR** = keyword for changing system palette.

**colour** = index to palette 0 -> 255.

**red** = red intensity 0 -> 15.

**green** = green intensity 0 -> 15.

**blue** = blue intensity 0 -> 15.

**Example:** Using **PALETTE**

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 1,0,0,2 :REM 320 X 200
30 SCREEN OPEN 1     :REM OPEN
40 SCREEN SET 1,1     :REM MAKE SCREEN ACTIVE
50 PALETTE 1,0, 0, 0, 0 :REM 0 = BLACK
60 PALETTE 1,1, 15, 0, 0 :REM 1 = RED
70 PALETTE 1,2, 0, 0,15 :REM 2 = BLUE
80 PALETTE 1,3, 0,15, 0 :REM 3 = GREEN
90 LINE 160,0,240,100   :REM 1ST. LINE
100 LINE 240,100,80,100 :REM 2ND. LINE
110 LINE 80,100,160,0   :REM 3RD. LINE
120 PRINT 160,10,0,2    :REM FILL TRIANGLE WITH BLUE (2)
130 GETKEY K$          :REM WAIT FOR KEY
140 SCREEN CLOSE 1      :REM END GRAPHICS
```

# PEEK

**Token:** \$C2

**Format:** PEEK(address)

**Usage:** Returns an unsigned 8 bit value (byte) read from address.

If the address is in the range (\$0000 to \$FFFF) the memory bank (set by **BANK**) is used.

Addresses >= \$10000 are assumed to be flat memory addresses and used as such, ignoring the bank setting.

**Remarks:** Banks 0 -> 127 give access to RAM or ROM banks. Banks > 127 are used to access I/O and SYSTEM like VIC, SID, FDC, etc.

**Example:** Using PEEK

```
10 BANK 128          :REM SELECT SYSTEM BANK
20 L = PEEK($02F8)    :REM USR JUMP TARGET LOW
30 H = PEEK($02F9)    :REM USR JUMP TARGET HIGH
40 T = L + 256 * H   :REM 16 BIT JUMP ADDRESS
50 PRINT "USR FUNCTION CALLS ADDRESS";T
```

# PEEKW

**Token:** \$C2 'W'

**Format:** **PEEKW(address)**

**Usage:** Returns an unsigned 16 bit value (word) read from address (low byte) and address+1 (high byte).

If the address is in the range (\$0000 to \$FFFF) the memory bank (set by **BANK**) is used.

Addresses >= \$10000 are assumed to be flat memory addresses and used as such, ignoring the bank setting.

**Remarks:** Banks 0 -> 127 give access to RAM or ROM banks. Banks > 127 are used to access I/O and SYSTEM like VIC, SID, FDC, etc.

**Example:** Using **PEEKW**

```
20 UA = PEEKW($02F8) :REM USR JUMP TARGET  
50 PRINT "USR FUNCTION CALL ADDRESS";UA
```

# PEN

**Token:** \$FE \$33

**Format:** **PEN pen colour**

**Usage:** Sets the colour for the graphic pen.

**pen** = pen number ( 0 -> 2 )

pen 0 : drawing pen

pen 1 : off bits in jam2 mode

pen 2 : currently unused

**colour** = palette index.

**Remarks:** **PEN** defined colours are used by all following drawing commands. For setting the drawing pen 0, the first parameter (0) may be omitted.

**Example:** Using **PEN**

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 1,0,0,2 :REM 320 X 200
30 SCREEN OPEN 1     :REM OPEN
40 SCREEN SET 1,1     :REM MAKE SCREEN ACTIVE
50 PALETTE 1,0, 0, 0, 0 :REM 0 = BLACK
60 PALETTE 1,1, 15, 0, 0 :REM 1 = RED
70 PALETTE 1,2, 0, 0,15 :REM 2 = BLUE
80 PALETTE 1,3, 0,15, 0 :REM 3 = GREEN
90 PEN 1             :REM PEN 0 = 1 RED
100 LINE 160,0,240,100 :REM DRAW RED LINE
110 PEN 2             :REM PEN 0 = 2 BLUE
120 LINE 240,100,80,100 :REM DRAW BLUE LINE
130 PEN 3             :REM PEN 0 = 3 GREEN
140 LINE 80,100,160,0   :REM DRAW GREEN LINE
150 GETKEY K$          :REM WAIT FOR KEY
160 SCREEN CLOSE 1    :REM END GRAPHICS
```

# **PIXEL**

**Token:** \$CE \$0C

**Format:** **PIXEL(x,y)**

**Usage:** Returns colour at given position.

**x** = absolute screen coordinate [pixel].

**y** = absolute screen coordinate [pixel].

# PLAY

**Token:** \$FE \$04

**Format:** **PLAY string**

**Usage:** Starts playing a tune with notes and directives embedded in the argument string.

A musical note is a letter (A,B,C,D,E,F,G) which may be preceded by an optional modifier.

Possible modifiers are:

| char | effect         |
|------|----------------|
| #    | sharp          |
| \$   | flat           |
| .    | dotted         |
| H    | half note      |
| I    | eighth note    |
| M    | wait for end   |
| Q    | quarter note   |
| R    | pause (rest)   |
| S    | sixteenth note |
| W    | whole note     |

Embedded directives consist of a letter followed by a digit:

| char | directive     | argument range |
|------|---------------|----------------|
| O    | octave        | 0 - 6          |
| T    | tune envelope | 0 - 9          |
| U    | volume        | 0 - 9          |
| V    | voice         | 1 - 3          |
| X    | filter        | 0 - 1          |

The envelope slots may be changed using the **ENVELOPE** statement. The default setting for the envelopes are:

| n | A  | D | S  | R | WF | PW   | Instrument  |
|---|----|---|----|---|----|------|-------------|
| 0 | 0  | 9 | 0  | 0 | 2  | 1536 | piano       |
| 1 | 12 | 0 | 12 | 0 | 1  |      | accordion   |
| 2 | 0  | 0 | 15 | 0 | 0  |      | calliope    |
| 3 | 0  | 5 | 5  | 0 | 3  |      | drum        |
| 4 | 9  | 4 | 4  | 0 | 0  |      | flute       |
| 5 | 0  | 9 | 2  | 1 | 1  |      | guitar      |
| 6 | 0  | 9 | 0  | 0 | 2  | 512  | harpsichord |
| 7 | 0  | 9 | 9  | 0 | 2  | 2048 | organ       |
| 8 | 8  | 9 | 4  | 1 | 2  | 512  | trumpet     |
| 9 | 0  | 9 | 0  | 0 | 0  |      | xylophone   |

**Remarks:** The **PLAY** statement sets up an interrupt driven routine that starts parsing the string and playing the tune. The execution continues with the next statement with no need waiting for the tune to be finished. However this can be forced, using the 'M' modifier.

**Example:** Using **PLAY**

```
10 ENVELOPE 9,10,5,10,5,2,4000
20 PLAY "T9"
30 VOL 8
40 TEMPO 100
50 PLAY "C D E F G A B"
60 PLAY "U5 V1 C D E F G A B"
```

# POINTER

**Token:** \$CE \$0A

**Format:** **POINTER(variable)**

**Usage:** Returns the current address of a variable or an array element. For string variables, it is the address of the string descriptor, not the string itself. The string descriptor consists of the three bytes (length,string address low, string address high).

Address values  $\geq \$F700$  are assigned to bank 0. All other addresses are assigned to bank 1.

**Remarks:** The address values of arrays and their elements are constant during a program execution.

The addresses of strings (not their descriptors) however may change at any time due to "garbage collection" in memory management.

**Example:** Using **POINTER**

```
10 H$="HELLO"
20 P=POINTER(H$):PRINT "descriptor AT: $";HEX$(P)
30 IF P> DEC("F700") THENBANK 0:ELSEBANK 1
40 L=PEEK(P):SP=PEEK(P+1)+256*PEEK(P+2)
50 PRINT"string address:$";HEX$(SP)
60 PRINT "length:";L
70 BANK 1:REM STRING BANK
80 FOR I=1TO L:PRINT PEEK(SP+I-1);:NEXT:PRINT
90 FOR I=1TO L:PRINT CHR$(PEEK(SP+I-1));:NEXT:PRINT

RUN
descriptor AT: $F702
string address:$F6F9
length= 5
72 69 76 76 79
HELLO
```

# POKE

**Token:** \$97

**Format:** **POKE address, byte [,byte ...]**

**Usage:** Puts one or more bytes into memory or memory mapped I/O, starting at **address**.

If the address is in the range (\$0000 to \$FFFF) the memory bank (set by **BANK**) is used.

Addresses >= \$10000 are assumed to be flat memory addresses and used as such, ignoring the bank setting.

**byte** = a value 0 -> 255.

**Remarks:** The address is increased by one for each data byte, so a memory range may be filled with a single command.

Banks > 127 are used to access I/O and SYSTEM like VIC, SID, FDC, etc.

**Example:** Using **POKE**

```
10 BANK 128      :REM SELECT SYSTEM BANK
20 POKE $02F8H,0,24 :REM SET USR VECTOR TO $1800
```

# POKEW

**Token:** \$97 'W'

**Format:** **POKEW address, word [,word ...]**

**Usage:** Puts on or more words into memory or memory mapped I/O, starting at **address**.

If the address is in the range (\$0000 to \$FFFF) the memory bank (set by **BANK**) is used.

Addresses >= \$10000 are assumed to be flat memory addresses and used as such, ignoring the bank setting.

**word** = a value 0 -> 65535

The first word is stored at address (low byte) and address+1 (high byte). The second one at address+2 (low byte) and address+3 (high byte), etc.

**Remarks:** The address is increased by two for each data word, so a memory range may be filled with a single PEEKW command.

Banks > 127 are used to access I/O and SYSTEM like VIC, SID, FDC, etc.

**Example:** Using **POKEW**

```
10 BANK 128      :REM SELECT SYSTEM BANK
20 POKEW $02F8,$1800  :REM SET USR VECTOR TO $1800
```

# POLYGON

**Token:** \$FE \$2F

**Format:** **POLYGON** **x, y, xrad, yrad, solid, angle, sides, n**

**Usage:** Draws a regular **n** sided polygon. The polygon is drawn using the current drawing context set with SCREEN, PALETTE and PEN.

**x,y** = centre coordinates.

**xrad,yrad** = radius in x- and y-direction.

**solid** = fill (1) or outline (0).

**angle** = start angle.

**sides** = sides to draw ( $\leq n$ ).

**n** = number of sides or edges.

**Remarks:** A regular polygon is both isogonal and isotonal, meaning all sides and angles are alike.

**Example:** Using **POLYGON**

```
POLYGON 320,100,50,50,0,0,6,6
```



# **POS**

**Token:** \$B9

**Format:** **POS(dummy)**

**Usage:** Returns the cursor column relative to the currently used window.

**dummy** = a numeric value, which is ignored.

**Remarks:** **POS** gives the column position for the screen cursor. It will not work for redirected output.

**Example:** Using **POS**

```
10 IF POS(0) > 72 THEN PRINT :REM INSERT RETURN
```

# POT

**Token:** \$CE \$02

**Format:** POT(**paddle**)

**Usage:** Returns the position of a paddle.

**paddle** = paddle number 1 -> 4.

The low byte of the return value is the paddle value with 0 at the clockwise limit and 255 at the counterclockwise limit.

A value > 255 indicates the simultaneous press of the fire button.

**Remarks:** Analogue paddles are noisy and inexact. The range may be less than 0 - 255 and there is some jitter in the data.

**Example:** Using POT

```
10 X = POT(1)      : REM READ PADDLE #1
20 B = X > 255    : REM TRUE (-1) IF FIRE BUTTON IS PRESSED
30 V = X AND 255  : PADDLE #1 VALUE
```

# PRINT

**Token:** \$99

**Format:** **PRINT arguments**

**Usage:** Evaluates the argument list and prints the values formatted to the current screen window. Standard formatting is used dependent on the argument type. For user controlled formatting see **PRINT USING**. Following argument types are processed:

**numeric** : The printout starts with a space for positive and zero values or a minus sign for negative values. Integer values are printed with the necessary number of digits. Real values are printed either in fixed point format with typically 9 digits or in scientific format, if the value is outside the range of 0.01 -> 999999999.

**string** : The string may consist of printable characters and control codes. Printable characters are printed to the cursor position, while control codes are executed.

, : A comma acts like a tabulator.

; : A semicolon acts as a separator between arguments of the list. Other than the comma character it does not put in any additional characters. A semicolon at the end of the argument list suppresses the automatic return character.

**Remarks:** The **SPC** and **TAB** functions may be used in the argument list for positioning. The **CMD** command can be used for redirection.

**Example:** Using **PRINT**

```
10 FOR I=1 TO 10    : REM START LOOP
20 PRINT I,I*I,SQR(I)
30 NEXT
```

# PRINT#

**Token:** \$98

**Format:** PRINT# **channel**, **arguments**

**Usage:** Evaluates the argument list and prints the values formatted to the device assigned to **channel**. Standard formatting is used dependent on the argument type. For user controlled formatting see **PRINT# USING**. Following argument types are processed:

**channel** : must be opened for output by an **OPEN** or **DOPEN** statement.

**numeric** : The printout starts with a space for positive and zero values or a minus sign for negative values. Integer values are printed with the necessary number of digits. Real values are printed either in fixed point format with typically 9 digits or in scientific format, if the value is outside the range of 0.01 -> 999999999.

**string** : The string may consist of printable characters and control codes. Printable characters are printed to the cursor position, while control codes are executed.

, : A comma acts like a tabulator.

; : A semicolon acts as a separator between arguments of the list. Other than the comma character it does not put in any additional characters. A semicolon at the end of the argument list suppresses the automatic return character.

**Remarks:** The **SPC** and **TAB** functions are not suitable for devices other than the screen.

**Example:** Using **PRINT#**

```
10 DOPEN#2,"TABLE",W,U9
20 FOR I=1 TO 10    : REM START LOOP
30 PRINT#2,I,I*I,SQR(I)
40 NEXT
50 DCLOSE#2
```

# PRINT USING

**Token:** \$98 \$FB or \$99 \$FB

**Format:** PRINT [# channel,] USING format;argument

**Usage:** Parses the format string and evaluates the argument. The argument can be either a string or a numeric value. The formatting of the resulting output is directed by the format string.

**channel** : must be opened for output by an **OPEN** or **DOPEN** statement. If no channel is specified, the output goes to the screen.

**format** : A string variable or a string constant which defines the rules for formatting.

**numeric argument** : The numeric formatting can be done in either CBM style providing a pattern like "###.##" or in C style using width.precision specifier like %3D %7.2F %4X for example.

If the argument does not fit into the format e.g. trying to print a 4 digit variable into "###" a series of asterisks fills the format character.

**string argument** : The string may consist of printable characters and control codes. Printable characters are printed to the cursor position, while control codes are executed. The number of '#' characters sets the width of the output. If the first character of the format string is a '=' sign, the argument string is centered within the width. If the first character of the format string is a ' '> sign, the argument string is right justified within the width.

**Remarks:** The format string is applied for one argument only. But it is possible to append more than one USING format;argument sequence.

**Example:** Using **PRINT# USING**

```
PRINT USING "##.###";#, USING "1%6.4F";SQR(2)
3.14 [1.4142]

PRINT USING "(< # # # >)";12*31
< 3 7 2 >

PRINT USING "####"; "ABCDE"
ABC

PRINT USING ">####"; "ABCDE"
CDE

PRINT USING "ADDRESS:$%4X";65000
ADDRESS:$FDE8

A$="####,####,####.##":PRINT USING A$;1E8/3
33,333,333.3
```

# **RCOLOR**

**Token:** \$CD

**Format:** **RCOLOR(**colour source**)**

**Usage:** Returns the current colour index for the selected colour source.

Colour sources are:

- 0: background colour (VIC \$D021)
- 1: text colour (\$F1)
- 2: highlight colour (\$2D8)
- 3: border colour (VIC \$D020)

**Example:** Using **RCOLOR**

```
10 C = RCOLOR(3) : REM C = colour index of border colour
```

# RCURSOR

**Token:** \$FE \$42

**Format:** **RCURSOR [colvar],[rowvar]**

**Usage:** Returns the current cursor column and cursor row.

**Remarks:** The row and column counts start at zero, the left most column is 0 and the top row is 0.

**Example:** Using **RCURSOR**

```
100 CURSOR ON,20,10  
110 PRINT "[HERE]";  
120 RCURSOR X,Y  
130 PRINT " COL:",X;" ROW:",Y
```

RUN

```
[HERE] COL: 26  ROW: 10
```

# READ

**Token:** \$87

**Format:** READ **variable list**

**Usage:** Reads values from program source into variables.

**variable list** = any legal variables.

All type of constants (integer, real, strings) can be read, but no expressions. Items are separated by commas. Strings containing commas, colons or spaces must be put in quotes.

A **RUN** command initialises the data pointer to the first item of the first **DATA** statement and advances it for every read item. It is in the responsibility of the programmer, that the type of the constant and the variable in the **READ** statement match. Empty items with no constant between commas are allowed and will be interpreted as zero for numeric variables and an empty string for string variables.

The **RESTORE** command may be used to set the data pointer to a specific line for subsequent readings.

**Remarks:** It is good programming style to put large amount of **DATA** statements at the end of the program. Otherwise **GOTO** and **GOSUB** statements, with target lines lower than the current one, start their search for linenumbers at the beginning of the program and have to skip through **DATA** lines wasting time.

**Example:** Using **READ**

```
10 READ NA$, VE
20 READ NX:FOR I=2 TO NX:READ GL(I):NEXT I
30 PRINT "PROGRAM: ";NA$;" VERSION: ";VE
40 PRINT "N-POINT GAUSS-LEGENDRE FACTORS E1":
50 FOR I=2 TO NX:PRINT I;GL(I):NEXT I
60 STOP
80 DATA "MEGA65",1,1
90 DATA 5,0.5120,0.3573,0.2760,0.2252
```

# RECORD

**Token:** \$FE \$12

**Format:** RECORD#**lfn**, record, [**byte**]

**Usage:** Positions the read/write pointer of a relative file.

**lfn** = logical file number

**record** = target record ( 1 -> 65535 ).

**byte** = byte position in record.

This command can be used only for files of type **REL**, which are relative files capable of direct access.

The **RECORD** command positions the file pointer to the specified record number. If this record number does not exist and the disk capacity is high enough, the file is expanded to this record count by adding empty records. This is not an error, but the disk status will give the message **RECORD NOT PRESENT**.

Any INPUT# or PRINT# command will then proceed on the selected record position.

**Remarks:** The original Commodore disk drives all had a bug in their DOS, which could destroy data by using relative files. A recommended workaround was to issue each **RECORD** command twice, before and after the I/O operation.

**Example:** Using **RECORD**

```
10 REM *** READ FIRST 10 INDEXED RECORDS FROM DATA BASE
15 N = 1000: DIM IX(N)
20 DOPEN#3,"DATA INDEX"
25 FOR I=1 TO N:INPUT#3,IX(I):NEXT
30 DCLOSE#3
35 DOPEN#2,"DATA BASE",L240
40 FOR J=1 TO 10
45 RECORD#2,IX(J)
50 INPUT#2,A$
55 PRINT A$
60 NEXT J
65 DCLOSE#2
```

# **REM**

**Token:** \$8F

**Format:** **REM**

**Usage:** Marks the rest of the line as comment.

All characters after **REM** are never executed but skipped.

**Example:** Using **REM**

```
10 REM *** PROGRAM TITLE ***
20 N=1000 :REM NUMBER OF ITEMS
30 DIM NA$(N)
```

# RENAME

**Token:** \$F5

**Format:** **RENAME old TO new [D drive] [,U unit]**

**Usage:** Renames a disk file.

**old** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**).

**new** is either a quoted string, e.g. "**backup**" or a string expression in parentheses, e.g. (**FS\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** The **RENAME** command is executed in the DOS of the disk drive. It can rename all regular file types (PRG, SEQ, USR, REL). The old file must exist, the new file must not exist. Only single files can be renamed, wild characters like '\*' and '?' are not allowed. The file type cannot be changed.

**Example:** Using **RENAME**

```
RENAME "CODES" TO "BACKUP" :REM RENAME SINGLE FILE
```

# RENUMBER

**Token:** \$F8

**Format:** **RENUMBER [new [,inc [range]]]**

**Usage:** Used to renumber all or a range of lines of a BASIC program.

**new** is the new starting line of the line range to renumber. The default value is 10.

**inc** is the increment to be used. The default value is 10.

**range** is the line range to renumber. The default values are from first to last line.

The **RENUMBER** changes all line numbers in the chosen range and also changes all references from statements like **GOTO**, **GOSUB**, **TRAP**, **RESTORE**, **RUN** etc.

**RENUMBER** can be executed in direct mode only. If it detects a problem, like memory overflow, unresolved references or line number overflow (greater than 64000) it will stop with an error message and leave the program unchanged.

The command may be called with 0-3 parameters. Unspecified parameters use their default values.

**Remarks:** The **RENUMBER** command may need several minutes to execute for large programs.

**Example:** Using **RENUMBER**

```
RENUMBER :REM NUMBERS WILL BE 10,20,30,...
RENUMBER 100,5 :REM NUMBERS WILL BE 100,105,110,115,...
RENUMBER 601,1,500 :REM RENUMBER STARTING AT 601 TO 601,602,...
RENUMBER 100,5,120-180 :REM RENUMBER LINES 120-180 TO 100,105,...
```

# RESTORE

**Token:** \$8C

**Format:** RESTORE [**line**]

**Usage:** Set or reset the internal pointer for **READ** from **DATA** statements.

**line** is the new position for the pointer to point at. The default is the first program line.

**Remarks:** The new pointer target **line** needs not to contain **DATA** statements. Every **READ** will automatically advance the pointer to the next **DATA** statement.

**Example:** Using **RESTORE**

```
10 DATA 3,1,4,1,5,9,2,6
20 DATA "MEGAB5"
30 DATA 2,7,1,8,2,8,9,5
40 FOR I=1 TO 8:READ P:PRINT P:NEXT
50 RESTORE 30
60 FOR I=1 TO 8:READ P:PRINT P:NEXT
70 RESTORE 20
80 READ A$:PRINT A$
```

# RESUME

**Token:** \$D6

**Format:** **RESUME** [line | NEXT]

**Usage:** is used inside a **TRAP** routine to resume normal program execution after handling the exception.

**line** : program execution resumes at the given line number.

**NEXT** : the keyword NEXT resumes execution at the statement following the statement, that caused the error.

**RESUME** with no parameters tries to re-execute the statement, that caused the error. The **TRAP** routine should have examined and corrected the variables in this case.

**Remarks:** **RESUME** cannot be used in direct mode.

**Example:** Using **RESUME**

```
10 TRAP 100
20 FOR I=1 TO 100
30 PRINT EXP(I)
40 NEXT
50 PRINT "STOPPED FOR I =";I
60 END
100 PRINT ERR$(ER): RESUME 50
```

# RETURN

**Token:** \$8E

**Format:** **RETURN**

**Usage:** Returns control from a subroutine, which was called with **GOSUB** or an event handler, declared with **COLLISION**.

The execution continues at the statement following the **GOSUB** call.

In the case of the **COLLISION** handler, the execution continues at the statement where it left to call the handler.

**Example:** Using **RETURN**

```
10 DOPEN#2,"DATA":GOSUB 100
20 FOR I=1 TO 100
30 INPUT#2,A$:GOSUB 100
40 PRINT A$
50 NEXT
60 DCLOSE#2
70 END
100 IF DS THEN PRINT DS$:STOP :REM DISK ERROR
110 RETURN :REM OK
```

# RGRAPHIC

**Token:** \$CC

**Format:** RGRAPHIC(screen,parameter)

**Usage:** Return graphic screen status and parameters

| param | description                           |
|-------|---------------------------------------|
| 0     | open (1), closed (0), or invalid (>1) |
| 1     | width (0=320, 1=640)                  |
| 2     | height (0=200, 1=400)                 |
| 3     | depth (1-8 bitplanes)                 |
| 4     | bitplanes used (bitmask)              |
| 5     | bank 4 blocks used (bitmask)          |
| 6     | bank 5 blocks used (bitmask)          |
| 7     | drawscreen # (0-3)                    |
| 8     | viewscreen # (0-3)                    |
| 9     | drawmodes (bitmask)                   |
| 10    | pattern type (bitmask)                |

**Example:** Using RGRAPHIC

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 0,1,0,4 :REM SCREEN 0:640 X 200 X 4
30 SCREEN OPEN 0      :REM OPEN
40 SCREEN SET 0,0      :REM DRAW = VIEW = 0
50 SCNCLR 0          :REM CLEAR
60 PEN 0,1            :REM SELECT COLOUR
70 LINE 0,0,639,199   :REM DRAW LINE
80 FOR I=0 TO 10:A(I)=RGRAPHIC(0,I) :NEXT
90 SCREEN CLOSE 0
100 FOR I=0 TO 6:PRINT I;A(I):NEXT :REM PRINT INFO
```

RUN

```
0 1
1 1
2 0
3 4
4 15
5 15
6 15
```

# **RIGHT\$**

**Token:** \$C9

**Format:** **RIGHT\$(string, n)**

**Usage:** Returns a string containing the last **n** characters from the argument **string**. If the length of **string** is equal or less than **n**, the result string will be identical to the argument string.

**string** = a string expression

**n** = a numeric expression (0 -> 255)

**Remarks:** Empty strings and zero lengths are legal values.

**Example:** Using **RIGHT\$**:

```
PRINT RIGHT$("MEGA-65",2)  
65
```

# RMOUSE

**Token:** \$FE \$3F

**Format:** **RMOUSE** *xvar, yvar, butvar*

**Usage:** Reads mouse position and button status.

**xvar** = numerical variable receiving x-position.

**yvar** = numerical variable receiving y-position.

**butvar** = numerical variable receiving button status.

left button sets bit 7, while right button sets bit 0.

| value | status       |
|-------|--------------|
| 0     | no button    |
| 1     | right button |
| 128   | left button  |
| 129   | both buttons |

The command puts a -1 into all variables, if the mouse is not connected or disabled.

**Remarks:** Two active mice on both ports merge the results.

**Example:** Using **RMOUSE**:

```
10 MOUSE ON, 1, 1      :REM MOUSE ON PORT 1 WITH SPRITE 1
20 RMOUSE XP, YP, BU   :REM READ MOUSE STATUS
30 IF XP < 0 THEN PRINT "NO MOUSE ON PORT 1":STOP
40 PRINT "MOUSE:";XP;YP;BU
50 MOUSE OFF            :REM DISABLE MOUSE
```

# RND

**Token:** \$BB

**Format:** **RND(type)**

**Usage:** Returns a pseudo random number

This is called a "pseudo" random number, because the numbers are not really random, but are derived from another number called "seed" and generate reproducible sequences. The **type** argument determines, which seed is used.

**type** = 0: use system clock.

**type** < 0: use the value of **type** as seed.

**type** > 0: derive value from previous random number.

**Remarks:** Seeded random number sequences produce the same sequence for identical seeds.

**Example:** Using **RND**:

```
10 DEF FNFDI(X) = INT(RND(0)*6)+1 :REM DICE FUNCTION
20 FOR I=1 TO 10                  :REM THROW 10 TIMES
30 PRINT I;FNFDI(0)                :REM PRINT DICE POINTS
40 NEXT
```

# RPEN

**Token:** \$D0

**Format:** RPEN(n)

**Usage:** Returns the colour index of pen n.

n = PEN number (0-2).

- 0: draw pen
- 1: erase pen
- 2: outline pen

**Example:** Using RPEN

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 0,1,0,4 :REM SCREEN 0:640 X 200 X 4
30 SCREEN OPEN 0     :REM OPEN
40 SCREEN SET 0,0    :REM DRAW = VIEW = 0
50 SCNCLR 0         :REM CLEAR
60 PEN 0,1           :REM SELECT COLOUR
70 X = RPEN(0)
80 Y = RPEN(1)
90 C = RPEN(2)
100 SCREEN CLOSE 0
110 PRINT "DRAW PEN COLOUR = ";X
RUN
DRAW PEN COLOUR = 1
```

# RREG

**Token:** \$FE \$09

**Format:** **RREG areg, xreg, yreg, zreg, sreg**

**Usage:** Reads the values, that were in the CPU registers after a SYS call, into the specified variables.

**areg** = variable gets accumulator value.

**xreg** = variable gets X register value.

**yreg** = variable gets Y register value.

**zreg** = variable gets Z register value.

**sreg** = variable gets status register value.

**Remarks:** The register values after a SYS call are stored in system memory. This enables the command **RREG** to retrieve these values.

**Example:** Using **RREG**:

```
10 BANK 128
20 BLOAD "ML PROG",8192
30 SYS 8192
40 RREG A,X,Y,Z,$
50 PRINT "REGISTER:";A;X;Y;Z,$
```

# RSPCOLOR

**Token:** \$CE \$07

**Format:** RSPCOLOR(n)

**Usage:** Returns multi-colour sprite colours.

n = 1 : get multi-colour # 1.

n = 2 : get multi-colour # 2.

**Remarks:** See also SPRITE and SPRCOLOR.

**Example:** Using RSPCOLOR:

```
10 SPRITE 1,1      :REM TURN SPRITE 1 ON
20 C1% = RSPCOLOR(1) :REM READ COLOUR #1
30 C2% = RSPCOLOR(2) :REM READ COLOUR #2
```

# RSPPPOS

**Token:** \$CE \$05

**Format:** RSPPPOS(sprite,n)

**Usage:** Returns sprite's position and speed

**sprite** : sprite number.

**n** = 0 : get X position.

**n** = 1 : get Y position.

**n** = 2 : get speed.

**Remarks:** See also SPRITE and MOVSPR.

**Example:** Using RSPPPOS:

```
10 SPRITE 1,1      :REM TURN SPRITE 1 ON
20 XP = RSPPPOS(1,0) :REM GET X OF SPRITE 1
30 YP = RSPPPOS(1,1) :REM GET Y OF SPRITE 1
30 SP = RSPPPOS(1,2) :REM GET SPEED OF SPRITE 1
```

# RSprite

**Token:** \$CE \$06

**Format:** RSsprite(sprite,n)

**Usage:** Returns sprite's parameter.

**sprite** : sprite number (0 -> 7)

**n** = 0 : turned on (0 or 1).

**n** = 1 : foreground colour (0 -> 15)

**n** = 2 : background priority (0 or 1).

**n** = 3 : X-expanded (0 or 1).

**n** = 4 : Y-expanded (0 or 1).

**n** = 5 : multi-colour (0 or 1).

**Remarks:** See also **SPRITE** and **MOVSPR**.

**Example:** Using RSsprite:

```
10 SPRITE 1,1      :REM TURN SPRITE 1 ON
20 EM = RSsprite(1,0) :REM SPRITE 1 ENABLED ?
30 FG = RSsprite(1,1) :REM SPRITE 1 FOREGROUND COLOUR INDEX
30 BP = RSsprite(1,2) :REM SPRITE 1 BACKGROUND PRIORITY
20 XE = RSsprite(1,3) :REM SPRITE 1 X EXPANDED ?
30 YE = RSsprite(1,4) :REM SPRITE 1 Y EXPANDED ?
30 MC = RSsprite(1,5) :REM SPRITE 1 MULTI-COLOUR ?
```

# RUN

**Token:** \$8A

**Format:** **RUN [line number]**  
**RUN filename [,D drive] [,U unit]**

**Usage:** Run a BASIC program.

If a filename is given, the program file is loaded into memory, otherwise the program that is currently in memory is used.

**line number** an existing line number of the program in memory.

**filename** is either a quoted string, e.g. "**prog**" or a string expression in parentheses, e.g. **(PR\$)**. The filetype must be "PRG".

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**RUN** resets first all internal pointers to their starting values. Therefore there are no variables, arrays and strings defined, the run-time stack is reset and the table of open files is cleared.

**Remarks:** In order to start or continue program execution without resetting everything, use the **GOTO** command.

**Example:** Using **RUN**

```
RUN "FLIGHTSIM" :LOAD AND RUN PROGRAM FLIGHTSIM
RUN 1000      :RUN PROGRAM IN MEMORY, START AT 1000
RUN          :RUN PROGRAM IN MEMORY
```

# RWINDOW

**Token:** \$CE \$09

**Format:** **RWINDOW(n)**

**Usage:** Returns information regarding the current text window

**n** = 0 : get width of current text window.

**n** = 1 : get height of current text window.

**n** = 2 : get width of screen (40 or 80).

**Remarks:** See also **WINDOW**.

Older versions of **RWINDOW** reported the width-1 and the height-1 for the arguments 0 and 1.

**Example:** Using **RWINDOW**:

```
10 W = RWINDOW(2)      :REM GET SCREEN WIDTH
20 IF W=80 THEN BEGIN   :REM IS 80 COLUMNS MODE ACTIVE?
30   PRINT CHR$(27)+"X"; :REM YES, SWITCH TO 40COLUMNS
40 BEND
```

# SAVE

**Token:** \$94

**Format:** **SAVE filename [,unit]**

**Format:** **← filename [,unit]**

**Usage:** The shortcut symbol ← can be used in direct mode only.

"Saves a BASIC program to a file of type PRG.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**F1\$**) The maximum length of the filename is 16 characters, not counting the optional save and replace character '@' and the in-file drive definition.. If the first character of the filename is an at-sign '@' it is interpreted as a "save and replace" operation. It is dangerous to use this replace option on drives 1541 and 1571, because they contain the notorious "save and replace bug" in their DOS. The filename may be preceded by the drive number definition "0:" or "1:" which is only relevant for dual drive disk units.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** This is an obsolete command, implemented only for compatibility to older BASIC dialects. The command **DSAVE** should be used instead.

**Example:** Using **SAVE**

```
SAVE "ADVENTURE"  
SAVE "ZORK-I",8  
SAVE "1:DUNGEON",9
```

# **SCNCLR**

**Token:** \$E8

**Format:** **SCNCLR [colour]**

**Usage:** Clears a text window or screen.

**SCNCLR** (with no arguments) clears the current text window. The default window occupies the whole screen.

**SCNCLR colour** clears the graphic screen by filling it with the **colour**.

**Example:** Using **SCNCLR**:

```
10 GRAPHIC CLR      :REM INITIALISE
20 SCREEN DEF 1,1,1,2 :REM 640 X 400 X 2
30 SCREEN SET 1,1     :REM VIEW IT
40 SCNCLR 0          :REM CLEAR SCREEN
50 LINE 50,50,590,350 :REM DRAW LINE
```

# SCRATCH

**Token:** \$F2

**Format:** SCRATCH **filename** [,D **drive**] [,U **unit**] [,R]

**Usage:** Used to erase a disk file.

**filename** is either a quoted string, e.g. "data" or a string expression in parentheses, e.g. (F1\$)

**drive** = drive # in dual drive disk units.

The drive # defaults to 0 and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to 8.

**R** = Recover a previously erased file. This will only work, if there were no write operations between erasure and recovery, which may have altered the contents of the file.

**Remarks:** The SCRATCH **filename** command works like the ERASE **filename** command.

The success and the number of erased files can be examined by printing or using the system variable DS\$. The second last number, which reports the track number in case of an disk error, now reports the number of successfully erased files.

**Example:** Using SCRATCH

```
SCRATCH "DRM",U9 :REM SCRATCH FILE DRM ON UNIT 9
PRINT DS$
01, FILES SCRATCHED,01,00
SCRATCH "OLD*"   :REM SCRATCH ALL FILES BEGINNING WITH "OLD"
PRINT DS$
01, FILES SCRATCHED,04,00
```

# SCREEN

**Token:** \$FE \$2E

**Format:**

- SCREEN [screen,] width, height, depth**
- SCREEN CLR colour**
- SCREEN DEF width flag, height flag, depth**
- SCREEN SET draw view**
- SCREEN OPEN screen [,errvar]**
- SCREEN CLOSE screen**

**Usage:** The first version of the **SCREEN** command is the easiest way to start a graphics screen and is the preferred method, if only one single screen is needed. This includes the actions **GRAPHIC CLR**, **SCREEN DEF**, **SCREEN SET**, **SCREEN OPEN** and **SCREEN CLR**.

**screen** the screen number (0-3) is optional. If no screen number is given, screen 0 is used.

**width** = 320 or 640 (default = 320)

**height** = 200 or 400 (default = 200)

**depth** = 1..8 (default = 8), colours =  $2 \uparrow \text{depth}$ .

The argument parser is error tolerant and uses default values for width (320) and height (200) if the parsed argument is not valid.

This version of SCREEN starts with a predefined palette and sets the background to black and the pen to white. So the drawing can start immediately using the default setup or palette and pen can be changed. The other versions require the setting of palette colours and pen colour, before any drawing can start.

The default palette for graphics mode:

| index | red | green | blue | colour      |
|-------|-----|-------|------|-------------|
| 0     | 0   | 0     | 0    | black       |
| 1     | 15  | 15    | 15   | white       |
| 2     | 15  | 0     | 0    | red         |
| 3     | 0   | 15    | 0    | green       |
| 4     | 0   | 0     | 15   | blue        |
| 5     | 15  | 15    | 0    | yellow      |
| 6     | 15  | 0     | 15   | magenta     |
| 7     | 0   | 15    | 15   | cyan        |
| 8     | 8   | 8     | 8    | gray        |
| 9     | 8   | 0     | 0    | maroon      |
| 10    | 0   | 8     | 0    | dark green  |
| 11    | 0   | 0     | 8    | dark blue   |
| 12    | 8   | 8     | 0    | olive       |
| 13    | 8   | 0     | 8    | deep purple |
| 14    | 0   | 8     | 8    | teal        |
| 15    | 11  | 11    | 11   | light grey  |

The other versions of the **SCREEN** command perform special actions, used for advanced graphics programs, that open multiple screens or double buffering. Except for **SCREEN CLOSE** they are not needed, if the first version was used.

**SCREEN CLR colour** clears the active graphics screen by filling it with colour.

**SCREEN DEF screen, width flag, height flag, depth** defines resolution parameters for the chosen screen. The width flag and height flag indicate, whether high resolution (1) or low resolution (0) is chosen.

screen = screen number 0-3

width flag = 0-1 (0:320, 1:640 pixel)

height flag = 0-1 (0:200, 1:400 pixel)

depth = 1-8 (2 - 256 colours)

**SCREEN SET draw view** sets screen numbers ( 0-3 ) for the drawing and the viewing screen.

**SCREEN OPEN screen** allocates resources and initialises the graphic context for the selected screen (0-3). An optional variable name as a further argument, gets the result of the command and can be tested afterwards for success.

**SCREEN CLOSE screen** closes screen (0-3) and frees resources.

**Example:** Using **SCREEN**:

```
10 SCREEN 320,200,2 :REM SCREEN #0: 320 X 200 X 2
20 PEN 1 :REM DRAWING PEN COLOUR = 1 (WHITE)
30 LINE 25,25,295,175 :REM DRAW LINE
40 GETKEY A$ :REM WAIT KEYPRESS
50 SCREEN CLOSE :REM CLOSE SCREEN 0
60 PALETTE RESTORE :REM BACK TO TEXT PALETTE
```

```
10 GRAPHIC CLR :REM INITIALISE
20 SCREEN DEF 1,0,0,2 :REM SCREEN #1: 320 X 200 X 2
30 SCREEN OPEN 1 :REM OPEN SCREEN 1
40 SCREEN SET 1,1 :REM USE SCREEN 1 FOR RENDERING AND VIEWING
50 SCREEN CLR 0 :REM CLEAR SCREEN
60 PALETTE 1,1,15,15,15:REM DEFINE COLOUR 1 AS WHITE
70 PEN 0,1 :REM DRAWING PEN
80 LINE 25,25,295,175 :REM DRAW LINE
90 SLEEP 10 :REM WAIT 10 SECONDS
100 SCREEN CLOSE 1 :REM CLOSE SCREEN 1
110 PALETTE RESTORE :REM BACK TO TEXT PALETTE
```

# SET

- Token:** \$FE \$2D
- Format:** **SET DEF unit**  
**SET DISK old to new**  
**SET VERIFY ON|OFF**
- Usage:** **SET DEF unit** redefines the default unit for disk access, which is initialised to 8 by the DOS. Commands, that do not explicitly specify a unit, will use this default unit.  
**SET DISK old to new** is used to change the unit number of a disk drive temporarily.  
**SET VERIFY ON|OFF** enables or disables the DOS verify-after-write mode for 3.5 drives.
- Remarks:** These settings are valid until a reset or shutdown.
- Example:** Using **SET**:

```
DIR      :REM SHOW DIRECTORY OF UNIT 8
SET DEF 11 :REM UNIT 11 BECOMES DEFAULT
DIR      :REM SHOW DIRECTORY OF UNIT 11
DLOAD "*"   :REM LOAD FIRST FILE FROM UNIT 11
SET DISK 8 TO 9 :REM CHANGE UNIT# OF DISK DRIVE 8 TO 9
DIR U9      :REM SHOW DIRECTORY OF UNIT 9 (FORMER 8)
SET VERIFY ON :REM ACTIVATE VERIFY-AFTER-WRITE MODE
```

# **SGN**

**Token:** \$B4

**Format:** **SGN(numeric expression)**

**Usage:** The **SGN** function extracts the sign from the argument and returns it as a number:

- 1 for a negative argument
- 0 for a zero
- 1 for a positive, non zero argument

**Example:** Using **SGN**

```
10 ON SGN(X)+2 GOTO 100,200,300 :REM TARGETS FOR MINUS,ZERO,PLUS  
20 Z = SGN(X) * ABS(Y) : REM COMBINE SIGN OF X WITH VALUE OF Y
```

# SIN

**Token:** \$BF

**Format:** **SIN(numeric expression)**

**Usage:** The **SIN** function returns the sine of the argument. The argument is expected in units of **[radians]**. The result is in the range (-1.0 to +1.0)

**Remarks:** An argument in units of **[degrees]** can be converted to **[radians]** by multiplication with  $\pi/180$ .

**Example:** Using **SIN**

```
PRINT SIN(0.7)
.644217687

X=30:PRINT SIN(X * pi / 180)
.5
```

# SLEEP

**Token:** \$FE \$0B

**Format:** **SLEEP seconds**

**Usage:** The **SLEEP** command pauses the execution for the given duration. The argument is a positive floating point number. The precision is 1 microsecond.

**Remarks:** Pressing the **STOP** key interrupts the sleep.

**Example:** Using **SLEEP**

```
20 SLEEP 10      :REM WAIT 10 SECONDS
40 SLEEP 0.0005 :REM SLEEP 500 MICRO SECONDS
50 SLEEP 0.01    :REM SLEEP 10 MILLI SECONDS
60 SLEEP DD      :REM TAKE SLEEP TIME FROM VARIABLE DD
70 SLEEP 600     :REM SLEEP 10 MINUTES
```

# **SLOW**

**Token:** \$FE \$26

**Format:** **SLOW**

**Usage:** Slow down system clock to 1 MHz.

**Example:** Using **SLOW**

```
50 SLOW      :REM SET SPEED TO MINIMUM  
60 GOSUB 100 :REM EXECUTE SUBROUTINE AT 1 MHZ  
70 FAST       :REM BACK TO HIGH SPEED
```

# SOUND

**Token:** \$DA

**Format:** **SOUND voice, freq, dur [,dir ,min, sweep, wave, pulse]**

**Usage:** plays a sound effect.

**voice** = voice number ( 1 -> 6 ).

**freq** = frequency ( 0 -> 65535 ).

**dur** = duration ( 0 -> 32767 ).

**dir** = direction (0:up, 1:down, 2:oscillate).

**min** = minimum frequency ( 0 -> 65535 ).

**sweep** = sweep range ( 0 -> 65535 ).

**wave** = waveform (0:triangle, 1:saw, 2:square, 3:noise).

**pulse** = pulse width ( 0 -> 5095 ).

For details on sound programming, read the **SOUND** chapter.

**Remarks:** The **SOUND** command starts playing the sound effect and immediately continues with the execution of the next BASIC statement, while the sound effect is played. This enables showing graphics or text and playing sounds simultaneously.

**Example:** Using **SOUND**

```
SOUND 1, 7382, 60 :REM PLAY SQUARE WAVE ON VOICE 1 FOR 1 SECOND  
SOUND 2, 800, 3600 :REM PLAY SQUARE WAVE ON VOICE 2 FOR 1 MINUTE  
SOUND 3, 4000, 120, 2, 2000, 400, 1  
REM PLAY SWEEPING SAWTOOTH WAVE AT VOICE 3
```

# SPC

**Token:** \$A6

**Format:** **SPC(columns)**

**Usage:** The **SPC** function skips **columns**.

The effect is like printing **column** times a **cursor right character**.

**Remarks:** The name of this function is derived from **SPACES**, which is misleading. The function prints **cursor right characters** not **SPACES**. The contents of those character cells, that are skipped, will not be changed.

**Example:** Using **SPC**

```
10 FOR I=8 TO 12
20 PRINT SPC(-(I<10));I :REM TRUE = -1, FALSE = 0
30 NEXT I
RUN
8
9
10
11
12
```

# SPRCOLOR

**Token:** \$FE \$08

**Format:** SPRCOLOR [mc1] [,mc2]

**Usage:** Sets multi-colour sprite colours.

The **SPRITE** command, which sets the attributes of a sprite, sets only the foreground colour. For the setting of the additional two colours, of multi-colour sprites, **SPRCOLOR** has to be used.

**Remarks:** See also **SPRITE**.

**Example:** Using **SPRCOLOR**:

```
10 SPRITE 1,1,2,,,1 :REM TURN SPRITE 1 ON (FG = 2)
20 SPRCOLOR 4,5      :REM MC1 = 4, MC2 = 5
```

# SPRITE

**Token:** \$FE \$07

**Format:** **SPRITE CLR**

**SPRITE LOAD filename [,D drive] [,U unit]**

**SPRITE SAVE filename [,D drive] [,U unit]**

**SPRITE no [switch, colour, prio, expx, expy, mode]**

**Usage:** **SPRITE CLR** clears all sprite data and sets all pointers and attributes to default values.

**SPRITE LOAD** loads sprite data from **filename** to sprite memory.

**SPRITE SAVE** saves sprite data from sprite memory to **filename**.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

The last form switches a sprite on or off and sets its attributes.

**no** = sprite number

**switch** = 1:ON, 0:OFF

**colour** = sprite foreground colour

**prio** = sprite(1) or screen(0) priority

**expx** = 1:sprite X expansion

**expy** = 1:sprite Y expansion

**mode** = 1:multi colour sprite

**Remarks:** The command **SPRCOLOR** must be used to set additional colours for multi colour sprites (mode = 1)

**Example:** Using **SPRITE**:

```
2290 CLR:SCNCLR:SPRITE CLR
2300 SPRITE LOAD "DEMOsprites1"
2320 FORI=0TO7: C=I: IFC=6THENC=8
2330 MOVSPR I, 60+30*I,0 TO 60+30*I,65+20*I, 3:SPRITE I,1,C,,1,1:NEXT: SLEEP3
2340 FORI=0TO7: SPRITE I,,,0,0 :NEXT: SLEEP3: SPRITE CLR
2350 FORI=0TO7: MOVSPR I,45*IH5 :NEXT: FORI=0TO7: SPRITE I,1: NEXT
2360 FORI=0TO7:X=60+30*I:Y=65+20*I:DO
2370 LOOPUNTIL(X=RSPPOS(I,,))AND(Y=RSPPOS(I,1)):MOVSPR I,,,:NEXT
```

# SPRSAV

**Token:** \$FE \$16

**Format:** **SPRSAV source, destination**

**Usage:** Copies sprite data.

**source** = sprite number or string variable.

**destination** = sprite number or string variable.

**Remarks:** Both, source and destination can be either a sprite number or a string variable. But they must not be both a string variable. A simple string assignment can be used for such cases.

This command can be used with the basic form of sprites (C64 compatible) only. These sprites have a size of 64 bytes, the length of the strings is 67.

The extended sprites and the variable height sprites cannot be used with this command.

**Example:** Using **SPRSAV**:

```
10 BLOAD "SPRITEDATA",P1600 :REM LOAD DATA FOR SPRITE 1
20 SPRITE 1,1 :REM TURN SPRITE 1 ON
30 SPRSAV 1,2 :REM COPY SPRITE 1 DATA TO 2
40 SPRITE 2,1 :REM TURN SPRITE 2 ON
50 SPRSAV 1,A$ :REM SAVE SPRITE 1 DATA IN STRING
```

# SQR

**Token:** \$BA

**Format:** **SQR(numeric expression)**

**Usage:** The **SQR** function returns the square root of the argument.

**Remarks:** The argument must not be negative.

**Example:** Using **SQR**

```
PRINT SQR(2)
```

```
1.41421356
```

# **ST**

**Format:** **ST** is a reserved system variable

**Usage:** **ST** holds the status of the last input/output operation. ST is set to zero, if there was no error, otherwise it is set to a device dependent error code.

**Example:** Using **ST**

```
100 DOPEN#1,"DATA"           :REM OPEN FILE
110 IF DS THEN PRINT"COULD NOT OPEN":STOP
120 LINE INPUT#1,T$(N):N=N+1  :REM READ ONE RECORD
130 IF ST=0 THEN 120         :REM ST = 64 FOR END-OF-FILE
140 DCLOSE#1
150 PRINT "READ";N;" RECORDS"
```

# STEP

**Token:** \$A9

**Format:** **FOR index=start TO end [STEP step] ... NEXT [index]**

**Usage:** The **STEP** keyword is an optional part of a **FOR** loop.

The **index** variable may be incremented or decremented by a constant value on each iteration. The default is to increment the variable by 1. The index variable must be a real variable.

The **start** value is used to initialise the index.

The **end** value is used at the end of the loop and controls, whether the next iteration will be started or the loop exited.

The **step** value defines the change applied to the index variable at the end of the loop. Positive step values increment it, while negative values decrement it. It defaults to 1.0 if not specified.

**Remarks:** For positive increments **end** must be greater or equal than **start**, for negative increments **end** must be less or equal than **start**.

It is bad programming style to change the value of the index variable inside the loop or to jump into or out of the loop body with **GOTO**.

**Example:** Using **STEP**

```
10 FOR D=0 TO 360 STEP 30
20 R = D * π / 180
30 PRINT D;R;SIN(R);COS(R);TAN(R)
40 NEXT D
```

# **STOP**

**Token:** \$90

**Format:** **STOP**

**Usage:** Stops the execution of the BASIC program. A message tells the line number of the break. The **READY.** prompt appears and the computer goes into direct mode waiting for keyboard input. The program execution can be resumed with the command **CONT.**

**Remarks:** All variable definitions are still valid after **STOP**. They may be inspected or altered and the program may be continued with the **CONT** statement. Every editing of the program source makes continuation impossible, however.

**Example:** Using **STOP**

```
10 IF V < 0 THEN STOP : REM NEGATIVE NUMBERS STOP THE PROGRAM
20 PRINT SQR(V)      : REM PRINT SQUARE ROOT
```

# **STR\$**

**Token:** \$C4

**Format:** **STR\$(numeric expression)**

**Usage:** Returns a string containing the formatted value of the argument, as if it were printed to the string.

**Example:** Using **STR\$**:

```
A$ = "THE VALUE OF PI IS " + STR$(π)  
PRINT A$  
THE VALUE OF PI IS 3.14159265
```

# SYS

**Token:** \$9E

**Format:** **SYS address [, areg, xreg, yreg, zreg, sreg]**

**Usage:** Calls a machine language subroutine. This can be a ROM resident kernel or BASIC subroutine or a routine in RAM, which was loaded or poked to RAM before.

The CPU registers are loaded with the arguments, if specified. Then a subroutine call **JSR address** is performed. The called routine should exit with a **RTS** instruction. Then the register contents will be saved and the execution of the BASIC program continues.

**address** = start address of the subroutine.

**areg** = variable gets accumulator value.

**xreg** = variable gets X register value.

**yreg** = variable gets Y register value.

**zreg** = variable gets Z register value.

**sreg** = variable gets status register value.

The **SYS** command uses the current bank as set with the **BANK** command.

**Remarks:** The register values after a **SYS** call are stored in system memory. This enables the command **RREG** to retrieve these values.

**Remarks:** The system leaves the memory setup in a somewhat strange state, which may or may not be a bug in BASIC 65. If you want to write an assembly routine that can access an entire memory bank, but with IO mapped, you need to set the CPU register \$01, like on the C64, and then also use the 4510's MAP instruction to set the memory map. For BANK 0, you could use the following sequence:

```
; Disable interrupts, because the C65 ROM won't be visible to
; handle them.
SEI
; Make IO visible
LDA #37
STA $01
; Prepare for MAP operation that resets memory map to BANK 0
LDA #00
TAX
TAY
TAZ
MAP
; End MAPping sequence
NOP
; Now comes your routine!
loop:
INC $D020
JMP loop
```

**Remarks:** You can alternatively use BANK 128, and place your routine in the lower-half of RAM, e.g., in the unallocated memory area at \$1600 - \$1FFF

**Remarks:** The RREG command can be useful for reading the register values when your routine completes.

**Example:** Using **SYS**:

```
10 BANK 128
20 BLOAD "ML PROG",8192
30 SYS 8192
40 RREG A,X,Y,Z,S
50 PRINT "REGISTER:";A;X;Y;Z,$
```

# TAB

**Token:** \$A3

**Format:** TAB(column)

**Usage:** Positions the cursor at **column**.

This is only done, if the target column is right of the current cursor column, otherwise nothing happens. The column count starts with 0 for the left most column.

**Remarks:** This function must not be confused with the **TAB** key, which advances the cursor to the next tab-stop.

**Example:** Using **TAB**

```
10 FOR I=1 TO 5
20 READ A$
30 PRINT "* " A$ TAB(10) " *"
40 NEXT I
50 END
60 DATA ONE,TWO,THREE,FOUR,FIVE
```

```
RUN
* ONE      *
* TWO      *
* THREE    *
* FOUR     *
* FIVE     *
```

# TAN

**Token:** \$C0

**Format:** **TAN(numeric expression)**

**Usage:** Returns the tangent of the argument. The argument is expected in units of [radians].

**Remarks:** An argument in units of [degrees] can be converted to [radians] by multiplication with  $\pi/180$ .

**Example:** Using **TAN**

```
PRINT TAN(0.7)
```

```
.8428838
```

```
X=45:PRINT TAN(X * pi / 180)
```

```
.99999999
```

# TEMPO

**Token:** \$FE \$05

**Format:** **TEMPO speed**

**Usage:** Sets the playback speed for the **PLAY** command.

**speed** = 1 -> 255.

The duration of a whole note is computed with  $duration = 24/speed$ .

**Example:** Using **TEMPO**

```
10 ENVELOPE 9,10,5,10,5,2,4000:PLAY "T9"  
20 VOL 8  
30 TEMPO 24      :REM PLAY EACH NOTE FOR ONE SECOND  
40 PLAY "C D E F G A B"  
50 PLAY "U5 V1 C D E F G A B"
```

# THEN

**Token:** \$A7

**Format:** **IF expression THEN true clause ELSE false clause**

**Usage:** The **THEN** keyword is part of an **IF** statement.

**expression** is a logical or numeric expression. A numerical expression is evaluated as **FALSE** if the value is zero and **TRUE** for any non zero value.

**true clause** are one or more statements starting directly after **THEN** on the same line. A linenumber after **THEN** performs a **GOTO** to that line.

**false clause** are one or more statements starting directly after **ELSE** on the same line. A linenumber after **ELSE** performs a **GOTO** to that line.

**Remarks:** The standard **IF ... THEN ... ELSE** structure is restricted to a single line. But the **true clause** or **false clause** may be expanded to several lines using a compound statement bracketed with the keywords **BEGIN** and **BEND**.

**Example:** Using **THEN**

```
10 IF V < 0 THEN PRINT RED$;:ELSE PRINT BLACK$;
20 PRINT V : REM PRINT NEGATIVE NUMBERS IN RED
30 INPUT "END PROGRAM:(Y/N)";A$
40 IF A$="Y" THEN END
50 IF A$="N" THEN 10:ELSE 30
```

# TI

**Format:** TI is a reserved system variable

**Usage:** TI is a high precision timer with the unit seconds and the resolution of 1 micro second.

It is started or reset with the command **CLR TI** and can be accessed like any other variable in expressions.

**Example:** Using TI

```
100 CLR TI           :REM START TIMER
110 FOR I% = 1 TO 10000:NEXT :REM DO SOMETHING
120 ET = TI           :REM STORE ELAPSED TIME IN ET
130 PRINT "EXECUTION TIME:";ET;" SECONDS"
```

# TI\$

**Format:** TI\$ is a reserved system variable

**Usage:** TI\$ holds the time information of the RTC (Real Time Clock) in text form of the format: "hh:mm:ss". It is updated on every usage.

TI\$ is a read-only variable, that just reads the registers of the RTC and puts the values formatted into a string.

The RTC can be set in the CONFIGURE menu.

**Remarks:** It is possible to access the RTC registers directly via PEEK. The start address of the registers is at \$FFD7110. For example:

```
100 REM ***** READ RTC ***** ALL VALUES ARE BCD ENCODED
110 RT = $FFD7110          :REM ADDRESS OF RTC
120 FOR I=0 TO 5           :REM SS,MM,HH,DD,MO,YY
130 T(I)=PEEK(RT+I)       :REM READ REGISTERS
140 NEXT I                :REM USE ONLY LAST TWO DIGITS
150 T(2) = T(2) AND 127    :REM REMOVE 24H MODE FLAG
160 T(5) = T(5) + $2000    :REM ADD YEAR 2000
170 FOR I=2 TO 0 STEP -1   :REM TIME INFO
180 PRINT USING ">## ";HEX$(T(I));
190 NEXT I
RUN
12 52 36
```

**Example:** Using TI\$

```
PRINT DT$;TI$
05-APR-2021 15:10:00
```

# TO

**Token:** \$A4

**Format:** keyword TO

**Usage:** TO is a secondary keyword used in combination with primary keywords like GO, FOR, BACKUP, BSAVE, CHANGE, CONCAT, COPY, RENAME and SET DISK

**Remarks:** The keyword TO cannot be used on its own.

**Example:** Using TO

```
10 GO TO 1000 :REM AS GOTO 1000
20 GOTO 1000 :REM SHORTER AND FASTER
30 FOR I=1 TO 10 :REM TO IS PART OF THE LOOP
40 PRINT I:NEXT :REM LOOP END
50 COPY "CODES" TO "BACKUP" :REM COPY SINGLE FILE
```

# TRAP

**Token:** \$D7

**Format:** **TRAP** [line number]

**Usage:** **TRAP** with a valid line number activates the BASIC error handler with following consequences: In case of an error the BASIC interpreter does not stop with an error message, but saves execution pointer and line number, places the error number into the system variable **ER** and jumps to the line number of the TRAP command. The trapping routine can examine **ER** and decide, whether to **STOP** or **RESUME** execution.

**TRAP** with no argument disables the error handler. Errors will be handled by the normal system routines.

**Example:** Using **TRAP**

```
10 TRAP 100
20 FOR I=1 TO 100
30 PRINT EXP(I)
40 NEXT
50 PRINT "STOPPED FOR I =";I
60 END
100 PRINT ERR$(ER): RESUME 50
```

# TROFF

**Token:** \$D9

**Format:** TROFF

**Usage:** Turns off trace mode (switched on by TRON).

**Example:** Using TROFF

```
10 TRON          :REM ACTIVATE TRACE MODE
20 FOR I=85 TO 100
30 PRINT I;EXP(I)
40 NEXT
50 TROFF         :REM DEACTIVATE TRACE MODE

RUN
[10][20][30] 85  8.22301268E+36
[40][30] 86  2.2352466E+37
[40][30] 87  6.0760302E+37
[40][30] 88  1.65163625E+38
[40][30] 89
?OVERFLOW ERROR IN 30
READY.
```

# TRON

**Token:** \$D8

**Format:** TRON

**Usage:** Turns on trace mode.

**Example:** Using TRON

```
10 TRON          :REM ACTIVATE TRACE MODE
20 FOR I=85 TO 100
30 PRINT I;EXP(I)
40 NEXT
50 TROFF         :REM DEACTIVATE TRACE MODE

RUN
[10][20][30] 85  8.22381268E+36
[40][30] 86  2.2352466E+37
[40][30] 87  6.0760382E+37
[40][30] 88  1.65163625E+38
[40][30] 89
?OVERFLOW ERROR IN 30
READY.
```

# TYPE

**Token:** \$FE \$27

**Format:** **TYPE filename [,D drive] [,U unit]**

**Usage:** types the contents of a file containing text in PETSCII code.

**filename** is either a quoted string, e.g. "**data**" or a string expression in parentheses, e.g. (**FI\$**)

**drive** = drive # in dual drive disk units.

The drive # defaults to **0** and can be omitted on single drive units like the 1581, 1571 or 1541 series.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** This command cannot be used to type BASIC programs. Use **LIST** for programs. **TYPE** can only process SEQ or USR files containing records of PETSCII text, delimited by the **CR** = CHR\$(13) character.

**Example:** Using **TYPE**

```
TYPE "README"  
TYPE "README 1ST",U9
```

# UNTIL

**Token:** \$FC

**Format:** **DO** ... **LOOP**

**DO** [ <**UNTIL** | **WHILE**> <logical expr.>]  
... statements [**EXIT**]  
**LOOP** [ <**UNTIL** | **WHILE**> <logical expr.>]

**Usage:** The **DO** and **LOOP** keywords define the start and end of the most versatile BASIC loop. Using **DO** and **LOOP** alone, without any modifiers creates an infinite loop, that can be left by the **EXIT** statement only. The loop can be controlled by adding an **UNTIL** or a **WHILE** statement after the **DO** or **LOOP**.

**Remarks:** **DO** loops may be nested. An **EXIT** statement exits the current loop only.

**Example:** Using **DO** and **LOOP**.

```
10 PW$="":DO  
20 GET A$:PW$=PW$+A$  
30 LOOP UNTIL LEN(PW$)>? OR A$=CHR$(13)  
  
10 DO : REM WAIT FOR USER DECISION  
20 GET A$  
30 LOOP UNTIL A$='Y' OR A$='N' OR A$='y' OR A$='n'  
  
10 DO WHILE ABS(EPS) > 0.001  
20 GOSUB 2000 : REM ITERATION SUBROUTINE  
30 LOOP  
  
10 IX=0 : REM INTEGER LOOP 1 -> 100  
20 DO IX=IX+1  
30 LOOP WHILE IX < 101
```

# USING

**Token:** \$FB

**Format:** PRINT [# channel,] USING format;argument

**Usage:** Parses the format string and evaluates the argument. The argument can be either a string or a numeric value. The formatting of the resulting output is directed by the format string.

**channel** : must be opened for output by an **OPEN** or **DOPEN** statement. If no channel is specified, the output goes to the screen.

**format** : A string variable or a string constant which defines the rules for formatting.

**numeric argument** : The numeric formatting can be done in either CBM style providing a pattern like "###.##" or in C style using width.precision specifier like %3D %7.2F %4X for example.

If the argument does not fit into the format e.g. trying to print a 4 digit variable into "###" a series of asterisks fills the format character.

**string argument** : The string may consist of printable characters and control codes. Printable characters are printed to the cursor position, while control codes are executed. The number of '#' characters sets the width of the output. If the first character of the format string is a '=' sign, the argument string is centered within the width. If the first character of the format string is a '}' sign, the argument string is right justified within the width.

**Remarks:** The format string is applied for one argument only. But it is possible to append more than one USING format;argument sequence.

**Example:** Using **PRINT# USING**

```
PRINT USING "##.###";#, USING "1%6.4F";SQR(2)
3.14 [1.4142]

PRINT USING "(< # # # >)";12*31
< 3 7 2 >

PRINT USING "####"; "ABCDE"
ABC

PRINT USING ">####"; "ABCDE"
CDE

PRINT USING "ADDRESS:$%4X";65000
ADDRESS:$FDE8

A$="####,###,###.##":PRINT USING A$;1E8/3
33,333,333.3
```

# USR

**Token:** \$B7

**Format:** **USR(numeric expression)**

**Usage:** Using the function **USR(X)** in a numeric expression, puts the argument into the floating point accumulator 1 and jumps to the address \$02F7 expecting the address of the machine language user routine in \$02F8 - \$02F9. After executing the user routine, BASIC returns the contents of the floating point accumulator 1, which should be set by the user routine..

**Remarks:** Banks 0 -> 127 give access to RAM or ROM banks. Banks > 127 are used to access I/O and SYSTEM like VIC, SID, FDC, etc.

**Example:** Using **USR**

```
10 UX = DEC("7F00")      :REM ADDRESS OF USER ROUTINE
20 BANK 128              :REM SELECT SYSTEM BANK
30 BLOAD "ML-PROG",P(UX) :REM LOAD USER ROUTINE
40 POKE (DEC("2F8")),UX AND 255 :REM USR JUMP TARGET LOW
50 POKE (DEC("2F9")),UX / 256   :REM USR JUMP TARGET HIGH
60 PRINT USR(\n)          :REM PRINT RESULT FOR ARGUMENT PI
```

# VAL

**Token:** \$C5

**Format:** **VAL(string expression)**

**Usage:** Converts a string to a floating point value.

This function acts like reading from a string.

**Remarks:** A string containing not a valid number will not produce an error but return 0 as result.

**Example:** Using **VAL**

```
PRINT VAL("78E2")
```

```
7800
```

```
PRINT VAL("7+5")
```

```
7
```

```
PRINT VAL("1.256")
```

```
1.256
```

```
PRINT VAL("$FFFF")
```

```
0
```

# VERIFY

**Token:** \$95

**Format:** **VERIFY filename [,unit [,binflag]]**

**Usage:** This command is obsolete in BASIC-10, where the commands **DVERIFY** and **BVERIFY** are better alternatives.

**VERIFY** with no **binflag** compares a BASIC program in memory with a disk file of type PRG. It does the same as **DVERIFY**, but with a different syntax.

**VERIFY** with **binflag** compares a binary file in memory with a disk file of type PRG. It does the same as **BVERIFY**, but with a different syntax.

**filename** is either a quoted string, e.g. "**prog**" or a string expression.

**unit** = device number on the IEC bus. Typically in the range 8 to 11 for disk units. If a variable is used, it must be put in parentheses. The unit # defaults to **8**.

**Remarks:** **VERIFY** can only test for equality. It gives no information about the number or position of different valued bytes. The command exits either with the message **OK** or with **VERIFY ERROR**.

**Example:** Using **VERIFY**

```
VERIFY "ADVENTURE"  
VERIFY "ZORK-I",9  
VERIFY "1:DUNGEON",10
```

# VIEWPORT

**Token:** \$FE \$31

**Format:** **VIEWPORT <CLR|DEF> X, Y, DX, DY**

**Usage:** **VIEWPORT** must be followed either by the keyword **CLR** or **DEF** and four integer parameters.

**Remarks:** The purpose of this command is currently not known and has to be determined.

**Example:** Using **VIEWPORT**

```
VIEWPORT CLR 0,0,2,2  
VIEWPORT DEF 0,0,2,2
```

# VOL

**Token:** \$DB

**Format:** VOL volume

**Usage:** Sets the volume for sound output with **SOUND** or **PLAY**.

volume = 0 (off) -> 15 (loudest).

**Remarks:** This volume setting affects all voices.

**Example:** Using VOL

```
10 ENVELOPE 9,10,5,10,5,2,4000:PLAY "T9"  
20 VOL 8  
30 TEMPO 100  
40 PLAY "C D E F G A B"  
50 PLAY "U5 V1 C D E F G A B"
```

# WAIT

**Token:** \$92

**Format:** **WAIT address, andmask [, xormask]**

**Usage:** Pauses the BASIC program until a requested bit pattern is read from the given address.

**address** = the address at the current memory bank, which is read.

**andmask** = and mask applied.

**xormask** = xor mask applied.

**WAIT** reads the byte value from **address** and applies the masks:

result = PEEK(address) AND andmask XOR xormask

The pause ends if the result is nonzero, otherwise the reading is repeated. This may hang the computer infinitely, if the condition is never met.

**Remarks:** This command is typically used to examine hardware registers or system variables and wait for an event, e.g. joystick event, mouse event, keyboard press or a special raster line.

**Example:** Using **WAIT**

```
10 BANK 128
20 WAIT 211,1      :REM WAIT FOR SHIFT KEY BEING PRESSED
```

# WHILE

**Token:** \$ED

**Format:** **DO ... LOOP**

**DO** [ <UNTIL | WHILE> <logical expr.>]  
. . . statements [**EXIT**]  
**LOOP** [ <UNTIL | WHILE> <logical expr.>]

**Usage:** The **DO** and **LOOP** keywords define the start and end of the most versatile BASIC loop. Using **DO** and **LOOP** alone, without any modifiers creates an infinite loop, that can be left by the **EXIT** statement only. The loop can be controlled by adding an **UNTIL** or a **WHILE** statement after the **DO** or **LOOP**.

**Remarks:** **DO** loops may be nested. An **EXIT** statement exits the current loop only.

**Example:** Using **DO** and **LOOP**

```
10 PW$="" : DO  
20 GET A$: PW$=PW$+A$  
30 LOOP UNTIL LEN(PW$)>? OR A$=CHR$(13)  
  
10 DO : REM WAIT FOR USER DECISION  
20 GET A$  
30 LOOP UNTIL A$='Y' OR A$='N' OR A$='y' OR A$='n'  
  
10 DO WHILE ABS(EPS) > 0.001  
20 GOSUB 2000 : REM ITERATION SUBROUTINE  
30 LOOP  
  
10 IX=0 : REM INTEGER LOOP 1 -> 100  
20 DO IX=IX+1  
30 LOOP WHILE IX < 101
```

# WINDOW

**Token:** \$FE \$1A

**Format:** **WINDOW left, top, right, bottom [,clear]**

**Usage:** Sets the text screen window.

**left** = left column

**top** = top row

**right** = right column

**bottom** = bottom row

**clear** = clear text window flag

The row values count from 0 to 24.

The column values count from 0 to 79 or 39 depending on the screen mode.

**Remarks:** There can be only one window on the screen. Striking the HOME key twice or printing CHR\$(19)CHR\$(19) will reset the window to the default full screen.

**Example:** Using **WINDOW**

```
10 WINDOW 0,1,79,24      :REM SCREEN WITHOUT TOP ROW
20 WINDOW 0,0,79,24,1    :REM FULL SCREEN WINDOW CLEARED
30 WINDOW 0,12,79,24     :REM LOWER HALF OF SCREEN
40 WINDOW 20,5,59,15     :REM SMALL CENTRED WINDOW
```

# XOR

**Token:** \$E9

**Format:** operand **XOR** operand

**Usage:** The Boolean **XOR** operator performs a bit-wise logical exclusive OR operation on two 16-bit values. Integer operands are used as they are. Real operands are converted to a signed 16 bit integer. Logical operands are converted to 16 bit integer using \$FFFF, decimal -1 for TRUE and \$0000, decimal 0, for FALSE.

```
0 XOR 0  -> 0
0 XOR 1  -> 1
1 XOR 0  -> 1
1 XOR 1  -> 0
```

**Remarks:** The result is of integer type. If the result is used in a logical context, the value of 0 is regarded as FALSE, all other, nonzero values are regarded as TRUE.

**Example:** Using **XOR**

```
FOR I = 0 TO 8: PRINT I XOR 5;; NEXT I
5 4 7 6 1 0 3 2 13
```



# C

## APPENDIX

# Special Keyboard Controls and Sequences

- PETSCII Codes and CHR\$
- Control codes
- Shifted codes
- Escape Sequences



# PETSCII CODES AND CHR\$

You can use the `PRINT CHR$(X)` statement to print a character. Below is the full table of PETSCII codes you can print by index. For example, by using index 65 from the table below as: `PRINT CHR$(65)` you will print the letter A.

You can also do the reverse with the `ASC` statement. For example: `PRINT ASC("A")` will output 65, which matches with the code in the table.

|                                                                                                                                                                                             |                                                                                     |                                                                                               |             |             |
|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------|-------------|-------------|
| <b>0</b>                                                                                                                                                                                    |                                                                                     | <b>19</b>    | <b>42</b> * | <b>65</b> A |
| <b>1</b>                                                                                                                                                                                    |                                                                                     | <b>20</b>    | <b>43</b> + | <b>66</b> B |
| <b>2</b> UNDERLINE ON                                                                                                                                                                       |                                                                                     | <b>21</b>                                                                                     | <b>44</b> , | <b>67</b> C |
| <b>3</b>                                                                                                                                                                                    |                                                                                     | <b>22</b>                                                                                     | <b>45</b> - | <b>68</b> D |
| <b>4</b>                                                                                                                                                                                    |                                                                                     | <b>23</b>                                                                                     | <b>46</b> . | <b>69</b> E |
| <b>5</b> WHITE                                                                                                                                                                              |                                                                                     | <b>24</b>                                                                                     | <b>47</b> / | <b>70</b> F |
| <b>6</b>                                                                                                                                                                                    |                                                                                     | <b>25</b>                                                                                     | <b>48</b> 0 | <b>71</b> G |
| <b>7</b> BELL                                                                                                                                                                               |                                                                                     | <b>26</b>                                                                                     | <b>49</b> 1 | <b>72</b> H |
| <b>8</b>                                                                                                                                                                                    |                                                                                     | <b>27</b> ESCAPE                                                                              | <b>50</b> 2 | <b>73</b> I |
| <b>9</b>                                                                                                                                                                                    |                                                                                     | <b>28</b> RED                                                                                 | <b>51</b> 3 | <b>74</b> J |
| <b>10</b> LINEFEED                                                                                                                                                                          |                                                                                     | <b>29</b>    | <b>52</b> 4 | <b>75</b> K |
| <b>11</b> DISABLE<br>     |                                                                                     | <b>30</b> GREEN                                                                               | <b>53</b> 5 | <b>76</b> L |
| <b>12</b> ENABLE<br>  |                                                                                     | <b>31</b> BLUE                                                                                | <b>54</b> 6 | <b>77</b> M |
| <b>13</b>                                                                                                                                                                                   | <b>RETURN</b>                                                                       | <b>32</b>  | <b>55</b> 7 | <b>78</b> N |
| <b>14</b>                                                                                                                                                                                   | LOWER CASE                                                                          | <b>33</b> !                                                                                   | <b>56</b> 8 | <b>79</b> O |
| <b>15</b>                                                                                                                                                                                   | BLINK ON                                                                            | <b>34</b> "                                                                                   | <b>57</b> 9 | <b>80</b> P |
| <b>16</b>                                                                                                                                                                                   |                                                                                     | <b>35</b> #                                                                                   | <b>58</b> : | <b>81</b> Q |
| <b>17</b>                                                                                                                                                                                   |  | <b>36</b> \$                                                                                  | <b>59</b> ; | <b>82</b> R |
| <b>18</b>                                                                                                                                                                                   | RVS ON                                                                              | <b>37</b> %                                                                                   | <b>60</b> < | <b>83</b> S |
|                                                                                                                                                                                             |                                                                                     | <b>38</b> &                                                                                   | <b>61</b> = | <b>84</b> T |
|                                                                                                                                                                                             |                                                                                     | <b>39</b> '                                                                                   | <b>62</b> > | <b>85</b> U |
|                                                                                                                                                                                             |                                                                                     | <b>40</b> (                                                                                   | <b>63</b> ? | <b>86</b> V |
|                                                                                                                                                                                             |                                                                                     | <b>41</b> )                                                                                   | <b>64</b> @ | <b>87</b> W |

|            |          |            |               |            |              |               |            |          |
|------------|----------|------------|---------------|------------|--------------|---------------|------------|----------|
| <b>88</b>  | <b>X</b> | <b>115</b> | <b>▀</b>      | <b>141</b> | <b>SHIFT</b> | <b>RETURN</b> | <b>166</b> | <b>☒</b> |
| <b>89</b>  | <b>Y</b> | <b>116</b> | <b>□</b>      | <b>142</b> | UPPERCASE    |               | <b>167</b> | <b>□</b> |
| <b>90</b>  | <b>Z</b> | <b>117</b> | <b>□</b>      | <b>143</b> | BLINK OFF    |               | <b>168</b> | <b>☒</b> |
| <b>91</b>  | <b>[</b> | <b>118</b> | <b>☒</b>      | <b>144</b> | BLACK        |               | <b>169</b> | <b>☒</b> |
| <b>92</b>  | <b>£</b> | <b>119</b> | <b>□</b>      | <b>145</b> | <b>↑</b>     |               | <b>170</b> | <b>□</b> |
| <b>93</b>  | <b>]</b> | <b>120</b> | <b>☒</b>      | <b>146</b> | RVS<br>OFF   |               | <b>171</b> | <b>☒</b> |
| <b>94</b>  | <b>↑</b> | <b>121</b> | <b>□</b>      | <b>147</b> | CLR<br>HOME  |               | <b>172</b> | <b>□</b> |
| <b>95</b>  | <b>←</b> | <b>122</b> | <b>☒</b>      | <b>148</b> | INST<br>DEL  |               | <b>173</b> | <b>□</b> |
| <b>96</b>  | <b>⤒</b> | <b>123</b> | <b>⤒</b>      | <b>149</b> | BROWN        |               | <b>174</b> | <b>⤒</b> |
| <b>97</b>  | <b>⤓</b> | <b>124</b> | <b>⤓</b>      | <b>150</b> | LT. RED      |               | <b>175</b> | <b>⤓</b> |
| <b>98</b>  | <b>⤔</b> | <b>125</b> | <b>⤔</b>      | <b>151</b> | DK. GRAY     |               | <b>176</b> | <b>⤔</b> |
| <b>99</b>  | <b>⤖</b> | <b>126</b> | <b>π</b>      | <b>152</b> | GRAY         |               | <b>177</b> | <b>⤖</b> |
| <b>100</b> | <b>⤘</b> | <b>127</b> | <b>⤘</b>      | <b>153</b> | LT. GREEN    |               | <b>178</b> | <b>⤘</b> |
| <b>101</b> | <b>⤙</b> | <b>128</b> |               | <b>154</b> | LT. BLUE     |               | <b>179</b> | <b>⤙</b> |
| <b>102</b> | <b>⤚</b> | <b>129</b> | ORANGE        | <b>155</b> | LT. GRAY     |               | <b>180</b> | <b>⤚</b> |
| <b>103</b> | <b>⤛</b> | <b>130</b> | UNDERLINE OFF | <b>156</b> | PURPLE       |               | <b>181</b> | <b>⤛</b> |
| <b>104</b> | <b>⤜</b> | <b>131</b> |               | <b>157</b> | <b>←</b>     |               | <b>182</b> | <b>⤜</b> |
| <b>105</b> | <b>⤝</b> | <b>132</b> |               | <b>158</b> | YELLOW       |               | <b>183</b> | <b>⤝</b> |
| <b>106</b> | <b>⤞</b> | <b>133</b> | F1            | <b>159</b> | CYAN         |               | <b>184</b> | <b>⤞</b> |
| <b>107</b> | <b>⤟</b> | <b>134</b> | F3            | <b>160</b> | SPACE        |               | <b>185</b> | <b>⤟</b> |
| <b>108</b> | <b>⤠</b> | <b>135</b> | F5            | <b>161</b> | <b>□</b>     |               | <b>186</b> | <b>⤠</b> |
| <b>109</b> | <b>⤡</b> | <b>136</b> | F7            | <b>162</b> | <b>□</b>     |               | <b>187</b> | <b>⤡</b> |
| <b>110</b> | <b>⤢</b> | <b>137</b> | F2            | <b>163</b> | <b>□</b>     |               | <b>188</b> | <b>⤢</b> |
| <b>111</b> | <b>⤣</b> | <b>138</b> | F4            | <b>164</b> | <b>□</b>     |               | <b>189</b> | <b>⤣</b> |
| <b>112</b> | <b>⤤</b> | <b>139</b> | F6            | <b>165</b> | <b>□</b>     |               | <b>190</b> | <b>⤤</b> |
| <b>113</b> | <b>⤥</b> | <b>140</b> | F8            |            |              |               | <b>191</b> | <b>⤥</b> |

NOTE: Codes for 192 to 223 are the equal to 96-127. Codes 224 to 254 equal to 160-190 and code 255 equal to 126.

# CONTROL CODES

| Keyboard Control                   | Function                                                                                                                                                                                                                              |
|------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <b>CTRL</b> + <b>1</b> to <b>8</b> | Choose from the first range of colours.                                                                                                                                                                                               |
| <b>CTRL</b> + <b>T</b>             | Backspace the character immediately to the left and to shift all rightmost characters one position to the left. This is the same function as the Backspace key.                                                                       |
| <b>CTRL</b> + <b>Z</b>             | Tabs the cursor to the left.                                                                                                                                                                                                          |
| <b>CTRL</b> + <b>E</b>             | Restores the colour of the cursor back to the default white.                                                                                                                                                                          |
| <b>CTRL</b> + <b>Q</b>             | moves the cursor down one line at a time. This is the same function produced by the Cursor Down key.                                                                                                                                  |
| <b>CTRL</b> + <b>G</b>             | produces a bell tone.                                                                                                                                                                                                                 |
| <b>CTRL</b> + <b>J</b>             | is a line feed and moves the cursor down one row. This is the same function produced by the <b>↓</b> key.                                                                                                                             |
| <b>CTRL</b> + <b>U</b>             | backs up to the start of the previous word, or unbroken string of characters. If there are no characters between the current cursor position and the start of the line, the cursor will move to the first column of the current line. |
| <b>CTRL</b> + <b>W</b>             | advances forward to the start of the next word, or unbroken string of characters. If there are no characters between the current cursor position and the end of the line, the cursor will move to the first column of the next line.  |

| Keyboard Control       | Function                                                                                                                                                                                                                                                                                                                                   |
|------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <b>CTRL</b> + <b>B</b> | turns on underline text mode. Turn off underline mode by pressing <b>ESC</b> then <b>O</b> .                                                                                                                                                                                                                                               |
| <b>CTRL</b> + <b>N</b> | changes the text case mode from uppercase to lowercase.                                                                                                                                                                                                                                                                                    |
| <b>CTRL</b> + <b>M</b> | is the carriage return. This is the same function as the <b>RETURN</b> key.                                                                                                                                                                                                                                                                |
| <b>CTRL</b> + <b>]</b> | is the same function as <b>→</b> .                                                                                                                                                                                                                                                                                                         |
| <b>CTRL</b> + <b>I</b> | tabs forward to the right.                                                                                                                                                                                                                                                                                                                 |
| <b>CTRL</b> + <b>X</b> | sets or clears the current screen column as a tab position. <b>CTRL</b> + <b>I</b> or <b>Z</b> will jump to all positions set with <b>X</b> . When there are no more tab positions, the cursor will stay at the end of the line with <b>CTRL</b> and <b>I</b> , or move to the start of the line in the case of <b>CTRL</b> and <b>Z</b> . |
| <b>CTRL</b> + <b>K</b> | locks the uppercase/lowercase mode switch usually performed with <b>M</b> and <b>SHIFT</b> keys.                                                                                                                                                                                                                                           |
| <b>CTRL</b> + <b>L</b> | enables the uppercase/lowercase mode switch that is performed with the <b>M</b> and <b>SHIFT</b> keys.                                                                                                                                                                                                                                     |
| <b>CTRL</b> + <b>[</b> | is the same as pressing the <b>ESC</b> key.                                                                                                                                                                                                                                                                                                |
| <b>CTRL</b> + <b>*</b> | enters the Matrix Mode Debugger.                                                                                                                                                                                                                                                                                                           |

# SHIFTED CODES

| Keyboard Control           | Function                                                                                                |
|----------------------------|---------------------------------------------------------------------------------------------------------|
| <b>SHIFT</b> + <b>INST</b> | Insert a character in the current cursor position and move all characters to the right by one position. |
| <b>SHIFT</b> + <b>HOME</b> | Clear home, clear the entire screen and move the cursor to the home position.                           |

# ESCAPE SEQUENCES

To perform an Escape Sequence, press and release the **ESC** key. Then press one of the following keys to perform the sequence:

| Key      | Sequence                                                                                                                                                                             |
|----------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <b>X</b> | Clears the screen and toggles between 40 and 80 column modes.                                                                                                                        |
| <b>@</b> | Clears the screen starting from the cursor to the end of the screen.                                                                                                                 |
| <b>A</b> | Enables the auto-insert mode. Any keys pressed will insert before other characters.                                                                                                  |
| <b>B</b> | Sets the bottom-right window area of the screen at the cursor position. All typed characters and screen activity will be restricted to the area. Also see <b>ESC</b> then <b>T</b> . |
| <b>C</b> | Disables auto-insert mode, going back to overwrite mode.                                                                                                                             |
| <b>D</b> | Deletes the current line and moves other lines up one position.                                                                                                                      |
| <b>E</b> | Sets the cursor to non-flashing mode.                                                                                                                                                |
| <b>F</b> | Sets the cursor to regular flashing mode.                                                                                                                                            |
| <b>G</b> | Enables the bell which can be sounded using <b>CTRL</b> and <b>G</b> .                                                                                                               |
| <b>H</b> | Disable the bell so that pressing <b>CTRL</b> and <b>G</b> will have no effect.                                                                                                      |
| <b>I</b> | Inserts an empty line in the current cursor position and moves all subsequent lines down one position.                                                                               |

| <b>Key</b> | <b>Sequence</b>                                                                                                                                                                                             |
|------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| <b>J</b>   | Moves the cursor to start of current line.                                                                                                                                                                  |
| <b>K</b>   | Move to end of the last non-white-space character on the current line.                                                                                                                                      |
| <b>L</b>   | Enables scrolling when the cursor down key is pressed at the bottom of the screen.                                                                                                                          |
| <b>M</b>   | Disables scrolling. When pressing the cursor down key at the bottom on the screen, the cursor will move to the top of the screen. The cursor is restricted at the top of the screen with the Cursor up key. |
| <b>O</b>   | Cancels the quote, reverse, underline and flash modes.                                                                                                                                                      |
| <b>P</b>   | Erases all characters from the cursor to the start of current line.                                                                                                                                         |
| <b>Q</b>   | Erases all characters from the cursor to the end of current line.                                                                                                                                           |
| <b>S</b>   | Switches the VIC-IV to colour range 16-31. These colours can be accessed with <b>CTRL</b> and keys <b>1</b> to <b>8</b> or <b>M</b> and keys <b>1</b> to <b>8</b> .                                         |
| <b>T</b>   | Set top-left window area of the screen at the cursor position. All typed characters and screen activity will be restricted to the area. Also see <b>ESC</b> then <b>B</b> .                                 |
| <b>U</b>   | Switches the VIC-IV to colour range 0-15. These colours can be accessed with <b>CTRL</b> and keys <b>1</b> to <b>8</b> or <b>M</b> and keys <b>1</b> to <b>8</b> .                                          |

| Key                  | Sequence                                                                                                         |
|----------------------|------------------------------------------------------------------------------------------------------------------|
| <b>V</b>             | Scrolls the entire screen up one line.                                                                           |
| <b>W</b>             | Scrolls the entire screen down one line.                                                                         |
| <b>X</b>             | Toggles the 40/80 column display.<br>The screen will also clear home.                                            |
| <b>Y</b>             | Set the default tab stops (every 8 spaces) for the entire screen.                                                |
| <b>Z</b>             | Clears all the tab stops. Any tabbing with <b>CTRL</b> and <b>I</b> will move the cursor to the end of the line. |
| <b>1</b> to <b>8</b> | Choose from the second range of colours.                                                                         |

# D APPENDIX

## The MEGA65 Keyboard

- **Hardware Accelerated Keyboard Scanning**
- **Keyboard Theory of Operation**
- **C65 Keyboard Matrix**
- **Synthetic Key Events**
- **Keyboard LED Control**
- **Native Keyboard Matrix**



The MEGA65 has a full mechanical keyboard which is compatible with the C65 and C64 keyboards, and features four distinct cursor keys which work in both C64 and C65 mode, as well as eleven new C65 keys that normally work only in C65 mode.

## HARDWARE ACCELERATED KEYBOARD SCANNING

To make use of the new extended keyboard easier, the MEGA65 features a hardware accelerated keyboard scan circuit, that provides ASCII (not PETSCII!) codes for keys and key-combinations. This makes it very simple to use the full capabilities of the MEGA65's keyboard, including the entry of ASCII symbols such as {, \_ and |, which are not possible to type on a normal C64 and C128 keyboards.

The hardware accelerated keyboard scanner has a buffer of 3 keys, which helps to make it easier to read from the keyboard without having check it too regularly. Further, the hardware accelerated keyboard scanner supports most Latin-1 code-page characters, allowing the entry of many accented characters. These keys are entered by holding down the  key and pressing other keys or key-combinations. The use of ASCII or Latin-1 symbols not present in the PETSCII character set requires the use of a font that contains these symbols, and software which supports them.

The hardware accelerated keyboard scanner is very simple to use: First, make sure that you have the MEGA65 IO context activated, then read memory location \$D610 (decimal 54800). If the register contains zero, no key has been pressed. Otherwise the value will be the ASCII code of the most recent key or key-combination that has been pressed. Reading \$D610 again will continue to read the same value until you POKE any value into \$D610. This clears the key from the input buffer.

The hardware accelerated keyboard scanner also provides a register that indicates which of the modifier keys are currently being held down. This is accessed via the read-only register \$D611 (decimal 54801):

**Bit 0** Right 

**Bit 4** 

**Bit 1** Left 

**Bit 5** 

**Bit 2** 

**Bit 6** 

**Bit 3** 

**Bit 7** Reserved

Note that the hardware accelerated keyboard scanner operates independently of the C64 or C65 KERNAL keyboard scanning routines. That is, the KERNAL will still have any keys that you have entered buffered in the normal way. For assembly language programs the easiest solution to this is to disable interrupts via the SEI instruction. This prevents the KERNAL keyboard scanner from running.

## Latin-1 Keyboard Map

## KEYBOARD THEORY OF OPERATION

The MEGA65 keyboard is a full mechanical keyboard, constructed as a matrix. Every key switch is fitted with a diode, which allows the keyboard hardware to detect when any combination of keys are pressed at the same time. This matrix is scanned by the firmware in the CPLD chip on the keyboard PCB many thousands of times per second. The matrix arrangement of the MEGA65 keyboard does not use the C65 matrix layout.

Instead, the CPLD also sorts the natural matrix of the keyboard into the C65 keyboard matrix order, and transmits this serially via the keyboard cable to the MEGA65 mainboard. The MEGA65 core reads this serial data and uses it to reconstruct a C65-compatible virtual keyboard in the FPGA. This virtual keyboard also takes input from the on-screen-keyboard, synthetic keyboard injection mechanism and/or other keyboard input sources depending on the MEGA65 model.

The end-to-end latency of the keyboard is less than one milli-second.

## C65 KEYBOARD MATRIX

The MEGA65 keyboard presents to legacy software as a C65-compatible keyboard. In this mode all keys are available for standard PETSCII scanning as per normal. There is also a hardware accelerated mechanism for detecting arbitrary combinations of keys that are held down. This is via \$D614 (decimal 54804). Writing a value between 0 and 8 to this register selects the corresponding row of the C65 keyboard matrix, which can then be read back from \$D613. If a bit is zero, then it means that the key is being pressed. If the bit is one, then the key is not being pressed.

The left and up cursor keys are special, because they logically press cursor right or down, and the right shift key. To be able to differentiate between these two situations, you can read \$D60F: Bit 0 is the state of the left cursor key and bit 1 is the state of the up cursor key.

The C65 keyboard matrix layout is as follows:

|          | <b>0</b>    | <b>1</b>      | <b>2</b> | <b>3</b> | <b>4</b> | <b>5</b> | <b>6</b>       | <b>7</b>    | <b>8</b>     |
|----------|-------------|---------------|----------|----------|----------|----------|----------------|-------------|--------------|
| <b>0</b> | INST<br>DEL | 3             | 5        | 7        | 9        | +        | £              | 1           | NO<br>SCROLL |
| <b>1</b> | RETURN      | W             | R        | Y        | I        | P        | *              | ←           | TAB          |
| <b>2</b> | →           | A             | D        | G        | J        | L        | ;              | CTRL        | ALT          |
| <b>3</b> | F7          | 4             | 6        | 8        | 0        | -        | CLR<br>HOME    | 2           | HELP         |
| <b>4</b> | F1          | Z             | C        | B        | M        | .        | SHIFT<br>right | SPC         | F9           |
| <b>5</b> | F3          | S             | F        | H        | K        | :        | =              | !           | F11          |
| <b>6</b> | F5          | E             | T        | U        | O        | @        | ↑              | Q           | F13          |
| <b>7</b> | ↓           | SHIFT<br>left | X        | V        | N        | ,        | /              | RUN<br>STOP | ESC          |

Note that the keyboard matrix is identical to the C64 keyboard matrix, except for the addition of one extra column on the right hand-side. The cursor left and up keys on the MEGA65 and C65 are implemented as cursor right and down, but with the right shift key applied. This enables them to work from in C64 mode. The CAPS key is not part of the matrix, but has its own dedicated line. Its status can be read from bit 6 of register \$D611 (decimal 54801):

The numbers across the top indicate the columns of the matrix, and the numbers down the left indicate the rows. The unique scan code of a key is calculated by multiplying the column by eight, and adding the row. For example, the CLR HOME key is in column 6 and row 3. Thus its scan code is  $6 \times 8 + 3 = 51$ .

## SYNTHETIC KEY EVENTS

The MEGA65 keyboard interface logic allows the use of a variety of keyboard types and alternatives. This is partly to cater for the early development on general purpose FPGA boards, the MEGAphone with its touch interface, and the desktop versions of the MEGA65 architecture. The depressing of up to 3 three keys can be simulated via the registers \$D615 - \$D617 (decimal 54,805 - 54,807). By setting the lower 7 bits of these registers to any C65 keyboard scan code, the MEGA65 will behave as though that key is being held down. The RESTORE key exists outside of the keyboard matrix, as on the C64. To simulate holding the RESTORE key down, write \$52 (ASCII code for a capital R), and to simulate a quick tap of the RESTORE, write \$72 (ASCII code for a lower-case R). Another value must be written after the \$72 value has been written, if you wish to simulate multiple presses of the RESTORE key.

To release a key, write \$7F (decimal 127) to the register containing the active key press. For example, to simulate briefly pressing the \* key, the following could be used:

```
POKE DEC("D615"),6*8+1:FOR I=1TO100:NEXT:POKE DEC("D615"),127
```

The FOR loop provides a suitable delay to simulate holding the key for a short time. All statements should be on a single line like this, if entered directly into the BASIC interpreter, because otherwise the MEGA65 will continue to act as though the \* key is being held down, making it rather difficult to enter the other commands!

## KEYBOARD LED CONTROL

The LEDs on the MEGA65's keyboard are normally controlled automatically by the system. However, it is also possible to place them under user control. This is activated by setting bit 7 (decimal 128) of \$D61D (decimal 54813). The lower bits indicate which keyboard LED to set. Values 0 through 11 correspond to the red, green and blue channels of the four LEDs. The table below shows the specific values:

- 0** left-half of DRIVE LED, RED
- 1** left-half of DRIVE LED, GREEN
- 2** left-half of DRIVE LED, BLUE
- 3** right-half of DRIVE LED, RED
- 4** right-half of DRIVE LED, GREEN
- 5** right-half of DRIVE LED, BLUE
- 6** left-half of POWER LED, RED
- 7** left-half of POWER LED, GREEN
- 8** left-half of POWER LED, BLUE
- 9** right-half of POWER LED, RED
- 10** right-half of POWER LED, GREEN
- 11** right-half of POWER LED, BLUE

Register \$D61E (decimal 54814) is used to specify the intensity that should be given to a specific LED (value between 0 and 255).

Note that whatever value is in \$D61E gets written to whatever register is currently selected in \$D61D. Therefore to safely change the intensity of one specific LED ensure \$D61D is

set to 255 first. This prevents affecting another LED when we set the intended intensity value into \$D61E. Now select the target LED by setting \$D61D to  $128 + x$ , where x is a value from the table above. Hold the \$D61D, \$D61E configuration for approximately one millisecond to give the keyboard logic enough time to pick up the new intensity value for the selected LED.

To return the keyboard LEDs to hardware control, clear bit 7 of \$D61D.

For example to pulse the keyboard LEDs red and blue, the following programme could be used:

```
10 REM ENABLE SOFTWARE CONTROL OF LEDS
20 POKEDEC("D61D"),128
30 REM SET ALL LEDS TO OFF
40 POKEDEC("D61E"),0
50 FORI=0TO11:POKEDEC("D61D"),128+I:NEXT
60 REM SELECT RED CHANNEL OF RIGHT MOST LED
70 POKEDEC("D61D"),128
80 REM CYCLE FROM BLACK TO RED AND BACK
90 FORI=0TO255:POKEDEC("D61E"),I:NEXT
100 FORI=255TO0STEP-1:POKEDEC("D61E"),I:NEXT
110 REM SELECT BLUE CHANNEL OF LEFT MOST LED
120 POKEDEC("D61D"),128+8
130 REM CYCLE FRO BLACK TO BLUE AND BACK
140 FORI=0TO255:POKEDEC("D61E"),I:NEXT
150 FORI=255TO0STEP-1:POKEDEC("D61E"),I:NEXT
160 GOT070
```

## NATIVE KEYBOARD MATRIX

The native keyboard matrix is accessible only from the CPLD on the MEGA65's keyboard. If you are programming the MEGA65 computer, you should not need to use this.

0 F5

4 <

1 9

5 INST  
DEL

2 |

6 CLR  
HOME

3 K

7 O

|    |                 |    |               |
|----|-----------------|----|---------------|
| 8  | F3              | 36 | V             |
| 9  | 8               | 37 | SPACE         |
| 10 | U               | 38 | 0             |
| 11 | J               | 39 | L             |
| 12 | M               | 40 | CAPS<br>LOCK  |
| 13 | →               | 41 | 4             |
| 14 | £               | 42 | E             |
| 15 | =               | 43 | D             |
| 16 | F1              | 44 | C             |
| 17 | 7               | 45 | Reserved      |
| 18 | Y               | 46 | HELP          |
| 19 | H               | 47 | RETURN        |
| 20 | N               | 48 | ALT           |
| 21 | ↓               | 49 | 3             |
| 22 | -               | 50 | W             |
| 23 | ;               | 51 | S             |
| 24 | Reserved        | 52 | X             |
| 25 | 6               | 53 | ↑ (cursor up) |
| 26 | T               | 54 | F13           |
| 27 | G               | 55 | ↑ (next to *) |
| 28 | B               | 56 | ESC           |
| 29 | ← (cursor left) | 57 | 2             |
| 30 | +               | 58 | Q             |
| 31 | :               | 59 | A             |
| 32 | NO<br>SCRL      | 60 | Z             |
| 33 | 5               | 61 | right SHIFT   |
| 34 | R               | 62 | F11           |
| 35 | F               |    |               |

**63** \*

**64** Reserved

**65** 1

**66** Reserved

**67** Reserved

**68** left  and 

**69** /

**70** 

**71** @

**72**



**73**



(next to 1)

**74**



**75**



**76**



**77** >

**78**



**79** P



# E

## APPENDIX

# Decimal, Binary and Hexadecimal

- **Numbers**
- **Notations and Bases**
- **Operations**
- **Signed and Unsigned Numbers**
- **Bit-wise Logical Operators**
- **Converting Numbers**



# NUMBERS

Simple computer programs, such as most of the introductory BASIC programs in this book, do not require an understanding of mathematics or much knowledge about the inner workings of the computer. This is because BASIC is considered a high-level programming language. It lets us program the computer somewhat indirectly, yet still gives us control over the computer's features. Most of the time, we don't need to concern ourselves with the computer's internal architecture, which is why BASIC is user friendly and accessible.

As you acquire deeper knowledge and become more experienced, you will often want to instruct the computer to perform complex or specialised tasks that differ from the examples given in this book. Perhaps for reasons of efficiency, you may also want to exercise direct and precise control over the contents of the computer's memory. This is especially true for applications that deal with advanced graphics and sound. Such operations are closer to the hardware and are therefore considered low-level. Some simple mathematical knowledge is required to be able to use these low-level features effectively.

The collective position of the tiny switches inside the computer—whether each switch is on or off—is the state of the computer. It is natural to associate numerical concepts with this state. Numbers let us understand and manipulate the internals of the machine via logic and arithmetic operations. Numbers also let us encode the two essential and important pieces of information that lie within every computer program: *instructions* and *data*.

A program's instructions tell a computer what to do and how to do it. For example, the action of outputting a text string to the screen via the statement **PRINT** is an instruction. The action of displaying a sprite and the action of changing the screen's border colour are instructions too. Behind the scenes, every instruction you give to the computer is associated with one or more numbers (which, in turn, correspond to the tiny switches inside the computer being switched on or off). Most of the time these instructions won't look like numbers to you. Instead, they might take the form of statements in BASIC.

A program's data consists of information. For example, the greeting "HELLO MEGA65!" is PETSCII character data in the form of a text string. The graphical design of a sprite might be pixel data in the form of a hero for a game. And the colour data of the screen's border might represent orange. Again, behind the scenes, every piece of data you give to the computer is associated with one or more numbers. Data is sometimes given directly next to the statement to which it applies. This data is referred to as a parameter or argument (such as when changing the screen colour with a **BACKGROUND 1** statement). Data may also be given within the program via the BASIC statement **DATA** which accepts a list of comma-separated values.

All such numbers—regardless of whether they represent instructions or data—reside in the computer's memory. Although the computer's memory is highly structured, the computer

does not distinguish between instructions and data, nor does it have separate areas of memory for each kind of information. Instead, both are stored in whichever memory location is considered convenient. Whether a given memory location's contents is part of the program's instructions or is part of the program's data largely depends on your viewpoint, the program being written and the needs of the programmer.

Although BASIC is a high-level language, it still provides statements that allow programmers to manipulate the computer's memory efficiently. The statement **PEEK** lets us read the information from a specified memory location: we can inspect the contents of a memory address. The statement **POKE** lets us store information inside a specified memory location: we can modify the contents of a memory address so that it is set to a given value.

## NOTATIONS AND BASES

We now take a look at numbers.

Numbers are ideas about quantity and magnitude. In order to manipulate numbers and determine relationships between them, it's important for them to have a unique form. This brings us to the idea of the symbolic representation of numbers using a positional notation. In this appendix we'll restrict our discussion to whole numbers, which are also called *integers*.

The *decimal* representation of numbers is the one with which you will be most comfortable since it is the one you were taught at school. Decimal notation uses the ten Hindu-Arabic numerals 0, 1, 2, 3, 4, 5, 6, 7, 8 and 9 and is thus referred to as a base 10 numeral system. As we shall see later, in order to express large numbers in decimal, we use a positional system in which we juxtapose digits into columns to form a bigger number.

For example, 53280 is a decimal number. Each such digit (0 to 9) in a decimal number represents a multiple of some power of 10. When a BASIC statement (such as **PEEK** or **POKE**) requires an integer as a parameter, that parameter is given in the decimal form.

Although the decimal notation feels natural and comfortable for humans to use, modern computers, at their most fundamental level, use a different notation. This notation is called *binary*. It is also referred to as a base 2 numeral system because it uses only two Hindu-Arabic numerals: 0 and 1. Binary reflects the fact that each of the tiny switches inside the computer must be in exactly one of two mutually exclusive states: on or off. The number 0 is associated with off and the number 1 is associated with on. Binary is the simplest notation that captures this idea. In order to express large numbers in binary, we use a positional system in which we juxtapose digits into columns to form a bigger number and prefix it with a % sign.

For example, %10010110 is a binary number. Each such digit (0 or 1) in a binary number represents a multiple of some power of 2.

We'll see later how we can use special BASIC statements to manipulate the patterns of ones and zeros present in a binary number to change the state of the switches associated with it. Effectively, we can toggle individual switches on or off, as needed.

A third notation called *hexadecimal* is also often used. This is a base 16 numeral system. Because it uses more than ten digits, we need to use some letters to represent the extra digits. Hexadecimal uses the ten Hindu-Arabic digits 0 to 9 as well as the six Latin alphabetic characters as "digits" (A, B, C, D, E and F) to represent the numbers 10 to 15. This gives a total of sixteen symbols for the numbers 0 to 15. To express a large number in hexadecimal, we use a positional system in which we juxtapose digits into columns to form a bigger number and prefix it with a \$ sign.

For example, \$E7 is a hexadecimal number. Each such digit (0 to 9 and A to F) in a hexadecimal number represents a multiple of some power of 16.

Hexadecimal is not often used when programming in BASIC. It is more commonly used when programming in low-level languages like machine code or assembly language. It also appears in computer memory maps and its brevity makes it a useful notation, so it is described here.

Always remember that decimal, binary and hexadecimal are just different notations for numbers. A notation just changes the way the number is written (i.e., the way it looks on paper or on the screen), but its intrinsic value remains unchanged. A notation is essentially different ways of representing the same thing. The reason that we use different notations is that each notation lends itself more naturally to a different task.

When using decimal, binary and hexadecimal for extended periods you may find it handy to have a scientific pocket calculator with a programmer mode. Such calculators can convert between bases with the press of a button. They can also add, subtract, multiply and divide, and perform various bit-wise logical operations. See Chapter/Appendix R on page R-3 as it contains a Base Conversion table for decimal, binary, and hexadecimal for integers between 0 and 255.

The BASIC listing for this appendix is a utility program that converts individual numbers into different bases. It can also convert multiple numbers within a specified range.

Although these concepts might be new now, with some practice they'll soon seem like second nature. We'll look at ways of expressing numbers in more detail. Later, we'll also investigate the various operations that we can perform on such numbers.

# Decimal

When representing integers using decimal notation, each column in the number is for a different power of 10. The rightmost position represents the number of units (because  $10^0 = 1$ ) and each column to the left of it is 10 times larger than the column before it. The rightmost column is called the units column. Columns to the left of it are labelled tens (because  $10^1 = 10$ ), hundreds (because  $10^2 = 100$ ), thousands (because  $10^3 = 1000$ ), and so on.

To give an example, the integer 53280 represents the total of 5 lots of 10000, 3 lots of 1000, 2 lots of 100, 8 lots of 10 and 0 units. This can be seen more clearly if we break the integer up into distinct parts, by column.

Since

$$53280 = 50000 + 3000 + 200 + 80 + 0$$

we can present this as a table with the sum of each column at the bottom.

| TEN THOUSANDS  | THOUSANDS     | HUNDREDS     | TENS        | UNITS      |
|----------------|---------------|--------------|-------------|------------|
| $10^4 = 10000$ | $10^3 = 1000$ | $10^2 = 100$ | $10^1 = 10$ | $10^0 = 1$ |
| 5              | 0             | 0            | 0           | 0          |
|                | 3             | 0            | 0           | 0          |
|                |               | 2            | 0           | 0          |
|                |               |              | 8           | 0          |
|                |               |              |             | 0          |
| 5              | 3             | 2            | 8           | 0          |

Another way of stating this is to write the expression using multiples of powers of 10.

$$53280 = (5 \times 10^4) + (3 \times 10^3) + (2 \times 10^2) + (8 \times 10^1) + (0 \times 10^0)$$

Alternatively

$$53280 = (5 \times 10000) + (3 \times 1000) + (2 \times 100) + (8 \times 10) + (0 \times 1)$$

We now introduce some useful terminology that is associated with decimal numbers.

The rightmost digit of a decimal number is called the least significant digit, because, being the smallest multiplier of a power of 10, it contributes the least to the number's magnitude. Each digit to the left of this digit has increasing significance. The leftmost (non-zero) digit of the decimal number is called the most significant digit, because, being the largest multiplier of a power of 10, it contributes the most to the number's magnitude.

For example, in the decimal number 53280, the digit 0 is the least significant digit and the digit 5 is the most significant digit.

A decimal number  $a$  is  $m$  orders of magnitude greater than the decimal number  $b$  if  $a = b \times (10^m)$ . For example, 50000 is three orders of magnitude greater than 50, because it has three more zeros. This terminology can be useful when making comparisons between numbers or when comparing the time efficiency or space efficiency of two programs with respect to the sizes of the given inputs.

Note that unlike binary (which uses a conventional % prefix) and hexadecimal (which uses a conventional \$ prefix), decimal numbers are given no special prefix. In some textbooks you might see such numbers with a subscript instead. So decimal numbers will have a sub-scripted 10, binary numbers will have a sub-scripted 2, and hexadecimal numbers will have a sub-scripted 16.

Another useful concept is the idea of signed and unsigned decimal integers.

A signed decimal integer can be positive or negative or zero. To represent a signed decimal integer, we prefix it with either a + sign or a - sign. (By convention, zero, which is neither positive nor negative, is given the + sign.)

If, on the other hand, a decimal integer is unsigned it must be either zero or positive and does not have a negative representation. This can be illustrated with the BASIC statements **PEEK** and **POKE**. When we use **PEEK** to return the value contained within a memory location, we get back an unsigned decimal number. For example, the statement **PRINT (PEEK (49152))** outputs the contents of memory location 49152 to the screen as an unsigned decimal number. Note that the memory address that we gave to **PEEK** is itself an unsigned integer. When we use **POKE** to store a value inside a memory location, both the memory address and the value to store inside it are given as unsigned integers. For example, the statement **POKE 49152, 128** stores the unsigned decimal integer 128 into the memory address given by the unsigned decimal integer 49152.

Each memory location in the MEGA65 can store a decimal integer between 0 and 255. This corresponds to the smallest and largest decimal integers that can be represented using eight binary digits (eight bits). Also, the memory addresses are decimal integers between 0 and 65535. This corresponds to the smallest and largest decimal integers that can be represented using sixteen binary digits (sixteen bits).

Note that the largest number expressible using  $d$  decimal digits is  $10^d - 1$ . (This number will have  $d$  nines in its representation.)

## Binary

Binary notation uses powers of 2 (instead of 10 which is for decimal). The rightmost position represents the number of units (because  $2^0 = 1$ ) and each column to the left of it is 2 times larger than the column before it. Columns to the left of the rightmost column

are the twos column (because  $2^1 = 2$ ), the fours column (because  $2^2 = 4$ ), the eights column (because  $2^3 = 8$ ), and so on.

As an example, the integer %1101 0011 uses exactly eight binary digits and represents the total of 1 lot of 128, 1 lot of 64, 0 lots of 32, 1 lot of 16, 0 lots of 8, 0 lots of 4, 1 lot of 2 and 1 unit.

We can break this integer up into distinct parts, by column.

Since

$$\%1101\ 0011 = \%1000\ 0000 + \%100\ 0000 + \%00\ 0000 + \%1\ 0000 + \%0000 + \%000 + \%10 + \%1$$

we can present this as a table with the sum of each column at the bottom.

| <b>ONE<br/>HUNDRED AND<br/>TWENTY-EIGHTS</b> | <b>SIXTY-<br/>FOURS</b> | <b>THIRTY-<br/>TWOS</b> | <b>SIXTEENS</b> | <b>EIGHTS</b> | <b>FOURS</b> | <b>TWOS</b> | <b>UNITS</b> |
|----------------------------------------------|-------------------------|-------------------------|-----------------|---------------|--------------|-------------|--------------|
| $2^7 = 128$                                  | $2^6 = 64$              | $2^5 = 32$              | $2^4 = 16$      | $2^3 = 8$     | $2^2 = 4$    | $2^1 = 2$   | $2^0 = 1$    |
| 1                                            | 0                       | 0                       | 0               | 0             | 0            | 0           | 0            |
|                                              | 1                       | 0                       | 0               | 0             | 0            | 0           | 0            |
|                                              |                         | 0                       | 0               | 0             | 0            | 0           | 0            |
|                                              |                         |                         | 1               | 0             | 0            | 0           | 0            |
|                                              |                         |                         |                 | 0             | 0            | 0           | 0            |
|                                              |                         |                         |                 |               | 0            | 0           | 0            |
|                                              |                         |                         |                 |               |              | 1           | 0            |
|                                              |                         |                         |                 |               |              |             | 1            |
| 1                                            | 1                       | 0                       | 1               | 0             | 0            | 1           | 1            |

Another way of stating this is to write the expression in decimal, using multiples of powers of 2.

$$\%11010011 = (1 \times 2^7) + (1 \times 2^6) + (0 \times 2^5) + (1 \times 2^4) + (0 \times 2^3) + (0 \times 2^2) + (1 \times 2^1) + (1 \times 2^0)$$

Alternatively

$$\%11010011 = (1 \times 128) + (1 \times 64) + (0 \times 32) + (1 \times 16) + (0 \times 8) + (0 \times 4) + (1 \times 2) + (1 \times 1)$$

which is the same as writing

$$\%11010011 = 128 + 64 + 16 + 2 + 1$$

Binary has terminology of its own. Each binary digit in a binary number is called a *bit*. In an 8-bit number the bits are numbered consecutively with the least significant (i.e., rightmost) bit as bit 0 and the most significant (i.e., leftmost) bit as bit 7. In a 16-bit number the most significant bit is bit 15. A bit is said to be *set* if it equals 1. A bit is said

to be *clear* if it equals 0. When a particular bit has a special meaning attached to it, we sometimes refer to it as a *flag*.

|       |       |       |       |       |       |       |       |
|-------|-------|-------|-------|-------|-------|-------|-------|
| 1     | 1     | 0     | 1     | 0     | 0     | 1     | 1     |
| Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 |

As mentioned earlier, each memory location can store an integer between 0 and 255. The minimum corresponds to %0000 0000 and the maximum corresponds to %1111 1111, which are the smallest and largest numbers that can be represented using exactly eight bits. The memory addresses use 16 bits. The smallest memory address, represented in exactly sixteen bits, is %0000 0000 0000 0000 and this corresponds to the smallest 16-bit number. Likewise, the largest memory address, represented in exactly sixteen bits, is %1111 1111 1111 1111 and this corresponds to the largest 16-bit number.

It is often convenient to refer to groups of bits by different names. For example, eight bits make a *byte* and 1024 bytes make a *kilobyte*. Half a byte is called a *nybble*. See Chapter/Appendix R on page R-3 for the Units of Storage table for further information.

Note that the largest number expressible using  $d$  binary digits is (in decimal)  $2^d - 1$ . (This number will have  $d$  ones in its representation.)

## Hexadecimal

Hexadecimal notation uses powers of 16. Each of the sixteen hexadecimal numerals has an associated value in decimal.

| <b>Hexadecimal Numeral</b> | <b>Decimal Equivalent</b> |
|----------------------------|---------------------------|
| \$0                        | 0                         |
| \$1                        | 1                         |
| \$2                        | 2                         |
| \$3                        | 3                         |
| \$4                        | 4                         |
| \$5                        | 5                         |
| \$6                        | 6                         |
| \$7                        | 7                         |
| \$8                        | 8                         |
| \$9                        | 9                         |
| \$A                        | 10                        |
| \$B                        | 11                        |
| \$C                        | 12                        |
| \$D                        | 13                        |
| \$E                        | 14                        |
| \$F                        | 15                        |

The rightmost position in a hexadecimal number represents the number of ones (since  $16^0 = 1$ ). Each column to the left of this digit is 16 times larger than the column before it. Columns to the left of the rightmost column are the 16-column (since  $16^1 = 16$ ), the 256-column (since  $16^2 = 256$ ), the 4096-column (since  $16^3 = 4096$ ), and so on.

As an example, the integer \$A3F2 uses exactly four hexadecimal digits and represents the total of 10 lots of 4096 (because \$A = 10), 3 lots of 256 (because \$3 = 3), 15 lots of 16 (because \$F = 15) and 2 units (because \$2 = 2). We can break this integer up into distinct parts, by column.

Since

$$\$A3F2 = \$A000 + \$300 + \$F0 + \$2$$

we can present this as a table with the sum of each column at the bottom.

| <b>FOUR THOUSAND AND NINETY-SIXES</b> | <b>TWO HUNDRED AND FIFTY-SIXES</b> | <b>SIXTEENS</b> | <b>UNITS</b> |
|---------------------------------------|------------------------------------|-----------------|--------------|
| $16^3 = 4096$                         | $16^2 = 256$                       | $16^1 = 16$     | $16^0 = 1$   |
| A                                     | 0                                  | 0               | 0            |
|                                       | 3                                  | 0               | 0            |
|                                       |                                    | F               | 0            |
|                                       |                                    |                 | 2            |
| A                                     | 3                                  | F               | 2            |

Another way of stating this is to write the expression in decimal, using multiples of powers of 16.

$$\$A3F2 = (10 \times 16^3) + (3 \times 16^2) + (15 \times 16^1) + (2 \times 16^0)$$

Alternatively

$$\$A3F2 = (10 \times 4096) + (3 \times 256) + (15 \times 16) + (2 \times 1)$$

which is the same as writing

$$\$A3F2 = 40960 + 768 + 240 + 2$$

Again, like binary and decimal, the rightmost digit is the least significant and the leftmost digit is the most significant.

Each memory location can store an integer between 0 and 255, and this corresponds to the hexadecimal numbers \$00 and \$FF. The hexadecimal number \$FFFF corresponds to 65535—the largest 16-bit number.

Hexadecimal notation is often more convenient to use and manipulate than binary. Binary numbers consist of a longer sequence of ones and zeros, while hexadecimal is much shorter and more compact. This is because one hexadecimal digit is equal to exactly four bits. So a two-digit hexadecimal number comprises of eight bits with the low nybble equalling the right digit and the high nybble equalling the left digit.

Note that the largest number expressible using  $d$  hexadecimal digits is (in decimal)  $16^d - 1$ . (This number will have  $d$  \$F symbols in its representation.)

## OPERATIONS

In this section we'll take a tour of some familiar operations like counting and arithmetic, and we'll see how they apply to numbers written in binary and hexadecimal.

Then we'll take a look at various logical operations using logic gates. These operations are easy to understand. They're also very important when it comes to writing programs that have extensive numeric, graphic or sound capabilities.

## Counting

If we consider carefully the process of *counting* in decimal, this will help us to understand how counting works when using binary and hexadecimal.

Let's suppose that we're counting in decimal and that we're starting at 0. Recall that the list of numerals for decimal is (in order) 0, 1, 2, 3, 4, 5, 6, 7, 8 and 9. Notice that when

we add 1 to 0 we obtain 1, and when we add 1 to 1 we obtain 2. We can continue in this manner, always adding 1:

$$\begin{aligned}0 + 1 &= 1 \\1 + 1 &= 2 \\2 + 1 &= 3 \\3 + 1 &= 4 \\4 + 1 &= 5 \\5 + 1 &= 6 \\6 + 1 &= 7 \\7 + 1 &= 8 \\8 + 1 &= 9\end{aligned}$$

Since 9 is the highest numeral in our list of numerals for decimal, we need some way of handling the following special addition:  $9 + 1$ . The answer is that we can reuse our old numerals all over again. In this important step, we reset the units column back to 0 and (at the same time) add 1 to the tens column. Since the tens column contained a 0, this gives us  $9 + 1 = 10$ . We say we "carried" the 1 over to the tens column while the units column cycled back to 0.

Using this technique, we can count as high as we like. The principle of counting for binary and hexadecimal is very much same, except instead of using ten symbols, we get to use two symbols and sixteen symbols, respectively.

Let's take a look at counting in binary. Recall that the list of numerals for binary is (in order) just 0 and 1. So, if we begin counting at  $\%0$  and then add  $\%1$ , we obtain  $\%1$  as the result:

$$\%0 + \%1 = \%1$$

Now, the sum  $\%1 + \%1$  will cause us to perform the analogous step: we reset the units column back to zero and (at the same time) add  $\%1$  to the twos column. Since the twos column contained a  $\%0$ , this gives us  $\%1 + \%1 = \%10$ . We say we "carried" the  $\%1$  over to the twos column while the units column cycled back to  $\%0$ . If we continue in this manner we can count higher.

$$\begin{aligned}\%1 + \%1 &= \%10 \\\%10 + \%1 &= \%11 \\\%11 + \%1 &= \%100 \\\%100 + \%1 &= \%101 \\\%101 + \%1 &= \%110 \\\%110 + \%1 &= \%111 \\\%111 + \%1 &= \%1000\end{aligned}$$

Now we'll look at counting in hexadecimal. The list of numerals for hexadecimal is (in order) 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A, B, C, D, E and F. If we begin counting at \$0 and repeatedly add \$1 we obtain:

$$\begin{aligned}\$0 + \$1 &= \$1 \\\$1 + \$1 &= \$2 \\\$2 + \$1 &= \$3 \\\$3 + \$1 &= \$4 \\\$4 + \$1 &= \$5 \\\$5 + \$1 &= \$6 \\\$6 + \$1 &= \$7 \\\$7 + \$1 &= \$8 \\\$8 + \$1 &= \$9 \\\$9 + \$1 &= \$A \\\$A + \$1 &= \$B \\\$B + \$1 &= \$C \\\$C + \$1 &= \$D \\\$D + \$1 &= \$E \\\$E + \$1 &= \$F\end{aligned}$$

Now, when we compute \$F + \$1 we must reset the units column back to \$0 and add \$1 to the sixteens column as that number is "carried".

$$\$F + \$1 = \$10$$

Again, this process allows us to count as high as we like.

## Arithmetic

The standard arithmetic operations of addition, subtraction, multiplication and division are all possible using binary and hexadecimal.

Addition is done in the same way that addition is done using decimal, except that we use base 2 or base 16 as appropriate. Consider the following example for the addition of two binary numbers.

$$\begin{array}{r} \% 1 \ 1 \ 0 \\ + \% 1 \ 1 \ 1 \\ \hline \% 1 \ 1 \ 0 \ 1 \end{array}$$

We obtain the result by first adding the units columns of both numbers. This gives us %0 + %1 = %1 with nothing to carry into the next column. Then we add the twos columns of both numbers: %1 + %1 = %0 with a %1 to carry into the next column. We then add the fours columns (plus the carry) giving (%1 + %1) + %1 = %1 with a %1 to carry into the next

column. Last of all are the eights columns. Because these are effectively both zero we only concern ourselves with the carry which is %1. So  $(\%0 + \%0) + \%1 = \%1$ . Thus, %1101 is the sum.

Next is an example for the addition of two hexadecimal numbers.

$$\begin{array}{r} \$ 7 \ D \\ + \$ 6 \ 9 \\ \hline \$ E \ 6 \end{array}$$

We begin by adding the units columns of both numbers. This gives us \$D + \$9 = \$6 with a \$1 to carry into the next column. We then add the sixteens columns (plus the carry) giving  $(\$7 + \$6) + \$1 = \$E$  with nothing to carry and so \$E6 is the sum.

We now look at subtraction. As you might suspect, binary and hexadecimal subtraction follows a similar process to that of subtraction for decimal integers.

Consider the following subtraction of two binary numbers.

$$\begin{array}{r} \% 1 \ 0 \ 1 \ 1 \\ - \% \quad 1 \ 1 \ 0 \\ \hline \% \quad 1 \ 0 \ 1 \end{array}$$

Starting in the units columns we perform the subtraction  $\%1 - \%0 = \%1$ . Next, in the twos columns we perform another subtraction  $\%1 - \%1 = \%0$ . Last of all we subtract the fours columns. This time, because  $\%0$  is less than  $\%1$ , we'll need to borrow a  $\%1$  from the eights column of the top number to make the subtraction. Thus we compute  $\%10 - \%1 = \%1$  and deduct  $\%1$  from the eights column. The eights columns are now both zeros. Since  $\%0 - \%0 = \%0$  and because this is the leading digit of the result we can drop it from the final answer. This gives  $\%101$  as the result.

Let's now look at the subtraction of two hexadecimal numbers.

$$\begin{array}{r} \$ 3 \ D \\ - \$ 1 \ F \\ \hline \$ 1 \ E \end{array}$$

To perform this subtraction we compute the difference of the units columns. In order to do this, we note that because \$D is less than \$F we will need to borrow \$1 from the sixteens column of the top number to make the subtraction. Thus, we compute \$1D - \$F = \$E and also compute  $\$3 - \$1 = \$2$  in the sixteens column for the \$1 that we just borrowed. Next, we compute the difference of the sixteens column as  $\$2 - \$1 = \$1$ . This gives us a final answer of \$1E.

We won't give in depth examples of multiplication and division for binary and hexadecimal notation. Suffice to say that principles parallel those for the decimal system. Multiplication is repeated addition and division is repeated subtraction.

We will, however, point out a special type of multiplication and division for both binary and hexadecimal. This is particularly useful for manipulating binary and hexadecimal numbers.

For binary, multiplication by two is simple—just shift all bits to the left by one position and fill in the least significant bit with a %0. Division by two is simple too—just shift all bits to the right by one position and fill in the most significant bit with a %0. By doing these repeatedly we can multiply and divide by powers of two with ease.

Thus the binary number %111, when multiplied by eight has three extra zeros on the end of it and is equal to %111000. (Recall that  $2^3 = 8$ .) And the binary number %10100, when divided by four has two less digits and equals %101. (Recall that  $2^2 = 4$ .)

These are called left and right *bit shifts*. So if we say that we shift a number to the left four bit positions, we really mean that we multiplied it by  $2^4 = 16$ .

For hexadecimal, the situation is similar. Multiplication by sixteen is simple—just shift all digits to the left by one position and fill in the rightmost digit with a \$0. Division by sixteen is simple too—just shift all digits to the right by one position. By doing this repeatedly we can multiply and divide by powers of sixteen with ease.

Thus the hexadecimal number \$F, when multiplied 256 has two extra zeros on the end of it and is equal to \$F00. (Recall that  $16^2 = 256$ .) And the hexadecimal number \$EA0, when divided by sixteen has one less digit and equals \$EA. (Recall that  $16^1 = 16$ .)

## Logic Gates

There exist several so-called *logic gates*. The fundamental ones are NOT, AND, OR and XOR.

They let us set, clear and invert specific binary digits. For example, when dealing with sprites, we might want to clear bit 6 (i.e., make it equal to 0) and set bit 1 (i.e., make it equal to 1) at the same time for a particular graphics chip register. Certain logic gates will, when used in combination, let us do this.

Learning how these logic gates work is very important because they are the key to understanding how and why the computer executes programs as it does.

All logic gates accept one or more inputs and produce a single output. These inputs and outputs are always single binary digits (i.e., they are 1-bit numbers).

The NOT gate is the only gate that accepts exactly one bit as input. All other gates—AND, OR, and XOR—accept exactly two bits as input. All gates produce exactly one output, and that output is a single bit.

First, let's take a look at the simplest gate, the NOT gate.

The NOT gate behaves by inverting the input bit and returning this resulting bit as its output. This is summarised in the following table.

| INPUT X | OUTPUT |
|---------|--------|
| 0       | 1      |
| 1       | 0      |

We write NOT  $x$  where  $x$  is the input bit.

Next, we take a look at the AND gate.

As mentioned earlier, the AND gate accepts two bits as input and produces a single bit as output. The AND gate behaves in the following manner. Whenever both input bits are equal to 1 the result of the output bit is 1. For all other inputs the result of the output bit is 0. This is summarised in the following table.

| INPUT X | INPUT Y | OUTPUT |
|---------|---------|--------|
| 0       | 0       | 0      |
| 0       | 1       | 0      |
| 1       | 0       | 0      |
| 1       | 1       | 1      |

We write  $x$  AND  $y$  where  $x$  and  $y$  are the input bits.

Next, we take a look at the OR gate.

The OR gate accepts two bits as input and produces a single bit as output. The OR gate behaves in the following manner. Whenever both input bits are equal to 0 the result is 0. For all other inputs the result of the output bit is 1. This is summarised in the following table.

| INPUT X | INPUT Y | OUTPUT |
|---------|---------|--------|
| 0       | 0       | 0      |
| 0       | 1       | 1      |
| 1       | 0       | 1      |
| 1       | 1       | 1      |

We write  $x$  OR  $y$  where  $x$  and  $y$  are the input bits.

Last of all we look at the XOR gate.

The XOR gate accepts two bits as input and produces a single bit as output. The XOR gate behaves in the following manner. Whenever both input bits are equal in value the output bit is 0. Otherwise, both input bits are unequal in value and the output bit is 1. This is summarised in the following table.

| INPUT X | INPUT Y | OUTPUT |
|---------|---------|--------|
| 0       | 0       | 0      |
| 0       | 1       | 1      |
| 1       | 0       | 1      |
| 1       | 1       | 0      |

We write  $x \text{ XOR } y$  where  $x$  and  $y$  are the input bits.

Note that there do exist some other gates. They are easy to construct.

- NAND gate: this is an AND gate followed by a NOT gate
- NOR gate: this is an OR gate followed by a NOT gate
- XNOR gate: this is an XOR gate followed by a NOT gate

## SIGNED AND UNSIGNED NUMBERS

So far we've largely focused on unsigned integers. Unsigned integer have no positive or negative sign. They are always assumed to be positive. (For this purpose, zero is regarded as positive.)

Signed numbers, as mentioned earlier, can have a positive sign or a negative sign.

Signed numbers are represented by treating the most significant bit as a sign bit. This bit cannot be used for anything else. If the most significant bit is 0 then the result is interpreted as having a positive sign. Otherwise, the most significant bit is 1, and the result is interpreted as having a negative sign.

A signed 8-bit number can represent positive-sign numbers between 0 and 127, and negative-sign numbers between -1 and -128.

A signed 16-bit number can represent positive-sign numbers between 0 and 32767, and negative-sign numbers between -1 and -32768.

Reserving the most significant bit as the sign of the signed number effectively halves the range of the available positive numbers (i.e., compared to unsigned numbers), with the trade-off being that we gain an equal quantity of negative numbers instead.

To negate any signed number, every bit in the signed number must be inverted and then %1 must added to the result. Thus, negating %0000 0101 (which is the signed number +5) gives %1111 1011 (which is the signed number -5). As expected, performing the negation of this negative number gives us +5 again.

# BIT-WISE LOGICAL OPERATORS

The BASIC statements **NOT**, **AND**, **OR** and **XOR** have functionality similar to that of the logic gates that they are named after.

The **NOT** statement must be given a 16-bit signed decimal integer as a parameter. It returns a 16-bit signed decimal integer as a result.

In the following example, all sixteen bits of the signed decimal number +0 are equal to 0. The **NOT** statement inverts all sixteen bits as per the NOT gate. This sets all sixteen bits. If we interpret the result as a signed decimal number, we obtain the answer of -1.

```
PRINT (NOT 0)  
-1
```

As expected, repeating the **NOT** statement on the parameter of -1 gets us back to where we started, since all sixteen set bits become cleared.

```
PRINT (NOT -1)  
0
```

The **AND** statement must be given two 16-bit signed decimal integers as parameters. The second parameter is called the *bit mask*. The statement returns a 16-bit signed decimal integer as a result, having changed each bit as per the AND gate.

In the following example, the number +253 is used as the first parameter. As a 16-bit signed decimal integer, this is equivalent to the following number in binary: %0000 0000 1111 1101. The **AND** statement uses a bit mask as the second parameter with a 16-bit signed decimal value of +239. In binary this is the number %0000 0000 1110 1110. If we use the AND logic gate table on corresponding pairs of bits, we obtain the 16-bit signed decimal integer +237 (which is %0000 0000 1110 1100 in binary).

```
PRINT (253 AND 239)  
237
```

We can see this process more clearly in the following table.

|            |             |           |           |         |
|------------|-------------|-----------|-----------|---------|
|            | % 0 0 0 0 0 | 0 0 0 0 0 | 1 1 1 1 1 | 1 1 0 1 |
| <b>AND</b> | % 0 0 0 0 0 | 0 0 0 0 0 | 1 1 1 0 0 | 1 1 1 0 |
|            | % 0 0 0 0 0 | 0 0 0 0 0 | 1 1 1 0 0 | 1 1 0 0 |

Notice that each bit in the top row passes through unchanged wherever there is a 1 in the mask bit below it. Otherwise the bit in that position gets cleared.

The **OR** statement must be given two 16-bit signed decimal integers as parameters. The second parameter is called the *bit mask*. The statement returns a 16-bit signed decimal integer as a result, having changed each bit as per the OR gate.

In the following example, the number +240 is used as the first parameter. As a 16-bit signed decimal integer, this is equivalent to the following number in binary: %0000 0000 1111 0000. The **OR** statement uses a bit mask as the second parameter with a 16-bit signed decimal value of +19. In binary this is the number %0000 0000 0001 0011. If we use the OR logic gate table on corresponding pairs of bits, we obtain the 16-bit signed decimal integer +243 (which is %0000 0000 1111 0011 in binary).

```
PRINT (240 OR 19)  
243
```

We can see this process more clearly in the following table.

|           |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|-----------|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| %         | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 1 | 1 | 1 | 1 | 0 | 0 | 0 | 0 |
| <b>OR</b> | % | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 1 | 0 | 0 | 1 | 1 |
|           | % | 0 | 0 | 0 | 0 | 0 | 0 | 1 | 1 | 1 | 1 | 0 | 0 | 1 | 1 |

Notice that each bit in the top row passes through unchanged wherever there is a 0 in the mask bit below it. Otherwise the bit in that position gets set.

Next we look at the **XOR** statement. This statement must be given two 16-bit unsigned decimal integers as parameters. The second parameter is called the *bit mask*. The statement returns a 16-bit unsigned decimal integer as a result, having changed each bit as per the XOR gate.

In the following example, the number 14091 is used as the first parameter. As a 16-bit unsigned decimal integer, this is equivalent to the following number in binary: %0011 0111 0000 1011. The **XOR** statement uses a bit mask as the second parameter with a 16-bit unsigned decimal value of 8653. In binary this is the number %0010 0001 1100 1101. If we use the XOR logic gate table on corresponding pairs of bits, we obtain the 16-bit unsigned decimal integer 5830 (which is %0001 0110 1100 0110 in binary).

```
PRINT (XOR(14091,8653))  
5830
```

We can see this process more clearly in the following table.

|            |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |   |
|------------|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|            | % | 0 | 0 | 1 | 1 | 0 | 1 | 1 | 1 | 0 | 0 | 0 | 0 | 1 | 0 | 1 | 1 |
| <b>XOR</b> | % | 0 | 0 | 1 | 0 | 0 | 0 | 1 | 1 | 1 | 0 | 0 | 1 | 1 | 0 | 1 | 0 |
|            | % | 0 | 0 | 0 | 1 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 0 | 0 | 1 | 1 | 0 |

Notice that when the bits are equal the resulting bit is 0. Otherwise the resulting bit is 1.

Much of the utility of these bit-wise logical operators comes through combining them together into a compound statement. For example, the VIC II register to enable sprites is memory address 53269. There are eight sprites (numbered 0 to 7) with each bit corresponding to a sprite's status. Now suppose we want to turn off sprite 5 and turn on sprite 1, while leaving the statuses of the other sprites unchanged. We can do this with the following BASIC statement which combines an **AND** statement with an **OR** statement.

```
POKE 53269, (((PEEK(53269)) AND 223) OR 2)
```

The technique of using **PEEK** on a memory address and combining the result with bit-wise logical operators, followed by a **POKE** to that same memory address is very common.

## CONVERTING NUMBERS

The program below is written in BASIC. It does number conversion for you. Type it in and save it under the name "CONVERT.BAS".

To execute the program, type **RUN** and press the **RETURN** key.

The program presents you with a series of text menus. You may choose to convert a single decimal, binary or hexadecimal number. Alternatively, you may choose to convert a range of such numbers.

The program can convert numbers in the range 0 to 65535.

```
10 REM *****
20 REM *          *
30 REM *  INTEGER BASE CONVERTER  *
40 REM *          *
50 REM *****
60 POKE 0,65: BORDER 6: BACKGROUND 6: FOREGROUND 1
70 DIM P(15)
80 E$ = "STARTING INTEGER MUST BE LESS THAN OR EQUAL TO ENDING INTEGER"
90 FOR N = 8 TO 15
100 : P(N) = 2 ↑ N
110 NEXT N
120 REM *** OUTPUT MAIN MENU ***
130 PRINT CHR$(147)
140 PRINT: PRINT "INTEGER BASE CONVERTER"
150 L = 22: GOSUB 1930: PRINT L$
160 PRINT: PRINT "SELECT AN OPTION (S, M OR Q)":: PRINT
170 PRINT "[S]{SPACE*2}SINGLE INTEGER CONVERSION"
180 PRINT "[M]{SPACE*2}MULTIPLE INTEGER CONVERSION"
190 PRINT "[Q]{SPACE*2}QUIT PROGRAM"
200 GET MS
210 IF (MS="S") THEN GOSUB 260: GOTO 140
220 IF (MS="M") THEN GOSUB 380: GOTO 140
230 IF (MS="Q") THEN END
240 GOTO 200
250 REM *** OUTPUT SINGLE CONVERSION MENU ***
260 PRINT: PRINT "{SPACE*2}SELECT AN OPTION (D, B, H OR R)":: PRINT
270 PRINT "{SPACE*2}[D]{SPACE*2}CONVERT A DECIMAL INTEGER"
280 PRINT "{SPACE*2}[B]{SPACE*2}CONVERT A BINARY INTEGER"
290 PRINT "{SPACE*2}[H]{SPACE*2}CONVERT A HEXADECIMAL INTEGER"
300 PRINT "{SPACE*2}[R]{SPACE*2}RETURN TO TOP MENU"
310 GET M1$ 
320 IF (M1$="D") THEN GOSUB 500: GOTO 260
330 IF (M1$="B") THEN GOSUB 760: GOTO 260
340 IF (M1$="H") THEN GOSUB 810: GOTO 260
350 IF (M1$="R") THEN RETURN
360 GOTO 310
370 REM *** OUTPUT MULTIPLE CONVERSION MENU ***
380 PRINT: PRINT "{SPACE*2}SELECT AN OPTION (D, B, H OR R)":: PRINT
390 PRINT "{SPACE*2}[D]{SPACE*2}CONVERT A RANGE OF DECIMAL INTEGERS"
400 PRINT "{SPACE*2}[B]{SPACE*2}CONVERT A RANGE OF BINARY INTEGERS"
410 PRINT "{SPACE*2}[H]{SPACE*2}CONVERT A RANGE OF HEXADECIMAL INTEGERS"
```

```

420 PRINT "(SPACE*2)[R](SPACE*2)RETURN TO TOP MENU"
430 GET M$
440 IF (M$="D") THEN GOSUB 1280: GOTO 380
450 IF (M$="B") THEN GOSUB 1670: GOTO 380
460 IF (M$="H") THEN GOSUB 1800: GOTO 380
470 IF (M$="R") THEN RETURN
480 GOTO 430
490 REM *** CONVERT SINGLE DECIMAL INTEGER ***
500 D$ = ""
510 PRINT: INPUT "ENTER DECIMAL INTEGER (UP TO 65535): ",D$
520 GOSUB 1030: REM VALIDATE DECIMAL INPUT
530 IF (V = 0) THEN GOTO 510
540 PRINT: PRINT " DEC";SPC(4); "BIN"; SPC(19); "HEX"
550 L = 5: GOSUB 1930: L1$ = L$
560 L = 20: GOSUB 1930: L2$ = L$
570 PRINT SPC(1);L1$;SPC(2);L2$;SPC(2);L1$
580 FOREGROUND 7
590 B$ = ""
600 D1 = 0
610 IF (D < 256) THEN GOTO 660
620 D1 = INT(D / 256)
630 FOR N = 1 TO 8
640 : IF ((D1 AND P(8 - N)) > 0) THEN B$ = B$ + "1": ELSE B$ = B$ + "0"
650 NEXT N
660 IF (D < 256) THEN B$ = "%" + B$: ELSE B$ = "%" + B$ + " "
670 D2 = D - 256*D1
680 FOR N = 1 TO 8
690 : IF ((D2 AND P(8 - N)) > 0) THEN B$ = B$ + "1": ELSE B$ = B$ + "0"
700 NEXT N
710 H$ = HEX$(D)
720 IF (D < 256) THEN H$ = "(SPACE*2)$" + RIGHT$(H$,2): ELSE H$ = "$" + H$
730 IF (D < 256) THEN PRINT SPC(6 - LEN(D$)); D$;SPC(12) + MID$(B$,1,5) +
" " + MID$(B$,6,10); "(SPACE*2)" + H$: FOREGROUND 1: RETURN
740 PRINT SPC(6 - LEN(D$)); D$;"(SPACE*2)" + MID$(B$,1,5) + " " + MID$(B$,6,4) +
MID$(B$,10,5) + " " + MID$(B$,15,4); "(SPACE*2)" + H$: FOREGROUND 1: RETURN
750 REM *** CONVERT SINGLE BINARY INTEGER ***
760 I$=""
770 PRINT: INPUT "ENTER BINARY INTEGER (UP TO 16 BITS): ",I$
780 GOSUB 1110: REM VALIDATE BINARY INPUT
790 IF (V = 0) THEN GOTO 760: ELSE GOTO 540
800 REM *** CONVERT SINGLE HEXADECIMAL INTEGER ***

```

```
810 H$=""  
820 PRINT: INPUT "ENTER HEXADECIMAL INTEGER (UP TO 4 DIGITS)": ",H$  
830 GOSUB 1220: REM VALIDATE HEXADECIMAL INPUT  
840 IF (V = 0) THEN GOTO 810: ELSE GOTO 540  
850 REM *** VALIDATE DECIMAL INPUT STRING ***  
860 FOR N = 1 TO LEN(D$)  
870 : M = ASC(MID$(D$,N,1)) - ASC("0")  
880 : IF ((M < 0) OR (M > 9)) THEN V = 0  
890 NEXT N: RETURN  
900 REM *** VALIDATE BINARY INPUT STRING ***  
910 FOR N = 1 TO LEN(I$)  
920 : M = ASC(MID$(I$,N,1)) - ASC("0")  
930 : IF ((M < 0) OR (M > 1)) THEN V = 0  
940 NEXT N: RETURN  
950 REM *** VALIDATE HEXADECIMAL INPUT STRING ***  
960 FOR N = 1 TO LEN(H$)  
970 : M = ASC(MID$(H$,N,1)) - ASC("0")  
980 : IF (NOT (((M >= 0) AND (M <= 9)) OR  
((M >= 17) AND (M <= 22)))) THEN V = 0  
990 NEXT N: RETURN  
1000 REM *** OUTPUT ERROR MESSAGE ***  
1010 FOREGROUND 2: PRINT: PRINT A$: FOREGROUND 1: RETURN  
1020 REM *** VALIDATE DECIMAL INPUT ***  
1030 V = 1: GOSUB 860: REM VALIDATE DECIMAL INPUT STRING  
1040 IF (V = 0) THEN A$ = "INVALID DECIMAL NUMBER": GOSUB 1010  
1050 IF (V = 1) THEN BEGIN  
1060 : D = VAL(D$)  
1070 : IF ((D < 0) OR (D > 65535)) THEN A$ = "DECIMAL NUMBER OUT OF RANGE":  
GOSUB 1010: V = 0  
1080 BEND  
1090 RETURN  
1100 REM *** VALIDATE BINARY INPUT ***  
1110 V = 1: GOSUB 910: REM VALIDATE BINARY INPUT STRING  
1120 IF (V = 0) THEN A$ = "INVALID BINARY NUMBER": GOSUB 1010: RETURN  
1130 IF (LEN(I$) > 16) THEN A$ = "BINARY NUMBER OUT OF RANGE":  
GOSUB 1010: V = 0 : RETURN  
1140 IF (V = 1) THEN BEGIN  
1150 : I = 0  
1160 : FOR N = 1 TO LEN(I$)  
1170 : I = I + VAL(MID$(I$,N,1)) * P(LEN(I$) - N)  
1180 : NEXT N
```

```

1190 BEND
1200 D$ = STR$(I): D = I: RETURN
1210 REM *** VALIDATE HEXADECIMAL INPUT ***
1220 V = 1: GOSUB 960: REM VALIDATE HEXADECIMAL INPUT STRING
1230 IF (V = 0) THEN A$ = "INVALID HEXADECIMAL NUMBER": GOSUB 1010: RETURN
1240 IF (LEN(H$) > 4) THEN A$ = "HEXADECIMAL NUMBER OUT OF RANGE":
GOSUB 1010: V = 0: RETURN
1250 D = DEC(H$): D$ = STR$(D): H = D: RETURN
1260 RETURN
1270 REM *** CONVERT MULTIPLE DECIMAL INTEGERS ***
1280 DB$=""
1290 PRINT: INPUT "ENTER STARTING DECIMAL INTEGER (UP TO 65535): ", DB$
1300 D$=DB$: GOSUB 1030: D$="": REM VALIDATE DECIMAL INPUT
1310 IF (V = 0) THEN GOTO 1290
1320 DE$=""
1330 PRINT: INPUT "ENTER ENDING DECIMAL INTEGER (UP TO 65535): ", DE$
1340 D$=DE$: GOSUB 1030: D$="": REM VALIDATE DECIMAL INPUT
1350 IF (V = 0) THEN GOTO 1330
1360 DB=VAL(DB$): DE=VAL(DE$)
1370 IF (DE < DB) THEN A$ = E$: GOSUB 1010: GOTO 1280
1380 SC = 1: SM = INT(((DE - DB) / 36) + 1)
1390 D = DB
1400 FOR J = SC TO SM
1410 : PRINT CHR$(147) + "RANGE: " + DB$ + " TO " + DE$ + "(SPACE*10)SCREEN: "
+ STR$(J) + " OF " + STR$(SM)
1420 : PRINT: PRINT "DEC";SPC(4);"BIN";SPC(19);"HEX";SPC(8);"DEC";SPC(4);
"BIN";SPC(19);"HEX"
1430 L = 5: GOSUB 1930: L1$ = L$
1440 L = 20: GOSUB 1930: L2$ = L$
1450 : PRINT SPC(1);L1$;SPC(2);L2$;SPC(2);L1$;SPC(6);L1$;SPC(2);
L2$;SPC(2);L1$
1460 : FOR K = 0 TO 17
1470 : FOREGROUND (7 + MOD(K,2))
1480 : D$ = STR$(D): GOSUB 590: D = D + 1
1490 : IF (D > DE) THEN GOTO 1630
1500 : NEXT K
1510 : PRINT CHR$(19): PRINT: PRINT: PRINT
1520 : FOR K = 0 TO 17
1530 : FOREGROUND (7 + MOD(K,2))
1540 : D$ = STR$(D): PRINT TAB(40): GOSUB 590: D = D + 1

```

```

1550 :      IF (D > DE) THEN GOTO 1630
1560 :      NEXT K
1570 :      FOREGROUND 1: PRINT: PRINT SPC(19); "PRESS X TO EXIT OR SPACEBAR TO CONTINUE..."
1580 :      GET B$
1590 :      IF B$="X" THEN RETURN
1600 :      IF B$="" THEN GOTO 1620
1610 :      GOTO 1580
1620 NEXT J
1630 PRINT CHR$(19): FOR I = 1 TO 22: PRINT: NEXT I
1640 PRINT SPC(20); "COMPLETE. PRESS SPACEBAR TO CONTINUE..."
1650 GET B$: IF B$<>" " THEN GOTO 1650: ELSE RETURN
1660 REM *** CONVERT MULTIPLE BINARY INTEGERS ***
1670 IB$=""
1680 PRINT: INPUT "ENTER STARTING BINARY INTEGER (UP TO 16 BITS): ", IB$
1690 IS=IB$: GOSUB 1110: IS=""": REM VALIDATE BINARY INPUT
1700 IF (V = 0) THEN GOTO 1680
1710 IB = I
1720 IE$=""
1730 PRINT: INPUT "ENTER ENDING BINARY INTEGER (UP TO 16 BITS): ", IE$
1740 IS=IE$: GOSUB 1110: IS=""": REM VALIDATE BINARY INPUT
1750 IF (V = 0) THEN GOTO 1730
1760 IE = I
1770 IF (IE < IB) THEN AS = E$: GOSUB 1010: GOTO 1670
1780 DB = IB: DE = IE: DB$ = STR$(IB): DE$ = STR$(IE): GOTO 1380
1790 REM *** CONVERT MULTIPLE HEXADECIMAL INTEGERS ***
1800 HB$=""
1810 PRINT: INPUT "ENTER STARTING HEXADECIMAL INTEGER (UP TO 4 DIGITS): ", HB$
1820 HS=HB$: GOSUB 1220: HS=""": REM VALIDATE HEXADECIMAL INPUT
1830 IF (V = 0) THEN GOTO 1810
1840 HB = H
1850 HE$=""
1860 PRINT: INPUT "ENTER ENDING HEXADECIMAL INTEGER (UP TO 4 DIGITS): ", HE$
1870 HS=HE$: GOSUB 1220: HS=""": REM VALIDATE HEXADECIMAL INPUT
1880 IF (V = 0) THEN GOTO 1860
1890 HE = H
1900 IF (HE < HB) THEN AS = E$: GOSUB 1010: GOTO 1880
1910 DB = HB: DE = HE: DB$ = STR$(HB): DE$ = STR$(HE): GOTO 1380
1920 REM *** MAKE LINES ***
1930 LS=""
1940 FOR K = 1 TO L: LS = LS + "-": NEXT K
1950 RETURN

```



# F

## APPENDIX

# System Memory Map

- **Introduction**
- **MEGA65 Native Memory Map**
- **\$D000 – \$DFFF IO Personalities**
- **CPU Memory Banking**
- **C64/C65 ROM Emulation**



# INTRODUCTION

The MEGA65 computer has a large 28-bit address space, which allows it to address up to 256MiB of memory and memory-mapped devices. This memory map has several different views, depending on which mode the computer is operating in. Broadly, there are five main modes: (1) Hypervisor mode; (2) C64 compatibility mode; (3) C65 compatibility mode; (4) UltiMAX compatibility mode; and (5) MEGA65 mode, or one of the other modes, where the programmer has made use of MEGA65 enhanced features.

It is important to understand that, unlike the C128, the C65 and MEGA65 allow access to all enhanced features from C64 mode, if the programmer wishes to do so. This means that while we frequently talk about "C64 Mode," "C65 Mode" and "MEGA65 Mode," these are simply terms of convenience for the MEGA65 with its memory map (and sometimes other features) configured to provide an environment that matches the appropriate mode. The heart of this is the MEGA65's flexibly memory map.

In this appendix, we will begin by describing the MEGA65's native memory map, that is, where all of the memory, IO devices and other features appear in the 28-bit address space. We will then explain how C64 and C65 compatible memory maps are accessed from this 28-bit address space.

# MEGA65 NATIVE MEMORY MAP

## The First Sixteen 64KB Banks

The MEGA65 uses a similar memory map to that of the C65 for the first MiB of memory, i.e., 16 memory banks of 64KiB each. This is because the C65's 4510 CPU can access only 1MiB of address space. These banks can be accessed from BASIC 65 using the **BANK**, **DMA**, **PEEK** and **POKE** commands. The following table summarises the contents of the first 16 banks:

| HEX | DEC | Address | Contents                                                         |
|-----|-----|---------|------------------------------------------------------------------|
| 0   | 0   | \$0xxxx | First 64KiB RAM. This is the RAM visible in C64 mode.            |
| 1   | 1   | \$1xxxx | Second 64KiB RAM. This is the 2nd 64KiB of RAM present on a C65. |
| 2   | 2   | \$2xxxx | First half of C65 ROM (C64 mode and shared components) or RAM    |
| 3   | 3   | \$3xxxx | Second half of C65 ROM (C65 mode components) or RAM              |
| 4   | 4   | \$4xxxx | Additional RAM (384KiB or larger chip-RAM models)                |
| 5   | 5   | \$5xxxx | Additional RAM (384KiB or larger chip-RAM models)                |
| 6   | 6   | \$6xxxx | Additional RAM (512KiB or larger chip-RAM models)                |
| 7   | 7   | \$7xxxx | Additional RAM (512KiB or larger chip-RAM models)                |
| 8   | 8   | \$8xxxx | Additional RAM (1MiB or larger chip-RAM models)                  |
| 9   | 9   | \$9xxxx | Additional RAM (1MiB or larger chip-RAM models)                  |
| A   | 10  | \$Axxxx | Additional RAM (1MiB or larger chip-RAM models)                  |
| B   | 11  | \$Bxxxx | Additional RAM (1MiB or larger chip-RAM models)                  |
| C   | 12  | \$Cxxxx | Additional RAM (1MiB or larger chip-RAM models)                  |
| D   | 13  | \$Dxxxx | Additional RAM (1MiB or larger chip-RAM models)                  |

continued ...

...continued

| HEX | DEC | Address | Contents                                        |
|-----|-----|---------|-------------------------------------------------|
| E   | 14  | \$Exxxx | Additional RAM (1MiB or larger chip-RAM models) |
| F   | 15  | \$Fxxxx | Additional RAM (1MiB or larger chip-RAM models) |

The key features of this address space are the 128KiB of RAM in the first two banks, which is also present on the C65. If you intend to write programmes which can also run on a C65, you should only use these two banks of RAM.

On all models it is possible to use all or part of the 128KiB of "ROM" space as RAM. To do this, you must first request that the Hypervisor removes the read-only protection on this area, before you will be able to change its contents. If you are writing a programme which will start from C64 mode, or otherwise switch to using the C64 part of the ROM, instead of the C65 part), then the second half of that space, i.e., BANK 3, can be safely used for your programmes. This gives a total of 192KiB of RAM, which is available on all models of the MEGA65.

On models that have 384KiB or more of chip RAM, BANK 4 and 5 are also available. Similarly, models which provide 1MiB or more of chip RAM will have BANK 6 through 15 also available, giving a total of 896KiB (or 960KiB, if only the C64 part of the ROM is required) of RAM available for your programmes. Note that the MEGA65's built-in freeze cartridge currently freezes only the first 384KiB of RAM.

## Colour RAM

The MEGA65's VIC-IV video controller supports much larger screens than the VIC-II or VIC-III. For this reason, it has access to a separate colour RAM, similar to on the C64. For compatibility with the C65, the first two kilo-bytes of this are accessible at \$1F800 – \$1FFF. The full 32KiB or 64KiB of colour RAM is located at \$FF80000. This is most easily accessed through the use of advanced DMA operations, or the 32-bit base-page indirect addressing mode of the processor.

At the time of writing, the **BANK** and **DMA** commands cannot be used to access the rest of the colour RAM, because the colour RAM is not located in the first mega-byte of address space. This may be corrected in a future revision of the MEGA65, allowing access to the full colour RAM via BANK 15 or an equivalent DMA job.

# 28-bit Address Space

In addition to the C65-style 1MB address space, the MEGA65 extends this to 256MiB, by using 28-bit addresses. The following shows the high-level layout of this address space.

| HEX                | DEC                     | Size      | Contents                                                      |
|--------------------|-------------------------|-----------|---------------------------------------------------------------|
| 0000000            | 0                       | 1         | CPU IO Port Data Direction Register                           |
| 0000001            | 1                       | 1         | CPU IO Port Data                                              |
| 0000002 - 005FFFF  | 2 - 384 KiB             | 384 KiB   | Fast chip RAM (40MHz)                                         |
| 0060000 - 0FFFFFF  | 384 KiB - 16 MiB        | 15.6 MiB  | Reserved for future chip RAM expansion                        |
| 1000000 - 3FFFFFF  | 16 MiB - 64 MiB         | 48 MiB    | Reserved                                                      |
| 4000000 - 7FFFFFF  | 64 MiB - 128 MiB        | 64 MiB    | Cartridge port and other devices on the slow bus (1 - 10 MHz) |
| 8000000 - 87FFFFFF | 128 MiB - 135 MiB       | 8 MiB     | 8MB ATTIC RAM (selected models only)                          |
| 8800000 - 8FFFFFF  | 135 MiB - 144 MiB       | 8 MiB     | 8MB CELLAR RAM (selected models only)                         |
| 9000000 - EFFFFFF  | 144 MiB - 240 MiB       | 96 MiB    | Reserved for future expansion RAM                             |
| F000000 - FF7DFFF  | 240 MiB - 255.49 MiB    | 15.49 MiB | Reserved for future IO expansion                              |
| FF7E000 - FF7EFFF  | 255.49 MiB - 255.49 MiB | 4 KiB     | VIC-IV Character ROM (write only)                             |
| FF80000 - FF87FFF  | 255.5 MiB - 255.53 MiB  | 32 KiB    | VIC-IV Colour RAM (32KiB colour RAM models)                   |
| FF88000 - FF8FFFF  | 255.53 MiB - 255.57 MiB | 32 KiB    | Additional VIC-IV Colour RAM (64KiB colour RAM models only)   |
| FF90000 - FFCAFFF  | 255.53 MiB - 255.80 MiB | 216 KiB   | Reserved                                                      |
| FFCB000 - FFCBFFF  | 255.80 MiB - 255.80 MiB | 4 KiB     | Emulated C1541 RAM                                            |
| FFCC000 - FFCFFFF  | 255.80 MiB - 255.81 MiB | 16 KiB    | Emulated C1541 ROM                                            |
| FFD0000 - FFD0FFF  | 255.81 MiB - 255.81 MiB | 4 KiB     | C64 \$Dxxx IO Personality                                     |

continued ...

...continued

| <b>HEX</b>        | <b>DEC</b>              | <b>Size</b> | <b>Contents</b>                                                                      |
|-------------------|-------------------------|-------------|--------------------------------------------------------------------------------------|
| FFD1000 - FFD1FFF | 255.81 MiB - 255.82 MiB | 4 KiB       | C65 \$Dxxx IO Personality                                                            |
| FFD2000 - FFD2FFF | 255.82 MiB - 255.82 MiB | 4 KiB       | MEGA65 \$Dxxx Ethernet IO Personality                                                |
| FFD3000 - FFD3FFF | 255.82 MiB - 255.82 MiB | 4 KiB       | MEGA65 \$Dxxx Normal IO Personality                                                  |
| FFD4000 - FFD5FFF | 255.82 MiB - 255.83 MiB | 8 KiB       | Reserved                                                                             |
| FFD6000 - FFD67FF | 255.83 MiB - 255.83 MiB | 2 KiB       | Hypervisor scratch space                                                             |
| FFD6000 - FFD6BFF | 255.83 MiB - 255.83 MiB | 3 KiB       | Hypervisor scratch space                                                             |
| FFD6C00 - FFD6DFF | 255.83 MiB - 255.83 MiB | 512         | F011 floppy controller sector buffer                                                 |
| FFD6E00 - FFD6FFF | 255.83 MiB - 255.83 MiB | 512         | SD Card controller sector buffer                                                     |
| FFD7000 - FFD70FF | 255.83 MiB - 255.83 MiB | 256         | MEGAphone r1 I2C peripherals                                                         |
| FFD7100 - FFD71FF | 255.83 MiB - 255.83 MiB | 256         | MEGA65 r2 I2C peripherals                                                            |
| FFD7200 - FFD72FF | 255.83 MiB - 255.83 MiB | 256         | MEGA65 HDMI I2C registers (only for models fitted with the ADV7511 HDMI driver chip) |
| FFD7300 - FFD7FFF | 255.83 MiB - 255.84 MiB | 3.25 KiB    | Reserved for future I2C peripherals                                                  |
| FFD8000 - FFDBFFF | 255.83 MiB - 255.86 MiB | 16 KiB      | Hypervisor ROM (only visible in Hypervisor Mode)                                     |
| FFDC000 - FFDDFFF | 255.86 MiB - 255.87 MiB | 8 KiB       | Reserved for Hypervisor Mode ROM expansion                                           |
| FFDE000 - FFDE7FF | 255.87 MiB - 255.87 MiB | 2 KiB       | Reserved for Ethernet buffer expansion                                               |
| FFDE800 - FFDEFFF | 255.87 MiB - 255.87 MiB | 2 KiB       | Ethernet frame read buffer (read only) and Ethernet frame write buffer (write only)  |
| FFDF000 - FFDFFFF | 255.87 MiB - 255.87 MiB | 4 KiB       | Virtual FPGA registers (selected models only)                                        |

continued ...

...continued

| <b>HEX</b>         | <b>DEC</b>           | <b>Size</b> | <b>Contents</b> |
|--------------------|----------------------|-------------|-----------------|
| FFE0000 - FFFFFFFF | 255.87 MiB - 256 MiB | 128 KiB     | Reserved        |

## \$D000 – \$DFFF IO PERSONALITIES

The MEGA65 supports four different IO personalities. These are selected by writing the appropriate values to the \$D02F KEY register, which is visible in all four IO personalities. There is more information in Chapter/Appendix 9 on page 9-3 about the use of the KEY register.

The following table shows which IO devices are visible in each of these IO modes, as well as the KEY register values that are used to select the IO personality.

| <b>HEX</b>      | <b>C64</b>          | <b>C65</b>              | <b>MEGA65<br/>ETHERNET</b> | <b>MEGA65</b>           |
|-----------------|---------------------|-------------------------|----------------------------|-------------------------|
| KEY             | \$00                | \$A5, \$96              | \$45, \$54                 | \$47, \$53              |
| \$D000 - \$D02F | VIC-II              | VIC-II                  | VIC-II                     | VIC-II                  |
| \$D030 - \$D07F | VIC-II <sup>1</sup> | VIC-III                 | VIC-III                    | VIC-III                 |
| \$D080 - \$D08F | VIC-II              | F011                    | F011                       | F011                    |
| \$D090 - \$D09F | VIC-II              | -                       | SD Card                    | SD Card                 |
| \$D0A0 - \$D0FF | VIC-II              | RAM EXPAND<br>CONTROL   | -                          | -                       |
| \$D100 - \$D1FF | VIC-II              | RED Palette             | RED Palette                | RED Palette             |
| \$D200 - \$D2FF | VIC-II              | GREEN<br>Palette        | GREEN<br>Palette           | GREEN<br>Palette        |
| \$D300 - \$D3FF | VIC-II              | BLUE Palette            | BLUE Palette               | BLUE Palette            |
| \$D400 - \$D41F | SID Right #1        | SID Right #1            | SID Right #1               | SID Right #1            |
| \$D420 - \$D43F | SID Right #2        | SID Right #2            | SID Right #2               | SID Right #2            |
| \$D440 - \$D45F | SID Left #1         | SID Left #1             | SID Left #1                | SID Left #1             |
| \$D460 - \$D47F | SID Left #2         | SID Left #2             | SID Left #2                | SID Left #2             |
| \$D480 - \$D49F | SID Right #1        | SID Right #1            | SID Right #1               | SID Right #1            |
| \$D4A0 - \$D4BF | SID Right #2        | SID Right #2            | SID Right #2               | SID Right #2            |
| \$D4C0 - \$D4DF | SID Left #1         | SID Left #1             | SID Left #1                | SID Left #1             |
| \$D4E0 - \$D4FF | SID Left #2         | SID Left #2             | SID Left #2                | SID Left #2             |
| \$D500 - \$D5FF | SID images          | -                       | Reserved                   | Reserved                |
| \$D600 - \$D63F | -                   | UART                    | UART                       | UART                    |
| \$D640 - \$D67F | -                   | UART images             | HyperTrap<br>Registers     | HyperTrap<br>Registers  |
| \$D680 - \$D6FF | -                   | -                       | MEGA65<br>Devices          | MEGA65<br>Devices       |
| \$D700 - \$D7FF | -                   | -                       | MEGA65<br>Devices          | MEGA65<br>Devices       |
| \$D800 - \$DBFF | COLOUR RAM          | COLOUR<br>RAM           | ETHERNET<br>Buffer         | COLOUR<br>RAM           |
| \$DC00 - \$DDFF | CIAs                | CIAs /<br>COLOUR<br>RAM | ETHERNET<br>Buffer         | CIAs /<br>COLOUR<br>RAM |
| \$DE00 - \$DFFF | CART IO             | CART IO                 | ETHERNET<br>Buffer         | CART IO /<br>SD SECTOR  |

<sup>1</sup> In the C64 IO personality, \$D030 behaves as on C128, allowing toggling between 1MHz and 2MHz CPU speed.

<sup>2</sup> The additional MEGA65 SIDs are visible in all IO personalities.

<sup>3</sup> Some models may replace the repeated images of the first four SIDs with four additional SIDs, for a total of 8 SIDs.

# CPU MEMORY BANKING

The 45GS10 processor, like the 6502, can only “see” 64KB of memory at a time. Access to additional memory is via a selection of bank-switching mechanisms. For backward-compatibility with the C64 and C65, the memory banking mechanisms for both of these computers existing the MEGA65:

1. C65-style MAP instruction banking
2. C65-style \$D030 banking
3. C64-style cartridge banking
4. C64-style \$00 / \$01 banking

It is important to understand that these different banking modes have a priority order: If a higher priority form of banking is being used, it takes priority over a lower priority form. The C65 banking methods take priority of the C64 mode banking methods. So, for example, if the 45GS10 MAP instruction has been used to provide a particular memory layout, the C64-style \$00 / \$01 banking will not be visible.

This makes the overall banking scheme more complex than on the C64. Thus to understand what the actual memory layout will be, you should start by considering the effects of C64 memory banking, and then if any C65 MAP instruction memory banking is enabled, using that to override the C64-style memory banking. Then if any C65 \$D030 memory banking is used, that overrides both the C64 and C65 MAP instruction memory banking. Finally, if IO is banked, or if there are any cartridges inserted and active, their effects are made.

The following diagram shows the different types of banking that can apply to the different areas of the 64KB that the CPU can see. The higher layers take priority over the lower layers, as described in the previous paragraph.

| IO/CART |                           | CART ROMLO         | CART ROMHI                |                    | IO                 | CART ROMHI         |
|---------|---------------------------|--------------------|---------------------------|--------------------|--------------------|--------------------|
| C65     |                           | BASIC              | BASIC                     | INTER-FACE         |                    | KERNEL             |
| MAP     | MAP LO<br>(4 x 8KB slabs) |                    | MAP HI<br>(4 x 8KB slabs) |                    |                    |                    |
| C64     |                           | BASIC              |                           | CHAR ROM           | KERNEL             |                    |
| RAM     | RAM*                      | RAM                | RAM                       | RAM                | RAM                | RAM                |
|         | \$0000 -<br>\$FFFF        | \$8000 -<br>\$9FFF | \$A000 -<br>\$BFFF        | \$C000 -<br>\$CFFF | \$D000 -<br>\$DFFF | \$E000 -<br>\$FFFF |

(There are actually a few further complications. For example, if the cartridge selects the UltiMAX(tm) game mode, then only the first 4KB of RAM will be visible, and the remaining address space will be un-mapped, and able to be supplied by the cartridge.)

For example, using \$D030 to bank in C65 ROM at \$A000, this will take priority over the C64 BASIC 2 ROM at the same address.

## C64/C65 ROM EMULATION

The C64 and C65 use ROM memories to hold the KERNAL and BASIC system. The MEGA65 is different: It uses 128KB of its 384KB fast chip RAM at \$20000 - \$3FFF (banks 2 and 3) to hold these system programmes. This makes it possible to change or upgrade the "ROM" that the MEGA65 is running, without having to open the computer. It is even possible to use the MEGA65's Freeze Menu to change the "ROM" being used while a programme is running.

The C64 and C65 memory banking methods use this 128KB of area when making ROM banks visible. When the RAM banks are mapped, they are always read-only. However, if the MAP instruction or DMA is used to access that address area, it is possible to write to it. For improved backward compatibility, the whole 128KB region of memory is normally set to read-only.

A programme can, however, request read-write access to this 128KB area of memory, so that it can make full use of the MEGA65's 384KB of chip RAM. This is accomplished by triggering the *Toggle Rom Write-protect* system trap of the hypervisor. The following code-fragment demonstrates how to do this. Calling it a second time will re-activate the write-protection.

```
LDA #$70  
STA $D640  
MOP
```

This fragment works by calling sub-function \$70 (toggle ROM write-protect) of Hypervisor trap \$00. Note that the `MOP` is mandatory. The MEGA65 IO personality must be first selected, so that the \$D640 register is un-hidden.

The current write-protection state can be tested by attempting to write to this area of memory. Also, you can examine and toggle the current state from in the MEGA65 Freeze Menu.

NOTE: If you are starting your programme from C65 mode, you must first make sure that the IO area is visible at \$D000-\$DFFF. The simplest way to do this is to use the MAP instruction with all zero values in the registers. The following fragment demonstrates this,

and also makes sure that the MEGA65 IO context is active, so that the hypervisor trap will be able to trigger:

```
; Clear C65 memory map
LDA #$00
TAX
TAY
TAZ
MAP
; Bank IO in via C64 mechanism
LDA #$35
STA $01
; Do MEGA65 / VIC-IV IO knock
LDA #$47
STA $D02F
LDA #$53
STA $D02F
; End MAP sequence, thus allowing interrupts to occur again
EOM
; Do Hypervisor call to un-write-protect the ROM area
LDA #$70
STA $D640
NOP
```

## C65 Compatibility ROM Layout

The layout of the C65 compatibility 128KB ROM area is identical to that of the C65:

| HEX                | Contents                                  |
|--------------------|-------------------------------------------|
| \$3E000 -- \$3FFFF | C65 KERNAL                                |
| \$3C000 -- \$3DFFF | RESERVED                                  |
| \$38000 -- \$3BFFF | C65 BASIC GRAPHICS ROUTINES               |
| \$32000 -- \$37FFF | C65 BASIC                                 |
| \$30000 -- \$31FFF | MONITOR (gets mapped at \$6000 -- \$7FFF) |
| \$2E000 -- \$2FFFF | C64 KERNAL                                |
| \$2D000 -- \$2DFFF | C64 CHARSET                               |
| \$2C000 -- \$2CFFF | INTERFACE                                 |
| \$24000 -- \$27FFF | RESERVED                                  |
| \$20000 -- \$23FFF | DOS (gets mapped at \$8000 -- \$BFFF)     |

The INTERFACE program is a series of routines that are used by the C65 to switch between C64 mode, C65 mode and the C65's built-in DOS. The DOS is located in the lower-eighth of the ROM.





## APPENDIX

# 45GS02 Microprocessor

- **Introduction**
- **Differences to the 6502**
- **C64 CPU Memory Mapped Registers**
- **New CPU Memory Mapped Registers**
- **MEGA65 CPU Maths Acceleration Registers**
- **MEGA65 Hypervisor Mode**



# INTRODUCTION

The 45GS02 is an enhanced version of the processor portion of the CSG4510 and of the F018 "DMAgic" DMA controller used in the Commodore 65 computer prototypes. The 4510 is, in turn, an enhanced version of the 65CE02. The reader is referred to the considerable documentation available for the 6502 and 65CE02 processors for the backwards-compatible operation of the 45GS02.

This chapter will focus on the differences between the 45GS02 and the earlier 6502-class processors, and the documentation of the many built-in memory-mapped IO registers of the 45GS02.

## DIFFERENCES TO THE 6502

The 45GS02 has a number of key differences to earlier 6502-class processors:

### Supervisor/Hypervisor Privileged Mode

Unlike the earlier 6502 variants, the 45GS02 has a privileged mode of operation. This mode is intended for use by an operating system or type-1 hypervisor. The ambiguity between operating system and Hypervisor on the MEGA65 stems from the fact that the operating system of the MEGA65 is effectively little more than a loader and task-switcher for C64 and C65 environments, i.e., effectively operating as a hypervisor, but provides only limited virtualisation of the hardware.

The key differences between normal and supervisor mode on the MEGA65, are that in supervisor mode:

- A special 16KiB memory area is mapped to \$8000 - \$BFFF, which is used to contain both the program and data of the Hypervisor / supervisor program. This is normally the Hypo program. This memory is not mappable by any means when the processor is in the normal mode (the chip-select line to it is inhibited), protecting it from accidental or malicious access.
- The 64 SYSCALL trap registers in the MEGA65 IO-mode at \$D640 - \$D67F are replaced by the virtualisation control registers. These registers allow complete control over the system, and it is their access that truly defines the privilege of the supervisor mode.
- The processor always operates at full speed (40MHz) and in the 4510 processor personality.

The Hypervisor Mode is described in more detail later in this appendix.

## 6502 Illegal Opcodes

The 65C02, 65CE02 and CSG4510 processors extended the original 6502 processor by using previously unallocated opcodes of the 6502 to provide additional instructions. All software that followed the official documentation of the 6502 processor will therefore work on these newer processors, possibly with different instruction timing. However, the common practice on the C64 and other home computers of using undefined opcodes (often called “illegal opcodes”, although there is no law against using them), means that many existing programs will not work on these newer processors.

To alleviate this problem the 45GS02 has the ability to switch processor personalities between the 4510 and 6502. The effect is that in 6502 mode, none of the new opcodes of the 65C02, 65CE02, 4510 or 45GS02 are available, and are replaced with the original, often strange, behaviour of the undefined opcodes of the 6502.

**WARNING:** This feature is incomplete and untested. Most undocumented 6502 opcodes do not operate correctly when the 6502 personality is enabled.

## Read-Modify-Write Instruction Bug Compatibility

The 65CE02 processor optimised a group of instructions called the Read-Modify-Write (RMW) instructions. For such instructions, such as INC, that increments the contents of a memory location, the 6502 would read the original value and then write it back unchanged, before writing it back with the new increased value. For most purposes, this did not cause any problems. However, it turned out to be a fast way to acknowledge VIC-II interrupts, because writing the original value back (which the instruction doesn't need to do) acknowledges the interrupt. This method is faster and uses fewer bytes than any alternative, and so became widely used in C64 software.

The problem came with the C65 with its 65CE02 derived CSG4510 that didn't do this extra write during the RMW instructions. This made the RMW instructions one cycle faster, which made software run slightly faster. Unfortunately, it also meant that a lot of existing C64 software simply won't run on a C65, unless the interrupt acknowledgement code in each program is patched to work around this problem. This is the single most common reason why many C64 games and other software titles won't run on a C65.

Because this problem is so common, the MEGA65's 45GS02 includes bug compatibility with this commonly used feature of the original 6502. It does this by checking if the

target of an RMW instruction is \$D019, i.e., the interrupt status register of the VIC-II. If it is, then the 45GS02 performs the dummy write, allowing many C64 software titles to run unmodified on the MEGA65, that do not run on a C65 prototype. By only performing the dummy write if the address is \$D019, the MEGA65 maintains C64 compatibility, without sacrificing the speed improvement for all other uses of these instructions.

## Variable CPU Speed

The 45GS02 is able to run at 1MHz, 2MHz, 3.5MHz and 40MHz, to support running software designed for the C64, C128 in C64 mode, C65 and MEGA65.

### Slow (1MHz – 3.5MHz) Operation

In these modes, the 45GS02 processor slows down, so that the same number of instructions per video frame are executed as on a PAL or NTSC C64, C128 in C64 mode or C65 prototype. This is to allow existing software to run on the MEGA65 at the correct speed, and with minimal display problems. The VIC-IV video controller provides cycle indication pulses to the 45GS02 that are used to keep time.

In these modes, opcodes take the same number of cycles as an 6502. However memory accesses within an instruction are not guaranteed to occur in the same cycle as on a 1MHz 6502. Normally the effect is that instructions complete faster, and the processor idles until the correct number of cycles have passed. This means that timing may be incorrect by up to 7 micro-seconds. This is not normally a problem, and even many C64 fast loaders will function correctly. For example, the GEOSTM Graphical Operating System for the C64 can be booted and used from a 1541 connected to the MEGA65's serial port.

However, some advanced VIC-II graphics tricks, such as Variable Screen Position (VSP) are highly unlikely to work correctly, due to the uncertainty in timing of the memory write cycles of instructions. However, in most cases such problems can be easily solved by using the advanced features of the MEGA65's VIC-IV video controller. For example, VSP is unnecessary on the MEGA65, because you can set the screen RAM address to any location in memory.

### Full Speed (40MHz) Instruction Timing

When the MEGA65's processor is operating at full speed (currently 40MHz), the instruction timing no longer exactly mirrors the 6502: Instructions that can be executed in fewer cycles will do so. For example, branches are typically require fewer instructions on the 45GS02. There are also some instructions that require more cycles on the 45GS02, in particular the LDA, LDX, LDY and LDZ instructions. Those instructions typically require one additional cycle. However as the processor is running at 40MHz, these instructions still execute much more quickly than on even a C65 or C64 with an accelerator.

## CPU Speed Fine-Tuning

It is also possible to more smoothly vary the CPU speed using the **SPEEDBIAS** register located at \$F7FA (55290), when MEGA65 IO mode is enabled. The default value is \$80 (128), which means no bias on the CPU speed. Higher values increase the CPU speed, with \$FF meaning 2× the expected speed. Lower values slow the processor down, with \$00 bring the CPU to a complete stand-still. Thus the speed can be varied between 0× and 2× the intended value.

This register is provided to allow tweaking the processor speed in games.

Note that this register has no effect when the processor is running at full-speed, because it only affects the way in which VIC-IV video cycle indication pulses are processed by the CPU.

## Direct Memory Access (DMA)

Direct Memory Access (DMA) is a method for quickly filling, copying or swapping memory regions. The MEGA65 implements an improved version of the F018 "DMAgic" DMA controller of the C65 prototypes. Unlike on the C65 prototypes, the DMA controller is part of the CPU on the MEGA65.

Detailed information on how to use the DMA controller and these advanced features can be found in Chapter/Appendix L on page L-3

# Accessing memory between the 64KB and 1MB points

The C65 included four ways to access memory beyond the 64KB point: three methods that are limited, specialised or both, and two general-purpose methods. We will first consider the limited methods, before documenting the general-purpose methods.

## C64-Style Memory Banking

The first method, is to use the C64-style \$00/\$01 ROM/RAM banking. This method is very limited, however, as it allows only the banking in and out of the two 8KB regions that correspond to the C64 BASIC and KERNEL ROMs. These are located at \$2A000 and \$2E000 in the 20-bit C65 address space, i.e., \$002A000 and \$002E000 in the 28-bit address space of the MEGA65. It can also provide access to the C64 character ROM data at \$D000, which is located at \$2D000 in the C65 memory map, and thus \$002D0000 in the MEGA65 address space. In addition to being limited to which regions this method can access, it also only provides read-only access to these memory regions, i.e., it cannot be used to modify these memory regions.

## VIC-III “ROM” Banking

Similar to the C64-style memory banking, the C65 included the facility to bank several other regions of the C65’s 128KB ROM. These are banked in and out using various bits of the VIC-III’s \$D030 register:

| \$D030 Bit | Signal Name | 20-bit Address                              | 16-bit Address  | Read-Write Access? |
|------------|-------------|---------------------------------------------|-----------------|--------------------|
| 0          | CRAM2K      | \$1F800 - \$1FFFF,<br>\$FF80000 - \$FF807FF | \$D800 - \$DFFF | Y                  |
| 3          | ROM8        | \$38000 - \$39FFF                           | \$8000 - \$9FFF | N                  |
| 4          | ROMA        | \$3A000 - \$3BFFF                           | \$A000 - \$BFFF | N                  |
| 5          | ROMC        | \$2C000 - \$2CFFF                           | \$C000 - \$CFFF | N                  |
| 6          | CROM9       | \$29000 - \$29FFF                           | \$D000 - \$DFFF | N                  |
| 7          | ROME        | \$3E000 - \$3FFFF                           | \$E000 - \$FFFF | N                  |

The CRAM2K signal causes the normal 1KB of colour RAM, which is located at \$1F800 - \$1FBFF and is visible at \$D800 - \$DBFF, to instead be visible from \$D800 - \$DFFF. That is, the entire range \$1F800 - \$1FFFF is visible, and can be both read from and written to. Unlike on the C64, the colour RAM on the MEGA65 is always visible as 8-bit bytes. Also, on the MEGA65, the colour RAM is 32KB in size, and exists at \$FF80000 - \$FF87FFF. The visibility of the colour RAM at \$1F800 - \$1FFFF is achieved by mirroring writes to both regions when accessing the colour RAM via this mechanism.

Note that these VIC-III memory banking signals take precedence over the C64-style memory banking.

## VIC-III Display Address Translator

The third specialised manner to access to memory above the 64KB point is to use the VIC-III’s Display Address Translator. Use of this mechanism is documented in Chapter/Appendix M on page M-5.

## The MAP instruction

The first general-purpose means of access to memory is the MAP instruction of the 4510 processor. The MEGA65's 45GS02 processor also supports this mechanism. This instruction divides the 64KB address of the 6502 into eight blocks of 8KB each. For each of these blocks, the block may either be accessed normally, i.e., accessing an 8KB region of the first 64KB of RAM of the system. Alternatively, each block may instead be re-mapped (hence the name of the MAP instruction) to somewhere else in the address space, by adding an offset to the address. Mapped addresses in the first 32KB use one offset, the lower offset, and the second 32KB uses another, the upper offset. Re-mapping of memory using the MAP instruction takes precedence over the C64-style memory banking, but not the C65's ROM banking mechanism.

The offsets must be a multiple of 256 bytes, and thus consist of 12 bits in order to allow an arbitrary offset in the 1MB address space of the C65. As each 8KB block in a 32KB half of memory can be either mapped or not, this requires one bit per 8KB block. Thus the processor requires 16 bits of information for each half of memory, for a total of 32 bits of information. This is achieved by setting the A and X registers for the lower half of memory and the Y and Z registers for the upper half of memory, before executing the MAP instruction.

The MAP instruction copies the contents of these registers into the processors internal registers that hold the mapping information. Note that there is no way to use the MAP instruction to determine the current memory mapping configuration, which somewhat limits its effectiveness.

The following diagram illustrates how the MAP instruction takes the values of the four A, X, Y and Z registers, and uses them to compute the upper and lower address offsets, and sets the bank enable bits for each of the eight 8KB memory regions of the 6502 address space:



That is, the contents of the A register and the lower-nybl of the X register form a 12-bit value that is multiplied by 256 to produce the offset used for any of the 8KB banks in the lower 32KB half of the 6502's 16-bit address space. The upper nybl of the X register is used as flags to indicate which of the four 8KB blocks in that 32KB half of the 6502 address space should have the offset added to their addresses to compute the actual address.

The Y and Z registers are used in a similar way to produce the offset for the upper 32KB half of the 6502 address space, and the flags to indicate whether the offset is used for each of the four 8KB blocks in that half of the address space.

Note that the lower 8 bits of the offset cannot be set. That is, the offset will be a multiple of 256 bytes, unlike on some extended 6502 processors. However, in practice this restriction is rarely limiting.

To understand how this works in practice, the following example shows how this works with a concrete example, showing the address ranges that would be visible in each of the 8KB slices of the 6502's 64KB address space:



Notice that the offsets for each of the two 32KB address ranges get added to the 6502 address. This is why the offset of \$48000 for the upper 32KB generates an address of \$50000 at the 6502 address \$8000.

See also under "Using the MAP instruction to access >1MB" for further explanation.

### **Direct Memory Access (DMA) Controller**

The C65's F018/F018A DMA controller allows for rapid filling, copying and swapping of the contents of memory anywhere in the 1MB address space. Detailed information about the F018 DMA controller, and the MEGA65's enhancements to this, refer to Chapter/Appendix L on page L-3

### **Flat Memory Access**

# Accessing memory beyond the 1MB point

The MEGA65 can support up to 256MiB of memory. This is more than the 1MiB address space of the CSG4510 on which it is based. There are several ways of performing this.

## Using the MAP instruction to access >1MB

The full address space is available to the MAP instruction for legacy C65-style memory mapping, although some care is required, as the MAP instruction must be called up to three times. The reason for this is that the MAP instruction must be called to first select which mega-byte of memory will be used for the lower and upper map regions, before it is again called in the normal way to set the memory mapping. Because between these two calls the memory mapping offset will be a mix of the old and new addresses, all mapping should be first disabled via the MAP instruction. This means that the code to re-map memory should live in the bottom 64KB of RAM or in one of the ROM-bankable regions, so that it can remain visible during the mapping process.

Failure to handle this situation properly will result in the processor executing instructions from somewhere unexpected half-way through the process, because the routine it is executing to perform the mapping will suddenly no longer be mapped.

Because of the relative complexity of this process, and the other problems with the MAP instruction as a means of memory access, we recommend that for accessing data outside of the current memory map that you use either DMA or the flat-memory address features of the 45GS02 that are described below. Indeed, access to the full address space via the MAP instruction is only provided for completeness.

As another example of how the MAP instruction can be used to map an area of memory from the expanded address space, the following program maps the Ethernet frame buffer from its natural location at \$FFDE8000 to appear at \$6800. To keep the example as simple as possible, we assume that the code is running from in the bottom 64KB of RAM, and not in the region between \$6000 - \$8000.

As the MAP instruction normally is only aware of the C65-style 20-bit addresses, the MEGA65 extension to the instruction must be used to set the upper 8 bits of the 28-bit MEGA65 addresses, i.e., which mega-byte of address space should be used for the address translation. This is done by setting the X register to \$0F when setting the mega-byte number for the lower-32KB of the C64-style 64KB address space. This does not create any incompatibility with any sensible use of the MAP instruction on a C65, because this value indicates that none of the four 8KB memory blocks will be re-mapped, but at the same time specifies that the upper 4 bits of the address offset for re-mapped block is the non-zero value of \$F. The mega-byte number is then specified by setting the A register.

The same approach applies to the upper 32KB, but using the Z and Y registers instead of the X and A registers. However, in this case, we do not need to re-map the upper 32KB of memory in this example, we will leave the Z and Y registers set to zero. We must however set X and A to set the mega-byte number for the lower-32KB to \$FF. Therefore A must have the value \$FF. To set the lower 20-bits of the address offset we use the MAP instruction a second time, this time using it in the normal C65 manner. As we want to remap \$6800 to \$FFDE800, and have already dealt with the \$FFxxxx offset via the mega-byte number, we need only to apply the offset to make \$6800 point to \$DE800. \$DE800 minus \$6800 = \$D8000. As the MAP instruction operates with a mapping granularity of 256 bytes = \$100, we can drop the last two digits from \$D8000 to obtain the MAP offset of \$D80. The lower 8-bits, \$80, must be loaded into the A register. The upper 4-bits, \$D, must be loaded into the low-nibble of the X register. As we wish to apply the mapping to only the fourth of the 8KB blocks that make up the lower 32KB half of the C64 memory map, we must set the 4th bit of the upper nibble. That is, the upper nibble must be set to %1000, i.e., \$8. Therefore the X register must be loaded with \$8D. Thus we yield the complete example program:

```
; Map Ethernet registers at $6000 - $7FFF

; Ethernet controller really lives $FFDE000 - $FFDEFFF, so select $FF megabyte section for MAP LO
LDA #$ff
LDX #$0f
LDY #$00
LDZ #$00
Map

; now enable mapping of $DE000-$DFFFF at $6000
; MAPs are offset based, so we need to subtract $6000 from the target address
; $DE000 - $6000 = $D8000
LDA #$80
LDX #$8d
LDY #$00
LDZ #$00
Map
EOM

; Ethernet buffer now visible at $6800 - $6FFF
```

Note that the EOM (End Of Mapping) instruction (which is the same as NOP on a 6502, i.e., opcode \$EA) was only supplied after the last MAP instruction, to make sure that no interrupts could occur while the memory map contained mixed values with the mega-byte number set, but the lower-bits of the mapping address had not been updated.

No example in BASIC for the MAP instruction is possible, because the MAP is an machine code instruction of the 4510 / 45GS02 processors.

## Flat-Memory Access

The 45GS02 makes it easy to read or write a byte from anywhere in memory by allowing the Zero-Page Indirect addressing mode to use a 32-bit pointer instead of the normal 16-bit pointer. This is accomplished by using the Z-indexed Zero-Page Indirect Addressing Mode for the access, and having the instruction directly preceded by a NOP instruction (opcode \$EA). For example:

```
NOP  
LDA ($45),Z
```

If you are using the ACME assembler, or another assembler that supports the 45GS02 extensions, you can instead use square-brackets to indicate that you are performing a flat-memory operation. Such assemblers will insert the \$EA prefix automatically for you. For example:

```
LDA [$45],Z
```

Regardless which tool you are using, this example would read the four bytes of Zero-Page memory at \$45 – \$48 to form a 32-bit memory address, and add the value of the Z register to this to form the actual address that will be read from. The byte order in the address is the same as the 6502, i.e., the right-most (least significant) byte of the address will be read from the first address (\$45 in this case), and so on, until the left-most (most significant) byte will be read from \$48. For example, to read from memory location \$12345678, the contents of memory beginning at \$45 should be 78 56 43 12.

This method is much more efficient and also simpler than either using the MAP instruction or the DMA controller for single memory accesses, and is what we generally recommend. The DMA controller can be used for moving/filler larger regions of memory. We recommend the MAP instruction only be used for banking code, or in rare situations where extensive access to a small region of memory is required, and the extra cycles of reading the 32-bit addresses is problematic.

## Virtual 32-bit Register

The 45GS02 allows the use of its four general purpose registers, A, X ,Y and Z as a single virtual 32-bit register. This can greatly simplify and speed up many common operations,

and help avoid many common programming errors. For example, adding two 16-bit or 32-bit values can now be easily accomplished with something like:

```
; Clear carry before performing addition, as normal  
CLC  
; Prefix an instruction with two NEG instructions to select virtual 32-bit register mode  
NEG  
NEG  
LDA $1234 ; Load the contents of $1234-$1237 into A,X,Y and Z respectively  
; And again, for the addition  
NEG  
NEG  
ADC $1238 ; Add the contents of $1238-$123B  
; The result of the addition is now in A, X, Y and Z.  
; And can be written out in whole or part  
  
; To write it all out, again, we need the NEG + NEG prefix  
NEG  
NEG  
STA $123C ; Write the whole out to $123C-$123F  
  
; Or to write out the bottom bytes, we can just write the contents of A and X as normal  
STA $1240  
STX $1241
```

This approach works with the LDA, STA, ADC, SBC, CMP, EOR, AND, BIT, ORA, ASL, ASR, LSR, ROL, ROR, INC and DEC instructions. If you are using ACME or another 45GS02 aware assembler, you can instead use the new **LDQ**, **STQ**, **ADCQ**, **SBCQ**, **CPQ**, **EORQ**, **ANDQ**, **BITQ**, **ORQ**, **ASLQ**, **ASRQ**, **LSRQ**, **ROLQ**, **RORQ**, **INQ** and **DEQ** mnemonics. The previous example would thus become:

```

; Clear carry before performing addition, as normal
CLC
LDQ $1234 ; Load the contents of $1234-$1237 into A,X,Y and Z respectively
; And again, for the addition
ADCQ $1238 ; Add the contents of $1238-$123B
; The result of the addition is now in A, X, Y and Z.
; And can be written out in whole or part

; To write it all out, again, we need the NEG + NEG prefix
STQ $123C ; Write the whole out to $123C-$123F

; Or to write out the bottom bytes, we can just write the contents of A and X as normal
STA $1240
STX $1241

```

The virtual 32-bit addressing mode works with any addressing mode. However, indexed addressing modes, where X, Y or Z are added to the address should be used with care, because these registers may in fact be holding part of a 32-bit value.

The exception is the Zero-Page Indirect Z-Indexed addressing mode: In this case the Z register is NOT added to the target address, unlike would normally be the case. This is to allow the virtual 32-bit register to be able to be used with flat-memory access with the combined prefix of **NEG NEG NOP**, before the instruction to allow accessing a 32-bit value anywhere in memory in a single instruction.

Note that the virtual 32-bit register cannot be used in immediate mode, e.g., to load a constant into the four general purpose registers, or to add or subtract a constant value. This is to avoid problems with variable length instructions.

For LDQ and STQ, it would save at most one byte compared to LDA #\$nn ... LDZ #\$nn, and would be no faster. In fact, for many common values, such as #\$00000000, there are short-cuts, such as:

```

LDA #$00
TAX
TAY
TAZ

```

If you need to add or subtract a 32-bit immediate value, this may require you to re-order the arguments, or perform other minor gymnastics. For example, to compute the sum of the contents of memory and an immediate value, you can load the A, X, Y and Z registers with the immediate value, and then use **ADCQ** with the memory address, e.g.:

```

; Get the immediate value #$12345678 into Q
LDA #$78
LDX #$56
LDY #$34
LDZ #$12
; Add the contents of memory locations $1234-$1237
NEG
NEG
ADC $1234
; Store the result back in $1234-$1237
NEG
NEG
STA $1234

```

Again, if you are using the ACME or another 45GS02-aware assembler, this can be more compactly and clearly written as follows. But note that in both cases the same byte-sequence of machine code is produced, and the programme will take the same number of cycles to execute.

```

; Get the immediate value #$12345678 into Q
LDA #$78
LDX #$56
LDY #$34
LDZ #$12
; Add the contents of memory locations $1234-$1237
ADCO $1234
; Store the result back in $1234-$1237
STQ $1234

```

## C64 CPU MEMORY MAPPED REGISTERS

| <b>HEX</b> | <b>DEC</b> | <b>Signal</b> | <b>Description</b>        |
|------------|------------|---------------|---------------------------|
| 00         | 0          | PORTDDR       | 6510/45GS10 CPU port DDR  |
| 01         | 1          | PORT          | 6510/45GS10 CPU port data |

## NEW CPU MEMORY MAPPED REGISTERS

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D640       | 54848      |            |            |            |            | HTRAP00    |            |            |            |
| D641       | 54849      |            |            |            |            | HTRAP01    |            |            |            |
| D642       | 54850      |            |            |            |            | HTRAP02    |            |            |            |
| D643       | 54851      |            |            |            |            | HTRAP03    |            |            |            |
| D644       | 54852      |            |            |            |            | HTRAP04    |            |            |            |
| D645       | 54853      |            |            |            |            | HTRAP05    |            |            |            |
| D646       | 54854      |            |            |            |            | HTRAP06    |            |            |            |
| D647       | 54855      |            |            |            |            | HTRAP07    |            |            |            |
| D648       | 54856      |            |            |            |            | HTRAP08    |            |            |            |
| D649       | 54857      |            |            |            |            | HTRAP09    |            |            |            |
| D64A       | 54858      |            |            |            |            | HTRAP0A    |            |            |            |
| D64B       | 54859      |            |            |            |            | HTRAP0B    |            |            |            |
| D64C       | 54860      |            |            |            |            | HTRAP0C    |            |            |            |
| D64D       | 54861      |            |            |            |            | HTRAP0D    |            |            |            |
| D64E       | 54862      |            |            |            |            | HTRAP0E    |            |            |            |
| D64F       | 54863      |            |            |            |            | HTRAP0F    |            |            |            |
| D650       | 54864      |            |            |            |            | HTRAP10    |            |            |            |
| D651       | 54865      |            |            |            |            | HTRAP11    |            |            |            |
| D652       | 54866      |            |            |            |            | HTRAP12    |            |            |            |
| D653       | 54867      |            |            |            |            | HTRAP13    |            |            |            |
| D654       | 54868      |            |            |            |            | HTRAP14    |            |            |            |
| D655       | 54869      |            |            |            |            | HTRAP15    |            |            |            |
| D656       | 54870      |            |            |            |            | HTRAP16    |            |            |            |
| D657       | 54871      |            |            |            |            | HTRAP17    |            |            |            |
| D658       | 54872      |            |            |            |            | HTRAP18    |            |            |            |
| D659       | 54873      |            |            |            |            | HTRAP19    |            |            |            |
| D65A       | 54874      |            |            |            |            | HTRAP1A    |            |            |            |
| D65B       | 54875      |            |            |            |            | HTRAP1B    |            |            |            |
| D65C       | 54876      |            |            |            |            | HTRAP1C    |            |            |            |
| D65D       | 54877      |            |            |            |            | HTRAP1D    |            |            |            |
| D65E       | 54878      |            |            |            |            | HTRAP1E    |            |            |            |
| D65F       | 54879      |            |            |            |            | HTRAP1F    |            |            |            |
| D660       | 54880      |            |            |            |            | HTRAP20    |            |            |            |
| D661       | 54881      |            |            |            |            | HTRAP21    |            |            |            |
| D662       | 54882      |            |            |            |            | HTRAP22    |            |            |            |
| D663       | 54883      |            |            |            |            | HTRAP23    |            |            |            |
| D664       | 54884      |            |            |            |            | HTRAP24    |            |            |            |
| D665       | 54885      |            |            |            |            | HTRAP25    |            |            |            |
| D666       | 54886      |            |            |            |            | HTRAP26    |            |            |            |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D667       | 54887      |            |            |            |            | HTRAP27    |            |            |            |
| D668       | 54888      |            |            |            |            | HTRAP28    |            |            |            |
| D669       | 54889      |            |            |            |            | HTRAP29    |            |            |            |
| D66A       | 54890      |            |            |            |            | HTRAP2A    |            |            |            |
| D66B       | 54891      |            |            |            |            | HTRAP2B    |            |            |            |
| D66C       | 54892      |            |            |            |            | HTRAP2C    |            |            |            |
| D66D       | 54893      |            |            |            |            | HTRAP2D    |            |            |            |
| D66E       | 54894      |            |            |            |            | HTRAP2E    |            |            |            |
| D66F       | 54895      |            |            |            |            | HTRAP2F    |            |            |            |
| D670       | 54896      |            |            |            |            | HTRAP30    |            |            |            |
| D671       | 54897      |            |            |            |            | HTRAP31    |            |            |            |
| D672       | 54898      |            |            |            |            | HTRAP32    |            |            |            |
| D673       | 54899      |            |            |            |            | HTRAP33    |            |            |            |
| D674       | 54900      |            |            |            |            | HTRAP34    |            |            |            |
| D675       | 54901      |            |            |            |            | HTRAP35    |            |            |            |
| D676       | 54902      |            |            |            |            | HTRAP36    |            |            |            |
| D677       | 54903      |            |            |            |            | HTRAP37    |            |            |            |
| D678       | 54904      |            |            |            |            | HTRAP38    |            |            |            |
| D679       | 54905      |            |            |            |            | HTRAP39    |            |            |            |
| D67A       | 54906      |            |            |            |            | HTRAP3A    |            |            |            |
| D67B       | 54907      |            |            |            |            | HTRAP3B    |            |            |            |
| D67C       | 54908      |            |            |            |            | HTRAP3C    |            |            |            |
| D67D       | 54909      |            |            |            |            | HTRAP3D    |            |            |            |
| D67E       | 54910      |            |            |            |            | HTRAP3E    |            |            |            |
| D67F       | 54911      |            |            |            |            | HTRAP3F    |            |            |            |
| D710       | 55056      | -          |            | BADEXTRA   | BRCOST     |            |            | -          |            |
| D7FA       | 55290      |            |            |            | FRAMECOUNT |            |            |            |            |
| D7FB       | 55291      |            | -          |            |            | CARTEN     |            | -          |            |
| D7FD       | 55293      | NOEXROM    | NOGAME     |            |            | -          |            |            | POWEREN    |
| D7FE       | 55294      |            |            | -          |            | OCEANA     | PREFETCH   |            |            |

- **BADEXTRA** Cost of badlines minus 40. ie. 00=40 cycles, 11 = 43 cycles.
- **BRCOST** 1=charge extra cycle(s) for branches taken
- **CARTEN** 1= enable cartridges
- **FRAMECOUNT** Count number of elapsed video frames
- **HTRAP00** Writing triggers hypervisor trap \$00

- **HTRAP01** Writing triggers hypervisor trap \$01
- **HTRAP02** Writing triggers hypervisor trap \$02
- **HTRAP03** Writing triggers hypervisor trap \$03
- **HTRAP04** Writing triggers hypervisor trap \$04
- **HTRAP05** Writing triggers hypervisor trap \$05
- **HTRAP06** Writing triggers hypervisor trap \$06
- **HTRAP07** Writing triggers hypervisor trap \$07
- **HTRAP08** Writing triggers hypervisor trap \$08
- **HTRAP09** Writing triggers hypervisor trap \$09
- **HTRAP0A** Writing triggers hypervisor trap \$0A
- **HTRAP0B** Writing triggers hypervisor trap \$0B
- **HTRAP0C** Writing triggers hypervisor trap \$0C
- **HTRAP0D** Writing triggers hypervisor trap \$0D
- **HTRAP0E** Writing triggers hypervisor trap \$0E
- **HTRAP0F** Writing triggers hypervisor trap \$0F
- **HTRAP10** Writing triggers hypervisor trap \$10
- **HTRAP11** Writing triggers hypervisor trap \$11
- **HTRAP12** Writing triggers hypervisor trap \$12
- **HTRAP13** Writing triggers hypervisor trap \$13
- **HTRAP14** Writing triggers hypervisor trap \$14
- **HTRAP15** Writing triggers hypervisor trap \$15
- **HTRAP16** Writing triggers hypervisor trap \$16
- **HTRAP17** Writing triggers hypervisor trap \$17
- **HTRAP18** Writing triggers hypervisor trap \$18
- **HTRAP19** Writing triggers hypervisor trap \$19
- **HTRAP1A** Writing triggers hypervisor trap \$1A
- **HTRAP1B** Writing triggers hypervisor trap \$1B
- **HTRAP1C** Writing triggers hypervisor trap \$1C

- **HTRAP1D** Writing triggers hypervisor trap \$1D
- **HTRAP1E** Writing triggers hypervisor trap \$1E
- **HTRAP1F** Writing triggers hypervisor trap \$1F
- **HTRAP20** Writing triggers hypervisor trap \$20
- **HTRAP21** Writing triggers hypervisor trap \$21
- **HTRAP22** Writing triggers hypervisor trap \$22
- **HTRAP23** Writing triggers hypervisor trap \$23
- **HTRAP24** Writing triggers hypervisor trap \$24
- **HTRAP25** Writing triggers hypervisor trap \$25
- **HTRAP26** Writing triggers hypervisor trap \$26
- **HTRAP27** Writing triggers hypervisor trap \$27
- **HTRAP28** Writing triggers hypervisor trap \$28
- **HTRAP29** Writing triggers hypervisor trap \$29
- **HTRAP2A** Writing triggers hypervisor trap \$2A
- **HTRAP2B** Writing triggers hypervisor trap \$2B
- **HTRAP2C** Writing triggers hypervisor trap \$2C
- **HTRAP2D** Writing triggers hypervisor trap \$2D
- **HTRAP2E** Writing triggers hypervisor trap \$2E
- **HTRAP2F** Writing triggers hypervisor trap \$2F
- **HTRAP30** Writing triggers hypervisor trap \$30
- **HTRAP31** Writing triggers hypervisor trap \$31
- **HTRAP32** Writing triggers hypervisor trap \$32
- **HTRAP33** Writing triggers hypervisor trap \$33
- **HTRAP34** Writing triggers hypervisor trap \$34
- **HTRAP35** Writing triggers hypervisor trap \$35
- **HTRAP36** Writing triggers hypervisor trap \$36
- **HTRAP37** Writing triggers hypervisor trap \$37
- **HTRAP38** Writing triggers hypervisor trap \$38

- **HTRAP39** Writing triggers hypervisor trap \$39
- **HTRAP3A** Writing triggers hypervisor trap \$3A
- **HTRAP3B** Writing triggers hypervisor trap \$3B
- **HTRAP3C** Writing triggers hypervisor trap \$3C
- **HTRAP3D** Writing triggers hypervisor trap \$3D
- **HTRAP3E** Writing triggers hypervisor trap \$3E
- **HTRAP3F** Writing triggers hypervisor trap \$3F
- **NOEXROM** Override for /EXROM : Must be 0 to enable /EXROM signal
- **NOGAME** Override for /GAME : Must be 0 to enable /GAME signal
- **OCEANA** Enable Ocean Type A cartridge emulation
- **POWEREN** Set to zero to power off computer on supported systems. WRITE ONLY.
- **PREFETCH** Enable expansion RAM pre-fetch logic

## MEGA65 CPU MATHS ACCELERATION REGISTERS

Every MEGA65 contains a combined 32-bit hardware multiplier and divider. This device takes two 32-bit inputs, MULTINA and MULTINB, and simultaneously calculates:

- the 64-bit product of MULTINA and MULTINB
- the 32-bit whole part of MULTINA / MULTINB
- the 32-bit fractional part of MULTINA / MULTINB

It is always updating the outputs based on the inputs, so there is no need to take special action when changing the inputs. The multiplier takes 1 cycle to calculate, and the updated result will thus be available immediately. The hardware divider, however, can take upto 16 cycles depending on the particular inputs. Thus programmers should insert a short delay after changing the inputs before reading the output. As this delay is so short, it can be implemented by simply reading the first byte of the result four times consecutively, as the 4th read will occur after the result has settled.

Some models of the MEGA65 also include a math unit, which helps to accelerate the calculation of fixed-point formulae. This presently disabled in all models of the MEGA65 and will be further documented if and when it becomes available.

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D70F       | 55055      | DIVBUSY    | MULBUSY    |            |            |            | -          |            |            |
| D768       | 55144      |            |            | DIVOUT     |            |            |            |            |            |
| D769       | 55145      |            |            | DIVOUT     |            |            |            |            |            |
| D76A       | 55146      |            |            | DIVOUT     |            |            |            |            |            |
| D76B       | 55147      |            |            | DIVOUT     |            |            |            |            |            |
| D76C       | 55148      |            |            | DIVOUT     |            |            |            |            |            |
| D76D       | 55149      |            |            | DIVOUT     |            |            |            |            |            |
| D76E       | 55150      |            |            | DIVOUT     |            |            |            |            |            |
| D76F       | 55151      |            |            | DIVOUT     |            |            |            |            |            |
| D770       | 55152      |            |            | MULTINA    |            |            |            |            |            |
| D771       | 55153      |            |            | MULTINA    |            |            |            |            |            |
| D772       | 55154      |            |            | MULTINA    |            |            |            |            |            |
| D773       | 55155      |            |            | MULTINA    |            |            |            |            |            |
| D774       | 55156      |            |            | MULTINB    |            |            |            |            |            |
| D775       | 55157      |            |            | MULTINB    |            |            |            |            |            |
| D776       | 55158      |            |            | MULTINB    |            |            |            |            |            |
| D777       | 55159      |            |            | MULTINB    |            |            |            |            |            |
| D778       | 55160      |            |            | MULTOUT    |            |            |            |            |            |
| D779       | 55161      |            |            | MULTOUT    |            |            |            |            |            |
| D77A       | 55162      |            |            | MULTOUT    |            |            |            |            |            |
| D77B       | 55163      |            |            | MULTOUT    |            |            |            |            |            |
| D77C       | 55164      |            |            | MULTOUT    |            |            |            |            |            |
| D77D       | 55165      |            |            | MULTOUT    |            |            |            |            |            |
| D77E       | 55166      |            |            | MULTOUT    |            |            |            |            |            |
| D77F       | 55167      |            |            | MULTOUT    |            |            |            |            |            |
| D780       | 55168      |            |            | MATHINO    |            |            |            |            |            |
| D781       | 55169      |            |            | MATHINO    |            |            |            |            |            |
| D782       | 55170      |            |            | MATHINO    |            |            |            |            |            |
| D783       | 55171      |            |            | MATHINO    |            |            |            |            |            |
| D784       | 55172      |            |            | MATHIN1    |            |            |            |            |            |
| D785       | 55173      |            |            | MATHIN1    |            |            |            |            |            |
| D786       | 55174      |            |            | MATHIN1    |            |            |            |            |            |
| D787       | 55175      |            |            | MATHIN1    |            |            |            |            |            |
| D788       | 55176      |            |            | MATHIN2    |            |            |            |            |            |
| D789       | 55177      |            |            | MATHIN2    |            |            |            |            |            |
| D78A       | 55178      |            |            | MATHIN2    |            |            |            |            |            |
| D78B       | 55179      |            |            | MATHIN2    |            |            |            |            |            |
| D78C       | 55180      |            |            | MATHIN3    |            |            |            |            |            |
| D78D       | 55181      |            |            | MATHIN3    |            |            |            |            |            |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D78E       | 55182      |            |            |            |            | MATHIN3    |            |            |            |
| D78F       | 55183      |            |            |            |            | MATHIN3    |            |            |            |
| D790       | 55184      |            |            |            |            | MATHIN4    |            |            |            |
| D791       | 55185      |            |            |            |            | MATHIN4    |            |            |            |
| D792       | 55186      |            |            |            |            | MATHIN4    |            |            |            |
| D793       | 55187      |            |            |            |            | MATHIN4    |            |            |            |
| D794       | 55188      |            |            |            |            | MATHIN5    |            |            |            |
| D795       | 55189      |            |            |            |            | MATHIN5    |            |            |            |
| D796       | 55190      |            |            |            |            | MATHIN5    |            |            |            |
| D797       | 55191      |            |            |            |            | MATHIN5    |            |            |            |
| D798       | 55192      |            |            |            |            | MATHIN6    |            |            |            |
| D799       | 55193      |            |            |            |            | MATHIN6    |            |            |            |
| D79A       | 55194      |            |            |            |            | MATHIN6    |            |            |            |
| D79B       | 55195      |            |            |            |            | MATHIN6    |            |            |            |
| D79C       | 55196      |            |            |            |            | MATHIN7    |            |            |            |
| D79D       | 55197      |            |            |            |            | MATHIN7    |            |            |            |
| D79E       | 55198      |            |            |            |            | MATHIN7    |            |            |            |
| D79F       | 55199      |            |            |            |            | MATHIN7    |            |            |            |
| D7A0       | 55200      |            |            |            |            | MATHIN8    |            |            |            |
| D7A1       | 55201      |            |            |            |            | MATHIN8    |            |            |            |
| D7A2       | 55202      |            |            |            |            | MATHIN8    |            |            |            |
| D7A3       | 55203      |            |            |            |            | MATHIN8    |            |            |            |
| D7A4       | 55204      |            |            |            |            | MATHIN9    |            |            |            |
| D7A5       | 55205      |            |            |            |            | MATHIN9    |            |            |            |
| D7A6       | 55206      |            |            |            |            | MATHIN9    |            |            |            |
| D7A7       | 55207      |            |            |            |            | MATHIN9    |            |            |            |
| D7A8       | 55208      |            |            |            |            | MATHIN10   |            |            |            |
| D7A9       | 55209      |            |            |            |            | MATHIN10   |            |            |            |
| D7AA       | 55210      |            |            |            |            | MATHIN10   |            |            |            |
| D7AB       | 55211      |            |            |            |            | MATHIN10   |            |            |            |
| D7AC       | 55212      |            |            |            |            | MATHIN11   |            |            |            |
| D7AD       | 55213      |            |            |            |            | MATHIN11   |            |            |            |
| D7AE       | 55214      |            |            |            |            | MATHIN11   |            |            |            |
| D7AF       | 55215      |            |            |            |            | MATHIN11   |            |            |            |
| D7B0       | 55216      |            |            |            |            | MATHIN12   |            |            |            |
| D7B1       | 55217      |            |            |            |            | MATHIN12   |            |            |            |
| D7B2       | 55218      |            |            |            |            | MATHIN12   |            |            |            |
| D7B3       | 55219      |            |            |            |            | MATHIN12   |            |            |            |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D7B4       | 55220      |            |            |            |            | MATHIN 13  |            |            |            |
| D7B5       | 55221      |            |            |            |            | MATHIN 13  |            |            |            |
| D7B6       | 55222      |            |            |            |            | MATHIN 13  |            |            |            |
| D7B7       | 55223      |            |            |            |            | MATHIN 13  |            |            |            |
| D7B8       | 55224      |            |            |            |            | MATHIN 14  |            |            |            |
| D7B9       | 55225      |            |            |            |            | MATHIN 14  |            |            |            |
| D7BA       | 55226      |            |            |            |            | MATHIN 14  |            |            |            |
| D7BB       | 55227      |            |            |            |            | MATHIN 14  |            |            |            |
| D7BC       | 55228      |            |            |            |            | MATHIN 15  |            |            |            |
| D7BD       | 55229      |            |            |            |            | MATHIN 15  |            |            |            |
| D7BE       | 55230      |            |            |            |            | MATHIN 15  |            |            |            |
| D7BF       | 55231      |            |            |            |            | MATHIN 15  |            |            |            |
| D7C0       | 55232      |            | UNIT0INB   |            |            |            |            | UNIT0INA   |            |
| D7C1       | 55233      |            | UNIT1INB   |            |            |            |            | UNIT1INA   |            |
| D7C2       | 55234      |            | UNIT2INB   |            |            |            |            | UNIT2INA   |            |
| D7C3       | 55235      |            | UNIT3INB   |            |            |            |            | UNIT3INA   |            |
| D7C4       | 55236      |            | UNIT4INB   |            |            |            |            | UNIT4INA   |            |
| D7C5       | 55237      |            | UNIT5INB   |            |            |            |            | UNIT5INA   |            |
| D7C6       | 55238      |            | UNIT6INB   |            |            |            |            | UNIT6INA   |            |
| D7C7       | 55239      |            | UNIT7INB   |            |            |            |            | UNIT7INA   |            |
| D7C8       | 55240      |            | UNIT8INB   |            |            |            |            | UNIT8INA   |            |
| D7C9       | 55241      |            | UNIT9INB   |            |            |            |            | UNIT9INA   |            |
| D7CA       | 55242      |            | UNIT10INB  |            |            |            |            | UNIT10INA  |            |
| D7CB       | 55243      |            | UNIT11INB  |            |            |            |            | UNIT11INA  |            |
| D7CC       | 55244      |            | UNIT12INB  |            |            |            |            | UNIT12INA  |            |
| D7CD       | 55245      |            | UNIT13INB  |            |            |            |            | UNIT13INA  |            |
| D7CE       | 55246      |            | UNIT14INB  |            |            |            |            | UNIT14INA  |            |
| D7CF       | 55247      |            | UNIT15INB  |            |            |            |            | UNIT15INA  |            |
| D7D0       | 55248      | -          |            |            |            |            |            | UNIT0OUT   |            |
| D7D1       | 55249      | -          |            |            |            |            |            | UNIT1OUT   |            |
| D7D2       | 55250      | -          |            |            |            |            |            | UNIT2OUT   |            |
| D7D3       | 55251      | -          |            |            |            |            |            | UNIT3OUT   |            |
| D7D4       | 55252      | -          |            |            |            |            |            | UNIT4OUT   |            |
| D7D5       | 55253      | -          |            |            |            |            |            | UNIT5OUT   |            |
| D7D6       | 55254      | -          |            |            |            |            |            | UNIT6OUT   |            |
| D7D7       | 55255      | -          |            |            |            |            |            | UNIT7OUT   |            |
| D7D8       | 55256      | -          |            |            |            |            |            | UNIT8OUT   |            |
| D7D9       | 55257      | -          |            |            |            |            |            | UNIT9OUT   |            |

continued ...

...continued

| HEX  | DEC   | DB7 | DB6 | DB5 | DB4 | DB3      | DB2 | DB1    | DB0       |
|------|-------|-----|-----|-----|-----|----------|-----|--------|-----------|
| D7DA | 55258 |     | -   |     |     |          |     |        | UNIT10OUT |
| D7DB | 55259 |     | -   |     |     |          |     |        | UNIT11OUT |
| D7DC | 55260 |     | -   |     |     |          |     |        | UNIT12OUT |
| D7DD | 55261 |     | -   |     |     |          |     |        | UNIT13OUT |
| D7DE | 55262 |     | -   |     |     |          |     |        | UNIT14OUT |
| D7DF | 55263 |     | -   |     |     |          |     |        | UNIT15OUT |
| D7E0 | 55264 |     |     |     |     | LATCHINT |     |        |           |
| D7E1 | 55265 |     |     |     | -   |          |     | CALCEN | WREN      |
| D7E2 | 55266 |     |     |     |     | RESERVED |     |        |           |
| D7E3 | 55267 |     |     |     |     | RESERVED |     |        |           |

- **CALCEN** Enable committing of output values from math units back to math registers (clearing effectively pauses iterative formulae)
- **DIVBUSY** Set if hardware divider is busy
- **DIVOUT** 64-bit output of MULTINA ÷ MULTINB
- **LATCHINT** Latch interval for latched outputs (in CPU cycles)
- **MATHINO** Math unit 32-bit input 0
- **MATHIN1** Math unit 32-bit input 1
- **MATHIN10** Math unit 32-bit input 10
- **MATHIN11** Math unit 32-bit input 11
- **MATHIN12** Math unit 32-bit input 12
- **MATHIN13** Math unit 32-bit input 13
- **MATHIN14** Math unit 32-bit input 14
- **MATHIN15** Math unit 32-bit input 15
- **MATHIN2** Math unit 32-bit input 2
- **MATHIN3** Math unit 32-bit input 3
- **MATHIN4** Math unit 32-bit input 4
- **MATHIN5** Math unit 32-bit input 5
- **MATHIN6** Math unit 32-bit input 6
- **MATHIN7** Math unit 32-bit input 7
- **MATHIN8** Math unit 32-bit input 8

- **MATHIN9** Math unit 32-bit input 9
- **MULBUSY** Set if hardware multiplier is busy
- **MULTINA** Multiplier input A / Divider numerator (32 bit)
- **MULTINB** Multiplier input B / Divider denominator (32 bit)
- **MULTOUT** 64-bit output of MULTINA × MULTINB
- **RESERVED** Reserved
- **UNIT0INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 0.
- **UNIT0INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 0.
- **UNIT0OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 0
- **UNIT10INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 10.
- **UNIT10INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 10.
- **UNIT10OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit A
- **UNIT11INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 11.
- **UNIT11INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 11.
- **UNIT11OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit B
- **UNIT12INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 12.
- **UNIT12INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 12.
- **UNIT12OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit C
- **UNIT13INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 13.

- **UNIT13INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 13.
- **UNIT13OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit D
- **UNIT14INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 14.
- **UNIT14INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 14.
- **UNIT14OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit E
- **UNIT15INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 15.
- **UNIT15INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 15.
- **UNIT15OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit F
- **UNIT1INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 1.
- **UNIT1INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 1.
- **UNIT1OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 1
- **UNIT2INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 2.
- **UNIT2INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 2.
- **UNIT2OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 2
- **UNIT3INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 3.
- **UNIT3INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 3.
- **UNIT3OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 3

- **UNIT4INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 4.
- **UNIT4INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 4.
- **UNIT4OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 4
- **UNIT5INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 5.
- **UNIT5INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 5.
- **UNIT5OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 5
- **UNIT6INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 6.
- **UNIT6INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 6.
- **UNIT6OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 6
- **UNIT7INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 7.
- **UNIT7INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 7.
- **UNIT7OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 7
- **UNIT8INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 8.
- **UNIT8INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 8.
- **UNIT8OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 8
- **UNIT9INA** Select which of the 16 32-bit math registers is input A for Math Function Unit 9.
- **UNIT9INB** Select which of the 16 32-bit math registers is input B for Math Function Unit 9.

- **UNIT9OUT** Select which of the 16 32-bit math registers receives the output of Math Function Unit 9
- **WREN** Enable setting of math registers (must normally be set)

## MEGA65 HYPERVISOR MODE

### Reset

On power-up or reset, the MEGA65 starts up in hypervisor mode, and expects to find a program in the 16KiB hypervisor memory, and begins executing instructions at address \$8100. Normally a JMP instruction will be located at this address, that will jump into a reset routine. That is, the 45GS02 does not use the normal 6502 reset vector. Its function is emulated by the Hyppo hypervisor program, which fetches the address from the 6502 reset vector in the loaded client operating system when exiting hypervisor mode.

The hypervisor memory is automatically mapped on reset to \$8000 - \$BFFF. This special memory is not able to be mapped or accessed, except when in hypervisor mode. It can, however, always be accessed from the serial monitor/debugger interface via its 28-bit address, \$FFF8000 - \$FFF8FFF. This is to protect it from accidental or malicious access from a guest operating system.

### Entering / Exiting Hypervisor Mode

Entering the Hypervisor occurs whenever any of the following events occurs:

- **Power-on** When the MEGA65 is first powered on.
- **Reset** If the reset line is lowered, or a watch-dog triggered reset occurs.
- **SYSCALL register accessed** The registers \$D640 - \$D67F in the MEGA65 IO context trigger SYSCALLs when accessed. This is intended to be the mechanism by which a client operating system or process requests the attention of the hypervisor or operating system.
- **Page Fault** On MEGA65s that feature virtual memory, a page fault will cause a trap to hypervisor mode.
- **Certain keyboard events** Pressing the **RESTORE** key for >0.5 seconds, or the **ALT** + **TAB** key combination traps to the hypervisor. Typically the first is used to launch the freeze menu and the second to toggle the display of debug interface.

- **Accessing virtualised IO devices** For example, if the F011 (internal 3.5" disk drive controller) has been virtualised, then attempting to read or write sectors using this device will cause traps to the hypervisor.
- **Executing an instruction that would lock up the CPU** A number of undocumented opcodes on the 6502 will cause the CPU to lockup. On the MEGA65, instead of locking up, the computer will trap to the hypervisor. This could be used to implement alternative instruction behaviours, or simply to tell the user that something bad has happened.
- **Certain special events** Some devices can generate hypervisor-level interrupts. These are implemented as traps to the hypervisor.

The 45GS02 handles all of these in a similar manner internally:

1. The SYSCALL or trap address is calculated, based on the event.
2. The contents of all CPU registers are saved into the virtualisation control registers.
3. The hypervisor mode memory layout is activated, the CPU decimal flag and special purpose registers are all set to appropriate values. The contents of the A,X,Y and Z and most other CPU flags are preserved, so that they can be accessed from the Hypervisor's SYSCALL/trap handler routine, without having to load them, thus saving a few cycles for each call.
4. The hypervisor-mode flag is asserted, and the programme counter (PC) register is set to the computed address.

All of the above happens in one CPU cycle, i.e., in 25 nano-seconds. Returning from a SYSCALL or trap consists simply of writing to \$D67F, which requires 125 nano-seconds, for a total overhead of 150 nano-seconds. This gives the MEGA65 SYSCALL performance rivalling – even beating – even the fastest modern computers, where the system call latency is typically hundreds to tens of thousands of cycles [2].

## Hypervisor Memory Layout

The hypervisor memory is 16KiB in size. The first 512 bytes are reserved for SYSCALL and system trap entry points, with four bytes for each. For example, the reset entry point is at \$8100 - \$8100 + 3 = \$8100 - \$8103. This allows 4 bytes for an instruction, typically a JMP instruction, followed by a NOP to pad it to 4 bytes.

The full list of SYSCALLs and traps is:

| <b>HEX</b> | <b>DEC</b> | <b>Name</b> | <b>Description</b>                       |
|------------|------------|-------------|------------------------------------------|
| 8000       | 32768      | SYSCALL00   | SYSCALL 0 entry point                    |
| 8004       | 32772      | SYSCALL01   | SYSCALL 1 entry point                    |
| 8008       | 32776      | SYSCALL02   | SYSCALL 2 entry point                    |
| 800C       | 32780      | SYSCALL03   | SYSCALL 3 entry point                    |
| 8010       | 32784      | SYSCALL04   | SYSCALL 4 entry point                    |
| 8014       | 32788      | SYSCALL05   | SYSCALL 5 entry point                    |
| 8018       | 32792      | SYSCALL06   | SYSCALL 6 entry point                    |
| 801C       | 32796      | SYSCALL07   | SYSCALL 7 entry point                    |
| 8020       | 32800      | SYSCALL08   | SYSCALL 8 entry point                    |
| 8024       | 32804      | SYSCALL09   | SYSCALL 9 entry point                    |
| 8028       | 32808      | SYSCALL0A   | SYSCALL 10 entry point                   |
| 802C       | 32812      | SYSCALL0B   | SYSCALL 11 entry point                   |
| 8030       | 32816      | SYSCALL0C   | SYSCALL 12 entry point                   |
| 8034       | 32820      | SYSCALL0D   | SYSCALL 13 entry point                   |
| 8038       | 32824      | SYSCALL0E   | SYSCALL 14 entry point                   |
| 803C       | 32828      | SYSCALL0F   | SYSCALL 15 entry point                   |
| 8040       | 32832      | SYSCALL10   | SYSCALL 16 entry point                   |
| 8044       | 32836      | SECURENTR   | Enter secure container trap entry point  |
| 8048       | 32840      | SECUREXIT   | Leave secure container trap entry point. |
| 804C       | 32844      | SYSCALL13   | SYSCALL 19 entry point                   |
| 8050       | 32848      | SYSCALL14   | SYSCALL 20 entry point                   |
| 8054       | 32852      | SYSCALL15   | SYSCALL 21 entry point                   |
| 8058       | 32856      | SYSCALL16   | SYSCALL 22 entry point                   |
| 805C       | 32860      | SYSCALL17   | SYSCALL 23 entry point                   |
| 8060       | 32864      | SYSCALL18   | SYSCALL 24 entry point                   |
| 8064       | 32868      | SYSCALL19   | SYSCALL 25 entry point                   |
| 8068       | 32872      | SYSCALL1A   | SYSCALL 26 entry point                   |
| 806C       | 32876      | SYSCALL1B   | SYSCALL 27 entry point                   |
| 8070       | 32880      | SYSCALL1C   | SYSCALL 28 entry point                   |
| 8074       | 32884      | SYSCALL1D   | SYSCALL 29 entry point                   |
| 8078       | 32888      | SYSCALL1E   | SYSCALL 30 entry point                   |
| 807C       | 32892      | SYSCALL1F   | SYSCALL 31 entry point                   |
| 8080       | 32896      | SYSCALL20   | SYSCALL 32 entry point                   |
| 8084       | 32900      | SYSCALL21   | SYSCALL 33 entry point                   |
| 8088       | 32904      | SYSCALL22   | SYSCALL 34 entry point                   |
| 808C       | 32908      | SYSCALL23   | SYSCALL 35 entry point                   |
| 8090       | 32912      | SYSCALL24   | SYSCALL 36 entry point                   |
| 8094       | 32916      | SYSCALL25   | SYSCALL 37 entry point                   |
| 8098       | 32920      | SYSCALL26   | SYSCALL 38 entry point                   |

continued ...

...continued

| <b>HEX</b>  | <b>DEC</b>    | <b>Name</b> | <b>Description</b>                           |
|-------------|---------------|-------------|----------------------------------------------|
| 809C        | 32924         | SYSCALL27   | SYSCALL 39 entry point                       |
| 80A0        | 32928         | SYSCALL28   | SYSCALL 40 entry point                       |
| 80A4        | 32932         | SYSCALL29   | SYSCALL 41 entry point                       |
| 80A8        | 32936         | SYSCALL2A   | SYSCALL 42 entry point                       |
| 80AC        | 32940         | SYSCALL2B   | SYSCALL 43 entry point                       |
| 80B0        | 32944         | SYSCALL2C   | SYSCALL 44 entry point                       |
| 80B4        | 32948         | SYSCALL2D   | SYSCALL 45 entry point                       |
| 80B8        | 32952         | SYSCALL2E   | SYSCALL 46 entry point                       |
| 80BC        | 32956         | SYSCALL2F   | SYSCALL 47 entry point                       |
| 80C0        | 32960         | SYSCALL30   | SYSCALL 48 entry point                       |
| 80C4        | 32964         | SYSCALL31   | SYSCALL 49 entry point                       |
| 80C8        | 32968         | SYSCALL32   | SYSCALL 50 entry point                       |
| 80CC        | 32972         | SYSCALL33   | SYSCALL 51 entry point                       |
| 80D0        | 32976         | SYSCALL34   | SYSCALL 52 entry point                       |
| 80D4        | 32980         | SYSCALL35   | SYSCALL 53 entry point                       |
| 80D8        | 32984         | SYSCALL36   | SYSCALL 54 entry point                       |
| 80DC        | 32988         | SYSCALL37   | SYSCALL 55 entry point                       |
| 80E0        | 32992         | SYSCALL38   | SYSCALL 56 entry point                       |
| 80E4        | 32996         | SYSCALL39   | SYSCALL 57 entry point                       |
| 80E8        | 33000         | SYSCALL3A   | SYSCALL 58 entry point                       |
| 80EC        | 33004         | SYSCALL3B   | SYSCALL 59 entry point                       |
| 80F0        | 33008         | SYSCALL3C   | SYSCALL 60 entry point                       |
| 80F4        | 33012         | SYSCALL3D   | SYSCALL 61 entry point                       |
| 80F8        | 33016         | SYSCALL3E   | SYSCALL 62 entry point                       |
| 80FC        | 33020         | SYSCALL3F   | SYSCALL 63 entry point                       |
| 8100        | 33024         | RESET       | Power-on/reset entry point                   |
| 8104        | 33028         | PAGFAULT    | Page fault entry point (not currently used)  |
| 8108        | 33032         | RESTORKEY   | Restore-key long press trap entry point      |
| 810C        | 33036         | ALTTABKEY   | ALT+TAB trap entry point                     |
| 8110        | 33040         | VF011RD     | F011 virtualised disk read trap entry point  |
| 8114        | 33044         | VF011WR     | F011 virtualised disk write trap entry point |
| 8118        | 33048         | BREAKPT     | CPU break-point encountered                  |
| 811C - 81FB | 33048 - 33275 | RESERVED    | Reserved traps point entry                   |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>Name</b> | <b>Description</b>                            |
|------------|------------|-------------|-----------------------------------------------|
| 81FC       | 33276      | CPUKIL      | KIL instruction in 6502-mode trap entry point |

The remainder of the 16KiB hypervisor memory is available for use by the programmer, but will typically use the last 512 bytes for the stack and zero-page, giving an overall memory map as follows:

| <b>HEX</b>  | <b>DEC</b>    | <b>Description</b>                                             |
|-------------|---------------|----------------------------------------------------------------|
| 8000 - 81FF | 32768 - 33279 | SYSCALL and trap entry points                                  |
| 8200 - BDFF | 33280 - 48639 | Available for hypervisor or operating system program           |
| 8E00 - BEFF | 48640 - 48895 | Processor stack for hypervisor or operating system             |
| 8F00 - BFFF | 48896 - 49151 | Processor zero-page storage for hypervisor or operating system |

The stack is used for holding the return address of function calls. The zero-page storage is typically used for holding variables and other short-term storage, as is customary on the 6502.

## Hypervisor Virtualisation Control Registers

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D640       | 54848      |            |            |            |            |            |            |            | REGA       |
| D641       | 54849      |            |            |            |            |            |            |            | REGX       |
| D643       | 54851      |            |            |            |            |            |            |            | REGZ       |
| D644       | 54852      |            |            |            |            |            |            |            | REGB       |
| D645       | 54853      |            |            |            |            |            |            |            | SPL        |
| D646       | 54854      |            |            |            |            |            |            |            | SPH        |
| D647       | 54855      |            |            |            |            |            |            |            | PFLAGS     |

continued ...

...continued

| HEX  | DEC   | DB7 | DB6      | DB5 | DB4 | DB3        | DB2   | DB1     | DB0 |
|------|-------|-----|----------|-----|-----|------------|-------|---------|-----|
| D648 | 54856 |     |          |     |     | PCL        |       |         |     |
| D649 | 54857 |     |          |     |     | PCH        |       |         |     |
| D64A | 54858 |     |          |     |     | MAPLO      |       |         |     |
| D64B | 54859 |     |          |     |     | MAPLO      |       |         |     |
| D64C | 54860 |     |          |     |     | MAPHI      |       |         |     |
| D64D | 54861 |     |          |     |     | MAPHI      |       |         |     |
| D64E | 54862 |     |          |     |     | MAPLOMB    |       |         |     |
| D64F | 54863 |     |          |     |     | MAPHIMB    |       |         |     |
| D650 | 54864 |     |          |     |     | PORT00     |       |         |     |
| D651 | 54865 |     |          |     |     | PORT01     |       |         |     |
| D652 | 54866 |     | -        |     |     |            | EXSID | VICMODE |     |
| D653 | 54867 |     |          |     |     | DMASRCMB   |       |         |     |
| D654 | 54868 |     |          |     |     | DMADSTMB   |       |         |     |
| D655 | 54869 |     |          |     |     | DMALADDR   |       |         |     |
| D656 | 54870 |     |          |     |     | DMALADDR   |       |         |     |
| D657 | 54871 |     |          |     |     | DMALADDR   |       |         |     |
| D658 | 54872 |     |          |     |     | DMALADDR   |       |         |     |
| D659 | 54873 |     | -        |     |     |            | VFLOP | VFLOP   |     |
| D670 | 54896 |     |          |     |     | GEORAMBASE |       |         |     |
| D671 | 54897 |     |          |     |     | GEORAMMASK |       |         |     |
| D672 | 54898 | -   | MATRIXEN |     |     |            | -     |         |     |
| D67C | 54908 |     |          |     |     | UARTDATA   |       |         |     |
| D67D | 54909 |     |          |     |     | WATCHDOG   |       |         |     |
| D67E | 54910 |     |          |     |     | HICKED     |       |         |     |
| D67F | 54911 |     |          |     |     | ENTEREXIT  |       |         |     |

- **ASCFAST** Hypervisor enable ASC/DIN CAPS LOCK key to enable/disable CPU slowdown in C64/C128/C65 modes
- **CPUFAST** Hypervisor force CPU to 48MHz for userland (userland can override via POKE0)
- **DMADSTMB** Hypervisor DMAgic destination MB
- **DMALADDR** Hypervisor DMAgic list address bits 0-7
- **DMASRCMB** Hypervisor DMAgic source MB
- **ENTEREXIT** Writing trigger return from hypervisor
- **EXSID** 0=Use internal SIDs, 1=Use external(1) SIDs

- **F4502** Hypervisor force CPU to 4502 personality, even in C64 IO mode.
- **GEORAMBASE** Hypervisor GeoRAM base address (x MB)
- **GEORAMMASK** Hypervisor GeoRAM address mask (applied to GeoRAM block register)
- **HICKED** Hypervisor already-upgraded bit (writing sets permanently)
- **JMP32EN** Hypervisor enable 32-bit JMP/JSR etc
- **MAPHI** Hypervisor MAPHI register storage (high bits)
- **MAPHIMB** Hypervisor MAPHI mega-byte number register storage
- **MAPLO** Hypervisor MAPLO register storage (high bits)
- **MAPLOMB** Hypervisor MAPLO mega-byte number register storage
- **MATRIXEN** Enable composited Matrix Mode, and disable UART access to serial monitor.
- **PCH** Hypervisor PC-high register storage
- **PCL** Hypervisor PC-low register storage
- **PFLAGS** Hypervisor P register storage
- **PIRQ** Hypervisor flag to indicate if an IRQ is pending on exit from the hypervisor / set 1 to force IRQ/NMI deferral for 1,024 cycles on exit from hypervisor.
- **PNMI** Hypervisor flag to indicate if an NMI is pending on exit from the hypervisor.
- **PORT00** Hypervisor CPU port \$00 value
- **PORT01** Hypervisor CPU port \$01 value
- **REGA** Hypervisor A register storage
- **REGB** Hypervisor B register storage
- **REGX** Hypervisor X register storage
- **REGZ** Hypervisor Z register storage
- **ROMPROT** Hypervisor write protect C65 ROM \$20000-\$3FFF
- **RSVD** RESERVED
- **SPH** Hypervisor SPH register storage
- **SPL** Hypervisor SPL register storage
- **UARTDATA** (write) Hypervisor write serial output to UART monitor

- **VFLOP** 1=Virtualise SD/Floppy0 access (usually for access via serial debugger interface)
- **VICMODE** VIC-II/VIC-III/VIC-IV mode select
- **WATCHDOG** Hypervisor watchdog register: writing any value clears the watch dog

## Programming for Hypervisor Mode

The easiest way to write a program for Hypervisor Mode on the MEGA65 is to use KickC, which is a special version of C made for writing programs for 6502-class processors. The following example programs are from KickC's supplied examples. KickC produces very efficient code, and directly supports the MEGA65's hypervisor mode quite easily through the use of a linker definition file with the following contents:

```
.file [name="%0.bin", type="bin", segments="XMega65Bin"]
.segmentdef XMega65Bin [segments="Syscall, Code, Data, Stack, Zeropage"]
.segmentdef Syscall [start=$8000, max=$81ff]
.segmentdef Code [start=$8200, min=$8200, max=$bdff]
.segmentdef Data [startAfter="Code", min=$8200, max=$bdff]
.segmentdef Stack [min=$be00, max=$beff, fill]
.segmentdef Zeropage [min=$bf00, max=$bfff, fill]
```

This file instructs KickC's assembler to create a 16KiB file with the 512 byte SYSCALL/trap entry point region at the start, followed by code and data areas, and then the stack and zero-page areas. It enforces the size and location of these fields, and will give an error during compilation if anything is too big to fit.

With this file in place, you can then create a KickC source file that provides data structures for the SYSCALL/trap table, e.g.:

```
// XMega65 KERNAL Development Template
// Each function of the KERNAL is a no-args function
// The functions are placed in the SYSCALLS table surrounded by JMP and NOP
```

```
import "string"
```

```
// Use a linker definition file (put the previous listing into that file)
#pragma link("mega65hyper.ld")
```

```
// Some definitions of addresses and special values that this program uses
```

```
const char* RASTER = 0xd012;
const char* VICMEMORY = 0xd018;
const char* SCREEN = 0x0400;
const char* BGCOL = 0xd021;
const char* COLS = 0xd000;
const char BLACK = 0;
const char WHITE = 1;
```

```
// Some text to display
```

```
char[] MESSAGE = "hello world!";
```

```
void Main() {
```

```
    // Initialise screen memory, and select correct font
```

```
    *VICMEMORY = 0x14;
```

```
    // Fill the screen with spaces
```

```
    memset(SCREEN, ' ', 40*25);
```

```
    // Set the colour of every character on the screen to white
```

```
    memset(COLS, WHITE, 40*25);
```

```
    // Print the "Hello world!" message
```

```
    char* sc = SCREEN+40; // Display it one line down on the screen
```

```
    char* msg = MESSAGE; // The message to display
```

```
    // A simple copy routine to copy the string
```

```
    while(*msg) {
```

```
        *sc++ = *msg++;
```

```
}
```

```
    // Loop forever showing two white lines as raster bars
```

```
    while(true) {
```

```
        if(*RASTER==54 || *RASTER==66) {
```

```
            *BGCOL = WHITE;
```

```
        } else {
```

```
            *BGCOL = BLACK;
```

```
        }
```

```
}
```

```
// Here are a couple sample SYSCALL handlers that just display a character on the screen
```

```
void syscall1() {
```

If you save the first listing into a file called mega65hyper.ld, and the second into a file called mega65hyper.kc, you can then compile them using KickC with a command like:

```
kickc -a mega65hyper
```

It will then produce a file called mega65hyper.bin, which you can then try out on your MEGA65, or run in the XMega65 emulator with a command like:

```
xmega65 -kickup mega65hyper.bin
```

# H

## APPENDIX

### 45GS02 & 6502 Instruction Sets

- Addressing Modes
- 6502 Instruction Set
- 4510 Instruction Set
- 45GS02 Compound Instructions



The 45GS02 CPU is able to operate in native mode, where it is compatible with the CSG 4510, and in 6502 compatibility mode, where 6502 undocumented instructions, also known as illegal instructions, are supported for compatibility.

When in 4510 compatibility mode, the 45GS02 also supports a number of extensions through *compound instructions*. These work by prefixing the desired instruction's opcode with one or more *prefix bytes*, which represent sequences of instructions that should not normally occur. For example, two **NEG** instructions in a row acts as a prefix to tell the 45GS02 that the following instruction will operate on 32 bits of data, instead of the usual 8 bits of data. This means that a 45GS02 instruction stream can be readily decoded or disassembled, without needing to set special instruction length flags, as is the case with the 65816 family of microprocessors. The trade-off is increased execution time, as the 45GS02 must skip over the prefix-bytes.

The remainder of this chapter introduces the addressing modes, instructions, opcodes and instruction timing data of the 45GS02, beginning with 6502 compatibility mode, before moving on to 4510 compatibility mode, and the 45GS02 extensions.

## ADDRESSING MODES

The 45GS02 supports 36 different addressing modes, which are explained below. Many of these are very similar to one another, being variations of the normal 6502 or 65CE02 addressing modes, except that they accept either 32-bit pointers, operate on 32-bits of data, or both.

### Implied

In this mode, there are no operands, as the precise function of the instruction is implied by the instruction itself. For example, the **INX** instruction increments the X Register.

### Accumulator

In this mode, the Accumulator is the operand. This is typically used to shift, rotate or modify the value of the Accumulator Register in some way. For example, **INC A** increments the value in the Accumulator Register.

# Q Pseudo Register

In this mode, the Q Pseudo Register is the operand. This is typically used to shift, rotate or modify the value of the Q Pseudo Register in some way. For example, `ASL Q` shifts the value in the Q Pseudo Register left one bit.

Remember that the Q Pseudo Register is simply the A, X, Y and Z registers acting together as a pseudo 32-bit register, where A contains the least significant bits, and Z the most significant bits. There are some cases where using a Q mode instruction can be helpful for operating on the four true registers.

## Immediate Mode

In this mode, the argument to the instruction is a value that is used directly. This is indicated by preceding the value with a # character. Most assemblers allow values to be entered in decimal, or in hexadecimal by preceding the value with a \$ sign, in binary, by preceding the value with a % sign. For example, to set the Accumulator Register to the value 5, you could use the following:

```
LDA #5
```

The immediate argument is encoded as a single byte following the instruction. For the above case, the instruction stream would contain \$A9, the opcode for LDA immediate mode, followed by \$05, the immediate operand.

## Immediate Word Mode

In this mode, the argument is a 16-bit value that is used directly. There is only one instruction which uses this addressing mode, `PWH`. For example, to push the word \$1234 onto the stack, you could use:

```
PWH #$1234
```

The low-byte of the immediate value follows the opcode of the instruction. The high-byte of the immediate value then follows that. For the above example, the instruction stream would thus be \$F4 \$34 \$12.

## Base Page (Zero-Page) Mode

In this mode, the argument is an 8-bit address. The upper 8-bits of the address are taken from the Base Page Register. On 6502 processors, there is no Base Page Register, and instead, the upper 8-bits are always set to zero – hence the name of this mode on the 6502: Zero-Page. On the 45GS02, it is possible to move this “Zero-Page” to any page in the processor’s 64KB view of memory by setting the Base Page Register using the TAB instruction. Base Page Mode allows faster access to a 256 region of memory, and uses less instruction bytes to do so.

The argument is encoded as a single byte that immediately follows the instruction opcode. For example, LDA \$12 would read the value stored in location \$12 in the Base Page, and put it into the Accumulator Register. The instruction byte stream for this would be \$85 \$12.

## Base Page (Zero-Page) Quad Mode

In this mode, the argument is an 8-bit address. The upper 8-bits of the address are taken from the Base Page Register. On 6502 processors, there is no Base Page Register, and instead, the upper 8-bits are always set to zero – hence the name of this mode on the 6502: Zero-Page. On the 45GS02, it is possible to move this “Zero-Page” to any page in the processor’s 64KB view of memory by setting the Base Page Register using the TAB instruction. Base Page Mode allows faster access to a 256 region of memory, and uses less instruction bytes to do so.

The argument is encoded as a single byte that immediately follows the instruction opcode. For example, LDQ \$12 would read the value stored in locations \$12 – \$15 in the Base Page, and put them into the Q Pseudo Register.

## Base Page (Zero-Page) X Indexed Mode

This mode is identical to Base Page Mode, except that the address is formed by taking the argument, and adding the value of the X Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Mode.

## **Base Page (Zero-Page) Quad X Indexed Mode**

This mode is identical to Base Page Quad Mode, except that the address is formed by taking the argument, and adding the value of the X Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Quad Mode.

## **Base Page (Zero-Page) Y Indexed Mode**

This mode is identical to Base Page Mode, except that the address is formed by taking the argument, and adding the value of the Y Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Mode.

## **Base Page (Zero-Page) Base Y Indexed Mode**

This mode is identical to Base Page Quad Mode, except that the address is formed by taking the argument, and adding the value of the Y Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Quad Mode.

## **Base Page (Zero-Page) Z Indexed Mode**

This mode is identical to Base Page Mode, except that the address is formed by taking the argument, and adding the value of the Z Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Mode.

# **Base Page (Zero-Page) Quad Z Indexed Mode**

This mode is identical to Base Page Quad Mode, except that the address is formed by taking the argument, and adding the value of the Z Register to it. In 6502 mode, the result will always be in the Base Page, that is, any carry due to the addition from the low-byte into the high-byte of the address will be ignored. The encoding for this addressing mode is identical to Base Page Quad Mode.

## **Absolute Mode**

In this mode, the argument is an 16-bit address. The low 8-bits of the address are taken from the byte immediately following the instruction opcode. The upper 8-bits are taken from the byte following that. For example, the instruction `LDA $1234`, would read the memory location \$1234, and place the read value into the Accumulator Register. This would be encoded as \$AD \$34 \$12.

## **Absolute Quad Mode**

In this mode, the argument is an 16-bit address. The low 8-bits of the address are taken from the byte immediately following the instruction opcode. The upper 8-bits are taken from the byte following that. For example, the instruction `LDQ $1234`, would read the memory locations \$1234 - \$1237, and place the read values into the Q Pseudo Register. This would be encoded as \$42 \$42 \$AD \$34 \$12.

## **Absolute X Indexed Mode**

This mode is identical to Absolute Mode, except that the address is formed by taking the argument, and adding the value of the X Register to it. If the indexing causes the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Mode.

## **Absolute Quad X Indexed Mode**

This mode is identical to Absolute Quad Mode, except that the address is formed by taking the argument, and adding the value of the X Register to it. If the indexing causes

the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Quad Mode.

## Absolute Y Indexed Mode

This mode is identical to Absolute Mode, except that the address is formed by taking the argument, and adding the value of the Y Register to it. If the indexing causes the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Mode.

## Absolute Quad Y Indexed Mode

This mode is identical to Absolute Quad Mode, except that the address is formed by taking the argument, and adding the value of the Y Register to it. If the indexing causes the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Quad Mode.

## Absolute Z Indexed Mode

This mode is identical to Absolute Mode, except that the address is formed by taking the argument, and adding the value of the Z Register to it. If the indexing causes the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Mode.

## Absolute Quad Z Indexed Mode

This mode is identical to Absolute Quad Mode, except that the address is formed by taking the argument, and adding the value of the Z Register to it. If the indexing causes the address to cross a page boundary, i.e., if the upper byte of the address changes, this may incur a 1 cycle penalty, depending on the processor mode and speed setting. The encoding for this addressing mode is identical to Absolute Quad Mode.

## Absolute Indirect Mode

In this mode, the 16-bit argument is the address that points to, i.e., contains the address of actual byte to read. For example, if memory location \$1234 contains \$78 and memory location \$1235 contains \$56, then `JMP ($1234)` would jump to address \$5678. The encoding for this addressing mode is identical to Absolute Mode.

## Absolute Indirect X-Indexed Mode

In this mode, the 16-bit argument is the address that points to, i.e., contains the address of actual byte to read. It is identical to Absolute Indirect Mode, except that the value of the X Register is added to the pointer address. For example, if the X Register contains the value \$04, memory location \$1238 contains \$78 and memory location \$1239 contains \$56, then `JMP ($1234)` would jump to address \$5678. The encoding for this addressing mode is identical to Absolute Mode.

## Base Page Indirect X-Indexed Mode

This addressing mode is identical to Absolute Indirect X-Indexed Mode, except that the address of the pointer is formed from the Base Page Register (high byte) and the 8-bit operand (low byte). The encoding for this addressing mode is identical to Base Page Mode.

## Base Page Quad Indirect X-Indexed Mode

This addressing mode is identical to Base PAge Indirect X-Indexed Mode, except that the address of the pointer is formed from the Base Page Register (high byte) and the 8-bit operand (low byte). The encoding for this addressing mode is identical to Base Page Quad Mode.

## Base Page Indirect Y-Indexed Mode

This addressing mode differs from the X-Indexed Indirect modes, in that the Y Register is added to the address that is read from the pointer, instead of being added to the pointer. This is a very useful mode, that is frequently because it effectively provides access to

"the Y-th byte of the memory at the address pointed to by the operand." That is, it de-references a pointer. The encoding for this addressing mode is identical to Base Page Mode.

## **Base Page Quad Indirect Y-Indexed Mode**

This addressing mode is identical to the Base Page Indirect Y-Indexed Mode, except that 32-bits of data are operated on. The encoding for this addressing mode is identical to Base Page Mode, except that it is prefixed by \$42, \$42.

## **Base Page Indirect Z-Indexed Mode**

This addressing mode differs from the X-Indexed Indirect modes, in that the Z Register is added to the address that is read from the pointer, instead of being added to the pointer. This is a very useful mode, that is frequently because it effectively provides access to "the Z-th byte of the memory at the address pointed to by the operand." That is, it de-references a pointer. The encoding for this addressing mode is identical to Base Page Mode.

That is, it is equivalent to the Base Page Indirect Y-Indexed Mode.

## **Base Page Quad Indirect Z-Indexed Mode**

This addressing mode is identical to the Base Page Indirect Z-Indexed Mode, except that 32-bits of data are operated on. The encoding for this addressing mode is identical to Base Page Mode, except that it is prefixed by \$42, \$42.

## **32-bit Base Page Indirect Z-Indexed Mode**

This mode is formed by preceding a Base Page Indirect Z-Indexed Mode instruction with the NOP instruction (opcode \$EA). This causes the 45GS02 to read a 32-bit address instead of a 16-bit address from the Base Page address indicated by its operand. The

Z index is added to that pointer. Importantly, the 32-bit address does not refer to the processor's current 64KB view of memory, but rather to the 45GS02's true 28-bit address space. This allows easy access to any memory, without requiring the use of complex bank-switching or DMA operations.

For example, if addresses \$12 to \$15 contained the bytes \$20, \$D0, \$FF, \$0D, and the Z index contained the value \$01, the following instruction sequence would change the screen colour to blue:

```
LDA #$06  
LDZ #$01  
STA [$12],Z
```

## 32-bit Base Page Indirect Quad Z-Indexed Mode

This addressing mode is identical to the 32-bit Base Page Indirect Z-Indexed Mode, except that it operates on 32-bits of data at the 32-bit address formed by the argument, in comparison to 32-bit Base Page Indirect Z-Indexed Mode which operates on only 8 bits of data. The encoding of this addressing mode is \$42, \$42, \$EA, followed by the natural 6502 opcode for the instruction being performed.

## 32-bit Base Page Indirect Mode

This mode is formed by preceding a Base Page Indirect Z-Indexed Mode instruction with the NOP instruction (opcode \$EA). This causes the 45GS02 to read a 32-bit address instead of a 16-bit address from the Base Page address indicated by its operand. Importantly, the 32-bit address does not refer to the processor's current 64KB view of memory, but rather to the 45GS02's true 28-bit address space. This allows easy access to any memory, without requiring the use of complex bank-switching or DMA operations.

For example, if addresses \$12 to \$15 contained the bytes \$20, \$D0, \$FF, \$0D, the following instruction sequence would change the screen border colour to blue:

```
LDA #$06  
STA [$12]
```

Note: The ACME assembler is the only assembler that currently supports this addressing mode. For other assemblers, you can achieve the same result by using a **NOP** instruction to prefix the equivalent *Base Page Indirect, Indexed by Z* instruction.

```
LDA #$06  
NOP  
STA ($12),Z
```

The encoding for this addressing mode is identical to Base Page Mode.

## 32-bit Base Page Indirect Mode

This addressing mode is identical to the 32-bit Base Page Indirect Mode, except that it operates on 32-bits of data at the 32-bit address formed by the argument, in comparison to 32-bit Base Page Indirect Mode which operates on only 8 bits of data. The encoding of this addressing mode is \$42, \$42, \$EA, followed by the natural 6502 opcode for the instruction being performed.

## Stack Relative Indirect, Indexed by Y

This addressing mode is similar to Base Page Indirect Y-Indexed Mode, except that instead of providing the address of the pointer in the Base Page, the operand indicates the offset in the stack to find the pointer. This addressing mode effectively de-references a pointer that has been placed on the stack, e.g., as part of a function call from a high-level language. It is encoded identically to the Base Page Mode.

## Relative Addressing Mode

In this addressing mode, the operand is an 8-bit signed offset to the current value of the Programme Counter (PC). It is used to allow branches to encode the near-by address at which execution should proceed if the branch is taken.

## Relative Word Addressing Mode

This addressing mode is identical to Relative Addressing Mode, except that the address offset is a 16-bit value. This allows a relative branch or jump to any location in the current 64KB memory view. This makes it possible to write software that is fully relocatable, by avoiding the need for absolute addresses when calling routines.

# 6502 INSTRUCTION SET

NOTE: The mechanisms for switching from 4510 to 6502 CPU personality have yet to be finalised.

NOTE: Not all 6502 illegal opcodes are currently implemented.

# Opcode Map

|      |     |     |     |     |     |     |     |     |     |     |     |     |     |     |     |     |
|------|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|-----|
| \$x0 | BRK | ORA | KIL | SLO | NOP | ORA | ASL | SLO | PHP | ORA | ASL | ANC | NOP | ORA | ASL | SLO |
| \$1x | BPL | ORA | KIL | SLO | NOP | ORA | ASL | SLO | CLC | ORA | NOP | SLO | NOP | ORA | ASL | SLO |
| \$2x | JSR | AND | KIL | RLA | BIT | AND | ROL | RLA | PLP | AND | ROL | ANC | BIT | AND | ROL | RLA |
| \$3x | BMI | AND | KIL | RLA | NOP | AND | ROL | RLA | SEC | AND | NOP | RLA | NOP | AND | ROL | RLA |
| \$4x | RTI | EOR | KIL | SRE | NOP | EOR | LSR | SRE | PHA | EOR | LSR | ALR | JMP | EOR | LSR | SRE |
| \$5x | BVC | EOR | KIL | SRE | NOP | EOR | LSR | SRE | CLI | EOR | NOP | SRE | NOP | EOR | LSR | SRE |
| \$6x | RTS | ADC | KIL | RRA | NOP | ADC | ROR | RRA | PLA | ADC | ROR | ARR | JMP | ADC | ROR | RRA |
| \$7x | BVS | ADC | KIL | RRA | NOP | ADC | ROR | RRA | SEI | ADC | NOP | RRA | NOP | ADC | ROR | RRA |
| \$8x | NOP | STA | NOP | SAX | STY | STA | STX | SAX | DEY | NOP | TXA | XAA | STY | STA | STX | SAX |
| \$9x | BCC | STA | KIL | SHA | STY | STA | STX | SAX | TYA | STA | TXS | TAS | SHY | STA | SHX | SHA |
| \$Ax | LDY | LDA | LDX | LAX | LDY | LDA | LDX | LAX | TAY | LDA | TAX | LAX | LDY | LDA | LDX | LAX |
| \$Bx | BCS | LDA | KIL | LAX | LDY | LDA | LDX | LAX | CLV | LDA | TSX | LAS | LDY | LDA | LDX | LAX |
| \$Cx | CPI | CMP | NOP | DCP | CPI | CMP | DEC | DCP | INY | CMP | DEX | SBX | CPI | CMP | DEC | DCP |
| \$Dx | BNE | CMP | KIL | DCP | NOP | CMP | DEC | DCP | CLD | CMP | NOP | DCP | NOP | CMP | DEC | DCP |
| \$Ex | CPX | SBC | NOP | ISC | CPX | SBC | INC | ISC | INX | SBC | NOP | SBC | CPI | SBC | INC | ISC |
| \$Fx | BEQ | SBC | KIL | ISC | NOP | SBC | INC | ISC | SED | SBC | NOP | ISC | NOP | SBC | INC | ISC |

# Instruction Timing

The following table summarises the base instruction timing for 6502 mode. Please also read the information for 4510 mode, as it discusses a number of important factors that affect these figures.

|      | \$x0           | \$x1           | \$x2 | \$x3           | \$x4 | \$x5           | \$x6 | \$x7 | \$x8 | \$x9           | \$xA | \$xB           | \$xC           | \$xD           | \$xE           | \$xF           |
|------|----------------|----------------|------|----------------|------|----------------|------|------|------|----------------|------|----------------|----------------|----------------|----------------|----------------|
| \$0x | 7              | 6              | 9    | 8              | 3    | 3              | 5    | 5    | 3    | 2              | 2    | 2              | 4              | 4              | 6              | 6              |
| \$1x | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4 <sup>p</sup> | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |
| \$2x | 6              | 6              | 9    | 8              | 3    | 3              | 5    | 5    | 4    | 2              | 2    | 2              | 4              | 4              | 6              | 6              |
| \$3x | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4              | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |
| \$4x | 6              | 6              | 9    | 8              | 3    | 3              | 5    | 5    | 3    | 2              | 2    | 2              | 3              | 4              | 6              | 6              |
| \$5x | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4              | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |
| \$6x | 6              | 6              | 9    | 8              | 3    | 3              | 5    | 5    | 4    | 2              | 2    | 2              | 5              | 4              | 6              | 6              |
| \$7x | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4              | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |
| \$8x | 2              | 6              | 2    | 6              | 3    | 3              | 3    | 3    | 2    | 2              | 2    | 2              | 4              | 4              | 4              | 4              |
| \$9x | 2 <sup>b</sup> | 6              | 9    | 6              | 4    | 4              | 4    | 4    | 2    | 5              | 2    | 5              | 5              | 5              | 5              | 5              |
| \$Ax | 2              | 6              | 2    | 6              | 3    | 3              | 3    | 3    | 2    | 2              | 2    | 2              | 4              | 4              | 4              | 4              |
| \$Bx | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 5 <sup>p</sup> | 4    | 4              | 4    | 4    | 2    | 4 <sup>p</sup> | 2    | 4 <sup>p</sup> |
| \$Cx | 2              | 6              | 2    | 8              | 3    | 3              | 5    | 5    | 2    | 2              | 2    | 2              | 4              | 4              | 6              | 6              |
| \$Dx | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4              | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |
| \$Ex | 2              | 6              | 2    | 8              | 3    | 3              | 5    | 5    | 2    | 2              | 2    | 2              | 4              | 4              | 6              | 6              |
| \$Fx | 2 <sup>b</sup> | 5 <sup>p</sup> | 9    | 8              | 4    | 4              | 6    | 6    | 2    | 4 <sup>p</sup> | 2    | 7              | 4 <sup>p</sup> | 4 <sup>p</sup> | 7              | 7              |

*b* Add one cycle if branch crosses a page boundary.  
*p* Add one cycle if indexing crosses a page boundary.

# Addressing Mode Table

|      | \$x0     | \$x1     | \$x2     | \$x3     | \$x4     | \$x5     | \$x6     | \$x7     | \$x8   | \$x9 | \$xA   | \$xB     | \$xC     | \$xD     | \$xE     | \$xF     |
|------|----------|----------|----------|----------|----------|----------|----------|----------|--------|------|--------|----------|----------|----------|----------|----------|
| \$0x | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$1x | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$2x | \$nnnn   | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$3x | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$4x |          | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$5x | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$6x |          | (\$nn,X) | #\$Snn | A    | #\$Snn | (\$nnnn) | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$7x | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$8x | #\$Snn   | (\$nn,X) | #\$Snn   | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$9x | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$Ax | #\$Snn   | (\$nn,X) | #\$Snn   | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$Bx | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$Cx | #\$Snn   | (\$nn,X) | #\$Snn   | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$Dx | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |
| \$Ex | #\$Snn   | (\$nn,X) | #\$Snn   | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | (\$nn,X) | #\$Snn | A    | #\$Snn | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   | \$nnnn   |
| \$Fx | \$rr     | (\$nn,Y) | #\$Snn | A    | #\$Snn | \$nnnn,Y | \$nnnn,X | \$nnnn,X | \$nnnn,X | \$nnnn,X |

# Official And Unintended Instructions

The 6502 opcode matrix has a size of  $16 \times 16 = 256$  possible opcodes. Those, that are officially documented, form the set of the [legal](#) instructions. All instructions of this legal set are headed by a blue coloured mnemonic.

The remaining opcodes form the set of the [unintended](#) instructions (sometimes called "illegal" instructions). For the sake of completeness these are documented too. All instructions of the unintended set are headed by a red coloured mnemonic.

The unintended instructions are implemented in the 6502 mode, but are not guaranteed to produce exactly the same results as on other CPU's of the 65xx family. Many of these instructions are known to be unstable, even running on old hardware.

## ADC

This instruction adds the argument to the contents of the Accumulator Register and the Carry Flag. If the D flag is set, then the addition is performed using Binary Coded Decimal.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The C flag will be set if the unsigned result is  $>255$ , or  $>99$  if the D flag is set.

**ADC : Add with carry****6502** $A \leftarrow A + M + C$ **N Z I C D V E**

+ + . + . + .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | ADC (\$nn,X)    | 61          | 2            | 6             |          |
| zero-page              | ADC \$nn        | 65          | 2            | 3             |          |
| immediate              | ADC #\$nn       | 69          | 2            | 2             |          |
| absolute               | ADC \$nnnn      | 6D          | 3            | 4             |          |
| (indirect),Y           | ADC (\$nn),Y    | 71          | 2            | 5             | <i>p</i> |
| zero-page,X            | ADC \$nn,X      | 75          | 2            | 4             |          |
| absolute,Y             | ADC \$nnnn,Y    | 79          | 3            | 4             | <i>p</i> |
| absolute,X             | ADC \$nnnn,X    | 7D          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.**ALR**

This instruction shifts the Accumulator one bit right after performing a binary AND of the Accumulator and the immediate mode argument. Bit 7 will be set to zero, and the bit 0 will be shifted out into the Carry Flag

**Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

**ALR : Binary AND and Logical Shift Right****6502** $A \leftarrow (A \text{ AND Value}) \gg 1, C \leftarrow A(0)$ **N Z I C D V E**

+ + . + . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | ALR #\$nn       | 4B          | 2            | 2             |

**ANC**

This instruction performs a binary AND operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumula-

tor, and that are set in the argument will be set in the accumulator on completion. Unlike the AND instruction, the Carry Flag is set as though the result were shifted left one bit. That is, the Carry Flag is set in the same way as the Negative Flag.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The C flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>ANC : Binary AND, and Set Carry</b>           | <b>6502</b>     |             |              |               |
|--------------------------------------------------|-----------------|-------------|--------------|---------------|
| A $\leftarrow$ A AND M, C $\leftarrow$ A7 AND M7 |                 |             |              |               |
| <b>Addressing Mode</b>                           | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| immediate                                        | ANC #\$nn       | 0B          | 2            | 2             |
| immediate                                        | ANC #\$nn       | 2B          | 2            | 2             |

## AND

This instruction performs a binary AND operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, and that are set in the argument will be set in the accumulator on completion.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**AND : Binary AND****6502**A  $\leftarrow$  A AND M

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | AND (\$nn,X)    | 21          | 2            | 6             |          |
| zero-page              | AND \$nn        | 25          | 2            | 3             |          |
| immediate              | AND #\$nn       | 29          | 2            | 2             |          |
| absolute               | AND \$nnnn      | 2D          | 3            | 4             |          |
| (indirect),Y           | AND (\$nn),Y    | 31          | 2            | 5             | <i>p</i> |
| zero-page,X            | AND \$nn,X      | 35          | 2            | 4             |          |
| absolute,Y             | AND \$nnnn,Y    | 39          | 3            | 4             | <i>p</i> |
| absolute,X             | AND \$nnnn,X    | 3D          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

**ARR**

This instruction shifts the Accumulator one bit right after performing a binary AND of the Accumulator and the immediate mode argument. Bit 7 is exchanged with the carry.

**Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The V flag will be apparently be affected in some way.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

**ARR : Binary AND and Rotate Right****6502**A  $\leftarrow$  (A AND Value)  $>>1$ , C  $\leftarrow$  A(7)

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | +        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | ARR #\$nn       | 6B          | 2            | 2             |

# ASL

This instruction shifts either the Accumulator or contents of the provided memory location one bit left. Bit 0 will be set to zero, and the bit 7 will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| <b>ASL : Arithmetic Shift Left</b>               |                 | <b>6502</b>          |              |               |  |  |
|--------------------------------------------------|-----------------|----------------------|--------------|---------------|--|--|
| $A \leftarrow A \ll 1$ or $M \leftarrow M \ll 1$ |                 | <b>N Z I C D V E</b> |              |               |  |  |
|                                                  |                 | + + . + . .          |              |               |  |  |
| <b>Addressing Mode</b>                           | <b>Assembly</b> | <b>Code</b>          | <b>Bytes</b> | <b>Cycles</b> |  |  |
| zero-page                                        | ASL \$nn        | 06                   | 2            | 5             |  |  |
| accumulator                                      | ASL A           | 0A                   | 1            | 2             |  |  |
| absolute                                         | ASL \$nnnn      | 0E                   | 3            | 6             |  |  |
| zero-page,X                                      | ASL \$nn,X      | 16                   | 2            | 6             |  |  |
| absolute,X                                       | ASL \$nnnn,X    | 1E                   | 3            | 7             |  |  |

# BCC

This instruction branches to the indicated address if the Carry Flag is clear.

| <b>BCC : Branch on Carry Flag Clear</b> |                 | <b>6502</b>          |              |               |   |  |
|-----------------------------------------|-----------------|----------------------|--------------|---------------|---|--|
| $PC \leftarrow PC + R8$                 |                 | <b>N Z I C D V E</b> |              |               |   |  |
|                                         |                 | . . . . .            |              |               |   |  |
| <b>Addressing Mode</b>                  | <b>Assembly</b> | <b>Code</b>          | <b>Bytes</b> | <b>Cycles</b> |   |  |
| relative                                | BCC \$rr        | 90                   | 2            | 2             | b |  |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# BCS

This instruction branches to the indicated address if the Carry Flag is set.

| BCS : Branch on Carry Flag Set                                                                            |          | 6502 |   |   |         |
|-----------------------------------------------------------------------------------------------------------|----------|------|---|---|---------|
| PC ← PC + R8                                                                                              |          | N    | Z | I | C D V E |
| relative                                                                                                  | BCS \$rr | B0   | 2 | 2 | b       |
| <i>b</i> Add one cycle if branch is taken.<br>Add one more cycle if branch taken crosses a page boundary. |          |      |   |   |         |

# BEQ

This instruction branches to the indicated address if the Zero Flag is set.

| BEQ : Branch on Zero Flag Set                                                                             |          | 6502 |   |   |         |
|-----------------------------------------------------------------------------------------------------------|----------|------|---|---|---------|
| PC ← PC + R8                                                                                              |          | N    | Z | I | C D V E |
| relative                                                                                                  | BEQ \$rr | F0   | 2 | 2 | b       |
| <i>b</i> Add one cycle if branch is taken.<br>Add one more cycle if branch taken crosses a page boundary. |          |      |   |   |         |

# BIT

This instruction is used to test the bits stored in a memory location. Bits 6 and 7 of the memory location's contents are directly copied into the Overflow Flag and Negative Flag. The Zero Flag is set or cleared based on the result of performing the binary AND of the Accumulator Register and the contents of the indicated memory location.

## Side effects

- The N flag will be set if the bit 7 of the memory location is set, else it will be cleared.
- The V flag will be set if the bit 6 of the memory location is set, else it will be cleared.
- The Z flag will be set if the result of A AND M is zero, else it will be cleared.

**BIT : Perform Bit Test****6502** $N \leftarrow M(7), V \leftarrow M(6), Z \leftarrow A \text{ AND } M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | +        | .        |

**Addressing Mode****Assembly Code****Bytes****Cycles**

zero-page

BIT \$nn

24

2

3

absolute

BIT \$nnnn

2C

3

4

# BMI

This instruction branches to the indicated address if the Negative Flag is set.

**BMI : Branch on Negative Flag Set****6502** $PC \leftarrow PC + R8$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

**Addressing Mode****Assembly Code****Bytes****Cycles**

relative

BMI \$rr

30

2

2

<sup>b</sup><sup>b</sup> Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# BNE

This instruction branches to the indicated address if the Zero Flag is clear.

**BNE : Branch on Zero Flag Clear****6502** $PC \leftarrow PC + R8$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

**Addressing Mode****Assembly Code****Bytes****Cycles**

relative

BNE \$rr

D0

2

2

<sup>b</sup><sup>b</sup> Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# BPL

This instruction branches to the indicated address if the Negative Flag is clear.

**BPL : Branch on Negative Flag Clear****6502**PC  $\leftarrow$  PC + R8**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b>  |
|------------------------|-----------------|-------------|--------------|----------------|
| relative               | BPL \$rr        | 10          | 2            | 2 <sup>b</sup> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

**BRK**

The break command causes the microprocessor to go through an interrupt sequence under program control. The address of the BRK instruction + 2 is pushed to the stack along with the status register with the Break flag set. This allows the interrupt service routine to distinguish between IRQ events and BRK events. For example:

```

PLA          ; load status
PHA          ; restore stack
AND #$10    ; mask break flag
BNE DO_BREAK ; -> it was a BRK
...          ; else continue with IRQ server

```

Cite from: MCS6500 Microcomputer Family Programming Manual, January 1976, Second Edition, MOS Technology Inc., Page 144:

"The BRK is a single byte instruction and its addressing mode is Implied."

There are debates, that BRK could be seen as a two byte instruction with the addressing mode immediate, where the operand byte is discarded. The byte following the BRK could then be used as a call argument for the break handler. Commodore however used the BRK, as stated in the manual, as a single byte instruction, which breaks into the ML monitor, if present. These builtin monitors decremented the stacked PC, so that it could be used to return or jump directly to the code byte after the BRK.

**BRK : Break to Interrupt****6502**PC  $\leftarrow$  (\$FFFF)**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | BRK             | 00          | 1            | 7             |

## BVC

This instruction branches to the indicated address if the Overflow (V) Flag is clear.

| BVC : Branch on Overflow Flag Clear |                      | 6502                |
|-------------------------------------|----------------------|---------------------|
| PC ← PC + R8                        |                      | N Z I C D V E       |
|                                     |                      | .....               |
| <b>Addressing Mode</b>              | <b>Assembly Code</b> | <b>Bytes</b>        |
| relative                            | BVC \$rr             | 50 2 2 <sup>b</sup> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## BVS

This instruction branches to the indicated address if the Overflow (V) Flag is set.

| BVS : Branch on Overflow Flag Set |                      | 6502                |
|-----------------------------------|----------------------|---------------------|
| PC ← PC + R8                      |                      | N Z I C D V E       |
|                                   |                      | .....               |
| <b>Addressing Mode</b>            | <b>Assembly Code</b> | <b>Bytes</b>        |
| relative                          | BVS \$rr             | 70 2 2 <sup>b</sup> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## CLC

This instruction clears the Carry Flag.

### Side effects

- The C flag is cleared.

**CLC : Clear Carry Flag**

6502

 $C \leftarrow 0$ 

N Z I C D V E

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | CLC             | 18          | 1            | 2             |

## CLD

This instruction clears the Decimal Flag. Arithmetic operations will use normal binary arithmetic, instead of Binary-Coded Decimal (BCD).

**Side effects**

- The D flag is cleared.

**CLD : Clear Decimal Flag**

6502

 $D \leftarrow 0$ 

N Z I C D V E

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | CLD             | D8          | 1            | 2             |

## CLI

This instruction clears the Interrupt Disable Flag. Interrupts will now be able to occur.

**Side effects**

- The I flag is cleared.

**CLI : Clear Interrupt Disable Flag**

6502

 $I \leftarrow 0$ 

N Z I C D V E

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | CLI             | 58          | 1            | 2             |

# CLV

This instruction clears the Overflow Flag.

## Side effects

- The V flag is cleared.

| CLV : Clear Overflow Flag | 6502     |      |       |        |
|---------------------------|----------|------|-------|--------|
| $V \leftarrow 0$          |          |      |       |        |
| Addressing Mode           | Assembly | Code | Bytes | Cycles |
| implied                   | CLV      | B8   | 1     | 2      |

# CMP

This instruction performs  $A - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

**CMP : Compare Accumulator****6502**

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | CMP (\$nn,X)    | C1          | 2            | 6             |          |
| zero-page              | CMP \$nn        | C5          | 2            | 3             |          |
| immediate              | CMP #\$nn       | C9          | 2            | 2             |          |
| absolute               | CMP \$nnnn      | CD          | 3            | 4             |          |
| (indirect),Y           | CMP (\$nn),Y    | D1          | 2            | 5             | <i>p</i> |
| zero-page,X            | CMP \$nn,X      | D5          | 2            | 4             |          |
| absolute,Y             | CMP \$nnnn,Y    | D9          | 3            | 4             | <i>p</i> |
| absolute,X             | CMP \$nnnn,X    | DD          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

**CPX**

This instruction performs  $X - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

**Side effects**

- The N flag will be set if the result of  $X - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $X - M$  is zero or positive, i.e., if X is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $X - M$  is zero, else it will be cleared.

**CPX : Compare X Register****6502**

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | CPX #\$nn       | E0          | 2            | 2             |
| zero-page              | CPX \$nn        | E4          | 2            | 3             |
| absolute               | CPX \$nnnn      | EC          | 3            | 4             |

# CPY

This instruction performs  $Y - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $Y - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $Y - M$  is zero or positive, i.e., if  $Y$  is not less than  $M$ , else it will be cleared.
- The Z flag will be set if the result of  $Y - M$  is zero, else it will be cleared.

| CPY : Compare Y Register |            | 6502 |       |       |        |   |   |   |
|--------------------------|------------|------|-------|-------|--------|---|---|---|
|                          |            | N    | Z     | I     | C      | D | V | E |
|                          |            | +    | +     | .     | +      | . | . | . |
| Addressing Mode          | Assembly   | Code | Bytes | Bytes | Cycles |   |   |   |
| immediate                | CPY #\$nn  | C0   | 2     | 2     |        |   |   |   |
| zero-page                | CPY \$nn   | C4   | 2     | 3     |        |   |   |   |
| absolute                 | CPY \$nnnn | CC   | 3     | 4     |        |   |   |   |

# DCP

This instruction decrements the contents of the indicated memory location, and then performs  $A - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if  $A$  is not less than  $M$ , else it will be cleared.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

**DCP : Decrement and Compare Accumulator****6502** $M \leftarrow M - 1, A - M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect,X)           | DCP (\$nn,X)    | C3          | 2            | 8             |
| zero-page              | DCP \$nn        | C7          | 2            | 5             |
| absolute               | DCP \$nnnn      | CF          | 3            | 6             |
| (indirect),Y           | DCP (\$nn),Y    | D3          | 2            | 8             |
| zero-page,X            | DCP \$nn,X      | D7          | 2            | 6             |
| absolute,Y             | DCP \$nnnn,Y    | DB          | 3            | 7             |
| absolute,X             | DCP \$nnnn,X    | DF          | 3            | 7             |

## DEC

This instruction decrements the Accumulator Register or indicated memory location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**DEC : Decrement Memory or Accumulator****6502** $A \leftarrow A - 1 \text{ or } M \leftarrow M - 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| zero-page              | DEC \$nn        | C6          | 2            | 5             |
| absolute               | DEC \$nnnn      | CE          | 3            | 6             |
| zero-page,X            | DEC \$nn,X      | D6          | 2            | 6             |
| absolute,X             | DEC \$nnnn,X    | DE          | 3            | 7             |

## DEX

This instruction decrements the X Register.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.

- The Z flag will be set if the result is zero, else it will be cleared.

| <b>DEX : Decrement X Register</b> | <b>6502</b>                       |      |       |        |
|-----------------------------------|-----------------------------------|------|-------|--------|
| $X \leftarrow X - 1$              |                                   |      |       |        |
|                                   | <b>N Z I C D V E</b><br>+ + . . . |      |       |        |
| Addressing Mode                   | Assembly                          | Code | Bytes | Cycles |
| implied                           | DEX                               | CA   | 1     | 2      |

## DEY

This instruction decrements the Y Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>DEY : Decrement Y Register</b> | <b>6502</b>                       |      |       |        |
|-----------------------------------|-----------------------------------|------|-------|--------|
| $Y \leftarrow Y - 1$              |                                   |      |       |        |
|                                   | <b>N Z I C D V E</b><br>+ + . . . |      |       |        |
| Addressing Mode                   | Assembly                          | Code | Bytes | Cycles |
| implied                           | DEY                               | 88   | 1     | 2      |

## EOR

This instruction performs a binary XOR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, but not both.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**EOR : Binary Exclusive OR**

6502

 $A \leftarrow A \text{ XOR } M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | EOR (\$nn,X)    | 41          | 2            | 6             |          |
| zero-page              | EOR \$nn        | 45          | 2            | 3             |          |
| immediate              | EOR #\$nn       | 49          | 2            | 2             |          |
| absolute               | EOR \$nnnn      | 4D          | 3            | 4             |          |
| (indirect),Y           | EOR (\$nn),Y    | 51          | 2            | 5             | <i>p</i> |
| zero-page,X            | EOR \$nn,X      | 55          | 2            | 4             |          |
| absolute,Y             | EOR \$nnnn,Y    | 59          | 3            | 4             | <i>p</i> |
| absolute,X             | EOR \$nnnn,X    | 5D          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

**INC**

This instruction increments the Accumulator Register or indicated memory location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**INC : Increment Memory or Accumulator**

6502

 $A \leftarrow A + 1 \text{ or } M \leftarrow M + 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |  |
|------------------------|-----------------|-------------|--------------|---------------|--|
| zero-page              | INC \$nn        | E6          | 2            | 5             |  |
| absolute               | INC \$nnnn      | EE          | 3            | 6             |  |
| zero-page,X            | INC \$nn,X      | F6          | 2            | 6             |  |
| absolute,X             | INC \$nnnn,X    | FE          | 3            | 7             |  |

**INX**

This instruction increments the X Register, i.e., adds 1 to it.

### **Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>INX : Increment X Register</b> |                      | <b>6502</b>  |               |   |
|-----------------------------------|----------------------|--------------|---------------|---|
| X $\leftarrow$ X + 1              |                      |              |               |   |
| <b>N Z I C D V E</b>              |                      |              |               |   |
| + + . . . .                       |                      |              |               |   |
| <b>Addressing Mode</b>            | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |   |
| implied                           | INX                  | E8           | 1             | 2 |

## **INY**

This instruction increments the Y Register, i.e., adds 1 to it.

### **Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>INY : Increment Y Register</b> |                      | <b>6502</b>  |               |   |
|-----------------------------------|----------------------|--------------|---------------|---|
| Y $\leftarrow$ Y + 1              |                      |              |               |   |
| <b>N Z I C D V E</b>              |                      |              |               |   |
| + + . . . .                       |                      |              |               |   |
| <b>Addressing Mode</b>            | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |   |
| implied                           | INY                  | C8           | 1             | 2 |

## **ISC**

This instruction increments the indicated memory location, and then performs A – M – 1 + C, and sets the processor flags accordingly. The result is stored in the Accumulator Register.

NOTE: This instruction is affected by the status of the Decimal Flag.

### **Side effects**

- The N flag will be set if the result of A – M is negative, i.e. bit 7 is set in the result, else it will be cleared.

- The C flag will be set if the result of A – M is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of A – M is zero, else it will be cleared.

| <b>ISC : Increment Memory, Subtract With Carry</b> | <b>6502</b> |
|----------------------------------------------------|-------------|
| $M \leftarrow M+1, A \leftarrow -M - 1 + C$        |             |

**N Z I C D V E**  
+ + . + . + .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect,X)           | ISC (\$nn,X)    | E3          | 2            | 8             |
| zero-page              | ISC \$nn        | E7          | 2            | 5             |
| absolute               | ISC \$nnnn      | EF          | 3            | 6             |
| (indirect),Y           | ISC (\$nn),Y    | F3          | 2            | 8             |
| zero-page,X            | ISC \$nn,X      | F7          | 2            | 6             |
| absolute,Y             | ISC \$nnnn,Y    | FB          | 3            | 7             |
| absolute,X             | ISC \$nnnn,X    | FF          | 3            | 7             |

## JMP

This instruction sets the Programme Counter (PC) Register to the address indicated by the instruction, causing execution to continue from that address.

| <b>JMP : Jump to Address</b> | <b>6502</b> |
|------------------------------|-------------|
| $PC \leftarrow M2:M1$        |             |

**N Z I C D V E**  
. . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| absolute               | JMP \$nnnn      | 4C          | 3            | 3             |
| indirect               | JMP (\$nnnn)    | 6C          | 3            | 5             |

## JSR

This instruction saves the address of the instruction following the JSR instruction onto the stack, and then sets the Programme Counter (PC) Register to the address indicated by the instruction, causing execution to continue from that address. Because the return address has been saved on the stack, the RTS instruction can be used to return from the called sub-routine and resume execution following the JSR instruction.

NOTE: This instruction actually pushes the address of the last byte of the JSR instruction onto the stack. The RTS instruction naturally is aware of this, and increments the address on popping it from the stack, before setting the Programme Counter (PC) register.

| <b>JSR : Jump to Sub-Routine</b> | <b>6502</b>          |              |               |
|----------------------------------|----------------------|--------------|---------------|
| PC ← M2:M1, Stack ← PCH:PCL      |                      |              |               |
| <b>Addressing Mode</b>           | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| absolute                         | JSR \$nnnn           | 20           | 3             |
|                                  |                      |              | 6             |

## KIL

On a 6502, these instructions cause the processor to enter an infinite loop in their internal logic that can only be aborted by resetting the computer. On the 45GS02 these instructions cause Hypervisor Traps. Or rather, they will, once this functionality has been implemented. Thus they can be used to detect whether running on a 6502 or a 45GS02: If on a 6502 processor, the instruction will never return, while they will cause an exception on a 45GS02, likely causing the calling programme to be aborted or crash.

| <b>KIL : Lock-up 6502 Processor</b> | <b>6502</b>          |              |               |
|-------------------------------------|----------------------|--------------|---------------|
|                                     |                      |              |               |
| <b>Addressing Mode</b>              | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                             | KIL                  | 02           | 1             |
| implied                             | KIL                  | 12           | 1             |
| implied                             | KIL                  | 22           | 1             |
| implied                             | KIL                  | 32           | 1             |
| implied                             | KIL                  | 42           | 1             |
| implied                             | KIL                  | 52           | 1             |
| implied                             | KIL                  | 62           | 1             |
| implied                             | KIL                  | 72           | 1             |
| implied                             | KIL                  | 92           | 1             |
| implied                             | KIL                  | B2           | 1             |
| implied                             | KIL                  | D2           | 1             |
| implied                             | KIL                  | F2           | 1             |

# LAS

NOTE: This monstrosity of an instruction, aside from being devoid of any conceivable useful purpose is unstable on many 6502 processors and should therefore also be avoided for that reason, if you had not already been put off.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- A feeling of hollow satisfaction, when you actually discover a useful purpose for this instruction. exactly how it works.

| <b>LAS : Set A, X and SPL Register With Useless Value</b> |              | <b>6502</b>       |
|-----------------------------------------------------------|--------------|-------------------|
| SP, A, X ← SP AND M                                       |              |                   |
|                                                           |              | N Z I C D V E     |
|                                                           |              | · · · · ·         |
| Addressing Mode                                           | Assembly     | Code Bytes Cycles |
| absolute,Y                                                | LAS \$nnnn,Y | BB 3 4 p          |

p Add one cycle if indexing crosses a page boundary.

# LAX

This instruction loads both the Accumulator Register and X Register with the indicated value, or with the contents of the indicated location.

NOTE: The LAX instruction is known to be unstable on many 6502 processors, and should not be used. Non-immediate modes MAY be stable enough to be usable, but should generally be avoided.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LAX : Load Accumulator and X Registers****6502**A, X  $\leftarrow$  M

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | LAX (\$nn,X)    | A3          | 2            | 6             |          |
| zero-page              | LAX \$nn        | A7          | 2            | 3             |          |
| immediate              | LAX #\$nn       | AB          | 2            | 2             |          |
| absolute               | LAX \$nnnn      | AF          | 3            | 4             |          |
| (indirect),Y           | LAX (\$nn),Y    | B3          | 2            | 5             | <i>p</i> |
| zero-page,Y            | LAX \$nn,Y      | B7          | 2            | 4             |          |
| absolute,Y             | LAX \$nnnn,Y    | BF          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

**LDA**

This instruction loads the Accumulator Register with the indicated value, or with the contents of the indicated location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDA : Load Accumulator****6502**A  $\leftarrow$  M

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| (indirect,X)           | LDA (\$nn,X)    | A1          | 2            | 6             |          |
| zero-page              | LDA \$nn        | A5          | 2            | 3             |          |
| immediate              | LDA #\$nn       | A9          | 2            | 2             |          |
| absolute               | LDA \$nnnn      | AD          | 3            | 4             |          |
| (indirect),Y           | LDA (\$nn),Y    | B1          | 2            | 5             | <i>p</i> |
| zero-page,X            | LDA \$nn,X      | B5          | 2            | 4             |          |
| absolute,Y             | LDA \$nnnn,Y    | B9          | 3            | 4             | <i>p</i> |
| absolute,X             | LDA \$nnnn,X    | BD          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

# LDX

This instruction loads the X Register with the indicated value, or with the contents of the indicated location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| LDX : Load X Register |              | 6502 |       |   |   |   |   |                |
|-----------------------|--------------|------|-------|---|---|---|---|----------------|
| X ← M                 |              | N    | Z     | I | C | D | V | E              |
|                       |              | +    | +     | . | . | . | . | .              |
| Addressing Mode       | Assembly     | Code | Bytes |   |   |   |   | Cycles         |
| immediate             | LDX #\$nn    | A2   | 2     |   |   |   |   | 2              |
| zero-page             | LDX \$nn     | A6   | 2     |   |   |   |   | 3              |
| absolute              | LDX \$nnnn   | AE   | 3     |   |   |   |   | 4              |
| zero-page,Y           | LDX \$nn,Y   | B6   | 2     |   |   |   |   | 4              |
| absolute,Y            | LDX \$nnnn,Y | BE   | 3     |   |   |   |   | <sup>p</sup> 4 |

*p* Add one cycle if indexing crosses a page boundary.

# LDY

This instruction loads the Y Register with the indicated value, or with the contents of the indicated location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDY : Load Y Register****6502** $Y \leftarrow M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| immediate              | LDY #\$nn       | A0          | 2            | 2             |          |
| zero-page              | LDY \$nn        | A4          | 2            | 3             |          |
| absolute               | LDY \$nnnn      | AC          | 3            | 4             |          |
| zero-page,X            | LDY \$nn,X      | B4          | 2            | 4             |          |
| absolute,X             | LDY \$nnnn,X    | BC          | 3            | 4             | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

**LSR**

This instruction shifts either the Accumulator or contents of the provided memory location one bit right. Bit 7 will be set to zero, and the bit 0 will be shifted out into the Carry Flag

**Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

**LSR : Logical Shift Right****6502** $A \leftarrow A >> 1, C \leftarrow A(0)$  or  $M \leftarrow M >> 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| zero-page              | LSR \$nn        | 46          | 2            | 5             |
| accumulator            | LSR A           | 4A          | 1            | 2             |
| absolute               | LSR \$nnnn      | 4E          | 3            | 6             |
| zero-page,X            | LSR \$nn,X      | 56          | 2            | 6             |
| absolute,X             | LSR \$nnnn,X    | 5E          | 3            | 7             |

**NOP**

These instructions act as null instructions: They perform the bus accesses as though they were real instructions, but then do nothing with the retrieved value. They can thus be used

either as delay instructions, or to read from registers that have side-effects when read, without corrupting a register.

Only \$EA is an intended opcode for NOP on the 6502. All others are only available on NMOS versions of the processor, or the 45GS02 in 6502 mode.

| <b>NOP : No-Operation (some are unintended opcodes)</b> |                 |          |          |          | <b>6502</b> |          |          |          |
|---------------------------------------------------------|-----------------|----------|----------|----------|-------------|----------|----------|----------|
|                                                         |                 | <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b>    | <b>D</b> | <b>V</b> | <b>E</b> |
| <b>Addressing Mode</b>                                  | <b>Assembly</b> |          |          |          |             |          |          |          |
| zero-page                                               | NOP \$nn        | 04       | 2        |          | 3           |          |          |          |
| absolute                                                | NOP \$nnnn      | 0C       | 3        |          | 4           |          |          |          |
| zero-page,X                                             | NOP \$nn,X      | 14       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | 1A       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | 1C       | 3        |          | 4           |          |          | <i>p</i> |
| zero-page,X                                             | NOP \$nn,X      | 34       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | 3A       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | 3C       | 3        |          | 4           |          |          | <i>p</i> |
| zero-page                                               | NOP \$nn        | 44       | 2        |          | 3           |          |          |          |
| zero-page,X                                             | NOP \$nn,X      | 54       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | 5A       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | 5C       | 3        |          | 4           |          |          | <i>p</i> |
| zero-page                                               | NOP \$nn        | 64       | 2        |          | 3           |          |          |          |
| zero-page,X                                             | NOP \$nn,X      | 74       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | 7A       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | 7C       | 3        |          | 4           |          |          | <i>p</i> |
| immediate                                               | NOP #\$nn       | 80       | 2        |          | 2           |          |          |          |
| immediate                                               | NOP #\$nn       | 82       | 2        |          | 2           |          |          |          |
| immediate                                               | NOP #\$nn       | 89       | 2        |          | 2           |          |          |          |
| immediate                                               | NOP #\$nn       | C2       | 2        |          | 2           |          |          |          |
| zero-page,X                                             | NOP \$nn,X      | D4       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | DA       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | DC       | 3        |          | 4           |          |          | <i>p</i> |
| immediate                                               | NOP #\$nn       | E2       | 2        |          | 2           |          |          |          |
| implied                                                 | NOP             | EA       | 1        |          | 2           |          |          |          |
| zero-page,X                                             | NOP \$nn,X      | F4       | 2        |          | 4           |          |          |          |
| implied                                                 | NOP             | FA       | 1        |          | 2           |          |          |          |
| absolute,X                                              | NOP \$nnnn,X    | FC       | 3        |          | 4           |          |          | <i>p</i> |

*p* Add one cycle if indexing crosses a page boundary.

# ORA

This instruction performs a binary OR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, or both.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| ORA : Decrement Memory or Accumulator        |              | 6502 |       |       |        |       |
|----------------------------------------------|--------------|------|-------|-------|--------|-------|
| $A \leftarrow A + 1$ or $M \leftarrow M + 1$ |              |      |       |       |        |       |
|                                              |              | N    | Z     | I     | C      | D V E |
|                                              |              | +    | +     | .     | .      | .     |
| Addressing Mode                              | Assembly     | Code | Bytes | Bytes | Cycles |       |
| (indirect,X)                                 | ORA (\$nn,X) | 01   | 2     | 6     |        |       |
| zero-page                                    | ORA \$nn     | 05   | 2     | 3     |        |       |
| immediate                                    | ORA #\$nn    | 09   | 2     | 2     |        |       |
| absolute                                     | ORA \$nnnn   | 0D   | 3     | 4     |        |       |
| (indirect),Y                                 | ORA (\$nn),Y | 11   | 2     | 5     | p      |       |
| zero-page,X                                  | ORA \$nn,X   | 15   | 2     | 4     | p      |       |
| absolute,Y                                   | ORA \$nnnn,Y | 19   | 3     | 4     | p      |       |
| absolute,X                                   | ORA \$nnnn,X | 1D   | 3     | 4     | p      |       |

p Add one cycle if indexing crosses a page boundary.

# PHA

This instruction pushes the contents of the Accumulator Register onto the stack, and decrements the value of the Stack Pointer by 1.

| PHA : Push Accumulator Register onto the Stack |          | 6502 |       |       |        |       |
|------------------------------------------------|----------|------|-------|-------|--------|-------|
| $STACK \leftarrow A$ , $SP \leftarrow SP - 1$  |          |      |       |       |        |       |
|                                                |          | N    | Z     | I     | C      | D V E |
|                                                |          | .    | .     | .     | .      | .     |
| Addressing Mode                                | Assembly | Code | Bytes | Bytes | Cycles |       |
| implied                                        | PHA      | 48   | 1     | 3     |        |       |

## PHP

This instruction pushes the contents of the Processor Flags onto the stack, and decrements the value of the Stack Pointer by 1.

| <b>PHP : Push Processor Flags onto the Stack</b> |                 | <b>6502</b> |
|--------------------------------------------------|-----------------|-------------|
| STACK $\leftarrow$ P, SP $\leftarrow$ SP - 1     |                 |             |
| N Z I C D V E                                    |                 |             |
| . . . . .                                        |                 |             |
|                                                  |                 |             |
| <b>Addressing Mode</b>                           | <b>Assembly</b> | <b>Code</b> |
| implied                                          | PHP             | 08          |
| <b>Bytes</b>                                     | <b>Cycles</b>   | 1 3         |

## PLA

This instruction replaces the contents of the Accumulator Register with the top value from the stack, and increments the value of the Stack Pointer by 1.

| <b>PLA : Pull Accumulator Register from the Stack</b> |                 | <b>6502</b> |
|-------------------------------------------------------|-----------------|-------------|
| A $\leftarrow$ STACK, SP $\leftarrow$ SP + 1          |                 |             |
| N Z I C D V E                                         |                 |             |
| + + . . .                                             |                 |             |
|                                                       |                 |             |
| <b>Addressing Mode</b>                                | <b>Assembly</b> | <b>Code</b> |
| implied                                               | PLA             | 68          |
| <b>Bytes</b>                                          | <b>Cycles</b>   | 1 4         |

## PLP

This instruction replaces the contents of the Processor Flags with the top value from the stack, and increments the value of the Stack Pointer by 1.

NOTE: This instruction does NOT replace the Extended Stack Disable Flag (E Flag), or the Software Interrupt Flag (B Flag)

| <b>PLP : Pull Accumulator Register from the Stack</b> |                 | <b>6502</b> |
|-------------------------------------------------------|-----------------|-------------|
| A $\leftarrow$ STACK, SP $\leftarrow$ SP + 1          |                 |             |
| N Z I C D V E                                         |                 |             |
| + + + + + + .                                         |                 |             |
|                                                       |                 |             |
| <b>Addressing Mode</b>                                | <b>Assembly</b> | <b>Code</b> |
| implied                                               | PLP             | 28          |
| <b>Bytes</b>                                          | <b>Cycles</b>   | 1 4         |

# RLA

This instruction shifts the contents of the provided memory location one bit left. Bit 0 will be set to the current value of the Carry Flag, and the bit 7 will be shifted out into the Carry Flag. The result is then ANDed with the Accumulator.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| <b>RLA : Rotate Left Memory, and AND with Accumulator</b>                                         |              | <b>6502</b> |       |       |        |   |
|---------------------------------------------------------------------------------------------------|--------------|-------------|-------|-------|--------|---|
| $M \leftarrow M \ll 1, C \leftarrow M(7), M(0) \leftarrow C, A \leftarrow A \text{ AND } M \ll 1$ |              |             |       |       |        |   |
| N                                                                                                 | Z            | I           | C     | D     | V      | E |
| +                                                                                                 | +            | .           | +     | .     | .      | . |
| Addressing Mode                                                                                   | Assembly     | Code        | Bytes | Bytes | Cycles |   |
| (indirect),X                                                                                      | RLA (\$nn,X) | 23          | 2     | 2     | 8      |   |
| zero-page                                                                                         | RLA \$nn     | 27          | 2     | 2     | 5      |   |
| absolute                                                                                          | RLA \$nnnn   | 2F          | 3     | 3     | 6      |   |
| (indirect),Y                                                                                      | RLA (\$nn),Y | 33          | 2     | 2     | 8      |   |
| zero-page,X                                                                                       | RLA \$nn,X   | 37          | 2     | 2     | 6      |   |
| absolute,Y                                                                                        | RLA \$nnnn,Y | 3B          | 3     | 3     | 7      |   |
| absolute,X                                                                                        | RLA \$nnnn,X | 3F          | 3     | 3     | 7      |   |

# ROL

This instruction shifts either the Accumulator or contents of the provided memory location one bit left. Bit 0 will be set to the current value of the Carry Flag, and the bit 7 will be shifted out into the Carry Flag.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

**ROL : Rotate Left Memory or Accumulator****6502** $M \leftarrow M << 1, C \leftarrow M(7), M(0) \leftarrow C$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| zero-page              | ROL \$nn        | 26          | 2            | 5             |
| accumulator            | ROL A           | 2A          | 1            | 2             |
| absolute               | ROL \$nnnn      | 2E          | 3            | 6             |
| zero-page,X            | ROL \$nn,X      | 36          | 2            | 6             |
| absolute,X             | ROL \$nnnn,X    | 3E          | 3            | 7             |

## ROR

This instruction shifts either the Accumulator or contents of the provided memory location one bit right. Bit 7 will be set to the current value of the Carry Flag, and the bit 0 will be shifted out into the Carry Flag

**Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

**ROR : Rotate Right Memory or Accumulator****6502** $M \leftarrow M >> 1, C \leftarrow M(0), M(7) \leftarrow C$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| zero-page              | ROR \$nn        | 66          | 2            | 5             |
| accumulator            | ROR A           | 6A          | 1            | 2             |
| absolute               | ROR \$nnnn      | 6E          | 3            | 6             |
| zero-page,X            | ROR \$nn,X      | 76          | 2            | 6             |
| absolute,X             | ROR \$nnnn,X    | 7E          | 3            | 7             |

# RRA

This instruction shifts either the contents of the provided memory location one bit right. Bit 7 will be set to the current value of the Carry Flag, and the bit 0 will be shifted out into the Carry Flag. The result is added to the Accumulator.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if the addition results in an overflow in the Accumulator.

| <b>RRA : Rotate Right Memory, and Add to Accumulator</b>                                            |  | <b>6502</b> |   |   |   |   |   |   |
|-----------------------------------------------------------------------------------------------------|--|-------------|---|---|---|---|---|---|
| M $\leftarrow$ M $>> 1$ , C $\leftarrow$ M(0), M(7) $\leftarrow$ C, A $\leftarrow$ A + M $>> 1$ + C |  | N           | Z | I | C | D | V | E |
|                                                                                                     |  | +           | + | . | + | . | . | . |

  

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect),X)          | RRA (\$nn,X)    | 63          | 2            | 8             |
| zero-page              | RRA \$nn        | 67          | 2            | 5             |
| absolute               | RRA \$nnnn      | 6F          | 3            | 6             |
| (indirect),Y           | RRA (\$nn),Y    | 73          | 2            | 8             |
| zero-page,X            | RRA \$nn,X      | 77          | 2            | 6             |
| absolute,Y             | RRA \$nnnn,Y    | 7B          | 3            | 7             |
| absolute,X             | RRA \$nnnn,X    | 7F          | 3            | 7             |

# RTI

This instruction pops the processor flags from the stack, and then pops the Programme Counter (PC) register from the stack, allowing an interrupted programme to resume.

- The 6502 Processor Flags are restored from the stack.
- Neither the B (Software Interrupt) nor E (Extended Stack) flags are set by this instruction.

**RTI : Return From Interrupt****6502**P  $\leftarrow$  STACK, PC  $\leftarrow$  STACK, SP  $\leftarrow$  SP + 3

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | ·        | ·        | +        | +        | +        | ·        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
|------------------------|-----------------|-------------|--------------|---------------|

|         |     |    |   |   |
|---------|-----|----|---|---|
| implied | RTI | 40 | 1 | 6 |
|---------|-----|----|---|---|

## RTS

This instruction adds optional argument to the Stack Pointer (SP) Register, and then pops the Programme Counter (PC) register from the stack, allowing a routine to return to its caller.

**RTS : Return From Subroutine****6502**PC  $\leftarrow$  STACK + N, SP  $\leftarrow$  SP + 2 + N

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| ·        | ·        | ·        | ·        | ·        | ·        | ·        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
|------------------------|-----------------|-------------|--------------|---------------|

|         |     |    |   |   |
|---------|-----|----|---|---|
| implied | RTS | 60 | 1 | 6 |
|---------|-----|----|---|---|

## SAX

This instruction acts as a combination of AND and CMP. The result is stored in the X Register. Because it includes functionality from CMP rather than SBC, the Carry Flag is not used in the subtraction, although it is modified by the instruction.

NOTE: This instruction is affected by the status of the Decimal Flag.

### Side effects

- The N flag will be set if the result of A – M is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of A – M is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of A – M is zero, else it will be cleared.

**SAX : AND Accumulator and X, and Subtract Without Carry 6502** $X \leftarrow (A \text{ AND } X) - \text{Value}$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | +        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect,X)           | SAX (\$nn,X)    | 83          | 2            | 6             |
| zero-page              | SAX \$nn        | 87          | 2            | 3             |
| absolute               | SAX \$nnnn      | 8F          | 3            | 4             |
| zero-page,Y            | SAX \$nn,Y      | 97          | 2            | 4             |

## SBC

This instruction performs  $A - M - 1 + C$ , and sets the processor flags accordingly. The result is stored in the Accumulator Register.

NOTE: This instruction is affected by the status of the Decimal Flag.

### Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

**SBC : Subtract With Carry****6502** $A \leftarrow -M - 1 + C$ **N Z I C D V E**

+ + . + . + .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect,X)           | SBC (\$nn,X)    | E1          | 2            | 6             |
| zero-page              | SBC \$nn        | E5          | 2            | 3             |
| immediate              | SBC #\$nn       | E9          | 2            | 2             |
| immediate              | SBC #\$nn       | EB          | 2            | 2             |
| absolute               | SBC \$nnnn      | ED          | 3            | 4             |
| (indirect),Y           | SBC (\$nn),Y    | F1          | 2            | 5 <i>p</i>    |
| zero-page,X            | SBC \$nn,X      | F5          | 2            | 4             |
| absolute,Y             | SBC \$nnnn,Y    | F9          | 3            | 4 <i>p</i>    |
| absolute,X             | SBC \$nnnn,X    | FD          | 3            | 4 <i>p</i>    |

*p* Add one cycle if indexing crosses a page boundary.

**SBX**

This instruction loads the X Register with the binary AND of the Accumulator Register and X Register, less the immediate argument.

NOTE: The subtraction effect in this instruction is due to CMP , not . Thus the Negative Flag is set according to the function of CMP, not SBC. That is, the carry flag is not used in the calculation.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if the result is zero or positive, else it will be cleared.

**SBX : AND and Subtract****6502** $X \leftarrow (A \text{ AND } X) - V$ **N Z I C D V E**

+ + . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | SBX #\$nn       | CB          | 2            | 2             |

# SEC

This instruction sets the Carry Flag.

## Side effects

- The C flag is set.

| SEC : Set Carry Flag |               | 6502          |        |   |  |  |
|----------------------|---------------|---------------|--------|---|--|--|
| C ← 1                |               | N Z I C D V E |        |   |  |  |
| Addressing Mode      | Assembly Code | Bytes         | Cycles |   |  |  |
| implied              | SEC           | 38            | 1      | 2 |  |  |

# SED

This instruction sets the Decimal Flag. Binary arithmetic will now use Binary-Coded Decimal (BCD) mode.

NOTE: The C64's interrupt handler does not clear the Decimal Flag, which makes it dangerous to set the Decimal Flag without first setting the Interrupt Disable Flag.

## Side effects

- The D flag is set.

| SED : Set Decimal Flag |               | 6502          |        |   |  |  |
|------------------------|---------------|---------------|--------|---|--|--|
| D ← 1                  |               | N Z I C D V E |        |   |  |  |
| Addressing Mode        | Assembly Code | Bytes         | Cycles |   |  |  |
| implied                | SED           | F8            | 1      | 2 |  |  |

# SEI

This instruction sets the Interrupt Disable Flag. Normal (IRQ) interrupts will no longer be able to occur. Non-Maskable Interrupts (NMI) will continue to occur, as their name suggests.

## Side effects

- The I flag is set.

| <b>SEI : Set Interrupt Disable Flag</b> |                 | <b>6502</b>          |
|-----------------------------------------|-----------------|----------------------|
| $I \leftarrow 1$                        |                 |                      |
|                                         |                 | <b>N Z I C D V E</b> |
| <b>Addressing Mode</b>                  | <b>Assembly</b> | <b>Code</b>          |
| implied                                 | SEI             | 78                   |
|                                         |                 | <b>Bytes Cycles</b>  |
|                                         |                 | 1 2                  |

## SHA

NOTE: This instruction is unstable on many 6502 processors, and should be avoided.

This instruction stores the binary AND of the contents of the Accumulator Register, X Register and the third byte of the instruction into the indicated location.

| <b>SHA : Store binary AND of A, X and 3rd Instruction Byte</b> |                 | <b>6502</b>          |
|----------------------------------------------------------------|-----------------|----------------------|
| $M \leftarrow A \text{ AND } X \text{ AND } B3$                |                 |                      |
|                                                                |                 | <b>N Z I C D V E</b> |
| <b>Addressing Mode</b>                                         | <b>Assembly</b> | <b>Code</b>          |
| (indirect),Y                                                   | SHA (\$nn),Y    | 93                   |
| absolute,Y                                                     | SHA \$nnnn,Y    | 9F                   |
|                                                                |                 | 2 6                  |
|                                                                |                 | 3 5                  |
|                                                                |                 | <b>Bytes Cycles</b>  |

## SHX

NOTE: This instruction is unstable on many 6502 processors, and should be avoided.

This instruction stores the binary AND of the contents of the X Register and the third byte of the instruction into the indicated location.

| <b>SHX : Store Binary AND of X Register and 3rd Instruction Byte</b> |                 | <b>6502</b>          |
|----------------------------------------------------------------------|-----------------|----------------------|
| $M \leftarrow X \text{ AND } B3$                                     |                 |                      |
|                                                                      |                 | <b>N Z I C D V E</b> |
| <b>Addressing Mode</b>                                               | <b>Assembly</b> | <b>Code</b>          |
| absolute,Y                                                           | SHX \$nnnn,Y    | 9E                   |
|                                                                      |                 | 3 5                  |
|                                                                      |                 | <b>Bytes Cycles</b>  |

**SHY**

NOTE: This instruction is unstable on many 6502 processors, and should be avoided.

This instruction stores the binary AND of the contents of the Y Register and the third byte of the instruction into the indicated location.

| <b>SHY : Store Binary AND of Y Register and 3rd Instruction Byte</b> | <b>6502</b>          |
|----------------------------------------------------------------------|----------------------|
| $M \leftarrow Y \text{ AND } B3$                                     |                      |
|                                                                      | <b>N Z I C D V E</b> |
|                                                                      | . . . . .            |
| <b>Addressing Mode</b>                                               | <b>Assembly</b>      |
| absolute,X                                                           | SHY \$nnnn,X         |
|                                                                      | 9C                   |
|                                                                      | 3                    |
|                                                                      | 5                    |

SLO

This instruction shifts either contents of the provided memory location one bit left, and then ORs the result with the Accumulator Register, and places the result in the Accumulator.

### **Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 of the Accumulator is set after the instruction completes, else it will be cleared.
  - The Z flag will be set if the Accumulator contains \$00 after the instruction has completed, else it will be cleared.
  - The C flag will be set if bit 7 of the memory contents was set, prior to being shifted.

| SLO : Shift Left and OR                    |              | 6502 |       |        |   |   |   |   |
|--------------------------------------------|--------------|------|-------|--------|---|---|---|---|
| $M \leftarrow M << 1, A \leftarrow M << 1$ |              | N    | Z     | I      | C | D | V | E |
| Addressing Mode                            | Assembly     | Code | Bytes | Cycles |   |   |   |   |
| (indirect,X)                               | SLO (\$nn,X) | 03   | 2     |        |   |   |   | 8 |
| zero-page                                  | SLO \$nn     | 07   | 2     |        |   |   |   | 5 |
| absolute                                   | SLO \$nnnn   | 0F   | 3     |        |   |   |   | 6 |
| (indirect),Y                               | SLO (\$nn),Y | 13   | 2     |        |   |   |   | 8 |
| zero-page,X                                | SLO \$nn,X   | 17   | 2     |        |   |   |   | 6 |
| absolute,Y                                 | SLO \$nnnn,Y | 1B   | 3     |        |   |   |   | 7 |
| absolute,X                                 | SLO \$nnnn,X | 1F   | 3     |        |   |   |   | 7 |

# SRE

This instruction shifts the contents of the provided memory location one bit right. Bit 7 will be set to zero, and the bit 0 will be shifted out into the Carry Flag. The result is exclusive ORed with the Accumulator and stored in the Accumulator.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

### SRE : Logical Shift Right and Exclusive OR with Accumulator 6502

$M \leftarrow M >> 1, A \leftarrow A \text{ XOR } M >> 1$

| N | Z | I | C | D | V | E |
|---|---|---|---|---|---|---|
| + | + | . | + | . | . | . |

| Addressing Mode | Assembly     | Code | Bytes | Cycles |
|-----------------|--------------|------|-------|--------|
| (indirect,X)    | SRE (\$nn,X) | 43   | 2     | 8      |
| zero-page       | SRE \$nn     | 47   | 2     | 5      |
| absolute        | SRE \$nnnn   | 4F   | 3     | 6      |
| (indirect),Y    | SRE (\$nn),Y | 53   | 2     | 8      |
| zero-page,X     | SRE \$nn,X   | 57   | 2     | 6      |
| absolute,Y      | SRE \$nnnn,Y | 5B   | 3     | 7      |
| absolute,X      | SRE \$nnnn,X | 5F   | 3     | 7      |

# STA

This instruction stores the contents of the Accumulator Register into the indicated location.

**STA : Store Accumulator****6502**

M ← A

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|              |              |    |   |   |
|--------------|--------------|----|---|---|
| (indirect,X) | STA (\$nn,X) | 81 | 2 | 6 |
| zero-page    | STA \$nn     | 85 | 2 | 3 |
| absolute     | STA \$nnnn   | 8D | 3 | 4 |
| (indirect),Y | STA (\$nn),Y | 91 | 2 | 6 |
| zero-page,X  | STA \$nn,X   | 95 | 2 | 4 |
| absolute,Y   | STA \$nnnn,Y | 99 | 3 | 5 |
| absolute,X   | STA \$nnnn,X | 9D | 3 | 5 |

**STX**

This instruction stores the contents of the X Register into the indicated location.

**STX : Store X Register****6502**

M ← X

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|             |            |    |   |   |
|-------------|------------|----|---|---|
| zero-page   | STX \$nn   | 86 | 2 | 3 |
| absolute    | STX \$nnnn | 8E | 3 | 4 |
| zero-page,Y | STX \$nn,Y | 96 | 2 | 4 |

**STY**

This instruction stores the contents of the Y Register into the indicated location.

**STY : Store Y Register****6502**

M ← Y

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|             |            |    |   |   |
|-------------|------------|----|---|---|
| zero-page   | STY \$nn   | 84 | 2 | 3 |
| absolute    | STY \$nnnn | 8C | 3 | 4 |
| zero-page,X | STY \$nn,X | 94 | 2 | 4 |

# TAS

NOTE: This monstrosity of an instruction, aside from being devoid of any conceivable useful purpose is unstable on many 6502 processors and should therefore also be avoided for that reason, if you had not already been put off.

## Side effects

- Remarkably, despite the over complicated operation that it performs, it modifies none of the processor flags.
- Loss of sanity if you attempt to use it, or even figure out exactly how it works.

| <b>TAS : Munge X Register and Stack Pointer</b>                                   |              | <b>6502</b>       |
|-----------------------------------------------------------------------------------|--------------|-------------------|
| SP $\leftarrow A \text{ AND } X, M \leftarrow (A \text{ AND } X) \text{ AND } B3$ |              |                   |
|                                                                                   |              | N Z I C D V E     |
|                                                                                   |              | . . . . .         |
| Addressing Mode                                                                   | Assembly     | Code Bytes Cycles |
| absolute,Y                                                                        | TAS \$nnnn,Y | 9B 3 5            |

# TAX

This instruction loads the X Register with the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>TAX : Transfer Accumulator Register into the X Register</b> |          | <b>6502</b>       |
|----------------------------------------------------------------|----------|-------------------|
| X $\leftarrow A$                                               |          |                   |
|                                                                |          | N Z I C D V E     |
|                                                                |          | + + . . .         |
| Addressing Mode                                                | Assembly | Code Bytes Cycles |
| implied                                                        | TAX      | AA 1 2            |

# TAY

This instruction loads the Y Register with the contents of the Accumulator Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>TAY : Transfer Accumulator Register into the Y Register    6502</b> |                      |              |               |   |  |
|------------------------------------------------------------------------|----------------------|--------------|---------------|---|--|
| Y ← A                                                                  |                      |              |               |   |  |
| N Z I C D V E                                                          |                      |              |               |   |  |
| + + . . . .                                                            |                      |              |               |   |  |
| <b>Addressing Mode</b>                                                 | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |   |  |
| implied                                                                | TAY                  | A8           | 1             | 2 |  |

## TSX

This instruction loads the X Register with the contents of the Stack Pointer High (SPL) Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>TSX : Transfer Stack Pointer High Register into the X Register    6502</b> |                      |              |               |   |  |
|-------------------------------------------------------------------------------|----------------------|--------------|---------------|---|--|
| X ← SPH                                                                       |                      |              |               |   |  |
| N Z I C D                V E                                                  |                      |              |               |   |  |
| + + . . . .                                                                   |                      |              |               |   |  |
| <b>Addressing Mode</b>                                                        | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |   |  |
| implied                                                                       | TSX                  | BA           | 1             | 2 |  |

## TXA

This instruction loads the Accumulator Register with the contents of the X Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TXA : Transfer X Register into the Accumulator Register 6502**A  $\leftarrow$  X

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
|------------------------|-----------------|-------------|--------------|---------------|

|         |     |    |   |   |
|---------|-----|----|---|---|
| implied | TXA | 8A | 1 | 2 |
|---------|-----|----|---|---|

**TXS**

This instruction sets the low byte of the Stack Pointer (SPL) register to the contents of the X Register.

**TXS : Transfer X Register into Stack Pointer Low Register 6502**SPL  $\leftarrow$  X

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
|------------------------|-----------------|-------------|--------------|---------------|

|         |     |    |   |   |
|---------|-----|----|---|---|
| implied | TXS | 9A | 1 | 2 |
|---------|-----|----|---|---|

**TYA**

This instruction loads the Accumulator Register with the contents of the Y Register.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TYA : Transfer Y Register into the Accumulator Register 6502**A  $\leftarrow$  Y

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
|------------------------|-----------------|-------------|--------------|---------------|

|         |     |    |   |   |
|---------|-----|----|---|---|
| implied | TYA | 98 | 1 | 2 |
|---------|-----|----|---|---|

# XAA

This instruction loads the Accumulator Register with the binary AND of the X Register and the immediate mode argument.

NOTE: This instruction is unstable on many 6502 processors, and should not be used.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>XAA : Transfer X into A and AND with operand</b> | <b>6502</b>          |              |               |
|-----------------------------------------------------|----------------------|--------------|---------------|
| A ← X AND VALUE                                     |                      |              |               |
|                                                     | N Z I C D V E        |              |               |
|                                                     | + + . . . . .        |              |               |
| <b>Addressing Mode</b>                              | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| immediate                                           | XAA #\$nn            | 8B           | 2             |

## 4510 INSTRUCTION SET

# Opcode Map

|      | \$x0 | \$x1 | \$x2 | \$x3 | \$x4 | \$x5 | \$x6 | \$x7 | \$x8 | \$x9 | \$xA | \$xB | \$xC | \$xD | \$xE | \$xF |  |
|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|--|
| \$0x | BRK  | ORA  | CLE  | SEE  | TSB  | ORA  | ASL  | RMB0 | PHP  | ORA  | ASL  | TSY  | TSB  | ORA  | ASL  | BBR0 |  |
| \$1x | BPL  | ORA  | ORA  | BPL  | TRB  | ORA  | ASL  | RMB1 | CLC  | ORA  | INC  | INZ  | TRB  | ORA  | ASL  | BBR1 |  |
| \$2x | JSR  | AND  | JSR  | JSR  | BIT  | AND  | ROL  | RMB2 | PLP  | AND  | ROL  | TYS  | BIT  | AND  | ROL  | BBR2 |  |
| \$3x | BMI  | AND  | BMI  | BIT  | AND  | ROL  | RMB3 | SEC  | AND  | DEC  | DEZ  | BIT  | AND  | ROL  | ROL  | BBR3 |  |
| \$4x | RTI  | EOR  | NEG  | ASR  | ASR  | EOR  | LSR  | RMB4 | PHA  | EOR  | LSR  | TAZ  | JMP  | EOR  | LSR  | BBR4 |  |
| \$5x | BVC  | EOR  | EOR  | BVC  | ASR  | EOR  | LSR  | RMB5 | CLI  | EOR  | PHY  | TAB  | MAP  | EOR  | LSR  | BBR5 |  |
| \$6x | RTS  | ADC  | RTS  | BSR  | STZ  | ADC  | ROR  | RMB6 | PLA  | ADC  | ROR  | TZA  | JMP  | ADC  | ROR  | BBR6 |  |
| \$7x | BVS  | ADC  | ADC  | BVS  | STZ  | ADC  | ROR  | RMB7 | SEI  | ADC  | PLY  | TBA  | JMP  | ADC  | ROR  | BBR7 |  |
| \$8x | BRA  | STA  | STA  | BRA  | STY  | STA  | STX  | SMB0 | DEY  | BIT  | TXA  | STY  | STA  | STX  | BBR0 |      |  |
| \$9x | BCC  | STA  | STA  | BCC  | STY  | STA  | STX  | SMB1 | TYA  | STA  | TXS  | STX  | STA  | STZ  | BBR1 |      |  |
| \$Ax | LDY  | LDA  | LDX  | LDY  | LDA  | LDX  | SMB2 | TAY  | LDA  | TAX  | LDZ  | LDY  | LDA  | LDX  | BBR2 |      |  |
| \$Bx | BCS  | LDA  | LDA  | BCS  | LDY  | LDA  | SMB3 | CLV  | LDA  | TSX  | LDZ  | LDY  | LDA  | LDX  | BBR3 |      |  |
| \$Cx | CPY  | CMP  | CPZ  | DEW  | CPY  | CMP  | DEC  | SMB4 | INY  | CMP  | DEX  | ASW  | CPY  | CMP  | DEC  | BBR4 |  |
| \$Dx | BNE  | CMP  | CMP  | BNE  | CPZ  | CMP  | DEC  | SMB5 | CID  | CMP  | PHX  | PHZ  | CPZ  | CMP  | DEC  | BBR5 |  |
| \$Ex | CPX  | SBC  | LDA  | INW  | Cpx  | SBC  | INC  | SMB6 | INX  | SBC  | EOM  | ROW  | CPX  | SBC  | INC  | BBR6 |  |
| \$Fx | BEQ  | SBC  | SBC  | BEQ  | PHW  | SBC  | INC  | SMB7 | SED  | SBC  | PLX  | PLZ  | PHW  | SBC  | INC  | BBR7 |  |

# Instruction Timing

The following table lists the base cycle count for each opcode. Note that the number of cycles depends on the speed setting of the processor: Some instructions take more or fewer cycles when the processor is running at full-speed, or a C65 compatibility 3.5MHz speed, or at C64 compatibility 1MHz/2MHz speed. More detailed information on this is listed under each instruction's information, but the high-level view is:

- When the processor is running at 1MHz, all instructions take at least two cycles, and dummy cycles are re-inserted into Read-Modify-Write instructions, so that all instructions take exactly the same number of cycles as on a 6502.
- The Read-Modify-Write instructions and all instructions that read a value from memory all require an extra cycle when operating at full speed, to allow signals to propagate within the processor.
- The Read-Modify-Write instructions require an additional cycle if the operand is \$D019, as the dummy write is performed in this case. This is to improve compatibility with C64 software that frequently uses this “bug” of the 6502 to more rapidly acknowledge VIC-II interrupts.
- Page-crossing and branch-taking penalties do not apply when the processor is running at full speed.
- Many instructions require fewer cycles when the processor is running at full speed, as generally most non-bus cycles are removed. For example, Pushing and Pulling values to and from the stack requires only 2 cycles, instead of the 4 that the 6502 requires for these instructions.

Note that it is possible that further changes to processor timing will occur.

Similar issues apply to when the processor is in 6502 mode.

|      | \$x0           | \$x1            | \$x2            | \$x3            | \$x4            | \$x5            | \$x6            | \$x7            | \$x8           | \$x9            | \$xA           | \$xB           | \$xC            | \$xD            | \$xE            | \$xF            |
|------|----------------|-----------------|-----------------|-----------------|-----------------|-----------------|-----------------|-----------------|----------------|-----------------|----------------|----------------|-----------------|-----------------|-----------------|-----------------|
| \$0x | 7              | 6 <sup>pr</sup> | 1 <sup>s</sup>  | 1 <sup>s</sup>  | 3 <sup>r</sup>  | 3 <sup>r</sup>  | 4 <sup>r</sup>  | 4 <sup>br</sup> | 2              | 2               | 1 <sup>s</sup> | 1 <sup>s</sup> | 5 <sup>r</sup>  | 4 <sup>r</sup>  | 5 <sup>r</sup>  |                 |
| \$1x | 2 <sup>b</sup> | 5 <sup>pr</sup> | 5 <sup>pr</sup> | 3 <sup>b</sup>  | 5 <sup>r</sup>  | 3 <sup>r</sup>  | 4 <sup>r</sup>  | 4 <sup>br</sup> | 1 <sup>s</sup> | 1 <sup>s</sup>  | 1 <sup>s</sup> | 1 <sup>s</sup> | 4 <sup>r</sup>  | 4 <sup>pr</sup> | 5 <sup>pr</sup> | 5 <sup>b</sup>  |
| \$2x | 5              | 5 <sup>pr</sup> | 5 <sup>r</sup>  | 5 <sup>pr</sup> | 3 <sup>r</sup>  | 3 <sup>r</sup>  | 4 <sup>r</sup>  | 4 <sup>r</sup>  | 4 <sup>m</sup> | 2               | 1 <sup>s</sup> | 1 <sup>s</sup> | 4 <sup>r</sup>  | 4 <sup>r</sup>  | 5 <sup>r</sup>  | 5 <sup>b</sup>  |
| \$3x | 2 <sup>r</sup> | 5 <sup>pr</sup> | 5 <sup>pr</sup> | 3 <sup>b</sup>  | 3 <sup>pr</sup> | 4 <sup>pr</sup> | 5 <sup>pr</sup> | 4 <sup>r</sup>  | 1 <sup>s</sup> | 4 <sup>r</sup>  | 1 <sup>s</sup> | 1 <sup>s</sup> | 4 <sup>pr</sup> | 4 <sup>pr</sup> | 5 <sup>pr</sup> | 4 <sup>b</sup>  |
| \$4x | 6 <sup>m</sup> | 5 <sup>r</sup>  | 1 <sup>s</sup>  | 1 <sup>s</sup>  | 4 <sup>r</sup>  | 3 <sup>r</sup>  | 4 <sup>r</sup>  | 4 <sup>r</sup>  | 2              | 2               | 1 <sup>s</sup> | 1 <sup>s</sup> | 3               | 4 <sup>r</sup>  | 5 <sup>r</sup>  | 4 <sup>br</sup> |
| \$5x | 2 <sup>b</sup> | 5 <sup>pr</sup> | 5 <sup>pr</sup> | 3 <sup>b</sup>  | 5 <sup>pr</sup> | 3 <sup>p</sup>  | 3 <sup>pr</sup> | 4 <sup>r</sup>  | 1 <sup>s</sup> | 4 <sup>pr</sup> | 2              | 1 <sup>s</sup> | 1 <sup>s</sup>  | 4 <sup>pr</sup> | 5 <sup>pr</sup> | 4 <sup>br</sup> |
| \$6x | 6 <sup>m</sup> | 5 <sup>r</sup>  | 4               | 3 <sup>b</sup>  | 3               | 3 <sup>r</sup>  | 5 <sup>r</sup>  | 5 <sup>r</sup>  | 4 <sup>m</sup> | 2               | 1 <sup>s</sup> | 1 <sup>s</sup> | 5 <sup>r</sup>  | 4 <sup>r</sup>  | 6 <sup>r</sup>  | 4 <sup>br</sup> |
| \$7x | 2 <sup>b</sup> | 5 <sup>pr</sup> | 5 <sup>pr</sup> | 3 <sup>b</sup>  | 3               | 3 <sup>r</sup>  |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$8x |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$9x |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Ax |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Bx |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Cx |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Dx |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Ex |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |
| \$Fx |                |                 |                 |                 |                 |                 |                 |                 |                |                 |                |                |                 |                 |                 |                 |

*b* Add one cycle if branch crosses a page boundary.

*m* Subtract non-bus cycles when at 40MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# Addressing Mode Table

| \$x0 | \$x1     | \$x2     | \$x3        | \$x4       | \$x5   | \$x6   | \$x7   | \$x8    | \$x9 | \$xA  | \$xB    | \$xC    | \$xD    | \$xE    | \$xF      |
|------|----------|----------|-------------|------------|--------|--------|--------|---------|------|-------|---------|---------|---------|---------|-----------|
| \$0x | (\$nn,X) |          |             | \$nn       | \$nn   | \$nn   | \$nn   | #\$nn   | A    | \$nnn | \$nnn   | \$nnn   | \$nnn   | \$nnn   | \$nn,\$rr |
| \$1x | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn   | \$nn,X | \$nn   | \$nnn,Y | A    | \$nnn | \$nnn,X | \$nnn,X | \$nnn,X | \$nnn,X | \$nn,\$rr |
| \$2x | \$nnn    | (\$nn,X) | (\$nnnn)    | (\$nnnn,X) | \$nn   | \$nn   | \$nn   | \$nnn   | A    |       | \$nnn   | \$nnn   | \$nnn   | \$nnn   | \$nn,\$rr |
| \$3x | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn,X | \$nn,X | \$nn   | \$nnn,Y | A    | \$nnn | \$nnn,X | \$nnn,X | \$nnn,X | \$nnn,X | \$nn,\$rr |
| \$4x |          |          |             |            | \$nn   | \$nn   | \$nn   | #\$nn   | A    | \$nnn | \$nnn   | \$nnn   | \$nnn   | \$nnn   | \$nn,\$rr |
| \$5x | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn,X | \$nn,X | \$nn   | \$nnn,Y | A    |       | \$nnn   | \$nnn,X | \$nnn,X | \$nnn,X | \$nn,\$rr |
| \$6x |          |          |             |            | \$nn   | \$nn   | \$nn   |         |      |       |         |         |         |         | \$nn,\$rr |
| \$7x | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn,X | \$nn,X | \$nn   | #\$nn   | A    |       |         |         |         |         |           |
| \$8x | \$rr     | (\$nn,X) | (\$nn,SP)   | Y          | \$rr   | \$nn   | \$nn   | \$nnn,Y |      |       |         |         |         |         |           |
| \$9x | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn,X | \$nn,X | \$nn   | #\$nn   |      |       |         |         |         |         |           |
| \$Ax | #\$nn    | (\$nn,X) | #\$nn       |            | \$nn   | \$nn   | \$nn   | #\$nn   |      |       |         |         |         |         |           |
| \$Bx | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn,X | \$nn,X | \$nn,Y | \$nnn,Y |      |       |         |         |         |         |           |
| \$Cx | #\$nn    | (\$nn,X) | #\$nn       |            | \$nn   | \$nn   | \$nn   | #\$nn   |      |       |         |         |         |         |           |
| \$Dx | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | \$nn   | \$nn,X | \$nn   | \$nnn,Y |      |       |         |         |         |         |           |
| \$Ex | #\$nn    | (\$nn,X) | (\$nn,SP),Y | \$nn       | \$nn   | \$nn   | \$nn   | #\$nn   |      |       |         |         |         |         |           |
| \$Fx | \$rr     | (\$nn),Y | (\$nn),Z    | \$rr       | #\$nnn | \$nn,X | \$nn   | \$nnn,Y |      |       |         |         |         |         |           |

# ADC

This instruction adds the argument to the contents of the Accumulator Register and the Carry Flag. If the D flag is set, then the addition is performed using Binary Coded Decimal.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The C flag will be set if the unsigned result is >255, or >99 if the D flag is set.

| ADC : Add with carry |              | 4510 |       |       |        |   |   |   |
|----------------------|--------------|------|-------|-------|--------|---|---|---|
| A ← A+M+C            |              | N    | Z     | I     | C      | D | V | E |
| Addressing Mode      | Assembly     | Code | Bytes | Bytes | Cycles |   |   |   |
| (indirect),X         | ADC (\$nn,X) | 61   | 2     | 5     | r      |   |   |   |
| base-page            | ADC \$nn     | 65   | 2     | 3     | r      |   |   |   |
| immediate            | ADC #\$nn    | 69   | 2     | 2     |        |   |   |   |
| absolute             | ADC \$nnnn   | 6D   | 3     | 4     | r      |   |   |   |
| (indirect),Y         | ADC (\$nn),Y | 71   | 2     | 5     | pr     |   |   |   |
| (indirect),Z         | ADC (\$nn),Z | 72   | 2     | 5     | pr     |   |   |   |
| base-page,X          | ADC \$nn,X   | 75   | 2     | 3     | r      |   |   |   |
| absolute,Y           | ADC \$nnnn,Y | 79   | 3     | 4     | r      |   |   |   |
| absolute,X           | ADC \$nnnn,X | 7D   | 3     | 4     | r      |   |   |   |

p Add one cycle if indexing crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

# AND

This instruction performs a binary AND operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, and that are set in the argument will be set in the accumulator on completion.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.

- The Z flag will be set if the result is zero, else it will be cleared.

| <b>AND : Binary AND</b>         |                 | <b>4510</b> |              |               |   |   |           |   |
|---------------------------------|-----------------|-------------|--------------|---------------|---|---|-----------|---|
| $A \leftarrow A \text{ AND } M$ |                 | N           | Z            | I             | C | D | V         | E |
|                                 |                 | +           | +            | .             | . | . | .         | . |
| <b>Addressing Mode</b>          | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |   |   |           |   |
| (indirect),X                    | AND (\$nn,X)    | 21          | 2            | 5             |   |   | <i>pr</i> |   |
| base-page                       | AND \$nn        | 25          | 2            | 3             |   |   | <i>r</i>  |   |
| immediate                       | AND #\$nn       | 29          | 2            | 2             |   |   |           |   |
| absolute                        | AND \$nnnn      | 2D          | 3            | 4             |   |   | <i>r</i>  |   |
| (indirect),Y                    | AND (\$nn),Y    | 31          | 2            | 5             |   |   | <i>pr</i> |   |
| (indirect),Z                    | AND (\$nn),Z    | 32          | 2            | 5             |   |   | <i>pr</i> |   |
| base-page,X                     | AND \$nn,X      | 35          | 2            | 4             |   |   | <i>pr</i> |   |
| absolute,Y                      | AND \$nnnn,Y    | 39          | 3            | 4             |   |   | <i>r</i>  |   |
| absolute,X                      | AND \$nnnn,X    | 3D          | 3            | 4             |   |   | <i>pr</i> |   |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## ASL

This instruction shifts either the Accumulator or contents of the provided memory location one bit left. Bit 0 will be set to zero, and the bit 7 will be shifted out into the Carry Flag

### Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

**ASL : Arithmetic Shift Left****4510** $A \leftarrow A << 1$  or  $M \leftarrow M << 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
|------------------------|-----------------|-------------|--------------|---------------|-----------|
| base-page              | ASL \$nn        | 06          | 2            | 4             | <i>r</i>  |
| accumulator            | ASL A           | 0A          | 1            | 1             | <i>s</i>  |
| absolute               | ASL \$nnnn      | 0E          | 3            | 5             | <i>r</i>  |
| base-page,X            | ASL \$nn,X      | 16          | 2            | 4             | <i>r</i>  |
| absolute,X             | ASL \$nnnn,X    | 1E          | 3            | 5             | <i>pr</i> |

*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## ASR

This instruction shifts either the Accumulator or contents of the provided memory location one bit right. Bit 7 is considered to be a sign bit, and is preserved. The contents of bit 0 will be shifted out into the Carry Flag

### Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

**ASR : Arithmetic Shift Right****4510** $A \leftarrow A >> 1$  or  $M \leftarrow M >> 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
|------------------------|-----------------|-------------|--------------|---------------|-----------|
| accumulator            | ASR A           | 43          | 1            | 1             | <i>s</i>  |
| base-page              | ASR \$nn        | 44          | 2            | 4             | <i>r</i>  |
| base-page,X            | ASR \$nn,X      | 54          | 2            | 5             | <i>pr</i> |

*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# ASW

This instruction shifts a 16-bit value in memory left one bit.

For example, if location \$1234 contained \$87 and location \$1235 contained \$A9, ASW \$1234 would result in location \$1234 containing \$0E and location \$1235 containing \$53, and the Carry Flag being set.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 of the upper byte is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the upper byte was set, prior to being shifted.

| <b>ASW : Arithmetic Shift Word Left</b> |                 | <b>4510</b>          |
|-----------------------------------------|-----------------|----------------------|
| $M \leftarrow M \ll 1$                  |                 |                      |
|                                         |                 | <b>N Z I C D V E</b> |
|                                         |                 | + + . + . . .        |
| <b>Addressing Mode</b>                  | <b>Assembly</b> | <b>Code</b>          |
| absolute                                | ASW \$nnnn      | CB                   |
|                                         |                 | 3                    |

# BBRO

This instruction branches to the indicated address if bit 0 is clear in the indicated base-page memory location.

| <b>BBRO : Branch on Bit 0 Reset</b> |                 | <b>4510</b>          |
|-------------------------------------|-----------------|----------------------|
| $PC \leftarrow PC + R8$             |                 |                      |
|                                     |                 | <b>N Z I C D V E</b> |
|                                     |                 | . . . . . .          |
| <b>Addressing Mode</b>              | <b>Assembly</b> | <b>Code</b>          |
| base-page and relative              | BBRO \$nn,\$rr  | OF                   |
|                                     |                 | 3                    |

# BBR1

This instruction branches to the indicated address if bit 1 is clear in the indicated base-page memory location.

**BBR1 : Branch on Bit 1 Reset**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|                        |                |    |   |   |              |
|------------------------|----------------|----|---|---|--------------|
| base-page and relative | BBR1 \$nn,\$rr | 1F | 3 | 5 | <sup>b</sup> |
|------------------------|----------------|----|---|---|--------------|

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## BBR2

This instruction branches to the indicated address if bit 2 is clear in the indicated base-page memory location.

**BBR2 : Branch on Bit 2 Reset**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|                        |                |    |   |   |              |
|------------------------|----------------|----|---|---|--------------|
| base-page and relative | BBR2 \$nn,\$rr | 2F | 3 | 5 | <sup>b</sup> |
|------------------------|----------------|----|---|---|--------------|

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## BBR3

This instruction branches to the indicated address if bit 3 is clear in the indicated base-page memory location.

**BBR3 : Branch on Bit 3 Reset**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

|                        |                |    |   |   |              |
|------------------------|----------------|----|---|---|--------------|
| base-page and relative | BBR3 \$nn,\$rr | 3F | 3 | 4 | <sup>b</sup> |
|------------------------|----------------|----|---|---|--------------|

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## BBR4

This instruction branches to the indicated address if bit 4 is clear in the indicated base-page memory location.

| BBR4 : Branch on Bit 4 Reset |                | 4510 |       |                  |
|------------------------------|----------------|------|-------|------------------|
| PC ← PC + R8                 |                | N    | Z     | I C D V E        |
| Addressing Mode              | Assembly       | Code | Bytes | Cycles           |
| base-page and relative       | BBR4 \$nn,\$rr | 4F   | 3     | 4 <sup>b</sup> r |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

## BBR5

This instruction branches to the indicated address if bit 5 is clear in the indicated base-page memory location.

| BBR5 : Branch on Bit 5 Reset |                | 4510 |       |                  |
|------------------------------|----------------|------|-------|------------------|
| PC ← PC + R8                 |                | N    | Z     | I C D V E        |
| Addressing Mode              | Assembly       | Code | Bytes | Cycles           |
| base-page and relative       | BBR5 \$nn,\$rr | 5F   | 3     | 4 <sup>b</sup> r |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

## BBR6

This instruction branches to the indicated address if bit 6 is clear in the indicated base-page memory location.

**BBR6 : Branch on Bit 6 Reset**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b>  |
|------------------------|-----------------|-------------|--------------|----------------|
| base-page and relative | BBR6 \$nn,\$rr  | 6F          | 3            | 4<br><i>br</i> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.**BBR7**

This instruction branches to the indicated address if bit 7 is clear in the indicated base-page memory location.

**BBR7 : Branch on Bit 7 Reset**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| base-page and relative | BBR7 \$nn,\$rr  | 7F          | 3            |               |

**BBS0**

This instruction branches to the indicated address if bit 0 is set in the indicated base-page memory location.

**BBS0 : Branch on Bit 0 Set**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| base-page and relative | BBS0 \$nn,\$rr  | 8F          | 3            |               |

**BBS1**

This instruction branches to the indicated address if bit 1 is set in the indicated base-page memory location.

| <b>BBS1 : Branch on Bit 1 Set</b> | <b>4510</b>     |             |              |               |
|-----------------------------------|-----------------|-------------|--------------|---------------|
| PC ← PC + R8                      |                 |             |              |               |
| N Z I C D V E                     |                 |             |              |               |
| . . . . .                         |                 |             |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| base-page and relative            | BBS1 \$nn,\$rr  | 9F          | 3            |               |

## BBS2

This instruction branches to the indicated address if bit 2 is set in the indicated base-page memory location.

| <b>BBS2 : Branch on Bit 2 Set</b> | <b>4510</b>     |             |              |               |
|-----------------------------------|-----------------|-------------|--------------|---------------|
| PC ← PC + R8                      |                 |             |              |               |
| N Z I C D V E                     |                 |             |              |               |
| . . . . .                         |                 |             |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| base-page and relative            | BBS2 \$nn,\$rr  | AF          | 3            |               |

## BBS3

This instruction branches to the indicated address if bit 3 is set in the indicated base-page memory location.

| <b>BBS3 : Branch on Bit 3 Set</b> | <b>4510</b>     |             |              |               |
|-----------------------------------|-----------------|-------------|--------------|---------------|
| PC ← PC + R8                      |                 |             |              |               |
| N Z I C D V E                     |                 |             |              |               |
| . . . . .                         |                 |             |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| base-page and relative            | BBS3 \$nn,\$rr  | BF          | 3            |               |

## BBS4

This instruction branches to the indicated address if bit 4 is set in the indicated base-page memory location.

**BBS4 : Branch on Bit 4 Set**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page and relative

BBS4 \$nn,\$rr

CF

3

## BBS5

This instruction branches to the indicated address if bit 5 is set in the indicated base-page memory location.

**BBS5 : Branch on Bit 5 Set**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page and relative

BBS5 \$nn,\$rr

DF

3

## BBS6

This instruction branches to the indicated address if bit 6 is set in the indicated base-page memory location.

**BBS6 : Branch on Bit 6 Set**

4510

PC ← PC + R8

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page and relative

BBS6 \$nn,\$rr

EF

3

## BBS7

This instruction branches to the indicated address if bit 7 is set in the indicated base-page memory location.

**BBS7 : Branch on Bit 7 Set**

4510

 $PC \leftarrow PC + R8$ 

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page and relative

BBS7 \$nn,\$rr

FF

3

## BCC

This instruction branches to the indicated address if the Carry Flag is clear.

**BCC : Branch on Carry Flag Clear**

4510

 $PC \leftarrow PC + R8$  or  $PC \leftarrow PC + R16$ 

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

relative

BCC \$rr

90

2

16-bit relative

BCC \$rrrr

93

3

## BCS

This instruction branches to the indicated address if the Carry Flag is set.

**BCS : Branch on Carry Flag Set**

4510

 $PC \leftarrow PC + R8$  or  $PC \leftarrow PC + R16$ 

N Z I C D V E

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

relative

BCS \$rr

B0

2

16-bit relative

BCS \$rrrr

B3

3

## BEQ

This instruction branches to the indicated address if the Zero Flag is set.

**BEQ : Branch on Zero Flag Set****4510**PC  $\leftarrow$  PC + R8 or PC  $\leftarrow$  PC + R16**N Z I C D V E****.....**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| relative               | BEQ \$rr        | F0          | 2            |               |
| 16-bit relative        | BEQ \$rrrr      | F3          | 3            |               |

**BIT**

This instruction is used to test the bits stored in a memory location. Bits 6 and 7 of the memory location's contents are directly copied into the Overflow Flag and Negative Flag. The Zero Flag is set or cleared based on the result of performing the binary AND of the Accumulator Register and the contents of the indicated memory location.

**Side effects**

- The N flag will be set if the bit 7 of the memory location is set, else it will be cleared.
- The V flag will be set if the bit 6 of the memory location is set, else it will be cleared.
- The Z flag will be set if the result of A AND M is zero, else it will be cleared.

**BIT : Perform Bit Test****4510**N  $\leftarrow$  M(7), V  $\leftarrow$  M(6), Z  $\leftarrow$  A AND M**N Z I C D V E****.....****.....****.....**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
|------------------------|-----------------|-------------|--------------|---------------|-----------|
| base-page              | BIT \$nn        | 24          | 2            | 3             | <i>r</i>  |
| absolute               | BIT \$nnnn      | 2C          | 3            | 4             | <i>r</i>  |
| base-page,X            | BIT \$nn,X      | 34          | 2            | 3             | <i>pr</i> |
| absolute,X             | BIT \$nnnn,X    | 3C          | 3            | 4             | <i>pr</i> |
| immediate              | BIT #\$nn       | 89          | 2            |               |           |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

**BMI**

This instruction branches to the indicated address if the Negative Flag is set.

**BMI : Branch on Negative Flag Set****4510**PC  $\leftarrow$  PC + R8 or PC  $\leftarrow$  PC + R16**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|----------------------|--------------|---------------|------------|
| relative               | BMI \$rr             | 30           | 2             | 2 <i>r</i> |
| 16-bit relative        | BMI \$rrrr           | 33           | 3             | 3 <i>b</i> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## BNE

This instruction branches to the indicated address if the Zero Flag is clear.

**BNE : Branch on Zero Flag Clear****4510**PC  $\leftarrow$  PC + R8 or PC  $\leftarrow$  PC + R16**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |  |
|------------------------|----------------------|--------------|---------------|--|
| relative               | BNE \$rr             | D0           | 2             |  |
| 16-bit relative        | BNE \$rrrr           | D3           | 3             |  |

## BPL

This instruction branches to the indicated address if the Negative Flag is clear.

**BPL : Branch on Negative Flag Clear****4510**PC  $\leftarrow$  PC + R8 or PC  $\leftarrow$  PC + R16**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|----------------------|--------------|---------------|------------|
| relative               | BPL \$rr             | 10           | 2             | 2 <i>b</i> |
| 16-bit relative        | BPL \$rrrr           | 13           | 3             | 3 <i>b</i> |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# BRA

This instruction branches to the indicated address.

| BRA : Branch Unconditionally                        |               | 4510          |        |
|-----------------------------------------------------|---------------|---------------|--------|
| PC $\leftarrow$ PC + R8 or PC $\leftarrow$ PC + R16 |               | N Z I C D V E |        |
| Addressing Mode                                     | Assembly Code | Bytes         | Cycles |
| relative                                            | BRA \$rr      | 80            | 2      |
| 16-bit relative                                     | BRA \$rrrr    | 83            | 3      |

# BRK

The break command causes the microprocessor to go through an interrupt sequence under program control. The address of the BRK instruction + 2 is pushed to the stack along with the status register with the Break flag set. This allows the interrupt service routine to distinguish between IRQ events and BRK events. For example:

```
PLA          ; load status
PHA          ; restore stack
AND #$10    ; mask break flag
BNE DO_BREAK ; -> it was a BRK
...          ; else continue with IRQ server
```

Cite from: MCS6500 Microcomputer Family Programming Manual, January 1976, Second Edition, MOS Technology Inc., Page 144:

"The BRK is a single byte instruction and its addressing mode is Implied."

There are debates, that BRK could be seen as a two byte instruction with the addressing mode immediate, where the operand byte is discarded. The byte following the BRK could then be used as a call argument for the break handler. Commodore however used the BRK, as stated in the manual, as a single byte instruction, which breaks into the ML monitor, if present. These builtin monitors decremented the stacked PC, so that it could be used to return or jump directly to the code byte after the BRK.

| <b>BRK : Break to Interrupt</b> | <b>4510</b>     |             |              |               |
|---------------------------------|-----------------|-------------|--------------|---------------|
| PC $\leftarrow$ (\$FFFE)        |                 |             |              |               |
| <b>N Z I C D V E</b>            |                 |             |              |               |
| <b>Addressing Mode</b>          | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                         | BRK             | 00          | 1            | 7             |

## BSR

This instruction branches to the indicated address, saving the address of the caller on the stack, so that the routine can be returned from using an RTS instruction.

This instruction is helpful for using relocatable code, as it provides a relative-addressed alternative to JSR.

| <b>BSR : Branch Sub-Routine</b>                     | <b>4510</b>     |             |              |                |
|-----------------------------------------------------|-----------------|-------------|--------------|----------------|
| PC $\leftarrow$ PC + R8 or PC $\leftarrow$ PC + R16 |                 |             |              |                |
| <b>N Z I C D V E</b>                                |                 |             |              |                |
| <b>Addressing Mode</b>                              | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b>  |
| 16-bit relative                                     | BSR \$rrrr      | 63          | 3            | 3 <sup>b</sup> |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

## BVC

This instruction branches to the indicated address if the Overflow (V) Flag is clear.

| <b>BVC : Branch on Overflow Flag Clear</b>          | <b>4510</b>     |             |              |                |
|-----------------------------------------------------|-----------------|-------------|--------------|----------------|
| PC $\leftarrow$ PC + R8 or PC $\leftarrow$ PC + R16 |                 |             |              |                |
| <b>N Z I C D V E</b>                                |                 |             |              |                |
| <b>Addressing Mode</b>                              | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b>  |
| relative                                            | BVC \$rr        | 50          | 2            | 2 <sup>b</sup> |
| 16-bit relative                                     | BVC \$rrrr      | 53          | 3            | 3 <sup>b</sup> |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# BVS

This instruction branches to the indicated address if the Overflow (V) Flag is set.

| <b>BVS : Branch on Overflow Flag Set</b>            |            | <b>4510</b> |   |   |   |   |              |   |
|-----------------------------------------------------|------------|-------------|---|---|---|---|--------------|---|
| PC $\leftarrow$ PC + R8 or PC $\leftarrow$ PC + R16 |            | N           | Z | I | C | D | V            | E |
| Addressing Mode                                     |            |             |   |   |   |   |              |   |
| relative                                            | BVS \$rr   | 70          | 2 |   | 2 |   | <sup>b</sup> |   |
| 16-bit relative                                     | BVS \$rrrr | 73          | 3 |   | 3 |   | <sup>b</sup> |   |

*b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

# CLC

This instruction clears the Carry Flag.

## Side effects

- The C flag is cleared.

| <b>CLC : Clear Carry Flag</b> |     | <b>4510</b> |   |   |   |   |              |   |
|-------------------------------|-----|-------------|---|---|---|---|--------------|---|
| C $\leftarrow$ 0              |     | N           | Z | I | C | D | V            | E |
| Addressing Mode               |     |             |   |   |   |   |              |   |
| implied                       | CLC | 18          | 1 |   | 1 |   | <sup>s</sup> |   |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# CLD

This instruction clears the Decimal Flag. Arithmetic operations will use normal binary arithmetic, instead of Binary-Coded Decimal (BCD).

## Side effects

- The D flag is cleared.

| <b>CLD : Clear Decimal Flag</b> | <b>4510</b>     |             |              |               |
|---------------------------------|-----------------|-------------|--------------|---------------|
| D ← 0                           |                 |             |              |               |
| N Z I C D V E                   |                 |             |              |               |
|                                 |                 |             |              |               |
| <b>Addressing Mode</b>          | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                         | CLD             | D8          | 1            |               |

## CLE

This instruction clears the Extended Stack Disable Flag. This causes the stack to be able to exceed 256 bytes in length, by allowing the processor to modify the value of the high byte of the stack address (SPH).

### Side effects

- The E flag is cleared.

| <b>CLE : Clear Extended Stack Disable Flag</b> | <b>4510</b>     |             |              |               |
|------------------------------------------------|-----------------|-------------|--------------|---------------|
| E ← 0                                          |                 |             |              |               |
| N Z I C D V E                                  |                 |             |              |               |
|                                                |                 |             |              |               |
| <b>Addressing Mode</b>                         | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                                        | CLE             | 02          | 1            | 1 <i>s</i>    |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## CLI

This instruction clears the Interrupt Disable Flag. Interrupts will now be able to occur.

### Side effects

- The I flag is cleared.

**CLI : Clear Interrupt Disable Flag****4510** $I \leftarrow 0$ **N Z I C D V E****Addressing Mode****Assembly****Code****Bytes****Cycles**

implied

CLI

58

1

1

*s**s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# CLV

This instruction clears the Overflow Flag.

**Side effects**

- The V flag is cleared.

**CLV : Clear Overflow Flag****4510** $V \leftarrow 0$ **N Z I C D V E****Addressing Mode****Assembly****Code****Bytes****Cycles**

implied

CLV

B8

1

# CMP

This instruction performs  $A - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

**Side effects**

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

**CMP : Compare Accumulator****4510**

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect),X)          | CMP (\$nn,X)    | C1          | 2            |               |
| base-page              | CMP \$nn        | C5          | 2            |               |
| immediate              | CMP #\$nn       | C9          | 2            |               |
| absolute               | CMP \$nnnn      | CD          | 3            |               |
| (indirect),Y           | CMP (\$nn),Y    | D1          | 2            |               |
| (indirect),Z           | CMP (\$nn),Z    | D2          | 2            |               |
| base-page,X            | CMP \$nn,X      | D5          | 2            |               |
| absolute,Y             | CMP \$nnnn,Y    | D9          | 3            |               |
| absolute,X             | CMP \$nnnn,X    | DD          | 3            |               |

**CPX**

This instruction performs  $X - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

**Side effects**

- The N flag will be set if the result of  $X - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $X - M$  is zero or positive, i.e., if X is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $X - M$  is zero, else it will be cleared.

**CPX : Compare X Register****4510**

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | CPX #\$nn       | E0          | 2            |               |
| base-page              | CPX \$nn        | E4          | 2            |               |
| absolute               | CPX \$nnnn      | EC          | 3            |               |

# CPY

This instruction performs  $Y - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $Y - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $Y - M$  is zero or positive, i.e., if  $Y$  is not less than  $M$ , else it will be cleared.
- The Z flag will be set if the result of  $Y - M$  is zero, else it will be cleared.

| CPY : Compare Y Register |            | 4510 |       |       |        |   |   |   |
|--------------------------|------------|------|-------|-------|--------|---|---|---|
|                          |            | N    | Z     | I     | C      | D | V | E |
|                          |            | +    | +     | .     | +      | . | . | . |
| Addressing Mode          | Assembly   | Code | Bytes | Bytes | Cycles |   |   |   |
| immediate                | CPY #\$nn  | C0   | 2     |       |        |   |   |   |
| base-page                | CPY \$nn   | C4   | 2     |       |        |   |   |   |
| absolute                 | CPY \$nnnn | CC   | 3     |       |        |   |   |   |

# CPZ

This instruction performs  $Z - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $Z - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $Z - M$  is zero or positive, i.e., if  $Z$  is not less than  $M$ , else it will be cleared.
- The Z flag will be set if the result of  $Z - M$  is zero, else it will be cleared.

**CPZ : Compare Z Register****4510**

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | CPZ #\$nn       | C2          | 2            |               |
| base-page              | CPZ \$nn        | D4          | 2            |               |
| absolute               | CPZ \$nnnn      | DC          | 3            |               |

**DEC**

This instruction decrements the Accumulator Register or indicated memory location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**DEC : Decrement Memory or Accumulator****4510**

$A \leftarrow A - 1$  or  $M \leftarrow M - 1$

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| accumulator            | DEC A           | 3A          | 1            | 1             | <i>s</i> |
| base-page              | DEC \$nn        | C6          | 2            |               |          |
| absolute               | DEC \$nnnn      | CE          | 3            |               |          |
| base-page,X            | DEC \$nn,X      | D6          | 2            |               |          |
| absolute,X             | DEC \$nnnn,X    | DE          | 3            |               |          |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

**DEW**

This instruction decrements the indicated memory word in the Base Page. The low numbered address contains the least significant bits. For example, if memory location \$12 contains \$78 and memory location \$13 contains \$56, the instruction DEW \$12 would cause memory location to be set to \$77.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| DEW : Decrement Memory Word |          | 4510          |
|-----------------------------|----------|---------------|
| M16 ← M16 - 1               |          |               |
|                             |          | N Z I C D V E |
|                             |          | + + . . . .   |
| Addressing Mode             | Assembly | Code          |
| base-page                   | DEW \$nn | C3            |
|                             |          | Bytes Cycles  |
|                             |          | 2             |

## DEX

This instruction decrements the X Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| DEX : Decrement X Register |          | 4510          |
|----------------------------|----------|---------------|
| X ← X - 1                  |          |               |
|                            |          | N Z I C D V E |
|                            |          | + + . . . .   |
| Addressing Mode            | Assembly | Code          |
| implied                    | DEX      | CA            |
|                            |          | Bytes Cycles  |
|                            |          | 1             |

## DEY

This instruction decrements the Y Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>DEY : Decrement Y Register</b> | <b>4510</b>                       |             |              |               |
|-----------------------------------|-----------------------------------|-------------|--------------|---------------|
| $Y \leftarrow Y - 1$              |                                   |             |              |               |
|                                   | <b>N Z I C D V E</b><br>+ + . . . |             |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b>                   | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                           | DEY                               | 88          | 1            |               |

## DEZ

This instruction decrements the Z Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>DEZ : Decrement Z Register</b> | <b>4510</b>                       |             |              |               |
|-----------------------------------|-----------------------------------|-------------|--------------|---------------|
| $Z \leftarrow Z - 1$              |                                   |             |              |               |
|                                   | <b>N Z I C D V E</b><br>+ + . . . |             |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b>                   | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| implied                           | DEZ                               | 3B          | 1            | 1 <i>s</i>    |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## EOM

In contrast with the 6502, the NOP instruction on the 45GS02 performs two additional roles when in 4502 mode.

First, indicate the end of a memory mapping sequence caused by a MAP instruction, allowing interrupts to occur again.

Second, it instructs the processor that if the following instruction uses Base-Page Indirect Z Indexed addressing, that the processor should use a 32-bit pointer instead of a 16-bit 6502 style pointer. Such 32-bit addresses are unaffected by C64, C65 or MEGA65 memory banking. This allows fast and easy access to the entire address space of the MEGA65 without having to perform or be aware of any banking, or using the DMA controller. This addressing mode causes a two cycle penalty, caused by the time required to read the extra two bytes of the pointer.

## Side effects

- Removes the prohibition on all interrupts caused by the MAP instruction, allowing Non-Maskable Interrupts to again occur, and IRQ interrupts, if the Interrupt Disable Flag is not set.

| <b>EOM : End of Mapping Sequence / No-Operation</b> |                 |             |              |               | <b>4510</b> |
|-----------------------------------------------------|-----------------|-------------|--------------|---------------|-------------|
| <b>N Z I C D V E</b>                                |                 |             |              |               |             |
| <b>Addressing Mode</b>                              | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |             |
| implied                                             | EOM             | EA          | 1            |               |             |

## EOR

This instruction performs a binary XOR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, but not both.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>EOR : Binary Exclusive OR</b> |                 |             |              |               | <b>4510</b> |
|----------------------------------|-----------------|-------------|--------------|---------------|-------------|
| <b>N Z I C D V E</b>             |                 |             |              |               |             |
| + + . . . .                      |                 |             |              |               |             |
| <b>Addressing Mode</b>           | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |             |
| (indirect,X)                     | EOR (\$nn,X)    | 41          | 2            | 5             | <i>r</i>    |
| base-page                        | EOR \$nn        | 45          | 2            | 3             | <i>r</i>    |
| immediate                        | EOR #\$nn       | 49          | 2            | 2             |             |
| absolute                         | EOR \$nnnn      | 4D          | 3            | 4             | <i>r</i>    |
| (indirect),Y                     | EOR (\$nn),Y    | 51          | 2            | 5             | <i>pr</i>   |
| (indirect),Z                     | EOR (\$nn),Z    | 52          | 2            | 5             | <i>pr</i>   |
| base-page,X                      | EOR \$nn,X      | 55          | 2            | 3             | <i>p</i>    |
| absolute,Y                       | EOR \$nnnn,Y    | 59          | 3            | 4             | <i>pr</i>   |
| absolute,X                       | EOR \$nnnn,X    | 5D          | 3            | 4             | <i>pr</i>   |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# INC

This instruction increments the Accumulator Register or indicated memory location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| INC : Increment Memory or Accumulator        |              | 4510 |       |        |   |   |   |          |
|----------------------------------------------|--------------|------|-------|--------|---|---|---|----------|
| A $\leftarrow$ A + 1 or M $\leftarrow$ M + 1 |              | N    | Z     | I      | C | D | V | E        |
| Addressing Mode                              | Assembly     | Code | Bytes | Cycles |   |   |   |          |
| accumulator                                  | INC A        | 1A   | 1     | 1      |   |   |   | <i>s</i> |
| base-page                                    | INC \$nn     | E6   | 2     |        |   |   |   |          |
| absolute                                     | INC \$nnnn   | EE   | 3     |        |   |   |   |          |
| base-page,X                                  | INC \$nn,X   | F6   | 2     |        |   |   |   |          |
| absolute,X                                   | INC \$nnnn,X | FE   | 3     |        |   |   |   |          |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# INW

This instruction increments the indicated memory word in the Base Page. The low numbered address contains the least significant bits. For example, if memory location \$12 contains \$78 and memory location \$13 contains \$56, the instruction DEW \$12 would cause memory location to be set to \$79.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| INW : Increment Memory Word |          | 4510 |       |        |   |   |   |   |
|-----------------------------|----------|------|-------|--------|---|---|---|---|
| M16 $\leftarrow$ M16 + 1    |          | N    | Z     | I      | C | D | V | E |
| Addressing Mode             | Assembly | Code | Bytes | Cycles |   |   |   |   |
| base-page                   | INW \$nn | E3   | 2     |        |   |   |   |   |

# INX

This instruction increments the X Register, i.e., adds 1 to it.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| INX : Increment X Register |  | 4510          |      |       |        |  |
|----------------------------|--|---------------|------|-------|--------|--|
| $X \leftarrow X + 1$       |  | N Z I C D V E |      |       |        |  |
|                            |  | + + . . . .   |      |       |        |  |
| Addressing Mode            |  | Assembly      | Code | Bytes | Cycles |  |
| implied                    |  | INX           | E8   | 1     |        |  |

# INY

This instruction increments the Y Register, i.e., adds 1 to it.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| INY : Increment Y Register |  | 4510          |      |       |        |  |
|----------------------------|--|---------------|------|-------|--------|--|
| $Y \leftarrow Y + 1$       |  | N Z I C D V E |      |       |        |  |
|                            |  | + + . . . .   |      |       |        |  |
| Addressing Mode            |  | Assembly      | Code | Bytes | Cycles |  |
| implied                    |  | INY           | C8   | 1     |        |  |

# INZ

This instruction increments the Z Register, i.e., adds 1 to it.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.

- The Z flag will be set if the result is zero, else it will be cleared.

| <b>INZ : Increment Z Register</b> | <b>4510</b>     |               |              |               |
|-----------------------------------|-----------------|---------------|--------------|---------------|
| Z ← Y + 1                         |                 |               |              |               |
|                                   |                 | N Z I C D V E |              |               |
|                                   |                 | + + . . . .   |              |               |
| <b>Addressing Mode</b>            | <b>Assembly</b> | <b>Code</b>   | <b>Bytes</b> | <b>Cycles</b> |
| implied                           | INZ             | 1B            | 1            | 1 <i>s</i>    |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## JMP

This instruction sets the Programme Counter (PC) Register to the address indicated by the instruction, causing execution to continue from that address.

| <b>JMP : Jump to Address</b> | <b>4510</b>     |               |              |               |
|------------------------------|-----------------|---------------|--------------|---------------|
| PC ← M2:M1                   |                 |               |              |               |
|                              |                 | N Z I C D V E |              |               |
|                              |                 | . . . . .     |              |               |
| <b>Addressing Mode</b>       | <b>Assembly</b> | <b>Code</b>   | <b>Bytes</b> | <b>Cycles</b> |
| absolute                     | JMP \$nnnn      | 4C            | 3            | 3             |
| indirect                     | JMP (\$nnnn)    | 6C            | 3            | 5 <i>r</i>    |
| indirect,X                   | JMP (\$nnnn,X)  | 7C            | 3            |               |

*r* Add one cycle if clock speed is at 40 MHz.

## JSR

This instruction saves the address of the instruction following the JSR instruction onto the stack, and then sets the Programme Counter (PC) Register to the address indicated by the instruction, causing execution to continue from that address. Because the return address has been saved on the stack, the RTS instruction can be used to return from the called sub-routine and resume execution following the JSR instruction.

NOTE: This instruction actually pushes the address of the last byte of the JSR instruction onto the stack. The RTS instruction naturally is aware of this, and increments the address on popping it from the stack, before setting the Programme Counter (PC) register.

**JSR : Jump to Sub-Routine****4510**PC  $\leftarrow$  M2:M1, Stack  $\leftarrow$  PCH:PCL**N Z I C D V E**

. . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| absolute               | JSR \$nnnn      | 20          | 3            | 5             |
| indirect               | JSR (\$nnnn)    | 22          | 3            | 5 <i>r</i>    |
| indirect,X             | JSR (\$nnnn,X)  | 23          | 3            | 5 <i>pr</i>   |

*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.**LDA**

This instruction loads the Accumulator Register with the indicated value, or with the contents of the indicated location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDA : Load Accumulator****4510**A  $\leftarrow$  M**N Z I C D V E**

+ + . . . .

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| (indirect,X)           | LDA (\$nn,X)    | A1          | 2            |               |
| base-page              | LDA \$nn        | A5          | 2            |               |
| immediate              | LDA #\$nn       | A9          | 2            |               |
| absolute               | LDA \$nnnn      | AD          | 3            |               |
| (indirect),Y           | LDA (\$nn),Y    | B1          | 2            |               |
| (indirect),Z           | LDA (\$nn),Z    | B2          | 2            |               |
| base-page,X            | LDA \$nn,X      | B5          | 2            |               |
| absolute,Y             | LDA \$nnnn,Y    | B9          | 3            |               |
| absolute,X             | LDA \$nnnn,X    | BD          | 3            |               |
| (indirect,SP),Y        | LDA (\$nn,SP),Y | E2          | 2            |               |

# LDX

This instruction loads the X Register with the indicated value, or with the contents of the indicated location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| LDX : Load X Register |              | 4510 |       |        |   |   |   |   |
|-----------------------|--------------|------|-------|--------|---|---|---|---|
| X ← M                 |              | N    | Z     | I      | C | D | V | E |
|                       |              | +    | +     | .      | . | . | . | . |
| Addressing Mode       | Assembly     | Code | Bytes | Cycles |   |   |   |   |
| immediate             | LDX #\$nn    | A2   | 2     |        |   |   |   |   |
| base-page             | LDX \$nn     | A6   | 2     |        |   |   |   |   |
| absolute              | LDX \$nnnn   | AE   | 3     |        |   |   |   |   |
| base-page,Y           | LDX \$nn,Y   | B6   | 2     |        |   |   |   |   |
| absolute,Y            | LDX \$nnnn,Y | BE   | 3     |        |   |   |   |   |

# LDY

This instruction loads the Y Register with the indicated value, or with the contents of the indicated location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDY : Load Y Register****4510**

Y ← M

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | LDY #\$nn       | A0          | 2            |               |
| base-page              | LDY \$nn        | A4          | 2            |               |
| absolute               | LDY \$nnnn      | AC          | 3            |               |
| base-page,X            | LDY \$nn,X      | B4          | 2            |               |
| absolute,X             | LDY \$nnnn,X    | BC          | 3            |               |

## LDZ

This instruction loads the Z Register with the indicated value, or with the contents of the indicated location.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDZ : Load Z Register****4510**

Z ← M

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | LDZ #\$nn       | A3          | 2            |               |
| absolute               | LDZ \$nnnn      | AB          | 3            |               |
| absolute,X             | LDZ \$nnnn,X    | BB          | 3            |               |

## LSR

This instruction shifts either the Accumulator or contents of the provided memory location one bit right. Bit 7 will be set to zero, and the bit 0 will be shifted out into the Carry Flag

**Side effects**

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.

- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

| <b>LSR : Logical Shift Right</b> |                 | <b>4510</b> |              |               |           |
|----------------------------------|-----------------|-------------|--------------|---------------|-----------|
| <b>Addressing Mode</b>           | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
| base-page                        | LSR \$nn        | 46          | 2            | 4             | <i>r</i>  |
| accumulator                      | LSR A           | 4A          | 1            | 1             | <i>s</i>  |
| absolute                         | LSR \$nnnn      | 4E          | 3            | 5             | <i>r</i>  |
| base-page,X                      | LSR \$nn,X      | 56          | 2            | 3             | <i>pr</i> |
| absolute,X                       | LSR \$nnnn,X    | 5E          | 3            | 5             | <i>pr</i> |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## MAP

This instruction sets the C65 or MEGA65 style memory map, depending on the values in the Accumulator, X, Y and Z registers.

Care should be taken to ensure that after the execution of an MAP instruction that appropriate memory is mapped at the location of the following instruction. Failure to do so will result in unpredictable results.

Further information on this instruction is available in Appendix G.

### Side effects

- The memory map is immediately changed to that requested.
- All interrupts, including Non-Maskable Interrupts (NMI) are blocked from occurring until an EOM (NOP) instruction is encountered.

| <b>MAP : Set Memory Map</b> |                 | <b>4510</b> |              |               |          |
|-----------------------------|-----------------|-------------|--------------|---------------|----------|
| <b>Addressing Mode</b>      | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
| implied                     | MAP             | 5C          | 1            | 1             | <i>s</i> |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# NEG

This instruction replaces the contents of the Accumulator Register with the two's complement of the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| NEG : Negate Accumulator | 4510     |      |       |                    |
|--------------------------|----------|------|-------|--------------------|
| A $\leftarrow$ -A        |          |      |       |                    |
| Addressing Mode          | Assembly | Code | Bytes | Cycles             |
| accumulator              | NEG A    | 42   | 1     | 1 <small>s</small> |

s Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# ORA

This instruction performs a binary OR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, or both.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**ORA : Decrement Memory or Accumulator****4510** $A \leftarrow A + 1$  or  $M \leftarrow M + 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
|------------------------|-----------------|-------------|--------------|---------------|-----------|
| (indirect),X           | ORA (\$nn,X)    | 01          | 2            | 6             | <i>pr</i> |
| base-page              | ORA \$nn        | 05          | 2            | 3             | <i>r</i>  |
| immediate              | ORA #\$nn       | 09          | 2            | 2             |           |
| absolute               | ORA \$nnnn      | 0D          | 3            | 4             | <i>r</i>  |
| (indirect),Y           | ORA (\$nn),Y    | 11          | 2            | 5             | <i>pr</i> |
| (indirect),Z           | ORA (\$nn),Z    | 12          | 2            | 5             | <i>pr</i> |
| base-page,X            | ORA \$nn,X      | 15          | 2            | 3             | <i>r</i>  |
| absolute,Y             | ORA \$nnnn,Y    | 19          | 3            | 4             | <i>r</i>  |
| absolute,X             | ORA \$nnnn,X    | 1D          | 3            | 4             | <i>pr</i> |

p Add one cycle if indexing crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

## PHA

This instruction pushes the contents of the Accumulator Register onto the stack, and decrements the value of the Stack Pointer by 1.

**PHA : Push Accumulator Register onto the Stack****4510** $STACK \leftarrow A, SP \leftarrow SP - 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | PHA             | 48          | 1            | 2             |

## PHP

This instruction pushes the contents of the Processor Flags onto the stack, and decrements the value of the Stack Pointer by 1.

**PHP : Push Processor Flags onto the Stack****4510**STACK  $\leftarrow$  P, SP  $\leftarrow$  SP - 1**N Z I C D V E**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | PHP             | 08          | 1            | 2             |

## PHW

This instruction pushes either a 16 bit literal value or the memory word indicated onto the stack, and decrements the value of the Stack Pointer by 2.

**PHW : Push Word onto the Stack****4510**STACK  $\leftarrow$  M1:M2, SP  $\leftarrow$  SP - 2**N Z I C D V E****+ + . . . .**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| immediate              | PHW #\$nnnn     | F4          | 3            |               |
| absolute               | PHW \$nnnn      | FC          | 3            |               |

## PHX

This instruction pushes the contents of the X Register onto the stack, and decrements the value of the Stack Pointer by 1.

**PHX : Push X Register onto the Stack****4510**STACK  $\leftarrow$  X, SP  $\leftarrow$  SP - 1**N Z I C D V E****. . . . .**

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | PHX             | DA          | 1            |               |

## PHY

This instruction pushes the contents of the Y Register onto the stack, and decrements the value of the Stack Pointer by 1.

**PHY : Push Y Register onto the Stack****4510**STACK  $\leftarrow$  Y, SP  $\leftarrow$  SP - 1**N Z I C D V E****.....****Addressing Mode****Assembly Code****Bytes****Cycles**

implied

PHY

5A

1

2

## PHZ

This instruction pushes the contents of the Z Register onto the stack, and decrements the value of the Stack Pointer by 1.

**PHZ : Push Z Register onto the Stack****4510**STACK  $\leftarrow$  z, SP  $\leftarrow$  SP - 1**N Z I C D V E****.....****Addressing Mode****Assembly Code****Bytes****Cycles**

implied

PHZ

DB

1

## PLA

This instruction replaces the contents of the Accumulator Register with the top value from the stack, and increments the value of the Stack Pointer by 1.

**PLA : Pull Accumulator Register from the Stack****4510**A  $\leftarrow$  STACK, SP  $\leftarrow$  SP + 1**N Z I C D V E****.....****.....****Addressing Mode****Assembly Code****Bytes****Cycles**

implied

PLA

68

1

4

*m*

*m* Subtract non-bus cycles when at 40MHz.

## PLP

This instruction replaces the contents of the Processor Flags with the top value from the stack, and increments the value of the Stack Pointer by 1.

NOTE: This instruction does NOT replace the Extended Stack Disable Flag (E Flag), or the Software Interrupt Flag (B Flag)

|                                                       |             |
|-------------------------------------------------------|-------------|
| <b>PLP : Pull Accumulator Register from the Stack</b> | <b>4510</b> |
| A $\leftarrow$ STACK, SP $\leftarrow$ SP + 1          |             |

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | + | + | + | + | . |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| implied                | PLP                  | 28           | 1 4 <i>m</i>  |

*m* Subtract non-bus cycles when at 40MHz.

## PLX

This instruction replaces the contents of the X Register with the top value from the stack, and increments the value of the Stack Pointer by 1.

|                                              |             |
|----------------------------------------------|-------------|
| <b>PLX : Pull X Register from the Stack</b>  | <b>4510</b> |
| X $\leftarrow$ STACK, SP $\leftarrow$ SP + 1 |             |

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| implied                | PLX                  | FA           | 1             |

## PLY

This instruction replaces the contents of the Y Register with the top value from the stack, and increments the value of the Stack Pointer by 1.

|                                              |             |
|----------------------------------------------|-------------|
| <b>PLY : Pull Y Register from the Stack</b>  | <b>4510</b> |
| Y $\leftarrow$ STACK, SP $\leftarrow$ SP + 1 |             |

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| implied                | PLY                  | 7A           | 1             |

## PLZ

This instruction replaces the contents of the Z Register with the top value from the stack, and increments the value of the Stack Pointer by 1.

| PLZ : Pull Z Register from the Stack |               | 4510          |
|--------------------------------------|---------------|---------------|
| Z ← STACK, SP ← SP + 1               |               |               |
|                                      |               | N Z I C D V E |
|                                      |               | + + . . . .   |
| Addressing Mode                      | Assembly Code | Bytes Cycles  |
| implied                              | PLZ           | FB 1          |

## RMBO

This instruction clears bit zero of the indicated address. No flags are modified, regardless of the result.

| RMBO : Reset Bit 0 in Base Page |               | 4510             |
|---------------------------------|---------------|------------------|
| M(0) ← 0                        |               |                  |
|                                 |               | N Z I C D V E    |
|                                 |               | . . . . .        |
| Addressing Mode                 | Assembly Code | Bytes Cycles     |
| base-page                       | RMB0 \$nn     | 07 2 4 <i>br</i> |

b Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

## RMB1

This instruction clears bit 1 of the indicated address. No flags are modified, regardless of the result.

**RMB1 : Reset Bit 1 in Base Page****4510** $M(1) \leftarrow 0$ **N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code Bytes Cycles**

base-page

RMB1 \$nn

17

2

4

*br**b* Add one cycle if branch is taken.

Add one more cycle if branch taken crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## RMB2

This instruction clears bit 2 of the indicated address. No flags are modified, regardless of the result.

**RMB2 : Reset Bit 2 in Base Page****4510** $M(2) \leftarrow 0$ **N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code Bytes Cycles**

base-page

RMB2 \$nn

27

2

4

*r**r* Add one cycle if clock speed is at 40 MHz.

## RMB3

This instruction clears bit 3 of the indicated address. No flags are modified, regardless of the result.

**RMB3 : Reset Bit 3 in Base Page****4510** $M(3) \leftarrow 0$ **N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code Bytes Cycles**

base-page

RMB3 \$nn

37

2

4

*r**r* Add one cycle if clock speed is at 40 MHz.

## RMB4

This instruction clears bit 4 of the indicated address. No flags are modified, regardless of the result.

| RMB4 : Reset Bit 4 in Base Page |                      | 4510          |
|---------------------------------|----------------------|---------------|
| $M(4) \leftarrow 0$             |                      | N Z I C D V E |
|                                 |                      | .....         |
| <b>Addressing Mode</b>          | <b>Assembly Code</b> | <b>Bytes</b>  |
| base-page                       | RMB4 \$nn            | 47 2 4        |

r Add one cycle if clock speed is at 40 MHz.

## RMB5

This instruction clears bit 5 of the indicated address. No flags are modified, regardless of the result.

| RMB5 : Reset Bit 5 in Base Page |                      | 4510          |
|---------------------------------|----------------------|---------------|
| $M(5) \leftarrow 0$             |                      | N Z I C D V E |
|                                 |                      | .....         |
| <b>Addressing Mode</b>          | <b>Assembly Code</b> | <b>Bytes</b>  |
| base-page                       | RMB5 \$nn            | 57 2 4        |

r Add one cycle if clock speed is at 40 MHz.

## RMB6

This instruction clears bit 6 of the indicated address. No flags are modified, regardless of the result.

| RMB6 : Reset Bit 6 in Base Page |                      | 4510          |
|---------------------------------|----------------------|---------------|
| $M(6) \leftarrow 0$             |                      | N Z I C D V E |
|                                 |                      | .....         |
| <b>Addressing Mode</b>          | <b>Assembly Code</b> | <b>Bytes</b>  |
| base-page                       | RMB6 \$nn            | 67 2 5        |

r Add one cycle if clock speed is at 40 MHz.

# RMB7

This instruction clears bit 7 of the indicated address. No flags are modified, regardless of the result.

| RMB7 : Reset Bit 7 in Base Page |           | 4510              |
|---------------------------------|-----------|-------------------|
| $M(7) \leftarrow 0$             |           |                   |
|                                 |           | N Z I C D V E     |
|                                 |           | . . . . . . .     |
| Addressing Mode                 | Assembly  | Code Bytes Cycles |
| base-page                       | RMB7 \$nn | 77 2              |

# ROL

This instruction shifts either the Accumulator or contents of the provided memory location one bit left. Bit 0 will be set to the current value of the Carry Flag, and the bit 7 will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| ROL : Rotate Left Memory or Accumulator                     |              | 4510              |
|-------------------------------------------------------------|--------------|-------------------|
| $M \leftarrow M << 1, C \leftarrow M(7), M(0) \leftarrow C$ |              |                   |
|                                                             |              | N Z I C D V E     |
|                                                             |              | + + . + . . .     |
| Addressing Mode                                             | Assembly     | Code Bytes Cycles |
| base-page                                                   | ROL \$nn     | 26 2 4 <i>r</i>   |
| accumulator                                                 | ROL A        | 2A 1 1 <i>s</i>   |
| absolute                                                    | ROL \$nnnn   | 2E 3 5 <i>r</i>   |
| base-page,X                                                 | ROL \$nn,X   | 36 2 5 <i>pr</i>  |
| absolute,X                                                  | ROL \$nnnn,X | 3E 3 5 <i>pr</i>  |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# ROR

This instruction shifts either the Accumulator or contents of the provided memory location one bit right. Bit 7 will be set to the current value of the Carry Flag, and the bit 0 will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| ROR : Rotate Right Memory or Accumulator                    |              |      |       |        |   | 4510          |
|-------------------------------------------------------------|--------------|------|-------|--------|---|---------------|
| $M \leftarrow M >> 1, C \leftarrow M(0), M(7) \leftarrow C$ |              |      |       |        |   |               |
|                                                             |              |      |       |        |   | N Z I C D V E |
|                                                             |              |      |       |        |   | + + . + . . . |
| Addressing Mode                                             | Assembly     | Code | Bytes | Cycles |   |               |
| base-page                                                   | ROR \$nn     | 66   | 2     | 5      | r |               |
| accumulator                                                 | ROR A        | 6A   | 1     | 1      | s |               |
| absolute                                                    | ROR \$nnnn   | 6E   | 3     | 6      | r |               |
| base-page,X                                                 | ROR \$nn,X   | 76   | 2     |        |   |               |
| absolute,X                                                  | ROR \$nnnn,X | 7E   | 3     |        |   |               |

r Add one cycle if clock speed is at 40 MHz.

s Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# ROW

This instruction rotates the contents of the indicated memory word one bit left. Bit 0 of the low byte will be set to the current value of the Carry Flag, and the bit 7 of the high byte will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the upper byte was set, prior to being shifted.

**ROW : Rotate Word Left****4510** $M2:M1 \leftarrow M2:M1 \ll 1, C \leftarrow M2(7), M1(0) \leftarrow C$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

**Addressing Mode****Assembly****Code****Bytes****Cycles**

absolute

ROW \$nnnn

EB

3

# RTI

This instruction pops the processor flags from the stack, and then pops the Programme Counter (PC) register from the stack, allowing an interrupted programme to resume.

- The 6502 Processor Flags are restored from the stack.
- Neither the B (Software Interrupt) nor E (Extended Stack) flags are set by this instruction.

**RTI : Return From Interrupt****4510** $P \leftarrow \text{STACK}, PC \leftarrow \text{STACK}, SP \leftarrow SP + 3$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | .        | +        | +        | +        | +        | .        |

**Addressing Mode****Assembly****Code****Bytes****Cycles**

implied

RTI

40

1

6 *m*

*m* Subtract non-bus cycles when at 40MHz.

# RTS

This instruction adds optional argument to the Stack Pointer (SP) Register, and then pops the Programme Counter (PC) register from the stack, allowing a routine to return to its caller.

**RTS : Return From Subroutine****4510** $PC \leftarrow \text{STACK} + N, SP \leftarrow SP + 2 + N$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

**Addressing Mode****Assembly****Code****Bytes****Cycles**

implied

RTS

60

1

6 *m*

immediate

RTS #\$nn

62

2

4

*m* Subtract non-bus cycles when at 40MHz.

# SBC

This instruction performs  $A - M - 1 + C$ , and sets the processor flags accordingly. The result is stored in the Accumulator Register.

NOTE: This instruction is affected by the status of the Decimal Flag.

## Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

| SBC : Subtract With Carry |              | 4510 |       |        |   |   |   |   |
|---------------------------|--------------|------|-------|--------|---|---|---|---|
| $A \leftarrow -M - 1 + C$ |              | N    | Z     | I      | C | D | V | E |
| Addressing Mode           | Assembly     | Code | Bytes | Cycles |   |   |   |   |
| (indirect,X)              | SBC (\$nn,X) | E1   | 2     |        |   |   |   |   |
| base-page                 | SBC \$nn     | E5   | 2     |        |   |   |   |   |
| immediate                 | SBC #\$nn    | E9   | 2     |        |   |   |   |   |
| absolute                  | SBC \$nnnn   | ED   | 3     |        |   |   |   |   |
| (indirect),Y              | SBC (\$nn),Y | F1   | 2     |        |   |   |   |   |
| (indirect),Z              | SBC (\$nn),Z | F2   | 2     |        |   |   |   |   |
| base-page,X               | SBC \$nn,X   | F5   | 2     |        |   |   |   |   |
| absolute,Y                | SBC \$nnnn,Y | F9   | 3     |        |   |   |   |   |
| absolute,X                | SBC \$nnnn,X | FD   | 3     |        |   |   |   |   |

# SEC

This instruction sets the Carry Flag.

## Side effects

- The C flag is set.

**SEC : Set Carry Flag****4510**

C ← 1

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly Code****Bytes****Cycles**

implied

SEC

38

1

1

s

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## SED

This instruction sets the Decimal Flag. Binary arithmetic will now use Binary-Coded Decimal (BCD) mode.

NOTE: The C64's interrupt handler does not clear the Decimal Flag, which makes it dangerous to set the Decimal Flag without first setting the Interrupt Disable Flag.

**Side effects**

- The D flag is set.

**SED : Set Decimal Flag****4510**

D ← 1

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly Code****Bytes****Cycles**

implied

SED

F8

1

## SEE

This instruction sets the Extended Stack Disable Flag. This causes the stack to operate as on the 6502, i.e., limited to a single page of memory. The page of memory in which the stack is located can still be modified by setting the Stack Pointer High (SPH) Register.

**Side effects**

- The E flag is set.

|                                              |                      |
|----------------------------------------------|----------------------|
| <b>SEE : Set Extended Stack Disable Flag</b> | <b>4510</b>          |
| $E \leftarrow 1$                             |                      |
|                                              | <b>N Z I C D V E</b> |
|                                              | . . . . .            |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| implied                | SEE                  | 03           | 1             |

s Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## SEI

This instruction sets the Interrupt Disable Flag. Normal (IRQ) interrupts will no longer be able to occur. Non-Maskable Interrupts (NMI) will continue to occur, as their name suggests.

### Side effects

- The I flag is set.

|                                         |                      |
|-----------------------------------------|----------------------|
| <b>SEI : Set Interrupt Disable Flag</b> | <b>4510</b>          |
| $I \leftarrow 1$                        |                      |
|                                         | <b>N Z I C D V E</b> |
|                                         | . . . . .            |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| implied                | SEI                  | 78           | 1             |

## SMBO

This instruction sets bit zero of the indicated address. No flags are modified, regardless of the result.

|                                      |                      |
|--------------------------------------|----------------------|
| <b>SMBO : Set Bit 0 in Base Page</b> | <b>4510</b>          |
| $M(0) \leftarrow 1$                  |                      |
|                                      | <b>N Z I C D V E</b> |
|                                      | . . . . .            |

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| base-page              | SMBO \$nn            | 87           | 2             |

## SMB1

This instruction sets bit 1 of the indicated address. No flags are modified, regardless of the result.

| SMB1 : Set Bit 1 in Base Page |                 | 4510          |
|-------------------------------|-----------------|---------------|
| $M(1) \leftarrow 1$           |                 | N Z I C D V E |
|                               |                 | .....         |
| <b>Addressing Mode</b>        | <b>Assembly</b> | <b>Code</b>   |
| base-page                     | SMB1 \$nn       | 97            |
|                               |                 | <b>Bytes</b>  |
|                               |                 | 2             |
|                               |                 | <b>Cycles</b> |

## SMB2

This instruction sets bit 2 of the indicated address. No flags are modified, regardless of the result.

| SMB2 : Set Bit 2 in Base Page |                 | 4510          |
|-------------------------------|-----------------|---------------|
| $M(2) \leftarrow 1$           |                 | N Z I C D V E |
|                               |                 | .....         |
| <b>Addressing Mode</b>        | <b>Assembly</b> | <b>Code</b>   |
| base-page                     | SMB2 \$nn       | A7            |
|                               |                 | <b>Bytes</b>  |
|                               |                 | 2             |
|                               |                 | <b>Cycles</b> |

## SMB3

This instruction sets bit 3 of the indicated address. No flags are modified, regardless of the result.

| SMB3 : Set Bit 3 in Base Page |                 | 4510          |
|-------------------------------|-----------------|---------------|
| $M(3) \leftarrow 1$           |                 | N Z I C D V E |
|                               |                 | .....         |
| <b>Addressing Mode</b>        | <b>Assembly</b> | <b>Code</b>   |
| base-page                     | SMB3 \$nn       | B7            |
|                               |                 | <b>Bytes</b>  |
|                               |                 | 2             |
|                               |                 | <b>Cycles</b> |

## SMB4

This instruction sets bit 4 of the indicated address. No flags are modified, regardless of the result.

| <b>SMB4 : Set Bit 4 in Base Page</b> |                      | <b>4510</b>         |
|--------------------------------------|----------------------|---------------------|
| M(4) ← 1                             |                      | N Z I C D V E       |
|                                      |                      | .....               |
| <b>Addressing Mode</b>               | <b>Assembly Code</b> | <b>Bytes Cycles</b> |
| base-page                            | SMB4 \$nn            | C7 2                |

## SMB5

This instruction sets bit 5 of the indicated address. No flags are modified, regardless of the result.

| <b>SMB5 : Set Bit 5 in Base Page</b> |                      | <b>4510</b>         |
|--------------------------------------|----------------------|---------------------|
| M(5) ← 1                             |                      | N Z I C D V E       |
|                                      |                      | .....               |
| <b>Addressing Mode</b>               | <b>Assembly Code</b> | <b>Bytes Cycles</b> |
| base-page                            | SMB5 \$nn            | D7 2                |

## SMB6

This instruction sets bit 6 of the indicated address. No flags are modified, regardless of the result.

| <b>SMB6 : Set Bit 6 in Base Page</b> |                      | <b>4510</b>         |
|--------------------------------------|----------------------|---------------------|
| M(6) ← 1                             |                      | N Z I C D V E       |
|                                      |                      | .....               |
| <b>Addressing Mode</b>               | <b>Assembly Code</b> | <b>Bytes Cycles</b> |
| base-page                            | SMB6 \$nn            | E7 2                |

## SMB7

This instruction sets bit 7 of the indicated address. No flags are modified, regardless of the result.

| SMB7 : Set Bit 7 in Base Page |                 | 4510          |
|-------------------------------|-----------------|---------------|
| $M(7) \leftarrow 1$           |                 | N Z I C D V E |
|                               |                 | . . . . .     |
| <b>Addressing Mode</b>        | <b>Assembly</b> | <b>Code</b>   |
| base-page                     | SMB7 \$nn       | F7            |
|                               |                 | 2             |

## STA

This instruction stores the contents of the Accumulator Register into the indicated location.

| STA : Store Accumulator |                 | 4510          |
|-------------------------|-----------------|---------------|
| $M \leftarrow A$        |                 | N Z I C D V E |
|                         |                 | . . . . .     |
| <b>Addressing Mode</b>  | <b>Assembly</b> | <b>Code</b>   |
| (indirect,X)            | STA (\$nn,X)    | 81            |
| (indirect,SP),Y         | STA (\$nn,SP),Y | 82            |
| base-page               | STA \$nn        | 85            |
| absolute                | STA \$nnnn      | 8D            |
| (indirect),Y            | STA (\$nn),Y    | 91            |
| (indirect),Z            | STA (\$nn),Z    | 92            |
| base-page,X             | STA \$nn,X      | 95            |
| absolute,Y              | STA \$nnnn,Y    | 99            |
| absolute,X              | STA \$nnnn,X    | 9D            |

## STX

This instruction stores the contents of the X Register into the indicated location.

**STX : Store X Register****4510**

M ← X

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page

STX \$nn

86

2

absolute

STX \$nnnn

8E

3

base-page,Y

STX \$nn,Y

96

2

absolute,Y

STX \$nnnn,Y

9B

3

## STY

This instruction stores the contents of the Y Register into the indicated location.

**STY : Store Y Register****4510**

M ← Y

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page

STY \$nn

84

2

absolute,X

STY \$nnnn,X

8B

3

absolute

STY \$nnnn

8C

3

base-page,X

STY \$nn,X

94

2

## STZ

This instruction stores the contents of the Z Register into the indicated location.

**STZ : Store Z Register****4510**

M ← Z

**N Z I C D V E**

. . . . .

**Addressing Mode****Assembly****Code****Bytes****Cycles**

base-page

STZ \$nn

64

2

3

base-page,X

STZ \$nn,X

74

2

3

absolute

STZ \$nnnn

9C

3

absolute,X

STZ \$nnnn,X

9E

3

## TAB

This instruction sets the Base Page register to the contents of the Accumulator Register. This allows the relocation of the 6502's Zero-Page into any page of memory.

| TAB : Transfer Accumulator into Base Page Register |               | 4510                |
|----------------------------------------------------|---------------|---------------------|
| B ← A                                              |               | N Z I C D V E       |
|                                                    |               | .....               |
| Addressing Mode                                    | Assembly Code | Bytes Cycles        |
| implied                                            | TAB           | 5B 1 1 <sup>s</sup> |

<sup>s</sup> Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## TAX

This instruction loads the X Register with the contents of the Accumulator Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| TAX : Transfer Accumulator Register into the X Register |               | 4510          |
|---------------------------------------------------------|---------------|---------------|
| X ← A                                                   |               | N Z I C D V E |
|                                                         |               | ++ .. . . .   |
| Addressing Mode                                         | Assembly Code | Bytes Cycles  |
| implied                                                 | TAX           | AA 1          |

## TAY

This instruction loads the Y Register with the contents of the Accumulator Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TAY : Transfer Accumulator Register into the Y Register 4510**

Y ← A

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | TAY             | A8          | 1            |               |

**TAZ**

This instruction loads the Z Register with the contents of the Accumulator Register.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TAZ : Transfer Accumulator Register into the Z Register 4510**

Z ← A

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | TAZ             | 4B          | 1            | 1 <i>s</i>    |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

**TBA**

This instruction loads the Accumulator Register with the contents of the Base Page Register.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TBA : Transfer Base Page Register into the Accumulator 4510**A  $\leftarrow$  B

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|-----------------|-------------|--------------|---------------|
| implied                | TBA             | 7B          | 1            |               |

## TRB

This instruction sets performs a binary AND of the negation of the Accumulator Register and the indicated memory location, storing the result there. That is, any bits set in the Accumulator Register will be reset in the indicated memory location.

It also performs a test for any bits in common between the accumulator and indicated memory location. This can be used to construct simple shared-memory multi-processor systems, by providing an atomic means of setting a semaphore or acquiring a lock.

### Side effects

- The Z flag will be set if the binary AND of the Accumulator Register and contents of the indicated memory location prior are zero, prior to the execution of the instruction.

**TRB : Test and Reset Bit****4510**M  $\leftarrow$  M AND (NOT A)

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| . | + | . | . | . | . | . |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| base-page              | TRB \$nn        | 14          | 2            | 5             | <i>r</i> |
| absolute               | TRB \$nnnn      | 1C          | 3            | 4             | <i>r</i> |

*r* Add one cycle if clock speed is at 40 MHz.

## TSB

This instruction sets performs a binary OR of the Accumulator Register and the indicated memory location, storing the result there. That is, any bits set in the Accumulator Register will be set in the indicated memory location.

It also performs a test for any bits in common between the accumulator and indicated memory location. This can be used to construct simple shared-memory multi-processor systems, by providing an atomic means of setting a semaphore or acquiring a lock.

### Side effects

- The Z flag will be set if the binary AND of the Accumulator Register and contents of the indicated memory location prior are zero, prior to the execution of the instruction.

| TSB : Test and Set Bit         | 4510                           |
|--------------------------------|--------------------------------|
| $M \leftarrow M \text{ OR } A$ |                                |
| N Z I C D V E                  |                                |
| · + · · · ·                    |                                |
| Addressing Mode                | Assembly Code Bytes Cycles     |
| base-page                      | TSB \$nn 04 2 3 <sup>r</sup>   |
| absolute                       | TSB \$nnnn 0C 3 5 <sup>r</sup> |

<sup>r</sup> Add one cycle if clock speed is at 40 MHz.

## TSX

This instruction loads the X Register with the contents of the Stack Pointer High (SPL) Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| TSX : Transfer Stack Pointer High Register into the X Register | 4510                       |
|----------------------------------------------------------------|----------------------------|
| $X \leftarrow \text{SPH}$                                      |                            |
| N Z I C D V E                                                  |                            |
| + + · · · ·                                                    |                            |
| Addressing Mode                                                | Assembly Code Bytes Cycles |
| implied                                                        | TSX BA 1                   |

## TSY

This instruction loads the Y Register with the contents of the Stack Pointer High (SPH) Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TSY : Transfer Stack Pointer High Register into the Y Register 4510**

Y ← SPH

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |          |
|------------------------|-----------------|-------------|--------------|---------------|----------|
| implied                | TSY             | 0B          | 1            | 1             | <i>s</i> |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## TXA

This instruction loads the Accumulator Register with the contents of the X Register.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TXA : Transfer X Register into the Accumulator Register 4510**

A ← X

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |  |
|------------------------|-----------------|-------------|--------------|---------------|--|
| implied                | TXA             | 8A          | 1            |               |  |

## TXS

This instruction sets the low byte of the Stack Pointer (SPL) register to the contents of the X Register.

**TXS : Transfer X Register into Stack Pointer Low Register 4510**

SPL ← X

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |  |
|------------------------|-----------------|-------------|--------------|---------------|--|
| implied                | TXS             | 9A          | 1            |               |  |

# TYA

This instruction loads the Accumulator Register with the contents of the Y Register.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>TYA : Transfer Y Register into the Accumulator Register</b> |                      | <b>4510</b>   |
|----------------------------------------------------------------|----------------------|---------------|
| A ← Y                                                          |                      |               |
|                                                                |                      | N Z I C D V E |
|                                                                |                      | + + . . . .   |
| <b>Addressing Mode</b>                                         | <b>Assembly Code</b> | <b>Bytes</b>  |
| implied                                                        | TYA                  | 98            |
|                                                                |                      | 1             |
| <b>Cycles</b>                                                  |                      |               |

# TYS

This instruction sets the high byte of the Stack Pointer (SPH) register to the contents of the Y Register. This allows changing the memory page where the stack is located (if the Extended Stack Disable Flag (E) is set), or else allows setting the current Stack Pointer to any page in memory, if the Extended Stack Disable Flag (E) is clear.

| <b>TYS : Transfer Y Register into Stack Pointer High Register</b> |                      | <b>4510</b>   |
|-------------------------------------------------------------------|----------------------|---------------|
| SPH ← Y                                                           |                      |               |
|                                                                   |                      | N Z I C D V E |
|                                                                   |                      | . . . .       |
| <b>Addressing Mode</b>                                            | <b>Assembly Code</b> | <b>Bytes</b>  |
| implied                                                           | TYS                  | 2B            |
|                                                                   |                      | 1             |
| <b>Cycles</b>                                                     |                      | s             |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

# TZA

This instruction loads the Accumulator Register with the contents of the Z Register.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**TZA : Transfer Z Register into the Accumulator Register 4510**A  $\leftarrow$  Z

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| Addressing Mode | Assembly | Code | Bytes | Cycles |          |
|-----------------|----------|------|-------|--------|----------|
| implied         | TZA      | 6B   | 1     | 1      | <i>s</i> |

*s* Instruction requires 2 cycles when CPU is run at 1MHz or 2MHz.

## 45GS02 COMPOUND INSTRUCTIONS

As the 4510 has no unallocated opcodes, the 45GS02 uses compound instructions to implement its extension. These compound instructions consist of one or more single-byte instructions placed immediately before a conventional instruction. These prefixes instruct the 45GS02 to treat the following instruction differently, as described in Chapter/Appendix G on page G-3.

## ADC

This instruction adds the argument to the contents of the Accumulator Register and the Carry Flag. If the D flag is set, then the addition is performed using Binary Coded Decimal.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The C flag will be set if the unsigned result is  $>255$ , or  $>99$  if the D flag is set.

**ADC : Add with carry****45GS02**A  $\leftarrow$  A+M+C

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | + | . | + | . |

| Addressing Mode | Assembly     | Code  | Bytes | Cycles |                       |
|-----------------|--------------|-------|-------|--------|-----------------------|
| [indirect],Z    | ADC [\$nn],Z | EA 72 | 3     | 7      | <i>i<sub>pr</sub></i> |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# ADCQ

This instruction adds the argument to the contents of the 32-bit Q pseudo-register Register and the Carry Flag. If the D flag is set, then the operation is undefined, and is subject to change.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The C flag will be set if the unsigned result is >255 if the D flag is clear.

| ADCQ : Add with carry    |                 | 45GS02      |              |               |            |   |   |   |
|--------------------------|-----------------|-------------|--------------|---------------|------------|---|---|---|
| $Q \leftarrow Q + M + C$ |                 | N           | Z            | I             | C          | D | V | E |
|                          |                 | +           | +            | .             | +          | . | + | . |
| <b>Addressing Mode</b>   | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |   |   |   |
| base-page quad           | ADCQ \$nn       | 42 42 65    | 4            | 8             | <i>r</i>   |   |   |   |
| absolute quad            | ADCQ \$nnnn     | 42 42 6D    | 5            | 9             | <i>r</i>   |   |   |   |
| (indirect quad)          | ADCQ (\$nn)     | 42 42 72    | 4            | 10            | <i>ipr</i> |   |   |   |
| [indirect quad]          | ADCQ [\$nn]     | 42 42 EA 72 | 5            | 13            | <i>ipr</i> |   |   |   |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# AND

This instruction performs a binary AND operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, and that are set in the argument will be set in the accumulator on completion.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**AND : Binary AND**

45GS02

 $A \leftarrow A \text{ AND } M$ 

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| Addressing Mode | Assembly     | Code  | Bytes | Cycles |            |
|-----------------|--------------|-------|-------|--------|------------|
| [indirect],Z    | AND [\$nn],Z | EA 32 | 3     | 7      | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

## ANDQ

This instruction performs a binary AND operation of the argument with the Q pseudo register, and stores the result in the accumulator. Only bits that were already set in the Q pseudo register, and that are set in the argument will be set in the Q pseudo register on completion.

Note that the indicated memory location is treated as the first byte of a 32-bit little-endian value.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**ANDQ : Binary AND**

45GS02

 $Q \leftarrow Q \text{ AND } M$ 

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| N | Z | I | C | D | V | E |
| + | + | . | . | . | . | . |

| Addressing Mode | Assembly    | Code        | Bytes | Cycles |            |
|-----------------|-------------|-------------|-------|--------|------------|
| base-page quad  | ANDQ \$nn   | 42 42 25    | 4     | 8      | <i>r</i>   |
| absolute quad   | ANDQ \$nnnn | 42 42 2D    | 5     | 9      | <i>r</i>   |
| (indirect quad) | ANDQ (\$nn) | 42 42 32    | 4     | 10     | <i>ipr</i> |
| [indirect quad] | ANDQ [\$nn] | 42 42 EA 32 | 5     | 13     | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

# ASLQ

This instruction shifts either the Q pseudo-register or contents of the provided memory location and following three one bit left, treating them as holding a little-endian 32-bit value. Bit 0 will be set to zero, and the bit i31 will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| ASLQ : Arithmetic Shift Left |               | 45GS02                                                  |       |       |        |   |   |   |
|------------------------------|---------------|---------------------------------------------------------|-------|-------|--------|---|---|---|
|                              |               | $Q \leftarrow Q \ll 1 \text{ or } M \leftarrow M \ll 1$ |       |       |        |   |   |   |
|                              |               | N                                                       | Z     | I     | C      | D | V | E |
|                              |               | +                                                       | +     | .     | +      | . | . | . |
| Addressing Mode              | Assembly      | Code                                                    | Bytes | Bytes | Cycles |   |   |   |
| base-page quad               | ASLQ \$nn     | 42 42 06                                                | 4     | 12    | dmr    |   |   |   |
| Q Pseudo Register            | ASLQ Q        | 42 42 0A                                                | 3     | 3     |        |   |   |   |
| absolute quad                | ASLQ \$nnnn   | 42 42 0E                                                | 5     | 13    | dmr    |   |   |   |
| base-page quad,X             | ASLQ \$nn,X   | 42 42 16                                                | 4     | 12    | dmpr   |   |   |   |
| absolute quad,X              | ASLQ \$nnnn,X | 42 42 1E                                                | 5     | 13    | dmpr   |   |   |   |

d Subtract one cycle when CPU is at 3.5MHz.

m Subtract non-bus cycles when at 40MHz.

p Add one cycle if indexing crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

# ASRQ

This instruction shifts either the Q pseudo-register or contents of the provided memory location and following three one bit left, treating them as holding a little-endian 32-bit value. Bit 0 will be set to zero, and the bit i31 will be shifted out into the Carry Flag

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 7 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

- The C flag will be set if bit 7 of the value was set, prior to being shifted.

| <b>ASRQ : Arithmetic Shift Left</b>              |             | 45GS02        |       |        |                  |  |
|--------------------------------------------------|-------------|---------------|-------|--------|------------------|--|
| $Q \leftarrow Q \ll 1$ or $M \leftarrow M \ll 1$ |             | N Z I C D V E |       |        |                  |  |
| Addressing Mode                                  | Assembly    | Code          | Bytes | Cycles |                  |  |
| Q Pseudo Register                                | ASRQ Q      | 42 42 43      | 3     | 3      |                  |  |
| base-page quad                                   | ASRQ \$nn   | 42 42 44      | 4     | 12     | d <sub>mr</sub>  |  |
| base-page quad,X                                 | ASRQ \$nn,X | 42 42 54      | 4     | 12     | d <sub>mpr</sub> |  |

*d* Subtract one cycle when CPU is at 3.5MHz.

*m* Subtract non-bus cycles when at 40MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## BITQ

This instruction is used to test the bits stored in a memory location and following three, treating them as holding a little-endian 32-bit value. Bits 30 and 31 of the memory location's contents are directly copied into the Overflow Flag and Negative Flag. The Zero Flag is set or cleared based on the result of performing the binary AND of the Q Register and the contents of the indicated memory location.

### Side effects

- The N flag will be set if the bit 31 of the memory location is set, else it will be cleared.
- The V flag will be set if the bit 30 of the memory location is set, else it will be cleared.
- The Z flag will be set if the result of  $Q \text{ AND } M$  is zero, else it will be cleared.

| <b>BITQ : Perform Bit Test</b>                                          |             | 45GS02        |       |        |   |  |
|-------------------------------------------------------------------------|-------------|---------------|-------|--------|---|--|
| $N \leftarrow M(31), V \leftarrow M(30), Z \leftarrow Q \text{ AND } M$ |             | N Z I C D V E |       |        |   |  |
| Addressing Mode                                                         | Assembly    | Code          | Bytes | Cycles |   |  |
| base-page quad                                                          | BITQ \$nn   | 42 42 24      | 4     | 8      | r |  |
| absolute quad                                                           | BITQ \$nnnn | 42 42 2C      | 5     | 9      | r |  |

*r* Add one cycle if clock speed is at 40 MHz.

# CMP

This instruction performs  $A - M$ , and sets the processor flags accordingly, but does not modify the contents of the Accumulator Register.

## Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

| CMP : Compare Accumulator |              | 45GS02 |       |   |        |   |   |             |
|---------------------------|--------------|--------|-------|---|--------|---|---|-------------|
|                           |              | N      | Z     | I | C      | D | V | E           |
|                           |              | +      | +     | . | +      | . | . | .           |
| Addressing Mode           | Assembly     | Code   | Bytes |   | Cycles |   |   |             |
| [indirect],Z              | CMP [\$nn],Z | EA D2  | 3     | 7 |        |   |   | <i>i</i> pr |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# CMPO

This instruction performs  $Q - M$ , and sets the processor flags accordingly, but does not modify the contents of the Q Register. The memory location is treated as the address of a little-endian 32-bit value.

## Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 31 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| base-page quad         | CMPQ \$nn       | 42 42 C5    | 4            | 8             | <i>r</i>   |
| absolute quad          | CMPQ \$nnnn     | 42 42 CD    | 5            | 9             | <i>r</i>   |
| (indirect quad)        | CMPQ (\$nn)     | 42 42 D2    | 4            | 10            | <i>ipr</i> |
| [indirect quad]        | CMPQ [\$nn]     | 42 42 EA D2 | 5            | 13            | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## DEQ

This instruction decrements the Accumulator Register or indicated memory location.

Note that the indicated memory location is treated as the first byte of a 32-bit little-endian value.

### Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

$Q \leftarrow Q - 1$  or  $M \leftarrow M - 1$

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |             |
|------------------------|-----------------|-------------|--------------|---------------|-------------|
| Q Pseudo Register      | DEQ Q           | 42 42 3A    | 3            | 3             |             |
| base-page quad         | DEQ \$nn        | 42 42 C6    | 4            | 12            | <i>dmr</i>  |
| absolute quad          | DEQ \$nnnn      | 42 42 CE    | 5            | 13            | <i>dmr</i>  |
| base-page quad,X       | DEQ \$nn,X      | 42 42 D6    | 4            | 12            | <i>dmpr</i> |
| absolute quad,X        | DEQ \$nnnn,X    | 42 42 DE    | 5            | 13            | <i>dmpr</i> |

*d* Subtract one cycle when CPU is at 3.5MHz.

*m* Subtract non-bus cycles when at 40MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# EOR

This instruction performs a binary XOR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, but not both.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| <b>EOR : Binary Exclusive OR</b> | <b>45GS02</b>                                |
|----------------------------------|----------------------------------------------|
| $A \leftarrow A \text{ XOR } M$  |                                              |
|                                  | <b>N Z I C D V E</b><br>+ + . . . . .        |
| <b>Addressing Mode</b>           | <b>Assembly</b>                              |
| [indirect],Z                     | EOR [\$nn],Z EA 52 3 7 <i>i<sup>pr</sup></i> |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# EORQ

This instruction performs a binary XOR operation of the argument with the Q pseudo register, and stores the result in the Q pseudo register. Only bits that were already set in the Q pseudo register, or that are set in the argument will be set in the accumulator on completion, but not bits that were set in both.

## Side effects

- The N flag will be set if the result is negative, i.e., bit 31 is set, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**EORQ : Binary Exclusive OR**

45GS02

 $Q \leftarrow Q \text{ XOR } M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| base-page quad         | EORQ \$nn       | 42 42 45    | 4            | 8             | <i>r</i>   |
| absolute quad          | EORQ \$nnnn     | 42 42 4D    | 5            | 9             | <i>r</i>   |
| (indirect quad)        | EORQ (\$nn)     | 42 42 52    | 4            | 10            | <i>ipr</i> |
| [indirect quad]        | EORQ [\$nn]     | 42 42 EA 52 | 5            | 13            | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.**INQ**

This instruction increments the Q pseudo register or indicated memory location.

Note that the indicated memory location is treated as the first byte of a 32-bit little-endian value.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**INQ : Increment Memory or Accumulator**

45GS02

 $Q \leftarrow Q + 1 \text{ or } M \leftarrow M + 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| Q Pseudo Register      | INQ Q           | 42 42 1A    | 3            | 3             |            |
| base-page quad         | INQ \$nn        | 42 42 E6    | 4            | 13            | <i>dmr</i> |
| absolute quad          | INQ \$nnnn      | 42 42 EE    | 5            | 14            | <i>dmr</i> |
| base-page quad,X       | INQ \$nn,X      | 42 42 F6    | 4            | 13            | <i>dmp</i> |
| absolute quad,X        | INQ \$nnnn,X    | 42 42 FE    | 5            | 14            | <i>dpr</i> |

*d* Subtract one cycle when CPU is at 3.5MHz.*m* Subtract non-bus cycles when at 40MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

# LDA

This instruction loads the Accumulator Register with the indicated value, or with the contents of the indicated location.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| LDA : Load Accumulator | 45GS02                       |
|------------------------|------------------------------|
| A ← M                  |                              |
|                        | N Z I C D V E<br>+ + . . . . |

*i* Add one cycle if clock speed is at 40 MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# LDO

This instruction loads the Q pseudo register with the indicated value, or with the contents of the indicated location. As the Q register is an alias for A, X, Y and Z used together, this operation will set those four registers. A contains the least significant bits, X the next least significant, then Y, and Z contains the most significant bits.

## Side effects

- The N flag will be set if the result is negative, i.e., bit 31 is set, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**LDQ : Load Q Pseudo Register**

45GS02

 $Q \leftarrow M$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| base-page quad         | LDQ \$nn        | 42 42 A5    | 4            | 8             | <i>r</i>   |
| absolute quad          | LDQ \$nnnn      | 42 42 AD    | 5            | 9             | <i>r</i>   |
| (indirect quad)        | LDQ (\$nn)      | 42 42 B2    | 4            | 10            | <i>ipr</i> |
| [indirect quad]        | LDQ [\$nn]      | 42 42 EA B2 | 5            | 13            | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

## LSRQ

This instruction shifts either the Q pseudo register or contents of the provided memory location one bit right. Bit 31 will be set to zero, and the bit 0 will be shifted out into the Carry Flag.

Note that the memory address is treated as the first address of a little-endian encoded 32-bit value.

### Side effects

- The N flag will be set if the result is negative, i.e., if bit 31 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 0 of the value was set, prior to being shifted.

**LSRQ : Logical Shift Right****45GS02** $Q \leftarrow Q >> 1, C \leftarrow A(0) \text{ or } M \leftarrow M >> 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |             |
|------------------------|-----------------|-------------|--------------|---------------|-------------|
| base-page quad         | LSRQ \$nn       | 42 42 46    | 4            | 12            | <i>dmr</i>  |
| Q Pseudo Register      | LSRQ Q          | 42 42 4A    | 3            | 3             |             |
| absolute quad          | LSRQ \$nnnn     | 42 42 4E    | 5            | 13            | <i>dmr</i>  |
| base-page quad,X       | LSRQ \$nn,X     | 42 42 56    | 4            | 12            | <i>dmpr</i> |
| absolute quad,X        | LSRQ \$nnnn,X   | 42 42 5E    | 5            | 13            | <i>dmpr</i> |

*d* Subtract one cycle when CPU is at 3.5MHz.*m* Subtract non-bus cycles when at 40MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

## ORA

This instruction performs a binary OR operation of the argument with the accumulator, and stores the result in the accumulator. Only bits that were already set in the accumulator, or that are set in the argument will be set in the accumulator on completion, or both.

**Side effects**

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

**ORA : Decrement Memory or Accumulator****45GS02** $A \leftarrow A + 1 \text{ or } M \leftarrow M + 1$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| [indirect],Z           | ORA [\$nn],Z    | EA 12       | 3            | 7             | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

# ORQ

This instruction performs a binary OR operation of the argument with the Q pseudo register, and stores the result in the Q pseudo register. Only bits that were already set in the Q pseudo register, or that are set in the argument, or both, will be set in the Q pseudo register on completion.

Note that this operation treats the memory address as the first address of a 32-bit little-endian value. That is, the memory address and the three following will be used.

## Side effects

- The N flag will be set if the result is negative, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.

| ORQ : Decrement Memory or Q                  | 45GS02        |             |       |              |
|----------------------------------------------|---------------|-------------|-------|--------------|
| $Q \leftarrow Q + 1$ or $M \leftarrow M + 1$ |               |             |       |              |
|                                              | N Z I C D V E |             |       |              |
|                                              | + + . . . .   |             |       |              |
| Addressing Mode                              | Assembly      | Code        | Bytes | Cycles       |
| base-page quad                               | ORQ \$nn      | 42 42 05    | 4     | 8 <i>r</i>   |
| absolute quad                                | ORQ \$nnnn    | 42 42 0D    | 5     | 9 <i>r</i>   |
| (indirect quad)                              | ORQ (\$nn)    | 42 42 12    | 4     | 10 <i>pr</i> |
| [indirect quad]                              | ORQ [\$nn]    | 42 42 EA 12 | 5     | 13 <i>pr</i> |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

# RESQ

These extended opcodes are reserved, and their function is undefined and subject to change in future revisions of the 45GS02. They should therefore not be used in any programme.

UNDEFINED

N Z I C D V E

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| (indirect quad,X)      | RESQ (\$nn,X)   | 42 42 01    | 4            | 10            | <i>ipr</i> |
| (indirect quad),Y      | RESQ (\$nn),Y   | 42 42 11    | 4            | 10            | <i>ipr</i> |
| base-page quad,X       | RESQ \$nn,X     | 42 42 15    | 4            | 8             | <i>pr</i>  |
| absolute quad,Y        | RESQ \$nnnn,Y   | 42 42 19    | 5            | 9             | <i>pr</i>  |
| absolute quad,X        | RESQ \$nnnn,X   | 42 42 1D    | 5            | 9             | <i>pr</i>  |
| (indirect quad,X)      | RESQ (\$nn,X)   | 42 42 21    | 4            | 10            | <i>ir</i>  |
| (indirect quad),Y      | RESQ (\$nn),Y   | 42 42 31    | 4            | 10            | <i>ipr</i> |
| base-page quad,X       | RESQ \$nn,X     | 42 42 34    | 4            | 8             | <i>pr</i>  |
| base-page quad,X       | RESQ \$nn,X     | 42 42 35    | 4            | 8             | <i>pr</i>  |
| absolute quad,Y        | RESQ \$nnnn,Y   | 42 42 39    | 5            | 10            | <i>pr</i>  |
| absolute quad,X        | RESQ \$nnnn,X   | 42 42 3C    | 5            | 9             | <i>pr</i>  |
| absolute quad,X        | RESQ \$nnnn,X   | 42 42 3D    | 5            | 10            | <i>pr</i>  |
| (indirect quad,X)      | RESQ (\$nn,X)   | 42 42 41    | 4            | 10            | <i>ipr</i> |
| (indirect quad),Y      | RESQ (\$nn),Y   | 42 42 51    | 4            | 10            | <i>ipr</i> |
| base-page quad,X       | RESQ \$nn,X     | 42 42 55    | 4            | 8             | <i>pr</i>  |
| absolute quad,Y        | RESQ \$nnnn,Y   | 42 42 59    | 5            | 9             | <i>pr</i>  |
| absolute quad,X        | RESQ \$nnnn,X   | 42 42 5D    | 5            | 9             | <i>pr</i>  |
| (indirect quad,X)      | RESQ (\$nn,X)   | 42 42 61    | 4            | 10            | <i>ir</i>  |
| (indirect quad),Y      | RESQ (\$nn),Y   | 42 42 71    | 4            | 10            | <i>ipr</i> |
| base-page quad,X       | RESQ \$nn,X     | 42 42 75    | 4            | 8             | <i>pr</i>  |
| absolute quad,Y        | RESQ \$nnnn,Y   | 42 42 79    | 5            | 10            | <i>pr</i>  |
| absolute quad,X        | RESQ \$nnnn,X   | 42 42 7D    | 5            | 10            | <i>pr</i>  |

i Add one cycle if clock speed is at 40 MHz.

p Add one cycle if indexing crosses a page boundary.

r Add one cycle if clock speed is at 40 MHz.

## ROLQ

This instruction shifts either the Accumulator or contents of the provided memory location one bit left. Bit 0 will be set to the current value of the Carry Flag, and the bit 31 will be shifted out into the Carry Flag.

Note: The memory address is treated as the first address of a little-endian encoded 32-bit value.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 31 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 31 of the value was set, prior to being shifted.

| <b>ROLQ : Rotate Left Memory or Q</b> |                 | <b>45GS02</b>                                                  |              |               |  |  |             |
|---------------------------------------|-----------------|----------------------------------------------------------------|--------------|---------------|--|--|-------------|
|                                       |                 | M $\leftarrow$ M<<1, C $\leftarrow$ M(31), M(0) $\leftarrow$ C |              |               |  |  |             |
|                                       |                 | N Z I C D V E                                                  |              |               |  |  |             |
|                                       |                 | + + . + . .                                                    |              |               |  |  |             |
| <b>Addressing Mode</b>                | <b>Assembly</b> | <b>Code</b>                                                    | <b>Bytes</b> | <b>Cycles</b> |  |  |             |
| base-page quad                        | ROLQ \$nn       | 42 42 26                                                       | 4            | 12            |  |  | <i>dmr</i>  |
| Q Pseudo Register                     | ROLQ Q          | 42 42 2A                                                       | 3            | 3             |  |  |             |
| absolute quad                         | ROLQ \$nnnn     | 42 42 2E                                                       | 5            | 13            |  |  | <i>dmr</i>  |
| base-page quad,X                      | ROLQ \$nn,X     | 42 42 36                                                       | 4            | 12            |  |  | <i>dmpr</i> |
| absolute quad,X                       | ROLO \$nnnn,X   | 42 42 3E                                                       | 5            | 13            |  |  | <i>dmpr</i> |

*d* Subtract one cycle when CPU is at 3.5MHz.

*m* Subtract non-bus cycles when at 40MHz.

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## RORQ

This instruction shifts either the Q pseudo register or contents of the provided memory location one bit right. Bit 31 will be set to the current value of the Carry Flag, and the bit 0 will be shifted out into the Carry Flag

Note that the address is treated as the first address of a little-endian 32-bit value.

## Side effects

- The N flag will be set if the result is negative, i.e., if bit 31 is set after the operation, else it will be cleared.
- The Z flag will be set if the result is zero, else it will be cleared.
- The C flag will be set if bit 31 of the value was set, prior to being shifted.

**RORQ : Rotate Right Memory or Accumulator**

45GS02

 $M \leftarrow M >> 1, C \leftarrow M(0), M(31) \leftarrow C$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |             |
|------------------------|-----------------|-------------|--------------|---------------|-------------|
| base-page quad         | RORQ \$nn       | 42 42 66    | 4            | 12            | <i>dmr</i>  |
| Q Pseudo Register      | RORQ Q          | 42 42 6A    | 3            | 3             |             |
| absolute quad          | RORQ \$nnnn     | 42 42 6E    | 5            | 13            | <i>dmr</i>  |
| base-page quad,X       | RORQ \$nn,X     | 42 42 76    | 4            | 12            | <i>dmpr</i> |
| absolute quad,X        | RORQ \$nnnn,X   | 42 42 7E    | 5            | 13            | <i>dmpr</i> |

*d* Subtract one cycle when CPU is at 3.5MHz.*m* Subtract non-bus cycles when at 40MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.**RSVQ**

These extended opcodes are reserved, and their function is undefined and subject to change in future revisions of the 45GS02. They should therefore not be used in any programme.

UNDEFINED

N Z I C D V E

| Addressing Mode      | Assembly         | Code     | Bytes | Cycles |            |
|----------------------|------------------|----------|-------|--------|------------|
| (indirect quad,X)    | RSVQ (\$nn,X)    | 42 42 81 | 4     | 10     | <i>ip</i>  |
| (indirect quad,SP),Y | RSVQ (\$nn,SP),Y | 42 42 82 | 4     | 10     | <i>ip</i>  |
| (indirect quad),Y    | RSVQ (\$nn),Y    | 42 42 91 | 4     | 10     | <i>ip</i>  |
| base-page quad,X     | RSVQ \$nn,X      | 42 42 95 | 4     | 8      | <i>p</i>   |
| absolute quad,Y      | RSVQ \$nnnn,Y    | 42 42 99 | 5     | 9      | <i>p</i>   |
| absolute quad,X      | RSVQ \$nnnn,X    | 42 42 9D | 5     | 9      | <i>p</i>   |
| (indirect quad,X)    | RSVQ (\$nn,X)    | 42 42 A1 | 4     | 10     | <i>ipr</i> |
| (indirect quad,X)    | RSVQ (\$nn,X)    | 42 42 C1 | 4     | 10     | <i>ipr</i> |
| (indirect quad),Y    | RSVQ (\$nn),Y    | 42 42 D1 | 4     | 10     | <i>ipr</i> |
| base-page quad,X     | RSVQ \$nn,X      | 42 42 D5 | 4     | 8      | <i>pr</i>  |
| absolute quad,Y      | RSVQ \$nnnn,Y    | 42 42 D9 | 5     | 9      | <i>pr</i>  |
| absolute quad,X      | RSVQ \$nnnn,X    | 42 42 DD | 5     | 9      | <i>pr</i>  |
| (indirect quad,X)    | RSVQ (\$nn,X)    | 42 42 E1 | 4     | 10     | <i>ipr</i> |
| (indirect quad),Y    | RSVQ (\$nn),Y    | 42 42 F1 | 4     | 10     | <i>ipr</i> |
| base-page quad,X     | RSVQ \$nn,X      | 42 42 F5 | 4     | 8      | <i>pr</i>  |
| absolute quad,Y      | RSVQ \$nnnn,Y    | 42 42 F9 | 5     | 8      | <i>pr</i>  |
| absolute quad,X      | RSVQ \$nnnn,X    | 42 42 FD | 5     | 9      | <i>pr</i>  |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.

## SBC

This instruction performs  $A - M - 1 + C$ , and sets the processor flags accordingly. The result is stored in the Accumulator Register.

NOTE: This instruction is affected by the status of the Decimal Flag.

### Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 7 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.

- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

| <b>SBC : Subtract With Carry</b> | <b>45GS02</b>   |             |              |               |
|----------------------------------|-----------------|-------------|--------------|---------------|
| $A \leftarrow -M - 1 + C$        |                 |             |              |               |
| <b>Addressing Mode</b>           | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |
| [indirect],Z                     | SBC [\$nn],Z    | EA F2       | 3            | 8 <i>pr</i>   |

*p* Add one cycle if indexing crosses a page boundary.

*r* Add one cycle if clock speed is at 40 MHz.

## SBCQ

This instruction performs  $Q - M - 1 + C$ , and sets the processor flags accordingly. The result is stored in the Q pseudo register.

Note: that the indicated memory location is treated as the first byte of a 32-bit little-endian value.

Note: The decimal (D) flag must be clear. Operation is reserved when D flag is set.

### Side effects

- The N flag will be set if the result of  $A - M$  is negative, i.e. bit 31 is set in the result, else it will be cleared.
- The C flag will be set if the result of  $A - M$  is zero or positive, i.e., if A is not less than M, else it will be cleared.
- The V flag will be set if the result has a different sign to both of the arguments, else it will be cleared. If the flag is set, this indicates that a signed overflow has occurred.
- The Z flag will be set if the result of  $A - M$  is zero, else it will be cleared.

**SBCQ : Subtract With Carry****45GS02** $Q \leftarrow -M - 1 + C$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| +        | +        | .        | +        | .        | +        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |            |
|------------------------|-----------------|-------------|--------------|---------------|------------|
| base-page quad         | SBCQ \$nn       | 42 42 E5    | 4            | 8             | <i>r</i>   |
| absolute quad          | SBCQ \$nnnn     | 42 42 ED    | 5            | 9             | <i>r</i>   |
| (indirect quad)        | SBCQ (\$nn)     | 42 42 F2    | 4            | 10            | <i>ipr</i> |
| [indirect quad]        | SBCQ [\$nn]     | 42 42 EA F2 | 5            | 13            | <i>ipr</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.*r* Add one cycle if clock speed is at 40 MHz.**STA**

This instruction stores the contents of the Accumulator Register into the indicated location.

**STA : Store Accumulator****45GS02** $M \leftarrow A$ 

|          |          |          |          |          |          |          |
|----------|----------|----------|----------|----------|----------|----------|
| <b>N</b> | <b>Z</b> | <b>I</b> | <b>C</b> | <b>D</b> | <b>V</b> | <b>E</b> |
| .        | .        | .        | .        | .        | .        | .        |

| <b>Addressing Mode</b> | <b>Assembly</b> | <b>Code</b> | <b>Bytes</b> | <b>Cycles</b> |           |
|------------------------|-----------------|-------------|--------------|---------------|-----------|
| [indirect],Z           | STA [\$nn],Z    | EA 92       | 3            | 8             | <i>ip</i> |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.**STQ**

This instruction stores the contents of the Q pseudo register into the indicated location.

As Q is composed of A, X, Y and Z, this means that these four registers will be written to the indicated memory location through to the indicated memory location plus 3, respectively.

**STQ : Store Q****45GS02** $M \leftarrow Q = M+0 \leftarrow A, \dots, M+3 \leftarrow Z$ **N Z I C D V E**

. . . . .

| <b>Addressing Mode</b> | <b>Assembly Code</b> | <b>Bytes</b> | <b>Cycles</b> |
|------------------------|----------------------|--------------|---------------|
| base-page quad         | STQ \$nn             | 42 42 85     | 4             |
| absolute quad          | STQ \$nnnn           | 42 42 8D     | 5             |
| (indirect quad)        | STQ (\$nn)           | 42 42 92     | 4             |
| [indirect quad]        | STQ [\$nn]           | 42 42 EA 92  | 5             |

*i* Add one cycle if clock speed is at 40 MHz.*p* Add one cycle if indexing crosses a page boundary.



## APPENDIX

# Developing System Programmes

- **Introduction**
- **Flash Menu**
- **Format/FDISK Utility**
- **Keyboard Test Utility**
- **MEGA65 Configuration Utility**
- **Freeze Menu**
- **Freeze Menu Helper Programmes**
- **Hypervisor**

- OpenROM





# INTRODUCTION

The MEGA65 has a number of system programmes and utilities that are used at various times to perform various functions. This includes the utilities accessible via the Utility Menu , the Freeze Menu and its own helper programmes, as well as the Flash Menu .

A number of these system programmes are pre-loaded into the MEGA65 bitstream, while others live on the SD card. For those that are pre-loaded into the MEGA65 bitstream, this works by having areas of pre-initialised memory, that contain the appropriate programme. For example, the utilities accessible via the Utility Menu are all located in the colour RAM, while the Flash Menu is located at \$50000 - \$57FFF.

In one sense, the easiest way to test new versions of these utilities is to generate a new bitstream with the updated versions. However, synthesising a new bitstream is very time consuming, typically taking an hour on a reasonably fast computer. Therefore this chapter explains the procedure for loading an alternate version of each of these system programmes, as well as providing some useful information about these programmes, how they operate, and the environment in which they operate compared with normal C64 or C65 mode programmes.

## FLASH MENU

The flash menu is located in pre-initialised RAM at \$50000 - \$57FFFF. It is executed during the first boot each time the MEGA65 is powered on. It is unusual in that it executes in the hypervisor context. This is so that it has access to the QSPI flash, which is not available outside of Hypervisor Mode, so that user programmes cannot corrupt the cores stored in the flash.

It is also important to note that the flash menu programme must fit *entirely* below \$8000 when loaded *and* executing, as the Hypervisor is still mapped at \$8000 - \$BFFF, and can easily be corrupted by an ill behaved flash menu programme. In this regard, the flash menu can be regarded as an extension of the hypervisor that is discarded after the first boot. This is unlike all other system programmes, that operate in a dedicated memory context, from where the Hypervisor is safe from corruption. It also means that you can't crunch the flash menu to make it fit, as it would overwrite the Hypervisor during decrunching.

Also, as the flash menu is executed very early in the boot process, only the pre-included OpenROM ROM image is available. Thus you must ensure that your flash menu programme is compatible with that ROM.

The Hypervisor maintains a flag that indicates whether the flash menu has been executed or not. This flag is updated at the point where the Hypervisor exits to user mode for the first time, since after that point, the contents of \$50000 - \$57FFF can no longer be

trusted to contain the flash menu. This means that if you wish to have the Hypervisor run a new version of the flash menu that you have loaded, you must prevent the Hypervisor from exiting to user mode first.

The easiest way to achieve this is to hold the ALT key down while powering on the MEGA65. This will cause the Hypervisor to display the Utility Menu, rather than exiting to user mode. It is safe at this time to use the `m65` utility to load the replacement flash menu programme using a command similar to the following:

```
m65 -@ newflashmenu.prg@50000
```

That command would load the file `newflashmenu.prg` at memory location \$50000.

After that, you can simply press the reset button the side of the MEGA65i while holding down the NO SCROLL key, and it will boot again, and because it never left Hypervisor Mode during the previous boot cycle, it will run your updated flash menu programme.

It should also be possible to completely automate this process, by first using `m65 -b` to load a new bitstream, thus simulating a cold boot, and then quickly calling `m65` again to simulate depressing the ALT key (or perhaps simply halting the processor), then `m65 -@ ...` and finally `m65 -F` to reset the machine. Writing a script or utility that correctly implements this automation is left as an exercise for the reader.

## FORMAT/FDISK UTILITY

The Format/FDISK utility is accessed as part of the Utility Menu system. These utilities are compiled, crunched and linked using the `utilpacker` programme. If you have checked out the `mega65-core` source repository, you can re-build the colour RAM image by using:

```
make bin/COLOURRAM.BIN
```

You will of course need to first have modified the Format/FDISK utility, which is normally located in the `src/mega65-freeze-menu` subdirectory.

You need to then load this modified colour RAM image into the running machine. Similar to when updating the flash menu, the Hypervisor will only present the utility menu on the first boot, before exiting to user mode for the first time, because it cannot otherwise be sure that the colour RAM contains the valid utility programmes.

So as for the flash menu, you would power the MEGA65 off, and then holding the ALT key down, you turn the MEGA65 back on, so that it displays the utility menu. At this point you can use the following command to load your modified COLOURRAM.BIN file:

```
m65 -c COLOURRAM.BIN
```

You can now hold down the ALT key, and press the reset button on the left side of the MEGA65, which should again present the utility menu, but this time with your modified format/FDISK utility in place.

## KEYBOARD TEST UTILITY

The process for updating the Keyboard test utility is essentially the same as for the format/FDISK utility, as it lives in the colour RAM

## MEGA65 CONFIGURATION UTILITY

The process for updating the MEGA65 Configuration utility is essentially the same as for the format/FDISK utility, as it lives in the colour RAM

## FREEZE MENU

The Freeze Menu is a normal programme, which is stored in FREEZER.M65 on the SD card's FAT32 file system.

To updated the Freeze Menu, simply use the m65ftp utility or some other means to upload your updated FREEZER.M65 file to the SD card's FAT32 file system. The format of the programme is simply a C64-mode PRG file, just renamed to FREEZER.M65.

## FREEZE MENU HELPER PROGRAMMES

The Freeze Menu helper programmes are updated in the same way as the Freeze Menu itself.

## HYPERVISOR

The Hypervisor is normally built as HICKUP.M65, a 16KiB file that contains the complete Hypervisor programme. MEGA65 bitstreams contain a pre-build version located

at \$FFF8000 – \$FFFFBFFF. Updated versions of the Hypervisor can be tested using two main approaches:

- 1. Place the updated HICKUP.M65 file on the FAT32 file system of the SD card, and then power the MEGA65 off and on. This works because the Hypervisor contains code that checks for an updated version of itself, and if found, loads it. However this approach is problematic in that if you install a newer bitstream, it will still downgrade the Hypervisor to whatever version is found in the HICKUP.M65 file on the SD card. This method is only recommended for developers who have a need to test their modified Hypervisor code from a cold start. Even then, it is recommended to remove the HICKUP.M65 file immediately after testing to avoid unexpected down-grading in the future.
- 2. Use the m65 command's `t -k` option to replace the Hypervisor in place, and then reset the MEGA65 using the reset button on the left side of the case. This should be done when the Hypervisor is *not* active, so that corruption of current execution cannot occur. However, it must also occur before any ROM has been loaded to replace the default OpenROM image. This is because the Hypervisor will attempt to call into the ROM on first-boot in preparation for calling the flash menu, and assumes that the OpenROM is present, because it uses a special OpenROM-specific call to initialise parts of the system state for the flash menu. This is best done by using a command like `m65 -k bin/HICKUP.M65 -R bin/MEGA65.ROM` to load both a new Hypervisor programme and re-load an OpenROM image.

## OPENROM

To load a new version of a ROM, there are several options, including replacing both the Hypervisor and ROM at the same time, as described above. However, typically the easiest is to copy the new ROM onto the FAT32 filesystem of the SD card as either MEGA65.ROM, or MEGA65 $x$ .ROM, where  $x$  is replaced by a digit between 0 and 9. When resetting the MEGA65, MEGA65.ROM will then be loaded as normal, or if a digit between 0 and 9 is held down on the keyboard while resetting, the Hypervisor will instead load MEGA65 $x$ .ROM, where  $x$  is the number being held down on the keyboard.

# J

## APPENDIX

# MEGA65 Hypervisor Services

- General Services
- Disk/Storage Hypervisor Calls
- Disk Image Management
- Task and Process Management
- System Partition & Freezing
- Secure Mode



The MEGA65's Hypervisor provides a number of services via Hypervisor Traps. This chapter will describe these services. For detailed information on how Hypervisor Traps are facilitated by the CPU is described in Chapter/Appendix G on page G-3.

The hypervisor calls are identified by the trap register (\$D640 - \$D67F), and the value of the accumulator register when writing to the trap register. Thus a hypervisor call \$00:\$02 would be made via the following sequence of instructions:

```
LDA #$02  
STA $D640+$00  
NOP ; All traps calls MUST be followed by a NOP instruction
```

The values of the other registers or other structures will be described for each individual call.

## GENERAL SERVICES

### \$00:\$00 – Get Hypervisor Version

Returns the version of the Hypervisor operating system and DOS components in the four registers:

- A = Hypervisor Operating System Major Version number.
- X = Hypervisor Operating System Minor Version number.
- Y = Hypervisor DOS Minor Version number.
- Z = Hypervisor DOS Major Version number.

These values can be used to check whether a given MEGA65 system's hypervisor supports features that become available (or are deprecated) at particular versions.

### \$00:\$38 – Get Current Error Code (geterrorcode)

Returns the current error code from the Hypervisor. The currently supported error codes are:

- \$01 (1) – Partition Not Interesting, indicating that an attempt to mount a partition was rejected because the partition was not of a supported type.

- \$02 (2) - Bad Signature, indicating that the signature bytes at the end of a partition table or of the first sector of a partition were missing or incorrect.
- \$03 (3) - An attempt was made to mount a FAT12 or FAT16 partition. Only FAT32 partitions are supported.
- \$04 (4) - An attempt was made to mount a partition that has too many reserved sectors. The number of reserved sectors must be less than 65,536.
- \$05 (5) - An attempt was made to mount a partition that does not have exactly two copies of the FAT structure.
- \$06 (6) - An attempt was made to mount a partition that contains a partition with too few clusters.
- \$07 (7) - A read timeout occurred.
- \$08 (8) - An unspecified error occurred while handling a partition.
- \$10 (16) - An invalid address was supplied to the Setup Transfer Area For Other Calls function.
- \$11 (17) - An illegal value was supplied to a Hypervisor call.
- \$20 (32) - A read error occurred.
- \$21 (33) - A write error occurred.
- \$80 (128) - An attempt was made to select or operate on a disk or partition that does not exist.
- \$81 (129) - The supplied filename was too long.
- \$82 (130) - A Hypervisor call was made to a function that is not implemented.
- \$83 (131) - An attempt was made to load a file into memory that is longer than 16MiB.
- \$84 (132) - Too many files are open, and no free file descriptor could be obtained for the requested operation.
- \$85 (133) - The supplied cluster number is invalid.
- \$86 (134) - An attempt was made to operate on a directory, where a normal file was expected.
- \$87 (135) - An attempt was made to operate on a normal file, where a directory was expected.
- \$88 (136) - The requested file could not be located.
- \$89 (137) - An invalid file descriptor was supplied.

- \$8A (138) - A disk image file had the wrong length, and was rejected for this reason.
- \$8B (139) - A disk image was attempted to be mounted, but could not because it is fragmented on the file system. Disk images must be stored contiguously on disk. This is because of the way that the SD card controller and floppy controller work: They load the starting sector of the disk image into special registers, and have no way to correctly handle a disk image that is stored in separate pieces on the disk.
- \$8C (140) - The disk has no free space for the requested operation.
- \$8D (141) - An attempt was made to create a file that already exists, or to rename a file to have the name of a file that already exists.
- \$8E (142) - An attempt was made to create a file in a directory that cannot accommodate any more entries.
- \$FF (255) - The end of a file or directory was encountered.

## \$00:\$3A – Setup Transfer Area for Other Calls (`setup_transfer_area`)

Setup the transfer area for various hypervisor calls. The page number of the transfer area is supplied in the Y register. This page must be between \$00 (0) and \$7E (126), thus indicating a transfer area starting between \$0000 and \$7E00 (0 and 32,256). The transfer area is 256 bytes long for most calls. Note that the transfer area is indicated using the processor's current memory mapping at the time that a function is called. However, it is good practice to always place it in the bottom 32KiB of main memory.

- Y = Page number of the transfer area (\$00 - \$7E).

This call can produce the following error codes:

- \$10 (16) - An invalid transfer area address was supplied, i.e., Y > \$7E (126).

## DISK/STORAGE HYPERVISOR CALLS

### \$00:\$02 – Get Default Drive (SD Card Partition)

This call returns the default drive (SD card partition) number in the A register.

## **\$00:\$04 - Get Current Drive (SD Card Partition)**

This call returns the current selected drive (SD card partition) in the A register.

## **\$00:\$06 - Select Drive (SD Card Partition)**

This call sets the currently selected drive (SD card partition) to the drive indicated in the X register.

- X = Selected drive (SD card partition) number.

This call can produce the following error codes:

- \$80 (128) - An attempt was made to select or operate on a disk or partition that does not exist.

## **\$00:\$08 - NOT IMPLEMENTED Get Disk Size**

When implemented, this call will return information on the size of the currently selected disk (SD card partition).

## **\$00:\$0A - NOT IMPLEMENTED Get Current Working Directory**

When implemented, this call will return information on the currently selected directory or sub-directory.

## **\$00:\$0C - Change Working Directory**

Changes the current working directory to the directory specified in the dirent structure. The dirent structure can be populated by using any of the findfirst, findnext, findfile, or readdir Hypervisor calls.

This call can produce the following error codes:

- \$87 (135) - An attempt was made to operate on a normal file, where a directory was expected.

## **\$00:\$0E – NOT IMPLEMENTED Create Directory**

When implemented, this call will allow the creation of new subdirectories.

## **\$00:\$10 – NOT IMPLEMENTED Remove Directory**

When implemented, this call will allow the removal of a directory.

## **\$00:\$12 – Open Directory (opendir)**

Open the current working directory.

On success, it returns the file descriptor of the opened directory in the A register.

This call can result in the following error codes:

- \$07 (7) - A read timeout occurred.
- \$08 (8) - An unspecified error occurred while handling a partition.
- \$10 (16) - An invalid address was supplied to the Setup Transfer Area For Other Calls function.
- \$11 (17) - An illegal value was supplied to a Hypervisor call.
- \$20 (32) - A read error occurred.
- \$21 (33) - A write error occurred.
- \$80 (128) - An attempt was made to select or operate on a disk or partition that does not exist.
- \$84 (132) - Too many files are open, and no free file descriptor could be obtained for the requested operation.

**\$00:\$14 – Read Next Directory Entry  
(readdir)**

**\$00:\$16 – Close Directory (closedir)**

**\$00:\$18 – Open File (openfile)**

**\$00:\$1A – Read From a File (readfile)**

**\$00:\$1C – NOT IMPLEMENTED Write to a  
File (writefile)**

**\$00:\$1E – NOT IMPLEMENTED Create  
File (mkfile)**

**\$00:\$20 – Close a File (closefile)**

**\$00:\$22 – Close All Open Files (closeall)**

**\$00:\$24 – NOT IMPLEMENTED Seek to a  
Given Offset in a File (seekfile)**

**\$00:\$26 – NOT IMPLEMENTED Delete a  
File (rmfile)**

**\$00:\$28 – NOT IMPLEMENTED Get Infor-  
mation About a File (fstat)**

**\$00:\$2A – NOT IMPLEMENTED Rename a File (rename)**

**\$00:\$2C – NOT IMPLEMENTED Set time stamp of a file (filedate)**

**\$00:\$2E – Set the current filename (set-name)**

**\$00:\$30 – Find first matching file (find-first)**

**\$00:\$32 – Find subsequent matching file (findnext)**

**\$00:\$34 – Find matching file (one only) (findfile)**

**\$00:\$36 – Load a File into Memory (loadfile)**

**\$00:\$3C – Change Working Directory to Root Directory of Selected Partition**

## **DISK IMAGE MANAGEMENT**

**\$00:\$40 – Attach a D81 Disk Image to Drive 0**

**\$00:\$42 – Detach All D81 Disk Images**

**\$00:\$44 – Write Enable All Currently Attached D81 Disk Images**

**\$00:\$46 – Attach a D81 Disk Image to Drive 1**

## **TASK AND PROCESS MANAGEMENT**

**\$00:\$50 – NOT IMPLEMENTED Get Task List**

**\$00:\$52 – NOT IMPLEMENTED Send Message to Another Task**

**\$00:\$54 – NOT IMPLEMENTED Receive Messages From Other Tasks**

**\$00:\$56 – NOT IMPLEMENTED Write Into Memory of Another Task**

**\$00:\$58 – NOT IMPLEMENTED Read From Memory of Another Task**

**\$00:\$60 – NOT IMPLEMENTED** Terminate Another Task

**\$00:\$62 – NOT IMPLEMENTED** Create a Native MEGA65 Task

**\$00:\$64 – NOT IMPLEMENTED** Load File Into Task

**\$00:\$66 – NOT IMPLEMENTED** Create a C64-Mode Task

**\$00:\$68 – NOT IMPLEMENTED** Create a C65-Mode Task

**\$00:\$6A – NOT IMPLEMENTED** Exit and Switch to Another Task

**\$00:\$6C – NOT IMPLEMENTED** Context-Switch to Another Task

**\$00:\$6E – NOT IMPLEMENTED** Exit This Task

**\$00:\$70 – Toggle Write Protection of ROM Area**

## **\$00:\$72 – Toggle 4510 vs 6502 Processor Mode**

## **\$00:\$74 – Get current 4510 memory MAPping**

Y = page where memory mapping is to be stored. Six bytes will be returned: Y must be <= \$7E.

- \$00 – Low byte of MAPLO (lower 32KB RAM mapping)
- \$01 – High byte of MAPLO (lower 32KB RAM mapping)
- \$02 – Low byte of MAPHI (upper 32KB RAM mapping)
- \$03 – High byte of MAPHI (upper 32KB RAM mapping)
- \$04 – Megabyte offset for MAPLO (lower 32KB RAM mapping)
- \$05 – Megabyte offset for MAPHI (upper 32KB RAM mapping)

## **\$00:\$76 – Set 4510 memory MAPping**

This call performs the opposite of the Get 4510 memory MAPping, reading 6 bytes from the memory page indicated by Y, and storing them into the current processor's mapping status. Y must be <= \$7E.

## **\$00:\$7C – Write Character to Serial Monitor/Matrix Mode Interface**

## **\$00:\$7E – Reset MEGA65**

## **\$01:\$00 – Enable Write Protection of ROM Area**

**\$01:\$02 – Disable Write Protection of ROM Area**

## **SYSTEM PARTITION & FREEZING**

**\$02:\$00 – Read System Config Sector Into Memory**

**\$02:\$02 – Write System Config Sector From Memory**

**\$02:\$04 – Apply System Config Sector Current Loaded Into Memory**

**\$02:\$06 – Set DMAgic Revision Based On Loaded ROM**

**\$02:\$10 – Locate First Sector of Freeze Slot**

**\$02:\$12 – Unfreeze From Freeze Slot**

**\$02:\$14 – Read Freeze Region List**

**\$02:\$16 – Get Number of Freeze Slots**

## **\$03:\$XX – Write Character to Serial Monitor/Matrix Mode Interface**

### **SECURE MODE**

**\$11:\$XX – Request Enter Secure Mode**

**\$12:\$XX – Request Exit Secure Mode**

**\$32:\$XX – DEPRECATED Set Protected Hardware Configuration**

This call will be removed once the secure mode framework is more completely implemented.

Until it is removed, this call allows a process to request the setting of the protected hardware configuration to allow or restrict access to various sub-systems, including the SD card storage system.

**\$3F:\$XX – Freeze Self**

# K

## APPENDIX

### Machine Language Monitor

- **Introduction**
- **C65 ROM Standard Machine Language Monitor**
- **Enhanced Machine Language Monitor**
- **MEGA65 Matrix Mode Monitor Interface**



# INTRODUCTION

Before we go any further, it is important to remember that the MEGA65 typically has two separate machine language monitors: The one included in C65 ROMs, and the one that is part of the Matrix Mode debug interface. It is also possible to replace the standard C65 monitor in the ROM with the enhanced MEGA65 OpenROMs machine language monitor, which corrects many bugs and adds many new features – including support for all enhanced CPU instructions of the MEGA65. This chapter describes all three of these machine language monitors.

## C65 ROM STANDARD MACHINE LANGUAGE MONITOR

The machine language monitor is a debugging tool for machine language programs. It includes a mini-assembler, a disassembler and many useful commands. When the program execution encounters the code 00 (zero) alias BRK, the default action of the operating system is, to call the monitor. This feature allows the debugging of programs by setting breakpoints.

# Table of C65 ROM Standard Monitor Commands

| C  | mnemonic     | description                                   |
|----|--------------|-----------------------------------------------|
| A  | ASSEMBLE     | Assemble a line of 45GS02 code                |
| C  | COMPARE      | Compare two sections of memory                |
| D  | DISASSEMBLE  | Disassemble a line of 45GS02 code             |
| F  | FILL         | Fill a section of memory with a value         |
| G  | GO           | Start execution at specified address          |
| H  | HUNT         | Find specified data in a section of memory    |
| L  | LOAD         | Load a file from disk                         |
| M  | MEMORY       | Dump a section of memory                      |
| R  | REGISTERS    | Display the contents of the 45GS02 registers  |
| S  | SAVE         | Save a section of memory to a disk file       |
| T  | TRANSFER     | Transfer memory to another location           |
| V  | VERIFY       | Compare a section of memory with a disk file  |
| X  | EXIT         | Exit Monitor mode                             |
| .  | <period>     | Assembles a line of 45GS02 code               |
| >  | <greater>    | Modifies memory                               |
| ;  | <:semicolon> | Modifies register contents                    |
| @  | <at sign>    | Disk command, directory or status             |
| \$ | <hex>        | Display hex, decimal, octal, and binary value |
| +  | <decimal>    | Display hex, decimal, octal, and binary value |
| &  | <octal>      | Display hex, decimal, octal, and binary value |
| %  | <binary>     | Display hex, decimal, octal, and binary value |

## Calling the Monitor

To enter the monitor from BASIC, type: **MONITOR**

The monitor responds with a display of register contents and waits for a command:

```
MONITOR
PC  SR  AC  XR  YR  ZR  BP  SP
; 000000 00 00 00 00 00 00 F8
```

# addresses and numbers

All addresses and numbers must be numbers of base 16 (hex), 10 (decimal), 8 (octal) or 2 (dual). Symbolic names like CHROUT or arithmetics like \$1000+5 are not allowed.

It is an old tradition since the first monitor of the Commodore PET, that the default base is 16. In fact the old monitors would not accept any other numbers, than hexadecimal (short hex). This may confuse beginners, because a statement like

LDA #10

loads the decimal value 16 into the accumulator. Later monitors, like that of the Commodore 128 accepted numbers of base 16,10,8 and 2 - like this one, but still used 16 (hex) as default. Additionally the MEGA65 monitor allows character entry, which uses the PETSCII value of the character. Following prefixes can be used to specify the base of the following number:

| base | name        | prefix | digits           | characters | example    |
|------|-------------|--------|------------------|------------|------------|
| 16   | hexadecimal |        | 0123456789ABCDEF |            | 100        |
| 16   | hexadecimal | \$     | 0123456789ABCDEF |            | \$100      |
| 10   | decimal     | +      | 0123456789       |            | +256       |
| 8    | octal       | &      | 01234567         |            | &400       |
| 2    | dual        | %      | 01               |            | %100000000 |
|      | character   | '      | all              |            | 'A         |

## D : DISASSEMBLE

**Format:** D [from [,to]]

**Usage:** Prints a machine language listing for the specified address range assuming, that it contains code. If only one argument is present, the disassembler disassembles the next 21 bytes. If no argument is given, the disassembly continues with the last used disassemble address. The contents are printed as hex values.

**Remarks:** The rows start with the dot character '..'. This enables direct full screen editing of the disassembly. Typing return in any row will assemble the changed command of the cursor row back to memory, if writable RAM is there. See monitor command ..

The disassembler knows the instruction set of the C65 CPU GS6502. Enhanced instructions from the 45GS02 CPU of the MEGA65 are not recognised.

**Example:** Using D

DL"\*

SEARCHING FOR 0:\*

LOADING

READY

MONITOR

MONITOR

|   | PC     | SR | AC | XR | YR  | ZR       | SP |
|---|--------|----|----|----|-----|----------|----|
| : | 000000 | 00 | 00 | 00 | 00  | 00       | F8 |
| D | 31F7   |    |    |    |     |          |    |
| : | 0031F7 | A4 | 29 |    | LDY | \$29     |    |
| : | 0031F9 | AB | 29 | 00 | LDZ | \$0029   |    |
| : | 0031FC | B1 | AC |    | LDA | (\$AC),Y |    |
| : | 0031FE | 91 | 57 |    | STA | (\$57),Y |    |
| : | 003200 | EA |    |    | NOP |          |    |
| : | 003201 | B2 | AE |    | LDA | (\$AE),Z |    |
| : | 003203 | EA |    |    | NOP |          |    |
| : | 003204 | 92 | 59 |    | STA | (\$59),Z |    |
| : | 003206 | 3B |    |    | DEZ |          |    |
| : | 003207 | 88 |    |    | DEY |          |    |
| : | 003208 | 10 | F2 |    | BPL | \$31FC   |    |
| : | 00320A | 18 |    |    | CLC |          |    |
| : | 00320B | A5 | AC |    | LDA | \$AC     |    |

## M : MEMORY

**Format:** M [from [,to]]

**Usage:** Prints a memory dump for the given address range. The dump displays memory contents, organised in rows of 16 consecutive addresses starting with the address, given as 1st. argument. The dump continues until a row has been printed, containing the value of the address given as 2nd. argument. If no 2nd. argument is present, the dump displays a full page of 256 bytes in 16 rows. The contents are printed as 16 byte values in hex, followed by the character representation.

**Remarks:** The rows start with the character '>'. This enables direct full screen editing of the dump. Typing return in any row will write the changed values of the cursor row back to memory, if writable RAM is there. See monitor command >.

**Example:** Using M

```

M 0322A0
>0322A0 20 20 42 41 53 49 43 20 31 30 2E 30 20 20 20 20: BASIC 10.0
>0322B0 56 30 2E 39 2E 39 31 30 31 31 31 20 20 20 20 41: U0.9.9i0iii A
>0322C0 4C 4C 20 52 49 47 48 54 53 20 52 45 53 45 52 56: ALL RIGHTS RESERV
>0322D0 45 44 0D 00 60 A2 17 BD E1 22 9D FA 02 CA 10 F7: ED., -J, "J", \,
>0322E0 60 ED 78 8A 2D 11 B0 4D 2E EC 2E 8C 23 6C 33 C9: -L, -, M., ..HL3,
>0322F0 2B A8 4C 9C 23 F0 33 C4 2C E3 3D A0 00 20 3B 23: +, L, #3, =, ;#
>032300 C9 20 F0 F5 C9 3A B0 06 38 E9 30 38 E9 D0 60 A9: P., R., 8/08
>032310 50 80 3A A9 3F 80 36 A9 52 80 32 A9 5C 80 2E A9: P., R., 2/08
>032320 66 80 2A A9 61 80 16 A9 70 80 12 A9 70 80 1E A9: -,*A, /, /, /, /
>032330 50 80 1A A9 61 80 16 A9 24 80 02 A9 3D DB DA AA: P., /, $., .+#
>032340 AB 84 00 20 74 FF FA FB 29 FF 60 A9 24 08 DB DA: [., ., ]n, $., +
>032350 AA AB 85 00 B5 01 C9 20 B0 03 AB 84 00 20 74 FF: [., ., ]n, /, /, /
>032360 FA FB 28 29 FF 60 08 DB 48 AB 85 00 B5 01 C9 20: [., ., ]n, +H, ., /
>032370 B0 03 AB 84 00 68 20 77 FF FB 28 60 A2 3D 08 DB: [., ., ]n, (-, =
>032380 AB 84 00 20 77 FF FB 28 60 6C 04 03 FC 3D 00 20: [., ., ]n, (-, =
>032390 FB 22 80 03 20 F9 22 90 FB 6C 0C 03 93 8E 00 C9: [., ., ]n, (-, =

```

## ENHANCED MACHINE LANGUAGE MONITOR

This machine language monitor is a new development for the MEAG65. It is available both in the 92xxxx ROMs and in the OpenROMs.

The enhanced monitor has following additional features:

### Addresses:

All addresses are used as 32 bit (4 bytes) addresses. This allows access to the whole MEGA65 address range, which needs 28 bit. This is especially useful for the access to the 8MB RAM blocks called attic RAM at \$8000000 (builtin) and cellar RAM at \$8800000 (optional). Setting bit 31 of an address to 1 gives access to a special (banked) configuration. In this case the I/O area at \$D000 and the ROM area \$6000 - \$7FFF (monitor ROM) and \$E000 - \$FFFF (kernal ROM) overlay the current bank.

### Commands:

The additional command **B** displays character bitmaps.

### Disk access:

The disk command character knows two more functions: **U1** for reading a sequence of disk blocks to memory and **U2** for writing a memory range to disk blocks. This enables disk editing, for example modifying directory entries or can be

even used to backup whole floppy contents or disk images. The attic RAM is large enough to hold the contents of 8 complete 1581 floppies.

#### **Disassembler:**

The disassembler can decode all additional address modes, like the 32 bit indirect mode  $[$nn]$ , Z and the compound instructions involving the use of the 32 bit Q register.

#### **Register:**

The register displays the full 16 bit stack pointer and the base page register and accepts new settings for them.

## **Table of MEGA65 Enhanced Monitor Commands**

| C  | mnemonic     | description                                   |
|----|--------------|-----------------------------------------------|
| A  | ASSEMBLE     | Assemble a line of 45GS02 code                |
| B  | BITMAPS      | Display 8x8 bitmaps (characters)              |
| C  | COMPARE      | Compare two sections of memory                |
| D  | DISASSEMBLE  | Disassemble a line of 45GS02 code             |
| F  | FILL         | Fill a section of memory with a value         |
| G  | GO           | Start execution at specified address          |
| H  | HUNT         | Find specified data in a section of memory    |
| L  | LOAD         | Load a file from disk                         |
| M  | MEMORY       | Dump a section of memory                      |
| R  | REGISTERS    | Display the contents of the 45GS02 registers  |
| S  | SAVE         | Save a section of memory to a disk file       |
| T  | TRANSFER     | Transfer memory to another location           |
| V  | VERIFY       | Compare a section of memory with a disk file  |
| X  | EXIT         | Exit Monitor mode                             |
| .  | <period>     | Assembles a line of 45GS02 code               |
| >  | <greater>    | Modifies memory                               |
| ;  | <:semicolon> | Modifies register contents                    |
| @  | <at sign>    | Disk command, directory or status             |
| \$ | <hex>        | Display hex, decimal, octal, and binary value |
| +  | <decimal>    | Display hex, decimal, octal, and binary value |
| &  | <octal>      | Display hex, decimal, octal, and binary value |
| %  | <binary>     | Display hex, decimal, octal, and binary value |

# Calling the Monitor

To enter the monitor from BASIC, type: **MONITOR**

The monitor responds with a display of register contents and waits for a command:

```
MONITOR
DS MONITOR COMMANDS: ABCDFGHJKLMNPXR,;)28+8Z/LSV
PC  SR  AC  XR  YR  ZR  BP  SP  MVEBDIZC
; 00CFA4 35 00 00 00 00 00 01F8 --11-1-1
```

## addresses and numbers

All addresses and numbers must be numbers of base 16 (hex), 10 (decimal), 8 (octal) or 2 (dual). Symbolic names like CHROUT or arithmetics like \$1000+5 are not allowed.

It is an old tradition since the first monitor of the Commodore PET, that the default base is 16. In fact the old monitors would not accept any other numbers, than hexadecimal (short hex). This may confuse beginners, because a statement like

**LDA #10**

loads the decimal value 16 into the accumulator. Later monitors, like that of the Commodore 128 accepted numbers of base 16,10,8 and 2 - like this one, but still used 16 (hex) as default. Additionally the MEGA65 monitor allows character entry, which uses the PETSCII value of the character. Following prefixes can be used to specify the base of the following number:

| base | name        | prefix | digits           | characters | example    |
|------|-------------|--------|------------------|------------|------------|
| 16   | hexadecimal |        | 0123456789ABCDEF |            | 100        |
| 16   | hexadecimal | \$     | 0123456789ABCDEF |            | \$100      |
| 10   | decimal     | +      | 0123456789       |            | +256       |
| 8    | octal       | &      | 01234567         |            | &400       |
| 2    | dual        | %      | 01               |            | %100000000 |
|      | character   | '      | all              |            | 'A         |

## Assembler

The monitor has a builtin mini-assembler, which can be used to write machine language code using the standard mnemonics like **LDA** or **STA**, etc. The most important difference to a full assembler is the necessity to use numeric constants as operands for the instructions

only. So you cannot use named variables, labels or subroutine names. A call to the kernel routine, which prints a character to the screen would be written **JSR CHROUT** in a full assembler, while the mini-assembler needs the syntax **JSR FFD2** (you need to know or lookup the addresses). There is the convenience for branch instructions, that the target address is written to the operand field and the mini-assembler computes the relative address, that is inserted in the code.

The assembler knows all instructions and address modes of the MEGA65 CPU 45GS02 (except the instructions using the Q register, these will be added later). So an instruction like **LDA [TXTPTR],Z** will be assembled as loading the accumulator using a 32 bit pointer at the addresses **TXTPTR, TXTPTR+1, TXTPTR+2, TXTPTR+3**.

## A : ASSEMBLE

**Format:** **A address mnemonic operand**

**Usage:** The mini assembler allows entry of machine language instructions using easy to remember mnemonics instead of opcodes. The operand may be entered as hex, decimal, binary or character. Branch targets are automatically converted to relative distances. After each entered instruction, the mini assembler generates the 1-3 byte long machine code, prints this code along with the instruction and advances the program counter. A new line is generated with the command **A** and the new value of the program counter printed. This eases the fast entry of instructions. The assembly input mode is stopped by pressing RETURN only. Any line of the entered code or a line in disassembly format can be changed by moving the cursor into that line and changing the desired element, for example the mnemonic or the operand. Listed hex values before the mnemonic are ignored.

If the monitor shall be reentered after executing the code, the last instruction must be a **BRK** instruction and the program must be called with I/O and monitor ROM active. This is done by setting the bit 31 of the execution address. If the program was entered in bank 0 on address 1500, it should be started with: **G 80001500**.

If the entered code is a subroutine, it must end with a **RTS** instruction.

**Remarks:** The assembler recognises all 45GS02 instructions of the MEGA65, except the instructions, that use the Q register. These instructions can be entered by typing the NEG NEG prefix explicit. E.g. instead of LDQ \$1234, entering the 3 instructions (on 3 different rows) NEG NEG LDA \$1234 is assembled to the equivalent code.

The screenshot shows the MEGA65 Xemu debugger interface. The title bar reads "MEGA65 - Xemu [100% 3%] running 3.5MHz". The monitor displays assembly code for a "Hello World" program. The code is as follows:

```

READY.
MONITOR
BS-MONITOR COMMANDS: A B C D E G H I M R U X C P Z S ? / ! L S U
PC  SR  AC  XR  YR  ZR  BP  SP  NVEBDI2C
: 00CFAF 00 00 00 00 00 00 00 01F8 -----
M 1500 1500
D1500 00 48 45 4C 4C 4F 20 57 4F 52 4C 44 0D 00 00 00 .HELLO WORLD.....
A 1510 A0 00 LDY #$00
A 1512 B9 00 15 LDA $1500,Y
A 1515 F0 06 BEQ $151D
A 1517 20 D2 FF JSR $FFD2
A 151A C8 INY
A 151B 80 F5 BRA $1512
A 151D 00 BRK
A 151E
G 80001510
HELLO WORLD

BREAK
BS-MONITOR COMMANDS: A B C D E G H I M R U X C P Z S ? / ! L S U
PC  SR  AC  XR  YR  ZR  BP  SP  NVEBDI2C
: 03151E 02 00 00 00 00 00 00 01F5 -----1-

```

## B : BITMAPS

**Format:** B display character bitmaps

**Usage:** B address

**Remarks:** The B command displays the contents of memory cells bitwise by printing an asterisk for 1 and a dot for 0. The special arrangement of character data with 8 bytes forming one character cell, is considered. 8 characters are displayed for each call.

There are three ROM character sets builtin in the 92xxxx ROMs:

|                                             |
|---------------------------------------------|
| FONT A : \$029000 : ASCII [£]† { } included |
| FONT B : \$03D000 : serif version of A      |
| FONT C : \$02D000 : original C64 font       |



The screenshot shows a terminal window titled "MEGA65 - Xemu [100% 8%] running 3.5MHz". The window displays a memory dump starting at address 020000. The first few lines of the dump are:

```
BS MONITOR COMMANDS: ABCODE1GHJMRUXC>25:87'LSV
PC SR AC XR VR ZR BP SP NUEBDIZC
: 00FFA2 00 00 00 00 00 00 01F8 -----
B 2D000
02D000 .***** .** .***** .**** .*** .***** .***** .*****
02D001 ** .** .**** .** .** .** .** .** .** .** .** .** .**
02D002 ** .** .** .** .** .** .** .** .** .** .** .** .**
02D003 ** .** .**** .**** .** .** .** .** .** .** .** .** .**
02D004 ** .** .** .** .** .** .** .** .** .** .** .** .**
02D005 ** .** .** .** .** .** .** .** .** .** .** .** .**
02D006 .***** .** .** .***** .**** .***** .***** .** .*****
02D007 .....
```

## MEGA65 MATRIX MODE MONITOR INTERFACE

This monitor is different to the other two: It is part of the MEGA65 system itself, and runs concurrently with MEGA65's processor. That is, you can view and modify the memory the MEGA65, *while a programme is running*.

This works using dedicated hardware in the MEGA65 design, that implements a little helper processor that runs this monitor interface, and has a special access mechanism to the memory and processor of the MEGA65.

In comparison with the ROM-based monitors that execute on the MEGA65's primary processor, the matrix mode monitor has several advantages and disadvantages:

- It can be used while a programme is running.
- It can be used, even if the ROM area is being used for programme code or data, instead of containing a standard C65 or MEGA65 ROM.

- It can be accessed via the serial debug interface, via the JB1 connector.
- It can be instructed to stop the processor as soon as the programme counter (PC) register of the main processor reaches a user specified address. That is, it supports a (single) hardware breakpoint.
- It can be instructed to stop the processor whenever a specified memory address is written to. That is, it supports a “write watch” on a single memory address. The memory address is specified as a full 28-bit address, allowing it to detect memory writes via any means. Note that DMA operations will complete, before the watch point takes effect.
- It can be instructed to stop the processor whenever specific CPU flags are set or cleared, which can also be used to support debugging of programmes.
- On some models of the MEGA65, the integrated ROM of the monitor processor is very small, which means that functionality may be limited. This is why, for example, there is no “assemble” command for this monitor. This may be corrected in future core updates for MEGA65 models that have capacity for a larger monitor processor ROM.

# Table of Matrix Mode Monitor Commands

| C | mnemonic    | description                                    |
|---|-------------|------------------------------------------------|
| # | HYPERTRAP   | Enable/disable CPU hypervisor traps            |
| + | UARTDIVISOR | Set UART bitrate divisor                       |
| ? | HELP        | help                                           |
| @ | CPUMEM      | (show memory from CPU context)                 |
| B | BREAKPOINT  | Set/clear CPU execution break point            |
| D | DISASSEMBLE | Disassemble memory                             |
| E | FLAGWATCH   | Set/clear CPU flags watch point                |
| F | FILL        | Fill memory with a value                       |
| G | SETPC       | Set CPU program counter                        |
| H | HELP        | help                                           |
| I | INTERRUPTS  | Enable/disable CPU interrupts                  |
| J | DEBUGMON    | Various debug functions for the monitor itself |
| L | LOADMEMORY  | Load data into memory                          |
| M | MEMORY      | Show memory contents                           |
| R | REGISTERS   | show registers                                 |
| S | SETMEMORY   | Set memory contents                            |
| T | TRACE       | set CPU trace/run mode                         |
| W | MEMORYWATCH | Set/clear memory write watch point             |
| Z | CPUHISTORY  | CPU history                                    |

## Calling the Monitor

To enter or exit the monitor hold down the  and press the  key. You will see an animation of green characters raining down from the top of the screen, and then be presented with a simple text terminal interface which is transparent, so that you can see the screen output of your running programme at the same time.

## # : Hypervisor trap enable/disable

**Format:** # Enable or disable Hypervisor Traps

**Usage:** #[0|1]

**Remarks:** If the argument is 1, then Hypervisor traps are enabled, otherwise they are disabled.

*It is not confirmed if this command is currently functional.*

## + : Set Serial Interface UART Divisor

**Format:** + Set Serial Interface UART Divisor

**Usage:** + divisor

**Remarks:** Sets the divisor for the serial monitor interface. This allows changing the baud rate of the serial monitor interface from the default 2,000,000 bits per second. The baud rate will be equal to  $40,500,000 / \text{divisor} + 1$ . This affects only the serial UART interface, and does not affect accessing this monitor via the Matrix Mode composited display.

For example, to slow the serial monitor interface down to 19,200 bits per second, the divisor would need to be  $40,500,000 / 19,200 - 1 = 2108$ . The + command then requires that you convert this value to hexadecimal, thus the command would be +83c.

Note that this command does *no* sanity checking of the provided value. If you accidentally provide an incorrect value for your needs, you can recover from this situation by activating the Matrix Mode interface by holding down the  and tapping the  key, and entering the appropriate command to correct the divisor, e.g., +14 to return to the default of 2,000,000 bits per second.

You must then exit the Matrix Mode again by repeating the  +  key combination, before the serial UART interface will become active again. This is because the Matrix Mode disables the serial UART interface when active.

## @ : CPUMEMORY

**Format:** @ [address]

**Usage:** Prints a memory dump for the given 16-bit address, as interpreted by the current CPU memory mapping. If you wish to inspect the contents of memory anywhere in the 28-bit address space, use the M command instead.

The dump displays memory contents, organised in rows of 16 consecutive addresses starting with the address. The dump displays a full page of 256 bytes in 16 rows. The contents are printed as 16 byte values in hex, followed by the character representation.

**Remarks:** If no address is provided, it will show the next 256 bytes.

## ? or H : HELP

**Format:** `? or h`

**Usage:** Displays a (very) brief message identifying the monitor. On some models of the MEGA65 that have more memory available to the monitor processor, this command may display information about each of the available commands.

## B : BREAKPOINT

**Format:** `b [address]`

**Usage:** Sets or clears the hardware breakpoint. If no address is provided, then the breakpoint will be disabled. Otherwise the breakpoint is set to the provided 16-bit address.

Whenever the programme counter (PC) register of the MEGA65's processor equals the value provided to this command, the processor will halt, and the Matrix Mode monitor interface will display the last instruction executed and current register values to alert the user to this event. It does not activate the Matrix Mode display when this occurs. It is normally expected that Matrix Mode will either already be active, or that the user is interacting via the serial interface.

## D : DISASSEMBLE

**Format:** `<d|D> [address]`

**Usage:** Disassembles and displays the instruction stored at the indicated 28-bit address.

To disassemble instructions from the CPU's current memory context, taking into account current memory banking, prefix the address with 777, e.g., d777080D would disassemble the instruction at \$080D, as currently visible to the MEGA65's processor.

Use D instead of d to disassemble 16 instructions at a time, instead of just one.

## E : FLAGWATCH

**Format:** `e [value]`

**Usage:** Sets or clears the CPU flag watch point: If no argument is provided, the flag watch point is disabled. If a value is provided, it is assumed to be a 16-bit value, where the first two hexadecimal digits indicate the processor flags that will trigger the watch point if they are set. The second two hexadecimal digits indicate which processor flags will trigger the watch point if they are clear. In this way any combination of processor flag values can be monitored.

*This command does not function correctly at the time of writing.*

**Example:** To cause the watch point to trigger when the Negative Flag is asserted, the command e8000 would be used.

## F : FILL

**Format:** f [start] [end+1] [value]

**Usage:** Fills the indicated 28-bit address range with the indicated value.

**Remarks:** The end address should be one more than the last address that is desired to be filled.

## G : SETPC

**Format:** g address

**Usage:** Sets the Programme Counter (PC) register of the MEGA65's processor to the supplied 16-bit address. If the processor is running at the time, execution will immediately proceed from that address. If the processor is halted at the time, e.g., due to the use of the tt t1 command, the processor remains halted, but with the Programme Counter set to the indicated address, ready for when the processor is again allowed to run.

## I : INTERRUPTS

**Format:** i[0|1]

**Usage:** Enables or disables interrupts on the MEGA65's processor. Disabling interrupts can be helpful when single-stepping through a programme, as otherwise you will tend to end up only stepping through the interrupt handler code,

because the interrupts will happen more frequently than the steps through the code.

**Remarks:** *This command is known to have problems, and may not currently function.*

## J : DEBUGMON

**Format:** **j [value]**

**Usage:** Display, and optionally set, internal signals of the matrix mode monitor interface.

## L : LOADMEMORY

**Format:** **l <start addr> <end addr + 1>**

**Usage:** Fast-load a block of memory via the serial monitor interface. Immediately after sending this command, the bytes of memory to be loaded should be sent to the serial monitor interface. The bytes are read as-is, and thus should be provided as natural bytes, not encoded in hexadecimal. This allows loading data at approximately 200KiB per second at the default serial baud rate of 2,000,000 bits per second.

## M : MEMORY

**Format:** **<m|M> [address]**

**Usage:** Prints a memory dump for the given 28-bit address. If you wish to inspect the contents of memory as currently seen by the processor's current banking configuration, use the `m` command instead.

The dump displays memory contents, organised in rows of 16 consecutive addresses starting with the address. The dump displays a full page of 256 bytes in 16 rows. The contents are printed as 16 byte values in hex, followed by the character representation.

**Remarks:** If no address is provided, it will show the next 256 bytes.

## R : REGISTERS

**Format:** **r**

**Usage:** Displays the current value of various processor registers and flags, as well as a disassembly of the most recently executed instruction.

## S : SETMEMORY

**Format:** `s addr <value ...>`

**Usage:** Sets the contents of the indicated memory location to the supplied value. If more than one space-separated value is provided, then multiple consecutive memory locations will be set.

This command uses 28-bit addresses, and therefore ignores the current selected memory banking configuration.

## T : TRACE

**Format:** `t<0|1|c>`

**Usage:** Selects the trace or run mode of the processor: t0 means that the processor runs freely, t1 halts the processor, and tc runs the processor in continuous-trace mode, where it displays each instruction and the register values immediately following its execution, as though t1 had been selected, and the user were to then immediately press return or enter to request the next instruction to be executed.

If t1 is selected, pressing enter or return in the matrix mode monitor will cause the next instruction to be executed.

The t0 command is also used following the triggering of a break-point or watch-point, to allow the processor to resume.

## W : WATCHPOINT

**Format:** `w [address]`

**Usage:** Sets or clears the hardware watch-point. If no address is provided, then the watch-point will be disabled. Otherwise the watch-point is set to the provided 28-bit address.

Whenever the MEGA65's processor writes to the address provided to this command, the processor will halt, and the Matrix Mode monitor interface will display the last instruction executed and current register values to alert the user to this event. It does not activate the Matrix Mode display when

this occurs. It is normally expected that Matrix Mode will either already be active, or that the user is interacting via the serial interface.

## Z : CPUHISTORY

**Format:** z [address]

**Usage:** Displays information about the instructions recently executed by the MEGA65's processor.

**Remarks:** *This command is suspected to not be correctly operational at the time of writing.*

## APPENDIX

# F018-Compatible Direct Memory Access (DMA) Controller

- F018A/B DMA Jobs
- MEGA65 Enhanced DMA Jobs
- Texture Scaling and Line Drawing
- Audio DMA
- F018 “DMAgic” DMA Controller
- MEGA65 DMA Controller Extensions
- Unimplemented Functionality



The MEGA65 includes an F018/F018A backward-compatible DMA controller. Unlike in the C65, where the DMA controller exists as a separate chip, it is part of the 45GS02 processor in the MEGA65. However, as the use of the DMA controller is a logically separate topic, it is documented separately in this appendix.

The MEGA65's DMA controller provides several important improvements over the F018/F018A DMAagic chips of the C65:

- **Speed** The MEGA65 performs DMA operations at 40MHz, allowing filling 40MiB or copying 20MiB per second. For example, it is possible to copy a complete 8KiB C64-style bitmap display in about 200 micro-seconds, equivalent to less than four raster lines!
- **Large Memory Access** The MEGA65's DMA controller allows access to all 256MiB of address space.
- **Texture Copying Support** The MEGA65's DMA controller can do fractional address calculations to support hardware texture scaling, as well as address striding, to make it possible in principle to simultaneously scale-and-draw a texture from memory to the screen. This would be useful, should anyone be crazy enough to try to implement a Wolfenstein or Doom style-game on the MEGA65.
- **Transparency/Mask Value Support** The MEGA65's DMA controller can be told to ignore a special value when copying memory, leaving the destination memory contents unchanged. This allows masking of transparent regions when performing a DMA copy, which considerably simplifies blitting of graphics shapes.
- **Per-Job Option List** A number of options can be configured for each job in a chained list of DMA jobs, for example, selecting F018 or F018B mode, changing the transparency value, fractional address stepping or the source or destination memory region.
- **Background Audio DMA** The MEGA65 includes background audio DMA capabilities similar to the Amiga™series of computers. Key differences are that the MEGA65 can use either 8 or 16 bit samples, supports very high sample rates up to approximately 1 MHz, has 256 volume settings per channel, and no inter-channel modulation.

## F018A/B DMA JOBS

To execute a DMA job using the F018 series of DMA controllers, you must construct the list of DMA jobs in memory, and then write the address of this list into the DMA address registers. The DMA job will execute when you write to the ADDRLSBTRIG register (\$D700). For this reason you must write the MSB and bank number of the DMA list into \$D701 and \$D702 first, and the LSB only after having set these other two registers. If you wish

to execute multiple DMA jobs using the same list structure in memory, you can simply write to ADDRLSBTRIG again after updating the list contents – provided that no other programme has modified the contents of \$D701 or \$D702. Note that BASIC 65 uses the DMA controller to scroll the screen, so it is usually safest to always write to all three registers.

When ADDRLSBTRIG has been written to, the DMA job completes immediately. Unlike on the C65, the DMA controller is part of the processor of the MEGA65. This means that the processor stops trying to execute instructions or *fetching audio samples for DMA audio playback* until the DMA job has completed. It also means that, unlike on the C65, DMA jobs cannot be interrupted. If your programme has sensitive timing requirements, you may need to break larger DMA jobs into several smaller jobs. This is somewhat mitigated by the high speed of the MEGA65's DMA, which is able to fill memory at 40.5MiB per second and copy memory at 20.25MiB per second, compared with circa 3.5 MiB and 1.7 MiB per second on a C65. This allows larger DMA jobs to be executed, without needing to worry about the impact on real-time elements of a programme. For example, it is possible to fill an 80 column 50 row text screen using the MEGA65's DMA controller in just 200 microseconds.

## F018 DMA Job List Format

The MEGA65's DMA controller supports the two different DMA job list formats used by the original F018 part that was used in the earlier C65 prototypes (upto Revision 2B) and the F018B and later revisions used in the Revision 3 – 5 C65 prototypes. The main difference is the addition of a second command byte, as the following tables show:

It is important to know which style the DMA controller is expecting. The MEGA65's Hypervisor sets the mode based on the detected version of C65 ROM, if one is running. If it is an older one, then the F018 style is expected, otherwise the newer F018B style is expected. You can check which style has been selected by querying bit 0 of \$D703: If it is a 1, then the newer F018B 12-byte list format is expected. If it is a 0, then the older F018 11-byte list format is expected. The expected style can be set by writing to this register.

Unless you are writing software that must also run on a C65 prototype, you should most probably use the MEGA65's Enhanced DMA Jobs, where the list format is explicitly specified in the list itself. As the Enhanced DMA Jobs are an extension of the F018/F018B DMA jobs, you should still read the following, unless you are already familiar with the behaviour of the F018 DMA controller.

## F018 11-byte DMA List Structure

| Offset | Contents                           |
|--------|------------------------------------|
| \$00   | Command LSB                        |
| \$01   | Count LSB                          |
| \$02   | Count MSB                          |
| \$03   | Source Address LSB                 |
| \$04   | Source Address MSB                 |
| \$05   | Source Address BANK and FLAGS      |
| \$06   | Destination Address LSB            |
| \$07   | Destination Address MSB            |
| \$08   | Destination Address BANK and FLAGS |
| \$09   | Modulo LSB                         |
| \$0a   | Modulo MSB                         |

\* The Command MSB is \$00 when using this list format.

## F018B 12-byte DMA List Structure

| Offset | Contents                           |
|--------|------------------------------------|
| \$00   | Command LSB                        |
| \$01   | Count LSB                          |
| \$02   | Count MSB                          |
| \$03   | Source Address LSB                 |
| \$04   | Source Address MSB                 |
| \$05   | Source Address BANK and FLAGS      |
| \$06   | Destination Address LSB            |
| \$07   | Destination Address MSB            |
| \$08   | Destination Address BANK and FLAGS |
| \$09   | Command MSB                        |
| \$0a   | Modulo LSB / Mode                  |
| \$0b   | Modulo MSB / Mode                  |

The structure of the command word is as follows:

| Bit(s)  | Contents                               |
|---------|----------------------------------------|
| 0 - 1   | DMA Operation Type                     |
| 2       | Chain (i.e., another DMA list follows) |
| 3       | Yield to interrupts                    |
| 4       | MINTERM -SA,-DA bit                    |
| 5       | MINTERM -SA,DA bit                     |
| 6       | MINTERM SA,-DA bit                     |
| 7       | MINTERM SA,DA bit                      |
| 8 - 9   | Addressing mode of source              |
| 10 - 11 | Addressing mode of destination         |
| 12 - 15 | RESERVED. Always set to 0's            |

The command field take the following four values:

| Value   | Contents           |
|---------|--------------------|
| %00 (0) | Copy               |
| %01 (1) | Mix (via MINTERMs) |
| %10 (2) | Swap               |
| %11 (3) | Fill               |

\* Only Copy and Fill are implemented at the time of writing.

The addressing mode fields take the following four values:

| Value   | Contents                               |
|---------|----------------------------------------|
| %00 (0) | Linear (normal) addressing             |
| %01 (1) | Modulo (rectangular) addressing        |
| %10 (2) | Hold (constant address)                |
| %11 (3) | XY MOD (bitmap rectangular) addressing |

\* Only Linear, Modulo and Hold are implemented at the time of writing.

The BANK and FLAGS field for the source address allow selection of addresses within a 1MB address space. To access memory beyond the first 1MB, it is necessary to use an Enhanced DMA Job with the appropriate option bytes to select the source and/or destination MB of memory. The BLANKS and FLAGS field has the following structure:

| Bit(s) | Contents                                                                                 |
|--------|------------------------------------------------------------------------------------------|
| 0 - 3  | Memory BANK within the selected MB                                                       |
| 4      | HOLD, i.e., do not change the address                                                    |
| 5      | MODULO, i.e., apply the MODULO field to wrap-around within a limited memory space        |
| 6      | DIRECTION. If set, then the address is decremented instead of incremented.               |
| 7      | I/O. If set, then IO registers are visible during the DMA controller at \$D000 - \$DFFF. |

# Performing Simple DMA Operations

For information on using the DMA controller from BASIC 65, refer to the **DMA** BASIC command in Chapter/Appendix B on page B-59.

To use the DMA controller from assembly language, set up a data structure with the DMA list, and then set \$D702 - \$D700 to the address of the list. For example, to clear the screen in C65 mode by filling it with spaces, the following routine could be used:

```
LDA #$00      ; DMA list exists in BANK 0
STA $D702
LDA #<dmalist ; Set MSB of DMA list address
STA $D701
LDA #>dmalist ; Set LSB of DMA list address, and execute DMA
STA $D700
RTS

dmalist:
.byte $03    ; Command low byte: FILL
.word 2000   ; Count: 80x25 = 2000 bytes
.word $0020  ; Fill with value $20
.byte $00    ; Source bank (ignored with FILL operation)
.word $0800  ; Destination address where screen lives
.byte $00    ; Screen is in bank 0
.byte $00    ; Command high byte
.word $0000  ; Modulo (ignored due to selected command)
```

It is also possible to execute more than one DMA job at the same time, by setting the CHAIN bit in the low-byte of the command word. For example to clear the screen as above, and also clear the colour RAM for the screen, you could use something like:

```

LDA #$00      ; DMA list exists in BANK 0
STA $D702
LDA #>dmalist ; Set MSB of DMA list address
STA $D701
LDA #(dmalist ; Set LSB of DMA list address, and execute DMA
STA $D700
RTS

dmalist:
.byte $04    ; Command low byte: FILL + CHAIN
.word 2000   ; Count: 80x25 = 2000 bytes
.word $0020   ; Fill with value $20
.byte $00    ; Source bank (ignored with FILL operation)
.word $0800   ; Destination address where screen lives
.byte $00    ; Screen is in bank 0
.byte $00    ; Command high byte
.word $0000   ; Modulo (ignored due to selected command)

; Second DMA job immediately follows the first
.byte $03    ; Command low byte: FILL
.word 2000   ; Count: 80x25 = 2000 bytes
.word $0001   ; Fill with value $01 = white
.byte $00    ; Source bank (ignored with FILL operation)
.word $F800   ; Destination address where colour RAM lives
.byte $01    ; colour RAM is in bank 1 ($1F800-$1FFFF)
.byte $00    ; Command high byte
.word $0000   ; Modulo (ignored due to selected command)

```

Copying memory is very similar to filling memory, except that the command low byte must be modified, and the source address field must be correctly initialised. For example, to copy the character set from where it lives in the ROM at \$2D000 - \$2DFFF to \$5000, you could use something like:

```

LDA #$00      ; DMA list exists in BANK 0
STA $D702
LDA #>dmalist ; Set MSB of DMA list address
STA $D701
LDA #<dmalist ; Set LSB of DMA list address, and execute DMA
STA $D700
RTS

dmalist:
.byte $00    ; Command low byte: COPY
.word $1000  ; Count: 4KB = 4096
.word $D000  ; Copy from $x0000
.byte $02    ; Source bank = $02 for $2xxxx
.word $5000  ; Destination address where screen lives
.byte $00    ; Screen is in bank 0
.byte $00    ; Command high byte
.word $0000  ; Modulo (ignored due to selected command)

```

It is also possible to perform a DMA operation from BASIC 2 in C64 mode by POKEing the necessary values, after first making sure that MEGA65 or C65 IO mode has been selected by writing the appropriate values to \$D02F (53295). For example, to clear the screen in C64 BASIC 2 using the DMA controller, you could use something like:

```

10 rem enable mega65 io
20 poke53295,asc("g"):poke53295,asc("s")
30 rem dma list in data statements
40 data 3: rem command lsb = fill
50 data 232,3 : rem screen is 1000 bytes = 3*256+232
60 data 32,0: rem fill with space = 32
70 data 0: rem source bank (unused for fill)
80 data 0,4: rem screen address = 1024 = 4*256
90 data 0: rem screen lives in bank 0
100 data 0: rem command high byte
110 data 0,0: rem modulo (unused in this job)
120 rem put dma list at $c000 = 49152
130 fori=0to11:reada:poke49152+i,a:next
140 rem execute job
150 poke55042,0: rem dma list is in bank 0
160 poke55041,192: rem dma list is in $0xx
170 poke55040,0: rem dma list is in $x00, and execute

```

While this is rather cumbersome to do each time, if you wanted to clear the screen again, all you would need to do would be to **POKE 55040,0** again, assuming that the DMA list and DMA controller registers had not been modified since the previous time the DMA job had been run.

The HOLD, IO and other options can also be used to create interesting effects. For example, to write a new value to the screen background colour very quickly, you could copy a region of memory to \$D021, with the IO flag set to make the IO register visible for writing in the DMA job, and the HOLD flag set, so that the same address gets written to repeatedly. This will write to the background colour at a rate of 20.5MHz, which is almost as fast as the video pixel clock (27MHz). Thus we can change the colour almost every pixel.

With a little care, we can make this routine such that it takes exactly one raster-line to run, and thus draw vertical raster bars, or to create a kind of frankenstein video mode that uses a linear memory layout - at the cost of consuming all of the processor's time during the active part of the display.

The following example does this to draw vertical raster bars on the screen. This programme assumes that the MEGA65 is set to PAL. For NTSC, the size of the DMA transfer would need to be decreased a little. The other thing to note with this programme, is that it uses MEGA65 Enhanced DMA Job option \$81 to set the destination mega-byte in memory to \$FFxxxx, and the bank is set to \$D, and the destination address to \$0021, to form the complete address \$FFD0021. This is the true location of the VIC-IV's border colour register. The programme is written using ACME-compatible syntax.

```
basicheader:  
;; 2020 SVS 2061  
!word $80a,2020  
!byte $9e,$32,$30,$36,$31,0,0,0  
  
;; Actual code begining at $000d = 2061  
Main:  
    sei  
    lda #$47      ; enable MEGA65 IO  
    sta $D02f  
    lda #$53  
    sta $d02f  
    lda #65      ; Set CPU speed to fast  
    sta 0  
    lda #0      ; disable screen to show only the border  
    sta $d011  
  
    lda $d012      ; Wait until start of the next raster  
rasterSync:   ; before beginning loop for horizontal alignment  
    cmp $d012  
    beq rasterSync  
  
;; The following loop takes exactly one raster line at 40.5MHz in PAL  
loop:  
    jsr triggerdma  
    jmp loop  
  
triggerdma:  
    lda #0 ; Make sure F010 list format  
    sta $d703  
  
    lda #0      ; dma list bank  
    sta $d702  
    lda #>rasterdmaList  
    sta $d701  
    lda #<rasterdmaList  
    sta $d705  
    rts
```

```

rasterdmaelist:
!byte $81,$ff,$00
!byte $00 ; COPY
!word 619 ; DMA transfer is 619 bytes long
!word rastercolours ; source address
!byte $00           ; source bank
!word $0020 ; destination address
!byte $1d           ; destination bank + HOLD
;; unused modulo field
!word $0000

```

```

rastercolours:
!byte 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
!byte 0,0,0,11,11,11,12,12,12,15,15,15,1,1,1,15,15,15,12,12,12,11,11,11,0,0,0
!byte 0,0,0,6,6,6,4,4,4,14,14,14,3,3,3,1,1,1,3,3,3,14,14,14,4,4,4,6,6,6,0,0,0
!byte 0,0,0,11,11,11,12,12,12,15,15,15,1,1,1,15,15,15,12,12,12,11,11,11,0,0,0
!byte 0,0,0,6,6,6,4,4,4,14,14,14,3,3,3,1,1,1,3,3,3,14,14,14,14,4,4,4,6,6,6,0,0,0

```

## MEGA65 ENHANCED DMA JOBS

The MEGA65's implementation of the DMAagic supports significantly enhanced DMA jobs. An enhanced DMA job is indicated by writing the low-byte of the DMA list address to \$D705 instead of to \$D700. The MEGA65 will then look for one or more *job option tokens* at the start of the DMA list. Those tokens will be interpreted, before executing the DMA job which immediately follows the *end of job options* token (\$00).

Job option tokens that take an argument have the most-significant bit set, and always take a 1 byte option. Job option tokens that take no argument have the most-significant-bit clear. Unsupported job option tokens are simply ignored. This allows for future revisions of the DMAgic to add support for additional options, without breaking backward compatibility.

These options are also used to achieve advanced features, such as hardware texture scaling at up to 20Mpixels per second, and hardware line drawing at up to 40Mpixels per second. These advanced functions are implemented by allowing complex calculations to be made to the source and/or destination address of DMA jobs as they execute.

The list of valid job option tokens is:

|      |                                                                                                                                                                            |
|------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| \$00 | End of job option list                                                                                                                                                     |
| \$06 | Disable use of transparent value                                                                                                                                           |
| \$07 | Enable use of transparent value                                                                                                                                            |
| \$0A | Use 11-byte F011A DMA list format                                                                                                                                          |
| \$0B | Use 12-byte F011B DMA list format                                                                                                                                          |
| \$53 | Enable 'Shallan Spiral' Mode                                                                                                                                               |
| \$80 | Source address bits 20 - 27                                                                                                                                                |
| \$81 | Destination address bits 20 - 27                                                                                                                                           |
| \$82 | Source skip rate (256 <sup>ths</sup> of bytes)                                                                                                                             |
| \$83 | Source skip rate (whole bytes)                                                                                                                                             |
| \$84 | Destination skip rate (256 <sup>ths</sup> of bytes)                                                                                                                        |
| \$85 | Destination skip rate (whole bytes)                                                                                                                                        |
| \$86 | Transparent value (bytes with matching value are not written)                                                                                                              |
| \$87 | Set X column bytes (LSB) for line drawing destination address                                                                                                              |
| \$88 | Set X column bytes (MSB) for line drawing destination address                                                                                                              |
| \$89 | Set Y row bytes (LSB) for line drawing destination address                                                                                                                 |
| \$8A | Set Y row bytes (MSB) for line drawing destination address                                                                                                                 |
| \$8B | Slope (LSB) for line drawing destination address                                                                                                                           |
| \$8C | Slope (MSB) for line drawing destination address                                                                                                                           |
| \$8D | Slope accumulator initial fraction (LSB) for line drawing destination address                                                                                              |
| \$8E | Slope accumulator initial fraction (MSB) for line drawing destination address                                                                                              |
| \$8F | Line Drawing Mode enable and options for destination address (set in argument byte): Bit 7 = enable line mode, Bit 6 = select X or Y direction, Bit 5 = slope is negative. |
| \$97 | Set X column bytes (LSB) for line drawing source address                                                                                                                   |
| \$98 | Set X column bytes (MSB) for line drawing source address                                                                                                                   |
| \$99 | Set Y row bytes (LSB) for line drawing source address                                                                                                                      |
| \$9A | Set Y row bytes (MSB) for line drawing source address                                                                                                                      |
| \$9B | Slope (LSB) for line drawing source address                                                                                                                                |
| \$9C | Slope (MSB) for line drawing source address                                                                                                                                |
| \$9D | Slope accumulator initial fraction (LSB) for line drawing source address                                                                                                   |
| \$9E | Slope accumulator initial fraction (MSB) for line drawing source address                                                                                                   |
| \$9F | Line Drawing Mode enable and options for source address (set in argument byte): Bit 7 = enable line mode, Bit 6 = select X or Y direction, Bit 5 = slope is negative.      |

# TEXTURE SCALING AND LINE DRAWING

The DMAgic supports an advanced internal address calculator that allows it to draw scaled textures and draw lines with arbitrary slopes on VIC-IV FCM video displays.

For texture scaling, the FCM screen must be arranged vertically, as shown below:



By lining the characters into vertical columns like this, advancing vertically by one pixel adds a constant 8 bytes each time, as shown below:



The source and destination skip rates also allow setting the scaling factors. A skip rate of \$0100 this corresponds to stepping \$01.00 pixels. To use the vertically stacked FCM layout as the target for copying vertical lines of textures, then the destination skip rate should be \$0800, i.e., 8.0 bytes per pixel. This would copy a vertical line of texture data without scaling. By setting the source stepping to < \$0100 will cause some pixels to be repeated, effectively zooming the texture in, while setting the source stepping to > \$0100 will cause some pixels to be skipped, effectively zooming the texture out. The destination stepping does not ordinary need to be adjusted. Note that the texture data must be stored with each vertical stripe stored contiguously, so that this mode can be used.

For line drawing, the DMA controller needs to know the screen layout, specifically, what number must be added to the address of a rightmost pixel in one column of FCM char-

acters in order to calculate the address of the pixel appearing immediately to its right. Similarly, it must also know how much must be added to the address of a bottom most pixel in one row of FCM characters in order to calculate the address of the pixel appearing immediately below it. This allows for flexible screen layout options, and arbitrary screen sizes. You must then also specify the slope of the line, and whether the line has the X or Y as its major axis, and whether the slope is positive or negative.

The file `test_290.c` in the <https://github.com/mega65/mega65-tools> repository provides an example of using these facilities to implement hardware accelerated line drawing. This is *very* fast, as it draws lines at the full DMA fill speed, i.e., approximately 40,500,000 pixels per second.

## AUDIO DMA

The MEGA65 includes four channels of DMA-driven audio playback that can be used in place of the direct digital audio registers at \$D6F8-\$D6FB. That is, you must select which of these two sources to feed to the audio cross-bar mixer. This is selected via the AUDEN signal (\$D711 bit 7), which simultaneously enables the audio DMA function in the processor, as well as instructing the audio cross-bar mixer to use the audio from this instead of the \$D6F8-\$D6FB digital audio registers. If you wish to have no other audio than the audio DMA channels, the audio cross-bar mixer can be bypassed, and the DMA audio played at full volume by setting the NOMIX signal (\$D711 bit 4). In that mode no audio from the SIDs, FM, microphones or other sources will be available. All other bits in \$D711 should ordinarily be left clear, i.e., write \$80 to \$D711 to enable audio DMA.

Two channels form the left digital audio channel, and the other two channels form the right digital audio channel. It is these left and right channels that are then fed into the MEGA65's audio cross-bar mixer.

As the DMA controller is part of the processor of the MEGA65, and the MEGA65 does not have reserved bus slots for multi-media operations, the MEGA65 uses idle CPU cycles to perform background DMA. This requires that the MEGA65 CPU be set to the "full speed" mode, i.e., approximately 40MHz. In this mode, there is a wait-state whenever reading an operand from memory. Thus each instruction that loads a byte from memory will create one implicit audio DMA slot. This is rarely a problem in practice, except if the processor idles in a very tight loop. To ensure that audio continues to play in the background, such loops should include a read instruction, such as:

```
loop: LDA $1234 // Ensure loop has at least one idle cycle for
           // audio DMA
    JMP loop
```

Each of the four DMA channels is configured using a block of 16 registers at \$D720, \$D730, \$D740 and \$D750, respectively. We will explain the registers for the first channel, channel 0, at \$D720 - \$D72F.

## Sample Address Management

To play an audio sample you must first supply the start address of the sample. This is a 24-bit address, and must be in the main chip memory of the MEGA65. This is done by writing the address into \$D72A - \$D72C. This is the address of the first sample value that will be played. You must then provide the end address of the sample in \$D727 - \$D728. But note that this is only 16 bits. This is because the MEGA65 compares only the bottom 16 bits of the address when checking if it has reached the end of a sample. In practice, this means that samples cannot be more than 64KB in size. If the sample contains a section that should be repeated, then the start address of the repeating part should be loaded into \$D721 - \$D723, and the CH0LOOP bit should be set (\$D720 bit 6).

You can determine the current sample address at any time by reading the registers at \$D72A - \$D72C. But beware: These registers are not latched, so it is possible that the values may be updated as you read the registers, unless you stop the channel first by clearing the CH0EN signal.

## Sample Playback frequency and Volume

The MEGA65 controls the playback rate of audio DMA samples by using a 24-bit counter. Whenever the 24-bit counter overflows, the next sample is requested. Sample speed control is achieved by setting the value added to this counter each CPU cycle. Thus a value of \$FFFFFF would result in a sample rate of almost 40.5 MHz. In practice, sample rates above a few megahertz are not possible, because there are insufficient idle CPU cycles, and distorted audio will result. Even below this, care must be taken to ensure that idle cycles come sufficiently often and dispersed throughout the processor's instruction stream to prevent distortion. At typical sample rates below 16KHz and using 8 bit samples these effects are typically negligible for normal instruction streams, and so no special action is normally required for typical audio playback.

At the other end of the scale, sample rates as low as  $40.5\text{MHz}/2^{24} = 2.4$  samples per second are possible. This is sufficiently low enough for even the most demanding infra-sound applications.

Volume is controlled by setting \$D729. Maximum volume is obtained with the value \$FF, while a value of \$00 will effectively mute the channel. The first two audio channels are

normally allocated to the left, and the second two to the right. However, the MEGA65 includes separate volume controls for the opposite channels. For example, to play audio DMA channel 0 at full volume on both left and right sides of the audio output, set both \$D729 and \$D71C to \$FF. This allows panning of the four audio DMA channels.

Both the frequency and volume can be freely adjusted while a sample is playing to produce various effects.

## Pure Sine Wave

Where it is necessary to produce a stable sine wave, especially at higher frequencies, there is a special mode to support this. By setting the CH0SINE signal, the audio channel will play a 32-byte 16-bit sine wave pattern. The sample addresses still need to be set, as though the sine wave table were located in the bottom 64 bytes of memory, as the normal address generation logic is used in this mode. However, no audio DMA fetches are performed when a channel is in this mode, thus avoiding all sources of distortion due to irregular spacing of idle cycles in the processor's instruction stream.

This can be used to produce sine waves in both the audible range, as well as well into the ultrasonic range, at frequencies exceeding 60,000Hz, provided that the MEGA65 is connected to an appropriately speaker arrangement.

## Sample playback control

To begin a channel playing a sample, set the CH0EN signal (\$D720 bit 7). The sample will play until its completion, unless the CH0LOOP signal has also been set. When a sample completes playing, the CH0STP flag will be set. The audio DMA subsystem cannot presently generate interrupts.

Unlike on the Amiga™, the MEGA65 audio DMA system supports both 8 and 16-bit samples. It also supports packed 4-bit samples, playing either the lower or upper nybl of each sample byte. This allows two separate samples to occupy the same byte, thus effectively halving the amount of space required to store two equal length samples.

# F018 “DMAGIC” DMA CONTROLLER

| <b>HEX</b> | <b>DEC</b> | <b>Signal</b> | <b>Description</b>                                                   |
|------------|------------|---------------|----------------------------------------------------------------------|
| D700       | 55040      | ADDRLSB-TRIG  | DMAagic DMA list address LSB, and trigger DMA (when written)         |
| D701       | 55041      | ADDRMSB       | DMA list address high byte (address bits 8 - 15).                    |
| D702       | 55042      | ADDRBANK      | DMA list address bank (address bits 16 - 22). Writing clears \$D704. |

## MEGA65 DMA CONTROLLER EXTENSIONS

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D703       | 55043      |            |            |            | -          |            |            |            | EN018B     |
| D704       | 55044      |            |            |            | ADDRMB     |            |            |            |            |
| D705       | 55045      |            |            |            | ETRIG      |            |            |            |            |
| D70E       | 55054      |            |            |            | ADDRLSB    |            |            |            |            |
| D711       | 55057      | AUDEN      | BLKD       | AUDWRBLK   | NOMIX      | -          |            |            | AUDBLKTO   |
| D71C       | 55068      |            |            |            | CH0RVOL    |            |            |            |            |
| D71D       | 55069      |            |            |            | CH1RVOL    |            |            |            |            |
| D71E       | 55070      |            |            |            | CH2LVOL    |            |            |            |            |
| D71F       | 55071      |            |            |            | CH3LVOL    |            |            |            |            |
| D720       | 55072      | CH0EN      | CH0LOOP    | CH0SGN     | CH0SINE    | CH0STP     | -          |            | CH0SBITS   |
| D721       | 55073      |            |            |            | CH0BADDR   |            |            |            |            |
| D722       | 55074      |            |            |            | CH0BADDR   |            |            |            |            |
| D723       | 55075      |            |            |            | CH0BADDR   |            |            |            |            |
| D724       | 55076      |            |            |            | CH0FREQ    |            |            |            |            |
| D725       | 55077      |            |            |            | CH0FREQ    |            |            |            |            |
| D726       | 55078      |            |            |            | CH0FREQ    |            |            |            |            |
| D727       | 55079      |            |            |            | CH0TADDR   |            |            |            |            |
| D728       | 55080      |            |            |            | CH0TADDR   |            |            |            |            |
| D729       | 55081      |            |            |            | CH0VOLUME  |            |            |            |            |
| D72A       | 55082      |            |            |            | CH0CURADDR |            |            |            |            |
| D72B       | 55083      |            |            |            | CH0CURADDR |            |            |            |            |
| D72C       | 55084      |            |            |            | CH0CURADDR |            |            |            |            |
| D72D       | 55085      |            |            |            | CH0TMRADDR |            |            |            |            |
| D72E       | 55086      |            |            |            | CH0TMRADDR |            |            |            |            |
| D72F       | 55087      |            |            |            | CH0TMRADDR |            |            |            |            |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D730       | 55088      | CH1EN      | CH1LOOP    | CH1SGN     | CH1SINE    | CH1STP     | -          | CH1SBITS   |            |
| D731       | 55089      |            |            |            | CH1BADDR   |            |            |            |            |
| D732       | 55090      |            |            |            | CH1BADDR   |            |            |            |            |
| D733       | 55091      |            |            |            | CH1BADDR   |            |            |            |            |
| D734       | 55092      |            |            |            | CH1FREQ    |            |            |            |            |
| D735       | 55093      |            |            |            | CH1FREQ    |            |            |            |            |
| D736       | 55094      |            |            |            | CH1FREQ    |            |            |            |            |
| D737       | 55095      |            |            |            | CH1TADDR   |            |            |            |            |
| D738       | 55096      |            |            |            | CH1TADDR   |            |            |            |            |
| D739       | 55097      |            |            |            | CH1VOLUME  |            |            |            |            |
| D73A       | 55098      |            |            |            | CH1CURADDR |            |            |            |            |
| D73B       | 55099      |            |            |            | CH1CURADDR |            |            |            |            |
| D73C       | 55100      |            |            |            | CH1CURADDR |            |            |            |            |
| D73D       | 55101      |            |            |            | CH1TMRADDR |            |            |            |            |
| D73E       | 55102      |            |            |            | CH1TMRADDR |            |            |            |            |
| D73F       | 55103      |            |            |            | CH1TMRADDR |            |            |            |            |
| D740       | 55104      | CH2EN      | CH2LOOP    | CH2SGN     | CH2SINE    | CH2STP     | -          | CH1SBITS   |            |
| D741       | 55105      |            |            |            | CH2BADDR   |            |            |            |            |
| D742       | 55106      |            |            |            | CH2BADDR   |            |            |            |            |
| D743       | 55107      |            |            |            | CH2BADDR   |            |            |            |            |
| D744       | 55108      |            |            |            | CH2FREQ    |            |            |            |            |
| D745       | 55109      |            |            |            | CH2FREQ    |            |            |            |            |
| D746       | 55110      |            |            |            | CH2FREQ    |            |            |            |            |
| D747       | 55111      |            |            |            | CH2TADDR   |            |            |            |            |
| D748       | 55112      |            |            |            | CH2TADDR   |            |            |            |            |
| D749       | 55113      |            |            |            | CH2VOLUME  |            |            |            |            |
| D74A       | 55114      |            |            |            | CH2CURADDR |            |            |            |            |
| D74B       | 55115      |            |            |            | CH2CURADDR |            |            |            |            |
| D74C       | 55116      |            |            |            | CH2CURADDR |            |            |            |            |
| D74D       | 55117      |            |            |            | CH2TMRADDR |            |            |            |            |
| D74E       | 55118      |            |            |            | CH2TMRADDR |            |            |            |            |
| D74F       | 55119      |            |            |            | CH2TMRADDR |            |            |            |            |
| D750       | 55120      | CH3EN      | CH3LOOP    | CH3SGN     | CH3SINE    | CH3STP     | -          | CH3SBITS   |            |
| D751       | 55121      |            |            |            | CH3BADDR   |            |            |            |            |
| D752       | 55122      |            |            |            | CH3BADDR   |            |            |            |            |
| D753       | 55123      |            |            |            | CH3BADDR   |            |            |            |            |
| D754       | 55124      |            |            |            | CH3FREQ    |            |            |            |            |
| D755       | 55125      |            |            |            | CH3FREQ    |            |            |            |            |

continued ...

...continued

| HEX  | DEC   | DB7 | DB6 | DB5 | DB4        | DB3 | DB2 | DB1 | DB0 |
|------|-------|-----|-----|-----|------------|-----|-----|-----|-----|
| D756 | 55126 |     |     |     | CH3FREQ    |     |     |     |     |
| D757 | 55127 |     |     |     | CH3TADDR   |     |     |     |     |
| D758 | 55128 |     |     |     | CH3TADDR   |     |     |     |     |
| D759 | 55129 |     |     |     | CH3VOLUME  |     |     |     |     |
| D75A | 55130 |     |     |     | CH3CURADDR |     |     |     |     |
| D75B | 55131 |     |     |     | CH3CURADDR |     |     |     |     |
| D75C | 55132 |     |     |     | CH3CURADDR |     |     |     |     |
| D75D | 55133 |     |     |     | CH3TMRADDR |     |     |     |     |
| D75E | 55134 |     |     |     | CH3TMRADDR |     |     |     |     |
| D75F | 55135 |     |     |     | CH3TMRADDR |     |     |     |     |

- **ADDRLSB** DMA list address low byte (address bits 0 - 7) WITHOUT STARTING A DMA JOB (used by Hypervisor for unfreezing DMA-using tasks)
- **ADDRMB** DMA list address mega-byte
- **AUDBLKTO** Audio DMA block timeout (read only) DEBUG
- **AUDEN** Enable Audio DMA
- **AUDWRBLK** Audio DMA block writes (samples still get read)
- **BLKD** Audio DMA blocked (read only) DEBUG
- **CHOBADDR** Audio DMA channel 0 base address LSB
- **CHOCURADDR** Audio DMA channel 0 current address LSB
- **CHOEN** Enable Audio DMA channel 0
- **CHOFREQ** Audio DMA channel 0 frequency LSB
- **CHOLOOP** Enable Audio DMA channel 0 looping
- **CHORVOL** Audio DMA channel 0 right channel volume
- **CHOSBITS** Audio DMA channel 0 sample bits (11=16, 10=8, 01=upper nybl, 00=lower nybl)
- **CHOSGN** Enable Audio DMA channel 0 signed samples
- **CHOSINE** Audio DMA channel 0 play 32-sample sine wave instead of DMA data
- **CHOSTP** Audio DMA channel 0 stop flag
- **CHOTADDR** Audio DMA channel 0 top address LSB
- **CHOTMRADDR** Audio DMA channel 0 timing counter LSB

- **CH0VOLUME** Audio DMA channel 0 playback volume
- **CH1BADDR** Audio DMA channel 1 base address LSB
- **CH1CURADDR** Audio DMA channel 1 current address LSB
- **CH1EN** Enable Audio DMA channel 1
- **CH1FREQ** Audio DMA channel 1 frequency LSB
- **CH1LOOP** Enable Audio DMA channel 1 looping
- **CH1RVOL** Audio DMA channel 1 right channel volume
- **CH1SBITS** Audio DMA channel 1 sample bits (11=16, 10=8, 01=upper nybl, 00=lower nybl)
- **CH1SGN** Enable Audio DMA channel 1 signed samples
- **CH1SINE** Audio DMA channel 1 play 32-sample sine wave instead of DMA data
- **CH1STP** Audio DMA channel 1 stop flag
- **CH1TADDR** Audio DMA channel 1 top address LSB
- **CH1TMRADDR** Audio DMA channel 1 timing counter LSB
- **CH1VOLUME** Audio DMA channel 1 playback volume
- **CH2BADDR** Audio DMA channel 2 base address LSB
- **CH2CURADDR** Audio DMA channel 2 current address LSB
- **CH2EN** Enable Audio DMA channel 2
- **CH2FREQ** Audio DMA channel 2 frequency LSB
- **CH2LOOP** Enable Audio DMA channel 2 looping
- **CH2LVOL** Audio DMA channel 2 left channel volume
- **CH2SGN** Enable Audio DMA channel 2 signed samples
- **CH2SINE** Audio DMA channel 2 play 32-sample sine wave instead of DMA data
- **CH2STP** Audio DMA channel 2 stop flag
- **CH2TADDR** Audio DMA channel 2 top address LSB
- **CH2TMRADDR** Audio DMA channel 2 timing counter LSB
- **CH2VOLUME** Audio DMA channel 2 playback volume
- **CH3BADDR** Audio DMA channel 3 base address LSB
- **CH3CURADDR** Audio DMA channel 3 current address LSB

- **CH3EN** Enable Audio DMA channel 3
- **CH3FREQ** Audio DMA channel 3 frequency LSB
- **CH3LOOP** Enable Audio DMA channel 3 looping
- **CH3LVOL** Audio DMA channel 3 left channel volume
- **CH3SBITS** Audio DMA channel 3 sample bits (11=16, 10=8, 01=upper nybl, 00=lower nybl)
- **CH3SGN** Enable Audio DMA channel 3 signed samples
- **CH3SINE** Audio DMA channel 3 play 32-sample sine wave instead of DMA data
- **CH3STP** Audio DMA channel 3 stop flag
- **CH3TADDR** Audio DMA channel 3 top address LSB
- **CH3TMRAADDR** Audio DMA channel 3 timing counter LSB
- **CH3VOLUME** Audio DMA channel 3 playback volume
- **ENO18B** DMA enable F018B mode (adds sub-command byte)
- **ETRIG** Set low-order byte of DMA list address, and trigger Enhanced DMA job (uses DMA option list)
- **NOMIX** Audio DMA bypasses audio mixer

## UNIMPLEMENTED FUNCTIONALITY

The MEGA65's DMAgic does not currently support either memory-swap or mini-term operations.

Miniterms were intended for bitplane blitting, which is not required for the MEGA65 which offers greatly advanced character modes and stepped and fractional DMA address incrementing which allows efficient texture copying and scaling. Also there exists no known software which ever used this facility, and it remains uncertain if it was ever implemented in any revision of the DMAgic chip used in C65 prototypes.

The memory-swap operation is intended to be implemented, but can be worked around in the meantime by copying the first region to a 3rd region that acts as a temporary buffer, then copying the 2nd region to the 1st, and the 3rd to the 2nd.

# M

## APPENDIX

# VIC-IV Video Interface Controller

- Features
- VIC-II/III/IV Register Access Control
- Video Output Formats, Timing and Compatibility
- Memory Interface
- Hot Registers
- New Modes
- Sprites

- **VIC-II / C64 Registers**
- **VIC-III / C65 Registers**
- **VIC-IV / MEGA65 Specific Registers**





# FEATURES

The VIC-IV is a fourth generation Video Interface Controller developed especially for the MEGA65, and featuring very good backwards compatibility with the VIC-II that was used in the C64, and the VIC-III that was used in the C65. The VIC-IV can be programmed as though it were either of those predecessor systems. In addition it supports a number of new features. It is easy to mix older VIC-II/III features with the new VIC-IV features, making it easy to transition from the VIC-II or VIC-III to the VIC-IV, just as the VIC-III made it easy to transition from the VIC-II. Some of the new features and enhancements of the VIC-IV include:

- **Direct access to 384KB RAM** (up from 16KB/64KB with the VIC-II and 128KB with the VIC-IV).
- Support for **32KB of 8-bit Colour/Attribute RAM** (up from 2KB on the VIC-III), to support very large screens.
- **HDTV  $720 \times 576$  /  $800 \times 600$  native resolution** at both 50Hz and 60Hz for **PAL and NTSC**, with **VGA and digital video** output.
- **81MHz pixel clock** (up from  $\sim 8$ MHz with the VIC-II/III), which enables a wide range of new features.
- New 16-colour ( $16 \times 8$  pixels per character cell) and 256-colour ( $8 \times 8$  pixels per character cell) **full-colour text modes**.
- Support for up to **8,192 unique characters in a character set**.
- **Four 256-colour palette banks** (versus the VIC-III's single palette bank), each supporting **23-bit colour depth** (versus the VIC-III's 12-bit colour depth), and which can be rapidly alternated to create even more colourful graphics than is possible with the VIC-III.
- Screen, bitmap, colour and character data can be positioned at any **address with byte-level granularity** (compared with fixed 1KB - 16KB boundaries with the VIC-II/III)
- **Virtual screen dimensioning**, which combined with byte-level data position granularity provides effective **hardware support for scrolling and panning in both X and Y directions**.
- **New sprite modes**: Bitplane modification, **full-colour** (15 foreground colours + transparency) and tiled modes, allowing a wide variety of new and exciting sprite-based effects
- The ability to stack sprites in a bit-planar manner to produce **sprites with up to 256 colours**.

- Sprites can use 64 bits of data per raster line, allowing **sprites to be 64 pixels wide** when using VIC-II/III mono/multi-colour mode, or 16 pixels wide when using the new VIC-IV full-colour sprite mode.
- **Sprite tile mode**, which allows a sprite to be repeated horizontally across an entire raster line, allowing sprites to be used to create animated backgrounds in a memory-efficient manner.
- Sprites can be configured to use a **separate 256-colour palette** to that used to draw other text and graphics, allowing for a more colourful display.
- **Super-extended attribute mode** which uses two screen RAM bytes and two colour RAM bytes per character mode, which supports a wide variety of new features including **alpha-blending/anti-aliasing**, **hardware kerning/variable-width characters**, hardware horizontal/vertical flipping, alternate palette selection and other powerful features that make it easy to create highly dynamic and colourful displays.
- **Raster-Rewrite Buffer** which allows **hardware-generated pseudo-sprites**, similar to “bobs” on Amiga™ computers, but with the advantage that they are rendered in the display pipeline, and thus do not need to be un-drawn and redrawn to animate them.
- **Multiple 8-bit colour play-fields** are also possible using the Raster-Rewrite Buffer.

In short, the VIC-IV is a powerful evolution of the VIC-II/III, while retaining the character and distinctiveness of the VIC-series of video controllers.

For a full description of the additional registers that the VIC-IV provides, as well as documentation of the legacy VIC-II and VIC-III registers, refer to the corresponding sections of this appendix. The remainder of the appendix will focus on describing the capabilities and use of many of the VIC-IV’s new features.

## VIC-II/III/IV REGISTER ACCESS CONTROL

Because the new features of the VIC-IV are all extensions to the existing VIC-II/III designs, there is no concept of having to select the mode in which the VIC-IV will operate: It is always in VIC-IV mode. However, for backwards compatibility with software, the many additional registers of the VIC-IV can be hidden, so that it appears to be either a VIC-II or VIC-III. This is done in the same manner that the VIC-III uses to hide its new features from legacy VIC-II software.

The mechanism is the VIC-III write-only KEY register (\$D02F, 53295 decimal). The VIC-III by default conceals its new features until a “knock” sequence is performed. This consists

of writing two special values one after the other to \$D02F. The following table summarises the knock sequences supported by the VIC-IV, and indicates which are VIC-IV specific, and which are supported by the VIC-III:

| <b>First Value Hex (Decimal)</b> | <b>Second Value Hex (Decimal)</b> | <b>Effect</b>                                                                                 | <b>VIC-IV Specific?</b> |
|----------------------------------|-----------------------------------|-----------------------------------------------------------------------------------------------|-------------------------|
| \$00 (0)                         | \$00 (0)                          | Only VIC-II registers visible (all VIC-III and VIC-IV new registers are hidden)               | No                      |
| \$A5 (165)                       | \$96 (150)                        | VIC-III new registers visible                                                                 | No                      |
| \$47 (71)                        | \$53 (83)                         | Both VIC-III and VIC-IV new registers visible                                                 | Yes                     |
| \$45 (69)                        | \$54 (84)                         | No VIC-II/III/IV registers visible.<br>45E100 Ethernet controller buffers are visible instead | Yes                     |

## Detecting VIC-II/III/IV

Detecting which generation of the VIC-II/III/IV a machine is fitted with can be important for programs that support only particular generations, or that wish to vary their graphical display based on the capabilities of the machine. While there are many possibilities for this, the following is a simple and effective method. It relies on the fact that the VIC-III and VIC-IV do not repeat the VIC-II registers throughout the IO address space. Thus while \$D000 and \$D100 are synonymous when a VIC-II is present (or a VIC-III/IV is hiding their additional registers), this is not the case when a VIC-III or VIC-IV is making all of its registers visible. Therefore presence of a VIC-III/IV can be determined by testing whether these two locations are aliases for the same register, or represent separate registers. The detection sequence consists of using the KEY register to attempt to make either VIC-IV or VIC-III additional registers visible. If either succeeds, then we can assume that the corresponding generation of VIC is installed. As the VIC-IV supports the VIC-III KEY knocks, we must first test for the presence of a VIC-IV. Also, we assume that the MEGA65 starts in VIC-IV mode, even when running C65 BASIC. Thus the test can be done in BASIC from either C64 or C65 mode as follows:

```

8 REM IN C65 MODE WE CANNOT SAFELY WRITE TO 53295, SO WE TEST A DIFFERENT WAY
10 IF PEEK(53272) AND 32 THEN GOTO 65
20 POKE53248,1:POKE53295,71:POKE53295,83
30 POKE53248+256,0:IFPEEK(53248)=1THENPRINT"VIC-IV PRESENT":END
40 PUKE53248,1:PUKE53295,165:PUKE53295,150
50 PUKE53248+256,0:IFPEEK(53248)=1THENPRINT"VIC-III PRESENT":END
60 PRINT "VIC-II PRESENT": END

65 REM WE ASSUME WE HAVE A C65 HERE
70 V1=PEEK(53248+80):V2=PEEK(53248+80):V3=PEEK(53248+80)
80 IF V1<>V2 OR V1<>V3 OR V2<>V3 THEN PRINT "VIC-IV PRESENT":END
90 GOTO 40

```

Line 10 of this programme checks whether the screen is a multiple of 2KiB. As the screen on the C64 is located at 1KiB, this test will fail, and execution will continue to line 20. Line 20 writes 1 to one of the VIC-II sprite position registers, 53248, before writing the MEGA65 knock to the key register, 53295. Line 30 writes to 53248 + 256, which on the C64 is a mirror of 53248, but on a MEGA65 with VIC-IV IO enabled will be one of the red palette registers. After writing to 53248 + 256, the programme checks if the register at 53248 has been modified by the write to 53248 + 256. If it has, then the two addresses point to the same register. This will happen on either a C64 or C65, but not on a computer with a VIC-IV. Thus if 53248 has not changed, we report that we have detected a VIC-IV. If writing to 53248 + 256 did change the value in register 53248, then we proceed to line 40, which writes to 53248 again, and this time writes the VIC-III knock to the key register. Line 50 is like line 30, but as it appears after a VIC-III knock, it allows the detection of a VIC-III. Finally, if neither a VIC-IV nor VIC-III is detected, we conclude that only a VIC-II must be present.

As the MEGA65 is the only C64-class computer that is fitted with a VIC-IV, this can be used as a *de facto* test for the presence of a MEGA65 computer. Detection of a VIC-III can be similarly assumed to indicate the presence of a C65.

## VIDEO OUTPUT FORMATS, TIMING AND COMPATIBILITY

# Integrated Marvellous Digital Hookup™(IMDH™) Digital Video Output

The MEGA65 features VGA analog video output and Integrated Marvellous Digital Hookup™ (IMDH™). This is different to existing common digital video standards in several key points:

1. We didn't invent a new connector for it: We instead used the most common digital video connector already in use. So your existing cables should work fine!
2. We didn't make it purposely incompatible with any existing digital video standard. So your existing TVs and monitors should work fine!
3. We don't engage in highway-robbery for other vendors to use the IMDH™ digital video standard, by trying to charge them \$10,000 every year, just for the permission to be able to sell a single device. This means that the MEGA65 is cheaper for you!
4. The IMDH™ standard does not allow content-protection or other sovereignty eroding flim-flam. If you produced the video, you can do whatever you like with it!

## Connecting to Naughty Proprietary Digital Video Standards

There are digital video standards that are completely backwards compared with IMDH™. Fortunately because of IMDH™'s open approach to interoperability, these should, in most cases, function with the MEGA65 without difficulty. Simply find a video cable fits the IMDH™ connector on the back of your MEGA65, and connect it to your MEGA65 and a TV, Monitor or Projector that has the same connector.

However, regrettably, not all manufacturers have submitted their devices for IMDH™ compliance testing with the MEGA65 team. This means that some TVs and Monitors are, unfortunately, not IMDH™ compliant. Thus while most TVs and Monitors will work with the MEGA65, you might find that you need to try a couple to get a satisfactory result. If you do find a monitor that doesn't work with the MEGA65, please let us know, and also report the problem to the Monitor vendor, recommending that they submit their devices for IMDH™ compliance testing.

The VIC-IV was designed for use in the MEGA65 and related systems, including the MEGA-phone family of portable devices. The VIC-IV supports both VGA and digital video output, using the non-proprietary IMDH™ interface. It also supports parallel digital video output suitable for driving LCD display panels. Considerable care has been taken to create a common video front-end that supports these three output modes.

For simplicity and accuracy of frame timing for legacy software, the video format is normally based on the HDTV PAL and NTSC  $720 \times 576/480$  (576p and 480p) modes using a 27MHz output pixel clock. This is ideal for digital video and LCD display panels. However not all VGA displays support these modes, especially  $720 \times 576$  at 50Hz.

In terms of VIC-II and VIC-III backwards compatibility, this display format has several effects that do not cause problems for most programs, but can cause some differences in behaviour:

1. Because the VIC-IV display is progressive rather than interlaced, two physical raster lines are produced for each logical VIC-II or VIC-III raster line. This means that there are either 63 or 65 cycles per logical double raster, rather than per physical 576p/480p physical raster. This can cause some minor visual artefacts, when programs make assumptions about where on a horizontal line the VIC is drawing when, for example, the border or screen colour is changed.
2. The VIC-IV does not follow the behaviour of the VIC-III, which allowed changes in video modes, e.g., between text and bitmap mode, on characters. Nor does it follow the VIC-II's policy of having such changes take effect immediately. Instead, the VIC-IV applies changes at the start of each raster line. This can cause some minor artefacts.
3. The VIC-IV uses a single-raster rendering buffer which is populated using the VIC-IV's internal 8.1MHz pixel clock, before being displayed using the 27MHz output pixel clock. This means that a raster lines display content tends to be rendered much earlier in a raster line than on either the VIC-II or VIC-III. This can cause some artefacts with displays, particularly in demos that rely on specific behaviour of the VIC-II at particular cycles in a raster line, for example for effects such as VSP or FLI. At present, such effects are unlikely to display correctly on the current revision of the VIC-IV. Improved support for these features is planned for a future revision of the VIC-IV.
4. The  $1280 \times 200$  and  $1280 \times 400$  display modes of the VIC-III are not currently supported, as they cannot be meaningfully displayed on any modern monitor, and no software is known to support or use this feature.

## Frame Timing

Frame timing is designed to match that of the 6502 + VIC-II combination of the C64. Both PAL and NTSC timing is supported, and the number of cycles per logical raster line, the number of raster lines per frame, and the number of cycles per frame are all adjusted accordingly. To achieve this, the VIC-IV ordinarily uses HDTV 576p 50Hz (PAL) and 480p 60Hz (NTSC) video modes, with timing tweaked to be as close as possible to double-scan PAL and NTSC composite TV modes as used by the VIC-II.

The VIC-IV produces timing impulses at approximately 1MHz which are used by the 45GS02 processor, so that the correct effective frequency is provided when operating at the 1MHz, 2MHz and 3.5MHz C64, C128 and C65 compatibility modes. This allows the single machine to switch between accurate PAL and NTSC CPU timing, as well as video modes. The exact frequency varies between PAL and NTSC modes, to mimic the behaviour of PAL versus NTSC C64, C128 and C65 processor and video timing.

The PAL frame is constructed from 624 physical raster lines, consisting of 864 pixel clock ticks. The pixel clock is 27MHz, which is 1/3 the VIC-IV pixel clock. The visible frame is  $720 \times 576$  pixels, the entirety of which can be used in VIC-IV mode. In VIC-II and VIC-III modes, the border area reduces the usable size to  $640 \times 400$  pixels. In VIC-II mode and VIC-III 200H modes, the display is double scanned, with two 31.5 micro-second physical rasters corresponding to a single 63 micro-second VIC-II-style raster line. Thus each frame consists of 312 VIC-II raster lines of 63 micro-seconds each, exactly matching that of a PAL C64.



The NTSC frame is constructed from 526 physical raster lines, consisting of 858 pixel clock ticks. The pixel clock is 27MHz, which is 1/3 the VIC-IV pixel clock. The visible frame is  $720 \times 480$  pixels, the entirety of which can be used in VIC-IV mode. In VIC-II and VIC-III modes, the border area reduces the usable size to  $640 \times 400$  pixels. In VIC-II mode and VIC-III 200H modes, the display is double scanned, with two 32 micro-second physical rasters corresponding to a single 64 micro-second VIC-II-style raster line. Thus

each frame consists of 263 VIC-II raster lines of 64 micro-seconds each, matching the most common C64 NTSC video timing.

**858 Horizontal Ticks  
(32 µSec per line)**



As these HDTV video modes are not supported by all VGA monitors, a compatibility mode is included that provides a  $640 \times 480$  VGA-style mode. However, as the pixel clock of the MEGA65 is fixed at 27MHz, this mode runs at 63Hz. Nonetheless, this should work on the vast majority of VGA monitors. There should be no problem with the PAL / NTSC modes when using the digital video output of the MEGA65 with the vast majority of IMDH™-enabled monitors and TVs.

To determine whether the MEGA65 is operating in PAL or NTSC, you can enter the freeze menu, which displays the current video mode, or from a program you can check the PAL-NTSC signal (bit 7 of \$D06F, 53359 decimal). If this bit is set, then the machine is operating in NTSC mode, and clear if operating in PAL mode. This bit can be modified to change between the modes, e.g.:

```

10 IFPEEK(53272)<32THENPOKE53295,ASC("6"):POKE53295,ASC("S"):REM ENABLE C65+MEGA65 IO
20 NTSC=PEEK(53359)AND128
30 IFNTSC<1 THENPRINT"MEGA65 IS IN NTSC MODE"
40 IFNTSC>0 THENPRINT"MEGA65 IS IN PAL MODE"
50 INPUT"SWITCH MODES (Y/N)? ",A$
60 IF A$="Y"THENPOKE53359,PEEK(53359)+128-NTSC
70 NTSC=PEEK(53359)AND128
80 IFNTSC<1 THENPRINT"MEGA65 IS NOW IN NTSC MODE"
90 IFNTSC>0 THENPRINT"MEGA65 IS NOW IN PAL MODE"

```

## Physical and Logical Rasters

Physical rasters per frame refers to the number of actual raster lines in the PAL or NTSC Enhanced Definition TV (EDTV) video modes used by the MEGA65. Logical Rasters refers to the number of VIC-II-style rasters per frame. Each logical raster consists of two physical rasters per line, since EDTV modes are double-scan modes compared with the original PAL and NTSC Standard Definition TV modes used by the C64. The frame parameters of the VIC-IV for PAL and NTSC are as follows:

| <b>Standard</b> | <b>Cycles per Raster</b> | <b>Physical Rasters per Frame</b> | <b>Logical Rasters per Frame</b> |
|-----------------|--------------------------|-----------------------------------|----------------------------------|
| PAL             | 63                       | 626                               | 312                              |
| NTSC            | 65                       | 526                               | 263                              |

The result is that the frames on the VIC-IV consist of exactly the same number of ~ 1MHz CPU cycles as on the VIC-II exactly.

## Bad Lines

The VIC-IV does not natively incur any “bad lines”, because the VIC-IV has its own dedicated memory busses to the main memory and colour RAM of the MEGA65. This means that both the processor and VIC-IV can access the memory at the same time, unlike on the C64 or C65, where they are alternated.

However, to improve compatibility, the VIC-IV signals when a “bad line” would have occurred on the VIC-II. The 45GS02 processor of the MEGA65 accepts these bad line signals, and pauses the CPU for 40 clock cycles, except if the processor is running at full speed, in which case they are ignored. This improves the timing compatibility with

the VIC-II considerably. However, the timing is not exact, because the current revision of the 45GS02 pauses for exactly 40 cycles, instead of 40 - 43 cycles, depending on the instruction being executed at the time. Also, the VIC-IV and 45GS02 do not currently pause for sprite fetches.

The bad line emulation is controlled by bit 0 of \$D710: setting this bit enables bad line emulation, and clearing it prevents any bad line from stealing time from the processor.

## MEMORY INTERFACE

The VIC-IV supports up to 64KB of colour RAM and, in principle, 16MB of direct access RAM for video data. However in typical installations 32KB of colour RAM and 384KB of addressable RAM is present. In MEGA65 systems, the second 128KB of RAM is typically used to hold a C65-compatible ROM, leaving 256KB available, unless software is written to avoid the need to use C65 ROM routines, in which case all 384KB can be used.

The VIC-IV supports all legacy VIC-II and VIC-III methods for accessing this RAM, including the VIC-II's use of 16KB banks, and the VIC-III's Display Address Translator (DAT). This additional memory can be used for character and bitmap displays, as well as for sprites. However, the VIC-III bitplane modes remain limited to using only the first 128KB of RAM, as the VIC-IV does not enhance the bitplane mode.

## Relocating Screen Memory

To use the additional memory for screen RAM, the screen RAM start address can be adjusted to any location in memory with byte-level granularity by setting the SCRN PTR registers (\$D060 - \$D063, 53344 - 53347 decimal). For example, to set the screen memory to address 12345:

```
IF PEEK(53272)<32 THEN POKE 53295, ASC("G") : POKE 53295, ASC("S") : REM ENABLE C65+MEGA65 IO  
POKE 53344+0, 69 : POKE 53344+1, 35 : POKE 53344+2, 1
```

## Relocating Character Generator Data

The location of the character generator data can also be set with byte-level precision via the CHAR PTR registers at \$D068 - \$D06A (53352 - 53354 decimal). As usual, the first of these registers holds the lowest-order byte, and the last the highest-order byte. The three bytes allow for placement of character data anywhere in the first 16MB of RAM.

For systems with less than 16MB of RAM accessible by the VIC-IV, the upper address bits should be zero.

For example, to indicate that character generator data should be sourced beginning at \$41200 (266752 decimal), the following could be used. Note that the AND binary operator only works with arguments between 0 and 65,535. Therefore we first subtract  $4 \times 65,536 = 262,144$  from the address (the 4 is determined by calculating  $\text{INT}(266752/65536)$  ), before we use the AND operator to compute the lower part of the address:

```
IFPEEK(53272)<32THENPOKE53295,ASC("6"):POKE53295,ASC("S"):REM ENABLE C65+MEGA65 IO  
POKE53352,(266752-INT(266752/65536)*65536)AND255  
POKE53353,INT((266752-INT(266752/65536)*65536)/256)  
POKE53354,INT(266752/65536)
```

## Relocating Colour / Attribute RAM

The area of colour RAM being used can be similarly set using the COLPTR registers (\$D064 - \$D065, 53348 – 53349 decimal). That is, the value is an offset from the start of the colour / attribute RAM. This is because, like on the C64, the colour / attribute RAM of the MEGA65 is a separate memory component, with its own dedicated connection to the VIC-IV. By default, the COLPTRs are set to zero, which replicates the behaviour of the VIC-II/III. To set the display to use the colour / attribute RAM beginning at offset 4000, one could use something like:

```
IFPEEK(53272)<32THENPOKE53295,ASC("6"):POKE53295,ASC("S"):REM ENABLE C65+MEGA65 IO  
POKE53348,4000 AND 255  
POKE53349,INT(4000/256)
```

## Relocating Sprite Pointers and Images

The location of the sprite pointers can also be moved, and sprites can be made to have their data anywhere in first 4MB of memory. This is accomplished by first setting the location of the sprite pointers by setting the SPRPTRADR registers (\$D06C - \$D06E, 53356 – 53358 decimal, but note that only the bottom 7 bits of \$D06E are used, as the highest bit is used for the SPRPTR16 signal). This allows the list of eight sprite pointers to be moved from the end of screen RAM to an arbitrary location in the first 8MB of RAM. To allow sprites themselves to be located anywhere in the first 4MB of RAM, the SPRPTR16 bit in \$D06E must be set. In this mode, two bytes are used to indicate the location of

each sprite, instead of one. That is, the list of sprite pointers will be 16 bytes long, instead of 8 bytes long as on the VIC-II/III. When SPRPTR16 is enabled, the location of the sprite pointers should always be set explicitly via the SPRPTRADR registers. For example, to position the sprite pointers at location 800 – 815, you could use something like the following code. Note that a little gymnastics is required to keep the SPRPTR16 bit unchanged, and also to work around the AND binary operator not working with values greater than 65535:

```
IF PEEK(53272)<32 THEN POKE 53295,ASC("G"):POKE 53295,ASC("S"):REM ENABLE C65+MEGA65 IO  
POKE 53356,(800-INT(800/65536)*65536) AND 255  
POKE 53357,INT(800/256)AND255  
POKE 53358,(PEEK(53358)AND128)+INT(800/65536)
```

The location of each sprite image remains a multiple of 64 bytes, thus allowing for up to 65,536 unique sprite images to be used at any point in time, if the system is equipped with sufficient RAM (4MB or more). In this mode, the VIC-II 16KB banking is ignored, and the location of sprite data is simply  $64 \times$  the pointer value. For example, to have the data for a sprite at \$C000 (49152 decimal), this would be sprite location 768, because  $49152 \div 64 = 768$ . We then need to split 768 into high and low bytes, to set the two pointer bytes:  $768 = 256 \times 3$ , with remainder 0, so this would require the two sprite pointer bytes to be 0 (low byte, which comes first) and 3 (high byte). Thus if the sprite pointers were located at \$7F8 (2040 decimal), setting the first sprite to sprite image 768 could be done with something like:

```
POKE 2040,768-256*INT(768/256)  
POKE 2041,INT(768/256)
```

## HOT REGISTERS

Because of the availability of precise vernier registers to set a wide range of video parameters directly, \$D011 (53265 decimal), \$D016 (53270 decimal) and other VIC-II and VIC-III video mode registers are implemented as virtual registers: by default, writing to any of these results in computed consistent values being applied to all of the relevant vernier registers. This means that writing to any of these virtual registers will reset the video mode. Thus some care has to be taken when using new VIC-IV features to not touch any of the “hot” VIC-II and VIC-III registers.

The “hot” registers to be careful with are:

\$D011, \$D016, \$D018, \$D031 (53265, 53270, 53272 and 53297 decimal) and the VIC-II bank bits of \$DD00 (56576 decimal).

If you write to any of those, various VIC-IV registers will need to be re-written with the values you wish to maintain.

This “hot” register behaviour is intended primarily for legacy software. It can be disabled by clearing the HOTREG signal (bit 7 of \$D05D, 53341 decimal).

## NEW MODES

# Why the new VIC-IV modes are Character and Bitmap modes, not Bitplane modes

The new VIC-IV video modes are derived from the VIC-II character and bitmap modes, rather than the VIC-III bitplane modes. This decision was based on several realities of programming a memory-constrained 8-bit home computer:

1. Bitplanes require that the same amount of memory is given to each area on screen, regardless of whether it is showing empty space, or complex graphics. There is no way with bitplanes to reuse content from within an image in another part of the image. However, most C64 games use highly repetitive displays, with common elements appearing in various places on the screen, of which Boulder Dash and Super Giana Sisters would be good examples.
2. Bitplanes also make it difficult to update a display, because every pixel is unique, in that there is no way to make a change, for example to the animation in an on-screen element, and have it take effect in all places at the same time. The diamond animations in Boulder Dash are a good example of this problem. The requirement to modify multiple separate bytes in each bitplane create an increased computational burden, which is why there were calls for the Amiga AAA chip-set to include so-called “chunky” modes, rather than just bitplane based modes. While the Display Address Translator (DAT) and DMAgic of the C65 provide some relief to this problem, the relief is only partial.
3. Scrolling using the C65 bitplanes requires copying the entire bitplane, as the hardware support for smooth scrolling does not extend to changing the bitplane source address in a fine manner. Even using the DMAgic to assist, scrolling a  $320 \times 200$  256-colour display requires 128,000 clock cycles in the best case (reading and writing  $320 \times 200 = 64000$  bytes). At 3.5MHz on the C65 this would require about 36 milli-seconds, or about 2 complete video frames. Thus for smooth scrolling of such

a display, a double buffered arrangement would be required, which would consume 128,000 of the 131,072 bytes of memory.

In contrast, the well known character modes of the VIC-II are widely used in games, due to their ability to allow a small amount of screen memory to select which  $8 \times 8$  block of pixels to display, allowing very rapid scrolling, reduced memory consumption, and effective hardware acceleration of animation of common elements. Thus the focus of improvements in the VIC-IV has been on character mode. As bitmap mode on the VIC-II is effectively a special case of character mode, with implied character numbers, it comes along free for the ride on the VIC-IV, and will only be mentioned in the context of a very few bitmap-mode specific improvements that were trivial to make, and it thus seemed foolish to not implement, in case they find use.

## Displaying more than 256 unique characters via "Super-Extended Attribute Mode"

The primary innovation is the addition of the Super-Extended Attribute Mode. The VIC-II already uses 12 bits per character: Each  $8 \times 8$  cell is defined by 12 bits of data: 8 bits of screen RAM data, by default from \$0400 - \$07E7 (1024 - 2023 decimal), indicating which characters to show, and 4 bits of colour data from the 1K nibble colour RAM at \$D800 - \$DBFF (55296 - 56319 decimal). The VIC-III of the C65 uses 16 bits, as the colour RAM is now 8 bits, instead of 4, with the extra 4 bits of colour RAM being used to support attributes (blink, bold, underline and reverse video). It is recommended to revise how this works, before reading the following. A good introduction to the VIC-II text mode can be found in many places. Super-Extended Attribute mode doubles the number of bits per character used from the VIC-III's 16, to 32: Two bytes of screen RAM and two bytes of colour/attribute RAM.

Super-Extended Attribute Mode is enabled by setting bit 0 in \$D054 (53332 decimal). Remember to first enable VIC-IV mode, to make this register accessible. When this bit is set, two bytes are used for each of the screen memory and colour RAM for each character shown on the display. Thus, in contrast to the 12 bits of information that the C64 uses per character, and the 16 bits that the VIC-III uses, the VIC-IV has 32 bits of information. How those 32 bits are used varies slightly among the particular modes. The default is as follows:

| <b>Bit(s)</b>                 | <b>Function</b>                                                                                                                                                                                                          |
|-------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Screen RAM byte 0             | Lower 8 bits of character number, the same as the VIC-II and VIC-III                                                                                                                                                     |
| Screen RAM byte 1, bits 0 - 4 | Upper 5 bits of character number, allowing addressing of 8,192 unique characters                                                                                                                                         |
| Screen RAM byte 1, bits 5 - 7 | Trim pixels from right side of character (bits 0 - 2) or Set character data Y offset if GOTOX set set (bits 0 - 2)                                                                                                       |
| Colour RAM byte 0, bit 7      | Vertically flip the character or enable transparency for subsequent characters if GOTOX is set                                                                                                                           |
| Colour RAM byte 0, bit 6      | Horizontally flip the character                                                                                                                                                                                          |
| Colour RAM byte 0, bit 5      | Alpha blend mode (leave 0, discussed later)                                                                                                                                                                              |
| Colour RAM byte 0, bit 4      | GOTO X (allows repositioning of characters along a raster via the Raster-Rewrite Buffer, discussed later), must be set to 0 for displaying characters                                                                    |
| Colour RAM byte 0, bits 3     | If set, Full-Colour characters use 4 bits per pixel and are 16 pixels wide (less any right side trim bits), instead of using 8 bits per pixel. When using 8 bits per pixels, the characters are the normal 8 pixels wide |
| Colour RAM byte 0, bits 2     | Trim pixels from right side of character (bit 3) or Set character data Y offset if GOTOX set set (bit 3)                                                                                                                 |
| Colour RAM byte 0, bits 0 - 1 | Number of pixels to trim from top or bottom of character                                                                                                                                                                 |
| Colour RAM byte 1, bits 0 - 3 | Low 4 bits of colour of character                                                                                                                                                                                        |
| Colour RAM byte 1, bits 4 - 7 | Upper 4 bits of colour of character (if VIC-II multi-colour mode is enabled)                                                                                                                                             |
| Colour RAM byte 1, bit 4      | Hardware blink of character (if VIC-III extended attributes are enabled)                                                                                                                                                 |
| Colour RAM byte 1, bit 5      | Hardware reverse video enable of character (if VIC-III extended attributes are enabled)*                                                                                                                                 |
| Colour RAM byte 1, bit 6      | Hardware bold attribute of character (if VIC-III extended attributes are enabled)*                                                                                                                                       |
| Colour RAM byte 1, bit 7      | Hardware underlining of character (if VIC-III extended attributes are enabled)                                                                                                                                           |

\* Enabling BOLD and REVERSE attributes at the same time on the MEGA65 selects an alternate palette, effectively allowing 512 colours on screen, but each 8x8 character can use colours only from one 256 colour palette.

If the GOTOX bit is set, some of the fields have different meanings:

| <b>Bit(s)</b>                    | <b>Function</b>                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
|----------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Screen RAM byte 0                | Lower 8 bits of new X position to start drawing the next character, relative to the start of character drawing.<br>Setting to 0 causes the next character to be drawn over the top of the left-most character.                                                                                                                                                                                                                                                                                                                    |
| Screen RAM byte 1,<br>bits 0 - 1 | Upper 2 bits of new X position                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |
| Screen RAM byte 1,<br>bits 3 - 4 | RESERVED, set to 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Screen RAM byte 1,<br>bits 5 - 7 | FCM Character data offset: Characters display normally when set to zero. When non-zero, $8 \times$ the value is added to the character address. With careful planning, this can be used to smoothly vertically scroll multiple layers of RRB content.                                                                                                                                                                                                                                                                             |
| Colour RAM byte 0,<br>bit 4 - 6  | RESERVED, set to 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Colour RAM byte 0,<br>bit 7      | If set, then background/transparent pixels will not be drawn, allowing layering                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
| Colour RAM byte 0,<br>bit 4      | GOTO X, set to 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| Colour RAM byte 0,<br>bits 3     | ROWMASK. If set, then the pixel row mask is used to determine which pixel rows of the following characters should be rendered. This can be used to vertically scroll characters using the Raster-Rewrite Buffer, by drawing each character twice, once shifted down on the screen line on which it appears, and a second time, shifted up in the following screen line, and masked so that only the pixel rows belonging to the scrolled character are displayed, and not data from either before or after that character's data. |
| Colour RAM byte 0,<br>bits 2     | If set, the following characters will be rendered as background, allowing sprites to appear in front of them, even when sprites are set to background.                                                                                                                                                                                                                                                                                                                                                                            |
| Colour RAM byte 0,<br>bits 0 - 1 | RESERVED, set to 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |
| Colour RAM byte 1,<br>bits 0 - 7 | Pixel row mask flags                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |

We can see that we still have the C64 style bottom 8 bits of the character number in the first screen byte. The second byte of screen memory gets five extra bits for that, allowing  $2^{13} = 8,192$  different characters to be used on a single screen. That's more than enough for unique characters covering an  $80 \times 50$  screen (which is possible to create with the

VIC-IV). The remaining bits allow for trimming of the character. This allows for variable width characters, which can be used to do things that would not normally be possible, such as using text mode for free horizontal placement of characters (or parts thereof). This was originally added to provide hardware support for proportional width fonts.

For the colour RAM, the second byte (byte 1) is the same as the C65, i.e., the lower half providing four bits of foreground colour, as on the C64, plus the optional VIC-III extended attributes. The C65 specifications document describes the behaviour when more than one of these are used together, most of which are logical, but there are a few combinations that behave differently than one might expect. For example, combining bold with blink causes the character to toggle between bold and normal mode. Bold mode itself is implemented by effectively acting as bit 4 of the foreground colour value, causing the colour to be drawn from different palette entries than usual.

However, if you do not need VIC-III extended attributes, you can instead use the upper four bits of the second byte of colour RAM to contain more bits for the colour index, allowing selection from the full range of 256 colour entries. This mode is activated by enabling the VIC-II's multi-colour mode while full-colour mode is active.

The C65 / VIC-III attributes and the use of 256 colour 8-bit values for various VIC-II colour registers is enabled by setting bit 5 of \$D031 (53297 decimal). Therefore this is highly recommended when using the VIC-IV mode, as otherwise certain functions will not behave as expected. Note that BOLD+REVERSE together has the meaning of selecting an alternate palette on the MEGA65, which differs from the C65.

Many effects are possible due to Super-Extended Attribute Mode. A few possibilities are explained in the following sub-sections.

## Using Super-Extended Attribute Mode

Super-Extended Attribute Mode requires double the screen RAM and colour RAM as the VIC-II/III text modes. This is because two bytes of each are required to define each character, instead of one. The screen RAM can be located anywhere in the 384KiB of main memory using registers \$D060 - \$D062 (53344 - 53346 decimal). The colour RAM can be located anywhere in the 32KiB colour RAM. Only the first 1 or 2 KiB of the colour RAM is visible at \$D800 - \$DBFF or \$D800 - \$DFFF (if the CRAM2K signal is set in bit 0 of \$D030, 53296 decimal). Thus if using a screen larger than  $40 \times 25$  characters use of the DMA controller or some other means may be required to access the full amount of colour RAM. Thus we will initially discuss using Super-Extender Attribute Mode with a  $40 \times 25$  character display, so that the use of DMA or other means to access the additional colour RAM.

The first step is to enable the Super-Extended Attribute Mode by asserting the *FCLRHI* and *CHR16* signals, by setting bits 2 and 0 of \$D054 (53332 decimal). As this is a VIC-IV

register, we must first enable the VIC-IV IO mode. The VIC-IV must also be configured to 40 column mode, by clearing the *H640* signal by clearing bit 7 of \$D031 (53297 decimal). This is because each pair of characters will be used to form a single character on screen, with one character requiring two screen RAM bytes, thus 80 screen RAM bytes are required to display 40 characters. Similarly 80 colour RAM bytes are required as well.

To understand this visually, it is helpful to first consider the normal C64 screen memory layout:

That is, each character cell uses one byte of screen RAM, and the addresses increase smoothly, both within lines, and between lines. Super-Extended Attribute Mode requires two bytes per character cell. So if you set \$D054 to \$05, for example, you will get screen addresses like this:

|      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|------|
| S400 | S402 | S404 | S406 | S408 | S409 | S40A | S40C | S40E | S410 | S412 | S414 | S416 | S418 | S41A | S41C | S41e | S420 | S422 | S424 | S426 | S428 | S42a | S42c | S42e | S430 | S432 | S434 | S436 | S438 | S43a | S43c | S43e | S440 | S442 | S444 | S446 | S448 | S44c | S44e | S44f | S44g | S44h | S44i | S44j | S44k | S44l | S44m | S44n | S44o | S44p | S44q | S44r | S44s | S44t | S44u | S44v | S44w | S44x | S44y | S44z |      |      |      |      |      |      |      |
| S428 | S429 | S42c | S42e | S42f | S42g | S42h | S42i | S42j | S42k | S42l | S42m | S42n | S42o | S42p | S42q | S42r | S42s | S42t | S42u | S42v | S42w | S42x | S42y | S42z | S42A | S42B | S42C | S42D | S42E | S42F | S42G | S42H | S42I | S42J | S42K | S42L | S42M | S42N | S42O | S42P | S42Q | S42R | S42S | S42T | S42U | S42V | S42W | S42X | S42Y | S42Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S450 | S452 | S454 | S456 | S458 | S459 | S45b | S45c | S456 | S45e | S460 | S462 | S464 | S466 | S468 | S469 | S46b | S46c | S46e | S470 | S472 | S474 | S476 | S478 | S479 | S47a | S47c | S47e | S47g | S47h | S47i | S47k | S47l | S47m | S47n | S47o | S47p | S47q | S47r | S47s | S47t | S47u | S47v | S47w | S47x | S47y | S47z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S478 | S479 | S47c | S47e | S480 | S482 | S484 | S486 | S488 | S48a | S48b | S48c | S48e | S48g | S48h | S48j | S48k | S48l | S48m | S48n | S48o | S48p | S48q | S48r | S48s | S48t | S48u | S48v | S48w | S48x | S48y | S48z | S48A | S48B | S48C | S48D | S48E | S48F | S48G | S48H | S48I | S48J | S48K | S48L | S48M | S48N | S48O | S48P | S48Q | S48R | S48S | S48T | S48U | S48V | S48W | S48X | S48Y | S48Z |      |      |      |      |      |      |      |      |      |      |
| S48o | S49d | S49a | S49b | S49c | S49e | S49f | S49g | S49h | S49i | S49j | S49k | S49l | S49m | S49n | S49o | S49p | S49q | S49r | S49s | S49t | S49u | S49v | S49w | S49x | S49y | S49z | S49A | S49B | S49C | S49D | S49E | S49F | S49G | S49H | S49I | S49J | S49K | S49L | S49M | S49N | S49O | S49P | S49Q | S49R | S49S | S49T | S49U | S49V | S49W | S49X | S49Y | S49Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S49c | S49a | S49c | S49e | S49f | S49g | S49h | S49i | S49j | S49k | S49l | S49m | S49n | S49o | S49p | S49q | S49r | S49s | S49t | S49u | S49v | S49w | S49x | S49y | S49z | S49A | S49B | S49C | S49D | S49E | S49F | S49G | S49H | S49I | S49J | S49K | S49L | S49M | S49N | S49O | S49P | S49Q | S49R | S49S | S49T | S49U | S49V | S49W | S49X | S49Y | S49Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S49b | S49c | S49d | S49e | S49f | S49g | S49h | S49i | S49j | S49k | S49l | S49m | S49n | S49o | S49p | S49q | S49r | S49s | S49t | S49u | S49v | S49w | S49x | S49y | S49z | S49A | S49B | S49C | S49D | S49E | S49F | S49G | S49H | S49I | S49J | S49K | S49L | S49M | S49N | S49O | S49P | S49Q | S49R | S49S | S49T | S49U | S49V | S49W | S49X | S49Y | S49Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S510 | S512 | S514 | S516 | S518 | S51a | S51b | S51c | S51d | S51e | S51f | S51g | S51h | S51i | S51j | S51k | S51l | S51m | S51n | S51o | S51p | S51q | S51r | S51s | S51t | S51u | S51v | S51w | S51x | S51y | S51z | S51A | S51B | S51C | S51D | S51E | S51F | S51G | S51H | S51I | S51J | S51K | S51L | S51M | S51N | S51O | S51P | S51Q | S51R | S51S | S51T | S51U | S51V | S51W | S51X | S51Y | S51Z |      |      |      |      |      |      |      |      |      |      |      |
| S518 | S51a | S51c | S51e | S51g | S520 | S522 | S524 | S526 | S528 | S52a | S52c | S52e | S52g | S52i | S52k | S52m | S52n | S52o | S52p | S52q | S52r | S52s | S52t | S52u | S52v | S52w | S52x | S52y | S52z | S52A | S52B | S52C | S52D | S52E | S52F | S52G | S52H | S52I | S52J | S52K | S52L | S52M | S52N | S52O | S52P | S52Q | S52R | S52S | S52T | S52U | S52V | S52W | S52X | S52Y | S52Z |      |      |      |      |      |      |      |      |      |      |      |      |
| S540 | S542 | S544 | S546 | S548 | S549 | S54a | S54c | S54e | S54g | S54h | S54i | S54j | S54k | S54l | S54m | S54n | S54o | S54p | S54q | S54r | S54s | S54t | S54u | S54v | S54w | S54x | S54y | S54z | S54A | S54B | S54C | S54D | S54E | S54F | S54G | S54H | S54I | S54J | S54K | S54L | S54M | S54N | S54O | S54P | S54Q | S54R | S54S | S54T | S54U | S54V | S54W | S54X | S54Y | S54Z |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S568 | S569 | S56c | S56e | S56g | S56h | S56j | S56k | S56l | S56m | S56n | S56o | S56p | S56q | S56r | S56s | S56t | S56u | S56v | S56w | S56x | S56y | S56z | S56A | S56B | S56C | S56D | S56E | S56F | S56G | S56H | S56I | S56J | S56K | S56L | S56M | S56N | S56O | S56P | S56Q | S56R | S56S | S56T | S56U | S56V | S56W | S56X | S56Y | S56Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S590 | S592 | S594 | S596 | S598 | S599 | S59b | S59c | S59d | S59e | S59f | S59g | S59h | S59i | S59j | S59k | S59l | S59m | S59n | S59o | S59p | S59q | S59r | S59s | S59t | S59u | S59v | S59w | S59x | S59y | S59z | S59A | S59B | S59C | S59D | S59E | S59F | S59G | S59H | S59I | S59J | S59K | S59L | S59M | S59N | S59O | S59P | S59Q | S59R | S59S | S59T | S59U | S59V | S59W | S59X | S59Y | S59Z |      |      |      |      |      |      |      |      |      |      |      |
| S59b | S59c | S59d | S59e | S59f | S59g | S59h | S59i | S59j | S59k | S59l | S59m | S59n | S59o | S59p | S59q | S59r | S59s | S59t | S59u | S59v | S59w | S59x | S59y | S59z | S59A | S59B | S59C | S59D | S59E | S59F | S59G | S59H | S59I | S59J | S59K | S59L | S59M | S59N | S59O | S59P | S59Q | S59R | S59S | S59T | S59U | S59V | S59W | S59X | S59Y | S59Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S600 | S602 | S604 | S606 | S608 | S609 | S610 | S612 | S614 | S616 | S618 | S61a | S61c | S61e | S61g | S61i | S61k | S61m | S61o | S61p | S61q | S61r | S61s | S61t | S61u | S61v | S61w | S61x | S61y | S61z | S61A | S61B | S61C | S61D | S61E | S61F | S61G | S61H | S61I | S61J | S61K | S61L | S61M | S61N | S61O | S61P | S61Q | S61R | S61S | S61T | S61U | S61V | S61W | S61X | S61Y | S61Z |      |      |      |      |      |      |      |      |      |      |      |      |
| S608 | S609 | S61a | S61b | S61c | S61d | S61e | S61f | S61g | S61h | S61i | S61j | S61k | S61l | S61m | S61n | S61o | S61p | S61q | S61r | S61s | S61t | S61u | S61v | S61w | S61x | S61y | S61z | S61A | S61B | S61C | S61D | S61E | S61F | S61G | S61H | S61I | S61J | S61K | S61L | S61M | S61N | S61O | S61P | S61Q | S61R | S61S | S61T | S61U | S61V | S61W | S61X | S61Y | S61Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S630 | S632 | S634 | S636 | S638 | S639 | S640 | S642 | S644 | S646 | S648 | S649 | S650 | S652 | S654 | S656 | S658 | S659 | S660 | S662 | S664 | S666 | S668 | S669 | S670 | S672 | S674 | S676 | S678 | S679 | S680 | S681 | S682 | S683 | S684 | S685 | S686 | S687 | S688 | S689 | S68a | S68b | S68c | S68d | S68e | S68f | S68g | S68h | S68i | S68j | S68k | S68l | S68m | S68n | S68o | S68p | S68q | S68r | S68s | S68t | S68u | S68v | S68w | S68x | S68y | S68z |      |      |
| S658 | S659 | S665 | S666 | S667 | S668 | S669 | S670 | S671 | S672 | S673 | S674 | S675 | S676 | S677 | S678 | S679 | S67a | S67b | S67c | S67d | S67e | S67f | S67g | S67h | S67i | S67j | S67k | S67l | S67m | S67n | S67o | S67p | S67q | S67r | S67s | S67t | S67u | S67v | S67w | S67x | S67y | S67z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S680 | S682 | S684 | S686 | S688 | S689 | S690 | S692 | S694 | S696 | S698 | S699 | S69b | S69c | S69d | S69e | S69f | S69g | S69h | S69i | S69j | S69k | S69l | S69m | S69n | S69o | S69p | S69q | S69r | S69s | S69t | S69u | S69v | S69w | S69x | S69y | S69z | S69A | S69B | S69C | S69D | S69E | S69F | S69G | S69H | S69I | S69J | S69K | S69L | S69M | S69N | S69O | S69P | S69Q | S69R | S69S | S69T | S69U | S69V | S69W | S69X | S69Y | S69Z |      |      |      |      |      |
| S698 | S699 | S69c | S69d | S69e | S69f | S69g | S69h | S69i | S69j | S69k | S69l | S69m | S69n | S69o | S69p | S69q | S69r | S69s | S69t | S69u | S69v | S69w | S69x | S69y | S69z | S69A | S69B | S69C | S69D | S69E | S69F | S69G | S69H | S69I | S69J | S69K | S69L | S69M | S69N | S69O | S69P | S69Q | S69R | S69S | S69T | S69U | S69V | S69W | S69X | S69Y | S69Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S69b | S69c | S69d | S69e | S69f | S69g | S69h | S69i | S69j | S69k | S69l | S69m | S69n | S69o | S69p | S69q | S69r | S69s | S69t | S69u | S69v | S69w | S69x | S69y | S69z | S69A | S69B | S69C | S69D | S69E | S69F | S69G | S69H | S69I | S69J | S69K | S69L | S69M | S69N | S69O | S69P | S69Q | S69R | S69S | S69T | S69U | S69V | S69W | S69X | S69Y | S69Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S69d | S69e | S69f | S69g | S69h | S69i | S69j | S69k | S69l | S69m | S69n | S69o | S69p | S69q | S69r | S69s | S69t | S69u | S69v | S69w | S69x | S69y | S69z | S69A | S69B | S69C | S69D | S69E | S69F | S69G | S69H | S69I | S69J | S69K | S69L | S69M | S69N | S69O | S69P | S69Q | S69R | S69S | S69T | S69U | S69V | S69W | S69X | S69Y | S69Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S700 | S702 | S704 | S706 | S708 | S709 | S70a | S70b | S70c | S70d | S70e | S70f | S70g | S70h | S70i | S70j | S70k | S70l | S70m | S70n | S70o | S70p | S70q | S70r | S70s | S70t | S70u | S70v | S70w | S70x | S70y | S70z | S70A | S70B | S70C | S70D | S70E | S70F | S70G | S70H | S70I | S70J | S70K | S70L | S70M | S70N | S70O | S70P | S70Q | S70R | S70S | S70T | S70U | S70V | S70W | S70X | S70Y | S70Z |      |      |      |      |      |      |      |      |      |      |
| S708 | S710 | S722 | S724 | S726 | S728 | S729 | S72e | S72f | S72g | S72h | S730 | S732 | S734 | S736 | S738 | S739 | S73b | S73c | S73d | S73e | S73f | S73g | S73h | S73i | S73j | S73k | S73l | S73m | S73n | S73o | S73p | S73q | S73r | S73s | S73t | S73u | S73v | S73w | S73x | S73y | S73z | S73A | S73B | S73C | S73D | S73E | S73F | S73G | S73H | S73I | S73J | S73K | S73L | S73M | S73N | S73O | S73P | S73Q | S73R | S73S | S73T | S73U | S73V | S73W | S73X | S73Y | S73Z |
| S720 | S722 | S724 | S726 | S728 | S729 | S730 | S732 | S734 | S736 | S738 | S739 | S73b | S73c | S73d | S73e | S73f | S73g | S73h | S73i | S73j | S73k | S73l | S73m | S73n | S73o | S73p | S73q | S73r | S73s | S73t | S73u | S73v | S73w | S73x | S73y | S73z | S73A | S73B | S73C | S73D | S73E | S73F | S73G | S73H | S73I | S73J | S73K | S73L | S73M | S73N | S73O | S73P | S73Q | S73R | S73S | S73T | S73U | S73V | S73W | S73X | S73Y | S73Z |      |      |      |      |      |
| S748 | S749 | S74c | S74e | S74f | S74g | S74h | S74i | S74j | S74k | S74l | S74m | S74n | S74o | S74p | S74q | S74r | S74s | S74t | S74u | S74v | S74w | S74x | S74y | S74z | S74A | S74B | S74C | S74D | S74E | S74F | S74G | S74H | S74I | S74J | S74K | S74L | S74M | S74N | S74O | S74P | S74Q | S74R | S74S | S74T | S74U | S74V | S74W | S74X | S74Y | S74Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S748 | S749 | S74c | S74e | S74f | S74g | S74h | S74i | S74j | S74k | S74l | S74m | S74n | S74o | S74p | S74q | S74r | S74s | S74t | S74u | S74v | S74w | S74x | S74y | S74z | S74A | S74B | S74C | S74D | S74E | S74F | S74G | S74H | S74I | S74J | S74K | S74L | S74M | S74N | S74O | S74P | S74Q | S74R | S74S | S74T | S74U | S74V | S74W | S74X | S74Y | S74Z |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |
| S770 | S772 | S774 | S776 | S778 | S77a | S77b | S77c | S77d | S77e | S77f | S77g | S77h | S77i | S77j | S77k | S77l | S77m | S77n | S77o | S77p | S77q | S77r | S    |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |      |

There are two things to notice in the above table: First, the address advances by two bytes for each character cell, because two bytes are required to define each character. Second, the start address of each screen line still only advances by 40 (\$28 in hexadecimal). This isn't what we really want, because it means that half of the previous row will get displayed again on each current row. This is fixed by setting the number of bytes to advance each screen row in \$D058 (LSB) and \$D059 (MSB). So in this case, we want to increase the number of bytes skipped each line from 40 bytes, to 80 bytes, which we can do by setting \$D058 to 80 (\$50 in hexadecimal), and \$D059 to 0. This gives us a screen layout like this:

It is possible to use Super-Extended Attribute Mode from C65 mode, by setting the screen to 80 columns, as the C65 ROM sets up 2KiB for both the screen RAM and colour RAM, and this automatically sets \$D058 and \$D059 to the correct value for  $40 \times 2 = 80$  bytes per screen line. The user need only to treat each character pair as a single Super-Extended Attribute character, and to enable Super-Extended Attribute Mode, as described above.

Because pairs of colour RAM and screen RAM bytes are used to define each character, care must be taken to initialise and manipulate the screen. A good approach is to set the text colour to black, because this is colour code 0, and then to fill the screen with @ characters, because that is character code 0. You can then have several ways to manipulate the screen. You can use the normal PRINT command and carefully construct strings that will put the correct values into each screen and colour byte pair. Another approach is to use the BANK and POKE commands to directly set the contents of the screen and colour RAM.

Managing a Super-Extended Attribute Mode screen in this way using BASIC 10 is of course rather a hack, and is only suggested as a relatively simple way to begin experimenting. You will almost certainly want to quickly move to using custom screen handling code, most probably in assembly, to manipulate Super-Extended Attribute Mode screens, although this approach of using BASIC 10 can be quite powerful, by allowing use of existing screen scrolling and other manipulations.

## XXX Example program

The following descriptions assume that you have implemented one of the methods described above to set the screen and colour RAM.

## **Full-Colour (256 colours per character) Text Mode (FCM)**

In normal VIC-II/III text mode, one byte is used for each row of pixels in a character. As a reminder for how those modes work, in hi-res mode, each pixel is either the background or foreground colour, based on the state of one bit in the byte. Multi-colour mode uses two bits to select between four possible colours, but as there are still only 8 bits to describe each row of 8 pixels, each pair of pixels has the same colour. The VIC-IV's full-colour text mode removes these limitations, and allows each pixel of a character to be chosen from the 256 colour of either the primary or alternate palette bank, without sacrificing horizontal resolution.

To do this, each character now requires 64 bytes of data. The address of the data is  $64 \times$  the character number, regardless of the character set address. FCM should normally be used with Super-Extended Attribute Mode (SEAM), so that more than 256 unique characters can be address. As SEAM allows the selection of 8,192 unique characters, this allows FCM character data to be placed anywhere in the first 512KiB of chip RAM (but note that most models of the MEGA65 have only 384KiB of chip RAM).

## **Nybl-colour (16 colours per character) Text Mode (NCM)**

The Nybl-Colour Mode (NCM) for text is similar to Full-Colour Text Mode, except that each byte of data describes two pixels using 4 bits each. This makes the NCM unique, because the characters will be 16 pixels wide, instead of the usual 8 pixels wide. This can be used to create colourful displays, without using as much memory as FCM, because fewer characters are required to cover the screen. Unlike the VIC-II's MCM, this mode does not result in a loss of horizontal resolution.

In NCM the lower four bits of the pixel colour comes from the upper or lower four bits of the pixel data. The upper four bits of the colour code come from the colour RAM data for the displayed character. This makes it possible to use all palette entries in NCM, although the limitation of 16 colours per character remains.

A further advantage of NCM is that it uses fewer bus cycles per pixel than FCM, because fewer character data fetches need to occur per raster line. Together with the reduced memory requirements, this makes NCM particularly useful for creating colourful multiple

layers of graphics. This allows the VIC-IV to display arcade style displays with more colours than many 16-bit computers.

XXX

## Alpha-Blending / Anti-Aliasing

XXX

## Flipping Characters

XXX

## Variable Width Fonts

There are 4 bits that allow trimming pixels from the right edge of characters when they are displayed. This has the effect of making characters narrower. This can be useful for making more attractive text displays, where narrow characters, such as "i" take less space than wider characters, such as "m", without having to use a bitmap display. This feature can be used to make it very efficient to display such variable-width text displays - both in terms of memory usage and processing time.

This feature can be combined with full-colour text mode, alpha blending mode and 4-bits per pixel mode to allow characters that consist of 15 levels of intensity between the background and foreground colour, and that are up to 16 pixels wide. Further, the GOTO bit can be used to implement negative kerning, so that character pairs like A and T do not have excessive white space between them when printed adjacently. The prudent use of these features can result in highly impressive text display, similar to that on modern 32-bit and 64-bit systems, but that are still efficient enough to be implemented on a relatively constrained system such as the MEGA65. The "MegaWAT!?" presentation software for the MEGA65 uses several of these features to produce its attractive anti-aliased proportional text display on slides.

XXX MEGAWat!? screenshot

XXX Example program

## Raster Re-write Buffer

If the GOTO bit is set for a character in Super-Extended Attribute Mode, instead of painting a character, the position on the raster is back-tracked (or advanced forward to) the pixel position specified in the low 10 bits of the screen memory bytes. If the vertical flip bit is set, then this has the alternate meaning of preventing the background colour from being painted. This combination can be used to print text material over the top of other text material, providing a crude supplement to the 8 hardware sprites. The amount of material is limited only by the raster time of the VIC-IV. Some experimentation will be required to determine how much can be achieved in PAL and NTSC modes.

If the GOTO bit is set for a character, and the character width reduction bits are also set, they are interpreted as a Y offset to add to the character data address, but only in Full Colour Mode. Setting Y=1 causes the character data to be fetched from 8 bytes later, i.e., the first row of character data will come from the address where the second row of character data would normally be fetched. Similarly for increased values the character data will be fetched from further character rows. With careful arrangement of characters in memory, it is possible to use this feature to provide free vertical placement of soft sprites, without needing to copy the character data.

This ability to draw multiple layers of text and graphics is highly powerful. For example, it can be used to provide multiple overlapping layers of separately scrollable graphics. This gives many of the advantages of bitplane-based play-fields on other computers, such as the Amiga, but without the disadvantages of bitplanes.

A good introduction to the Raster Re-write Buffer and its uses can be found in this video:

<https://www.youtube.com/watch?v=00bm5uBeBos&feature=youtu.be>

One important aspect of the RRB, is that the VIC-IV will display only the character data to the left of, and including, the last drawn character. This means that if you use the GOTO token to overwrite multiple layers of graphics, you must either make sure that the last layer reaches to the right-hand edge of the display, or you must include a GOTO token that moves the render position to the right-hand edge of the display.

XXX Example program

## SPRITES

## VIC-II/III Sprite Control

The control of sprites for C64 / VIC-II/III compatibility is unchanged from the C64. The only practical differences are very minor. In particular the VIC-IV uses ring-buffer for each

sprites data when rendering a raster. This means that a sprite can be displayed multiple times per raster line, thus potentially allowing for horizontal multiplexing.

## Extended Sprite Image Sets

On the VIC-II and VIC-III, all sprites must draw their image data from a single 16KB region of memory at any point in time. This limits the number of different sprite images to 256, because each sprite image occupies 64 bytes. In practice, the same 16KB region must also contain either bitmap, text or bitplane data, considerably reducing the number of sprite images that can be used at the same time.

The VIC-IV removes this limitation, by allowing sprite data to be placed anywhere in memory, although still on 64-byte boundaries. This is done by setting the SPRPTR16 signal (bit 7, \$D06E, decimal 53358), which tells the VIC-IV to expect two bytes per sprite pointer instead of one. These addresses are then absolute addresses, and ignore the 16KB VIC-II bank selection logic. Thus 16 bytes are required instead of 8 bytes. The list of pointers can also be placed anywhere in memory by setting the SPRPTRADR (\$D06C - \$D06D, 53356 - 53357 decimal) and SPRPTRBNK signals (bits 0 - 6, \$D06E, 53358 decimal). This allows for sprite data to be located anywhere in the first 4MB of RAM, and the sprite pointer list to be located anywhere in the first 8MB of RAM. Note that typical installations of the VIC-IV have only 384KB of connected RAM, so these limitations are of no practical effect. However, the upper bits of the SPRPTRBNK signal should be set to zero to avoid forward-compatibility problems.

One reason for supporting more sprite images is that sprites on the VIC-IV can require more than one 64 byte image slot. For example, enabling Extra-Wide Sprite Mode means that a sprite will require  $8 \times 21 = 168$  bytes, and will thus occupy four VIC-II style 64 byte sprite image slots. If variable height sprites are used, this can grow to as much as  $8 \times 255 = 2,040$  bytes per sprite.

## Variable Sprite Size

Sprites can be one of three widths with the VIC-IV:

1. Normal VIC-II width (24 pixels wide).
2. Extra Wide, where 64 bits (8 bytes) of data are used per raster line, instead of the VIC-II's 24. This results in sprites that are 64 pixels wide, unless Full-Colour Sprite Mode is selected for a sprite, in which case the sprite will be  $64 \text{ bits} \div 4 \text{ bits per pixel} = 16$  pixels wide.
3. Tiled mode, where the sprite is drawn repeatedly until the end of the raster line. Tiled mode should normally only be used with Extra Wide sprite mode, as the tiling

always occurs using the full 64 bit sprite data. Thus if you use tiled mode with normal 24 pixel wide mono or multi-colour sprites, the tiling will treat each 2 and 2/3 rows of sprite data as a single row, resulting in garbled displays.

To enable a sprite to be 64 pixels (or 16 pixels if in Full-Colour Sprite Mode), set the corresponding bit for the sprite in the SPRX64EN register at (\$D057, 53335 decimal). Enabling Full Colour mode for a sprite implicitly enables extended width mode, causes these sprites to be 16 pixels wide.

Similarly, sprites can be various heights: Sprites will be either the 21 pixels high of the VIC-II, or if the corresponding bit for the sprite is enabled in the SPRHGTEN signal (\$D055, 53333 decimal), then that sprite will be the number of pixels tall that is set in the SPRHGT register (\$D056, 53334 decimal).

## Variable Sprite Resolution

By default, sprites are the same resolution as on the VIC-II, i.e., each sprite pixel is two physical pixels wide and high. However, sprites can be made to use the native resolution, where sprite pixels are one physical pixel wide and/or high. This is achieved by setting the relevant bit for the sprite in the SPRENV400 (\$D076, 53366 decimal) registers to increase the vertical resolution on a sprite-by-sprite basis. The horizontal resolution for all sprites is either the normal VIC-II resolution, or if the SPR640 signal is set (bit 4 of \$D054, 53332 decimal), then sprites will have the same horizontal resolution as the physical pixels of the display.

## Sprite Palette Bank

The VIC-IV has four palette banks, compared with the single palette bank of the VIC-III. The VIC-IV allows the selection of separate palette banks for bitmap/text graphics and for sprites. This makes it easy to have very colourful displays, where the sprites have different colours to the rest of the display, or to use palette animation to achieve interesting visual effects in sprites, without disturbing the palette used by other elements of the display.

The sprite palette bank is selected by setting the SPRPALSEL signal in bits 2 and 3 of the register \$D070 (53360 decimal). It is possible to set this to the same bank as the bitmap/text display, or to select a different palette bank. Palette bank selection takes effect immediately. Don't forget that to be able to modify a palette, you have to also bank it to be the palette accessible via the palette bank registers at \$D100 - \$D3FF by setting the MAPEDPAL signal in bits 6 and 7 of \$D070.

# Full-Colour Sprite Mode

In addition to monochrome and multi-colour modes, the VIC-IV supports a new full-colour sprite mode. In this mode, four bits are used to encode each sprite pixel. However, unlike multi-colour mode where pairs of bits encode pairs of pixels, in full-colour mode the pixels remain at their normal horizontal resolution. The colour zero is considered transparent. If you wish to use black in a full-colour sprite, you must configure the palette bank that is selected for sprites so that one of the 15 colours for the specific sprite encodes black.

Full-colour sprite mode is selectable for each sprite by setting the appropriate bit in the SPR16EN register (\$D06B, 53355 decimal).

To enable the eight sprites to have 15 unique colours each, the sprite colour is drawn using the palette entry corresponding to:  $spritenumber \times 16 + nibblevalue$ , where *spritenumber* is the number of the sprite (from 0 to 7), and *nibblevalue* is the value of the half-byte that contains the sprite data for the pixel. In addition, if bitplane mode is enabled for this sprite, then 128 is added to the colour value, which makes it easy to switch between two colour schemes for a given sprite by changing only one bit in the SPRBPMEN register.

Because Full-Colour Sprite Mode requires four bits per pixel, sprites will be only six pixels wide, unless Extra Wide Sprite Mode is enabled for a sprite, in which case the sprite will be 16 pixels wide. Tiled Mode also works with Full-Colour Sprite Mode, and will result in the 16 full-colour pixels of the sprite being repeated until the end of the raster line.

The following BASIC program draws a Full-Colour Sprite in either C64 or C65 mode:

```
8 AD=56*64:IF PEEK(53272) AND 32 THEN GOTO 30: REM C65/C64 MODE DETECT
10 POKE 53295,ASC("G"):POKE 53295,ASC("S"): REM ENABLE MEGA65 VIC-IV FEATURES
20 AD=768+64: REM $0340 HEX FOR SPRITE
30 FOR I=AD TO AD+168:POKEI ,0:NEXT I
40 POKE 2040,AD/64: REM SET SPRITE NUMBER
50 POKE 53269,1: REM ENABLE SPRITE 0
60 POKE 53248,100:POKE 53249,100: REM PUT SPRITE ON SCREEN
70 POKE 53355,1: REM MAKE SPRITE 0 16-COLOUR
80 POKE 53335,1: REM MAKE SPRITE 0 USE 64 BITS OF DATA = 16 X 4-BIT PIXELS
90 POKE 53287,10: REM MAKE PINK THE TRANSPARENT COLOUR
100 GOSUB 900: REM READ MULTI-COLOUR SPRITE
```

```
899 END
```

```
900 REM LOAD SPRITE
910 READN$:IFN$="END"THEN RETURN
920 GOSUB 1000
930 GOTO 910
```

```
1000 REM DECODE STRING OF NIBBLES IN N$ AT ADDRESS AD
1010 L=LEN(N$)
1020 FOR I=1 TO (L/2+1):POKE AD+I,0
1030 FOR I= 1 TO L:N=ASC(MID$(N$,I,1))-ASC("0")
1040 A=AD+INT((I-1)/2):IF (I AND 1)=1 THEN N=N*16
1050 V=PEEK(A):POKE A,V OR N:NEXTI
1060 AD=AD+INT(I/2)
1070 IF (L AND 1) THEN AD=AD+1
1080 RETURN
```

```
1998 REM SPRITE DATA FOLLOWS
1999 REM 0 = TRANSPARENT, A-0 = COLOURS 1 TO 15
2000 DATA "BABCDEFHGIJKLMNOP"
2010 DATA "Aooooooooooooooo"
2020 DATA "ACoooooooooooooo"
2030 DATA "ACCAACCCCCCCCCCCC"
2040 DATA "ACCCCAACCCCCCCCCC"
2050 DATA "ACCCCCACCCCCCCCCC"
2060 DATA "ACCCCCAACCCCCCCCC"
2070 DATA "ACCCCCCACCACCCCCC"
2080 DATA "ACCCCCCCCCACCCCCCCC"
2090 DATA "ACCCCCCCCCACCCCCC"
2100 DATA "ACCCCCCCCCACCCCCC"
2110 DATA "ACCCCCCCCCCACCCCC"
2120 DATA "ACCCCCCCCCCCCCACCC"
2130 DATA "ACCCCCCCCCCCCCACCC"
2140 DATA "ACCCCCCCCCCCCCACCC"
2150 DATA "ACCCCCCCCCCCCCCCCCA"
```

# VIC-II / C64 REGISTERS

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D000       | 53248      |            |            |            |            |            |            |            | S0X        |
| D001       | 53249      |            |            |            |            |            |            |            | S0Y        |
| D002       | 53250      |            |            |            |            |            |            |            | S1X        |
| D003       | 53251      |            |            |            |            |            |            |            | S1Y        |
| D004       | 53252      |            |            |            |            |            |            |            | S2X        |
| D005       | 53253      |            |            |            |            |            |            |            | S2Y        |
| D006       | 53254      |            |            |            |            |            |            |            | S3X        |
| D007       | 53255      |            |            |            |            |            |            |            | S3Y        |
| D008       | 53256      |            |            |            |            |            |            |            | S4X        |
| D009       | 53257      |            |            |            |            |            |            |            | S4Y        |
| D00A       | 53258      |            |            |            |            |            |            |            | S5X        |
| D00B       | 53259      |            |            |            |            |            |            |            | S5Y        |
| D00C       | 53260      |            |            |            |            |            |            |            | S6X        |
| D00D       | 53261      |            |            |            |            |            |            |            | S6Y        |
| D00E       | 53262      |            |            |            |            |            |            |            | S7X        |
| D00F       | 53263      |            |            |            |            |            |            |            | S7Y        |
| D010       | 53264      |            |            |            |            |            |            |            | SXMSB      |
| D011       | 53265      | RC         | ECM        | BMM        | BLNK       | RSEL       |            |            | YSCL       |
| D012       | 53266      |            |            |            |            |            |            |            | RC         |
| D013       | 53267      |            |            |            |            |            |            |            | LPX        |
| D014       | 53268      |            |            |            |            |            |            |            | LPY        |
| D015       | 53269      |            |            |            |            |            |            |            | SE         |
| D016       | 53270      | -          | RST        | MCM        | CSEL       |            |            |            | XSCL       |
| D017       | 53271      |            |            |            |            |            |            |            | SEXY       |
| D018       | 53272      |            | VS         |            |            | CB         |            |            | -          |
| D019       | 53273      | -          |            |            | ILP        | ISSC       | ISBC       |            | RIRO       |
| D01A       | 53274      |            | -          |            | MISSC      | MISBC      |            |            | MRIRO      |
| D01B       | 53275      |            |            |            |            |            |            |            | BSP        |
| D01C       | 53276      |            |            |            |            |            |            |            | SCM        |
| D01D       | 53277      |            |            |            |            |            |            |            | SEXX       |
| D01E       | 53278      |            |            |            |            |            |            |            | SSC        |
| D01F       | 53279      |            |            |            |            |            |            |            | SBC        |
| D020       | 53280      |            | -          |            |            |            |            |            | BORDERCOL  |
| D021       | 53281      |            | -          |            |            |            |            |            | SCREENCOL  |
| D022       | 53282      |            | -          |            |            |            |            |            | MC1        |
| D023       | 53283      |            | -          |            |            |            |            |            | MC2        |

continued ...

...continued

| HEX  | DEC   | DB7 | DB6 | DB5 | DB4 | DB3     | DB2 | DB1 | DB0      |
|------|-------|-----|-----|-----|-----|---------|-----|-----|----------|
| D024 | 53284 |     | -   |     |     |         |     |     | MC3      |
| D025 | 53285 |     |     |     |     | SPRMC0  |     |     |          |
| D026 | 53286 |     |     |     |     | SPRMC1  |     |     |          |
| D027 | 53287 |     |     |     |     | SPR0COL |     |     |          |
| D028 | 53288 |     |     |     |     | SPR1COL |     |     |          |
| D029 | 53289 |     |     |     |     | SPR2COL |     |     |          |
| D02A | 53290 |     |     |     |     | SPR3COL |     |     |          |
| D02B | 53291 |     |     |     |     | SPR4COL |     |     |          |
| D02C | 53292 |     |     |     |     | SPR5COL |     |     |          |
| D02D | 53293 |     |     |     |     | SPR6COL |     |     |          |
| D02E | 53294 |     |     |     |     | SPR7COL |     |     |          |
| D030 | 53296 |     |     |     | -   |         |     |     | C128FAST |

- **BLNK** disable display
- **BMM** bitmap mode
- **BORDERCOL** display border colour (16 colour)
- **BSP** sprite background priority bits
- **C128FAST** 2MHz select (for C128 2MHz emulation)
- **CB** character set address location ( $\times$  1KiB)
- **CSEL** 38/40 column select
- **ECM** extended background mode
- **ILP** light pen indicate or acknowledge
- **ISBC** sprite:bitmap collision indicate or acknowledge
- **ISSC** sprite:sprite collision indicate or acknowledge
- **LPX** Coarse horizontal beam position (was lightpen X)
- **LPY** Coarse vertical beam position (was lightpen Y)
- **MC1** multi-colour 1 (16 colour)
- **MC2** multi-colour 2 (16 colour)
- **MC3** multi-colour 3 (16 colour)
- **MCM** Multi-colour mode
- **MISBC** mask sprite:bitmap collision IRQ

- **MISSC** mask sprite:sprite collision IRQ
- **MRIRQ** mask raster IRQ
- **RC** raster compare bit 8
- **RIRQ** raster compare indicate or acknowledge
- **RSEL** 24/25 row select
- **RST** Disables video output on MAX Machine(tm) VIC-II 6566. Ignored on normal C64s and the MEGA65
- **S0X** sprite 0 horizontal position
- **S0Y** sprite 0 vertical position
- **S1X** sprite 1 horizontal position
- **S1Y** sprite 1 vertical position
- **S2X** sprite 2 horizontal position
- **S2Y** sprite 2 vertical position
- **S3X** sprite 3 horizontal position
- **S3Y** sprite 3 vertical position
- **S4X** sprite 4 horizontal position
- **S4Y** sprite 4 vertical position
- **S5X** sprite 5 horizontal position
- **S5Y** sprite 5 vertical position
- **S6X** sprite 6 horizontal position
- **S6Y** sprite 6 vertical position
- **S7X** sprite 7 horizontal position
- **S7Y** sprite 7 vertical position
- **SBC** sprite/foreground collision indicate bits
- **SCM** sprite multicolour enable bits
- **SCREENCOL** screen colour (16 colour)
- **SE** sprite enable bits
- **SEX<sub>X</sub>** sprite horizontal expansion enable bits
- **SEX<sub>Y</sub>** sprite vertical expansion enable bits

- **SPR0COL** sprite 0 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR1COL** sprite 1 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR2COL** sprite 2 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR3COL** sprite 3 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR4COL** sprite 4 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR5COL** sprite 5 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR6COL** sprite 6 colour / 16-colour sprite transparency colour (lower nybl)
- **SPR7COL** sprite 7 colour / 16-colour sprite transparency colour (lower nybl)
- **SPRMC0** Sprite multi-colour 0
- **SPRMC1** Sprite multi-colour 1
- **SSC** sprite/sprite collision indicate bits
- **SXMSB** sprite horizontal position MSBs
- **VS** screen address ( $\times$  1KiB)
- **XSCL** horizontal smooth scroll
- **YSCL** 24/25 vertical smooth scroll

## VIC-III / C65 REGISTERS

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D020       | 53280      |            |            |            |            | BORDERCOL  |            |            |            |
| D021       | 53281      |            |            |            |            | SCREENCOL  |            |            |            |
| D022       | 53282      |            |            |            |            | MC1        |            |            |            |
| D023       | 53283      |            |            |            |            | MC2        |            |            |            |
| D024       | 53284      |            |            |            |            | MC3        |            |            |            |
| D025       | 53285      |            |            |            |            | SPRMC0     |            |            |            |
| D026       | 53286      |            |            |            |            | SPRMC1     |            |            |            |
| D02F       | 53295      |            |            |            |            | KEY        |            |            |            |
| D030       | 53296      | ROME       | CROM9      | ROMC       | ROMA       | ROM8       | PAL        | EXTSYNC    | CRAM2K     |
| D031       | 53297      | H640       | FAST       | ATTR       | BPM        | V400       | H1280      | MONO       | INT        |
| D033       | 53299      | B0ADODD    |            |            | -          | B0ADEVN    |            |            | -          |
| D034       | 53300      | B1ADODD    |            |            | -          | B1ADEVN    |            |            | -          |
| D035       | 53301      | B2ADODD    |            |            | -          | B2ADEVN    |            |            | -          |

continued ...

...continued

| <b>HEX</b>  | <b>DEC</b>          | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|-------------|---------------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D036        | 53302               |            | B3ADODD    |            | -          |            | B3ADEVN    |            | -          |
| D037        | 53303               |            | B4ADODD    |            | -          |            | B4ADEVN    |            | -          |
| D038        | 53304               |            | B5ADODD    |            | -          |            | B5ADEVN    |            | -          |
| D039        | 53305               |            | B6ADODD    |            | -          |            | B6ADEVN    |            | -          |
| D03A        | 53306               |            | B7ADODD    |            | -          |            | B7ADEVN    |            | -          |
| D03B        | 53307               |            |            |            |            | BPCOMP     |            |            |            |
| D03C        | 53308               |            |            |            |            | BPX        |            |            |            |
| D03D        | 53309               |            |            |            |            | BPY        |            |            |            |
| D03E        | 53310               |            |            |            |            | HPOS       |            |            |            |
| D03F        | 53311               |            |            |            |            | VPOS       |            |            |            |
| D040        | 53312               |            |            |            |            | B0PIX      |            |            |            |
| D041        | 53313               |            |            |            |            | B1PIX      |            |            |            |
| D042        | 53314               |            |            |            |            | B2PIX      |            |            |            |
| D043        | 53315               |            |            |            |            | B3PIX      |            |            |            |
| D044        | 53316               |            |            |            |            | B4PIX      |            |            |            |
| D045        | 53317               |            |            |            |            | B5PIX      |            |            |            |
| D046        | 53318               |            |            |            |            | B6PIX      |            |            |            |
| D047        | 53319               |            |            |            |            | B7PIX      |            |            |            |
| D100 - D1FF | 53504<br>-<br>53759 |            |            |            |            | PALRED     |            |            |            |
| D200 - D2FF | 53760<br>-<br>54015 |            |            |            |            | PALGREEN   |            |            |            |
| D300 - D3FF | 54016<br>-<br>54271 |            |            |            |            | PALBLUE    |            |            |            |

- **ATTR** Enable extended attributes and 8 bit colour entries
- **BOADEVN** - Bitplane 0 address, even lines
- **BOADODD** - Bitplane 0 address, odd lines
- **B0PIX** Display Address Translater (DAT) Bitplane 0 port
- **B1ADEVN** - Bitplane 1 address, even lines
- **B1ADODD** - Bitplane 1 address, odd lines
- **B1PIX** Display Address Translater (DAT) Bitplane 1 port
- **B2ADEVN** - Bitplane 2 address, even lines

- **B2ADODD** - Bitplane 2 address, odd lines
- **B2PIX** Display Address Translater (DAT) Bitplane 2 port
- **B3ADEVN** - Bitplane 3 address, even lines
- **B3ADODD** - Bitplane 3 address, odd lines
- **B3PIX** Display Address Translater (DAT) Bitplane 3 port
- **B4ADEVN** - Bitplane 4 address, even lines
- **B4ADODD** - Bitplane 4 address, odd lines
- **B4PIX** Display Address Translater (DAT) Bitplane 4 port
- **B5ADEVN** - Bitplane 5 address, even lines
- **B5ADODD** - Bitplane 5 address, odd lines
- **B5PIX** Display Address Translater (DAT) Bitplane 5 port
- **B6ADEVN** - Bitplane 6 address, even lines
- **B6ADODD** - Bitplane 6 address, odd lines
- **B6PIX** Display Address Translater (DAT) Bitplane 6 port
- **B7ADEVN** - Bitplane 7 address, even lines
- **B7ADODD** - Bitplane 7 address, odd lines
- **B7PIX** Display Address Translater (DAT) Bitplane 7 port
- **BORDERCOL** display border colour (256 colour)
- **BPCOMP** Complement bitplane flags
- **BPM** Bit-Plane Mode
- **BPX** Bitplane X
- **BPY** Bitplane Y
- **CRAM2K** Map 2nd KB of colour RAM \$DC00-\$FFFF
- **CROM9** Select between C64 and C65 charset.
- **EXTSYNC** Enable external video sync (genlock input)
- **FAST** Enable C65 FAST mode (~3.5MHz)
- **H1280** Enable 1280 horizontal pixels (not implemented)
- **H640** Enable C64 640 horizontal pixels / 80 column mode

- **HPOS** Bitplane X Offset
- **INT** Enable VIC-III interlaced mode
- **KEY** Write \$A5 then \$96 to enable C65/VIC-III IO registers
- **MC1** multi-colour 1 (256 colour)
- **MC2** multi-colour 2 (256 colour)
- **MC3** multi-colour 3 (256 colour)
- **MONO** Enable VIC-III MONO video output (not implemented)
- **PAL** Use PALETTE ROM or RAM entries for colours 0 - 15
- **PALBLUE** blue palette values (reversed nybl order)
- **PALGREEN** green palette values (reversed nybl order)
- **PALRED** red palette values (reversed nybl order)
- **ROM8** Map C65 ROM \$8000
- **ROMA** Map C65 ROM \$A000
- **ROMC** Map C65 ROM \$C000
- **ROME** Map C65 ROM \$E000
- **SCREENCOL** screen colour (256 colour)
- **SPRMC0** Sprite multi-colour 0 (8-bit for selection of any palette colour)
- **SPRMC1** Sprite multi-colour 1 (8-bit for selection of any palette colour)
- **V400** Enable 400 vertical pixels
- **VPOS** Bitplane Y Offset

## VIC-IV / MEGA65 SPECIFIC REGISTERS

| HEX  | DEC   | DB7 | DB6 | DB5 | DB4       | DB3 | DB2 | DB1 | DB0 |
|------|-------|-----|-----|-----|-----------|-----|-----|-----|-----|
| D020 | 53280 |     |     |     | BORDERCOL |     |     |     |     |
| D021 | 53281 |     |     |     | SCREENCOL |     |     |     |     |
| D022 | 53282 |     |     |     | MC1       |     |     |     |     |
| D023 | 53283 |     |     |     | MC2       |     |     |     |     |
| D024 | 53284 |     |     |     | MC3       |     |     |     |     |
| D025 | 53285 |     |     |     | SPRMC0    |     |     |     |     |

continued ...

...continued

| HEX  | DEC   | DB7     | DB6      | DB5    | DB4      | DB3  | DB2       | DB1    | DB0      |
|------|-------|---------|----------|--------|----------|------|-----------|--------|----------|
| D026 | 53286 |         |          |        |          |      |           |        | SPRMC1   |
| D02F | 53295 |         |          |        |          |      |           |        | KEY      |
| D048 | 53320 |         |          |        |          |      |           |        | TBDRPOS  |
| D049 | 53321 |         |          |        | SPRBPMEN |      |           |        | TBDRPOS  |
| D04A | 53322 |         |          |        |          |      |           |        | BBDRPOS  |
| D04B | 53323 |         |          |        | SPRBPMEN |      |           |        | BBDRPOS  |
| D04C | 53324 |         |          |        |          |      |           |        | TEXTXPOS |
| D04D | 53325 |         |          |        | SPRTILEN |      |           |        | TEXTXPOS |
| D04E | 53326 |         |          |        |          |      |           |        | TEXTYPOS |
| D04F | 53327 |         |          |        | SPRTILEN |      |           |        | TEXTYPOS |
| D050 | 53328 |         |          |        |          |      |           |        | XPOS     |
| D051 | 53329 | NORRDEL | DBLRR    |        |          |      |           |        | XPOS     |
| D052 | 53330 |         |          |        |          |      |           |        | FNRASTER |
| D053 | 53331 | FNRST   | SHDEMU   |        |          | -    |           |        | FNRASTER |
| D054 | 53332 | ALPHEN  | VFAST    | PALEMU | SPR640   | SMTH | FCLRHI    | FCLRLO | CHR1     |
| D055 | 53333 |         |          |        |          |      | SPRHGTEN  |        |          |
| D056 | 53334 |         |          |        |          |      | SPRHGHT   |        |          |
| D057 | 53335 |         |          |        |          |      | SPRX64EN  |        |          |
| D058 | 53336 |         |          |        |          |      | LINESTEP  |        |          |
| D059 | 53337 |         |          |        |          |      | LINESTEP  |        |          |
| D05A | 53338 |         |          |        |          |      | CHRXSCL   |        |          |
| D05B | 53339 |         |          |        |          |      | CHRYSCL   |        |          |
| D05C | 53340 |         |          |        |          |      | SIDBDRWD  |        |          |
| D05D | 53341 | HOTREG  | RSTDELEN |        |          |      | SIDBDRWD  |        |          |
| D05E | 53342 |         |          |        |          |      | CHRCOUNT  |        |          |
| D05F | 53343 |         |          |        |          |      | SPRXSMSBS |        |          |
| D060 | 53344 |         |          |        |          |      | SCRNPTR   |        |          |
| D061 | 53345 |         |          |        |          |      | SCRNPTR   |        |          |
| D062 | 53346 |         |          |        |          |      | SCRNPTR   |        |          |
| D063 | 53347 | EXGLYPH | -        |        | CHRCOUNT |      |           |        | SCRNPTR  |
| D064 | 53348 |         |          |        |          |      | COLPTR    |        |          |
| D065 | 53349 |         |          |        |          |      | COLPTR    |        |          |
| D068 | 53352 |         |          |        |          |      | CHARPTR   |        |          |
| D069 | 53353 |         |          |        |          |      | CHARPTR   |        |          |
| D06A | 53354 |         |          |        |          |      | CHARPTR   |        |          |
| D06B | 53355 |         |          |        |          |      | SPRP16EN  |        |          |
| D06C | 53356 |         |          |        |          |      | SPRPTRADR |        |          |
| D06D | 53357 |         |          |        |          |      | SPRPTRADR |        |          |

continued ...

...continued

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b>   | <b>DB6</b> | <b>DB5</b>  | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|--------------|------------|-------------|------------|------------|------------|------------|------------|
| D06E       | 53358      | SPRPTR16     |            |             |            | SPRPTRBNK  |            |            |            |
| D06F       | 53359      | PALNTSC      | VGAHDTV    |             |            | RASLINE0   |            |            |            |
| D070       | 53360      | MAPEDPAL     |            | BTPALSEL    |            | SPRPALSEL  |            | ABTPALSEL  |            |
| D071       | 53361      |              |            | BP16ENS     |            |            |            |            |            |
| D072       | 53362      |              |            | SPRYADJ     |            |            |            |            |            |
| D073       | 53363      | RASTERHEIGHT |            |             |            | ALPHADELAY |            |            |            |
| D074       | 53364      |              |            | SPRENALPHA  |            |            |            |            |            |
| D075       | 53365      |              |            | SPRALPHAVAL |            |            |            |            |            |
| D076       | 53366      |              |            | SPRENV400   |            |            |            |            |            |
| D077       | 53367      |              |            | SRPYMSBS    |            |            |            |            |            |
| D078       | 53368      |              |            | SPRYSMSBS   |            |            |            |            |            |
| D079       | 53369      |              |            | RSTCOMP     |            |            |            |            |            |
| D07A       | 53370      | FNRSTCMP     | EXTIROS    | RESERVED    | SPTRCONT   |            | RSTCMP     |            |            |
| D07B       | 53371      |              |            | Number      |            |            |            |            |            |
| D07C       | 53372      | DEBUGC       |            | VSYNCN      | HSYNCN     | RESERVED   |            | BITPBANK   |            |

- **ABTPALSEL** VIC-IV bitmap/text palette bank (alternate palette)
- **ALPHADELAY** Alpha delay for compositor
- **ALPHEN** Alpha compositor enable
- **BBDRPOS** bottom border position
- **BITPBANK** Set which 128KB bank bitplanes
- **BORDERCOL** display border colour (256 colour)
- **BP16ENS** VIC-IV 16-colour bitplane enable flags
- **BTPALSEL** bitmap/text palette bank
- **CHARPTR** Character set precise base address (bits 0 - 7)
- **CHR16** enable 16-bit character numbers (two screen bytes per character)
- **CHRCOUNT** Number of characters to display per row (LSB)
- **CHRXSCL** Horizontal hardware scale of text mode (pixel 120ths per pixel)
- **CHRYSCL** Vertical scaling of text mode (number of physical rasters per char text row)
- **COLPTR** colour RAM base address (bits 0 - 7)

- **DBLRR** When set, the Raster Rewrite Buffer is only updated every 2nd raster line, limiting resolution to V200, but allowing more cycles for Raster-Rewrite actions.
- **DEBUGC** VIC-IV debug pixel select red(01), green(10) or blue(11) channel visible in \$D07D
- **EXGLYPH** source full-colour character data from expansion RAM
- **EXTIRQS** Enable additional IRQ sources, e.g., raster X position.
- **FCLRHI** enable full-colour mode for character numbers >\$FF
- **FCLRLO** enable full-colour mode for character numbers <=\$FF
- **FNRASTER** Read physical raster position
- **FNRST** Raster compare source (0=VIC-IV fine raster, 1=VIC-II raster)
- **FNRSTCMP** Raster compare is in physical rasters if set, or VIC-II raster if clear
- **HOTREG** Enable VIC-II hot registers. When enabled, touching many VIC-II registers causes the VIC-IV to recalculate display parameters, such as border positions and sizes
- **HSYNCP** hsync polarity
- **KEY** Write \$47 then \$53 to enable C65GS/VIC-IV IO registers
- **LINESTEP** number of bytes to advance between each text row (LSB)
- **MAPEDPAL** palette bank mapped at \$D100-\$D3FF
- **MC1** multi-colour 1 (256 colour)
- **MC2** multi-colour 2 (256 colour)
- **MC3** multi-colour 3 (256 colour)
- **NORRDEL** When clear, raster rewrite double buffering is used
- **Number** of text rows to display
- **PALEMU** Enable PAL CRT-like scan-line emulation
- **PALNTSC** NTSC emulation mode (max raster = 262)
- **RASLINE0** first VIC-II raster line
- **RASTERHEIGHT** physical rasters per VIC-II raster (1 to 16)
- **RESERVED**
- **RSTCMP** Raster compare value MSB
- **RSTCOMP** Raster compare value

- **RSTDELEN** Enable raster delay (delays raster counter and interrupts by one line to match output pipeline latency)
- **SCREENCOL** screen colour (256 colour)
- **SCRNPTR** screen RAM precise base address (bits 0 - 7)
- **SHDEMU** Enable simulated shadow-mask (PALEMU must also be enabled)
- **SIDBDRWD** Width of single side border
- **SMTH** video output horizontal smoothing enable
- **SPR16EN** sprite 16-colour mode enables
- **SPR640** Sprite H640 enable;
- **SPRALPHAVAL** Sprite alpha-blend value
- **SPRBPMEN** Sprite bitplane-modify-mode enables
- **SPRENALPHA** Sprite alpha-blend enable
- **SPRENV400** Sprite V400 enables
- **SPRHGHT** Sprite extended height size (sprite pixels high)
- **SPRHGTEN** sprite extended height enable (one bit per sprite)
- **SPRMCO** Sprite multi-colour 0 (8-bit for selection of any palette colour)
- **SPRMCI** Sprite multi-colour 1 (8-bit for selection of any palette colour)
- **SPRPALSEL** sprite palette bank
- **SPRPTR16** 16-bit sprite pointer mode (allows sprites to be located on any 64 byte boundary in chip RAM)
- **SPRPTRADR** sprite pointer address (bits 7 - 0)
- **SPRPTRBNK** sprite pointer address (bits 22 - 16)
- **SPRTILEN** Sprite horizontal tile enables.
- **SPRX64EN** Sprite extended width enables (8 bytes per sprite row = 64 pixels wide for normal sprites or 16 pixels wide for 16-colour sprite mode)
- **SPRXSMSBS** Sprite H640 X Super-MSBs
- **SPRYADJ** Sprite Y position adjustment
- **SPRYSMSBS** Sprite V400 Y position super MSBs
- **SPTRCONT** Continuously monitor sprite pointer, to allow changing sprite data source while a sprite is being drawn

- **SRPYMSBS** Sprite V400 Y position MSBs
- **TBDRPOS** top border position
- **TEXTXPOS** character generator horizontal position
- **TEXTYPOS** Character generator vertical position
- **VFAST** C65GS FAST mode (48MHz)
- **VGAHDTV** Select more VGA-compatible mode if set, instead of HDMI/HDTV VIC-II cycle-exact frame timing. May help to produce a functional display on older VGA monitors.
- **VSYNCP** vsync polarity
- **XPOS** Read horizontal raster scan position LSB



# N

## APPENDIX

### 6526 Complex Interface Adaptor (CIA) Registers

- CIA 6526 Registers
- CIA 6526 Hypervisor Registers



# CIA 6526 REGISTERS

| HEX  | DEC   | DB7     | DB6   | DB5   | DB4  | DB3   | DB2   | DB1   | DB0     |
|------|-------|---------|-------|-------|------|-------|-------|-------|---------|
| DC00 | 56320 |         |       |       |      |       |       |       | PORTA   |
| DC01 | 56321 |         |       |       |      |       |       |       | PORTB   |
| DC02 | 56322 |         |       |       |      |       |       |       | DDRA    |
| DC03 | 56323 |         |       |       |      |       |       |       | DDRB    |
| DC04 | 56324 |         |       |       |      |       |       |       | TIMER A |
| DC05 | 56325 |         |       |       |      |       |       |       | TIMER A |
| DC06 | 56326 |         |       |       |      |       |       |       | TIMER B |
| DC07 | 56327 |         |       |       |      |       |       |       | TIMER B |
| DC08 | 56328 |         | -     |       |      |       |       |       | TODJIF  |
| DC09 | 56329 | -       |       |       |      |       |       |       | TODSEC  |
| DC0A | 56330 | -       |       |       |      |       |       |       | TODSEC  |
| DC0B | 56331 | TODAMPM | -     |       |      |       |       |       | TODHOUR |
| DC0C | 56332 |         |       |       |      | SDR   |       |       |         |
| DC0D | 56333 | IR      | -     |       | FLG  | SP    | ALRM  | TB    | TA      |
| DC0E | 56334 | TOD50   | SPMOD | IMODA | -    | RMODA | OMODA | PBONA | STRTA   |
| DC0F | 56335 | -       |       | IMODB | LOAD | RMODB | OMODB | PBONB | STRTB   |

- **ALRM** TOD alarm
- **DDRA** Port A DDR
- **DDRB** Port B DDR
- **FLG** FLAG edge detected
- **IMODA** Timer A Timer A tick source
- **IMODB** Timer B Timer A tick source
- **IR** Interrupt flag
- **LOAD** Strobe input to force-load timers
- **OMODA** Timer A toggle or pulse
- **OMODB** Timer B toggle or pulse
- **PBONA** Timer A PB6 out
- **PBONB** Timer B PB7 out
- **PORTA** Port A

- **PORTB** Port B
- **RMODA** Timer A one-shot mode
- **RMODB** Timer B one-shot mode
- **SDR** shift register data register(writing starts sending)
- **SP** shift register full/empty
- **SPMOD** Serial port direction
- **STRTA** Timer A start
- **STRTB** Timer B start
- **TA** Timer A underflow
- **TB** Timer B underflow
- **TIMERA** Timer A counter (16 bit)
- **TIMERB** Timer B counter (16 bit)
- **TOD50** 50/60Hz select for TOD clock
- **TODAMPM** TOD PM flag
- **TODHOUR** TOD hours
- **TODJIF** TOD tenths of seconds
- **TODSEC** TOD seconds

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| DC0F       | 56335      | TODEDIT    |            |            |            |            | -          |            |            |
| DD00       | 56576      |            |            |            |            |            | PORTA      |            |            |
| DD01       | 56577      |            |            |            |            |            | PORTB      |            |            |
| DD02       | 56578      |            |            |            |            |            | DDRA       |            |            |
| DD03       | 56579      |            |            |            |            |            | DDRB       |            |            |
| DD04       | 56580      |            |            |            |            |            | TIMERA     |            |            |
| DD05       | 56581      |            |            |            |            |            | TIMERA     |            |            |
| DD06       | 56582      |            |            |            |            |            | TIMERB     |            |            |
| DD07       | 56583      |            |            |            |            |            | TIMERB     |            |            |
| DD08       | 56584      |            | -          |            |            |            |            | TODJIF     |            |
| DD09       | 56585      | -          |            |            |            |            |            | TODSEC     |            |
| DD0A       | 56586      | -          |            |            |            |            |            | TODSEC     |            |
| DD0B       | 56587      | TODAMPM    | -          |            |            |            |            | TODHOUR    |            |
| DD0C       | 56588      |            |            |            |            |            | SDR        |            |            |

continued ...

...continued

| HEX  | DEC   | DB7     | DB6   | DB5   | DB4  | DB3   | DB2   | DB1   | DB0   |
|------|-------|---------|-------|-------|------|-------|-------|-------|-------|
| DD0D | 56589 |         | -     |       | FLG  | SP    | ALRM  | TB    | TA    |
| DD0E | 56590 | TOD50   | SPMOD | IMODA | -    | RMODA | OMODA | PBONA | STRTA |
| DD0F | 56591 | TODEDIT |       | IMODB | LOAD | RMODB | OMODB | PBONB | STRTB |

- **ALRM** TOD alarm
- **DDRA** Port A DDR
- **DDRB** Port B DDR
- **FLG** FLAG edge detected
- **IMODA** Timer A Timer A tick source
- **IMODB** Timer B Timer A tick source
- **LOAD** Strobe input to force-load timers
- **OMODA** Timer A toggle or pulse
- **OMODB** Timer B toggle or pulse
- **PBONA** Timer A PB6 out
- **PBONB** Timer B PB7 out
- **PORTA** Port A
- **PORTB** Port B
- **RMODA** Timer A one-shot mode
- **RMODB** Timer B one-shot mode
- **SDR** shift register data register(writing starts sending)
- **SP** shift register full/empty
- **SPMOD** Serial port direction
- **STRTA** Timer A start
- **STRTB** Timer B start
- **TA** Timer A underflow
- **TB** Timer B underflow
- **TIMERA** Timer A counter (16 bit)
- **TIMERB** Timer B counter (16 bit)

- **TOD50** 50/60Hz select for TOD clock
- **TODAMPM** TOD PM flag
- **TODEDIT** TOD alarm edit
- **TODHOUR** TOD hours
- **TODJIF** TOD tenths of seconds
- **TODSEC** TOD seconds

## CIA 6526 HYPERVISOR REGISTERS

In addition to the standard CIA registers available on the C64 and C65, the MEGA65 provides an additional set of registers that are visible only when the system is in Hypervisor Mode. These additional registers allow the internal state of the CIA to be more fully extracted when freezing, thus allowing more programs to function correctly after being frozen. They are not visible when using the MEGA65 normally, and can be safely ignored by programmers who are not programming the MEGA65 in Hypervisor Mode.

| HEX  | DEC   | DB7      | DB6  | DB5    | DB4  | DB3 | DB2 | DB1 | DB0      |
|------|-------|----------|------|--------|------|-----|-----|-----|----------|
| DC10 | 56336 |          |      |        |      |     |     |     | TALATCH  |
| DC11 | 56337 |          |      |        |      |     |     |     | TALATCH  |
| DC12 | 56338 |          |      |        |      |     |     |     | TALATCH  |
| DC13 | 56339 |          |      |        |      |     |     |     | TALATCH  |
| DC14 | 56340 |          |      |        |      |     |     |     | TALATCH  |
| DC15 | 56341 |          |      |        |      |     |     |     | TALATCH  |
| DC16 | 56342 |          |      |        |      |     |     |     | TALATCH  |
| DC17 | 56343 |          |      |        |      |     |     |     | TALATCH  |
| DC18 | 56344 | IMFLG    | IMSP | IMALRM | IMTB |     |     |     | TODJIF   |
| DC19 | 56345 |          |      |        |      |     |     |     | TODSEC   |
| DC1A | 56346 |          |      |        |      |     |     |     | TODMIN   |
| DC1B | 56347 | TODAMPM  |      |        |      |     |     |     | TODHOUR  |
| DC1C | 56348 |          |      |        |      |     |     |     | ALRMJIF  |
| DC1D | 56349 |          |      |        |      |     |     |     | ALRMSEC  |
| DC1E | 56350 |          |      |        |      |     |     |     | ALRMMIN  |
| DC1F | 56351 | ALRMAMPM |      |        |      |     |     |     | ALRMHOUR |

- **ALRMAMPM** TOD Alarm AM/PM flag
- **ALRMHOUR** TOD Alarm hours value
- **ALRMJIF** TOD Alarm 10ths of seconds value

- **ALRMMIN** TOD Alarm minutes value
- **ALRMSEC** TOD Alarm seconds value
- **IMALRM** Interrupt mask for TOD alarm
- **IMFLG** Interrupt mask for FLAG line
- **IMSP** Interrupt mask for shift register (serial port)
- **IMTB** Interrupt mask for Timer B
- **TALATCH** Timer A latch value (16 bit)
- **TODAMPM** TOD AM/PM flag
- **TODHOUR** TOD hours value
- **TODJIF** TOD 10ths of seconds value
- **TODMIN** TOD Alarm minutes value
- **TODSEC** TOD Alarm seconds value

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| DD10       | 56592      |            |            |            | TALATCH    |            |            |            |            |
| DD11       | 56593      |            |            |            | TALATCH    |            |            |            |            |
| DD12       | 56594      |            |            |            | TALATCH    |            |            |            |            |
| DD13       | 56595      |            |            |            | TALATCH    |            |            |            |            |
| DD14       | 56596      |            |            |            | TALATCH    |            |            |            |            |
| DD15       | 56597      |            |            |            | TALATCH    |            |            |            |            |
| DD16       | 56598      |            |            |            | TALATCH    |            |            |            |            |
| DD17       | 56599      |            |            |            | TALATCH    |            |            |            |            |
| DD18       | 56600      | IMFLG      | IMSP       | IMALRM     | IMTB       |            |            |            | TODJIF     |
| DD19       | 56601      |            |            |            |            | TODSEC     |            |            |            |
| DD1A       | 56602      |            |            |            |            | TODMIN     |            |            |            |
| DD1B       | 56603      | TODAMPM    |            |            |            | TODHOUR    |            |            |            |
| DD1C       | 56604      | DD00DELAY  |            |            |            |            | ALRMJIF    |            |            |
| DD1D       | 56605      |            |            |            |            | ALRMSEC    |            |            |            |
| DD1E       | 56606      |            |            |            |            | ALRMMIN    |            |            |            |
| DD1F       | 56607      | ALRMAMPM   |            |            |            |            | ALRM HOUR  |            |            |

- **ALRMAMPM** TOD Alarm AM/PM flag
- **ALRM HOUR** TOD Alarm hours value
- **ALRMJIF** TOD Alarm 10ths of seconds value
- **ALRMMIN** TOD Alarm minutes value

- **ALRMSEC** TOD Alarm seconds value
- **DDOODELAY** Enable delaying writes to \$DD00 by 3 cycles to match real 6502 timing
- **IMALRM** Interrupt mask for TOD alarm
- **IMFLG** Interrupt mask for FLAG line
- **IMSP** Interrupt mask for shift register (serial port)
- **IMTB** Interrupt mask for Timer B
- **TALATCH** Timer A latch value (16 bit)
- **TODAMPM** TOD AM/PM flag
- **TODHOUR** TOD hours value
- **TODJIF** TOD 10ths of seconds value
- **TODMIN** TOD Alarm minutes value
- **TODSEC** TOD Alarm seconds value



## APPENDIX

# 4551 UART, GPIO and Utility Controller

- C65 6551 UART Registers
- 4551 General Purpose IO & Miscellaneous Interface Registers



# C65 6551 UART REGISTERS

| HEX  | DEC   | DB7     | DB6     | DB5     | DB4     | DB3    | DB2    | DB1      | DB0    |  |  |
|------|-------|---------|---------|---------|---------|--------|--------|----------|--------|--|--|
| D600 | 54784 | DATA    |         |         |         |        |        |          |        |  |  |
| D601 | 54785 | -       |         |         |         | FRMERR | PTYERR | RXOVRRUN | RXRDY  |  |  |
| D602 | 54786 | TXEN    | RXEN    | SYNCMOD |         | CHARSZ |        | PTYEN    | PTYEVE |  |  |
| D603 | 54787 | DIVISOR |         |         |         |        |        |          |        |  |  |
| D604 | 54788 | DIVISOR |         |         |         |        |        |          |        |  |  |
| D605 | 54789 | IMTXIRQ | IMRXIRQ | IMTXNMI | IMRXNMI |        |        | -        |        |  |  |
| D606 | 54790 | IFTXIRQ | IFRXIRQ | IFTXNMI | IFRXNMI |        |        | -        |        |  |  |

- **CHARSZ** UART character size: 00=8, 01=7, 10=6, 11=5 bits per byte
- **DATA** UART data register (read or write)
- **DIVISOR** UART baud rate divisor (16 bit). Baud rate =  $7.09375\text{MHz} / \text{DIVISOR}$ , unless MEGA65 fast UART mode is enabled, in which case baud rate =  $80\text{MHz} / \text{DIVISOR}$
- **FRMERR** UART RX framing error flag (clear by reading \$D600)
- **IFRXIRQ** UART interrupt flag: IRQ on RX (not yet implemented on the MEGA65)
- **IFRXNMI** UART interrupt flag: NMI on RX (not yet implemented on the MEGA65)
- **IFTXIRQ** UART interrupt flag: IRQ on TX (not yet implemented on the MEGA65)
- **IFTXNMI** UART interrupt flag: NMI on TX (not yet implemented on the MEGA65)
- **IMRXIRQ** UART interrupt mask: IRQ on RX (not yet implemented on the MEGA65)
- **IMRXNMI** UART interrupt mask: NMI on RX (not yet implemented on the MEGA65)
- **IMTXIRQ** UART interrupt mask: IRQ on TX (not yet implemented on the MEGA65)
- **IMTXNMI** UART interrupt mask: NMI on TX (not yet implemented on the MEGA65)
- **PTYEN** UART Parity enable: 1=enabled
- **PTYERR** UART RX parity error flag (clear by reading \$D600)
- **PTYEVEN** UART Parity: 1=even, 0=odd
- **RXEN** UART enable receive
- **RXOVRRUN** UART RX overrun flag (clear by reading \$D600)
- **RXRDY** UART RX byte ready flag (clear by reading \$D600)

- **SYNCMOD** UART synchronisation mode flags (00=RX & TX both async, 01=RX sync, TX async, 1x=TX sync, RX async (unused on the MEGA65)
- **TXEN** UART enable transmit

## 4551 GENERAL PURPOSE IO & MISCELLANEOUS INTERFACE REGISTERS

| HEX  | DEC   | DB7       | DB6      | DB5     | DB4      | DB3      | DB2    | DB1    | DB0      |
|------|-------|-----------|----------|---------|----------|----------|--------|--------|----------|
| D609 | 54793 |           |          |         | -        |          |        |        | UFAS     |
| D60B | 54795 | OSKZEN    | OSKZON   |         |          |          |        |        | PORTF    |
| D60C | 54796 |           | PORTFDDR |         |          |          |        |        | PORTFDDR |
| D60D | 54797 | HDSCL     | HDSDA    | SDBSH   | SDCS     | SDCLK    | SDDATA | RST41  | CONN     |
| D60E | 54798 |           |          |         | BASHDDR  |          |        |        |          |
| D60F | 54799 | ACCESSKEY | OSKDIM   | REALHW  |          | -        |        | KEYUP  | KEYLE    |
| D610 | 54800 |           |          |         | ASCIIKEY |          |        |        |          |
| D611 | 54801 | -         | MCAPS    | MSCRL   | MALT     | MMEGA    | MCTRL  | MLSHFT | MRSH     |
| D612 | 54802 | LJOYB     | LJOYA    | JOYSWAP | OSKDEBUG |          |        |        | -        |
| D615 | 54805 | OSKEN     |          |         |          | VIRTKEY1 |        |        |          |
| D616 | 54806 | OSKALT    |          |         |          | VIRTKEY2 |        |        |          |
| D617 | 54807 | OSKTOP    |          |         |          | VIRTKEY3 |        |        |          |
| D618 | 54808 |           |          |         | KSCNRATE |          |        |        |          |
| D619 | 54809 |           |          |         | UNUSED   |          |        |        |          |
| D61A | 54810 |           |          |         | SYSCTL   |          |        |        |          |
| D61D | 54813 | Keyboard  |          |         | Keyboard |          |        |        |          |
| D61E | 54814 |           |          |         | Keyboard |          |        |        |          |
| D620 | 54816 |           |          |         | POTAX    |          |        |        |          |
| D621 | 54817 |           |          |         | POTAY    |          |        |        |          |
| D622 | 54818 |           |          |         | POTBX    |          |        |        |          |
| D623 | 54819 |           |          |         | POTBY    |          |        |        |          |
| D625 | 54821 |           |          |         | J21L     |          |        |        |          |
| D626 | 54822 |           |          |         | J21H     |          |        |        |          |
| D627 | 54823 |           |          |         | J21LDDR  |          |        |        |          |
| D628 | 54824 |           |          |         | J21HDDR  |          |        |        |          |

- **ACCESSKEY** Enable accessible keyboard input via joystick port 2 fire button

- **ASCIIKEY** Last key press as ASCII (hardware accelerated keyboard scanner). Write to clear event ready for next.
- **BASHDDR** Data Direction Register (DDR) for \$D60D bit bashing port.
- **CONN41** Internal 1541 drive connect (1=connect internal 1541 drive to IEC bus)
- **HDSCL** HDMI I2C control interface SCL clock
- **HDSDA** HDMI I2C control interface SDA data line
- **J21H** J21 pins 11 – 14 input/output values
- **J21HDDR** J21 pins 11 – 14 data direction register
- **J21L** J21 pins 1 – 6, 9 – 10 input/output values
- **J21LDDR** J21 pins 1 – 6, 9 – 10 data direction register
- **JOYSWAP** Exchange joystick ports 1 & 2
- **KEYLEFT** Directly read C65 Cursor left key
- **KEYUP** Directly read C65 Cursor up key
- **KSCNRATE** Physical keyboard scan rate (\$00=50MHz, \$FF=~200KHz)
- **Keyboard** LED control enable
- **LJOYA** Rotate inputs of joystick A by 180 degrees (for left handed use)
- **LJOYB** Rotate inputs of joystick B by 180 degrees (for left handed use)
- **MALT** ALT key state (hardware accelerated keyboard scanner).
- **MCAPS** CAPS LOCK key state (hardware accelerated keyboard scanner).
- **MCTRL** CTRL key state (hardware accelerated keyboard scanner).
- **MLSHFT** Left shift key state (hardware accelerated keyboard scanner).
- **MMEGA** MEGA/C= key state (hardware accelerated keyboard scanner).
- **MRSHFT** Right shift key state (hardware accelerated keyboard scanner).
- **MSCRCL** NOSCRCL key state (hardware accelerated keyboard scanner).
- **OSKALT** Display alternate on-screen keyboard layout (typically dial pad for MEGA65 telephone)
- **OSKDEBUG** Debug OSK overlay (WRITE ONLY)
- **OSKDIM** Light or heavy dimming of background material behind on-screen keyboard
- **OSKEN** Enable display of on-screen keyboard composited overlay

- **OSKTOP** 1=Display on-screen keyboard at top, 0=Dispaly on-screen keyboard at bottom of screen.
- **OSKZEN** Display hardware zoom of region under first touch point for on-screen keyboard
- **OSKZON** Display hardware zoom of region under first touch point always
- **PORTF** PMOD port A on FPGA board (data) (Nexys4 boards only)
- **PORTFDDR** PMOD port A on FPGA board (DDR)
- **POTAX** Read Port A paddle X, without having to fiddle with SID/CIA settings.
- **POTAY** Read Port A paddle Y, without having to fiddle with SID/CIA settings.
- **POTBX** Read Port B paddle X, without having to fiddle with SID/CIA settings.
- **POTBY** Read Port B paddle Y, without having to fiddle with SID/CIA settings.
- **REALHW** Set to 1 if the MEGA65 is running on real hardware, set to 0 if emulated (Xemu) or simulated (ghdl)
- **RST41** Internal 1541 drive reset ( 1=reset, 0=operate)
- **SDBSH** Enable SD card bitbash mode
- **SDCLK** SD card SCLK
- **SDCS** SD card CS\_BO
- **SDDATA** SD card MOSI/MISO
- **SYSCTL** System control flags (target specific)
- **UFAST** C65 UART BAUD clock source: 1 = 7.09375MHz, 0 = 80MHz (VIC-IV pixel clock)
- **UNUSED** port o output value
- **VIRTKEY1** Set to \$7F for no key down, else specify virtual key press.
- **VIRTKEY2** Set to \$7F for no key down, else specify 2nd virtual key press.
- **VIRTKEY3** Set to \$7F for no key down, else specify 3nd virtual key press.

# P

## APPENDIX

### 45E100 Fast Ethernet Controller

- Overview
- Memory Mapped Registers
- Example Programs



# OVERVIEW

The 45E100 is a new and simple Fast Ethernet controller that has been designed specially for the MEGA65 and for 8-bit computers generally. In addition to supporting 100Mbit Fast Ethernet, it is radically different from other Ethernet controllers, such as the RR-NET.

The 45E100 includes four receive buffers, allowing upto three frames to be received while another is being processed, or to allow less frequent processing of interrupts. These receive buffers can be memory mapped, and also directly accessed using the MEGA65's DMA controller. Together with automatic CRC32 checking on reception, and automatic CRC32 generation for transmit, these features considerably reduce the burden on the processor, and make it much simpler to write ethernet-enabled programs.

The 45E100 also supports true full-duplex operation at 100Mbit per second, allowing for total bi-directional throughput exceeding 100Mbit per second. The MAC address is software configurable, and promiscuous mode is supported, as are individual control of the reception of broadcast and multi-cast Ethernet frames.

The 45E100 also supports both transmit and receive interrupts, allowing greatly improved real-world performance. When especially low latency is required, it is also possible to immediately abort the transmission of the current Ethernet frame, so that a higher-priority frame can be immediately sent. These features combine to enable sub-millisecond round trip latencies, which can be of particular value for interactive applications, such as multi-player network games.

## Differences to the RR-NET and similar solutions

The RR-NET and other Ethernet controllers for the Commodore™ line of 8-bit home computers generally use an Ethernet controller that was designed for 16-bit PCs, but that also supports a so-called "8-bit mode," which suffers from a number of disadvantages. These disadvantages include the lack of working interrupts, as well as processor intensive access to the Ethernet frame buffers. The lack of interrupts forces programs to use polling to check for the arrival of new Ethernet frames. This, together with the complexities of accessing the buffers results in an Ethernet interface that is very slow, and whose real-world throughput is considerably less than its theoretical 10Mbits per second. Even a Commodore 64 with REU cannot achieve speeds above several tens of kilobytes per second.

In contrast, the 45E100 supports both RX (Ethernet frame received) interrupts and TX (ready to transmit) interrupts, freeing the processor from having to poll the device. Be-

cause the 45E100 supports RX interrupts, there is no need for large numbers of receive buffers, which is why the 45E100 requires only two RX buffers to achieve very high levels of performance.

Further, the 45E100 supports direct memory mapping of the Ethernet frame buffers, allowing for much more efficient access, including by DMA. Using the MEGA65's integrated DMA controller it is quite possible to achieve transfer rates of several mega-bytes per second – some 100x faster than the RR-NET.

## Theory of Operation: Receiving Frames

The 45E100 is simple to operate: To begin receiving Ethernet frames, the programmer needs only to clear the RST and TXRST bits (bit 0 of register \$D6E0) to ensure that the Ethernet controller is reset, and then set these bits to 1, to release the controller from the reset state. It will then auto-negotiate connection at the highest available speed, typically 100Mbit, full-duplex.

If you wish to simply poll for the arrival of ethernet frames, check the RXQ bit (bit 5 of \$D6E1). If it is set, then there is at least one frame that has been received. To access the next frame that has been received, write \$01 to \$D6E1, and then \$03 to \$D6E1. This will rotate the ring of receive buffers, to make the next received frame accessible by the processor. The receive buffer that was previously accessible by the processor is marked free, and the 45E100 will use it to receive another ethernet frame when required.

Because the 45E100 has four receive buffers, it is possible that to process multiple frames in succession by following this procedure. If all receive buffers contain received frames, and the processor has not accepted them, then the RXBLKD signal will be asserted, so that the processor knows that if any more frames are received, they will be lost. Programmers should take care to avoid this situation. As the 45E100 supports receive interrupts, this is generally easy to manage – but don't underestimate how often ethernet frames can arrive on a 100mbit Fast Ethernet connection: If a sender sends a continuous stream of minimum-length ethernet frames, they can arrive every 6 microseconds or so! While, it is unlikely that you will have to deal with such a high rate of packet reception, you should anticipate the need to process packets at least every milli-second. In particular, a once-per-frame CIA or raster IRQ may cause some packets to be lost, more than three arrive in a 16 - 20 ms video frame. The RXBLKD signal can be used to determine if this situation is likely to have occurred. But note that it indicates only when all receive buffers are occupied, not if any further frames arrived while there were no free receive buffers.

The receive buffers are 2KiB bytes each, and can each hold only one received ethernet frame at a time. This is different to some ethernet controllers that use their total receive buffer memory as a simple ring buffer. The reason for this is to keep the mechanism for programmers as simple as possible. By having the fixed buffers, it means that the

controller can memory map the received ethernet frames in exactly the same location each time, making it possible to write much simpler receiver programmes, because the location of the received ethernet frames can be assumed to be constant.

The structure of a receive buffer containing an ethernet frame is quite simple: The first two bytes indicate the length of the received frame. The frame then follows immediately. The effective Maximum Transport Unit (MTU) length is 2,042 bytes, as the last four bytes are occupied by the CRC32 checksum of the received ethernet frame. The layout of the receive buffers is thus as follows:

| <b>HEX</b>  | <b>DEC</b>    | <b>Length</b> | <b>Description</b>                                                                                                                                                                                                                                                                                                                                                                                                              |
|-------------|---------------|---------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0000        | 0             | 1             | The low byte of the length of the received ethernet frame.                                                                                                                                                                                                                                                                                                                                                                      |
| 0001        | 1             | 1             | The lower four bits contain the upper bits of the length of the received ethernet frame. Bit 4 is set if the received ethernet frame is a multi-cast frame. Bit 5 if it is a broadcast frame. Bit 6 is set if the frame's destination address matches the 45E100's programmed MAC address. Bit 7 is set if the CRC32 check for the received frame failed, i.e., that the frame is either truncated or was corrupted in transit. |
| 0002 - 07FB | 2 - 2,043     | 2,042         | The received frame. Frames shorter than 2,042 bytes will begin at offset 2.                                                                                                                                                                                                                                                                                                                                                     |
| 07FC - 07FF | 2,044 - 2,047 | 4             | Reserved space for holding the CRC32 code during reception. The CRC32 code is, however, always located directly after the received frame, and thus will only occupy this space if the received frame is more than 2,038 bytes long. "                                                                                                                                                                                           |

Because of the very rapid rate at which Fast Ethernet frames can be received, a programmer should use the receive interrupt feature, enabled by setting RXQEN (bit 7 of \$D6E1). Polling is possible as an alternative, but is not recommended with the 45E100, because at the 100Mbit Fast Ethernet speed, packets can arrive as often as every 5 microseconds. Fortunately, at the MEGA65's 40MHz full speed mode, and using the 20MiB per second DMA copy functionality, it is possible to keep up with such high data rates.

# Accessing the Ethernet Frame Buffers

Unlike on the RR-NET, the 45E100's ethernet frame buffers are able to be memory mapped, allowing rapid access via DMA or through assembly language programs. It is also possible to access the buffers from BASIC with some care.

The frame buffers can either be accessed from their natural location in the MEGA65's extended address space at address \$FFDE800 - \$FFDEFFF, or they can be mapped into the normal C64/C65 \$D000 IO address space. Care must be taken as mapping the ethernet frame buffers into the \$D000 IO address space causes all other IO devices to unavailable during this time. Therefore CIA-based interrupts MUST be disabled before doing so, whether using BASIC or machine code. Therefore when programming in assembly language or machine code, it is recommended to use the natural location, and to access this memory area using one of the three mechanisms for accessing extended address space, which are described in Chapter/Appendix G on page G-11.

The method of disabling interrupts differs depending on the context in which a program is being written. For programs being written using C64 mode's BASIC 2, the following will work:

```
POKE56333 ,127: REM DISABLE CIA TIMER IRQS
```

While for MEGA65's BASIC 65, the following must instead be used, because a VIC-III raster interrupt is used instead of a CIA-based timer interrupt:

```
POKE53274 ,0: REM DISABLE VIC-II/III/IV RASTER IRQS
```

Once this has been done, the IO context for the ethernet controller can be activated by writing \$45 (69 in decimal, equal to the character 'E' in PETSCII) and \$54 (84 in decimal, equal to the character 'T' in PETSCII) into the VIC-IV's KEY register (\$D02F, 53295 in decimal), for example:

```
POKE53295 ,ASC("E"):POKE53295 ,ASC("T")
```

At this point, the ethernet RX buffer can be read beginning at location \$D000 (53248 in decimal), and the TX buffer can be written to at the same address. Refer to 'Theory of Operation: Receiving Frames' above for further explanation on this.

Once you have finished accessing the ethernet frame buffer, you can restore the normal C64, C65 or MEGA65 IO context by writing to the VIC-III/IV's KEY register. In most cases, it will make the most sense to revert to the MEGA65's IO context by writing \$47 (71 decimal) in and \$53 (83 in decimal) to the KEY register, for example:

```
POKE53295,ASC("G"):POKE53295,ASC("$")
```

Finally, you should then re-enable interrupts, which will again depend on whether you are programming from C64 or C65 mode. For C64 mode:

```
POKE56333,129
```

For C65 mode it would be:

```
POKE53274,129
```

## Theory of Operation: Sending Frames

Sending frames is similarly simple: The program must simply load the frame to be transmitted into the transmit buffer, write its length into TXSZLSB and TXSZMSB registers, and then write \$01 into the COMMAND register. The frame will then begin to transmit, as soon as the transmitter is idle. There is no need to calculate and attach an ethernet CRC32 field, as the 45E100 does this automatically.

Unlike for the receiver, there is only one frame buffer for the transmitter (this may be changed in a future revision). This means that you cannot prepare the next frame until the previous frame has already been sent. This slightly reduces the maximum data throughput, in return for a very simple architecture.

Also, note that the transmit buffer is write-only from the processor bus interface. This means that you cannot directly read the contents of the transmit buffer, but must load values "blind". Finally, the 45E100 allows you to send ethernet.

## Advanced Features

In addition to operating as a simple and efficient ethernet frame transceiver, the 45E100 includes a number of advanced features, described here.

## Broadcast and Multicast Traffic and Promiscuous Mode

The 45E100 supports filtering based on the destination Ethernet address, i.e., MAC address. By default, only frames where the destination Ethernet address matches the eth-

ernet address programmed into the MACADDR1 – MACADDR6 registers will be received. However, if the MCST bit is set, then multicast ethernet frames will also be received. Similarly, setting the BCST bit will allow all broadcast frames, i.e., with MAC address ff:ff:ff:ff:ff:ff, to be received. Finally, if the NOPROM bit is cleared, the 45E100 disables the filter entirely, and will receive all valid ethernet frames.

## Debugging and Diagnosis Features

The 45E100 also supports several features to assist in the diagnosis of ethernet problems. First, if the NOCRC bit is set, then even ethernet frames that have invalid CRC32 values will be received. This can help debug faulty ethernet devices on a network.

If the STRM bit is set, the ethernet transmitter transmits a continuous stream of debugging frames supplied via a special high-bandwidth logging interface. By default, the 45E100 emits a stream of approximately 2,200 byte ethernet frames that contain compressed video provided by a VIC-IV or compatible video controller that supports the MEGA65 video-over-ethernet interface. By writing a custom decoder for this stream of ethernet frames, it is possible to create a remote display of the MEGA65 via ethernet. Such a remote display can be used, for example, to facilitate digital capture of the display of a MEGA65.

The size and content of the debugging frames can be controlled by writing special values to the COMMAND register. Writing \$F1 allows the selection of frames that are 1,200 bytes long. While this reduces the performance of the debugging and streaming features, it allows the reception of these frames on systems whose ethernet controllers cannot be configured to receive frames of 2,200 bytes.

If the STRM bit is set and bit 2 of \$D6E1 is also set, a compressed log of instructions executed by the 45gs02 CPU will instead be streamed, if a compatible processor is connected to this interface. This mechanism includes back-pressure, and will cause the 45gs02 processor to slowdown, so that the instruction data can be emitted. This typically limits the speed of the connected 45gs02 processor to around 5MHz, depending on the particular instruction mix.

Note also that the status of bit 2 of \$D6E1 cannot currently be read directly. This may be corrected in a future revision.

Finally, if the video streaming functionality is enabled, this also enables reception of synthetic keyboard events via ethernet. These are delivered to the MEGA65's Keyboard Complex Interface Adapter (KCIA), allowing full remote interaction with a MEGA65 via its ethernet interface. This feature is primarily intended for development.

# MEMORY MAPPED REGISTERS

The 45E100 Fast Ethernet controller is a MEGA65-specific feature. It is therefore only available in the MEGA65 IO context. This is enabled by writing \$53 and then \$47 to VIC-IV register \$D02F. If programming in BASIC, this can be done with:

```
POKE53295,ASC("G"):POKE53295,ASC("S")
```

The 45E100 Fast Ethernet controller has the following registers

| HEX  | DEC   | DB7    | DB6     | DB5  | DB4      | DB3   | DB2     | DB1    | DB0 |
|------|-------|--------|---------|------|----------|-------|---------|--------|-----|
| D6E0 | 55008 | TXIDLE | RXBLKD  | -    | KEYEN    | DRXDV | DRXD    | TXRST  | RST |
| D6E1 | 55009 | RXQEN  | TXQEN   | RXQ  | TXQ      | STRM  |         | RXBF   | -   |
| D6E2 | 55010 |        |         |      | TXSZLSB  |       |         |        |     |
| D6E3 | 55011 |        |         |      | TXSZMSB  |       |         |        |     |
| D6E4 | 55012 |        |         |      | COMMAND  |       |         |        |     |
| D6E5 | 55013 | -      | MCST    | BCST |          | RXPH  | NOCRC   | NOPROM |     |
| D6E6 | 55014 |        | MIIMPHY |      |          |       | MIIMREG |        |     |
| D6E7 | 55015 |        |         |      | MIIMVLSB |       |         |        |     |
| D6E8 | 55016 |        |         |      | MIIMVMSB |       |         |        |     |
| D6E9 | 55017 |        |         |      | MACADDR1 |       |         |        |     |
| D6EA | 55018 |        |         |      | MACADDR2 |       |         |        |     |
| D6EB | 55019 |        |         |      | MACADDR3 |       |         |        |     |
| D6EC | 55020 |        |         |      | MACADDR4 |       |         |        |     |
| D6ED | 55021 |        |         |      | MACADDR5 |       |         |        |     |
| D6EE | 55022 |        |         |      | MACADDR6 |       |         |        |     |

- **BCST** Accept broadcast frames
- **COMMAND** Ethernet command register (write only)
- **DRXD** Read ethernet RX bits currently on the wire
- **DRXDV** Read ethernet RX data valid (debug)
- **KEYEN** Allow remote keyboard input via magic ethernet frames
- **MACADDR1** Ethernet MAC address
- **MACADDR2** Ethernet MAC address
- **MACADDR3** Ethernet MAC address
- **MACADDR4** Ethernet MAC address

- **MACADDR5** Ethernet MAC address
- **MACADDR6** Ethernet MAC address
- **MCST** Accept multicast frames
- **MIIMPHY** Ethernet MIIM PHY number (use 0 for Nexys4, 1 for MEGA65 r1 PCBs)
- **MIIMREG** Ethernet MIIM register number
- **MIIMVLSB** Ethernet MIIM register value (LSB)
- **MIIMVMSB** Ethernet MIIM register value (MSB)
- **NOCRC** Disable CRC check for received packets
- **NOPROM** Ethernet disable promiscuous mode
- **RST** Write 0 to hold ethernet controller under reset
- **RXBF** Number of free receive buffers
- **RXBLKD** Indicate if ethernet RX is blocked until RX buffers freed
- **RXPH** Ethernet RX clock phase adjust
- **RXQ** Ethernet RX IRQ status
- **RXQEN** Enable ethernet RX IRQ
- **STRM** Enable streaming of CPU instruction stream or VIC-IV display on ethernet
- **TXIDLE** Ethernet transmit side is idle, i.e., a packet can be sent.
- **TXPH** Ethernet TX clock phase adjust
- **TXQ** Ethernet TX IRQ status
- **TXQEN** Enable ethernet TX IRQ
- **TXRST** Write 0 to hold ethernet controller transmit sub-system under reset
- **TXSZLSB** TX Packet size (low byte)
- **TXSZMSB** TX Packet size (high byte)

## COMMAND register values

The following values can be written to the COMMAND register to perform the described functions. In normal operation only the STARTTX command is required, for example, by performing the following POKE:

POKE55812 ,1

| HEX | DEC | Signal    | Description                                                                                                                                                  |
|-----|-----|-----------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 00  | 0   | STOPTX    | Immediately stop transmitting the current ethernet frame. Will cause a partially sent frame to be received, most likely resulting in the loss of that frame. |
| 01  | 1   | STARTTX   | Transmit packet                                                                                                                                              |
| D0  | 208 | RXNORMAL  | Disable the effects of RXONLYONE                                                                                                                             |
| D4  | 212 | DEBUGVIC  | Select VIC-IV debug stream via ethernet when \$D6E1.3 is set                                                                                                 |
| DC  | 220 | DEBUGCPU  | Select CPU debug stream via ethernet when \$D6E1.3 is set                                                                                                    |
| DE  | 222 | RXONLYONE | Receive exactly one ethernet frame only, and keep all signals states (for debugging ethernet sub-system)                                                     |
| F1  | 241 | FRAME1K   | Select 1KiB frames for video/cpu debug stream frames (for receivers that do not support MTUs of greater than 2KiB)                                           |
| F2  | 242 | FRAME2K   | Select 2KiB frames for video/cpu debug stream frames, for optimal performance.                                                                               |

## EXAMPLE PROGRAMS

Example programs for the ethernet controller exist in imperfect for in the MEGA65 Core repository on github in the src/tests and src/examples directories.

If you wish to use the ethernet controller for TCP/IP traffic, you may wish to examine the port of WeeIP to the MEGA65 at <https://github.com/mega65/mega65-weeip>. The code that controls the ethernet controller is located in eth.c.





## APPENDIX

# 45IO27 Multi-Function IO Controller

- Overview
- F011-compatible Floppy Controller
- SD Card Controller and F011 Virtualisation Functions
- Touch Panel Interface
- Audio Support Functions
- Miscellaneous IO Functions



# OVERVIEW

The 45IO27 is a multi-purpose IO controller that incorporates the functions of the C65's F011 floppy controller, together with the MEGA65's SD card controller interface, and a number of other miscellation IO functions.

Each of these major functions is covered in a separate section of this chapter

## F011-COMPATIBLE FLOPPY CONTROLLER

The MEGA65 computer is one of very few modern computers that still includes first-class support for magnetic floppy drives. It includes a floppy controller that is backwards compatible with the C65's F011D floppy drive controller.

However, unlike the F011D, the MEGA65's floppy disk controller supports HD and ED media, and similar to the 1541 floppy drive, it also supports variable data rates, so that a determined user could develop disk formats that store more data, includ robust copy protection schemes, or both.

GCR encoding is not currently supported, but may be supported by a future revision of the controller. It may also be possible with some creativity and effort to use the debug register interface to read double-density GCR formatted media. This is because there are debug registers that can be queried to indicate the gap between each successive magnetic domain – which is sufficient to decode any disk format.

## Multiple Drive Support

Like the C65's F011 floppy drive controller, the 45IO27 supports up to 8 drives. The first two of those drives, drive 0 and drive 1, are assumed to be connected to a standard 34-pin floppy cable, the same as used in standard PCs, i.e., with a twist in the cable to allow the use of two unjumpered drives.

As is described in later sections, it is possible to switch drive 0 and drive 1's position, without having to change cabling. Similarly, either or both of the first two drives may reference a real floppy drive, a D81 disk image stored on an attached SD card, or redirected to the floppy drive virtualisation service, so that the sector accesses can be handled by a connected computer, e.g., as part of a comfortable and efficient cross-development environment.

The remaining six drives are supported only in conjunction with a future C1565-compatible external drive port.

## Buffered Sector Operations

The 45IO27 support two main modes of reading sectors from a disk: byte-by-byte, and via a memory-mapped sector buffer.

The byte-by-byte mechanism consists of having a loop wait for the DRQ signal to be asserted, and then reading the byte of data from the DATA register (\$D087).

The memory-mapped sector buffer method consists of waiting for the BUSY flag to clear, indicating that the entire sector has been read, and then directly accessing the sector buffer located at \$FFD6C00 - \$FFD6dff. Care should be taken to ensure that the BUFSEL signal (bit 7 of \$D689) is cleared, so that the floppy sector buffer is visible, rather than the SD card sector buffer for programmes other than the Hypervisor. This is because only the Hypervisor has access to the full 4KiB SD controller buffer space: Normal programmes see either the floppy sector buffer or the SD card sector buffer repeated 8 times between \$FFD6000 and \$FFD6FFF.

Alternatively, the sector buffer can be mapped at \$DE00 - \$DFFF, i.e., in the 4KB IO area, by writing the \$81 to the SD command register at \$D680. This will hide any IO peripherals that are otherwise using this area, e.g., from cartridges, or REU emulation. This function can be disabled again by writing \$82 to the SD command register. As with the normal sector buffer memory mapping at \$FFD6xxx, the BUFSEL signal (bit 7 of \$D689) affects whether the FDC or the SD card sector buffer is visible, for software not running in Hypervisor mode. Note that if you use the matrix mode / serial monitor interface to inspect the contents of the sector buffer, that this occurs in the Hypervisor context, and so the BUFSEL signal will be ignored, and the full 4KiB buffer will be visible.

The memory-mapped sector buffer has the advantage that it can be accessed via DMA, allowing for very efficient copies. Also, it allows for loading a sector to occur in the background, while your programme gets on with more interesting things in the meantime.

## Reading Sectors from a Disk

There are several steps that you must follow in order to successfully read a sector from a disk. If you follow these instructions, your code will work with both physical disks, as well as D81 disk images that exist on the SD card:

- First, enable the motor and select the appropriate drive. The F011 supports upto 8 physical drives, although it is rare for more than two to be physically connected. To enable the motor, write \$60 to \$D080. You should then write a SPINUP command

(\$20) to \$D081, and wait for the BUSY flag (bit 7 of \$D082) to clear. The drive is now spinning at speed, and ready to service requests.

- Next, select the correct side of the disk by either setting or clearing the SIDE1 flag (bit 3 of \$D080). This takes effect immediately.
- Third, use the step-in and step-out commands (writing \$10 and \$18 to \$D081) as required to move the head to the correct track. Again, after each command, you should wait for the BUSY flag (bit 7 of \$D082) to clear, before issuing the next command.

Note that you can check if the head is at track 0 by checking the TRACK0 flag, but there is no fool-proof way to know if you are on any other specific track. You can use the registers at \$D6A3 - \$D6A5 to see the track, sector and side value from the last sector header which passed under the head to make an informed guess as to which track is currently selected. Note that this only works for real disks, as disk images do not spin under the read head. Also note that it is possible for tracks to contain sectors which purposely or accidentally have incorrect track numbers in the sector headers.

- Fourth, you need to load the desired track, sector and side number into the TRACK, SECTOR and SIDE registers (\$D084, \$D085 and \$D086, respectively). The FDC is now primed ready to read a sector.
- Fifth, you should write an appropriate read command value into \$D081. This will normally be \$40 (64). You then wait for the RDREQ signal (\$D083, bit 7) to go high, to indicate that the sector has been found. You then either wait for each occasion when DRQ goes high, and read byte-by-byte in such a loop, or wait for the BUSY flag to clear and the DRQ and EQ flags to go high, which indicates that the complete sector has been read into the buffer.

## Track Auto-Tune Function Deprecated

The 45IO27 also includes a track “auto-tune” function, which is enabled by clearing bit 4 of \$D689. That function reads the sector headers to determine which track the head is currently over, and steps the head in or out to try to get to the correct track. This was implemented to work around a known problem where the floppy drive would mis-step sometimes. However that problem has since been corrected, and thus the auto-tune function is no-longer required, and it is not recommended to use it.

The reason we recommend that you do not use the auto-tune function is that if a disk is not properly formatted or purposely includes incorrect track numbers in sector headers, this can easily result in “head chatter”, where the drive seeks continuously back and forth between two or more tracks. This creates an annoying noise, and will probably shorten

the life-span of your floppy drive. However, it can likely be used to make disks that when inserted will play simple tunes using the floppy drive noises, and when carefully used (and disabled after) can automate the stepping of the head between tracks to reduce CPU load.

## Sector Skew and Target Any Mode

It is also worth noting that the TARGANY signal can be asserted to tell the floppy controller to simply read the next sector that passes under the head. This applies only when using real floppy disks, where it offers the considerable advantage of letting you read the sectors in the order in which they exist on the disk. This allows you to read a track at once, without having to wait for the index hole to pass by, or having to know which sector will next pass under the head.

For example, the C65 DOS formats disks using a skew factor of 7, while PCs may use a different skew-factor. If you don't know the skew factor of the disk, you may schedule the reading of the sectors on the track in a sub-optimal order. This can result in transfer rates as low as 5 sectors per second, compared with the optimal case of 50 sectors per second. Thus with either correct sector order, or using the target any mode, it is possible to read approximately two full tracks per second, i.e., two sides  $\times$  two tracks, or approximately 20KiB/second on DD disks, or double that on HD disks, at around 40KiB/second. This compares very favourably with the C65 DOS loading speed, which is typically nearer 1KiB/sec in C64 mode.

## Disk Layout and 1581 Logical Sectors

The 1581 disk format is unusual in that the physical sectors on the disk are a different size of the size of the data blocks that it presents to the user. Specifically, the disks use 512 byte sectors, while the 1581 (and C65) DOS present 256 byte data blocks. Two blocks are stored in each physical sector. Also, the physical track numbers are from 0 to 79, while the logical track numbers of the DOS are 1 to 80. Physical sectors are also numbered from 1 to 10, while logical block numbers begin are 0 to 39.

This means that if you want to find a 1581 logical sector, you need to know which physical sector it will be found in. To determine the physical sector that contains a block, you first subtract one from the track number, and then divide the sector number by two. Logical sectors 0 to 19 of each track are located in physical sectors 1 to 10 on the first side of the disk. Logical sectors 20 to 39 are located in physical sectors 1 to 10 on the reverse side of the disk.

Thus we can map a some logical track and sector  $t,s$  to the physical track, side and sector as follows:

*track* =  $t - 1$

*sector* =  $(s/2) + 1$ , *IFFs* < 20, *ELSE* =  $((s - 20)/2) + 1$

*side* = 0 *IFF sector* < 20

It is also worth noting that the 45IO27 is capable of reading from tracks beyond track 80, provided that the disk drive is capable of this. Almost all 3.5 inch floppy drives are capable of reading at least one extra track, as historically manufacturers of floppy disks stored information about the disk on the 81st track. In our experience almost all drives will also be able to access an 82nd track.

## FD2000 Disks

The CMD(tm) FD2000(tm) high-density 3.5" disk drives for Commodore(tm) computers use an unusual disk layout that is quite different from PCs: They use 10 sectors, the same as on 720KB double-density (DD) disks, but double the *sector* size from 512 bytes to 1,024 bytes. The 45IO27 does not currently support these larger sectors. Support is planned to be added via a core update in the future.

## Formatting Disks

Formatting disks is now possible with the 45IO27, with two caveats:

- The CPU should be set to 40MHz mode, due to an unresolved timing error in instruction execution that can cause problems when formatting disks using the C65 DOS.
- The C65 BASIC HEADER command will successfully *format* a disk, but will fail to write the directory structure to the disk, because buffered sector writing is not yet implemented, as explained below.

The formatting process is unbuffered, that is, the CPU must write each clock and data byte to the appropriate FDC registers each time the DRQ flag is asserted. Failure to supply the next byte in time will result in bit-slippage, and a disk with unreadable sectors that will require formatting again.

To successfully format a track, the user must provide the full structure of the track, including all sync marks, sector headers, CRC bytes, data regions and gaps. The explanation of this is beyond the scope of this document. The C65 Specifications Manual, the C65 ROM DOS source code and the `floppytest.c` programme from the <https://github.com/mega65/mega65-tools> repository each describe and/or implement the process for formatting tracks.

# Buffered Sector Writing

The 45IO27 can write to disk images that are located on the SD card, or when using virtualised disk access.

However, the current revision of the 45IO27 does not support writing to real floppy disks. This will be corrected in a future revision. However, the determined user may still be able to write to disks by manipulating the WGATE and WDATA signals of the \$D6A0 register. However, this is not recommended.

To write a sector, you follow a similar process to reading, except that you write \$84 to the command byte instead of \$40. The \$80 indicates a write, and the \$04 activates write-precompensation. This is important when writing to real floppy disks, especially HD and ED disks. Write-precompensation causes bits to be written slightly early or slightly late, using an algorithm that models how the magnetic domains on a disk tend to move after being written.

If you do not wish to use the sector buffer, but instead provide each byte one at a time during the write operation, you must add \$01 to the command code. However, this is not recommended on the MEGA65, because when writing to the SD card or using virtualised disk images the entire sector operation can happen instantaneously from the perspective of your programme. This means that it is not possible to supply data reliably when in this mode. Thus apart from being less convenient, it is also less reliable.

Once a write operation has been triggered, the DRQ signal indicates when you should provide the next byte if performing a byte-by-byte write. Otherwise, it is assumed that you will have pre-filled the sector buffer with the complete 512 bytes of data required.

## Using HD, ED Drives and/or variable Data Rates

The mechanism for using high-density (HD) and extended-density (ED) media and variable data rates are all identical on the 45IO27: You simply set the interval rate (DATARATE) in the \$D6A2 register.

The register is used as a divisor of the system bus frequency (40.5MHz). Thus for 1581-compatible double-density (DD) media which uses a data rate of 500KHz, this register should contain  $40.5\text{MHz} / 500\text{KHz} = 81$ .

High-density media uses a 1MHz data rate, and thus should contain 40, and extended-density media with its 2MHz data rate should have this field set to 20. Depending on the HD or ED drive used, it may be necessary to clear or assert the DENSITY signal (\$D6A0 bit 7), or take other specific actions. However, in our experience, most HD drives do not

require any manual manipulation of this signal: Instead simply set the DATARATE register to the correct value, and begin accessing the disk.

Note that the DATARATE register takes effect instantaneously. This means that it is possible to change the data rate for different tracks, sectors, or even bytes within sectors.

This means that it would be possible to, for example, use standard 500KHz DD encoding for the directory track and one data track, and then switch to HD encoding for the other 79 tracks of a disk. The result would be a disk that could contain a boot-loader programme and directory that can be read in a 1581, and that could be used to switch to the faster and higher-density HD encoding for the remaining data tracks. This could even be done for half of a disk, so that when used in a 1581, it loads at the DD speed, but when inserted in a MEGA65, it uses the HD data tracks, allowing the programme to load twice as fast, and fit twice as much data.

You are really only limited by your imagination, available time, and the limited number of people who are still interested in inserting a floppy disk into their computer!

## F011 Floppy Controller Registers

The following are the set of F011 compatibility registers of the 45IO47. Note that registers related to the use of SD-card based storage are found in the corresponding section below.

| HEX  | DEC   | DB7   | DB6   | DB5   | DB4   | DB3    | DB2   | DB1  | DB0    |
|------|-------|-------|-------|-------|-------|--------|-------|------|--------|
| D080 | 53376 | IRQ   | LED   | MOTOR | SWAP  | SIDE   |       |      | DS     |
| D081 | 53377 | WRCMD | RDCMD | FREE  | STEP  | DIR    | ALGO  | ALT  | NOBUF  |
| D082 | 53378 | BUSY  | DRQ   | EQ    | RNF   | CRC    | LOST  | PROT | TK0    |
| D083 | 53379 | RDREQ | WTREQ | RUN   | WGATE | DISKIN | INDEX | IRQ  | DSKCHG |
| D084 | 53380 |       |       |       |       | TRACK  |       |      |        |
| D085 | 53381 |       |       |       |       | SECTOR |       |      |        |
| D086 | 53382 |       |       |       |       | SIDE   |       |      |        |
| D087 | 53383 |       |       |       |       | DATA   |       |      |        |
| D088 | 53384 |       |       |       |       | CLOCK  |       |      |        |
| D089 | 53385 |       |       |       |       | STEP   |       |      |        |
| D08A | 53386 |       |       |       |       | PCODE  |       |      |        |

- **ALGO** Selects reading and writing algorithm (currently ignored).
- **ALT** Selects alternate DPLL read recovery method (not implemented)
- **BUSY** F011 FDC busy flag (command is being executed) (read only)

- **CLOCK** Set or read the clock pattern to be used when writing address and data marks. Should normally be left \$FF
- **COMMAND** F011 FDC command register
- **CRC** F011 FDC CRC check failure flag (read only)
- **DATA** F011 FDC data register (read/write) for accessing the floppy controller's 512 byte sector buffer
- **DIR** Sets the stepping direction (inward vs
- **DISKIN** F011 Disk sense (read only)
- **DRQ** F011 FDC DRQ flag (one or more bytes of data are ready) (read only)
- **DS** Drive select (0 to 7). Internal drive is 0. Second floppy drive on internal cable is 1. Other values reserved for C1565 external drive interface.
- **DSKCHG** F011 disk change sense (read only)
- **EQ** F011 FDC CPU and disk pointers to sector buffer are equal, indicating that the sector buffer is either full or empty. (read only)
- **FREE** Command is a free-format (low level) operation
- **INDEX** F011 Index hole sense (read only)
- **IRQ** The floppy controller has generated an interrupt (read only). Note that interrupts are not currently implemented on the 45GS27.
- **LED** Drive LED blinks when set
- **LOST** F011 LOST flag (data was lost during transfer, i.e., CPU did not read data fast enough) (read only)
- **MOTOR** Activates drive motor and LED (unless LED signal is also set, causing the drive LED to blink)
- **NOBUF** Reset the sector buffer read/write pointers
- **PCODE** (Read only) returns the protection code of the most recently read sector. Was intended for rudimentary copy protection. Not implemented.
- **PROT** F011 Disk write protect flag (read only)
- **RDCMD** Command is a read operation if set
- **RDREQ** F011 Read Request flag, i.e., the requested sector was found during a read operation (read only)
- **RNF** F011 FDC Request Not Found (RNF), i.e., a sector read or write operation did not find the requested sector (read only)

- **RUN** F011 Successive match. A synonym of RDREQ on the 45IO47 (read only)
- **SECTOR** F011 FDC sector selection register
- **SIDE** Directly controls the SIDE signal to the floppy drive, i.e., selecting which side of the media is active.
- **STEP** Writing 1 causes the head to step in the indicated direction
- **SWAP** Swap upper and lower halves of data buffer (i.e. invert bit 8 of the sector buffer)
- **TK0** F011 Head is over track 0 flag (read only)
- **TRACK** F011 FDC track selection register
- **WGATE** F011 write gate flag. Indicates that the drive is currently writing to media. Bad things may happen if a write transaction is aborted (read only)
- **WRCMD** Command is a write operation if set
- **WTREQ** F011 Write Request flag, i.e., the requested sector was found during a write operation (read only)

The following registers apply to the 45IO27 only, i.e., are MEGA65 specific:

| HEX  | DEC   | DB7     | DB6       | DB5       | DB4    | DB3      | DB2      | DB1  |
|------|-------|---------|-----------|-----------|--------|----------|----------|------|
| D6A0 | 54944 | DENSITY | DBGMOTORA | DBGMOTORA | DBGDIR | DBGDIR   | DBGWDATA | DBGW |
| D6A2 | 54946 |         |           |           |        | DATARATE |          |      |

- **DATARATE** Set number of bus cycles per floppy magnetic
- **DBGDIR** Control floppy drive STEPDIR line
- **DBGMOTORA** Control floppy drive MOTOR line
- **DBGWDATA** Control floppy drive WDATA line
- **DBGWGATE** Control floppy drive WGATE line
- **DENSITY** Control floppy drive density select line

## SD CARD CONTROLLER AND F011 VIRTUALISATION FUNCTIONS

For those situations where you do not wish to use real floppy disks, the 45IO27 supports two complementary alternative modes:

- SD-Card Based Disk Image Access.
- Virtualised Disk Image Access.

This is in addition to providing direct access to a dual-bus SD card interface.

## SD-Card Based Disk Image Access

The 45IO27 is both a floppy drive and SD card controller. This enables it to transparently allow access to D81 disk images stored on the SD card. Further, because the controller is combined, it is possible to still have the floppy drive step and spin as though it were being used, providing considerable atmosphere and sense of realism, even when using disk images.

The 45IO27 supports both 800KB standard D81 disk images, as well as 64MiB “MEGA Images”. While an operating system may impose restrictions based on the name of a file, the 45IO27 is blind to these requirements. Instead, it requires only that a contiguous 800KiB or 64MiB of the SD card is used to contain a disk image.

When a disk image is enabled, the corresponding set of sectors on the SD card are effectively placed under user control, and the operating system is no longer able to prevent the reading or writing of any of those sectors. Thus you should never enable access to an image that is shorter than the required size, as it will otherwise allow the user to unwittingly or maliciously access and/or modify data that is not part of the image file.

For the same reason, only the hypervisor can change the sector number where a disk image starts (the D?STARTSEC? signals), or allow the use of disk images instead of the real floppy drive (USEREAL0 and USEREAL1 signals). Once the Hypervisor has set the start sector of a disk image, and cleared the USEREAL0 or USEREAL1 signal, the user can still control whether an access will go to the real floppy drive or to the disk image by respectively clearing or setting the appropriate signal. For drive 0, this is D0IMG, and for drive 1, it is D1IMG.

There are also signals to control whether a disk image is an 800KiB D81 image or a 64MiB MEGA Disk image, and whether a disk image is present, and whether it is write protected. These are all located in the \$D68B register. Because of the ability of manipulation of these registers to corrupt or improperly access data, these signals are all read-only, except from within the hypervisor.

The following table lists the registers that are used to control access to disk images resident on the SD card:

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b> | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|------------|------------|------------|------------|
| D68B       | 54923      | D1MD       | D0MD       | D1WP       | D1P        | D1IMG      | D0WP       | D0P        | D0IMG      |

continued ...

...continued

| HEX  | DEC   | DB7 | DB6 | DB5 | DB4 | DB3         | DB2     | DB1      | DB0 |
|------|-------|-----|-----|-----|-----|-------------|---------|----------|-----|
| D68C | 54924 |     |     |     |     | D0STARTSEC0 |         |          |     |
| D68D | 54925 |     |     |     |     | D0STARTSEC1 |         |          |     |
| D68E | 54926 |     |     |     |     | D0STARTSEC2 |         |          |     |
| D68F | 54927 |     |     |     |     | D0STARTSEC3 |         |          |     |
| D690 | 54928 |     |     |     |     | D0STARTSEC0 |         |          |     |
| D691 | 54929 |     |     |     |     | D1STARTSEC1 |         |          |     |
| D692 | 54930 |     |     |     |     | D2STARTSEC2 |         |          |     |
| D693 | 54931 |     |     |     |     | D3STARTSEC3 |         |          |     |
| D6A1 | 54945 |     | -   |     |     | USEREAL1    | TARGANY | USEREAL0 |     |

- **DOIMG** F011 disk 0 use disk image if set, otherwise use real floppy drive.
- **DOMD** F011 drive 0 disk image is 64MiB mega image if set (otherwise 800KiB 1581 image)
- **DOP** F011 drive 0 media present
- **DOSTARTSEC0** F011 disk 1 disk image address on SD card (LSB)
- **DOSTARTSEC1** F011 disk 1 disk image address on SD card (2nd byte)
- **DOSTARTSEC2** F011 disk 1 disk image address on SD card (3rd byte)
- **DOSTARTSEC3** F011 disk 1 disk image address on SD card (MSB)
- **DOWP** Write protect F011 drive 0
- **D1IMG** F011 disk 1 use disk image if set, otherwise use real floppy drive.
- **D1MD** F011 drive 1 disk image is 64MiB mega image if set (otherwise 800KiB 1581 image)
- **D1P** F011 drive 1 media present
- **D1STARTSEC1** F011 disk 2 disk image address on SD card (2nd byte)
- **D1WP** Write protect F011 drive 1
- **D2STARTSEC2** F011 disk 2 disk image address on SD card (3rd byte)
- **D3STARTSEC3** F011 disk 2 disk image address on SD card (MSB)
- **TARGANY** Read next sector under head if set, ignoring the requested side, track and sector number.
- **USEREAL0** Use real floppy drive for drive 0 if set (read-only, except for from hypervisor)

- **USEREAL1** Use real floppy drive for drive 1 if set (read-only, except for from hypervisor)

## F011 Virtualisation

In addition to allowing automatic read and write access to SD-card based D81 images, it is possible to connect a programme to the serial monitor interface that provides and accepts data as though it were the floppy disk.

This is commonly used in a cross-development environment, where you wish to frequently modify a disk image that is used by a programme you are developing – without the need to continually push new versions of the disk image on the MEGA65's SD-Card first. It also has the added benefit that it allows you to easily visualise which sectors are being read from and written to, which can help speed up development and debugging of your programme.

This function operates together with the MEGA65's Hypervisor by triggering hyperrupts (that is, interrupts that activate the Hypervisor). There is then special code in the Hypervisor that communicates with the `m65` programme via the serial monitor interface.

If that all sounds rather complex, all you need to know is that to use this function, you run the `m65` utility with arguments like `-d image.d81`. This should automatically establish the link with the MEGA65. If the BASIC interpreter stops responding, press the reset button (not the power switch) on the left side of your MEGA65, and it should return to the BASIC's READY. prompt – and if your supplied disk image has a C65 auto-boot function, then it should automatically start booting.

This function works very well if the host computer runs Linux, and will allow loading at a speed of around 60KiB per second. However, it may be much slower on Windows or Apple OSX-based systems.

Of course to use this, you will also need an interface module and/or cable to connect your cross-development system to the MEGA65's serial monitor interface. This is most easily done using a Trenz TE0790-03 JTAG adapter and mini-USB cable.

More information on using this interface and the `m65` tool can be found in Chapter/Appendix 12 on page 12-3.

## Dual-Bus SD Card Controller

The 45IO27 contains a high-speed dual-bus SD card controller. This controller operates in SPI x1 mode at a clock speed of 20MHz, providing a maximum throughput of approximately 2MiB/sec. The quality of the SD card makes a significant difference to

performance, with some cards routinely delivering 1.7MiB/sec, while others 1MiB/sec or less. Generally speaking, newer cards marketed as being suitable for video recording perform better. The controller supports SDHC cards, and has experimental support for SDXC cards. Legacy SD cards with a capacity of 2GiB or less are not supported, as these use a different addressing mode.

The SD controller itself is very simple to drive: Supply the sector number in \$D861-\$D684, and then issue a read or write command to the command register (\$D680). The SD controller supports only sector-based buffered operations, using the sector buffer. In hypervisor mode, the sector buffer is located at \$FFD6E00 - \$FFD6FFF, while when the computer is in normal operating mode, the SD card and the floppy controller share a single address for both the floppy drive and SD card sector buffers. Which buffer is visible at that address is dictated by the BUFSEL signal. If it is 1, then the SD card buffer is visible, while if it is 0, then the floppy drive sector buffer is visible. See also Sub-section Q on page Q-4 for further discussion on the precise behaviour of this buffer with regard to normal mode versus Hypervisor mode, and how it can also be mapped at \$DE00.

## Write Gate

When writing a sector, you must, however, first open the “write gate”. This is a mechanism to prevent accidental corruption of data on the SD card, as it requires two different values to be written to the command register (\$D680) in quick succession: You have approximately 1 milli second after opening the write gate to command the write, before the write gate effectively closes again, write-protecting the SD card until the write gate is opened again. There are two different write gates: One for the master boot record (sector 0), and the other for all other sectors, both of which are listed in the command table below. This is designed to provide additional protection to the very important master boot record sector against programmes accidentally calculating sector 0 as the target for an ordinary write.

## Fill Mode

Where you wish to fill sectors with a constant value, the 45IO27 supports a mode for this, so that you do not need to overwrite the contents of the sector buffer. This is activated by placing the desired fill value into the FILLVAL register (\$D686), and then issuing the enable fill mode command (\$83), performing the sector write operations, and then issuing the disable fill mode command (\$84).

# Selecting Among Multiple SD Cards

The controller supports two SD Card interfaces, and it is possible to have a card in both at the same time. However, each card needs to be reset and commanded separately. Only one card can be commanded at a time. That said, it is possible to reset each card once, and then switch between the cards to perform individual operations.

To select the first SD Card slot, write \$C0 to the SD Controller Command Register (\$D680). To select the second SD Card slot, write \$C1 instead.

## SD Controller Command Table

The SD controller supports the following commands that can be written to the command register at \$D680:

| Command   | Function                                                             |
|-----------|----------------------------------------------------------------------|
| \$00 (0)  | Place SD card under reset (deprecated. Use command \$10 instead)     |
| \$01 (1)  | Release SD card from reset                                           |
| \$02 (2)  | Read a sector from the SD card                                       |
| \$03 (3)  | Write a single sector to the SD card                                 |
| \$04 (4)  | Write the first sector of a multi-sector write to the SD card        |
| \$05 (5)  | Write a subsequent sector of a multi-sector write to the SD card     |
| \$06 (6)  | Write the final sector of a multi-sector write to the SD card        |
| \$0C (12) | Request flush of SD card write buffers (experimental)                |
| \$0E (14) | Pull SD handshake line low (debug only)                              |
| \$0F (15) | Pull SD handshake line high (debug only)                             |
| \$10 (16) | Place SD card under reset with flags set (preferred method)          |
| \$11 (17) | Release SD card from reset (alternate method)                        |
| \$40 (64) | Clear the SDHC/SDXC flag, selecting legacy SD card mode (deprecated) |
| \$41 (65) | Set the SDHC/SDXC mode flag                                          |
| \$44 (68) | End force clearing of SD card state machine error flag               |
| \$45 (69) | Begin force clearing of SD card state machine error flag             |

continued ...

...continued

| Command    | Function                                                                          |
|------------|-----------------------------------------------------------------------------------|
| \$4D (77)  | Open write-gate to sector 0 (master boot record) for approximately 1 milli-second |
| \$57 (87)  | Open write-gate for all sectors > 0 for approximately 1 milli-second              |
| \$81 (129) | Enable mapping of the SD/FDC sector buffer at \$DE00 - \$DFFF                     |
| \$82 (130) | Disable mapping of the SD/FDC sector buffer at \$DE00 - \$DFFF                    |
| \$83 (131) | Enable SD Card Fill Mode                                                          |
| \$84 (132) | Disable SD Card Fill Mode                                                         |
| \$C0 (192) | Select SD Card Slot 0                                                             |
| \$C1 (193) | Select SD Card Slot 1                                                             |

Note that the hypervisor can enable or disable direct access to the SD controller. The hypervisor operating system may provide a mechanism for requesting permission to access the SD card controller, e.g., for disk management utilities.

The SD card controller registers are as follows:

| HEX  | DEC   | DB7        | DB6   | DB5   | DB4   | DB3    | DB2      | DB1     | DB0 |
|------|-------|------------|-------|-------|-------|--------|----------|---------|-----|
| D680 | 54912 | CMDANDSTAT |       |       |       |        |          |         |     |
| D681 | 54913 | SECTOR0    |       |       |       |        |          |         |     |
| D682 | 54914 | SECTOR1    |       |       |       |        |          |         |     |
| D683 | 54915 | SECTOR2    |       |       |       |        |          |         |     |
| D684 | 54916 | SECTOR3    |       |       |       |        |          |         |     |
| D686 | 54918 | FILLVAL    |       |       |       |        |          |         |     |
| D68A | 54922 | -          |       | VFDC1 | VFDC0 | VICIII | CDC00    |         |     |
| D6AF | 54959 | -          | VLOST | VDRQ  | VRNF  | VEQINH | VWFFOUND | VRFOUND |     |

- **CDC00** (read only) Set if colour RAM at \$DC00
- **CMDANDSTAT** SD controller status/command
- **FILLVAL** WRITE ONLY set fill byte for use in fill mode, instead of SD buffer data
- **SECTOR0** SD controller SD sector address (LSB)
- **SECTOR1** SD controller SD sector address (2nd byte)
- **SECTOR2** SD controller SD sector address (3rd byte)
- **SECTOR3** SD controller SD sector address (MSB)

- **VDRQ** Manually set f011\_drq signal (indented for virtual F011 mode only)
- **VEQINH** Manually set f011\_eq\_inhibit signal (indented for virtual F011 mode only)
- **VFDC0** (read only) Set if drive 0 is virtualised (sectors delivered via serial monitor interface)
- **VFDC1** (read only) Set if drive 1 is virtualised (sectors delivered via serial monitor interface)
- **VICIII** (read only) Set if VIC-IV or ethernet IO bank visible
- **VLOST** Manually set f011\_lost signal (indented for virtual F011 mode only)
- **VRFOUND** Manually set f011\_rsector\_found signal (indented for virtual F011 mode only)
- **VRNF** Manually set f011\_rnf signal (indented for virtual F011 mode only)
- **VWFOUND** Manually set f011\_wsector\_found signal (indented for virtual F011 mode only)

## TOUCH PANEL INTERFACE

Some MEGA65 variants include an LCD touch panel, primarily the MEGAphone hand-held version of the MEGA65. The touch interface supports the detection of two simultaneous touch events. Some variants may also support gesture detection, however, this is still very experimental.

The touch detection interface that is contained in the 45IO27 is complemented by the on-screen-keyboard interface of the 4551 UART and GPIO controller. Refer to section O for further information. Of particular relevance are bit 7 of the registers \$D615 – \$D617 which allow activating the on-screen keyboard interface, selecting whether the on-screen keyboard is placed in the upper or lower portion of the screen, and whether the primary or secondary on-screen keyboard is displayed.

Direct connections between the 4551 and the 45IO27 combine information about any currently displayed on-screen keyboard and the touch interface controller, allowing synthetic keyboard events to be automatically triggered when the on-screen keyboard portion of the touch interface is pressed. This allows the touch interface to be used to drive the on-screen keyboard without requiring any support from user programmes. This works even when the on-screen keyboard is moving during activation or transitioning between the top and bottom of the screen.

As touch interfaces can require calibration, the 45IO27 allows for a linear transformation of both the X and Y coordinates of a touch event. Specifically, there are scale (TCHXS-

CALE and TCHYSCALE) and offset registers (TCHXDELTA and TCHYDELTA) that provide for this transformation. It is also possible to flip the touch screen coordinates in either or both the X and Y axes. These calibration registers also affect the operation of the on-screen keyboard.

It should also be noted that some touch interfaces do not have constant horizontal or vertical resolution. For example, some panels have a low horizontal resolution region in the middle of the panel, which can require some care to accommodate.

To detect the primary touch event, the TOUCH1XLSB, TOUCH1XMSB, TOUCH1YLSB, TOUCH1YMSB registers can be read. Similar registers exist for the 2nd touch event: TOUCH2XLSB, TOUCH2XMSB, TOUCH2YLSB, TOUCH2YMSB. Each touch event has a single bit flag that indicates whether the touch event is currently valid: the EV1 and EV2 bits of the register \$D6B0. There are also corresponding bit-fields that indicate whether a given touch event has been made or released, allowing the detection of when a finger both makes and breaks contact with the screen. The UPDN1 and UPDN2 signals provide this information. Binary values of 01 and 10, respectively indicate if the finger has been removed or pressed against the touch panel. Values of 00 and 11 mean that a finger is either being held or not being held against the touch panel.

The primary touch event is also fed into the lightpen input of the VIC-IV, and can be detected using the normal light pen registers of the VIC-IV.

The registers for the touch panel interface are as follows:

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b>   | <b>DB3</b> | <b>DB2</b> | <b>DB1</b>  | <b>DB0</b> |
|------------|------------|------------|------------|------------|--------------|------------|------------|-------------|------------|
| D6B0       | 54960      | YINV       | XINV       |            | UPDN2        |            | UPDN1      | EV2         | EV1        |
| D6B1       | 54961      |            |            |            | CALXSCALELSB |            |            |             |            |
| D6B2       | 54962      |            |            |            | CALXSCALEMSB |            |            |             |            |
| D6B3       | 54963      |            |            |            | CALYSCALELSB |            |            |             |            |
| D6B4       | 54964      |            |            |            | CALYSCALEMSB |            |            |             |            |
| D6B5       | 54965      |            |            |            | CALXDELTALSB |            |            |             |            |
| D6B7       | 54967      |            |            |            | CALYDELTALSB |            |            |             |            |
| D6B8       | 54968      |            |            |            | CALYDELTAMSB |            |            |             |            |
| D6B9       | 54969      |            |            |            | TOUCH1XLSB   |            |            |             |            |
| D6BA       | 54970      |            |            |            | TOUCH1YLSB   |            |            |             |            |
| D6BB       | 54971      | -          |            | TOUCH1YMSB |              | -          |            | TOUCH1XMSB  |            |
| D6BC       | 54972      |            |            |            | TOUCH2XLSB   |            |            |             |            |
| D6BD       | 54973      |            |            |            | TOUCH2YLSB   |            |            |             |            |
| D6BE       | 54974      | -          |            | TOUCH2YMSB |              | -          |            | TOUCH2XMSB  |            |
| D6C0       | 54976      |            |            | GESTUREID  |              |            |            | GESTUREDIRE |            |

- **CALXDELTALSB** Touch pad X delta LSB

- **CALXSCALELSB** Touch pad X scaling LSB
- **CALXSCALEMSB** Touch pad X scaling MSB
- **CALYDELTALSB** Touch pad Y delta LSB
- **CALYDELTAMSB** Touch pad Y delta MSB
- **CALYSCALELSB** Touch pad Y scaling LSB
- **CALYSCALEMSB** Touch pad Y scaling MSB
- **EV1** Touch event 1 is valid
- **EV2** Touch event 2 is valid
- **GESTUREDIR** Touch pad gesture directions (left,right,up,down)
- **GESTUREID** Touch pad gesture ID
- **TOUCH1XLSB** Touch pad touch 1 X LSB
- **TOUCH1XMSB** Touch pad touch #1 X MSBs
- **TOUCH1YLSB** Touch pad touch 1 Y LSB
- **TOUCH1YMSB** Touch pad touch #1 Y MSBs
- **TOUCH2XLSB** Touch pad touch #2 X LSB
- **TOUCH2XMSB** Touch pad touch #2 X MSBs
- **TOUCH2YLSB** Touch pad touch #2 Y LSB
- **TOUCH2YMSB** Touch pad touch #2 Y MSBs
- **UPDN1** Touch event 1 up/down state
- **UPDN2** Touch event 2 up/down state
- **XINV** Invert horizontal axis
- **YINV** Invert vertical axis

## AUDIO SUPPORT FUNCTIONS

The 45IO27 provides the primary interface into the MEGA65's full cross-bar audio mixer. This includes the interface for reading or modifying the mixer co-efficients, as well as accessing the mixer feedback registers, and setting the 16-bit digital sample values that are two of the input channels into the audio mixer.

The audio mixer consists of 128 coefficients, each of which is 16 bits. Each audio output channel, e.g., left speaker, right speaker, left headphone, right headphone, cellular modem 1 (MEGAphone models only) and so on, are generated by taking each of the audio input channels, multiplying them by the appropriate coefficient, and adding it to the total output of the audio output channel.

Because each audio output channel has its own set of coefficients that are applied to all of the audio input channels, this means that it is possible to produce totally different audio out each audio channel: For example, it is possible to play your favourite quadrophonic SID music out of the headphones while rick-rolling passers by with Amiga-style MOD audio. This is why the audio mixer is referred to as a full cross-bar mixer, because there are no restrictions on how you mix each audio output channel. In this regard, it is very similar to a full-function audio desk, allowing different mixing levels for different speakers.

Because the audio coefficients are 16 bits each, each one is formed using two successive bytes of the audio coefficient space. Changes to the audio coefficients take effect immediately, so care should be taken when changing coefficients to avoid audible clicks and pops. Also, you must allow 32 cycles to elapse before changing the selected audio coefficient, as otherwise the change may be discarded if the audio mixer accumulator has not had time to re-visit that coefficient.

The audio sources on the MEGA65 and MEGAphone devices are as follows:

| <b>Input Channel ID</b> | <b>Connection</b>                                     |
|-------------------------|-------------------------------------------------------|
| \$0 (0)                 | Left SIDs                                             |
| \$1 (1)                 | Right SIDs                                            |
| \$2 (2)                 | Modem Bay 1 (MEGAphone only)                          |
| \$3 (3)                 | Modem Bay 2 (MEGAphone only)                          |
| \$4 (4)                 | Bluetooth(tm) Left                                    |
| \$5 (5)                 | Bluetooth(tm) Right                                   |
| \$6 (6)                 | Headphone Interface 1                                 |
| \$7 (7)                 | Headphone Interface 2                                 |
| \$8 (8)                 | Digital audio Left                                    |
| \$9 (9)                 | Digital audio Right                                   |
| \$A (10)                | MEMs Microphone 0 (Nexys4 and MEGAphone only)         |
| \$B (11)                | MEMs Microphone 1 (MEGAphone only)                    |
| \$C (12)                | MEMs Microphone 2 (MEGAphone only)                    |
| \$D (13)                | MEMs Microphone 3 (MEGAphone only)                    |
| \$E (14)                | Headphone jack microphone (Nexys4 and MEGAphone only) |

continued ...

...continued

| <b>Input Channel ID</b> | <b>Connection</b>                                                    |
|-------------------------|----------------------------------------------------------------------|
| \$F (15)                | OPL-compatible FM audio ( <i>shares co-efficient with input 14</i> ) |

The OPL-compatible FM audio which is on source 15 is controlled by the coefficient for source 14. This is because the coefficient for source 15 provides the master volume level for each output.

The audio cross-bar mixer supports the following eight output channels:

| <b>Output Channel ID</b> | <b>Connection</b>                                                                                                    |
|--------------------------|----------------------------------------------------------------------------------------------------------------------|
| \$0 (0)                  | Left Primary Speaker (digital audio on MEGA65 R2/R3, physical speaker on MEGAphone, headphone jack audio on Nexys4)  |
| \$1 (1)                  | Right Primary Speaker (digital audio on MEGA65 R2/R3, physical speaker on MEGAphone, headphone jack audio on Nexys4) |
| \$2 (2)                  | Modem Bay 1 audio output (MEGAphone only)                                                                            |
| \$3 (3)                  | Modem Bay 2 audio output (MEGAphone only)                                                                            |
| \$4 (4)                  | Bluetooth Left Audio (MEGAphone only)                                                                                |
| \$5 (5)                  | Bluetooth Right Audio (MEGAphone only)                                                                               |
| \$6 (6)                  | Headphone Left output (MEGA65 R2/R3 and MEGAphone only. On Nexys4 boards the primary speaker drives the 3.5mm jack)  |
| \$7 (7)                  | Headphone Right output (MEGA65 R2/R3 and MEGAphone only. On Nexys4 boards the primary speaker drives the 3.5mm jack) |

To determine the coefficient register number for a given source and output, multiply the output number by 32 and multiply the source number by 2. This will be the register number for the LSB of the 16-bit coefficient. The MSB will be the next register. For example, to set the coefficient of the right SIDs to the 2nd modem bay audio output, the coefficient would be  $32 \times 3 + 1 \times 2 = 96 + 2 = 98$ .

XXX - mixer stuff XXX - mixer feedback registers XXX - Left and right digi XXX - CPU register for selecting PWM/PDM

| <b>HEX</b> | <b>DEC</b> | <b>DB7</b> | <b>DB6</b> | <b>DB5</b> | <b>DB4</b> | <b>DB3</b>  | <b>DB2</b> | <b>DB1</b> | <b>DB0</b> |
|------------|------------|------------|------------|------------|------------|-------------|------------|------------|------------|
| D6F4       | 55028      |            |            |            |            | MIXREGSEL   |            |            |            |
| D6F5       | 55029      |            |            |            |            | MIXREGDATA  |            |            |            |
| D6F8       | 55032      |            |            |            |            | DIGILLSB    |            |            |            |
| D6F9       | 55033      |            |            |            |            | DIGILMSB    |            |            |            |
| D6FA       | 55034      |            |            |            |            | DIGIRLSB    |            |            |            |
| D6FB       | 55035      |            |            |            |            | DIGIRMSB    |            |            |            |
| D6FC       | 55036      |            |            |            |            | READBACKLSB |            |            |            |
| D6FD       | 55037      |            |            |            |            | READBACKMSB |            |            |            |
| D711       | 55057      |            | -          |            | PWMPDM     |             | -          |            |            |

- **DIGILEFTLSB** Digital audio, left channel, LSB
- **DIGILEFTMSB** Digital audio, left channel, MSB
- **DIGILLSB** 16-bit digital audio out (left LSB)
- **DIGILMSB** 16-bit digital audio out (left MSB)
- **DIGIRIGHTLSB** Digital audio, left channel, LSB
- **DIGIRIGHTMSB** Digital audio, left channel, MSB
- **DIGIRLSB** 16-bit digital audio out (right LSB)
- **DIGIRMSB** 16-bit digital audio out (right MSB)
- **MIXREGDATA** Audio Mixer register read port
- **MIXREGSEL** Audio Mixer register select
- **PWMPDM** PWM/PDM audio encoding select
- **READBACKLSB** audio read-back LSB (source selected by \$D6F4)
- **READBACKMSB** audio read-back MSB (source selected by \$D6F4)

## MISCELLANEOUS IO FUNCTIONS



# R

## APPENDIX

### Reference Tables

- Units of Storage
- Base Conversion



# UNITS OF STORAGE

| <b>Unit</b> | <b>Equals</b>          | <b>Abbreviation</b> |
|-------------|------------------------|---------------------|
| 1 Bit       |                        |                     |
| 1 Nibble    | 4 Bits                 |                     |
| 1 Byte      | 8 bits                 | B                   |
| 1 Kilobyte  | 1024 B                 | KB                  |
| 1 Megabyte  | 1024 KB or 1,048,576 B | MB                  |

# BASE CONVERSION

| Decimal | Binary | Hexadecimal |
|---------|--------|-------------|
| 0       | %0     | \$0         |
| 1       | %1     | \$1         |
| 2       | %10    | \$2         |
| 3       | %11    | \$3         |
| 4       | %100   | \$4         |
| 5       | %101   | \$5         |
| 6       | %110   | \$6         |
| 7       | %111   | \$7         |
| 8       | %1000  | \$8         |
| 9       | %1001  | \$9         |
| 10      | %1010  | \$A         |
| 11      | %1011  | \$B         |
| 12      | %1100  | \$C         |
| 13      | %1101  | \$D         |
| 14      | %1110  | \$E         |
| 15      | %1111  | \$F         |
| 16      | %10000 | \$10        |
| 17      | %10001 | \$11        |
| 18      | %10010 | \$12        |
| 19      | %10011 | \$13        |
| 20      | %10100 | \$14        |
| 21      | %10101 | \$15        |
| 22      | %10110 | \$16        |
| 23      | %10111 | \$17        |
| 24      | %11000 | \$18        |
| 25      | %11001 | \$19        |
| 26      | %11010 | \$1A        |
| 27      | %11011 | \$1B        |
| 28      | %11100 | \$1C        |
| 29      | %11101 | \$1D        |
| 30      | %11110 | \$1E        |
| 31      | %11111 | \$1F        |

| Decimal | Binary  | Hexadecimal |
|---------|---------|-------------|
| 32      | %100000 | \$20        |
| 33      | %100001 | \$21        |
| 34      | %100010 | \$22        |
| 35      | %100011 | \$23        |
| 36      | %100100 | \$24        |
| 37      | %100101 | \$25        |
| 38      | %100110 | \$26        |
| 39      | %100111 | \$27        |
| 40      | %101000 | \$28        |
| 41      | %101001 | \$29        |
| 42      | %101010 | \$2A        |
| 43      | %101011 | \$2B        |
| 44      | %101100 | \$2C        |
| 45      | %101101 | \$2D        |
| 46      | %101110 | \$2E        |
| 47      | %101111 | \$2F        |
| 48      | %110000 | \$30        |
| 49      | %110001 | \$31        |
| 50      | %110010 | \$32        |
| 51      | %110011 | \$33        |
| 52      | %110100 | \$34        |
| 53      | %110101 | \$35        |
| 54      | %110110 | \$36        |
| 55      | %110111 | \$37        |
| 56      | %111000 | \$38        |
| 57      | %111001 | \$39        |
| 58      | %111010 | \$3A        |
| 59      | %111011 | \$3B        |
| 60      | %111100 | \$3C        |
| 61      | %111101 | \$3D        |
| 62      | %111110 | \$3E        |
| 63      | %111111 | \$3F        |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 64             | %1000000      | \$40               |
| 65             | %1000001      | \$41               |
| 66             | %1000010      | \$42               |
| 67             | %1000011      | \$43               |
| 68             | %1000100      | \$44               |
| 69             | %1000101      | \$45               |
| 70             | %1000110      | \$46               |
| 71             | %1000111      | \$47               |
| 72             | %1001000      | \$48               |
| 73             | %1001001      | \$49               |
| 74             | %1001010      | \$4A               |
| 75             | %1001011      | \$4B               |
| 76             | %1001100      | \$4C               |
| 77             | %1001101      | \$4D               |
| 78             | %1001110      | \$4E               |
| 79             | %1001111      | \$4F               |
| 80             | %1010000      | \$50               |
| 81             | %1010001      | \$51               |
| 82             | %1010010      | \$52               |
| 83             | %1010011      | \$53               |
| 84             | %1010100      | \$54               |
| 85             | %1010101      | \$55               |
| 86             | %1010110      | \$56               |
| 87             | %1010111      | \$57               |
| 88             | %1011000      | \$58               |
| 89             | %1011001      | \$59               |
| 90             | %1011010      | \$5A               |
| 91             | %1011011      | \$5B               |
| 92             | %1011100      | \$5C               |
| 93             | %1011101      | \$5D               |
| 94             | %1011110      | \$5E               |
| 95             | %1011111      | \$5F               |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 96             | %1100000      | \$60               |
| 97             | %1100001      | \$61               |
| 98             | %1100010      | \$62               |
| 99             | %1100011      | \$63               |
| 100            | %1100100      | \$64               |
| 101            | %1100101      | \$65               |
| 102            | %1100110      | \$66               |
| 103            | %1100111      | \$67               |
| 104            | %1101000      | \$68               |
| 105            | %1101001      | \$69               |
| 106            | %1101010      | \$6A               |
| 107            | %1101011      | \$6B               |
| 108            | %1101100      | \$6C               |
| 109            | %1101101      | \$6D               |
| 110            | %1101110      | \$6E               |
| 111            | %1101111      | \$6F               |
| 112            | %1110000      | \$70               |
| 113            | %1110001      | \$71               |
| 114            | %1110010      | \$72               |
| 115            | %1110011      | \$73               |
| 116            | %1110100      | \$74               |
| 117            | %1110101      | \$75               |
| 118            | %1110110      | \$76               |
| 119            | %1110111      | \$77               |
| 120            | %1111000      | \$78               |
| 121            | %1111001      | \$79               |
| 122            | %1111010      | \$7A               |
| 123            | %1111011      | \$7B               |
| 124            | %1111100      | \$7C               |
| 125            | %1111101      | \$7D               |
| 126            | %1111110      | \$7E               |
| 127            | %1111111      | \$7F               |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 128            | %10000000     | \$80               |
| 129            | %10000001     | \$81               |
| 130            | %10000010     | \$82               |
| 131            | %10000011     | \$83               |
| 132            | %10000100     | \$84               |
| 133            | %10000101     | \$85               |
| 134            | %10000110     | \$86               |
| 135            | %10000111     | \$87               |
| 136            | %10001000     | \$88               |
| 137            | %10001001     | \$89               |
| 138            | %10001010     | \$8A               |
| 139            | %10001011     | \$8B               |
| 140            | %10001100     | \$8C               |
| 141            | %10001101     | \$8D               |
| 142            | %10001110     | \$8E               |
| 143            | %10001111     | \$8F               |
| 144            | %10010000     | \$90               |
| 145            | %10010001     | \$91               |
| 146            | %10010010     | \$92               |
| 147            | %10010011     | \$93               |
| 148            | %10010100     | \$94               |
| 149            | %10010101     | \$95               |
| 150            | %10010110     | \$96               |
| 151            | %10010111     | \$97               |
| 152            | %10011000     | \$98               |
| 153            | %10011001     | \$99               |
| 154            | %10011010     | \$9A               |
| 155            | %10011011     | \$9B               |
| 156            | %10011100     | \$9C               |
| 157            | %10011101     | \$9D               |
| 158            | %10011110     | \$9E               |
| 159            | %10011111     | \$9F               |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 160            | %10100000     | \$A0               |
| 161            | %10100001     | \$A1               |
| 162            | %10100010     | \$A2               |
| 163            | %10100011     | \$A3               |
| 164            | %10100100     | \$A4               |
| 165            | %10100101     | \$A5               |
| 166            | %10100110     | \$A6               |
| 167            | %10100111     | \$A7               |
| 168            | %10101000     | \$A8               |
| 169            | %10101001     | \$A9               |
| 170            | %10101010     | \$AA               |
| 171            | %10101011     | \$AB               |
| 172            | %10101100     | \$AC               |
| 173            | %10101101     | \$AD               |
| 174            | %10101110     | \$AE               |
| 175            | %10101111     | \$AF               |
| 176            | %10110000     | \$B0               |
| 177            | %10110001     | \$B1               |
| 178            | %10110010     | \$B2               |
| 179            | %10110011     | \$B3               |
| 180            | %10110100     | \$B4               |
| 181            | %10110101     | \$B5               |
| 182            | %10110110     | \$B6               |
| 183            | %10110111     | \$B7               |
| 184            | %10111000     | \$B8               |
| 185            | %10111001     | \$B9               |
| 186            | %10111010     | \$BA               |
| 187            | %10111011     | \$BB               |
| 188            | %10111100     | \$BC               |
| 189            | %10111101     | \$BD               |
| 190            | %10111110     | \$BE               |
| 191            | %10111111     | \$BF               |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 192            | %11000000     | \$C0               |
| 193            | %11000001     | \$C1               |
| 194            | %11000010     | \$C2               |
| 195            | %11000011     | \$C3               |
| 196            | %11000100     | \$C4               |
| 197            | %11000101     | \$C5               |
| 198            | %11000110     | \$C6               |
| 199            | %11000111     | \$C7               |
| 200            | %11001000     | \$C8               |
| 201            | %11001001     | \$C9               |
| 202            | %11001010     | \$CA               |
| 203            | %11001011     | \$CB               |
| 204            | %11001100     | \$CC               |
| 205            | %11001101     | \$CD               |
| 206            | %11001110     | \$CE               |
| 207            | %11001111     | \$CF               |
| 208            | %11010000     | \$D0               |
| 209            | %11010001     | \$D1               |
| 210            | %11010010     | \$D2               |
| 211            | %11010011     | \$D3               |
| 212            | %11010100     | \$D4               |
| 213            | %11010101     | \$D5               |
| 214            | %11010110     | \$D6               |
| 215            | %11010111     | \$D7               |
| 216            | %11011000     | \$D8               |
| 217            | %11011001     | \$D9               |
| 218            | %11011010     | \$DA               |
| 219            | %11011011     | \$DB               |
| 220            | %11011100     | \$DC               |
| 221            | %11011101     | \$DD               |
| 222            | %11011110     | \$DE               |
| 223            | %11011111     | \$DF               |

| <b>Decimal</b> | <b>Binary</b> | <b>Hexadecimal</b> |
|----------------|---------------|--------------------|
| 224            | %11100000     | \$E0               |
| 225            | %11100001     | \$E1               |
| 226            | %11100010     | \$E2               |
| 227            | %11100011     | \$E3               |
| 228            | %11100100     | \$E4               |
| 229            | %11100101     | \$E5               |
| 230            | %11100110     | \$E6               |
| 231            | %11100111     | \$E7               |
| 232            | %11101000     | \$E8               |
| 233            | %11101001     | \$E9               |
| 234            | %11101010     | \$EA               |
| 235            | %11101011     | \$EB               |
| 236            | %11101100     | \$EC               |
| 237            | %11101101     | \$ED               |
| 238            | %11101110     | \$EE               |
| 239            | %11101111     | \$EF               |
| 240            | %11110000     | \$F0               |
| 241            | %11110001     | \$F1               |
| 242            | %11110010     | \$F2               |
| 243            | %11110011     | \$F3               |
| 244            | %11110100     | \$F4               |
| 245            | %11110101     | \$F5               |
| 246            | %11110110     | \$F6               |
| 247            | %11110111     | \$F7               |
| 248            | %11111000     | \$F8               |
| 249            | %11111001     | \$F9               |
| 250            | %11111010     | \$FA               |
| 251            | %11111011     | \$FB               |
| 252            | %11111100     | \$FC               |
| 253            | %11111101     | \$FD               |
| 254            | %11111110     | \$FE               |
| 255            | %11111111     | \$FF               |



# S APPENDIX

## Flashing the FPGAs and CPLDs in the MEGA65

- Suggested PC specifications
- Warning
- Installing Vivado
- Installing the FTDI drivers
- Flashing the main FPGA using Vivado
- Flashing the CPLD in the  
MEGA65's Keyboard with  
Lattice Diamond

**• Flashing the MAX10 FPGA on  
the MEGA65's Mainboard  
with INTEL QUARTUS**





The MEGA65 is an open-source and open-hardware computer. This means you are free, not only to write programs that run on the MEGA65 as a finished computer, but also to use the re-programmable chips in the MEGA65 to turn it into all sorts of other things.

If you just want to install an upgrade core for the MEGA65, or a core that lets you use your MEGA65 as another type of computer, you probably want to look in Chapter/Appendix 5 on page 5-5 instead.

This chapter is more intended for people who want to help develop cores for the MEGA65. This chapter may also be of interest to Nexys4 board owners that are interested booting their devices from the on-board QSPI flash memory chip (rather than a bitstream file on the sd-card). This will require flashing an .mcs file onto their board's QSPI chip, so as to provide an initial bitstream in the 'Slot 0' position.

These re-programmable chips are called Field Programmable Gate Arrays (FPGAs) or Complex Programmable Logic Devices (CPLDs), and can implement a wide variety of circuits. They are normally programmed using a language like VHDL or Verilog. These are languages that are not commonly encountered by most people. They are also quite different in some ways to "normal" programming languages, and it can take a while to understand how they work. But with some effort and perseverance, exciting things can be created with them.

## SUGGESTED PC SPECIFICATIONS

Be prepared to install many gigabytes of software on a Linux or Windows PC, before you will be able to write programs for the FPGAs and CPLDs in the MEGA65. Also, "compiling" complex designs can take up to several hours, depending on the speed and memory capacity of your computer. We recommend a computer with at least 12GB RAM (preferably 16GB) if you want to write programs for FPGAs and CPLDs. On the other hand, if all you want to do is load programs onto your MEGA65's FPGAs and CPLDs that other people have written, then most computers running a recent version of Windows or Linux should be able to cope.

- OS: Linux or Windows
- CPU Speed: As fast as you can get your hands on!
- Number of cores: Ideally, 8 or more, as the free license of Vivado can make use of a max of 8 cores.
- Hard disk space: Have about 70GB or more. The exact amount used depends on how many components within Vivado you install (bear in mind that the full install file is about 50GB in itself)
- Memory: minimum of 12GB (ideally, have more, to play it safe)

# WARNING

Before we go any further, we do have to provide a warning about reprogramming the FPGAs and CPLDs in the MEGA65. Re-programming the MEGA65 FPGA can potentially cause damage, or leave your MEGA65 in an unresponsive state from which it is very difficult to recover, i.e., "bricked". Therefore if you choose to open your MEGA65 and re-program any of the FPGAs it contains, it is no longer possible to guarantee its correct operation. Therefore, we cannot reasonably honour the warranty of the device as a computer. You have been warned!

## INSTALLING VIVADO

Installation of Vivado is required to flash the QSPI flash memory within your MEGA65 target device, whether it be a MEGA65 R2/R3, Nexys4/Nexys4DDr/NexysA7, MEGAphone or other.

Vivado is also the tool used to perform compilation (synthesis, as it is preferably called) of FPGA bitstreams.

To get started, connect to <https://www.xilinx.com/support/download.html>

Select 2020.2 version

The screenshot shows the Xilinx website for downloading the Vivado Design Suite. The top navigation bar includes 'Solutions', 'Products', and 'Support'. The Xilinx logo is on the right. A sidebar on the left lists versions: 2020.3 (selected), 2020.2 (highlighted with a yellow box), 2020.1, 2019.2, Vivado Archive, ISE Archive, and CAE Vendor Libraries Archive. The main content area is titled 'Vivado Design Suite - HLx Editions - 2020.2 Full Product Installation'. It features an 'Important Information' section with a bulleted list of new features: Public access support for the Xilinx® Versal™ Platforms, Petalinux now a part of Xilinx Unified Installer, Access Block Design container now to create team-based designs, Abstract Shell for Dynamic Function eXchange, 2020.2 Introduces Vitis™ HLS for Vivado flows, and Add-on for MATLAB® and Simulink® (Unified Model Composer and System Generator). To the right of this section are columns for 'Download Includes', 'Download Type', 'Last Updated', 'Answers', 'Documentation', and 'Support Forums'. At the bottom, a note says 'We strongly recommend to use the web installers as it reduces download time and saves significant disk space.'

NOTE : Some users still have success with using older versions, as the main aim here is to install a version that supports the FPGA of your target hardware.

i.e., the Artix7 100T (for Nexys and R2) or 200T (R3).

Click on Xilinx Unified Installer 2020.2: Windows Self Extracting Web Installer EXE - 248.44 MB



You will be asked to create an account in order to sign in and be able to download the installation program.

Your credentials will also be requested when doing the installation.



After having signed in, you have to provide some personal information and then click on Download



The screenshot shows a registration form for Xilinx. It includes fields for Business E-mail, Company Name, Address 1, Address 2, Location, City, State/Province, Postal Code, Phone, and Job Function. A note at the bottom states: "For more information about how we process your personal information, please see our [privacy policy](#)". A red "Download" button is at the bottom left.

Business E-mail\*

Company Name\*

NoCompany

Please enter the name of your business or institution.

Address 1\*

Please enter your Company Address.

Address 2

Location\*

State/Province

City\*

Postal Code

Phone

Job Function\*

Hardware Enthusiast/Pro-Hobbyist

For more information about how we process your personal information, please see our [privacy policy](#).

Download

Execute the installer as Administrator (Xilinx\_Unified\_2020.2\_1118\_1232\_Win64.exe).



Click on Allow Access.



Click on Next.



Enter your credentials and click on Next.



Select Vivado and click on Next.



Select "Vivado HL WebPACK" and click on "Next"



We'd suggest selecting only the "**7 Series**" devices, as our chosen FPGA is within this series, and de-selecting the other series will save you about 6GB in download size. Then click on "Next"



Warning: As stated, disconnect any USB cable that would be connected to your PC from the Nexys board.

Agree with all the End User Licence Agreement and Terms and conditions and click on "Next".



Choose the location where you want to install the software and click on "Next".



Warning : You are about to download 20GB of software and you need 70GB to perform the installation.

Click on "Yes"



Click on "Install"



Wait for the installation to complete. At the very end of the installation you will be asked if you want to install Xilinx device software.



Click on "Install"



Let the installation complete.

The installation is completed. Click on "OK"



You end up with the following icons on your desktop:



Launch Vivado 2020.2



Click on "Help" -> "Obtain a licence Key"



This launches the Vivado licence manager

Select "Get Free ISE WebPACK, ISE/Vivado IP or PetaLinux Licenses"

Click on "Connect Now"



Connect with the user account you have created to be able to download the Vivado software. If you were not already connected to Xilinx website, this will take you to the main webpage. Go back in the licence manager (which is not closed)



Click again on "Connect Now" (ensure "Get Free ISE WebPACK, ISE/Vivado IP or PetaLinux Licenses" is still selected)



You then register your personal information on the Vivado website:

Product Licensing - Name and Address Verification

U.S. Government Export Approval

- U.S. export regulations require that your First Name, Last Name, Company Name and Shipping Address be verified before Xilinx can fulfill your download request. [Please provide accurate and complete information.](#)
- Addresses with Post Office Boxes and names/addresses with Non-Roman Characters with accents such as grave, tilde or colon [are not supported](#) by US export compliance systems.

First Name\* [Redacted]  
Last Name\* [Redacted]

Business Email\* [Redacted]

Company Name\* [Redacted]  
Please enter the name of your business or institution.

Address 1\* [Redacted]  
Please enter your Company Address.

Address 2  
[Redacted]

Location\* [Redacted] State/Province [Redacted]

City\* [Redacted] Postal Code [Redacted]

Phone [Redacted]

Job Function\* [Redacted]  
Hardware Enthusiast/Pro-Hobbyist

For more information about how we process your personal information, please see our [privacy policy](#).

**Next**

Click on "Next".

Select "ISE WebPACK Licence" and "Vivado Design Suite: HL WebPACK 2015 and Earlier License"

Then click on "Generate Node-Locked Licence"

#### Create a New License File

Create a new license file by making your product selections from the table below. [?](#)

#### Certificate Based Licenses

| Product                                                                                      | Type                     | License | Available Seats | Status  | Subscription End Date |
|----------------------------------------------------------------------------------------------|--------------------------|---------|-----------------|---------|-----------------------|
| Xilinx add-on for Matlab and Simulink, 90 Day Evaluation                                     | Certificate - Evaluation | Node    | 1/1             | Current | 90 days               |
| ISE Embedded Edition License                                                                 | Certificate - No Charge  | Node    | 1/1             | Current | None                  |
| Vivado Design Suite: 30-Day Evaluation License                                               | Certificate - Evaluation | Node    | 1/1             | Current | 30 days               |
| SDSoC Environment, 60 Day Evaluation License                                                 | Certificate - Evaluation | Node    | 1/1             | Current | 60 days               |
| SDAccel OpenCL Development Environment: 30 Day Node Locked Evaluation Lice...                | Certificate - Evaluation | Node    | 1/1             | Current | 30 days               |
| <input checked="" type="checkbox"/> Vivado Design Suite: HL WebPACK 2015 and Earlier License | Certificate - No Charge  | Node    | 1/1             | Current | None                  |
| <input checked="" type="checkbox"/> ISE WebPACK License                                      | Certificate - No Charge  | Node    | 1/1             | Current | None                  |
| Xilinx MicroBlaze/All Programmable SoC Software Development Kit – Standalone                 | Certificate - No Charge  | Node    | 1/1             | Current | None                  |
| PetaLinux Tools License                                                                      | Certificate - Evaluation | Node    | 1/1             | Current | 365 days              |
| Vivado HLS Evaluation License                                                                | Certificate - Evaluation | Node    | 1/1             | Current | 30 days               |

**Generate Node-Locked License**

Click on "Next"

Generate Node License  
Fields marked with an asterisk \* are required.

**1 PRODUCT SELECTION**

| Product Selections | Product             | Type      | Available Seats | Subscription End Date | Requested Seats | Borrowed Seats |
|--------------------|---------------------|-----------|-----------------|-----------------------|-----------------|----------------|
| *                  | ISE WebPACK License | No Charge | 1/1             | None                  | 1               |                |

**2 SYSTEM INFORMATION**

|           |      |
|-----------|------|
| License   | Node |
| Host ID * | Any  |

**3 COMMENTS**

|            |  |  |
|------------|--|--|
| Comments ? |  |  |
|------------|--|--|

**Next** **Cancel**

Click on "Next"

Generate Node License

**4 REVIEW LICENSE REQUEST**

| Product Selections | Product             | Subscription End Date | Available Seats | Requested Seats |
|--------------------|---------------------|-----------------------|-----------------|-----------------|
|                    | ISE WebPACK License | 1/1                   | 1               |                 |

System Information

|         |      |
|---------|------|
| License | Node |
| Host ID | ANY  |

Note: WebTalk is always enabled for WebPACK users. WebTalk ignores user and install preference when a bitstream is generated using the WebPACK license. If a design is using a device contained in WebPACK and a WebPACK license is available, the WebPACK license will always be used. To get additional information on WebTalk, go to [www.xilinx.com/webtalk](http://www.xilinx.com/webtalk).

**Previous** **Next** **Cancel**

Check your email box : You should have received an email from Xilinx, Inc. with a licence file attached and named "Xilinc.lic".

Retrieve this file on your PC and keep it in safe place.

Congratulations

Your new license file has been successfully generated and e-mailed to [REDACTED] You can also view the license file under the Manage Licenses tab.

Please add this sender ([xilinx.notification@entitlenow.com](mailto:xilinx.notification@entitlenow.com)) to your address book.

#### License File Details

Node License

Host ID: ANY

#### Products

ISE WebPACK License (No Charge): 1 seats

Go back to the licence manager (which is still running).

Set "Load License" and click on "Copy License"



Browse to the location where you saved "Xilinx.lic" file, select it and click on "Open".



Click on "OK" and close the Vivado licence manager.



Your Vivado software is registered and you can now use it.

## INSTALLING THE FTDI DRIVERS

The FTDI drivers are needed in order for your PC to communicate with the hardware's JTAG port and serial comms port (note that the single physical USB connection made to your PC actually provides these two ports).

# Linux drivers

Some Linux users have reported that they have found the FTDI drivers to be installed within their Linux distributions out-of-the-box, while others have found they needed to run this extra command after installing Vivado:

```
cd /opt/Xilinx/Vivado/2018.3/data/xicom/cable_drivers/lin64/install_script/install_drivers  
sudo ./install_drivers
```

# Windows drivers

Download the following archive to install the drivers:

- [https://www.ftdichip.com/Drivers/CDM/CDM21228\\_Setup.zip](https://www.ftdichip.com/Drivers/CDM/CDM21228_Setup.zip)

Unzip the file **CDM21228\_Setup.zip**, you get the file **CDM21228\_Setup.exe**.

#### Warning:

Before installing the drivers, it is imperative to switch off the Nexys4 board and to disconnect the USB cable from the PC.

Review the devices already installed before the installation:



Run the file CDM21228\_Setup.exe as administrator:



Confirm that you want to run the program.

Click on "Extract".



Click on "Next >"



Accept the agreement and click on "Next >".



The installation of the drivers starts.

Click on "Finish".



Connect the USB cable to a USB port on the PC **without turning on the Nexys4 board**.

Connecting the USB cable triggers the appearance of new devices.

The screenshot compares two device lists: "BEFORE DRIVERS INSTALLATION" and "AFTER DRIVERS INSTALLATION". Both lists are for a system with its power switch off and the USB cable unplugged from the PC's USB port. The "AFTER" list shows additional entries, specifically new COM ports, which are highlighted with yellow boxes and red outlines.

**BEFORE DRIVERS INSTALLATION (Nexys power switch off and USB cable unplugged from PC USB port)**

- > Mice and other pointing devices
- > Monitors
- > Network adapters
  - Ports (COM & LPT)
    - Intel(R) Active Management Technology - SOL (COM3)
    - Port imprimante ECP (LPT1)
  - Print queues
- > Processors
- > Proximity devices
- > Smart card readers
- > Software devices
- > Sound, video and game controllers
  - AMD High Definition Audio Device
  - Realtek Audio
  - Son Intel(R) pour écrans
- > Storage controllers
- > System devices
  - Universal Serial Bus controllers
    - Intel(R) USB 3.0 eXtensible Host Controller - 1.0 (Microsoft)
    - USB Composite Device
    - USB Composite Device
    - USB Root Hub (USB 3.0)

**AFTER DRIVERS INSTALLATION (Nexys power switch off and USB cable plugged in PC USB port)**

- > Mice and other pointing devices
- > Monitors
- > Network adapters
- > Ports (COM & LPT)
  - Intel(R) Active Management Technology - SOL (COM3)
  - Port imprimante ECP (LPT1)
  - USB Serial Port (COM5)
- > Print queues
- > Processors
- > Proximity devices
- > Smart card readers
- > Software devices
- > Sound, video and game controllers
  - AMD High Definition Audio Device
  - Realtek Audio
  - Son Intel(R) pour écrans
- > Storage controllers
- > System devices
  - Universal Serial Bus controllers
    - Intel(R) USB 3.0 eXtensible Host Controller - 1.0 (Microsoft)
    - USB Composite Device
    - USB Composite Device
    - USB Root Hub (USB 3.0)
    - USB Composite Device
    - USB Serial Converter A
    - USB Serial Converter B

- An additional COM port has been installed: **This is the COM port that will be used to communicate with the Nexys4 board.**

- An additional USB composite device has been installed.
- Two USB serial converter devices have been installed.

At this point the Nexys4 board has still not been powered up.

For more information about the installed drivers, you can download the corresponding documentation:

- [https://ftdichip.com/wp-content/uploads/2020/08/AN\\_396-FTDI-Drivers-Installation-Guide-for-Windows-10.pdf](https://ftdichip.com/wp-content/uploads/2020/08/AN_396-FTDI-Drivers-Installation-Guide-for-Windows-10.pdf)
- [https://ftdichip.com/wp-content/uploads/2021/01/AN\\_119\\_FTDI\\_Drivers\\_Installation\\_Guide\\_for\\_Windows7.pdf](https://ftdichip.com/wp-content/uploads/2021/01/AN_119_FTDI_Drivers_Installation_Guide_for_Windows7.pdf)

## FLASHING THE MAIN FPGA USING VIVADO

Firstly, to clarify that when we say 'flashing the FPGA', in reality, what we mean is that we are flashing the QSPI flash memory chip that the FPGA makes use of upon startup in order to quickly load the bitstream from.

The diagram below shows two common pathways that the FPGA can load bitstreams at startup:



- We can first flash a bitstream/core-file onto the QSPI flash memory chip, and the FPGA can load this quickly at power-up. Flashing the QSPI is quite slow, but the reward of a fast boot-up time is an advantage.
- We can drop a bitstream file onto our sd-card and let the FPGA load it (somewhat more slowly) from there at power-up. This way is popular amongst Nexys4 board users, and allows them to swap/upgrade bitstreams quickly.

In this section, we describe the pathway that makes use of the QSPI.

Many of the following steps in this section are applicable not only to MEGA65 R2/R3 owners, but Nexys4 board owners too. There are a few points of distinction along the way that readers will be made aware of.

If you choose to proceed, you will need a functioning installation of Xilinx's Vivado software, and the FTDI drivers installed, as described in the earlier sections.

You will also need to download or build an .mcs bitstream file (and optional .prm checksum verification file) that you intend flash onto the QSPI chip via Vivado. See Bitstream files for more details on where such files can be downloaded.

For MEGA65 R2/R3 owners:

You will need a TE0790-03 JTAG programming module. It is also necessary to have dip-switches 1 and 3 in the ON position and dip-switches 2 and 4 in the OFF position on the TE-0790. With your MEGA65 disconnected from the power, the TE-0790 must be installed on the JB1 connector which is located between the floppy data cable and the audio jack. The gold-plated hole of the TE-0790 must line up with the screw hole below. The mini-USB cable will then connect on the side towards the 3.5" floppy drive. The following image shows the correct position: The TE0790 is surrounded by the yellow box, and the dip-switches by the red box. Dip-switch 1 is the one nearest the floppy data cable.



For Nexys4 board owners:

Simply connect your micro-usb cable between your Nexys4 board and your PC via the port labeled 'PROG UART' (J6), as shown:



Also, set J1 jumper to the QSPI position:



- Connect your non-8-bit computer to the FPGA programming device using the appropriate USB cable.
- Switch the MEGA65 computer ON.
- Open Vivado.

Step 1a: Create a new Vivado project with "File", "Project", "New...".

NOTE: On future occasions that you need to flash the QSPI, just re-open this project (no need to create a new project each time).



Step 1b: The 'New Project' wizard appears. Click on "Next":



Step 1c: Name your project and choose the location you like, then click on "Next":



Step 1d: Keep the default selected options and click on "Next":



Step 1e: Do not add any sources, keep the default selected options and click on "Next":



Step 1f: Keep the default selected options and click on "Next":



## Step 1g: Click on "Finish":



## Step 2: In the left column, select "Open Hardware Manager" at the very bottom.



### Step 3: Connect to the FPGA:

Under "Open Hardware Manager", choose "Open Target", then "Auto Connect".



Step 4: Wait a moment, "Connecting to server..." should automatically close without dropping an error to the console.



Step 5: Under "Open Hardware Manager", choose "Add Configuration Memory Device", then:

- For MEGA65R3: "xc7a200t\_0"
- For Nexys4 and MEGA65R2: "xc7a100t\_0".



## Step 6a: Select Memory Part:

In the newly opened dialogue:

- For MEGA65R2/R3: type "S25fl256s" (without quotes), then select "s25fl256xxxxxxxxx0-spi-x1\_x2\_x4" (the upper one) and click "OK".
- For Nexys4: type "S25fl128s" (without quotes), then select "s25fl128xxxxxxxxx0-spi-x1\_x2\_x4" (the upper one) and click "OK".



Step 6b: Click on "OK" to confirm you want to program the configuration memory device now.



Step 6c: If you do not see such a popup, or wish to reprogram the QSPI on a future occasion, in "Hardware" window, right click on the memory configuration and select "Program Configuration Memory Device":



## Step 7: Set programming options:

In the next dialogue, set the "Configuration file" to the path of your ".mcs" bitstream file. You can also optionally set the "PRM file" field to the path of your ".prm" file. Leave all other parameters as they are (see screenshot below).



Step 8: Patiently wait for the programming to finish. This can take several minutes as the Vivado software erases and then reprograms the flash memory that is used to initialise the FPGA on power-up.



Step 9: If your screen looks like the screenshot below, your new bitstream has been successfully flashed into Slot0 of your QSPI flash memory!



Step 10: If you want to reflash the FPGA, you might find the "Add Configuration Memory Device" option in step 5 greyed out. Instead, select "s25fl256xxxxxxxx0-spi-x1\_x2\_x4" in the "Hardware" window, press right mouse button and select "Program Configuration Memory Device" to flash.



## FLASHING THE CPLD IN THE MEGA65'S KEYBOARD WITH LATTICE DIAMOND

If you choose to proceed, you will need a TE0790-03 JTAG programming module and a functioning installation of Lattice Diamond Programmer software. This can be done on either Windows or Linux, but in both cases you will need to install any necessary USB drivers. It is also necessary to have dip-switches 1 and 3 in the ON position and dip-switches 2 and 4 in the OFF position on the TE-0790. With your MEGA65 disconnected from the power, the TE-0790 must be installed on the JB1 connector, which is located between the floppy data cable and the audio jack. The gold-plated hole of the TE-0790 must line up with the screw hole below. The mini-USB cable will then connect on the side towards the 3.5" floppy drive. The following image shows the correct position: The TE0790 is surrounded by the yellow box, and the dip-switches by the red box. Dip-switch 1 is the one nearest the floppy data cable.



On the PCB R2 MEGA65 mainboard, dip switch 1 (the one nearest to the user sitting in front of the machine) must be in the ON position. The other switches must be OFF. The keyboard will go into "ambulance mode" (blue flashing lights) when set correctly.

Connect your non-8-bit computer to the FPGA programming device using a mini-USB cable. Switch the MEGA65 computer ON. Open the Diamond Programmer which can be downloaded from the Internet.

**Step 1: Open DIAMOND PROGRAMMER:**

Select "Create a new project from a JTAG scan". If entry under "Cable:" is empty, click "Detect Cable".



**Step 2: Create a new project:**

If dialog "Programmer: Multiple Cables Detected" appears, select the first entry ("Location 0000") and click "OK".



### Step 3: Select cable:

You have now created a new project which should display "MachXO2" under "Device Family" and "LCMXO2-1200HC" under "Device"



### Step 4: New Diamond Programmer project:

Choose "File" then "Open File" to load the Diamond Pprogrammer project with the MEGA65 keyboard firmware update.



## Step 5: Open project:

Navigate into the folder with the extracted MEGA65 keyboard firmware files you have received and select the file ending with ".xcf".



## Step 6: Select project file:

Click the three dots under "File Name" to set the correct path and find the file ending with ".jed".



Step 7: Choose correct path of .jed file:  
Select the file ending with ".jed" and click "OK".



Step 8: Select .jed file:

Click on the icon with the green arrow facing down "PROGRAM", which looks similar to the Diamond Programmer program icon.



## Step 9: Select cable:

After a moment the Output window should display "INFO - Operation: successful." and the "Status" cell should go green (does not always happen).



## Step 10: Operation successful:

You have now successfully flashed the MEGA65 keyboard. If you wish you can now save the project for later use.



# FLASHING THE MAX10 FPGA ON THE MEGA65'S MAINBOARD WITH INTEL QUARTUS

If you choose to proceed, you will need a TEI0004 - Arrow USB Programmer2 module with TEI0004 driver installed and a functioning installation of Quartus Prime Programmer Lite Edition. This can be done on either Windows or Linux, but in both cases you will need to install any necessary USB drivers. With your MEGA65 disconnected from the power, the TEI0004 must be installed on the J17 connector, which is located between the floppy data cable and the ARTIX 7 FPGA on the Mainboard. The micro-USB port of the TEI0004 must face in the opposite direction of the HDMI and LAN sockets, towards the trap door. The following image shows the correct position.

On the PCB R2 MEGA65 mainboard, all dip switches must be in the OFF position. The main FPGA of the MEGA65 R2 must not contain a valid bitstream. See section Flashing the main FPGA using Vivado on how to erase the bitstream from the main FPGA.



Connect your non-8-bit computer to the FPGA programming device using a micro-USB cable. Open Quartus Prime Programmer Lite Edition, which can be downloaded from the Internet.

**Step 1: Open Quartus Prime Programmer Lite Edition:**

Click the "Hardware Setup" button in the top left corner of the Quartus Prime Programmer window.



**Step 2: Enter Hardware Setup:**

In the newly appeared window under "Currently selected hardware" choose "Arrow-USB-Blaster". If "Arrow-USB-Blaster" does not appear, verify cable and drivers being correctly installed.



**Step 3: Select Arrow USB-Blaster:**

Click the "Add File" button from the left row and choose the latest ".pof" file. Then click "Open".



## Step 4: Select Programming File:

Tick at least the three boxes under "Program/Configure". Also enabling all boxes under "Verify" and "Blank-Check" will make the process more reliable.



## Step 5: Select Program/Configure Options:



While keeping the Reset-Button pressed, switch the MEGA65 computer ON. The keyboard will go into “ambulance mode” (blue flashing lights). If it does not, the main FPGA is not empty – restart the whole process.

Now click on “Start” in the left row of buttons. The progress bar in the top right corner should quickly go to 100 percent and turn green. You have now successfully updated your MAX10 FPGA.

If you receive an error message instead, make sure the main FPGA bitstream has been erased and that you did not release the reset-button on the MEGA65 beforehand. Switch off the MEGA65 and restart this step.

#### Step 6: Programming successful:





# T

## APPENDIX

### Trouble shooting

- Vivado



# VIVADO

## RAM requirements

```
INFO: [Synth 8-256] done synthesizing module 'ram32x1024' [/home/....]
INFO: [Synth 8-256] synthesizing module 'charrom' [/home/....]
/opt/Xilinx/Vivado/2018.2/bin/loader: line 280: 2317 killed
WARNING: [Vivado 12-8222] Failed run(s) : 'synth#1'
ERROR: Application Exception: failed to launch run 'impl#1' due to failures in the following run(s):
synth#1
These failed run(s) need to be reset prior to launching 'impl#1' again.
```

This error is due to vivado crashing because the machine doesn't have enough RAM for vivado to run. Vivado requires at least 4GB to synthesise the MEGA65 target, but 8GB is better.



# U

## APPENDIX

### Model Specific Features

- Detecting MEGA65 Models
- MEGA65 Desktop Computer, Revision 3 onwards
- MEGA65 Desktop Computer, Revision 2
- MEGAphone Handheld, Revisions 1 and 2
- Nexys4 DDR FPGA Board



# DETECTING MEGA65 MODELS

While we expect the production version of the MEGA65 to be a stable platform, there may still be cases where detecting which hardware your programme is running on. This is particularly important for the MEGA65 system software, which may need to initialise different pieces of hardware on the different models. Also, because there is a hand-held version of the MEGA65 already in development, which uses a slightly different resolution screen (800x480 instead of 720x576), and has a touch screen but no hardware keyboard, you may wish to make programmes that adapt to the hand-held devices in a more graceful way. For example, you may enable touch-screen input, and restructure on-screen selections to be large enough to be easily activated by a finger.

The simple way to detect which model of MEGA65 your programme is running on, is to check the \$D629 register (but don't forget to enable the MEGA65 IO personality first, via \$D02F). This contains an 8-bit hardware identifier. The following values are currently defined:

**\$01 (1)** MEGA65 R1

**\$02 (2)** MEGA65 R2

**\$03 (3)** MEGA65 R3

**\$21 (33)** MEGAphone (hand-held) R1

**\$40 (64)** Nexys4 PSRAM

**\$41 (65)** Nexys4DDR

**\$42 (66)** Nexys4DDR with widget board

**\$FD (253)** QMTECH Wukong A100T board

**\$FE (254)** Simulation run of VHDL

# MEGA65 DESKTOP COMPUTER, REVISION 3 ONWARDS

The R3 desktop PCB is very similar to the R2 desktop PCB, with two key changes:

- First, the R3 PCB does not have an ADV7511 digital video driver chip, and so the I2C register block for that device is not present.
- Second, the R3 PCB uses a different on-board amplifier for the PC speakers, which are now present in stereo, rather than mono as on the R2 PCB. The amplifier on the

R3 PCB is the same as on the MEGAphone R1 – R2 PCBs. However, the I2C registers are at a different address. On the MEGA65 R3 PCB, the registers are located at \$FFD71DC – \$FFD71EF.

## MEGA65 DESKTOP COMPUTER, REVISION 2

The desktop version of the MEGA65 contains a Real-Time Clock (RTC), which also includes a small amount of non-volatile memory (NVRAM) that retains its value, even if the computer is turned off and disconnected from its power supply. The NVRAM will hold its values for as long as the internal battery has sufficient charge. This battery also powers the Real-Time Clock (RTC) itself, which includes a 100 year calendar spanning the years 2000 – 2099.

The main trick with accessing the RTC from BASIC, is that we will need to use a MEGA65 Enhanced DMA operation to fetch the RTC registers, because the RTC registers sit above the 1MB barrier, which is the limit of the C65's normal DMA operations. The easiest way to do this is to construct a little DMA list in memory somewhere, and make an assembly language routine that uses it. Something like this (using BASIC 65 in C65 mode):

```
10 RESTORE 110:FORT=0T043:READA$:POKE1024+I,DEC(A$):NEXT:BANK 128:SYS1042
20 S=PEEK(1056):M=PEEK(1057):H=PEEK(1058)
30 D=PEEK(1059):MM=PEEK(1060):Y=PEEK(1061)+DEC("2000")
40 IF H AND 128 GOTO 80
50 PRINT "THE TIME IS ";RIGHT$(HEX$(H AND 63),2);":",RIGHT$(HEX$(M),2);".",RIGHT$(HEX$(S),2)
60 IF H AND 32 THEN PRINT "PM": ELSE PRINT "AM"
70 GOTO 90
80 PRINT "THE TIME IS ";RIGHT$(HEX$(H AND 63),1);":",RIGHT$(HEX$(M),2);".",RIGHT$(HEX$(S),2)
90 PRINT "THE DATE IS ";RIGHT$(HEX$(D),2);".",RIGHT$(HEX$(MM),2);".",HEX$(Y)
100 END
110 DATA 0B,00,FF,81,00,00,00,00,00,10,71,00,20,04,00,00,00
120 DATA A9,47,B0,2F,D0,A9,53,B0,2F,D0,A9,00,B0,02,D7,A9
130 DATA 04,B0,01,D7,A9,00,B0,05,D7,60
```

This program works by setting up a DMA list in memory at 1,024 (hex \$0400) (unused normally on the C65), followed by a routine at 1,042 (hex \$0412) which ensures we have MEGA65 registers un-hidden, and then sets the DMA controller registers appropriately to trigger the DMA job, and then returns. The rest of the BASIC code PEEKs out the RTC registers that the DMA job copied to 1,024 – 1,032 (hex \$0400 – \$0407), and interprets them appropriately to print the time.

The curious can use the MONITOR command, and then D1012 to see the routine.

If you want a running clock, you could replace line 100 with GOTO 10. Doing that, you will get a result something like the following:

```
THE TIME IS 10:05:36 PM  
THE DATE IS 20.02.2020  
THE TIME IS 10:05:36 PM  
THE DATE IS 20.02.2020  
THE TIME IS 10:05:36 PM  
THE DATE IS 20.02.2020  
THE TIME IS 10:05:36 PM  
THE DATE IS 20.02.2020  
...  
...
```

If you first POKE0,65 to set the CPU to full speed, the whole program can run many times per second. There is an occasional glitch, if the RTC registers are read while being updated by the machine, so we really should de-bounce the values by reading the time a couple of times in succession, and if the values aren't the same both times, then repeat the process until they are. This is left as an exercise for the reader.

NOTE: These registers are not yet fully documented.

## MEGAPHONE HANDHELD, REVISIONS 1 AND 2

The MEGAphone revision 1 and 2 contain a Real-Time Clock (RTC), however this RTC does not include a non-volatile memory (NVRAM) area. Other specific features of the MEGAphone revisions 1 and 2 include a 3-axis accelerometer, including analog to digital converters (ADCs), amplifier controller for loud speakers, and several I2C IO expanders, that are used to connect the joy-pad and other peripherals. The IO expanders are fully integrated into the MEGAphone design, and thus there should be no normal need to read these registers directly. The IO expanders are, however, also responsible for power control of the various sub-systems of the MEGAphone.

NOTE: These registers are not yet fully documented.

# NEXYS4 DDR FPGA BOARD

NOTE: These registers are not yet fully documented.

# V

## APPENDIX

### Schematics

- **MEGA65 R3 Schematics**
- **MEGA65 R2 Schematics**



# MEGA65 R3 SCHEMATICS











MEGA65

Rev. 03

Page 5 of 23

3





Title: MEGA65

A4 Number: TE0765  
[No Variations]  
date: 2020-06-29 Copyright: Tenz Electronic GmbH  
filename: B16.SchDoc

Rev. 03  
Page 7 of 23





---

le: MEGA65

Rev. 03  
Page 8 of 23



|                           |                                   |                                                  |
|---------------------------|-----------------------------------|--------------------------------------------------|
| Title: MEGA65             |                                   | Rev. 03                                          |
| A4                        | Number: TE0765<br>[No Variations] | Date: 2020-06-29 Copyright: Trez Electronic GmbH |
|                           |                                   | Page 9 of 24                                     |
| Filename: FPGA-CFG.SchDoc |                                   |                                                  |

|               |            |
|---------------|------------|
| Title:        | MEGA65     |
| Number:       | A4         |
| Date:         | 2020-06-22 |
| Filename: FPG |            |



MEGA65



|           |                                      |            |                       |
|-----------|--------------------------------------|------------|-----------------------|
| Title:    | MEGA65                               |            |                       |
| A4        | Number:<br>TE0765<br>[No Variations] | Rev.<br>03 |                       |
| Date:     | 2020-06-29                           | Copyright: | Trenz Electronic GmbH |
| Filename: | FPGA-MGT-SchDoc                      |            |                       |
|           | Page 10 of 23                        |            |                       |









1

2

3

4

A

A

B

B

C

C

1

2

3

4



Title: MEGA65  
A4 Number: TE0765  
[No Variations]  
Date: 2020-06-29 Rev. 03  
Filename: Floppy.SchDoc



4

2

1





Title: MEGA65

|                       |                                   |               |
|-----------------------|-----------------------------------|---------------|
| A4                    | Number: TE0765<br>[No Variations] | Rev. 03       |
| Date: 2020-06-29      | Copyright: Trenz Electronic GmbH  | Page 17 of 23 |
| Filename: HDMI_SchDoc |                                   |               |





lei: MEGA65

Rev. 03  
Page 18 of 23  
Documentname: Ethernet SchDoc



4

3

2

1

A



B



C



D

2

1

3

4







Title: MEGA65

A4 Number: TE0765  
[No Variations]

Rev. 03  
Date: 2020-06-29 Copyright: Trenz Electronic GmbH

Page 22 of 23  
Filename: SOUND.SchDoc



| A                | 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 2  | 3 | 4 |
|------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----|---|---|
| B                |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |    |   |   |
| C                |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |    |   |   |
| D                |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |    |   |   |
| REVISION HISTORY |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |    |   |   |
| REV              | Description                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    | 1G |   |   |
| -03              | <p>1. Added a VRP resistor on bank 63;</p> <p>2. LDO U133 is changed on ADP7102ACPZ;</p> <p>3. Signal FPGA Q00 is connected on AE18 pin of FPGAs;</p> <p>4. Signal DIG, LED3 is connected on AD18 pin of FPGAs;</p> <p>5. Signal MIO13_25 connected to I1 pin 33 instead MIO25;</p> <p>6. Resistor R84 is removed;</p> <p>7. LED D1 moved on edge of PCB;</p> <p>8. Added TH1 resistors J4 on CPLD_ITAGEN, R76 was removed;</p> <p>9. Signals B9_Y, X are renamed in B88_XX_X;</p> <p>10. C241 is changed on inf;</p> <p>11. Length of CLK signals on RFADC and RFDAC are adjusted;</p> <p>12. Wrong connection R83 on back of PCB;</p> <p>13. Wrong connection PGOOD1 pin of U7 is fixed;</p> |    |   |   |
| A                | 1                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              | 2  | 3 | 4 |



Title: MEGA65  
 A4 Number: TE0835 [No Variations]  
 Date: 2020-06-29 Rev. 03  
 Copyright: trenz Electronic GmbH  
 Page 23 of 23  
 Filename: Revision\_Changes\_SchDoc



# MEGA65 R2 SCHEMATICS





PMOD Conn



Grove Conn



CPLD CONFIG



CPLD JTAG



TE JTAG



MEGA65



Rev. 02

Page 2 of 23

Page 1 of 1

Page 2 of 1

Page 3 of 1

Page 4 of 1

Page 5 of 1

Page 6 of 1

Page 7 of 1

Page 8 of 1

Page 9 of 1

Page 10 of 1

Page 11 of 1

Page 12 of 1

Page 13 of 1

Page 14 of 1

Page 15 of 1

Page 16 of 1

Page 17 of 1

Page 18 of 1

Page 19 of 1

Page 20 of 1

Page 21 of 1

Page 22 of 1

Page 23 of 1

Page 24 of 1







|                  |                                 |         |       |
|------------------|---------------------------------|---------|-------|
| Title:           | MEGA65                          |         |       |
| A4               | Number: TE0765<br>Default       | Rev. 02 |       |
| Date: 2019-03-11 | Copyright: Tenz Electronic GmbH | Page 5  | of 23 |
| Filename:        | B14.SchDoc                      |         |       |





|           |               |
|-----------|---------------|
| Title:    | <b>MEGA65</b> |
| A4        | Number:       |
| Date:     | 2019-03-11    |
| Filename: |               |

**trenz**  
electronic

Rev. 02  
Page 6 of 23

1      2      3      4



Title: MEGA65

Date: 2019-03-11

Filename: B16.SchDoc

Page 7 of 23

A4 Number: TE0765  
Detail

Rev. 02

Page 7 of 23

Title: trenz electronic

Date: 2019-03-11

Filename: B16.SchDoc

Page 7 of 23

1

2

3

4

















104

32

4

1

5

2

1

3

1

1



[View Details](#)

4

3

2

1

A



C



C



D

1

2

3

3

4



|                  |                                  |               |  |
|------------------|----------------------------------|---------------|--|
| Title:           | MEGA65                           |               |  |
| A4               | Number: TE0765<br>Default        | Rev. 02       |  |
| Date: 2019-03-11 | Copyright: Trez7 Electronic GmbH | Page 18 of 23 |  |
| Filename:        | VGA SchDoc                       |               |  |

The logo for tre-electr features the word "tre" in a bold, black, sans-serif font, positioned above the word "electr" in a smaller, italicized, black, sans-serif font. The letter "e" in both words is stylized with a horizontal bar extending from its top right, which then curves down and to the left to form the vertical stroke of the next letter. To the left of the text, there is a circular arrangement of approximately 12 blue, semi-transparent, rounded ovals of varying sizes, creating a sense of motion or a network.







3: MEGA 65

Number:

4

e: 2019-03-11



D





# W

## APPENDIX

### Supporters & Donors

- **Organisations**
- **Contributors**
- **Supporters**



The MEGA65 would not have been possible to create without the generous support of many organisations and individuals.

We are still compiling these lists, so apologies if we haven't included you yet. If you know anyone we have left out, please let us know, so that we can recognise the contribution of everyone who has made the MEGA65 possible, and into the great retro-computing project that it has become.

## ORGANISATIONS

- The MEGA Museum of Electronic Games & Art e.V. Germany **EVERYTHING**
- Trenz Electronik, Germany **MOTHERBOARD**
- Hintsteiner, Austria **CASE**
- GMK, Germany **KEYBOARD**

# CONTRIBUTORS

## **Andreas Liebeskind**

(*libi in paradize*)

CFO MEGA eV

## **Gurce Isikyildiz**

(*Gurce*)

Tools and enhancements

## **Russell Peake**

(*rdpeake*)

Bug Herding

## **Alexander Nik Petra**

(*n0d*)

Early Case Design

## **Ralph Egas**

(*0-limits*)

Business Advisor

## **Lucas Moss**

MEGAPhone PCB Design

## **Dr. Canan Hastik**

(*indica*)

Chairwoman MEGA eV

## **Simon Jameson**

(*Shallan*)

Platform Enhancements

## **Stephan Kleinert**

(*ubik*)

Destroyer of BASIC 10

## **Wayne Johnson**

(*sausage*)

Manual Additions

## **Lukas Kleiss**

(*LAK132*)

MegaWAT Presentation Software

# SUPPORTERS

@11110110100  
3c74ce64  
8-Bit Classics  
Aaron Smith  
Achim Mrotzek  
Adolf Nefischer  
Adrian Esdaile  
Adrien Guichard  
Ahmed Kablaoui  
Alan Bastian Witkowski  
Alan Field  
Alberto Mercuri  
Alexander Kaufmann  
Alexander Niedermeier  
Alexander Soppert  
Alfonso Ardire  
Amiga On The Lake  
Andre Kapp  
André Kudra  
André Simeit  
André Wösten  
Andrea Farolfi  
Andrea Minutello  
Andreas Behr  
Andreas Freier  
Andreas Grabski  
Andreas Millinger  
Andreas Nopper  
Andreas Wendel Manufaktur  
Andreas Zschunke  
Andrew Bingham  
Andrew Dixon  
Andrew Mondt  
Andrzej Hłuchyj  
Andrzej Sawiniec  
Andrzej Śliwa  
Anthony W. Leal  
Arkadiusz Bronowicki  
Arkadiusz Kwasny  
Arnaud Léandre  
Arne Drews  
Arne Neumann  
Arne Richard Tyarks  
Axel Klahr  
Balaz Ondrej  
Barry Thompson  
Benjamin Maas  
Bernard Alaiz  
Bernhard Zorn  
Bieno Marti-Braitmaier  
Bigby  
Bill LaGrue  
Bjoerg Stojalowski  
Björn Johannesson  
Bjørn Melbøe  
Bo Goeran Kvamme  
Boerge Noest  
Bolko Beutner  
Brett Hallen  
Brian Gajewski  
Brian Green  
Brian Juul Nielsen  
Brian Reiter  
Bryan Pope  
Burkhard Franke  
Byron Goodman  
Cameron Roberton (KONG)  
Carl Angervall  
Carl Danowski  
Carl Stock  
Carl Wall  
Carlo Pastore  
Carlos Silva  
Carsten Sørensen  
Cenk Miroglu Miroglu  
Chang sik Park  
Charles A. Hutchins Jr.  
Chris Guthrey  
Chris Hooper  
Chris Stringer  
Christian Boettcher  
Christian Eick  
Christian Gleinser  
Christian Gräfe  
Christian Heffner  
Christian Kersting  
Christian Streck  
Christian Wyk  
Christoph Haug  
Christoph Huck  
Christoph Pross  
Christopher Christopher  
Christopher Kalk  
Christopher Kohlert  
Christopher Nelson  
Christopher Taylor  
Christopher Whillock  
Claudio Piccinini  
Collen Blijenberg  
Constantine Lignos  
Crnjaninja  
Daniel Auger  
Daniel Julien  
Daniel Lobitz  
Daniel O'Connor  
Daniel Teicher  
Daniel Tootill  
Daniel Wedin

Daniele Gaetano Capursi  
Dariusz Szczesniak  
Darrell Westbury  
David Asenjo Raposo  
David Dillard  
David Gorgon  
David Norwood  
David Raulo  
David Ross  
de voughn accooe  
Dean Scully  
Dennis Jeschke  
Dennis Schaffers  
Dennis Schierholz  
Dennis Schneck  
denti  
Dick van Ginkel  
Diego Barzon  
Dierk Schneider  
Dietmar Krueger  
Dietmar Schinnerl  
Dirk Becker  
Dirk Wouters  
Domingo Fivoli  
DonChaos  
Donn Lasher  
Douglas Johnson  
Dr. Leopold Winter  
Dusan Sobotka  
Earl Woodman  
Ed Reilly  
Edoardo Auteri  
Eduardo Gallardo  
Eduardo Luis Arana  
Eirik Juliussen Olsen  
Emilio Monelli

EP Technical Services  
Epic Sound  
Erasmus Kuhlmann  
ergoGnomik  
Eric Hildebrandt  
Eric Hill  
Erwin Reichel  
Espen Skog  
Evangelos Mpouras  
Ewan Curtis  
Fabio Zanicotti  
Fabrizio Di Dio  
Fabrizio Lodi  
FARA Gießen GmbH  
FeralChild  
First Choice Auto's  
Florian Rienhardt  
Forum64. de  
Francesco Baldassarri  
Frank Fechner  
Frank Glaush  
Frank Gulasch  
Frank Haaland  
Frank Hempel  
Frank Koschel  
Frank Linhares  
Frank Wolf  
FranticFreddie  
Fredrik Ramsberg  
Fridun Nazaradeh  
Friedel Kropp  
Garrick West  
Gary Lake-Schaal  
Gary Pearson  
Gavin Jones  
Geir Sigmund Straume

Gerd Mitlaender  
Giampietro Albiero  
Giancarlo Valente  
Gianluca Girelli  
Giovanni Medina  
Glen Fraser  
Glen R Perye III  
Glenn Main  
Gordon Rimac  
GRANT BYERS  
Gregor Bubek  
Gregor Gramlich  
Guido Ling  
Guido von Gösseln  
Guillaume Serge  
Gunnar Hemmerling  
Günter Hummel  
Guy Simmons  
Guybrush Threepwood  
Hakan Blomqvist  
Hans Pronk  
Hans-Jörg Nett  
Harald Dosch  
Harri Salokorpi  
Harry Culpan  
Heath Gallimore  
Heinz Roesner  
Heinz Stampfli  
Helge Förster  
Hendrik Fensch  
Henning Harperath  
Henri Parfait  
Henrik Kühn  
Holger Burmester  
Holger Sturk  
Howard Knibbs

|                                |                            |                     |
|--------------------------------|----------------------------|---------------------|
| Hubert de Hollain              | Jens-Uwe Wessling          | Karsten Engstler    |
| Huberto Kusters                | Jesse DiSimone             | Karsten Westebbe    |
| Hugo Maria Gerardus v.d. Aa    | Johan Arneklev             | Keith McComb        |
| Humberto Castaneda             | Johan Svensson             | Kenneth Dyke        |
| Ian Cross                      | Johannes Fitz              | Kenneth Joensson    |
| IDE64 Staff                    | John Cook                  | Kevin Edwards       |
| Igor Ianov                     | John Deane                 | Kevin Thomasson     |
| Immo Beutler                   | John Nagi                  | Kim Jorgensen       |
| Ingo Keck                      | John Rorland               | Kim Rene Jensen     |
| Insanely Interested Publishing | John Sargeant              | Kimmo Hamalainen    |
| IT-Dienstleistungen Obsieger   | John Traeholt              | Konrad Buryto       |
| Ivan Elwood                    | Jon Sandelin               | Kosmas Einbrodt     |
| Jaap HUIJSMAN                  | Jonas Bernemann            | Kurt Klemm          |
| Jace Courville                 | Jonathan Prosise           | Lachlan Glaskin     |
| Jack Wattenhofer               | Joost Honig                | Large bits collider |
| Jakob Schönpflug               | Jordi Pakey-Rodriguez      | Lars Becker         |
| Jakub Tyszko                   | Jöre Weber                 | Lars Berntsson      |
| James Hart                     | Jörg Jungermann            | Lars Edelmann       |
| James McClanahan               | Jörg Schaeffer             | Lars Slivsgaard     |
| James Sutcliffe                | Jörg Weese                 | Lasse Lambrecht     |
| Jan Bitruff                    | Josef Hesse                | Lau Olivier         |
| Jan Hildebrandt                | Josef Soucek               | Lee Chatt           |
| Jan Iemhoff                    | Josef Stohwasser           | Loan Leray          |
| Jan Kösters                    | Joseph Gerth               | Lorenzo Quadri      |
| Jan Peter Borsje               | Jovan Crnjanin             | Lorenzo Travagli    |
| Jan Schulze                    | Juan Pablo Schisano        | Lorin Millsap       |
| Jan Stoltenberg-Lerche         | Juan S. Cardona Iguina     | Lothar James Foss   |
| Janne Tompuri                  | JudgeBeeb                  | Lothar Serra Mari   |
| Jannis Schulte                 | Juliusen Olsen             | Luca Papinutti      |
| Jari Loukasmäki                | Juna Luis Fernandez Garcia | Ludek Smetana       |
| Jarrod Adams                   | Jürgen Endras              | Lukas Burger        |
| Jason Smith                    | Jürgen Herm Stapelberg     | Lutz-Peter Buchholz |
| Javier Gonzalez Gonzalez       | Jyrki Laurila              | Luuk Spaetgens      |
| Jean-Paul Lauque               | Kai Pernau                 | Mad Web Skills      |
| Jeffrey van der Schilden       | Kalle Pöyhönen             | MaDCz               |
| Jens Schneider                 | Karl-Heinz Blum            | Magnus Wiklander    |

Maik Diekmann  
Malte Mundt  
Manfred Wittemann  
Manuel Beckmann  
Manzano Mérida  
Marc "3D-vice" Schmitt  
Marc Bartel  
Marc Jensen  
Marc Theunissen  
Marc Tutor  
Marc Wink  
Marcel Buchtmann  
Marcel Kante  
Marco Beckers  
Marco Cappellari  
Marco Rivela  
Marco van de Water  
Marcus Gerards  
Marcus Herbert  
Marcus Linkert  
Marek Pernicky  
Mario Esposito  
Mario Fetka  
Mario Teschke  
Mariusz Tymków  
Mark Adams  
Mark Green  
Mark Hucker  
Mark Leitiger  
Mark Spezzano  
Mark Watkin  
Marko Rizvic  
Markus Bieler  
Markus Bonet  
Markus Dauberschmidt  
Markus Fehr  
Markus Fuchs  
Markus Guenther-Hirn  
Markus Liukka  
Markus Merz  
Markus Roesgen  
Markus Uttenweiler  
Martin Bauhuber  
Martin Benke  
Martin Gendera  
Martin Groß  
Martin Gutenbrunner  
Martin Johansen  
Martin Marbach  
Martin Sonnleitner  
Martin Steffen  
Marvin Hardy  
Massimo Villani  
Mathias Dellacherie  
Mathieu Chouinard  
Matthew Adams  
Matthew Browne  
Matthew Carnevale  
Matthew Palmer  
Matthew Santos  
Matthias Barthel  
Matthias Dolenc  
Matthias Fischer  
Matthias Frey  
Matthias Grandis  
Matthias Guth  
Matthias Lampe  
Matthias Meier  
Matthias Mueller  
Matthias Nofer  
Matthias Schonder  
Maurice van Gils  
Max Ihlenfeldt  
Meeso Kim  
Michael Dailly  
Michael Dötsch  
Michael Dreßel  
Michael Fichtner  
Michael Fong  
Michael Geoffrey Stone  
Michael Gertner  
Michael Grün  
Michael Habel  
Michael Härtig  
Michael Haynes  
Michael J Burkett  
Michael Jensen  
Michael Jurisch  
Michael Kappelgaard  
Michael Kleinschmidt  
Michael Lorenz  
Michael Mayerhofer  
Michael Nursey  
Michael Rasmussen  
Michael Richmond  
Michael Sachse  
Michael Sarbak  
Michael Scholz  
Michael Timm  
Michael Traynor  
Michal Ursiny  
Michele Chiti  
Michele Perini  
Michele Porcu  
Miguel Angel Rodriguez Jodar  
Mikael Lund  
Mike Betz  
Mike Kastrantas

Mike Pikowski  
Mikko Hämäläinen  
Mikko Suontausta  
Mirko Roller  
Miroslav Karkus  
Morgan Antonsson  
Moritz  
Morten Nielsen  
MUBIQUO APPS,SL  
Myles Cameron-Smith  
Neil Moore  
Nelson  
neoman  
Nicholas Melnick  
Nikolaj Brinch Jørgensen  
Nils Andreas  
Nils Eilers  
Nils Hammerich  
Nils77  
Norah Smith  
Norman King  
Normen Zoch  
Olaf Grunert  
Ole Etels  
Oliver Boerner  
Oliver Brüggmann  
Oliver Smith  
Olivier Bori  
ONEPSI LLC  
oRdYNe  
Osaühing Trioflex  
OSHA-PROS USA  
Padawer  
Patrick Becher  
Patrick Bürckstümmer  
Patrick de Zoete  
Patrick Toal  
Patrick Vogt  
Paul Alexander Warren  
Paul Gerhardt (KONG)  
Paul Jackson  
Paul Johnson  
Paul Massay  
Paul Westlake  
Paul Woegerer  
Pauline Brasch  
Paulo Apolonia  
Pete Collin  
Pete of Retrohax.net  
Peter Eliades  
Peter Gries  
Peter Habura  
Peter Herklotz  
Peter Huyoff  
Peter Leswell  
Peter Weile  
Petri Alvinen  
Philip Marien  
Philip Timmermann  
Philipp Rudin  
Pierre Kressmann  
Pieter Labie  
Piotr Kmiecik  
Power-on.at  
Przemysław Safonow  
Que Labs  
R Welbourn  
R-Flux  
Rafat Michno  
Rainer Kappler  
Rainer Weninger  
Ralf Griewel  
Ralf Reinhardt  
Ralf Schenden  
Ralf Smolarek  
Ralf Zenker  
Ralph Bauer  
Ralph Wernecke  
Rédl Károly  
Reiner Lanowski  
Remi Veilleux  
Riccardo Bianchi  
Richard Englert  
Richard Good  
Richard Menedetter  
Richard Sopuch  
Rick Reynolds  
Rico Gruninger  
Rob Dean  
Robert Bernardo  
Robert Eaglestone  
Robert Grasböck  
Robert Miles  
Robert Schwan  
Robert Shively  
Robert Tangmar  
Robert Trangmar  
Rodney Xerri  
Roger Olsen  
Roger Pugh  
Roland Attila Kett  
Roland Evers  
Roland Schatz  
Rolf Hass  
Ronald Cooper  
Ronald Hunn  
Ronny Preiß  
Roy van Zundert

Rüdiger Wohlfomm  
Ruediger Schlenter  
Rutger Willemse  
Sampo Peltonen  
Sarmad Gilani  
SAS74  
Scott Halman  
Scott Hollier  
Scott Robison  
Sebastian Baranski  
Sebastian Bölling  
Sebastian Felzmann  
Sebastian Lipp  
Sebastian Rakel  
Şemseddin Moldibi  
Seth Morabito  
Shawn McKee  
Siegfried Hartmann  
Sigurbjorn Larusson  
Sigurdur Finnsson  
Simon Lawrence  
Simon Wolf  
spreen.digital  
Stefan Haberl  
Stefan Kramperth  
Stefan Richter  
Stefan Schultze  
Stefan Sonnek  
Stefan Theil  
Stefan Vrampe  
Stefano Canali  
Stefano Mozzi  
Steffen Reiersen  
Stephan Bielmann  
Stephen Jones  
  
Steve Gray  
Steve Kurlin  
Steve Lemieux  
Steven Combs  
Stewart Dunn  
Stuart Marsh  
Sven Neumann  
Sven Stache  
Sven Sternberger  
Szabolcs Bence  
Tantrumedia Limited  
Techvana Operations Ltd.  
Teddy Turmeaux  
Teemu Korvenpää  
The Games Foundation  
Thierry Supplisson  
Thieu-Duy Thai  
Thomas Bierschenk  
Thomas Edmister  
Thomas Frauenknecht  
Thomas Gitzen  
Thomas Gruber  
Thomas Haidler  
Thomas Jager  
Thomas Karlsen  
Thomas Laskowski  
Thomas Marschall  
Thomas Niemann  
Thomas Scheelen  
Thomas Schilling  
Thomas Tahsin-Bey  
Thomas Walter  
Thomas Wirtzmann  
Thorsten Knoll  
Tim Krome  
  
Tim Waite  
Timo Weirich  
Timothy Blanks  
Timothy Henson  
Timothy Prater  
Tobias Butter  
Tobias Heim  
Tobias Köck  
Tobias Lüthi  
Tommi Vasarainen  
Toni Ammer  
Tore Olsen  
Torleif Strand  
Torsten Schröder  
Tuan Nguyen  
Uffe Jakobsen  
Ulrich Hintermeier  
Ulrich Nieland  
Ulrik Kruse  
Ursula Förstle  
Uwe Boschanski  
Vedran Vrbanc  
Verm Project  
Wayne Sander  
Wayne Steele  
Who Knows  
Winfried Falkenhahn  
Wolfgang Becker  
Wolfgang Stabla  
Worblehat  
www.patop69.net  
Yan B  
Zoltan Markus  
Zytex Online Store



# **Bibliography**



- [1] N. Montfort, P. Baudoin, J. Bell, I. Bogost, J. Douglass, M. C. Marino, M. Mateas, C. Reas, M. Sample, and N. Vawter, *10 PRINT CHR \$(205.5+ RND (1));: GOTO 10*. MIT Press, 2012.
- [2] L. Soares and M. Stumm, "Flexsc: Flexible system call scheduling with exception-less system calls." in *Osdi*, vol. 10, 2010, pp. 1–8.
- [3] Actraiser, "Vic-ii for beginners: Screen modes, cheaper by the dozen," 2013. [Online]. Available: <http://dustlayer.com/vic-ii/2013/4/26/vic-ii-for-beginners-screen-modes-cheaper-by-the-dozen>



# INDEX



, (comma), 7-22, 7-39  
.bit files, 5-6  
.cor files, 5-6  
.mcs files, 5-6  
.prm files, 5-6  
: (colon), 7-22, 7-39, 7-40  
<> (not equal to), 7-31  
\$00 (PORTDDR), G-16  
\$00 (STOPTX), P-11  
\$01 (PORT), G-16  
\$01 (STARTTX), P-11  
\$D0 (RXNORMAL), P-11  
\$D4 (DEBUGVIC), P-11  
\$DC (DEBUGCPU), P-11  
\$DE (RXONLYONE), P-11  
\$F1 (FRAME1K), P-11  
\$F2 (FRAME2K), P-11

ABS, B-9  
ADC, H-17, H-62, H-116  
ADCQ, G-14, H-117  
ALR, H-18  
Amiga(tm) style audio, L-17  
ANC, H-18  
AND, 7-44, B-10, H-19, H-62, H-117  
ANDQ, G-14, H-118  
APPEND, B-11  
ARR, H-20  
ASC, B-12  
ASL, H-21, H-63  
ASLQ, G-14, H-119  
ASR, H-64  
ASRQ, H-119  
ASSEMBLE, K-10  
ASW, H-65  
ATN, B-13  
audio cross-bar switch, Q-20  
audio mixer, Q-20  
AUTO, B-14

BACKGROUND, 7-44, B-15  
BACKUP, B-16  
BANK, B-17

BASIC 65 Commands, B-199  
APPEND, B-11  
Examples, B-11  
AUTO, B-14  
Examples, B-14  
BACKGROUND, 7-44, B-15  
Examples, B-15, B-75  
BACKUP, B-16  
Examples, B-16  
BANK, B-17, F-4  
Examples, B-17, B-143,  
B-149-B-151, B-174, B-202,  
B-217, B-222

BEGIN, B-18  
Examples, B-18, B-19, B-179  
BEND, B-19  
Examples, B-18, B-19, B-179  
BLOAD, B-20  
Examples, B-21, B-132, B-174,  
B-195, B-202, B-217

BOOT, B-22  
Examples, B-22  
BORDER, 7-44, B-23  
Examples, B-23, B-35, B-75  
BOX, B-24  
Examples, B-24  
BSAVE, B-25  
Examples, B-25  
BUMP, B-26  
Examples, B-26, B-40  
BVERIFY, B-27  
Examples, B-27  
CATALOG, B-28  
Examples, B-28, B-29  
CHANGE, B-30  
Examples, B-30  
CHAR, B-31  
Examples, B-32, B-75  
CIRCLE, B-34  
Examples, B-35  
CLOSE, B-36

Examples, B-36, B-38, B-75,  
B-82, B-95, B-100, B-121,  
B-141, B-142, B-145, B-169,  
B-173, B-185

CLR, B-37  
Examples, B-35, B-37, B-75,  
B-100, B-117, B-121, B-132,  
B-141, B-142, B-145, B-169,  
B-173, B-181, B-185, B-194,  
B-207, B-220

CMD, B-38  
Examples, B-38, B-139

COLLECT, B-39  
Examples, B-39

COLLISION, B-40  
Examples, B-40

COLOR, B-41  
Examples, B-136, B-138

CONCAT, B-42  
Examples, B-42

CONT, 7-39, B-43  
Examples, B-43

COPY, B-44  
Examples, B-45, B-209

CURSOR, B-47  
Examples, B-47, B-160

DATA, B-48  
Examples, B-48, B-54, B-121,  
B-161, B-166, B-203

DCLEAR, B-49  
Examples, B-49

DCLOSE, B-50  
Examples, B-49, B-50, B-92,  
B-93, B-98, B-99, B-108,  
B-118, B-156, B-162, B-168,  
B-197

DEF  
Examples, B-35, B-52, B-75,  
B-87, B-88, B-100, B-117,  
B-121, B-141, B-142, B-145,  
B-169, B-172, B-173, B-181,  
B-185, B-186, B-220

DEF FN, B-52

DELETE, B-53  
Examples, B-53

DIM, B-54  
Examples, B-54, B-89, B-92,  
B-93, B-106, B-108, B-118,  
B-134, B-162, B-163

DIR, B-55  
Examples, B-186

DISK, B-57  
Examples, B-57, B-186

DLOAD, B-58  
Examples, B-58, B-126, B-186

DMA, B-59, F-4  
Examples, B-59

DMODE, B-60

DO, B-61  
Examples, B-61, B-82, B-94,  
B-95, B-98, B-99, B-106,  
B-124, B-194, B-214, B-223

DOPEN, B-62  
Examples, B-63, B-65, B-66,  
B-92, B-93, B-98, B-99,  
B-108, B-118, B-156, B-162,  
B-168, B-197

DPAT, B-64

DSAVE, B-67  
Examples, B-67

DVERIFY, B-69  
Examples, B-69

EDMA, B-72  
Examples, B-72

ELLIPSE, B-74  
Examples, B-75

ELSE, B-76  
Examples, B-105, B-206

END, 7-39, B-77  
Examples, B-40, B-48, B-54,  
B-77, B-105, B-167, B-168,  
B-203, B-206, B-210

ENVELOPE, B-78

Examples, B-78, B-148, B-205,  
B-221  
ERASE, B-79  
EXIT, B-82  
    Examples, B-82, B-95, B-106  
FAST, B-84  
    Examples, B-84, B-190  
FILTER, B-85  
    Examples, B-85  
FIND, B-86  
FONT, B-88  
FOR, 7-7, B-89  
    Examples, B-40, B-48, B-52,  
    B-54, B-87-B-89, B-108,  
    B-118, B-128, B-134, B-149,  
    B-155, B-156, B-161, B-162,  
    B-166-B-169, B-172, B-192,  
    B-198, B-203, B-207-B-212,  
    B-225  
FOREGROUND, 7-44, B-90  
    Examples, B-75  
GET, B-94  
    Examples, B-18, B-61, B-82,  
    B-94, B-95, B-124, B-214,  
    B-223  
GET#, B-95  
GETKEY, B-96  
    Examples, B-35, B-75, B-96,  
    B-100, B-141, B-142, B-145,  
    B-185  
GO64, B-97  
    Examples, B-97  
GOSUB, B-98  
    Examples, B-61, B-84, B-98,  
    B-111, B-124, B-138, B-168,  
    B-190, B-214, B-223  
GOTO, 7-46, B-99  
    Examples, B-18, B-43, B-94,  
    B-96, B-98, B-99, B-106,  
    B-108, B-111, B-118, B-138,  
    B-187, B-209  
GRAPHIC, B-100  
Examples, B-35, B-75, B-100,  
B-117, B-121, B-141, B-142,  
B-145, B-169, B-173, B-181,  
B-185  
HEADER, B-101  
    Examples, B-101  
HELP, B-102  
    Examples, B-102  
HIGHLIGHT, B-104  
IF, 7-30, B-105  
    Examples, B-10, B-18, B-19,  
    B-26, B-35, B-65, B-66, B-73,  
    B-76, B-77, B-80-B-82,  
    B-94-B-96, B-98, B-99,  
    B-105, B-106, B-108, B-111,  
    B-118, B-135, B-138, B-140,  
    B-149, B-153, B-168, B-171,  
    B-179, B-197, B-199, B-206  
INPUT, 7-17, 7-24, B-106  
    Examples, B-76, B-82, B-95,  
    B-98, B-99, B-105, B-106,  
    B-108, B-118, B-162, B-168,  
    B-197, B-206  
INPUT#, B-107  
INSTR, B-109  
    Examples, B-109  
KEY, B-112  
    Examples, B-4, B-112, B-113,  
    B-136, B-138  
LET, 7-15, B-116  
    Examples, B-116  
LINE, B-117  
    Examples, B-82, B-95, B-100,  
    B-117, B-118, B-141, B-142,  
    B-145, B-169, B-181, B-185,  
    B-197  
LINE INPUT#, B-118  
LIST, 7-11, 7-12, B-119  
    Examples, B-38, B-119, B-139  
LOAD, B-120  
    Examples, B-113, B-120, B-121,  
    B-178, B-194

LOADIFF, B-121  
    Examples, B-121

LOOP, B-124  
    Examples, B-61, B-82, B-94,  
        B-95, B-98, B-99, B-106,  
        B-124, B-214, B-223

MERGE, B-126  
    Examples, B-126

MONITOR, B-129  
    Examples, B-129

MOUSE, B-130  
    Examples, B-130, B-136, B-138,  
        B-171

MOVSPR, B-131  
    Examples, B-40, B-132, B-194

NEW, B-133  
    Examples, B-133

NEXT, B-134  
    Examples, B-35, B-40, B-48,  
        B-52, B-54, B-87-B-89, B-92,  
        B-93, B-108, B-118, B-121,  
        B-128, B-132, B-134, B-149,  
        B-155, B-156, B-161, B-162,  
        B-166-B-169, B-172, B-192,  
        B-194, B-198, B-203,  
        B-207-B-212, B-225

OFF, B-136  
    Examples, B-14, B-113, B-130,  
        B-136, B-171

ON, B-137  
    Examples, B-47, B-111, B-113,  
        B-130, B-138, B-160, B-171,  
        B-186, B-187

OPEN, B-139  
    Examples, B-35, B-36, B-38,  
        B-75, B-82, B-95, B-100,  
        B-121, B-139, B-141, B-142,  
        B-145, B-169, B-173, B-185

PAINT, B-141  
    Examples, B-141, B-142

PALETTE, B-142

Examples, B-35, B-75, B-100,  
        B-121, B-142, B-145, B-185

PEN, B-145  
    Examples, B-35, B-75, B-100,  
        B-145, B-169, B-173, B-185

PLAY, B-147  
    Examples, B-78, B-148, B-205,  
        B-221

POLYGON, B-152  
    Examples, B-152

PRINT, 7-11, B-155  
    Examples, B-5, B-9, B-10, B-12,  
        B-13, B-18, B-19, B-26, B-33,  
        B-36-B-38, B-40, B-43, B-46,  
        B-48, B-51, B-52, B-66, B-68,  
        B-73, B-76, B-77, B-79-B-83,  
        B-87-B-89, B-91, B-95, B-98,  
        B-99, B-102, B-103, B-105,  
        B-106, B-108, B-111, B-114,  
        B-115, B-118, B-122, B-123,  
        B-125, B-127, B-128, B-134,  
        B-135, B-138, B-140, B-143,  
        B-144, B-149, B-153, B-155,  
        B-156, B-158, B-160-B-162,  
        B-166-B-174, B-179, B-182,  
        B-188, B-192, B-196-B-200,  
        B-202-B-204, B-206-B-212,  
        B-216-B-218, B-225

PRINT USING, B-157

PRINT#, B-156

RCURSOR, B-160  
    Examples, B-160

READ, B-161  
    Examples, B-48, B-54, B-161,  
        B-166, B-203

RECORD, B-162  
    Examples, B-162

REM, B-163  
    Examples, B-15-B-19, B-21,  
        B-23, B-26, B-33, B-35, B-36,  
        B-38, B-40, B-45, B-47, B-48,  
        B-50, B-52-B-54, B-57, B-59,

B-61, B-63, B-72, B-75-B-77,  
B-79, B-82, B-84, B-85, B-87,  
B-88, B-94-B-96,  
B-98-B-100, B-105, B-106,  
B-108-B-111, B-113,  
B-116-B-119, B-121, B-124,  
B-125, B-130, B-133, B-136,  
B-138, B-139, B-141-B-145,  
B-149-B-151, B-153-B-156,  
B-159, B-162-B-165, B-168,  
B-169, B-171-B-173,  
B-175-B-177, B-179, B-181,  
B-182, B-185-B-187,  
B-189-B-193, B-195, B-197,  
B-199, B-205-B-209, B-211,  
B-212, B-214, B-217,  
B-222-B-224

RENAME, B-164  
    Examples, B-164

RENUMBER, 7-38, B-165  
    Examples, B-165

RESTORE, B-166  
    Examples, B-35, B-75, B-100,  
        B-121, B-133, B-166, B-185

RESUME, B-167  
    Examples, B-73, B-80, B-81,  
        B-167, B-210

RETURN, B-168  
    Examples, B-40, B-98, B-99,  
        B-111, B-138, B-168

RMOUSE, B-171  
    Examples, B-171

RREG, B-174  
    Examples, B-174, B-202

RUN, B-178  
    Examples, B-5, B-43, B-48,  
        B-52, B-87, B-88, B-102,  
        B-108, B-127, B-149, B-160,  
        B-169, B-173, B-178, B-192,  
        B-203, B-208, B-211, B-212

SAVE, B-180  
    Examples, B-113, B-180

SCNCLR, B-181  
    Examples, B-35, B-75, B-100,  
        B-117, B-121, B-132, B-169,  
        B-173, B-181, B-194

SCRATCH, B-182  
    Examples, B-79, B-182

SCREEN, B-183  
    Examples, B-35, B-75, B-100,  
        B-117, B-121, B-141, B-142,  
        B-145, B-169, B-173, B-181,  
        B-185

SET, B-186  
    Examples, B-35, B-75, B-100,  
        B-117, B-121, B-141, B-142,  
        B-145, B-169, B-173, B-181,  
        B-185, B-186

SLEEP, B-189  
    Examples, B-121, B-132, B-185,  
        B-189

SLOW, B-190  
    Examples, B-84, B-190

SOUND, B-191  
    Examples, B-191

SPRCOLOR, B-193  
    Examples, B-193

SPRITE, B-194  
    Examples, B-40, B-175-B-177,  
        B-193-B-195

SPRSAV, B-195  
    Examples, B-195

STEP, 7-13, B-198  
    Examples, B-52, B-87-B-89,  
        B-134, B-198, B-208

STOP, 7-39

SYS, B-201  
    Examples, B-22, B-174, B-202

TEMPO, B-205  
    Examples, B-78, B-148, B-205,  
        B-221

THEN, 7-30, B-206  
    Examples, B-10, B-18, B-19,  
        B-26, B-65, B-66, B-73, B-77,

B-80-B-82, B-94-B-96, B-98,  
B-99, B-105, B-106, B-108,  
B-111, B-118, B-135, B-138,  
B-140, B-153, B-168, B-171,  
B-179, B-197, B-199, B-206

TO, B-209  
Examples, B-16, B-25, B-30,  
B-40, B-42, B-45, B-48, B-52,  
B-54, B-87-B-89, B-108,  
B-118, B-128, B-134, B-155,  
B-156, B-161, B-162, B-164,  
B-166-B-169, B-172, B-186,  
B-192, B-194, B-198, B-203,  
B-207-B-212, B-225

TRAP, B-210  
Examples, B-73, B-80, B-81,  
B-167, B-210

TROFF, B-211  
Examples, B-211, B-212

TRON, B-212  
Examples, B-211, B-212

TYPE, B-213  
Examples, B-108, B-213

UNTIL, B-214  
Examples, B-61, B-94, B-98,  
B-99, B-106, B-124, B-214,  
B-223

USING, B-215  
Examples, B-52, B-87, B-88,  
B-158, B-208, B-216

VERIFY, B-219  
Examples, B-186, B-219

VIEWPORT, B-220  
Examples, B-220

VOL, B-221  
Examples, B-78, B-148, B-205,  
B-221

WAIT, B-222  
Examples, B-222

WHILE, B-223  
Examples, B-61, B-124, B-214,  
B-223

WINDOW, B-224  
Examples, B-224

BASIC 65 Functions

ABS, B-9  
ASC, B-12  
ATN, B-13  
CHR\$, B-33  
COS, B-46  
DEC, B-51  
ERR\$, B-81  
EXP, B-83  
FN, B-52, B-87  
FRE, B-91  
FREAD, B-92  
FWRITE, B-93  
HEX\$, B-103  
INT, B-110  
JOY, B-111  
LEFT\$, B-114  
LEN, B-115  
LOG, B-122  
LOG10, B-123  
LPEN, B-125  
MID\$, B-127  
MOD, B-128  
PEEK, B-143  
PEEKW, B-144  
PIXEL, B-146  
POINTER, B-149  
POKE, B-150  
POKEW, B-151  
POS, B-153  
POT, B-154  
RCOLOR, B-159  
RGRAPHIC, B-169  
RIGHT\$, B-170  
RND, 7-45, B-172  
RPEN, B-173  
RSPCOLOR, B-175  
RSPPPOS, B-176  
RSprite, B-177  
RWINDOW, B-179

SGN, B-187  
SIN, B-188  
SPC, B-192  
SQR, B-196  
STR\$, B-200  
TAB, B-203  
TAN, B-204  
USR, B-217  
VAL, B-218  
**BASIC 65 Operators**  
AND, 7-44, B-10  
NOT, B-135  
OR, B-140  
XOR, B-225  
**BASIC 65 System Commands**  
EDIT, B-70  
**BASIC 65 System Variables**  
DS, B-65  
DS\$, B-66  
DT\$, B-68  
EL, B-73  
ER, B-80  
ST, B-197  
TI, B-207  
TI\$, B-208  
BBR0, H-65  
BBR1, H-65  
BBR2, H-66  
BBR3, H-66  
BBR4, H-67  
BBR5, H-67  
BBR6, H-67  
BBR7, H-68  
BBS0, H-68  
BBS1, H-68  
BBS2, H-69  
BBS3, H-69  
BBS4, H-69  
BBS5, H-70  
BBS6, H-70  
BBS7, H-70  
BCC, H-21, H-71  
BCS, H-22, H-71  
BEGIN, B-18  
BEND, B-19  
BEQ, H-22, H-71  
bgcolor, 15-7  
BIT, H-22, H-72  
BITMAPS, K-11  
BITQ, G-14, H-120  
blink, 15-9  
BLOAD, B-20  
blocked, 7-21  
BMI, H-23, H-72  
BNE, H-23, H-73  
BOOT, B-22  
BORDER, 7-44, B-23  
bordercolor, 15-7  
BOX, B-24  
box, 15-10  
BPL, H-23, H-73  
BRA, H-74  
BREAKPOINT, K-16  
BRK, H-24, H-74  
BSAVE, B-25  
BSR, H-75  
BUMP, B-26  
BVC, H-25, H-75  
BVERIFY, B-27  
BVS, H-25, H-76  
CATALOG, B-28  
cellcolor, 15-9  
cgetc, 15-18  
CHANGE, B-30  
CHAR, B-31  
character, 7-13  
character set, 7-13  
CHR\$, B-33  
cinput, 15-19  
CIRCLE, B-34  
CLC, H-25, H-76  
CLD, H-26, H-76  
CLE, H-77  
CLI, H-26, H-77

CLOSE, B-36  
CLR, B-37  
clrscr, 15-5  
CLV, H-27, H-78  
CMD, B-38  
CMP, H-27, H-48, H-78, H-121  
CMPO, H-121  
COLLECT, B-39  
COLLISION, B-40  
COLOR, B-41  
CONCAT, B-42  
conionit, 15-3  
CONT, 7-39, B-43  
context dependent, 7-19  
COPY, B-44  
copyright, ii  
COS, B-46  
CPQ, G-14  
cprintf, 15-17  
CPUHISTORY, K-20  
CPUMEMORY, K-15  
cputc, 15-14  
cputcxy, 15-16  
cputdec, 15-15  
cputhex, 15-15  
cputnc, 15-15  
cputncxy, 15-17  
cputs, 15-16  
cputsxy, 15-16  
CPX, H-28, H-79  
CPY, H-29, H-80  
CPZ, H-80  
cross-bar switch, audio, Q-20  
CURSOR, B-47  
  
DATA, B-48  
DCLEAR, B-49  
DCLOSE, B-50  
DCP, H-29  
DEBUGCPU, P-11  
DEBUGMON, K-18  
DEBUGVIC, P-11  
DEC, B-51, H-30, H-81  
  
DEF FN, B-52  
DELETE, B-53  
DEQ, G-14, H-122  
DEW, H-81  
DEX, H-30, H-82  
DEY, H-31, H-82  
DEZ, H-83  
Digital Audio, L-17  
digital video, M-9  
DIM, B-54  
DIR, B-55  
Direct Mode, 7-18  
DISASSEMBLE, K-5, K-16  
DISK, B-57  
DLOAD, B-58  
DMA, B-59  
DMA Audio, L-17  
DMODE, B-60  
DO, B-61  
DOPEN, B-62  
DPAT, B-64  
DS, B-65  
DS\$, B-66  
DSAve, B-67  
DT\$, B-68  
DVERIFY, B-69  
  
EDIT, B-70  
EDMA, B-72  
EL, B-73  
ELLIPSE, B-74  
ELSE, B-76  
END, 7-39, B-77  
ENVELOPE, B-78  
EOM, H-83  
EOR, H-31, H-84, H-123  
EORQ, G-14, H-123  
ER, B-80  
ERASE, B-79  
ERR\$, B-81  
Errors  
    Extra Ignored, 7-22  
    Illegal Direct, 7-18

Syntax, 7-4  
Type mismatch, 7-17  
EXIT, B-82  
EXP, B-83  
Extra Ignored, 7-22

FAST, B-84  
FILL, K-17  
fillrect, 15-10  
FILTER, B-85  
FIND, B-86  
FLAGWATCH, K-16  
Flash Menu, I-5  
flushkeybuf, 15-19  
FN, B-52, B-87  
FONT, B-88  
FOR, 7-7, B-89  
FOREGROUND, 7-44, B-90  
FRAME1K, P-11  
FRAME2K, P-11  
FRE, B-91  
FREAD, B-92  
Freeze Menu, I-5  
FWRITE, B-93

Games  
    Guess the number, 7-33

GET, B-94  
GET#, B-95  
getcharsetaddr, 15-5  
getcolramoffset, 15-4  
GETKEY, B-96  
getkeymodstate, 15-18  
getscreenaddr, 15-4  
getscreensize, 15-5  
GO64, B-97  
gohome, 15-12  
GOSUB, B-98  
GOTO, 7-46, B-99  
gotox, 15-12  
gotoxy, 15-12  
gotoy, 15-12  
GRAPHIC, B-100

Guess the number, 7-33  
HEADER, B-101  
HELP, B-102, K-16  
HEX\$, B-103  
HIGHLIGHT, B-104  
highlight, 15-8  
hline, 15-11  
HYPERTRAP, K-14

IF, 7-30, B-105  
Illegal Direct Error, 7-18  
IMDH™, M-9  
INC, H-32, H-85  
INPUT, 7-17, 7-24, B-106  
INPUT#, B-107  
INQ, G-14, H-124  
INSTR, B-109  
INT, B-110  
Integrated Marvellous Digital Hookup™,  
    M-9  
INTERRUPTS, K-17  
INW, H-85  
INX, H-32, H-86  
INY, H-33, H-86  
INZ, H-86  
IO  
    blocking, 7-21  
ISC, H-33

JMP, H-34, H-87  
JOY, B-111  
JSR, H-34, H-87

kbhit, 15-18  
KEY, B-112  
Keyboard  
    matrix, D-4  
KIL, H-35

LAS, H-36  
LAX, H-36  
LDA, H-37, H-88, H-125

LDQ, G-14, H-125  
LDX, H-38, H-89  
LDY, H-38, H-89  
LDZ, H-90  
LEFT\$, B-114  
LEN, B-115  
LET, 7-15, B-116  
light pen, Q-19  
LINE, B-117  
Line Drawing, L-13  
DMA Option Bytes, L-13  
LINE INPUT#, B-118  
Lines  
    editing, 7-25  
    renumbering, 7-38  
    replacing, 7-25  
LIST, 7-11, 7-12, B-119  
LOAD, B-120  
LOADIFF, B-121  
LOADMEMORY, K-18  
LOG, B-122  
LOG10, B-123  
LOOP, B-124  
LPEN, B-125  
LSR, H-39, H-90  
LSRQ, G-14, H-126

MAP, G-8-G-10, H-91  
MEGA Flash, I-5  
MEMORY, K-6, K-18  
Memory banking, G-8-G-10  
MERGE, B-126  
MID\$, B-127  
mixer, audio, Q-20  
MOD, B-128  
MOD-file style audio, L-17  
MONITOR, B-129  
    Enhanced MEGA65 ROM Monitor,  
        K-7  
    Matrix Mode/Serial Monitor  
        Interface, K-12  
    Standard C65 ROM Monitor, K-3  
MONITOR Commands  
ASSEMBLE, K-10  
BITMAPS, K-11  
BREAKPOINT, K-16  
CPUHISTORY, K-20  
CPUMEMORY, K-15  
DEBUGMON, K-18  
DISASSEMBLE, K-5, K-16  
FILL, K-17  
FLAGWATCH, K-16  
HELP, K-16  
HYPERTRAP, K-14  
INTERRUPTS, K-17  
LOADMEMORY, K-18  
MEMORY, K-6, K-18  
REGISTERS, K-18  
SETMEMORY, K-19  
SETPC, K-17  
TRACE, K-19  
UARTDIVISOR, K-15  
WATCHPOINT, K-19

MOUSE, B-130  
movedown, 15-13  
moveleft, 15-13  
moveright, 15-14  
moveup, 15-13  
MOVSPR, B-131

name spaces, 7-17  
NEG, H-92  
NEW, B-133  
NEXT, B-134  
NO SCROLL, 7-46  
NOP, H-39  
NOT, B-135  
not equal, 7-31

OFF, B-136  
ON, B-137  
OPEN, B-139  
OpenROMs, 11-4  
operators  
    relational, 7-30  
OR, B-140

|                        |                          |
|------------------------|--------------------------|
| ORA, H-41, H-92, H-127 | Registers                |
| ORQ, G-14, H-128       | \$D000, M-32             |
| PAINT, B-141           | \$D001, M-32             |
| PALETTE, B-142         | \$D002, M-32             |
| PEEK, B-143            | \$D003, M-32             |
| PEEKW, B-144           | \$D004, M-32             |
| PEN, B-145             | \$D005, M-32             |
| PHA, H-41, H-93        | \$D006, M-32             |
| PHP, H-42, H-93        | \$D007, M-32             |
| PHW, H-94              | \$D008, M-32             |
| PHX, H-94              | \$D009, M-32             |
| PHY, H-94              | \$D00A, M-32             |
| PHZ, H-95              | \$D00B, M-32             |
| PIXEL, B-146           | \$D00C, M-32             |
| PLA, H-42, H-95        | \$D00D, M-32             |
| PLAY, B-147            | \$D00E, M-32             |
| PLP, H-42, H-95        | \$D00F, M-32             |
| PLX, H-96              | \$D010, M-32             |
| PLY, H-96              | \$D011, M-32             |
| PLZ, H-97              | \$D012, M-32             |
| POINTER, B-149         | \$D013, M-32             |
| POKE, B-150            | \$D014, M-32             |
| POKEW, B-151           | \$D015, M-32             |
| POLYGON, B-152         | \$D016, M-32             |
| PORT, G-16             | \$D017, M-32             |
| PORTDDR, G-16          | \$D018, M-32             |
| POS, B-153             | \$D019, M-32             |
| POT, B-154             | \$D01A, M-32             |
| PRINT, 7-11, B-155     | \$D01B, M-32             |
| PRINT USING, B-157     | \$D01C, M-32             |
| PRINT#, B-156          | \$D01D, M-32             |
| Programmes             | \$D01E, M-32             |
| editing, 7-25          | \$D01F, M-32             |
| replacing lines, 7-25  | \$D020, M-32, M-35, M-38 |
| quote mode, 7-35       | \$D021, M-32, M-35, M-38 |
| RCOLOR, B-159          | \$D022, M-32, M-35, M-38 |
| RCURSOR, B-160         | \$D023, M-32, M-35, M-38 |
| READ, B-161            | \$D024, M-33, M-35, M-38 |
| RECORD, B-162          | \$D025, M-33, M-35, M-38 |
| REGISTERS, K-18        | \$D026, M-33, M-35, M-39 |
|                        | \$D027, M-33             |
|                        | \$D028, M-33             |

|                    |              |
|--------------------|--------------|
| \$D029, M-33       | \$D054, M-39 |
| \$D02A, M-33       | \$D055, M-39 |
| \$D02B, M-33       | \$D056, M-39 |
| \$D02C, M-33       | \$D057, M-39 |
| \$D02D, M-33       | \$D058, M-39 |
| \$D02E, M-33       | \$D059, M-39 |
| \$D02F, M-35, M-39 | \$D05A, M-39 |
| \$D030, M-33, M-35 | \$D05B, M-39 |
| \$D031, M-35       | \$D05C, M-39 |
| \$D033, M-35       | \$D05D, M-39 |
| \$D034, M-35       | \$D05E, M-39 |
| \$D035, M-35       | \$D05F, M-39 |
| \$D036, M-36       | \$D060, M-39 |
| \$D037, M-36       | \$D061, M-39 |
| \$D038, M-36       | \$D062, M-39 |
| \$D039, M-36       | \$D063, M-39 |
| \$D03A, M-36       | \$D064, M-39 |
| \$D03B, M-36       | \$D065, M-39 |
| \$D03C, M-36       | \$D068, M-39 |
| \$D03D, M-36       | \$D069, M-39 |
| \$D03E, M-36       | \$D06A, M-39 |
| \$D03F, M-36       | \$D06B, M-39 |
| \$D040, M-36       | \$D06C, M-39 |
| \$D041, M-36       | \$D06D, M-39 |
| \$D042, M-36       | \$D06E, M-40 |
| \$D043, M-36       | \$D06F, M-40 |
| \$D044, M-36       | \$D070, M-40 |
| \$D045, M-36       | \$D071, M-40 |
| \$D046, M-36       | \$D072, M-40 |
| \$D047, M-36       | \$D073, M-40 |
| \$D048, M-39       | \$D074, M-40 |
| \$D049, M-39       | \$D075, M-40 |
| \$D04A, M-39       | \$D076, M-40 |
| \$D04B, M-39       | \$D077, M-40 |
| \$D04C, M-39       | \$D078, M-40 |
| \$D04D, M-39       | \$D079, M-40 |
| \$D04E, M-39       | \$D07A, M-40 |
| \$D04F, M-39       | \$D07B, M-40 |
| \$D050, M-39       | \$D07C, M-40 |
| \$D051, M-39       | \$D080, Q-9  |
| \$D052, M-39       | \$D081, Q-9  |
| \$D053, M-39       | \$D082, Q-9  |

|                       |                    |
|-----------------------|--------------------|
| \$D083, Q-9           | \$D628, O-4        |
| \$D084, Q-9           | \$D640, G-17, G-33 |
| \$D085, Q-9           | \$D641, G-17, G-33 |
| \$D086, Q-9           | \$D642, G-17       |
| \$D087, Q-9           | \$D643, G-17, G-33 |
| \$D088, Q-9           | \$D644, G-17, G-33 |
| \$D089, Q-9           | \$D645, G-17, G-33 |
| \$D08A, Q-9           | \$D646, G-17, G-33 |
| \$D100 - \$D1FF, M-36 | \$D647, G-17, G-33 |
| \$D200 - \$D2FF, M-36 | \$D648, G-17, G-34 |
| \$D300 - \$D3FF, M-36 | \$D649, G-17, G-34 |
| \$D600, O-3           | \$D64A, G-17, G-34 |
| \$D601, O-3           | \$D64B, G-17, G-34 |
| \$D602, O-3           | \$D64C, G-17, G-34 |
| \$D603, O-3           | \$D64D, G-17, G-34 |
| \$D604, O-3           | \$D64E, G-17, G-34 |
| \$D605, O-3           | \$D64F, G-17, G-34 |
| \$D606, O-3           | \$D650, G-17, G-34 |
| \$D609, O-4           | \$D651, G-17, G-34 |
| \$D60B, O-4           | \$D652, G-17, G-34 |
| \$D60C, O-4           | \$D653, G-17, G-34 |
| \$D60D, O-4           | \$D654, G-17, G-34 |
| \$D60E, O-4           | \$D655, G-17, G-34 |
| \$D60F, O-4           | \$D656, G-17, G-34 |
| \$D610, O-4           | \$D657, G-17, G-34 |
| \$D611, O-4           | \$D658, G-17, G-34 |
| \$D612, O-4           | \$D659, G-17, G-34 |
| \$D615, O-4           | \$D65A, G-17       |
| \$D616, O-4           | \$D65B, G-17       |
| \$D617, O-4           | \$D65C, G-17       |
| \$D618, O-4           | \$D65D, G-17       |
| \$D619, O-4           | \$D65E, G-17       |
| \$D61A, O-4           | \$D65F, G-17       |
| \$D61D, O-4           | \$D660, G-17       |
| \$D61E, O-4           | \$D661, G-17       |
| \$D620, O-4           | \$D662, G-17       |
| \$D621, O-4           | \$D663, G-17       |
| \$D622, O-4           | \$D664, G-17       |
| \$D623, O-4           | \$D665, G-17       |
| \$D625, O-4           | \$D666, G-17       |
| \$D626, O-4           | \$D667, G-18       |
| \$D627, O-4           | \$D668, G-18       |

|                    |              |
|--------------------|--------------|
| \$D669, G-18       | \$D6AF, Q-17 |
| \$D66A, G-18       | \$D6B0, Q-19 |
| \$D66B, G-18       | \$D6B1, Q-19 |
| \$D66C, G-18       | \$D6B2, Q-19 |
| \$D66D, G-18       | \$D6B3, Q-19 |
| \$D66E, G-18       | \$D6B4, Q-19 |
| \$D66F, G-18       | \$D6B5, Q-19 |
| \$D670, G-18, G-34 | \$D6B7, Q-19 |
| \$D671, G-18, G-34 | \$D6B8, Q-19 |
| \$D672, G-18, G-34 | \$D6B9, Q-19 |
| \$D673, G-18       | \$D6BA, Q-19 |
| \$D674, G-18       | \$D6BB, Q-19 |
| \$D675, G-18       | \$D6BC, Q-19 |
| \$D676, G-18       | \$D6BD, Q-19 |
| \$D677, G-18       | \$D6BE, Q-19 |
| \$D678, G-18       | \$D6C0, Q-19 |
| \$D679, G-18       | \$D6E0, P-9  |
| \$D67A, G-18       | \$D6E1, P-9  |
| \$D67B, G-18       | \$D6E2, P-9  |
| \$D67C, G-18, G-34 | \$D6E3, P-9  |
| \$D67D, G-18, G-34 | \$D6E4, P-9  |
| \$D67E, G-18, G-34 | \$D6E5, P-9  |
| \$D67F, G-18, G-34 | \$D6E6, P-9  |
| \$D680, Q-17       | \$D6E7, P-9  |
| \$D681, Q-17       | \$D6E8, P-9  |
| \$D682, Q-17       | \$D6E9, P-9  |
| \$D683, Q-17       | \$D6EA, P-9  |
| \$D684, Q-17       | \$D6EB, P-9  |
| \$D686, Q-17       | \$D6EC, P-9  |
| \$D68A, Q-17       | \$D6ED, P-9  |
| \$D68B, Q-12       | \$D6EE, P-9  |
| \$D68C, Q-13       | \$D6F4, Q-23 |
| \$D68D, Q-13       | \$D6F5, Q-23 |
| \$D68E, Q-13       | \$D6F8, Q-23 |
| \$D68F, Q-13       | \$D6F9, Q-23 |
| \$D690, Q-13       | \$D6FA, Q-23 |
| \$D691, Q-13       | \$D6FB, Q-23 |
| \$D692, Q-13       | \$D6FC, Q-23 |
| \$D693, Q-13       | \$D6FD, Q-23 |
| \$D6A0, Q-11       | \$D700, L-20 |
| \$D6A1, Q-13       | \$D701, L-20 |
| \$D6A2, Q-11       | \$D702, L-20 |

|                    |              |
|--------------------|--------------|
| \$D703, L-20       | \$D73F, L-21 |
| \$D704, L-20       | \$D740, L-21 |
| \$D705, L-20       | \$D741, L-21 |
| \$D70E, L-20       | \$D742, L-21 |
| \$D70F, G-22       | \$D743, L-21 |
| \$D710, G-18       | \$D744, L-21 |
| \$D711, L-20, Q-23 | \$D745, L-21 |
| \$D71C, L-20       | \$D746, L-21 |
| \$D71D, L-20       | \$D747, L-21 |
| \$D71E, L-20       | \$D748, L-21 |
| \$D71F, L-20       | \$D749, L-21 |
| \$D720, L-20       | \$D74A, L-21 |
| \$D721, L-20       | \$D74B, L-21 |
| \$D722, L-20       | \$D74C, L-21 |
| \$D723, L-20       | \$D74D, L-21 |
| \$D724, L-20       | \$D74E, L-21 |
| \$D725, L-20       | \$D74F, L-21 |
| \$D726, L-20       | \$D750, L-21 |
| \$D727, L-20       | \$D751, L-21 |
| \$D728, L-20       | \$D752, L-21 |
| \$D729, L-20       | \$D753, L-21 |
| \$D72A, L-20       | \$D754, L-21 |
| \$D72B, L-20       | \$D755, L-21 |
| \$D72C, L-20       | \$D756, L-22 |
| \$D72D, L-20       | \$D757, L-22 |
| \$D72E, L-20       | \$D758, L-22 |
| \$D72F, L-20       | \$D759, L-22 |
| \$D730, L-21       | \$D75A, L-22 |
| \$D731, L-21       | \$D75B, L-22 |
| \$D732, L-21       | \$D75C, L-22 |
| \$D733, L-21       | \$D75D, L-22 |
| \$D734, L-21       | \$D75E, L-22 |
| \$D735, L-21       | \$D75F, L-22 |
| \$D736, L-21       | \$D768, G-22 |
| \$D737, L-21       | \$D769, G-22 |
| \$D738, L-21       | \$D76A, G-22 |
| \$D739, L-21       | \$D76B, G-22 |
| \$D73A, L-21       | \$D76C, G-22 |
| \$D73B, L-21       | \$D76D, G-22 |
| \$D73C, L-21       | \$D76E, G-22 |
| \$D73D, L-21       | \$D76F, G-22 |
| \$D73E, L-21       | \$D770, G-22 |

|              |              |
|--------------|--------------|
| \$D771, G-22 | \$D79B, G-23 |
| \$D772, G-22 | \$D79C, G-23 |
| \$D773, G-22 | \$D79D, G-23 |
| \$D774, G-22 | \$D79E, G-23 |
| \$D775, G-22 | \$D79F, G-23 |
| \$D776, G-22 | \$D7A0, G-23 |
| \$D777, G-22 | \$D7A1, G-23 |
| \$D778, G-22 | \$D7A2, G-23 |
| \$D779, G-22 | \$D7A3, G-23 |
| \$D77A, G-22 | \$D7A4, G-23 |
| \$D77B, G-22 | \$D7A5, G-23 |
| \$D77C, G-22 | \$D7A6, G-23 |
| \$D77D, G-22 | \$D7A7, G-23 |
| \$D77E, G-22 | \$D7A8, G-23 |
| \$D77F, G-22 | \$D7A9, G-23 |
| \$D780, G-22 | \$D7AA, G-23 |
| \$D781, G-22 | \$D7AB, G-23 |
| \$D782, G-22 | \$D7AC, G-23 |
| \$D783, G-22 | \$D7AD, G-23 |
| \$D784, G-22 | \$D7AE, G-23 |
| \$D785, G-22 | \$D7AF, G-23 |
| \$D786, G-22 | \$D7B0, G-23 |
| \$D787, G-22 | \$D7B1, G-23 |
| \$D788, G-22 | \$D7B2, G-23 |
| \$D789, G-22 | \$D7B3, G-23 |
| \$D78A, G-22 | \$D7B4, G-24 |
| \$D78B, G-22 | \$D7B5, G-24 |
| \$D78C, G-22 | \$D7B6, G-24 |
| \$D78D, G-22 | \$D7B7, G-24 |
| \$D78E, G-23 | \$D7B8, G-24 |
| \$D78F, G-23 | \$D7B9, G-24 |
| \$D790, G-23 | \$D7BA, G-24 |
| \$D791, G-23 | \$D7BB, G-24 |
| \$D792, G-23 | \$D7BC, G-24 |
| \$D793, G-23 | \$D7BD, G-24 |
| \$D794, G-23 | \$D7BE, G-24 |
| \$D795, G-23 | \$D7BF, G-24 |
| \$D796, G-23 | \$D7C0, G-24 |
| \$D797, G-23 | \$D7C1, G-24 |
| \$D798, G-23 | \$D7C2, G-24 |
| \$D799, G-23 | \$D7C3, G-24 |
| \$D79A, G-23 | \$D7C4, G-24 |

|              |                  |
|--------------|------------------|
| \$D7C5, G-24 | \$DC07, N-3      |
| \$D7C6, G-24 | \$DC08, N-3      |
| \$D7C7, G-24 | \$DC09, N-3      |
| \$D7C8, G-24 | \$DC0A, N-3      |
| \$D7C9, G-24 | \$DC0B, N-3      |
| \$D7CA, G-24 | \$DC0C, N-3      |
| \$D7CB, G-24 | \$DC0D, N-3      |
| \$D7CC, G-24 | \$DC0E, N-3      |
| \$D7CD, G-24 | \$DC0F, N-3, N-4 |
| \$D7CE, G-24 | \$DC10, N-6      |
| \$D7CF, G-24 | \$DC11, N-6      |
| \$D7D0, G-24 | \$DC12, N-6      |
| \$D7D1, G-24 | \$DC13, N-6      |
| \$D7D2, G-24 | \$DC14, N-6      |
| \$D7D3, G-24 | \$DC15, N-6      |
| \$D7D4, G-24 | \$DC16, N-6      |
| \$D7D5, G-24 | \$DC17, N-6      |
| \$D7D6, G-24 | \$DC18, N-6      |
| \$D7D7, G-24 | \$DC19, N-6      |
| \$D7D8, G-24 | \$DC1A, N-6      |
| \$D7D9, G-24 | \$DC1B, N-6      |
| \$D7DA, G-25 | \$DC1C, N-6      |
| \$D7DB, G-25 | \$DC1D, N-6      |
| \$D7DC, G-25 | \$DC1E, N-6      |
| \$D7DD, G-25 | \$DC1F, N-6      |
| \$D7DE, G-25 | \$DD00, N-4      |
| \$D7DF, G-25 | \$DD01, N-4      |
| \$D7E0, G-25 | \$DD02, N-4      |
| \$D7E1, G-25 | \$DD03, N-4      |
| \$D7E2, G-25 | \$DD04, N-4      |
| \$D7E3, G-25 | \$DD05, N-4      |
| \$D7FA, G-18 | \$DD06, N-4      |
| \$D7FB, G-18 | \$DD07, N-4      |
| \$D7FD, G-18 | \$DD08, N-4      |
| \$D7FE, G-18 | \$DD09, N-4      |
| \$DC00, N-3  | \$DD0A, N-4      |
| \$DC01, N-3  | \$DD0B, N-4      |
| \$DC02, N-3  | \$DD0C, N-4      |
| \$DC03, N-3  | \$DD0D, N-5      |
| \$DC04, N-3  | \$DD0E, N-5      |
| \$DC05, N-3  | \$DD0F, N-5      |
| \$DC06, N-3  | \$DD10, N-7      |

|                     |                         |
|---------------------|-------------------------|
| \$DD11, N-7         | 53272, M-32             |
| \$DD12, N-7         | 53273, M-32             |
| \$DD13, N-7         | 53274, M-32             |
| \$DD14, N-7         | 53275, M-32             |
| \$DD15, N-7         | 53276, M-32             |
| \$DD16, N-7         | 53277, M-32             |
| \$DD17, N-7         | 53278, M-32             |
| \$DD18, N-7         | 53279, M-32             |
| \$DD19, N-7         | 53280, M-32, M-35, M-38 |
| \$DD1A, N-7         | 53281, M-32, M-35, M-38 |
| \$DD1B, N-7         | 53282, M-32, M-35, M-38 |
| \$DD1C, N-7         | 53283, M-32, M-35, M-38 |
| \$DD1D, N-7         | 53284, M-33, M-35, M-38 |
| \$DD1E, N-7         | 53285, M-33, M-35, M-38 |
| \$DD1F, N-7         | 53286, M-33, M-35, M-39 |
| 53504 - 53759, M-36 | 53287, M-33             |
| 53760 - 54015, M-36 | 53288, M-33             |
| 54016 - 54271, M-36 | 53289, M-33             |
| 53248, M-32         | 53290, M-33             |
| 53249, M-32         | 53291, M-33             |
| 53250, M-32         | 53292, M-33             |
| 53251, M-32         | 53293, M-33             |
| 53252, M-32         | 53294, M-33             |
| 53253, M-32         | 53295, M-35, M-39       |
| 53254, M-32         | 53296, M-33, M-35       |
| 53255, M-32         | 53297, M-35             |
| 53256, M-32         | 53299, M-35             |
| 53257, M-32         | 53300, M-35             |
| 53258, M-32         | 53301, M-35             |
| 53259, M-32         | 53302, M-36             |
| 53260, M-32         | 53303, M-36             |
| 53261, M-32         | 53304, M-36             |
| 53262, M-32         | 53305, M-36             |
| 53263, M-32         | 53306, M-36             |
| 53264, M-32         | 53307, M-36             |
| 53265, M-32         | 53308, M-36             |
| 53266, M-32         | 53309, M-36             |
| 53267, M-32         | 53310, M-36             |
| 53268, M-32         | 53311, M-36             |
| 53269, M-32         | 53312, M-36             |
| 53270, M-32         | 53313, M-36             |
| 53271, M-32         | 53314, M-36             |

|             |             |
|-------------|-------------|
| 53315, M-36 | 53359, M-40 |
| 53316, M-36 | 53360, M-40 |
| 53317, M-36 | 53361, M-40 |
| 53318, M-36 | 53362, M-40 |
| 53319, M-36 | 53363, M-40 |
| 53320, M-39 | 53364, M-40 |
| 53321, M-39 | 53365, M-40 |
| 53322, M-39 | 53366, M-40 |
| 53323, M-39 | 53367, M-40 |
| 53324, M-39 | 53368, M-40 |
| 53325, M-39 | 53369, M-40 |
| 53326, M-39 | 53370, M-40 |
| 53327, M-39 | 53371, M-40 |
| 53328, M-39 | 53372, M-40 |
| 53329, M-39 | 53376, Q-9  |
| 53330, M-39 | 53377, Q-9  |
| 53331, M-39 | 53378, Q-9  |
| 53332, M-39 | 53379, Q-9  |
| 53333, M-39 | 53380, Q-9  |
| 53334, M-39 | 53381, Q-9  |
| 53335, M-39 | 53382, Q-9  |
| 53336, M-39 | 53383, Q-9  |
| 53337, M-39 | 53384, Q-9  |
| 53338, M-39 | 53385, Q-9  |
| 53339, M-39 | 53386, Q-9  |
| 53340, M-39 | 54784, O-3  |
| 53341, M-39 | 54785, O-3  |
| 53342, M-39 | 54786, O-3  |
| 53343, M-39 | 54787, O-3  |
| 53344, M-39 | 54788, O-3  |
| 53345, M-39 | 54789, O-3  |
| 53346, M-39 | 54790, O-3  |
| 53347, M-39 | 54793, O-4  |
| 53348, M-39 | 54795, O-4  |
| 53349, M-39 | 54796, O-4  |
| 53352, M-39 | 54797, O-4  |
| 53353, M-39 | 54798, O-4  |
| 53354, M-39 | 54799, O-4  |
| 53355, M-39 | 54800, O-4  |
| 53356, M-39 | 54801, O-4  |
| 53357, M-39 | 54802, O-4  |
| 53358, M-40 | 54805, O-4  |

|                   |                   |
|-------------------|-------------------|
| 54806, O-4        | 54875, G-17       |
| 54807, O-4        | 54876, G-17       |
| 54808, O-4        | 54877, G-17       |
| 54809, O-4        | 54878, G-17       |
| 54810, O-4        | 54879, G-17       |
| 54813, O-4        | 54880, G-17       |
| 54814, O-4        | 54881, G-17       |
| 54816, O-4        | 54882, G-17       |
| 54817, O-4        | 54883, G-17       |
| 54818, O-4        | 54884, G-17       |
| 54819, O-4        | 54885, G-17       |
| 54821, O-4        | 54886, G-17       |
| 54822, O-4        | 54887, G-18       |
| 54823, O-4        | 54888, G-18       |
| 54824, O-4        | 54889, G-18       |
| 54848, G-17, G-33 | 54890, G-18       |
| 54849, G-17, G-33 | 54891, G-18       |
| 54850, G-17       | 54892, G-18       |
| 54851, G-17, G-33 | 54893, G-18       |
| 54852, G-17, G-33 | 54894, G-18       |
| 54853, G-17, G-33 | 54895, G-18       |
| 54854, G-17, G-33 | 54896, G-18, G-34 |
| 54855, G-17, G-33 | 54897, G-18, G-34 |
| 54856, G-17, G-34 | 54898, G-18, G-34 |
| 54857, G-17, G-34 | 54899, G-18       |
| 54858, G-17, G-34 | 54900, G-18       |
| 54859, G-17, G-34 | 54901, G-18       |
| 54860, G-17, G-34 | 54902, G-18       |
| 54861, G-17, G-34 | 54903, G-18       |
| 54862, G-17, G-34 | 54904, G-18       |
| 54863, G-17, G-34 | 54905, G-18       |
| 54864, G-17, G-34 | 54906, G-18       |
| 54865, G-17, G-34 | 54907, G-18       |
| 54866, G-17, G-34 | 54908, G-18, G-34 |
| 54867, G-17, G-34 | 54909, G-18, G-34 |
| 54868, G-17, G-34 | 54910, G-18, G-34 |
| 54869, G-17, G-34 | 54911, G-18, G-34 |
| 54870, G-17, G-34 | 54912, Q-17       |
| 54871, G-17, G-34 | 54913, Q-17       |
| 54872, G-17, G-34 | 54914, Q-17       |
| 54873, G-17, G-34 | 54915, Q-17       |
| 54874, G-17       | 54916, Q-17       |

|             |                   |
|-------------|-------------------|
| 54918, Q-17 | 55020, P-9        |
| 54922, Q-17 | 55021, P-9        |
| 54923, Q-12 | 55022, P-9        |
| 54924, Q-13 | 55028, Q-23       |
| 54925, Q-13 | 55029, Q-23       |
| 54926, Q-13 | 55032, Q-23       |
| 54927, Q-13 | 55033, Q-23       |
| 54928, Q-13 | 55034, Q-23       |
| 54929, Q-13 | 55035, Q-23       |
| 54930, Q-13 | 55036, Q-23       |
| 54931, Q-13 | 55037, Q-23       |
| 54944, Q-11 | 55040, L-20       |
| 54945, Q-13 | 55041, L-20       |
| 54946, Q-11 | 55042, L-20       |
| 54959, Q-17 | 55043, L-20       |
| 54960, Q-19 | 55044, L-20       |
| 54961, Q-19 | 55045, L-20       |
| 54962, Q-19 | 55054, L-20       |
| 54963, Q-19 | 55055, G-22       |
| 54964, Q-19 | 55056, G-18       |
| 54965, Q-19 | 55057, L-20, Q-23 |
| 54967, Q-19 | 55068, L-20       |
| 54968, Q-19 | 55069, L-20       |
| 54969, Q-19 | 55070, L-20       |
| 54970, Q-19 | 55071, L-20       |
| 54971, Q-19 | 55072, L-20       |
| 54972, Q-19 | 55073, L-20       |
| 54973, Q-19 | 55074, L-20       |
| 54974, Q-19 | 55075, L-20       |
| 54976, Q-19 | 55076, L-20       |
| 55008, P-9  | 55077, L-20       |
| 55009, P-9  | 55078, L-20       |
| 55010, P-9  | 55079, L-20       |
| 55011, P-9  | 55080, L-20       |
| 55012, P-9  | 55081, L-20       |
| 55013, P-9  | 55082, L-20       |
| 55014, P-9  | 55083, L-20       |
| 55015, P-9  | 55084, L-20       |
| 55016, P-9  | 55085, L-20       |
| 55017, P-9  | 55086, L-20       |
| 55018, P-9  | 55087, L-20       |
| 55019, P-9  | 55088, L-21       |

|             |             |
|-------------|-------------|
| 55089, L-21 | 55131, L-22 |
| 55090, L-21 | 55132, L-22 |
| 55091, L-21 | 55133, L-22 |
| 55092, L-21 | 55134, L-22 |
| 55093, L-21 | 55135, L-22 |
| 55094, L-21 | 55144, G-22 |
| 55095, L-21 | 55145, G-22 |
| 55096, L-21 | 55146, G-22 |
| 55097, L-21 | 55147, G-22 |
| 55098, L-21 | 55148, G-22 |
| 55099, L-21 | 55149, G-22 |
| 55100, L-21 | 55150, G-22 |
| 55101, L-21 | 55151, G-22 |
| 55102, L-21 | 55152, G-22 |
| 55103, L-21 | 55153, G-22 |
| 55104, L-21 | 55154, G-22 |
| 55105, L-21 | 55155, G-22 |
| 55106, L-21 | 55156, G-22 |
| 55107, L-21 | 55157, G-22 |
| 55108, L-21 | 55158, G-22 |
| 55109, L-21 | 55159, G-22 |
| 55110, L-21 | 55160, G-22 |
| 55111, L-21 | 55161, G-22 |
| 55112, L-21 | 55162, G-22 |
| 55113, L-21 | 55163, G-22 |
| 55114, L-21 | 55164, G-22 |
| 55115, L-21 | 55165, G-22 |
| 55116, L-21 | 55166, G-22 |
| 55117, L-21 | 55167, G-22 |
| 55118, L-21 | 55168, G-22 |
| 55119, L-21 | 55169, G-22 |
| 55120, L-21 | 55170, G-22 |
| 55121, L-21 | 55171, G-22 |
| 55122, L-21 | 55172, G-22 |
| 55123, L-21 | 55173, G-22 |
| 55124, L-21 | 55174, G-22 |
| 55125, L-21 | 55175, G-22 |
| 55126, L-22 | 55176, G-22 |
| 55127, L-22 | 55177, G-22 |
| 55128, L-22 | 55178, G-22 |
| 55129, L-22 | 55179, G-22 |
| 55130, L-22 | 55180, G-22 |

|             |             |
|-------------|-------------|
| 55181, G-22 | 55223, G-24 |
| 55182, G-23 | 55224, G-24 |
| 55183, G-23 | 55225, G-24 |
| 55184, G-23 | 55226, G-24 |
| 55185, G-23 | 55227, G-24 |
| 55186, G-23 | 55228, G-24 |
| 55187, G-23 | 55229, G-24 |
| 55188, G-23 | 55230, G-24 |
| 55189, G-23 | 55231, G-24 |
| 55190, G-23 | 55232, G-24 |
| 55191, G-23 | 55233, G-24 |
| 55192, G-23 | 55234, G-24 |
| 55193, G-23 | 55235, G-24 |
| 55194, G-23 | 55236, G-24 |
| 55195, G-23 | 55237, G-24 |
| 55196, G-23 | 55238, G-24 |
| 55197, G-23 | 55239, G-24 |
| 55198, G-23 | 55240, G-24 |
| 55199, G-23 | 55241, G-24 |
| 55200, G-23 | 55242, G-24 |
| 55201, G-23 | 55243, G-24 |
| 55202, G-23 | 55244, G-24 |
| 55203, G-23 | 55245, G-24 |
| 55204, G-23 | 55246, G-24 |
| 55205, G-23 | 55247, G-24 |
| 55206, G-23 | 55248, G-24 |
| 55207, G-23 | 55249, G-24 |
| 55208, G-23 | 55250, G-24 |
| 55209, G-23 | 55251, G-24 |
| 55210, G-23 | 55252, G-24 |
| 55211, G-23 | 55253, G-24 |
| 55212, G-23 | 55254, G-24 |
| 55213, G-23 | 55255, G-24 |
| 55214, G-23 | 55256, G-24 |
| 55215, G-23 | 55257, G-24 |
| 55216, G-23 | 55258, G-25 |
| 55217, G-23 | 55259, G-25 |
| 55218, G-23 | 55260, G-25 |
| 55219, G-23 | 55261, G-25 |
| 55220, G-24 | 55262, G-25 |
| 55221, G-24 | 55263, G-25 |
| 55222, G-24 | 55264, G-25 |

|                 |                     |
|-----------------|---------------------|
| 55265, G-25     | 56579, N-4          |
| 55266, G-25     | 56580, N-4          |
| 55267, G-25     | 56581, N-4          |
| 55290, G-18     | 56582, N-4          |
| 55291, G-18     | 56583, N-4          |
| 55293, G-18     | 56584, N-4          |
| 55294, G-18     | 56585, N-4          |
| 56320, N-3      | 56586, N-4          |
| 56321, N-3      | 56587, N-4          |
| 56322, N-3      | 56588, N-4          |
| 56323, N-3      | 56589, N-5          |
| 56324, N-3      | 56590, N-5          |
| 56325, N-3      | 56591, N-5          |
| 56326, N-3      | 56592, N-7          |
| 56327, N-3      | 56593, N-7          |
| 56328, N-3      | 56594, N-7          |
| 56329, N-3      | 56595, N-7          |
| 56330, N-3      | 56596, N-7          |
| 56331, N-3      | 56597, N-7          |
| 56332, N-3      | 56598, N-7          |
| 56333, N-3      | 56599, N-7          |
| 56334, N-3      | 56600, N-7          |
| 56335, N-3, N-4 | 56601, N-7          |
| 56336, N-6      | 56602, N-7          |
| 56337, N-6      | 56603, N-7          |
| 56338, N-6      | 56604, N-7          |
| 56339, N-6      | 56605, N-7          |
| 56340, N-6      | 56606, N-7          |
| 56341, N-6      | 56607, N-7          |
| 56342, N-6      | ABTPALSEL, M-40     |
| 56343, N-6      | ACCESSKEY, O-4      |
| 56344, N-6      | ADDRBANK, L-20      |
| 56345, N-6      | ADDRLSB, L-20, L-22 |
| 56346, N-6      | ADDRLSBTRIG, L-20   |
| 56347, N-6      | ADDRMB, L-20, L-22  |
| 56348, N-6      | ADDRMSB, L-20       |
| 56349, N-6      | ALGO, Q-9           |
| 56350, N-6      | ALPHADELAY, M-40    |
| 56351, N-6      | ALPHEN, M-40        |
| 56576, N-4      | ALRM, N-3, N-5      |
| 56577, N-4      | ALRMAMPM, N-6, N-7  |
| 56578, N-4      | ALRMHOUR, N-6, N-7  |

ALRMJIF, N-6, N-7  
ALRMMIN, N-6, N-7  
ALRMSEC, N-6-N-8  
ALT, Q-9  
ASCFAST, G-34  
ASCIKEY, O-4, O-5  
ATTR, M-36  
AUDBLKTO, L-20, L-22  
AUDEN, L-22  
AUDWRBLK, L-22  
B0ADEVN, M-35, M-36  
B0ADODD, M-35, M-36  
B0PIX, M-36  
B1ADEVN, M-35, M-36  
B1ADODD, M-35, M-36  
B1PIX, M-36  
B2ADEVN, M-35, M-36  
B2ADODD, M-35, M-37  
B2PIX, M-36, M-37  
B3ADEVN, M-36, M-37  
B3ADODD, M-36, M-37  
B3PIX, M-36, M-37  
B4ADEVN, M-36, M-37  
B4ADODD, M-36, M-37  
B4PIX, M-36, M-37  
B5ADEVN, M-36, M-37  
B5ADODD, M-36, M-37  
B5PIX, M-36, M-37  
B6ADEVN, M-36, M-37  
B6ADODD, M-36, M-37  
B6PIX, M-36, M-37  
B7ADEVN, M-36, M-37  
B7ADODD, M-36, M-37  
B7PIX, M-36, M-37  
BADEXTRA, G-18  
BASHDDR, O-4, O-5  
BBDRPOS, M-39, M-40  
BCST, P-9  
BITPBANK, M-40  
BLKD, L-22  
BLNK, M-33  
BMM, M-33  
BORDERCOL, M-32, M-33, M-35,  
M-37, M-38, M-40  
BP16ENS, M-40  
BPCOMP, M-36, M-37  
BPM, M-37  
BPX, M-36, M-37  
BPY, M-36, M-37  
BRCOST, G-18  
BSP, M-32, M-33  
BTPALSEL, M-40  
BUSY, Q-9  
C128FAST, M-33  
CALCEN, G-25  
CALXDELTALSB, Q-19  
CALXSCALELSB, Q-19, Q-20  
CALXSCALEMSB, Q-19, Q-20  
CALYDELTALSB, Q-19, Q-20  
CALYDELTAMSB, Q-19, Q-20  
CALYSCALELSB, Q-19, Q-20  
CALYSCALEMSB, Q-19, Q-20  
CARTEN, G-18  
CB, M-32, M-33  
CDC00, Q-17  
CH0BADDR, L-20, L-22  
CH0CURADDR, L-20, L-22  
CH0EN, L-22  
CH0FREQ, L-20, L-22  
CH0LOOP, L-22  
CH0RVOL, L-20, L-22  
CH0SBITS, L-20, L-22  
CH0SGN, L-22  
CH0SINE, L-22  
CH0STP, L-22  
CH0TADDR, L-20, L-22  
CH0TMRADDR, L-20, L-22  
CH0VOLUME, L-20, L-23  
CH1BADDR, L-21, L-23  
CH1CURADDR, L-21, L-23  
CH1EN, L-23  
CH1FREQ, L-21, L-23  
CH1LOOP, L-23  
CH1RVOL, L-20, L-23

CH1SBITS, L-21, L-23  
CH1SGN, L-23  
CH1SINE, L-23  
CH1STP, L-23  
CH1TADDR, L-21, L-23  
CH1TMRADDR, L-21, L-23  
CH1VOLUME, L-21, L-23  
CH2BADDR, L-21, L-23  
CH2CURADDR, L-21, L-23  
CH2EN, L-23  
CH2FREQ, L-21, L-23  
CH2LOOP, L-23  
CH2LVOL, L-20, L-23  
CH2SGN, L-23  
CH2SINE, L-23  
CH2STP, L-23  
CH2TADDR, L-21, L-23  
CH2TMRADDR, L-21, L-23  
CH2VOLUME, L-21, L-23  
CH3BADDR, L-21, L-23  
CH3CURADDR, L-22, L-23  
CH3EN, L-24  
CH3FREQ, L-21, L-22, L-24  
CH3LOOP, L-24  
CH3LVOL, L-20, L-24  
CH3SBITS, L-21, L-24  
CH3SGN, L-24  
CH3SINE, L-24  
CH3STP, L-24  
CH3TADDR, L-22, L-24  
CH3TMRADDR, L-22, L-24  
CH3VOLUME, L-22, L-24  
CHARPTR, M-39, M-40  
CHARSZ, O-3  
CHR16, M-40  
CHRCOUNT, M-39, M-40  
CHRXSCL, M-39, M-40  
CHRYSCL, M-39, M-40  
CLOCK, Q-9, Q-10  
CMDANDSTAT, Q-17  
COLPTR, M-39, M-40  
COMMAND, P-9, Q-10  
CONN41, O-5  
CPUFAST, G-34  
CRAM2K, M-37  
CRC, Q-10  
CROM9, M-37  
CSEL, M-33  
D0IMG, Q-13  
D0MD, Q-13  
D0P, Q-13  
D0STARTSEC0, Q-13  
D0STARTSEC1, Q-13  
D0STARTSEC2, Q-13  
D0STARTSEC3, Q-13  
D0WP, Q-13  
D1IMG, Q-13  
D1MD, Q-13  
D1P, Q-13  
D1STARTSEC1, Q-13  
D1WP, Q-13  
D2STARTSEC2, Q-13  
D3STARTSEC3, Q-13  
DATA, O-3, Q-9, Q-10  
DATARATE, Q-11  
DBGDIR, Q-11  
DBGMOTORA, Q-11  
DBGWDATA, Q-11  
DBGWGATE, Q-11  
DBLRR, M-41  
DD00DELAY, N-8  
DDRA, N-3-N-5  
DDRБ, N-3-N-5  
DEBUGC, M-40, M-41  
DENSITY, Q-11  
DIGILEFTLSB, Q-23  
DIGILEFTMSB, Q-23  
DIGILLSB, Q-23  
DIGILMSB, Q-23  
DIGIRIGHTLSB, Q-23  
DIGIRIGHTMSB, Q-23  
DIGIRLSB, Q-23  
DIGIRMSB, Q-23  
DIR, Q-10

DISKIN, Q-10  
DIVBUSY, G-25  
DIVISOR, O-3  
DIVOUT, G-22, G-25  
DMADSTMB, G-34  
DMAADDR, G-34  
DMASRCMB, G-34  
DRQ, Q-10  
DRXD, P-9  
DRXDV, P-9  
DS, Q-9, Q-10  
DSKCHG, Q-10  
ECM, M-33  
EN018B, L-24  
ENTEREXIT, G-34  
EQ, Q-10  
ETRIG, L-20, L-24  
EV1, Q-20  
EV2, Q-20  
EXGLYPH, M-41  
EXSID, G-34  
EXTIROS, M-41  
EXTSYNC, M-37  
F4502, G-35  
FAST, M-37  
FCLRHI, M-41  
FCLRLO, M-41  
FILLVAL, Q-17  
FLG, N-3, N-5  
FNRASTER, M-39, M-41  
FNRST, M-41  
FNRSTCMP, M-41  
FRAMECOUNT, G-18  
FREE, Q-10  
FRMERR, O-3  
GEORAMBASE, G-34, G-35  
GEORAMMASK, G-34, G-35  
GESTUREDIR, Q-19, Q-20  
GESTUREID, Q-19, Q-20  
H1280, M-37  
H640, M-37  
HDSCL, O-5  
HDSDA, O-5  
HICKED, G-34, G-35  
HOTREG, M-41  
HPOS, M-36, M-38  
HSYNC, M-41  
HTRAP00, G-17, G-18  
HTRAP01, G-17, G-19  
HTRAP02, G-17, G-19  
HTRAP03, G-17, G-19  
HTRAP04, G-17, G-19  
HTRAP05, G-17, G-19  
HTRAP06, G-17, G-19  
HTRAP07, G-17, G-19  
HTRAP08, G-17, G-19  
HTRAP09, G-17, G-19  
HTRAP0A, G-17, G-19  
HTRAP0B, G-17, G-19  
HTRAP0C, G-17, G-19  
HTRAP0D, G-17, G-19  
HTRAP0E, G-17, G-19  
HTRAP0F, G-17, G-19  
HTRAP10, G-17, G-19  
HTRAP11, G-17, G-19  
HTRAP12, G-17, G-19  
HTRAP13, G-17, G-19  
HTRAP14, G-17, G-19  
HTRAP15, G-17, G-19  
HTRAP16, G-17, G-19  
HTRAP17, G-17, G-19  
HTRAP18, G-17, G-19  
HTRAP19, G-17, G-19  
HTRAP1A, G-17, G-19  
HTRAP1B, G-17, G-19  
HTRAP1C, G-17, G-19  
HTRAP1D, G-17, G-20  
HTRAP1E, G-17, G-20  
HTRAP1F, G-17, G-20  
HTRAP20, G-17, G-20  
HTRAP21, G-17, G-20  
HTRAP22, G-17, G-20  
HTRAP23, G-17, G-20  
HTRAP24, G-17, G-20

HTRAP25, G-17, G-20  
HTRAP26, G-17, G-20  
HTRAP27, G-18, G-20  
HTRAP28, G-18, G-20  
HTRAP29, G-18, G-20  
HTRAP2A, G-18, G-20  
HTRAP2B, G-18, G-20  
HTRAP2C, G-18, G-20  
HTRAP2D, G-18, G-20  
HTRAP2E, G-18, G-20  
HTRAP2F, G-18, G-20  
HTRAP30, G-18, G-20  
HTRAP31, G-18, G-20  
HTRAP32, G-18, G-20  
HTRAP33, G-18, G-20  
HTRAP34, G-18, G-20  
HTRAP35, G-18, G-20  
HTRAP36, G-18, G-20  
HTRAP37, G-18, G-20  
HTRAP38, G-18, G-20  
HTRAP39, G-18, G-21  
HTRAP3A, G-18, G-21  
HTRAP3B, G-18, G-21  
HTRAP3C, G-18, G-21  
HTRAP3D, G-18, G-21  
HTRAP3E, G-18, G-21  
HTRAP3F, G-18, G-21  
IFRXIRQ, O-3  
IFRXNMI, O-3  
IFTXIRQ, O-3  
IFTXNMI, O-3  
ILP, M-33  
IMALRM, N-7, N-8  
IMFLG, N-7, N-8  
IMODA, N-3, N-5  
IMODB, N-3, N-5  
IMRXIRQ, O-3  
IMRXNMI, O-3  
IMSP, N-7, N-8  
IMTB, N-7, N-8  
IMTXIRQ, O-3  
IMTXNMI, O-3  
INDEX, Q-10  
INT, M-38  
IR, N-3  
IRQ, Q-10  
ISBC, M-33  
ISSC, M-33  
J21H, O-4, O-5  
J21HDDR, O-4, O-5  
J21L, O-4, O-5  
J21LDDR, O-4, O-5  
JMP32EN, G-35  
JOYSWAP, O-5  
KEY, M-35, M-38, M-39, M-41  
Keyboard, O-4, O-5  
KEYEN, P-9  
KEYLEFT, O-5  
KEYUP, O-5  
KSCNRATE, O-4, O-5  
LATCHINT, G-25  
LED, Q-10  
LINESTEP, M-39, M-41  
LJOYA, O-5  
LJOYB, O-5  
LOAD, N-3, N-5  
LOST, Q-10  
LPX, M-32, M-33  
LPY, M-32, M-33  
MACADDR1, P-9  
MACADDR2, P-9  
MACADDR3, P-9  
MACADDR4, P-9  
MACADDR5, P-9, P-10  
MACADDR6, P-9, P-10  
MALT, O-5  
MAPEDPAL, M-40, M-41  
MAPHI, G-34, G-35  
MAPHIMB, G-34, G-35  
MAPLO, G-34, G-35  
MAPLOMB, G-34, G-35  
MATHIN0, G-22, G-25  
MATHIN1, G-22, G-25  
MATHIN10, G-23, G-25

MATHIN11, G-23, G-25  
MATHIN12, G-23, G-25  
MATHIN13, G-24, G-25  
MATHIN14, G-24, G-25  
MATHIN15, G-24, G-25  
MATHIN2, G-22, G-25  
MATHIN3, G-22, G-23, G-25  
MATHIN4, G-23, G-25  
MATHIN5, G-23, G-25  
MATHIN6, G-23, G-25  
MATHIN7, G-23, G-25  
MATHIN8, G-23, G-25  
MATHIN9, G-23, G-26  
MATRIXEN, G-35  
MC1, M-32, M-33, M-35, M-38,  
    M-41  
MC2, M-32, M-33, M-35, M-38,  
    M-41  
MC3, M-33, M-35, M-38, M-41  
MCAPS, O-5  
MCM, M-33  
MCST, P-10  
MCTRL, O-5  
MIIMPHY, P-9, P-10  
MIIMREG, P-9, P-10  
MIIMVLSB, P-9, P-10  
MIIMVMSB, P-9, P-10  
MISBC, M-33  
MISSC, M-34  
MIXREGDATA, Q-23  
MIXREGSEL, Q-23  
MLSHFT, O-5  
MMEGA, O-5  
MONO, M-38  
MOTOR, Q-10  
MRIRQ, M-34  
MRSHFT, O-5  
MSCRL, O-5  
MULBUSY, G-26  
MULTINA, G-22, G-26  
MULTINB, G-22, G-26  
MULTOUT, G-22, G-26  
NOBUF, Q-10  
NOCRC, P-10  
NOEXROM, G-21  
NOGAME, G-21  
NOMIX, L-24  
NOPROM, P-10  
NORRDEL, M-41  
Number, M-40, M-41  
OCEANA, G-21  
OMODA, N-3, N-5  
OMODB, N-3, N-5  
OSKALT, O-5  
OSKDEBUG, O-5  
OSKDIM, O-5  
OSKEN, O-5  
OSKTOP, O-6  
OSKZEN, O-6  
OSKZON, O-6  
PAL, M-38  
PALBLUE, M-36, M-38  
PALEMU, M-41  
PALGREEN, M-36, M-38  
PALNTSC, M-41  
PALRED, M-36, M-38  
PBONA, N-3, N-5  
PBONB, N-3, N-5  
PCH, G-34, G-35  
PCL, G-34, G-35  
PCODE, Q-9, Q-10  
PFLAGS, G-33, G-35  
PIRQ, G-35  
PNMI, G-35  
PORT00, G-34, G-35  
PORT01, G-34, G-35  
PORTA, N-3-N-5  
PORTB, N-3-N-5  
PORTF, O-4, O-6  
PORTFDDR, O-4, O-6  
POTAX, O-4, O-6  
POTAY, O-4, O-6  
POTBX, O-4, O-6  
POTBY, O-4, O-6

POWEREN, G-21  
PREFETCH, G-21  
PROT, Q-10  
PTYEN, O-3  
PTYERR, O-3  
PTYEVEN, O-3  
PWMPDM, Q-23  
RASLINE0, M-40, M-41  
RASTERHEIGHT, M-40, M-41  
RC, M-32, M-34  
RDCMD, Q-10  
RDREQ, Q-10  
READBACKLSB, Q-23  
READBACKMSB, Q-23  
REALHW, O-6  
REGA, G-33, G-35  
REGB, G-33, G-35  
REGX, G-33, G-35  
REGZ, G-33, G-35  
RESERVED, G-25, G-26, M-40,  
    M-41  
IRQ, M-34  
RMODA, N-4, N-5  
RMODB, N-4, N-5  
RNF, Q-10  
ROM8, M-38  
ROMA, M-38  
ROMC, M-38  
ROME, M-38  
ROMPROT, G-35  
RSEL, M-34  
RST, M-34, P-10  
RST41, O-6  
RSTCMP, M-40, M-41  
RSTCOMP, M-40, M-41  
RSTDELEN, M-42  
RSVD, G-35  
RUN, Q-11  
RXBF, P-9, P-10  
RXBLKD, P-10  
RXEN, O-3  
RXOVRRUN, O-3  
RXPH, P-9, P-10  
RXQ, P-10  
RXQEN, P-10  
RXRDY, O-3  
S0X, M-32, M-34  
S0Y, M-32, M-34  
S1X, M-32, M-34  
S1Y, M-32, M-34  
S2X, M-32, M-34  
S2Y, M-32, M-34  
S3X, M-32, M-34  
S3Y, M-32, M-34  
S4X, M-32, M-34  
S4Y, M-32, M-34  
S5X, M-32, M-34  
S5Y, M-32, M-34  
S6X, M-32, M-34  
S6Y, M-32, M-34  
S7X, M-32, M-34  
S7Y, M-32, M-34  
SBC, M-32, M-34  
SCM, M-32, M-34  
SCREENCOL, M-32, M-34, M-35,  
    M-38, M-42  
SCRNPTR, M-39, M-42  
SDBSH, O-6  
SDCLK, O-6  
SDCS, O-6  
SDDATA, O-6  
SDR, N-3-N-5  
SE, M-32, M-34  
SECTOR, Q-9, Q-11  
SECTOR0, Q-17  
SECTOR1, Q-17  
SECTOR2, Q-17  
SECTOR3, Q-17  
SEXX, M-32, M-34  
SEXY, M-32, M-34  
SHDEMU, M-42  
SIDBDRWD, M-39, M-42  
SIDE, Q-9, Q-11  
SMTH, M-42

SP, N-4, N-5  
SPH, G-33, G-35  
SPL, G-33, G-35  
SPMOD, N-4, N-5  
SPR0COL, M-33, M-35  
SPR16EN, M-39, M-42  
SPR1COL, M-33, M-35  
SPR2COL, M-33, M-35  
SPR3COL, M-33, M-35  
SPR4COL, M-33, M-35  
SPR5COL, M-33, M-35  
SPR640, M-42  
SPR6COL, M-33, M-35  
SPR7COL, M-33, M-35  
SPRALPHAVAL, M-40, M-42  
SPRBPMEN, M-39, M-42  
SPRENALPHA, M-40, M-42  
SPRENV400, M-40, M-42  
SPRHGHT, M-39, M-42  
SPRHGTEN, M-39, M-42  
SPRMCO, M-33, M-35, M-38,  
    M-42  
SPRMCI, M-33, M-35, M-38,  
    M-39, M-42  
SPRPALSEL, M-40, M-42  
SPRPT16, M-42  
SPRPTRADR, M-39, M-42  
SPRPTRBNK, M-40, M-42  
SPRTILEN, M-39, M-42  
SPRX64EN, M-39, M-42  
SPRXSMSBS, M-39, M-42  
SPRYADJ, M-40, M-42  
SPRYSMSBS, M-40, M-42  
SPTRCONT, M-42  
SRPYMSBS, M-40, M-43  
SSC, M-32, M-35  
STEP, Q-9, Q-11  
STRM, P-10  
STRTA, N-4, N-5  
STRTB, N-4, N-5  
SWAP, Q-11  
SXMSB, M-32, M-35  
SYNCMOD, O-3, O-4  
SYSCTL, O-4, O-6  
TA, N-4, N-5  
TALATCH, N-6-N-8  
TARGANY, Q-13  
TB, N-4, N-5  
TBDRPOS, M-39, M-43  
TEXTXPOS, M-39, M-43  
TEXTYPOS, M-39, M-43  
TIMERA, N-3-N-5  
TIMERB, N-3-N-5  
TK0, Q-11  
TOD50, N-4, N-6  
TODAMPM, N-4, N-6-N-8  
TODEDIT, N-6  
TODHOUR, N-3, N-4, N-6-N-8  
TODJIF, N-3, N-4, N-6-N-8  
TODMIN, N-6-N-8  
TODSEC, N-3, N-4, N-6-N-8  
TOUCH1LSB, Q-19, Q-20  
TOUCH1XMSB, Q-19, Q-20  
TOUCH1YLSB, Q-19, Q-20  
TOUCH1YMSB, Q-19, Q-20  
TOUCH2XLSB, Q-19, Q-20  
TOUCH2YMSB, Q-19, Q-20  
TRACK, Q-9, Q-11  
TXEN, O-4  
TXIDLE, P-10  
TXPH, P-10  
TXQ, P-10  
TXQEN, P-10  
TXRST, P-10  
TXSZLSB, P-9, P-10  
TXSZMSB, P-9, P-10  
UARTDATA, G-34, G-35  
UFAST, O-6  
UNIT0INA, G-24, G-26  
UNIT0INB, G-24, G-26  
UNIT0OUT, G-24, G-26  
UNIT10INA, G-24, G-26

UNIT10INB, G-24, G-26  
UNIT10OUT, G-25, G-26  
UNIT11INA, G-24, G-26  
UNIT11INB, G-24, G-26  
UNIT11OUT, G-25, G-26  
UNIT12INA, G-24, G-26  
UNIT12INB, G-24, G-26  
UNIT12OUT, G-25, G-26  
UNIT13INA, G-24, G-26  
UNIT13INB, G-24, G-27  
UNIT13OUT, G-25, G-27  
UNIT14INA, G-24, G-27  
UNIT14INB, G-24, G-27  
UNIT14OUT, G-25, G-27  
UNIT15INA, G-24, G-27  
UNIT15INB, G-24, G-27  
UNIT15OUT, G-25, G-27  
UNIT1INA, G-24, G-27  
UNIT1INB, G-24, G-27  
UNIT1OUT, G-24, G-27  
UNIT2INA, G-24, G-27  
UNIT2INB, G-24, G-27  
UNIT2OUT, G-24, G-27  
UNIT3INA, G-24, G-27  
UNIT3INB, G-24, G-27  
UNIT3OUT, G-24, G-27  
UNIT4INA, G-24, G-28  
UNIT4INB, G-24, G-28  
UNIT4OUT, G-24, G-28  
UNIT5INA, G-24, G-28  
UNIT5INB, G-24, G-28  
UNIT5OUT, G-24, G-28  
UNIT6INA, G-24, G-28  
UNIT6INB, G-24, G-28  
UNIT6OUT, G-24, G-28  
UNIT7INA, G-24, G-28  
UNIT7INB, G-24, G-28  
UNIT7OUT, G-24, G-28  
UNIT8INA, G-24, G-28  
UNIT8INB, G-24, G-28  
UNIT8OUT, G-24, G-28  
UNIT9INA, G-24, G-28  
UNIT9INB, G-24, G-28  
UNIT9OUT, G-24, G-29  
UNUSED, O-4, O-6  
UPDN1, Q-19, Q-20  
UPDN2, Q-19, Q-20  
USEREAL0, Q-13  
USEREAL1, Q-14  
V400, M-38  
VDRO, Q-18  
VEQINH, Q-18  
VFAST, M-43  
VFDC0, Q-18  
VFDC1, Q-18  
VFLOP, G-36  
VGAHDTV, M-43  
VICIII, Q-18  
VICMODE, G-34, G-36  
VIRTKEY1, O-4, O-6  
VIRTKEY2, O-4, O-6  
VIRTKEY3, O-4, O-6  
VLOST, Q-18  
VPOS, M-36, M-38  
VRFOUND, Q-18  
VRNF, Q-18  
VS, M-32, M-35  
VSYNCP, M-43  
VWFOUND, Q-18  
WATCHDOG, G-34, G-36  
WGATE, Q-11  
WRCMD, Q-11  
WREN, G-29  
WTREQ, Q-11  
XINV, Q-20  
XPOS, M-39, M-43  
XSCL, M-32, M-35  
YINV, Q-20  
YSCL, M-32, M-35  
relational operators, 7-30  
REM, B-163  
RENAME, B-164  
RENUMBER, 7-38, B-165  
RESQ, H-128

RESTORE, B-166  
RESUME, B-167  
RETURN, B-168  
revers, 15-8  
RGRAPHIC, B-169  
RIGHT\$, B-170  
RLA, H-43  
RMB0, H-97  
RMB1, H-97  
RMB2, H-98  
RMB3, H-98  
RMB4, H-99  
RMB5, H-99  
RMB6, H-99  
RMB7, H-100  
RMOUSE, B-171  
RND, B-172  
RND(), 7-45  
ROL, H-43, H-100  
ROLQ, G-14, H-129  
ROR, H-44, H-101  
RORQ, G-14, H-130  
ROW, H-101  
RPEN, B-173  
RRA, H-45  
RREG, B-174  
RSPCOLOR, B-175  
RSPPOS, B-176  
RSprite, B-177  
RSVQ, H-131  
RTI, H-45, H-102  
RTS, H-46, H-102  
RUN, B-178  
RWINDOW, B-179  
RXNORMAL, P-11  
RXONLYONE, P-11  
SAVE, B-180  
SAX, H-46  
SBC, H-47, H-48, H-103, H-132  
SBCQ, G-14, H-133  
SBX, H-48  
scientific notation, 7-45  
SCNCLR, B-181  
SCRATCH, B-182  
SCREEN, B-183  
SEC, H-49, H-103  
SED, H-49, H-104  
SEE, H-104  
SEI, H-49, H-105  
SET, B-186  
set16bitcharmode, 15-6  
setcharsetaddr, 15-5  
setcolramoffset, 15-4  
setextendedattrib, 15-7  
sethotregs, 15-6  
SETMEMORY, K-19  
SETPC, K-17  
setscreenaddr, 15-4  
setscreensize, 15-6  
SGN, B-187  
SHA, H-50  
SHX, H-50  
SHY, H-51  
SIN, B-188  
SLEEP, B-189  
SLO, H-51  
SLOW, B-190  
SMB0, H-105  
SMB1, H-106  
SMB2, H-106  
SMB3, H-106  
SMB4, H-107  
SMB5, H-107  
SMB6, H-107  
SMB7, H-108  
SOUND, B-191  
SPC, B-192  
SPRCOLOR, B-193  
SPRITE, B-194  
SPRSAV, B-195  
SQR, B-196  
SRE, H-52  
ST, B-197  
STA, H-52, H-108, H-134

STARTTX, P-11  
STEP, 7-13, B-198  
STOP, 7-39, B-199  
STOPTX, P-11  
STQ, G-14, H-134  
STR\$, B-200  
string, 7-13  
STX, H-53, H-108  
STY, H-53, H-109  
STZ, H-109  
SYNTAX ERROR, 7-4  
SYS, B-201  
  
TAB, B-203, H-110  
TAN, B-204  
TAS, H-54  
TAX, H-54, H-110  
TAY, H-54, H-110  
TAZ, H-111  
TBA, H-111  
TEMPO, B-205  
textcolor, 15-8  
Texture Scaling, L-15  
THEN, 7-30, B-206  
TI, B-207  
TI\$, B-208  
TO, B-209  
togglecase, 15-7  
TRACE, K-19  
TRAP, B-210  
TRB, H-112  
TROFF, B-211  
TRON, B-212  
TSB, H-112  
TSX, H-55, H-113  
TSY, H-113  
  
TXA, H-55, H-114  
TXS, H-56, H-114  
TYA, H-56, H-115  
TYPE, B-213  
Type mismatch error, 7-17  
TYS, H-115  
TZA, H-115  
  
UARTDIVISOR, K-15  
underline, 15-9  
unequal, 7-31  
UNTIL, B-214  
USING, B-215  
USR, B-217  
Utility Menu, I-5  
  
VAL, B-218  
variable, 7-8  
    numeric, 7-15  
    string, 7-15  
VERIFY, B-219  
VIEWPORT, B-220  
vline, 15-11  
VOL, B-221  
  
WAIT, B-222  
Warnings  
    Extra Ignored, 7-22  
WATCHPOINT, K-19  
wherex, 15-14  
wherey, 15-14  
WHILE, B-223  
WINDOW, B-224  
  
XAA, H-57  
XOR, B-225

# About the MEGA65 Book

---

C64 and C65 programme and peripheral compatibility ... amazing sound ... true arcade-class graphics ... beautifully finished hardware ... full mechanical keyboard ... rich networking capabilities and one of the fastest 6502-class processors available make the MEGA65 truly unique for home, business or educational use.

**The MEGA65 Book** collects into a single huge volume all the information you need to know about your MEGA65. Collecting all of the key information of the various MEGA65 user guides and reference manuals in once place, this book provides detailed information on every topic, including BASIC programming, sound, graphics, networking, assembly language programming, cross-platform development, including using high-level languages like C or KickC, the MEGA65's powerful 8-bit chipset, and how to use its powerful FPGA core to implement other computer systems.

This book is the must-have reference for the MEGA65, that every user, whether beginner or advanced, should have with them when exploring the full potential of the MEGA65.

With authors including professional writers, university lecturers and 8-bit experts, the content is both accessible and extensive. Beginners will find detailed explanations of how to get started, while advanced users will find highly detailed technical information on the inner workings of every aspect of the MEGA65.

In short, this book is designed for you to get the most out of the MEGA65's extensive capabilities.

