A small 6502 system with MS BASIC built in a Lattice UltraPlus FPGA. This system includes the following features:
- Up to 52kB SRAM with optional write protect (using two of the four available SPRAM cores)
- 8 bits input, 8 bits output
- 115200bps serial I/O port
- NTSC color composite video with text/glyph and graphic modes, 32kB video RAM (4 8kB pages) and original OSI 2kB character ROM
- 2kB ROM for startup and I/O support
- 8kB Ohio Scientific C1P Microsoft BASIC loaded from spi flash into protected RAM
- SPI port with access to 4MB flash memory
- LED PWM driver
- PS/2 Keyboard port with tx and rx capability
- 4-voice sound generator with 1-bit sigma-delta output
To build this you will need the following FPGA tools
- Icestorm - ice40 FPGA tools
- Yosys - Synthesis
- Nextpnr - Place and Route (version newer than Mar 23 2019 is needed to support IP cores)
Info on these can be found at http://www.clifford.at/icestorm/
You will also need the following 6502 tools to build the startup ROM:
- cc65 6502 C compiler (for default option) https://github.com/cc65/cc65
git clone https://github.com/emeb/up5k_basic.git
cd up5k_basic
git submodule update --init
cd icestorm
make
I built this system on a custom up5k board and programmed it with a custom USB->SPI board that I built so you will definitely need to tweak the programming target of the Makefile in the icestorm directory to match your own hardware. Note that the 8kB BASIC ROM must now be loaded into the SPI configuration flash memory starting at offset 0x40000 in order for BASIC to run correctly. You can find a link to the ROM data at the end of this document.
You will need to connect a PS/2 keyboard to the ps2_clk/dat pins, or a 115200bps serial terminal port to the TX/RX pins of the FPGA - the data input routines can take characters from either or both. Load the bitstream an you'll see the boot prompt:
D/C/W/M?
This is asking which initial service to start.
- D - diagnostics. Currently unused, just enters an infinite loop.
- C - Cold start BASIC. This is what you'll normally want.
- W - Warm start BASIC. Use this only after BASIC has been running and the system has been reset.
- M - Machine Language Monitor.
Hit the key of choice (uppercase only!) to continue.
After hitting "C", BASIC will initialize. It will prompt you:
MEMORY SIZE?
to which you answer with 'enter' to let it use all memory. It then prompts with:
TERMINAL WIDTH?
Again, hit 'enter' to use the default. It then prints a welcome message and is ready to accept BASIC commands and code. You can find out more about how to use this version of BASIC here: https://www.pcjs.org/docs/c1pjs/
I've upgraded the LOAD and SAVE commands in BASIC from the original bare-bones features found in the OSI ROMs which were intended for simple audio tape storage. Now, LOAD/SAVE operate on "slots" of up to 32kB stored in the SPI Flash memory connected to the FPGA. Use LOAD [n] or SAVE [n] where [n] is an integer from 0-99 that refers to the memory slot in which you wish to save or load your BASIC program.
Memory slots contain raw ASCII text of the programs (un-tokenized), so you could conceivably pre-load the SPI Flash with code from an external source. Slots start at 0x050000 in the flash memory space and are spaced every 0x8000. Program text is terminated with 0xFF, so just leave unused bytes in the default erased state.
The BASIC line input routine has been patched to allow use of the the Backspace key instead of the underline character. The video text output driver has not been altered so the cursor doesn't actually back up.
C'MON is a simple hex machine language monitor for the 6502 written by Bruce Clark and placed in the public domain which allows examining and editing memory contents as well as executing machine code.
Answering "M" to the boot prompt will print a quick help header and start the C'MON monitor.
The 2kB ROM located at $f800 - $ffff contains the various reset/interrupt vectors, initialization code, the C'MON monitor and I/O routines needed to load and support BASIC.
You can revise this ROM with your own additional support code - you may need to edit the linker script to change the memory sizes. The cc65 assembler and linker are required to put it all together into the final .hex file needed by the FPGA build.
Originally this project relied on a FOSS USART coded in Verilog that I came across elsewhere on github. I discovered that there were some issues with it so it has been replaced with a design that I built myself and which seems to behave much more reliably while also using fewer resources. The serial I/O parameters are fixed at 115200bps 8/N/1 but the data rate can be easily changed over a wide range with simple tweaks to parameters in the acia.v file.
This is an NTSC color composite video generator which is a significant improvement over the original OSI Challenger 1P/Superboard II video that the system started out with. Features are:
- 32x28 text/glyph mode using the OSI 8x8 character generator
- Separate foreground / background colors per-character in text/glyph mode
- 256x224 two-color pixel graphics mode
- 4 8kB memory pages for text or graphics
The 75 ohm baseband composite video signal is synthesized by means of a 4-bit weighted-R DAC driven directly by four bits from the FPGA.
A PS/2 keyboard port is provided which (along with the ACIA) feeds the ASCII input. Host to keyboard communication is supported and the caps lock LED should toggle on/off to indicate status.
An active-low "soft" reset input will reset the 6502 system without reconfiguring the FPGA. This will return the system the the "D/C/W/M" prompt and allow a warm- start into BASIC for recovery from some situations.
The iCE40 Ultra Plus features two SPI and two I2C ports as hard IP cores that are accessible through a "system bus" that's similar to the popular Wishbone standard. I've added a 6502 to Wishbone bridge mapped to addresses $F100-$F1FF which provides access to all four cores. Currently only the SPI core at addresses $F106-$F10F is connected and it is used to read the BASIC ROM from flash into SPRAM and support LOAD and SAVE operations.
Many FPGAs in the iCE40 family provide hard IP cores for driving RGB LEDs. A simple interface to this is provided so the 6502 may control the LED driver.
A 4-voice sound generator is provided which supports pitch from 0-32kHz in roughly 0.5Hz steps, choice of waveform (saw/square/triangle/noise) and volume control on each voice. Output is via a 1-bit sigma-delta process which requires a simple 1-pole RC filter (100ohm + 0.1uf) lowpass filter to smooth the digital pulse waveform down to analog audio.
Simulation is supported and requires the following prerequisites:
- Icarus Verilog simulator http://iverilog.icarus.com/
- GTKWave waveform viewer http://gtkwave.sourceforge.net/
To simulate, use the following commands
cd icarus
make
make wave
This will build the simulation executable, run it and then view the output.
There have been questions raised about the copyright status of the MS BASIC provided in this project. To the best of my knowledge, the contents of the file src/basic_8k.hex is still property of Microsoft and is used here for educational purposes only. The full source code for this can be found at:
https://github.com/brajeshwar/Microsoft-BASIC-for-6502-Original-Source-Code-1978
The ROM files from which I created the .hex file are also available in many places - I used this archive: http://www.osiweb.org/misc/OSI600_RAM_ROM.zip
Thanks to the developers of all the tools used for this, as well as the authors of the IP cores I snagged for the 6502 and UART. I've added those as submodules so you'll know where to get them and who to give credit to.