From 9c0b548aab584d53469abd6d0ba3fe6dd345ddd9 Mon Sep 17 00:00:00 2001 From: Terence Boldt Date: Tue, 15 Dec 2020 21:05:12 -0500 Subject: [PATCH] Add ProDOS date/time support --- Firmware/Driver.lst | 2 +- Firmware/Firmware.asm | 12 ++- Firmware/Firmware.bin | Bin 2048 -> 2048 bytes README.md | 2 +- RaspberryPi/Driver.go | 165 +++++++++++++++++++++++++++++++----------- 5 files changed, 137 insertions(+), 44 deletions(-) diff --git a/Firmware/Driver.lst b/Firmware/Driver.lst index 966d058..fa96aad 100644 --- a/Firmware/Driver.lst +++ b/Firmware/Driver.lst @@ -1,4 +1,4 @@ -ca65 V2.19 - Git 59c58ac +ca65 V2.19 - Git 59c58acb Main file : Driver.asm Current file: Driver.asm diff --git a/Firmware/Firmware.asm b/Firmware/Firmware.asm index 0d63fb9..3532b01 100644 --- a/Firmware/Firmware.asm +++ b/Firmware/Firmware.asm @@ -16,6 +16,7 @@ InputByte = $c08e OutputByte = $c08d ReadBlockCommand = $01 WriteBlockCommand = $02 +GetTimeCommand = $03 NibbleStorage = $1d .org STARTSLOT @@ -85,7 +86,16 @@ GetStatus: ; ProDOS Read Block Command ReadBlock: - lda #ReadBlockCommand + ldy #$00 ;Get the current time on each block read for now + lda #GetTimeCommand + jsr SendByte +getTimeByte: + jsr GetByte + sta $bf90,y + iny + cpy #$04 + bne getTimeByte + lda #ReadBlockCommand ;read the block after setting the clock jsr SendByte lda BlockLo jsr SendByte diff --git a/Firmware/Firmware.bin b/Firmware/Firmware.bin index 4772c61b09009bac2876670058529c3e7758c0eb..a16034064ead1a5a4214a82878bcfb674a6eb429 100644 GIT binary patch literal 2048 zcmb`?KT8}z7{~ElF9DT+SEdoBm4H~Lu+qCzLb^1TTV9#g7qF!vDN_hH0Rv-|RDpnG z=f8HcuV5jN`WivBxIDjPhn-Ea*&^NWcB?jNjK@_FY?1n zk>s=dD4&S+e9Cr<*XjD_!;6FEvzv4|zFAw&UWzx_r`2L5lIPhX5hHe4u(J60{>OFk z});V&cv6$#opt_k~TaWKhCp+&SkfVy`6c;fw_gUo9MZ3P_!TQv>{swB_!ayLeii(x z@wYpUj@KRhD)KAxt8Tvvel?_D_wZ|0>e#R0*YIojHT*jGb>nY$938Ja_;uvh8TvcKdDc+adj~hu^W%#eN6BgWtjL;CI3A8h^Xv=y=`1 z?;^h=zw7q9;CDm%eGk89rH}m{eh@xXlp_erBPNZaWa4U_HYXmhC)7H<9pd(p#*LTR-0Liy>^+>Z#r5~mmLhKx8b+$$0!bdo%UM#XFsp?bND&@9DWWz_kN!H%N;>SvJQS;@N?(q z)qd{%yjj1f^$Yj~`~rRfzwmyM`^z0cN3srnQSb}r7u9~@{i0dFto2LyCHxY83BUAy znfuEfK}WIHV@SvJQS-@N4JS)qd^$x>>)e^&9vN{04plzwv&P`^z0c zN3srnQ}7$-H`RXQ{ia#Jt@T^@E&LXK3%~V#oBPWhK}WI;3jW{!fSQ BM1lYS diff --git a/README.md b/README.md index bd10ed6..02ad737 100644 --- a/README.md +++ b/README.md @@ -13,7 +13,7 @@ This is an early stage project. Currently one board has been assembled and teste 1. DONE - Build initial prototype that reads/writes virtual hard drive 2. DONE - Create firmware to make the card a bootable device 3. DONE - Fix board with updated second prototype PCB -4. Add ProDOS clock driver +4. DONE - Add ProDOS clock driver 5. Add RPi terminal access 6. Add web service call support 7. Proxy VNC connection, rendering as Apple II compatible graphics diff --git a/RaspberryPi/Driver.go b/RaspberryPi/Driver.go index 192b846..7d98eb9 100644 --- a/RaspberryPi/Driver.go +++ b/RaspberryPi/Driver.go @@ -1,13 +1,15 @@ package main import ( - //"time" "periph.io/x/periph/conn/gpio" "periph.io/x/periph/conn/gpio/gpioreg" "periph.io/x/periph/host" "fmt" "os" + "time" + "errors" ) +var edgeTimeout time.Duration var out_write gpio.PinIO var out_read gpio.PinIO @@ -26,7 +28,11 @@ var in_bit2 gpio.PinIO var in_bit1 gpio.PinIO var in_bit0 gpio.PinIO -func readNibble() byte { +const ReadBlockCommand = 1 +const WriteBlockCommand = 2 +const GetTimeCommand = 3 + +func readNibble() (byte, error) { // let the Apple II know we are ready to read //fmt.Printf("let the Apple II know we are ready to read\n") out_read.Out(gpio.Low) @@ -34,7 +40,10 @@ func readNibble() byte { // wait for the Apple II to write //fmt.Printf("wait for the Apple II to write\n") for in_write.Read() == gpio.High { - in_write.WaitForEdge(-1) + if !in_write.WaitForEdge(edgeTimeout) { + //fmt.Printf("Timed out reading nibble -- write stuck high\n") + return 0, errors.New("Timed out reading nibble -- write stuck high\n") + } } // get a nibble of data @@ -66,24 +75,34 @@ func readNibble() byte { // wait for the Apple II to finish writing //fmt.Printf("wait for the Apple II to finish writing\n") for in_write.Read() == gpio.Low { - in_write.WaitForEdge(-1) + if !in_write.WaitForEdge(edgeTimeout) { + //fmt.Printf("Timed out reading nibble -- write stuck low\n") + return 0, errors.New("Timed out reading nibble -- write stuck low") + } } - return nibble + return nibble, nil } -func readByte() byte { - data := byte(0) - data = readNibble() << byte(4) - data += readNibble() - return data +func readByte() (byte, error) { + data, err := readNibble() + data = data << byte(4) + if err != nil { return 0, err } + highNibble, err := readNibble() + if err != nil { return 0, err } + data += highNibble + //fmt.Printf("R%02X ", data) + return data, nil } -func writeNibble(data byte) { +func writeNibble(data byte) error { // wait for the Apple II to be ready to read //fmt.Printf("wait for the Apple II to be ready to read\n") for in_read.Read() == gpio.High { - in_read.WaitForEdge(-1) + if !in_read.WaitForEdge(edgeTimeout) { + //fmt.Printf("Timed out writing nibble -- read stuck high\n") + return errors.New("Timed out writing nibble -- read stuck high") + } } bit3 := gpio.Low @@ -118,23 +137,35 @@ func writeNibble(data byte) { // wait for the Apple II to finsih reading //fmt.Printf("wait for the Apple II to finsih reading\n") for in_read.Read() == gpio.Low { - in_read.WaitForEdge(-1) + if !in_read.WaitForEdge(edgeTimeout) { + //fmt.Printf("Timed out writing nibble -- read stuck low\n") + return errors.New("Timed out writing nibble -- read stuck low") + } } // let the Apple II know we are done writing //fmt.Printf("let the Apple II know we are done writing\n") out_write.Out(gpio.High) + return nil } -func writeByte(data byte) { - writeNibble(data >> 4) - writeNibble(data & 15) +func writeByte(data byte) error { + //fmt.Printf("W%02X ", data) + err := writeNibble(data >> 4) + if err != nil { return err } + err = writeNibble(data & 15) + if err != nil { return err } + + return nil } -func readBlock(buffer []byte) { +func readBlock(buffer []byte) error { for i := 0; i < 512; i++ { - writeByte(buffer[i]) + err := writeByte(buffer[i]) + if err != nil { return err } } + + return nil } func dumpBlock(buffer []byte) { @@ -143,10 +174,14 @@ func dumpBlock(buffer []byte) { } } -func writeBlock(buffer []byte) { +func writeBlock(buffer []byte) error { + var err error for i := 0; i < 512; i++ { - buffer[i] = readByte() + buffer[i], err = readByte() + if err != nil { return err } } + + return nil } func main() { @@ -169,11 +204,17 @@ func main() { in_bit1 = gpioreg.ByName("GPIO13") in_bit0 = gpioreg.ByName("GPIO6") + in_write.In(gpio.PullDown, gpio.BothEdges) + in_read.In(gpio.PullDown, gpio.BothEdges) + + edgeTimeout = time.Second * 5 + fmt.Printf("Starting Apple II RPi...\n") - fileName := "Total Replay v4.0-rc.1.hdv" + fileName := os.Args[1] file, err := os.OpenFile(fileName, os.O_RDWR, 0755) if err != nil { - //log.Fatal(err) + fmt.Printf("ERROR: %s", err.Error()) + os.Exit(1) } //if err := f.Close(); err != nil { //log.Fatal(err) @@ -187,34 +228,76 @@ func main() { //dumpBlock(buffer) for { - command := readByte(); - if (command == 1) { - blockLow := readByte(); - blockHigh := readByte(); + //fmt.Printf("Check for command") + + command,err := readByte(); + if err != nil { + //fmt.Printf("Timed out waiting for command\n") + } else { + switch command { + case ReadBlockCommand: + blockLow,_ := readByte(); + blockHigh,_ := readByte(); - var block int64 - block = int64(blockHigh) * 256 + int64(blockLow) + var block int64 + block = int64(blockHigh) * 256 + int64(blockLow) - fmt.Printf("Read block %d\n", block) + fmt.Printf("Read block %d\n", block) - file.ReadAt(buffer, int64(block) * 512) - //dumpBlock(buffer) - readBlock(buffer) - } - if (command == 2) { - blockLow := readByte(); - blockHigh := readByte(); + file.ReadAt(buffer, int64(block) * 512) + //dumpBlock(buffer) + readBlock(buffer) + break + case WriteBlockCommand: + blockLow,_ := readByte(); + blockHigh,_ := readByte(); - var block int64 - block = int64(blockHigh) * 256 + int64(blockLow) + var block int64 + block = int64(blockHigh) * 256 + int64(blockLow) - fmt.Printf("Write block %d\n", block) + fmt.Printf("Write block %d\n", block) - writeBlock(buffer) - file.WriteAt(buffer, int64(block) * 512) + writeBlock(buffer) + file.WriteAt(buffer, int64(block) * 512) + break + case GetTimeCommand: + /* + 49041 ($BF91) 49040 ($BF90) + + 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + DATE: | year | month | day | + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + + 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + TIME: | hour | | minute | + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + + 49043 ($BF93) 49042 ($BF92) + */ + now := time.Now() + + year := now.Year() % 100 + month := now.Month() + day := now.Day() + hour := now.Hour() + minute := now.Minute() + + bf91 := (byte(year) << 1) + (byte(month) >> 3) + bf90 := ((byte(month) & 15) << 5) + byte(day) + bf93 := byte(hour) + bf92 := byte(minute) + + writeByte(bf90) + writeByte(bf91) + writeByte(bf92) + writeByte(bf93) + break + } } } }