ti/lcd: allow for fonts smaller than 5 pixels

That's a lot of code for such a small change, but there's a big difference
between 5 pixels and 4 pixels: 4 pixels requires compositing.
This commit is contained in:
Virgil Dupras 2019-11-09 22:55:15 -05:00
parent 6d9f96aee6
commit c4658591bd
3 changed files with 141 additions and 41 deletions

Binary file not shown.

View File

@ -26,6 +26,25 @@
; when active row is 0, Z is FNT_HEIGHT+1, when row is 1, Z is (FNT_HEIGHT+1)*2, ; when active row is 0, Z is FNT_HEIGHT+1, when row is 1, Z is (FNT_HEIGHT+1)*2,
; When row is 8, Z is 0. ; When row is 8, Z is 0.
; ;
; *** 6/8 bit columns and smaller fonts ***
;
; If your glyphs, including padding, are 6 or 8 pixels wide, you're in luck
; because pushing them to the LCD can be done in a very efficient manner.
; Unfortunately, this makes the LCD unsuitable for a Collapse OS shell: 6
; pixels per glyph gives us only 16 characters per line, which is hardly
; usable.
;
; This is why we have this buffering system. How it works is that we're always
; in 8-bit mode and we hold the whole area (8 pixels wide by FNT_HEIGHT high)
; in memory. When we want to put a glyph to screen, we first read the contents
; of that area, then add our new glyph, offsetted and masked, to that buffer,
; then push the buffer back to the LCD. If the glyph is split, move to the next
; area and finish the job.
;
; That being said, it's important to define clearly what CURX and CURY variable
; mean. Those variable keep track of the current position *in pixels*, in both
; axes.
;
; *** Requirements *** ; *** Requirements ***
; fnt/mgm ; fnt/mgm
; ;
@ -37,7 +56,9 @@
.equ LCD_CMD_8BIT 0x01 .equ LCD_CMD_8BIT 0x01
.equ LCD_CMD_DISABLE 0x02 .equ LCD_CMD_DISABLE 0x02
.equ LCD_CMD_ENABLE 0x03 .equ LCD_CMD_ENABLE 0x03
.equ LCD_CMD_XDEC 0x04
.equ LCD_CMD_XINC 0x05 .equ LCD_CMD_XINC 0x05
.equ LCD_CMD_YDEC 0x06
.equ LCD_CMD_YINC 0x07 .equ LCD_CMD_YINC 0x07
.equ LCD_CMD_COL 0x20 .equ LCD_CMD_COL 0x20
.equ LCD_CMD_ZOFFSET 0x40 .equ LCD_CMD_ZOFFSET 0x40
@ -45,19 +66,22 @@
.equ LCD_CMD_CONTRAST 0xc0 .equ LCD_CMD_CONTRAST 0xc0
; *** Variables *** ; *** Variables ***
; Current row being written on. In terms of pixels, not of glyphs. During a ; Current Y position on the LCD, that is, where re're going to spit our next
; linefeed, this increases by FNT_HEIGHT+1. ; glyph.
.equ LCD_CURROW LCD_RAMSTART .equ LCD_CURY LCD_RAMSTART
; Current column ; Current X position
.equ LCD_CURCOL @+1 .equ LCD_CURX @+1
.equ LCD_RAMEND @+1 ; two pixel buffers that are 8 pixels wide (1b) by FNT_HEIGHT pixels high.
; This is where we compose our resulting pixels blocks when spitting a glyph.
.equ LCD_BUF @+1
.equ LCD_RAMEND @+FNT_HEIGHT*2
; *** Code *** ; *** Code ***
lcdInit: lcdInit:
; Initialize variables ; Initialize variables
xor a xor a
ld (LCD_CURROW), a ld (LCD_CURY), a
ld (LCD_CURCOL), a ld (LCD_CURX), a
; Clear screen ; Clear screen
call lcdClrScr call lcdClrScr
@ -82,12 +106,8 @@ lcdInit:
ld a, LCD_CMD_CONTRAST+0x34 ld a, LCD_CMD_CONTRAST+0x34
call lcdCmd call lcdCmd
; Enable 6-bit mode. ; Enable 8-bit mode.
ld a, LCD_CMD_6BIT ld a, LCD_CMD_8BIT
call lcdCmd
; Enable X-increment mode
ld a, LCD_CMD_XINC
call lcdCmd call lcdCmd
ret ret
@ -109,10 +129,15 @@ lcdCmd:
jr lcdWait jr lcdWait
; Send data A to LCD ; Send data A to LCD
lcdData: lcdDataSet:
out (LCD_PORT_DATA), a out (LCD_PORT_DATA), a
jr lcdWait jr lcdWait
; Get data from LCD into A
lcdDataGet:
in a, (LCD_PORT_DATA)
jr lcdWait
; Turn LCD off ; Turn LCD off
lcdOff: lcdOff:
push af push af
@ -140,52 +165,126 @@ lcdSetRow:
pop af pop af
ret ret
; Send the 5x7 glyph that HL points to to the LCD, at its current position. ; Send the glyph that HL points to to the LCD, at its current position.
; After having called this, the LCD's position will have advanced by one ; After having called this, the LCD's position will have advanced by one
; position ; position
lcdSendGlyph: lcdSendGlyph:
push af push af
push bc push bc
push hl push hl
push ix
ld a, (LCD_CURROW) ld a, (LCD_CURY)
call lcdSetRow call lcdSetRow
ld a, (LCD_CURCOL) ld a, (LCD_CURX)
srl a \ srl a \ srl a ; div by 8
call lcdSetCol call lcdSetCol
; First operation: read the LCD memory for the "left" side of the
; buffer. We assume the right side to always be empty, so we don't
; read it. After having read each line, compose it with glyph line at
; HL
; Before we start, what is our bit offset?
ld a, (LCD_CURX)
and 0b111
; that's our offset, store it in C
ld c, a
ld a, LCD_CMD_XINC
call lcdCmd
ld ix, LCD_BUF
ld b, FNT_HEIGHT ld b, FNT_HEIGHT
.loop: ; A dummy read is needed after a movement.
call lcdDataGet
.loop1:
; let's go get that glyph data
ld a, (hl) ld a, (hl)
ld (ix), a
call .shiftIX
; now let's go get existing pixel on LCD
call lcdDataGet
; and now let's do some compositing!
or (ix)
ld (ix), a
inc hl inc hl
call lcdData inc ix
djnz .loop djnz .loop1
; Buffer set! now let's send it.
ld a, (LCD_CURY)
call lcdSetRow
ld hl, LCD_BUF
ld b, FNT_HEIGHT
.loop2:
ld a, (hl)
call lcdDataSet
inc hl
djnz .loop2
; And finally, let's send the "right side" of the buffer
ld a, (LCD_CURY)
call lcdSetRow
ld a, (LCD_CURX)
srl a \ srl a \ srl a ; div by 8
inc a
call lcdSetCol
ld hl, LCD_BUF+FNT_HEIGHT
ld b, FNT_HEIGHT
.loop3:
ld a, (hl)
call lcdDataSet
inc hl
djnz .loop3
; Increase column and wrap if necessary ; Increase column and wrap if necessary
ld a, (LCD_CURCOL) ld a, (LCD_CURX)
inc a add a, FNT_WIDTH+1
ld (LCD_CURCOL), a ld (LCD_CURX), a
cp 16 cp 96-FNT_WIDTH
jr nz, .skip jr c, .skip ; A < 96-FNT_WIDTH
call lcdLinefeed call lcdLinefeed
.skip: .skip:
pop ix
pop hl pop hl
pop bc pop bc
pop af pop af
ret ret
; Shift glyph in (IX) to the right C times, sending carry into (IX+FNT_HEIGHT)
.shiftIX:
dec c \ inc c
ret z ; zero? nothing to do
push bc ; --> lvl 1
xor a
ld b, a
ld a, (ix)
; TODO: support SRL (IX) and RR (IX) in zasm
.shiftLoop:
srl a
rr b
dec c
jr nz, .shiftLoop
ld (ix), a
ld a, b
ld (ix+FNT_HEIGHT), a
pop bc ; <-- lvl 1
ret
; Changes the current line and go back to leftmost column ; Changes the current line and go back to leftmost column
lcdLinefeed: lcdLinefeed:
push af push af
ld a, (LCD_CURROW) ld a, (LCD_CURY)
call .addFntH call .addFntH
ld (LCD_CURROW), a ld (LCD_CURY), a
call lcdClrLn call lcdClrLn
; Now, lets set Z offset which is CURROW+FNT_HEIGHT+1 ; Now, lets set Z offset which is CURROW+FNT_HEIGHT+1
call .addFntH call .addFntH
add a, LCD_CMD_ZOFFSET add a, LCD_CMD_ZOFFSET
call lcdCmd call lcdCmd
xor a xor a
ld (LCD_CURCOL), a ld (LCD_CURX), a
pop af pop af
ret ret
.addFntH: .addFntH:
@ -201,8 +300,6 @@ lcdLinefeed:
lcdClrX: lcdClrX:
push af push af
call lcdSetRow call lcdSetRow
ld a, LCD_CMD_8BIT
call lcdCmd
.outer: .outer:
push bc ; --> lvl 1 push bc ; --> lvl 1
ld b, 11 ld b, 11
@ -211,16 +308,14 @@ lcdClrX:
xor a xor a
call lcdSetCol call lcdSetCol
.inner: .inner:
call lcdData call lcdDataSet
djnz .inner djnz .inner
ld a, LCD_CMD_XINC ld a, LCD_CMD_XINC
call lcdCmd call lcdCmd
xor a xor a
call lcdData call lcdDataSet
pop bc ; <-- lvl 1 pop bc ; <-- lvl 1
djnz .outer djnz .outer
ld a, LCD_CMD_6BIT
call lcdCmd
pop af pop af
ret ret
@ -251,10 +346,10 @@ lcdPutC:
pop hl pop hl
ret ret
.bs: .bs:
ld a, (LCD_CURCOL) ld a, (LCD_CURX)
or a or a
ret z ; going back one line is too complicated. ret z ; going back one line is too complicated.
; not implemented yet ; not implemented yet
dec a sub FNT_WIDTH+1
ld (LCD_CURCOL), a ld (LCD_CURX), a
ret ret

View File

@ -3,11 +3,16 @@ use strict;
# This script converts "space-dot" fonts to binary "glyph rows". One byte for # This script converts "space-dot" fonts to binary "glyph rows". One byte for
# each row. In a 5x7 font, each glyph thus use 7 bytes. # each row. In a 5x7 font, each glyph thus use 7 bytes.
# Resulting bytes are aligned to the **left** of the byte. Therefore, for
# a 5-bit wide char, ". . ." translates to 0b10101000
# Left-aligned bytes are easier to work with when compositing glyphs.
my $fn = @ARGV[0]; my $fn = @ARGV[0];
unless ($fn =~ /.*(\d)x(\d)\.txt/) { die "$fn isn't a font filename" }; unless ($fn =~ /.*(\d)x(\d)\.txt/) { die "$fn isn't a font filename" };
my ($width, $height) = ($1, $2); my ($width, $height) = ($1, $2);
if ($width > 8) { die "Can't have a width > 8"; }
print STDERR "Reading a $width x $height font.\n"; print STDERR "Reading a $width x $height font.\n";
my $handle; my $handle;
@ -21,9 +26,9 @@ while (<$handle>) {
unless (/( |\.){${width}}\n/) { die "Invalid line format '$_'"; } unless (/( |\.){${width}}\n/) { die "Invalid line format '$_'"; }
my @line = split //, $_; my @line = split //, $_;
my $num = 0; my $num = 0;
for (my $i=$width-1; $i>=0; $i--) { for (my $i=0; $i<8; $i++) {
if (@line[$width-$i-1] eq '.') { if (@line[$i] eq '.') {
$num += (1 << $i); $num += (1 << (7-$i));
} }
} }
print pack('C', $num); print pack('C', $num);