--- /dev/null
+;--------------------------------------------------------------------
+; Atari 8-bit mouse routines -- 05/07/2000 Freddy Offenga
+; Some changes by Christian Groessler, Ullrich von Bassewitz
+;
+; The following devices are supported:
+; - Atari trak-ball
+; - ST mouse
+; - Amiga mouse
+;
+; Mouse checks are done in the timer 1 IRQ and the mouse arrow is
+; drawn in player 0 during the vertical blank
+;--------------------------------------------------------------------
+
+       .export         _mouse_init, _mouse_done, _mouse_box
+       .export         _mouse_show, _mouse_hide, _mouse_move
+       .export         _mouse_buttons, _mouse_pos, _mouse_info
+       .constructor    initmouse,27
+
+       .import         popax
+       .importzp       ptr1
+
+       .include "atari.inc"
+
+TRAK_BALL      = 0     ; device Atari trak-ball
+ST_MOUSE       = 1     ; device ST mouse
+AMIGA_MOUSE    = 2     ; device Amiga mouse
+MAX_TYPE       = 3     ; first illegal device type
+
+; the default values force the mouse cursor inside the test screen (no access to border)
+defxmin = 48           ; default x minimum
+defymin = 31           ; default y minimum
+defxmax = 204          ; default x maximum
+defymax = 211          ; default y maximum
+
+pmsize = 16            ; y size pm shape
+
+xinit  = defxmin       ; init. x pos.
+yinit  = defymin       ; init. y pos.
+
+;--------------------------------------------------------------------
+; reserve memory for the mouse pointer
+
+initmouse:
+       lda     APPMHI+1
+       and     #%11111000      ; make 2k aligned
+       sec
+       sbc     #%00001000      ; reserve 2k
+       tax
+       adc     #3              ; add 4 (C = 1)
+       sta     mouse_pm0
+       lda     #0
+       sta     APPMHI
+       stx     APPMHI+1
+       rts
+
+
+;--------------------------------------------------------------------
+; Initialize mouse routines
+; void __fastcall__ mouse_init (unsigned char type);
+
+_mouse_init:
+       cmp     #MAX_TYPE+1     ; Check for a valid type
+       bcc     setup
+
+ifail: lda     #0              ; init. failed
+       tax
+       rts
+
+setup: tax
+       lda     lvectab,x
+       sta     mouse_vec+1
+       lda     hvectab,x
+       sta     mouse_vec+2
+
+       jsr     pminit
+
+       lda     VTIMR1
+       sta     old_t1
+       lda     VTIMR1+1
+       sta     old_t1+1
+
+       lda     #<t1_vec
+       sta     VTIMR1
+       lda     #>t1_vec
+       sta     VTIMR1+1
+
+       lda     #%00000001
+       sta     AUDCTL
+
+       lda     #0
+       sta     AUDC1
+
+       lda     #15
+       sta     AUDF1
+       sta     STIMER
+
+       sei
+       lda     POKMSK
+       ora     #%00000001              ; timer 1 enable
+       sta     POKMSK
+       sta     IRQEN
+       cli
+
+       lda     VVBLKI
+        sta     vbi_jmp+1
+       lda     VVBLKI+1
+        sta     vbi_jmp+2
+
+       lda     #6
+       ldy     #<vbi
+       ldx     #>vbi
+       jsr     SETVBV
+
+       lda     #$C0
+       sta     NMIEN
+
+       ldx     #0
+       lda     #1
+       sta     mouse_off
+       rts
+
+;--------------------------------------------------------------------
+; Finish mouse routines
+; void mouse_done(void)
+
+_mouse_done:
+       sei
+       lda     POKMSK
+       and     #%11111110              ; timer 1 disable
+       sta     IRQEN
+       sta     POKMSK
+       cli
+
+       lda     old_t1
+       sta     VTIMR1
+       lda     old_t1+1
+       sta     VTIMR1+1
+
+       lda     #$40
+       sta     NMIEN
+
+       lda     #6
+        ldy     vbi_jmp+1
+        ldx     vbi_jmp+2
+       jsr     SETVBV
+
+       ldx     #0
+       stx     GRACTL
+       stx     HPOSP0
+       inx
+       stx     mouse_off
+       rts
+
+;--------------------------------------------------------------------
+; Set mouse limits
+; void __fastcall__ mouse_box(int xmin, int ymin, int xmax, int ymax)
+
+_mouse_box:
+       sta     ymax
+        jsr     popax          ; always ignore high byte
+       sta     xmax
+       jsr     popax
+       sta     ymin
+        jsr     popax
+       sta     xmin
+       rts
+
+;--------------------------------------------------------------------
+; Set mouse position
+; void __fastcall__ mouse_move(int xpos, int ypos)
+
+_mouse_move:
+       sta     mousey          ; always ignore high byte
+       jsr     popax
+       sta     mousex
+       rts
+
+;--------------------------------------------------------------------
+; Show mouse arrow
+; void mouse_show(void)
+
+_mouse_show:
+       lda     mouse_off       ; Already on?
+       beq     @L1
+               dec     mouse_off
+@L1:   rts
+
+;--------------------------------------------------------------------
+; Hide mouse arrow
+; void mouse_hide(void)
+
+_mouse_hide:
+       inc     mouse_off
+       rts
+
+;--------------------------------------------------------------------
+; Ask mouse button
+; unsigned char mouse_buttons(void)
+
+_mouse_buttons:
+       ldx     #0
+       lda     STRIG0
+       bne     nobut
+;      lda     #14
+;???   sta     COLOR1
+       lda     #1
+       rts
+nobut: txa
+       rts
+
+;--------------------------------------------------------------------
+; Get the mouse position
+; void mouse_pos (struct mouse_pos* pos);
+
+_mouse_pos:
+       sta     ptr1
+       stx     ptr1+1                  ; Store argument pointer
+       ldy     #0
+       lda     mousex                  ; X position
+       sta     (ptr1),y
+       lda     #0
+       iny
+       sta     (ptr1),y
+       lda     mousey                  ; Y position
+       iny
+       sta     (ptr1),y
+       lda     #0
+       iny
+       sta     (ptr1),y
+       rts
+
+;--------------------------------------------------------------------
+; Get the mouse position and button information
+; void mouse_info (struct mouse_info* info);
+
+_mouse_info:
+
+; We're cheating here to keep the code smaller: The first fields of the
+; mouse_info struct are identical to the mouse_pos struct, so we will just
+; call _mouse_pos to initialize the struct pointer and fill the position
+; fields.
+
+        jsr    _mouse_pos
+
+; Fill in the button state
+
+       jsr     _mouse_buttons          ; Will not touch ptr1
+       ldy     #4
+       sta     (ptr1),y
+
+       rts
+
+;--------------------------------------------------------------------
+; Atari trak-ball check, A,Y = 4-bit port value
+
+trak_check:
+       eor     oldval
+       and     #%00001000
+       beq     horiz
+
+       tya
+       and     #%00000100
+       beq     mmup
+
+       inc     mousey
+       bne     horiz
+
+mmup:  dec     mousey
+
+horiz: tya
+       eor     oldval
+       and     #%00000010
+       beq     mmexit
+
+       tya
+       and     #%00000001
+       beq     mmleft
+
+       inc     mousex
+       bne     mmexit
+
+mmleft: dec    mousex
+
+mmexit: sty    oldval
+       rts
+
+;--------------------------------------------------------------------
+; ST mouse check, A,Y = 4-bit port value
+
+st_check:
+       and     #%00000011
+       ora     dumx
+       tax
+       lda     sttab,x
+       bmi     nxst
+
+       beq     xist
+       dec     mousex              ; 1 = left
+       bne     nxst
+xist:  inc     mousex              ; 0 = right
+
+nxst:  tya
+       and     #%00001100
+       ora     dumy
+       tax
+       lda     sttab,x
+       bmi     nyst
+
+       bne     yst
+       dec     mousey              ; 0 = up
+       bne     nyst
+yst:   inc     mousey              ; 1 = down
+
+; store old readings
+
+nyst:  tya
+       and     #%00000011
+       asl
+       asl
+       sta     dumx
+       tya
+       and     #%00001100
+       lsr
+       lsr
+       sta     dumy
+       rts
+
+;--------------------------------------------------------------------
+; Amiga mouse check, A,Y = 4-bit port value
+
+amiga_check:
+
+       lsr
+       and     #%00000101
+       ora     dumx
+       tax
+       lda     amitab,x
+       bmi     nxami
+
+       bne     xiami
+       dec     mousex              ; 0 = left
+       bne     nxami
+xiami: inc     mousex              ; 1 = right
+
+nxami: tya
+
+       and     #%00000101
+       ora     dumy
+       tax
+       lda     amitab,x
+       bmi     nyami
+
+       bne     yiami
+       dec     mousey              ; 0 =  up
+       bne     nyami
+yiami: inc     mousey              ; 1 = down
+
+; store old readings
+
+nyami: tya
+       and     #%00001010
+       sta     dumx
+       tya
+       and     #%00000101
+       asl
+       sta     dumy
+       rts
+
+;--------------------------------------------------------------------
+; timer 1 IRQ routine - check mouse
+
+t1_vec: tya
+       pha
+       txa
+       pha     
+
+.ifdef DEBUG
+       lda     RANDOM
+       sta     COLBK           ; debug
+.endif
+
+       lda     PORTA
+       tay
+
+mouse_vec:
+        jsr     st_check        ; will be modified; won't be ROMmable
+
+       pla
+       tax
+       pla
+       tay
+        pla
+       rti
+
+;--------------------------------------------------------------------
+; VBI - check mouse limits and display mouse arrow
+
+vbi:   lda     mousex
+       cmp     xmin
+        bcs     ok1             ; xmin <= mousex
+        lda     xmin
+       sta     mousex
+
+ok1:   lda     mousey
+       cmp     ymin
+        bcs     ok2             ; ymin <= mousey
+       lda     ymin
+       sta     mousey
+
+ok2:    lda     xmax
+        cmp     mousex
+        bcs     ok3             ; xmax >= mousex
+       lda     xmax
+       sta     mousex
+
+ok3:    lda     ymax
+        cmp     mousey
+        bcs     ok4             ; ymax >= mousey
+       lda     ymax
+       sta     mousey
+
+ok4:    jsr     clrpm
+
+       lda     mouse_off
+        beq     mon
+        lda     #0
+        sta     HPOSP0
+        beq     moff
+
+mon:    jsr     drwpm
+       lda     mousey
+       sta     omy
+
+        lda     #3
+moff:  sta     GRACTL
+
+vbi_jmp:
+        jmp     SYSVBV          ; will be modified; won't be ROMmable
+
+;--------------------------------------------------------------------
+; initialize mouse pm
+
+pminit:        lda     mouse_pm0
+       sta     mpatch1+2
+       sta     mpatch2+2
+       sta     mpatch3+2
+
+       ldx     #0
+       txa
+mpatch1:
+clpm:  sta     $1000,x         ; will be patched
+       inx
+       bne     clpm
+
+       lda     mouse_pm0
+       sec
+       sbc     #4
+       sta     PMBASE
+
+       lda     #62
+       sta     SDMCTL
+
+        lda     #1
+       sta     GPRIOR
+
+       lda     #0
+       sta     PCOLR0
+        sta     SIZEP0
+       rts
+
+;--------------------------------------------------------------------
+; draw new mouse pm
+
+drwpm: lda     mousex
+       sta     HPOSP0
+
+       lda     mousey
+       tax
+
+       ldy     #0
+fmp2:  lda     mskpm,y
+mpatch2:
+       sta     $1000,x         ; will be patched
+       inx
+       iny
+       cpy     #pmsize
+       bne     fmp2
+       rts
+
+;--------------------------------------------------------------------
+; clear old mouse pm
+
+clrpm: lda     omy
+        tax
+
+       ldy     #0
+       tya
+mpatch3:
+fmp1:  sta     $1000,x         ; will be patched
+       inx
+       iny
+       cpy     #pmsize
+       bne     fmp1
+       rts
+
+;--------------------------------------------------------------------
+        .rodata
+
+; mouse arrow - pm shape
+
+mskpm: .byte %00000000
+       .byte %10000000
+       .byte %11000000
+       .byte %11000000
+
+       .byte %11100000
+       .byte %11100000
+       .byte %11110000
+       .byte %11100000
+
+       .byte %11100000
+       .byte %00100000
+       .byte %00100000
+       .byte %00110000
+
+       .byte %00110000
+       .byte %00000000
+       .byte %00000000
+       .byte %00000000
+
+; ST mouse lookup table
+
+sttab:  .byte $FF,$01,$00,$01
+        .byte $00,$FF,$00,$01
+        .byte $01,$00,$FF,$00
+        .byte $01,$00,$01,$FF
+
+; Amiga mouse lookup table
+
+amitab: .byte $FF,$01,$00,$FF
+        .byte $00,$FF,$FF,$01
+        .byte $01,$FF,$FF,$00
+        .byte $FF,$00,$01,$FF
+
+; Device vectors
+
+lvectab:
+       .byte <trak_check, <st_check, <amiga_check
+hvectab:
+       .byte >trak_check, >st_check, >amiga_check
+
+; default values
+
+xmin:   .byte  defxmin
+ymin:   .byte  defymin
+xmax:  .byte   defxmax
+ymax:  .byte   defymax
+
+mousex:        .byte   xinit
+mousey:        .byte   yinit
+
+;--------------------------------------------------------------------
+       .bss
+
+; Misc. vars
+
+old_t1: .res 2         ; old timer interrupt vector
+oldval: .res 1         ; used by trakball routines
+dumx:  .res 1
+dumy:  .res 1
+omy:   .res 1          ; old y pos
+
+mouse_off:
+       .res 1
+mouse_pm0:
+       .res 1