; ; File: PatchIIciROM.a ; ; Contains: patches for the ROMs first shipped in a Macintosh IIci ($067C) ; ; Copyright: © 1985-1991 by Apple Computer, Inc., all rights reserved. ; ; Change History (most recent first): ; ; <146> 12/2/91 SAM Using official boxflag equates now. ; <145> 12/2/91 pvh Added cache flushing to fixup patch table code. ; <144> 10/28/91 SAM/KSM Rolled in Regatta Changes. ; ; Regatta Change History: ; ; <15> 8/30/91 SAM Added a check for Tim boxFlag to the code that changes boxFlag ; from boxTim to BoxTimLC. This prevents non-Tim/TimLC machines ; using the TERROR ROM with power mgrs and no-FPU being identified ; as a TimLC. ; <14> 8/30/91 SAM Added a check for an FPU in the previous backpatch patch so it ; will not install on machines without one. ; <13> 8/29/91 SAM Added a check for Zydeco to the "should we remove (ie override) ; the PACK 4 & 5 SANE rsrcs in ROM" check. (CCH) Added a patch to ; _FP68k for 030 TERROR machines that will not allow _FP68k calls ; made from ROM to be backpatched. ; <12> 8/15/91 SAM (RMP) Added v32SetHandleSize patch. SetHandleSize calls the same ; common routine that v32SetPtrSize calls, a32SetSize. SetPtrSize ; was patched earlier to fix a bug in a32SetSize <121> but ; SetHandleSize was never patched. ; <11> 8/9/91 SAM Moved the disposal of the Gestalt tables made by ROM to after ; the reinstallation of Gestalt from disk. This prevents crashes ; caused by someone calling gestalt after the old tables have been ; trashed prior to the new table's creation. ; <10> 8/8/91 SAM (eva) patch level 4 autovector pointed to by vbr, not just the ; zero-offset cuz when VM is running VBR 0.This routine is a ; patch to the _VM trap. It fixes a bug in the Terror ROM which ; occurs when the page the stack is marked uncacheable in the MMU ; tables, then a subsequent PFLUSH'd is executed with data still ; in the cache. ; <9> 8/7/91 SAM Moved the boxFlag slam before Gestalt. Brainfade. ; <8> 8/6/91 SAM Added a check for TIM LC that will stuff the correct boxFlag ; into BoxFlag if we're running on a TIm LC. Gestalt's private ; storage allocated by the ROM version is disposed of prior to ; reinstalling gestalt from the disk. ; <7> 8/5/91 SAM (eva) fix patch <6>: Port B dispatcher fixed to call serial ; service routines and to have same number of bytes before the rts ; so that async serial driver Rx and SCx patches (which fiddle ; with return address on stack) return to the proper place. ; <6> 7/30/91 SAM (eva) fix the Farallon MacRecorder bug by modifing patch to test ; for port A RxChar and loop back only if we are servicing a channel ; A interrupt. For port B we just jump to service routine like ROM ; code. ; <5> 7/29/91 SAM (eva) patched SCx interrupt handler to fix the StyleWriter bug. ; On faster machines we are able to send out a byte fast enough ; after the TBE bit in RR0 gets set that the byte gets written ; before the TBE interrupt is generated. Instead of just not ; generating the interrupt, the SCC goes ahead and generates it, ; but changes the interrupt source code to read 'no interrupt' ; instead of 'TBE'. This 'no interrupt' code is identical to the ; 'Special Condition on port B' code, which was the source of our ; problem. Fix is to just exit SCIntHnd if no special condition ; bits are set. ; <4> 7/17/91 SAM (EH) Added Lvl4 irq patch to improve serial performance. ; <3> 6/30/91 SAM Conditionalizing the SCSI Chaining Bus Error handler for non ; C96. ; <2> 6/14/91 SAM Removed the GestaltHardware patch in favor of the real code in ; GestaltFunction.a. Fixed a check for an unallocated Palette Mgr ; handle in the InitPalettes Patch. (RMP) Added the IIfx IOP irq ; MPW fix to the set of Eclipse fixes. (bg) Changed the Egret ; "CheckPacket()" patch to allow the full ; range check of possible pseudo-commands. ; <1> 5/15/91 SAM Split off from 7.0 GM sources. Added code to use PACK 4 & 5 from ; disk on IIciROM machines that are not TERROR A7 or newer. Do ; not save GestaltÕs old Logical mem size across PtchInst 5 ; (which is Gestalt) cuz the old number is wrong. ; 7.0 Change History: ; ; <143> 9/18/91 JSM Cleanup header, donÕt include InternalOnlyEqu.a (already in dump ; file). ; <142> 8/30/91 DTY Define isUniversal here since itÕs no longer defined in ; BBSStartup. This file used to use {Defs32}, so isUniversal is ; defined to be 1. ; <141> 6/12/91 LN added #include 'ROMPrivateEqu.a' ; <140> 6/12/91 LN removed #include 'HardwareEqu.a' ; <139> 4/12/91 dba ewa: add magic signature to Time Manager task used to implement ; VBLs; the signature will prevent VM from deferring the task ; <138> 4/8/91 eh Fixed bug in iop serial driver that was causing bad printing to ; Imagewriter. Actually, it was two bugs. CTS interrupt disable ; code had a bad branch and the IOP timer never got kickstarted. ; <137> 4/5/91 eh Fix to Serial driver SerSetbuf call. Missed one place in <135> ; that sign extended a register. ; <136> 3/28/91 SAM As per DarinÕs request, I have added a TestFor SCSI96_1Exists ; before the Quantum 7.9 ROM fix from the Regatta deltas (without ; forRegattaCPUs). ; <135> 3/17/91 eh Serial driver status calls 9 and $8000 now return hardcoded ; driver version instead of getting it from DCE. This obviates the ; need for the linked patch to _Open to patch the version number ; in the DCE, a fix which was causing FileShare to crash. Also ; fixed bug in IOP SerSetBuff (control 9) call -- was sign ; extending a register unintentionally. ; <134> 3/8/91 DFH VL,WS#DFH-910308a: Fixed SetEntries patch to not chew on a2. ; This led to crashes, such as in AfterDark. ; <133> 3/4/91 dba dty: get rid of SysVers checks ; <132> 2/26/91 DFH csd,WS#910226a: Fixed bug in ptchPMgrExit where it assumed that ; graphics has been initialized. Bug introduced by change <124>. ; <131> 2/25/91 gbm csd, #83338: Fix the fix to Gestalt. Save the original Gestalt ; globals pointer, and put some important (but hard to figure out) ; data back in it after the RAM copy installs ; <130> 2/21/91 gbm csd, #83338: Start including the Gestalt ptch even on ci ROMs. ; <129> 2/21/91 eh (djw) Fixed bug in serial driver that was preventing use of one ; port when Nike was printing on the other. ; <128> 2/19/91 gbm csd, #Go5 approved: Make sure _MemoryDispatch is NOT implemented ; on machines without PMMUs. ; <127> 1/29/91 eh (djw) Fixed a bug in the External Status interrupt handler. On ; port A we crashed on receiving a break if a read was pending ; because we weren't getting the .Ain DCE ptr properly. Now we get ; the DCE ptr directly from the Unit table, just like for port B. ; <126> 1/25/91 eh Fix iop serial driver to: time out an waiting for all-sent in ; the close, and to disable HWHS properly when external clocking ; is set. ; <125> 1/19/91 eh (djw) Patch async serial bypass driver headers to insert ; signature longword used by linked patch to Open in Serial ; Patches.a . Added signature longword before already patch IOP ; driver headers. ; <124> 1/18/91 dvb Don't restore color environment if it hasn't been changed. ; Modify SetEntries to mark flag in QDSpare. ; <123> 1/14/91 eh (djw) Added external clocking support for the NIke printer to ; the Async Serial Driver. ; <122> 12/19/90 djw (ewa) Patch ROM SCSIDispatch to the ROM SCSIDispatch to resolve ; a VM patch conflict. ; <121> 12/19/90 TL Fixed an old bug in setptrsize for old time sake, in calls ; between relocRel, the tag is restored in the old block, instead ; of the new block location. And also discard flag when two block ; sizes are identical. ; <120> 12/19/90 dvb Don't draw windowframes after InitGDevice. Removed some ; duplicate code between PatchIIciROM.a and PaletteMgr.a. ; <119> 12/15/90 djw (jwk) Add SCSI Mgr support for Quantum 7.9 ROM problem by ; replacing blind write routine ; <118> 12/14/90 dnf (jsm) Turn all patches on ExtFSHook off for 7.0 since they have ; been moved to LaterFileMgrPatches.a ; <117> 12/14/90 SMC Patched DisposeAppPalettes via PMgrExit to use ApplZone->bklim ; instead of HeapEnd to determine if palette is in the application ; heap. Also, patched PMgrExit to only throw away palettes if app ; is front process. With DFH. ; <116> 12/14/90 dba <JDR> get rid of VMCalls.a ; <115> 12/6/90 BBM (dba) Protect the Come-from patches. Eleminate TextFont patch to ; the MDEF, as the MDEF in ROM is not used anymore, RomOverride at ; its best. ; <114> 11/15/90 JSM <bbm> Move come-from patch on _OpenCPort inside NewGWorld to ; QuickDrawPatches.a. ; <113> 10/31/90 dba & csd; kill 8¥24 GC loading code for 7.0 (StartSystem.a does it ; now) ; <112> 10/22/90 JJ Rex V8: Change VISAChipBit to V8ChipBit. ; <111> 10/22/90 JJ Rex V8: Change all box flag references to use boxMacLC. Get rid ; of boxElsie and boxElsieV8. ; <110> 10/9/90 SAM Updated the AppleTalk _Open patch so that it stuffs PortNotCf ; into ioResult if SPConfig >= 2. ; <109> 9/26/90 SAM Redoing the patch to Open that fixes the .MPP crash. ; <108> 9/25/90 KIP Change Sound Mgr. to a linked patch. ; <107> 9/18/90 SAM Added changes from SixPack code review (search for <107>). ; Removed EclipseNOPs (as per BGs request). Fixed the last of the ; build warnings for this file (rah). ; <106> 9/15/90 DC added GetGray ; <105> 9/15/90 GMR Fixed Gestalt parity routine to set good parity before testing ; the byte with bad parity, so the RPU interrupt would not cause ; bad data to be stacked. ; <104> 9/5/90 EH Removed duplicate declarations for PollProc, Pollstack, and ; HiIntMask from EgretMgr patch <101>. ; <103> 8/31/90 SAM Adding fix to ReallocHandle patch that checks to see if SysZone ; = AppZone and if so it only checks the current heap then jumps ; back to ROM. Added a patch to open so that it clears out the low ; nibble of SPConfig when the .MPP driver is opened. Made ; GestaltHardwareAttr recognize Visa/V8. ; <102> 8/29/90 GMR Fixed Gestalt parity check on FX so it does not destroy A4 (this ; reg needs to be preserved throughout patches and inits!) ; <101> 8/22/90 SAM Adding a series of REgret patches & a SANE ptch (search for ; <101> for more explicit descriptions). ; <100> 8/20/90 dvb Remove "DrawGrayishText" ; <99> 8/19/90 SAM Adding changes to make sound (MACE) work on ci ROMs ; <98> 8/18/90 dba get rid of ptchInst 7 and 8 (Sony Format and Eject patches) as ; they are now linked patches ; <97> 8/17/90 gbm start loading soundmgr patch 23 on erickson and elsie, cause ; BigBang don't work otherwise ; <96> 8/14/90 DTY Removed TextEditPatchIIciROM since itÕs a linked patch now. (For ; real this time.) Also deleted an INIT 18 patch for Darin. ; <95> 8/13/90 JWK NEEDED FOR SIXPACK: Added NuBus and Serial Gestalt selectors. ; <94> 8/11/90 PKE NEEDED FOR SIXPACK: Fixed bug in previous revision - (per KIP) ; ptch 3 must load before ptch 23 so SoundMgr globals are ; initialized!! Was causing ci & fx to hang during boot. ; <93> 8/10/90 SAM Made ptch 3 (7.0 Sound Mgr) load on all IIciROM machines. It ; used to *not* load on Erickson/Elsie. ; <92> 8/8/90 SAM Changing DispatchHelper & ProcHelper into an old style ptch. ; ¥¥¥--> Temporary <--¥¥¥ Remove when the Sound ptch get converted ; into an Lptch. ; <91> 8/7/90 PKE Fixed the SixPack build by adding an eclipseDebug conditional ; around a BRA.S. Build was broken because BG changed BBSStartup ; to turn on eclipseDebug for Mac32 (which applies to this file), ; so EclipseNOPs are expanded and a branch went out of range. ; Also, altered the previous comment to say needed for SixPack. ; <90> 8/5/90 PKE NEEDED FOR SIXPACK: Moved installation of Script Mgr ptch 39 and ; 27 to Mac and A/UX section of install code (was formerly for Mac ; OS only). Search for <90> to see changes. ; <89> 7/30/90 SAM Added Egret TickHandler patch so it doesnt loose time while IRQs ; are off. ; <88> 7/30/90 dnf Remove installation of ptch 18 (File Manager) and ptch 6 (Btree ; Manager), now linked patches ; <87> 7/23/90 dba get rid of ptch 25 for 7.0; it is covered by DialogMgrPatches; ; get rid of extraneous pre-6.0.6 SysVers conditionals; removed ; ptchInst 16 since PrGlue is a linked patch ; <86> 7/21/90 BG Added EclipseNOPs for flakey 040s. ; <85> 7/20/90 CSL Needed for SIXPACK: Totally replaced Memory manager routine ; a32SetSize, patch <68> only replaced part of a32SetSize and is ; not sufficient. ; <84> 7/20/90 DTY Removed Bass patches since itÕs a linked patch now. ; <83> 7/20/90 gbm Get rid of some warnings ; <82> 7/19/90 CCH NEEDED FOR SIXPACK: Removed HwPriv patch since it is now a ; linked patch. ; <81> 7/19/90 EH Add VM support for async.a, SetSerBuf control call ; <80> 7/19/90 GMR Install ptch 7 to change eject track from 79 to 40. ; <79> 7/18/90 DDG NEEDED FOR SIXPACK: Fixed previous comment. ; <78> 7/18/90 DDG NEEDED FOR SIXPACK: clean up lomem $B50 after the netbooting socket ; listener. ; <77> 7/17/90 DVB Fix DrawGrayishText, and include PaletteMgr.a ; <76> 7/14/90 DVB Add in new PMgr dispatches ; <75> 7/11/90 gbm get rid of a few duplicate symbols ; <74> 7/11/90 DDG NEEDED FOR SIXPACK: Fixed a bug in our new ReadDateTime call. We ; now return an error code of zero, instead of garbage. ; <73> 7/2/90 DTY Remove ptchInst 21 since Resource Manager extensions are now in ; a linked patch. ; <72> 6/28/90 DDG NEEDED FOR SIXPACK: Put in a fix for egret. Now the ; _ReadDateTime call does not actually call the clock chip. It ; simply reads the low mem global (Time) and returns that. This is ; OK, since Time is automatically updated. This fix only happens ; on Elsie and Erickson. ; <71> 6/25/90 DTY Removed ptchInst 9. ; <70> 6/25/90 KON Remove rSwapMMUMode macro 'cause it's in QDHooks. ; <69> 6/19/90 VL Remove ptchinst 29 since MiscPatches is a linked patch now. ; <68> 6/14/90 CSL Needed for Six-Pack. Fixed _setptrsize problem in memory ; mananger for 32 bit mode only. It will not de-reference from ; address $0 now. ; <67> 6/12/90 JSM Remove PtchInst 33 since PPC Toolbox is a linked patch now. ; <66> 6/7/90 EMT Remove PtchInst 17 since Layer Manager is a linked patch now. ; <65> 6/7/90 VL Remove PtchInst 28 since HelpMgr is a linked patch now. ; <64> 5/31/90 DDG Adding a special system error for machines that donÕt have an ; FPU. ; <63> 5/31/90 DDG Added the extFS patch from the PatchIIROM.a file. This will fix ; a strange bug where the wrong error code is being returned from ; the MountVol call. ; <62> 5/29/90 DDG NEEDED FOR SIXPACK: Changed all the sixpack conditionals from ; six-point-oh-seven to six-point-oh-six. ; <61> 5/17/90 KON Clear high word of d3 and d4 so compare is done as a word in ; ResizePalette. ; <60> 5/10/90 JSM AliasMgr now a linked patch, don't install it here anymore. ; <59> 5/2/90 CCH Fixed bug in parity routine for pre-IIci machines with IIci ; ROMs. ; <58> 5/1/90 CCH Added Scrolling throttle and Square menus to misc selector. ; <57> 4/22/90 csd commented out the installation of SCSI DMA so we stop trashing ; hard disks on the IIfx. ; <56> 4/19/90 NB Add PrGlue to the PtchInst list (#16). It's already on all other ; machines ; <55> 4/18/90 DDG Rolling in the ReallocHandle patches from the sys6 sources. ; <54> 4/16/90 DDG Rolled over changes from the split off sources including: a ; patch to hwPriv, an IOP serial driver fix, SCSI DMA is now ; disabled on the IIfx due to bugs, added an init (18) to fix the ; comm toolbox, added the patch for tiburon video cards, added ; patch 25 (generic system fixes), and finally changed the equate ; for spline fonts to use the build variable hasSplineFonts. ; <53> 4/11/90 dba move PPC after B-Tree Manager; get rid of old AppleShare patch ; for 7.0 ; <52> 3/30/90 DVB Add "WhatPal" PmgrDispatch ; <51> 3/27/90 PKE Deleted import (currently conditionalized out) of NewSwapIcon, ; which no longer exists in this PTCH. ; <50> 3/26/90 PKE Moved definition of symbols that control Script Mgr patch and ; 'ptch' installation into ScriptPriv.a. Renamed symbols and added ; some. ; <49> 3/23/90 NC Added ptch 43 for System 6.0.6 on up. This is for Sound. ; <48> 3/20/90 DDG Added code to not install patch 3 on Ericksons and above for ; Sys607 and above. ; <47> 3/12/90 DVB Fix PMgrExit so that ExitToShell doesn't crash! ; <46> 3/8/90 JAL Took out include of QuickEqu.a because it is already in ; StandardEqu.d. ; <45> 3/4/90 PKE Changed _UprText to _UpperText to match new Traps.a. ; <44> 3/1/90 GMR Fixed bug in Gestalt parity check for Zone-5, where longword was ; not being written back with good parity after each SIMM check. ; <43> 2/28/90 GMR Added include of EgretEqu.a to fix system disk build. ; <42> 2/28/90 PKE For 7.0, moved Script Mgr ROM patches into ÔptchÕ 39: patches to ; _Pack6 and _GetIndADB, fixes to String2Date/InitDateCache, ; installation of some new vectors, and the change to SwapIcon. ; For all of these except the SwapIcon change, the code is still ; here but conditionalized differently (in case we need it for ; 6.1). ; <41> 2/26/90 EH Rewrote <8.7> to use patch macros. Added on patch for 68000 IOP ; Serial Driver read bug. ; <40> 2/23/90 DVB PMgrExit no longer clears the windowlist. ; <39> 2/22/90 PKE Replaced obsolete _LwrStringToUpr opword with _UprText. Removed ; setting of temporary sVectFixSpExtra vector (see <12>); we no ; longer need it. ; <38> 2/21/90 JS NEEDED FOR 6.0.5: SetDepth also takes a mode ; <37> 2/16/90 EH Fixed CTS status bug and low baud rate bug in IOP Serial Driver. ; Fixed <27> to use the patch macros properly. ; <36> 2/15/90 EMT Moved <1.9> before <1.2> - both patch SetWinColor, SetCtlColor. ; <35> 2/15/90 DVB Change SetDepth's return to OSErr ; <34> 2/14/90 DVB Fix SetDepth's register trashage ; <33> 2/14/90 EH NEEDED FOR 6.0.5: 'Come from' patched IOPMoveData trap to fix ; IOP Serial Driver DMA hang bug. ; <32> 2/14/90 CCH NEEDED FOR 6.0.5 - Made sure external cache on Zone 5 does not ; interfere with the check to see if parity RAM is installed. ; <31> 2/13/90 SMB NEEDED for 6.0.5 - Added installation code for ptchGetLRPosition ; and ptchPixelWidths for TextEdit. ; <30> 2/9/90 EH NEEDED FOR 6.0.5: Patched Level4SccIopInt to set flags such that ; all the IOP completion routines be called immediately from the ; IOP Manager, rather than be run as deferred tasks. This, so we ; don't break Timbuktu/Remote. ; <29> 2/2/90 JWK NEEDED FOR 6.0.5: Changed SCSI Mgr bus error handler to allow ; chaining to previously installed handler if a non-SCSI bus error ; occurs. ; <28> 2/2/90 GGD NEEDED FOR 6.0.5: Modified the Deferred Task Manager to enable ; interrupts when returning just so QuickMail servers will ; continue to work (their bug, but we'll fix them for now). ; <27> 1/31/90 EH NEEDED FOR 6.0.5: 'Come from' patched IOPMsgRequest trap to fix ; IOP serial driver control call bug ; <26> 1/30/90 CCH NEEDED FOR 6.0.5: Modified gestaltHardwareAttr patch to set ; gestaltHasSCC bit in result if in IOP bypass mode. ; <25> 1/29/90 GGD NEEDED FOR 6.0.5, Adding a temporary patch to make cmd-period ; not hang the MPW shell on Zone-5, until the shell can be reved ; to fix the real problem. ; <24> 1/29/90 DVB NEEDED FOR 605 - Add PMgrVersion call ; <23> 1/25/90 KON NEEDED FOR 6.0.5: QDversion is offscreenVersion + $100, not ; $200. ; <22> 1/23/90 KON NEEDED FOR 6.0.5: Gestalt now gets the Quickdraw version from ; GWorld Offscreen version and adds 200. ; <21> 1/23/90 DVB NEEDED FOR 6.0.5 - Add HasDepth and SetDepth PMgr Dispatches ; <20> 1/22/90 CCH NEEDED FOR 6.0.5 - Added notification mgr selector. ; <19> 1/19/90 JWK NEEDED FOR 6.0.5 - Turn on SCSI DMA code in Zone5 final ROM's. ; <18> 1/18/90 DVB Include PalettePriv.a ; <17> 1/17/90 CCH NEEDED FOR 6.0.5: moved restoration of cache register in parity ; check routine. ; <16> 1/16/90 SMB NEEDED FOR 6.0.5: added install code for TENew and ; TEStylNew patch and included TextEditPriv.a. ; <15> 1/16/90 CCH NEEDED FOR 6.0.5: Turn off cache when checking RPU for parity. ; <14> 1/16/90 KON NEEDED FOR 6.0.5: Added ptch 26 ; <13> 1/12/90 CCH Added include of ÒHardwarePrivateEqu.aÓ. ; <12> 1/11/90 PKE Initialize Script Manager's new sVectFixSpExtra vector. ; <11> 1/11/90 DVB Remove _Zaplinks macro. ; <10> 1/8/90 CCH Added gestaltMiscAttr selector. ; <9> 1/5/90 GMR Added ptchInst 8; Sony Format patch is now in it's own patch ; file (FormatPatch.a). ; <8> 1/5/90 CCH NEEDED FOR 6.0.5. Added parity support for Zone 5, removed base ; address selectors. ; <7> 1/4/90 BBM NEEDED FOR 6.0.5. Make the new sound manager patch out the rom ; on all machines. ; <6> 1/4/90 PKE NEEDED FOR 6.0.5 (smb's changes from SS-6.0.4b23 to SS-6.0.4f1): ; ¥ Add install code for TEFindLine patch. ALSO: Updated ; conditionals so SS-6.0.4 changes go in 6.0.5 as well as 7.0. ; Cleaned up the BBS header and eliminated the EASE header. ; <5> 1/3/90 BAL NEEDED FOR 6.0.5: Fixed Zero width character problem in ; drawtext. ; <4> 12/19/89 EH REALLY fix the pound signs! ; <3> 12/19/89 EH try to fix the pound signs! ; <2> 12/19/89 EH NEEDED FOR 6.0.5: Fix bug in 7.1 async patch; switched registers ; <1> 12/17/89 CCH Adding for the first time into BBS. ; <8.8> 12/14/89 SWC NEEDED FOR 6.0.5: Added RecoverHandle patch to also check if the ; pointer points to a ROM resource. Removed the NewSCSI trap ; initialization since async SCSI will not be in Zone 5 until 7.0 ; or later. ; <8.7> 12/14/89 EVA NEEDED FOR 6.0.5 Fix SerialDriver.a patch to check SCCIOPFlag ; instead of PRAM to see which driver to load ; <8.6> 12/11/89 smb NEEDED FOR Scripts 604 AND (6.0.5 <= SysVers < 7.0): added ; installation code for TE routines CaretInsideRun, ; InvrtRectangle, SetKeyboard2Font. ; <8.5> 12/8/89 PKE NEEDED FOR Scripts604 AND FOR (6.0.5 <= SysVers < 7.0): Patch ; LwrString to handle 2-byte chars via Transliterate (yuck). For ; 7.0, we do this elsewhere. ; <8.4> 11/29/89 GGD NEEDED FOR 6.0.5: Re-Enabled the SANE optimization to bypass the ; Package Manager which never made it into 6.0.4. Fixed bug ; StripAddress that caused it to sometimes not strip when in a ; interrupt handler. Modified ADB auto/srq polling to not poll ; device addresses that are not in the device table. On OSS based ; machines, install a time manager task to simulate a 60.15Hz ; pseudo VBL because the OSS generates a 60.00Hz interrupt which ; is the wrong frequency. ; <8.3> 11/28/89 dvb Fixed bug in PMgr Zaplinks by working over pmgrDispatch. Also ; fixed bug in dispatch. ; <8.2> 11/22/89 EMT NEEDED FOR 6.0.5: Added humane scrolling. ; <8.1> 11/21/89 smb Added installation code for TE routines ptchGetWidth and ; ptchGetRLPosition. ; <8.0> 11/17/89 PKE Tail patch on _GetIndADB to fix Script Mgr SwapKybd routine, ; which cleared ADB keyboard driver dead state as a word: should ; be a long. ; <7.9> 11/10/89 KON PTCH'ed GetNewCWindow so when a palette with the same ID as the ; window exists in the resource file, the palette is attached to ; the window. ; <7.8> 11/6/89 PKE NEEDED FOR 6.0.5!! Bug fixes for InitDateCache and String2Date ; needed for HyperCard. InitDateCache: Fixed CopyArray to use ; correct register (A4) for source pointer, and to initialize all ; relevant bytes of length register (D0). String2Date: if first ; relevant (i.e., day or month name) alpha token is a month name, ; we now search the day name list if we find another alpha token ; (fixes BRC #54946). Rearranged Cache structure to fix invalid ; use of month/day name index. ; <7.7> 10/12/89 smb Moved TextEdit patches to TextEditPatchIIciROM.a. Fixed bug in ; old highlighting code that I introduced accidently (changed ; move.w -> move.l). ; <7.6> 10/11/89 CCH Added INCLUDE of "MMUEqu.a" ; <7.5> 10/11/89 CCH Set up MMUType if not already initialized. ; <7.4> 10/6/89 JSM Removed SnarfMan 'ptch', now PACK 13. ; <7.3> 10/5/89 CCH Changed an ORI to a BSET in gestaltAddressingModeAttr patch, and ; added installation of it. ; <7.2> 10/5/89 smb Added old highlighting code for TextEdit for HyperCard ; outlining. ; <7.1> 10/5/89 EVA NEEDED FOR 6.0.5!!! Fixed bug in async serial driver Ext/Stat ; int handler ; <7.0> 10/4/89 smb Fixed 6.0.4 build by moving around some of my conditionals due ; to dependencies! Fixed TEInit to set TEDoText like it used to. ; Cleaned up TEOutlineHilite. Fixed caret display for outline ; highlighting. ; <6.9> 9/29/89 CCH Added patch for gestaltAddressingModeAttr selector to add ; gestalt32BitCapable attribute. (version 6.8 breaks 6.0.4 builds, ; so I just tested on bigbang - because alpha 0 is in two days, ; thats why) ; <6.8> 9/29/89 smb Added patches for TextEdit for TEFeatureFlag, SetFont2Keyboard, ; and fixed CursorMovement for double-byte characters. ; <6.7> 9/27/89 EMT Fixed 6.2 to use Lea rather than LeaROM for RAM address. ; <6.6> 9/26/89 CVC Added PPC Toolbox as a 'ptch'. ; <6.5> 9/25/89 EMT Added ptchs 18, 6, 29. ; <6.4> 9/19/89 RLC Add Balloon Help (ptch 28) to this file. ; <6.3> 9/18/89 BBM Install resource manager extensions, ptch 21 ; <6.2> 9/18/89 PKE For 7.0, patch Script Manager internal routine SwapIcon to use ; new 'kscn' resource type instead of 'SICN'. ; <6.1> 9/17/89 PKE Decided to patch out GetScript/SetScript entirely in ptch 27, so ; we no longer patch them here (see 5.9). Also, for 7.0 we now ; install the Script Manager's Gestalt function in ptch 27 instead ; of here (see 2.5). ; <6.0> 9/12/89 EMT PaintOne patch (3.7) nukes the layer manager. Only use on pre ; 7.0 systems. ; <5.9> 9/6/89 PKE Patch Script Manager GetScript/SetScript routines to support ; additional font info verbs for 7.0, and set up this font info in ; Roman ScriptRecord. Set new sVectCallInterface vector. Redo ; Script Manager version setup, using new smgrVersPTCHRom equate. ; <5.8> 9/5/89 jwk NEEDED FOR 6.0.5: Turn on NewSCSI in ROM if on F19. ; <5.7> 9/4/89 PKE Install Script Manager 7.0 extensions, ptch 27. Set Script Mgr ; version for 7.0. ; <5.6> 8/31/89 JSM Added SnarfMan support. ; <5.5> 8/30/89 smb NEEDED FOR Scripts604 and Big Bang! Bug fix in ptchTEFindWord ; (fixed word dragging bug and now uses old FindWord code when ; RecalLines calls. Also added ptchMeasureWidth to fix a ; hilighting bug at line ends when the line direction is RL on a ; system that is normally LR (ie - Roman, Kanji, etc.) ; <5.4> 8/28/89 PKE For 7.0, initialize additional Script Mgr vectors for ; SMgrCalcRect and SMgrInitFonts. ; <5.3> 8/27/89 PKE NEEDED FOR Scripts604, 6.0.5: Bump Script Manager version for ; Scripts604 or SysVers > $604, since IntlForce bug fix (4.5) is ; installed. ; <5.2> 8/22/89 PKE NEEDED FOR 6.0.4: (really dba) Tail patch to TextFont which ; fixes an MDEF bug in calculating the icon height. Replaces ; GetResource patch in 3.7 which attempted to fix the same ; problem. ; <5.1> 8/21/89 PKE NEEDED FOR 6.0.4: Really put in 4.1, which was only present as a ; comment until now (although it existed on Fiction sources). ; <5.0> 8/21/89 PKE NEEDED FOR 6.0.4: (really CSD) Fixed MDEF 3.7 patch (again. Last ; time, right?) to load #$00200020 into D1 when appropriate ; instead of jumping back into ROM at the same place for both ; cases cicn found and cicn not found. ; <4.9> 8/21/89 PKE NEEDED FOR 6.0.4: (really dba) Changed MDEF patch 3.7 to load A0 ; with the right value instead of preserving its value. ; <4.8> 8/21/89 PKE NEEDED FOR 6.0.4: (Per CSD) Correct ROMNoColor address in MDEF ; patch (part of 3.7). ; <4.7> 8/21/89 PKE NEEDED FOR 6.0.4: ; ¥ (per dba) Only re-load the standard clut into QDColors if ; QDColors exists; this is so that A/UX will work ; ¥ Conditionalize 4.5 for Scripts604 OR (SysVers > $604) ; <4.6> 8/21/89 PKE NEEDED FOR 6.0.4!: Patch RelString,UprString,CmpString to ; re-introduce the bug in which "`" is converted to uppercase as ; "a". This is necessary to prevent HFS problems, since existing ; disk catalogs use the old, buggy sorting. Also remove SANE ; optimization (2.0); breaks Excel. ; <4.5> 8/21/89 PKE NEEDED FOR 6.0.4 SCRIPTS BUILD, 6.0.5: Extend Pack6 patch to fix ; a problem in LwrString,CharType,Transliterate,FindWord. These ; routines should save the IntlForce flags then clear them before ; the IUGetIntl call, restoring the flags afterward. This is so we ; get the itl2 tables for the correct script (instead of the ; tables for the system script). ; <4.4> 8/21/89 PKE NEEDED FOR 6.0.4!: Fixed accidental deletion of INCLUDE ; 'VideoPatch.a' in (4.3). ; <4.3> 8/21/89 PKE NEEDED FOR 6.0.4!!: Patch Pack6 to fix me-too problem with ; pointer to unlocked handle in Transliterate; affects Roman ; system (this problem looks familiar; did I dream that I already ; fixed this one?). ; <4.2> 8/18/89 smb NEEDED FOR 6.0.4: Another unlocked handle problem! ; AAAARRRRRGGGGHHHHHH! (See ptchOnSameLine below) ; <4.1> 8/18/89 DAF FOR 6.0.4 BUILD - Fixed a bug on non-CLUT devices in the ; SaveEntries trap on Aurora ; <4.0> 8/16/89 dba NEEDED FOR 6.0.4: InitProcMenu bug fixed: MBDF ID was not put in ; MenuList if MenuList was already allocated (InitMenus already ; called) ; <3.9> 8/15/89 prp Added Alias Manager and Folder Manager support. ; <3.8> 8/15/89 smb NEEDED FOR 6.0.4: Fixed some problems in my patch code for the ; bad pointer bug. ; <3.7> 8/15/89 dba NEEDED FOR 6.0.4: ; ¥ Shortened the SetWin/CtlColor patch by returning to ROM earlier. ; ¥ Patch PaintOne to do nothing if WWExist is non-0. ; ¥ Patch OpenCPort so that the current port is saved across ; NewGWorld. ; ¥ Patch GetResource so that GetIconSize in MDEF does not trash A0. ; ¥ Moved VideoPatch.a so that it does not happen on A/UX. ; ¥ Re-load the standard clut into QDColors so that the green channel ; is fixed ; <3.6> 8/14/89 djw NEEDED FOR 6.0.4: Fixed bug in slot mgr in pInitEntry routine. ; When calling _InsertSRTRec, and the sRsrc has a refnum ; associated with it, _sFindDevBase has bad spId. ; <3.5> 8/13/89 BAL NEEDED FOR 6.0.4: Added tail patch to InitPalettes to defeat ; spurious ExitToShell patching code (leftover from JP builds). ; (dvb really) ; <3.4> 8/11/89 smb Finally rolled in patches for a bad pointer bug (#51602) and an ; up & down arrow keys bug (#51961). See comments below. ; <3.3> 8/11/89 djw NEEDED FOR 6.0.4: Cleaned up file by deleting old comments, ; adding new comments, laying out patches in a "prettier" and more ; orderly format. Added A/UX runtime conditional check for ; Mac-only patches. Moved patches <2.4> and <2.7> to Mac OS only. ; Modified slot manager calls in patche <2.1> as per code review. ; <3.2> 8/9/89 CCH NEEDED FOR 6.0.4: Added patch to gestaltQuickdrawVersion to ; return two digit versions. ; <3.1> 8/8/89 smb NEEDED FOR 6.0.4: Reverted TEFindWord patch since the previous ; patch introduces an uglier bug that won't be fixed now. Also ; added gestalt function for TE. ; <¥3.0> 8/8/89 dba RealityÕs hard disk was full! Try submitting again. ; <3.0> 8/8/89 dba NEEDED FOR 6.0.4: added Menu Manager patches; InitProcMenu patch ; because it simply didnÕt work; MenuSelect patch to reinstate ; WaitMouseUp check ; <2.9> 8/8/89 PKE NEEDED FOR 6.0.4: In gestaltScriptMgr function, if gestalt ; selector is undefined, just leave gestalt result alone. ; <2.8> 8/8/89 smb NEEDED FOR 604: Added more TE patches for Aurora! ; <2.7> 8/7/89 GGD Changed SIntInstall again, to leave $FF as highest priority, but ; reverse handling of equal priorities. Modified to use less ; space. Changed GetBoardROM from <2.4> to be a ROM Offset, ; instead of absolute ROM address. ; <2.6> 8/6/89 DAF FOR 6.0.4 BUILD - Added TFB video card driver override for boot ; screen. This is the same code as is included in PatchIIROM.a ; (VideoPatch.a) ; <2.5> 8/5/89 PKE NEEDED FOR 6.0.4: Add Gestalt function for Script Manager, ; install corresponding Gestalt selectors. ; <2.4> 8/4/89 djw NEEDED FOR 6.0.4: Deleted patch <1.3> (no longer needed). Added ; code to re-init board id's in slot pram (fixes a bug in Aurora ; in InitPRAMRecs). ; <2.3> 8/2/89 EVA (forgot to) include hardwareEqu.a and SlotMgrEqu.a ; <2.2> 8/2/89 EVA FOR 6.0.4 (and above) Patched _SIntInstall to reverse slot ; interrupt queueing order ; <2.1> 8/1/89 CSD FOR 6.0.4: Added patch to suppress appleshare arrows on 32-bit ; video cards. ; <2.0> 7/26/89 GGD NEEDED FOR 6.0.4 Added SANE optimization to bypass the package ; manager ; <1.9> 7/25/89 DAF FOR 6.0.4 BUILD - Fixes for SetWinColor, SetCtlColor in 32-bit ; mode ; <1.8> 7/25/89 smb NEEDED FOR 6.0.4: Added TextEdit patches for Aurora. ; <1.7> 7/3/89 NJC Sound Manager Extensions (ptch 23) rolled in for real. ; <1.6> 6/21/89 NJC Added a ptchinst 3 to if it wasn't already there and added in a ; commented-out ptchinst 23 for the sound dispatcher,DJ, and S.M. ; enhancements. ; <1.5> 5/31/89 CEL Only defined Spline_Font variable if it is undefined - makes it ; easier to build test 6.0.4 systems ; <1.4> 5/22/89 AG Fixed my mistake in gettrapaddress call ; <1.3> 5/21/89 AG Remove NEW SCSI manager trap (A089) ; <1.2> 5/13/89 EMT Added Window Manager extensions (Layers). ; <1.1> 5/3/89 CEL Rolling in Bass for the first time into EASEÉ ; <1.0> 3/6/89 CCH Adding to EASE for first time. ;___________________________________________________________________________________________________ ; To Do: ; ;___________________________________________________________________________________________________ IF (&TYPE('SPLINE_FONT') = 'UNDEFINED') THEN SPLINE_FONT EQU hasSplineFonts ;<1.1-4april89-CEL> <DDG> ENDIF IF (&TYPE('Scripts604') = 'UNDEFINED') THEN Scripts604 EQU 0 ;<4.7><08/21/89 pke> ENDIF if (&type('isUniversal') = 'UNDEFINED') then isUniversal: equ 1 ; <142> Define isUniversal here instead of BBSStartup endif BLANKS ON STRING ASIS MACHINE MC68020 ; changed from MC68000 to MC68020 <24July89smb> MC68881 LOAD 'StandardEqu.d' ; INCLUDE 'PatchMacros.a' INCLUDE 'SlotMgrEqu.a' ;<2.3> INCLUDE 'HardwarePrivateEqu.a' INCLUDE 'UniversalEqu.a' ;<7.1> INCLUDE 'VideoEqu.a' INCLUDE 'ROMEqu.a' INCLUDE 'MMUEqu.a' ;<7.6> Include 'ScriptPriv.a' ; INCLUDE 'GestaltEqu.a' ; INCLUDE 'GestaltPrivateEqu.a' ; INCLUDE 'TextEditPriv.a' ; <1/16/90smb> <16> INCLUDE 'PackMacs.a' ; INCLUDE 'InternalMacros.a' ; <54> INCLUDE 'PaletteEqu.a' ; INCLUDE 'PalettePriv.a' INCLUDE 'ApplDeskBus.a' ; <8.0> INCLUDE 'EgretEqu.a' ; <43> INCLUDE 'IOPEqu.a' ; <8.4> INCLUDE 'AppleDeskBusPriv.a' ; <8.4> INCLUDE 'SCSIEqu.a' ; <19> INCLUDE 'SCSIPriv.a' ; <19> include 'ColorEqu.a' INCLUDE 'QDHooks.a' TERRORmajorVers EQU $067C ; TERROR major ROM version TERRORminorVers EQU $15 ; TERROR ROM minor version number ZYDECOminorVers EQU $17 ; Zydeco ROM minor version number ; The following symbols are defined in ScriptPriv.a. They control whether <50> ; various Script Mgr patches are resident and installed here (since many ; are also included in 'ptch' 39 or 27), and also control installation of ; Script Mgr 'ptch' resources. ; ; doScriptMgrGestalt ; in ptch 27 ; doScriptMgrPack6Fix ; in ptch 39 ; doScriptMgrStr2DatFix ; in ptch 39 ; doScriptMgrRstKCHRFix ; in ptch 39 ; doScriptMgrLwrString2 ; in ptch 27 ; doScriptMgrSetROMVers ; in ptch 39 ; doScriptMgrNewVectors ; in ptch 39 ; installScriptMgrPtch27 ; installScriptMgrPtch39 ; ;------------------------------------------------------------------------------- ROM67CFix PROC EXPORT IMPORT PatchInit EXPORT StartPatch,CutBack ; Cut back Code: ; ; StartPatch is the entry point for ROM67CFix. Upon entry D1.L contains our handle. ; StartPatch Bra PatchInit ; Carry out the patches <1.1-4april89-CEL> DC.B 'PTCH' ; resource type DC.W $67C ; patch ID $67C. DC.W 1 ; current version number. ; cut back the ram-based system code to exclude this initialization code ; d0 = size to cut patch down to (EndOfPatch - StartPatch) ; a0 = handle to this PTCH resource (passed into StartPatch in D1 from SysPatch) CutBack _SetHandleSize ; adjust our size <1.1-4april89-CEL> MOVEQ #$7F,D0 ; a soon to be large number SWAP D0 _CompactMem ,SYS ; of course there must be a comma! RTS ; all done ENDPROC ; <24July89smb> ;========================================================================================= ;========================================================================================= ;= = ;= RESIDENT PATCH INSTALL CODE IS BELOW THIS POINT = ;= = ;= Add patch code which remains resident in the system in the following section. The = ;= installation code for the resident patches are further down in the file. Please = ;= begin your patch code with a header which serves to delimit your code from the = ;= previous person's patch code. Use another patches header as an example. Leave at = ;= least 2 or 3 blank lines between your code and the next code. Be sure to be careful = ;= about your conditionals. The safest thing to do is to make your code self-contained = ;= and not depend on someone elses conditionals encompassing your code. = ;= = ;= To find a place to insert your resident patch code, search for 'EndOfPatch' = ;= = ;= Please be neat and follow the format = ;= Drive safely = ;= = ;========================================================================================= ;========================================================================================= ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ;********************** PaletteMgr Patches Start Here ************************ ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ INCLUDE 'paletteMgr.a' ptchInitPalettes PROC EXPORT RomInitPalettes EQU $3E1F0 ; <dvb 8 August 1989> 3.5 ; This neutralizes some spurious code in InitPalettes that patches ExitToShell ; the first time [InitPalettes is called and MultiFinder is active]. We depend ; on getting at least one InitPalettes before MultiFinder (from INIT 31). JSRRom RomInitPalettes ; Call the old one. MOVE.L PMgrHandle,A0 ; A0 = handle to pm globals MOVE.L A0,D0 ; Are we a pmgr? BEQ.S ptchIPEnd ; No=>bye. CMPI.L #-1,D0 ; Is it -1? <2> BEQ.S ptchIPEnd ; -> Yes, bail out. <2> MOVE.L (A0),A0 ; A0->pm globals TST.L OldExitToShell(A0) ; Has it been patched yet? BNE.S ptchIPEnd ; Yes => do nothing MOVE #1,OldExitToShell(A0) ; Put something there; pretend its patched ptchIPEnd RTS ENDPROC ;-------------- Start of PMgrDispatch patch added by dvb 28Nov89 <8.3> -------------------- PROC EXPORT ptchPMgrDispatch,pMgrProcs,pMgrProcsEnd IMPORT SetDepth,HasDepth,PMgrVersion IMPORT SetPaletteUpdates,GetPaletteUpdates,CheckColors IMPORT GetGray RomEntry2Index EQU $3E8C0 RomIndex2Entries EQU $3E9A0 RomDeltaRGB EQU $3CBF0 RomNewHiliteColor EQU $3F8D0 RomPMgrExit EQU $3DAA0 RomSaveFore EQU $3F910 RomSaveBack EQU $3F914 RomRestoreFore EQU $3F970 RomRestoreBack EQU $3F98E RomReinstantiateColors EQU $3F9B0 ;this routine should do something heh heh RomReleaseList EQU $3F9C0 RomPMgrDispatch EQU $3FA36 RomPMgrFinalError EQU $3FA4A RomRestoreDeviceClut EQU $3F8A0 RomResizePaletteplusEight EQU $3F818 RomFindLink EQU $3E460 RomUnhookAfterZapLinks EQU $3F61E RomUnhookGoHome EQU $3F642 RomZapLinksAtA EQU $3F70A ;ctReserveBit EQU 6 ptchResizePalette ;--------------------------------------------------- ; ; PROCEDURE ResizePalette(srcPalette:PaletteHandle; dstSize:INTEGER); AAA2/3; ; ; Set the number of entries in the palette, and properly dispose of some ; of them if the palette is shrinking. ;ResizePalette PROC EXPORT RPVars RECORD {A6Link},DECREMENT result DS.B 0 ; Result: none srcPal DS.B 4 ; input: palette handle dstSize DS.B 2 ; input: new size for palette return DS.B 4 A6Link DS.B 4 ; old contents of A6 linkSize DS.B 0 ;linky number ENDR WITH RPVars LINK A6,#linkSize MOVEM.L A2/D3-D4,-(SP) ; save 'em. moveq #0,d3 ;<KON 17May90> move.l d3,d4 ;<KON 17May90> JMPRom RomResizePaletteplusEight ptchZapLinks ; Following code is spliced out of CQD:PaletteMgr.a <dvb> ; ROM67C: Zaplinks bug is that it doesn't clear the ctReserve bit if its ; not on the main screen. Zaplinks is called by Pillage, UnreserveDevices, ; and ReleaseList. Pillage and UnreserveDevices end up clearing the bit ; themselves, so only ReleaseList must be patched here. ; ZLVars RECORD {A6Link},DECREMENT result DS.B 0 ;Result length: none myIndex DS.B 2 ;Input: device number and index to unlink return DS.B 4 A6Link DS.B 4 ;old contents of A6 linkSize DS.B 0 ;linky number ENDR WITH ZLVars LINK A6,#linkSize MOVEM.L A0-A2/D0-D4,-(SP) MOVE.L PMgrHandle,A1 MOVE.L (A1),A1 ; A1->PMgrData LEA LinkTabs(A1),A0 ; A0->Link Tables MOVE.L PListHandle(A1),A2 ; A2 = palette list handle MOVE.L (A2),A2 ; A2 -> palette info list MOVE myIndex(A6),D0 ; D0 = tricky link AND #$1FFF,D0 ; clear key bits MOVE D0,D1 LSR #8,D1 ; D1 = device number MOVE.L DevHandles(A1,D1*8),A1 ; A1 = device handle MOVE.L (A1),A1 ; A1->device MOVE.L gdPMap(A1),A1 ; A1 = pixmap handle MOVE.L (A1),A1 ; A1->pixmap MOVE.L pmTable(A1),D1 ; D0 = color table for this device BEQ.S @a ; Skip this if none MOVE.L D1,A1 ; A1 = color table handle MOVE.L (A1),A1 ; A1 -> color table CLR D1 MOVE.B D0,D1 ; We need to clear the upper bits CMP ctSize(A1),D1 ; Index in range? <fixed> BHI.S @a ; No=>skip this BCLR #ctReserveBit,ctTable+value(A1,D1*8) ; unreserve! @a JMPRom RomZapLinksAtA ; return to ROM code ;--------------------------------------------------- ; ; PROCEDURE PMgrExit; AAA2/12; ; ; Someone should call here when an application quits. Please. ; ptchPMgrExit PXVars RECORD {A6Link},DECREMENT result DS.B 0 ;Result: nothing return DS.B 4 A6Link DS.B 4 ; old contents of A6 frontPSN DS.B 8 ; a 64-bit process serial number myPSN DS.B 8 ; likewise inFront DS.B 2 ; a boolean linkSize DS.B 0 ; linky number ENDR ROMINTOPMGREXIT EQU $3DAB2 WITH PXVars MOVE.L WindowList,-(SP) ; Save the windowlist value LINK A6,#linkSize TST.B QDExist ; if InitGraf has not been called yetÉ BNE.S @doNothing ; (a5) not valid and there is nothing to do CMP.L #PMgrNil,PMgrHandle ; If the palette manager doesnt exist... BEQ.S @doNothing CLR.L AppPalette BSR.S DisposeAppPalettes MOVE.L (A5),A0 ; < 124 > BTST #3,QDSpare0(A0) ; Any change to color env? < 124 > BEQ.S @doNothing ; < 124 > BSR.S CheckForProcessMgr BEQ.S @front ; if no process mgr, assume we're in front MOVE #$0100,inFront(A6) ; default inFront to true SUBQ #2,SP ; space for result PEA frontPSN(A6) _GetFrontProcess TST (SP)+ ; problem getting front psn? BNE.S @front ; YES => just zap the cluts CLR.L myPSN(A6) MOVE.L #kCurrentProcess,myPSN+4(A6) SUBQ #2,SP ; space for OSErr result PEA frontPSN(A6) PEA myPSN(A6) PEA inFront(A6) _SameProcess ; are we the front process? ADDQ #2,SP ; OSErr leaves inFront true from above TST inFront(A6) BEQ.S @doNothing @front JSRROM ROMINTOPMGREXIT ; JSR CheckAllDeviceCluts ; A pretty simple patch, really. ; CLR -(SP) ; No Setentries on Scatter ; JSR ScatterDevices ; BSR.S CheckForJuggler ; Is Jugglertm active? ; BNE.S @doNothing ; No=>WMgr is void, next line dangerous ; JSR UpdateDevices ; MOVE.L mainDevice,theGDevice @doNothing UNLK A6 MOVE.L (SP)+,WindowList RTS ;--------------------------------------------------- ; ; FUNCTION CheckForProcessMgr:Boolean; ; ; Determine if the process mgr is present using Gestalt. ; (we assume that gestaltLaunchControl iff processMgr is here) ; Affect only A0/D0, since Gestalt is an OS trap. ; return Z-flag Clear if the process mgr is here (BNE ProcMgrTrue) CheckForProcessMgr MOVE.L #'os ',D0 ; type of question _Gestalt TST D0 ; OSErr from Gestalt? BEQ.S @a ; No=>test result SUBA A0,A0 ; Yes=>assume there's no procMgr @a MOVE.L A0,D0 BTST #gestaltLaunchControl,D0 ; clear the Z-flag if procmgr here RTS ;--------------------------------------------------- ; ; PROCEDURE DisposeAppPalettes(); ; ; Look through the palette list and dispose of any in the current App heap. DisposeAppPalettes MOVEM.L A0-A3/D0-D3,-(SP) ; save all registers MOVE.L PMgrHandle,A2 ; get paletteMgr handle CMP.L MinusOne,A2 ; is it there? BEQ GoHome ; => no, just return MOVE.L (A2),A1 ; point to data structure MOVE.L PListHandle(A1),A0 ; get handle to palette list _HLock ; and lock it down MOVE.L (A0),A3 ; point to palette list Move APalettes(A1),D3 ; get number of active handles Beq.s NoPals ; no friends => go home Add FreeSpaces(A1),D3 ; calculate total number of entries ;<13> Clr.L WindowList ; clear the list of windows for FrontWindow AWC.PB506 BRA.S FindEnd ; => check for no entries FindLoop Move.L PaletteRef(A3),D1 ; get first entry BEQ.S FindNext ; => no palette in entry MOVE.L ApplZone,A0 ; get application heap zone <117> CMP.L A0,D1 ; Are we before the heap zone? <117> BLO.S FindNext ; Yes => not in app heap CMP.L bklim(A0),D1 ; Are we past the heap zone <117> BHS.S FindNext ; => not in app heap MOVE.L D1,-(SP) ; push palette handle _DisposePalette ; and dispose it in place FindNext AddQ #PLstEntrySz,A3 ; bump to the next entry FindEnd DBra D3,FindLoop ; repeat for all spaces NoPals MOVE.L (A2),A1 ; point to palette stuff MOVE.L PListHandle(A1),A0 ; get handle to palette list _HUnlock ; and unlock it GoHome Clr.L AppPalette ; set us up for the next guy AWC.PB508 MOVEM.L (SP)+,A0-A3/D0-D3 RTS ; This is the Palette Manager's dispatch table. ROM routines ; which work are included with DcRom to give absolute ; addresses. New routines are offsets from the table base, with ; the low bit set, just for clarity. ptchPMgrError JMPRom RomPMgrFinalError pMgrProcs switch00 DcRom RomEntry2Index switch01 DcRom RomIndex2Entries switch02 DC.L RestoreDeviceClut-pMgrProcs+1 switch03 DC.L ptchResizePalette-pMgrProcs+1 switch04 DC.L ptchZapLinks-pMgrProcs+1 switch05 DC.L WhatPal-pMgrProcs+1 switch06 DC.L ptchPMgrError-pMgrProcs+1 ; Some unused dispatchs switch07 DC.L ptchPMgrError-pMgrProcs+1 switch08 DC.L ptchPMgrError-pMgrProcs+1 switch09 DC.L ptchPMgrError-pMgrProcs+1 switch10 DcRom RomDeltaRGB switch11 DcRom RomNewHiliteColor switch12 DC.L ptchPMgrExit-pMgrProcs+1 switch13 DcRom RomSaveFore switch14 DcRom RomSaveBack switch15 DcRom RomRestoreFore switch16 DcRom RomRestoreBack switch17 DcRom RomReinstantiateColors switch18 DC.L ReleaseList-pMgrProcs+1 switch19 DC.L SetDepth-pMgrProcs+1 switch20 DC.L HasDepth-pMgrProcs+1 switch21 DC.L PMgrVersion-pMgrProcs+1 switch22 DC.L SetPaletteUpdates-pMgrProcs+1 switch23 DC.L GetPaletteUpdates-pMgrProcs+1 switch24 DC.L CheckColors-pMgrProcs+1 switch25 DC.L GetGray-pMgrProcs+1 pMgrProcsEnd ptchPMgrDispatch CMP.B #(PMgrProcsEnd-PMgrProcs)/4,D0 BHS.S ptchPMgrError ; Too high a dispatch MOVE.L A0,-(SP) ; Preserve A0 LEA pMgrProcs,A0 ; A0->absolute table of dispatchs AND #$00FF,D0 ; Kill arg count from dispatch MOVE.L (A0,D0*4),-(SP) ; Push target as return address MOVE.L 4(SP),A0 ; Restore A0 RTD #4 ; And go to routine. ENDPROC ; <107> ;--- This equate must be kept outside of code/data modules --- ; <107> pMgrProcsSize EQU pMgrProcsEnd-pMgrProcs ; <107> ;-------------- Start of GetNewCWindow patch added by KON 10Nov89 -------------------- ptchGetNewCWindow PROC EXPORT RomGetNewCWindow EQU $21c90 ; ; First we call the existing routine to get a CWindow. Then we add a palette if one ; with the same ID exists in the resource file. ; result EQU 18 ; parameter equates winID EQU 16 wStor EQU 12 behind EQU 8 link a6, #0 ;no local parameters <10Nov89 KON> clr.l -(sp) ;room for result from ROM GetNewCWindow <10Nov89 KON> move.w winID(a6),-(sp) ;do same call as requested <10Nov89 KON> move.l wStor(a6),-(sp) move.l behind(a6),-(sp) JSRRom RomGetNewCWindow ;call the old one <10Nov89 KON> move.l (sp)+,result(a6) ;return result <10Nov89 KON> ; ; Check if palette with same ID as window exists (taken from getmgr.a) ; clr.l -(sp) ; make room for palette handle <erich> move winID(a6),-(SP) ; push window ID <erich> _GetNewPalette ; fetchez la palette <erich> move.l (sp)+,d0 ; well? <erich> beq.s NoWindowPltt ; sorry, no automatic palette today AWC.PB459 move.l D0,A0 ; get the palette AWC.PB459 move.l (A0),A0 ; dereference it AWC.PB459 move PmPrivate(A0),D1 ; grab the update bits AWC.PB459 lsr #1,D1 ; put them in position AWC.PB459 bset #NNewBit,D1 ; use the new CUpdates format AWC.PB459 bset #DisposeBit,PmPrivate(A0) ; set for automatic disposal AWC.PB459 move.l result(a6),-(sp) ; push the window <10Nov89 KON> move.l d0,-(sp) ; push the palette <erich> move D1,-(sp) ; push cUpdates AWC.PB459 _SetPalette ; <erich> NoWindowPltt ; AWC.PB459 unlk a6 move.l (sp)+,a0 ;return address add #10,sp ;strip parameters, leave result jmp (a0) ENDPROC ;-------------- End of GetNewCWindow patch added by KON 10Nov89 ---------------------- ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ;********************** End Of PaletteMgr Patches? ************************ ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ;____________________________________________________________________________ DAF <2.6> ; Window manager patch ; ; This patch corrects a bug in SetWin/CtlColor in 32-bit addressing mode. These routines ; (which operate on identical data structures) can accept a colortable whose ctSize=-1 ; which allows an application to specify that the wcTable be set to the default color ; set (via a copy of the color table handle). In 32-bit mode it is important that there ; be exactly one auxWinRec per window. This routine has a bug where -1 color tables can ; cause extra auxWinRecs to be allocated in 32-bit mode. ; ; ; procedure SetWinColor ( theWindow:windowPtr; newColorTable:CTabHandle ); ; ; This procedure changes the awCTable field of the theWindow's auxWinRec. ; If newColorTable is equal to the default record's CTabHandle, then ; the auxWinRec is deallocated. If this window does not have an ; auxWinRec, then one is allocated at the head of the list. If theWindow ; is NIL, then the default colors are changed. If a record gets deallocated, ; it's colorTable is disposed unless it has it's resource bit set (then the ; app must take responsibility) ; ; Most of this code is shared with the control manager. Since the auxRec ; offsets are the same for both types of auxrecs, the code should be OK, ; even though the code uses the auxWinRec names. ; NewSetWinColor PROC EXPORT EXPORT NewSetCtlColor ROMSetWinColorAfterSetGutsCall EQU $19744 ROMSetCtlColorAfterSetGutsCall EQU $1979C ROMGetAuxGuts EQU $19888 ; GetAuxGuts in Aurora ROM ROMSetGutsAfterSpecAlloc32Call EQU $197F6 ROMDeAlAux EQU $1983A ROMSpecAlloc EQU $1981A ROMSWCDone EQU $19816 MOVEM.L A2/A3,-(SP) ;get some address registers MOVE.L AuxWinHead,A2 ;get the list head LEA AuxWinHead,A3 ;in case we need to deallocate BSR.S SetGuts ;bsr so that we can redraw on return JMPROM ROMSetWinColorAfterSetGutsCall NewSetCtlColor MOVEM.L A2/A3,-(SP) ;get some address registers MOVE.L AuxCtlHead,A2 ;get the list head LEA AuxCtlHead,A3 ;in case we need to deallocate BSR.S SetGuts ; JMPROM ROMSetCtlColorAfterSetGutsCall SetGuts tWind EQU $18 ;some equates for parameters nCTab EQU $14 ; after LINK (don't forget the movem above!) LINK A6,#0 ;no globals CLR.L -(SP) ;return AuxWinHandle here CLR.W -(SP) ;function return here CLR.L -(SP) ;NIL for default PEA 6(SP) ;pointer to the placeholder MOVE.L A2,A1 ;set up list head for GetAuxGuts JSRROM ROMGetAuxGuts ;get the default record ADDQ #2,SP ;ignore boolean result MOVE.L (SP)+,A0 ;get the default handle MOVE.L (A0),A0 ;get a pointer MOVE.L awCTable(A0),A0 ;get the cTabHandle CMP.L nCTab(A6),A0 ;if the same then deallocate BEQ DeAlAux ; CMP.L #-1,nCTab(A6) ;if -1, then allocate a rec with default table BEQ.S SpecAlloc ; MOVE.L A0,-(SP) ;save a reg (yuck, but registers are tight!) MOVE.L nCTab(A6),A0 ;get the new color table handle MOVE.L (A0),A0 ;get a pointer CMP.W #-1,ctSize(A0) ;is ctSize -1? MOVE.L (SP)+,A0 ;get the register back (don't change flags) BEQ.S SpecAlloc32 ;If so, alloc a rec with default table JMPROM ROMSetGutsAfterSpecAlloc32Call DeAlAux JMPROM ROMDeAlAux SpecAlloc JMPROM ROMSpecAlloc ; if this routine is called in 24-bit mode, then it falls through to SpecAlloc and makes ; a new auxWinRec with a copy of the default colortable. If it is called in 32-bit ; mode, then it finds the window's pre-existing auxWinRec (one is always guaranteed to ; exist) and sets it's colortable to be a copy of the default colortable handle. SpecAlloc32 TST.B MMU32Bit ; are we in 32-bit mode? BEQ.S SpecAlloc ; no, so allocate a new record with a copy of the default colors MOVE.L A0,nCTab(A6) ; stick the default colors in the stack frame CLR.L -(SP) ; return AuxWinHandle here CLR.W -(SP) ; leave room for return value here MOVE.L tWind(A6),-(SP) ; find this window PEA 6(SP) ; point to placeholder MOVE.L A2,A1 ; set up list head for GetAuxGuts (windows or controls) JSRROM ROMGetAuxGuts ; get it (it will always be here in 32-bit mode) ADDQ #2,SP ; the result should have always returned true MOVE.L (SP)+,A0 ; get the win/ctl rec handle> MOVE.L (A0),A0 ; get a pointer MOVE.L nCTab(A6),awCTable(A0) ;and set the default colortable (in nCTab) field JMPROM ROMSWCDone ;____________________________________________________________________________ GGD <2.7> ; Slot interrupt patch for _sIntInstall ; ; <2.2/2.7> Slot Interrupt patch: patches _SIntInstall, and reverses Slot ; interrupt queue order ; ; Change the treatment of Equal Priorities to match the Mac II, which ran the most ; recently installed element soonest when priorities were equal. This code matches ; Inside Mac Vol 5, and treats $FF as highest priority. The Mac II has a bug and ; treats $FF as LOWEST priority within the slot, but as HIGHEST priority across slots. ; Most cards seemed to beleive the documentation, and none have multiple handlers per slot, ; One developer figured out that the Mac II was backwards within the slot, and wanted to ; insert an element ahead of a priority zero (highest with Mac II bug) element. To do ; this, he took advantage of the was that equal priorities were treated on the Mac II, ; which is why we are now patching this to match. ; SIntCorePtch proc export WITH SlotIntQElement,slotIntGlobals EXPORT SIntINSTALL ROMGetSlotIntQHead equ $00006E30 ; offset to GetSlotIntQHead ;_______________________________________________________________________ ; ; Routine: SIntINSTALL ; ; Arguments: D0 (input) : slot number for routine ($0..$E possible) ; A0 (input) : ptr to queue element ; D0 (output): error code - 0 no error ; ; Function: Installs a slot interrupt poll routine which is called when ; a slot interrupt occurs. The queue element is installed in ; in priority order, $FF highest (first), $00 lowest (last). ; ; When elements with EQUAL priorities exist, the most recently ; installed one is run sooner. ; ; <2.7> changes the treatment of Equal priorities to match Mac II. ; ; Only low byte of priority word is used (0-255 range). ; Priorities 200-255 are reserved by Apple.] ; ; The Mac II used to maintain priority across slots, which was ; used when multiple slots interrupted AT THE SAME TIME. ; We no longer support this, and prioritize by slot number instead. ; This will be removed from the Mac II in system 7.0 ; ; The Mac II had a bug in SIntInstall, which had $FF as the highest ; priority across slots, but had $FF as the lowest within a slot. ; Most cards used the priority order as documented, ($FF highest), so ; we will retain that ordering. ; This will be fixed in the Mac II in System 7.0. ; ; Registers Used: D0,D1,D2,A0,A1 ;_______________________________________________________________________ SIntINSTALL JsrROM ROMGetSlotIntQHead ; a1 := pointer to qheader move.b SQPrio+1(a0),d0 ; get the priority (low byte of word) move.w sr,-(sp) ; save interrupt level ori.w #HiIntMask,sr ; disable interrupts @PrioritySearch move.l SQLink(a1),d1 ; get next queue element to check beq.s @insert ; if end of list, insert before this exg.l a1,d1 ; swap previous and current pointers cmp.b SQPrio+1(a1),d0 ; compare priorities blo.s @PrioritySearch ; loop until lower priority found (was BLS.S) <2.7> ; if equal, insert newer in front of older <2.7> exg.l d1,a1 ; a1 points to previous, d1 points to current @insert ; insert before the lower priority element move.l d1,SQLink(a0) ; remainder of queue goes after new element move.l a0,SQLink(a1) ; new element goes after previous element move.w (sp)+,sr ; restore interrupt level moveq.l #noErr,d0 ; return success rts ; all done ENDP ;____________________________________________________________________________ <20> ; Patches to Gestalt Functions ;____________________________________________________________________________ ; The following gestalt selectors are patched or added in this file: ; - gestaltHardwareAttr ; - gestaltParityAttr ; - gestaltMiscAttr ; - gestaltVersion ; - gesaltNotificationMgrAttr ; - gestaltQuickdrawVersion ; - gestaltAddressingModeAttr ; - gestaltNuBusConnectors <95> ; - gestaltSerialAttr <95> ; ; The following gestalt selectors are removed in this file: ; - gestaltRBVAddr ; - gestaltSCCReadAddr ; - gestaltSCCWriteAddr ; - gestaltVIA1Addr ; - gestaltVIA2Addr ; - gestaltSlotAttr <95> ; - gestaltFirstSlot <95> ; - gestaltSlotCount <95> ; ;----------------------------------------------------------------------------------------- ; ; Record to describe gestalt function parameters ; ;----------------------------------------------------------------------------------------- gestaltParmFrame record {oldA6},decrement result ds.w 1 ; OSErr argSize equ *-8 gestaltSelector ds.l 1 ; packed array [1..4] of char gestaltResult ds.l 1 ; addr of longint result return ds.l 1 oldA6 ds.l 1 localFrame equ * endR ;____________________________________________________________________________ CCH <8> ; Gestalt function to effectively remove a selector ; ; The following is a gestalt function that will return an error as if the ; selector were undefined. ; ; Routine gestaltUndef ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; gestaltUndef PROC EXPORT with gestaltParmFrame link a6,#localFrame move.w #gestaltUndefSelectorErr,result(a6) ; return an error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ CCH <10> ; Gestalt function for gestaltMiscAttr ; ; The following is a patch to the gestaltMiscAttr selector. It returns ; miscellaneous information about things. ; ; Routine gestaltMisc ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; UserDelayTrap EQU $A84C ; user delay trap <17> gestaltMisc PROC EXPORT with gestaltParmFrame link a6,#localFrame clr.l d3 ; clear result @hasBootGlobs bset #gestaltBootGlobals,d3 ; we have boot globals @userDelay move.w #UnimplementedTrap,D0 ; get loc of unimplemented trap <58> _GetTrapAddress ,newTool ; get the address into A0 Move.l A0,D2 ; save it for a sec Move.l #UserDelayTrap,D0 ; trap ID to check for scrolling throttle _GetTrapAddress ,newTool ; get the address of it cmp.l a0,d2 ; is it unimplemented? beq.s @squareMenus ; nope.. <58> bset #gestaltScrollingThrottle,d3 ; <58> @squareMenus move.b NTSC,d0 ; get a copy of the NTSC byte <58> andi.b #$0F,d0 ; only look at bottom nibble <58> bne.s @next ; if it's non-zero, menus aren't square <58> bset #gestaltSquareMenuBar,d3 ; otherwise, it is! <58> @next move.l d3,d0 ; put result into d0 <58> move.l gestaltResult(a6),a0 ; get address to place result move.l d0,(a0) ; move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ CCH <20> ; Gestalt function for gestaltNotificationMgrAttr ; ; The following is a patch to the gestaltNotificationMgrAttr selector. ; It indicates existence of the notification manager trap. ; ; Routine gestaltNMgr ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; NMInstallTrap EQU $A05E ; notification manager install trap <20> gestaltNMgr PROC EXPORT with gestaltParmFrame link a6,#localFrame clr.l d0 ; clear result Move.w #UnimplementedTrap,D0 ; get loc of unimplemented trap _GetTrapAddress ,newTool ; get the address into A0 Move.l A0,D2 ; save it for a sec Move.l #NMInstallTrap,D0 ; trap ID to check for notification mgr _GetTrapAddress ,newTool ; get the address of it Cmp.l A0,D2 ; is it unimplemented? Beq.s @noNMgr ; it is?! bset #gestaltNotificationPresent,d0 ; otherwise it exists @noNMgr move.l gestaltResult(a6),a0 ; get address to place result move.l d0,(a0) ; move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ CCH <8> ; Gestalt function for gestaltVersion ; ; The following is a patch to the gestaltVersion selector. It returns the ; correct version number. ; ; Routine getGestaltVers ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; getGestaltVers PROC EXPORT with gestaltParmFrame link a6,#localFrame move.l gestaltResult(a6),a0 ; get address to place result move.l #1,(a0) ; return response determined at install time move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ jwk <95> ; Gestalt function for gestaltNuBusConnectors ; ; The following returns a bitmap of slots with NuBus connectors. ; gestaltNBCon PROC EXPORT MaxSlots EQU 16 ; max number of slots possible with gestaltParmFrame, NuBusInfo, ProductInfo link a6,#localFrame movea.l UnivInfoPtr,a0 ; get universal info ptr adda.l NuBusInfoPtr(a0),a0 ; add in offset to NuBus info clr.l d0 ; clear index clr.l d1 ; clear bitmap of NuBus connectors @slotLoop btst #hasConnector,(a0,d0.w) ; does this slot have a connector? beq.s @next ; if not, go look at next slot bset.l d0,d1 ; note that this slot has a NuBus connector @next addq #1,d0 ; bump up to next slot cmpi.l #MaxSlots,d0 ; have we gone through all slots? bne.s @slotLoop ; if not, keep going move.l gestaltResult(a6),a0 ; get address to place result move.l d1,(a0) ; return the bitmap of NuBus connectors move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ EH <95> ; Gestalt function for gestaltSerialAttr ; ; The following is a patch to the gestaltSerialAttr selector. ; It indicates existence of serial attributes on the machine. ; gestaltSerial PROC EXPORT with gestaltParmFrame,ExpandMemRec,GestaltGlobals link a6,#localFrame movea.l ExpandMem,a0 ; get ptr to expandmem rec movea.l emGestalt(a0),a0 ; get gestalt global ptr moveq.l #0,d0 ; assume no GPI connections cmpi.w #gestaltMacLC,machType(a0) ; <107> use real machine name now... beq.s @noGPI ; IIci and IIsi have GPI connected moveq.l #(1<<gestaltHasGPIaToDCDa)|\ ; GPIa connected to DCDa (1<<gestaltHasGPIaToRTxCa)|\ ; GPIa connected to RTxCa clock input) (1<<gestaltHasGPIbToDCDb),d0 ; GPIb connected to DCDb @noGPI move.l gestaltResult(a6),a0 ; get address to place result move.l d0,(a0) ; move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ CCH <8> ; Gestalt function for gestaltParityAttr ; ; The following is a patch to the gestaltParityAttr selector. It correctly ; handles parity on Zone 5, as well as the IIci. ; ; Routine gestaltParity ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; gestaltParity PROC EXPORT EXPORT parityValue with gestaltParmFrame link a6,#localFrame move.l gestaltResult(a6),a0 ; get address to place result move.l parityValue,(a0) ; return response determined at install time move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return parityValue dc.l 0 ; response for parity selector ENDP ;____________________________________________________________________________ CCH <3.2> ; Gestalt function for gestaltQuickdrawVersion ; ; The following is a patch to the gestaltQuickdrawVersion selector. ; ; Routine gestaltQDVers ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; gestaltQDVers PROC EXPORT with gestaltParmFrame link a6,#localFrame clr.l -(sp) ;make room for result _OffScreenVersion move.l gestaltResult(a6),a0 ; get address to place result move.l (sp)+, d0 ; get QD version from GWorld add.w #$100, d0 move.l d0, (a0) ; return version number for ci move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP ;____________________________________________________________________________ CCH <6.9> ; Gestalt function for gestaltAddressingModeAttr ; ; The following is a patch to the gestaltAddressingModeAttr selector to add the ; gestalt32BitCapable bit to this selector. ; ; Routine gestaltAddrMode ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; gestaltAddrMode PROC EXPORT with gestaltParmFrame link a6,#localFrame clr.l d0 ; clear result value <6.9> move.b SystemInfo,d0 ; get status byte into d1 <6.9> not.b d0 ; invert status bits <6.9> and.l #3,d0 ; only want bit 0 and bit 1 <6.9> bset #gestalt32BitCapable,d0 ; return that ROM is 32-bit clean <6.9> move.l gestaltResult(a6),a0 ; get address to place result move.l d0,(a0) ; save result <6.9> move.w #noErr,result(a6) ; return no error unlk a6 move.l (sp)+,a0 ; get return value add.l #argSize,sp ; restore stack pointer jmp (a0) ; return ENDP IF doScriptMgrGestalt AND (NOT installScriptMgrPtch27) THEN ; <6.1><50> ;____________________________________________________________________________ pke <2.5> ; Gestalt function for Script Mgr <08/05/89 pke> ; ; The following Gestalt Function is an interface between the Gestalt mechanism ; and the Script Manager's GetEnvirons routine. Currently, it only supports two ; Gestalt selectors: gestaltScriptMgrVersion and gestaltScriptCount. ; ; Routine gestaltScriptMgr ( ; gestaltSelector: OSType; = PACKED ARRAY [1..4] OF CHAR; ; VAR gestaltResult: Longint; ; ): OSErr; = Integer; ; ; <08/05/89 pke> New today ; gestaltScriptMgr proc export export gestaltSMgrTable ; table for installation <8/05/89 pke> with gestaltParmFrame ; <3.2> link a6,#localFrame ; initialize loop, set up default return values move.l gestaltSelector(a6),d0 ; selector value ;; move.l gestaltResult(a6),a0 ; addr for result ;; clr.l (a0) ; initially set result to 0 move.l #gestaltUndefSelectorErr,result(a6) ; assume unknown selector lea gestaltSMgrTable,a1 ; loop to find Gestalt selector in table @gestaltLoop move.l (a1)+,d1 ; get next table entry beq.s @gestaltDone ; end of list, quit move.w (a1)+,d2 ; now get GetEnvirons verb cmp.l d0,d1 ; is selector correct? bne.s @gestaltLoop ; no, get next one ; ok, we found the Gestalt selector. Now call GetEnvirons with correct verb clr.l -(sp) ; space for result move.w d2,-(sp) ; push verb _GetEnvirons move.l gestaltResult(a6),a0 ; addr for result move.l (sp)+,(a0) ; pop result into Gestalt move.w #noErr,result(a6) ; return no error ; all done @gestaltDone unlk a6 move.l (sp)+,a0 add.l #argSize,sp jmp (a0) endWith ; Table for converting between gestalt selectors and GetEnvirons verbs. Each pair ; consists of a long with the gestalt selector, followed by a word with the ; GetEnvirons verb. This table and the equate files where the Gestalt selectors ; are defined are the only things that need to change to add new Gestalt selectors ; for the Script Manager. gestaltSMgrTable dc.l gestaltScriptMgrVersion dc.w smVersion dc.l gestaltScriptCount dc.w smEnabled dc.l 0 ; terminator endProc ENDIF ; <6.1> IF NOT SPLINE_FONT THEN ; <6.0.5> ;____________________________________________________________________________ CEL <5> ; NewRSect (fixes DrawText) ; ; DrText had a check to see if the advance width of a character was zero. If so, ; it would skip the character. This was bad since International uses zero width ; characters to lay out certain kanji or vertical text. only one line needed to ; be taken out. The patch is big but the change is small. ; NewRSect proc export ;------------------------------------------- ; ; KERNED STRIKE FONT FORMAT OFFSETS: ; FORMAT EQU 0 ;WORD MINCHAR EQU 2 ;WORD MAXCHAR EQU 4 ;WORD MAXWD EQU 6 ;WORD FBBOX EQU 8 ;WORD fKernMax FBBOY EQU 10 ;WORD save as nDescent; unused except as high owTLoc FBBDX EQU 12 ;WORD fRectWidth FBBDY EQU 14 ;WORD fRectHeight LENGTH EQU 16 ;WORD owTLoc locASCENT EQU 18 ;WORD Conflict with other Ascent and Descent defines locDESCENT EQU 20 ;WORD XOFFSET EQU 22 ;WORD leading RASTER EQU 24 ;WORD rowWords ;------------------------------------------------------ ; ; A6 OFFSETS OF PARAMETERS AFTER LINK: ; PARAMSIZE EQU 14 ;SIZE OF PARAMETERS COUNT EQU PARAMSIZE+8-2 ;WORD TEXTADDR EQU COUNT-4 ;LONG NUMER EQU TEXTADDR-4 ;LONG, POINT DENOM EQU NUMER-4 ;LONG, POINT ;----------------------------------------------------------- ; ; A6 OFFSETS OF LOCAL VARIABLES AFTER LINK: ; SAVESTK EQU -4 ;long stack before allocating buffers TEXTRECT EQU SAVESTK-8 ;RECT original bounding rect TEXTR2 EQU TEXTRECT-8 ;RECT copy of textRext used by maprect SRCRECT EQU TEXTR2-8 ;RECT original shadow bounding rect DSTRECT EQU SRCRECT-8 ;RECT copy of srcRect used by maprect MINRECT EQU DSTRECT-8 ;RECT minimum rect for clipping BUFEND EQU MINRECT-4 ;LONG where the offscreen buffer ends BUFSIZE EQU BUFEND-2 ;WORD the size of the offscreen buffer BUFROW EQU BUFSIZE-2 ;WORD the rowBytes for the offscreen buffer BUF2START EQU BUFROW-4 ;LONG second buffer used for shadow BUF2END EQU BUF2START-4 ;LONG where shadow buffer ends BUFLEFT EQU BUF2END-2 ;WORD output left edge HEIGHT EQU BUFLEFT-2 ;WORD font fRectHeight copy SRCPTR EQU HEIGHT-4 ;LONG used only in wide character case as temp DSTPTR EQU SRCPTR-4 ;LONG used only in wide character case as temp BUFSTART EQU DSTPTR-4 ;LONG \ start of bits buffer SRCADDR EQU BUFSTART-4 ;LONG >- these 3 grouped: address of font bits SRCROW EQU SRCADDR-4 ;LONG / rowbytes of font bits maskStart EQU SRCROW-4 ;LONG \ start of mask buffer maskAddr EQU maskStart-4 ;LONG >- these 3 grouped: address of mask bits mSrcRow EQU maskAddr-4 ;LONG / rowbytes of mask bits FROMRECT EQU mSrcRow-8 ;RECT mapRect parameter TORECT EQU FROMRECT-8 ;RECT mapRect parameter PENLOC EQU TORECT-4 ;POINT copy of original pnLoc SPWIDTH EQU PENLOC-4 ;FIXED POINT width of space CHARLOC EQU SPWIDTH-4 ;FIXED POINT fractional pen position HEIGHTAB EQU CHARLOC-4 ;LONG pointer to font height table WIDTAB EQU HEIGHTAB-4 ;LONG pointer to font offset/width table LOCTAB EQU WIDTAB-4 ;LONG pointer to font location table SAVEA5 EQU LOCTAB-4 ;LONG register saved so can be reused characterExtra EQU SAVEA5-4 ;LONG fixed point extra added to each character maskBitsPtr EQU characterExtra-4 ;LONG pointer to maskBits, sourcePix, or 0 bkCol EQU maskBitsPtr-4 ;LONG full of the background color leftBack EQU bkCol-4 ;LONG bkCol masked to the left part of the character rightBack EQU leftBack-4 ;LONG bkCol masked to the right part of the character REALBOUNDS EQU rightBack-4 ;LONG USED FOR SHIELDCURSOR leftOffset EQU REALBOUNDS-8 ;2 LONGs offset used by bit field instructions in charblt maskSize EQU leftOffset-2 ;WORD size of mask buffer mBufRow EQU maskSize-2 ;WORD \ these 2 mask buffer row size maskBlts EQU mBufRow-6 ;3 WORDS / grouped: saved state for mask blit FAKERGN EQU maskBlts-10 ;RECTANGULAR REGION FAKEPTR EQU FAKERGN-4 ;LONG, FAKE MASTER POINTER INFO EQU FAKEPTR-8 ;4 WORDS font info record returned by txMeasure (unused) NUMER2 EQU INFO-4 ;Point copy of numer for iterative case DENOM2 EQU NUMER2-4 ;Point copy of denom for iterative case charsRemain EQU DENOM2-2 ;word remaining characters to draw in iterative case SRCBITS EQU charsRemain-14 ;bitMap input to shadow stretchBits, bitsToPix SRCPIX EQU SRCBITS-(PMREC+CTREC+20) ;pixMap input to normal stretchbits maskBits EQU SRCPIX-14 ;bitMap input to bitsToPix for font mask DSTPIX EQU maskBits-(PMREC+CTREC+20) ;pixMap destination FASTFLAG EQU DSTPIX-1 ;BYTE flag set if source ORing to screen is OK maskFont EQU FASTFLAG-1 ;byte flag set if a maskFont is available + requested STRETCH EQU maskFont-1 ;BOOLEAN flag set if numerator not equal denominator HEIGHTFLAG EQU STRETCH-1 ;byte flag set if font has a height table TOPHT EQU HEIGHTFLAG-2 ;word character top & height from font or clip MAXMIN EQU TOPHT-2 ;word number of characters in font MINCH EQU MAXMIN-2 ;word first character in font bitDepth EQU MINCH-2 ;word \ These two bits per pixel in font bkCol1 EQU bitDepth-2 ;word / grouped. 1 pixel worth of background color italicSlop EQU bkCol1-2 ;word extra width due to italic, bold, outline, shadow kernAdjust EQU italicSlop-2 ;word left kerning due to kerning and italic penLocHFrac EQU kernAdjust-2 ;word fractional pen position for recursive calls longCount EQU penLocHFrac-2 ;word loop counter for doMove charWidth EQU longCount-2 ;word width in pixels of current character blt stackOffset EQU charWidth-2 ;word 2 if stack was word aligned before link countCopy EQU stackOffset-2 ;word copy of character count, decremented as drawn CRSRFLAG EQU countCopy-1 ;BYTE set if crsr is shielded (to screen) MMUSave EQU CRSRFLAG-1 ;BYTE MMU mode on entry to drawText locMode EQU MMUSave-2 ;word copy of text mode, adjusted if arith. + 1 bit bitShift EQU locMode-2 ;word how far to shift to multiply by bitDepth orNotOK EQU bitShift-1 ;Boolean true if bit extract/insert must be used instead notMaskPass EQU orNotOK-1 ;Boolean true if blit is not creating font mask textCopyMode EQU notMaskPass-1 ;Boolean true if blit must use extract/insert orMode EQU textCopyMode-1 ;Boolean true if mode is srcOr & forecolor is black colorSource EQU orMode-1 ;Boolean true if font contains colors (nonblack/white) saveHilite EQU colorSource-1 ;byte saved hilite flag for iterative state run32bit EQU saveHilite-1 ;byte flag to run 32 bit VARSIZE EQU (((run32bit+1)/4)-1)*4 ;SIZE OF VARIABLES long aligned FromDrText EQU $28468 ROMRSect EQU $2C6D4 RomIterate EQU $281AA cmpRA FromDrText,(sp) ; is this called from DrText? beq.s @doDrTextPatch ; yes, go fix the patch jmpROM ROMRSect ; call back to the ROM @doDrTextPatch ADDQ #4, sp ; Pop off return addr and come back here jsrROM ROMRSect ; call back to the ROM ;--------------------------------------------------------------------------------------- ; < BELOW COPIED VERBATIM FROM THE ROM UNTIL FURTHER NOTICE > ; BEQ GOHOME ;QUIT IF NO INTERSECTION MOVE (SP)+,textR2+right(A6) ;restore text right ; ; Test to see if spline: ; If spline set up clipping variables ; ; ; Set up srcAddr, srcRow, and height ; LEA 26(A2),a0 ;GET START OF FONT BITMAP MOVE.L a0,SRCADDR(A6) ;SAVE FOR LATER MOVEQ #0,D1 ;zero high word MOVE RASTER(A2),D1 ;GET WORDS PER ROW IN FONT MOVE bitShift(A6),D0 LSL.L D0,D1 ;scale up font rowWords ADD.L D1,D1 ;DOUBLE FOR BYTES PER ROW MOVE.L D1,SRCROW(A6) ;REMEMBER FOR LATER MOVE FBBDY(A2),HEIGHT(A6) ;SETUP HEIGHT FOR LATER ; ; Test for fast case: ; not stretched, no color mapping, txMode = srcOr, same bits per pixel ; not bold, italic, underlined, outlined or shadowed, ; visRgn and clipRgn both rectangular. ; TST.B fmOutItalic(A4) ;TEST BOLD $$$ BNE NOTFAST ;NOT FAST UNLESS BOTH ZERO TST.B fmOutBold(A4) ;TEST BOLD BNE NOTFAST ;NOT FAST TST.B orMode(A6) ;IS TEXT MODE SRCOR ? (or srcCopy + mask, see above) BEQ NOTFAST ;NO, NOT FAST TST.W 10(A4) ;TEST ULTHICK AND SHADOW BNE NOTFAST ;NOT FAST UNLESS BOTH ZERO TST.B STRETCH(A6) ;IS TEXT STRETCHED ? BNE NOTFAST ;YES, NOT FAST MOVE.L ([CLIPRGN,A3]),A0 ;GET CLIPRGN HANDLE, dereferenced MOVEQ #10,D0 CMP RGNSIZE(A0),D0 ;IS CLIPRGN RECTANGULAR ? BNE NOTFAST ;NO, NOT FAST MOVEQ #1,D0 MOVE bitShift(A6),D1 ;get the depth of the source map LSL D1,D0 ;turn into 1 ÉÊ8 CMP DSTPIX+PIXELSIZE(A6),D0 ;same depth per pixel? BNE NOTFAST ;=>NOPE cmp #16,dstPix+pixelType(A6) ;is it a direct device? @@@@ BAL 16Jun88 bne.s @modeok seq textCopyMode(a6) ;yes, can't go fast for now @@@@ BAL 16Jun88 clr locmode(a6) ;this essentially alters mode to srcCopy bra NotFast @modeok MOVE.L VISRGN(A3),A1 ;GET VISRGN HANDLE MOVE.L (A1),A0 ;DE-REFERENCE IT CMP #10,RGNSIZE(A0) ;IS VISRGN RECTANGULAR ? BEQ.S FAST ;YES, TAKE FAST OPTIMIZATION ; ; All systems go except for VisRgn not rectangular. ; Check if visRgn sect minRect is rectangular. ; IF TrimRect(visRgn,minRect) THEN take the fast way. ; MOVE.L A1,-(SP) ;PUSH VISRGN PEA MINRECT(A6) ;PUSH MINRECT MOVE.W #-1,-(SP) ;pass Trim = True _TRIMRECT ;CALL TRIMRECT BLT GOHOME ;quit if intersection empty BGT NOTFAST ;continue if non-rectangular ; ; Fast case, go directly to screen. ; If text is clipped vertically, then clear heightflag and update TOPHT ; FAST ST FASTFLAG(A6) ;REMEMBER WE'RE GOING FAST CLR.B maskFont(A6) ;no need for second mask pass in fast case CLR.L bkCol(A6) ;zero out back color long MOVE MINRECT+TOP(A6),D0 ;GET MINRECT.TOP MOVE MINRECT+BOTTOM(A6),D1 ;GET MINRECT.BOTTOM SUB TEXTRECT+TOP(A6),D0 ;was top clipped ? BNE.S VCLIP ;yes, handle clip CMP TEXTRECT+BOTTOM(A6),D1 ;was bottom clipped ? BEQ.S VCLIPOK ;no, continue VCLIP CLR.B HEIGHTFLAG(A6) ;can't use height table MOVE.B D0,TOPHT(A6) ;use adjusted top SUB MINRECT+TOP(A6),D1 ;calc clipped height MOVE.B D1,TOPHT+1(A6) ;replace TOPHT VCLIPOK MOVE TEXTRECT+TOP(A6),D0 ;GET DST TOP SUB DSTPIX+BOUNDS+TOP(A6),D0 ;CONVERT TO GLOBAL COORDINATES MOVE DSTPIX+ROWBYTES(A6),D1 ;GET ROWBYTES AND #nuRBMask,D1 ;CLEAR OFF FLAG BITS MULS D1,D0 ;MULT BY ROWBYTES ADD.L DSTPIX+BASEADDR(A6),D0 ;ADD START OF DST BITMAP MOVE.L D0,BUFSTART(A6) ;SET UP BUFSTART FOR LATER MOVE D1,BUFROW(A6) ;SET UP BUFROW FOR LATER MOVE DSTPIX+BOUNDS+LEFT(A6),BUFLEFT(A6) ;REMEMBER BUFLEFT TST.B CRSRFLAG(A6) ;IS DST TO A SCREEN? <BAL 19Sep88> BEQ GETPTRS ;=>NO PEA MINRECT(A6) ;PUSH SHIELD RECT MOVE.L REALBOUNDS(A6),-(SP) ;PUSH DELTA FOR GLOBAL _SHIELDCURSOR ;HIDE CURSOR IF IT INTERSECTS ;----------------------------------------------------------- ; ; Clean pointers and then ; Switch to 32 bit addressing mode ; MOVE.L SRCADDR(A6),d0 ;get 24 bit base addr _Translate24To32 ;mask off high byte BAL/MCF 03Dec88 MOVE.L d0,SRCADDR(A6) ;SAVE FOR LATER move.l a2,d0 ;get FontPtr _Translate24To32 ;mask off high byte BAL/MCF 03Dec88 move.l d0,a2 ;clean font ptr @setMMUMode MOVE stackOffset(A6),D1 MOVE.L TEXTADDR(A6,D1),D0 ;GET TEXTPTR _Translate24To32 ;mask off high byte BAL/MCF 03Dec88 MOVE.L d0,TEXTADDR(A6,D1) ;SAVE FOR LATER moveq #true32b,d0 ;switch to 32 bit addressing move.l a2,-(sp) ;save FontPtr _rSwapMMUMode ;get previous mode in d0.b (can trash a0/a1/a2, d0/d1/d2) move.l (sp)+,a2 ;restore FontPtr move.b d0,MMUsave(a6) ;save previous state for later MOVE.B #1, run32bit(A6) ; Set flag to state running 32 bit clean BRA GETPTRS NOTSCREEN ; ; Slow case: Setup for an off-screen buffer. ; ; Calc bufLeft: (LONG-align to avoid shift) ; NOTFAST SF FASTFLAG(A6) ;NOT GOING DIRECTLY TO SCREEN SF CRSRFLAG(A6) ;CRSR HAS NOT BEEN SHIELDED <BAL 19Sep88> MOVE TEXTRECT+LEFT(A6),D0 ;GET TEXTRECT LEFT SUB DSTPIX+BOUNDS+LEFT(A6),D0 ;CONVERT TO GLOBAL AND #$FFE0,D0 ;TRUNC TO LONG BOUND ADD DSTPIX+BOUNDS+LEFT(A6),D0 ;RETURN TO LOCAL COORDS MOVE D0,BUFLEFT(A6) ;REMEMBER FOR LATER ; ; Calculate buffer size ; MOVEQ #0,D1 ;clear high word MOVE TEXTRECT+RIGHT(A6),D1 ;BUFRIGHT := TEXTRECT RIGHT SUB D0,D1 ;WIDTH:=BUFRIGHT-BUFLEFT MOVE D1,D2 ;mask depth as well MOVE bitShift(A6),D4 ;convenient constant LSL.L D4,D1 ;scale up by pixel size + 1 for mask LSR.L #5,D1 ;CONVERT DOTS TO LONGS ADD #2,D1 ;ROUND UP PLUS EXTRA LONG MOVE HEIGHT(A6),D3 ;GET HEIGHT MULU D1,D3 ;BUFSIZE:=HEIGHT*BUFROW LONGS ; if the intermediate result is too big, stop before going any further CMP.L #$1C00,D3 ;is it bigger than 28K? (unit is longs) BGT.S DoSubDivide ;if so, draw fewer characters at a time MOVE D3,BUFSIZE(A6) ;SAVE FOR LATER ADD D1,D3 ;add for stretch srcBuf LSL #2,D1 ;QUAD BUFROW FOR BYTES MOVE D1,BUFROW(A6) ;SAVE FOR LATER MOVE TEXTR2+RIGHT(A6),D5 SUB TEXTR2+LEFT(A6),D5 LSR #5,D5 ;convert to longs ADDQ #2,D5 ;account for slop MOVE D5,D0 ADD D0,D3 ;in case clip is nonrectangular ADD D0,D3 ;in case vis is nonrectangular LSL D4,D0 ;scale up by source depth ADD D0,D3 ;add stretch destination buffer size MULU dstPix+pixelSize(A6),D5 ;size of composite mask ADD D5,D3 ;include it ;if srcDepth not equal to dstDepth, add stretch dest. buf. again ADD D5,D3 ;include space for scale buffer ;set up maskSize, maskRow TST.B maskFont(A6) ;do we need a mask to pass to stretchBits? BEQ.S @noMask LSR #5,D2 ;CONVERT DOTS TO LONGS ADD #2,D2 ;ROUND UP PLUS EXTRA LONG ADD D2,D1 ;total number of longs MOVE HEIGHT(A6),D0 ;GET HEIGHT MULU D2,D0 ;BUFSIZE:=HEIGHT*BUFROW LONGS MOVE D0,maskSize(A6) ;SAVE FOR LATER LSL #2,D2 ;QUAD BUFROW FOR BYTES MOVE D2,mBufRow(A6) ;SAVE FOR LATER ADD D0,D3 ;add for stack check calculation ADD D2,D3 ;add for stretch srcMaskBuf MOVE TEXTR2+RIGHT(A6),D0 SUB TEXTR2+LEFT(A6),D0 LSR #5,D0 ;convert from dots to longs ADD D0,D3 ;add stretch mask destination buffer size @noMask ; ; Calculate total stack requirements for off-screen buffers. ; TST.B 11(A4) ;ARE WE SHADOWING ? BEQ.S @1 ;NO, CONTINUE ADD bufSize(A6),D3 ;YES, CALC 2*BUFSIZE ADD D1,D3 ;add in 4 * source rows for shadow @1 LSL.L #2,D3 ;CALC TOTAL STACK BYTES NEEDED ; how much slop? size of scale table (256 bytes maximum) ; size of stretch stack frame (750 bytes for parameters, local stack frame, saved regs), ; and about 1K for interrupts. ADD.L #2048,D3 ;ADD 2 KBYTE SLOP ; ; If stack is too small to allocate buffer(s), then draw half as many characters at a time. ; _StackAvail ;Get StackAvail IN D0 CMP.L D0,D3 ;IS stackNeeded > stackAvail ? BLE.S StackAlmost ;NO, CONTINUE DoSubDivide MOVE stackOffset(A6),D6 ;2 if stack was not aligned, otherwise, 0 MOVE count(A6,D6),D0 ;how many characters to draw MOVE D0,D7 ;figure half LSR #1,D7 BNE.S @subDivide ;if more than 1 left, can subdivide problem SUBQ #1,charsRemain(A6) ;pretend the one character was drawn BRA GoHome ;if only 1 character, punt @subDivide ; *** look for space character? Could adjust D7 to coincide with a space if any, making ; *** drawing look better (for italics, kerns) TST charsRemain(A6) ;if zero, this is the first time through BNE.S @notFirst MOVE.B HiliteMode,saveHilite(A6) ;save original in case stretch is called multiple times MOVE D0,charsRemain(A6) ;initialize partial count drawing location @notFirst MOVE.L PENLOC(A6),PNLOC(A3) ;RESTORE PNLOC TO ORIGINAL TST PORTBITS+ROWBYTES(A3) ; is it a new port? BPL.S @useOld ; no, no fraction to restore MOVE PenLocHFrac(A6),pnLocHFrac(A3) ;restore fraction if applicable @useOld MOVE D7,count(A6,D6) ;reset count to draw doIterate MOVE.L grafGlobals(A5),A4 ;set up grafGlobals pointer for getting real width MOVE.L numer2(A6),numer(A6,D6) ;restore numerator MOVE.L denom2(A6),denom(A6,D6) ;restore denominator MOVE.B saveHilite(A6),HiliteMode ;restore hilite bit ; ; < ABOVE COPIED VERBATIM FROM THE ROM > ;--------------------------------------------------------------------------------------- ; JMP Iterate ;draw first half of string JMPRom RomIterate ; ;--------------------------------------------------------------------------------------- ; < BELOW COPIED VERBATIM FROM THE ROM > ; ; draw the second half of the string, but drawing no more characters than could be successfully ; drawn by the first half. secondHalf MOVEQ #0,D7 ;zero high word MOVE stackOffset(A6),D6 ;2 if stack was not aligned, otherwise, 0 MOVE count(A6,D6),D7 ;number of characters drawn last MOVE charsRemain(A6),D0 ;how many characters remain? CMP D0,D7 ;donÕt try to draw more than worked last BLE.S @ok MOVE D0,count(A6,D6) ;draw what remains for the second half @ok ADD.L D7,textAddr(A6,D6) ;bump source address by half already drawn BRA.S doIterate ; ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ; ; Allocate and clear an off-screen buffer ; ; If the source font is only 1 bit deep, clear the screen to white. Also, if the transfer ; mode is xor, bic, or, clear the screen to white. If an arithmetic mode or copy and ; source is multibits deep, assume depth of source font is equal to destination depth. ; Also assume that background of buffer must be colored the same as the port background color. StackAlmost MOVEQ #0,D0 ;get a long of white TST bitShift(A6) ;is source 1 bit? BEQ.S @whiteBackCol CMP #srcXor,locMode(A6) ;srcXor? BEQ.S @whiteBackCol ;even if font contains colors, leave background white TST.B colorSource(A6) ;font contains color? BNE.S @useBackCol ;if so, must color the buffer CMP #$32,locMode(A6) ;hilite? BEQ.S @whiteBackCol ;leave background white if so BTST #5,locMode+1(A6) ;arithmetic? BEQ.S @whiteBackCol ;if copy, or leave background white @useBackCol SUBQ #6,SP ;allocate VAR RGBColor MOVE.L SP,-(SP) ;point to VAR RGBColor _GetBackColor ;ask for the background color CLR.L -(SP) ;make room for function result PEA 4(SP) ;point to VAR RGBColor _Color2Index MOVEQ #0,D0 MOVEQ #32,D1 ; MOVE bitDepth(A6),D2 ;get destination bits per pixel @nxtPixel ASL.L D2,D0 OR.l (SP),D0 ;or in result of color2Index <BAL 04July88> SUB D2,D1 BGT.S @nxtPixel ADD #10,SP ;strip RGBColor and long result @whiteBackCol MOVE.L D0,bkCol(A6) ;save for comparing with background, later STACKOK MOVE bufSize(A6),D3 SUBQ #1,D3 ;INIT DBRA LOOP COUNT MOVE.L D0,-(SP) ;PAD BUFFER WITH AN EXTRA ZERO MOVE.L SP,BUFEND(A6) ;REMEMBER WHERE BUFFER ENDS CLRLOOP MOVE.L D0,-(SP) DBRA D3,CLRLOOP ;ALLOCATE AND CLEAR BUFFER MOVE.L SP,BUFSTART(A6) ;REMEMBER START OF BUFFER MOVE.L D0,-(SP) ;PAD BUFFER WITH AN EXTRA ZERO TST.B maskFont(A6) BEQ.S @noMask MOVE maskSize(A6),D3 SUBQ #1,D3 CLR.L -(SP) @clrMask CLR.L -(SP) DBRA D3,@clrMask MOVE.L SP,maskStart(A6) @noMask CLR.L -(SP) ; ; Get pointers to location table, width table, and height table in font ; GETPTRS LEA 26(A2),A0 ;GET START OF FONT BITMAP MOVEQ #0,D0 ;zero high word MOVE FBBDY(A2),D0 ;GET HEIGHT OF FONT BITMAP MULU.L SRCROW(A6),D0 ;CALC TOTAL SIZE OF STRIKE ADD.L D0,A0 ;A1 := START OF LOC TABLE MOVE.L A0,LOCTAB(A6) ;SAVE FOR LATER ; *** could test bitShift first and only check fNDescent if bitDepth > 1 É MOVE.W fNDescent(A2),D0 ;possibly the high word of owTLoc SWAP D0 ;put it in the high word BPL.S @notNegative MOVEQ #0,D0 ;old fonts have negative of ascent here @notNegative MOVE LENGTH(A2),D0 ;HOW MANY WORDS IN STRIKE BODY LEA 16(A2,D0.L*2),A1 ;GET START OF WIDTH TABLE MOVE.L A1,WIDTAB(A6) ;SAVE FOR LATER MOVE MAXCHAR(A2),D0 ;GET MAXCHAR MOVE MINCHAR(A2),D1 ;GET MINCHAR MOVE D1,MINCH(A6) ;STASH MINCHAR FOR LATER SUB D1,D0 ;CALC MAXCHAR-MINCHAR MOVE D0,MAXMIN(A6) ;SAVE FOR LATER ADD #3,D0 ;CALC MAXMIN+3 BTST #1,1(A2) ;DOES FONT HAVE WIDTH TABLE ? BEQ.S NOWID ;NO, CONTINUE ADD D0,D0 ;SKIP WIDTH TABLE NOWID LEA 0(A1,D0*2),A0 ;POINT TO HEIGHT TABLE MOVE.L A0,HEIGHTAB(A6) ;SAVE FOR LATER ; ; Set up space width ; MOVE.L widthTabHandle,A0 ;point to width table MOVE.L (A0),A0 MOVE.L 128(A0),SPWIDTH(A6) ;get width of the space char ; ; Setup misc stuff in registers for speed ; MOVE BUFLEFT(A6),D1 ;GET BUFLEFT MOVE PENLOC+H(A6),D0 ;GET PEN LOCATION ADD kernAdjust(A6),D0 ;ADJUST FOR KERNING SUB D1,D0 ;MAKE CHARLOC RELATIVE TO BUFLEFT MOVE.W D0,CHARLOC(A6) ;INIT INT PART OF CHARLOC TST PORTBITS+ROWBYTES(A3) ;is it a new port? BPL.S @useOld ;no, set fraction to 1/2 MOVE penLocHFrac(A6),CHARLOC+2(A6) ;set up fractional part <PB362> BAL BRA.S @goOn @useOld MOVE #$8000,CHARLOC+2(A6) @goOn SUB D1,MINRECT+LEFT(A6) ;MAKE MINRECT.LEFT AND SUB D1,MINRECT+RIGHT(A6) ;MINRECT.RIGHT BUFFER RELATIVE MOVE stackOffset(A6),D0 MOVE.L TEXTADDR(A6,D0),A1 ;GET TEXTPTR BRA.S NEXTCH ;GO TO LOOP START ;--------------------------------------------------- ; ; Here's the main character drawing loop: ; MISSING MOVE MAXMIN(A6),D0 ;NO, USE MISSING SYMBOL ADD #1,D0 ;WHICH IS ONE PAST MAXCHAR MOVE.L WIDTAB(A6),A0 ;POINT TO WIDTH TABLE MOVE 0(A0,D0*2),D3 ;GET OFFSET AND WIDTH BYTES CMP #-1,D3 ;missing? BNE NOTMISS ;IS MISSING CHAR MISSING ? BRA.S NEXTCH ;YES, SKIP THIS CHAR SPACECH MOVE.L SPWIDTH(A6),D1 ;GET SPACE WIDTH ADD.L D1,CHARLOC(A6) ;BUMP CHARLOC SKIPCH SUB #1,charsRemain(A6) ;decrement total character count SUB #1,countCopy(A6) ;decrement partial draw character count BLE STRDONE ;QUIT IF CHARCOUNT <= 0 NEXTCH CLR D0 ;get ready for byte MOVE.B (A1)+,D0 ;GET NEXT CHAR CMP.B #32,D0 ;IS IT A SPACE ? BEQ.S SPACECH ;YES, HANDLE IT MOVE.L WidthTabHandle,A0 ;POINT TO WIDTH TABLE MOVE.L (A0),A0 MOVE D0,D4 ;COPY CHARACTER MOVE.L 0(A0,D4*4),D4 ;GET FIXED POINT WIDTH ADD.L characterExtra(A6),D4 ;add in character extra, if any ; ; < ABOVE COPIED VERBATIM FROM THE ROM > ; ;--------------------------------------------------------------------------------------- ; ; BLE.S SKIPCH ;SKIP CharBlt if Advance width <= 0 <1.6-11april89-CEL> ; ;--------------------------------------------------------------------------------------- ; ; < BELOW COPIED VERBATIM FROM THE ROM UNTIL FURTHER NOTICE > ; SUB MINCH(A6),D0 ;SUBTRACT SAVED MINCHAR CMP MAXMIN(A6),D0 ;IS CH BETWEEN MINCHAR AND MAXCHAR ? BHI MISSING ;Missing so skip it OKCHAR MOVE.L WIDTAB(A6),A0 ;POINT TO WIDTH TABLE MOVE 0(A0,D0*2),D3 ;GET OFFSET AND WIDTH BYTES CMP #-1,D3 ;missing? BEQ MISSING ;OFFSET NEG = MISSING CHAR NOTMISS LSR #8,D3 ;GET OFFSET BYTE ADD.W CHARLOC(A6),D3 ;DSTLEFT := CHARLOC.INT + OFFSET ADD.L D4,CHARLOC(A6) ;ADD FIXED POINT WIDTH TO CHARLOC MOVE.L LOCTAB(A6),A0 ;POINT TO LOCATION TABLE MOVEQ #0,D1 ; *** clear top word MOVE 0(A0,D0*2),D1 ;GET SRCLEFT MOVE 2(A0,D0*2),D2 ;GET SRCRIGHT SUB D1,D2 ;CALC WIDTH OF BITS BLE SKIPCH ;SKIP CHARBLT IF WIDTH <= 0 MOVE D2,charWidth(A6) ;save for later ADD D3,D2 ;DSTRIGHT := DSTLEFT + WIDTH TST.B HEIGHTFLAG(A6) ;does font have height table ? BEQ.S NOHEIGHT ;no, continue MOVE.L HEIGHTAB(A6),A0 ;get height table MOVE 0(A0,D0*2),TOPHT(A6) ;get this char's top and height NOHEIGHT ;----------------------------------------------------------------------- ; ; Horizontal clip only if FastFlag: (else stretch,outline,shadow,italic die) ; ; skip if hidden on left, string done if hidden on right. ; ; at this point: D1: srcLeft ; D2: dstRight ; D3: dstLeft ; TST.B FastFlag(A6) ;ARE WE GOING FAST ? BEQ.S HORIZOK ;NO, DON'T CLIP CMP MINRECT+LEFT(A6),D3 ;IS DSTLEFT < MINRECT.LEFT ? BGE.S LEFTOK ;NO, CONTINUE CMP MINRECT+LEFT(A6),D2 ;IS DSTRIGHT <= MINRECT.LEFT ? BLE SKIPCH ;YES, SKIP THIS CHAR TRIMLFT MOVE MINRECT+LEFT(A6),D0 ;NO, GET MINRECT.LEFT SUB D3,D0 ;DISCARD:=MINRECT.LEFT-DSTLEFT ADD D0,D1 ;SRCLEFT:=SRCLEFT+DISCARD ADD D0,D3 ;DSTLEFT:=DSTLEFT+DISCARD CMP MINRECT+RIGHT(A6),D2 ;IS DSTRIGHT > MINRECT.RIGHT ? BLE.S fixCharWidth ;NO, CONTINUE BRA.S TRIMRT ;YES, TRIM RIGHT LEFTOK CMP MINRECT+RIGHT(A6),D2 ;IS DSTRIGHT <= MINRECT.RIGHT ? BLE.S HORIZOK ;YES, CONTINUE CMP MINRECT+RIGHT(A6),D3 ;IS DSTLEFT >= MINRECT.RIGHT ? BGE STRDONE ;YES, IGNORE REST OF STRING TRIMRT MOVE MINRECT+RIGHT(A6),D2 ;NO, TRIM DSTRIGHT fixCharWidth MOVE D2,D0 ;char width has been trimmed left, right, or both SUB D3,D0 ;so recalculate char width to draw MOVE D0,charWidth(A6) ;and save this tidbit of info. for later HORIZOK ;-------------------------------------------------------------- ; ; Inputs to local block CharBlt: ; ; srcAddr(A6) ; srcRow(A6) ; bufStart(A6) = dstAddr ; bufRow(A6) = dstRow ; D1 = srcLeft ; D2 = dstRight relative to buffer, > dstLeft ; D3 = dstLeft relative to buffer ; TOPHT(A6) top and height ; ; CLOBBERS: D0,D1,D2,D3,D4,D5,D6,D7,A0,A2,A3,A4,A5 ; PRESERVES: A1,A6,A7 ; ; ; Setup shift count in D5 (relative shift between src and dst) ; MOVEM D1-D3,maskBlts(A6) MOVE.B textCopyMode(A6),orNotOK(A6) ;if copy or if mask mode, do bit moves instead of OR ST notMaskPass(A6) ;use the bitDepth this pass MOVE bitShift(A6),D0 ;get font scale factor LSL D0,D1 LSL D0,D2 LSL D0,D3 ;scale up all three MOVE BUFROW(A6),A3 MOVEM.L SRCROW(A6),A2/A4-A5 ;srcRow, srcAddr (src bitmap), bufStart (dst bitmap) maskBlt ; ; Get char height into D7 ; MOVE.W #255,D7 ;GET -1 BYTE, CLEAR HI BYTE ADD.B TOPHT+1(A6),D7 ;CALC HEIGHT-1 FOR DBRA LOOP BMI SKIPCH ;OOPS HEIGHT WAS ZERO ! ; ; Adjust srcPtr and dstPtr for charTop ; MOVEQ #0,D0 ;get ready for byte MOVE.B TOPHT(A6),D0 ;get char top MOVE.L A2,D4 ;get srcRow in D-reg MULU.L D0,D4 ;calc charTop * srcRow ADD.L D4,A4 ;add to srcPtr MOVE A3,D4 ;get dstRow in D-reg MULU D0,D4 ;calc charTop * dstRow ADD.L D4,A5 ;add to dstPtr ; ; figure alignment offset in source so that dest is long aligned ; MOVEQ #$1F,D0 ;get a 5 bit mask, needed later MOVE.L D1,D5 ; ; Setup dstPtr in A5, address of leftmost dst word ; MOVE D3,D6 ;GET A COPY OF DSTLEFT ASR #5,D6 ;CONVERT FROM DOTS TO LONGS MOVE D6,D4 ;copy dest left in longs for later LSL #2,D6 ;quad for bytes ADD D6,A5 ;move dest pointer to top of character ; ; figure number of longs to move ; MOVE D2,D6 ;copy dstRight ASR #5,D6 SUB D4,D6 ;if same, it fits into a single long BEQ fitsInLong ;most 1 bit deep characters fit into a long ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ; ; Character spans long in destination, so it must be moved in pieces ; Registers used at this point: ; D0 = 5 bit mask D1 = src offset D2 = dstRight D3 = dstLeft ; D4 = -- D5 = src offset D6 = longs to move D7 = character height ; A0 = -- A1 = character ptr A2 = source row sz A3 = dest row size ; A4 = source row ptr A5 = dest row ptr A6 = locals A7 = stack NEG D3 ;negate for bit counting AND D0,D3 ;get bottom 5 bits of dstLeft BNE.S @skip32 MOVEQ #32,D3 ;if 0, make it 32 @skip32 MOVE D6,D4 ;total longs (not including left part) LSL #2,D4 ;make it a byte count SUB D4,A2 ;adjust source bump count SUB D4,A3 ;adjust dest. bump count SUBQ #1,D6 ;number of whole longs, less left & right parts ADD D3,D1 ;the right hand start MOVEQ #32,D4 SUB.L D4,D1 ; ; if not 1 bit font & mode is not srcOr, do moves instead ; TST.B orNotOK(A6) BNE.S DoMove ; ; OR words to the screen; set up right mask ; AND D0,D2 ;get bottom 5 bits of dstRight MOVEQ #1,D4 ROR.L #1,D4 ;set the high bit ASR.L D2,D4 ;$80000000 to $FFFFFFFF ASL.L #1,D4 ;$00000000 to $FFFFFFFE MOVE D6,D2 ;initialize whole long count MOVE D6,A0 ;save long count for later MOVE.L D5,D6 ADD D3,D6 MOVEQ #32,D0 SUB.L D0,D6 ; ; Slow loop only taken in wierd cases where dst wider than a long. ; Or the more common case that the font is more than 1 bit deep ; MAIN1 BFEXTU (A4){D5:D3},D0 ;GET SRC FROM BITMAP BRA.S @checkLong ;see if there is some number of longs to do @main2 BFEXTU (A4){D6:0},D0 ;GET SRC FROM BITMAP @checkLong OR.L D0,(A5)+ ;OR SRC INTO DST ADDQ #4,A4 ;BUMP SRCPTR RIGHT DBRA D2,@main2 ;LOOP TILL LAST long BFEXTU (A4){D1:0},D0 ;GET SRC FROM BITMAP AND.L D4,D0 OR.L D0,(A5) ;OR SRC INTO DST ADD.L A2,A4 ;BUMP TO NEXT ROW ADD.L A3,A5 ;BUMP DST TO NEXT ROW WIDE1 MOVE A0,D2 ;GET long count DBRA D7,MAIN1 ;LOOP ALL ROWS BRA decCount ;skip mask blt part ; ; Handle case of drawing into the offscreen bit map in color separately ; DoMove MOVE D2,D4 ;copy dstRight AND D0,D4 ;get bottom 5 bits of dstRight MOVE D6,longCount(A6) ;save # of whole longs MOVE D6,D2 ;set up first loop count MOVE.L bkCol(A6),D6 ;get a long full of the background color BFEXTU D6{0:D3},D0 ;background color & left mask MOVE.L D0,leftBack(A6) ;save for quick compare BFEXTU D6{0:D4},D0 ;background color & right mask MOVE.L D0,rightBack(A6) ;right mask color compare MOVE bitDepth(A6),D0 ;bits in a single pixel BFEXTU D6{0:D0},D0 ;get a single pixel of the background color MOVE D0,bkCol1(A6) ;a single background color for quick compare MOVEQ #32,D6 ; SUB D3,D6 ; MOVE.L D5,A0 ADD D3,A0 SUB #32,A0 ;set up center move offset ; ; Slow loop taken in cases where dst wider than a long (common for fonts more than 1 bit ; deep.) Need to move only the pixels that are not colored same as background. ; MoveLeft BFEXTU (A5){D6:D3},D0 ;get current background of destination CMP.L leftBack(A6),D0 ;compare it against the background color AND leftMask BNE.S leftOnePix ;if different, must move 1 pixel at a time ; ; Here the destination has only background color pixels, so the whole thing can be moved ; BFEXTU (A4){D5:D3},D0 ;GET SRC FROM BITMAP BFINS D0,(A5){D6:D3} checkLong EXG A0,D5 ;set up center move offset midChkLong ADDQ #4,A5 ;bump destination BRA.S checkMLong ;see if there is some number of longs to do MoveMid MOVE.L bkCol(A6),D0 CMP.L (A5),D0 ;same as background color? BNE.S midOnePix ;if different, move 1 at a time BFEXTU (A4){D5:0},D0 ;GET SRC FROM BITMAP MOVE.L D0,(A5)+ ;OR SRC INTO DST checkMLong ADDQ #4,A4 ;BUMP SRCPTR RIGHT DBRA D2,MoveMid ;LOOP TILL LAST WORD MoveRight EXG A0,D5 ;set up left move offset TST D4 BEQ.S nextRow BFEXTU (A5){0:D4},D0 ;look at the destination CMP.L rightBack(A6),D0 ;compare it against the background AND rightMask BNE.S rightOnePix ;if different, must move 1 pixel at a time ; ; Here the destination has only background color pixels, so the whole thing can be moved ; BFEXTU (A4){D1:D4},D0 ;get the right part of the character BFINS D0,(A5){0:D4} ;move it to the destination nextRow ADD.L A2,A4 ;BUMP TO NEXT ROW ADD.L A3,A5 ;BUMP DST TO NEXT ROW offMove MOVE longCount(A6),D2 ;GET long count DBRA D7,MoveLeft ;LOOP ALL ROWS BRA checkMask ; ; need to move only the pixels that are not colored same as background ; leftOnePix MOVEM.L D1/D3-D6,-(SP) ;save temporary registers MOVEM bkCol1(A6),D0/D1 ;D0 = back color D1 = pixel width @nxtPixel BFEXTU (A4){D5:D1},D4 ;pick up a source pixel CMP D4,D0 ;is it the same as the back color? BEQ.S @skipInsert ;if so, donÕt add to destination BFINS D4,(A5){D6:D1} ;if not, copy it to the destination @skipInsert ADD.L D1,D5 ;move to next source pixel ADD D1,D6 ;move to next destination pixel SUB D1,D3 ;decrement count BGT.S @nxtPixel ;do until weÕre done MOVEM.L (SP)+,D1/D3-D6 ;restore bkCol, left & right widths, src & dest offsets BRA.S checkLong midOnePix MOVEQ #32,D0 ;full long size in bits MOVEM.L D1/D3-D6,-(SP) ;save temporary registers MOVEQ #0,D6 MOVEM bkCol1(A6),D1/D3 ;D1 = back color D3 = pixel width @nxtPixel BFEXTU (A4){D5:D3},D4 ;pick up a source pixel CMP D4,D1 ;is it the same as the back color? BEQ.S @skipInsert ;if so, donÕt add to destination BFINS D4,(A5){D6:D3} ;if not, copy it to the destination @skipInsert ADD.L D3,D5 ;move to next source pixel ADD D3,D6 ;move to next destination pixel SUB D3,D0 ;decrement count BGT.S @nxtPixel ;do until weÕre done MOVEM.L (SP)+,D1/D3-D6 ;restore bkCol, left & right widths, src & dest offsets BRA midChkLong rightOnePix MOVEM.L D1/D3-D6,-(SP) ;save temporary registers MOVEQ #0,D6 MOVEM bkCol1(A6),D3/D5 ;D3 = back color D5 = pixel width @nxtPixel BFEXTU (A4){D1:D5},D0 ;pick up a source pixel CMP D0,D3 ;is it the same as the back color? BEQ.S @skipInsert ;if so, donÕt add to destination BFINS D0,(A5){D6:D5} ;if not, copy it to the destination @skipInsert ADD.L D5,D1 ;move to next source pixel ADD D5,D6 ;move to next destination pixel SUB D5,D4 ;decrement count BGT.S @nxtPixel ;do until weÕre done MOVEM.L (SP)+,D1/D3-D6 ;restore bkCol, left & right widths, src & dest offsets BRA nextRow ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ; Here, the character does not cross a long boundary, so faster loops are in order. ; fitsInLong MOVE charWidth(A6),D3 ;set up character width in bits TST.B notMaskPass(A6) ;if dest is 1 bit, or this is the mask, BEQ.S @skipMul ;then skip the mul scale MOVE bitShift(A6),D4 ;set up bit depth LSL D4,D3 @skipMul TST.B orNotOK(A6) ;can we OR? BNE.S offLong ;if not, bit extract/insert instead ; ; Optimize if dst fits in one long. (most 1 bit deep characters do) ; LONG1 ; ; OR words to the screen; set up right mask MOVE D2,D1 ;copy dest right NEG D1 ;figure not bits from left but bits from right AND D0,D1 ;but only from 0 to 31 ADD D1,D3 ;add right mask to extract width AND D0,D2 ;get bottom 5 bits of dstRight MOVEQ #1,D4 ROR.L #1,D4 ;set the high bit ASR.L D2,D4 ;$80000000 to $FFFFFFFF ASL.L #1,D4 ;$00000000 to $FFFFFFFE ; if D5 + D3 < 32, use a faster set of code: longLoop BFEXTU (A4){D5:D3},D0 ;GET SRC DATA AND.L D4,D0 OR.L D0,(A5) ;OR RESULT INTO DST ADD.L A2,A4 ;BUMP SRCPTR TO NEXT ROW ADD.L A3,A5 ;BUMP DSTPTR TO NEXT ROW DBRA D7,longLoop ;LOOP ALL ROWS BRA.S decCount ; ; Optimize if dst fits in one long. (most normal characters do) ; offLong MOVE D2,D6 AND.L D0,D6 SUB D3,D6 BFEXTU bkCol(A6){0:D3},D2 MOVEM.L D5/D6,leftOffset(A6) ;save source left, destination left offsets MOVE D5,D0 ;source left offset less ADD D3,D0 ; width of pixels to be moved MOVE D0,longCount(A6) ; determines inner loop bounds MOVE bitDepth(A6),D4 ;get source depth BFEXTU D2{0:D4},D1 ;set up 1 pixel of background color offLongLoop ; check to see if existing destination is untouched (equal to background) so that there is no ; concern about obliterating over a part of a character that kerns into this space. BFEXTU (A5){D6:D3},D0 ;get the existing background CMP.L D2,D0 ;is it background colored? BEQ.S @okToBlast ;if so, it is safe to cut and fill ; inner loop taken if some kerning has occured so that bits can not be moved directly over bkground @nxtPixel BFEXTU (A4){D5:D4},D0 CMP D0,D1 BEQ.S @skipInsert BFINS D0,(A5){D6:D4} @skipInsert ADD D4,D6 ADD D4,D5 CMP longCount(A6),D5 ;where to stop with the mask BLT.S @nxtPixel MOVEM.L leftOffset(A6),D5/D6 ;restore source left, destination left offsets BRA.S @nextRow ; if no kerning, data can be moved without concern for what is already there. @okToBlast BFEXTU (A4){D5:D3},D0 ;GET SRC DATA BFINS D0,(A5){D6:D3} ;move to destination @nextRow ADD.L A2,A4 ;BUMP SRCPTR TO NEXT ROW ADD.L A3,A5 ;BUMP DSTPTR TO NEXT ROW DBRA D7,offLongLoop ;LOOP ALL ROWS ; ; here, set up things for mask blt ; checkMask TST bitShift(A6) ;is source depth 1? BEQ.S decCount ;if so, skip mask (just copy result to mask later) TST.B maskFont(A6) BEQ.S decCount SF orNotOK(A6) ;next time through, can OR the mask SF notMaskPass(A6) ;force bitDepth to 1 this pass MOVEM maskBlts(A6),D1-D3/A3 ;set up words MOVEM.L mSrcROW(A6),A2/A4/A5 ;set up longs BRA maskBlt ;then blt mask decCount SUB #1,charsRemain(A6) ;decrement total character count SUB #1,countCopy(A6) ;decrement partial draw character count BGT NEXTCH ;LOOP IF MORE CHARS LEFT ;------ STRDONE MOVE.L SAVEA5(A6),A5 ;RESTORE GLOBAL PTR TST.B FASTFLAG(A6) ;WERE WE GOING DIRECT TO SCREEN ? BEQ.S @1 ;NO, CONTINUE TST.B CRSRFLAG(A6) ;DRAWING TO SCREEN? BEQ.S @0 ;=>IF NOT, JUST RETURN move.b MMUsave(a6),d0 ;get previous MMU state in d0 _rSwapMMUMode ;get previous mode in d0.b (can trash a0/a1/a2, d0/d1/d2) @32bitNot _SHOWCURSOR ;YES, RESTORE CURSOR @0 BRA GOHOME ;AND QUIT @1 MOVE.L GRAFGLOBALS(A5),A0 ;POINT TO QUICKDRAW GLOBALS MOVE.L THEPORT(A0),A3 ;GET CURRENT GRAFPORT MOVE.L FONTPTR(A0),A4 ;POINT TO FMOUTPUT MOVE.L 2(A4),A2 ;GET FONT HANDLE MOVE.L (A2),A2 ;DE-REFERENCE IT MOVE CHARLOC(A6),D7 ;GET FINAL CHARLOC INT PART SUB fKernMax(A2),D7 ;UN-ADJUST FROM KERNING, but not italic ; ; Make buffer bold if necessary: ; CKBOLD CLR D2 ;GET READY FOR BYTE MOVE.B 6(A4),D2 ;GET NUMBER OF OVERSTRIKES BRA.S NXTBOLD ;BOLD BUFFER IF ANY BOLDIT MOVE.L BUFSTART(A6),A0 ;POINT TO START OF BUFFER MOVE BUFSIZE(A6),D1 ;HOW MANY LONGS IN BUF SUB D0,D0 ;CLEAR X-BIT BOLDLP MOVE.L (A0),D0 ;GET ONE LONG ROXR.L #1,D0 ;ROTATE RIGHT WITH EXTEND OR.L D0,(A0)+ ;OR BACK INTO BUFFER DBRA D1,BOLDLP ;LOOP ENTIRE BUFFER NXTBOLD DBRA D2,BOLDIT ;LOOP FOR EACH OVERSTRIKE ; ; Slant the buffer if necessary: ; Work from bottom of buffer up, shifting each row right. ; Work right to left to avoid clobbering src. ; MOVEQ #0,D2 ;GET READY FOR BYTE MOVE.B 7(A4),D2 ;DO WE NEED ITALIC ? BEQ.S CHECKUL ;NO, CONTINUE MOVE.L BUFEND(A6),A1 ;DSTPTR:=END OF BUFFER MOVE BUFROW(A6),D3 ;GET BUFFER ROWBYTES SUB D3,A1 ;BACK UP DSTPTR TO END OF 2ND ROW LSR #2,D3 ;LONGCNT:=ROWBYTES DIV 4 SUB #1,D3 ;LONGCOUNT-1 FOR DBRA LOOP MOVE HEIGHT(A6),D6 ;INIT ROW COUNTER MOVEQ #0,D4 ;INIT OFFSET BRA.S DOSLANT ;GO TO LOOP START NXTROW ADD.L D2,D4 ;OFFSET:=OFFSET+ITALIC MOVE.L D4,D5 ;COPY OFFSET LSR.L #4,D5 ;DELTA := OFFSET SCALED BY 16 NEG.L D5 ;make negative for BFEXTU MOVE.L A1,A0 ;SRCPTR:=DSTPTR SUB #4,A0 ;POINT TO LAST LONG MOVE D3,D1 ;INIT LOOP TO LONGCNT NXTLONG BFEXTU (A0){D5:0},D0 ;GET A SHIFTED LONG OF SRC SUB #4,A0 ;BUMP SRCPTR LEFT ONE LONG MOVE.L D0,-(A1) ;STORE IN DST AND BUMP DSTPTR DBRA D1,NXTLONG ;LOOP ALL LONG THIS ROW DOSLANT DBRA D6,NXTROW ;LOOP FOR ALL ROWS IN BUFFER ; ; Underline characters in buffer if necessary. ; ; Use characters in buffer to hide parts of the underline. ; CHECKUL TST.B 10(A4) ;IS ULTHICK ZERO ? BEQ.S NOTUL ;YES, CONTINUE MOVE.L BUFSTART(A6),A0 ;POINT TO BUFFER START MOVE BUFROW(A6),D1 ;GET BYTES PER ROW OF BUFFER MOVE locASCENT(A2),D0 ;GET ASCENT MOVE locDESCENT(A2),D2 ;GET DESCENT MULU D1,D0 ADD.L D0,A0 ;POINT TO BASELINE ROW MOVE.L A0,A1 MOVE.L A0,A2 ADD D1,A1 ;POINT TO BASELINE+1 CMP #2,D2 ;IS DESCENT AT LEAST 2 ? BLT.S NOTUL ;NO, SKIP UNDERLINE BEQ.S ONLY2 ;ONLY USE 2 IF DESCENT=2 ADD D1,A2 ADD D1,A2 ;POINT TO BASELINE+2 ONLY2 SUB D1,SP ;ALLOCATE TEMP SCANBUF MOVE.L A3,-(SP) ;SAVE GRAFPORT LEA 4(SP),A3 ;POINT TO START OF TEMP LSR #2,D1 ;CONVERT BYTES TO LONGS SUB #1,D1 ;INIT DBRA LOOP COUNT MOVE D1,D2 ;COPY LOOP COUNT SUB D0,D0 ;CLEAR X-BIT UL1 MOVE.L (A0)+,D0 ;GET FROM BASELINE OR.L (A1)+,D0 ;OR WITH BASELINE+1 OR.L (A2)+,D0 ;OR WITH BASELINE+2 MOVE.L D0,(A3) ;PUT RESULT TO TEMP ROXR.L #1,D0 ;SHIFT WITH CARRY OR.L D0,(A3)+ ;OR INTO TEMP DBRA D1,UL1 ;LOOP ALL LONGS IN ROW MOVE.L A1,A0 ;COPY END PTR SUB D0,D0 ;CLEAR X-BIT UL2 MOVE.L -(A3),D0 ;GET FROM TEMP ROXL.L #1,D0 ;SHIFT LEFT WITH CARRY OR.L (A3),D0 ;OR WITH TEMP NOT.L D0 ;INVERT OR.L D0,-(A1) ;DRAW SOME UNDERLINE DBRA D2,UL2 ;LOOP ALL LONGS IN ROW MOVE.L (SP)+,A3 ;RESTORE GRAFPORT ; ; Setup fakeRgn, a dummy rectangular region ; NOTUL MOVE #10,FAKERGN+RGNSIZE(A6) ;SIZE=10 BYTES FOR RECT RGN MOVE.L DSTPIX+BOUNDS(A6),FAKERGN+RGNBBOX(A6) MOVE.L DSTPIX+BOUNDS+4(A6),FAKERGN+RGNBBOX+4(A6) LEA FAKERGN(A6),A0 ;GET ADDR OF FAKERGN MOVE.L A0,FAKEPTR(A6) ;POINT FAKE MASTERPTR TO IT ; ; SET UP SOURCE BITMAP TO TRANSFER TO SCREEN ; ; SRCPIX := buffer ; LEA SRCBITS(A6),A0 ;POINT TO source bits MOVE.L A0,A1 MOVE.L BUFSTART(A6),(A0)+ ;SET UP BASEADDR MOVE BUFROW(A6),(A0)+ ;SET UP ROWBYTES MOVE TEXTRECT+TOP(A6),(A0)+ ;SET UP BOUNDS TOP MOVE BUFLEFT(A6),(A0)+ ;SET UP BOUNDS LEFT MOVE.L TEXTRECT+BOTRIGHT(A6),(A0)+ ;SET UP BOTTOM RIGHT LEA SRCPIX(A6),A2 _BitsToPix TST.B maskFont(A6) BEQ.S @finishSrcPix LEA maskBits(A6),A0 ;POINT TO mask bits MOVE.L maskStart(A6),(A0)+ ;SET UP BASEADDR MOVE mBufRow(A6),(A0)+ ;SET UP ROWBYTES MOVE TEXTRECT+TOP(A6),(A0)+ ;SET UP BOUNDS TOP MOVE BUFLEFT(A6),(A0)+ ;SET UP BOUNDS LEFT MOVE.L TEXTRECT+BOTRIGHT(A6),(A0)+ ;SET UP BOTTOM RIGHT @finishSrcPix TST bitShift(A6) ;is the bit depth 1? BEQ.S @skipColorJam MOVE bitDepth(A6),srcPix+pixelSize(A6) ;set up bit depth MOVE.L dstPix+pmTable(A6),srcPix+pmTable(A6) ;set up color table @skipColorJam ; ; check if any shadowing: ; CLR D3 ;GET READY FOR BYTE MOVE.B 11(A4),D3 ;GET SHADOW COUNT BEQ NOSHAD ;SKIP IF NO SHADOWING ; ; Shadowing will be used. Allocate buf2, 4 scans taller than BUF1. ; Clear out new 4 scanlines, and copy BUF1 into the rest. ; MOVE BUFROW(A6),D0 ;GET 4 * NUMBER OF LONGS PER ROW SUB #1,D0 ;INIT LOOP COUNTER CLR.L -(SP) ;ALLOW ONE LONG OF SLOP MOVE.L SP,BUF2END(A6) ;REMEMBER END OF BUF2 CLR2 CLR.L -(SP) ;ALLOCATE AND CLEAR A LONG DBRA D0,CLR2 ;CLEAR 4 SCANLINES WORTH MOVE BUFSIZE(A6),D0 ;GET NUMBER OF LONGS IN BUF1 SUB #1,D0 ;INIT DBRA COUNTER MOVE.L BUFEND(A6),A0 ;POINT TO END OF BUF1 COPYLP MOVE.L -(A0),-(SP) ;COPY FROM BUF1 TO NEW BUF2 DBRA D0,COPYLP ;COPY ALL OF BUF1 MOVE.L SP,BUF2START(A6) ;REMEMBER START OF BUF2 CLR.L -(SP) ;ALLOW ONE LONG OF SLOP ; ; Bold buf2 across to the right enough for shadow. ; AND #3,D3 ;RESTRICT SHADOW COUNT TO 1..3 MOVE D3,D2 ;INIT BOLD COUNTER ACROSS1 MOVE.L BUF2START(A6),A0 ;POINT TO START OF BUFFER2 MOVE BUFSIZE(A6),D1 ;INIT COUNT OF LONGS SUB D0,D0 ;CLEAR X-BIT ACROSS2 MOVE.L (A0),D0 ;GET A LONG FROM BUF2 ROXR.L #1,D0 ;SHIFT IT RIGHT EXTENDED OR.L D0,(A0)+ ;OR IT BACK INTO BUFFER DBRA D1,ACROSS2 ;LOOP FOR ALL LONGS DBRA D2,ACROSS1 ;BOLD RIGHT 2,3, OR 4 TIMES ; ; Bold BUF2 down enough for shadow. ; MOVE.L BUF2START(A6),A2 ;GET LIMIT POINTER DOWN1 MOVE.L BUF2END(A6),A1 ;DSTPTR:=END OF BUF2 MOVE.L A1,A0 MOVE SRCBITS+ROWBYTES(A6),D2 ;GET SRC ROWBYTES SUB D2,A0 ;SRCPTR:=END - 1 SCANLINE DOWN2 MOVE.L -(A0),D0 ;GET A LONG FROM LINE ABOVE OR.L D0,-(A1) ;OR INTO THIS LINE CMP.L A2,A0 ;IS SRCPTR <= BUF2START ? BGT.S DOWN2 ;NO, LOOP ALL LONGS DBRA D3,DOWN1 ;BOLD DOWN 2,3, OR 4 TIMES ; XOR the plain text into the shadow buffer MOVE bufSize(A6),D1 ;size of buffers in longs MOVE.L bufStart(A6),A0 ;start of plain text MOVE.L A2,A1 ;start of shadowed text ADD D2,A1 ;bump down a line in the shadowed text SUB D0,D0 ;clear x bit @xor MOVE.L (A0)+,D0 ;a line of plain text ROXR.L #1,D0 ;shift it right, extended EOR.L D0,(A1)+ ;combine with shadowed text DBRA D1,@xor ;repeat for all longs ; ; Alter SRCBITS to use BUF2 ; MOVE.L A2,SRCBITS+BASEADDR(A6) ;SRC BASEADDR:=BUF2START ADD #4,SRCBITS+BOUNDS+BOTTOM(A6) ;4 SCANS TALLER ; ; Push params and call StretchBits to transfer shadow to screen ; MOVE.L TEXTRECT(A6),SRCRECT(A6) ;DSTRECT := SRCRECT MOVE.L TEXTRECT+4(A6),SRCRECT+4(A6) ADD #4,SRCRECT+BOTTOM(A6) ;PLUS 4 SCANS TALLER MOVE.L SRCRECT(A6),DSTRECT(A6) MOVE.L SRCRECT+4(A6),DSTRECT+4(A6) LEA DSTRECT+TOP(A6),A0 ;OFFSET BY (-1,-1) SUB #1,(A0)+ ;TOP SUB #1,(A0)+ ;LEFT SUB #1,(A0)+ ;BOTTOM SUB #1,(A0)+ ;RIGHT TST.B STRETCH(A6) BEQ.S @1 PEA DSTRECT(A6) PEA FROMRECT(A6) PEA TORECT(A6) _MAPRECT ;THEN MAPPED FOR SCALING @1 PEA SRCBITS(A6) ;PUSH SRCPIX MOVE.L maskBitsPtr(A6),-(SP) ;a mask? BEQ.S @noMask MOVE.L 4(SP),(SP) ;replace mask with copy of source @noMask PEA PORTBITS(A3) ;TRANSFER TO SCREEN PEA SRCRECT(A6) ;PUSH SRCRECT = TEXTRECT PEA SRCRECT(A6) ;maskRect same as source rect PEA DSTRECT(A6) ;PUSH DSTRECT = TEXTR2 MOVE locMode(A6),-(SP) ;PUSH TEXTMODE CLR.L -(SP) ;NO PATTERN FOR NOW MOVE.L CLIPRGN(A3),-(SP) ;PUSH CLIPRGN HANDLE MOVE.L VISRGN(A3),-(SP) ;PUSH VISRGN HANDLE PEA FAKEPTR(A6) ;PUSH FAKE HANDLE CLR -(SP) ;pass multicolor flag false _StretchBits ;TRANSFER BUFFER TO SCREEN ; if old port, srcOr, draw center part in bic; if bic, in srcOr ; (This is done for compatibility with older applications like MacProject and MacDraw that ; expect to put text on an arbitrary background, relying on the center of the shadow to show.) TST portBits+rowBytes(A3) ;a new port? BMI.S GOHOME MOVE locMode(A6),D0 CMP #srcOr,D0 BEQ.S @useBic CMP #srcBic,D0 BNE.S GOHOME @useBic ; restored altered srcBits MOVE.L bufStart(A6),srcBits+baseAddr(A6) SUB #4,srcBits+bounds+bottom(A6) ; adjust params for center portion StretchBits CLR.L maskBitsPtr(A6) ;no mask EOR #2,D0 ;change or to bic, bic to or MOVE D0,locMode(A6) ;adjust textmode ; ; Push params and call StretchBits to transfer buffer to screen ; NOSHAD PEA SRCPIX(A6) ;PUSH SRCBITS MOVE.L maskBitsPtr(A6),-(SP) ;may be 0 if no mask bitmap PEA PORTBITS(A3) ;TRANSFER TO SCREEN PEA TEXTRECT(A6) ;PUSH SRCRECT = TEXTRECT PEA TEXTRECT(A6) ;maskRect same as source rect PEA TEXTR2(A6) ;PUSH DSTRECT = TEXTR2 MOVE locMode(A6),-(SP) ;PUSH TEXTMODE CLR.L -(SP) ;NO PATTERN FOR NOW MOVE.L CLIPRGN(A3),-(SP) ;PUSH CLIPRGN HANDLE MOVE.L VISRGN(A3),-(SP) ;PUSH VISRGN HANDLE PEA FAKEPTR(A6) ;PUSH FAKE HANDLE MOVE.B colorSource(A6),-(SP) ;pass multicolor flag true if source is multibit _StretchBits ;TRANSFER BUFFER TO SCREEN GOHOME MOVE.L saveStk(A6),SP ;throw away the buffers MOVE charsRemain(A6),D0 SUB countCopy(A6),D0 MOVE D0,charsRemain(A6) ;if iterative, this will be greater than zero BGT secondHalf ;so there are more characters to draw BSET #hiliteBit,HiliteMode ;reset hilite override, in case colormap was skipped TST stackOffset(A6) ;was the stack aligned? MOVEM.L (SP)+,D0-D7/A1-A4 ;RESTORE REGS UNLK A6 BEQ.S @skipAlign ADDQ #2,SP @skipAlign RTD #PARAMSIZE ;Return and pop the stack <1.6-11april89-CEL> ENDPROC ; ; < ABOVE COPIED VERBATIM FROM THE ROM > ; ;--------------------------------------------------------------------------------------- ENDIF ; <6.0.5> <CEL> ;____________________________________________________________________________ dba <3.0> ; Menu manager patch ; ; The following patch fixes the head of InitProcMenu. The wrong version was ; rolled into the ROM, so we replace the head with this. NewInitProcMenu proc export ROMIMCommon equ $16D52 ; place to rejoin ROM in InitMenus ; the following code is copied out of the Menu Manager move.l (sp)+,a0 ; get the return address move.w (sp)+,d1 ; get the MBDF resource ID move.l a0,-(sp) ; put return address back on the stack jsrROM ROMIMCommon ; call InitMenus <4.0> move.l MenuList,a0 ; get the menu list <4.0> move.l (a0),a0 ; dereference the menu list <4.0> move.w d1,mbResID(a0) ; store MBDF ID <4.0> rts ; return to InitProcMenu caller <4.0> endproc ; The following patch reinstates the WaitMouseUp in MenuSelect. It was removed as part ; of the tear-off menu stuff (in an attempt to support double-clicking in the menu bar), ; but that change was probably not a good idea, even for the tear-off menu case. NewMenuSelect proc export ROMContinueMenuSelect equ $171A2 ; place to rejoin ROM in MenuSelect ROMDoneMenuSelect equ $17296 ; place to rejoin ROM in MenuSelect ; the following code is copied out of the Menu Manager ;----------------------------------------------- ; MenuSelect stack frame definitions ;----------------------------------------------- STARTPT EQU 8 ; 1st parameter is 8, and is 4 bytes long FUNCRESULT EQU 12 ; longword function result (menu,item) ITEMRESULT EQU 14 ; item portion of funcResult MENURECT EQU -8 ; rectangle (8 bytes) MSavePort EQU MENURECT-4 ; saved grafPort msLastMenu EQU MSavePort-2 ; how many menus are on the screen DelayTicks EQU msLastMenu - 4 ; how long mouse in HItem DragTicks EQU DelayTicks - 4 ; how long mouse moving towards HMenu menuDelaySave EQU DragTicks - 1 ; store menuDelay from pRAM menuDragSave EQU menuDelaySave - 1 ; store menuDrag from pRAM lastFuncResult EQU menuDragSave - 2 ; funcResult last time thru MSLoop lastItemResult EQU lastFuncResult - 2 ; itemResult last time thru MSLoop firstMenuID EQU lastItemResult - 2 ; ID of REGULAR menu up NewHMenuOffset EQU firstMenuID - 2 ; offset into MenuList of new HMenu tRect1 EQU NewHMenuOffset - 8 ; temp rectangle SelectFrame EQU tRect1 ; number of bytes to save ;----------------------------------------------- link a6,#SelectFrame ; set up a stack frame movem.l d2-d7/a2-a4,-(sp) ; save work registers clr.w -(sp) ; push zero for this _HiliteMenu ; make sure no menu is hilited clr.l funcResult(a6) ; assume nothing will be selected clr.l MenuDisable ; clear the low-memory location MenuDisable clr.w -(sp) ; push zero for this _WaitMouseUp ; is the mouse still down? tst.b (sp)+ ; examine result beq.s DoneMenuSelect ; if not still down, donÕt bother jmpROM ROMContinueMenuSelect ; continue in ROM DoneMenuSelect jmpROM ROMDoneMenuSelect ; continue in ROM endproc ;____________________________________________________________________________ djw <3.6> ; Slot Manager pInitEntry patch ; ; Fixes a bug when calling _InsertSRTRec, inserting a ROM sRsrc with a valid ; refnum, pInitEntry is called. pInitEntry calls _sFindDevBase to calc a base ; addr to place in the DCE. The bug is _sFindDevBase is called with a bad ; sRsrc id. The fix is one line to restore the spId before calling ; _sFindDevBase. ; pInitEntry Proc Export With spBlock,srrBlock,SlotRsrcType ROMpInitEntry Equ $00006904 ; addr to continue to in ROM ROMpInitError Equ $0000691c ; addr of @Error label in ROM ROMpInitDone Equ $00006920 ; addr of @Done label in ROM ROMMapUnit Equ $00006930 ; addr of MapUnit routine in ROM MOVEM.L D1/A2,-(SP) ; save regs SUBA.W #SRSize,SP ; alloc size to read sRsrc sType field MOVEA.L SP,A2 ; save A2 = ptr to buffer ; Fill in the fields of a new SRT entry MOVE.B spSlot(A0),srrSlot(A1) ; slot number MOVE.B spId(A0),srrId(A1) ; sRsrc id MOVE.B spExtDev(A0),srrExtDev(A1) ; external device id CLR.B srrHWDev(A1) ; clear hardware device id MOVE.W spRefNum(A0),srrRefNum(A1) ; driver reference number MOVE.W spIOReserved(A0),srrIOReserved(A1) ; IO reserved field (good for nothin?) MOVE.L spsPointer(A0),srrsRsrcPtr(A1) ; set the ptr to the RAM or ROM sRsrc CLR.L srrFlags(A1) ; clear flag field MOVE.L spParamData(A0),D0 ; test state of enable/disable value BEQ.S @ReadStuff ; zero - sRsrc is marked as enabled BSET.B #srDisable,srrFlags+3(A1) ; set flag to indicate disabled ; Read the hardware sRsrc id @ReadStuff MOVE.B #sRsrcHWDevId,spId(A0) ; get hardware id (if present) _sReadByte ; no error if not present BNE.S @Continue MOVE.B spResult+3(A0),srrHWDev(A1) ; only a byte value ; Find and read the sRsrc type field <category><type><drvrSW><drveHW> @Continue MOVE.B #sRsrcType,spId(A0) ; get ptr to sRsrc type _sFindStruct BNE.S @Error ; error - sRsrc type field is required MOVE.L #SRSize,spSize(A0) ; set size to read MOVE.L A2,spResult(A0) ; set buffer ptr _sReadStruct BNE.S @Error ; error - found but can't read ? MOVE.L sCategory(A2),srrCategory(A1) ; copy <cat><type> MOVE.L sDrvrSw(A2),srrDrvrSw(A1) ; copy <drvrSW><drvrHW> ; If there is a valid reference number, then find the DCE and calculate the dCtlDevBase MOVEQ #0,D0 ; set a good return MOVE.W spRefNum(A0),D1 ; get ref num BEQ.S @Done ; no ref num - done jsrROM ROMMapUnit ; return reg A2 = ptr to DCE BNE.S @Error ; some error with ref num or DCE ; Read a 24 or 32 bit base address from the sResource move.b srrId(a1),spId(a0) ; set spId of sRsrc to get base addr (the fix) jmpROM ROMpInitEntry ; continue in ROM ; Jump points back into ROM @Error jmpROM ROMpInitError ; jump to @Error label @Done jmpROM ROMpInitDone ; jump to @Done label EndProc ;____________________________________________________________________________ DAF <4.1> ; This patch really didn't get in until <5.1> ; ; SaveEntries Patch ; ; OK, I blew it on a patch roll-in from ROM78Fix into the Aurora ROM. In SaveEntries ; the a test was added for non-CLUT devices, but when I rolled it inline, I jumped ; back to the wrong place. This patch head patches the routine just like ROM78Fix ; so that it never hits the ROM routine if there is an error. ; mySaveEntries PROC EXPORT ROMSaveEntries EQU $3C920 MOVE.L ([theGDevice]),A0 ; get a pointer to the current device CMP.W #CLUTType,GDType(A0) ; is it a clut device? BNE.S SEDevErr ; nope, so report an error JMPRom ROMSaveEntries ; OK, so execute normal saveEntries SEDevErr MOVE.W #cDevErr,QDErr ; post error code RTD #12 ; and return ;____________________________________________________________________________ ; SetEntries, dvb < 124 > 1991 ; Mark bit 3 of QDSpare0 when the color environment changes, so pmgrexit ; only cleans up if bit was set for that layer. mySetEntries PROC EXPORT ROMSetEntries EQU $3C6C0 ; Mark QDSpare that the color environment has changed, but only if ; QD exists TST.B QDExist ; is QD around? BNE.S @noQDE ; no, skip this next MOVE.L (A5),A0 BSET #3,QDSpare0(A0) ; Tell PMgrExit to do color cleaning @noQDE JMPRom ROMSetEntries ; and on with our regularly scheduled program. < 124 > ;____________________________________________________________________________ ; Script Manager patch to Pack6, fixes problem with pointer to unlocked ; handle in Transliterate. <4.3><08/21/89 pke> ; Also fixes another problem in LwrString, CharType, Transliterate, FindWord: ; These routines need to have the Script Manager IntlForce flag cleared ; before calling IUGetIntl (this flag must be saved beforehand and restored ; afterward). This ensures that the itl2 tables used by these routines apply ; to the current script. <4.5><08/21/89 pke> ; Only do the second fix for Scripts604 OR (SysVers > $604) <4.7><08/21/89 pke> ; ; This patch is not installed for A/UX, since A/UX doesn't support the Script ; Manager. ; IF doScriptMgrPack6Fix AND (NOT installScriptMgrPtch39) THEN ; <42><50> ptchPack6 PROC Export fromROMLwrString EQU $0C0B2 fromROMCharType EQU $1D986 fromROMFindWord EQU $1DBE0 fromROMTranslit EQU $1DB34 toROMTranslit EQU $1DB3A ROMPack6 EQU $262B2 tlRecord record {a6link},decr result ds.w 1 ; function result. tlArgs equ *-8 ; size of arguments. srcHandle ds.l 1 ; srcHandle. dstHandle ds.l 1 ; dstHandle. target ds.w 1 ; target. srcMask ds.l 1 ; srcMask. selector ds.l 1 ; selector return ds.l 1 ; return address. a6link ds.l 1 ; old a6 register. tlLocals equ * ; size of local variables. endr cmp.w #iuGetIntl,4(sp) ; is it an IUGetIntl call? bne.s @NoPatch ; skip if not (optimization) CmpRA fromROMLwrString,(sp) beq.s @PatchOther CmpRA fromROMCharType,(sp) beq.s @PatchOther CmpRA fromROMFindWord,(sp) beq.s @PatchOther CmpRA fromROMTranslit,(sp) beq.s @PatchTrans ; Translit is special - 2 bugs @NoPatch JmpROM ROMPack6 ; nothing interesting to do @PatchTrans ; We get here if we came from the _IUGetIntl in Transliterate (IUGetIntl is a ; macro that pushes a selector and then does a _Pack6 trap). A few instructions ; before the _IUgetIntl, we had handles in a1 and a2 which were then dereferenced ; into the same registers. We want to postpone this dereferencing till after ; _Pack6. Fortunately, the original handles are still available in Transliterate's ; a6 frame, so we can get them after the Pack6 call, dereference them again, and ; stuff them where they belong. We also need to clear the IntlForce flag around ; Pack6 if it is set. WITH tlRecord, SMgrRecord addq #4,sp ; kill old return address GetSMgrCore a0 ; get pointer to SMgrRecord tst.b smgrIntlForce(a0) ; check IntlForce flag bne.s @fixFlag ; if set, go fix it JsrROM ROMPack6 bra.s @doDeref @fixFlag clr.b smgrIntlForce(a0) ; now force it to 0 JsrROM ROMPack6 GetSMgrCore a0 ; get pointer to SMgrRecord st smgrIntlForce(a0) ; restore IntlForce @doDeref ; The next two lines are what follow the _IUGetIntl in the ROM Transliterate move.l (sp)+,a4 ; movem.l (sp)+,a1/d1-d2 ; ; Now we expect to have pointers in a1 and a2, so set them up again (this is ; the fix) and jump back into ROM: move.l srcHandle(a6),a1 ; get source handle. move.l dstHandle(a6),a2 ; get destination handle. move.l (a1),a1 ; get source pointer. move.l (a2),a2 ; get destination pointer. JmpROM toROMTranslit ENDWITH @PatchOther ; We get here if we came from the _IUGetIntl in LwrString, CharType, or ; FindWord. If IntlForce is currently zero, we just JMP to the old Pack6. ; Otherwise, we copy part of the stack in order to save the return address, ; then JSR to Pack6, then fix up the stack. WITH SMgrRecord GetSMgrCore a0 ; get pointer to SMgrRecord tst.b smgrIntlForce(a0) ; check IntlForce flag beq.s @NoPatch ; if already 0, nothing to do clr.b smgrIntlForce(a0) ; now force it to 0 ; At this point the stack has 12 bytes that are significant for Pack6: ; 8(sp).L = space for returned handle ; 6(sp).W = argument, specifies which itlx ; 4(sp).W = Pack6 routine selector = #iuGetIntl ; 0(sp).L = return address ; We now copy the selector, argument, and return space below the return address. ; This sets up the stack correctly for the JsrROM (which will add a new return ; address) while preserving our original return address. clr.l -(sp) ; new place for returned handle move.l 8(sp),-(sp) ; copy selector and arguments ; Now this piece of stack has changed as follows: ; 16(sp).L = old space for returned handle ; 14(sp).W = old copy of argument (unnecessary) ; 12(sp).W = old copy of selector (unnecessary) ; 8(sp).L = old return address ; 4(sp).L = new space for returned handle ; 2(sp).W = new copy of argument, specifies which itlx ; 0(sp).W = new copy of Pack6 routine selector = #iuGetIntl ; We are now set up to call Pack6: JsrROM ROMPack6 ; The last stack picture: ; 12(sp).L = old space for returned handle ; 10(sp).W = nothing useful ; 8(sp).W = nothing useful ; 4(sp).L = old return address ; 0(sp).L = returned handle ; We need to copy the returned handle to where it should go, clean up the stack, ; set IntlForce, and return. move.l (sp)+,8(sp) ; copy returned handle to its place move.l (sp)+,a1 ; get return address addq #4,sp ; finish cleaning up stack GetSMgrCore a0 ; get pointer to SMgrRecord st smgrIntlForce(a0) ; restore IntlForce ; Now we just have the returned handle on the stack (see, no picture). jmp (a1) ; go home (I'm about to). ENDWITH ENDPROC ENDIF ; <42> ;============================================================================ pke <4.6> ; Patch UprString,RelString,CmpString to reintroduce the bug in which "`" is ; converted to uppercase as "a". This is necessary to prevent HFS problems, ; since existing disk catalogs use the old, buggy sorting. All we need to do ; is patch them up to the point where they access the UpperTab table, which ; does not have the necessary bug in ROM. ; ; A/UX uses the ROM versions of these routines, so we can install them for ; A/UX. ; PROC EXPORT NewUprString,NewCmpString,NewRelString ROMUprStringAfterEntString equ $0C10C ROMRelStringAfterEntString equ $0C136 ROMEntStringAfterLEAUpperTab equ $0C196 ;______________________________________________________________________ ; ; Routine: UprString ; Arguments: D0.W (input) -- string length ; A0.L (input) -- pointer to string to canonize ; Opcode bit 10 -- 0 - map to upper case; 1 - case sensitive ; Opcode bit 9 -- 0 - diacritical sensitive; 1 - strip diacrits ; All registers are preserved. ; Function: UprString is a utility routine which, according to ; a pair of input booleans, strips diacritical marks ; and/or maps all characters to upper case. ; ;______________________________________________________________________ NewUprString BSR EntString ; changed jmpRom ROMUprStringAfterEntString ;______________________________________________________________________ ; ; Routine: CmpString, RelString ; Arguments: D0.L (input) -- high-order word = string 0 length ; low-order word = string 1 length ; A0.L (input) -- pointer to string 0 ; A1.L (input) -- pointer to string 1 ; D0.W (output) -- ; CmpString -- result code = 0, 1 for equal, unequal. ; RelString -- result code = -1, 0, 1 according as string 0 ; is <, =, > than string 1, respectively. ; Opcode bit 10 -- 0 - map to upper case; 1 - case sensitive ; Opcode bit 9 -- 0 - diacritical sensitive; 1 - strip diacrits ; Pascal register conventions are observed. ; ; Calling Sequence for Pascal strings: ; LEA str0,A0 ; LEA str1,A1 ; MOVEQ #0,D0 ; MOVE.B (A0)+,D0 ; SWAP D0 ; MOVE.B (A1)+,D0 ; _CmpString ; ; Function: CmpString is a utility routine to compare two strings for ; equality. The strings are pointed to by A0 and A1 ; on entry; the result is returned in D0. ; Two booleans determine the flavor of the compare: ; Marks -- ; if set, all diacritical marks are ignored. ; Case -- ; if set, all characters are mapped to upper case. ;______________________________________________________________________ ; ; To expedite CmpString, test for length differences first. If they're the same, ; use the absolute value of RelString's return. ; NewCmpString MOVE.W D0,D2 ; str 1 length SWAP D0 ; str 0 length CMP.W D0,D2 BNE.S CSQuickNE SWAP D0 ; realign for fresh start in RelString BSR.S NewRelString ; count on CCR from last D0 store <4.6> BPL.S @1 NEG.L D0 ; -1 ---> 1 @1 RTS CSQuickNE MOVEQ #1,D0 ; must differ since lengths do RTS ; ; Register usage: ; D0 = shorter string length = loop index ; D1 = trap opcode, with embedded booleans ; D2 = string 0 input buffer ; D3 = string 1 input buffer ; D4 = -1, 0, 1 according to string lengths ; ; A0 = string 0 pointer ; A1 = string 1 pointer ; ; Output: ; D0 = -1, 0, 1 ; CCR reflects D0 ; D1-D2/A0-A1 trash ; NewRelString BSR.S EntString jmpROM ROMRelStringAfterEntString ; <4.6> ;______________________________________________________________________ ; ; Utility EntString saves regs D3-D4/A2-A3, sets up opword for quick ; test in Canonizer, and primes A2 with UpperTab and A3 with CmpTab. ; D2 is cleared for use by Canonizer. ; ; Utility ExtString cleans up and returns, leaving CCR. It is jumped to... ;______________________________________________________________________ EntString MOVE.L (SP)+,D2 ; return address MOVEM.L D2-D4/A2-A3,-(SP) ; return on top LEA UpperTab,A2 jmpROM ROMEntStringAfterLEAUpperTab ; <4.6> ; ; Handy macro allows for ease in building big, regular tables. ; ASSEMBLER WON'T ALLOW EIGHTH ARGUMENT ON SAME LINE!!!!!!!!!!!!!! ; MACRO crow DC.B (&Syslst[1]),( &Syslst[1]+1),( &Syslst[1]+2),( &Syslst[1]+3),( &Syslst[1]+4),( &Syslst[1]+5),( &Syslst[1]+6) DC.B (&Syslst[1]+7) ENDM MACRO dcrow crow &Syslst[1] crow &Syslst[1]+8 ENDM ;_________________________________________________________________________________________________________ ; ; <4.6> <08/21/89 dba/pke> ; ; Note the $61 at the beginning of the lower-case letters. In Feb. 89 for the ROMs, Peter E. fixed the ; upper casing of the "`" (backquote) character. Previously, it would turn into "a" (ASCII $61). ; Unfortunately, to our dismay, all of our old HFS disks require that this bug be perpetuated forever, ; at least for the ordering of HFS names in the B-Tree, which use RelString. We also realized that ; others may be using the string traps to order internal data structure. Because of all of this, we ; are reintroducing the bug. The uppercase version of backquote is hereby declared to be lower-case a. ; UpperTab ; Special chars, punctuation, digits, upper-case letters. dcrow $00 dcrow $10 dcrow $20 dcrow $30 dcrow $40 dcrow $50 ; Lower-case letters and some symbols. DC.B $61,( 'A'),( 'B'),( 'C'),( 'D'),( 'E'),( 'F'),( 'G') ; fixed ` <02/29/89 pke> unfixed <4.6> crow $48 crow $50 DC.B ('X'),( 'Y'),( 'Z'), $7B, $7C, $7D, $7E, $7F ; Accented characters. crow $80 DC.B $CB, $89, $80, $CC, $81, $82, $83, $8F DC.B $90, $91, $92, $93, $94, $95, $84, $97 DC.B $98, $99, $85, $CD, $9C, $9D, $9E, $86 ; Symbols and upper-case AE and O-slash dcrow $A0 ; Symbols and ae, o-slash crow $B0 DC.B $B8, $B9, $BA, $BB, $BC, $BD, $AE, $AF ; Symbols, upper A-acute, A-tilde, O-tilde, OE, and oe crow $C0 DC.B $C8, $C9, $CA, $CB, $CC, $CD, $CE, $CE ; Symbols and undefineds. y-umlaut cannot map up. dcrow $D0 dcrow $E0 dcrow $F0 ; End of UpperTab ;_________________________________________________________________________________________________________ ENDPROC IF doScriptMgrStr2DatFix AND (NOT installScriptMgrPtch39) THEN ; <7.8><42><50> ;============================================================================ pke <7.8> ; Patch Script Manager routines InitDateCache and String2Date for bug fixes ; needed by HyperCard. ; ---------------------------------------------------------------------------- proc export NewInitDateCache,NewString2Date ROMValidLong equ $1EE60 ; <7.8> ROMBlock2String equ $1EEA0 ; <7.8> ROMMatchString equ $1EF20 ; <7.8> ; in InitDateCache ($1EF80): ROMAfter2ndJSRCopyArray equ $1F0E8 ; <7.8> ROMExit equ $1F160 ; <7.8> ; in/after String2Date ($1F190): ROMAfterDBNE equ $1F2D6 ; <7.8> ROMDTComExit equ $1F426 ; <7.8> ROMFixValue equ $1F46E ; <7.8> ROMDateTimeEntry equ $1F47E ; <7.8> WhiteSpace EQU 1 ; token number for white space AlphaToken EQU 4 ; token number for string NumericToken EQU 5 ; token number for number AltNumericToken equ $B ; alternate token type for numbers NonAlphNumToken EQU 16 ; token numbers starting here correspond to non-alpha numeric tokens MaxTokens EQU 32 ; number of tokens for which there is space omdy EQU 0 ; date order constants odmy EQU 1 oymd EQU 2 Str15 EQU 16 ; length of string[15] DayMonthLen equ 15 ; length of days and months NumDays EQU 7 ; length of various arrays NumMonths EQU 12 DayList EQU NumDays*Str15 MonthList EQU NumMonths*Str15 NumStrBytes EQU 300 Cache RECORD 0 version ds.w 1 CurrentDate DS LongDateRec BaseDate DS LongDateRec theDays DS.B DayList ; <7.8> theMonths DS.B MonthList ; <7.8> theEveStr DS.B Str15 theMornStr DS.B Str15 the24hrStr DS.B Str15 theTimeSep DS.W 1 theDateSep DS.W 1 theDateOrder DS.b 1 longDateOrder ds.b 1 theAbbrLen DS.W 1 TBlock DS tokenBlock CacheSize equ * theTimeStrings equ theEveStr ENDR ; ---------------------------------------------------------------------------- ; function InitDateCache(theCache: CachePtr): OSErr; ; ; InitDateCache will initialize the items in the DateCacheRecord pointed to by theCache^ ; and set the initialized item to true ; ; The bug fixes are in the CopyArray subroutine <7.8>. ; ---------------------------------------------------------------------------- idcSaveRegs REG A2-A4/D3/d4 InitCacheRec RECORD {A6link},decr Result DS.W 1 ; offset to function result (integer) paramBegin EQU * theCache DS.L 1 ; pointer to the cache record selector ds.l 1 ; added for resource paramEnd EQU * return DS.L 1 A6link DS.L 1 theTokens DS.B 2*TokenRec.tokenRecSize Intl0 DS.L 1 ; handle, not pointer Intl1 DS.L 1 ; handle, not pointer altSpace ds.w 1 ; use top byte aLongDate ds LongDateTime ; localsize EQU * ENDR WITH InitCacheRec,LongDateField NewInitDateCache LINK A6,#localsize ; Establish local frame and variables MOVEM.L idcSaveRegs,-(SP) ; saved used registers CLR.W Result(A6) ; clear function result MOVE.L theCache(A6),A2 ; cache addr move.b #' ',altSpace(a6) ; assume no alt space sub.l #6,sp ; reserve returns _IntlScript ; get the script move.w #smScriptRight,-(sp) ; get flag _GetScript ; get value tst.l (sp)+ ; got it? beq.s @NoAltSpace ; no move.b #' '+$80,altSpace(a6) ; use alt space. @NoAltSpace WITH Cache lea aLongDate(a6),a0 ; @temp clr.l (a0)+ ; no high move.l Time,(a0)+ ; low = current pea aLongDate(a6) ; extended pea CurrentDate(A2) ; get current date from system global time and convert to date _LongSecs2Date lea aLongDate(a6),a0 ; @temp clr.l (a0)+ ; no high clr.l (a0)+ ; low = current pea aLongDate(a6) ; extended pea BaseDate(A2) ; get base date and convert to date _LongSecs2Date ; IUGetIntl(0): Handle CLR.L -(SP) CLR.W -(SP) _IUGetIntl MOVE.L (SP)+,Intl0(A6) move.w ResErr,d0 ; did intl0 load in all right BEQ.S GotIntl0Ok ; if so, go on MOVE.W d0,Result(A6) BRA Exit GotIntl0Ok ; IUGetIntl(1): Handle CLR.L -(SP) MOVE.W #1,-(SP) _IUGetIntl MOVE.L (SP)+,Intl1(A6) TST.W ResErr ; did intl0 load in all right BEQ.S GetSeparators ; if so, go on MOVE.W ResErr,Result(A6) BRA Exit GetSeparators MOVE.L Intl0(A6),A3 ; Lock itl0 across Block2String calls (which can now move memory) move.l a3,a0 _MoveHHi move.l a3,a0 _HLock MOVE.L (A3),A3 ; now a3 is pointer to itl0 ; Block2String(eveStr,theEveStr,4,true) LEA eveStr(A3),A0 ; get international 0 record for evening string LEA theEveStr(A2),A1 MOVEQ #4,D0 MOVE.B #1,D1 move.b altSpace(a6),d2 ; pass alt string JsrROM ROMBlock2String ; convert packed array of char into string <7.8> ; with no leading spaces ; Block2String(mornStr,theMornStr,4,true) LEA mornStr(A3),A0 ; get international 0 record for morning string LEA theMornStr(A2),A1 MOVEQ #4,D0 MOVE.B #1,D1 move.b altSpace(a6),d2 ; pass alt string JsrROM ROMBlock2String ; convert packed array of char into string <7.8> ; with no leading spaces ; Block2String(time1Stuff,the24hrStr,1,true) LEA timeSuff(A3),A0 ; get international 0 record for 24 hour string LEA the24hrStr(A2),A1 MOVEQ #1,D0 MOVE.B #1,D1 move.b altSpace(a6),d2 ; pass alt string JsrROM ROMBlock2String ; convert packed array of char into string <7.8> ; with no leading spaces MOVE.B timeSep(A3),theTimeSep(A2) ; get time separator from intl 0 MOVE.B dateOrder(A3),theDateOrder(A2) MOVE.B dateSep(A3),(theTimeSep + 1)(A2) ; Now unlock itl0 move.l Intl0(A6),a0 _HUnlock MOVE.L Intl1(A6),A3 ; Lock itl1 across Copy Array calls; it calls Block2String, which can now move memory move.l a3,a0 _MoveHHi move.l a3,a0 _HLock MOVE.L (A3),A3 ; now a3 is pointer to itl1 ; Get the long date order and convert ; false => dmy, true => mdy; otherwise fancy stuff move.l #omdy,d1 ; assume false clr.w d0 ; wordize move.b lngDateFmt(a3),d0 ; get long format beq.s @GotLong ; done move.l #odmy,d1 ; assume true cmp.b #$FF,d0 ; true? beq.s @GotLong ; yes, got it ; if we are looking at the long date, only the day and year are important, ; since the dayOfWeek and month are both strings. Walk through the format until ; we find either one or the other @LongFmtLoop move.b d0,d2 ; get bottom half-nybble and.b #3,d2 ; got it cmp.b #longDay,d2 ; day? beq.s @GotLong ; yes, return #odmy cmp.b #longYear,d2 ; year? beq.s @LongYearFirst ; go for it lsr.b #2,d0 ; strip bottom bne.s @LongFmtLoop ; repeat until done @LongYearFirst move.l #oymd,d1 ; year first (also if none found) @GotLong move.b d1,longDateOrder(a2) ; set it ; continue MOVE.B abbrLen(A3),theAbbrLen(A2) ; get abbrLen ; changed order for more convenience in the later routine lea months(a3),a4 ; source of days LEA theMonths(A2),a1 ; destination names of the day and months MOVE.L #(numMonths - 1),D3 ; load day and months from intl 1 rec into space jsr CopyArray ; copy days lea days(a3),a4 ; source of days LEA theDays(A2),a1 ; destination names of the day and months MOVE.L #(numDays - 1),D3 ; load day and months from intl 1 rec into space jsr CopyArray ; copy days ; Now continue with ROM code JmpROM ROMAfter2ndJSRCopyArray ; <7.8> Exit JmpROM ROMExit ; <7.8> ; little subroutine for code savings ; a4 is source pointer ; a1 is dest pointer ; d3 is byte length CopyArray move.l a1,d4 ; dest ptr @LOOP MOVE.L A4,A0 ; move string from intl 1 rec <7.8> MOVE.L d4,A1 ; into local frame moveq #0,d0 ; Block2String wants a long! <7.8> MOVE.B (A0)+,D0 ; with length from first byte of string MOVEQ #1,D1 ; with no local frame move.b altSpace(a6),d2 ; pass alt string JsrROM ROMBlock2String ; search <7.8> ADD.L #Str15,d4 ; get next string to transfer ADD.w #Str15,A4 ; into next string in local frame <7.8> DBRA D3,@LOOP rts ENDWITH ; Cache ENDWITH ; ---------------------------------------------------------------------------- ;function String2Date( textPtr: Ptr; ; textLen: longint; ; theCache: DateCachePtr; ; var lengthUsed: longint; ; var DateTime: LongDateRec): Integer; ; ; String2Date will look for a date in the text given it according to the international ; resource format. Using the Tokenizer routine it will look for a dayofWeek (which will ; be a string), a month (either a string or a number) and a day and year (both numbers). ; If the month is a number, order is decided by the ShortDate format in INTL 0; otherwise ; String2Date uses a table. Note if only two numbers are found they are assumed to be day ; and month. If one number is found it is assumed to be a date. Missing fields are ; filled in by the current date and time ; ; Register Usage ; ; A2 - Work register D3 - loop control register ; A3 - Work register D4 - MonthFound ; A4 - Cache Addr D5 - ResultNum ; D6 - AbbrLen ; D7 - NumDelimsFound ; ; The bug fixes are: É <7.8> ; ---------------------------------------------------------------------------- s2dSaveRegs REG D3-D7/A2-a4 ; removed a5 DateFrame RECORD {A6link},decr Result DS.W 1 ; offset to function result (integer) paramBegin EQU * textPtr DS.L 1 ; offset to Date2Time's paramters textLen DS.L 1 theCache DS.L 1 ; @DateCacheRecord RestofText DS.L 1 ; @Longint DateTime DS.L 1 ; @LongDateRec selector ds.l 1 ; added for resource paramEnd EQU * return DS.L 1 A6link DS.L 1 theTokens DS.B MaxTokens*TokenRec.tokenRecSize ; storage for tokens found by tokenizer theDate DS LongDateRec ; date time rec results DS.W 3 ; three temporary results myDateOrder DS.W 1 DayFound DS.W 1 lastItemSep DS.W 1 lastToken DS.W 1 lastTokenAddr DS.L 1 lastExToken DS.W 1 dummyLongDate ds LongDateTime stringStorage DS.B NumStrBytes localsize EQU * ENDR WITH DateFrame With Cache,LongDateField NewString2Date LINK A6,#localsize ; Establish local frame and variables MOVEM.L s2dSaveRegs,-(SP) ; saved used registers JsrROM ROMDateTimeEntry ; initialize & call _IntlTokenize ; Note that DateTimeEntry also does the following: ; 1. moves theCache(a6) into a4 ; 2. moves TheTokens(a6) into a3 ; 3. sets result(a6) ; 4. clears D7,D6; sets D5.L = -1 ; 5. after IntlTokenize, sets D3= number of tokens - 1 tst.w result(a6) ; entry failed? bne DTComExit ; bail if so ; specific stuff MOVEQ #-1,D4 ; initialize MonthFound to -1 CLR.W DayFound(A6) LEA CurrentDate(A4),A1 ; source LEA theDate(A6),A0 ; MOVE.L (A1)+,(A0)+ ; era, year MOVE.L (A1)+,(A0)+ ; month, day MOVE.L (A1)+,(A0)+ ; hour, minute move.w (a1)+,(a0)+ ; second CLR.W (A0)+ ; clear out dayOfWeek. Will either be set by user or be set ; in the course of the validity check of the date MOVE.B theAbbrLen(A4),D6 WITH TokenRec @TokenLoop MOVE.W theToken(A3),D1 ; get token code from TokenRec record at (A3) CMP.W #NonAlphNumToken,D1 ; is it a separator BGE.S @FoundSeparator SUB.W #WhiteSpace,D1 ; was it a white space? BEQ @TokenLoopEnd ; ignore SUBQ.W #(AlphaToken - WhiteSpace),D1 ; is it an alpha token? beq.s @FoundAlpha SUBQ.W #(NumericToken - AlphaToken),D1 ; is it a number token? BEQ.S @NumberToken SUBQ.W #(AltNumericToken - NumericToken),D1 ; is it a number token? BEQ.S @NumberToken ; BRA @TokenLoopEnd ; ignore any other characters @FoundSeparator TST.B D7 ; possible separator. Has a separator already been found BEQ.S @OneDelimFound ; no, so no error yet OR.W #TooManySeps,Result(A6) ; yes, so now we have too many separators @OneDelimFound ADDQ.B #1,D7 ; record separator found TST.w D5 ; is this after the first number loaded? (.w) <1/5/88med> BMI @TokenLoopEnd ; if we have not reached a number yet go on BNE.S @CheckDateSep ; if we loaded in a number then check consistency MOVE.W D1,lastItemSep(A6) ; otherwise update lastItemSep with just found separator BRA @TokenLoopEnd @CheckDateSep CMP.W lastItemSep(A6),D1 ; are they consistent BEQ @TokenLoopEnd ; if yes, than go on OR.W #SepNotIntlSep+sepNotConsistent,Result(A6) ; record warning BRA @TokenLoopEnd @NumberToken ; check for numbers separated only by white space tst.w d7 ; got a separator? bne.s @1 ; yes tst.w d5 ; first number? blt.s @1 ; yes, skip bne.s @0 ; have 1 number move.w #WhiteSpace,lastItemSep(A6) ; set separator bra.s @1 ; continue @0 cmp.w #WhiteSpace,lastItemSep(a6) ; same? beq.s @1 ; yes, continue or.w #SepNotIntlSep+sepNotConsistent,Result(A6) ; record warning @1 ; end of white space check CLR.L D7 ; clear out past separators CMP.W #2,D5 ; make sure that results array is not full BGE @ExtrnsToken ; if full, record warning and search <7.8> ; for final string (remove .s) ; ValidLong(position,textLength): longint MOVE.L stringPosition(A3),A0 JsrROM ROMValidLong ; get number in text at TokenRec.position <7.8> BMI DTComExit JsrROM ROMFixValue ; <7.8> bra @TokenRecognized ; get next token (remove .s) <7.8> @FoundAlpha CLR.L D7 ; clear out past separators ; UprString(newTextPtr,AbbrLen) ; and make it uppercase for case insensitive search MOVE.L stringPosition(A3),A2 ADDQ.L #1,A2 MOVE.L A2,A0 MOVE.L length(A3),D0 ; use real length _UpperText ; _LwrString with uppercase function <39><45> TST.W D4 BPL.S @SearchForDay ; <7.8> ; MatchString(newTextPtr,AbbrLen,@months,15,12): integer CLR.W -(SP) ; attempt to find a day/month string which matches alpha token MOVE.L A2,-(SP) MOVE.W length+2(a3),-(SP) ; push length as integer MOVE.W D6,-(SP) ; push minlen as integer PEA theDays(A4) ; <7.8> MOVE.W #DayMonthLen,-(SP) MOVE.W #NumMonths+NumDays,-(SP) ; check for both at once JsrROM ROMMatchString ; <7.8> MOVE.W (SP)+,D2 ; save in D2 BLE.S @ExtrnsToken ; checking both ; decide between days and months sub.w #NumDays,d2 ; months? ble.s @CheckForDay ; no, do days ; got a month CMP.W #2,D5 ; is result array already full? BGE.S @ExtrnsToken ; if so, then its not a needed token ADDQ.W #1,D5 ; is in range, result is not full and string MOVE.W D5,D1 ; did not match as a day ADD.W D1,D1 ; double index for integer array LEA results(A6),A1 ; ResultNum:= ResultNum + 1 MOVE.W D2,(A1,D1.W) ; results[ResultNum]:= match result MOVE.W D5,D4 ; MonthFound:= resultNum BRA.S @TokenRecognized ; if day not found yet, we need to see if current string matches a day <7.8> @SearchForDay TST.B DayFound(a6) ; have we found one yet? BNE.S @ExtrnsToken ; day already found, record warning CLR.W -(SP) ; attempt to find a day string which matches alpha token MOVE.L A2,-(SP) MOVE.W length+2(a3),-(SP) ; push length as integer MOVE.W D6,-(SP) ; push minlen as integer PEA theDays(A4) MOVE.W #DayMonthLen,-(SP) MOVE.W #NumDays,-(SP) ; just check days this time JsrROM ROMMatchString ; <7.8> MOVE.W (SP)+,D2 ; save in D2 BLE.S @ExtrnsToken ; checking both BRA.S @HaveDay @CheckForDay TST.B DayFound(a6) ; off of a6! BNE.S @ExtrnsToken ; day already found, record warning add.w #NumDays,d2 ; restore days <7.8> @HaveDay ; <7.8> ST DayFound(A6) ; record that day was found <7.8> MOVE.W D2,theDate.dayOfWeek(A6) ; otherwise, load dayOfWeek into date time record TST.W D5 ; is dayOfWeek between two numbers (.w) BMI.S @NoWarning ; if no number has been found yet, go on OR.W #fieldOrderNotIntl,Result(A6) ; record warning @NoWarning BRA.S @TokenRecognized @ExtrnsToken TST.W lastExToken(A6) ; is this the first extraneous token found? BPL.S @TokenLoopEnd ; no, go on MOVE.W D3,lastExToken(A6) ; a string which we don't recognize has been found BRA.S @TokenLoopEnd ; record warning and go on @TokenRecognized MOVE.W D3,lastToken(A6) MOVE.L A3,lastTokenAddr(A6) ; save loop and token addr for later use @TokenLoopEnd ADD.L #tokenRecSize,A3 ; add size of TokenRec record to A3 to get next token CMP.W #2,D5 ; if the result array full (.w) SGE D0 AND.B DayFound(A6),D0 ; and dayofWeek string been found DBNE D3,@TokenLoop ; then stop loop, otherwise go until all tokens are looked at JmpROM ROMAfterDBNE DTComExit JmpROM ROMDTComExit endproc ENDIF IF doScriptMgrRstKCHRFix AND (NOT installScriptMgrPtch39) THEN ; <8.0><42><50> ;============================================================================ pke <8.0> ; Tail patch on _GetIndADB to fix bug in Script Manager ResetKCHR routine ; called by SwapKybd: it clears ADB keyboard driver dead state as a word, ; should be a long. ; ---------------------------------------------------------------------------- proc export ptchGetIndADB ROMAfterGetIndADB equ $1E048 ROMAfterClrDeadKey equ $1E058 ROM@1dbra equ $1E064 ptchGetIndADB CmpRA ROMAfterGetIndADB,OSTrapReturnAddressDepth(sp) ; From ResetKCHR? bne.s @0 ; if not, just continue with trap lea ptchResetKCHR,a1 ; a1 is not a param for GetIndADB move.l a1,OSTrapReturnAddressDepth(sp) ; so trap returns to patch, not ROM @0 BackToTrap oldGetIndADB ; now get on with the trap ; When we reach the patch below, we are in ResetKCHR just after the _GetIndADB rkRecord record {a6link},decr rkArgs equ *-8 ; size of arguments. pointer ds.l 1 ; new KCHR pointer. return ds.l 1 ; return address. a6link ds.l 1 ; old a6 register. adb ds ADBDataBlock ; ADB data structure. oldKeyboard ds.l 1 ; old keyboard pointer rkLocals equ * ; size of local variables. endr ptchResetKCHR with rkRecord cmp.b #kybdADBAddr,adb.origADBAddr(a6) ; is it a keyboard? bne.s @1 ; no -> skip it. move.l adb.dbDataAreaAddr(a6),a0 ; load data area pointer. clr.l KybdDriverData.deadKey(a0) ; clear dead state - long! <8.0> JmpROM ROMAfterClrDeadKey ; <8.0> @1 JmpROM ROM@1dbra ; <8.0> endwith endproc ENDIF IF doScriptMgrLwrString2 AND (NOT installScriptMgrPtch27) THEN ; <8.5><50> ;============================================================================ pke <8.5> ; Patch LwrString to handle 2-byte chars via Transliterate. ; ---------------------------------------------------------------------------- ; routine: LwrString ; input: a0 textPtr ; d0.w length ; d1.w trap word; the following bits are significant in 7.0 only: ; Opcode bits ; 10 9 Function ; -- -- -------- ; 0 0 convert to lower-case ; 0 1 strip diacritics ; 1 0 convert to upper-case ; 1 1 convert to upper-case and strip diacritics ; output: d0.w error ; function: Change text pointed to by a0 according to opcode bits. Before 7.0, we ; just assume that these bits are 0 and act accordingly. ; ---------------------------------------------------------------------------- proc export ptchLwrString lwrTrFrame record {a6link},decr return ds.l 1 ; return address a6link ds.l 1 ; link pointer sourcePtr ds.l 1 ; orig source ptr sourceLen ds.l 1 ; orig source len sourceHndl ds.l 1 ; new handle with copy of source destHndl ds.l 1 ; new handle for transliterate result errCode ds.w 1 ; err code to be returned in d0 lwrTrLocals equ * ; size of locals endr ; ---------------------------------------------------------------------------- ; NOTE: For 2-byte scripts, we need to call Transliterate. So, the first thing to ; do is figure out what script we're in. We also test for length <= 0 (tests >32K). ; ---------------------------------------------------------------------------- ptchLwrString ext.l d0 ; as fast as tst.w and we need ext.l later ble lwrRTS ; if bad length, quit movem.l a0/d0/d1,-(sp) ; save important registers subq.l #2,sp ; make room for return _FontScript ; find script of port move.w (sp)+,d2 ; pop it into d2 movem.l (sp)+,a0/d0/d1 ; restore important registers ; ---------------------------------------------------------------------------- ; Here we make use of the fact that a word redraw flag of 1 (or anything >0) ; indicates a 2-byte script. ; ---------------------------------------------------------------------------- with SMgrRecord,ScriptRecord lsl.w #2,d2 ; make script code a long offset move.l IntlSpec,a1 ; get SMgrRecord pointer move.l smgrEntry(a1,d2.w),d2 ; get ScriptRecord pointer beq lwrRTS ; if nil (script not installed), do nothing move.l d2,a1 tst.b scriptEnabled(a1) ; is script enabled? beq lwrRTS ; if not, do nothing tst.b scriptRedraw(a1) ; is it a 2-byte script? ble not2Byte ; if not, go use normal LwrString endwith ; ---------------------------------------------------------------------------- ; OK, we need to set up for Transliterate: set up source & dest handles ; ---------------------------------------------------------------------------- with lwrTrFrame link a6,#lwrTrLocals ; create local storage move.l a0,sourcePtr(a6) ; save source ptr move.l d0,sourceLen(a6) ; save length _PtrToHand ; make new handle containing copy of text move.w d0,errCode(a6) ; save err code bne.s lwrTrUnlk ; if error in PtrToHand, quit move.l a0,sourceHndl(a6) ; save new source handle move.l sourceLen(a6),d0 ; get length again _NewHandle ; make new handle with random contents move.w d0,errCode(a6) ; save err code bne.s lwrTrDisp1Hndl ; if error in NewHandle, quit ; ---------------------------------------------------------------------------- ; Now do Transliterate and check result: dest length should equal source length ; ---------------------------------------------------------------------------- move.l a0,destHndl(a6) ; save new dest handle subq.l #2,sp ; space for Transliterate error code move.l sourceHndl(a6),-(sp) ; push source handle move.l a0,-(sp) ; push dest handle (still in a0) move.w #smTransLower+smTransAscii,-(sp) ; lower-case conversion, target=Roman move.l #smMaskAscii,-(sp) ; convert Roman only _Transliterate move.w (sp)+,errCode(a6) ; save error code bne.s lwrTrDisp2Hndl ; if error in Transliterate, quit move.l destHndl(a6),a0 ; now check resultÉ _GetHandleSize move.w d0,errCode(a6) ; save err code tst.l d0 ; .l, because the result is really a long blt.s lwrTrDisp2Hndl ; if error in GetHandleSize, quit move.w #-1,errCode(a6) ; assume len err; need a better err code! cmp.l sourceLen(a6),d0 ; should be same as source len bne.s lwrTrDisp2Hndl ; if not, set err and bail ; ---------------------------------------------------------------------------- ; Copy result (destHndl) to original text buffer (sourcePtr). ; NOTE: For large text blocks, _BlockMove might be faster than this loop. ; ---------------------------------------------------------------------------- move.l destHndl(a6),a0 ; get dest handle andÉ move.l (a0),a0 ; deref it to get source ptr for this copy move.l sourcePtr(a6),a1 ; old source ptr is dest for this copy move.l sourceLen(a6),d0 ; length for copy (we know it is >= 1) subq.l #1,d0 ; set up for dbra @1 move.b (a0)+,(a1)+ ; copy a byte dbra d0,@1 ; loop till done clr.w errCode(a6) ; no errors! ; ---------------------------------------------------------------------------- ; Exits - need to dispose of handles we created ; ---------------------------------------------------------------------------- lwrTrDisp2Hndl move.l destHndl(a6),a0 ; _DisposHandle ; lwrTrDisp1Hndl move.l sourceHndl(a6),a0 ; _DisposHandle ; lwrTrUnlk move.w errCode(a6),d0 ; set err code unlk a6 ; rts endwith lwrRTS clr.w d0 rts ; ---------------------------------------------------------------------------- ; This is not a 2-byte script, just use old LwrString. ; ---------------------------------------------------------------------------- not2Byte BackToTrap oldLwrString ; endproc ENDIF ;_______________________________________________________________________ ; <7.1><8.7><41><81><123><125><126><128><135><137> Async Serial Driver Patch ; <4><5> ; ; Start of patch code for fix of bugs in the serial driver ; introduced into the aurora ROM (and hence the F19 overpatch ROM) by the ; port arbitration code, which was accidently included in those ROMS. ; ; The patch is installed in a patch vector for the async driver. ; Note the 'simple' come-from patching mechanism has been really hosed by the ; addition of the pic chip code; now we have to include all that code in the ; patch. Something to watch for when fixing the ROM driver. ; ; Needed to include all code which jumped TO the ExtSts interrupt handler. ; ; Traps patched: -- The AsyncVector, which is in the middle of the OS ToolTable ; is patched, Trap number = $A0BE ; -- all the PIC driver headers patched. <41> ; ; ; HISTORY: ; <7.1> 10/2/89 patch ExtAIntHnd and ExtBIntHnd ; <8.7> check SCCIOPFlag now instead of PRAM to determine which driver to load. ; changed sense of bypass bit to agree with new ROM definition. ; <41> fixed <8.7> to use the patch macros. Also added in patch for the ; IOPSerialDriver for the prime code. As a result, had to patch ALL ; the IOP driver headers. yuck. ; <81> added VM support for the SerSetBuf control call. ; <123> added nike printer support: ; patched control call 16 in bypass driver to use bit 6 for setting ; external/internal clocking modes. ; Patched both bypass and IOP code to add status call to return version. ; Patched InitSCC routine in bypass driver to use this RAM-based table so ; that clocking mode is now variable instead of hard-wired to internal only. ; -- set clock divide to 1 instead of 16 for external clocking ; -- clear HWHS enable since we can't clock and handshake on same line ; Patched bypass mode driver to set bit 3 of AsyncErr if break rcvd. ; <125> We patch all the bypass driver headers (iop headers are already patched) ; so that we can put a signature word before each header that we can use ; to identify ourselves as the Apple async serial driver. We do this ; right now so that that our linked patch on Open (in Serialpatches.a) will ; not stomp the version number of some third party driver that sticks them- ; selves into our spot in the unit table. I think this idea will come in ; handy later as well. ; I think i'll use the signature word 'wong'. ; <126> fixed a bug in EXTSTS interrupt handler for port A. Because of the BAP ; stuff, AInDCE no longer has a valid ptr the .Ain driver's DCE. We solve ; the problem the same way we did in the port B code, by getting the DCE ; ptr straight from the unit table. ; <127> Fixed a bug in the External Status interrupt handler. On ; port A we crashed on receiving a break if a read was pending ; because we weren't getting the .Ain DCE ptr properly. Now we get ; the DCE ptr directly from the Unit table, just like for port B. ; <129> Fixed bug in external clock stuff that was causing us to disable ; internal clocking on both ports if we were trying to do external ; clocking on just one. So Nike would print but Tabasco would just ; sit there. The fix was to set clocking params at each time we go ; thru InitSCC, rather than just once during the external clocking ; control call. This way even tho the ports share the InitSCCTable, ; we set the Table properly by keying off of the value in CtlOptions ; variable, which they don't share. ; <135> Status call 9 and $8000 return fixed version equate instead of ; what's in the DCE. Fixed SerSetBuff (control call 9) to not sign ; extend the buffer count. ; <137> Fixed bug in IOP code that caused SerSetBuff to fail for zero size ; buffer. Wasn't clearing upper word of count passed to UnholdMemory. ; <4> patched Rx and SCx interrupt handlers to help thruput on Apollo and TIM LC ; <5> patched SCx interrupt handler to fix the StyleWriter bug. On faster ; machines we are able to send out a byte fast enough after the TBE bit ; in RR0 gets set that the byte gets written before the TBE interrupt ; is generated. Instead of just not generating the interrupt, the SCC goes ; ahead and generates it, but changes the interrupt source code to read ; 'no interrupt' instead of 'TBE'. This 'no interrupt' code is identical ; to the 'Special Condition on port B' code, which was the source of our ; problem. Fix is to just exit SCIntHnd if no special condition bits are ; set. ; ;_______________________________________________________________________ AsyncPatch proc export SerialPatch ; ; equates ; PortAVars EQU SerialVars ; serial chan A variables and buffer SerialVers EQU 5 ; current version 3/91 <135> ; ; Bypass driver var offsets ; OutDCE EQU 0 ;(4) long DCE pointer for output driver SCCOffset EQU 4 ;(2) word of SCC offset . . . InBufPtr EQU 6 ;(4) pointer to local input buffer BufSize EQU 10 ;(2) size of local input buffer BufLow EQU 12 ;(2) low buf byte count to send XOn BufHigh EQU 14 ;(2) bytes from end of buffer to send XOff SWHS EQU 16 ;(1) software handshake enable HWHS EQU 17 ;(1) hardware handshake enable XONChar EQU 18 ;(1) input char which continues output (SWHS) XOFFChar EQU 19 ;(1) input char which stops output Options EQU 20 ;(1) bit 4 = abort on parity error ; bit 5 = abort on overrun ; bit 6 = abort on framing error PostOptions EQU 21 ;(1) bit 7=1 enables posting break changes ; bit 5=1 enables posting handshake changes InSWHS EQU 22 ;(1) input XOn/XOff flow control enable InHWHS EQU 23 ;(1) input RTS (DTR) flow ctl enb <14Oct85> AsyncErr EQU 24 ;(1) error indications (cumulative) SoftOR EQU 0 ; bit 0 = soft overrun ; bit 4 = parity error ; bit 5 = overrun error ; bit 6 = framing error FlowOff EQU 25 ;(1) $80 = input flow shut off by XOff, $40 by DTR ReadCmd EQU 26 ;(1) FF = read command pending WriteCmd EQU 27 ;(1) FF = write command pending CTSFlag EQU 28 ;(1) FF = CTS asserted XOFFlag EQU 29 ;(1) FF = XOFF pending LastWR5 EQU 30 ;(1) WR5 value with last DTR state <14Oct85> DTRNegVal EQU 31 ;(1) WR5 value used to negate DTR <14Oct85> SCCReset EQU 32 ;(1) WR9 value for reset StopBits EQU 33 ;(1) stop bits/parity option (WR4 value) WR1AVal EQU 34 ;(1) first WR1 value to write WR3AVal EQU 35 ;(1) first WR3 value to write WR5AVal EQU 36 ;(1) first WR5 value to write BaudLoCnst EQU 37 ;(2) 2 byte baud rate constant (WR12-13) BaudHiCnst EQU 38 RcvrBits EQU 39 ;(1) 1 byte receiver bits/char (WR3 value) XmitBits EQU 40 ;(1) 1 byte xmitter bits/char (WR5 value) WReqPin EQU 41 ;(1) w/req pin state (WR1 value) lastSetup EQU 42 ;(2) last SCC init values . . . BufIndex EQU 44 ;(2) index into local buffer (insert) BufOutdex EQU 46 ;(2) index into local buffer (remove) LocalBuf EQU 48 ;(64) local buffer for input chars LclBufSize EQU 64 ; default input buffer size = 64 bytes HSCount EQU 112 ;(2) count of CTS pulses in VBL time (clk detect) <14Oct85> LastTime EQU 114 ;(4) ticks time of last CTS pulse (clk detect) <14Oct85> SendXOnff EQU 118 ;(1) flag to xmit logic to send XOn/XOff <14Oct85> CharMask EQU 119 ;(1) $1F,$3F,$7F, or $FF mask for input chars <14Oct85> PEChar EQU 120 ;(1) char to change incoming parity errors to <14Oct85> AltChar EQU 121 ;(1) char to change incoming PEChars to <14Oct85> InSWHS1 EQU 122 ;(1) saved InSWHS state <14Oct85> CtlOptions EQU 123 ;(1) bits 0-6=0 (reserved). bit 7=1 to leave <14Oct85> ; DTR state unchanged at close. SaveExInt EQU 124 ;(4) saved Ext int vector <14Oct85> SaveTxInt EQU 128 ;(4) saved TxD int vector <14Oct85> SaveRxInt EQU 132 ;(4) saved RxD int vector <14Oct85> SaveSxInt EQU 136 ;(4) saved Special Rx int vector <14Oct85> InDCE EQU 140 VarSize EQU 144 ; output driver storage size LclVarSize EQU 144 ; output driver storage size ; ; PIC driver var offsets and misc equ ; PICOpenFlag EQU 0 ; marks if driver is open OpenMsgPB EQU 22 EventMsgPB EQU 106 OpenMsg EQU 270 PICInBufPtr EQU 368 ; ptr to current input buffer PICBufSize EQU 372 ; size of current input buffer PICBufIndex EQU 374 ; index for putting bytes into input buffer PICBufOutdex EQU 376 ; index for taking bytes out of input buffer PICLocalBuf EQU 378 ; ptr to driver's local buffer ; General Equates for message fields Results Equ 0 ;(1) Message return code OpCode Equ Results+1 ;(1) Some messages slots are shared CloseOpCode Equ 2; Close ;____________________________________________________________________ ; ; ROM address offsets ; ;____________________________________________________________________ ; bypass driver ROM entrypoint addresses <125> ;port A ROM_AInEntryOpen EQU $6ac2a ROM_AInEntryClose EQU $6af82 ROM_AInEntryPrime EQU $6b2a8 ROM_AOutEntryOpen EQU $6acf2 ROM_AOutEntryClose EQU $6af0e ROM_AOutEntryPrime EQU $6b1ec ROM_AEntryControl EQU $6b002 ROM_AEntryStatus EQU $6afac ;Port B ROM_BInEntryOpen EQU $6ac30 ROM_BInEntryClose EQU $6af82 ROM_BInEntryPrime EQU $6b2ae ROM_BOutEntryOpen EQU $6ad22 ROM_BOutEntryClose EQU $6af22 ROM_BOutEntryPrime EQU $6b1f2 ROM_BEntryControl EQU $6b008 ROM_BEntryStatus EQU $6afb2 ; Open patch equates for bypass driver fromAOutOpen EQU $0006B542 ; at end of drvr, in IOP stuff (ChkConfig called there) fromBOutOpen EQU $0006B550 fromAInOpen EQU $0006B55E fromBInOpen EQU $0006B56C fromControl EQU $6B01a fromClose EQU $6af3e fromStatus EQU $6afc4 fromInitSCC EQU $6aeb8 backToAOutOpen EQU $0006AD12 ; returning to finish up Open routines in ROM backToBOutOpen EQU $0006AD44 PlugInDrvr EQU $0006B594 ; plug header addr into DCE and continue backToInOpen EQU $0006ac34 ; finish In Open routines in ROM ROM_PollDtain EQU $0006B372 ; rom routines addresses we must push on the stack ROM_SCAIntHnd EQU $0006B466 ROM_SCBIntHnd EQU $0006B45C ROM_RAIntHnd EQU $0006B3B0 ROM_RBIntHnd EQU $0006B3A6 ROM_TAIntHnd EQU $0006B276 ROM_TBIntHnd EQU $0006B26C ROM_AsyncAIn EQU $0006abca ; ROM header addresses for byass drivers ROM_AsyncAOut EQU $0006ABE2 ROM_AsyncBIn EQU $0006abfa ROM_AsyncBOut EQU $0006AC12 ; Control patch equates for bypass driver backToControl EQU $6b02c ; for fixing control calls ROM_CtlGood EQU $6b062 ROM_CtlExit EQU $6b064 FinishCall9 EQU $6b0c0 InitSCC EQU $6aea8 ; Close patch equates for bypass driver SyncOutput EQU $6af86 InitSCC1 EQU $6aeae backToClose EQU $6af64 ResetData EQU $6aefe ResetLth EQU $10 ; InitSCC patch equates for bypass driver backToInitSCC EQU $6aeb8 initData EQU $6ae74 ; Status patch equates for bypass driver backToStatus EQU $6afde ; Interrupt Handler equates for the bypass driver ; External/Status interrupt handler equates ToContOut1 EQU $0006B29C ; return to finish up interrupt routines in ROM ToRdReqDone EQU $0006B4B6 ToCtlSet EQU $0006B14C ; Rx interrupt handler equates ROM_PutInOurBuf EQU $6b400 ROM_toStash EQU $6b312 ROM_GoodFinish EQU $6b234 ; SC interrupt handler equates ROM_abortReq EQU $6b4ae ; PIC driver ROM entrypoint addresses ; (note some are here for documentation purposes only) ROM_PICAInOpen EQU $0006B60A ; .AIn ROM_PICAInPrime EQU $0006BA7C ROM_PICAInControl EQU $0006B870 ROM_PICAInStatus EQU $0006B7B8 ROM_PICAInClose EQU $0006B7AC ROM_PICAOutOpen EQU $0006B618 ; .Aout ROM_PICAOutPrime EQU $0006B9DC ROM_PICAOutControl EQU $0006B868 ROM_PICAOutStatus EQU $0006B7B0 ROM_PICAOutClose EQU $0006B72C ROM_PICBInOpen EQU $0006B60A ; .BIn ROM_PICBInPrime EQU $0006BA82 ROM_PICBInControl EQU $0006B876 ROM_PICBInStatus EQU $0006B7BE ROM_PICBInClose EQU $0006B7AC ROM_PICBOutOpen EQU $0006B648 ; .BOut ROM_PICBOutPrime EQU $0006B9DC ROM_PICBOutControl EQU $0006B868 ROM_PICBOutStatus EQU $0006B7B0 ROM_PICBoutClose EQU $0006B72C ; equates for PIC patches IntoChkAConfig EQU $0006AC8A ; for the Open patches IntoChkBConfig EQU $0006AC4E ROM_BufferShuffle EQU $0006BD1E ; for the Input Prime patches backToPICControl EQU $6b898 ; for Control patches backToPICKillIO EQU $6b884 ROM_PICCtlExit EQU $6b8d4 backToPICClose EQU $6b764 ; for Close patch StatusErrExit EQU $6b85a ; for Status patch BacktoPICStatus EQU $6b7d8 PICStatusOkExit EQU $6b858 ;_______________________________________; ; ; IOP Driver Headers ; ;_______________________________________; DC.b 'wong' ; our personal signature PICAsyncAIn DC.W $4D00 ; read, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W PICAInOpen-PICAsyncAIn ; Initialization routine DC.W PICAInPrime-PICAsyncAIn ; input Prime routine DC.W PICAInControl-PICAsyncAIn ; shared Control routine DC.W PICAInStatus-PICAsyncAIn ; shared Status routine DC.W PICAInClose-PICAsyncAIn ; Close routine DC.B 4 ; channel A input driver DC.B '.AIn ' DC.b 'wong' ; our personal signature PICAsyncAOut DC.W $4E00 ; write, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W PICAOutOpen-PICAsyncAOut ; Initialization routine DC.W PICAOutPrime-PICAsyncAOut ; output Prime routine DC.W PICAOutControl-PICAsyncAOut ; Control routine DC.W PICAOutStatus-PICAsyncAOut ; Status routine DC.W PICAOutClose-PICAsyncAOut ; Close routine DC.B 5 ; channel A output driver DC.B '.AOut' DC.b 'wong' ; our personal signature PICAsyncBIn DC.W $4D00 ; read, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W PICBInOpen-PICAsyncBIn ; Initialization routine DC.W PICBInPrime-PICAsyncBIn ; input Prime routine DC.W PICBInControl-PICAsyncBIn ; Control routine DC.W PICBInStatus-PICAsyncBIn ; Status routine DC.W PICBInClose-PICAsyncBIn ; Close routine DC.B 4 ; channel B input driver DC.B '.BIn ' DC.b 'wong' ; our personal signature PICAsyncBOut DC.W $4E00 ; write, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W PICBOutOpen-PICAsyncBOut ; Initialization routine DC.W PICBOutPrime-PICAsyncBOut ; output Prime routine DC.W PICBOutControl-PICAsyncBOut ; Control routine DC.W PICBOutStatus-PICAsyncBOut ; Status routine DC.W PICBOutClose-PICAsyncBOut ; Close routine DC.B 5 ; channel B output driver DC.B '.BOut' ;_______________________________________; ; ; IOP Driver Stubs -- routines that are ; just along for the ride, and jump back ; directly to ROM ; ;_______________________________________; PICAInClose jmpROM ROM_PICAInClose ; input Close routine ;__________________________________ PICAOutPrime jmpROM ROM_PICAOutPrime ; output Prime routine ;__________________________________ PICBInClose jmpROM ROM_PICBInClose ; input Close routine ;__________________________________ PICBOutPrime jmpROM ROM_PICBoutPrime ; output Prime routine ;__________________________________ ; ; Patched PIC Routines ; ;__________________________________ ; Open Routines ; We patch out the Open routines because if we jump into ROM Open ; we'll stuff the ROM PIC driver into the DCE instead of this patched one. ; So instead we load up the value we want in the DCE ourselves, and ; then jump into the open patch for the Bypass driver below. ; Also note for the Open calls, that because of the way PICChkXXXDrvr ; is written in ROM, we must the SAME instruction exactly 4 bytes ; into our Open patch as is in ROM. Else PICChkXXXDrvr would try ; to jump back to some weird place it wasn't meant to. PICAInOpen jmp @10 ; need 4 bytes here to correspond to ROM driver jmpROM ROM_PICAInOpen ; finish up in ROM @10 leaROM ROM_AsyncAIn,a2 ; load up the inputs lea PICAsyncAIn,a3 toAOpenPatch leaROM IntoChkAConfig,a0 ; continue on in ChkAConfig moveq #4,d1 ; use hyigh 4 bits in SPConfig moveq #8,d2 ; and PortAUse bra.w OpnFx1 ; go to our byPass driver patch PICAOutOpen jmp @10 ; need 4 bytes here to correspond to ROM driver jmpROM ROM_PICAOutOpen ; finish up in ROM @10 leaROM ROM_AsyncAOut,a2 ; load up the inputs lea PICAsyncAOut,a3 bra.s toAOpenPatch PICBInOpen jmp @10 ; need 4 bytes here to correspond to ROM driver jmpROM ROM_PICBInOpen ; finish up in ROM @10 leaROM ROM_AsyncBIn,a2 ; load up the inputs lea PICAsyncBIn,a3 toBOpenPatch leaROM IntoChkBConfig,a0 ; continue on in ChkBConfig bra.w OpnFx1 PICBOutOpen jmp @10 ; need 4 bytes here to correspond to ROM driver jmpROM ROM_PICBOutOpen ; finish up in ROM @10 leaROM ROM_AsyncBOut,a2 ; load up the inputs lea PICAsyncBOut,a3 bra.s toBOpenPatch ; ; Prime Routines ; ; We patch the input prime routines because they're trashing the error code ; value thru one thread. PICAInPrime move.l SerialVars,d0 ;Port A uses serial vars bra.s InPrime PICBInPrime move.l ExpandMem,a2 ;Port B uses expand mem move.l ExpandMemRec.emSerdVars(a2),d0 InPrime move.l d0,a2 ;a2 = serial storage ptr (assume ok) bgt.s @10 ;output driver is open move.w #ReadErr,d0 move.l JIODone,-(sp) ;output side not open rts @10 clr.l ioActCount(a0) ;get ready for read move.l a1,10(a2) ;save dce showing cmd active into ReadCmdDCE variable moveq.l #NoErr,d0 ;clear out the error code!!! <41> jmpROM ROM_BufferShuffle ;return to the twisted code in ROM ; ; Close Routines ; ; So far we only need to patch control for 7.0 VM to unhold the buffer ; we may have held durning ctl call #9. ; PICAOutClose PICBOutClose with IOPRequestInfo move.l a1,-(sp) ; save dce on stack MOVE.L DCtlStorage(A1),a2 ; a2 = handle to local storage MOVE.L (A2),A2 ; a2=local storage pointer clr.b PICOpenFlag(a2) ;mark driver as closed ;kill the waiters on the event message ; fixed rmv to use same pb. lea EventMsgPB(a2),a0 ; get address of the waiter move.l a0,d0 _StripAddress ; and strip it move.l d0,a0 move.b #irRemoveRcvWaiter,irRequestKind(a0) ;remove a wait for msg _IOPMsgRequest ;remove it ;send a close msg to pic move.b #CloseOpCode,OpenMsg+OpCode(a2) ;close opcode lea OpenMsgPB(a2),a0 ;send msg to pic _IOPMsgRequest @10 tst.b OpenMsgPB+irReqActive(a2) ;is it done yet? bne.s @10 ;no, wait some more ; and finally do what we came here for... jsr UnHoldPICBuf ; unhold the buffer jmpROM backToPICClose ; and go back to the ROM endwith ; ; UnHoldPICBuf -- unholds the the buffer that may have been held ; Called by: control, close ; Inputs: a2 -- ptr to the serial vars UnHoldPICBuf ; if old buffer was held, we better unhold it. it will have been held ; if it was not our LocalBuf, i.e. was not in the system heap ; first time thru this loop InBufPtr is 0. move.l #gestaltVMAttr,d0 ; is VM installed? _Gestalt move.w a0,d0 beq.s @goodEnd ; if not, we're outta here lea PICLocalBuf(a2),a0 ; inBufPtr = LocalBufPtr? move.l a0,d0 _StripAddress ; being careful movea.l PICInBufPtr(a2),a0 ; addr of old buffer cmp.l a0,d0 beq.s @goodEnd ; yes, then don't need to unhold cmpa.l #0,a0 ; inBufPtr = 0? (for first time thru) beq.s @goodEnd ; yes, then don't need to unhold clr.l d0 ; load count.l into a1 <137> move.w PICBufSize(a2),d0 ; <137> move.l d0,a1 ; <137> _UnHoldMemory ; unhold the old buffer rts @goodEnd moveq #0,d0 ; everything a-ok rts ; ; Control Routines ; ; So far we only need to patch control for 7.0 VM to hold the buffer in ; ctl call #9 in memory. ; PICAOutControl PICBOutControl move.l DCtlStorage(a1),a2 ;recover handle from DCE move.l (a2),a2 ;recover pointer bra.s Control_1 PICAInControl ;Requires output driver open move.l SerialVars,d0 ;Port A uses serial Vars bra.s Control PICBInControl ;Requires output driver open move.l ExpandMem,a2 ;Port B uses expand mem area move.l ExpandMemRec.emSerdVars(a2),d0 Control ;a0 = iopb ;a1 = dce ;d0 = address of serial storage (maybe) move.l d0,a2 ;a2 = address of serial storage -- assume it is ok bgt Control_1 ;is ok if not zero or < 0 jmpROM backToPICKillIO ;we don't want to be here unless storage ptr is ok Control_1 move.w CSCode(a0),d1 ; cmpi #9,d1 ; opcode 9? beq.s Ctl9OpCode ; if so, handle here in the patch jmpROM backToPICControl ; if not, let ROM handle it ;------------------------------------------------- ; Ctl9OpCode - install/remove user supplied buffers ; a0 = iopb ; a1 = dce ; a2 = output driver local storage ;------------------------------------------------- Ctl9OpCode movem.l a0-a1,-(sp) ; save out PB ptr and DCE ptr clr.w PICBufIndex(a2) ;clear input and output indexes clr.w PICBufOutdex(a2) move.l CSParam(a0),a4 ; nubuf ptr clr.l d1 ; clear out buffer count <135> move.w CSParam+4(a0),d1 ;is nubuf being supplied (cnt) beq.s @10 ;no, use our own ;::::::::::::::::::::::VM STUFF START:::::::::::::::::::::::::: ; hold the new non-local buffer so VM doesn't page it out move.l #gestaltVMAttr,d0 ; is VM installed? not on a mac II w/o MMU _Gestalt move.w a0,d0 beq.s @15 ; if not, skip the hold movea.l a4,a0 ; hold buffer addr movea.l #0,a1 ; clear out count movea.l d1,a1 ; hold buffer count <135> ; since VM won't let us call _holdmemory with interrupts ; masked, we return the error that VM spits at us. ; note that PIC driver does not set interrupt mask at all _HoldMemory ; hold the new buffer in memory tst.w d0 ; did we get our physical buffer? beq.s @15 ; yes, go on bra.s toPicCtlExit ; no, punt with a VM error @10 ;install our own buffers lea PICLocalBuf(a2),a4 move.w #LclBufSize,d1 @15 jsr UnHoldPICBuf ; unhold the old buffer if necessary tst.w d0 ; did we release the old buffer? bne.s toPicCtlExit ; no, punt with VM error ;::::::::::::::::::::::VM STUFF STOP::::::::::::::::::::::::::: move.l a4,PICInBufPtr(a2) ; actually install the buffer move.w d1,PICBufSize(a2) clr.w d0 ;all done toPicCtlExit movem.l (sp)+,a0-a1 ; restore PB ptr and DCE ptr jmpROM ROM_PICCtlExit ; ; Status Routines ; ; We patch the status routines in order to provide a new Version number ; status call (9 and $8000 csCodes). We are adding negative csCodes for ; system required codes (see David Wong). ; PICAOutStatus PICBOutStatus move.l DCtlStorage(a1),a2 ;recover storage ptr from our own DCE move.l (a2),a2 bra.s Status_1 PICAInStatus move.l SerialVars,d0 ;Port A uses serial Vars bra.s Status PICBInStatus move.l ExpandMem,a2 ;Port B uses expand mem area move.l ExpandMemRec.emSerdVars(a2),d0 Status ;a0 = iopb ;a1 = dce ;d0 = address of serial storage (maybe) move.l d0,a2 ;a2 = address of serial storage -- assume it is ok bne Status_1 ;is ok if not zero or < 0 move.w #StatusErr,d0 ;mark as status error jmpROM StatusErrExit ;error output driver not open Status_1 ;process the status call always will complete immediately ;a0 = iopb ;a1 = dce ;a2 = output driver storage MOVE.W CSCode(A0),D1 ; get opcode cmpi.w #9,d1 ; do we care? beq.s @version ; yes cmpi.w #$8000,d1 ; largest negative number csCode for Version beq.s @version ; yes jmpROM BacktoPICStatus @version move.b #SerialVers,CSParam(a0); return the version <135> jmpROM PICStatusOkExit; ; end of patched PIC routines ;_______________________________________ ;_______________________________________ ; ; Bypass Patch Code Entry Point ; (Patched Bypass mode routines here) ; ; Called By: we get here via the async serial patch vector ($a0be) ; SerialPatch movem.l d0/a2-a3,-(SP) ; save reg d0,a2,a3 <PMAB401> move.l 16(SP),D0 ; get D0 = return address <PMAB401> lea AsyncAIn,a2 ; assume we are coming from AIn lea PICAsyncAIn,a3 ; cmpRA fromAInOpen,D0 ; test for ChkAConfig call from AInOpen beq.w OpenFix lea AsyncAOut,a2 ; assume we are coming from Aout lea PICAsyncAOut,a3 ; cmpRA fromAOutOpen,D0 ; test for ChkAConfig call from AOutOpen beq.w OpenFix lea AsyncBIn,a2 ; assume we are coming from BIn lea PICAsyncBIn,a3 ; cmpRA fromBInOpen,D0 ; test for ChkBConfig call from BBInOpen beq.w OpenFix lea AsyncBOut,a2 ; assume we are coming from Bout lea PICAsyncBOut,a3 ; cmpRA fromBOutOpen,D0 ; test for ChkBConfig call from BOutOpen beq.w OpenFix move.l 12(sp),d0 ; one less level indirection for non-open calls cmpRA fromControl,d0 ; to fix bypass control calls beq.w ControlFix cmpRA fromStatus,d0 ; to fix bypass control calls beq.w StatusFix cmpRA fromClose,d0 ; to fix bypass close calls beq.w CloseFix cmpRA fromInitSCC,d0 ; to fix initSCC routine beq.w InitSCCFix movem.l (sp)+,d0/a2-a3 ; restore regs <PMAB401> RTS ; back to ROM ; ; Close fixes-- so far we only need to do it for 7.0 VM to unhold ; the buffer we may have held in control call #9. ; CloseFix ADD.L #16,SP ; pop save reg d0, for some reason the driver saved ; out ptr to globals in a6 move.l a1,-(sp) ; save out the DCE ptr MOVE.L (A6),A2 ; get locals pointer <14Oct85> TST.B CtlOptions(A2) ; leave DTR unchanged? <14Oct85> bmi.s @1 ; <C216/16oct86> bclr #7,XmitBits(A2) ; no, clear the DTR bit <C216/16oct86> bclr #7,WR5AVal(A2) ; <C216/16oct86> @1 bclr #3,XmitBits(A2) ; always clear Tx enable <C216/16oct86> jsrROM SyncOutput ; delay until last char has cleared <14Oct85> ; output buffer LeaROM ResetData,A3 MOVEQ #ResetLth,D1 jsrROM InitSCC1 ; shut down the channel ; now that we've shut down the channel it should be safe to unhold ; the input buffer. we won't check for error since we can't do much ; about it anyway. jsr UnHoldInputBuf ; unhold the buffer if necessary move.l (sp)+, a1 ; restore the DCE ptr jmpROM backToClose ;::::::::::::::::::::::VM STUFF START:::::::::::::::::::::::::: ; ; UnHoldInputBuf -- unholds the the buffer that may have been held ; Called by: control, close ; Inputs: a2 -- ptr to the serial vars ; if old buffer was held, we better unhold it. it will have been held ; if it was not our LocalBuf, i.e. was not in the system heap ; first time thru this loop InBufPtr is 0. UnHoldInputBuf move.l #gestaltVMAttr,d0 ; is VM installed? _Gestalt move.w a0,d0 beq.s @goodEnd ; if not, we're outta here ; check usual case...is inBufPtr equal to LocalBufPtr? move.l d1,-(sp) ; save a reg lea LocalBuf(a2),a0 ;inBufPtr = LocalBufPtr? move.l a0,d1 move.l InBufPtr(a2),d0 eor.l d1,d0 _StripAddress ; ignore differences in the high bits move.l (sp)+,d1 ; restore a reg tst.l d0 ; beq.s @goodEnd ; same, so don't need to unhold ; check first time thru case, when InBufPtr ; would equal zero move.l InBufPtr(a2),d0 ; for first time thru, is InBufPtr zero? beq.s @goodEnd ; yes, then don't need to unhold ; unhold the puppy movea.l d0,a0 ; a0 <- ptr moveq #0,d0 ; clear out count move.w BufSize(a2),d0 ; movea.l d0,a1 ; a1<- cnt _UnHoldMemory ; unhold the old buffer rts @goodEnd moveq #0,d0 ; everything a-ok rts ;::::::::::::::::::::::VM STUFF STOP::::::::::::::::::::::::::: ; ; Routine: InitSCC ; Patch: --We patch this routine to use the InitData table here in RAM ; instead of the one in ROM. We do this so that the values in ; WR11 and WR14 aren't hardcoded, so that we can support external ; clocking. ; --We also fix the value of StopBits (WR4) here to have the clock ; divide bits correspond to the the external/internal clock state ; indicated in CtlOptions. We do this here so that CtlConfig calls ; won't stomp StopBits in the ToSCCInit routine. ; --We also disable HWHS so we don't try to use CTS line for both ; clocking and handshaking. clkBit equ 6 ; bit 6 in CtlOptions controls ext/int SCC clk clkDvdBit equ 6 ; divide clock bit in WR4 clkMask equ %01000000 ; mask for getting at clkBit <129> extClkSrc equ %00101000 ; SCC clk src is TRxC (CTS) pin (WR11) <129> intClkSrc equ %01010000 ; SCC clk src is baud rate generator (WR11) <129> BRGEnbl equ %00000001 ; enable baud rate generator (WR14) <129> BRGDsbl equ %00000000 ; disable baud rate generator (WR14) <129> SCCDataTable DC.B $02,9 ; status in low bits, MIE disabled clkDvd DC.B 4,$FF ; x16 clk, stop bits, parity options DC.B 1,$FF ; WR1 reg, first write DC.B 3,$FF ; bits/char option rcvr DC.B 5,$FF ; bits/char option xmitter DC.B $00,2 ; zero interrupt vector DC.B $00,10 ; NRZ encoding ClkMode DC.B $50,11 ; brgen/TRxC clk to rcvr, xmitter--default to internal DC.B 12,$FF ; set baud rate low byte DC.B 13,$FF ; set baud rate high byte DC.B 3,$FF ; enable rcvr DC.B 5,$FF ; enable xmitter BRGEnable DC.B $01,14 ; enb/disable baud rate generator from RTxC pin --default to on dc.b $A0,15 ; Break, CTS external ints (dcd not needed) <2.3> DC.B $10,0 ; reset ext/status twice DC.B $10,0 DC.B 1,$FF ; w/req pin configuration DC.B $0A,9 ; enable interrupts, status in low bits SCCDataTableLth EQU *-SCCDataTable ; InitSCCFix movem.l (sp)+,d0/a2-a3 ; restore drvr storage ptr in a2, InitSCC table in a3 add.l #4,sp ; pop saved rtrn addr--we jump back move.l a3,d0 cmpRA InitData,d0 ; are we initializing SCC? bne @done lea SCCDataTable,a3 ; use our RAM table instead of the ROM one moveq #SCCDataTableLth,d1 movem.l d1/a0,-(sp) ; save out reg's <129> ; default to internally clocked state <129> bset.b #clkDvdBit,StopBits(a2) ; default to a divide-by-16 clock <129> moveq #intClkSrc,d0 ; internal clocking source <129> moveq #BRGEnbl,d1 ; enable baud rate generator <129> btst.b #clkBit,CtlOptions(a2) ; are we externally clocked? beq.s @load ; not externally clocked, so load ; set to externally clocked state bclr.b #clkDvdBit,StopBits(a2) ; set to a divide-by-one clock <129> moveq #extClkSrc,d0 ; external clock source <129> moveq #BRGDsbl,d1 ; disable baud rate generator <129> clr.b HWHS(a2) ; make sure we're not trying to do HWHS <129> @load lea ClkMode,a0 ; load params into InitSCC data table <129> move.b d0,(a0) ; <129> lea BRGEnable,a0 ; <129> move.b d1,(a0) ; <129> movem.l (sp)+,d1/a0 ; restore reg's <129> @done jmpROM backToInitSCC ; finish up in ROM ; ; Routine: Status ; Patch: We patch status to add a call to return the driver's version. StatusFix movem.l (sp)+,d0/a2-a3 ; restore drvr storage ptr in a2 add.l #4,sp ; pop saved rtrn addr--we jump back MOVE.W IOTrap(A0),-(SP) ; save trap to distinguish immed calls<14Oct85> MOVE.L A1,-(SP) ; save passed DCE for in/out <14Oct85> MOVE.W SR,-(SP) ; disable interrupts for ctl call <14Oct85> ORI #HiIntMask,SR ; <A357/06nov86> LEA CSCode(A0),A0 ; get pointer to return parameters MOVEQ #StatusErr,D0 ; assume status error MOVE.L A2,D1 ; have our variables been set up? <14Oct85> Bgt @stat1 ; exit if not (only input side open) jmpROM ROM_CtlExit ; just like ROM @stat1 MOVE.W (A0)+,D1 ; get opcode cmpi.w #9,d1 ; do we care? beq.s @version ; yes cmpi.w #$8000,d1 ; largest negative number csCode for Version beq.s @version ; yes jmpROM BackToStatus ; no @version move.b #SerialVers,(a0) ; return the version <135> jmpROM ROM_CtlGood ; ; Routine: Control ; Patch: We patch control for the following reasons: ; 1) to add VM support for control call 9--lock/unlock the buffer ; 2) to add external clock support in control call 16 ; ControlFix movem.l (sp)+,d0/a2-a3 ; restore drvr storage ptr in a2 add.l #4,sp ; pop saved rtrn addr--we jump back MOVE.W IOTrap(A0),-(SP) ; save trap to distinguish immed calls<14Oct85> MOVE.L A1,-(SP) ; save passed DCE for in/out <14Oct85> MOVE.W SR,-(SP) ; disable interrupts for ctl call <14Oct85> ORI #HiIntMask,SR ; <A357/06nov86> moveq #0,d0 ; for the killIO call LEA CSCode(A0),A0 ; get parameters MOVE.W (A0)+,D1 ; get opcode cmpi.w #9,d1 ; opcode 9? (buffer) beq CtlBuffer ; if so, handle here in the patch cmpi.w #16,d1 ; opcode 16? (ctlOptions) <extClk> beq CtlSwitchCTSClock ; <extClk> jmpROM backToControl ; if not, let ROM handle it ;_______________________________________; ; ; Install a new input buffer. ; ; The patch is needed for 7.0 VM. We must 'hold' the buffer in memory ; so that there is no possibility it is paged out at interrupt time (which ; would have the potential for causing a double page fault). ;_______________________________________; CtlBuffer CLR.L BufIndex(A2) ; clear in and out indices MOVE.L (A0)+,A4 ; buffer pointer moveq #0,d1 ; clear out length reg MOVE.W (A0),D1 ; length of buffer? BEQ.S InstllLBuf ; if zero, revert to our own buffer ; hold the new non-local buffer so VM doesn't page it out move.l #gestaltVMAttr,d0 ; is VM installed? not on a mac II w/o MMU _Gestalt move.w a0,d0 beq.s InstllABuf ; if not, skip the hold movea.l a4,a0 ; hold buffer addr movea.l d1,a1 ; hold buffer count ; since VM won't let us call _holdmemory with interrupts ; masked, we set the interrupt level back to that app's ; interrupt level, which hopefully was zero. If so, then ; everything should be peachy. If not, then we return the ; error that VM spits at us. move.w sr,-(sp) ; save our int lvl move.w 2(sp),d0 ; get app's int lvl and.w (sp),d0 ; and set us to that move.w d0,sr _HoldMemory ; hold the new buffer in memory move.w (sp)+,sr ; and restore our int lvl tst.w d0 ; did we get our physical buffer? beq.s InstllABuf ; yes, go on Ctl_Err jmpROM ROM_CtlExit ; no, then pass back VM error ; InstllLBuf is also called from OpenInstall! InstllLBuf LEA LocalBuf(A2),A4 ; use our meager local buffer for now MOVEQ #LclBufSize,D1 ; InstllABuf ; since VM won't let us call _holdmemory with interrupts ; masked, we set the interrupt level back to that app's ; interrupt level, which hopefully was zero. If so, then ; everything should be peachy. If not, then we return the ; error that VM spits at us. move.w sr,-(sp) ; save our int lvl move.w 2(sp),d0 ; get app's int lvl and.w (sp),d0 ; and set us to that move.w d0,sr jsr UnHoldInputBuf ; unhold the buffer if necessary move.w (sp)+,sr ; restore our int lvl tst.w d0 ; did we release old buffer? bne.s Ctl_Err ; no, pass back VM error jsrROM FinishCall9 ; and finish up the call in ROM jmpROM ROM_CtlGood ; and go back to ctl call dispatcher ;::::::::::::::::::::::VM STUFF STOP::::::::::::::::::::::::::: ; Routine: SetCtlOptions -- Opcode 16 ; Patch: We patch this routine so that bit 6 of CtlOptions variable now ; controls a switch to internal/external clocking on the CTS (HSIn) line. ; We must call InitSCC to get the clocking option to kick in. ; Inputs: byte number in IOPB value ; (26) [$0010] opcode ; (28) bit 7 = 0 for drop DTR at close ; bit 7 = 1 for leave DTR unchanged at close ; bit 6 = 0 for internal clocking ; bit 6 = 1 for external clocking ; bits 0-5 reserved for future use ; Notes: We will not put control for the GPI internal/external ; clocking switch here, as that requires the control of ; HW external to the SCC (namely, the VIA vSync pin). Control ; of the GPIa line is in _HWPriv. ; CtlSwitchCTSClock move.b (a0),CtlOptions(a2) ; store new value <129> jsrROM InitSCC ; set up the SCC according to new value <129> jmpROM ROM_CtlGood ; finish up in ROM <129> ; ; Open Fixes ; ; ; note that no stack fiddling is allowed, as the chkConfigure EXPECTS to be called directly ; from the open routine, and will not go to IOCore with error properly should one occur. ; Common code for open patches OpenFix move.l d0,a4 ; save where we came from w/o use of the stack ADD.L #12,SP ; pop save reg d0,a2,a3 - not needed <PMAB401> MOVE.L (SP)+,A0 ; pop A0 = addr to continue ChkConfig addq #4,SP ; pop async open ret addr - will jmp back OpnFx1 jsr (a0) ; finish ChkConfig (A1,D1,D2 params) ; check to see whether we are in IOP or Bypass mode if isUniversal then ; <2.1> TestFor SCCIOPExists ;check to see if we even have an IOP <2.1> beq.s @05 ;if not, just use bypass mode driver <2.1> endif ; <2.1> btst #0,SCCIOPFlag ;are we in bypass mode? <8.7> test low mem, not PRAM bne.s @05 ;yes, in bypass mode; use SCC driver <8.7> change bit sense move.l a3,a2 ; plug in PIC driver header address @goback jmpROM PlugInDrvr ; go finish open call ; install our RAM-based bypass driver header into DCE <125> ; a2 has pointer to RAM driver header ; a1 has pointer to DCE @05 move.l a2,(a1) ; install ourselves into the DCE ; check again where we came from to see if which open fix to branch off to. move.l a4,d0 ; restore where we came from cmpRA fromAOutOpen,d0 ; test for ChkAConfig call from AOutOpen beq.s FixAout cmpRA fromBOutOpen,d0 ; test for ChkAConfig call from AOutOpen beq.s FixBout jmpROM backToInOpen ; for In driver just go finish open call <125> FixAout ; replace receive routine vectors installed by async open LEA PortAVars,A2 ; local variables address peaROM ROM_PollDtaIn,-(sp) ; ROM--this proc handles disk poll data <20Oct85> pea NewSCAIntHnd ; Special RxD int handler <4> <20Oct85> pea NewRAIntHnd ; int handler <4> <20Oct85> peaROM ROM_TAIntHnd,-(sp) ; ROM--TxD int handler <20Oct85> PEA Lvl2DT+16 ; SCC interrupt dispatch table, chan A <20Oct85> PEA NewExtAIntHnd ; External int handler <20Oct85> jmpROM backToAOutOpen ; JUMP BACK INTO ROM FixBout ; replace receive routine vectors installed by async open move.l ExpandMem,a2 ; local variable address in is <2.0> lea ExpandMemRec.emSerdVars(a2),a2 ; expandmem <2.0> CLR.L -(SP) ; no disk poll routine pea NewSCBIntHnd ; Special RxD int handler <4> <20Oct85> pea NewRBIntHnd ; RxD int handler <4> <20Oct85> peaROM ROM_TBIntHnd,-(sp) ; ROM--TxD int handler <20Oct85> PEA Lvl2DT ; SCC interrupt dispatch table, chan B <20Oct85> PEA NewExtBIntHnd ; External int handler <20Oct85> jmpROM backToBOutOpen ; JUMP BACK INTO ROM ;----------------------------------------------------------------------- ; ; Interrupt Handlers ; ;----------------------------------------------------------------------- ;________________________________________________________________________ ; ; Routine: RXIntHnd ; ; Arguments: A0 (input) -- chan A/B control read address ; A1 (input) -- chan A/B control write address ; ; Function: This routine handles SCC receiver interrupts for ; both channels; the data is read and stashed, IODone called ; if necessary. ; ; Patches: Patched to fix the serial thruput problem on Apollo and TimLC. <4> ; If we complete a device manager request we change the return ; address on the stack to return to the primary ; level 4 interrupt dispatcher, rather than to the code in the ; secondary dispatcher that checks to see if there is another ; RxChar available (and loops to process the next interrupt if so). ; This way we guarantee that we return all the way out of the interrupt ; handlers at least once per device manager request, allowing ; things like deferred tasks a chance to run. ;________________________________________________________________________ Lvl4IntPatchSize equ 10 ; size of interrupt patch code NewRBIntHnd move.l ExpandMem,a3 ; get appropriate variables (chan B) <2.0> lea ExpandMemRec.emSerdVars(a3),a3 ; <2.0> BRA.S RXIntHnd ; go to shared code NewRAIntHnd LEA PortAVars,A3 ; get appropriate variables (chan A) RXIntHnd MOVE.B SCCData(A0),D0 ; get the data byte PollStash MOVE.L (A3)+,A2 ; get pointer to local variables move.l InDCE(a2),a3 ; and DCE pointer <2.0> AND.B CharMask(A2),D0 ; zero unused high bits <14Oct85> MOVE.B PEChar(A2),D1 ; Are we translating PE Characters? BEQ.S @1 ; No, just save the char. CMP.B D1,D0 ; Is the new char a PEChar? BNE.S @1 ; No, just do normal stuff MOVE.B AltChar(A2),D0 ; Make sure that a good char <14Oct85> ; doesn't look like PEChar. @1 TST.B SWHS(A2) ; software handshake enabled? BEQ.S StashIt ; branch if not CMP.B XONChar(A2),D0 ; was this an XON? <14Oct85> bne.s chkXoff ; not an XOn <4> ContOut CLR.B XOFFlag(A2) ; we got an XON, so clear Xoff-received flag <4> jmpROM ToContOut1 ; XOn, so start output again <4> chkXoff CMP.B XOFFChar(A2),D0 ; how about an XOFF? <14Oct85><4> BNE.S StashIt ; if not, then stash the character ST XOFFlag(A2) ; if so, then note it InputRTS rts ; and exit <4> ; stash byte in the user's buffer if a request is pending, otherwise use our own StashIt TST.B ReadCmd(A2) ; read request pending? bne.s PutInUserBuf ; request pending, so stash byte in user buffer <4> jmpROM ROM_PutInOurBuf ; no request pending, so stash byte in our buffer <4> PutInUserBuf MOVE.L A3,A1 ; get DCE pointer jsrROM ROM_toStash ; use utility routine to save code <4> BPL.S InputRTS ; if request isn't finished, just RTS CLR.B ReadCmd(A2) ; no longer a read request pending add.l #Lvl4IntPatchSize,(sp); return to rts at end of int dispatcher <patch><4> ; so we return out of Interrupt handler completely ; since we have finished a request jmpROM ROM_GoodFinish ; we have a good finish ;________________________________________________________________________ ; ; Routine: SCIntHnd ; ; Arguments: A0 (input) -- channel A/B control read address ; A1 (input) -- channel A/B control write address ; ; Function: This routine handles SCC special condition interrupts: ; these occur when an input character is received that has ; a parity error, framing error, or causes an overrun. ; If the option is set to abort on the error, the character ; is discarded and the input request (if any) aborted; otherwise, ; the error is noted and the character buffered as usual. ; ; Patch: Must patch this rtn because we patch the RxIntHnd which <4> ; includes the 'Stashit' label called here. ; ; If there are no special condition bits set, then this is NOT <5> ; a special-condition interrupt, but an interrupt for which ; the cause has already been removed. This occured with the ; Stylewriter on faster machines, when we wrote out a byte ; faster than the TBE interrupt could be generated (after ; the TBE bit was set in RR0). ;________________________________________________________________________ NewSCBIntHnd move.l ExpandMem,a3 ; get appropriate variables (chan B) <2.0> lea ExpandMemRec.emSerdVars(a3),a3 ; <2.0> BRA.S SCIntHnd ; go to shared code NewSCAIntHnd LEA PortAVars,A3 ; get appropriate variables (chan A) SCIntHnd MOVE.B #1,(A1) ; point to error reg MOVE.L (A3)+,A2 ; get local variables pointer move.l InDCE(a2),A3 ; and DCE pointer (delay, too) <2.0> MOVE.B (A0),D1 ; read the error condition MOVEQ #$70,D3 ; form $70 mask AND.B D3,D1 ; isolate error bits beq.s InputRTS ; no errors posted, then this is not a SC interrupt <5> OR.B D1,AsyncErr(A2) ; accumulate errors (delay, too) MOVE.B SCCData(A0),D0 ; get the data byte AND.B CharMask(A2),D0 ; zero unused high-order bits <14Oct85> MOVE.B Options(A2),D2 ; get abort options AND.B D1,D2 ; should we abort? MOVE.B #$30,(A1) ; reset the error flag AND.B D3,D2 beq.s @dontAbort ; options say we should NOT abort request <4> jmpROM ROM_abortReq ; options say we should abort request <4> @dontAbort MOVE.B PEChar(A2),D3 ; alternate char for parity errors? <4> BEQ.S @1 ; br if not CMP.B D0,D3 ; Is the incoming char equal to the PEChar? BNE.S @0 ; No, no substitution needed. MOVE.B AltChar(A2),D0 ; Make sure that a good char <14Oct85> ; doesn't look like PEChar. @0 BTST #4,D1 ; parity error? BEQ.S @1 ; br if not MOVE.B D3,D0 ; replace it @1 bra.s StashIt ; go stash it . . .<StashIt is IN RAM NOW> <4> ;________________________________________________________________________ ; ; Routine: ExtIntHnd ; ; Arguments: A0 (input) -- channel A/B control read address ; A1 (input) -- channel A/B control write address ; D0 (input) -- SCC read reg 0 value ; D1 (input) -- SCC read reg 0 changed bits ; ; Function: This routine handles SCC external/status interrupts for ; both channels; mouse (DCD) interrupts are passed along to the ; mouse interrupt handler in CrsrCore. Only Break/Abort and CTS ; external interrupts are enabled (besides DCD). ; ; Note that CTS low in read reg 0 currently means that the ; hardware handshake line is asserted which means 'ok to transmit'. ;________________________________________________________________________ NewExtBIntHnd move.l ExpandMem,a3 ; get appropriate variables - chan B <2.0> move.l ExpandMemRec.emSerdVars(a3),a2 ; <2.0> move.w #28,d2 ; .Bin is offset 28 into unit table <126> bra.s ExtIntHnd NewExtAIntHnd LEA PortAVars,A3 ; get appropriate variables - chan A MOVE.L (A3)+,A2 ; get pointer to local variables move.w #20,d2 ; Ain is offset 20 in the unit table <126> ExtIntHnd movea.l UTableBase,a3 ; get addr of Utable <126> movea.l 0(a3,d2.w),a3 ; get DCE handle (d2 has offset) <126> movea.l (a3),a3 ; get DCE ptr <126> MOVE.B D1,D2 ; changed bits AND.B postOptions(A2),D2 ; post this change? BEQ.S @0 ; br if not MOVEM.L D0/A0,-(SP) ; preserve these registers MOVE.W #IODrvrEvt,A0 ASL.W #8,D0 ; make room for 'changed' values MOVE.B D1,D0 SWAP D0 ; make room for driver refnum MOVE.W DCtlRefnum(A3),D0 _PostEvent ; and post the event MOVEM.L (SP)+,D0/A0 @0 TST.B D1 ; see if it's a change in break status BMI.S extBreak ; branch if it was a break interrupt LSL.B #2,D0 ; must be CTS change SMI CTSFlag(A2) ; set flags according to CTS ; This piece of code is used to detect a clock into the HWHS line ; and shut off the ext/sts interrupt for the handshake line. CMP.W #80,HSCount(A2) ; exceeded 80 transitions in 16 MS? <14Oct85> BCS.S @2 ; br if not <14Oct85> MOVEQ #-128,D0 ; ($80) leave break ints enabled <14Oct85> MOVEQ #15,D1 ; write register 15 <14Oct85> jsrROM ToCtlSet ; <patch> @2 MOVE.L Ticks,D2 ; get current tick time <14Oct85> CMP.L LastTime(A2),D2 ; same as last? <14Oct85> BEQ.S @3 ; br if so <14Oct85> MOVE.L D2,LastTime(A2) ; new last time <14Oct85> CLR.W HSCount(A2) ; restart count for new time <14Oct85> @3 ADDQ #1,HSCount(A2) ; update count <14Oct85> jmpROM toContOut1 ;<patch> BRA ContOut1--if freshly asserted, continue output<14Oct85> extBreak TST.B D0 ; check break level BMI.S @1 ; if it's asserted, terminate any input MOVE.B SCCData(A0),D0 ; otherwise (end of break), discard null @0 RTS ; and return @1 MOVEQ #BreakRecd,D0 ; note the break bset.b #3,AsyncErr(a2) ; we now note break level in status TST.B ReadCmd(A2) ; read request pending? beq.s @0 ; just return if done <7.1> jmpROM ToRdReqDone ; return to ROM code ;_______________________________________________________________________ ; ; Patched Driver headers (with signature long word before each header) ; DC.b 'wong' ; our personal signature AsyncAIn DC.W $4D00 ; read, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W AInOpen-AsyncAIn ; Initialization routine DC.W AInPrime-AsyncAIn ; input Prime routine DC.W AControl-AsyncAIn ; shared Control routine DC.W AStatus-AsyncAIn ; shared Status routine DC.W AInClose-AsyncAIn ; Close routine DC.B 4 ; channel A input driver DC.B '.AIn ' DC.b 'wong' ; our personal signature AsyncAOut DC.W $4E00 ; write, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W AOutOpen-AsyncAOut ; Initialization routine DC.W AOutPrime-AsyncAOut ; output Prime routine DC.W AControl-AsyncAOut ; shared Control routine DC.W AStatus-AsyncAOut ; shared Status routine DC.W AOutClose-AsyncAOut ; Close routine DC.B 5 ; channel A output driver DC.B '.AOut' DC.b 'wong' ; our personal signature AsyncBIn DC.W $4D00 ; read, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W BInOpen-AsyncBIn ; Initialization routine DC.W BInPrime-AsyncBIn ; input Prime routine DC.W BControl-AsyncBIn ; shared Control routine DC.W BStatus-AsyncBIn ; shared Status routine DC.W BInClose-AsyncBIn ; Close routine DC.B 4 ; channel B input driver DC.B '.BIn ' DC.b 'wong' ; our personal signature AsyncBOut DC.W $4E00 ; write, control, status, lock DC.W 0,0 ; not an ornament DC.W 0 ; no menu DC.W BOutOpen-AsyncBOut ; Initialization routine DC.W BOutPrime-AsyncBOut ; output Prime routine DC.W BControl-AsyncBOut ; shared Control routine DC.W BStatus-AsyncBOut ; shared Status routine DC.W BOutClose-AsyncBOut ; Close routine DC.B 5 ; channel B output driver DC.B '.BOut' ;_______________________________________________________________________ ; ; jumping to ROM from our patched entry points ; AInOpen jmpROM ROM_AInEntryOpen ;port a <125> AInClose jmpROM ROM_AInEntryClose AInPrime jmpROM ROM_AInEntryPrime AOutOpen jmpROM ROM_AOutEntryOpen AOutClose jmpROM ROM_AOutEntryClose AOutPrime jmpROM ROM_AOutEntryPrime AControl jmpROM ROM_AEntryControl AStatus jmpROM ROM_AEntryStatus BInOpen jmpROM ROM_BInEntryOpen ;port b BInClose jmpROM ROM_BInEntryClose BInPrime jmpROM ROM_BInEntryPrime BOutOpen jmpROM ROM_BOutEntryOpen BOutClose jmpROM ROM_BOutEntryClose BOutPrime jmpROM ROM_BOutEntryPrime BControl jmpROM ROM_BEntryControl BStatus jmpROM ROM_BEntryStatus endproc ;_______________________________________________________________________ ; <7.1><8.7><81><123><125> Async Serial Driver Patch ; ; End of patch code for fix of bugs in the serial driver ; introduced into the aurora ROM (and hence the F19 overpatch ROM) by the ; port arbitration code, which was accidently included in those ROMS. ;_______________________________________________________________________ ;_______________________________________________________________________ ;________________________________________________________________________________ ; <8.8> RecoverHandle() patch for 32-bit Memory Manager ; ; This patch fixes RecoverHandle so that if the passed block pointer isn't in ; either the system or app heaps, it makes a last-ditch pass and sees if the ; pointer points to a ROM resource. If so, great, else it bails with an error. ; This is especially useful since several new AppleTalk resources are in ROM ; and their users have a tendency to do RecoverHandles on them because they ; don't know any better (and shouldn't have to). ; ; No traps are patched. These routines are patched into the 32-bit Memory Manager's ; vector tables in lo-mem at JMemMgr24 ($1E00) and JMemMgr32 ($1F00). ;________________________________________________________________________________ PROC EXPORT v24RecoverHandle,v32RecoverHandle ;________________________________________________________________________________ ; ; Routine: v24RecoverHandle,v32RecoverHandle ; ; Inputs: A0 -- pointer to block ; A6 -- pointer to current heap zone ; ; Outputs: A0 -- handle to block ; D0 -- unchanged from input ; MemErr contains the error code ; ; Function: Given a pointer to an alleged heap block, it returns the block's handle. ;________________________________________________________________________________ v24RecoverHandle ; 24 bit version of RecoverHandle MOVE.L Lo3Bytes,D2 ; mask to strip extraneous bits BRA.S vRecoverHandle v32RecoverHandle ; 32 bit version of RecoverHandle MOVEQ #-1,D2 ; no extraneous address bits in 32-bit mode vRecoverHandle Move.L BusErrVct,A2 ; save current bus error handler LEA RHBusErrHandler,A3 ; our ZC version Move.L A3,BusErrVct Move.L -4(A0),A1 ; Handle, or ROMBase on bus error Move.L A2,BusErrVct ;restore old one after danger Add.L A6,A1 ; The following code tests for bad mp, but can only set MemErr Move.L A0,D1 ; original pointer And.L D2,D1 ; stripped to RAW address AND.L (A1),D2 ; master pointer value stripped to RAW address Sub.L D1,D2 ; if equal, have required error code! Beq.S RecoverExit ; If we get here, the pointer either didn't point to a block in the suspected ; heap or is a bogus block pointer. We'll assume it's the first case and try ; looking in the ROZ created for the ROM resource map before giving up completely. ; This will not affect the performance of RecoverHandle() since this code is only ; executed as part of the error path (which shouldn't be taken most of the time). MOVEA.L A6,A3 ; save the previous zone start pointer MOVEA.L RomMapHndl,A6 ; get the handle to the ROM resource map MOVEA.L (A6),A2 ; and de-reference it SUBA.L -4(A2),A6 ; calculate heap start based on rel hdl in MP CMPA.L A3,A6 ; have we been here before? BNE.S v32RecoverHandle ; -> no, try again with the ROZ MoveQ #memBCErr,D2 ; block check failure RecoverExit Move.W D2,MemErr ; stuff global error code (is anybody out there?) MoveA.L A1,A0 ; restore A0=handle Move.L vMMNoErrEpilogue,-(SP) ; get MMNoErrEpilogue vector RTS ;________________________________________________________________________________ ; ; RHBusErr -- bus error handler for RecoverHandle ; A bus error in RecoverHandle causes ROMBase to be jammed for ; the relative handle. ; Clear DF bit in frame to inhibit retry of failed read from RAM, and force ; ROMBase into result, i.e. A1. ; Input: none ; Output: potentially forces ROMBase to result field (ultimately A1) ; Regs: A3 ; Triggered by bus errors in Move.L -4(A0),A1 in RecoverHandle. ;________________________________________________________________________________ RHBusErrHandler AndI.W #$FEFF,$A(SP) ;clear DF bits Move.L ROMBase,$2C(SP) ;fake Nil value in DIB RtE ENDPROC ;________________________________________________________________________________ ; ; <8.8> End of RecoverHandle() patch ;________________________________________________________________________________ ;________________________________________________________________________________ ; RMP <12> ; ; SetHandleSize() patch for 32-bit Memory Manager ; ; This patch fixes A32SetSize not to de-reference off from a nil handle (i.e. A1) ; when it is called by v32SetHandleSize. ; ; No traps are patched. These routines are patched into the 32-bit Memory Manager's ; vector tables in lo-mem at JMemMgr32+jsetHandleSize ($1F64). ;________________________________________________________________________________ PROC EXPORT v32SetHandleSize IMPORT a32SetSize ;---------------------------------------------------------------------- ; ; FUNCTION SetHandleSize(h: Handle; cb: LongInt): result; ; ; This routine is used to change the size of a relocatable memory block. ; The block handle remains the same; the Master Pointer is unchanged if ; the new length is less than or equal to the old. ; ; Arguments: ; D0 - cb: new block size ; A0 - handle: pointer to master pointer ; ; Result: ; D0 - 0 => success; memFullErr => failure ; ; Registers: ; ; A0 - next free block ; A1 - handle ; v32SetHandleSize BNE.S SHSOk SHSNil MoveQ #nilHandleErr,D0 ;return error code BrA.S SHSExit SHSOk BSR.S a32SetSize ;Set size as specified in D0 SHSExit Move.L A1,A0 ;restore A0 for RecoverHandle Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9> RTS ; jump to it <v1.9> ENDPROC ;________________________________________________________________________________ ; ; <12> End of SetHandleSize() patch ;________________________________________________________________________________ ;________________________________________________________________________________ ; <68> <85> ; ; SetPtrSize() patch for 32-bit Memory Manager ; ; This patch fixes A32SetSize not to de-reference off from a nil handle (i.e. A1) ; when it is called by v32SetPtrSize. ; ; No traps are patched. These routines are patched into the 32-bit Memory Manager's ; vector tables in lo-mem at JMemMgr32+jsetptrSize ($1F4C). ;________________________________________________________________________________ PROC EXPORT v32SetPtrSize,a32SetSize ; addresses in ROM a32ActualS EQU $E5A2 a32GetSize EQU $E5D2 HBlockMove EQU $E334 a32MakeSpace EQU $EB40 ClearGZStuff EQU $E5F2 SubtractFree EQU $E92A A32MakeBkF EQU $DDF4 AdjustFree EQU $E91C A32SetLogSize EQU $E2A0 a32RelocRel EQU $EC66 a32SetSizePatch EQU $87996 SmartSetSize EQU 1 ;---------------------------------------------------------------------- ; ; FUNCTION SetPtrSize(blk: Ptr; cb: Size): result; ; ; This routine is used to set the size of non-relocatable memory blocks ; ; Arguments: ; D0 - cb: new block size ; A0 - blk: pointer to block data ; A1 - h: NIL, set up by MMPPrologue ; ; Result: ; D0 - 0 => success; memFullErr => failure ; ; Registers: ; ; A0 - next free block ; A1 - current free block ; A2 - theZone ; D0 - number of bytes needed ; D1 - number of bytes in current free block ; D2 - new block length (temporary) ; v32SetPtrSize BSR.S a32SetSize ; set its size <v1.1> Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9> RTS ; jump to it <v1.9> ; ----- 32 bit version ----- a32SetSize MoveM.L D1-D6/A0-A3,-(SP) ;Save registers. <18Apr85> Move.L A0,A2 ;Points to block text. <v1.4> Move.B MPtag32-BlkData32(A2),-(SP) ;save flags of block <v1.1> Move.L D0,D1 ;Save desired logical size. JSRROM a32ActualS ; Move.L D0,D3 ;New physical size. JSRROM.S a32GetSize ;Get current logical size in D0. Cmp.L D0,D1 ;sNew = sCur? BEq @SSEqual ;Yes, return to caller. <121> Move.L A0,A2 ;Points to block text. Sub.L #BlkData32,A2 ;Point to block header. <v1.1> Move.L BlkSize32(A2),D2 ;Get physical size. <v1.1> Move.L D2,D4 ;Old physical size. Cmp.L D2,D3 ;sNAct = sCAct? BEq @SSLogical ;Yes, just set logical size. BCS @SSShrink ;sNAct < sCAct, shrink the block. ******************************** Start conditional block. ******************************** <18Apr85> IF SmartSetSize THEN ; The block must be grown. <18Apr85> ; Registers: D0=D3=new phys size; D1=new log size; D2=D4=phys size; D5=?? <18Apr85> ; A0=text ptr; A1=handle; A2=block ptr; A3=?? <18Apr85> ; Across the code use registers: <18Apr85> ; D0=tmp; D1=new log size; D2=D4=phys size; D3=new phys size; D5=delta size <18Apr85> ; A0=tmp; A1=handle; A2=block ptr; A3=tmp <18Apr85> ; <18Apr85> ; First, set up some registers and the GZ's. <18Apr85> Move.L D3,D5 Sub.L D2,D5 ;delta physical size Move.L A1,GZRootHnd ;record handle being sized Move.L A0,GZRootPtr ;record pointer " " ; In this loop, A3 steps across the blocks and D6 accumulates free space. <18Apr85> MoveA.L A2,A3 AddA.L D4,A3 ;next block after growing MoveQ #0,D6 ;initial accumulation @SSAftFree CmpA.L bkLim(A6),A3 ;have we reached the heap end? BCC.S @SSAFEnd ;Carry Clear => beyond the end Tst.B (A3) ;Eq => free block BNE.S @SSAFEnd Add.L BlkSize32(A3),D6 ;accumulate free space <v1.1> AddA.L BlkSize32(A3),A3 ;bump on to next block <v1.1> BrA.S @SSAftFree ;and continue looping @SSAFEnd ; Detail: be sure to go on if exactly enough space is there -- for no sliding will be done! <18Apr85> Sub.L D6,D5 ;D5 := amount unrequited by free space at end BLS @SSAsUsual ;Lower or Same => enough space for easy growth! ; Provided the block is unlocked, relocatable, try coalescing free space from below, remembering <18Apr85> ; the 'last' block to give a backward peek when A2 is reached. The logic follows BFindS farily <18Apr85> ; closely. Starting at the heap's beginning, outer loop skips over allocated blocks (watching <18Apr85> ; for the one being grown); when a free block is hit, the inner loop starts coalescing until <18Apr85> ; allocated block (or growing one) is found. The loop terminates with A0 pointing to the last <18Apr85> ; free block found, and D0 equal to the amount of free space immmediately before the growing block. <18Apr85> Move.L A1,D0 BEq @SSAsUsual ;Eq => it's a ptr, cannot move Tst.B (SP) ;remember, the flags were saved earlier BMi @SSAsUsual ;Mi => locked, cannot move LEA heapData(A6),A3 ;start with the first block @CoalLoop CmpA.L A2,A3 BCC.S @CoalFini ;Carry Clear => beyond growing pointer Tst.B TagBC32(A3) ; <v1.1> BEq.S @CoalFree ;Eq => free block found <v1.1> Move.L BlkSize32(A3),D0 ;get block size <v1.1> AddA.L D0,A3 ;bump up to next block MoveQ #0,D0 ;mark 'no immediately preceeding free space' BrA.S @CoalLoop @CoalFree MoveA.L A3,A0 ;start of free space MoveQ #0,D0 ;init accumulated free space @CoalFLoop CmpA.L allocPtr(A6),A3 ;is the rover at a disappearing block? <23Apr85> BNE.S @1 ;NotEqual => no... <23Apr85> Move.L A0,allocPtr(A6) ;don't let allocptr pt to nonexistent block <23Apr85> @1 ; <23Apr85> Add.L BlkSize32(A3),D0 ;accumulate free space <v1.1> AddA.L BlkSize32(A3),A3 ;bump on to next block <v1.1> Tst.B TagBC32(A3) BEq.S @CoalFLoop ;Eq => more free space! Clr.L TagBC32(A0) ;clear 1st long word of block <v1.1> Move.L D0,BlkSize32(A0) ;update size of coalesced block <v1.1> BrA.S @CoalLoop ;continue with this non-free block @CoalFini ; We exit the above loop with D0=free space immediately below growing block, A0 pointing <18Apr85> ; to it. If at least 1/4 of the unrequited space can be made up by sliding, DO IT. <18Apr85> ; In case of a slide: if the free space is within a few minFree's of the needed space, <18Apr85> ; go the whole way, otherwise slide just enough to meet the needs. <18Apr85> ; A special case arises: when D5 is tiny, a fourth is zero, and this is BAD news if D0 is <18Apr85> ; also zero! The test BLS (as opposed to BCS) catches this case along with tinies. <18Apr85> ; Yet another goof -- must bound unrequited size by minFree, since that's the size of block <18Apr85> ; of free block we'll make at the end. <18Apr85> Move.L D5,D6 ;unrequited space LSR.L #2,D6 ;1/4 of it Cmp.L D6,D0 BLS.S @SSNoSlide ;Lower or Same => too little space free (or none available) MoveQ #minFree32,D6 ;Lower bound for unrequited <v1.2> Cmp.L D5,D6 BLS.S @5 ;Lower or Same => D5 big enough for slide Move.L D6,D5 ;Make at least minFree space... @5 MoveQ #minFree32+minFree32,D6 ;unscientific slop <v1.2> Add.L D5,D6 ;unrequited space + slop Cmp.L D6,D0 BCS.S @SSFullSlide ;Carry Set => not enough extra to split free space ; To do a partial slide, decrement the size of the lower block by the amount we'll slide. <18Apr85> Sub.L D5,D0 ;Size minus amount of slide <v1.1> Clr.L TagBC32(A0) ;set free block <v1.1> Move.L D0,BlkSize32(A0) ;Force size of free block (don't need A0 now)<C883> AddA.L D0,A0 ;Target slot for growing block Move.L D5,D0 ;Size of slide, for 'sliding' into FullSlide ; Full slide code: D0=amt of slide; D4=size of sliding block; D6=tmp; A0=dst of slide; <18Apr85> ; A1=handle to sliding block; A2=ptr to sliding block. Be sure to update the handle! <18Apr85> ; And after the block is moved, update GZRootPtr and allocPtr, since the latter may point <18Apr85> ; to a nonexistent free block. <18Apr85> @SSFullSlide Move.L A0,(A1) ;new location for sliding/growing block Add.L #blkData32,(A1) ;be sure handle points to text of block!!!! <v1.1> Move.B (SP),MPtag32(A0) ;restore the tags from home on the stack <v1.1> MoveM.L D0/A1,-(SP) ;save amt of slide and handle CmpA.L allocPtr(A6),A2 ;is the rover at the block we're moving? <23Apr85> BNE.S @10 ;NotEqual => rover is elsewhere... <23Apr85> Move.L A0,allocPtr(A6) ;point rover at new location <23Apr85> @10 ; <23Apr85> Exg A0,A2 ;A0=src of growing block, A1=dst MoveA.L A2,A1 ;dst, for blockmove Move.L D4,D0 ;current phys length is length of move! ;+++ BSr HBlockMove JSRROM HBlockMove MoveA.L A2,A0 ;ptr to moved block AddA.L D4,A0 ;ptr just beyond -- to new free block Move.L (SP)+,BlkSize32(A0) ;stuff size of new free block <v1.1> Clr.L TagBC32(A0) ;free block <v1.1> MoveA.L (SP)+,A1 ;restore handle LEA blkData32(A2),A0 ;text ptr to growing block Move.L A0,GZRootPtr BrA.S @SSAsUsual ; Our last resort: check the block following ours; if it's larger and allocated, don't even <18Apr85> ; bother trying to move it. <18Apr85> @SSNoSlide MoveA.L A2,A0 AddA.L D4,A0 ;ptr to following block Move.L BlkSize32(A0),D0 ;get size <v1.1> Cmp.L D0,D3 BCC.S @SSAsUsual ;Carry Clear => new size is large, so go the normal route Tst.B TagBC32(A0) BNE.S @SSDoReloc ;Not Equal => larger, alloc block following ********************************* End conditional block. ******************************** <18Apr85> ENDIF ; SmartSetSize ; This resumes the original SetSize growth code. The register setup must be adjusted to: <18Apr85> ; D0=??; D1=new log size; D2=D4=phys size; D3=new phys size; D4=phy size; D5=??; D6=?? <18Apr85> ; A0=??; A1=handle; A2=block ptr; A3=?? <18Apr85> @SSAsUsual Move.L A2,A0 ;Points to block header. Move.L D3,D0 ;New physical size. Sub.L D2,D0 ;Compute delta bytes. Add.L D2,A2 ;Point to where space is needed ; doesn't move it by accident. BSet #Lock,MPtag32(A0) ;lock block <v1.1> JSRROM a32MakeSpace ;Try making room after this block. BClr #Lock,MPtag32(A0) ;Unlock block <v1.1> @200 Move.L A2,D4 ;Did we succeed? BNE.S @SSEnough ;Yes, keep going ; New entry point to bypass MakeSpace when following block is bigger than one we're growing. <18Apr85> @SSDoReloc Move.L A1,D4 ;No, maybe we can relocate block. BEq.S @SSFail ;No, block is Non-Relocatable. Tst.B (SP) ;Can we move the growing block? BMi.S @SSFail ;No, block is locked. Move.L D1,D0 ;Desired new logical size. ; Try to grow the (unlocked) block by relocating&growing simultaneously. Across the call <10Apr85> ; mark it nonpurgeable, since we WANT it! Note must retrieve previous state from stack <10Apr85> ; unlike the BSet/BClr pair bracketing MakeSpace above. To get here, the Lock bit must be <10Apr85> ; clear, so the blithe BClr above jibes with our restoration from the stack. <10Apr85> Move.L (A1),A3 ;dereference handle <v1.1> BClr #Purge,MPtag32-BlkData32(A3);Unlock block <v1.1> JSRROM a32RelocRel ;Relocate the old block into a new. Move.L (A1),A0 ; get new address of block header <121> Sub.L #BlkData32,A0 ;Point to block header. <121> ; MOVE.B (SP),MPtag32-BlkData32(A3) ;Restore previous state. <v1.1> JSRROM ClearGZStuff ;mark gz stuff as done BrA.S @SSExit ;Return to caller. @SSFail ;Unable to find space. JSRROM ClearGZStuff ;mark gz stuff as done MoveQ #memFullErr,D0 ;Not enough room to reloc. block. BrA.S @SSExit ;Return to caller. @SSEnough ;Enough room has been found. JSRROM ClearGZStuff ;mark gz stuff as done Move.L BlkSize32(A2),D0 ;Length of free space after. <v1.1> Move.L D0,D4 ;Physical size of free space. Add.L D2,D4 ;Physical size of new combined blk. Move.L A0,A2 ;Points to block. Move.L D4,BlkSize32(A2) ;set block physical size. <v1.1> ; Neg.L D0 ;Minus size of new space. ; BSR.S AdjustFree ;Account for less free space. JSRROM SubtractFree ;Account for less free space. <v1.7> @SSShrink ;Shrink the block. Sub.L D3,D4 ;Number of bytes to shrink block. Cmp.L #MinFree32,D4 ;sFree < MinFree? <v1.2> BCS.S @SSLogical ;Yes, set logical size only. BTST #ROZ,flags(A6) ;don't shrink ROZ blocks <04Mar85> BNE.S @SSSkip ; <04Mar85> Move.L A2,A0 ;Points to block. Add.L D3,A0 ;Point to new free block. Move.L D3,BlkSize32(A2) ;set block physical size. <v1.1> Move.L D4,D0 ;Size of new free block. JSRROM a32MakeBkF ;Release free space. JSRROM AdjustFree ;And account for new free space. @SSLogical ;Set the logical size of the block. Move.L A2,A0 ;Points to block. Move.L D1,D0 ;New logical size. JSRROM a32SetLogSize ;Set logical size. @SSDone MoveQ #0,D0 ;Success result code. @SSExit ; Move.L (A1),A3 ;dereference handle <v1.1> ; Move.B (SP)+,MPtag32-BlkData32(A3) ;Restore MP flag byte. <v1.1> Move.B (SP)+,MPtag32(A0) ;Restore MP flag byte. <v1.1> @SSFin MoveM.L (SP)+,D1-D6/A0-A3 ;Restore registers. <18Apr85> RTS ;Return to caller. @SSSkip MoveQ #memROZWarn,D0 ; <04Mar85> BRA.S @SSExit ; <04Mar85> @SSEqual Move.B (SP)+,D0 ; Don't care about saved MP flag <121> MoveQ #0,D0 ;Success result code. <121> Bra.S @SSFin ; <121> ENDPROC ;________________________________________________________________________________ ; ; <68><85> End of SetPtrSize() patch ;________________________________________________________________________________ ;________________________________________________________________________ ; ; <54> Rolled over changes from split-off sources on 4/16/90 by DDG ; ; Patch To IOPMsgRequest Trap to Fix IOP Serial Driver Control Call Bug ; ; Traps Patched: IOPMsgRequest ; ; The problem: IOP Serial Driver calls are being mis-queued because ; the control rtn is jumping to IODone before the control calls are ; actually completed on the 6502 side. ; ; The fix: to make the patch small, and because the control calls call ; the IOPMsgRequest trap, we do a come-from patch on that trap. If the ; 'come from' addresses match any of those in the driver, then the patch ; code is exectued. Else the ROM code (or previous patch) is jumped to. ; All the patch code does is jsr to the ROM code, then wait around for ; the IOP request to have completed before returning. ; ;________________________________________________________________________ proc export IOPMsgRequest ;offsets to ROM addresses IOPControlNormal EQU $6B932 IOPControl08 EQU $6B95A IOPControl13 EQU $6B9C2 with IOPRequestInfo IOPMsgRequest ; We can use the new protection scheme, if the first two instructions in the <115> ; patch are a bra.s followed by the jump to the old rom address. bra.s @protection @toRom BackToTrap OldIOPMsgRequest ; no, then just execute ROM Code @protection cmpRA IOPControlNormal,OSTrapReturnAddressDepth(sp) ; from IOP serial drvr ctl call? beq.s @doPatch ; yes, then come back here cmpRA IOPControl08,OSTrapReturnAddressDepth(sp) beq.s @doPatch cmpRA IOPControl13,OSTrapReturnAddressDepth(sp) bne.s @toRom @doPatch move.l a0,-(sp) ; ROM IOPMsgRequest does not save a0 bsr.s @toRom ; bsr to IOPMsgRequest in ROM move.l (sp)+,a0 @syncLoop tst.b irReqActive(a0) bne.s @syncLoop rts endwith endproc ;_______________________________________________________________________________ ; ; <54> End of Patch To IOPMsgRequest Trap to Fix IOP Serial Driver Control Call Bug ;_______________________________________________________________________________ ;____________________________________________________________________________ GGD <28> ; Patch the Deferred Task Manager Task Dispatcher to re-enable interrupts upon ; exiting to fix QuickMail servers (for now at least), because they call through ; jDisptch at application level and wind up with all interrupts being masked, ; and the Task Dispatcher expects to only be called from then end of an interrupt ; handler, which will restore the interrupt level via an RTE. Note that Applications ; should NOT be calling through this UNDOCUMENTED LOW MEMORY VECTOR, and we are ; changing this just so QuickMail servers don't hang with interrupts disabled, ; until CE Software can fix their bug. NewvDisptch proc export JsrTrap ROMvDisptch ; call the existing ROM code andi.w #$F8FF,sr ; enable all interrupts rts ; all done endproc ; End of Deferred Task Manager Task Dispatcher Patch Installation ;____________________________________________________________________________ GGD <28> ;____________________________________________________________________________ jwk <29> ; ; BusErrHandler -- New and improved chaining SCSI bus error handler. ; When the SCSI Mgr is performing a blind data transfer, it patches ; out the bus error vector. The old SCSI Mgr bus error handler ; assumed that if it got called, it must be handling a SCSI bus error. ; Unfortunately, NuBus cards could bus error while the SCSI Mgr is ; installed. To be a better bus error citizen, the SCSI bus error ; handler now checks for a SCSI address as the fault address, and ; chains to the bus error handler that it replaced. ; BusErrHandler: PROC EXPORT ROMBusErrRecovery EQU $00008a6c ; recovery delay routine in ROM <29> move.l d0,-(sp) ; save d0 <29> moveq.l #$ffffff9c,d0 ; mask = $ffffff9c <start> and.l 4+$10(sp),d0 ; clear variable bits of the fault address cmp.l SCSIHsk,d0 ; was it a SCSI chip access ? beq.s @start ; if so, start processing the bus error move.l (sp)+,d0 ; restore d0 move.l OldBusErrVct(a6),-(sp) ; put old bus error handler addr on stack <end> rts ; jump to old handler, assuming it'll RTE <29> @start subq.w #1,BusErrCount(a6) ; retry until we get tired beq.s CleanUp jmpROM ROMBusErrRecovery ; back off and try again (RTE) <29> Cleanup addq.l #4,sp ; throw away copy of d0 on stack <29> move.w (sp),d0 ; get sr from stack bfextu 6(sp){0:4},d1 ; get format code from stack cmp.b #$0A,d1 ; short exception frame? bne.s Drop46w ; no, so use larger frame adda.w #16*2,sp ; dispose of the 16-word frame bra.s DummyFrame ; and finish up Drop46w add.w #46*2,sp ; size of exception frame DummyFrame ; ; Fake a format code 0 exception frame (4 words) to finish cleaning up ; move.w zeroReg,-(sp) ; format code 0 pea FinishErr ; PC value move.w d0,-(sp) ; sr value rte ; 'return' from the fake exception FinishErr moveq.l #scBusTOErr,d0 ; assume slow peripheral btst.b #bPM,sBSR(a3) ; phase change? <C478/10Dec86> <v1.5><29> bne.s @ErrorDone ; phase matches: slow peripheral moveq.l #scPhaseErr,d0 ; return phase change error @ErrorDone rts ; ; End of chaining SCSI bus error handler patch ;____________________________________________________________________________ jwk <29> ;_________________________________________________________________________________________ <119> djw ; Beginning code for Quantum 7.9 ROM fix ; ; Detailed description of the Quantum problem and solution: ; ; Quantum drives with the firmware version 7.9, has a problem with loosing the last byte ; of a block during a multi-block write. The problem occurs when the time between blocks ; written to the SCSI bus on the CPU side, is greater than 482 microseconds. This may ; occur in situations where there are a lot of interrupts. When conditions are right and ; the problem happens, the last byte of the previous block (in a multi-block transaction), ; is "eaten" by the drive. Any checksum or CRC calculated by the drive is correct, ; because the drive calculates it after the byte has been corrupted. This problem is ; especially frequent when email packages are installed on the Mac, since they generate ; a lot of interrupts which take a long time. This problem only occurs during fast ; writes in the SCSI manager (pseudo-dma mode). ; ; The way the SCSI manager currently works is it gets a TIB packet which contains the ; instructions on how to talk to a particular device. Included in the TIB is when to ; re-synchronize with the drive by waiting for a *REQ. When we see *REQ, the SCSI ; manager then begins the next TIB data transfer instruction and loads a byte of data ; into the 53C80's output register. After the first byte is "manually" sent, the ; hardware handshaking automatically takes care of the *REQ and *ACK handshaking. When ; the TIB write transfer is complete, pseudo-dma is disable, which releases *ACK, ; completing the handshake. This allows the target to assert *REQ when it is ready. ; ; The window of vulnerability is between the last byte of the previous block and the ; first byte of the next block. More correctly, it is between the rising edge of *ACK ; and the falling edge of the next *ACK. Because we release *ACK to synchronize, an ; interrupt may come in and delay the next transfer. ; ; The solution to the problem is to pre-load a data byte into the 53C80's data output ; register before we release *ACK. There are two ways of releasing *ACK: disable pseudo ; dma, and write a byte to the data output register. Leaving pseudo-dma enabled through ; the entire TIB will mean that whenever a *REQ occurs, there will be data available in ; the output register. This means there will be no delay between bytes because the ; hardware is not subject to interrupt delays. ; ; This patch therefore involves patching the TIB interpreter and the fast write routine ; in the SCSI manager. There are five versions of the SCSI manager to patch: Mac Plus, ; SE, Mac II, Portable, and IIci. ; ;_________________________________________________________________________________________ ; QuantumWBlindIIci - patch to NewSCSIWBlind ; ; This code replaces the original NewSCSIWBlind entry point. The new entry for blind ; writes enables pseudo-dma on a per-transaction basis. Pseudo-dma was previously ; enabled only on a per-TIB-instruction basis. We jump back into the ROM to the original ; entry point to continue executing the TIB. ; ; Input: reg a3 = base of SCSI read addr ; a6 = SCSI stack frame ; d7 = zero ; Export QuantumWBlindIIci Export FastWriteFix QuantumWBlindIIci romNewSCSIWBlind Equ $07574 ; rom offset to NewSCSIWBlind entry point move.b d7,sTCR+WrOffs(a3) ; set to match data out phase (to zero) move.b #iDMA,sMR+WrOffs(a3) ; enable DMA in mode register move.b #iDB,sICR+WrOffs(a3) ; assert data bus in initiator command reg move.b d7,sDMAtx+WrOffs(a3) ; start write DMA ; Do I need to test for DRQ or *REQ at this point? jmp ([RomBase],romNewSCSIWBlind); continue at original entry pt in ROM ;_________________________________________________________________________________________ ; FastWriteFix - patch to FastWrite ; ; This code replaces the low level data transfer routine for fast writes in the SCSI ; manager. It is rewritten to assume psuedo-dma is always on, and to do the device ; synchronization (looking for *REQ and DRQ), after loading a data byte into the ; output register of the 5380. We don't have to worry about zero-byte transfers. They ; are filtered out in the calling routine (Transfer). ; ; Entry: d2 = number of bytes to transfer ; a1 = addr of sBSR(a3) - by convention, and used by bus error handler ; a2 = ptr to data buffer to transfer ; a3 = base addr of NCR 53C80 ; a4 = ptr to SCSI globals ; ; Exit: d1 = number of bytes actually transfered. This value is only good if ; the transfer was good with no errors. It is inaccurate when an ; error aborts the transfer. With scsiGlobalRecord FastWriteFix move.l hhsk5380(a4),a0 ; point to addr for hardware handshaking adda.w wrOffset(a4),a0 ; add in write offset move.l d2,d1 ; save to return number of bytes transfered ; Pre-load the NCR 53C80's output register with a byte of data. If we are in ; the middle of a multi-block write, *ACK is currently asserted. Writing a byte ; to the output register will release *ACK, completing the handshaking, allowing ; the target to assert *REQ. Is there a chance We do not need to check for DRQ at this point (to see ; if the chip is ready for the next byte), because all data out routines always ; wait to make sure the last byte got out. sub.l #1,d2 ; dec number of bytes to xfer move.b (a2)+,(a0) ; write data byte to output register ; With *ACK released, determine if the target is in sync. We cannot look for ; *REQ to be asserted, because the target may have already accepted the data byte ; and released *REQ at this point. We can sync on DRQ which will signal when ; the NCR 53C80 is ready to accept a data byte, meaning a *REQ from the target ; must have already occurred and our data byte was taken. @syncWait btst.b #bDMAR,(a1) ; check bus & status reg for DRQ bne.s @doWrite ; DRQ present - sync-ed up so proceed btst.b #bREQ,sCSR(a3) ; no DRQ - is *REQ present ? beq.s @syncWait ; no *REQ yet - wait for sync btst.b #bPM,(a1) ; with *REQ, check phase lines bne.s @syncWait ; still in data out phase - wait moveq.l #0,d1 ; out of phase - did not xfer any bytes moveq.l #scPhaseErr,d0 ; return error bra.s @Done ; exit ; Perform the write to the SCSI chip. First align the bytes to longs, then ; align them to 32 byte chunks. Transfer the bulk of the data in 32 byte ; blocks. ; Reg d2.l = number of bytes to move @doWrite cmpi.l #3,d2 ; check for very short copy bls.s @veryShort ; skip alignment if very short move.l a2,d0 ; get addr of data buffer andi.l #$00000003,d0 ; check for long word alignment beq.s @Aligned ; if no alignment needed subq.l #4,d0 ; bias by 4 to get correct index jmp @Aligned(d0.w*2) ; do the alignment move.b (a2)+,(a0) ; move a byte move.b (a2)+,(a0) ; move a byte move.b (a2)+,(a0) ; move a byte @Aligned add.l d0,d2 ; adjust the byte count (d0 = neg) move.l d2,d4 ; save tail byte count lsr.w #2,d2 ; adjust to number of longs to move moveq.l #7,d0 ; mask for starting index and.l d2,d0 ; number of long words to move first neg.w d0 ; negate to index backwards lsr.l #3,d2 ; number of 32 byte blocks to move move.l d2,d3 ; get number of 32*64K byte blks to move swap d3 ; count in low word jmp @CopyStart(d0.w*2) ; jump into the loop @CopyLoop move.l (a2)+,(a0) ; move a 32 byte block of data.... move.l (a2)+,(a0) ; ... 4 bytes at a time move.l (a2)+,(a0) move.l (a2)+,(a0) move.l (a2)+,(a0) move.l (a2)+,(a0) move.l (a2)+,(a0) move.l (a2)+,(a0) @CopyStart dbra d2,@CopyLoop ; loop in chunks of 32 bytes dbra d3,@CopyLoop ; loop in chunks of 32*64K bytes andi.l #$00000003,d4 ; check for tail alignment move.l d4,d2 ; d2 = number of bytes remaining @veryShort neg.w d2 ; negate to index backwards jmp @Remaining(d2.w*2) ; write remaining bytes move.b (a2)+,(a0) ; move a byte move.b (a2)+,(a0) ; move a byte move.b (a2)+,(a0) ; move a byte @Remaining ; Before exiting this routine, make sure that the peripheral has actually accepted ; the data byte. Wait for a DRQ (meaning the SCSI chip is ready for another byte) ; before exiting. @DoneWait moveq.l #0,d0 ; set good return btst.b #bDMAR,(a1) ; check for DRQ (a1 = sBSR(a3)) bne.s @Done ; if DRQ, peripheral got the byte btst.b #bREQ,sCSR(a3) ; no DRQ - is *REQ present ? beq.s @DoneWait ; no *REQ yet - wait for it btst.b #bPM,(a1) ; are we still in phase ? bne.s @DoneWait ; if so, keep waiting @Done rts ; we're done Endp ; end scsi patches ;____________________________________________________________________________ EH <33><37> ; ; Patch To IOPMoveData Trap to Fix IOP Serial Driver DMA Hang Bug ; ; Traps Patched: IOPMoveData ; ; The problem: the iop serial DMA hangs at if more than one byte ; accumulates in the SCC fifo. This is really a problem with the ; timing in the DMA hardware, but we'll work around it in software. ; ; The fix: In the main driver loop, we'll manually remove extra ; bytes from SCC fifo to DMA buffer before restarting the DMA. ; ; Note:If one of these manually moved bytes is in error, it will be ; bucketed (not moved to DMA buffer), as will subsequent bytes in the ; fifo. ; ; MECHANISM: To make the patch small, we do a come-from on IOPMoveData. ; We use the ROM IOPMove Data to download the patches to the 6502 code ; after we've already downloaded the unpatched image. ; ; <37> added in two fixes. ; Fix slow baud rate chars getting munched if a control call is made to ; change the baud rate. Now we wait for an ALL SENT from the SCC before ; dispatching all control calls. ; ; Fix CTS status on open to be reported properly back to 68000 side. ; We now properly set CTSFlag value during the open call, which is the ; value that gets reported in the status call. ; ; <123> added in external control call feature ; -- Use bit 6 now of the CtlOptions byte in control call 16 ; bit 6 = 0 means internal clocking ; bit 6 = 1 means external clocking ; -- Incidentally, fixed bug in break handling code such that the null ; char is now discarded when break is de-asserted. ; -- changed ExtStatInt handler code to set bit 3 of Async Err if break is rcv'd. ; -- changed InitSCC rtn to check for ext clocking in setting WR4 clock divide bits. ; we also disable HWHS for the external clock case so we won't try to use the ; CTS line for clocking AND handshaking. ; -- changed the EXTInt handler to disable CTS interrupts if we receive them ; too quickly. This was never implemented in the IOP driver. ; Note we are doing a slime-bucket thing here by using ZP addresses that the ROM ; driver isn't using. But it doesn't look like anybody's using it, so hey, what ; the living heck. ZPSize is 32, so we'll start our stuff at offset $33. ; CTSCount = ZPBase+$33 the count of CTS transitions we started timing ; LastTime = ZPBase+$34 the last timer value (low byte) for a CTS interrupt ; -- Modify the Zero Page initialization to clear all $55 bytes we have ; ; <126> ; -- fix disable to HWHS for external clocking. it now gets disabled write before ; CtlOptions (#16) calls DoInitSCC instead of in the InitSCC routine. ; -- rolled in a fix to close that i had meant to add in <123>. We time out now ; on waiting for the all-sent on the close. ; ; <138> ; -- fix the IOP timer to actually count! i think the code to disable CTS interrupts ; was just working by chance, since i'm also fixing a branch instruction in ; that code to fix the bug with printing to the imagewriter. ;________________________________________________________________________ IOPMoveDataPatch proc export IOPMoveData ;offsets to ROM addresses From_IOPOpenCall EQU $0006bC48 ;return addr of IOPMoveData call in IOP Serial Open ROM_IOPMoveData EQU $00004ed2 ;rom addr for the trap ;Pic RAM addresses DrvrAPicAddr EQU $2d55 ; driver A start addr for Pic RAM DrvrBPicAddr EQU $56aa ; driver B start addr for Pic RAM numPatches EQU 24 ; number of times thru loop PatchDrvrSize EQU $09ae ; size of the patched driver ;__________________________________________________________ ; ; 6502 IOP Driver Patches -- code to fix in downloaded image. ; Patches are in order by offset from beginning of driver. ; Each set of patch bytes is preceded by a word offset to ; start of patch, and a byte for size of patch. ; ; There is one set of patches for Driver A and one for Driver B. ;__________________________________________________________ DrvrAFix ; changes zero page init to clear all $55 bytes ; this one-byte change is inline. dc.b $00,$09 ; offset to this patch dc.b @a20end-@a20 ; number patchbytes @a20 dc.b $55 ; y<-$55, the index to clear out z-page ; fixes the new location of the driver buffers ; new buffer start addr is $3703 = $2d55(base) + $9ae(new offset) @a20end dc.b $00,$46 ; offset to this patch dc.b @a1end-@a1 ; number patchbytes @a1 dc.b $a9,$37 ; lda #>Buffers (new high byte) dc.b $a2,$03 ; lda #>Buffers (new low byte) ; fix one byte to load into DMA count @a1end dc.b $00,$94 ; offset to this patch dc.b @a2end-@a2 ; number patchbytes @a2 dc.b $FB ; put FB in the DMA count instead of FF ; calls the fix to start the IOP timer since ; patch wouldn't fit inline @a2end dc.b $00,$98 ; offset to this patch <138> dc.b @a23end-@a23 ; number patchbytes <138> ; <138> @a23 dc.b $4C,$f8,$36 ; jmp to fix at addr $36f8 = $2d55(base) +$9a3(offset) <138> ; calls the fix to CTSFlag on Open (see patch at $8D5) since the ; patch wouldn't fit inline @a23end dc.b $01,$00 ; offset to this patch <37> dc.b @a3end-@a3 ; number patchbytes <37> ; <37> @a3 dc.b $4C,$2b,$36 ; jmp to fix at addr $362B = $2d55(base) +$8D6(offset) <37> ; calls the fix to DoPollEvent (see patch at $880) since the ; patch wouldn't fit inline @a3end dc.b $01,$2F ; offset to this patch dc.b @a4end-@a4 ; number patchbytes @a4 dc.b $20,$D5,$35 ; jsr to fix at address $2d55(base) +$880(offset) ; calls the fix to DMA code (see patch at $886) since the ; patch wouldn't fit inline @a4end dc.b $01,$46 ; offset to this patch dc.b @a5end-@a5 ; number patchbytes @a5 dc.b $F0,$03 ; beq to just below dc.b $4C,$db,$35 ; jmp here if DMA was NOT stopped by interrupt rtn $2d55(base)+$886(offset) dc.b $4C,$02,$36 ; jmp here if DMA was stopped by interrupt rtn $2d55(base)+$8ad(offset) ; fix one byte to load into DMA count @a5end dc.b $01,$6E ; offset to this patch dc.b @a6end-@a6 ; number patchbytes @a6 dc.b $FB ; put FB in the DMA count instead of FF ; calls patch to close to time-out on the all sent loop since the ; patch wouldn't fit inline @a6end dc.b $02,$6F ; offset to this patch dc.b @a21end-@a21 ; number patchbytes @a21 dc.b $4C,$e0,$36 ; jmp to fix at address $36e0 = $2d55(base) +$98b(offset) <37> ; calls the fix to all sent in control calls (see patch at 8C6) since the <37> ; patch wouldn't fit inline <37> @a21end dc.b $03,$39 ; offset to this patch <37> dc.b @a7end-@a7 ; number patchbytes <37> ; <37> @a7 dc.b $4C,$1b,$36 ; jmp to fix at address $361b = $2d55(base) +$8C6(offset) <37> ; calls patch to add ext clking to control call 16 since the ; patch wouldn't fit inline @a7end dc.b $4,$4a ; offset to this patch dc.b @a12end-@a12 @a12 dc.b $4c,$38,$36 ; jump to fix at address $3638= $2d55(base) + $8e3(offset) ; calls patch to disable CTS interrupts if we get them too fast since ; patch wouldn't fit inline @a12end dc.b $07,$36 ; offset to this patch dc.b @a18end-@a18 @a18 dc.b $4c,$b5,$36 ; jump to fix at address $36b5= $2d55(base) + $960(offset) ; calls patch to set bit 3 of AsyncErr if break received since ; patch wouldn't fit inline @a18end dc.b $07,$55 ; offset to this patch dc.b @a14end-@a14 @a14 dc.b $4c,$82,$36 ; jump to fix at address $3682= $2d55(base) + $92d(offset) <126> ; calls patch to set clock divide bits for WR4 since ; patch wouldn't fit inline @a14end dc.b $07,$79 ; offset to this patch dc.b @a16end-@a16 @a16 dc.b $4c,$a1,$36 ; jump to fix at address $36a1= $2d55(base) + $94c(offset) <126> ; fixes the code at DoPollEvent to compare to DMA count, ; not the DMA address. @a16end dc.b $08,$80 ; offset to this patch dc.b @a8end-@a8 ; number patchbytes @a8 dc.b $AD,$23,$F0 ; lda DMACntLo dc.b $C9,$FB ; cmp #$FB dc.b $60 ; rts ; fixes DMA code to check for bytes left over in the SCC fifo @a8end dc.b $08,$86 ; offset to this patch dc.b @a9end-@a9 ; number patchbytes @a9 dc.b $AD,$22,$F0 ; lda DMA_AddrHi Set up temp pointer dc.b $85,$5A ; sta TempBufPtr+1 into the DMA buffer. $55(base)+$05(offset) dc.b $64,$59 ; stz TempBufPtr $55(base)+$04(offset) dc.b $AC,$21,$F0 ; ldy DMA_AddrLo Get the byte offset into buffer ;@NxtChar dc.b $AD,$41,$F0 ; lda SCC_Ctl Read RRO dc.b $29,$01 ; and #Rx_CharAvail is there a receive char? dc.b $F0,$2B ; beq @Done no chars, then we are done dc.b $AD,$43,$F0 ; lda SCC_Data get the data char dc.b $91,$59 ; sta (TempBufPtr),y store char in buffer $55(base)+$04(offset) dc.b $84,$7F ; Sty HoldDMACount $55(base)+$2a(offset) dc.b $C8 ; iny ; if any errors reset the scc dc.b $A9,$01 ; lda #1 read RR1 dc.b $8D,$41,$F0 ; sta SCC_Ctl dc.b $AD,$41,$F0 ; lda SCC_Ctl dc.b $29,$70 ; and #$70 parity, overrun, framing err? dc.b $85,$5E ; sta HoldRR1 save errors $55(base)+$09(offset) dc.b $F0,$E3 ; beq @NxtChar no get the next char ; Suppose Special Rx Int has turned off DMA. ; There could be chars in FIFO after interrupt. ; Driver only flags errors on last byte DMAd in. ; To insure this, drop all bytes after error. ; SHOULD BE FIXED WHEN DRIVER REVVED ;@ResetSCC dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$41,$F0 ; sta SCC_Ctl dc.b $AD,$43,$F0 ; lda SCC_Data flush SCC FIFO dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$41,$F0 ; sta SCC_Ctl dc.b $AD,$43,$F0 ; lda SCC_Data flush SCC FIFO dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$41,$F0 ; sta SCC_Ctl ;@Done dc.b $4C,$b0,$2E ; jump back to our regularly scheduled code $2d55(base) +$15b(offset) ; fixes ctrl calls to wait for all sent from SCC before executing @a9end dc.b $08,$C6 ; offset to this patch dc.b @a10end-@a10 ; number patchbytes ;@wait_more <37> @a10 dc.b $78 ; sei turn off interrupts <37> dc.b $A9,$01 ; lda #$01 read RR1 <37> dc.b $8D,$41,$F0 ; sta SCC_Ctl <37> dc.b $AD,$41,$F0 ; lda SCC_Ctl <37> dc.b $58 ; cli turn on interrupts <37> dc.b $6A ; ror a test the all sent bit, bit 0 <37> dc.b $90,$F3 ; bcc @wait_more <37> dc.b $7C,$33,$35 ; jmp (ControlDispatch,x) <37> ; ControlDispatch at $3533 = $2d55(base) + $7DE(offset) <37> ; fixes ctrl calls to wait for all sent from SCC before executing <37> @a10end dc.b $08,$D6 ; offset to this patch <37> dc.b @a11end-@a11 ; number patchbytes <37> ; <37> ; ctsFlag = 0 on input. Acc holds RRO value <37> @a11 dc.b $29,$A0 ; and #$A0 isolate DTR and break bits <37> dc.b $85,$58 ; sta holdRR0 $55(base) + $17(offset) <37> dc.b $29,$20 ; and #$20 is DTR asserted? <37> dc.b $F0,$02 ; beq @1 yes, then CTSFlag stays 0 <37> dc.b $C6,$6C ; dec CTSFlag no, then CTSFlag goes to $FF CTSFlag at $55(base) + $17(offset) dc.b $4C,$59,$2E ;@1 jmp backToCode at $2e59 = $2d55(base) + $104 offset <37> ; adds external clocking control into bit 6 of the ; CtlOptions byte for control call 16 @a11end dc.b $08,$e3 ; offset to this patch dc.b @a13end-@a13 ; number of patchbytes @a13 dc.b $a5,$73 ;lda CtlOptions get old clk bit $55(base)+$1e(offset) dc.b $29,$40 ;and #clkMsk bit 6 dc.b $aa ;tax dc.b $ad,$84,$02 ;lda C16_CtlOptions get new byte $200(base)+$80(msg#)+$4(offset) dc.b $a8 ;tay dc.b $29,$40 ;and #clkMsk bit 6 dc.b $85,$73 ;sta CtlOptions $55(base)+$1e(offset) dc.b $8a ;txa dc.b $c5,$73 ;cmp CtlOptions is old clock bit same as new? $55(base)+$1e(offset) dc.b $d0,$05 ;bne @switchIt nope, switch clk mode dc.b $84,$73 ;@out sty CtlOptions store new byte $55(base)+$1e(offset) dc.b $4c,$7f,$30 ;jmp DoControlEvent_Return $2d55(base) +$32a(offset) dc.b $98 ;@switchIt tya dc.b $0a ;asl shift into minus bit dc.b $30,$14 ;bmi @ext dc.b $a9,$50 ;@int lda #intClkSrc set int clk params dc.b $8d,$17,$35 ;sta WR11_Data $2d55(base) +$7c2(offset) dc.b $a9,$01 ;lda #BRGEnbl dc.b $8d,$1d,$35 ;sta WR14_Data $2d55(base) +$7c8(offset) dc.b $ad,$0d,$35 ;lda WR4_data set clock divide to 16 $2d55(base) +$7b8(offset) dc.b $09,$40 ;ora #dvd16Msk clock divide bit is bit 6 dc.b $8d,$0d,$35 ;sta WR4_data in WR4 $2d55(base) +$7b8(offset) dc.b $80,$12 ;bra @load dc.b $a9,$28 ;@ext lda #extClkSrc set ext clk params dc.b $8d,$17,$35 ;sta WR11_Data $2d55(base) +$7c2(offset) dc.b $a9,$00 ;lda #BRGDsbl dc.b $8d,$1d,$35 ;sta WR14_Data $2d55(base) +$7c8(offset) dc.b $ad,$0d,$35 ;lda WR4_data set clock divide to one $2d55(base) +$7b8(offset) dc.b $29,$BF ;and #dvdOneMsk clock divide bit is bit 6 dc.b $8d,$0d,$35 ;sta WR4_data in WR4 $2d55(base) +$7b8(offset) dc.b $5a ;@load phy ; save the control byte dc.b $64,$63 ;stz HWHSenable disable hardware handshaking $55(base)+$0e(offset) <126> dc.b $20,$f4,$34 ;jsr DoInitSCC $2d55(base) +$79f(offset) dc.b $7a ;ply dc.b $80,$c8 ;bra @out <126> ; patch to set bit 3 of AsyncErr if break received @a13end dc.b $09,$2d ; offset to this patch <126> dc.b @a15end-@a15 ; number of patchbytes @a15 dc.b $8a ;txa a=changed bits dc.b $30,$03 ;bmi @change break changed dc.b $4c,$ad,$34 ;@done jmp ExtStatEnd $2d55(base) +$758(offset) dc.b $a5,$58 ;@change lda LastRRO get break value $55(base)+$03(offset) dc.b $30,$05 ;bmi @post break was set dc.b $ad,$43,$f0 ;lda SCC_data break unset, discard null dc.b $80,$f4 ;bra @done dc.b $a9,$08 ;@post lda #$08 set bit 3 dc.b $24,$67 ;bit ResetAsyncErr has host read stats $55(base)+$12(offset) dc.b $10,$04 ;bpl @unread dc.b $64,$68 ;stz AsyncErr $55(base)+$13(offset) dc.b $64,$67 ;stz ResetAsyncErr $55(base)+$12(offset) dc.b $05,$68 ;@unread ora AsyncErr $55(base)+$13(offset) dc.b $85,$68 ;sta AsyncErr $55(base)+$13(offset) dc.b $80,$e4 ;bra @done ; patch to set clock divide bits for WR4 and disable HWHS for ext clock @a15end dc.b $09,$4c ; offset to this patch <126> dc.b @a17end-@a17 ; number of patchbytes ;acc=%0400xxxx default to divide-by-16 @a17 dc.b $da ;phx save reg dc.b $aa ;tax save acc dc.b $a5,$73 ;lda CtlOptions are we ext clk'd? $55(base)+$1e(offset) dc.b $0a ;asl shift ext clk bit into minus bit dc.b $30,$03 ;bmi @ext yes, go fix clk dvd bits and disable HWHS dc.b $8a ;@int txa no, just restore acc dc.b $80,$03 ;bra @done <126> dc.b $8a ;@ext txa restore acc dc.b $29,$0f ;and #$0f fix clk dvd bits to div-by-one ; dc.b $64,$63 ;stz HWHSenable disable hardware handshaking $55(base)+$0e(offset)<126> dc.b $fa ;@done plx restore reg dc.b $8d,$0d,$35 ;sta WR4_Data and write out WR4 value $2d55(base) +$7b8(offset) dc.b $4c,$d1,$34 ;jmp FinishInitSCC and finish up $2d55(base) +$77c(offset) ; patch to disable CTS interrupts if we get them too fast @a17end dc.b $09,$60 ; offset to this patch dc.b @a19end-@a19 ; number of patchbytes @a19 dc.b $a5,$88 ;lda CTSCount exceeded 5 transitions in 1 ms? $55(base)+$33(offset) dc.b $c9,$05 ;cmp #5 dc.b $90,$0c ;bcc @fixTime no, then fix the time dc.b $a9,$0f ;lda #15 yes, then disable CTS interrupts dc.b $8d,$41,$f0 ;sta SCC_Ctl write to WR15 dc.b $a9,$80 ;lda #$80 bit 5 clear to disable CTS ints dc.b $8d,$41,$f0 ;sta SCC_Ctl dc.b $80,$12 ;bra @out go finish in ROM ;@fixTime dc.b $a5,$89 ;lda LastTime get last count (remember it's a countdown timer) dc.b $ed,$11,$f0 ;sbc TimerCounterH time elasped > threshold? $55(base)+$34(offset) dc.b $c9,$08 ;cmp #threshold threshold is 1 msÅ 8 ticks dc.b $90,$07 ;bcc @update no, just increment <138> dc.b $ad,$11,$f0 ;lda TimerCounterH yes, set new last time dc.b $85,$89 ;sta LastTime $55(base)+$34(offset) dc.b $64,$88 ;stz CTSCount and restart count $55(base)+$33(offset) ;@update inc the transition count dc.b $e6,$88 ;inc CTSCount $55(base)+$33(offset) ;@out dc.b $64,$6c ;stz CTSFlag do stuff we stomped on in overpatch $55(base)+$17(offset) dc.b $a5,$58 ;lda RRO $55(base)+$03(offset) dc.b $4c,$8f,$34 ;jmp BackToROM finish up in ROM $2d55(base) +$73a(offset) ; patch to close to time-out on the all sent loop @a19end dc.b $09,$8b ; offset to this patch dc.b @a22end-@a22 ; number of patchbytes @a22 dc.b $a5,$FF ;ldx #$FF load up $FFFF for timeout loop dc.b $c9,$FF ;ldy #$FF 24 cycles * .5 us/cycle * 64K Å .8 seconds dc.b $a9,$01 ;@loop lda #1 read from RR1 dc.b $8d,$41,$f0 ;sta SCC_Ctl dc.b $ad,$41,$f0 ;lda SCC_Ctl dc.b $6a ;ror get all sent bit in carry dc.b $b0,$06 ;bcs @done if allsent, then outta here dc.b $ca ;dex not allsent, so keep waiting dc.b $d0,$f2 ;bne @loop dc.b $88 ;dey dc.b $d0,$ef ;bne @loop dc.b $4c,$cf,$2f ;@done jumpBacktoClose $2d55(base) +$27a(offset) ; patch to start the IOP timer @a22end dc.b $09,$a3 ; offset to this patch dc.b @a24end-@a24 ; number of patchbytes @a24 dc.b $9c,$24,$f0 ;stz DMA_CntHi do code we stomped with patch dc.b $a9,$ff ;lda #$FF dc.b $8d,$11,$f0 ;sta TimerCounterH get the timer started (just need to write to it) dc.b $4c,$f0,$2d ;@done jumpBacktoOpen $2d55(base) +$09b(offset) @a24end dc.w 0 ; filler DrvrBFix ; changes zero page init to clear all $55 bytes ; this one-byte change is inline. dc.b $00,$09 ; offset to this patch dc.b @b20end-@b20 ; number patchbytes @b20 dc.b $55 ; y<-$55, the index to clear out z-page ; fixes the new location of the driver buffers ; new buffer start addr is $604d = $56aa(base) + $9a3(new offset) @b20end dc.b $00,$46 ; offset to this patch dc.b @b1end-@b1 ; number patchbytes @b1 dc.b $a9,$60 ; lda #>Buffers (new high byte) dc.b $a2,$4d ; lda #>Buffers (new low byte) ; fix one byte to load into DMA count @b1end dc.b $00,$94 ; offset to this patch dc.b @b2end-@b2 ; number patchbytes @b2 dc.b $FB ; put FB in the DMA count instead of FF ; calls the fix to start the IOP timer since ; patch wouldn't fit inline @b2end dc.b $00,$98 ; offset to this patch <138> dc.b @b23end-@b23 ; number patchbytes <138> ; <138> @b23 dc.b $4C,$4d,$60 ; jmp to fix at addr $604d = $56aa(base) +$9a3(offset) <138> ; calls the fix to CTSFlag on Open (see patch at $8D5) since the ; patch wouldn't fit inline @b23end dc.b $01,$00 ; offset to this patch <37> dc.b @b3end-@b3 ; number patchbytes <37> ; <37> @b3 dc.b $4c,$80,$5F ; jmp to fix at addr $5F80 = $56aa(base) +$8D6(offset) <37> ; calls the fix to DoPollEvent (see patch at $880) since the ; patch wouldn't fit inline @b3end dc.b $01,$2F ; offset to this patch dc.b @b4end-@b4 ; number patchbytes @b4 dc.b $20,$2a,$5F ; jsr to fix at address $56aa(base) +$880(offset) ; calls the fix to DMA code (see patch at 886) since the ; patch wouldn't fit inline @b4end dc.b $01,$46 ; offset to this patch dc.b @b5end-@b5 ; number patchbytes @b5 dc.b $F0,$03 ; beq to just below dc.b $4C,$30,$5F ; jmp here if DMA was NOT stopped by interrupt rtn $56aa(base)+$886(offset) dc.b $4C,$57,$5F ; jmp here if DMA was stopped by interrupt rtn $56aa(base)+$8ad(offset) ; fix one byte to load into DMA count @b5end dc.b $01,$6E ; offset to this patch dc.b @b6end-@b6 ; number patchbytes @b6 dc.b $FB ; put FB in the DMA count instead of FF ; calls patch to close to time-out on the all sent loop since the ; patch wouldn't fit inline @b6end dc.b $02,$6F ; offset to this patch dc.b @b21end-@b21 ; number patchbytes @b21 dc.b $4C,$35,$60 ; jmp to fix at address $6035 = $56aa(base) +$98b(offset) <37> ; calls the fix to all sent in control calls (see patch at 8C6) since the <37> ; patch wouldn't fit inline <37> @b21end dc.b $03,$39 ; offset to this patch <37> dc.b @b7end-@b7 ; number patchbytes <37> ; <37> @b7 dc.b $4C,$70,$5F ; jmp to fix at address $5F70 = $56aa(base) +$8C6(offset) <37> ; calls patch to add ext clking to control call 16 since the ; patch wouldn't fit inline @b7end dc.b $4,$4a ; offset to this patch dc.b @b12end-@b12 @b12 dc.b $4c,$8d,$5f ; jump to fix at address $5f8d= $56aa(base) + $8e3(offset) ; calls patch to disable CTS interrupts if we get them too fast since ; patch wouldn't fit inline @b12end dc.b $07,$36 ; offset to this patch dc.b @b18end-@b18 @b18 dc.b $4c,$0a,$60 ; jump to fix at address $600a= $56aa(base) + $960(offset) ; calls patch to set bit 3 of AsyncErr if break received since ; patch wouldn't fit inline @b18end dc.b $07,$55 ; offset to this patch dc.b @b14end-@b14 @b14 dc.b $4c,$d7,$5f ; jump to fix at address $5fd7= $56aa(base) + $92d(offset) ; calls patch to set clock divide bits for WR4 since ; patch wouldn't fit inline @b14end dc.b $07,$79 ; offset to this patch dc.b @b16end-@b16 @b16 dc.b $4c,$f6,$5f ; jump to fix at address $5ff6= $56aa(base) + $94c(offset) ; fixes the code at DoPollEvent to compare to DMA count, ; not the DMA address. @b16end dc.b $08,$80 ; offset to this patch dc.b @b8end-@b8 ; number patchbytes @b8 dc.b $AD,$2B,$F0 ; lda DMACntLo dc.b $C9,$FB ; cmp #$FB dc.b $60 ; rts ; fixes DMA code to check for bytes left over in the SCC fifo @b8end dc.b $08,$86 ; offset to this patch dc.b @b9end-@b9 ; number patchbytes @b9 dc.b $AD,$2A,$F0 ; lda DMA_AddrHi Set up temp pointer dc.b $85,$AF ; sta TempBufPtr+1 into the DMA buffer. $aa(base)+$05(offset) dc.b $64,$AE ; stz TempBufPtr $aa(base)+$04(offset) dc.b $AC,$29,$F0 ; ldy DMA_AddrLo Get the byte offset into buffer ;@NxtChar dc.b $AD,$40,$F0 ; lda SCC_Ctl Read RRO dc.b $29,$01 ; and #Rx_CharAvail is there a receive char? dc.b $F0,$2B ; beq @Done no chars, then we are done dc.b $AD,$42,$F0 ; lda SCC_Data get the data char dc.b $91,$AE ; sta (TempBufPtr),y store char in buffer $aa(base)+$04(offset) dc.b $84,$D4 ; Sty HoldDMACount $aa(base)+$2a(offset) dc.b $C8 ; iny ; if any errors reset the scc dc.b $A9,$01 ; lda #1 read RR1 dc.b $8D,$40,$F0 ; sta SCC_Ctl dc.b $AD,$40,$F0 ; lda SCC_Ctl dc.b $29,$70 ; and #$70 parity, overrun, framing err? dc.b $85,$B3 ; sta HoldRR1 save errors $aa(base)+$09(offset) dc.b $F0,$E3 ; beq @NxtChar no get the next char ; Suppose Special Rx Int has turned off DMA. ; There could be chars in FIFO after interrupt. ; Driver only flags errors on last byte DMAd in. ; To insure this, drop all bytes after error. ; SHOULD BE FIXED WHEN DRIVER REVVED ;@ResetSCC dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$40,$F0 ; sta SCC_Ctl dc.b $AD,$42,$F0 ; lda SCC_Data flush SCC FIFO dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$40,$F0 ; sta SCC_Ctl dc.b $AD,$42,$F0 ; lda SCC_Data flush SCC FIFO dc.b $A9,$30 ; lda #$30 send err reset to SCC dc.b $8D,$40,$F0 ; sta SCC_Ctl ;@Done dc.b $4C,$05,$58 ; jump back to our regularly scheduled code $56aa(base) +$15B(offset) ; fixes ctrl calls to wait for all sent from SCC before executing <37> @b9end dc.b $08,$C6 ; offset to this patch <37> dc.b @b10end-@b10 ; number patchbytes <37> ; <37> ;@wait_more <37> @b10 dc.b $78 ; sei turn off interrupts <37> dc.b $A9,$01 ; lda #$01 read RR1 <37> dc.b $8D,$40,$F0 ; sta SCC_Ctl <37> dc.b $AD,$40,$F0 ; lda SCC_Ctl <37> dc.b $58 ; cli turn on interrupts <37> dc.b $6A ; ror a test the all sent bit, bit 0 <37> dc.b $90,$F3 ; bcc @wait_more <37> dc.b $7C,$88,$5E ; jmp (ControlDispatch,x) <37> ; ControlDispatch at $5E88 = $56aa(base) + $7DE(offset) <37> ; fixes ctrl calls to wait for all sent from SCC before executing <37> @b10end dc.b $08,$D6 ; offset to this patch <37> dc.b @b11end-@b11 ; number patchbytes <37> ; <37> ; ctsFlag = 0 on input. Acc holds RRO value <37> @b11 dc.b $29,$A0 ; and #$A0 isolate DTR and break bits <37> dc.b $85,$AD ; sta holdRR0 $aa(base) + $17(offset) <37> dc.b $29,$20 ; and #$20 is DTR asserted? <37> dc.b $F0,$02 ; beq @1 yes, then CTSFlag stays 0 <37> dc.b $C6,$C1 ; dec CTSFlag no, then CTSFlag goes to $FF CTSFlag at $aa(base) + $17(offset) dc.b $4C,$AE,$57 ;@1 jmp backToCode at $57AE = $56aa(base) + $104 offset <37> ; adds external clocking control into bit 6 of the ; CtlOptions byte for control call 16 @b11end dc.b $08,$e3 ; offset to this patch dc.b @b13end-@b13 ; number of patchbytes @b13 dc.b $a5,$c8 ;lda CtlOptions get old clk bit $aa(base)+$1e(offset) dc.b $29,$40 ;and #clkMsk bit 6 dc.b $aa ;tax dc.b $ad,$e4,$02 ;lda C16_CtlOptions get new byte $200(base)+$e0(msg#)+$04 dc.b $a8 ;tay dc.b $29,$40 ;and #clkMsk bit 6 dc.b $85,$c8 ;sta CtlOptions $aa(base)+$1e(offset) dc.b $8a ;txa dc.b $c5,$c8 ;cmp CtlOptions is old clock bit same as new? $aa(base)+$1e(offset) dc.b $d0,$05 ;bne @switchIt nope, switch clk mode dc.b $84,$c8 ;@out sty CtlOptions store new byte $aa(base)+$1e(offset) dc.b $4c,$d4,$59 ;jmp DoControlEvent_Return $56aa(base) +$32a(offset) dc.b $98 ;@switchIt tya dc.b $0a ;asl shift into minus bit dc.b $30,$14 ;bmi @ext dc.b $a9,$50 ;@int lda #intClkSrc set int clk params dc.b $8d,$6c,$5e ;sta WR11_Data $56aa(base) +$7c2(offset) dc.b $a9,$01 ;lda #BRGEnbl dc.b $8d,$72,$5e ;sta WR14_Data $56aa(base) +$7c8(offset) dc.b $ad,$62,$5e ;lda WR4_data set clock divide to 16 $56aa(base) +$7b8(offset) dc.b $09,$40 ;ora #dvd16Msk clock divide bit is bit 6 dc.b $8d,$62,$5e ;sta WR4_data in WR4 $56aa(base) +$7b8(offset) dc.b $80,$12 ;bra @load dc.b $a9,$28 ;@ext lda #extClkSrc set ext clk params dc.b $8d,$6c,$5e ;sta WR11_Data $56aa(base) +$7c2(offset) dc.b $a9,$00 ;lda #BRGDsbl dc.b $8d,$72,$5e ;sta WR14_Data $56aa(base) +$7c8(offset) dc.b $ad,$62,$5e ;lda WR4_data set clock divide to one $56aa(base) +$7b8(offset) dc.b $29,$BF ;and #dvdOneMsk clock divide bit is bit 6 dc.b $8d,$62,$5e ;sta WR4_data in WR4 $56aa(base) +$7b8(offset) dc.b $5a ;@load phy ; save the control byte dc.b $64,$b8 ;stz HWHSenable disable hardware handshaking $aa(base)+$0e(offset) dc.b $20,$49,$5e ;jsr DoInitSCC $56aa(base) +$79f(offset) dc.b $7a ;ply dc.b $80,$c8 ;bra @out ; patch to set bit 3 of AsyncErr if break received @b13end dc.b $09,$2d ; offset to this patch dc.b @b15end-@b15 ; number of patchbytes @b15 dc.b $8a ;txa a=changed bits dc.b $30,$03 ;bmi @change break changed dc.b $4c,$02,$5e ;@done jmp ExtStatEnd $56aa(base) +$758(offset) dc.b $a5,$ad ;@change lda LastRRO get break value $aa(base)+$03(offset) dc.b $30,$05 ;bmi @post break was set dc.b $ad,$42,$f0 ;lda SCC_Data break unset, discard null dc.b $80,$f4 ;bra @done dc.b $a9,$08 ;@post lda #$08 set bit 3 dc.b $24,$bc ;bit ResetAsyncErr has host read stats $aa(base)+$12(offset) dc.b $10,$04 ;bpl @unread dc.b $64,$bd ;stz AsyncErr $aa(base)+$13(offset) dc.b $64,$bc ;stz ResetAsyncErr $aa(base)+$12(offset) dc.b $05,$bd ;@unread ora AsyncErr $aa(base)+$13(offset) dc.b $85,$bd ;sta AsyncErr $aa(base)+$13(offset) dc.b $80,$e4 ;bra @done ; patch to set clock divide bits for WR4 and disable HWHS for external clocking @b15end dc.b $09,$4c ; offset to this patch dc.b @b17end-@b17 ; number of patchbytes ;acc=%0400xxxx default to divide-by-16 @b17 dc.b $da ;phx save reg dc.b $aa ;tax save acc dc.b $a5,$c8 ;lda CtlOptions are we ext clk'd? $aa(base)+$1e(offset) dc.b $0a ;asl shift ext clk bit into minus bit dc.b $30,$03 ;bmi @ext yes, go fix clk dvd bits dc.b $8a ;@int txa no, just restore acc dc.b $80,$03 ;bra @done dc.b $8a ;@ext txa restore acc dc.b $29,$0f ;and #$0f fix clk dvd bits to div-by-one ; dc.b $64,$b8 ;stz HWHSenable disable hardware handshaking $aa(base)+$0e(offset) dc.b $fa ;@done plx restore reg dc.b $8d,$62,$5e ;sta WR4_Data and write out WR4 value $56aa(base) +$7b8(offset) dc.b $4c,$26,$5e ;jmp FinishInitSCC and finish up $56aa(base) +$77c(offset) ; patch to disable CTS interrupts if we get them too fast @b17end dc.b $09,$60 ; offset to this patch dc.b @b19end-@b19 ; number of patchbytes @b19 dc.b $a5,$dd ;lda CTSCount exceeded 5 transitions in 1 ms? $aa(base)+$33(offset) dc.b $c9,$05 ;cmp #5 dc.b $90,$0c ;bcc @fixTime no, then fix the time dc.b $a9,$0f ;lda #15 yes, then disable CTS interrupts dc.b $8d,$40,$f0 ;sta SCC_Ctl write to WR15 dc.b $a9,$80 ;lda #$80 bit 5 clear to disable CTS ints dc.b $8d,$40,$f0 ;sta SCC_Ctl dc.b $80,$12 ;bra @out go finish in ROM ;@fixTime dc.b $a5,$de ;lda LastTime get last count (remember it's a countdown timer) $aa(base)+$34(offset) dc.b $ed,$11,$f0 ;sbc TimerCounterH time elasped > threshold? dc.b $c9,$08 ;cmp #threshold threshold is 1 msÅ 8 ticks dc.b $90,$07 ;bcc @update no, just increment <138> dc.b $ad,$11,$f0 ;lda TimerCounterH yes, set new last time dc.b $85,$de ;sta LastTime $aa(base)+$34(offset) dc.b $64,$dd ;stz CTSCount and restart count $aa(base)+$33(offset) ;@update inc the transition count dc.b $e6,$dd ;inc CTSCount $aa(base)+$33(offset) ;@out dc.b $64,$c1 ;stz CTSFlag do stuff we stomped on in overpatch $aa(base)+$17(offset) dc.b $a5,$ad ;lda RRO $aa(base)+$03(offset) dc.b $4c,$e4,$5d ;jmp BackToROM finish up in ROM $56aa(base) +$73a(offset) ; patch to close to time-out on the all sent loop @b19end dc.b $09,$8b ; offset to this patch dc.b @b22end-@b22 ; number of patchbytes @b22 dc.b $a5,$FF ;ldx #$FF load up $FFFF for timeout loop dc.b $c9,$FF ;ldy #$FF 24 cycles * .5 us/cycle * 64K Å .8 seconds dc.b $a9,$01 ;@loop lda #1 read from RR1 dc.b $8d,$40,$f0 ;sta SCC_Ctl dc.b $ad,$40,$f0 ;lda SCC_Ctl dc.b $6a ;ror get all sent bit in carry dc.b $b0,$06 ;bcs @done if allsent, then outta here dc.b $ca ;dex not allsent, so keep waiting dc.b $d0,$f2 ;bne @loop dc.b $88 ;dey dc.b $d0,$ef ;bne @loop dc.b $4c,$24,$59 ;@done jumpBacktoClose $56aa(base) +$27a(offset) ; patch to start the IOP timer @b22end dc.b $09,$a3 ; offset to this patch dc.b @b24end-@b24 ; number of patchbytes @b24 dc.b $9c,$2c,$f0 ;stz DMA_CntHi do code we stomped with patch dc.b $a9,$ff ;lda #$FF dc.b $8d,$11,$f0 ;sta TimerCounterH get the timer started (just need to write to it) dc.b $4c,$45,$57 ;@done jumpBacktoOpen $56aa(base) +$09b(offset) @b24end dc.w 0 ; filler ;_______________________________________________________________________ ; ; Routine: IOPMoveData ; Inputs: A0 - pointer to IOPMoveInfo paramater block ; Outputs: D0 - Result Code (NoErr/paramErr) ; Destroys: A0, A1, D0, D1, D2 ; Calls: CopyToIOP, CopyFromIOP, CompareWithIop ; Called by: OsTrap Dispatch Table ; ; Function: Moves, or compares data between the IOP and the Host ; memories, using the parameters inthe IOPMoveInfo parameter ; block. Also has a special mode to apply patches to IOP ; memory as an atomic operation. ; ;_______________________________________________________________________ with IOPMoveInfo IOPMoveData ; We can use the new protection scheme, if the first two instructions in the <115> ; patch are a bra.s followed by the jump to the old rom address. bra.s @protection @NoPatch BackToTrap oldIOPMoveData ; no, then just execute ROM Code @protection ; are we coming from the IOP serial driver open call? cmpRA From_IOPOpenCall,OSTrapReturnAddressDepth(sp) bne.s @NoPatch @chkA lea DrvrAFix,a1 ; get address of our patchbytes cmpi.w #DrvrAPicAddr,imIOPAddr(a0) beq.s @doPatch @chkB lea DrvrBFix,a1 ; get address of our patchbytes cmpi.w #DrvrBPicAddr,imIOPAddr(a0) bne.s @noPatch ; not our drivers? well, don't patch then @doPatch ; Call the ROM to download the unpatched image _IOPMoveData ; now patch the downloaded driver. Each section of patch bytes ; is preceded by an offset word and a count byte. ; a2 has start addr of our patch bytes. moveq #numpatches-1,d1 ; number of patches (minus 1 for dbra) move.w imIOPAddr(a0),d2 ; get base addr of downloaded driver moveq #0,d0 ; for additions move.w d0,imByteCount(a0) ; clear out high byte of count @patchloop move.w d2,d0 ; get base addr of downloaded driver add.w (a1)+,d0 ; add in offset to patch start move.w d0,imIOPAddr(a0) ; set dest addr for patchbytes move.b (a1)+,imByteCount+1(a0) ; number patchbytes move.l a1,imHostAddr(a0) ; set source addr of patchbytes adda.w imByteCount(a0),a1 ; increment ptr past patchbytes _IOPMoveData ; download the patch bytes dbra d1,@patchloop ; go on to next patch ; note that when dbra expires, d0 will contain output for last IOPMove data ; performed, which is an approximation of what should be in d0 ; note also that we've trashed the IOPMgr PB fields, but since the driver ; never refers to them again, and in the interest of STAYING SMALL, we'll ; ignore it. rts endwith endproc ; ; ; End of Patch To IOPMoveData Trap to Fix IOP Serial Driver Control Call Bug ;____________________________________________________________________________ EH <33> ;____________________________________________________________________________ NewFLineRoutine PROC EXPORT EXPORT OldFLineRoutine cmp.w #$002C,6(sp) ;make sure that this is our stack frame. If not bne.s OldFLineRoutine ;..then jump thru the old exception vector MOVEM.L A0/D0,-(SP) ; Save these two registers <107> move.l 8+2(sp),a0 ;get the address of the instruction word that caused move.w (a0),d0 ;..this exception. Then copy the word into DO and.w #%0000111000000000,d0 ;get just the co-processor ID from the instruction cmp.w #%0000001000000000,d0 ;if itÕs not one, then it isnÕt a FPU instruction bne.s stackError ;..so return thru the old FPU replacement routine lea continue,a0 ;change the saved return address to our continuation move.l a0,8+2(sp) ;..point and return out of exception mode. MOVEM.L (SP)+,A0/D0 ; Restore the regs <107> rte continue move.w #dsNoFPU,d0 ;this displays the dialog that lets the user _SysError ;..return to the finder (FPU type) stackError MOVEM.L (SP)+,A0/D0 ; Restore the regs <107> OldFLineRoutine JMP $12345678 ; This gets filled in at install time <107> ENDP ;____________________________________________________________________________ ;---------------------------------------------------------------------------------------------------- ; This routine will read the Time LowMem global without ; issuing the call to Egret. Egret will keep the time updated ; in the background without needing system intervention. The ; Egret will post a unsolicited time packet which will update the ; LowMem global when it is not able to post the Tick packet ; within one second. ;---------------------------------------------------------------------------------------------------- RdDateTimePtch PROC EXPORT move.l Time,(a0) ; return the time to the caller clr.w d0 ; return no error <74> rts ENDP ;---------------------------------------------------------------------------------------------------- ;---------------------------------------------------------------------------------------------------- ; This patch fixes a bug in Egret Manager TickHandler routine. <89><101> ; The routine will store the 32 bit time passed by Egret to the ; system if the packet is a ReadRealTime into TIME lowmen. ; Decrements the lowmem Time by one to compensate for the interrupt ; Handler incrementing it by one and then calls the LVL1DT interrupt ; handler. ; If the packet is a tick packet then it calls LVL1DT to increment the ; time setting and check the alarm state. ; ; Entry: ; A1.l = Points at Response data buffer At Flags byte: ; ; +---------+----------+---------+---------+---------//-------+ ; | Attn | Pkt Type | Flags | Pkt Cmd | 8 Bytes data Max | ; +---------+----------+---------+---------+---------//-------+ ; ; A2.l = Points at Egret Manager Globals ; ;---------------------------------------------------------------------------------------------------- TickHandler PROC EXPORT WITH respPacket,EgretGlobals subq.l #2,a1 ; point to the beginning of the Response data buffer cmpi.b #TickPkt,RespType(a1) ; Check the Packet type beq.s @CallLVL1DT ; If tick packet just call LVL1DT handler ; ; The packet was a readTime packet from Egret. Update the Time lowmem and ; adjust it to compensate for the increment in the LVL1DT handler. ; move.l RespData(a1),Time ; write the new time in lowmem TIME subq.l #1,Time ; adjust to compensate for increment in LVL1DT ; ; JUMP to routine pointed to by the Contents of LVL1DT ; @CallLVL1DT move.l VIA,a1 ; point to the VIA move.l LVL1DT,a0 ; get vector jmp (a0) ENDWITH ENDP ;---------------------------------------------------------------------------------------------------- ; This patch saves the processor status register, calls the <101> SAM ; ShiftRegIrq routine in Egret Manager and then restore the ; status register and exits. ;---------------------------------------------------------------------------------------------------- EgretIRQPtch PROC Export EXPORT Vector_Hold move.w sr,-(sp) ; save the status register @IrqJmp jsr $10000000 ; this JSR will call the routine pointed to by the old ; contents of LVL1DT+8 vector (ShiftRegIrq) move.w (sp)+,sr ; restore the status register rts Vector_hold equ @IrqJmp+2 ; point to the operand field ENDP ;---------------------------------------------------------------------------------------------------- ; This patch fixes possible loss of data to the SCC by calling <101> SAM ; the Poll Proc when SCC data is available. ;---------------------------------------------------------------------------------------------------- EgretDispatchPtch PROC Export EMJmpAddr equ $14868 ; Offset added to Contents of Rombase for jmp back to Rom Egret Manager ; <104> removed declarations for Pollstack,PollProc, and HiIntMsk with EgretPB, EgretGlobals bsr CheckPacket ; Validate the Packet type & command byte bne @done ; Exit if Error packet movea.l EgretBase,a2 ; a2 gets ptr to globals movea.l VIA,a1 ; a1 points to VIA base cmpi.b #specialPkt,pbCmdType(a0) ; is this a special command? bne.s @EgretRestart move.l a0,ADBpb(a2) ; yes, is to set the ADB param block (for autopoll data). bra @done @EgretRestart move.w sr,-(sp) ; save SR ori.w #hiIntMask,sr ; mask interrupts btst.b #xcvrSes,vBufB(a1) ; does Egret want to abort? beq.s @abort ; yes, wait for it to go away bset.b #busy,flags(a2) ; not an abort, mark that we're busy. beq.s @sendPackType ; we were not busy before, so try to send the first byte @abort move.w (sp)+,sr ; we were busy, enable interrupts bsr pollByte ; poll shift reg, calling handler if interrupts masked bra.s @EgretRestart ; and keep waiting for busy to go away... @sendPackType ; interrupts masked here bset.b #sysSes,vBufB(a1) ; assert System Session (we're starting command packet) bset.b #SRdir,vACR(a1) ; switch to output move.b pbCmdType(a0),vSR(a1) ; send command packet to shift reg bset.b #viaFull,vBufB(a1) ; tell Egret we sent it ;_________________________ ; If PollProc exists, Poll the SCC and save any available data ; When the shift register irq comes in call the PollProc ; then process the shift register irq data ; movem.l d0/d1/a0-a4/a6,-(sp) ; save some registers lea @zero, a3 move.l sp,PollStack ; Pointer to buffer for polled bytes btst.b #0,SccIopFlag ; Check if we are in IOP mode (On Eclipse...) beq.s @wait tst.l PollProc ; Check for a Poll Proc available beq.s @wait ; movea.l SccRd,a3 ; SCC may have data to get movea.l a3,a6 addq.l #Actl,a3 ; Point to data available register (RR0) addq.l #AData,a6 ; Point to the SCC data register @wait btst.b #RxCa,(a3) ; Test for SCC data available beq.s @2 move.b (a6),-(sp) ; Push the data on the stack @2 btst.b #vShift,vIFR(a1) ; now wait for shift reg IRQ beq.s @wait cmpa.l PollStack,SP ; Is there any poll data beq.s @NoSCCData ; ; We have SCC data and a Poll Proc to call. Go call it ; pea @NoSCCData ; Return addr for PollProc move.l PollProc,-(SP) ; Point to the PollProc rts ; Call the PollProc @zero dc.w 0 @NoSCCData movem.l (sp)+,d0/d1/a0-a4/a6 ; restore work registers @VsrIrq btst.b #xcvrSes,vBufB(a1) ; did Egret abort? bne.s @accepted ; no, then it will accept our packet bclr.b #SRdir,vACR(a1) ; yes, switch back to input bclr.b #sysSes,vBufB(a1) ; ack the abort bsr.s CallShiftRegIRQ ; handle it bra.s @abort ; and wait @accepted JmpROM EMJmpAddr ; Jump back into Rom Egret Manager <107> @done moveq #0,d0 ; no errors for now rts ;________________________________________________________________________________________________ CallShiftRegIRQ movem.l a0-a3/d0-d3,-(sp) ; save regs like interrupt handler does movea.l Lvl1DT+8,a0 ; get the shift reg IRQ handler jsr (a0) ; call it movem.l (sp)+,a0-a3/d0-d3 ; restore regs rts ;________________________________________________________________________________________________ ; Routine: PollByte ; ; Function: This routine checks to see if level 1 interrupts are masked, exits if not. ; If masked, it polls the flag register for a shift reg interrupt, and ; calls the handler if found. ; ; Inputs: a1 - VIA base ptr ; a2 - globals pointer ; ; Outputs: none ; ; Destroys: d0,d1 ;________________________________________________________________________________________________ PollByte move.w sr,d0 ; get 68xxx interrupt mask andi.w #hiIntMask,d0 ; are we at interrupt level? beq.s @exit ; no, just exit btst.b #vShift,vIFR(a1) ; yes, poll the shift reg beq.s @exit ; no shift reg interrupt, return bsr.s CallShiftRegIRQ ; yes, handle it @exit rts ;________________________________________________________________________________________________ ; ; Validate the Packet type and Command if Pseudo Pkt. ; ; If an error is encountered with the packet a error packet will be ; built in the pbParam area of the parameter block as if Egret had ; returned the packet. Also, the pbResult field will contain a System ; error code for bad parameter block format. ; ; Entry: A0 = Parameter block pointer ; ; Exit: sr.Z = 0 if valid packet/pseudocmd nonzero otherwise ; _________________________________________________________________________________________________ CheckPacket cmp.b #SpecialPkt,pbCmdType(a0) ; could be an ADB initialization packet beq.s @OkExit cmp.b #ErrorPkt,pbCmdType(a0) ; ADB ($00) and Pseudo ($01) only bhi.s @BadPkt ; Invalid packet cmp.b #PseudoPkt,pbCmdType(a0) ; Check for pseudo commands bne.s @OkExit ; On Pseudo Packets check the command cmp.b #MaxPseudoCmd,pbCmd(a0) ; Validate the Pseudo command number <bg> bls.s @OkExit @BadPseudo move.w #InvPseudo,pbParam(a0) ; report a pseudo command error move.w pbCmdType(a0),pbParam+2(a0) ; we are faking an error packet bra.s @ErrorExit @BadPkt move.w #InvPkt,pbParam(a0) ; report invalid packet error move.w pbCmdType(a0),pbParam+2(a0) ; we are faking an error packet @ErrorExit move.w #paramErr,pbResult(a0) ; parameter error in result field bra.s @Exit @OkExit move.w #0,pbResult(a0) ; So far packet OK @Exit rts ENDWITH ENDPROC ;------------------------------------------------------------------ ;__________________________________________________________________________ <101> <BEGIN> ; PACK 4 overpatch code for MC68020 software SANE to correct binary-to ; decimal conversion bug which may cause an extra byte to be written to ; the destination decimal record. ; ; Copyright © 1990 Apple Computer, Inc. All rights reserved. ; Written by: Jon Okada. ; ; The overpatch code intercepts every PACK 4 call and determines if the SANE ; operation requested is a binary-to-decimal conversion. If it is not, ; control is passed to the original, underlying SANE engine. If it is, ; the SANE engine is called with output directed to a local decimal record, ; which is copied to the user's destination decimal record, taking care not ; to exceed the desired precision. ; ; Entry point is FP020PATCH. Upon entry, the stack is as follows: ; ; &return < SANE opword (2 bytes) < &dst < &src < &src2, ; ; where &src and &src2 may not be present, depending on the SANE operation ; requested by the caller. PACK4inROM equ $73940 ;Offset from ROMBase to non-fpu PACK4. FP020Patch PROC MOVE.L D0,-(SP) ; STACK: D0sv < &ret < opcode < etc. MOVEQ #$1F,D0 ; D0 <- opcode mask AND.W 8(SP),D0 ; isolate opcode CMPI.B #$0B,D0 ; binary-to-decimal conversion? MOVEM.L (SP)+,D0 ; (restore register) BEQ.S @DOBINDEC ; yes jmpROM PACK4inROM ; Let PACK4 handle call. ; ; Binary-to-decimal conversion is accomplished via the underlying PACK 4 ; implementation, but the result is directed to an intermediate buffer ; @DOBINDEC: LINK A6,#-28 ; reserve space for decimal record MOVEM.L D0/A0-A1,-(SP) ; save registers LEA -28(A6),A0 ; A0 <- addr of local decimal record MOVE.L 18(A6),-(SP) ; push &src2 MOVE.L 14(A6),-(SP) ; push &src MOVE.L A0,-(SP) ; push local decimal record addr MOVE.W 8(A6),-(SP) ; push opword jsrROM PACK4inROM ; call pack4. MOVEA.L 10(A6),A1 ; A1 <- &dst MOVE.L (A0)+,(A1)+ ; copy sign/exp to dst record CLR.W D0 ; clear counter MOVE.B (A0)+,D0 ; read decimal string length MOVE.B D0,(A1)+ ; write it to dst record SUBQ #1,D0 ; initialize loop counter @1: MOVE.B (A0)+,(A1)+ ; copy string to dst record DBRA D0,@1 MOVEM.L (SP)+,D0/A0-A1 ; restore registers UNLK A6 ; unlink MOVE.L (SP),14(SP) ; move return addr up ADDA.W #14,SP ; kill arguments on stack RTS ; return to caller ENDP ;__________________________________________________________________________ <101> <END> ;__________________________________________________________________________ <xyz> <cch 8/6/91> ; Routine: GetPageDescPatch ; ; Function: This routine is a patch to the GetPageDescPatch, which is ; an internal routine called by LockMemory and WriteProtectMemory. ; The bug it fixes is that the page descriptor pointer is ; converted from physical to logical twice in GetPageDesc. ; This patch is only needed on 68030 machines with ROM version $67C, ; sub-version 1.5 without VM. ; ; How: This is a come-from patch to the _Gestalt trap, which is ; called by the GetMMUInfo routine in GetReal.a. If the ; come-from patch is triggered, and the GetPageProc flag on ; the stack is set, the return address from GetPageDescProc is ; replaced with the address of a routine which fixes the ; result and returns to the original caller. ; ; Note: This patch is NOT reentrant. This is OK, however, since ; _LockMemory may not be called at interrupt level, which is ; the only way reentrancy could occur with this routine. ; ; Inputs: A0/D0 - parameters to gestalt ; (sp).L - address routine was called from plus 2 ; ; Outputs: none ; ; Destroys: a1 ;__________________________________________________________________________ GetMMUInfoPatchAddr EQU $40886d1e ; return address when Gestalt is called SetCacheInhibitAddr EQU $408873a2 ; return address when correct swapMMUMode is called getPageDesc EQU 7 ; getPageDesc selector GetPageDescPatch PROC EXPORT MemDispPatch cmp.l #GetMMUInfoPatchAddr,$18(sp) ; check addr _Gestalt was called from bne.s @skipPatch ; IF we had a match THEN cmp.w #getPageDesc,$4e(sp) ; check if user wants the page descriptor bne.s @skipPatch ; IF user wants a page descriptor THEN lea GetPageDescRetAddr,a1 ; where to save getPageDesc return addr move.l $50(sp),(a1) ; save return address to getMMUInfo lea fixResult,a1 ; get address of routine to fix result move.l a1,$50(sp) ; stuff it in the stack frame ; ENDIF ; ENDIF @skipPatch BackToTrap origGestalt ; jump to original Gestalt trap ;----------------- ; getPageDesc fix up routine ;----------------- fixResult sub.l Phys2Log,a0 ; fix up result jmp ([GetPageDescRetAddr]) ; jump back into normal code stream GetPageDescRetAddr ; return address for getMMUInfo dc.l $87654321 ;__________________________________________________________________________ <10> <cch 8/6/91> ; Routine: MemDispPatch ; ; Function: This routine is a patch to the _VM trap. It fixes a bug ; in the Terror ROM which occurs when the page the stack is ; is is marked uncacheable in the MMU tables, then a subsequent ; PFLUSH'd is executed with data still in the cache. ; ; How: This is a simple patch which is installed only on 68030's when ; VM is not installed on Terror ROMs, version $67C, sub-version 1.5. ; It disables the data cache during all _VM calls with the exception ; of _GetPhysical. ; ; Inputs: A0-A1/D0 - parameters to _VM call ; ; Outputs: none ; ; Destroys: d1 ;__________________________________________________________________________ GetPhysSel EQU 5 MemDispPatch cmp.b #GetPhysSel,d0 ; check for get physical selector beq.s @skipPatch ; IF this is anything but getPhysical THEN movec cacr,d1 ; get cache value move.w d1,-(sp) ; save cache state on stack andi.w #$001f,d1 ; we want to save current instruction cache bits addi.w #$0800,d1 ; value to flush data cache movec d1,cacr ; flush and disable data cache jsrTrap origMemDisp ; do original call move.w (sp)+,d1 ; get original cache state movec d1,cacr ; reenable caches rts ; return to caller @skipPatch ; ELSE BackToTrap origMemDisp1 ; do original call ; ENDIF ENDP ;__________________________________________________________________________ <10> <end> ;__________________________________________________________________________ <13><cch 8/28/91> ; Routine: FP68k patch for Terror-based 030 machines ; ; Inputs: stack has: <RET> <OPWORD> <ADRS1> <ADRS2> <ADRS3> ; ; Destroys: d1 ;__________________________________________________________________________ QADDX EQU $408EC1E4 FP881CASE EQU $408EB50C OMEGA881BASE EQU $408EB1FC OPERRHANDLER EQU $408EDCE6 FP68kPatch PROC EXPORT tst.b MMU32bit ; Are we in 24 bit mode? beq.s @inMode24 ; -> Yes @inMode32 cmp.l #$40800000,(SP) ; check address we're patching bhs.s @dontBackpatch ; IF we're being called from ROM THEN @CallOld BackToTrap origFP68k ; do original call @inMode24 move.l D0,-(SP) ; save D0 move.l 4(SP),D0 ; get Return address in D0 and.l MaskBC,D0 ; poor man's strip address cmp.l #$00800000,D0 ; Is the PC less than ROM? movem.l (SP)+,D0 ; Restore D0 blo.s @CallOld ; -> Yes, Call the old Backpatch routine ; No. Fall into DontBackPatch @dontBackpatch ; ELSE SUBQ #2,SP ; MAKE A HOLE MOVEM.L D0/A0,-(SP) ; SAVE 2 REGISTERS LEA QADDX,A0 ; ALWAYS STORE &QADDX INTO ITS JUMP ISLAND MOVE.L A0,$0B6E ; HOME SO TRAPS HAVE A POINT OF REFERENCE MOVE.W 14(SP),D0 ; GET OPWORD INTO D0 ROL.W #8,D0 ; XXXX383F --> XXXX3F38 ... SWAP BYTES LSL.B #2,D0 ; XXXX3F38 --> XXXX3FE0 ... SCRUNCH TOGETHER BFEXTS D0{18:9},D0 ; D0 ::= INDEX := 8 * OPCODE + FORMAT MOVE (FP881CASE,D0*2),D0 ; DO := OFFSET AT INDEX IN TABLE LEA OMEGA881BASE,A0 ; GET OMEGABASE ADD.L D0,A0 ; GO DO IT MOVE.L 10(SP),12(SP) ; SLIDE RETURN ADDRESS ON TOP OF OPWORD MOVE.L A0,8(SP) ; FILL RTS HOLE PROVIDED ABOVE LEA OPERRHANDLER,A0 ; INSTALL OPERR TRAP HANDLER INTO MOVE.L A0,$0D0 ; LOW-MEM VECTOR (SEE TABLE 7-6 882 MAN) FMOVE.L FPCR,D0 ; ENABLE OPERAND ERROR TRAPPING BSET #13,D0 FMOVE.L D0,FPCR MOVEM.L (SP)+,D0/A0 ; RESTORE 2 REGISTERS RTS ; DISPATCH ; ENDIF ENDP ;__________________________________________________________________________ <13> ;---------------------------------------------------------------------------------------------------- ;========================================================================================= ;========================================================================================= ;= = ;= END OF RESIDENT PATCH CODE SECTION = ;= = ;========================================================================================= ;========================================================================================= EndOfPatch PROC EXPORT ;========================================================================================= ;========================================================================================= ;= = ;= PATCH INSTALL CODE IS BELOW THIS POINT = ;= = ;= Patches may fall into two different categories: = ;= 1. Required for both the Mac OS and A/UX = ;= 2. Required only for the Mac OS = ;= = ;= Macintosh OS-only patches are usually characterized by code which touches hardware = ;= directly, or patches for traps not supported by A/UX. The safest thing to do when = ;= in doubt is to check with your friendly A/UX representative (only a phone call = ;= away!). A/UX loads all patches, but conditionalizes the installation code. The = ;= first section of installation code which follows, is for patches required on both = ;= Mac OS and A/UX. The second section (further down), is for Mac OS-only patch = ;= installation code. = ;= = ;= To find a place to add your Mac OS and A/UX patch, search for 'EndBothSection' = ;= = ;= To find a place to add your Mac OS only patch, search for 'EndMacSection' = ;= = ;========================================================================================= ;========================================================================================= PatchInit MAIN Export Import EndOfPatch Import StartPatch Import Cutback Import FixupTbl ; added FixupTbl <24July89smb> MOVE.L D1,-(SP) ; Save our handle <1.1-4april89-CEL> ;____________________________________________________________________________ ; Fixup patch addresses ; ; The Fixing up of addresses generated by the CMPRA, JSRROM & JMPROM ; Breaks down as follows: ; ; Addressing High Byte in Action ; mode: instruction: ; ---------- ------------ ------ ; 24 bit NZ Do not alter the address in the instruction. ; Used with CMPRA to a ROM Resource. ; 32 bit NZ Mask out 12 high bits from the address ; in the instruction, leaving a ROM offset ; Then add in ROMBase. ; either Zero Normal Case. Add ROMBase to the address ; in the instruction. ; ; Registers: ; D0: Address Size Flag. Pos: 24 bit mode; Neg: 32 bit Mode ; D1: Entry from table of offsets of locations to be fixed up ; D2: (RomBase) ; A0: Pointer into table of offsets of locations to be fixed up. ; A1: Base of fixup table; also location from which offsets are computed. move.l RomBase, D2 moveq #-1, D0 _StripAddress lea FixupTbl, A0 move.l A0, A1 FixUpLooP move.l (A0)+, D1 beq.s @1 tst.b 0(A1, D1.L) beq.s @AddIt tst.l D0 ; If Address has any high bits set bpl.s FixUpLoop ; 24 Bit mode; Dont alter this one. and.w #$f, 0(A1, D1.L); 32 Bit Mode; Clear High 12 bits. @AddIt add.l D2, 0(A1, D1.L) ; Normal Case: Add in ROMBase bra.s FixUpLoop @1 move.l jCacheFlush, a0 ; flush caches (make sure tables are consistent) jsr (a0) ;========================================================================================= ;========================================================================================= ;= = ;= PATCH INSTALLATION CODE FOR BOTH MAC OS AND A/UX STARTS HERE = ;= = ;= Please be neat and follow the format = ;= = ;========================================================================================= ;========================================================================================= ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ GGD <8.4> ; Fix _StripAddress to not check the MMStartMode bit of MMFlags at runtime, ; because the memory manager changes that bit when accessing ROM resources, ; and if an interrupt occurs while the bit is changed, calling _StripAddress ; from an interrupt routine (eg VBL task) might not work correctly. To correct ; this, we will install either a 24 or 32 bit version of _StripAddress once ; and eliminate the runtime check, making the code both faster and correct. ; NOTE: This is installed early in the patch file to reduce the chances of ; encountering this bug while installing the remaining patches. StripAddressOld equ $0000D764 ; ROM offset of old StripAddress StripAddress24 equ $0000D76C ; ROM offset of new 24 bit mode version StripAddress32 equ $0000D770 ; ROM offset of new 32 bit mode version FixStripAddress lea OsTable+($55*4),a0 ; point to dispatch table entry (trap A055) CmpRA StripAddressOld,(a0) ; see if it already patched (by A/UX maybe?) bne.s @Done ; if already patched, don't patch again moveq.l #StripAddress24-StripAddressOld,d0 ; get adjustment for 24 bit mode btst.b #Systemis24bit,SystemInfo ; what kind of OS are we running bne.s @patchIt ; if 24 bit OS, change it to 24 bit StripAddress addq.l #StripAddress32-StripAddress24,d0 ; if 32 bit, adjust offset to 32 bit entry @patchIt add.l d0,(a0) ; update the dispatch table entry @done ; all done <8.4> ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ <8> ; The TERROR ROM does not distinguish between a Tim and a Tim LC. We have to. ; We need to fix boxFlag if we're running on a Tim LC (a Tim w/o an FPU) ; MOVE.L ROMBase,A0 ; Get Bas o ROM CMPI.B #TERRORMinorVers,18(A0) ; Is this a TERROR 67C ROM? BNE.S @DoneTimLC ; -> Nope, don't futz with anything CMPI.B #boxPowerBook170,boxFlag ; Does the ROM think this is a Tim? <146> BNE.S @DoneTimLC ; -> No, don't assume anything. Bail out. MOVE.W HwCfgFlags,D0 ; Get them Hardware Config Flags BTST.L #hwCbFPU,D0 ; Does we gots an FPU? BNE.S @DoneTimLC ; -> Yes, we're not on a Tim LC. Exit. BTST.L #hwCbPwrMgr,D0 ; Do we have a PowerMgr? BEQ.S @DoneTimLC ; -> No. This is not a Portable of any kind ; We're on a TERROR $67C ROM with a PwrMgr and no FPU. MOVE.B #boxPowerBook140,boxFlag ; We're on a Tim LC. Stuff the right boxFlag <146> @DoneTimLC ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ <130><131><8><9> ; The GestaltFuction.a file is used to build a ptch for machines without ; Gestalt in ROM, but it also contains fixes and improvements that are not ; in the ci ROMs OR in the GestaltPatches.a file... The changes are not ; trivial, so weÕre punting and loading the Gestalt ptch on ci ROMs, too... ; ; Before we do, though, we throw away any storage that Gestalt has allocated. ; move.l ExpandMem,A1 move.l ExpandMemRec.emGestalt(A1),A0 ; Get the ROM GestaltÕs Storage pointer move.l A0,-(SP) ; Save the Ptr on the Stack move.l (A0),-(SP) ; Save Gestalt's Selector table handle PtchInst 5 ; Reinstall all of Gestalt ptch move.l (SP)+,A0 ; Get the old Selector Table Handle _DisposHandle ; Throw the storage away! move.l (SP)+,A0 ; Get the storage Ptr _DisposPtr ; Make it go away too! ;____________________________________________________________________________ NJC <1.6> ; ci Quickdraw ptch added 1/16/90 KON ; PtchInst 26 ; ci Quickdraw patches ;____________________________________________________________________________ DAF <1.9> ; Install window manager fixes ; InstToolTp newSetWinColor,$241 InstToolTp NewSetCtlColor,$243 ;____________________________________________________________________________ dvb <3.5> ; Palette Mgr Patches ; InstToolTp ptchInitPalettes,$290 InstToolTp ptchGetNewCWindow,$246 ;<10Nov89 KON> InstToolTp ptchPMgrDispatch,$2A2 ; Changed a couple of dispatches dvb <8.3> ; Next, fix up the dispatch table LEA pMgrProcs,A0 MOVE.L A0,D1 ; D1-> base of table SUBQ.L #1,D1 ; Less 1, 'coz lobit = patched flag MOVE #((pMgrProcsSize)/4)-1,D0 ; we'll loop for this many fields <107> (pMgrProcsSize) @FixPMLoop MOVE.L (A0),D2 ; pull entry from table BTST #0,D2 BEQ.S @FixPMLoopEnd ADD.L D1,D2 @FixPMLoopEnd MOVE.L D2,(A0)+ ; stash correct address, bump A0 DBRA D0,@FixPMLoop ; and count down, including zero. dvb <8.3> end ;__________________________________________________________________________ ;__________________________________________________________________________ FixBackSANE MOVE.L #'fpu ',D0 ; Gestalt FPU selector _Gestalt ; Do it MOVE.L A0,D0 ; Do we have an FPU? BEQ @ReallyDone ; -> No, do nothing MOVE.L ROMBase,A0 ; Get ROM base CMPI.B #TERRORminorVers,18(A0) ; Is this TERROR 067C ROM? BEQ @ReallyDone ; -> Yes, leave the ROM SANE enabled! CMPI.B #ZYDECOminorVers,18(A0) ; Is this Zydeco 067C ROM? BEQ @ReallyDone ; -> Yes, leave the ROM SANE enabled! @DisableSANEinROM MOVE.B ResLoad,-(SP) ; Save current resLoad state MOVE.W CurMap,-(SP) ; Save Current Rsrc Map refNum CLR.W CurMap ; Make the System the current Map (CurMap = 0) SF ResLoad ; Don't actually read the rsrc CLR.L -(SP) ; Result MOVE.L #'PACK',-(SP) ; Type MOVE.W #4,-(SP) ; Id 4 _Get1Resource MOVE.L (SP)+,D0 ; Did we get it? BEQ @Done ; -> No, PACK 4 is not on the disk. Exit Now! (Use the ROM SANE) MOVE.L D0,-(SP) _ReleaseResource ; Make the System SANE handle go away. (or we'll get it at the _RmveRsrc) SF ResLoad ; Don't actually read the rsrc CLR.L -(SP) ; Result MOVE.L #'PACK',-(SP) ; Type MOVE.W #5,-(SP) ; Id 5 _Get1Resource MOVE.L (SP)+,D0 ; Did we get it? BEQ @Done ; -> No, PACK 5 is not on the disk. Exit Now! (Use the ROM SANE) MOVE.L D0,-(SP) _ReleaseResource ; Make the System SANE handle go away. (or we'll get it at the _RmveRsrc) ST RomMapInsert ; Put the ROM map in first MOVE.W #1,CurMap ; Make the ROM map current CLR.L -(SP) ; Result MOVE.L #'PACK',-(SP) ; Type MOVE.W #4,-(SP) ; Id 4 _Get1Resource MOVE.L (SP)+,D0 ; Did we get it? BEQ.S @Try45 ; -> No, try 4 five. MOVE.L D0,-(SP) ; The Handle (save a copy of it) MOVE.L ROMMapHndl,-(SP) ; Save this MOVE.L #-1,ROMMapHndl ; To fool SetResAttr into letting use change ROM rsrc attrs MOVE.L D0,-(SP) ; The Handle (push it. Push it good) MOVE.W #$50,-(SP) ; A good value (Unprotected) _SetResAttrs MOVE.L (SP)+,ROMMapHndl MOVE.W #1,CurMap ; Set CurMap to the ROM rsrc map ST RomMapInsert ; Dont load the rsrc into memory _RmveResource @Try45 ST RomMapInsert ; Put the ROM map in first CLR.L -(SP) ; Result MOVE.L #'PACK',-(SP) ; Type MOVE.W #5,-(SP) ; Id 5 _Get1Resource MOVE.L (SP)+,D0 ; Did we get it? BEQ.S @Done ; -> No, exit MOVE.L D0,-(SP) ; The Handle (save a copy of it) MOVE.L ROMMapHndl,-(SP) ; Save this MOVE.L #-1,ROMMapHndl ; To fool SetResAttr into letting use change ROM rsrc attrs MOVE.L D0,-(SP) ; The Handle (push it. Push it good) MOVE.W #$50,-(SP) ; A good value (Unprotected) _SetResAttrs MOVE.L (SP)+,ROMMapHndl MOVE.W #1,CurMap ; Set CurMap to the ROM rsrc map ST RomMapInsert ; Dont load the rsrc into memory _RmveResource @Done MOVE.W (SP)+,CurMap ; Restore the current res map MOVE.B (SP)+,resLoad ; Restore ResLoad @ReallyDone ;__________________________________________________________________________ ;__________________________________________________________________________ ;____________________________________________________________________________ <2> RMP ; This patch is used to fix a problem in the MPW Shell on Eclipse. See above. ROMVIA2DT equ $00009D70 ; ROM offset of old IOP interrupt handler bra.s EclipseMPWPatchEnd ; skip over patch body into install code EclipseMPWPatchStart move.b #1<<ifCA2,VIFR(A1) ; reset the VIA2 interrupt flag move.l #(%00001000<<24)+\ ; allow xmt msg 3 (ADB) (%00001000<<16)+\ ; allow rcv msg 3 (ADB) (0<<8)+\ ; this byte must be zero 1,d0 ; ADB is on SWIM IOP (IOP 1) jmpROM IOPInterrupt ; handle the interrupt and return EclipseMPWPatchEnd @EclipseMPWPatchSize equ EclipseMPWPatchEnd-EclipseMPWPatchStart cmpRA ROMVIA2DT,(VIA2DT+4*ifCA2) ; see if we have an IOP interrupt handler bne.s @done ; if not, don't need to install it moveq.l #@EclipseMPWPatchSize,d0; space needed _NewPtr ,Sys,Clear ; allocate it bne.s @done ; if can't allocate, don't install movea.l a0,a1 ; dest address lea EclipseMPWPatchStart,a0 ; source address moveq.l #@EclipseMPWPatchSize,d0 ; byte count _BlockMove ; copy the code move.w sr,-(sp) ; save old int level ori.w #HiIntMask,sr ; disable all ints move.l a1,(Via2DT+4*ifCA2) ; install the new VIA2 interrupt handler move.w (sp)+,sr ; restore the interrupt level @done ; End of Eclipse Cmd-Shift-Esc patch ;____________________________________________________________________________ <2> RMP ;____________________________________________________________________________ GGD <2.0> ; SANE Package Optimization ; ; Optimize the SANE packages which are ROM Resources, by having the dispatch ; table entry point right to the first instruction of the package, instead ; of going through the Package Manager which finds the package by de-referencing ; a handle. ; IF 1 THEN ; removed <4.6> <08/21/89 pke> re-enabled 11/29/89 <8.4> OptimizePACKs lea @PackOptTable,a3; point to the optimization table @optLoop move.w (a3)+,d0 ; get the package number bmi.s @done ; exit at end of list move.w d0,-(sp) ; push the package number _InitPack ; get the ROM resource, update AppPacks table move.w (a3)+,d1 ; get the trap word move.w (a3)+,a0 ; get the low mem address move.l (a0),d0 ; get the handle beq.s @optLoop ; if null, skip it movea.l d0,a0 ; setup to de-reference it move.l (a0),d0 ; get a pointer to the routine beq.s @optLoop ; if null, skip it movea.l d0,a0 ; setup pointer to code cmpi.b #$60,(a0) ; see if first instruction is a BRA.S bne.s @branchOptDone ; if not, don't optimize it move.w (a0)+,d0 ; get the BRA.S instruction ext.w d0 ; extend the branch displacement adda.w d0,a0 ; update the entry point address @branchOptDone move.w d1,d0 ; setup the trap word _SetTrapAddress ; go directly to the first instr of the package bra.s @optLoop ; optimize the next package @PackOptTable dc.w 4 ; package number 4 _FP68K ; trap name for PACK 4 dc.w AppPacks+4*4 ; low mem address of handle to PACK 4 dc.w 5 ; package number 5 _Elems68K ; trap name for PACK 5 dc.w AppPacks+5*4 ; low mem address of handle to PACK 5 dc.w -1 ; package number -1, end of list @done ENDIF ;__________________________________________________________________________ <101> <BEGIN> ; ; Install PACK 4 patch if: ; ; - BoxFlag = 13 (//si - Erickson), or 14 (Mac LC - Elsie) ; and - No FPU present ; and - Current non-fpu PACK 4 lives at ROMBase + $73940 ; ; NOTE: This code MUST execute after OptimizePACKs! ; FP68KTrap equ $A9EB ;Trap number. PACK4Handle equ AppPacks+16 ;Lomem contains PACK4 handle. PACKID equ 4 ;Resource ID. PACK4Patch ; ; Are we running on IIsi or LC? ; ; Commented out boxFlag check. Its not really needed. <107> SAM ; cmpi.b #boxMacLC,BoxFlag ;Is this an LC? ; beq.s @doPatch ;Yep, so install patch. ; cmpi.b #boxErickson,BoxFlag ;No, is it a IIsi? ; bne @endPACK4Patch ;Nope, so skip install. ; ; Yes, is there an FPU? ; @doPatch move.l #gestaltFPUType,d0 ;Do we have an FPU? _Gestalt cmp.w #gestaltNoFPU,a0 bne @endPACK4Patch ;Yes, so don't install. ; ; Nope, does the current PACK4 live at ROMBase + $73940? ; move.w #FP68KTrap,d0 ;Get trap number. _GetTrapAddress ;Get current vector in a0. cmpra PACK4inROM,a0 ;Matching address? bne.s @endPACK4Patch ;Nope, so don't install. ; ; Yes, so install the patch... ; move.w #FP68KTrap,d0 ;Get trap number. lea FP020Patch,a0 ;Get address of our patch code. _SetTrapAddress ;Patch it. @endPACK4Patch ;__________________________________________________________________________ <101> <END> ;____________________________________________________________________________ ; Gestalt parity calculation ; ; This code figures out if the board has parity capability, and if so, ; if there are parity SIMMs installed. It stores the result in the ; gestalt parity function in the resident portion of the patch above. ; IMPORT parityValue WITH DecoderInfo,ExpandMemRec,GestaltGlobals,ProductInfo MACHINE MC68030 movem.l a3/a4/d3/d4,-(sp) ; save registers! <102> movea.l ExpandMem,a0 ; get ptr to expandmem rec movea.l emGestalt(a0),a0 ; get gestalt global ptr clr.l d0 ; assume no parity, clear result <59> cmpi.w #gestaltMacIIci,machType(a0) ; Mac IIci is first parity machine blt @parityExit ; if before ci, definitely no parity move.l UnivROMFlags,d1 ; get external feature bits btst #PGCInstalled,d1 ; check if a parity chip exists bne.s @checkPGC ; yep, go see if parity is enabled move.l AddrMapFlags,d1 ; get universal info attributes btst #RPUExists,d1 ; do we have an RPU chip? bne.s @checkRPU ; then go check RPU bra @parityExit ; if not, return zero as result ; checkPGC - check the parity VIA bit to see if it is enabled. @checkPGC move.l VIA,a0 ; load up Via base address btst #6,VBufB(a0) ; check if parity enabled beq @parityOn ; if clear, parity is enabled bra @parityOff ; otherwise, it's off ; checkRPU - First set RPU to generate bad parity, then access all four SIMMs in ; both banks by accessing location 0 and the last longword in memory. ; Parity is only fully active if all four SIMMs generate a parity interrupt. ; trashes a0/a1/a3/a4/d1-d4 @checkRPU movec cacr,d3 ; get contents of cache move.l d3,-(sp) ; save them move.l #$1111,d3 ; enable caches with bursting <32> movec d3,cacr ; write into cache register <32> move.l RomBase,a0 ; point to base of rom <32> move.l #$2000,d3 ; cache size/long = 32k/4 = 8k <32> @fillLoop tst.l (a0)+ ; load a long <32> dbra d3,@fillLoop ; keep doing it for 8k <32> move.l #$0808,d3 ; disable and freeze caches (bursting off) movec d3,cacr ; write into cache register move.l UnivInfoPtr,a0 ; get pointer to universal information add.l (a0),a0 ; point to decoderInfo record move.l RPUAddr(a0),a1 ; get RPU base address move.l sp,d3 ; save our stack pointer move.l AutoInt7,d4 ; save NMI vector lea @parityInt,a4 ; address to return to from parity intrpt lea @nmiHandler,a0 ; get address of our NMI handler move #$2700,sr ; turn all interrupts off move.l a0,AutoInt7 ; point vector to our handler moveq #0,a3 ; check first bank (0 is in first bank) @checkBank moveq #4-1,d2 ; check all four SIMMs move.l (a3),d1 ; get contents of longword @nextByte clr.l (a3) ; write a long with correct parity <44> st.b rpuReset(a1) ; reset serial ptr st.b (a1) ; write wrong parity mode clr.b (a3,d2.w) ; write one SIMM with bad parity st.b rpuReset(a1) ; reset serial ptr <105> clr.b (a1) ; write good parity mode <105> tst.l (a3) ; parity error if it's a parity SIMM nop ; wait for an interrupt nop move.l d1,(a3) ; restore contents of longword move #$2000,sr ; turn interrupts back on move.l d4,AutoInt7 ; restore NMI vector move.l (sp)+,d3 ; get original state of cache <17> movec d3,cacr ; restore original state of cache <17> bra.s @parityOff ; if we get here, it's a non-parity SIMM @parityInt st.b rpuReset(a1) ; sync up RPU serial line clr.b (a1) ; write good parity mode dbra d2,@nextByte ; go check next SIMM @allParityBank move.l d1,(a3) ; restore contents of longword move.l a3,d1 ; did we just check location 0? bne.s @restoreNMI ; if not, we're done and parity is on movea.l ExpandMem,a0 ; get ptr to expandmem rec movea.l emGestalt(a0),a0 ; get gestalt global ptr movea.l memSize(a0),a3 ; total amount of memory subq #4,a3 ; point to last longword in memory bra.s @checkBank ; go check bank B @restoreNMI move #$2000,sr ; turn interrupts back on move.l d4,AutoInt7 ; restore NMI vector move.l (sp)+,d3 ; get original state of cache <17> movec d3,cacr ; restore original state of cache <17> bra.s @parityOn ; go return that parity is on ; Interrupt handler for checkRPU routine. It is assumed that the NMI button is ; not pressed during this routine. ; d3 = stack pointer of routine ; a1 = RPU base address ; a4 = return address @nmiHandler st.b rpuReset(a1) ; reset serial ptr clr.b (a1) ; write good parity mode st.b (a1) ; clear the parity error move.l d3,sp ; restore the stack pointer jmp (a4) ; return to the checkRPU routine @parityOn bset #gestaltParityEnabled,d0 ; by here it is @parityOff bset #gestaltHasParityCapability,d0 ; set parity capability bit @parityExit lea parityValue,a0 ; get address of variable in gestalt function move.l d0,(a0) ; save response in gestalt function movem.l (sp)+,a3/a4/d3/d4 ; restore registers! <102> ENDWITH ;____________________________________________________________________________ ; Gestalt function patch installation ; ; These functions either did not make it in the Portable ROM, or changed ; between ROM final and System Disk final. ; gestaltRBVAddr EQU 'rbv ' gestaltSCCReadAddr EQU 'sccr' gestaltSCCWriteAddr EQU 'sccw' gestaltVIA1Addr EQU 'via1' gestaltVIA2Addr EQU 'via2' gestaltSlotAttr EQU 'slot' ; removed in favor of gestaltNuBusConnectors <95> gestaltFirstSlot EQU 'slt1' ; removed in favor of gestaltNuBusConnectors <95> gestaltSlotCount EQU 'nubs' ; removed in favor of gestaltNuBusConnectors <95> lea gestaltQDVers,a0 ; address of gestaltQDVersion patch move.l #gestaltQuickdrawVersion,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltAddrMode,a0 ; addr of gestaltAddressingModeAttr patch move.l #gestaltAddressingModeAttr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltParity,a0 ; addr of gestaltParityAttr patch move.l #gestaltParityAttr,d0 ; selector to REPLACE _ReplaceGestalt lea getGestaltVers,a0 ; addr of getGestaltVersion patch move.l #gestaltVersion,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltMisc,a0 ; addr of gestaltMisc patch move.l #gestaltMiscAttr,d0 ; selector to ADD _NewGestalt lea gestaltNBCon,a0 ; addr of gestaltNuBusConnectors patch <95> move.l #gestaltNuBusConnectors,d0 ; selector to ADD _NewGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch <95> move.l #gestaltSlotAttr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch <95> move.l #gestaltFirstSlot,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch <95> move.l #gestaltSlotCount,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltNMgr,a0 ; addr of gestaltNotificationMgrAttr patch move.l #gestaltNotificationMgrAttr,d0 ; selector to ADD _NewGestalt lea gestaltSerial,a0 ; addr of gestaltSerialAttr patch <95> move.l #gestaltSerialAttr,d0 ; selector to ADD _NewGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch move.l #gestaltRBVAddr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch move.l #gestaltSCCReadAddr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch move.l #gestaltSCCWriteAddr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch move.l #gestaltVIA1Addr,d0 ; selector to REPLACE _ReplaceGestalt lea gestaltUndef,a0 ; addr of gestaltUndef patch move.l #gestaltVIA2Addr,d0 ; selector to REPLACE _ReplaceGestalt IF doScriptMgrGestalt AND (NOT installScriptMgrPtch27) THEN ; <6.1><50> ;____________________________________________________________________________ pke <2.5> ; Gestalt install for Script Mgr <2.5> <08/05/89 pke> ; import gestaltScriptMgr,gestaltSMgrTable lea gestaltSMgrTable,a3 ; table of Gestalt selectors ; and GetEnvirons verbs @loopInstallGestalt move.l (a3)+,d0 ; get next Gestalt selector beq.s @doneInstallGestalt ; 0 means we're done addq.l #2,a3 ; skip GetEnvirons verb lea gestaltScriptMgr,a0 ; push gestaltFunction ProcPtr _NewGestalt ; ignore OSErr in d0 (what can ; we do if it is not noErr?) bra.s @loopInstallGestalt @doneInstallGestalt ENDIF ; <6.1> ;____________________________________________________________________________ dba <3.7> ; Install default colors 'clut' from System file ; Since the first InitGraf was before the ROvr code, the ROM 'clut' is ; always used. We want to use a better 'clut' from the System file. ; NOTE: This code is straight out of GrafAsm.a. FixQDColors ; <4.7> cmp.l #-1,QDColors ; is QDColors initialized? <4.7> beq.s @done ; no, no need to fix it up <4.7> SUBQ #4,SP ; make room for function result MOVE #DefQDColors,-(SP) ; push resource ID _GetCTable ; get default colors MOVE.L QDColors,a0 ; get pointer to QDColors MOVE.L (SP),A1 ; get handle to default colors MOVE.L (A1),A1 ; point to default colors ADD #CTTable,A1 ; skip over header MOVEQ #15,D0 ; need to move 16 longs @nextLong MOVE.L (A1)+,(A0)+ ; move a long DBRA D0,@nextLong ; => repeat for all longs _DisposCTable ; dispose of color table @done ; <4.7> ;____________________________________________________________________________ dba <3.0> ; Install Menu Manager patches ; InstToolTp NewInitProcMenu,$8 InstToolTp NewMenuSelect,$13D IF NOT SPLINE_FONT THEN ;____________________________________________________________________________ cel <5> ; Install NewRSect Patch for DrawText zero width chars ; setTrap NewRSect,$AB21 ENDIF ;____________________________________________________________________________ DAF <4.1> ; This really didn't get in until <5.1> ; ; Install SetEntries patch InstToolTp mySaveEntries,$249 InstToolTp mySetEntries,$23F ; < 124 > ;____________________________________________________________________________ pke <4.6> ; Install patches to UprString,CmpString,RelString (OK for A/UX) InstOSTp NewUprString,$54 InstOSTp NewCmpString,$3C InstOSTp NewRelString,$50 ;____________________________________________________________________________ DDG <58> ; Install vector for the ÒFPU not installedÓ dialog box if we donÕt have an ; FPU or an MMU move.l #gestaltFPUType,d0 ;do we have an FPU ? _Gestalt cmp.w #gestaltNoFPU,a0 ;if this is NOT noFPU, then we skip bne.s @skipInstall ;..the installation of this patch lea NewFLineRoutine,a0 ;get the address of our F-Line routine into A0 lea OldFLineRoutine,a1 ;get the address of the JMP to the old F-Line routine lea Line1111,A2 move.l (A2),2(A1) ; Save the old Vector <107> SAM move.l A0,(A2) ; Make 1111 traps come to us first <107> SAM @skipInstall ;---------------------------------------------------------------------------------------------------- ;__________*** This set of routines gets run only if we have a REgret chip ***_________________ WITH ProductInfo ; <101> SAM MOVE.L UnivROMFlags,D0 ; Do we have a REgret? AND.B #ClockMask,D0 ; (bits 4-6) <107> SAM (Use ClockMask!) CMP.B #ClockEgret,D0 ; Is bit 5 on? BNE.S @EndOfEgretOnly ; -> Nope, skip this install code ;---------------------------------------------------------------------------------------------------- ; This patch bypasses the physical read of egret when reading the time. The time is automatically ; updated in the background, therefore the low mem global always reflects the acurate time. ; (REgret only) ;---------------------------------------------------------------------------------------------------- InstOSTp RdDateTimePtch,$39 ;---------------------------------------------------------------------------------------------------- ; Install the TickHandler Patch into the TickComp vector kept in <89><101> ; the Egret Manager Global variables. ; (REgret only) ;---------------------------------------------------------------------------------------------------- WITH EgretGlobals ; Dont need to turn off IRQs here. <107> SAM ; move.w sr,-(sp) ; no interruptions ; ori.w #LVL7IRQ,sr lea TickHandler,a0 ; get TickHandler address move.l EgretBase,a1 ; point to EgretManager Globals move.l a0,tickComp(a1) ; setup vector to new tick handler ; move.w (sp)+,sr ; restore interrupt state ENDWITH ;---------------------------------------------------------------------------------------------------- ; This patch will save the processor status register, call the <101> SAM ; ShiftRegIrq routine in Egret Manager and then restore the ; status register and exits. ; (REgret only) ;---------------------------------------------------------------------------------------------------- lea Vector_hold,a1 ; point to the interrupt vector placeholder move.l LVL1DT+8,(a1) ; shift register interrupt handler lea EgretIRQPtch,a1 ; Point to the patch move.l a1,LVL1DT+8 ;---------------------------------------------------------------------------------------------------- ; This patch fixes possible loss of data to the SCC by calling <101> SAM ; the Poll Proc when SCC data is available. ; (REgret only) ;---------------------------------------------------------------------------------------------------- InstOSTp EgretDispatchPtch, $92 @EndOfEgretOnly ENDWITH ;__________________________ End of REgret Only Section ______________________________________ ;---------------------------------------------------------------------------------------------------- move.l MinusOne, $B50 ; clean up after netbooting socket listener.<78> ;---------------------------------------------------------------------------------------------------- ;____________________________________________________________________________ pke <90> ; Move Script Mgr patches here so they are installed for A/UX. <90> ; Currently assumes that installScriptMgrPtch39 and installScriptMgrPtch27 are ; true, but this may be changed, so the inner conditionals are being left in. IF doScriptMgrForAUX THEN ; <90> IF installScriptMgrPtch39 THEN ; <50> PtchInst 39 ; <42> ENDIF IF installScriptMgrPtch27 THEN ; <50> PtchInst 27 ; <5.7> ENDIF ENDIF ; <90> EndBothSection ;========================================================================================= ;========================================================================================= ;= = ;= ALL PATCHES BELOW THIS POINT ARE INSTALLED ON THE MACINTOSH OS ONLY. ALL PATCHES = ;= ABOVE THIS POINT ARE INSTALLED BOTH THE MAC OS AND A/UX. = ;= = ;========================================================================================= ;========================================================================================= ; Runtime check to determine whether we are running A/UX. If so, skip this section ; of patch install code. move.w HwCfgFlags,d0 ; check Ôem the compulsive way btst.l #hwCbAUX,d0 ; is it A/UX time? bne skipMacOnlyPatches ; if so skip the mac-only patches ;========================================================================================= ;========================================================================================= ;= = ;= PATCH INSTALLATION CODE FOR THE MAC OS ONLY STARTS HERE = ;= = ;= Please be neat and follow the format = ;= = ;========================================================================================= ;========================================================================================= ;____________________________________________________________________________ ggd <2.2/2.7> ; Slot Interrupt patch ; ; Patches _SIntInstall, and reverses Slot interrupt queue order ; PatchSIntInstall Import SIntInstall With SlotIntQElement,slotIntGlobals ; We reverse the order of equal priority elements in the queue for Aurora, such that ; they again blissfully agree with what the Mac II does for Equal Priorities ; turn off interrupts since we are mucking with a queue move.w SR, -(sp) ori.w #HiIntMask,SR ; disable all ints InstOSTp SIntInstall,$A075 ; install the new trap ; cycle thru the slots for reversing queue order moveq #TotalSlots-1,d1 ; set up for dbra @nextQueue lea ([SlotQDT],d1.w*4,slotIntQHeads),a0 ; get address of queue header movea.l (a0),a1 ; get ptr to first queue element clr.l (a0) ; destroy old queue bra.s @start ; re-install the elements @loop movea.l a1,a0 ; set up for SIntinstall movea.l SQLink(a1),a1 ; save out ptr to next queue element move.l d1,d0 ; set up for call _SIntInstall ; re-install the queue element @start move.l a1,d0 ; test for end of queue bne.s @loop ; loop until end of queue @queueDone dbra d1,@nextQueue ; re-enable interrupts now that we are finished mucking move.w (sp)+, SR ;____________________________________________________________________________ djw <2.4> ; Slot manager PRAM patch ; ; Fixes a problem in InitPRAMRecs where when initializing slot PRAM from a ; PRAM initialization record from the declaration ROM, the board id would ; not be set correctly. This code writes the correct board id to the slot ; PRAM for all good cards. GetBoardROM Equ $0000636E ; addr of GetBoard routine in ROM <2.7> With spBlock,sInfoRecord FixSlotPRAM suba.w #spBlockSize,sp ; alloc spBlock movea.l sp,a0 ; a0 = ptr to spBlock subq.l #8,sp ; alloc space for pram buffer movea.l sp,a1 ; a1 = ptr to pram read/write buffer clr.l (a1) ; clr pram buffer (8 bytes) clr.l 4(a1) clr.b spSlot(a0) ; start from slot zero ; Loop through the slots and check if the board id in PRAM matches the ; board id on the declaration ROM. If not, then write the correct one ; to PRAM. move.w sr,-(sp) ; save status reg ori.w #HiIntMask,sr ; disable interrupts @Loop jsrROM GetBoardROM ; jsr into ROM to GetBoard routine to get board sRsrc bne.s @EndLoop ; no board sRsrc - skip this slot movea.l spResult(a0),a2 ; a2 = ptr to sInfo record tst.w siInitStatusA(a2) ; test card status bmi.s @EndLoop ; bad slot - skip it move.b #Boardid,spId(a0) ; read board id field from board sRsrc _sReadWord bne.s @EndLoop ; board id is gone - must be a bad slot move.w spResult+2(a0),d1 ; reg d1 = board id from decl rom move.l a1,spResult(a0) ; pass ptr to buffer _sReadPRAMRec bne.s @EndLoop ; cannot read PRAM - bad slot ; Compare PRAM board id with declaration ROM board id. If different, update ; the PRAM board id. cmp.w (a1),d1 ; same board id's? beq.s @EndLoop ; same - go to next slot move.w d1,(a1) ; update board id in buffer move.l a1,spsPointer(a0) ; set pointer to write buffer _InitSlotPRAM @EndLoop addq.b #1,spSlot(a0) ; inc to next slot cmp.b #sLastSlot,spSlot(a0) ble.s @Loop ; continue till all slots are done move.w (sp)+,sr ; restore interrupts adda.w #spBlockSize+8,sp Endwith ;____________________________________________________________________________ djw <3.6> ; Slot manager pInitEntry patch ; ; Install the pInitEntry patch which corrects the spId passed to _sFindDevBase ; from _InsertSRTRec. ; Import pInitEntry lea pInitEntry,a0 ; entry point for resident patch move.l a0,([SDMJmpTblPtr],InitEntry*4) ; insert addr in slot mgr tbl ;____________________________________________________________________________ DAF <2.6> ; TFB Video Driver Override ; INCLUDE 'VideoPatch.a' ;________________________________________________________________ <4.3><08/21/89 pke> ; Patch to Pack6 to fix Script Manager routines (which are not ; supported by A/UX). ; IF doScriptMgrPack6Fix AND (NOT installScriptMgrPtch39) THEN ; <42><50> InstToolTp ptchPack6,$1ED ENDIF ;____________________________________________________________________________ ; Bump Script Manager version number for IntlForce bug fix (SysVers < $700) pke <5.3> ; Set Script Manager version (SysVers >= $700) pke <5.7> ; redo using new symbol smgrVersPTCHRom pke <5.9> IF doScriptMgrSetROMVers AND (NOT installScriptMgrPtch39) THEN ; <42><50> With SMgrRecord GetSMgrCore a0 move.w #smgrVersPTCHRom,smgrVersion(a0) EndWith ENDIF ;____________________________________________________________________________ ; Initialize additional Script Manager vectors pke <5.4> ; Add CallInterface pke <5.9> ; Add NewSwapIcon (7.0 only, moved to ptch 39) pke <6.2> IF doScriptMgrNewVectors AND (NOT installScriptMgrPtch27) THEN ; <42><50> ROMSMgrCalcRect EQU $1E1C2 ROMSMgrInitFonts EQU $1E21C ROMCallInterface EQU $1E3B0 ; <5.9> With SMgrRecord GetSMgrCore a0 leaRom ROMSMgrCalcRect,a1 move.l a1,sVectSMgrCalcRect(a0) leaRom ROMSMgrInitFonts,a1 move.l a1,sVectSMgrInitFonts(a0) leaRom ROMCallInterface,a1 ; <5.9> move.l a1,sVectCallInterface(a0) ; <5.9> EndWith ENDIF ;____________________________________________________________________________ pke <5.7> ; Install Script Manager 7.0 extensions, ROM patches ; Moved these installs up if doScriptMgrForAUX is true. <90> IF NOT doScriptMgrForAUX THEN ; <90> IF installScriptMgrPtch39 THEN ; <50> PtchInst 39 ; <42> ENDIF IF installScriptMgrPtch27 THEN ; <50> PtchInst 27 ; <5.7> ENDIF ENDIF ; <90> ;____________________________________________________________________________ pke <7.8> ; Install patches for Script Manager routines InitDateCache, String2Date IF doScriptMgrStr2DatFix AND (NOT installScriptMgrPtch39) THEN ; <42><50> smInitDateCacheOff equ -16 ; DispTable offset for InitDateCache smString2DateOff equ -20 ; DispTable offset for String2Date With SMgrRecord GetSMgrCore a1 move.l smgrDispTable(a1),a1 lea NewInitDateCache,a0 move.l a0,smInitDateCacheOff(a1) lea NewString2Date,a0 move.l a0,smString2DateOff(a1) EndWith ENDIF ;____________________________________________________________________________ pke <8.0> ; Install Script Manager tail patch to _GetIndADB, fixes the way SwapKybd ; clears dead key state in ADB keyboard driver data structure IF doScriptMgrRstKCHRFix AND (NOT installScriptMgrPtch39) THEN ; <42><50> PatchOSJump oldGetIndADB,$78 ; set addr for BackToTrap InstOSTp ptchGetIndADB,$78 ; Éand then install patch ENDIF ;____________________________________________________________________________ pke <8.5> ; Install Script Manager patch to LwrString so it handles 2-byte chars IF doScriptMgrLwrString2 AND (NOT installScriptMgrPtch27) THEN ; <8.5><50> PatchOSJump oldLwrString,$56 ; set addr for BackToTrap InstOSTp ptchLwrString,$56 ; Éand then install patch ENDIF ;____________________________________________________________________________ EVA <7.1><123> ; Install Async Serial Driver Vector Address Import SerialPatch InstOSTp SerialPatch,$A0BE SERDVersion equ 5 ; install new version number move.l #5*4,d0 ; first serial DCE is 5th entry in Utbl moveq #4-1,d1 ; index through 4 dce's (adjusted for dbra) @next movea.l UTableBase,a0 ; get ptr to the unit table movea.l 0(a0,d0.w),a0 ; get DCE handle movea.l (a0),a0 ; get DCE ptr move.w #SERDVersion,dCtlQueue(a0) ; post new version addq #4,d0 ; get next DCE dbra d1,@next ;____________________________________________________________________________ GGD <8.4> ; Install ADB Polling patch for ViaADB implementations only ; ; If we are running on a machine with ViaADB (ie. No IOPs), then we allocate a ; block for this patch, copy the code into it, and install it (saving heap space ; on machines that don't need it). ; ; This patch changes Auto/SRQ polling to only poll device addresses which have entries ; in the device table. OldFDBShiftInt equ $0000a700 ; ROM offset of old FDBShiftInt SendByte2Rtn equ $0000A6D0 ; ROM offset of return from second @sendNextByte call GetByte2Rtn equ $0000A63C ; ROM offset of return from second @getNextByte call ROM@sendNextByte equ $0000A6E6 ; ROM offset of @sendNextByte label ROM@fetchLoop equ $0000A63A ; ROM offset of @fetchLoop label ROM@handleSRQ equ $0000A724 ; ROM offset of SRQ handling code ROM@skipUpdate equ $0000A738 ; ROM offset of ReqDoneVIA exit without updating ADDR InstallADBPollPatch cmpRA OldFDBShiftInt,Lvl1dt+8 ; see if correct shift int handler to patch bne.s @exit ; if not, don't patch it (might not be VIA based ADB) lea ADBPollPatchStart,a1 ; get address of code to copy move.l (a1),d0 ; length of code _NewPtr ,SYS ; allocate space for the code bne.s @exit ; if couldn't allocate, don't install it move.l (a1)+,d0 ; length of code exg.l a0,a1 ; setup src/dst for block move _BlockMove ; copy the code move.l a1,Lvl1DT+8 ; install new shift int handler @exit bra.w ADBPollPatchEnd ; jump around patch body ADBPollPatchStart dc.l ADBPollPatchEnd-ADBPollPatchStart-4 with ADBVars ;_______________________________________________________________________ ; Routine: FDBShiftInt ; Inputs: A1 - base address of VIA1 (setup by IntHnd) ; Outputs: A1 - base address of VIA1 ; A3 - pointer to ADBBase ; ccr.z - result of BTST #vFDBInt,vBufB(a1) ; ; Function: handles shift interrupt, resumes asynchronous processing ;_______________________________________________________________________ FDBShiftInt movea.l ADBBase,a3 ; point to ADB globals in low memory movea.l ShiftIntResume(a3),a0 ; get address to resume at cmpRA SendByte2Rtn,a0 ; see if resuming from sending second byte beq.s SendPatch ; if so, use the patched code cmpRA GetByte2Rtn,a0 ; see if resuming from fetching second byte beq.s FetchPatch ; if so, use the patched code btst.b #vFDBInt,vBufB(a1) ; test the FDBInt~ status jmp (a0) ; resume async processing SendPatch btst.b #vFDBInt,vBufB(a1) ; test the FDBInt~ status bne.s @sendLoop ; if no SRQ, send the data bset.b #fDBSRQ,FDBAuFlag(a3) ; remember that a service request was returned @sendLoop tst.b fDBCnt(a3) ; see if end of buffer reached beq.s ReqDoneVIA ; leave when count is zero, no reply data jsrROM ROM@sendNextByte ; send another byte bra.s @sendLoop ; loop until count exhausted FetchPatch btst.b #vFDBInt,vBufB(a1) ; test the FDBInt~ status beq.s @fetchDone ; exit if end of data reached cmpi.b #8,fDBCnt(a3) ; see if end of buffer reached bhs.s @fetchDone ; keep fetching until end of data jmpROM ROM@fetchLoop @fetchDone btst.b #fDBNoReply,FDBAuFlag(a3) ; see if buffer data is valid seq.b d0 ; $FF if data is valid, $00 if no reply and.b d0,fDBCnt(a3) ; set count to zero if timeout * bra.s ReqDoneVIA ReqDoneVIA move.b FDBAuFlag(a3),d0 ; get the flags move.b fDBCmd(a3),d1 ; get the command moveq.l #(1<<fDBSRQ),d3 ; mask to test for SRQ pending and.b d0,d3 ; isolate the bit neg.l d3 ; set bit 31 if SRQ pending move.b d1,d3 ; insert the command byte lsr.b #4,d1 ; isolate the device address tst.l d3 ; was there an SRQ? bpl.s @noSRQ ; if not, don't advance poll address jmpROM ROM@handleSRQ ; otherwise, go handle the SRQ @noSRQ move.w DevMap(a3),d2 ; list of possible address to search btst.l d1,d2 ; see if it is in the table beq.s @skipUpdate ; if not, don't switch to that address move.b d1,PollAddr(a3) ; remember where to auto/SRQ poll next @skipUpdate jmpROM ROM@skipUpdate ; back to ROM, just past the update ADBPollPatchEnd ; End of ADB Polling patch for ViaADB implementations ;____________________________________________________________________________ GGD <8.4> ;____________________________________________________________________________ GGD <8.4> ; Install a Time Manager Task to simulate the 60.15Hz pseudo VBL for OSS based systems. ; ; If we are running on a machine with an OSS, then we allocate a block for this patch, ; copy the code into it, and install it (saving heap space on machines that don't need it). ; ; This patch corrects the pseudo VBL frequency on OSS based systems, the OSS generates ; a 60.00Hz interrupt, but the Macintosh Pseudo VBL rate should be 60.14742 Hz. ; To correct this, we simply ignore and disable the OSS interrupt, and use the ; Time Manager to generate interrupts at the correct rate. FixMPWShell equ 1 ; Temporarily fix CMD-Period <25> ; This same patch is also modified to fix a problem in the MPW Shell on Zone-5. ; The real bug is in the MPW Shell, but we will temporarily fix it here until they ; have a chance to rev, so that we don't hear bitching from developers. ; The MPW Shell will sometimes cut back the stack from within a patch they make to ; PostEvent, when CMD-Period is pressed. Therefore, it may never return back to the ; keyboard driver, which never returns to the ADBManager, which never returns to the ; IOP Manager, which never returns to the Deferred Task Manager, which never returns ; to the Interrupt Handler, which never returns to whatever code was running at the ; time of the interrupt (which may have been another interrupt). This causes data ; structures to be left in inconsistent states, such as the deferred task manager thinks ; that a deferred task is still running, and won't run any new tasks. ; The work around is to not run ADB as a deferred task, and instead run it at level 1 <25> ROMLevel1OSSInt equ $00009B50 ; ROM offset of old Level1OSSInt ServiceIntMMU equ $00009B68 ; ROM offset of ServiceIntMMU if FixMPWShell then ; <25> IOPInterrupt equ $00005036 ; ROM offset of IOP Manager interrupt handler <25> ContinueOssLevel1Decode equ $00009D8E ; ROM offset of continuation of OssLevel1Decode <25> else ; <25> ContinueOssLevel1Decode equ $00009D88 ; ROM offset of continuation of OssLevel1Decode endif ; <25> OssVBL60HzInt equ $00009DF0 ; ROM offset of OssVBL60HzInt IntRegs reg a0-a3/d0-d3 ; registers saved by all interrupt handlers bra.s Oss60HzPatchEnd ; skip over patch body into install code Oss60HzPatchStart Oss60HzTimeHandler movea.l a1,a0 ; point to the TMTask move.l #-16626,d0 ; 16.626 ms, 60.14742 Hz (pseudo VBL rate) movea.l jPrimeTime,a1 ; point to the routine jsr (a1) ; initiate the delay movea.l OSS,a1 ; get OSS chip base address jmpROM OssVBL60HzInt ; run the VBL interrupt OssLevel1Decode movea.l OSS,a1 ; get OSS chip base address move.w OSSIntStat(a1),d0 ; see who is interrupting bclr.l #OSSIntVBL60Hz,d0 ; ignore OSS 60Hz pseudo VBL interrupt if FixMPWShell then ; <25> btst #OSSIntIOPSWIM,d0 ; check SWIM IOP first <25> beq.s OSSPSwmMasked ; if no SWIM IOP Interrupt, skip it <25> moveq.l #7,d1 ; mask for int level <25> and.b OSSMskPSwm(a1),d1 ; see if SWIM IOP was disabled <25> beq.s OSSPSwmMasked ; if disabled, ignore it <25> move.l #(%00001000<<24)+\ ; allow xmt msg 3 (ADB) <25> (%00001000<<16)+\ ; allow rcv msg 3 (ADB) <25> (0<<8)+\ ; this byte must be zero <25> 1,d0 ; ADB is on SWIM IOP (IOP 1) <25> jmpROM IOPInterrupt ; handle the interrupt and return <25> OSSPSwmMasked ; <25> endif ; <25> jmpROM ContinueOssLevel1Decode ; resume with the ROM code Level1OSSInt movem.l IntRegs,-(SP) ; preserve registers lea OssLevel1Decode,a3 ; use the OSS level 1 dispatcher jmpROM ServiceIntMMU ; call primary interrupt dispatcher Oss60HzPatchEnd @PatchSize equ Oss60HzPatchEnd-Oss60HzPatchStart cmpRA ROMLevel1OSSInt,AutoInt1; see if we have an OSS interrupt handler bne.s @done ; if not, don't need to install it moveq.l #tmXQSize+@PatchSize,d0 ; space needed _NewPtr ,Sys,Clear ; allocate it bne.s @done ; if can't allocate, don't install movea.l a0,a2 ; save a copy lea Oss60HzPatchStart,a0 ; source address lea tmXQSize(a2),a1 ; dest address moveq.l #@PatchSize,d0 ; byte count _BlockMove ; copy the code lea Oss60HzTimeHandler-Oss60HzPatchStart(a1),a0 ; point to the handler move.l a0,tmAddr(a2) ; initialize the task address movea.l a2,a0 ; get the time manager task address move.l #'eada',(a0) ; put in magic signature (see VM for details) <139> _InsXTime ; install the task (fixed frequency) move.l #-16626,d0 ; 60.14742 Hz (pseudo VBL rate) movea.l jPrimeTime,a2 ; point to the routine lea Level1OSSInt-Oss60HzPatchStart(a1),a1 move.l Ticks,d1 ; get the current time @sync cmp.l Ticks,d1 ; sync with the VBL manager beq.s @sync ; wait for it to change move.w sr,-(sp) ; save old int level ori.w #HiIntMask,sr ; disable all ints move.l a1,AutoInt1 ; install the new level 1 handler movea.l OSS,a1 ; get the OSS base clr.b OSSMsk60Hz(a1) ; disable the OSS 60Hz interrupt source move.w (sp)+,sr ; restore the interrupt level jsr (a2) ; start the pseudo VBL timer @done ; End of OSS VBL frequency patch ;____________________________________________________________________________ GGD <8.4> ;____________________________________________________________________________ SWC <8.8> ; RecoverHandle() patch for 32-bit Memory Manager IMPORT v24RecoverHandle,v32RecoverHandle JRecoverHandle EQU $6C ;offset into 32-bit Memory Manager's vector table LEA v24RecoverHandle,A0 ;patch in the 24-bit version of RecoverHandle MOVE.L A0,JMemMgr24+JRecoverHandle LEA v32RecoverHandle,A0 ;patch in the 32-bit version of RecoverHandle MOVE.L A0,JMemMgr32+JRecoverHandle ; End of RecoverHandle() patch install ;____________________________________________________________________________ SWC <8.8> ;____________________________________________________________________________ RMP <12> ; SetHandleSize() patch for 32-bit Memory Manager IMPORT v32SetHandleSize JSetHandleSize EQU $64 ;offset into 32-bit Memory Manager's vector table LEA v32SetHandleSize,A0 ;patch in the 32-bit version of SetHandleSize MOVE.L A0,JMemMgr32+JSetHandleSize ; End of SetHandleSize() patch install ;____________________________________________________________________________ RMP <12> ;____________________________________________________________________________ CSL ; SetPtrSize() patch for 32-bit Memory Manager IMPORT v32SetPtrSize JSetPtrSize EQU $4C ;offset into 32-bit Memory Manager's vector table LEA v32SetPtrSize,A0 ;patch in the 32-bit version of SetPtrSize MOVE.L A0,JMemMgr32+JSetPtrSize ; End of SetPtrSize() patch install ;____________________________________________________________________________ CSL IF 0 THEN ;____________________________________________________________________________ jwk <19> ; ; Patch in FastReadOSS() and FastWriteOSS() ROM routines to use the SCSIDMA chip. ; WITH ROMHeader, scsiGlobalRecord ROMFastWriteOSS EQU $00009042 ROMFastReadOSS EQU $000092ea cmpi.b #boxF19,BoxFlag ; is it an F19 ? bne.s @jwkNoInstall ; only applies to F19's move.w #$11f0,d0 ; check for any final F19 ROM movea.l ROMBase,a1 ; point to the base of the ROM and.w ROMRelease(a1),d0 ; we'll take 1.1f1, 1.1f2, etc. cmpi.w #$11f0,d0 ; is it a 1.1fÉ ROM ? bne.s @jwkNoInstall ; only install on a Zone5 with 1.1fÉ ROM's ; <57> 04-22-90 csd ; commented out the installation of SCSI DMA so we stop trashing ; hard disks on the IIfx. movea.l SCSIGlobals,a1 ; point to SCSI Manager globals move.l #250*10000,G_Reserved1(a1) ; set up 250ms byte-to-byte timeout leaROM ROMFastWriteOSS,a0 ; move.l a0,jvVFWO(a1) ; use DMA code for SCSIWBlind leaROM ROMFastReadOSS,a0 ; move.l a0,jvVFRO(a1) ; use DMA code for SCSIRBlind @jwkNoInstall ENDWITH ; ; End of FastReadOSS() and FastWriteOSS() patch install ;____________________________________________________________________________ jwk <19> ENDIF ;____________________________________________________________________________ jwk <29> ; ; Patch in chaining SCSI bus error handler ; TestFor SCSI96_1Exists ; check universal tables for C96 SCSI Chip <3> bne.s @ChaningBErrDone ; has C96 - do not install patch (or die) <3> WITH scsiGlobalRecord movea.l SCSIGlobals,a1 ; point to SCSI Manager globals lea.l BusErrHandler,a0 ; point to bus error handler patch move.l a0,jvBusErr(a1) ; patch vector in the globals ENDWITH @ChaningBErrDone ;_________________________________________________________________________________________ <119> djw ; Install SCSI fix for Quantum 7.9 rom fix with scsiGlobalRecord ; Check the universal tables in this ROM version for a SCSI DMA chip (the GDU). If ; the GDU exist, then we must be on a Mac IIfx (dammed overpatch ROM's). The Quantum ; fix cannot be applied to the IIfx because on the GDU, *ACK cannot be controlled. TestFor SCSIDMAExists ; check universal tables for SCSI DMA bne.s @noSCSIpatch ; has SCSI DMA - do not install patch TestFor SCSI96_1Exists ; check universal tables for C96 SCSI Chip <136> bne.s @noSCSIpatch ; has C96 - do not install patch (or die) <136> ; Apply the patch to this Mac IIci family ROM lea QuantumWBlindIIci,a0 ; a0 = TIB interpreter patch move.l a0,jvWBlind(a1) ; replace vector lea FastWriteFix,a0 ; patch to FastWrite data xfer routine move.l a0,jvVFWO(a1) ; virtual fast writes (hhsk) move.l a0,jvVFW(a1) ; virtual fast writes (hhsk) @noSCSIpatch ;_________________________________________________________________________________________ <122> djw ; Install SCSIDispatch patch for VM conflict ; ; Patch SCSIDispatch back to ROM to avoid the situation where VM patches SCSIDispatch ; before patches, and after patches. VM will be in the chain of patches twice. This ; coupled with the SCSI mgr calling the file system gets into a circular calling sequence ; of the SCSI mgr calling VM and VM calling the SCSI mgr. @ROMSCSIDispatch Equ $00007460 ; offset to SCSIDispatch in ROM movea.l RomBase,a0 ; get base of ROM adda.l #@ROMSCSIDispatch,a0 ; add offset to SCSIDispatch in ROM setTrapA0 $a815 ; patch SCSIDispatch to ROM ;____________________________________________________________________________ ; ; <54> Install patch to IOPMsgRequest to fix IOP Serial Driver Ctrl Call bug ;____________________________________________________________________________ IMPORT IOPMsgRequest move.w sr, -(sp) ; disable interrupts while we patch ori.w #HiIntMask,sr ; ; Install the patch only if IOPMsgRequest trap already exists ; gettrapA0 $A087 ; get IOPMsgRequest trap addr move.l a0,a1 ; set it aside gettrapA0 $A89F ; get unimplemented trap addr cmpa.l a0,a1 ; check 'em beq.s @noTrap ; nope, so don't install patch PatchOSJump oldIOPMsgRequest,$87 ; Stash original IOPMsgRequest routine <12> InstOSTp IOPMsgRequest,$87 ; replace IOPMsgRequest trap <12> @noTrap ; restore state move.w (sp)+,sr ; disable interrupts while we patch ;____________________________________________________________________________ ; ; <54> End Install patch to IOPMsgRequest to fix IOP Serial Driver Ctrl Call bug ;____________________________________________________________________________ ;____________________________________________________________________________ GGD <28> ; Patch the Deferred Task Manager Task Dispatcher to re-enable interrupts upon ; exiting to fix QuickMail servers (for now at least), because they call through ; jDisptch at application level and wind up with all interrupts being masked, ; and the Task Dispatcher expects to only be called from then end of an interrupt ; handler, which will restore the interrupt level via an RTE. PatchOSJump ROMvDisptch,$B9 ; set addr for BackToTrap InstOSTp NewvDisptch,$B9 ; Éand then install patch ; End of Deferred Task Manager Task Dispatcher Patch Installation ;____________________________________________________________________________ GGD <28> ;____________________________________________________________________________ EH <4> ; Patch the secondary dispatcher for the Level 4 IOP interrupt handler ; ; This patch fixes some of the serial thruput problems encountered on Apollo ; and the TimLC when the CPUs are running in 24 bit mode. These are the first (slow) ; machines to run in 32 bit mode a good portion of the time. Every time we ; take an interrupt the primary interrupt handler switches the CPU into 24 bit ; mode, forcing cache flushes and generally incurring a lot of overhead. At high ; serial baud rates, this overhead interferes with our ability to keep pace. ; ; The solution is to NOT take an interrupt for each received character, but ; rather to empty out the SCC FIFO before returning to the primary interrupt ; dispatch. So we patch the secondary dispatcher to call the serial driver's ; interrupt routines, rather than jumping to them. The service routine then ; returns to the secondary dispatcher, with checks to see if there is another ; character available on either port. If there is another character pending, ; the secondary dispatch routine loops to dispatch to the new interrupt (that ; must be pending) rather than returning to the primary dispatcher (and incurring ; the overhead). ; ; we do this ONLY FOR CHANNEL A, since it's the only one that's documented to be ; fast, and also so that appletalk won't pound us at its 230Kbaud rate. ; ; The Serial Driver Rx and SC interrupt service routines are patched in conjuction ; with this so that if a device manager request completes, we exit to the primary ; dispatcher regardless of whether a character is pending. This way we exit the ; interrupt handlers completely at least once per device manager request, allowing ; things like deferred tasks to run more in sequence. ; ; <6> fix the Farallon MacRecorder bug ; modified patch to test for port A RxChar and loop back only if we are servicing ; a channel A interrupt. For port B we just jump to service routine like ROM code. ; ; <7> fix patch <6> ; Port B dispatcher fixed to call serial service routines and to have same number ; of bytes before the rts so that async serial driver Rx and SCx patches (which ; fiddle with return address on stack) return to the proper place. ; ; <10> for VM, patch level 4 autovector pointed to by vbr, not just the zero-offset. ROMLevel4SccInt equ $00009b20 ; ROM offset of old SCCIOPBypassInt IntRegsSize equ 8*4 ; size of IntRegs in bytes, must change if IntRegs changes SccDT equ Lvl2DT ; Dispatch table for SCC interrupts bra.s SCCPatchEnd ; branch to install code SCCPatchStart ; ; this is the secondary dispatcher code we are patching ; SCCIOPBypassInt SccDecode MOVE.L SCCRd,A0 ; get SCC read address MOVE.B (A0),D0 ; read to sync up SCC (just to be safe) MOVE.L SCCWr,A1 ; get SCC channel B write control address SccDecodeCommon MOVE.B #2,(A1) ; point to SCC register 2, B channel LEA SccDT,A2 ; point to dispatch table and delay MOVEQ #$0E,D0 ; 'and mask' and extra delay AND.B (A0),D0 ; read the modified interrupt vector CMPI.B #8,D0 ; channel A interrupt? bge.S @GoChannelA ; branch if for A <6> MOVE.L 0(A2,D0.W*2),A2 ; get dispatch vector <6> jsr (A2) ; call service routine <6> <7> bra.s @fixedRTS ; fill in the 10 bytes as for port A <7> dcb.b 8,0 ; so that Rx,SCx handlers return properly <7> @fixedRTS rts @GoChannelA ; <6> ADDQ #2,A0 ; adjust SCC addresses for port A ADDQ #2,A1 @GoLvl2 MOVE.L 0(A2,D0.W*2),A2 ; get dispatch vector jsr (A2) ; call service routine <6> ; note we return here UNLESS we finish a device manager request ; note this is 10 bytes-- CHANGE THE SCC RxIntHnd patch equate if this changes!!!!! movea.l SCCRd,a0 ; point to SCC read base address <6> btst.b #RxCA,2(a0) ; SCC data available on channel A? <6> bne.s SCCDecode ; data available, so dispatch again <6> ; note we return here if we are finishing a device manager request rts ; return to primary dispatcher <6> SCCIOPBypassIntEnd ; for IOP patch below <7> SCCIOPBypassIntSize equ SCCIOPBypassIntEnd-SCCIOPBypassInt; <7> ; ; ; this is the primary dispatch code we patch in order to get to the secondary dispatcher ; Level4SccInt movem.l IntRegs,-(SP) ; preserve registers lea SCCIOPBypassInt,a3 ; use the SCC IOP interrupt dispatcher jmpROM ServiceIntMMU ; call primary interrupt dispatcher SccPatchEnd @SccPatchSize equ SccPatchEnd-SccPatchStart ; ; this is the install code for the patch ; cmpRA ROMLevel4SccInt,AutoInt4 ; see if we have an SCC bypass interrupt handler bne.s @done ; if not, don't need to install it moveq.l #@SccPatchSize,d0 ; space needed _NewPtr ,Sys,Clear ; allocate it bne.s @done ; if can't allocate, don't install movea.l a0,a1 ; dest address lea SccPatchStart,a0 ; source address moveq.l #@SccPatchSize,d0 ; byte count _BlockMove ; copy the code move.w sr,-(sp) ; save old int level ori.w #HiIntMask,sr ; disable all ints lea Level4SccInt-SccPatchStart(a1),a1 ; get offset to primary dispatcher code move.l a1,AutoInt4 ; install the new level 4 handler movec vbr,a0 ; get the vbr in case VM moved it <10> move.l a1,AutoInt4(a0) ; install the new level 4 handler <10> move.w (sp)+,sr ; restore the interrupt level @done ; End of Lvl4Int secondary dispatcher patch ;____________________________________________________________________________ EH <4> ;____________________________________________________________________________ EH <30> ; Patch the SCCIOP interrupt handler to set all completion flags to immediate. ; ; If we are running on a machine with an OSS, then we allocate a block for this patch, ; copy the code into it, and install it (saving heap space on machines that don't need it). ; ; This patch fixes things so that TIMBUKTU/REMOTE, slime code that it is, will ; run on a Zone 5 in IOP mode. Timbuktu/Remote makes SYNCHRONOUS read calls to ; the IOP Serial driver FROM A VBL TASK. Sometimes the VBL gets called when the ; IOP Manager completion routine (for a call from the serial driver) is waiting ; to run (as a deferred task). ; The result is that the synchronous call waits in the Device Manager for IODone ; to be called, but IODone is called by the serial driver, which is returned to ; from the IOP Manager completion routine, which is NEVER returned to because we ; made a synchronous call from an elevated interrupt level and are now just sitting ; in a loop in the device manager: DEADLOCK. ; ; The reason Timbuktu/Remote worked on previous Macs is that the bypass driver ; does NOT use a deferred task to call IODone. ; ; The work around is to not call the IOP Manager completion routines for the SCC ; as deferred tasks, but instead call them at level 1 (immediately). ; ; <10> for VM, patch level 4 autovector pointed to by vbr, not just the zero-offset. ROMLevel4SccIopInt equ $00009B10 ; ROM offset of old Level4SccIopInt bra.s SccIopPatchEnd ; skip over patch body into install code SCCIopPatchStart SCCIopInt move.l #(%11111111<<24)+\ ; complete xmt msgs 0..7 immediate (%11111111<<16)+\ ; complete rcv msg 0..7 immediate (0<<8)+\ ; this byte must be zero 0,d0 ; SCC IOP number 0 jmpROM IOPInterrupt ; handle the interrupt and return Level4SccIopInt movem.l IntRegs,-(SP) ; preserve registers lea SccIopInt,a3 ; use the SCC IOP interrupt dispatcher jmpROM ServiceIntMMU ; call primary interrupt dispatcher SccIopPatchEnd @SccIopPatchSize equ SccIopPatchEnd-SccIopPatchStart cmpRA ROMLevel4SccIopInt,AutoInt4; see if we have an SCC IOP interrupt handler bne.s @done ; if not, don't need to install it moveq.l #@SccIopPatchSize,d0 ; space needed _NewPtr ,Sys,Clear ; allocate it bne.s @done ; if can't allocate, don't install movea.l a0,a1 ; dest address movea.l a0,a2 ; save dest addr <7> lea SccIopPatchStart,a0 ; source address moveq.l #@SccIopPatchSize,d0 ; byte count _BlockMove ; copy the code ;patch the bypass interrupt handler moveq.l #SCCIOPBypassIntSize,d0 ; space needed _NewPtr ,Sys,Clear ; allocate it bne.s @done ; if can't allocate, don't install movea.l a0,a1 ; dest address lea SCCIOPBypassInt,a0 ; source address moveq.l #SCCIOPBypassIntSize,d0 ; byte count _BlockMove ; copy the code ; install the patches move.w sr,-(sp) ; save old int level ori.w #HiIntMask,sr ; disable all ints movea.l IOPmgrVars,a0 ; get IOP mgr globals <7> move.w #SccIopNum,d0 ; get SCC IOP number as index <7> ; fortunately the IOPInfoPtrs are declared at offset zero to IOPMgrVars <7> ; as shown here below <7> ; IOPMgrGlobals record {IOPInfoPtrs},increment ; globals used by the IOP manager <7> move.l (a0,d0.w*4),d0 ; get SCCIOP info pointer <7> move.l d0,a0 ; <7> with IOPInfo ; <7> move.l a1,BypassHandler(a0) ; a1-> secondary SCC dispatcher <7> endwith ; <7> lea Level4SccIOPInt-SccIopPatchStart(a2),a1 ;a2-> primary handler <7> move.l a1,AutoInt4 ; install the new level 4 handler movec vbr,a0 ; get the vbr in case VM moved it <10> move.l a1,AutoInt4(a0) ; install the new level 4 handler <10> move.w (sp)+,sr ; restore the interrupt level @done ; End of SccIopInt patch ;____________________________________________________________________________ EH <30> ;____________________________________________________________________________ EH <33> ; ; Install patch to IOPMoveData to fix IOP Serial Driver DMA Hang bug ; IMPORT IOPMoveData ;save state move.w sr, -(sp) ; disable interrupts while we patch ori.w #HiIntMask,sr movem.l a0/d0-d1,-(sp) ; save some registers ; Install patch only if IOPMoveData trap already exists clr.l d0 ; clear result move.w #UnimplementedTrap,d0 ; get loc of unimplemented trap _GetTrapAddress ,newTool ; get the address into A0 move.l a0,d1 ; save it for a sec move.l #$A088,d0 ; trap ID to check for IOPMoveData _GetTrapAddress ,NewOS ; get the address of it cmp.l a0,d1 ; is it implemented? beq.s @noTrap ; nope, so don't install patch PatchOSJump oldIOPMoveData,$88 ; set addr for BackToTrap InstOSTp IOPMoveData,$88 ; implemented, so install new trap addr @noTrap ; restore state movem.l (sp)+,a0/d0-d1 ; restore saved registers move.w (sp)+,sr ; disable interrupts while we patch ; ; ; End Install patch to IOPMoveData to fix IOP Serial Driver DMA Hang bug ;____________________________________________________________________________ EH <33> ;____________________________________________________________________________ <10> <cch 8/6/91> ; ; Install patch to getPageDesc and _VM. These patches take effect only on ; 68030 machines with the Terror ROM, version $67c, sub-version 1.5 when VM ; is disabled. ; ROMVers EQU $08 ROMSubVers EQU $12 VMGlobals EQU $b78 IMPORT GetPageDescPatch ; check to see if we're on the Terror ROM, subversion 1.5. cmp.b #cpu68030,CPUFlag ; check if we're on an 030 bne.s @skipPatch ; ignore if not move.l RomBase,a0 ; get base of ROM cmp.w #TERRORmajorVers,ROMVers(a0); check major version bne.s @skipPatch ; ignore if incorrect version cmp.b #TERRORminorVers,ROMSubvers(a0); check subversion bne.s @skipPatch ; ignore if incorrect version tst.l VMGlobals ; check if VM is running bpl.s @skipPatch ; ignore if VM is on ; patch _Gestalt PatchOSJump origGestalt,$AD ; set addr for BackToTrap to orig _Gestalt InstOSTp GetPageDescPatch,$AD ; implemented, so install new trap addr PatchOSJump origMemDisp,$5C ; set addr for JsrTrap back to orig _VM PatchOSJump origMemDisp1,$5C ; set addr for BackToTrap back to orig _VM InstOSTp MemDispPatch,$5C ; implemented, so install new trap addr @skipPatch ; ; ; End Install patch to getPageDesc proc ;____________________________________________________________________________ <10> <cch 8/6/91> ;____________________________________________________________________________ <13> <cch 8/28/91> ; ; Install patch to flush after backpatching fp68k calls made from ROM on ; 68030 running Terror. ; FP68kPatchInstall ; check to see if we're on the Terror ROM, subversion 1.5. cmp.b #cpu68030,CPUFlag ; check if we're on an 030 bne.s @skipPatch ; ignore if not move.l RomBase,a0 ; get base of ROM cmp.w #TERRORmajorVers,ROMVers(a0) ; check major version bne.s @skipPatch ; ignore if incorrect version cmp.b #TERRORminorVers,ROMSubvers(a0) ; check subversion bne.s @skipPatch ; ignore if incorrect version MOVE.W HwCfgFlags,D0 ; Get them Hardware Config Flags <14> SAM BTST.L #hwCbFPU,D0 ; Does we gots an FPU? <14> BEQ.S @skipPatch ; -> No, don't patch <14> ; patch _FP68k ($A9EB) PatchToolJump origFP68k,$1EB ; set addr for BackToTrap to orig _Gestalt InstToolTp FP68kPatch,$1EB ; implemented, so install new trap addr @skipPatch ; ; ; End Install patch to fp68k ;____________________________________________________________________________ <13> <cch 8/28/91> EndMacSection ;========================================================================================= ;========================================================================================= ;= = ;= END OF MAC-ONLY INSTALL CODE SECTION = ;= = ;========================================================================================= ;========================================================================================= skipMacOnlyPatches ; above code is for Mac OS only ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ ;############################### END PATCH INSTALL CODE ################################# ;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ CalcPatchSize Lea StartPatch,A1 ; start of patch code Lea EndOfPatch,A0 ; end of patch code Sub.l A1,A0 ; calc size of resident code Move.l A0,D0 ; SetHandleSize takes new size in D0 Move.l (SP)+,A0 ; restore the handle passed by SysPatch Bra CutBack ; go cut the install code off and exit ENDPROC UsesPtchInst ; Patchinstll code Macro <1.1-4april89-CEL> ; added Proc FixupTbl from PatchIIROM.a <24July89smb> Proc Export FixupTbl ;################## TABLE OF LOCATIONS TO BE OFFSET BY [ROMBase] #################### ; ; Table contains one entry for each reference to a ROM address from the ; JmpROM, JsrROM, and CmpRA macros. &RomFixIndex is a count of the total ; number of these references, and is also used to synthesize the label ; names for them. The reference labels have the form ; RXXX000, RXXX001, RXXX002, ... ; See the Macro definitions for JmpROM, JsrROM and CmpRA (near the top of ; this file) for more info. ; ; MkTbl is a macro only because assembler while loops will not work ; outside of macros. ; MACRO MkTbl gbla &RomFixIndex while &RomFixIndex <> 0 do &RomFIxIndex seta &RomFixIndex - 1 dc.l RXXX&I2S(&ROMFIXINDEX,-3) - FixUpTbl endwhile dc.l 0 ; Zero entry marks end of table ENDM FixupTbl MkTbl ;########################### END PATCHES CUT BACK CODE ################################ EndProc END