forked from Apple-2-HW/gp2io
Update README.md
This commit is contained in:
parent
8807187ed1
commit
c40dcbd5e1
59
README.md
59
README.md
@ -38,7 +38,7 @@ The first two buttons addresses are shared with the Apple keys, so it's best not
|
|||||||
PB2 = $C063 ; game Pushbutton 2 (read) <- Our victim
|
PB2 = $C063 ; game Pushbutton 2 (read) <- Our victim
|
||||||
|
|
||||||
PB3 (GS only) = $C060 ; game Pushbutton 3 (read)
|
PB3 (GS only) = $C060 ; game Pushbutton 3 (read)
|
||||||
$C060 bit 7 = data from cassette on Apple II, II+, IIe
|
$C060 bit 7 = data from cassette on Apple II, II+, IIe
|
||||||
|
|
||||||
Button 3 and the cassette share the same memory address $C060. So, we could toggle pin 9 on the game port (PB3) to load data directly into the memory address for the built-in cassette routines. But that pin is only on the GS. Poo. And the GS, having no cassette input, has no cassette routine in ROM. Double poo.
|
Button 3 and the cassette share the same memory address $C060. So, we could toggle pin 9 on the game port (PB3) to load data directly into the memory address for the built-in cassette routines. But that pin is only on the GS. Poo. And the GS, having no cassette input, has no cassette routine in ROM. Double poo.
|
||||||
|
|
||||||
@ -46,8 +46,6 @@ Luckily, there are already examples of reading in data on those pins.
|
|||||||
|
|
||||||
I dug into [Michael Mahon's NadaNet](http://michaeljmahon.com/NadaNetPaper.html) implementation for inspiration, and found proof of speedy and reliable communication on the game port. This was encouraging.
|
I dug into [Michael Mahon's NadaNet](http://michaeljmahon.com/NadaNetPaper.html) implementation for inspiration, and found proof of speedy and reliable communication on the game port. This was encouraging.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
But, rather than require developers to implement NadaNet network packet reads just to get a few bytes into and out of an AVR, we can continue to take inspiration from the cassette routines. Using those as a starting point, I look for a transition from LOW to HIGH in pushbutton 1, then count the loops until the transition from HIGH to LOW. If it's a short time, the bit is zero, if it's longer, a one.
|
But, rather than require developers to implement NadaNet network packet reads just to get a few bytes into and out of an AVR, we can continue to take inspiration from the cassette routines. Using those as a starting point, I look for a transition from LOW to HIGH in pushbutton 1, then count the loops until the transition from HIGH to LOW. If it's a short time, the bit is zero, if it's longer, a one.
|
||||||
|
|
||||||
===
|
===
|
||||||
@ -123,15 +121,15 @@ entry point $0303, or $0300 for sending single bit
|
|||||||
uses $EF for storing outgoing byte
|
uses $EF for storing outgoing byte
|
||||||
|
|
||||||
0300- A0 01 LDY #$01 ; load 1, for a 1 bit send - OR -
|
0300- A0 01 LDY #$01 ; load 1, for a 1 bit send - OR -
|
||||||
0302- 2C BIT ;
|
0302- 2C BIT ;
|
||||||
0303- A0 08 LDY #$08 ; load 8 for a full byte, loop counter
|
0303- A0 08 LDY #$08 ; load 8 for a full byte, loop counter
|
||||||
0305- 85 EF STA $EF ; put byte in zero page for safe keeping
|
0305- 85 EF STA $EF ; put byte in zero page for safe keeping
|
||||||
0307- 8D 59 C0 STA $C059 ; annunciator 0 high, RTS
|
0307- 8D 59 C0 STA $C059 ; annunciator 0 high, RTS
|
||||||
030A- 26 EF ROL $EF ; rotate byte left, high/MSB out to carry (sendbit)
|
030A- 26 EF ROL $EF ; rotate byte left, high/MSB out to carry (sendbit)
|
||||||
030C- 90 03 BCC $0311 ; "branch on carry clear" - if carry/bit = 0, goto #311 (short loop)
|
030C- 90 03 BCC $0311 ; "branch on carry clear" - if carry/bit = 0, goto #311 (short loop)
|
||||||
030E- A2 14 LDX #$14 ; if carry/bit = 1, load X with 20 (long loop) - OR -
|
030E- A2 14 LDX #$14 ; if carry/bit = 1, load X with 20 (long loop) - OR -
|
||||||
0310- 2C BIT
|
0310- 2C BIT
|
||||||
0311- A2 0A LDX #$0A ; if carry/bit = 0, load X with 10 (short loop)
|
0311- A2 0A LDX #$0A ; if carry/bit = 0, load X with 10 (short loop)
|
||||||
0313- 8D 5B C0 STA $C05B ; set annunciator 1 HIGH
|
0313- 8D 5B C0 STA $C05B ; set annunciator 1 HIGH
|
||||||
0316- CA DEX ; decrement, countdown to setting ANN1 low (countdown)
|
0316- CA DEX ; decrement, countdown to setting ANN1 low (countdown)
|
||||||
0317- D0 FD BNE $0316 ; if X > 0, keep counting (countdown)
|
0317- D0 FD BNE $0316 ; if X > 0, keep counting (countdown)
|
||||||
@ -151,52 +149,37 @@ uses $EF for storing outgoing byte
|
|||||||
|
|
||||||
On the GP2IO side, the AVR has two interrupts set on ANN0 and ANN1: as the RTS pin goes HIGH (rising), and on changes to the data pin. If the data pin changes state in less than 70 microseconds (.07 milliseconds) the bit is read as zero. Longer than 70 microseconds, a one.
|
On the GP2IO side, the AVR has two interrupts set on ANN0 and ANN1: as the RTS pin goes HIGH (rising), and on changes to the data pin. If the data pin changes state in less than 70 microseconds (.07 milliseconds) the bit is read as zero. Longer than 70 microseconds, a one.
|
||||||
|
|
||||||
|
~~~
|
||||||
attachInterrupt(0, APPLERTS, RISING); // ANNUNCIATOR 0, Apple sending byte
|
attachInterrupt(0, APPLERTS, RISING); // ANNUNCIATOR 0, Apple sending byte
|
||||||
attachInterrupt(1, RECEIVINGBITS, CHANGE); // ANNUNCIATOR 1, Apple sending bits
|
attachInterrupt(1, RECEIVINGBITS, CHANGE); // ANNUNCIATOR 1, Apple sending bits
|
||||||
|
|
||||||
void APPLERTS() {
|
void APPLERTS() {
|
||||||
// signal to start receiving bits from Apple II
|
// signal to start receiving bits from Apple II
|
||||||
bitCount = 0;
|
bitCount = 0;
|
||||||
changeCount = 0;
|
changeCount = 0;
|
||||||
returnByte = B00000000;
|
returnByte = B00000000;
|
||||||
}
|
}
|
||||||
|
|
||||||
void RECEIVINGBITS()
|
void RECEIVINGBITS()
|
||||||
{
|
{
|
||||||
// ignore short "reset" transitions
|
// ignore short "reset" transitions
|
||||||
currentMicros = micros();
|
currentMicros = micros();
|
||||||
|
|
||||||
if (changeCount % 2 == 1) {
|
if (changeCount % 2 == 1) {
|
||||||
|
|
||||||
if ((currentMicros - lastMicros) > 70) {
|
if ((currentMicros - lastMicros) > 70) {
|
||||||
|
|
||||||
receivedBit = 1;
|
receivedBit = 1;
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
receivedBit = 0;
|
receivedBit = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
byteArray[7 - bitCount] = receivedBit;
|
byteArray[7 - bitCount] = receivedBit;
|
||||||
|
|
||||||
bitCount++;
|
bitCount++;
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
changeCount++;
|
changeCount++;
|
||||||
|
|
||||||
if (bitCount == 8) { // got a BYTE
|
if (bitCount == 8) { // got a BYTE
|
||||||
receivedByte = arrayToByte(byteArray, 8);
|
receivedByte = arrayToByte(byteArray, 8);
|
||||||
PROCESSBYTE( byte(receivedByte) );
|
PROCESSBYTE( byte(receivedByte) );
|
||||||
TIMEOUTCLOCK = millis();
|
TIMEOUTCLOCK = millis();
|
||||||
}
|
}
|
||||||
|
|
||||||
lastMicros = currentMicros;
|
lastMicros = currentMicros;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
~~~
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
===
|
===
|
||||||
@ -222,15 +205,23 @@ Processing Bytes on the GP2IO
|
|||||||
|
|
||||||
The AVR starts in a null "standby" state, waiting for a "control" byte from the Apple II to set its function. Subsequent bytes after the function is set are the "message". The value of the first byte from the Apple determines the function according to this table:
|
The AVR starts in a null "standby" state, waiting for a "control" byte from the Apple II to set its function. Subsequent bytes after the function is set are the "message". The value of the first byte from the Apple determines the function according to this table:
|
||||||
|
|
||||||
$01 Sets the RGB LED on the GP2IO to the color value of the next three bytes, in order Red, Green and Blue. Requires 3 byte message
|
$01 Sets the RGB LED on the GP2IO to the color value of the next three bytes, in order Red, Green and Blue. Requires 3 byte message
|
||||||
$02 Sets the RGB LED to white, intensity based on the 1 byte message that follows (0-255).
|
|
||||||
$04 Write the following message to an internal buffer, for retrieval later. First byte is message length (0-255 bytes follow)
|
$02 Sets the RGB LED to white, intensity based on the 1 byte message that follows (0-255).
|
||||||
$08 Write the following message to UART serial (buffered). First byte is message length (0-255 bytes follow)
|
|
||||||
$10 Write the following 8 bytes to I2C bus. For the demo, this is connected to either an 8x8 LED matrix or a 4 character 7-segment display. Requires 8 byte message.
|
$04 Write the following message to an internal buffer, for retrieval later. First byte is message length (0-255 bytes follow)
|
||||||
$20 SPI (not yet implemented)
|
|
||||||
$40 Query the buffer. Triggers the AVR to respond with one byte, containing the length of the current buffer.
|
$08 Write the following message to UART serial (buffered). First byte is message length (0-255 bytes follow)
|
||||||
$80 Send the buffer. Triggers the AVR to respond with the first N bytes of the buffer, where N is the byte following the $80 trigger.
|
|
||||||
$00 Debug mode. At the moment, simply echoes the bytes received from the Apple side into the serial buffer, and prints them to USB serial out. Also sends one byte from the GP2IO buffer to the Apple whenever the APPLECTS signal (Annunciator 2) goes high.
|
$10 Write the following 8 bytes to I2C bus. For the demo, this is connected to either an 8x8 LED matrix or a 4 character 7-segment display. Requires 8 byte message.
|
||||||
|
|
||||||
|
$20 SPI (not yet implemented)
|
||||||
|
|
||||||
|
$40 Query the buffer. Triggers the AVR to respond with one byte, containing the length of the current buffer.
|
||||||
|
|
||||||
|
$80 Send the buffer. Triggers the AVR to respond with the first N bytes of the buffer, where N is the byte following the $80 trigger.
|
||||||
|
|
||||||
|
$00 Debug mode. At the moment, simply echoes the bytes received from the Apple side into the serial buffer, and prints them to USB serial out. Also sends one byte from the GP2IO buffer to the Apple whenever the APPLECTS signal (Annunciator 2) goes high.
|
||||||
|
|
||||||
|
|
||||||
Each function, as outlined, expects a certain number of bytes per message. For example, to set an RGB LED requires three bytes - one each for Red, Green, and Blue values, plus the first $01 byte to set the function, for a total of four.
|
Each function, as outlined, expects a certain number of bytes per message. For example, to set an RGB LED requires three bytes - one each for Red, Green, and Blue values, plus the first $01 byte to set the function, for a total of four.
|
||||||
@ -259,6 +250,7 @@ EXAMPLE: RGB LED Pulse white
|
|||||||
|
|
||||||
SLIGHTLY MORE COMPLEX EXAMPLE: RGB LED pulsing
|
SLIGHTLY MORE COMPLEX EXAMPLE: RGB LED pulsing
|
||||||
|
|
||||||
|
|
||||||
SETRGB:
|
SETRGB:
|
||||||
one-shot, sets the LED with red value in $06, green at $07, blue at $08
|
one-shot, sets the LED with red value in $06, green at $07, blue at $08
|
||||||
|
|
||||||
@ -272,6 +264,7 @@ SETRGB:
|
|||||||
0811- 20 03 03 JSR $0303 ; SENDBYTE
|
0811- 20 03 03 JSR $0303 ; SENDBYTE
|
||||||
0814- 60 RTS ; return.
|
0814- 60 RTS ; return.
|
||||||
|
|
||||||
|
|
||||||
CYCLERGB
|
CYCLERGB
|
||||||
|
|
||||||
0816- A9 00 LDA #$00 ; set accumulator to 0
|
0816- A9 00 LDA #$00 ; set accumulator to 0
|
||||||
|
Loading…
x
Reference in New Issue
Block a user