2013-02-28 07:01:51 +00:00
|
|
|
// Simplest possible Apple II that will possibly boot. ~ (tilde) to quit.
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2016-09-10 00:49:21 +00:00
|
|
|
"flag"
|
2013-02-28 07:01:51 +00:00
|
|
|
"fmt"
|
2016-09-10 00:49:21 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2016-09-17 19:32:44 +00:00
|
|
|
"strings"
|
2013-02-28 07:01:51 +00:00
|
|
|
|
|
|
|
"github.com/nsf/termbox-go"
|
2013-03-28 04:49:38 +00:00
|
|
|
"github.com/zellyn/goapple2"
|
|
|
|
"github.com/zellyn/goapple2/util"
|
|
|
|
"github.com/zellyn/goapple2/videoscan"
|
2013-02-28 07:01:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Mapping of screen bytes to character values
|
|
|
|
var AppleChars = "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_ !\"#$%&'()*+,-./0123456789:;<=>?"
|
|
|
|
|
2016-09-10 00:49:21 +00:00
|
|
|
var ColorFG = termbox.ColorGreen
|
|
|
|
var ColorBG = termbox.ColorBlack
|
|
|
|
|
2013-02-28 07:01:51 +00:00
|
|
|
// Translate to termbox
|
|
|
|
func translateToTermbox(value byte) (char rune, fg, bg termbox.Attribute) {
|
|
|
|
// BUG(zellyn): change this to return char, MODE_ENUM.
|
|
|
|
ch := rune(AppleChars[value&0x3F])
|
|
|
|
if value&0x80 > 0 {
|
2016-09-10 00:49:21 +00:00
|
|
|
return ch, ColorFG, ColorBG
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
2016-09-10 00:49:21 +00:00
|
|
|
return ch, ColorBG, ColorFG
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func termboxToAppleKeyboard(ev termbox.Event) (key byte, err error) {
|
|
|
|
if ev.Key > 0 && ev.Key <= 32 {
|
|
|
|
return byte(ev.Key), nil
|
|
|
|
}
|
|
|
|
if ev.Ch >= '!' && ev.Ch <= 'Z' || ev.Ch == '^' {
|
|
|
|
return byte(ev.Ch), nil
|
|
|
|
}
|
|
|
|
if ev.Ch >= 'a' && ev.Ch <= 'z' {
|
|
|
|
return byte(ev.Ch - 'a' + 'A'), nil
|
|
|
|
}
|
|
|
|
switch ev.Key {
|
|
|
|
case termbox.KeyBackspace2:
|
|
|
|
return 8, nil // backspace
|
|
|
|
case termbox.KeyArrowLeft:
|
|
|
|
return 8, nil // left arrow
|
|
|
|
case termbox.KeyArrowRight:
|
|
|
|
return 21, nil // right arrow
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("hi")
|
|
|
|
}
|
|
|
|
|
2013-03-28 04:49:38 +00:00
|
|
|
func ProcessEvents(events chan termbox.Event, a2 *goapple2.Apple2) bool {
|
2013-02-28 07:01:51 +00:00
|
|
|
select {
|
2013-03-28 04:49:38 +00:00
|
|
|
case ev := <-events:
|
2013-02-28 07:01:51 +00:00
|
|
|
if ev.Type == termbox.EventKey && ev.Ch == '~' {
|
2013-03-28 04:49:38 +00:00
|
|
|
return true
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
if ev.Type == termbox.EventKey {
|
|
|
|
if key, err := termboxToAppleKeyboard(ev); err == nil {
|
2013-03-30 05:43:15 +00:00
|
|
|
a2.Keypress(key)
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
|
2013-03-28 04:49:38 +00:00
|
|
|
return false
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
|
2013-03-28 04:49:38 +00:00
|
|
|
type TextPlotter int
|
2013-02-28 07:01:51 +00:00
|
|
|
|
2013-03-28 04:49:38 +00:00
|
|
|
func (p TextPlotter) Plot(data videoscan.PlotData) {
|
|
|
|
y := int(data.Row / 8)
|
|
|
|
x := int(data.Column)
|
|
|
|
value := data.RawData
|
|
|
|
ch, fg, bg := translateToTermbox(value)
|
|
|
|
termbox.SetCell(x+1, y+1, ch, fg, bg)
|
2013-04-08 01:31:14 +00:00
|
|
|
}
|
|
|
|
func (p TextPlotter) OncePerFrame() {
|
|
|
|
termbox.Flush()
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
|
2016-09-17 19:32:44 +00:00
|
|
|
// Run the emulator. If file is not empty, load it at address 0x6000,
|
|
|
|
// add a PCAction to quit if address 0 is called, clear the screen,
|
|
|
|
// call 0x6000, call 0, and dump the screen contents (minus trailing
|
|
|
|
// whitespace).
|
2016-09-19 03:27:16 +00:00
|
|
|
func RunEmulator(file string, quit bool) error {
|
2016-09-10 00:49:21 +00:00
|
|
|
var options []goapple2.Option
|
|
|
|
if file != "" {
|
|
|
|
ColorFG = termbox.ColorDefault
|
|
|
|
ColorBG = termbox.ColorDefault
|
|
|
|
bytes, err := ioutil.ReadFile(file)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
options = append(options, goapple2.WithRAM(0x6000, bytes))
|
|
|
|
}
|
|
|
|
rom := util.ReadRomOrDie("../data/roms/apple2+.rom", 12288)
|
2016-09-19 03:27:16 +00:00
|
|
|
charRom := util.ReadSmallCharacterRomOrDie("../data/roms/apple2-chars.rom")
|
2013-03-28 04:49:38 +00:00
|
|
|
plotter := TextPlotter(0)
|
2016-09-10 00:49:21 +00:00
|
|
|
a2 := goapple2.NewApple2(plotter, rom, charRom, options...)
|
2013-03-28 04:49:38 +00:00
|
|
|
if err := termbox.Init(); err != nil {
|
2016-09-10 00:49:21 +00:00
|
|
|
return err
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
2013-03-28 04:49:38 +00:00
|
|
|
events := make(chan termbox.Event)
|
2016-09-17 19:32:44 +00:00
|
|
|
done := false
|
|
|
|
|
|
|
|
if file != "" {
|
|
|
|
a2.AddPCAction(0, goapple2.PCAction{
|
|
|
|
Type: goapple2.ActionCallback,
|
|
|
|
Callback: func() { done = true },
|
|
|
|
})
|
|
|
|
}
|
2013-03-28 04:49:38 +00:00
|
|
|
go func() {
|
2016-09-10 00:49:21 +00:00
|
|
|
if file != "" {
|
2016-09-19 03:27:16 +00:00
|
|
|
for _, ch := range "HOME:CALL 24576" {
|
2016-09-10 00:49:21 +00:00
|
|
|
a2.Keypress(byte(ch))
|
|
|
|
}
|
2016-09-19 03:27:16 +00:00
|
|
|
if quit {
|
|
|
|
for _, ch := range ":CALL 0" {
|
|
|
|
a2.Keypress(byte(ch))
|
|
|
|
}
|
|
|
|
}
|
2016-09-10 00:49:21 +00:00
|
|
|
a2.Keypress(13)
|
|
|
|
}
|
2013-03-28 04:49:38 +00:00
|
|
|
for {
|
|
|
|
events <- termbox.PollEvent()
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
2013-03-28 04:49:38 +00:00
|
|
|
}()
|
2016-09-17 19:32:44 +00:00
|
|
|
for !ProcessEvents(events, a2) && !done {
|
2013-03-28 04:49:38 +00:00
|
|
|
err := a2.Step()
|
2013-02-28 07:01:51 +00:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2013-03-28 04:49:38 +00:00
|
|
|
termbox.Close()
|
2016-09-17 19:32:44 +00:00
|
|
|
if file != "" {
|
|
|
|
dumpscreen(a2)
|
|
|
|
}
|
2016-09-10 00:49:21 +00:00
|
|
|
return nil
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|
|
|
|
|
2016-09-17 19:32:44 +00:00
|
|
|
func dumpscreen(a2 *goapple2.Apple2) {
|
|
|
|
chars := []byte{}
|
|
|
|
for third := 0x400; third <= 0x450; third += 0x28 {
|
|
|
|
for base := third; base <= third+0x380; base += 0x80 {
|
|
|
|
for x := 0; x < 40; x++ {
|
|
|
|
ch := a2.RamRead(uint16(base + x))
|
|
|
|
chars = append(chars, ch&0x7f)
|
|
|
|
}
|
|
|
|
chars = append(chars, '\n')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
screen := string(chars)
|
|
|
|
screen = strings.TrimRight(screen, "\r\n\t ")
|
|
|
|
fmt.Println(screen)
|
|
|
|
}
|
|
|
|
|
2016-09-10 00:49:21 +00:00
|
|
|
var binfile = flag.String("binfile", "", "binary file to load at $6000 and CALL")
|
2016-09-19 03:27:16 +00:00
|
|
|
var quit = flag.Bool("quit", false, "quit after running binary")
|
2016-09-10 00:49:21 +00:00
|
|
|
|
2013-02-28 07:01:51 +00:00
|
|
|
func main() {
|
2016-09-10 00:49:21 +00:00
|
|
|
flag.Parse()
|
2016-09-19 03:27:16 +00:00
|
|
|
if err := RunEmulator(*binfile, *quit); err != nil {
|
2016-09-10 00:49:21 +00:00
|
|
|
fmt.Fprintln(os.Stderr, err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
2013-02-28 07:01:51 +00:00
|
|
|
}
|