mirror of
https://github.com/cc65/cc65.git
synced 2025-01-25 11:30:06 +00:00
Mouse driver implementation
git-svn-id: svn://svn.cc65.org/cc65/trunk@2957 b7a2c559-68d2-44c3-8de9-860c34a00d81
This commit is contained in:
parent
5f3ed8826f
commit
31595342d6
@ -37,11 +37,6 @@ HEADER:
|
||||
.addr IOCTL
|
||||
.addr IRQ
|
||||
|
||||
; Data that is visible to the outside. Initialized by the kernel.
|
||||
|
||||
XPos: .word 0 ; Current mouse position, X
|
||||
YPos: .word 0 ; Current mouse position, Y
|
||||
|
||||
; Callback table, set by the kernel before INSTALL is called
|
||||
|
||||
CHIDE: jmp $0000 ; Hide the cursor
|
||||
@ -57,9 +52,8 @@ SCREEN_HEIGHT = 200
|
||||
SCREEN_WIDTH = 320
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
;
|
||||
; Global variables
|
||||
;
|
||||
; Global variables. The bounding box values are sorted so that they can be
|
||||
; written with the least effort in the BOX routine, so don't reorder them.
|
||||
|
||||
.bss
|
||||
|
||||
@ -67,10 +61,12 @@ Vars:
|
||||
OldPotX: .res 1 ; Old hw counter values
|
||||
OldPotY: .res 1
|
||||
|
||||
XMin: .res 2 ; X1 value of bounding box
|
||||
YMin: .res 2 ; Y1 value of bounding box
|
||||
XMax: .res 2 ; X2 value of bounding box
|
||||
YPos: .res 2 ; Current mouse position, Y
|
||||
XPos: .res 2 ; Current mouse position, X
|
||||
YMax: .res 2 ; Y2 value of bounding box
|
||||
XMax: .res 2 ; X2 value of bounding box
|
||||
YMin: .res 2 ; Y1 value of bounding box
|
||||
XMin: .res 2 ; X1 value of bounding box
|
||||
|
||||
OldValue: .res 1 ; Temp for MoveCheck routine
|
||||
NewValue: .res 1 ; Temp for MoveCheck routine
|
||||
@ -81,9 +77,12 @@ NewValue: .res 1 ; Temp for MoveCheck routine
|
||||
|
||||
.proc DefVars
|
||||
.byte 0, 0 ; OldPotX/OldPotY
|
||||
.word 0, 0 ; XMin/YMin
|
||||
.word SCREEN_WIDTH ; XMax
|
||||
.word SCREEN_HEIGHT/2 ; YPos
|
||||
.word SCREEN_WIDTH/2 ; XPos
|
||||
.word SCREEN_HEIGHT ; YMax
|
||||
.word SCREEN_WIDTH ; XMax
|
||||
.word 0 ; YMin
|
||||
.word 0 ; XMin
|
||||
.endproc
|
||||
|
||||
.code
|
||||
@ -103,18 +102,31 @@ INSTALL:
|
||||
dex
|
||||
bpl @L1
|
||||
|
||||
; Be sure the mouse cursor is invisible and at the default location. We
|
||||
; need to do that here, because our mouse interrupt handler doesn't set the
|
||||
; mouse position if it hasn't changed.
|
||||
|
||||
sei
|
||||
jsr CHIDE
|
||||
lda XPos
|
||||
ldx XPos+1
|
||||
jsr CMOVEX
|
||||
lda YPos
|
||||
ldx YPos+1
|
||||
jsr CMOVEY
|
||||
cli
|
||||
|
||||
; Done, return zero (= MOUSE_ERR_OK)
|
||||
|
||||
inx ; X = 0
|
||||
ldx #$00
|
||||
txa
|
||||
; rts ; Run into UNINSTALL instead
|
||||
rts ; Run into UNINSTALL instead
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
; UNINSTALL routine. Is called before the driver is removed from memory.
|
||||
; No return code required (the driver is removed from memory on return).
|
||||
|
||||
UNINSTALL:
|
||||
rts
|
||||
UNINSTALL = HIDE ; Hide cursor on exit
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
; HIDE routine. Is called to hide the mouse pointer. The mouse kernel manages
|
||||
@ -123,7 +135,10 @@ UNINSTALL:
|
||||
; no special action is required besides hiding the mouse cursor.
|
||||
; No return code required.
|
||||
|
||||
HIDE = CHIDE
|
||||
HIDE: sei
|
||||
jsr CHIDE
|
||||
cli
|
||||
rts
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
; SHOW routine. Is called to show the mouse pointer. The mouse kernel manages
|
||||
@ -132,15 +147,31 @@ HIDE = CHIDE
|
||||
; no special action is required besides enabling the mouse cursor.
|
||||
; No return code required.
|
||||
|
||||
SHOW = CSHOW
|
||||
SHOW: sei
|
||||
jsr CSHOW
|
||||
cli
|
||||
rts
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
; BOX: Set the mouse bounding box. No checks are done if the mouse is
|
||||
; currently inside the box, this is the job of the caller. It is not necessary
|
||||
; to validate the parameters, trust the caller and save some code here.
|
||||
; No return code required.
|
||||
; BOX: Set the mouse bounding box. The parameters are passed as they come from
|
||||
; the C program, that is, maxy in a/x and the other parameters on the stack.
|
||||
; The C wrapper will remove the parameters from the stack when the driver
|
||||
; routine returns.
|
||||
; No checks are done if the mouse is currently inside the box, this is the job
|
||||
; of the caller. It is not necessary to validate the parameters, trust the
|
||||
; caller and save some code here. No return code required.
|
||||
|
||||
BOX:
|
||||
BOX: ldy #5
|
||||
sei
|
||||
sta YMax
|
||||
stx YMax+1
|
||||
|
||||
@L1: lda (sp),y
|
||||
sta XMax,y
|
||||
dey
|
||||
bpl @L1
|
||||
|
||||
cli
|
||||
rts
|
||||
|
||||
;----------------------------------------------------------------------------
|
||||
@ -243,8 +274,7 @@ IRQ: lda SID_ADConv1 ; Get mouse X movement
|
||||
|
||||
; Skip processing if nothing has changed
|
||||
|
||||
tay
|
||||
beq @SkipX
|
||||
bcc @SkipX
|
||||
|
||||
; Calculate the new X coordinate (--> a/y)
|
||||
|
||||
@ -275,7 +305,7 @@ IRQ: lda SID_ADConv1 ; Get mouse X movement
|
||||
; Move the mouse pointer to the new X pos
|
||||
|
||||
tya
|
||||
jsr CMOVEY
|
||||
jsr CMOVEX
|
||||
|
||||
; Calculate the Y movement vector
|
||||
|
||||
@ -286,8 +316,7 @@ IRQ: lda SID_ADConv1 ; Get mouse X movement
|
||||
|
||||
; Skip processing if nothing has changed
|
||||
|
||||
tay
|
||||
beq @SkipY
|
||||
bcc @SkipY
|
||||
|
||||
; Calculate the new Y coordinate (--> a/y)
|
||||
|
||||
@ -321,7 +350,7 @@ IRQ: lda SID_ADConv1 ; Get mouse X movement
|
||||
; Move the mouse pointer to the new X pos
|
||||
|
||||
tya
|
||||
jsr CMOVEX
|
||||
jmp CMOVEY
|
||||
|
||||
; Done
|
||||
|
||||
@ -349,6 +378,7 @@ MoveCheck:
|
||||
lsr a ; a /= 2;
|
||||
beq @L2 ; if (a != 0)
|
||||
ldy NewValue ; y = NewValue
|
||||
sec
|
||||
rts ; return
|
||||
|
||||
@L1: ora #%11000000 ; else or in high order bits
|
||||
@ -358,8 +388,10 @@ MoveCheck:
|
||||
ror a ; a /= 2
|
||||
dex ; high byte = -1 (X = $FF)
|
||||
ldy NewValue
|
||||
sec
|
||||
rts
|
||||
|
||||
@L2: txa ; A = $00
|
||||
clc
|
||||
rts
|
||||
|
||||
|
@ -2,6 +2,9 @@
|
||||
; Default mouse callbacks for the C64
|
||||
;
|
||||
; Ullrich von Bassewitz, 2004-03-20
|
||||
;
|
||||
; All functions in this module should be interrupt safe, because they may
|
||||
; be called from an interrupt handler
|
||||
;
|
||||
|
||||
.export _mouse_def_callbacks
|
||||
@ -9,6 +12,7 @@
|
||||
.include "mouse-kernel.inc"
|
||||
.include "c64.inc"
|
||||
|
||||
.macpack generic
|
||||
|
||||
; Sprite definitions. The first value can be changed to adjust the number
|
||||
; of the sprite used for the mouse.
|
||||
@ -18,71 +22,74 @@ MOUSE_SPR_NMASK = .lobyte(.not MOUSE_SPR_MASK) ; Negative mask
|
||||
VIC_SPR_X = (VIC_SPR0_X + 2*MOUSE_SPR) ; Sprite X register
|
||||
VIC_SPR_Y = (VIC_SPR0_Y + 2*MOUSE_SPR) ; Sprite Y register
|
||||
|
||||
|
||||
.code
|
||||
|
||||
; --------------------------------------------------------------------------
|
||||
; Hide the mouse pointer
|
||||
; Hide the mouse pointer. Always called with interrupts disabled.
|
||||
|
||||
.proc hide
|
||||
|
||||
lda #MOUSE_SPR_NMASK
|
||||
sei
|
||||
and VIC_SPR_ENA
|
||||
sta VIC_SPR_ENA
|
||||
cli
|
||||
rts
|
||||
|
||||
.endproc
|
||||
|
||||
; --------------------------------------------------------------------------
|
||||
; Show the mouse pointer
|
||||
; Show the mouse pointer. Always called with interrupts disabled.
|
||||
|
||||
.proc show
|
||||
|
||||
lda #MOUSE_SPR_MASK
|
||||
sei
|
||||
ora VIC_SPR_ENA
|
||||
sta VIC_SPR_ENA
|
||||
cli
|
||||
rts
|
||||
|
||||
.endproc
|
||||
|
||||
; --------------------------------------------------------------------------
|
||||
; Move the mouse pointer X position to the value in a/x
|
||||
; Move the mouse pointer X position to the value in a/x. Always called with
|
||||
; interrupts disabled.
|
||||
|
||||
.proc movex
|
||||
|
||||
; Set the low byte, this frees A
|
||||
; Add the X correction and set the low byte. This frees A.
|
||||
|
||||
add #24 ; X correction
|
||||
sta VIC_SPR_X
|
||||
|
||||
; Set the high byte
|
||||
|
||||
txa ; Test high byte of X coord
|
||||
bne @L1
|
||||
sei
|
||||
txa
|
||||
adc #0
|
||||
bne @L1 ; Branch if high byte not zero
|
||||
lda VIC_SPR_HI_X ; Get high X bits of all sprites
|
||||
and #MOUSE_SPR_NMASK ; Clear high bit for sprite
|
||||
sta VIC_SPR_HI_X
|
||||
cli
|
||||
rts
|
||||
|
||||
@L1: sei
|
||||
lda VIC_SPR_HI_X ; Get high X bits of all sprites
|
||||
ora #MOUSE_SPR_NMASK ; Set high bit for sprite
|
||||
@L1: lda VIC_SPR_HI_X ; Get high X bits of all sprites
|
||||
ora #MOUSE_SPR_MASK ; Set high bit for sprite
|
||||
sta VIC_SPR_HI_X
|
||||
cli
|
||||
rts
|
||||
|
||||
.endproc
|
||||
|
||||
; --------------------------------------------------------------------------
|
||||
; Move the mouse pointer Y position to the value in a/x
|
||||
; Move the mouse pointer Y position to the value in a/x. Always called with
|
||||
; interrupts disabled.
|
||||
|
||||
.proc movey
|
||||
|
||||
clc
|
||||
ldx PALFLAG
|
||||
bne @L1
|
||||
adc #50 ; FIXME: Should be NTSC, is PAL value
|
||||
sta VIC_SPR_Y ; Set Y position
|
||||
rts
|
||||
|
||||
@L1: adc #50 ; Add PAL correction
|
||||
sta VIC_SPR_Y ; Set Y position
|
||||
rts
|
||||
|
||||
@ -100,4 +107,3 @@ _mouse_def_callbacks:
|
||||
.addr movey
|
||||
|
||||
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user