mirror of
https://github.com/cc65/cc65.git
synced 2025-01-24 04:34:35 +00:00
230 lines
4.7 KiB
ArmAsm
230 lines
4.7 KiB
ArmAsm
;
|
|
; 2022-04-03, Karri Kaksonen
|
|
;
|
|
; setcursor (unsigned char on);
|
|
;
|
|
; The Atari 7800 does not have a hw cursor.
|
|
; This module emulates a cursor to be used with the conio
|
|
; implementation.
|
|
;
|
|
; The actual cursor display is included in the conio dll
|
|
; but every scanline has the element silenced by default
|
|
; at the end of every zone.
|
|
;
|
|
; The way the cursor works is to silence it before the cursor
|
|
; position changes and enable it afterwards.
|
|
;
|
|
; In order to get some performance we have a pointer to the
|
|
; cursor header structure. This structure is always at the
|
|
; end of the zone. So the pointer changes when CURS_Y changes.
|
|
;
|
|
; There is so many dependencies that it makes sense to
|
|
; deal with all CURS_X, CURS_Y stuff in this file and
|
|
; definitely not allow direct access to the variables.
|
|
;
|
|
|
|
.export gotoxy, _gotoxy, _gotox, _gotoy, _wherex, _wherey
|
|
.export CURS_X, CURS_Y
|
|
.constructor init_cursor
|
|
.interruptor blink_cursor
|
|
|
|
.importzp sp
|
|
.import _zones
|
|
.import cursor
|
|
.import pusha, incsp1, pusha0, pushax, popa
|
|
.include "atari7800.inc"
|
|
.include "extzp.inc"
|
|
|
|
.macpack generic
|
|
|
|
.data
|
|
;-----------------------------------------------------------------------------
|
|
; The variables used by cursor functions
|
|
;
|
|
|
|
CURS_X:
|
|
.byte 0
|
|
CURS_Y:
|
|
.byte 0
|
|
blink_time:
|
|
.byte 140
|
|
|
|
.code
|
|
|
|
;---------------------------------------------------------------------------
|
|
; 8x16 routine
|
|
|
|
umula0:
|
|
ldy #8 ; Number of bits
|
|
lda #0
|
|
lsr ptr7800 ; Get first bit into carry
|
|
@L0: bcc @L1
|
|
|
|
clc
|
|
adc ptrtmp
|
|
tax
|
|
lda ptrtmp+1 ; hi byte of left op
|
|
adc ptr7800+1
|
|
sta ptr7800+1
|
|
txa
|
|
|
|
@L1: ror ptr7800+1
|
|
ror a
|
|
ror ptr7800
|
|
dey
|
|
bne @L0
|
|
tax
|
|
lda ptr7800 ; Load the result
|
|
rts
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Calculate cursorzone address
|
|
; You also need to set the cursorzone to point to the correct cursor Header
|
|
; at the end of line CURS_Y.
|
|
; Offset to cursor zone 5. To next line offset 11
|
|
; cursorzone points to _zones + CURS_Y * 11 + 5
|
|
; A = CURS_Y
|
|
.proc calccursorzone
|
|
|
|
sta ptr7800
|
|
lda #11
|
|
sta ptrtmp
|
|
lda #0
|
|
sta ptr7800+1
|
|
sta ptrtmp+1
|
|
jsr umula0
|
|
clc
|
|
adc #5
|
|
bcc @L1
|
|
inx
|
|
@L1: clc
|
|
adc #<_zones
|
|
sta cursorzone ; calculate new cursorzone
|
|
txa
|
|
adc #>_zones
|
|
sta cursorzone+1
|
|
rts
|
|
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Set cursor to Y position.
|
|
; You also need to set the cursorzone to point to the correct cursor Header
|
|
; at the end of line CURS_Y.
|
|
; Offset to cursor zone 5. To next line offset 11
|
|
; cursorzone points to _zones + CURS_Y * 11 + 5
|
|
;
|
|
; cursorzone[1] = 0 when not CURS_Y, 30 if CURS_Y
|
|
;
|
|
; Disable cursor
|
|
; cursorzone[1] = 0
|
|
;
|
|
; Enable cursor
|
|
; if showcursor cursorzone[1] = 30
|
|
;
|
|
.proc _gotoy
|
|
|
|
pha
|
|
lda CURS_Y
|
|
jsr calccursorzone
|
|
ldy #1
|
|
lda #0
|
|
sta (cursorzone),y ; disable cursor
|
|
pla
|
|
sta CURS_Y
|
|
jsr calccursorzone
|
|
lda cursor
|
|
beq @L1
|
|
lda #30 ; enable cursor
|
|
@L1: ldy #1
|
|
sta (cursorzone),y
|
|
rts
|
|
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Set cursor to X position.
|
|
; You also need to set the hpos offset to the correct value on this line
|
|
; cursorzone[3] = 8 * CURS_X
|
|
;
|
|
.proc _gotox
|
|
|
|
sta CURS_X
|
|
ldy #3
|
|
clc
|
|
rol
|
|
rol
|
|
rol
|
|
sta (cursorzone),y
|
|
rts
|
|
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Set cursor to desired position (X,Y)
|
|
;
|
|
.proc _gotoxy
|
|
|
|
jsr _gotoy
|
|
jsr popa
|
|
jsr _gotox
|
|
rts
|
|
.endproc
|
|
|
|
.proc gotoxy
|
|
jsr popa
|
|
jmp _gotoxy
|
|
.endproc
|
|
;-----------------------------------------------------------------------------
|
|
; Get cursor X position
|
|
;
|
|
.proc _wherex
|
|
|
|
lda CURS_X
|
|
rts
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Get cursor Y position
|
|
;
|
|
.proc _wherey
|
|
|
|
lda CURS_Y
|
|
rts
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Initialize cursorzone at startup
|
|
; Offset to cursor zone 5.
|
|
;
|
|
.proc blink_cursor
|
|
inc blink_time
|
|
bne @L3
|
|
lda #140
|
|
sta blink_time
|
|
ldy #0
|
|
lda (cursorzone),y
|
|
bne @L1
|
|
lda #254
|
|
bne @L2
|
|
@L1: lda #0
|
|
@L2: sta (cursorzone),y
|
|
@L3: rts
|
|
.endproc
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; Initialize cursorzone at startup
|
|
; Offset to cursor zone 5.
|
|
;
|
|
.segment "ONCE"
|
|
init_cursor:
|
|
lda #0
|
|
jsr calccursorzone
|
|
rts
|
|
|
|
;-----------------------------------------------------------------------------
|
|
; force the init constructor to be imported
|
|
|
|
.import initconio
|
|
conio_init = initconio
|