2019-11-07 11:52:29 -05:00
|
|
|
; kbd
|
|
|
|
;
|
|
|
|
; Control TI-84+'s keyboard.
|
|
|
|
;
|
|
|
|
; *** Constants ***
|
|
|
|
.equ KBD_PORT 0x01
|
|
|
|
|
|
|
|
; *** Code ***
|
2019-11-08 14:04:56 -05:00
|
|
|
|
2019-11-08 14:58:29 -05:00
|
|
|
; Wait for a digit to be pressed and sets the A register ASCII value
|
|
|
|
; corresponding to that key press.
|
2019-11-08 14:04:56 -05:00
|
|
|
;
|
|
|
|
; This routine waits for a key to be pressed, but before that, it waits for
|
|
|
|
; all keys to be de-pressed. It does that to ensure that two calls to
|
|
|
|
; waitForKey only go through after two actual key presses (otherwise, the user
|
|
|
|
; doesn't have enough time to de-press the button before the next waitForKey
|
|
|
|
; routine registers the same key press as a second one).
|
|
|
|
;
|
2019-11-07 11:52:29 -05:00
|
|
|
; Sending 0xff to the port resets the keyboard, and then we have to send groups
|
|
|
|
; we want to "listen" to, with a 0 in the group bit. Thus, to know if *any* key
|
|
|
|
; is pressed, we send 0xff to reset the keypad, then 0x00 to select all groups,
|
|
|
|
; if the result isn't 0xff, at least one key is pressed.
|
2019-11-08 14:04:56 -05:00
|
|
|
kbdGetC:
|
|
|
|
push bc
|
2019-11-08 14:58:29 -05:00
|
|
|
push hl
|
2019-11-08 14:04:56 -05:00
|
|
|
|
2019-11-08 15:45:45 -05:00
|
|
|
; During this GetC loop, register C holds the modificators (Alpha, 2nd)
|
|
|
|
ld c, 0
|
|
|
|
; loop until a digit is pressed
|
2019-11-08 14:04:56 -05:00
|
|
|
.loop:
|
2019-11-08 14:58:29 -05:00
|
|
|
ld hl, .dtbl
|
|
|
|
; we go through the 7 rows of the table
|
|
|
|
ld b, 7
|
2019-11-08 15:45:45 -05:00
|
|
|
; is alpha mod enabled?
|
|
|
|
bit 0, c
|
|
|
|
jr z, .inner ; unset? skip next
|
|
|
|
ld hl, .atbl ; set? we're in alpha mode
|
2019-11-08 14:58:29 -05:00
|
|
|
.inner:
|
|
|
|
ld a, (hl) ; group mask
|
2019-11-08 14:04:56 -05:00
|
|
|
call .get
|
2019-11-08 14:58:29 -05:00
|
|
|
cp 0xff
|
|
|
|
jr nz, .something
|
|
|
|
; nothing for that group, let's scan the next group
|
|
|
|
ld a, 9
|
|
|
|
call addHL ; go to next row
|
|
|
|
djnz .inner
|
|
|
|
; found nothing, loop
|
|
|
|
jr .loop
|
|
|
|
.something:
|
|
|
|
; We have something on that row! Let's find out which char. Register A
|
|
|
|
; currently contains a mask with the pressed char bit unset.
|
|
|
|
ld b, 8
|
|
|
|
inc hl
|
|
|
|
.findchar:
|
|
|
|
rrca ; is next bit unset?
|
|
|
|
jr nc, .gotit ; yes? we have our char!
|
|
|
|
inc hl
|
|
|
|
djnz .findchar
|
|
|
|
.gotit:
|
|
|
|
ld a, (hl)
|
|
|
|
or a ; is char 0?
|
|
|
|
jr z, .loop ; yes? unsupported. loop.
|
2019-11-08 15:45:45 -05:00
|
|
|
cp 0x80 ; is it alpha?
|
|
|
|
jr nz, .notalpha
|
|
|
|
set 0, c
|
|
|
|
jr .loop
|
|
|
|
.notalpha:
|
2019-11-08 14:58:29 -05:00
|
|
|
|
|
|
|
; wait until all keys are de-pressed
|
|
|
|
push af ; --> lvl 1
|
|
|
|
.wait:
|
2019-11-08 14:04:56 -05:00
|
|
|
xor a
|
|
|
|
call .get
|
|
|
|
inc a ; if a was 0xff, will become 0 (nz test)
|
2019-11-08 14:58:29 -05:00
|
|
|
jr nz, .wait ; non-zero? something is pressed
|
2019-11-08 14:04:56 -05:00
|
|
|
|
2019-11-08 14:58:29 -05:00
|
|
|
pop af ; <-- lvl 1
|
2019-11-08 14:04:56 -05:00
|
|
|
|
2019-11-08 14:58:29 -05:00
|
|
|
pop hl
|
2019-11-08 14:04:56 -05:00
|
|
|
pop bc
|
|
|
|
ret
|
|
|
|
.get:
|
|
|
|
ex af, af'
|
2019-11-07 11:52:29 -05:00
|
|
|
ld a, 0xff
|
2019-11-08 14:04:56 -05:00
|
|
|
di
|
2019-11-07 11:52:29 -05:00
|
|
|
out (KBD_PORT), a
|
2019-11-08 14:04:56 -05:00
|
|
|
ex af, af'
|
2019-11-07 11:52:29 -05:00
|
|
|
out (KBD_PORT), a
|
|
|
|
in a, (KBD_PORT)
|
2019-11-08 14:04:56 -05:00
|
|
|
ei
|
2019-11-07 11:52:29 -05:00
|
|
|
ret
|
2019-11-08 14:58:29 -05:00
|
|
|
|
|
|
|
; digits table. each row represents a group. first item is group mask.
|
|
|
|
; 0 means unsupported. no group 7 because it has no keys.
|
2019-11-08 15:45:45 -05:00
|
|
|
; 0x80 is a special value for ALPHA key which is never returned directly.
|
2019-11-08 14:58:29 -05:00
|
|
|
.dtbl:
|
|
|
|
.db 0xfe, 0, 0, 0, 0, 0, 0, 0, 0
|
|
|
|
.db 0xfd, 0x0d, '+' ,'-' ,'*', '/', '^', 0, 0
|
|
|
|
.db 0xfb, 0, '3', '6', '9', ')', 0, 0, 0
|
|
|
|
.db 0xf7, '.', '2', '5', '8', '(', 0, 0, 0
|
|
|
|
.db 0xef, '0', '1', '4', '7', ',', 0, 0, 0
|
2019-11-08 15:45:45 -05:00
|
|
|
.db 0xdf, 0, 0, 0, 0, 0, 0, 0, 0x80
|
|
|
|
.db 0xbf, 0, 0, 0, 0, 0, 0, 0, 0x7f
|
|
|
|
|
|
|
|
; alpha table. same as .dtbl, for when we're in alpha mode.
|
|
|
|
.atbl:
|
|
|
|
.db 0xfe, 0, 0, 0, 0, 0, 0, 0, 0
|
|
|
|
.db 0xfd, 0x0d, '"' ,'W' ,'R', 'M', 'H', 0, 0
|
|
|
|
.db 0xfb, '?', 0, 'V', 'Q', 'L', 'G', 0, 0
|
|
|
|
.db 0xf7, ':', 'Z', 'U', 'P', 'K', 'F', 'C', 0
|
|
|
|
.db 0xef, '_', 'Y', 'T', 'O', 'J', 'E', 'B', 0
|
|
|
|
.db 0xdf, 0, 'X', 'S', 'N', 'I', 'D', 'A', 0x80
|
2019-11-08 14:58:29 -05:00
|
|
|
.db 0xbf, 0, 0, 0, 0, 0, 0, 0, 0x7f
|