; Copyright 1996 Acorn Computers Ltd ; ; Licensed under the Apache License, Version 2.0 (the "License"); ; you may not use this file except in compliance with the License. ; You may obtain a copy of the License at ; ; http://www.apache.org/licenses/LICENSE-2.0 ; ; Unless required by applicable law or agreed to in writing, software ; distributed under the License is distributed on an "AS IS" BASIS, ; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ; See the License for the specific language governing permissions and ; limitations under the License. ; ; > $.Source.PMF.Mouse ; Mouse driving code ; Author: Steve Cormie ; Started: 24-Feb-93 ; Change history: ; ; Date Who Description ; ---- --- ----------- ; 24-Feb-93 SMC Created. ; ***************************************************************************** ; ; Mouse initialisation ; MouseInit Push "lr" LDR r11, =ZeroPage+KeyWorkSpace [ :LNOT: AssemblingArthur MOV r0, #MouseV ADRL r1, ReadMouse SWI OS_Claim ] MOV r0, #MouseStepCMOS ; setup mouse multipliers from CMOS BL Read MOV r0, r0, LSL #24 ; sign extend it MOVS r0, r0, ASR #24 MOVEQ r0, #1 ; if would be zero, set to 1 STR r0, MouseXMult STR r0, MouseYMult MOV r0, #0 STRB r0, MouseButtons [ STB STRB r0, MousePresent ] MOV r0, #MouseCMOS BL Read STRB r0, MouseType Pull "pc" ; ***************************************************************************** ; ; MouseButtonChange - Called by keyboard handler when mouse button change ; ; in: R0 = state of buttons (bit0=R, bit1=C, bit2=L) ; R11 -> KeyWorkSpace ; MouseButtonChange ROUT Push "R0-R5, R12, R14" VDWS WsPtr STRB R0, MouseButtons ; save it for ReadMouse calls MOV R3, R0 LDR R1, MouseX LDR R0, [WsPtr, #OrgX] SUB R1, R1, R0 ; mouse X LDR R2, MouseY LDR R0, [WsPtr, #OrgY] SUB R2, R2, R0 ; mouse Y [ AssemblingArthur :LOR: Module LDR R4, =ZeroPage LDR R4, [R4, #MetroGnome] ; use monotonic variable now | BYTEWS WsPtr LDR R4, RealTime ; doesn't exist in my world ] MOV R0, #Event_Mouse BL OSEVEN MOV WsPtr, #IOC [ :LNOT:MouseBufferManager [ MouseBufferFix LDR R0, MouseX | MOV R5, R2 ; save mouse Y MOV R0, R1 ] BL MouseInsert ; send mouse X low BCS %FT10 ; buffer full, so don't send rest MOV R0, R0, LSR #8 ; send mouse X high BL MouseInsert [ MouseBufferFix LDR R0, MouseY | MOV R0, R5 ] BL MouseInsert ; send mouse Y low MOV R0, R0, LSR #8 ; send mouse Y high BL MouseInsert MOV R0, R3 BL MouseInsert ; send buttons MOV R0, R4 BL MouseInsert ; send realtime(0) MOV R0, R4, LSR #8 BL MouseInsert ; send realtime(1) MOV R0, R4, LSR #16 BL MouseInsert ; send realtime(2) MOV R0, R4, LSR #24 BL MouseInsert ; send realtime(3) | ; Use buffer manager's 'block insert' function [ {TRUE} ; TMD 26-Feb-93: Fix bug - if X is negative, Y would be inserted in the buffer as -1 LDR R0, MouseX ; 16 bits, sign-extended to 32 bits MOV R0, R0, LSL #16 LDR R1, MouseY ; ditto MOV R1, R1, LSL #16 ORR R0, R1, R0, LSR #16 ; combine, having knocked off the troublesome bits | LDR R0, MouseX ; 16 bits LDR R1, MouseY ; 16 bits ORR R0, R0, R1, LSL #16 ; R0 = Combined 16bit X/Y mouse position ] ORR R1, R3, R4, LSL #8 ; R1 = Combined 8bit buttons and 24 LSB's of time MOV R2, R4, LSR #24 ; R2 = MSB of time SUB SP, SP, #3*4 ; Create local mouse data buffer STMIA SP, {R0,R1,R2} ; Write mouse data to buffer MOV R3, #9 ; Mouse packet size MOV R2, SP ; R2-> block to insert MOV R1, #(Buff_Mouse:OR:(1:SHL:31)) ; Block insert to mouse buffer Push "R10,R12" MOV R10, #INSV ; Insert BL GoVec2 ; Call the vector in R10 Pull "R10,R12" ADD SP, SP, #3*4 ; Destroy mouse data buffer ] 10 Pull "R0-R5, R12, PC" [ :LNOT:MouseBufferManager MouseInsert Push "R10,R12,R14" MOV R10, #INSV MOV R1, #Buff_Mouse B GoVec ] ; ***************************************************************************** ; ; Read mouse position ; ReadMouse ROUT Push "R4-R6,R10-R12" LDR R11, =ZeroPage+KeyWorkSpace [ :LNOT:MouseBufferManager MOV R1, #Buff_Mouse BL KeyREMOVE BCS %FT10 ; MouseAhead buffer empty MOV R4, R2, LSL #16 ; Mouse X Low BL KeyREMOVE ORR R4, R4, R2, LSL #24 ; R4 := Mouse X << 16 BL KeyREMOVE MOV R5, R2, LSL #16 ; Mouse Y Low BL KeyREMOVE ORR R5, R5, R2, LSL #24 ; R5 := Mouse Y << 16 BL KeyREMOVE MOV R6, R2 ; Button state BL KeyREMOVE ; get realtime MOV R3, R2 BL KeyREMOVE ORR R3, R3, R2, LSL #8 BL KeyREMOVE ORR R3, R3, R2, LSL #16 BL KeyREMOVE ORR R3, R3, R2, LSL #24 MOV R0, R4, ASR #16 ; sign extend mouse coords MOV R1, R5, ASR #16 MOV R2, R6 | SUB SP, SP, #3*4 ; Create 9 byte local mouse data buffer MOV R3, #9 ; Mouse packet size MOV R2, SP ; R2-> buffer for data MOV R1, #(Buff_Mouse:OR:(1:SHL:31)) ; Block remove from mouse buffer CLRV ; Remove not examine Push "R10,R12" MOV R10, #REMV BL GoVec2 ; Call the vector in R10 Pull "R10,R12" LDMCCIA SP, {R4,R5,R6} ADD SP, SP, #3*4 ; Destroy mouse data buffer BCS %FT10 ; Jump if no buffered data MOV R0, R4, LSL #16 MOV R0, R0, ASR #16 ; R0 = sign extended x coord MOV R1, R4, ASR #16 ; R1 = sign extended y coord AND R2, R5, #&FF ; R2 = button state MOV R3, R5, LSR #8 ; R3 = 3 low order bytes of time ORR R3, R3, R6, LSL #24 ; R3 = time ] ; code inserted here 12-Aug-88 to force position read from buffer to be inside ; CURRENT bounding box; this removes the need to flush buffer when changing ; the bounding box. ADR R4, MouseBounds LDMIA R4, {R4-R6,R10} ; R4=LCol; R5=BRow; R6=RCol; R10=TRow; CMP R0, R4 MOVLT R0, R4 CMP R0, R6 MOVGT R0, R6 CMP R1, R5 MOVLT R1, R5 CMP R1, R10 MOVGT R1, R10 [ MouseBufferFix B %FT20 ; correct for origin after clipping | Pull "R4-R6,R10-R12,PC" ] 10 LDRB R2, MouseButtons [ AssemblingArthur :LOR: Module LDR R3, =ZeroPage LDR R3, [R3, #MetroGnome] ; use monotonic variable now | BYTEWS WsPtr LDR R3, RealTime ; doesn't exist in my world ] LDR R0, MouseX LDR R1, MouseY 20 VDWS WsPtr LDR R4, [WsPtr, #OrgX] SUB R0, R0, R4 LDR R4, [WsPtr, #OrgY] SUB R1, R1, R4 Pull "R4-R6,R10-R12,PC" Abso DCB "Abso" ; ***************************************************************************** ; ; ProcessMouseXY - Called to update mouse position. ; ; in: r2 = signed 32-bit X movement ; r3 = signed 32-bit Y movement ; r4 = "Abso" if absolute movement ; r11 ->KeyWorkSpace ; out: r2,r3 corrupted ; ProcessMouseXY Push "r4,lr" ; check for absolute position LDR lr, Abso TEQ r4, lr BEQ %FT40 ; process X movement CMP r2, #0 BEQ %FT10 MOV r2, r2, LSL #16 ; move delta X to top 16 bits LDR r4, MouseXMult MUL r2, r4, r2 LDR r4, MouseX ADD r2, r2, r4, LSL #16 ; add signed value in top 16 bits MOV r2, r2, ASR #16 ; sign extend to 32 bits LDR r4, MouseBoundLCol ; bound to bounding box CMP r2, r4 MOVLT r2, r4 LDR r4, MouseBoundRCol CMP r4, r2 MOVLT r2, r4 STR r2, MouseX 10 ; process Y movement CMP r3, #0 Pull "r4,pc",EQ MOV r3, r3, LSL #16 ; move delta Y to top 16 bits LDR r4, MouseYMult MUL r3, r4, r3 LDR r4, MouseY ADD r3, r3, r4, LSL #16 ; add signed value in top 16 bits MOV r3, r3, ASR #16 ; sign extend to 32 bits LDR r4, MouseBoundBRow ; bound to bounding box CMP r3, r4 MOVLT r3, r4 LDR r4, MouseBoundTRow CMP r4, r3 MOVLT r3, r4 STR r3, MouseY Pull "r4,pc" 40 ; process absolute position MOV r2, r2, ASL #16 ; look only at bottom 16 bits, MOV r3, r3, ASL #16 ; sign extended MOV r2, r2, ASR #16 MOV r3, r3, ASR #16 LDR r4, MouseBoundLCol ; bound to bounding box CMP r2, r4 MOVLT r2, r4 LDR r4, MouseBoundRCol CMP r4, r2 MOVLT r2, r4 STR r2, MouseX LDR r4, MouseBoundBRow ; bound to bounding box CMP r3, r4 MOVLT r3, r4 LDR r4, MouseBoundTRow CMP r4, r3 MOVLT r3, r4 STR r3, MouseY Pull "r4,pc" [ AssemblePointerV ; ***************************************************************************** ; ; PollPointer - Called on VSync to get mouse changes. ; ; out: corrupts r0-r3,r9-r11 ; PollPointer Push "r4,r12,lr" LDR r11, =ZeroPage+KeyWorkSpace LDRB r0, MouseReporting TEQ r0, #0 Pull "r4,r12,pc",NE MOV r0, #0 ; Request pointer state. LDRB r1, MouseType MOV r2, #0 ; Default to no movement. MOV r3, #0 MOV r4, #0 ; They might fill this in SavePSR r9 ; Save current PSR. WritePSRc SVC_mode+I_bit, r10 ; Call PointerV in SVC mode, no IRQs. MOV r10, #PointerV ; Call PointerV to get movements & button states Push "lr" ; Save SVC lr. BL CallVector Pull "lr" ; Restore SVC lr. RestPSR r9 [ STB TEQ r2, #0 TEQEQ r3, #0 MOVNE lr, #1 STRNEB lr, MousePresent BLNE ProcessMouseXY | BL ProcessMouseXY ] Pull "r4,r12,pc" ; ***************************************************************************** ; ; PointerVector - the default PointerV claimant ; PointerVector CMP r0, #PointerReason_Report Pull pc, NE Push "r2,r3,r11" LDR r11, =ZeroPage+KeyWorkSpace LDRB lr, MouseType TEQ r1, lr MOVEQ lr, #1 STREQB lr, MouseReporting [ STB TEQ r2, #0 TEQEQ r3, #0 MOVNE lr, #1 STRNEB lr, MousePresent BLNE ProcessMouseXY | BL ProcessMouseXY ] Pull "r2,r3,r11,pc" ; ***************************************************************************** ; ; PointerSWI - Handle SWI OS_Pointer calls (read/set pointer type). ; PointerSWI LDR r11, =ZeroPage+KeyWorkSpace TEQ r0, #0 LDREQB r0, MouseType BEQ SLVK [ STB TEQ r0, #2 LDREQB r0, MousePresent BEQ SLVK ] TEQ r0, #1 BNE %FT10 Push "r0,r10,r12,lr" STRB r1, MouseType MOV r0, #0 STRB r0, MouseReporting MOV r0, #2 MOV r10, #PointerV BL CallVector Pull "r0,r10,r12,lr" B SLVK 10 ADRL r0, ErrorBlock_BadParameters [ International BL TranslateError ] B SLVK_SetV ] END