IIcSystemClock/clock.system.s
2020-10-02 21:02:13 -07:00

1173 lines
28 KiB
ArmAsm

.include "opcodes.inc"
.include "apple2.inc"
.include "common.inc"
.feature string_escapes
.org $2000
.setcpu "65C02"
.enum MessageCode
kInstall = 0
kNoSysFile = 1
kDiskError = 2
kIIc = 3
kIIe = 4
kCurrentYear = 5
kOkPrompt = 6
kNoClock = 7
kRunning = 8
kSeikoIIc = 9
kSeikoIIe = 10
kRemoveWriteProtect = 11
.endenum
kErrDiskWriteProtected = $2B
;;; Zero Page Locations
mach_type := $07 ; 0 = IIc, 1 = II+/IIe
case_mask := $08 ; mask with chars before COUT
has_80col := $09 ; 1 = has 80 column card
month := $0A ; month from current date
msg_num := $0B ; stashed message number
PRODOS_SYS_START := $2000
SLOT3_FIRMWARE := $C300
PORT2_ACIA_STATUS := $C0A9
PORT2_ACIA_COMMAND := $C0AA
PORT2_ACIA_CONTROL := $C0AB
DISXY := $C058
ENBXY := $C059
X0EDGE1 := $C05C
X0EDGE2 := $C05D
MOUSE_BTN := $C063
kClockRoutineMaxLength = 125 ; Per ProDOS 8 TRM
kDefaultDriveSlot = 6
kIIcVersionByte = HI('L')
kDefaultVersionByte = HI(' ')
kDefaultYear = 1986
L2000:
default_slot := * + 1
lda #kDefaultDriveSlot ; For easy patching???
version_test_byte := *+1
lda #kDefaultVersionByte
cmp #kIIcVersionByte
bne L2013
sta L239E + (L1272 - L1000) ; In relocated code
lda #$5C
sta L2285
sta L2336
L2013:
;; Clear stack
ldx #$FF
txs
;; Clear interpreter version
;; TODO: Remove this
lda #0
sta IBAKVER
sta IVERSION
;; Trash reset vector to cause reboot
sta ROMIN2
lda $03F3
eor #$FF
sta $03F4
;; Identify machine type
ldy #0
ldx #0
lda MACHID
and #%10001000 ; =0 if II+, >0 if IIe/IIc
beq :+
inx ; Not a II+
: cmp #%10001000 ; IIc ?
beq is_iic
lda version_test_byte ; Hack ???
cmp #kIIcVersionByte
beq is_iic
iny ; Not a IIc
is_iic: sty mach_type
;; Set case mask (II+ vs. IIe/IIc)
lda case_mask_table,x
sta case_mask
;; Check 40/80 columns; wrap strings if 40 columns.
lda MACHID
and #%00000010 ; 80 Column card?
lsr a
sta has_80col
bne init_80_col
;; Convert spaces to newlines if 40 Columns
lda #CR|$80
sta chain + (wrap1 - L1000)
sta chain + (wrap4 - L1000)
sta chain + (wrap5 - L1000)
sta chain + (wrap3 - L1000)
sta chain + (wrap2 - L1000)
inc L2095
inc L2099
bne :+
init_80_col:
jsr SLOT3_FIRMWARE
:
;; --------------------------------------------------
;; Copy to $1000
.proc RelocateChainingCode
ldy #0
;; End signature is two adjacent $FFs
loop:
read_msb := * + 2
lda L239E,y
cmp #$FF
bne :+
check_msb := * + 2 ; check for second $FF
ldx L239E+1,y
cpx #$FF
beq L208F ; done!
:
write_msb := * + 2
sta L1000,y
iny
bne loop
inc read_msb
inc check_msb
inc write_msb
bne loop ; always
.endproc
L208F: ldy #MessageCode::kInstall
jsr ShowMessage
L2095 := * + 1
ldy #$03
sty $22
L2099 := * + 1
ldy #$05
sty $23
jsr SetPrefixAndCheckSysFile
lda $1204
lsr a
sta year
jsr ConvertToBCD
sta bcd_year
;; --------------------------------------------------
;; Identify drive slot, needed for patch1
.scope
;; Search device list for Disk II device
ldx DEVCNT
: lda DEVLST,x
and #%00001111 ; low nibble = "device identification"
beq found ; 0 = Disk II
dex
bpl :-
;; Did not find Disk II, use default
lda default_slot
and #%00000111
bne :+
lda #6 ; default to slot 6 if was tweaked to 0
: asl a ; shift to 0sss0000 (like a unit number)
asl a
asl a
asl a
bne assign
found: lda DEVLST,x
assign: and #%01110000 ; slot
ora #$80
sta patch1_firmware_byte ; Set $C0nn address
.endscope
;; --------------------------------------------------
lda L11FE
and #$03
beq select_patch ; apply patch2 if IIc, patch1 otherwise
cmp #$02
bcs L20FE
lda mach_type
bne not_iic ; apply patch3
beq L20FE ; always
;;; ------------------------------------------------------------
kPatchLength = $38
kPatch1Offset = $0
kPatch2Offset = $38
kPatch3Offset = $70
;;; Patch1 - otherwise
;;; Patch2 - for mach_type = IIc
;;; Patch3
select_patch:
ldy #kPatch2Offset
lda mach_type
beq apply_patch ; if IIc
ldy #kPatch1Offset
jmp apply_patch
;; Patch bytes on top of driver
not_iic:
ldy #kPatch3Offset
apply_patch:
ldx #$00
: lda Patches,y
sta L2269,x
iny
inx
cpx #kPatchLength
bne :-
;;
L20FE: lda #2
sta tries
: jsr CalcMonthValidateDateTime
bcs L2111
dec tries
bpl :-
bmi install_clock ; always
tries: .byte 0
;;; ------------------------------------------------------------
L2110: .byte 0
L2111: lda mach_type
bne L2135 ; not IIc
lda L2110
bne L2135
inc L2110
ldy #$99
sty L228A
iny
sty L226A
sty L2274
lda L11FE
and #$03
beq L20FE
sty L229F
bne L20FE
L2135: ldy #MessageCode::kNoClock
sty msg_num
lda MACHID ; Check for clock card
ror a
bcc no_clock ; Bit 0 = 0 means no clock card
jsr MON_HOME
jmp L1000
no_clock:
lda #0
sta DATELO
sta DATELO+1
sta TIMELO
sta TIMELO+1
jmp ShowMessageAndMaybeChain
;; --------------------------------------------------
install_clock:
lda #OPC_JMP_abs
sta DATETIME
lda MACHID
ora #%00000001 ; has clock
sta MACHID
bit KBD
bmi L218C
lda month
cmp #11
bcc L2176
bit L11FE
bpl L2176
sta $1204
L2176: lda $1204
lsr a
cmp #(kDefaultYear .mod 100)
bcc L218C
;; Two byte compare
lda DATELO
cmp file_info_params + $A ; mod_date
lda DATELO+1
sbc file_info_params + $A + 1 ; mod_date + 1
bcs L21DF
L218C: bit KBDSTRB
rol L11FE
lda #3
cmp month
ror L11FE
;; --------------------------------------------------
;; Show current year prompt
.scope
retry:
ldy #MessageCode::kCurrentYear
jsr ShowMessage
lda bcd_year ; 2-digit year
jsr PRBYTE
ldy #MessageCode::kOkPrompt
jsr ShowMessage
;; Wait for keypress
: jsr RDKEY
and #%11011111 ; lowercase --> uppercase
cmp #'Y'|$80
beq year_ok
cmp #'N'|$80
bne :-
;; Prompt for two digit year
ldy #MessageCode::kCurrentYear
jsr ShowMessage
jsr GetDigitKey ; Decade
asl a ; move tens digit to high nibble
asl a
asl a
asl a
sta bcd_year
jsr GetDigitKey ; Year
and #$0F ; mask ones digit
ora bcd_year
sta bcd_year
jmp retry
.endscope
;; --------------------------------------------------
;; Current year is okay
year_ok:
lda $1204
jsr YearFromBCD
jsr InvokeDriver
jsr WriteFileInfo
L21DF: lda RWRAM1 ; Driver lives in LC Bank 1
lda RWRAM1 ; so bank that in
lda DATETIME+1 ; driver destination
sta install_ptr
clc
adc #time_offset_table - Driver ; patch an internal reference (LSB)
sta offset_table_addr
lda DATETIME+2 ; driver destination
sta install_ptr+1
adc #0
sta offset_table_addr+1 ; patch an internal reference (MSB)
;; Relocate clock driver
ldy #kClockRoutineMaxLength - 1
: lda Driver,y
install_ptr := * + 1
sta $F000,y
dey
bpl :-
;; Initialize the time (via driver)
jsr DATETIME
lda ROMIN2 ; Bank ROM back in
;; Chain
jmp L1000
;;; ------------------------------------------------------------
;;; Convert year from BCD
;;; Input: bcd_year
;;; Output: year
;;; $06 is trashed
.proc YearFromBCD
lda bcd_year
pha
lsr a
lsr a
lsr a
lsr a
sta $06
asl a
asl a
adc $06
asl a
sta $06
pla
and #$0F
clc
adc $06
sta year
rts
.endproc
;;; ------------------------------------------------------------
;;; Convert to BCD
;;; Input: A = number
;;; Output: A = BCD number
;;; $06 is trashed
.proc ConvertToBCD
ldx #$FF
: inx
sec
sbc #10
bcs :-
adc #10
sta $06
txa
asl a
asl a
asl a
asl a
ora $06
rts
.endproc
;;; ------------------------------------------------------------
;;; Returns with carry set if date is not valid.
.proc CalcMonthValidateDateTime
jsr InvokeDriver
;; Check month
lda DATELO+1
ror a
lda DATELO
rol a
rol a
rol a
rol a
and #$0F
sec
beq done ; Month = 0 is failure
cmp #13
bcs done ; Month >= 13 is failure
sta month
;; Check hour
lda TIMELO+1
cmp #24
bcs done ; Hour >= 24 is failure
;; Check minute
lda TIMELO
cmp #60 ; Min >= 60 is failure
done: rts
.endproc
;;; ============================================================
;;; Clock Driver (Relocatable)
;;; ============================================================
Driver: cld
cld ; TODO: Remove duplicate CLD
php
sei
L2269:
L226A := * + 1
lda PORT2_ACIA_COMMAND
pha ; Save command register
ldy #$03
ldx #$16
lda #%00001000
L2273:
L2274 := * + 1
sta PORT2_ACIA_COMMAND
L2276: dex
bne L2276
eor #%00001010
ldx #$09
dey
bne L2273
ldy #$04
bne L2289
L2284:
L2285 := * + 1
lda #$5D
L2286: dec ; 65C02
bne L2286
L2289:
L228A := * + 1
lda PORT2_ACIA_STATUS
rol a
rol a
rol a
ror $0200,x
lsr $0201,x
dey
bne L2284
ldy #$04
dex
bpl L2284
pla ; Restore command register
L229F := * + 1
sta PORT2_ACIA_COMMAND
;; Process digits
ldx #6
L22A3: lda $0201,x
: dec $0200,x ;
bmi :+
clc
adc #10
bcc :-
:
offset_table_addr := * + 1
ldy time_offset_table,x ; Offset table
sta DEVNUM,y ; y is offset from DEVNUM for some reason
dex
dex
bne L22A3
;; --------------------------------------------------
;; Assign month in DATELO/DATEHI
L22BA: lda $0200 ; top nibble is month
asl a ; DATELO = mmmddddd
and #%11100000
ora DATELO
sta DATELO
year := * + 1
lda #(kDefaultYear .mod 100)
rol a ; DATEHI = yyyyyyym, shift in month bit
sta DATELO+1
;; --------------------------------------------------
ldy #$01
L22CE: lda $0208,y
ora #$B0
sta $020F,y
dey
bpl L22CE
dex
bne L22BA
plp
rts
time_offset_table := * - 3
.byte $FF ; dummy
.byte TIMELO+1 - DEVNUM ; offset
.byte $FF ; dummy
.byte TIMELO - DEVNUM ; offset
;; End of relocated clock driver
;;; ============================================================
;; Patches applied to driver (length $38, at offset 0)
Patches:
;; Patch #1
patch1:
patch1_firmware_byte := * + 1
lda $C0E0 ; Set to $C0x0, n=slot+8
lda DISVBL
ldy #$01
ldx #$16
L22EC: dex
bne L22EC
lda DISVBL,y
ldx #$0B
dey
bpl L22EC
L22F7: dex
bne L22F7
ldx #$09
ldy #$04
bne L2307
L2300: lda #$5D
sec
L2303: sbc #$01
bne L2303
L2307: lda MOUSE_BTN
rol a
ror $0200,x
lsr $0201,x
nop
dey
bne L2300
ldy #$04
dex
bpl L2300
.assert * - patch1 = kPatchLength, error, "Patch length"
;; --------------------------------------------------
;; Patch #2
patch2:
.assert * = Patches + kPatch2Offset, error, "Offset changed"
lda PORT2_ACIA_COMMAND
nop
ldy #$03
ldx #$16
lda #%00000010
L2324: sta PORT2_ACIA_COMMAND
L2327: dex
bne L2327
eor #$0A
ldx #$09
dey
bne L2324
ldy #$04
bne L233A
L2335:
L2336 := * + 1
lda #$5D
L2337: dec ; 65C02
bne L2337
L233A: lda PORT2_ACIA_STATUS
eor #$20
rol a
rol a
rol a
ror $0200,x
lsr $0201,x
dey
bne L2335
ldy #$04
dex
bpl L2335
nop
nop
.assert * - patch2 = kPatchLength, error, "Patch length"
;; --------------------------------------------------
;; Patch #3
patch3:
.assert * = Patches + kPatch3Offset, error, "Offset changed"
lda ENVBL
sta ENBXY
sta X0EDGE1
nop
nop
nop
nop
lda DISXY
ldx #$15
L2364: dex
bne L2364
ldx #$09
L2369: ldy #$04
L236B: lda MOUSE_BTN
rol a
ror $0200,x
lsr $0201,x
sta X0EDGE2
nop
nop
nop
nop
nop
nop
sta X0EDGE1
dey
bne L236B
dex
bpl L2369
lda DISVBL
.assert * - patch3 = kPatchLength, error, "Patch length"
;;; ============================================================
.proc InvokeDriver
jsr Driver
rts
.endproc
;;; ------------------------------------------------------------
;;; Prompt, loop until digit key is pressed
.proc GetDigitKey
jsr RDKEY
cmp #'0' | $80
bcc GetDigitKey
cmp #('9'+1) | $80
bcs GetDigitKey
jmp COUT
.endproc
;;; ------------------------------------------------------------
case_mask_table:
.byte %11011111 ; map lowercase to uppercase
.byte %11111111 ; preserve case
chain:
L239E:
;; Relocated to $1000
;;; ============================================================
;;; Chaining code???
;;; ============================================================
.org $1000
L1000:
ldy #$00
sty read_block_block_num+1
iny
sty $04
iny
sty read_block_block_num
jsr L119F
lda $1C23
sta $02
lda $1C24
sta $03
lda #$2B
sta $00
lda #$1C
sta $01
L1021: ldy #$10
lda ($00),y
cmp #$FF
bne L10A8
ldy #$00
lda ($00),y
and #$30
beq L10A8
lda ($00),y
and #$0F
sta $05
tay
ldx #strlen_str_system - 1
L103A: lda ($00),y
cmp str_system,x
bne L10A8
dey
dex
bpl L103A
ldy #strlen_str_clock_system
L1047: lda ($00),y
cmp str_clock_system,y
bne L1053
dey
bne L1047
beq L10A8
L1053: lda $05
sta $121D
sta $0280
inc $05
lda msg_num
cmp #$07
beq L1070
ldy #$03
jsr L10E5
lda mach_type
beq L106D
iny
L106D: jsr ShowMessage
L1070: ldy #MessageCode::kRunning
jsr ShowMessage
ldy #$01
L1077: lda ($00),y
sta $121D,y
sta $0280,y
ora #$80
jsr COUT
iny
cpy $05
bne L1077
jsr LoadSysFile
lda #$00
sta WNDTOP
lda #$18
sta WNDBTM
jsr MON_HOME
lda has_80col
beq L10A5
lda #$15
jsr COUT
lda #$8D
jsr COUT
L10A5: jmp PRODOS_SYS_START
L10A8: clc
lda $00
adc $02
sta $00
lda $01
adc #$00
sta $01
inc $04
lda $04
cmp $03
bne L10E2
ldy $1C02
sty read_block_block_num
lda $1C03
sta read_block_block_num+1
bne :+
tya
bne :+
ldy #MessageCode::kNoSysFile
jmp ShowMessageAndMaybeChain
: jsr L119F
lda #$00
sta $04
lda #$04
sta $00
lda #$1C
sta $01
L10E2: jmp L1021
L10E5: lda L11FE
and #$03
bne L10EE
ldy #$09
L10EE: rts
;;; ------------------------------------------------------------
;;; Call with message number in Y.
;;; Clears screen unless kOkPromptor or kRunning.
.proc ShowMessage
lda message_table_lo,y
sta ptr
lda message_table_hi,y
sta ptr+1
cpy #MessageCode::kOkPrompt
beq :+
cpy #MessageCode::kRunning
beq :+
jsr MON_HOME
: ldy #0
ptr := *+1
loop: lda $F000,y
beq done
cmp #'`'|$80 ; Not lower case?
bcc :+
and case_mask
: jsr COUT
iny
bne loop
done: rts
.endproc
;;; ------------------------------------------------------------
;;; Write current date/time to a file.
.proc WriteFileInfo
retry:
lda #7 ; SET_FILE_INFO count
sta file_info_params
;; Fill params with current date/time.
ldy #3
: lda DATELO,y
sta file_info_params + $A,y ; mod_date
dey
bpl :-
lda #OPC_RTS ; Temporarily disable driver
sta DATETIME
PRODOS_CALL MLI_SET_FILE_INFO, file_info_params
bne :+
lda #OPC_JMP_abs ; Re-enable driver
sta DATETIME
rts
;; Error; maybe retry?
: cmp #kErrDiskWriteProtected
bne ShowDiskErrorAndChain ; Failed
;; Write protected - show error and retry
ldy #MessageCode::kRemoveWriteProtect
jsr ShowMessage
jsr Bell
jsr RDKEY
jmp retry
.endproc
;;; ------------------------------------------------------------
.proc LoadSysFile
PRODOS_CALL MLI_OPEN, open_params
bne ShowDiskErrorAndChain
lda open_params_ref_num
sta read_params_ref_num
PRODOS_CALL MLI_READ, read_params
bne ShowDiskErrorAndChain
PRODOS_CALL MLI_CLOSE, close_params
bne ShowDiskErrorAndChain
rts
.endproc
;;; ------------------------------------------------------------
.proc SetPrefixAndCheckSysFile
lda DEVNUM ; Most recently accessed device
sta on_line_unit_num
sta read_block_unit_num
PRODOS_CALL MLI_ON_LINE, on_line_params
bne ShowDiskErrorAndChain
lda $120C
and #$0F
tay
iny
sty $120B
lda #'/'
sta $120C
PRODOS_CALL MLI_SET_PREFIX, set_prefix_params
bne ShowDiskErrorAndChain
PRODOS_CALL MLI_GET_FILE_INFO, file_info_params
bne ShowDiskErrorAndChain
rts
.endproc
;;; ------------------------------------------------------------
L119F: PRODOS_CALL MLI_READ_BLOCK, read_block_params
bne ShowDiskErrorAndChain
rts
;;; ------------------------------------------------------------
.proc ShowDiskErrorAndChain
ldy #MessageCode::kDiskError
;; fall through
.endproc
;;; Show message and chain to next system file, unless
;;; kNoClock (in which case: hang)
.proc ShowMessageAndMaybeChain
sty msg_num
jsr ShowMessage
jsr Bell
lda msg_num
cmp #MessageCode::kNoClock
bne loop
lda #OPC_RTS
sta DATETIME
jmp L1000
loop: jmp loop ; Infinite loop!
.endproc
;;; ------------------------------------------------------------
;;; Make a tone (from ProDOS Technical Reference Manual)
;;; $0C is trashed
.proc Bell
length := $0C
lda #$20 ; duration of tone
sta length
bell1: lda #$2 ; short delay...click
jsr WAIT
sta SPKR
lda #$24 ; long delay...click ($20 in TRM)
jsr WAIT
sta SPKR
dec length
bne bell1 ; repeat length times
rts
.endproc
;;; ------------------------------------------------------------
;;; MLI call params
on_line_params:
.byte 2 ; param_count
on_line_unit_num:
.byte $60 ; unit_num
.addr $120C ; data_buffer
set_prefix_params:
.byte 1 ; param_count
.addr $120B ; pathname
read_block_params:
.byte 3 ; param_count
read_block_unit_num:
.byte $60 ; unit_num
.addr $1C00 ; data_buffer
read_block_block_num:
.word $0000 ; block_num
open_params:
.byte 3 ; param_count
.addr $121D ; pathname
.addr $1C00 ; io_buffer
open_params_ref_num:
.byte 0 ; ref_num
read_params:
.byte 4 ; param_count
read_params_ref_num:
.byte 0 ; ref_num
.addr PRODOS_SYS_START ; data_buffer
.word PRODOS-PRODOS_SYS_START ; request_count
.word 0 ; trans_count
close_params:
.byte 1 ; param_count
.byte 0 ; ref_num
file_info_params:
.byte $A ; param_count
.addr str_clock_system ; pathname
.byte 0 ; access
.byte 0 ; file_type
;; ...
;;; ------------------------------------------------------------
;;; Misc variables
L11FE: .byte 0 ; ???
bcd_year: .byte 0 ; 2-digit (shared)
L1200:
;; buffer for variables, filename
.res 46, 0
str_system:
.byte ".SYSTEM"
strlen_str_system = .strlen(".SYSTEM")
str_clock_system:
PASCAL_STRING "CLOCK.SYSTEM"
strlen_str_clock_system = .strlen("CLOCK.SYSTEM")
;;; ------------------------------------------------------------
;;; Message strings
message_table_lo:
.byte <msgInstall,<msgNoSysFile,<msgDiskError,<msgIIc
.byte <msgIIe,<msgCurrentYear,<msgOkPrompt,<msgNoClock
.byte <msgRunning,<msgSeikoIIc,<msgSeikoIIe,<msgRemoveWriteProtect
message_table_hi:
.byte >msgInstall,>msgNoSysFile,>msgDiskError,>msgIIc
.byte >msgIIe,>msgCurrentYear,>msgOkPrompt,>msgNoClock
.byte >msgRunning,>msgSeikoIIc,>msgSeikoIIe,>msgRemoveWriteProtect
msgInstall:
HIASCII "Install Clock Driver 1.5"
L1272: HIASCII " " ; Modified at launch
HIASCII "\rCopyright (c) 1986 "
wrap1 := *-1
HIASCIIZ "Creative Peripherals Unlimited, Inc."
msgNoSysFile:
HIASCIIZ "Unable to find a '.SYSTEM' file!"
msgRemoveWriteProtect:
HIASCII "Remove Write-Protect tab, "
wrap2 := *-1
HIASCIIZ "Replace disk, and Press a key..."
msgDiskError:
HIASCIIZ "Disk error! Unable to continue!!!"
msgSeikoIIc:
HIASCII "Seiko "
msgIIc: HIASCIIZ "//c driver installed. "
wrap4 := *-2
msgSeikoIIe:
HIASCII "Seiko "
msgIIe: HIASCIIZ "//e driver installed. "
wrap5 := *-2
msgCurrentYear:
HIASCIIZ "Current year is 19"
msgOkPrompt:
HIASCII "."
wrap3 := *
HIASCIIZ " OK? (Y/N) "
msgNoClock:
HIASCIIZ "No clock! Driver not installed...\r"
msgRunning:
HIASCIIZ "Running "
;; Signature for end of range to copy to $1000
.byte $FF,$FF
;;; ============================================================
.byte $00,$04,$00
.byte $FF,$00,$FF,$00,$FF,$00,$00,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$00
.byte $FF,$00,$FF,$00,$FF,$00,$FF,$65
.byte $00,$FF,$00,$FF,$00,$FF,$00,$FF
.byte $00,$FF,$00,$FF,$00,$FF,$00,$FF
.byte $00,$FF,$00,$FF,$00,$FF,$00,$FF
.byte $00