1
0
mirror of https://github.com/cc65/cc65.git synced 2025-01-19 02:33:19 +00:00
cc65/libsrc/lynx/tgi/lynx-160-102-16.s
2019-10-13 09:03:46 -04:00

1080 lines
30 KiB
ArmAsm

;
; Graphics driver for the 160x102x16 mode on the Lynx.
;
; All the drawing functions simply are done by sprites, as the sprite
; engine is the only way to do fast graphics on a Lynx.
;
; This code was written by Karri Kaksonen, 2004 for the cc65 compiler.
;
.include "zeropage.inc"
.include "../extzp.inc"
.include "tgi-kernel.inc"
.include "tgi-error.inc"
.include "lynx.inc"
.macpack generic
.macpack module
; ------------------------------------------------------------------------
; Header. Includes jump table and constants.
module_header _lynx_160_102_16_tgi
; First part of the header is a structure that has a magic and defines the
; capabilities of the driver
.byte $74, $67, $69 ; "tgi"
.byte TGI_API_VERSION ; TGI API version number
libref: .addr $0000 ; Library reference
.word 160 ; X resolution
.word 102 ; Y resolution
.byte 16 ; Number of drawing colors
.byte 2 ; Number of screens available
.byte 8 ; System font X size
.byte 8 ; System font Y size
.word $0100 ; Aspect ratio (square pixel LCD)
.byte TGI_BM_FONT_FINESCALE ; TGI driver flags
; Next comes the jump table. Currently all entries must be valid and may point
; to an RTS for test versions (function not implemented). A future version may
; allow for emulation: In this case the vector will be zero. Emulation means
; that the graphics kernel will emulate the function by using lower level
; primitives - for example ploting a line by using calls to SETPIXEL.
.addr INSTALL
.addr UNINSTALL
.addr INIT
.addr DONE
.addr GETERROR
.addr CONTROL
.addr CLEAR
.addr SETVIEWPAGE
.addr SETDRAWPAGE
.addr SETCOLOR
.addr SETPALETTE
.addr GETPALETTE
.addr GETDEFPALETTE
.addr SETPIXEL
.addr GETPIXEL
.addr LINE
.addr BAR
.addr TEXTSTYLE
.addr OUTTEXT
; ------------------------------------------------------------------------
; Data.
; Variables mapped to the zero page segment variables. Some of these are
; used for passing parameters to the driver.
X1 := ptr1
Y1 := ptr2
X2 := ptr3
Y2 := ptr4
STRPTR := ptr3
FONTOFF := ptr4
STROFF := tmp3
STRLEN := tmp4
; Absolute variables used in the code
.bss
ERROR: .res 1 ; Error code
DRAWINDEX: .res 1 ; Pen to use for drawing
VIEWPAGEL: .res 1
VIEWPAGEH: .res 1
DRAWPAGEL: .res 1
DRAWPAGEH: .res 1
; Text output stuff
TEXTMAGX: .res 1
TEXTMAGY: .res 1
TEXTDIR: .res 1
BGINDEX: .res 1 ; Pen to use for text background
; Double buffer IRQ stuff
DRAWPAGE: .res 1
SWAPREQUEST: .res 1
; 8 rows with (one offset-byte plus 20 character bytes plus one fill-byte) plus one 0-offset-byte.
; (As an experiment, the fill-byte isn't being generated.
; It might not be needed to work around a Suzy bug.)
text_bitmap: .res 8*(1+20+1)+1
; Constants and tables
.rodata
DEFPALETTE: .byte >$011
.byte >$34d
.byte >$9af
.byte >$9b8
.byte >$777
.byte >$335
.byte >$448
.byte >$75e
.byte >$d5f
.byte >$c53
.byte >$822
.byte >$223
.byte >$484
.byte >$8e5
.byte >$cf5
.byte >$fff
.byte <$011
.byte <$34d
.byte <$9af
.byte <$9b8
.byte <$777
.byte <$335
.byte <$448
.byte <$75e
.byte <$d5f
.byte <$c53
.byte <$822
.byte <$223
.byte <$484
.byte <$8e5
.byte <$cf5
.byte <$fff
PALETTESIZE = * - DEFPALETTE
.code
; ------------------------------------------------------------------------
; INSTALL routine. Is called after the driver is loaded into memory. May
; initialize anything that has to be done just once. Is probably empty
; most of the time.
;
; Must set an error code: NO
;
INSTALL:
lda #1
sta TEXTMAGX
sta TEXTMAGY
stz BGINDEX
stz DRAWPAGE
stz SWAPREQUEST
lda libref
ldx libref+1
sta ptr1
stx ptr1+1
ldy #1
lda #<irq
sta (ptr1),y
iny
lda #>irq
sta (ptr1),y
lda #$4C ; Jump opcode
sta (ptr1) ; Activate IRQ routine
rts
; ------------------------------------------------------------------------
; UNINSTALL routine. Is called before the driver is removed from memory. May
; clean up anything done by INSTALL but is probably empty most of the time.
;
; Must set an error code: NO
;
UNINSTALL:
lda libref
ldx libref+1
sta ptr1
stx ptr1+1
lda #$60 ; RTS opcode
sta (ptr1) ; Disable IRQ routine
rts
; ------------------------------------------------------------------------
; INIT: Changes an already installed device from text mode to graphics
; mode.
; Note that INIT/DONE may be called multiple times while the driver
; is loaded, while INSTALL is only called once, so any code that is needed
; to initializes variables and so on must go here. Setting palette and
; clearing the screen is not needed because this is called by the graphics
; kernel later.
; The graphics kernel will never call INIT when a graphics mode is already
; active, so there is no need to protect against that.
;
; Must set an error code: YES
;
INIT:
; Enable interrupts for VBL
lda #$80
tsb VTIMCTLA
; Set up collision buffer to $A058
lda #$58
sta COLLBASL
lda #$A0
sta COLLBASH
; Put collision index before sprite data
lda #$FF
sta COLLOFFL
lda #$FF
sta COLLOFFH
; Done, reset the error code
lda #TGI_ERR_OK
sta ERROR
rts
; ------------------------------------------------------------------------
; DONE: Will be called to switch the graphics device back into text mode.
; The graphics kernel will never call DONE when no graphics mode is active,
; so there is no need to protect against that.
;
; Must set an error code: NO
;
DONE:
rts
; ------------------------------------------------------------------------
; GETERROR: Return the error code in A and clear it.
GETERROR:
ldx #TGI_ERR_OK
lda ERROR
stx ERROR
rts
; ------------------------------------------------------------------------
; CONTROL: Platform/driver specific entry point.
;
; Must set an error code: YES
;
; The TGI lacks a way to draw sprites. As that functionality is vital to
; Lynx games we borrow this CONTROL function to implement the still
; missing tgi_draw_sprite funtion. To use this in your C-program
; do a #define tgi_draw_sprite(spr) tgi_ioctl(0, spr)
;
; To do a flip-screen call tgi_ioctl(1, 0)
;
; To set the background index for text outputs call tgi_ioctl(2, bgindex)
;
; To set the frame rate for the display hardware call tgi_ioctl(3, rate)
;
; To check if the drawing engine is busy with the previous swap you can
; call tgi_ioctl(4, 0). It returns 0 if idle and 1 if busy
;
; To update displays you can call tgi_ioctl(4, 1) it will wait for the
; next VBL interrupt and swap draw and view buffers.
;
; Activate or deactivate collision detection by calling tgi_ioctl(5, 0/1).
CONTROL:
pha ; Almost all control routines succeed
lda #TGI_ERR_OK
sta ERROR
pla
cmp #5
bne ControlSwap
lda ptr1 ; Activate/deactivate collision detection
bne @L0
lda #%00000001 ; tgi_clear does not erase collision buffer
sta cls_sprite
lda #%00100000
sta cls_sprite+2
lda __sprsys
ora #$20
bra @L1
@L0: lda #%00000000 ; tgi_clear erases collision buffer
sta cls_sprite
sta cls_sprite+2
lda __sprsys
and #$df
@L1: sta __sprsys
sta SPRSYS
rts
ControlSwap:
cmp #4
bne ControlFramerate
lda ptr1 ; Swap request
bne @L0
lda SWAPREQUEST
rts
@L0: sta SWAPREQUEST
rts
ControlFramerate:
cmp #3
bne ControlTextBG
lda ptr1
cmp #75 ; Set framerate
beq rate75
cmp #60
beq rate60
cmp #50
beq rate50
lda #TGI_ERR_INV_ARG
sta ERROR
rts
rate50: lda #$bd ; 50 Hz
ldx #$31
bra setRate
rate60: lda #$9e ; 60 Hz
ldx #$29
bra setRate
rate75: lda #$7e ; 75 Hz
ldx #$20
setRate:
sta HTIMBKUP
stx PBKUP
rts
ControlTextBG:
cmp #2
bne ControlFlipScreen
lda ptr1 ; Set text background color
sta BGINDEX
rts
ControlFlipScreen:
cmp #1
bne ControlDrawSprite
lda __sprsys ; Flip screen
eor #8
sta __sprsys
sta SPRSYS
lda __viddma
eor #2
sta __viddma
sta DISPCTL
ldy VIEWPAGEL
ldx VIEWPAGEH
and #2
beq NotFlipped
clc
tya
adc #<8159
tay
txa
adc #>8159
tax
NotFlipped:
sty DISPADRL
stx DISPADRH
rts
ControlDrawSprite:
lda ptr1 ; Get the sprite address
ldx ptr1+1
draw_sprite: ; Draw it in render buffer
sta SCBNEXTL
stx SCBNEXTH
lda DRAWPAGEL
ldx DRAWPAGEH
sta VIDBASL
stx VIDBASH
lda #1
sta SPRGO
stz SDONEACK
@L0: stz CPUSLEEP
lda SPRSYS
lsr
bcs @L0
stz SDONEACK
lda #TGI_ERR_OK
sta ERROR
rts
; ------------------------------------------------------------------------
; CLEAR: Clears the screen.
;
; Must set an error code: NO
;
.rodata
pixel_bitmap:
.byte 3,%10000100,%00000000, $0 ; A pixel bitmap
.data
cls_coll:
.byte 0
cls_sprite:
.byte %00000001 ; A pixel sprite
.byte %00010000
.byte %00100000
.addr 0,pixel_bitmap
.word 0
.word 0
.word $a000 ; 160
.word $6600 ; 102
.byte $00
.code
CLEAR: lda #<cls_sprite
ldx #>cls_sprite
bra draw_sprite
; ------------------------------------------------------------------------
; SETVIEWPAGE: Set the visible page. Called with the new page in A (0..n).
; The page number is already checked to be valid by the graphics kernel.
;
; Must set an error code: NO (will only be called if page ok)
;
; It is a good idea to call this function during the vertical blanking
; period. If you call it in the middle of the screen update then half of
; the drawn frame will be from the old buffer and the other half is
; from the new buffer. This is usually noticed by the user.
SETVIEWPAGE:
cmp #1
beq @L1 ; page == maxpages-1
ldy #<$e018 ; page 0
ldx #>$e018
bra @L2
@L1:
ldy #<$c038 ; page 1
ldx #>$c038
@L2:
sty VIEWPAGEL ; Save viewpage for getpixel
stx VIEWPAGEH
lda __viddma ; Process flipped displays
and #2
beq @L3
clc
tya
adc #<8159
tay
txa
adc #>8159
tax
@L3:
sty DISPADRL ; $FD94
stx DISPADRH ; $FD95
rts
; ------------------------------------------------------------------------
; SETDRAWPAGE: Set the drawable page. Called with the new page in A (0..n).
; The page number is already checked to be valid by the graphics kernel.
;
; Must set an error code: NO (will only be called if page ok)
;
SETDRAWPAGE:
cmp #1
beq @L1 ; page == maxpages-1
lda #<$e018 ; page 0
ldx #>$e018
bra @L2
@L1:
lda #<$c038 ; page 1
ldx #>$c038
@L2:
sta DRAWPAGEL
stx DRAWPAGEH
rts
irq:
lda INTSET ; Poll all pending interrupts
and #VBL_INTERRUPT
beq @L0 ; Exit if not a VBL interrupt
lda SWAPREQUEST
beq @L0
lda DRAWPAGE
jsr SETVIEWPAGE
lda DRAWPAGE
eor #1
sta DRAWPAGE
jsr SETDRAWPAGE
stz SWAPREQUEST
@L0:
clc
rts
; ------------------------------------------------------------------------
; SETCOLOR: Set the drawing color (in A). The new color is already checked
; to be in a valid range (0..maxcolor-1).
;
; Must set an error code: NO (will only be called if color ok)
;
SETCOLOR:
sta DRAWINDEX
rts
; ------------------------------------------------------------------------
; SETPALETTE: Set the palette (not available with all drivers/hardware).
; A pointer to the palette is passed in ptr1. Must set an error if palettes
; are not supported
;
; Must set an error code: YES
;
SETPALETTE:
ldy #31
@L1: lda (ptr1),y
sta GCOLMAP,y ; $FDA0
dey
bpl @L1
; Done, reset the error code
lda #TGI_ERR_OK
sta ERROR
rts
; ------------------------------------------------------------------------
; GETPALETTE: Return the current palette in A/X. Even drivers that cannot
; set the palette should return the default palette here, so there's no
; way for this function to fail.
;
; Must set an error code: NO
;
GETPALETTE:
lda #<GCOLMAP ; $FDA0
ldx #>GCOLMAP
rts
; ------------------------------------------------------------------------
; GETDEFPALETTE: Return the default palette for the driver in A/X. All
; drivers should return something reasonable here, even drivers that don't
; support palettes, otherwise the caller has no way to determine the colors
; of the (not changeable) palette.
;
; Must set an error code: NO (all drivers must have a default palette)
;
GETDEFPALETTE:
lda #<DEFPALETTE
ldx #>DEFPALETTE
rts
; ------------------------------------------------------------------------
; SETPIXEL: Draw one pixel at X1/Y1 = ptr1/ptr2 with the current drawing
; color. The coordinates passed to this function are never outside the
; visible screen area, so there is no need for clipping inside this function.
;
; Must set an error code: NO
;
.data
pixel_coll:
.byte 0
pixel_sprite:
.byte %00000001 ; A pixel sprite
.byte %00010000
.byte %00100000
.addr 0,pixel_bitmap
pix_x: .word 0
pix_y: .word 0
.word $100
.word $100
pix_c: .byte $00
.code
SETPIXEL:
lda X1
sta pix_x
lda Y1
sta pix_y
lda DRAWINDEX
sta pix_c
lda #<pixel_sprite
ldx #>pixel_sprite
jmp draw_sprite
; ------------------------------------------------------------------------
; GETPIXEL: Read the color value of a pixel and return it in A/X. The
; coordinates passed to this function are never outside the visible screen
; area, so there is no need for clipping inside this function.
GETPIXEL:
lda Y1
sta MATHD ; Hardware multiply
stz MATHC
lda #80
sta MATHB
stz MATHA
lda X1
lsr A
php
tay
clc
lda VIEWPAGEL
adc MATHH
sta ptr1
lda VIEWPAGEH
adc MATHG
sta ptr1+1
ldx #0
lda #15
sta MAPCTL
lda (ptr1),y
tay
lda #$0c
sta MAPCTL
tya
plp
bcc @L1
and #$f
rts
@L1: lsr A
lsr A
lsr A
lsr A
rts
; ------------------------------------------------------------------------
; LINE: Draw a line from X1/Y1 to X2/Y2, where X1/Y1 = ptr1/ptr2 and
; X2/Y2 = ptr3/ptr4 using the current drawing color.
;
; Must set an error code: NO
;
.data
line_coll:
.byte 0
line_sprite:
.byte 0 ; Will be replaced by the code
.byte %00110000
.byte %00100000
.word 0,pixel_bitmap
line_x:
.word 0
line_y:
.word 0
line_sx:
.word $100
line_sy:
.word $100
.word 0
line_tilt:
.word 0
line_c:
.byte $e
.code
LINE:
lda DRAWINDEX
sta line_c
stz line_sx
stz line_sy
sec
lda X2
sbc X1
lda X2+1
sbc X1+1
bpl @L1
lda X1
ldx X2
sta X2
stx X1
lda X1+1
ldx X2+1
sta X2+1
stx X1+1
lda Y1
ldx Y2
sta Y2
stx Y1
lda Y1+1
ldx Y2+1
sta Y2+1
stx Y1+1
@L1:
lda #%00000000 ; Not flipped
sta line_sprite
sec
lda Y2
sbc Y1
sta Y2
lda Y2+1
sbc Y1+1
sta Y2+1
bpl @L2
sec
lda #0
sbc Y2
sta Y2
lda #0
sbc Y2+1
sta Y2+1
lda #%00010000 ; Vertical flip
sta line_sprite
@L2:
lda X1
sta line_x
lda X1+1
sta line_x+1
lda Y1
sta line_y
lda Y1+1
sta line_y+1
lda Y2
ina
sta line_sy+1
sta MATHP ; hardware divide
stz MATHN
stz MATHH
stz MATHG
sec
lda X2
sbc X1
ina
sta MATHF
stz MATHE
@L3:
lda SPRSYS
bmi @L3 ; wait for math done (bit 7 of sprsys)
lda MATHC
sta line_tilt
lda MATHB
sta line_tilt+1
bne @L4
lda #1
sta line_sx+1
bra @L6
@L4:
bit line_tilt
bpl @L5
ina
@L5:
sta line_sx+1
@L6:
lda #<line_sprite
ldx #>line_sprite
jmp draw_sprite
; ------------------------------------------------------------------------
; BAR: Draw a filled rectangle with the corners X1/Y1, X2/Y2, where
; X1/Y1 = ptr1/ptr2 and X2/Y2 = ptr3/ptr4 using the current drawing color.
; Contrary to most other functions, the graphics kernel will sort and clip
; the coordinates before calling the driver, so on entry the following
; conditions are valid:
; X1 <= X2
; Y1 <= Y2
; (X1 >= 0) && (X1 < XRES)
; (X2 >= 0) && (X2 < XRES)
; (Y1 >= 0) && (Y1 < YRES)
; (Y2 >= 0) && (Y2 < YRES)
;
; Must set an error code: NO
;
.data
bar_coll:
.byte 0
bar_sprite:
.byte %00000001 ; A pixel sprite
.byte %00010000
.byte %00100000
.addr 0,pixel_bitmap
bar_x: .word 0
bar_y: .word 0
bar_sx: .word $0100
bar_sy: .word $0100
bar_c: .byte $00
.code
BAR: lda X1
sta bar_x
lda Y1
sta bar_y
lda X2
sec
sbc X1
ina
sta bar_sx+1
lda Y2
sec
sbc Y1
ina
sta bar_sy+1
lda DRAWINDEX
sta bar_c
lda #<bar_sprite
ldx #>bar_sprite
jmp draw_sprite
; ------------------------------------------------------------------------
; TEXTSTYLE: Set the style used when calling OUTTEXT. Text scaling in X and Y
; direction is passend in X/Y, the text direction is passed in A.
;
; Must set an error code: NO
;
TEXTSTYLE:
stx TEXTMAGX
sty TEXTMAGY
sta TEXTDIR
rts
; ------------------------------------------------------------------------
; OUTTEXT: Output text at X/Y = ptr1/ptr2 using the current color and the
; current text style. The text to output is given as a zero-terminated
; string with address in ptr3.
;
; Must set an error code: NO
;
OUTTEXT:
lda TEXTMAGX ; Scale sprite
sta text_sx+1
lda TEXTMAGY
sta text_sy+1
lda BGINDEX
beq @L1 ; Choose opaque black sprite?
lda #$04 ; No, choose normal sprite
@L1:
sta text_sprite
lda DRAWINDEX ; Set color
asl
asl
asl
asl
ora BGINDEX
sta text_c
lda X1 ; Set start position
sta text_x
lda X1+1
sta text_x+1
lda Y1
sta text_y
lda Y1+1
sta text_y+1
ldy #<-1 ; Calculate string length
@L2:
iny
lda (STRPTR),y
bne @L2
cpy #20
bmi @L3
ldy #20
@L3:
sty STRLEN
tya
bne @L4
rts ; Zero-length string
@L4:
iny ; Prepare text_bitmap
; The next instruction is commented because the code won't include a fill-byte.
; iny
sty STROFF
ldy #8-1 ; 8 pixel lines per character
ldx #$00
clc
@L5:
lda STROFF
sta text_bitmap,x
txa
adc STROFF
tax
; This was the fill-byte.
; lda #$FF
; sta text_bitmap-1,x
dey
bpl @L5
stz text_bitmap,x
stz tmp2
iny ;(ldy #$00)
@L6:
lda (STRPTR),y
sty tmp1
sub #' ' ; (ch - ' ') * 8
stz FONTOFF+1
asl
asl
rol FONTOFF+1
asl
rol FONTOFF+1
;clc ; (cleared by rol)
adc #<font ; Choose font
sta FONTOFF
lda FONTOFF+1
adc #>font
sta FONTOFF+1
; And now, copy the 8 bytes of that glyph.
ldx tmp2
inx
stx tmp2
; Draw char. from top to bottom, reading char-data from offset 8-1 to offset 0.
ldy #8-1
@L7:
lda (FONTOFF),y ; *chptr
sta text_bitmap,x ; textbuf[y*(1+len+1)+1+x]
txa
adc STROFF
tax
dey
bpl @L7
; Goto next char.
ldy tmp1
iny
dec STRLEN
bne @L6
lda #<text_sprite
ldx #>text_sprite
jmp draw_sprite
.data
text_coll:
.byte 0
text_sprite:
.byte $00,$90,$20
.addr 0, text_bitmap
text_x:
.word 0
text_y:
.word 0
text_sx:
.word $100
text_sy:
.word $100
text_c:
.byte 0
.rodata
; The Font
; 96 characters from ASCII 32 to 127
; 8 pixels wide, 8 pixels high
; bit value 0 = foreground, bit value 1 = background / transparent
font:
; VERSAIL
.byte $FF, $FF, $FF, $FF, $FF, $FF, $FF, $FF ;32
.byte $FF, $E7, $FF, $FF, $E7, $E7, $E7, $E7 ;33
.byte $FF, $FF, $FF, $FF, $FF, $99, $99, $99 ;34
.byte $FF, $D7, $D7, $01, $D7, $01, $D7, $D7 ;35
.byte $FF, $E7, $83, $F9, $C3, $9F, $C1, $E7 ;36
.byte $FF, $B9, $99, $CF, $E7, $F3, $99, $9D ;37
.byte $81, $B3, $31, $8F, $87, $33, $87, $FF ;38
.byte $FF, $FF, $FF, $FF, $FF, $E7, $F3, $F9 ;39
.byte $FF, $F3, $E7, $CF, $CF, $CF, $E7, $F3 ;40
.byte $FF, $CF, $E7, $F3, $F3, $F3, $E7, $CF ;41
.byte $FF, $99, $C3, $81, $C3, $99, $FF, $FF ;42
.byte $FF, $FF, $E7, $E7, $81, $E7, $E7, $FF ;43
.byte $CF, $E7, $E7, $FF, $FF, $FF, $FF, $FF ;44
.byte $FF, $FF, $FF, $FF, $81, $FF, $FF, $FF ;45
.byte $FF, $E7, $E7, $FF, $FF, $FF, $FF, $FF ;46
.byte $FF, $BF, $9F, $CF, $E7, $F3, $F9, $FD ;47
.byte $FF, $C3, $99, $99, $89, $91, $99, $C3 ;48
.byte $FF, $81, $E7, $E7, $E7, $C7, $E7, $E7 ;49
.byte $FF, $81, $9F, $CF, $F3, $F9, $99, $C3 ;50
.byte $FF, $C3, $99, $F9, $E3, $F9, $99, $C3 ;51
.byte $FF, $F3, $F3, $01, $33, $C3, $E3, $FB ;52
.byte $FF, $C3, $99, $F9, $F9, $83, $9F, $81 ;53
.byte $FF, $C3, $99, $99, $83, $9F, $99, $C3 ;54
.byte $FF, $E7, $E7, $E7, $E7, $F3, $99, $81 ;55
.byte $FF, $C3, $99, $99, $C3, $99, $99, $C3 ;56
.byte $FF, $C3, $99, $F9, $C1, $99, $99, $C3 ;57
.byte $FF, $FF, $E7, $FF, $FF, $E7, $FF, $FF ;58
.byte $CF, $E7, $E7, $FF, $FF, $E7, $FF, $FF ;59
.byte $FF, $F1, $E7, $CF, $9F, $CF, $E7, $F1 ;60
.byte $FF, $FF, $FF, $81, $FF, $81, $FF, $FF ;61
.byte $FF, $8F, $E7, $F3, $F9, $F3, $E7, $8F ;62
.byte $FF, $E7, $FF, $E7, $F3, $F9, $99, $C3 ;63
.byte $FF, $C3, $9D, $9F, $91, $91, $99, $C3 ;0
.byte $FF, $99, $99, $99, $81, $99, $C3, $E7 ;1
.byte $FF, $83, $99, $99, $83, $99, $99, $83 ;2
.byte $FF, $C3, $99, $9F, $9F, $9F, $99, $C3 ;3
.byte $FF, $87, $93, $99, $99, $99, $93, $87 ;4
.byte $FF, $81, $9F, $9F, $87, $9F, $9F, $81 ;5
.byte $FF, $9F, $9F, $9F, $87, $9F, $9F, $81 ;6
.byte $FF, $C3, $99, $99, $91, $9F, $99, $C3 ;7
.byte $FF, $99, $99, $99, $81, $99, $99, $99 ;8
.byte $FF, $C3, $E7, $E7, $E7, $E7, $E7, $C3 ;9
.byte $FF, $C7, $93, $F3, $F3, $F3, $F3, $E1 ;10
.byte $FF, $99, $93, $87, $8F, $87, $93, $99 ;11
.byte $FF, $81, $9F, $9F, $9F, $9F, $9F, $9F ;12
.byte $FF, $39, $39, $39, $29, $01, $11, $39 ;13
.byte $FF, $99, $99, $91, $81, $81, $89, $99 ;14
.byte $FF, $C3, $99, $99, $99, $99, $99, $C3 ;15
.byte $FF, $9F, $9F, $9F, $83, $99, $99, $83 ;16
.byte $FF, $F1, $C3, $99, $99, $99, $99, $C3 ;17
.byte $FF, $99, $93, $87, $83, $99, $99, $83 ;18
.byte $FF, $C3, $99, $F9, $C3, $9F, $99, $C3 ;19
.byte $FF, $E7, $E7, $E7, $E7, $E7, $E7, $81 ;20
.byte $FF, $C3, $99, $99, $99, $99, $99, $99 ;21
.byte $FF, $E7, $C3, $99, $99, $99, $99, $99 ;22
.byte $FF, $39, $11, $01, $29, $39, $39, $39 ;23
.byte $FF, $99, $99, $C3, $E7, $C3, $99, $99 ;24
.byte $FF, $E7, $E7, $E7, $C3, $99, $99, $99 ;25
.byte $FF, $81, $9F, $CF, $E7, $F3, $F9, $81 ;26
.byte $FF, $C3, $CF, $CF, $CF, $CF, $CF, $C3 ;27
.byte $FF, $03, $9D, $CF, $83, $CF, $ED, $F3 ;28
.byte $FF, $C3, $F3, $F3, $F3, $F3, $F3, $C3 ;29
.byte $E7, $E7, $E7, $E7, $81, $C3, $E7, $FF ;30
.byte $FF, $DF, $9F, $01, $01, $9F, $DF, $FF ;31
; gemena
.byte $FF, $C3, $9D, $9F, $91, $91, $99, $C3 ;224
.byte $FF, $C1, $99, $C1, $F9, $C3, $FF, $FF ;225
.byte $FF, $83, $99, $99, $83, $9F, $9F, $FF ;226
.byte $FF, $C3, $9F, $9F, $9F, $C3, $FF, $FF ;227
.byte $FF, $C1, $99, $99, $C1, $F9, $F9, $FF ;228
.byte $FF, $C3, $9F, $81, $99, $C3, $FF, $FF ;229
.byte $FF, $E7, $E7, $E7, $C1, $E7, $F1, $FF ;230
.byte $83, $F9, $C1, $99, $99, $C1, $FF, $FF ;231
.byte $FF, $99, $99, $99, $83, $9F, $9F, $FF ;232
.byte $FF, $C3, $E7, $E7, $C7, $FF, $E7, $FF ;233
.byte $C3, $F9, $F9, $F9, $F9, $FF, $F9, $FF ;234
.byte $FF, $99, $93, $87, $93, $9F, $9F, $FF ;235
.byte $FF, $C3, $E7, $E7, $E7, $E7, $C7, $FF ;236
.byte $FF, $39, $29, $01, $83, $93, $FF, $FF ;237
.byte $FF, $99, $99, $99, $99, $83, $FF, $FF ;238
.byte $FF, $C3, $99, $99, $99, $C3, $FF, $FF ;239
.byte $9F, $9F, $83, $99, $99, $83, $FF, $FF ;240
.byte $F9, $F9, $C1, $99, $99, $C1, $FF, $FF ;241
.byte $FF, $9F, $9F, $9F, $99, $83, $FF, $FF ;242
.byte $FF, $83, $F9, $C3, $9F, $C1, $FF, $FF ;243
.byte $FF, $F1, $E7, $E7, $E7, $81, $E7, $FF ;244
.byte $FF, $C1, $99, $99, $99, $99, $FF, $FF ;245
.byte $FF, $E7, $C3, $99, $99, $99, $FF, $FF ;246
.byte $FF, $93, $83, $01, $29, $39, $FF, $FF ;247
.byte $FF, $99, $C3, $E7, $C3, $99, $FF, $FF ;248
.byte $87, $F3, $C1, $99, $99, $99, $FF, $FF ;249
.byte $FF, $81, $CF, $E7, $F3, $81, $FF, $FF ;250
.byte $FF, $C3, $CF, $CF, $CF, $CF, $CF, $C3 ;251
.byte $FF, $03, $9D, $CF, $83, $CF, $ED, $F3 ;252
.byte $FF, $C3, $F3, $F3, $F3, $F3, $F3, $C3 ;253
.byte $E7, $E7, $E7, $E7, $81, $C3, $E7, $FF ;254
.byte $FF, $DF, $9F, $01, $01, $9F, $DF, $FF ;255