mirror of
https://github.com/elliotnunn/mac-rom.git
synced 2025-01-20 12:30:40 +00:00
4325cdcc78
Resource forks are included only for .rsrc files. These are DeRezzed into their data fork. 'ckid' resources, from the Projector VCS, are not included. The Tools directory, containing mostly junk, is also excluded.
329 lines
9.8 KiB
Plaintext
329 lines
9.8 KiB
Plaintext
;
|
|
; File: SeekRgn.a
|
|
;
|
|
; Copyright: © 1981-1993 by Apple Computer, Inc., all rights reserved.
|
|
;
|
|
; Change History (most recent first):
|
|
;
|
|
; <SM3> 9/12/93 SAM Fixed a comment.
|
|
; <SM2> 9/12/93 SAM Changed all instances of _Translate24to32 to _rTranslate24to32
|
|
; so they can conditionalized out of the build.
|
|
;
|
|
; To Do:
|
|
;_____________________________________________________________________________________________________
|
|
;EASE$$$ READ ONLY COPY of file ÒSEEKRGN.aÓ
|
|
;¥1.4 BAL 05/29/1989 Blasting in 32-Bit QuickDraw version 1.0 Final
|
|
;¥1.3 BAL 04/12/1989 Blasting in 32-Bit QuickDraw 1.0B1
|
|
; File SeekRgn.a
|
|
;
|
|
; Copyright Apple Computer, Inc. 1981-1986
|
|
; All Rights Reserved
|
|
|
|
|
|
BLANKS ON
|
|
STRING ASIS
|
|
|
|
MACHINE MC68020
|
|
|
|
;------------------------------------------------------------------
|
|
;
|
|
; --> SEEKRGN.TEXT
|
|
;
|
|
; Routines to play back a region into a scanline buffer.
|
|
;
|
|
|
|
;------------------------------------------------------------------
|
|
;
|
|
; MODIFICATION HISTORY
|
|
;
|
|
; 8May86 EHB Added backwards seekRgn
|
|
; 9May86 EHB Optimized InvPair (to make up for time spent calling it)
|
|
; 2Oct88 BAL Altered seekRgn to return number of scans until mask changes
|
|
|
|
;------------------------------------------------------------------
|
|
;
|
|
; REGION STATE RECORD:
|
|
;
|
|
; RGNPTR EQU $0 ; [LONG] POINTER TO REGION
|
|
; DATAPTR EQU $4 ; [LONG] POINTER TO REGION'S DATA
|
|
; SCANBUF EQU $8 ; [LONG] POINTER TO REGION'S SCANLINE BUFFER
|
|
; SCANSIZE EQU $C ; [WORD] SIZE OF SCANBUF IN LONGS
|
|
; THISV EQU $E ; [WORD] CURRENT VERT POSITION OF SCANBUF
|
|
; NEXTV EQU $10 ; [WORD] NEXT VERT IN REGION STRUCTURE
|
|
; MINH EQU $12 ; [WORD] LEFT EDGE OF MINRECT
|
|
; MAXH EQU $14 ; [WORD] RIGHT EDGE OF MINRECT
|
|
; LEFTH EQU $16 ; [WORD] LEFT EDGE OF BUFFER
|
|
|
|
|
|
INITRGN PROC EXPORT
|
|
;------------------------------------------------------
|
|
;
|
|
; INPUTS: A0: RGNPTR
|
|
; A1: STATE RECORD
|
|
; D0: MINH
|
|
; D1: MAXH
|
|
; D2: BUFLEFT
|
|
;
|
|
; OUTPUTS: ALL FIELDS OF STATE RECORD,
|
|
; SCANBUF ALLOCATED ON STACK
|
|
;
|
|
; CLOBBERS: D0,D1,A0
|
|
;
|
|
MOVE D0,MINH(A1) ;INSTALL MINH
|
|
MOVE D1,MAXH(A1) ;INSTALL MAXH
|
|
MOVE D2,LEFTH(A1) ;INSTALL LEFTH
|
|
move.l a0,d0 ;get master ptr @@@@ BAL 09Apr88
|
|
_rTranslate24To32 ;and with high byte mask @@@@ BAL 09Apr88
|
|
move.l d0,a0 ;put back in a0 @@@@ BAL 09Apr88
|
|
MOVE.L A0,RGNPTR(A1) ;INSTALL RGNPTR
|
|
MOVE #-32767,THISV(A1) ;THISV := -32767
|
|
MOVE RGNBBOX+TOP(A0),NEXTV(A1) ;NEXTV := RGN BBOX TOP
|
|
LEA RGNDATA(A0),A0 ;POINT TO FIRST DATA
|
|
MOVE.L A0,DATAPTR(A1) ;INIT DATAPTR
|
|
MOVE.L (SP)+,A0 ;POP RETURN ADDR
|
|
SUB D2,D1 ;CALC BUFFER WIDTH IN DOTS
|
|
SUBQ #1,D1 ;MINUS ONE FOR EDGE CASES
|
|
LSR #5,D1 ;DIV BY 32 FOR #LONGS-1
|
|
MOVE D1,SCANSIZE(A1) ;SAVE SCANSIZE FOR LATER
|
|
|
|
CLRLOOP CLR.L -(SP) ;ALLOCATE AND CLEAR BUFFER
|
|
DBRA D1,CLRLOOP
|
|
MOVE.L SP,SCANBUF(A1) ;REMEMBER BUFFER START
|
|
JMP (A0) ;RETURN
|
|
|
|
|
|
|
|
SEEKRGN PROC EXPORT
|
|
EXPORT SEEKDOWN,SEEKUP
|
|
;------------------------------------------------------------------
|
|
;
|
|
; SeekRgn(rgnState,vert);
|
|
;
|
|
; ROUTINE TO PLAY BACK A REGION FORWARD OR BACKWARD UNTIL ITS SCAN
|
|
; BUFFER CONTAINS THE BITMAP FOR THE GIVEN VERTICAL COORDINATE.
|
|
;
|
|
; INPUTS: A1 POINTS TO A REGION STATE RECORD
|
|
; DO CONTAINS THE DESIRED VERTICAL COORD
|
|
;
|
|
; OUTPUTS: UPDATES THISV, NEXTV, DATAPTR, AND SCANBUF^ OF STATE RECORD
|
|
; D2--> number of scanlines before mask changes
|
|
;
|
|
; CLOBBERS: A0,D1,D2.
|
|
;
|
|
|
|
;----------------------------------------------------
|
|
;
|
|
; RETURN QUICKLY IF SCANBUF IS ALREADY CURRENT.
|
|
;
|
|
CMP NEXTV(A1),D0 ;IS DESIRED VERT >= NEXTV ?
|
|
BGE.S SEEKDOWN ;YES, BUMP DOWNWARD
|
|
CMP THISV(A1),D0 ;IS DESIRED VERT < CURRENT VERT ?
|
|
BLT.S SEEKUP ;YES, BUMP UPWARD
|
|
move nextV(a1),d1 ;compute nextV-vert
|
|
sub d0,d1
|
|
move d0,d2 ;compute vert-thisV+1
|
|
sub thisV(a1),d2
|
|
cmp d1,d2 ;return d2=min(d1,d2)
|
|
blt.s @d2
|
|
move d1,d2
|
|
@d2 RTS ;AND RETURN
|
|
|
|
;-----------------------------------------------------
|
|
;
|
|
; RESET TO START AND MOVE DOWN.
|
|
; CALLED BY UP1 WHEN DESIRED LINE < RGNBBOX.TOP
|
|
;
|
|
UP2 MOVE.L SCANBUF(A1),A0 ;POINT TO SCANBUF
|
|
MOVE SCANSIZE(A1),D1 ;GET BUFFER SIZE
|
|
CLRLP CLR.L (A0)+ ;CLEAR A LONG
|
|
DBRA D1,CLRLP ;LOOP ENTIRE SCANBUF
|
|
MOVE.L RGNPTR(A1),A0 ;GET RGNPTR
|
|
MOVE RGNBBOX+TOP(A0),NEXTV(A1) ;NEXTV := TOP VERT
|
|
MOVE #-32767,THISV(A1) ;RESET THISV TO -32767
|
|
LEA RGNDATA(A0),A0 ;POINT TO START OF REGION DATA
|
|
MOVE.L A0,DATAPTR(A1) ;RESET DATAPTR
|
|
CMP NEXTV(A1),D0 ;IS DESIRED VERT >= NEXTV ?
|
|
bge.s SEEKDOWN ;yes, SEEKDOWN
|
|
moveq #1,d2 ;no, set count to 1 and return
|
|
bra.s done
|
|
|
|
|
|
;------------------------------------------------------
|
|
;
|
|
; WHILE DESIRED VERT >= NEXTV DO BUMP DOWN.
|
|
;
|
|
SEEKDOWN MOVEM.L D0/d3-D7/A2-A4,-(SP) ; SAVE REGS
|
|
MOVE D0,D2 ; SAVE VERT
|
|
MOVE.L DATAPTR(A1),A2 ; POINT TO VERT COORD
|
|
LEA NEXTHOR,A4 ; RETURN ADDRESS FOR INVPAIR
|
|
MOVE.W MINH(A1),D6 ; D6 = MINH FOR INVPAIR
|
|
MOVE.W MAXH(A1),D7 ; D7 = MAXH FOR INVPAIR
|
|
|
|
DOWN1 MOVE (A2)+,THISV(A1) ; UPDATE CURRENT VERT
|
|
|
|
;-------------------------------------------------
|
|
;
|
|
; GET LEFT AND RIGHT HORIZ COORDS
|
|
; AND CALL INVPAIR TO TRIM AND INVERT BUFFER
|
|
;
|
|
NEXTHOR MOVE (A2)+,D3 ; GET LEFT COORD
|
|
CMP #32767,D3 ; IS IT A TERMINATOR ?
|
|
BEQ.S DONE1 ; YES, QUIT
|
|
MOVE (A2)+,D4 ; GET RIGHT COORD
|
|
BRA.S INVPAIR ; INVERT THIS PAIR OF COORDINATES
|
|
; AND RETURN TO NEXTHOR
|
|
DONE1 CMP (A2),D2 ; IS DESIRED VERT >= NEXTV ?
|
|
BGE.S DOWN1 ; YES, BUMP DOWN SOME MORE
|
|
sub (a2),d2 ; compute scan count = nextV-vert
|
|
neg d2 ; return in d1
|
|
|
|
MOVE (A2),NEXTV(A1) ; UPDATE NEXT VERT
|
|
MOVE.L A2,DATAPTR(A1) ; UPDATE DATAPTR
|
|
MOVEM.L (SP)+,D0/d3-D7/A2-A4 ; RESTORE REGS
|
|
DONE RTS ; AND RETURN
|
|
|
|
|
|
;-----------------------------------------------------
|
|
;
|
|
; TO MOVE UPWARDS, SCAN BACKWARDS THROUGH RGNDATA.
|
|
;
|
|
SEEKUP MOVE.L RGNPTR(A1),A0 ; GET THE REGION POINTER
|
|
MOVE.W RGNBBOX+TOP(A0),D1 ; GET TOPMOST VERT
|
|
CMP.W D1,D0 ; ABOVE REGION?
|
|
BLT.S UP2 ; =>YES, RESET TO START
|
|
|
|
MOVEM.L D0/d3-D7/A2-A5,-(SP) ; SAVE REGS
|
|
MOVE D0,D2 ; SAVE VERT
|
|
LEA RGNDATA(A0),A5 ; POINT TO START OF REGION DATA
|
|
LEA NEXTH1,A4 ; RETURN ADDRESS FOR INVPAIR
|
|
MOVE.W MINH(A1),D6 ; D6 = MINH FOR INVPAIR
|
|
MOVE.W MAXH(A1),D7 ; D7 = MAXH FOR INVPAIR
|
|
|
|
NEXTH MOVE.L DATAPTR(A1),A2 ; POINT TO VERT COORD
|
|
SUBQ #2,A2 ; POINT TO TERMINATOR
|
|
|
|
;-------------------------------------------------
|
|
;
|
|
; GET LEFT AND RIGHT HORIZ COORDS
|
|
; AND CALL INVPAIR TO TRIM AND INVERT BUFFER
|
|
;
|
|
NEXTH1 MOVE -(A2),D4 ; GET RIGHT COORD (OR VERT)
|
|
MOVE -(A2),D3 ; GET LEFT COORD (OR TERMINATOR)
|
|
CMP #32767,D3 ; IS IT A TERMINATOR ?
|
|
BEQ.S DONEH1 ; => YES, DONE WITH THIS SCAN
|
|
BRA.S INVPAIR ; ELSE INVERT THIS PAIR
|
|
|
|
DONEH1 MOVE.L A2,D0 ; GET POINTER TO TERMINATOR
|
|
ADDQ.L #2,D0 ; BUMP PAST TERMINATOR
|
|
MOVE.L D0,DATAPTR(A1) ; SAVE DATA POINTER
|
|
MOVE.W D4,NEXTV(A1) ; AND SAVE VERT
|
|
|
|
; DONE WITH THIS SCAN. SCAN BACK TO PRIOR VERT TO SET THISV.
|
|
|
|
NEXTH2 MOVE -(A2),D4 ; GET RIGHT COORD (OR VERT)
|
|
CMP.L A5,A2 ; AT BEGINNING OF DATA?
|
|
BLE.S DONEH2 ; => YES, DONE
|
|
MOVE -(A2),D3 ; GET LEFT COORD (OR TERMINATOR)
|
|
CMP #32767,D3 ; IS IT A TERMINATOR ?
|
|
BNE.S NEXTH2 ; => NO, KEEP SCANNING
|
|
DONEH2 MOVE.W D4,THISV(A1) ; save as first scan in this range
|
|
CMP D4,D2 ; IS DESIRED VERT < NEXTV ?
|
|
BLT.S NEXTH ; YES, BUMP UP SOME MORE
|
|
|
|
sub d4,d2 ; compute vert-thisV+1
|
|
addq #1,d2 ;
|
|
|
|
MOVEM.L (SP)+,D0/d3-D7/A2-A5 ; RESTORE REGS
|
|
RTS ; AND RETURN
|
|
|
|
|
|
INVPAIR
|
|
;-------------------------------------------------
|
|
;
|
|
; LOCAL PROCEDURE INVPAIR
|
|
;
|
|
; CLIP THE LEFT AND RIGHT COORDINATES IN D3 AND D4
|
|
; TO THE MINRECT, THEN INVERT THAT RANGE IN SCANBUF.
|
|
;
|
|
; D6 = MINH
|
|
; D7 = MAXH
|
|
; A4 = RETURN ADDRESS
|
|
|
|
CMP D6,D4 ;IS RIGHT <= MINH ?
|
|
BLE.S IDONE ;YES, IGNORE ON LEFT
|
|
CMP D7,D3 ;IS LEFT >= MAXH ?
|
|
BGE.S IDONE ;YES, IGNORE ON RIGHT
|
|
CMP D6,D3 ;IS LEFT < MINH ?
|
|
BGE.S LOK ;NO, CONTINUE
|
|
MOVE D6,D3 ;YES, TRIM LEFT
|
|
LOK CMP D7,D4 ;IS RIGHT > MAXH ?
|
|
BLE.S ROK ;NO, CONTINUE
|
|
MOVE D7,D4 ;YES, TRIM RIGHT
|
|
|
|
ROK SUB LEFTH(A1),D3 ;MAKE COORDS REL TO BUFFER
|
|
SUB LEFTH(A1),D4
|
|
|
|
;------------------------------------------
|
|
;
|
|
; GET LEFTMASK IN D5 AND RIGHTMASK IN D1
|
|
;
|
|
MOVEQ #$1F,D0 ;GET MASK FOR LO 5 BITS
|
|
|
|
MOVE D3,D1 ;COPY LEFT COORD
|
|
AND D0,D1 ;CALC LEFT MOD 32
|
|
MOVEQ #-1,D5 ;GET ALL ONES
|
|
LSR.L D1,D5 ;SHIFT IN 0'S FROM LEFT
|
|
|
|
AND D4,D0 ;CALC RIGHT MOD 32
|
|
MOVEQ #-1,D1 ;GET ALL ONES
|
|
LSR.L D0,D1 ;SHIFT IN 0'S FROM LEFT
|
|
NOT.L D1 ;AND COMPLEMENT FOR RIGHTMASK
|
|
|
|
|
|
;------------------------------------------
|
|
;
|
|
; CALC LEFTWORD, BUFPTR, WORDCOUNT
|
|
;
|
|
LSR #5,D3 ;CONVERT DOTS TO LONGS
|
|
MOVE.L SCANBUF(A1),A3 ;COPY BUFSTART
|
|
MOVE D3,D0 ;GET LEFT LONGS
|
|
LSL #2,D0 ;CONVERT TO BYTES
|
|
ADD D0,A3 ;INIT BUFPTR TO LEFTLONG
|
|
LSR #5,D4 ;CALC RIGHT DIV 32
|
|
SUB D3,D4 ;LONGCOUNT:=RIGHTLONG-LEFTLONG
|
|
BGT.S NOTIN1 ;BR IF NOT ALL IN ONE
|
|
|
|
|
|
;------------------------------------------
|
|
;
|
|
; LEFT AND RIGHT ARE ALL IN ONE WORD
|
|
;
|
|
AND.L D5,D1 ;COMBINE LEFT AND RIGHT MASKS
|
|
EOR.L D1,(A3) ;XOR COMBINATION INTO BUFFER
|
|
IDONE JMP (A4) ; => RETURN TO CALLER
|
|
|
|
|
|
;------------------------------------------
|
|
;
|
|
; NOT ALL IN ONE WORD. DO LEFT, MIDDLE IF ANY, THEN RIGHT
|
|
;
|
|
NOTIN1 EOR.L D5,(A3)+ ;XOR LEFTMASK INTO BUFFER
|
|
BRA.S TEST ;SEE IF ANY FULL LONGS
|
|
DO2LONG NOT.L (A3)+ ;INVERT 2 WHOLE LONGS
|
|
NOT.L (A3)+
|
|
TEST SUBQ #2,D4 ;ANY LONGS LEFT ?
|
|
BGT.S DO2LONG ;YES, AT LEAST 2
|
|
BLT.S ENDLONG ;NO, FINISH UP LAST WITH MASK
|
|
NOT.L (A3)+ ;YES, DO LAST FULL LONG
|
|
ENDLONG EOR.L D1,(A3) ;XOR RIGHTMASK INTO BUFFER
|
|
JMP (A4) ; => RETURN TO CALLER
|
|
|
|
|
|
|
|
ENDPROC
|
|
|
|
|
|
|
|
|
|
|