mirror of
https://github.com/elliotnunn/mac-rom.git
synced 2025-01-01 11:29:27 +00:00
0ba83392d4
Resource forks are included only for .rsrc files. These are DeRezzed into their data fork. 'ckid' resources, from the Projector VCS, are not included. The Tools directory, containing mostly junk, is also excluded.
3854 lines
154 KiB
Plaintext
3854 lines
154 KiB
Plaintext
;__________________________________________________________________________________________________
|
|
;
|
|
; File: MemoryMgr.a
|
|
;
|
|
; Contains: Client interface routines for 32-Bit Macintosh Memory Manager
|
|
;
|
|
; Supports multiple heap zones with mixed relocatable and non-relocatable storage blocks.
|
|
; Reorganized from code designed and written by Bud Tribble, 27-Nov-81, and maintained
|
|
; and modified by Angeline Lo, Larry Kenyon, and Andy Hertzfeld.
|
|
;
|
|
; Written by: Bud Tribble
|
|
;
|
|
; Copyright © 1981-1993 by Apple Computer, Inc. All rights reserved.
|
|
;
|
|
; Change History (most recent first):
|
|
;
|
|
; <SM32> 10/12/93 SAM Roll in <MC2> from mc900ftjesus.
|
|
; <MC2> 10/12/93 SAM Updated InitMemMgr to be in sync with the figment version. From
|
|
; now on we should always build with Figment and choose which
|
|
; memmgr we want at a per launch basis.
|
|
; <SM31> 5/20/93 BT Update sources to conditionally compile Figment under the build
|
|
; flag "hasNewHeapMgr".
|
|
; <SM30> 5/18/93 BT Change names of InitMemMgr, InitMemVect, and
|
|
; SetApplBaseAfterBoot to "old_Å". Changed some BSR's to BSR.L's.
|
|
; All in the name of Figment, our beloved Walt Disney friend.
|
|
; <SM29> 4/23/93 kc Roll in MoveHHi, NewPtr and RealocHandle patches from
|
|
; MemoryMgrPatches.a. Integrate some of the other patches to save
|
|
; a few cycles.
|
|
; <SM28> 4/22/93 rab Removed code from FlushFontCaches that skipped the routine if
|
|
; CurApName wasn't valid. This fixes Radar# 1077877 and 1078314.
|
|
; <SM27> 4/12/93 chp Rewrote MoveHHi head patch using half the number of
|
|
; instructions.
|
|
; <SM26> 3/9/93 rab Bumped StkSlop in MoveHHi from 1024 to 3072 per Jim Reekes
|
|
; suggestion in bug #1067917.
|
|
; <SM25> 2/22/93 PN Quad-word align the stackbuffer in MoveHHi call for 040
|
|
; machines.
|
|
; <SM24> 1/27/93 kc Fix bug in MaxAppleZone where we were rounding down the address
|
|
; of the HeapZone to the nearest quadword but not the address we
|
|
; were asking for. Fixes bug 1061295.
|
|
; <SM23> 1/5/93 PN In the SuperMario ROM, many resources are put in ROM that used
|
|
; to be in the system disk. DisposeHandle is modified to leave the
|
|
; handle alone if it is passed a handle to a table or resource in
|
|
; ROM.
|
|
; <SM22> 12/4/92 RB Rolled in changes to support NuKernel from Wayne Meretski and
|
|
; Russell Williams. Make RHBusErrHandler compatible with NuKernel.
|
|
; <SM21> 12/4/92 SWC Moved InitMemMgr here from StartInit.a.
|
|
; <SM20> 12/3/92 PN Back out previous change. This is the patch
|
|
; BracketSetAppBaseWithInitApplZones
|
|
; It will patch out the SetApplBase at Gibbly time when
|
|
; InstallMgr get called. The time when this patch applied is
|
|
; crucial (during boot time up to gibbly time this patch is NOT to
|
|
; be used, it is used after gibbly time)
|
|
; <SM19> 11/19/92 PN Radar #1050264 the SetAppBase was calling bsr IAZ instead of
|
|
; _InitApplZone so a lot of clean up code was missing and lead to
|
|
; the bug. So I bracket the routine SetApplBase with the whole
|
|
; _InitApplZone. This is the patch
|
|
; BracketSetAppBasewithInitApplZone.
|
|
; <SM18> 11/3/92 RB Do not force 32 bit mode for the LC930 by checking the feature
|
|
; conditional Supports24Bit.
|
|
; <SM17> 10/26/92 CSS Fix some short branches and remove cache flushing from
|
|
; HLock. Also, added Supports24Bit conditional to conditionalize
|
|
; _StripAddress.
|
|
; <SM16> 9/16/92 CSS Fix bsr to IOPram to use bsr.l.
|
|
; <SM15> 7/28/92 PN Put CacheFlush back in HLock for compatibilty's sake
|
|
; <SM14> 7/28/92 PN Bug fix. NewPtr call that grows the system heap will crash into
|
|
; Macsbug. The setting of semaphore flag was done incorrectly in
|
|
; ResrvMem and consequently caused the problem. It is fixed. Also,
|
|
; I take out the 040 cache flush in HLock to improve performance.
|
|
; <SM13> 7/16/92 PN Take out cpu ³ 020 conditionals
|
|
; <SM12> 6/11/92 PN Roll in FixStripAddress from PatchIIciROM.a
|
|
; <SM11> 5/22/92 kc Append "Trap" to the names of
|
|
; CompactMem,GetPtrSize,HandleZone,InitZone,MaxBlock,NewEmptyHandl
|
|
; e,PtrZone,PurgeSpace,ReAllocHandle,RecoverHandle,SetHandleSize,S
|
|
; etPtrSize and StackSpace to avoid name conflict with the glue.
|
|
; <SM10> 5/7/92 stb Added purge proc in InitZone for the TrueType disk cache (from
|
|
; DiskCachePatches.a). Moved some "other mgr" cleanup code around
|
|
; so itÕs all grouped nicely.
|
|
; <SM9> 4/22/92 TN Clean up BracketSetAppBaseWithInitApplZones, roll in
|
|
; AllowNILDisposePtr, ResrvMemGrowSystemHeap,
|
|
; NewPtrGrowSystemHeapCleanly MoveHHiPatch from MemoryMgrPatches.a
|
|
; <SM8> 4/16/92 PN Roll in SetAppBaseCacheCleaner and InitAppZoneCacheCleaner from
|
|
; BassPatches.a to remove any app memory allocated by Bass
|
|
; whenever the application heap is reformatted.
|
|
; <SM7> 3/26/92 kc Force 32-bit mode. (see <SM3> in MMU.a)ÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊÊ
|
|
; ¥¥¥¥ This should be backed out as soon as we carve out space for
|
|
; the 2 Meg rom from the slot address space. ¥¥¥¥;
|
|
; <SM6> 3/24/92 PN Roll in BracketSetAppBaseWithInitApplZones into _SetappBase By
|
|
; calling bsr IAZ three times.
|
|
; <SM5> 3/10/92 PN Back out the last change
|
|
; <SM4> 3/9/92 kc Don't call jSoundDead untill we install the SoundMgr.
|
|
; <SM3> 3/4/92 kc Add a semicolon.
|
|
; <SM2> 3/4/92 kc Roll in vxxHlockPatch, vxxMaxMemPatch, v??CMGutsPatch
|
|
; InitxxZoneAlign, vxxMaxApplPatch, InitxxZoneAlign,
|
|
; AlignxxTrailer and SetApplBasePatch from Terror. Add patch to
|
|
; HLock that flushes cache on 040's.
|
|
; <T8> 7/10/91 RP Patched MaxApplZone for line alignment.
|
|
; <T7> 7/7/91 RP Patched MaxMem, CompactMem and MaxBlock to line-align free space
|
|
; values on 040s.
|
|
; <T6> 6/25/91 RP Patched to line align all blocks on 040 CPUs
|
|
; <T5> 5/29/91 CCH Backed out revision 4.
|
|
; <T4> 5/29/91 WM Changed MOVE to SR into ANDI to SR
|
|
; <T3> 4/21/91 CCH Removed HSetState patch since it was not used and is not needed.
|
|
; <T2> 4/2/91 CCH Patched HLock and HSetState to flush the cache.
|
|
; <1> 10/2/91 JSM first checked in
|
|
; <0> 10/1/91 JSM Created from Heap32.a.
|
|
;
|
|
; Modification history from Heap32.a below:
|
|
;
|
|
; <15> 9/27/91 JSM DonÕt use hasCQD conditional, all future ROMs will have color
|
|
; QuickDraw.
|
|
; <14> 9/16/91 JSM Cleanup header.
|
|
; <13> 6/12/91 LN removed #include 'HardwareEqu.a'
|
|
; <12> 12/19/90 DFH Removed StackSpace and MoveHHi jamming of HiHeapMark when
|
|
; HiHeapMark >= ApplLimit (from change <4>). This was really just
|
|
; a workaround for an app bug (SuperCard 1.00 on a portable), but
|
|
; that app has been fixed. Furthermore, the jamming interferes
|
|
; with parts of the system that want to use stacks other than the
|
|
; current application stack.
|
|
; <11> 7/30/90 BG Removed CPU = 040 conditional in preparation to move from Mac040
|
|
; to Mac32 build.
|
|
; <10> 7/17/90 dba change names of GetHandleSize and StripAddress so they donÕt
|
|
; conflict with glue
|
|
; <9> 3/22/90 MSH New PRAM routines in clock.a require a small change to the
|
|
; caller.
|
|
; <8> 3/8/90 JAL Removed include of QuickEqu.a because it was colliding with StandardEqu.d
|
|
; <7> 1/25/90 BG Removed AERRORs from RecoverHandle and RHBusErrHandler. Added
|
|
; RH040BusErrHandler, and modified RecoverHandle to use the 040
|
|
; bus error handler when necessary.
|
|
; <6> 1/18/90 DVB Include PalettePriv.a
|
|
; <5> 1/11/90 CCH Added include of ÒHardwarePrivateEqu.aÓ.
|
|
; <4> 1/11/90 MSH StkSpacePatch - Because some apps jam ApplLimit directly, it is
|
|
; possible for HiHeapMark>ApplLimit which should never be. This
|
|
; fix will jam ApplLimit into HiHeapMark if this is the case.
|
|
; MoveHHIPatch - MoveHHi uses HiHeapMark which may be in error due
|
|
; to the above problem. This patch forces MoveHHi to make the same
|
|
; check as above.
|
|
; <3> 1/3/90 BG Modifying the AERROR statements related to the 040 to use the
|
|
; correct quoting (single quotes, not double quotes).
|
|
; <2> 1/2/90 SWC Modified RecoverHandle to do a last-ditch check to see if a
|
|
; handle points to a ROM resource if the handle isn't in the
|
|
; system or app heap. Also compressed v24RecoverHandle and
|
|
; v32RecoverHandle into essentially one routine since very little
|
|
; is different. They'll enter with a value to mask off either 24
|
|
; or 32 bits of an address.
|
|
; <2.2> 8/27/89 PKE Delete superfluous withs in InitApplZone; remove unnecessary
|
|
; include of 'ScriptPriv.a'. No code changes.
|
|
; <2.1> 8/22/89 SES Removed references to nFiles.
|
|
; <2.0> 7/14/89 CSL initialized sysheapis32bit flag in systemInfo byte.
|
|
; <1.9> 6/15/89 CSL Changed InitMemVect's way to initialize the 24 and 32 bit memory
|
|
; trap vector table. Used value from RealMemTop instead of MaxSize
|
|
; whenever an outlandish request for memory size. Added vectors
|
|
; for MMPrologue, MMPPrologue, MMHPrologue, MMRHPrologue,
|
|
; MMnoPrologue, MMEpilogue, MMNoErrEpilogue. MMnoPrologue is a new
|
|
; routine to handle routine that did not jump throught any
|
|
; Prologue routine.
|
|
; <¥1.8> 6/10/89 CEL Moved Private.a QuickDraw Equates into proper QuickDraw private
|
|
; file (colorequ.a), got rid of QuickDraw nFiles dependencies and
|
|
; fixed up necessary filesÉ
|
|
; <1.7> 5/24/89 GGD Converted to feature based conditionals, added checks for 68040
|
|
; in places where changes will most likely be needed and generate
|
|
; assembly errors if assembling for a 68040. No code changes.
|
|
; <1.6> 4/28/89 CSL fixed bug in Reallochandle
|
|
; <1.5> 4/17/89 CSL Fixed the slimy Microsoft bugs in MaxBlock per BBM request.
|
|
; <1.4> 3/30/89 CSL write a safe address in address 0 for nil handle.
|
|
; <1.3> 3/22/89 CSL now Hlock, Hunlock, HPurge, HnoPurge, HRSRC, HnoRsrc, HGetFlags
|
|
; and HSetFlags all jump thu MMHPrologue, also added prevention
|
|
; code for slimy Light Speed C.
|
|
; <1.2> 3/11/89 CSL replaced reference of minfree by minfree24 and minfree32.
|
|
; <1.1> 2/28/89 CSL Added initialization for Jump vectors for MM traps and
|
|
; duplicate routines for 32 bit operations.
|
|
; <1.0> 2/13/89 CCH Adding to EASE for first time from 24-Bit source.
|
|
;
|
|
; <C914> 10/29/87 rwh Port to Modern Victorian (on MvMac)
|
|
; <C897> 9/30/87 MSH Port to HcMac (Laguna) by closing sound like numac
|
|
; 6/10/87 EHB Roll in patch to iazInit to clean up palette manager
|
|
; <PB092> 6/2/87 JTC Roll in patch to MoveHHi to handle teeny leftovers.
|
|
;
|
|
; <C817> 2/14/87 JTC In the spirit of C778, adjust the initialization of ApplLimit
|
|
; and HiHeapMark in IAZ to use simply SP-DefltStack rather than
|
|
; old TomaxLimit value.
|
|
; <C778> 2/10/87 JTC Yet YET YET! another HiHeapMark change. C765 above fails because
|
|
; of the Pascal glue in the world that implements MaxApplZone sans
|
|
; trap; that code will grow the zone without changing HHM, causing
|
|
; many deaths to old apps. So we pin things at ApplLimit, even
|
|
; though thatÕs pessimistic. Also, add simple bus error handler to
|
|
; RecoverHandle, to jam ROMBase as bogus relative handle.
|
|
; <C765> 2/5/87 JTC Yet another HiHeapMark (HackHeapMark?) to accommodate the
|
|
; nonstandard mem config at boot time. Until now, HHM was set to
|
|
; ApplLimit (just below BufPtr) when the ApplZone was created.
|
|
; This breaks at start time, when the stack is artificially
|
|
; created down at MemTop/2. The change is to tighten HiHeapMark to
|
|
; indicate more accurately where the highest zone end is, as
|
|
; opposed to where it COULD BE. When the ApplZone is created, we
|
|
; set HiHeapMark to HeapEnd; thereafter, InitZone can set it
|
|
; higher if a strange zone is built above HiHeapMark, and the
|
|
; HeapGuts GrowZone can set it if ApplZone is grown beyond
|
|
; HiHeapMark.
|
|
; <C741> 2/2/87 JTC Undo C587 above, since weÕre using the more conservative
|
|
; approach of running with the stack mid-zone. Change to HeapGuts
|
|
; only.
|
|
; <C731> 2/1/87 JTC Fix MaxApplZone to jam HiHeapMark, in case itÕs out of sync with
|
|
; ApplLimit due to appÕs jamming latter without calling SetApplLimit.
|
|
; <C681> 1/24/87 JTC Move init code from Launch to InitApplZone. When 64K ROM patches
|
|
; were rolled into Heap, IAZ (the old InitApplZone) was kept
|
|
; separate from InitApplZone (the patch stuff, which called IAZ to
|
|
; do the real work). The effect was to deprive SetApplBase (who
|
|
; called IAZ) of the good init stuff. Because SetApplBase is
|
|
; called so early, it cannot benefit from the init code. WARNING:
|
|
; after the earliest time, callers of SetAB should also call
|
|
; InitApplZone to guarantee the best initialization. Among the new
|
|
; init things are the VBL and Snd cleanups and a call to
|
|
; _RDrvrInstall, which code is spread between the sound world and
|
|
; SegmentLoader.a.
|
|
; <C636> 1/14/87 JTC Make SetApplLimit sensitive to minimum stack size in DefltStack.
|
|
; Old-style apps like MacDraw and MacPaint jam ApplLimit to within
|
|
; 8K of SP, causing problems in color. Add bus error handlers to
|
|
; xxxPrologue routines DerefHandle and ZoneCheck, to allow bogus
|
|
; address to return with error codes rather than Deep6ing.
|
|
; <C587> 1/3/87 JTC Support boot time sizing of SysZone by making special case of
|
|
; (SP < bkLim(SysZone)) in HeapGuts routine MaxLimit. That routine
|
|
; constrains the (growing) application zone according to SP and a
|
|
; proposed limit; the test against SP is omitted in the case above
|
|
; to permit jamming the stack into a system heap block. The basis
|
|
; of the trick is that HiHeapMark is the stack lower bound; by
|
|
; forcing it to point into the stack block during boot time, stack
|
|
; can be allocated correctly by QD, _StackSpace, etc. while the
|
|
; heap merrily grows. StackSpace is changed to use HiHeapMark, as
|
|
; it should have from Mac+ days.
|
|
; <C415> 12/9/86 JTC Reinstate C251 alignment. Change to HeapGuts.a only.
|
|
; <A415> 11/17/86 JTC Back out some of the wonderful <C251> alignment feature to
|
|
; support our friends at Microsoft who gang-bang a large set of
|
|
; handles into a large, fixed block without checking for the 2
|
|
; slop bytes we would like to give them to force longword
|
|
; alignment when their request is (2n+1)*2. Parts of <C251> that
|
|
; force alignment of the beginning of the zone are left in for the
|
|
; modest improvement they admit, and because there may be a way to
|
|
; salvage the Microsoft-style apps. The principal change is to
|
|
; routine ActualS in HeapGuts.a, where the rounding to a multiple
|
|
; of 4 is done.
|
|
; <A305> 10/30/86 JTC Roll in Mac+ patch to RecoverHandle to use the more reasonable
|
|
; of SysZone and ApplZone as ÒTheZoneÓ when incoming TheZone is
|
|
; one of Sys or Appl. Since RecoverHandle is strange, passing a
|
|
; pointer to a handle-related routine, the cleanest way is to
|
|
; customize a MMRHPrologue, ÒRHÓ as in RecoverHandle, of course,
|
|
; to do the A6 fixing. The only change to the code here in Heap is
|
|
; to JSR to MMRHPrologue rather than MMPrologue. MMRHPrologue has
|
|
; several extra lines to choose A6 wisely.
|
|
; <C251> 10/24/86 JTC Changes to support longword alignment of blocks. Since the heap
|
|
; header as well as individual block headers are multiples of 4 in
|
|
; size, the problem reduces to getting heaps started on longword
|
|
; boundaries and then allocating blocks in multiples of longwords.
|
|
; The first issue depends on the start code, which determines the
|
|
; system zone, maintainence of a 4x system heap (since the app
|
|
; heap always starts at its end), and a little change in
|
|
; SetAppBase to keep a resized system heap on a long boundary. For
|
|
; home-brew heaps that start on 4x+2 boundaries, InitZone contains
|
|
; some code to fudge the initial block, necessarily a master
|
|
; pointer block, to ensure that all following blocks are longword
|
|
; aligned. Code in InitZone ensures that the heap limit is on a
|
|
; long boundary.
|
|
; Once the zone is aligned, itÕs up to internal code to keep
|
|
; allocations properly aligned. Internal routine ActualS is the key.
|
|
; It now rounds up to the nearest multiple of 4 (not 2, as before);
|
|
; the slop is picked up by internal routine SetLogSize as usual.
|
|
; ApplLimit and its associated routine SetApplLimit are left as is,
|
|
; since the variable may be tweaked from within an app with impunity.
|
|
; Instead, uses of ApplLimit for resizing the zone or calculating
|
|
; available size are carefully monitored. Internal routine MaxLimit
|
|
; usually brackes uses of ApplLimit, so it is made to round down to
|
|
; the nearest multiple of 4. MaxApplZone, which uses ApplLimit
|
|
; directly, are modified to round it down to a multiple of 4 first.
|
|
; StackSpace rounds the SP down to a multiple of 4 before computing
|
|
; available space.
|
|
; 10/17/86 CRC InitApplZone sets up FMExist, LastSpExtra only for Font Manager.
|
|
; <C206> 10/9/86 bbm Modified to mpw aincludes/ changed resourc to resource.
|
|
; <C169> 9/23/86 JTC Add _StripAddress.
|
|
; 3/31/86 BBM added a blank line in order to work with the assembler.
|
|
; 2/19/86 BBM Made some modifications to work under MPW
|
|
; 1/13/86 JTC 1. Patch InitApplZone to inval font cach by setting LastSPExtra
|
|
; to -1. <13Jan86 JTC>
|
|
; 2. Save/Restore theZone across MoveHHi so theZone can be set to
|
|
; that of the moving handle.
|
|
; 3. Add 4 calls to set/clr rsrc bit of handle and pull flag byte
|
|
; from master ptr.
|
|
;
|
|
; 11/1/85 JTC New global HiHeapMark tracks heap ends below SP -- InitApplZone,
|
|
; SetApplLimit, <01Nov85> InitZone, MoveHHi.
|
|
;
|
|
; 10/27/85 BBM needed more space, added three branch shorts.
|
|
; 5/29/85 JTC Made a MOVE a MOVEQ to be sure error code is long. <29May85>
|
|
; 5/21/85 JTC Distinguish Nil and empty in Disposehandle. <21May85>
|
|
; 5/10/85 KWK Removed JSR to OS code. <10May85>
|
|
; 5/10/85 JTC InitMem moved to Start code to facilitate separate linking. <10May85>
|
|
; 5/9/85 JTC Added call StackSpace. <09May85>
|
|
; 5/6/85 JTC Initialize new FontMgr globals. <06May85>
|
|
; 5/4/85 JTC Incorporated MoveHHi. <04May85>
|
|
; 4/27/85 JTC Moved MakeStkPB into code with conditional assembly, to
|
|
; facilitate RAM-based... <27Apr85>
|
|
; 4/25/85 JTC Add new common exit for non-MMEpilogue routines to stuff return
|
|
; code in MemErr. <25Apr85> Amend RecoverHandle to use special
|
|
; MMNoErrEpilogue.
|
|
; 4/24/85 JTC Modify HLock, etc. to bypass MMHPrologue to weaken their error
|
|
; checking (since <24Apr85> MicroSoft calls with bad handles) but
|
|
; to bypass 0 when passed in.
|
|
; 4/24/85 JTC Added MaxApplLimit, with slight modifications from OSTraps
|
|
; source. <24Apr85>
|
|
; 4/23/85 JTC Backed out validity check and error code in RecoverHandle --
|
|
; it's documented as <23Apr85> as such in Inside Mac! Code
|
|
; restored to original version.
|
|
; 4/17/85 JTC Add call PurgeSpace. <17Apr85>
|
|
; 4/16/85 JTC Fix no-error code at end of InitApplZone. <16Apr85>
|
|
; 4/14/85 JTC Change InitApplZone to start zone up high in stack. Subject to
|
|
; further tuning... <14Apr85>
|
|
; 4/13/85 JTC Add validity check and error code to RecoverHandle. <13Apr85>
|
|
; 4/10/85 JTC Add call MaxBlock, to return max available free space that could
|
|
; be gotten <10Apr85> without purge or grow. Revitalize
|
|
; EmptyHandle and DisposeHandle to be effective even in ROZ, so
|
|
; that apps trying to make space by deleting, say, Pack5 will get
|
|
; past that and into some real space savings. EmptyHandle will
|
|
; indeed zero the (bogus) MP, but not attempt to actually free
|
|
; space. DisposeHandle will do likewise; and it won't return the
|
|
; MP to the free list, since there is a permanent connection
|
|
; between ROM bogus MP's and ROM resources.
|
|
; 3/12/85 LAK&JTC Modified sys heap start to $1400, making room for 1K ToolTable @ $C00
|
|
; 3/4/85 JTC Changes to support so-called read-only zones (for ROM map zone,
|
|
; currently). <04Mar85>
|
|
; 1) Add ROZ .EQU 0 to NewEqu, reusing bit fOnCheck.
|
|
; 2) Add memROZWarn .EQU -99 to NewEqu -- report error on abuse.
|
|
; 3) Add memROZError .EQU -99 to NewEqu -- same thing.
|
|
; 4) DisposPtr, DisposHandle -- warning if ROZ.
|
|
; 5) ReallocHandle -- if nonempty, don't empty and warn if try to grow
|
|
; 6) SetHandleSize, SetPtrSize -- don't shrink it.
|
|
; 7) EmptyHandle (the big one) -- don't do it and DON'T signal error!
|
|
; 8) Add _NewEmptyHandle, parallel to NewHandle.
|
|
; 1/30/85 LAK Added ram patch to InitApplZone for world reinit stuff (with REF
|
|
; for MakeStackPB).
|
|
; 1/29/85 LAK New system heap start is $C00.
|
|
; 1/29/85 JTC Rolled in patches, marked <29jan85>:
|
|
; 1) SetApplLimit -- check for memfull error
|
|
; 2) MoreMasters -- add new call
|
|
; 3) InitZone, SetGrowZone -- change to 2-level GrowZoneProc
|
|
; 4) StdGZ -- add standard GrowZoneProc
|
|
; 5) MakePtrSpc -- (internal routine) new code after return from MakeSpace
|
|
; 1/23/85 LAK Adapted for new equate files. Changed 'MoreMasters' to mAllocCnt.
|
|
; 9/12/83 LAK Added GZ semaphores to ReallocHandle.
|
|
; 9/10/83 LAK Changed CompactMem to force heap compact by setting FNSelCompct
|
|
; bit in flags word.
|
|
; 9/4/83 LAK Fixed bug in ReallocHandle (don't dispose handle on memFullErr).
|
|
; 8/18/83 LAK Fixed potential problem with ResrvMem (if used to make room for
|
|
; a handle, it needs to reserve extra space for a master pointer
|
|
; block if we are out of master pointers . . .)
|
|
; 8/12/83 LAK Added clear option for NewPtr, NwHandle, and ReAllocHandle.
|
|
; 7/31/83 LAK DefltFlags now reside in low memory location mmDefFlags: placed
|
|
; there by InitMem and loaded into zone vars by InitZone.
|
|
; 7/28/83 LAK CompactMem, PurgeMem, MaxMem, and FreeMem now call MMPrologue
|
|
; and MMEpilogue; added ResrvMem.
|
|
; 7/24/83 LAK Removed all free-list references; removed .word constants from
|
|
; checking code; made numerous optimizations; initialize DefltStack,
|
|
; MinStack in InitMem and now use these values instead of hard-wired
|
|
; constants.
|
|
; 6/21/83 MPH Put FreeList code under assembly switch: FList.
|
|
; 6/18/83 MPH Put maintenance of minCBFree under Statistics switch.
|
|
; 6/17/83 MPH Space Saving Changes:
|
|
; - Removed CallMoveProc calls
|
|
; - InitMem: tightened Mask Init
|
|
; - SetApplBase:
|
|
; Sub.L #x,A0 => Sub.W #x,A0
|
|
; JSR InitApplZone => BSR.S
|
|
; - InitApplZone:
|
|
; Add.L #x,A0 => Add.W #x,A0
|
|
; Move.L y,x(SP) => Move.L y,-(SP)
|
|
; - Removed unneeded Tst.L D0
|
|
; - GetZone: Repackaged checking code
|
|
; - SetZone: Repackaged checking code
|
|
; - PurgeMem: Moved PMFail to save space,
|
|
; - HLock, HUnLock, HPurge, HNoPurge:
|
|
; Removed unneeded Move.L A1,A0 at HGood.
|
|
; 6/16/83 MPH Removed error return from InitMem, InitApplZone in all cases.
|
|
; Fix CompactMem to adjust size correctly.
|
|
; Remove unused code at MMFail from MaxMem.
|
|
; 6/11/83 theGang Fixed register trashed bug in InitApplZone
|
|
; 6/10/83 MPH Changed order of _RsrcZoneInit, _InitZone in InitApplZone.
|
|
; Included GrafTypes so that Nil is defined.
|
|
; 6/5/83 MPH Fixed InitApplZone by moving IAZCommon to module Heap.
|
|
; 6/2/83 MPH More minor revisions for release, share common code.
|
|
; 4/12/83 MPH Final revisions for release.
|
|
; 2/22/83 LAK,MPH changed to observe pascal regsave conventions
|
|
; 2/17/83 LAK default ApplLimit is now figured from current stack pointer;
|
|
; initmem just inits the system zone.
|
|
; 2/10/83 MPH Changed SetApplLimit to take argument in A0 instead of D0
|
|
; 1/26/83 LAK Changed system heap size to 12K.
|
|
; 1/14/83 AJH Changed RsrcZoneInit invocation to a trap
|
|
; 10/27/82 MPH Begin Zone refinement per suggestions of Andy J. Hertzfeld
|
|
; and Larry A. Kenyon
|
|
; 8/16/82 AL Added ReAllocHandle core routine call
|
|
; 7/30/82 AL Modified for 128K... 16 MBytes memory expansion
|
|
; 4/28/82 AL added locked, purgeble, permanent relocatable blocks
|
|
; 4/7/82 AL use A0 to interface
|
|
; 1/25/82 BT All routines added, preliminary testing done.
|
|
; 12/13/81 BT More routines added.
|
|
;__________________________________________________________________________________________________
|
|
|
|
|
|
BLANKS ON
|
|
STRING ASIS
|
|
|
|
LOAD 'StandardEqu.d'
|
|
INCLUDE 'ColorEqu.a'
|
|
INCLUDE 'palettePriv.a'
|
|
INCLUDE 'MemoryMgrPriv.a'
|
|
INCLUDE 'FontPrivate.a'
|
|
INCLUDE 'HardwarePrivateEqu.a'
|
|
INCLUDE 'UniversalEqu.a'
|
|
|
|
Heap PROC EXPORT ; moved up here to contain equates!
|
|
|
|
IMPORT a24ActualS
|
|
IMPORT a32ActualS
|
|
IMPORT a24AllocBk
|
|
IMPORT a32AllocBk
|
|
IMPORT a24BkCompactS
|
|
IMPORT a32BkCompactS
|
|
IMPORT ClearGZStuff
|
|
IMPORT a24CompactHp
|
|
IMPORT a32CompactHp
|
|
IMPORT a24EH
|
|
IMPORT a32EH
|
|
IMPORT a24FreeBk
|
|
IMPORT a32FreeBk
|
|
IMPORT a24GetSize
|
|
IMPORT a32GetSize
|
|
IMPORT a24HMakeMoreMasters
|
|
IMPORT a32HMakeMoreMasters
|
|
IMPORT a24MakeBkf
|
|
IMPORT a32MakeBkf
|
|
IMPORT a24MakePtrSpc
|
|
IMPORT a32MakePtrSpc
|
|
IMPORT a24MaxLimit
|
|
IMPORT a32MaxLimit
|
|
IMPORT ToMaxLimit
|
|
IMPORT MMNoErrEpilogue
|
|
IMPORT MMEpilogue
|
|
IMPORT MMHPrologue
|
|
IMPORT MMPPrologue
|
|
IMPORT MMPrologue
|
|
IMPORT MMRHPrologue
|
|
IMPORT MMNoPrologue
|
|
IMPORT MMMMPrologue
|
|
IMPORT a24NextMaster
|
|
IMPORT a32NextMaster
|
|
|
|
IMPORT a24PurgeHeap
|
|
IMPORT a32PurgeHeap
|
|
IMPORT ReleaseMP
|
|
IMPORT a24SetSize
|
|
IMPORT a32SetSize
|
|
IMPORT StdGZ
|
|
IMPORT a24ZoneAdjustEnd
|
|
IMPORT a32ZoneAdjustEnd
|
|
|
|
IMPORT FlushApplVBLs ; code borrowed from SegLoader
|
|
IMPORT AppZoneAddr ; code borrowed from SegLoader
|
|
|
|
IMPORT FlushApplNM ; from NotificationMgr.c
|
|
|
|
IF RAMTest THEN
|
|
ELSE
|
|
IMPORT MakeStkPB ;<30jan85> for initapplzone stuff
|
|
ENDIF
|
|
|
|
IMPORT a24TotePurgeables ;new guts routine
|
|
IMPORT a32TotePurgeables ;new guts routine
|
|
|
|
IMPORT BaseofRom
|
|
IMPORT PramIO
|
|
|
|
IMPORT DiskCachePurgeProcedure ; <sm10>
|
|
|
|
EXPORT SetApplBase ; <SM20>
|
|
If hasNewHeapMgr Then
|
|
EXPORT Old_AfterBootSetApplBase ;<SM20>, <SM30 BT>
|
|
else
|
|
EXPORT AfterBootSetApplBase ;<SM20>
|
|
endif
|
|
EXPORT IAZ ; <SM19>
|
|
EXPORT InitApplZone
|
|
EXPORT InitZoneTrap
|
|
EXPORT GetZone
|
|
EXPORT SetZone
|
|
|
|
EXPORT MaxBlockTrap
|
|
EXPORT PurgeSpaceTrap
|
|
EXPORT CompactMemTrap
|
|
EXPORT PurgeMem
|
|
EXPORT FreeMem
|
|
EXPORT ResrvMem
|
|
EXPORT MaxMemTrap
|
|
|
|
EXPORT SetGrowZone
|
|
EXPORT SetApplLimit
|
|
EXPORT StackSpaceTrap
|
|
EXPORT MaxApplZone
|
|
|
|
EXPORT NewPtrTrap
|
|
EXPORT DisposePtr
|
|
EXPORT GetPtrSizeTrap
|
|
EXPORT SetPtrSizeTrap
|
|
EXPORT PtrZoneTrap
|
|
|
|
EXPORT NwHandle
|
|
EXPORT NewEmptyHandleTrap
|
|
EXPORT DsposeHandle
|
|
EXPORT __GetHandleSize
|
|
EXPORT SetHandleSizeTrap
|
|
EXPORT HandleZoneTrap
|
|
EXPORT RecoverHandleTrap
|
|
EXPORT EmptyHandle
|
|
EXPORT ReAllocHandleTrap
|
|
|
|
EXPORT HLock
|
|
EXPORT HUnlock
|
|
EXPORT HPurge
|
|
EXPORT HNoPurge
|
|
EXPORT HRSRC ; set the RSRC bit
|
|
EXPORT HNoRSRC ; clear the RSRC bit
|
|
EXPORT HGetFlags ; get flag byte of master ptr
|
|
EXPORT HSetFlags ; set flag byte of master ptr
|
|
|
|
EXPORT MoreMasters ; new trap added
|
|
EXPORT __StripAddress
|
|
EXPORT MoveHHi ; moved in from glue land
|
|
|
|
EXPORT vIAZInit ; init at top of InitApplZone for vector
|
|
EXPORT vIAZPostInit ; init after zone is created for vector
|
|
|
|
if hasNewHeapMgr Then
|
|
EXPORT Old_InitMemVect ; initialization for Memory Vectors <SM30 BT>
|
|
EXPORT Old_InitMemMgr ; initialize Memory Manager low mem, vectors <SM21>,<SM30 BT>
|
|
else
|
|
EXPORT InitMemVect ; initialization for Memory Vectors <SM30 BT>
|
|
EXPORT InitMemMgr ; initialize Memory Manager low mem, vectors <SM21>,<SM30 BT>
|
|
endif
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; PROCEDURE InitMemVect;
|
|
;
|
|
; This routine initialize the jump vector tables for both 24 and 32 bit
|
|
; memory traps.
|
|
;
|
|
|
|
MACHINE MC68030
|
|
|
|
MMTrapTable ; table for all Memory Manager traps
|
|
; entries have to be in pairs.....
|
|
DC.L v24SetApplBase-BaseofROM ;
|
|
DC.L v32SetApplBase-BaseofROM ;
|
|
DC.L vInitApplZone-BaseofROM ;
|
|
DC.L vInitApplZone-BaseofROM ;
|
|
DC.L v24InitZone-BaseofROM ;
|
|
DC.L v32InitZone-BaseofROM ;
|
|
DC.L vGetZone-BaseofROM ;
|
|
DC.L vGetZone-BaseofROM ;
|
|
DC.L vSetZone-BaseofROM ;
|
|
DC.L vSetZone-BaseofROM ;
|
|
DC.L v24MaxBlock-BaseofROM ;
|
|
DC.L v32MaxBlock-BaseofROM ;
|
|
DC.L v24PurgeSpace-BaseofROM ;
|
|
DC.L v32PurgeSpace-BaseofROM ;
|
|
DC.L v24CompactMem-BaseofROM ;
|
|
DC.L v32CompactMem-BaseofROM ;
|
|
DC.L v24PurgeMem-BaseofROM ;
|
|
DC.L v32PurgeMem-BaseofROM ;
|
|
DC.L vFreeMem-BaseofROM ;
|
|
DC.L vFreeMem-BaseofROM ;
|
|
DC.L v24ResrvMem-BaseofROM ;
|
|
DC.L v32ResrvMem-BaseofROM ;
|
|
DC.L v24MaxMem-BaseofROM ;
|
|
DC.L v32MaxMem-BaseofROM ;
|
|
DC.L vSetGrowZone-BaseofROM ;
|
|
DC.L vSetGrowZone-BaseofROM ;
|
|
DC.L v24SetApplLimit-BaseofROM ;
|
|
DC.L v32SetApplLimit-BaseofROM ;
|
|
DC.L vStackSpace-BaseofROM ;
|
|
DC.L vStackSpace-BaseofROM ;
|
|
DC.L v24MaxApplZone-BaseofROM ;
|
|
DC.L v32MaxApplZone-BaseofROM ;
|
|
DC.L v24NewPtr-BaseofROM ;
|
|
DC.L v32NewPtr-BaseofROM ;
|
|
DC.L v24DisposePtr-BaseofROM ;
|
|
DC.L v32DisposePtr-BaseofROM ;
|
|
DC.L v24GetPtrSize-BaseofROM ;
|
|
DC.L v32GetPtrSize-BaseofROM ;
|
|
DC.L v24SetPtrSize-BaseofROM ;
|
|
DC.L v32SetPtrSize-BaseofROM ;
|
|
DC.L vPtrZone-BaseofROM ;
|
|
DC.L vPtrZone-BaseofROM ;
|
|
DC.L v24NwHandle-BaseofROM ;
|
|
DC.L v32NwHandle-BaseofROM ;
|
|
DC.L v24NewEmptyHandle-BaseofROM ;
|
|
DC.L v32NewEmptyHandle-BaseofROM ;
|
|
DC.L v24DsposeHandle-BaseofROM ;
|
|
DC.L v32DsposeHandle-BaseofROM ;
|
|
DC.L v24GetHandleSize-BaseofROM ;
|
|
DC.L v32GetHandleSize-BaseofROM ;
|
|
DC.L v24SetHandleSize-BaseofROM ;
|
|
DC.L v32SetHandleSize-BaseofROM ;
|
|
DC.L vHandleZone-BaseofROM ;
|
|
DC.L vHandleZone-BaseofROM ;
|
|
DC.L v24RecoverHandle-BaseofROM ;
|
|
DC.L v32RecoverHandle-BaseofROM ;
|
|
DC.L v24EmptyHandle-BaseofROM ;
|
|
DC.L v32EmptyHandle-BaseofROM ;
|
|
DC.L v24ReAllocHandle-BaseofROM ;
|
|
DC.L v32ReAllocHandle-BaseofROM ;
|
|
DC.L v24Hlock-BaseofROM ;
|
|
DC.L v32Hlock-BaseofROM ;
|
|
DC.L v24HUnlock-BaseofROM ;
|
|
DC.L v32HUnlock-BaseofROM ;
|
|
DC.L v24HPurge-BaseofROM ;
|
|
DC.L v32HPurge-BaseofROM ;
|
|
DC.L v24HNoPurge-BaseofROM ;
|
|
DC.L v32HNoPurge-BaseofROM ;
|
|
DC.L v24HRSRC-BaseofROM ;
|
|
DC.L v32HRSRC-BaseofROM ;
|
|
DC.L v24HNoRSRC-BaseofROM ;
|
|
DC.L v32HNoRSRC-BaseofROM ;
|
|
DC.L v24HGetFlags-BaseofROM ;
|
|
DC.L v32HGetFlags-BaseofROM ;
|
|
DC.L v24HSetFlags-BaseofROM ;
|
|
DC.L v32HSetFlags-BaseofROM ;
|
|
DC.L v24MoreMasters-BaseofROM ;
|
|
DC.L v32MoreMasters-BaseofROM ;
|
|
DC.L v24MoveHHi-BaseofROM ;
|
|
DC.L v32MoveHHi-BaseofROM ;
|
|
TableEnd
|
|
|
|
;----------------------------------------------------------------
|
|
; InitMemMgr: Set up some low memory constants related to the memory manager.
|
|
; Destroys none
|
|
; Called by System initialization sequence.
|
|
;----------------------------------------------------------------
|
|
IF hasNewHeapMgr Then
|
|
Old_InitMemMgr
|
|
ELSE
|
|
InitMemMgr
|
|
ENDIF
|
|
|
|
IF hasNewHeapMgr Then
|
|
Old_InitMemVect
|
|
ELSE
|
|
InitMemVect
|
|
ENDIF
|
|
BSET #MMStartMode,MMFlags ; Force 32-bit mode
|
|
BSET #MMMixed,MMFlags ; Force 32-bit mode
|
|
BSET #MMSysheap,MMFlags ; Force 32-bit mode
|
|
BSET #MMROZheap,MMFlags ; Force 32-bit mode
|
|
BCLR #mmHighSysHeap,MMFlags ; Force 32-bit mode
|
|
|
|
BCLR.B #Systemis24bit,SystemInfo ; set system to be 32 bit
|
|
BCLR.B #Sysheapis24bit,SystemInfo ; set system heap to be 32 bit
|
|
@2
|
|
MOVE.L ROMBase,D0 ; get ROMbase <v1.4>
|
|
LEA MMTrapTable,A0 ; A3 gets beginning of table
|
|
LEA JMemMgr24,A1 ; set A1 to start of 24 bit table <v1.1>
|
|
LEA JMemMgr32,A2 ; set A2 to start of 32 bit table <v1.1>
|
|
MOVE.L #(((TableEnd-MMTrapTable)>>3)-1),D1 ; get number of entry pairs <v1.9>
|
|
@NextEntry
|
|
MOVE.L (A0)+,(A1) ; copy offset to entries in 24 bit table <v1.9>
|
|
ADD.L D0,(A1)+ ; Add ROM Base to offset <v1.9>
|
|
MOVE.L (A0)+,(A2) ; copy offset to entries in 32 bit table <v1.9>
|
|
ADD.L D0,(A2)+ ; Add ROM Base to offset <v1.9>
|
|
DBRA D1,@NextEntry ; loop until end <v1.9>
|
|
|
|
ADD.L #$00010000,D0 ; makeup a safe address <v1.9>
|
|
MOVE.L D0,$0 ; write address for nil handle <v1.9>
|
|
MOVE.L D0,$4 ; write address for nil window pointer <v1.9>
|
|
|
|
LEA MMNoPrologue,A0 ; get address <v1.9>
|
|
Move.L A0,vMMNoPrologue ;
|
|
LEA MMPPrologue,A0 ; get address
|
|
Move.L A0,vMMPPrologue ;
|
|
LEA MMPrologue,A0 ; get address
|
|
Move.L A0,vMMPrologue ;
|
|
LEA MMHPrologue,A0 ; get address
|
|
Move.L A0,vMMHPrologue ;
|
|
LEA MMRHPrologue,A0 ; get address
|
|
Move.L A0,vMMRHPrologue ;
|
|
LEA MMMMPrologue,A0 ; get address
|
|
Move.L A0,vMMMMPrologue ;
|
|
LEA MMEpilogue,A0 ; get address
|
|
Move.L A0,vMMEpilogue ;
|
|
LEA MMNoErrEpilogue,A0 ; get address
|
|
Move.L A0,vMMNoErrEpilogue ; <v1.9>
|
|
|
|
|
|
MOVEM.L (SP)+,D0-D3/A0-A3 ; restore registers <v1.1>
|
|
RTS ; <v1.1>
|
|
|
|
;----------------------------------------------------------------------
|
|
; PROCEDURE SetApplBase(a: Address);
|
|
;
|
|
; Sets the Application Zone Base, and Inits the Application Zone
|
|
; No error setting required here,since IAZ does all the hard work. <25Apr85>
|
|
; These days use the full splendor of InitApplZone.
|
|
;
|
|
; Argument:
|
|
; A0: New Base Address for Applic Zone.
|
|
;
|
|
; Result:
|
|
; D0: ec: error code.
|
|
;
|
|
; Registers:
|
|
; D0: amount to adjust counter by, new counter value.
|
|
; A0: points to counter, max pair.
|
|
;
|
|
|
|
SetApplBase
|
|
; Roll in SetAppBaseCacheCleaner from BassPatches.a <SM8> <PN>
|
|
; Remove any application memory allocated by Bass whenever the application heap is
|
|
; reformatted. SetAppBase and InitApplZone must be patched since they both trash the
|
|
; application heap.
|
|
TST.B CurApName ;checking for app name <SM8> <PN>
|
|
BMI @SkipIt ;still in booting, skip the patch <SM8> <PN>
|
|
MOVEM.L A0-A2/D0-D3,-(sp) ; save registers <SM8> <PN>
|
|
CLR.L -(SP) ; room for result <SM8> <PN>
|
|
MOVE.L applZone,A0 ; point to zone hdr <SM8> <PN>
|
|
MOVE.L A0,-(SP) ; push start ptr <SM8> <PN>
|
|
MOVE.L (A0),-(SP) ; push zone trailer <SM8> <PN>
|
|
_sbKillSomeCaches ; <SM8> <PN>
|
|
ADDQ #4,SP ; ignore result <SM8> <PN>
|
|
MOVEM.L (sp)+,A0-A2/D0-D3 ; restore registers <SM8> <PN>
|
|
@SkipIt
|
|
Move.L #JSetApplBase,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
; 24 bit version
|
|
v24SetApplBase
|
|
MoveM.L D3/A2-A6,-(SP) ; Save Registers.
|
|
Move.L SysZone,A6 ; Get sysZone pointer.
|
|
Move.L A0,D0 ; round base up to 4x <C251><sm9>stb
|
|
Cmpi.b #cpu68040,CPUFlag ; on an 040 or greater? <2> kc.start
|
|
Blo.s @0 ; No? Then long align.
|
|
|
|
Add.L #15,D0 ; Quad-word align
|
|
AndI.W #$FFF0,D0 ; D0 = 16x; D0 >= A0
|
|
bra.s @continue
|
|
|
|
@0 AddQ.L #3,D0 ;
|
|
AndI.W #$FFFC,D0 ; D0 = 4x; D0 >= A0
|
|
@continue ; <2> kc.end
|
|
MoveA.L D0,A0 ; <C251>
|
|
Sub.W #MinFree24,A0 ; Point to new end block. <v1.2>
|
|
Cmp.L BkLim(A6),A0 ; New sysZone end <= current?
|
|
BLS.S @SABInitAppl ; Yes, just init applZone.
|
|
BSR a24ZoneAdjustEnd ; Adjust the sysZone end. <v1.1>
|
|
@SABInitAppl
|
|
BSR IAZ ; use historical IAZ call <C681>
|
|
MoveM.L (SP)+,D3/A2-A6
|
|
RTS
|
|
|
|
; 32 bit version
|
|
v32SetApplBase
|
|
MoveM.L D3/A2-A6,-(SP) ; Save Registers.
|
|
Move.L SysZone,A6 ; Get sysZone pointer.
|
|
Move.L A0,D0 ; round base up to 4x <C251><sm9>stb
|
|
Cmpi.b #cpu68040,CPUFlag ; on an 040 or greater? <2> kc.start
|
|
Blo.s @0 ; No? Then long align.
|
|
|
|
Add.L #15,D0 ; Quad-word align
|
|
AndI.W #$FFF0,D0 ; D0 = 16x; D0 >= A0
|
|
bra.s @continue
|
|
|
|
@0 AddQ.L #3,D0 ;
|
|
AndI.W #$FFFC,D0 ; D0 = 4x; D0 >= A0
|
|
@continue ; <2> kc.end
|
|
MoveA.L D0,A0 ; <C251>
|
|
Sub.W #MinFree32,A0 ; Point to new end block. <v1.2>
|
|
Cmp.L BkLim(A6),A0 ; New sysZone end <= current?
|
|
BLS.S @SABInitAppl ; Yes, just init applZone.
|
|
BSR a32ZoneAdjustEnd ; Adjust the sysZone end. <v1.1>
|
|
@SABInitAppl
|
|
BSR.S IAZ ; use historical IAZ call <C681>
|
|
MoveM.L (SP)+,D3/A2-A6
|
|
RTS
|
|
|
|
;----------------------------------------------------------------------
|
|
; PROCEDURE AfterBootSetApplBase(a: Address);
|
|
;
|
|
; This is the patch BracketSetAppBaseWithInitApplZones
|
|
; It will patch out the SetApplBase at Gibbly time when InstallMgr
|
|
; get called. The time when this patch applied is crucial (during boot
|
|
; time up to gibbly time this patch is NOT to be used, it is used
|
|
; after gibbly time) <SM20> PN
|
|
;
|
|
|
|
If hasNewHeapMgr Then
|
|
; <SM30 BT>: Change AfterBootSetApplBase to Old_AfterBootSetApplBase so we can export a wrapper to
|
|
; BootRetry instead. The wrapper muxes the call based on if figment is active
|
|
Old_AfterBootSetApplBase ;<SM20> PN
|
|
else
|
|
AfterBootSetApplBase
|
|
endif
|
|
|
|
_InitApplZone ;<SM20> PN
|
|
jsr SetApplBase ;<SM20> PN
|
|
_InitApplZone ;<SM20> PN
|
|
rts ;<SM20> PN
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; Procedure InitApplZone;
|
|
;
|
|
; Common code for initializing the Application Zone. Requires SPBot,
|
|
; sysZone to be set up. Sets up ApplLimit, applZone. Invokes
|
|
; RsrcZoneInit, if needed. Sets up HiHeapMark := ApplLimit, as always, <C765><C778>
|
|
; but uses the more conservative result (D1) from ToMaxLimit rather than <C765><C778>
|
|
; the BufPtr-based result (D2) which was historical. <C778>
|
|
;
|
|
; New wrinkle -- start ApplZone up high, then let it grow downwards. <14Apr85>
|
|
; Stuff error codes in IAZ (for SetApplBase) and InitApplZone. <25Apr85>
|
|
; Init ApplZone and OrgApplZone before InitZone, to simplify MakeMoreMasters <27Apr85>
|
|
; Roll all of InitApplZone into one blob, leaving IAZ for history. <C681>
|
|
;
|
|
; Arguments:
|
|
; None.
|
|
;
|
|
; Result:
|
|
; D0: ec: error code.
|
|
; <0: InitZone failed.
|
|
; 0: Success.
|
|
;
|
|
; Registers:
|
|
; D1: appLim: Computed ApplLimit value.
|
|
; D2: lTemp: Proposed appl Limit value.
|
|
; A0: start: Points to proposed applZone start.
|
|
; A0: args: Points to InitZone argument list.
|
|
; A1: end: Points to proposed applZone end.
|
|
; A6: z: Points to sysZone, eventually applZone.
|
|
;
|
|
|
|
|
|
InitApplZone
|
|
Move.L #JInitApplZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
vInitApplZone
|
|
|
|
MoveA.L jIAZInit,A0 ; get pre-processing routine vector <C681>
|
|
JSR (A0) ; do it <C681>
|
|
BSR.S IAZ ; get the real zone <C681>
|
|
MoveA.L jIAZPostInit,A0 ; get post-processing routine vector <C681>
|
|
JSR (A0) ; do it <C681>
|
|
|
|
MoveQ #0,D0 ; re-stuff the no-error code <16Apr85>
|
|
Move.W D0,MemErr ; Record it globally, too <25Apr85>
|
|
RTS
|
|
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; IAZ <C681>
|
|
; - Do the real work of initing the appl zone.
|
|
; Major change to use stack&BufPtr-based ApplLimit and HiHeapMark, not
|
|
; just to hammer BufPtr-MinStack, as has been done since the early days.
|
|
; This fixes problems at start time, when SetApplBase is called with the
|
|
; stack in mid-ram and StackSpace is forced to say there is no
|
|
; Registers: D0-D2/A0-A1
|
|
; Called by InitApplZone and SetApplBase.
|
|
;----------------------------------------------------------------------
|
|
IAZ Move.L A6,-(SP) ;Save A6.
|
|
|
|
_RsrcZoneInit ;initialize resources in appl heap
|
|
Clr.L applZone ;No applZone exists during init.
|
|
|
|
; Do the Launch-type computation of ApplLimit, to give ample stack <C817>
|
|
; at boot time. Remove use of antiquated ToMaxLimit. <C817>
|
|
Move.l SP,D1 ;current stack <C817>
|
|
Sub.L DefltStack,D1 ;enough space for even QD <C817>
|
|
Move.L D1,ApplLimit ;Use stack limit, NOT BufPtrÕs <C778>
|
|
Move.L D1,HiHeapMark ;Start HiHeapMark in sync with <C778>
|
|
;ApplLimit in case of glue MaxApplZone<C778>
|
|
|
|
Move.L sysZone,A6 ;SysZone pointer.
|
|
Move.L BkLim(A6),A0 ;Current Last Block in sysZone.
|
|
|
|
TST.B Zone32Flag(A6) ;is it 32 bit zone <v1.2>
|
|
BNE.S @1 ;branch if yes <v1.2>
|
|
|
|
; Ordinarily start ApplZone just after SysZone, but if GrowDown is true, <14Apr85>
|
|
; and there's room, start up offset by CacheSize. <27Apr85>
|
|
LEA MinFree24(A0),A0 ;Points to first of cur. applZone. <v1.2>
|
|
BRA.S @5 ; <v1.2>
|
|
@1
|
|
LEA MinFree32(A0),A0 ;Points to first of cur. applZone. <v1.2>
|
|
|
|
@5
|
|
; Now proceed building the zone... <14Apr85>
|
|
Move.L A0,A1 ;New Application zone base.
|
|
Add.W #AppZoneSize,A1 ;Compute new end address.
|
|
Cmp.L D1,A1 ;New End Address < ApplLimit?
|
|
BCS.S @10 ;Yes, use new end address.
|
|
|
|
Move.L D1,A1 ;use ApplLimit as end address.
|
|
|
|
@10
|
|
; Move.L A0,StartPtr(SP) ;New Applic Zone Base.
|
|
; Move.L A1,LimitPtr(SP) ;New Applic Zone End.
|
|
; Move.W #<2*dfltMasters>,CMoreMasters(SP)
|
|
; Move.L #0,PGrowZone(SP) ;No Grow Zone procedure.
|
|
|
|
Move.L #0,-(SP) ;No Grow Zone procedure.
|
|
Move.W #(2*dfltMasters),-(SP) ;Number of masters to alloc.
|
|
Move.L A1,-(SP) ;New Applic Zone End.
|
|
Move.L A0,-(SP) ;New Applic Zone Base.
|
|
|
|
Move.L A0,applZone ;Save as the application heap zone.
|
|
|
|
Move.L SP,A0 ;Point to argument block.
|
|
_InitZone ;Ask OS to do request.
|
|
|
|
Add.W #14,SP ;Restore stack pointer.
|
|
|
|
Move.L theZone,A6 ;Application heap zone.
|
|
Move.L BkLim(A6),HeapEnd ;Set up HeapEnd.
|
|
|
|
MoveQ #0,D0 ;Success result code.
|
|
Move.W D0,MemErr ;Record it globally, too <25Apr85>
|
|
|
|
Move.L (SP)+,A6 ;Restore A6.
|
|
RTS ; <C681>
|
|
;------------------------------------------------------------------- <C681>
|
|
; End of historical IAZ.
|
|
;------------------------------------------------------------------- <C681>
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; vIAZInit new <C681>
|
|
; - Inits before ApplZone is created.
|
|
; Registers: D0-D2/A0-A1
|
|
; Called by InitApplZone via jIAZInit vector.
|
|
;----------------------------------------------------------------------
|
|
vIAZInit
|
|
MOVE.L IAZNotify,D0 ; anyone to notify?
|
|
BLE.S @0 ; br if not
|
|
MOVE.L D0,A0 ; give them a call if so
|
|
JSR (A0)
|
|
|
|
@0 MOVE.W #-1,WWExist ; $FF for non-existence (also sets QDExist)
|
|
ST FMExist ; $FF for InitFonts hasnÕt been called
|
|
; BSET #7,DSWndUpdate ; cancel pending PaintBehind
|
|
CLR.L MenuList
|
|
; Clr.L LastFOND ; <06May85>
|
|
; Move.W #-1,FONDID ; <06May85>
|
|
Move.L MinusOne,LastSPExtra ; flag to inval cache <13Jan86 JTC>
|
|
ST SEvtEnb ; re-enable system event
|
|
|
|
MoveQ #(IOVQElSize/2)-1,D0 ; <10May85>
|
|
@99
|
|
Clr.W -(SP)
|
|
DBrA D0,@99
|
|
MoveA.L SP,A0
|
|
|
|
MOVE.L VCBQHdr+QHead,D0
|
|
@2 BEQ.S @4
|
|
MOVE.L D0,A1
|
|
MOVE.W VCBDrvNum(A1),IOVDrvNum(A0)
|
|
BEQ.S @3
|
|
_FlushVol ; flush all on-line volumes
|
|
@3 MOVE.L QLink(A1),D0
|
|
BRA.S @2
|
|
@4
|
|
ADD #IOVQElSize,SP ; clean up stack
|
|
|
|
; Init code moved from Launch
|
|
CLR.L DragHook ;No drag hook yet
|
|
CLR.L DeskHook ;No desk hook for hit-testing desk.
|
|
LEA CloseOrnHook, A0 ; Point to closeOrnHook
|
|
CLR.L (A0)+ ;clear closeOrnHook
|
|
CLR.L (A0)+ ;clear RestProc
|
|
CLR.L (A0)+ ;clear saveProc
|
|
CLR.W TaskLock ;clear taskLock, fScaleDisable.
|
|
CLR.L ResErrProc ;and resource error proc.
|
|
CLR.L EjectNotify ;moved here from InitApplZone (from patches) <24Apr85>
|
|
|
|
IF hasASC THEN ; <C897><C914><1.7>
|
|
MOVE.L jSoundDead,A0 ;get vector to kill all current sounds in ... <C681>
|
|
JSR (A0) ;...the application heap <C681>
|
|
ENDIF
|
|
|
|
BSR.L FlushApplVbls ;kill off doomed vbl tasks <27Mar85>, <SM30 BT>
|
|
BigJSR FlushApplNM,A0 ; And bogus NM requests <V1.1><1.2>
|
|
|
|
BSR.S FlushFontCaches ; flush TrueType caches <sm10>stb
|
|
BSR.S FlushApplPorts ;kill off doomed grafPorts <C699>
|
|
BSR.S FlushPalettes ;as well as doomed palettes <10Jun87 EHB>
|
|
|
|
; this cleanup is needed, for instance, if an ES is done from MacsBug. <10Jun87 EHB>
|
|
|
|
MOVE.L MainDevice,A0 ; get the main device <10Jun87 EHB>
|
|
MOVE.L A0,SrcDevice ; set the src device <10Jun87 EHB>
|
|
MOVE.L A0,theGDevice ; and the current device <10Jun87 EHB>
|
|
MOVE #$2000,SR ; make sure interrupts enabled <10Jun87 EHB>
|
|
RTS
|
|
|
|
FlushFontCaches
|
|
;----------------------------------------------------------------------
|
|
; Remove any application memory allocated by TrueType whenever the application heap is
|
|
; reformatted. SetAppBase and InitApplZone are both patched since they both trash the
|
|
; application heap.
|
|
; Rolled in InitApplZoneCacheCleaner from BassPatches.a <SM8> <PN><sm10>stb
|
|
; Registers: saves all registers, trashes condition codes
|
|
;----------------------------------------------------------------------
|
|
|
|
MOVEM.L A0-A2/D0-D3,-(sp) ; save registers <SM8> <PN>
|
|
CLR.L -(SP) ; room for result <SM8> <PN>
|
|
MOVE.L applZone,A0 ; point to zone hdr <SM8> <PN>
|
|
MOVE.L A0,-(SP) ; push start ptr <SM8> <PN>
|
|
MOVE.L (A0),-(SP) ; push zone trailer <SM8> <PN>
|
|
_sbKillSomeCaches ; <SM8> <PN>
|
|
ADDQ #4,SP ; ignore result <SM8> <PN>
|
|
MOVEM.L (sp)+,A0-A2/D0-D3 ; restore registers <SM8> <PN>
|
|
@SkipIt
|
|
RTS
|
|
|
|
FlushApplPorts
|
|
;----------------------------------------------------------------------
|
|
; FlushApplPorts new <C681>
|
|
; -- clean up portList just before ApplZone is eliminated
|
|
; Registers: D0-D2/A0-A1
|
|
; Called by vIAZInit.
|
|
;----------------------------------------------------------------------
|
|
; Remove from the PortList any ports in the application heap zone
|
|
|
|
@REPEAT MOVE.L portList,A1 ;get the portList <C699>
|
|
CMP.L MinusOne,A1 ;are they equal? <C699>
|
|
BEQ.S @DONE ;=>if so,invalid portlist, just return <C699>
|
|
MOVE.L (A1),A1 ;point to it <C699>
|
|
MOVE.L A1,A0 ;save pointer in A0 <C699>
|
|
MOVE (A1)+,D1 ;get count of elements? <C699>
|
|
BRA.S @NEXT ;=> and dive into loop <C699>
|
|
|
|
@SEARCH MOVE.L (A1)+,D0 ;get next element from list <C699>
|
|
BSR.L AppZoneAddr ;in application area (or zero)? <C699>,<SM30 BT>
|
|
BEQ.S @DELETE ;=>yes, delete it <C699>
|
|
@NEXT DBRA D1,@SEARCH ;=>else try next in list <C699>
|
|
BRA.S @DONE ;=>none left, continue <C699>
|
|
|
|
@DELETE SUB #1,(A0) ;decrement port count <C699>
|
|
MOVE.L -(A1),-(SP) ;stk: port < etc. <C699>
|
|
CLR.L -(SP) ;stk: result < port < etc. <C699>
|
|
MOVE.L portList,-(SP) ;stk: plist < res < port < etc. <C699>
|
|
MOVEQ #2,D0 ;search after length <C699>
|
|
MOVE.L D0,-(SP) ;stk: #2 < plist < res < port < etc. <C699>
|
|
PEA 12(SP) ;PTR1 = GrafPtr <C699>
|
|
MOVEQ #4,D0 ;LEN1 = 4 bytes <C699>
|
|
MOVE.L D0,-(SP) ;stk: #4 < ptr1 < #2 < plist < res < port < <C699>
|
|
PEA @DONE ;PTR2 = XXX <C699>
|
|
CLR.L -(SP) ;LEN2 = 0 <C699>
|
|
_Munger ;rhymes with plunger <C699>
|
|
ADDQ #8,SP ;strip result, grafPort <C699>
|
|
BRA.S @REPEAT ;=>repeat until no more <C699>
|
|
@DONE RTS
|
|
|
|
|
|
FlushPalettes ; patch rolled in <10Jun87 EHB>
|
|
;-----------------------------------------------------------------------
|
|
; Call DisposePalette for all palettes in the app heap.
|
|
;
|
|
; Registers: D0-D2/A0-A1
|
|
; Called by vIAZInit.
|
|
;----------------------------------------------------------------------
|
|
MOVEM.L A2-A3/D3,-(SP) ; save work registers
|
|
MOVE.L PMgrHandle,A2 ; get paletteMgr handle
|
|
CMP.L MinusOne,A2 ; is it there?
|
|
BEQ.S @DONE ; => 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
|
|
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 D1,D0 ; and get for routine
|
|
BSR.L AppZoneAddr ; in application area (or zero)? <SM30 BT>
|
|
BNE.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
|
|
|
|
@DONE MOVEM.L (SP)+,A2-A3/D3 ; restore work registers
|
|
RTS
|
|
|
|
;----------------------------------------------------------------------
|
|
; vIAZPostInit new <C681>
|
|
; - last-minute inits after ApplZone has been created.
|
|
; Registers: D0-D2/A0-A1
|
|
; Called by InitApplZone via jIAZPostInit vector.
|
|
;----------------------------------------------------------------------
|
|
vIAZPostInit
|
|
_InitAllPacks ; reinstall the packages -- may trash MemErr <25Apr85>
|
|
|
|
CLR.W CurApRefNum ; since all app heap res files closed <C681>
|
|
_RDrvrInstall ; fix up ram based drivers <C681>
|
|
|
|
; clear the pack6 resource cache when launching (for Int'l Utilities / Script Mgr)
|
|
; moved here from InitApplZone because it seems to make more sense. <sm10>stb
|
|
with ExpandMemRec
|
|
move.l expandMem,a0 ; expand pointer
|
|
add.w #emItlCache,a0 ; point to cache
|
|
move.l #emItlCacheSize/4-1,d0 ; dbra ptr
|
|
@ClearCache
|
|
clr.l (a0)+ ; clear part of cache
|
|
dbra d0,@ClearCache ; until done
|
|
endWith
|
|
|
|
RtS
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; InitZone(startPtr, limitPtr: Ptr; mAllocCnt: INTEGER;
|
|
; growZone: ProcPtr);
|
|
; - Initialize a heap zone, and set theZone to that zone.
|
|
; Signal noErr globally, nowadays. <25Apr85>
|
|
; Set HiHeapMark if new heap end is below SP. <01Nov85 JTC>
|
|
; startPtr - start of heap zone
|
|
; limitPtr - end of heap zone
|
|
; mAllocCnt - number of master pointers to allocate initially
|
|
; and when more are needed.
|
|
; growZone - user supplied procedure to grow zone (must be
|
|
; supplied):
|
|
; FUNCTION growZone(bytesNeeded: LongInt): LongInt;
|
|
; (returns the number of bytes zone was grown by)
|
|
; <29jan85> User growZone in GZProc, StdGZ salted in spare1
|
|
;
|
|
;----------------------------------------------------------------------
|
|
|
|
InitZoneTrap
|
|
Move.L #jInitZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
|
|
v24InitZone
|
|
MoveM.L A2-A3/A6,-(SP) ;Save registers
|
|
|
|
Move.L A0,A1 ;Argument vector pointer
|
|
Move.L (A1)+,A6 ;StartPtr (zone obj.)
|
|
SetZ A6 ;set theZone
|
|
|
|
Move.L A6,A0 ;zone object ptr
|
|
MOVE.L A6,A2 ;used for filling in fields
|
|
MoveQ #(HeapData-4)/4,D1
|
|
|
|
@IZClr24 Clr.L (A0)+ ;Zero heapzone header
|
|
DBrA D1,@IZClr24 ;leave A0=HeapData(A6)
|
|
|
|
; note that AllocPtr is zeroed . . .
|
|
; Set up hi heap mark according to LimitPtr. <01Nov85 JTC>
|
|
Move.L (A1)+,D1 ;LimitPtr, to be stripped of high bits <01Nov85 JTC>
|
|
And.L Lo3Bytes,D1 ;24 bits only <01Nov85 JTC>
|
|
|
|
;---------------------------------------------------------------------- <T6> <2> kc.start
|
|
; Align the trailer block on 040 cpus so that when a MoveHHi call is made the block will
|
|
; be line aligned.
|
|
|
|
Align24Trailer
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blo.s @0 ; No? Then long align.
|
|
Tst.l pVMGlobals ; VM running? <T7>
|
|
Bpl.s @0 ; Yea? Then don't patch.<T7>
|
|
|
|
Subq.l #16-MinFree24,D1
|
|
AndI.w #$FFF0,D1
|
|
Add.l #16-MinFree24,D1
|
|
MoveA.L D1,A3 ; LimitPtr
|
|
bra.s @05
|
|
|
|
@0 AndI.W #$FFFC,D1 ; LimitPtr = 4x
|
|
MoveA.L D1,A3 ; LimitPtr
|
|
@05 ; <2> kc.end
|
|
|
|
CmpA.L HiHeapMark,A3 ;24-bit subtract A3-HiHeapMark <01Nov85 JTC>
|
|
BLT.S @1 ;Less Than (unsigned OK) ==> below HiHeap <01Nov85 JTC>
|
|
Move.L SP,D1 ;Must strip to compare successfully! <01Nov85 JTC>
|
|
And.L Lo3Bytes,D1 ;24-bit sp <01Nov85 JTC>
|
|
CmpA.L D1,A3 ;LimitPtr-SP <01Nov85 JTC>
|
|
BGT.S @1 ;Greater Than (signed OK) ==> in stack <01Nov85 JTC>
|
|
Move.L A3,HiHeapMark ;a new 24-bit limit! <01Nov85 JTC>
|
|
@1 ;bypass point for setting HiHeapMark <01Nov85 JTC>
|
|
|
|
MoveQ #MinFree24,D1 ;MinFree=16 bytes <v1.2>
|
|
Sub.L D1,A3 ;LimitPtr-MinFree
|
|
Move.L D1,TagBC24(A3) ;make limit block <v1.1>
|
|
|
|
Move.L A3,(A2)+ ;BkLim(A6)
|
|
Move.L A0,(A2)+ ;PurgePtr(A6) - rover starts at
|
|
; HeapData(A6)
|
|
AddQ #4,A2 ;skip HFstFree
|
|
|
|
Sub.L A0,A3 ;BkLim(A6) - HeapData(A6)
|
|
Move.L A3,D0
|
|
Move.L D0,(A2)+ ;ZCBFree(A6)
|
|
|
|
Move.W (A1)+,D1 ;CMoreMasters
|
|
Move.L (A1)+,(A2)+ ;PGrowZone -> GZProc(A6)
|
|
|
|
; <29jan85> Stuff system growZone into spare1. Use fact that A1 is free; A2 = mAllocCnt(A6)
|
|
LEA StdGZ,A1 ;system GZProc <29jan85>
|
|
Move.L A1,(spare1-mAllocCnt)(A2)
|
|
|
|
Move.W D1,(A2)+ ;CMoreMasters -> mAllocCnt(A6)
|
|
Move.W mmDefFlags,(A2)+ ;Flags(A6)
|
|
|
|
Move.B #Is24Zone,Zone32Flag(A6); set zone to be 24 bit <v1.1>
|
|
Move.L #JMemMgr24,ZoneJumpV(A6); set jump vector in the zone <v1.1>
|
|
|
|
BSR a24MakeBKF ;start with 1 large free block <v1.1>
|
|
;A0=blk ptr, D0=size
|
|
BSR a24HMakeMoreMasters ;HFstFree(A6)=0 <v1.1>
|
|
|
|
; After first block of master pointers is allocated, check that zone is line aligned on 040s. <2> kc.start
|
|
; If not, pad the mp block out by two bytes, diminishing the free block following and the
|
|
; the global free count. This ensures that all subsequent allocations will be aligned.
|
|
; Be sure to update the allocation rover, which may be pointing to the big free block.
|
|
|
|
Init24ZoneAlign
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blt.s @0 ; No? Then long align.
|
|
Tst.l pVMGlobals ; VM running? <T7>
|
|
Bpl.s @0 ; Yea? Then don't patch.<T7>
|
|
|
|
LEA heapData(A6),A0 ; A0 -> mp block
|
|
Move.l A0,D0 ;
|
|
Add.l #BlkData24,D0
|
|
Move.L (A0),D1 ; D0 := length of mp block
|
|
And.L Lo3Bytes,D1 ;
|
|
Add.l D1,D0
|
|
And.l #$F,D0 ; Eq => multiple of 16, ok
|
|
BEq.S @postMPAlign ;
|
|
|
|
Not.b D0 ; Get remainder.
|
|
And.b #$F,D0 ; Only care about bottom 4 bits.
|
|
Addq.b #1,D0 ; Add in 1.
|
|
|
|
Sub.L D0,ZCBFree(A6) ; taking up xx free bytes for pad
|
|
MoveA.L A0,A1 ;
|
|
AddA.L D1,A1 ; A1 -> free block after mp
|
|
Add.B D0,(A0) ; increment slop count
|
|
Add.L D0,(A0) ; increment block size
|
|
Sub.L D0,(A1) ; decrement free block size
|
|
Move.L (A1),0(A1,D0) ; move relevant part of free block header
|
|
Add.L D0,A1 ; A1 -> new free block
|
|
Move.L A1,allocPtr(A6) ; be sure rover is valid
|
|
Bra.s @postMPAlign
|
|
|
|
@0 Move.L A6,D0 ;
|
|
BTst #1,D0 ; Eq => multiple of 4, ok
|
|
BEq.S @postMPAlign ;
|
|
|
|
SubQ.L #2,ZCBFree(A6) ; taking up 2 free bytes for pad
|
|
LEA heapData(A6),A0 ; A0 -> mp block
|
|
MoveA.L A0,A1 ;
|
|
Move.L (A0),D0 ; D0 := length of mp block
|
|
And.L Lo3Bytes,D0 ;
|
|
AddA.L D0,A1 ; A1 -> free block after mp
|
|
AddQ.B #2,(A0) ; increment slop count
|
|
AddQ.L #2,(A0) ; increment block size
|
|
SubQ.L #2,(A1) ; decrement free block size
|
|
Move.L (A1),2(A1) ; move relevant part of free block header
|
|
AddQ.L #2,A1 ; A1 -> new free block
|
|
Move.L A1,allocPtr(A6) ; be sure rover is valid <2> kc.end
|
|
@postMPAlign
|
|
|
|
; TrueType has a disk-based cache which puts font caches onto disk when they <sm10>stb
|
|
; get purged. This purgeProc will check to see whether a block which is being <sm10>stb
|
|
; purged is a font cache, and save it to disk if it is. <sm10>stb
|
|
lea DiskCachePurgeProcedure,A0 ; TrueType purge procedure <sm10>stb
|
|
move.l A0,purgeProc(A6) ; install purge procedure <sm10>stb
|
|
|
|
MoveM.L (SP)+,A2-A3/A6 ;Restore registers
|
|
MoveQ #0,D0 ;Success return <25Apr85>
|
|
Move.W D0,MemErr ;Record it globally, too <25Apr85>
|
|
RTS ;and return to caller
|
|
|
|
|
|
v32InitZone
|
|
MoveM.L A2-A3/A6,-(SP) ;Save registers
|
|
|
|
Move.L A0,A1 ;Argument vector pointer
|
|
Move.L (A1)+,A6 ;StartPtr (zone obj.)
|
|
SetZ A6 ;set theZone
|
|
|
|
Move.L A6,A0 ;zone object ptr
|
|
MOVE.L A6,A2 ;used for filling in fields
|
|
MoveQ #(HeapData-4)/4,D1
|
|
|
|
@IZClr32 Clr.L (A0)+ ; Zero heapzone header
|
|
DBrA D1,@IZClr32 ; leave A0=HeapData(A6)
|
|
|
|
; note that AllocPtr is zeroed . . .
|
|
; Set up hi heap mark according to LimitPtr. <01Nov85 JTC>
|
|
Move.L (A1)+,D1 ; LimitPtr, to be stripped of high bits <01Nov85 JTC>
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater? <2> kc.start
|
|
Blt.s @0 ; No? Then long align.
|
|
Subq.l #4,D1
|
|
AndI.w #$FFF0,D1
|
|
Addq.l #4,D1
|
|
MoveA.L D1,A3 ; LimitPtr
|
|
bra.s @05
|
|
|
|
@0 AndI.W #$FFFC,D1 ; LimitPtr = 4x
|
|
MoveA.L D1,A3 ; LimitPtr
|
|
@05 ; <2> kc.end
|
|
CmpA.L HiHeapMark,A3 ; ??-bit subtract A3-HiHeapMark <01Nov85 JTC>
|
|
BLT.S @1 ; Less Than (unsigned OK) ==> below HiHeap <01Nov85 JTC>
|
|
Move.L SP,D1 ; Must strip to compare successfully! <01Nov85 JTC>
|
|
CmpA.L D1,A3 ; LimitPtr-SP <01Nov85 JTC>
|
|
BGT.S @1 ; Greater Than (signed OK) ==> in stack <01Nov85 JTC>
|
|
Move.L A3,HiHeapMark ; new limit <01Nov85 JTC>
|
|
@1 ; bypass point for setting HiHeapMark <01Nov85 JTC>
|
|
|
|
MoveQ #MinFree32,D1 ; MinFree=16 bytes <v1.2>
|
|
Sub.L D1,A3 ; LimitPtr-MinFree
|
|
Clr.L TagBC32(A3) ; set limit block as free block <v1.1>
|
|
Move.L D1,blkSize32(A3) ;make limit block <v1.1>
|
|
|
|
Move.L A3,(A2)+ ; BkLim(A6)
|
|
Move.L A0,(A2)+ ; PurgePtr(A6) - rover starts at
|
|
; HeapData(A6)
|
|
AddQ #4,A2 ; skip HFstFree
|
|
|
|
Sub.L A0,A3 ; BkLim(A6) - HeapData(A6)
|
|
Move.L A3,D0
|
|
Move.L D0,(A2)+ ; ZCBFree(A6)
|
|
|
|
Move.W (A1)+,D1 ; CMoreMasters
|
|
Move.L (A1)+,(A2)+ ; PGrowZone -> GZProc(A6)
|
|
|
|
; <29jan85> Stuff system growZone into spare1. Use fact that A1 is free; A2 = mAllocCnt(A6)
|
|
LEA StdGZ,A1 ; system GZProc <29jan85>
|
|
Move.L A1,(spare1-mAllocCnt)(A2)
|
|
|
|
Move.W D1,(A2)+ ; CMoreMasters -> mAllocCnt(A6)
|
|
Move.W mmDefFlags,(A2)+ ; Flags(A6)
|
|
|
|
Move.B #Is32Zone,Zone32Flag(A6); set zone to be 32 bit <v1.1>
|
|
Move.L #JMemMgr32,ZoneJumpV(A6); set jump vector in the zone <v1.1>
|
|
|
|
BSR a32MakeBKF ; start with 1 large free block <v1.1>
|
|
; A0=blk ptr, D0=size
|
|
BSR a32HMakeMoreMasters ; HFstFree(A6)=0 <v1.1>
|
|
|
|
; After first block of master pointers is allocated, check that zone is longword aligned.
|
|
; If not, pad the mp block out by two bytes, diminishing the free block following and the
|
|
; the global free count. This ensures that all subsequent allocations will be aligned.
|
|
; Be sure to update the allocation rover, which may be pointing to the big free block.
|
|
;
|
|
; This attempt is feeble given the removal of alignment in routine ActualS in HeapGuts.a <A415>
|
|
; but the code is left in to ensure that at least the beginning of the heap is cool. <A415>
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater? <2> kc.start
|
|
Blt.s @not040 ; No? Then long align.
|
|
|
|
LEA heapData(A6),A0 ; A0 -> mp block
|
|
Move.l A0,D0 ;
|
|
Add.l #BlkData32,D0
|
|
Move.L BlkSize32(A0),D1 ; D0 := length of mp block
|
|
Add.l D1,D0
|
|
And.l #$F,D0 ; Eq => multiple of 16, ok
|
|
BEq.S @postMPAlign ;
|
|
|
|
Not.b D0 ; Get remainder.
|
|
And.b #$F,D0 ; Only care about bottom 4 bits.
|
|
Addq.b #1,D0 ; Add in 1.
|
|
|
|
Sub.L D0,ZCBFree(A6) ; taking up xx free bytes for pad
|
|
MoveA.L A0,A1 ;
|
|
AddA.L D1,A1 ; A1 -> free block after mp
|
|
Add.B D0,SizeCor32(A0) ; increment slop count
|
|
Add.L D0,BlkSize32(A0) ; increment block size
|
|
Sub.L D0,BlkSize32(A1) ; decrement free block size
|
|
Move.L BlkSize32(A1),BlkSize32(A1,D0) ; move relevant part of free block header
|
|
|
|
Add.L D0,A1 ; A1 -> new free block
|
|
Clr.l tagBC32(A1) ; Clear off old blocksize.
|
|
|
|
Move.L A1,allocPtr(A6) ; be sure rover is valid
|
|
Bra.s @postMPAlign
|
|
|
|
@not040 Move.L A6,D0 ;
|
|
BTst #1,D0 ; Eq => multiple of 4, ok
|
|
BEq.S @postMPAlign ;
|
|
|
|
SubQ.L #2,ZCBFree(A6) ; taking up 2 free bytes for pad
|
|
LEA heapData(A6),A0 ; A0 -> mp block
|
|
MoveA.L A0,A1 ;
|
|
Move.L BlkSize32(A0),D0 ; D0 := length of mp block
|
|
AddA.L D0,A1 ; A1 -> free block after mp
|
|
AddQ.B #2,SizeCor32(A0) ; increment slop count
|
|
AddQ.L #2,BlkSize32(A0) ; increment block size
|
|
SubQ.L #2,BlkSize32(A1) ; decrement free block size
|
|
Move.L BlkSize32(A1),2+BlkSize32(A1) ; move block size
|
|
Move.L (A1),2(A1) ; move relevant part of free block header
|
|
AddQ.L #2,A1 ; A1 -> new free block
|
|
Move.L A1,allocPtr(A6) ; be sure rover is valid
|
|
@postMPAlign ; <2> kc.end
|
|
|
|
; TrueType has a disk-based cache which puts font caches onto disk when they <sm10>stb
|
|
; get purged. This purgeProc will check to see whether a block which is being <sm10>stb
|
|
; purged is a font cache, and save it to disk if it is. <sm10>stb
|
|
lea DiskCachePurgeProcedure,A0 ; TrueType purge procedure <sm10>stb
|
|
move.l A0,purgeProc(A6) ; install purge procedure <sm10>stb
|
|
|
|
MoveM.L (SP)+,A2-A3/A6 ; Restore registers
|
|
IZCleanExit ; Common no-error exit <25Apr85>
|
|
MoveQ #0,D0 ; Success return <25Apr85>
|
|
Move.W D0,MemErr ;Record it globally, too <25Apr85>
|
|
RTS ;and return to caller
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION GetZone: Ptr;
|
|
;
|
|
; - Returns the global pointer to the current heap
|
|
; Uses A0 only.
|
|
;
|
|
|
|
GetZone
|
|
Move.L #JGetZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
|
|
vGetZone ; <v1.1>
|
|
GetZ A0 ;Set up return value
|
|
|
|
BrA.S IZCleanExit ;Use common routine above <25Apr85>
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
;PROCEDURE SetZone(Hz: Ptr);
|
|
;
|
|
; - Set global pointer to the current heap zone
|
|
; Uses A0 only.
|
|
;
|
|
|
|
SetZone
|
|
Move.L #JSetZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
|
|
vSetZone ; <v1.1>
|
|
SetZ A0 ;Set up return value
|
|
|
|
BrA.S IZCleanExit ;Use common routine above <25Apr85>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION CompactMem(cb: LongInt):[pblk: Ptr; cbFound: LongInt];
|
|
; FUNCTION MaxBlock :[pblk: Ptr; cbFound: LongInt];
|
|
;
|
|
; MaxBlock is modeled on CompactMem, but calls with #MaxMem as requested size
|
|
; (in order to find the largest available space) and with FNSelCompct=0 to
|
|
; prevent costly compaction.
|
|
;
|
|
MaxBlockTrap ;new routine <10Apr85>
|
|
PEA @Slime ;slimy solution for slimy problem <v1.9>
|
|
MOVE.L RealMemTop,D0 ;make an outlandish request <v1.9>
|
|
|
|
MOVE.L #JMaxBlock,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
|
|
@Slime ; <v1.5>
|
|
SUBQ.L #4,D0 ;reduce size by 4 byte to account for <v1.5>
|
|
;block starts and ends in non long word <v1.5>
|
|
;address, if 4 is too big, the extra 2 <v1.5>
|
|
;byte can act as margin. <v1.5>
|
|
BGE.S @Return ;if less than zero, then return 0 <v1.5>
|
|
MoveQ.L #0,D0 ;return 0 for length <v1.5>
|
|
@Return ; <v1.5>
|
|
RTS ; <v1.5>
|
|
|
|
CompactMemTrap
|
|
MOVE.L #JCompactMem,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24MaxBlock
|
|
MOVEQ #1,D1 ; mark routine as MaxBlock <v1.9>
|
|
BRA.S v24CMGuts ; branch to common code <v1.9>
|
|
|
|
v24CompactMem ;
|
|
MOVEQ #0,D1 ; mark as CompactMem <v1.9>
|
|
|
|
v24CMGuts
|
|
BSR a24ActualS ;adjust size <v1.1>
|
|
|
|
LEA Flags(A6),A1 ;useful address
|
|
Move.B (A1),D2 ;preserve current flags in D2
|
|
|
|
TST.B D1 ;nonzero means MaxBlock <10Apr85>
|
|
BNE.S @1 ;don't do compaction <10Apr85>
|
|
BSet #FNSelCompct,(A1) ;compact always for CompactMem <10Apr85>
|
|
BRA.S @3 ; <10Apr85>
|
|
@1 ; <10Apr85>
|
|
BClr #FNSelCompct,(A1) ;never compact for MaxBlock <10Apr85>
|
|
@3 ; <10Apr85>
|
|
|
|
Bsr.l a24CompactHp ;compact heap <2> kc.start
|
|
Move.B D2,(A1) ;restore flags
|
|
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blt.s @0 ; No? Then no alignment.
|
|
Tst.l pVMGlobals ; VM running?
|
|
Bpl.s @0 ; Yea? Then no alignment.
|
|
|
|
AndI.W #$FFF0,D1 ; Line align result.
|
|
|
|
@0 Move.L D1,D0 ;Size of largest block found <2> kc.end
|
|
BEq PMExit24 ;hasty exit through our epilogue <v1.1>
|
|
SubQ.L #BlkData24,D0 ;convert back to logical cb <v1.1>
|
|
|
|
Move.L A0,D1 ;was a large enough blk found?
|
|
BEq.S PMExit24 ;Eq => NIL, return that
|
|
BrA.S PMOK24
|
|
|
|
|
|
v32MaxBlock
|
|
MOVEQ #1,D1 ; mark routine as MaxBlock <v1.9>
|
|
BRA.S v32CMGuts ; branch to common code <v1.9>
|
|
|
|
v32CompactMem ;
|
|
MOVEQ #0,D1 ; mark as CompactMem <v1.9>
|
|
|
|
v32CMGuts ;entry point from MaxBlock <10Apr85>
|
|
BSR a32ActualS ;adjust size <v1.1>
|
|
|
|
LEA Flags(A6),A1 ;useful address
|
|
Move.B (A1),D2 ;preserve current flags in D2
|
|
|
|
TST.B D1 ;nonzero means MaxBlock <10Apr85>
|
|
BNE.S @1 ;don't do compaction <10Apr85>
|
|
BSet #FNSelCompct,(A1) ;compact always for CompactMem <10Apr85>
|
|
BRA.S @3 ; <10Apr85>
|
|
@1 ; <10Apr85>
|
|
BClr #FNSelCompct,(A1) ;never compact for MaxBlock <10Apr85>
|
|
@3 ; <10Apr85>
|
|
Bsr.l a32CompactHp ;compact heap <2> kc.start
|
|
Move.B D2,(A1) ;restore flags
|
|
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blt.s @0 ; No? Then no alignment.
|
|
|
|
AndI.W #$FFF0,D1 ; Line align result.
|
|
|
|
@0 Move.L D1,D0 ;Size of largest block found <2> kc.end
|
|
BEq.S PMExit32 ;hasty exit through our epilogue <v1.1>
|
|
Sub.L #BlkData32,D0 ;convert back to logical cb <v1.1>
|
|
|
|
Move.L A0,D1 ;was a large enough blk found? <v1.1>
|
|
BEq.S PMExit32 ;Eq => NIL, return that <v1.1>
|
|
BrA.S PMOK32
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION PurgeMem(cb: LongInt): pblk: Ptr;
|
|
;
|
|
|
|
PurgeMem
|
|
MOVE.L #JPurgeMem,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24PurgeMem ; 24 bit version of Purgemem <v1.1>
|
|
BSR a24ActualS ; adjust size <v1.1>
|
|
BSR a24PurgeHeap ; Purge the Heap <v1.1>
|
|
PMRMExit24 Move.L A0,D0 ; was a large enough blk found?
|
|
BNE.S PMNNil24 ; Yes, return success <v1.1>
|
|
|
|
MoveQ #memFullErr,D0 ; No, not enough room in mem.
|
|
BrA.S PMExit24
|
|
PMNNil24
|
|
MoveQ #0,D0 ; and show success return
|
|
PMOk24
|
|
AddQ.L #BlkData24,A0 ; else, point to data field.
|
|
PMExit24
|
|
BRA.S PMExit32 ; go to common exit <v1.9>
|
|
|
|
v32PurgeMem ; 32 bit version of Purgemem <v1.1>
|
|
BSR a32ActualS ; adjust size <v1.1>
|
|
BSR a32PurgeHeap ; Purge the Heap <v1.1>
|
|
PMRMExit32 Move.L A0,D0 ; was a large enough blk found?
|
|
BNE.S PMNNil32 ; Yes, return success <v1.1>
|
|
|
|
MoveQ #memFullErr,D0 ; No, not enough room in mem.
|
|
BrA.S PMExit32 ; <v1.1>
|
|
PMNNil32
|
|
MoveQ #0,D0 ; and show success return
|
|
PMOk32
|
|
Add.L #BlkData32,A0 ; else, point to data field. <v1.1>
|
|
PMExit32
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION PurgeSpace: [totalPurgeAndFree, contigPurgeAndFree : LongInt]; <17Apr85>
|
|
; On exit:
|
|
; D0 - total number of purgeable and free bytes in zone
|
|
; A0 - maximum possible contiguous space after purge (that is, max total of free
|
|
; and purgeable by 'region'.
|
|
; This function scans only; it does not purge, compact, nor allocate!
|
|
;
|
|
|
|
PurgeSpaceTrap
|
|
MOVE.L #JPurgeSpace,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24PurgeSpace ; 24 bit version of PurgeSpace <v1.1>
|
|
BSR a24TotePurgeables ; returns D0,A0 trashes others... <17Apr85>
|
|
BRA.S PurgeSExit ; go to common exit routine <v1.9>
|
|
|
|
v32PurgeSpace ; 32 bit version of PurgeSpace <v1.1>
|
|
BSR a32TotePurgeables ; returns D0,A0 trashes others... <17Apr85>
|
|
|
|
PurgeSExit Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION FreeMem: Longint;
|
|
; On exit:
|
|
; D0 - Returns the number of free bytes in the zone.
|
|
|
|
FreeMem
|
|
MOVE.L #JFreeMem,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
vFreeMem ; <v1.1>
|
|
Move.L ZCBFree(A6),D0 ; set result
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE ResrvMem;
|
|
; On exit:
|
|
; A0 - ptr to free block made or found.
|
|
; D0 - 0 if large enough block was found, memfullerr if not.
|
|
;
|
|
; This routine is used to open a block of free space as low in the heap
|
|
; as possible; it should be called before NwHandle if that handle is to
|
|
; be locked permanently (i.e., when using handles as non-relocatables) to
|
|
; prevent fragmentation of the heap.
|
|
;
|
|
; Arguments:
|
|
; D0: size of block required (logical size).
|
|
;
|
|
|
|
; ÑÑÑÑÑÑÑÑÑ SuperMario roll in <SM9> tcn ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
; ResrvMem - make it grow System Heap, too.
|
|
; ResrvMem will not grow any heap other than ApplZone. This is a problem with the new, growable system heap.
|
|
; Process Mgr will grow the system heap for NewHandle calls. ResrvMem usually precedes a NewHandle,
|
|
; HLock sequence. If ResrvMem fails, the following NewHandle will cause the heap to grow (if it can).
|
|
; Then the block is allocated where the free space was made -- at the top.
|
|
;
|
|
; This proc not only causes the heap to grow, it also calls ResrvMem again to see how low the free block
|
|
; can be made.
|
|
;
|
|
; Notes:
|
|
; NewPtr calls ResrvMem again while doing the StdGZProc. We want it to call the user gzproc, so
|
|
; itÕs important that ResrvMem return an error when itÕs reentered. ThatÕs why we have a semaphore.
|
|
;
|
|
; See also: NewPtr
|
|
;
|
|
;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
|
|
ResrvMem
|
|
; -- roll in start --
|
|
move.l A1,-(SP) ; save A1
|
|
move.l ExpandMem,A1 ; is this a reentry?
|
|
tst.l ExpandMemRec.emResrvMemSemaphore(A1)
|
|
bne.s @leaveNowStillBusy ; yup, get out.
|
|
move.l #1,ExpandMemRec.emResrvMemSemaphore(a1) ; <PN> nope, set the semaphore
|
|
|
|
movem.l d0-d1,-(sp) ; save how much was asked for
|
|
jsr GetTheMem ; go get it
|
|
tst.w d0 ; test for os error
|
|
beq.s @leaveNowPopEm ; if zero, space was reserved and weÕre done
|
|
|
|
; move.l 4(sp),d1 ; restore the trap word our caller used <5>
|
|
btst #tSysOrCurZone,d1 ; is this a _Resrvmem ,SYS?
|
|
bne.s @yupItsTheSystemHeap
|
|
move.l sysZone,a0 ; is this the (growable) system zone?
|
|
cmp.l theZone,a0
|
|
bne.s @leaveNowPopEm ; no, get out and leave other heaps alone
|
|
|
|
@yupItsTheSystemHeap
|
|
move.l (sp),d0 ; restore the amount being requested <5>
|
|
_NewPtr ,SYS ; force open a slot, even though itÕs probably at the top of the heap
|
|
; because we just made the heap grow.
|
|
bne @leaveNowPopEm ; leave if we couldnÕt get any space
|
|
|
|
_DisposPtr ; ok, we got it. now free it
|
|
movem.l (sp),d0-d1 ; ask ResrvMem again for this much. It will do a better job now that
|
|
jsr GetTheMem ; we know we have enough free space.
|
|
; dispatcher sets the z flag based on the function result
|
|
@leaveNowPopEm
|
|
addq #8,sp
|
|
|
|
@leaveNow
|
|
move.l ExpandMem,A1
|
|
clr.l ExpandMemRec.emResrvMemSemaphore(A1)
|
|
move.l (SP)+,A1 ; restore A1
|
|
rts
|
|
|
|
@leaveNowStillBusy
|
|
move.l (SP)+,A1 ; restore A1
|
|
moveq #memFullErr,d0 ; failure
|
|
move.w d0,MemErr
|
|
rts
|
|
; -- roll in end --
|
|
|
|
GetTheMem
|
|
MOVE.L #JResrvMem,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24ResrvMem ; 24 bit version of PurgeSpace <v1.1>
|
|
TST.L HFstFree(A6) ;Any masters left?
|
|
BNE.S @1 ;br if so
|
|
Move.W mAllocCnt(A6),D2 ;Number of masters to make.
|
|
Ext.L D2
|
|
|
|
ASL.L #2,D2 ;Convert MP count to byte count.
|
|
AddQ.L #BlkData24,D2 ;And block header <v1.1>
|
|
Add.L D2,D0 ;Increase req count by ptr blk
|
|
|
|
@1 BSR a24MakePtrSpc ;Make room at the low end. <v1.1>
|
|
Move.L A2,A0 ;Ptr to block
|
|
BrA.S PMRMExit24 ;Do the purge mem exit trip . . . <v1.1>
|
|
|
|
v32ResrvMem ; 32 bit version of PurgeSpace <v1.1>
|
|
TST.L HFstFree(A6) ;Any masters left?
|
|
BNE.S @1 ;br if so
|
|
Move.W mAllocCnt(A6),D2 ;Number of masters to make.
|
|
Ext.L D2
|
|
ASL.L #2,D2 ;Convert MP count to byte count.
|
|
|
|
Add.L #BlkData32,D2 ;And block header <v1.1>
|
|
Add.L D2,D0 ;Increase req count by ptr blk
|
|
|
|
@1 BSR a32MakePtrSpc ;Make room at the low end. <v1.1>
|
|
Move.L A2,A0 ;Ptr to block
|
|
BrA.S PMRMExit32 ;Do the purge mem exit trip . . . <v1.1>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION MaxMem: LongInt;
|
|
; On exit:
|
|
; If request pertains to application heap:
|
|
; A0 - returns the size of memory available for the application heap to grow
|
|
; D0 - Returns the size of the largest available contiguous free space.
|
|
; If request pertains to any other heap heap:
|
|
; A0 - 0, since only application heap may grow
|
|
; D0 - Returns the size of the largest available contiguous free space.
|
|
;----------------------------------------------------------------------
|
|
|
|
MaxMemTrap
|
|
MOVE.L #JMaxMem,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24MaxMem ; 24 bit version of MaxMem <v1.1>
|
|
|
|
Move.L RealMemTop,D0 ;BkCompactS(MaxSize <v1.9>
|
|
|
|
MoveQ #0,D2 ; False <2> kc.start
|
|
Bsr.l a24BkCompactS
|
|
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blt.s @0 ; No? Then no alignment.
|
|
Tst.l pVMGlobals ; VM running?
|
|
Bpl.s @0 ; Yea? Then no alignment.
|
|
|
|
AndI.W #$FFF0,D1 ; Line align result.
|
|
|
|
@0 Move.L D1,D0 ;return adjusted max <2> kc.end
|
|
BEq.S @1 ;block size, unless 0
|
|
|
|
SubQ.L #BlkData24,D0 ;Allow for overhead
|
|
|
|
@1
|
|
Sub.L A0,A0 ;if not applZone => return 0 in A0
|
|
Move.L applZone,A1 ;If request not for
|
|
Cmp.L theZone,A1 ;applic heap...
|
|
BNE.S MMExit24 ;then return <v1.1>
|
|
|
|
; Now return the size of memory available for the application heap to grow
|
|
|
|
Move.L ApplLimit,D2 ;static applLimit
|
|
BSR a24MaxLimit ;don't get closer than MinStack <v1.1>
|
|
Sub.L HeapEnd,D1
|
|
Move.L D1,A0 ;return in A0
|
|
MMExit24
|
|
BRA.S MMExit32 ; jump to common exit <v1.9>
|
|
|
|
v32MaxMem ; 32 bit version of MaxMem <v1.1>
|
|
|
|
Move.L RealMemTop,D0 ;BkCompactS(MaxSize <v1.9>
|
|
|
|
MoveQ #0,D2 ; False <2> kc.start
|
|
Bsr.l a32BkCompactS
|
|
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater?
|
|
Blt.s @0 ; No? Then no alignment.
|
|
|
|
AndI.W #$FFF0,D1 ; Line align result.
|
|
|
|
@0 Move.L D1,D0 ;return adjusted max <2> kc.end
|
|
BEq.S @1 ;block size, unless 0
|
|
|
|
Sub.L #BlkData32,D0 ;Allow for overhead <v1.1>
|
|
|
|
@1
|
|
Sub.L A0,A0 ;if not applZone => return 0 in A0
|
|
Move.L applZone,A1 ;If request not for
|
|
Cmp.L theZone,A1 ;applic heap...
|
|
BNE.S MMExit32 ;then return <v1.1>
|
|
|
|
; Now return the size of memory available for the application heap to grow
|
|
|
|
Move.L ApplLimit,D2 ;static applLimit
|
|
BSR a32MaxLimit ;don't get closer than MinStack <v1.1>
|
|
Sub.L HeapEnd,D1
|
|
Move.L D1,A0 ;return in A0
|
|
|
|
MMExit32
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE SetGrowZone(GrowZone: Ptr);
|
|
;
|
|
; - sets the GrowZone procedure for the current heap zone to be
|
|
; that supplied by the user
|
|
; On entry, Registers:
|
|
; A0 - GrowZone procedures' address
|
|
;
|
|
|
|
SetGrowZone
|
|
Move.L #JSetGrowZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
vSetGrowZone
|
|
GetZ A1 ; A1 - theZone
|
|
Move.L A0,GZProc(A1) ; set user's GrowZone procedure
|
|
BrA.S SALOk
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE SetApplLimit(zonelimit: Long);
|
|
;
|
|
; - sets the limit of the application zone
|
|
; - <29jan85> Now checks it, too!
|
|
; Return error code globally now. <25Apr85>
|
|
; Mask A0 with Lo3Bytes before compare with HiHeapMark or BKLim. <01Nov85 JTC>
|
|
; Pin minimum allowed stack to DefltStack in low memory, since larger <C636 JTC>
|
|
; stacks are required on color machines are old-style apps are starving them. <C636 JTC>
|
|
; On entry, Registers:
|
|
; A0 - zone limit (in bytes)
|
|
;
|
|
|
|
SetApplLimit
|
|
Move.L #JSetApplLimit,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
v24SetApplLimit ; 24 bit version of setApplLimit <v1.1>
|
|
Move.L A0,D0 ; suggested new value <01Nov85 JTC>
|
|
And.L Lo3Bytes,D0 ; 24-bit address candidate for new limit <01Nov85 JTC>
|
|
MoveA.L D0,A0 ; presume this is lower value <01Nov85 JTC>
|
|
Move.L SP,D1 ; try SP-DefltStack <C636>
|
|
And.L Lo3Bytes,D1 ; 24-bit SP <C636>
|
|
Sub.L DefltStack,D1 ; masked(SP-DefltStack) <C636>
|
|
Cmp.L D0,D1 ; (SP-DefltStack) - <suggested> <C636>
|
|
BGT.S @1 ; D0 < D1 --> suggested value OK <C636>
|
|
MoveA.L D1,A0 ; D1 ² D0 --> use SP-DefltStack <C636>
|
|
@1 ; bypass using DefltStack pinning <C636>
|
|
Move.L A0,ApplLimit ; Set the application limit
|
|
; (don't worry, GrowZone checks it dynamically)
|
|
; the above claim is B.S. <29jan85>
|
|
Move.L A0,HiHeapMark ; New high! <01Nov85 JTC><C765><C778>
|
|
|
|
Move.L ApplZone,A1 ; point to the app zone <29jan85>
|
|
Move.L BKLim(A1),A1 ; point to limit block <29jan85>
|
|
MoveQ #MemFullErr,D0 ; presume the worst <29jan85>
|
|
Cmp.L A0,A1 ; BKLim - ApplZone <29jan85>
|
|
BHi.S SALNotOk ; Hi ==> already beyond limit <29jan85>
|
|
; LowOrSame ==> fall through Ok <29jan85>
|
|
SALOk
|
|
MoveQ #0,D0 ; return success
|
|
SALNotOk ; get here if memFull above <29jan85>
|
|
Move.W D0,MemErr ; stuff error globally <25Apr85>
|
|
RTS
|
|
|
|
v32SetApplLimit ; 32 bit version of setApplLimit <v1.1>
|
|
Move.L A0,D0 ; suggested new value <v1.1>
|
|
Move.L SP,D1 ; try SP-DefltStack <v1.1>
|
|
Sub.L DefltStack,D1 ; masked(SP-DefltStack) <C636>
|
|
Cmp.L D0,D1 ; (SP-DefltStack) - <suggested> <C636>
|
|
BGT.S @1 ; D0 < D1 --> suggested value OK <C636>
|
|
MoveA.L D1,A0 ; D1 ² D0 --> use SP-DefltStack <C636>
|
|
@1 ; bypass using DefltStack pinning <C636>
|
|
Move.L A0,ApplLimit ; Set the application limit
|
|
; (don't worry, GrowZone checks it dynamically)
|
|
; the above claim is B.S. <29jan85>
|
|
Move.L A0,HiHeapMark ; New high! <01Nov85 JTC><C765><C778>
|
|
|
|
Move.L ApplZone,A1 ; point to the app zone <29jan85>
|
|
Move.L BKLim(A1),A1 ; point to limit block <29jan85>
|
|
MoveQ #MemFullErr,D0 ; presume the worst <29jan85>
|
|
Cmp.L A0,A1 ; BKLim - ApplZone <29jan85>
|
|
BHi.S @SALNotOk ; Hi ==> already beyond limit <29jan85>
|
|
; LowOrSame ==> fall through Ok <29jan85>
|
|
MoveQ #0,D0 ; return success
|
|
@SALNotOk ; get here if memFull above <29jan85>
|
|
Move.W D0,MemErr ; stuff error globally <25Apr85>
|
|
RTS
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION StackSpace : LongInt; <09May85>
|
|
;
|
|
; - returns Delta between SP and HiHeapMark. <C587>
|
|
; Stack on entry is:
|
|
; ret to dispatch << A0-A1 << ret within dispatch << D1-D2/A2 << trash long << ret to app
|
|
; so there are 4 + 8 + 4 + 12 + 4 + 4 = 36 more bytes available than indicated by the current
|
|
; SP. Since bkLim(xxx) refers to the beginning of a 16-byte block (change from 12 byte < v1,1>), <v1.1>
|
|
; and HiHeapMark is bkLim of the highest zone to date, the correct arithmetic is <C587>
|
|
; StackSpace := ( SP - BKLim(xxx) ) - 16 + 36 = SP - bkLim(xxx) + 20 <v1.1>
|
|
; The high bits of ApplZone and HiHeapMark are guaranteed to be clean. <C587>
|
|
;
|
|
|
|
StackSpaceTrap
|
|
Move.L #JStackSpace,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
vStackSpace ; <v1.1>
|
|
MoveQ #20,D0 ;fudge factor <v1.1>
|
|
Add.L SP,D0 ; <09May85>
|
|
AndI.W #$FFFC,D0 ;round down to multiple of 4 <C251>
|
|
Sub.L HiHeapMark,D0 ; <C587>
|
|
Clr.W MemErr ;no problem here <09May85>
|
|
RtS
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE MaxApplZone;
|
|
;
|
|
; - lengthens the application zone out to ApplLimit, without extending the latter. <24Apr85>
|
|
; This routine depends on the values of bkLim(ApplZone), ApplLimit, and HeapEnd being
|
|
; free of extraneous high bits, since address arithmetic (comparable to that of ZoneAdjustEnd,
|
|
; MaxLimit, and GrowZone in HeapGuts) is performed. Also, ApplLimit is forced to be (unsigned)
|
|
; greater or equal to HeapEnd, so their difference may be interpreted as unsigned.
|
|
;
|
|
; Uses SALOk to stuff noErr code. <25Apr85>
|
|
; Take care to force HiHeapMark if it was less than current ApplLimit. <C731>
|
|
;
|
|
; On entry: none
|
|
; On exit: D0=0
|
|
; Trashes: A0-A1/D0-D2
|
|
;
|
|
|
|
MaxApplZone
|
|
Move.L #JMaxApplZone,-(SP) ; get offset onto stack <v1.9>
|
|
Move.L vMMNoPrologue,-(SP) ; get vector to MMnoPrologue<v1.9>
|
|
RTS
|
|
|
|
v24MaxApplZone ; 24 bit version of MaxApplZone <v1.1>
|
|
Move.L ApplLimit,D0 ; alleged end of the zone
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater? <2> kc.start
|
|
Blt.s @0 ; No? Then no alignment.
|
|
Tst.l pVMGlobals ; VM running?
|
|
Bpl.s @0 ; Yea? Then no alignment.
|
|
|
|
Andi.w #$FFF0,D0 ; Force line alignment.
|
|
Sub.l #BlkData24,D0 ; Header size.
|
|
Bra.s @1
|
|
|
|
@0 AndI.W #$FFFC,D0 ; round down to multiple of 16
|
|
@1 Move.L D0,D1 ; <2> kc.end
|
|
LEA HeapEnd,A1 ; we'll use and possibly set this value
|
|
Sub.L (A1),D0 ; amount of heap growth, if any; (this value is NONNEGATIVE) NOT!
|
|
BCS.S @MAZFini ; Carry Set => minFree > delta, no dice... <SM24>
|
|
MoveQ #minFree24,D2 ; beneath this epsilon we will not grow <v1.2>
|
|
Cmp.L D2,D0
|
|
BCS.S @MAZFini ; Carry Set => minFree > delta, no dice...
|
|
|
|
MoveA.L (A1),A0 ; current bkLim ptr is the new (large) free block
|
|
Move.L D0,(A0) ; force large free block there, freeing A0
|
|
Move.L D1,(A1) ; new heapEnd := bkLim, freeing A1
|
|
MoveA.L D1,A0 ; we need it as a ptr, in an A-register
|
|
CmpA.L HiHeapMark,A0 ; is this a new high for heap extents? <C731>
|
|
BLS.S @noNewHigh ; HiHeapMark ³ A0 => no change <C731>
|
|
Move.L A0,HiHeapMark ; HiHeapMark < A0 => reset HiHeapMark <C731>
|
|
@noNewHigh ; skip to here when HiHeapMark OK <C731>
|
|
MoveA.L ApplZone,A1 ; now adjust the zone header
|
|
Move.L A0,bkLim(A1)
|
|
Move.L D2,(A0) ; make it a minFree block
|
|
Add.L D0,ZCBFree(A1) ; adjust free block count
|
|
@MAZFini
|
|
BrA.S SALOk ; stuff the noErr code there <25Apr85>
|
|
|
|
|
|
v32MaxApplZone ; 32 bit version of MaxApplZone <v1.1>
|
|
Move.L ApplLimit,D0 ; alleged end of the zone
|
|
Cmpi.b #cpu68040,CPUFlag ; Are we on an 040 or greater? <2> kc.start
|
|
Blt.s @0 ; No? Then no alignment.
|
|
|
|
Andi.w #$FFF0,D0 ; Force line alignment.
|
|
Sub.l #BlkData32,D0 ; Header size.
|
|
Bra.s @1
|
|
|
|
@0 AndI.W #$FFFC,D0 ; round down to multiple of 16
|
|
@1 Move.L D0,D1 ; <2> kc.end
|
|
LEA HeapEnd,A1 ; we'll use and possibly set this value
|
|
Sub.L (A1),D0 ; amount of heap growth, if any; this value is NONNEGATIVE!!
|
|
BCS.S @MAZFini ; Carry Set => minFree > delta, no dice... <SM24>
|
|
MoveQ #minFree32,D2 ; beneath this epsilon we will not grow <v1.2>
|
|
Cmp.L D2,D0
|
|
BCS.S @MAZFini ; Carry Set => minFree > delta, no dice...
|
|
|
|
MoveA.L (A1),A0 ; current bkLim ptr is the new (large) free block
|
|
Clr.L TagBC32(A0) ; set as free block <v1.1>
|
|
Move.L D0,BlkSize32(A0); get block size <v1.1>
|
|
Move.L D1,(A1) ; new heapEnd := bkLim, freeing A1
|
|
MoveA.L D1,A0 ; we need it as a ptr, in an A-register
|
|
CmpA.L HiHeapMark,A0 ; is this a new high for heap extents? <C731>
|
|
BLS.S @noNewHigh ; HiHeapMark ³ A0 => no change <C731>
|
|
Move.L A0,HiHeapMark ; HiHeapMark < A0 => reset HiHeapMark <C731>
|
|
@noNewHigh ; skip to here when HiHeapMark OK <C731>
|
|
MoveA.L ApplZone,A1 ; now adjust the zone header
|
|
Move.L A0,bkLim(A1)
|
|
Clr.L tagBC32(A0) ; clear 1st long word in header block <v1.1>
|
|
Move.L D2,BlkSize32(A0); make it a minFree block <v1.1>
|
|
Add.L D0,ZCBFree(A1) ; adjust free block count
|
|
@MAZFini
|
|
BrA.S SALOk ; stuff the noErr code there <25Apr85>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION NewPtr(byteCount: LongInt): Ptr;
|
|
;
|
|
; On entry: D0 - size in bytes to allocate
|
|
; On exit : A0 - Returns a pointer to a block of (data) size byteCount.
|
|
;
|
|
;----------------------------------------------------------------------
|
|
|
|
; ÑÑÑÑÑÑÑÑÑÑÑSuperMario roll in <SM9> tcn ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
; NewPtr - make it grow System Heap and place the block down low, too.
|
|
; Now that the system heap can grow, all of a sudden NewPtr is dropping locked blocks at the top of
|
|
; the heap. This stems from the fact the routine which reserves the block (MkPtrSpace) only grows ApplZone.
|
|
; This proc calls NewPtr, then checks to see whether the heap grew. If it did, this throws away the
|
|
; newly allocated block and calls NewPtr again. This gives MkPtrSpace the chance to allocate space
|
|
; down low because now thereÕs room to put relocatable blocks (the space the heap growing gave us).
|
|
;
|
|
; See also: ResrvMemGrowSystemHeap
|
|
;
|
|
; I re-rolled this patch so it won't overload the cpu :-) <SM29> kc
|
|
;
|
|
;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
NewPtrTrap
|
|
MOVE.L #JNewPtr,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
; --- 24 bit version ----
|
|
v24NewPtr ; 24 bit version of NewPtr <v1.1>
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap?
|
|
BEQ.S NPSysZone24 ; yup
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword?
|
|
BEQ.S NPNotSysZone24 ; nope
|
|
NPSysZone24
|
|
MOVE.L bkLim(a6),-(sp) ; save the end of system zone
|
|
SUB.L A1,A1 ; Handle Nil => fixed block
|
|
BSR a24MakePtrSpc ; Make room at the low end. <v1.1>
|
|
BSR a24AllocBk ; Allocate the block <v1.1>
|
|
MOVE.L (sp)+,A1 ; Recover the old end of system zone
|
|
BEQ.S NPNoMem24 ; Br if out of mem
|
|
|
|
CMP.L bkLim(a6),A1 ; Did the system heap grow?
|
|
BEQ.S NPCheckClear24 ; No, carry on as normal
|
|
BSR a24FreeBk ; Yes, free the block and try again.
|
|
NPNotSysZone24
|
|
SUB.L A1,A1 ; Handle Nil => fixed block
|
|
BSR a24MakePtrSpc ; Make room at the low end. <v1.1>
|
|
BSR a24AllocBk ; Allocate the block <v1.1>
|
|
BEQ.S NPNoMem24 ; Br if out of mem
|
|
NPCheckClear24
|
|
BSR CheckClear ; Clear block if requested
|
|
NPOk24
|
|
MOVEQ #0,D0
|
|
NPExit24
|
|
BRA.S NPExit ; jump to common exit <v1.9>
|
|
NPNoMem24
|
|
MOVEQ #memFullErr,D0
|
|
BRA.S NPExit24
|
|
|
|
; --- 32 bit version ----
|
|
v32NewPtr ; 32 bit version of NewPtr <v1.1>
|
|
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap?
|
|
BEQ.S NPSysZone32 ; yup
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword?
|
|
BEQ.S NPNotSysZone32 ; nope
|
|
NPSysZone32
|
|
MOVE.L bkLim(a6),-(sp) ; save the end of system zone
|
|
SUB.L A1,A1 ; Handle Nil => fixed block
|
|
BSR a32MakePtrSpc ; Make room at the low end. <v1.1>
|
|
BSR a32AllocBk ; Allocate the block <v1.1>
|
|
MOVE.L (sp)+,A1 ; Recover the old end of system zone
|
|
BEQ.S NPNoMem ; Br if out of mem
|
|
|
|
CMP.L bkLim(a6),A1 ; Did the system heap grow?
|
|
BEQ.S NPCheckClear32 ; No, carry on as normal
|
|
BSR a32FreeBk ; Yes, free the block and try again.
|
|
NPNotSysZone32
|
|
SUB.L A1,A1 ; Handle Nil => fixed block
|
|
|
|
BSR a32MakePtrSpc ; Make room at the low end. <v1.1>
|
|
BSR a32AllocBk ; Allocate the block <v1.1>
|
|
BEQ.S NPNoMem ; Br if out of mem
|
|
NPCheckClear32
|
|
BSR CheckClear ; Clear block if requested
|
|
NPOk
|
|
MOVEQ #0,D0
|
|
NPExit
|
|
MOVE.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
NPNoMem
|
|
MOVEQ #memFullErr,D0
|
|
BrA.S NPExit
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE DisposePtr(p: Ptr);
|
|
;
|
|
; On entry: A0 - ptr to block to be disposed
|
|
; - Disposes of previousely allocated Ptr object -- unless ROZ <04Mar85>
|
|
;
|
|
;----------------------------------------------------------------------
|
|
|
|
DisposePtr
|
|
move.l a0,d0 ; be cool when a NIL ptr comes through <sm9>stb
|
|
bne.w @NonNIL ; not nil, so call the real thing <sm9>stb
|
|
move.w d0,MemErr ; return no error even though itÕs not a ptr stb
|
|
rts ; <sm9>stb
|
|
|
|
@NonNIL MOVE.L #jDisposePtr,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPPrologue,-(SP) ; get vector to MMpprologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24DisposePtr ; 24 bit version of DisposePtr <v1.1>
|
|
; Move.L (SP)+,A0 ; restore A0 <v1.1>
|
|
DisposeTail24
|
|
BTST #ROZ,flags(A6) ; skip if read-only <04Mar85>
|
|
BNE.S DisposeSkip ; <04Mar85>
|
|
|
|
BSR a24FreeBk ; <v1.1>
|
|
BrA.S NPOk
|
|
|
|
v32DisposePtr ; 32 bit version of DisposePtr <v1.1>
|
|
; Move.L (SP)+,A0 ; restore A0 <v1.1>
|
|
DisposeTail32
|
|
BTST #ROZ,flags(A6) ; skip if read-only <04Mar85>
|
|
BNE.S DisposeSkip ; <04Mar85>
|
|
|
|
BSR a32FreeBk ; <v1.1>
|
|
BrA.S NPOk
|
|
DisposeSkip
|
|
MOVEQ #memROZWarn,D0 ; deliver warning <04Mar85>
|
|
BRA.S NPExit ; <04Mar85>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION GetPtrSize(p: Ptr): Size;
|
|
;
|
|
; On entry: A0.L- ptr
|
|
; On exit : D0.L- size in bytes
|
|
; Returns the actual data size of a Ptr object.
|
|
|
|
GetPtrSizeTrap
|
|
MOVE.L #jGetPtrSize,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPPrologue,-(SP) ; get vector to MMpprologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24GetPtrSize ; 24 bit version of MaxMem <v1.1>
|
|
BSR a24GetSize ; Get its size <v1.1>
|
|
BrA.S NPExit ; Check Zone, Restore, Return
|
|
|
|
v32GetPtrSize ; 32 bit version of MaxMem <v1.1>
|
|
BSR a32GetSize ; Get its size <v1.1>
|
|
BrA.S NPExit ; Check Zone, Restore, Return
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; 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)
|
|
;
|
|
|
|
SetPtrSizeTrap
|
|
MOVE.L #jSetPtrSize,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPPrologue,-(SP) ; get vector to MMpprologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24SetPtrSize ; 24 bit version of MaxMem <v1.1>
|
|
BSR a24SetSize ; set its size <v1.1>
|
|
BrA.S NPExit
|
|
|
|
v32SetPtrSize ; 32 bit version of MaxMem <v1.1>
|
|
BSR a32SetSize ; set its size <v1.1>
|
|
BrA.S NPExit
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION PtrZone(p: Ptr): Zone;
|
|
;
|
|
; Returns the zone in which a memory block belongs.
|
|
;
|
|
; Argument:
|
|
; A0: p: pointer for block
|
|
;
|
|
; Result:
|
|
; D0: ec: error code
|
|
; A0: zone: zone in which block belongs
|
|
;
|
|
|
|
PtrZoneTrap
|
|
MOVE.L #jPtrZone,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPPrologue,-(SP) ; get vector to MMpprologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
vPtrZone ; 24/32 bit version of PtrZone <v1.1>
|
|
Move.L A6,A0
|
|
BrA.S NPOk
|
|
|
|
CheckClear
|
|
BTst #ClearBit,D1 ;clear the allocated memory?
|
|
BEq.S @2 ;br if not
|
|
Move.L A0,-(SP) ;preserve A0
|
|
AddQ.L #1,D0 ;increment allocation size
|
|
LSR.L #1,D0 ;get word size
|
|
|
|
@1 Clr.W (A0)+ ;clear it out
|
|
SubQ.L #1,D0
|
|
BGT.S @1
|
|
Move.L (SP)+,A0 ;restore A0
|
|
|
|
@2 RTS
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION NewEmptyHandle: Handle;
|
|
; On entry:
|
|
; no input assumptions
|
|
; On exit:
|
|
; A0 - handle
|
|
; D0 - error code
|
|
; - Allocate an empty master pointer in the zone pool.
|
|
; Registers:
|
|
|
|
NewEmptyHandleTrap
|
|
MOVE.L #JNewEmptyHandle,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24NewEmptyHandle ; 24 bit version of NewEmptyHandle <v1.1>
|
|
BSR a24NextMaster ; Get next master in A1 <v1.1>
|
|
BEQ.S NHNoMaster ; If NIL, exit as in NewHandle
|
|
BRA.S NewEHExit ; use common code <v1.1>
|
|
|
|
v32NewEmptyHandle ; 32 bit version of NewEmptyHandle <v1.1>
|
|
BSR a32NextMaster ; Get next master in A1 <v1.1>
|
|
BEQ.S NHNoMaster ; If NIL, exit as in NewHandle
|
|
NewEHExit
|
|
Move.L A1,A0 ; Get it to A0 for return
|
|
CLR.L (A0) ; Make it NIL
|
|
MoveQ #0,D0 ; Show success
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION NwHandle(cb: Size): Handle;
|
|
; On entry:
|
|
; D0 - size to allocate
|
|
; On exit:
|
|
; A0 - handle
|
|
; - Allocate a relocatable object with its master pointer in the
|
|
; zone pool.
|
|
; Registers:
|
|
|
|
NwHandle
|
|
MOVE.L #JNwHandle,-(SP) ; get offset to routine <v1.9>
|
|
MOVE.L vMMPrologue,-(SP) ; get vector to MMPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24NwHandle ; 24 bit version of NwHandle <v1.1>
|
|
; from SetRovingPointerBeforeNewHandle patch<SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap? <SM29> kc
|
|
BEQ.S @inSysZone ; yup <SM29> kc
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword? <SM29> kc
|
|
BEQ.S @notInSysZone ; nope <SM29> kc
|
|
@inSysZone ; <SM29> kc
|
|
CLR.L allocPtr(a6) ; clear the rover <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
BSR a24NextMaster ; Get next master <v1.1>
|
|
BNE.S @NHMaster ; If not NIL, go get block <v1.1>
|
|
|
|
BRA.S NHNoMaster ; if NIL, report error <v1.1>
|
|
@NHMaster
|
|
BSR a24AllocBk ; try to allocate block <v1.1>
|
|
BEq.S NHNoMem ; No luck, fail return
|
|
|
|
BRA.S HCheckHandle
|
|
|
|
|
|
v32NwHandle ; 32 bit version of NwHandle <v1.1>
|
|
; from SetRovingPointerBeforeNewHandle patch<SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap? <SM29> kc
|
|
BEQ.S @inSysZone ; yup <SM29> kc
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword? <SM29> kc
|
|
BEQ.S @notInSysZone ; nope <SM29> kc
|
|
@inSysZone ; <SM29> kc
|
|
CLR.L allocPtr(a6) ; clear the rover <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
BSR a32NextMaster ; Get next master <v1.1>
|
|
BNE.S NHMaster ; If not NIL, go get block <v1.1>
|
|
NHNoMaster
|
|
MoveQ #memFullErr,D0
|
|
BrA.S HNilResult
|
|
NHMaster
|
|
BSR a32AllocBk ; try to allocate block <v1.1>
|
|
BEq.S NHNoMem ; No luck, fail return
|
|
|
|
HCheckHandle BSR.S CheckClear ; Clear memory if requested
|
|
Move.L A1,A0
|
|
NHOk
|
|
MoveQ #0,D0
|
|
NHExit
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
NHNoMem
|
|
MoveQ #memFullErr,D0
|
|
BrA.S HReleaseMP
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE DsposeHandle(h: Handle);
|
|
; On entry:
|
|
; A0 - handle to block to be disposed -- unless ROZ <04Mar85>
|
|
; Reactivated even if ROZ to just clear (bogus) MP <10Apr85>
|
|
; Distinguish Nil and Empty cases from MMHPrologue. <21May85>
|
|
;----------------------------------------------------------------------
|
|
|
|
DsposeHandle
|
|
Move.L #JDsposeHandle,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24DsposeHandle ; 24 bit version of MaxMem <v1.1>
|
|
BNE.S DisposeTail24 ; Release block <v1.1>
|
|
Bra.S DsposeCommon ; branch to commond code <v1.1>
|
|
|
|
|
|
v32DsposeHandle ; 32 bit version of DsposeHandle <v1.1>
|
|
BNE.S DisposeTail32 ; Release block <v1.1>
|
|
DsposeCommon ; commond code <v1.1>
|
|
Move.L A1,D0 ; Masked handle: EQ => Nil passed <21May85>
|
|
BEq.S HNilResult ; No master ptr to release <21May85>
|
|
|
|
BTST #ROZ,flags(A6) ; skip if ROZ <10Apr85>
|
|
BNE.S HROZResult ; <10Apr85>
|
|
|
|
cmp.l ROMBase,d0 ; is it a ROM resource ? <SM23> PN
|
|
bgt.s Nothing ; if in ROM, leave it alone <SM23> PN
|
|
|
|
MoveQ #0,D0 ; Success result
|
|
HReleaseMP
|
|
BSR ReleaseMP ; Return MP to list
|
|
HNilResult
|
|
Sub.L A0,A0 ; Return NIL in A0 for NwHandle
|
|
BrA.S NHExit
|
|
HROZResult
|
|
CLR.L (A1) ; just clear the (bogus) MP <10Apr85>
|
|
Nothing BRA.S NHOk ; go to MMEpilogue, sans error <10Apr85>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; FUNCTION GetHandleSize(h: Handle): LongInt;
|
|
; On entry: A0 - handle
|
|
; On exit : D0 - size in bytes
|
|
; Returns the actual data size of a Handle object.
|
|
|
|
__GetHandleSize
|
|
Move.L #JGetHandleSize,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24GetHandleSize ; 24 bit version of GetHandleSize <v1.1>
|
|
BEq.S SHSNil
|
|
BSR a24GetSize ; <v1.1>
|
|
|
|
BrA.S SHSExit
|
|
|
|
|
|
v32GetHandleSize ; 32 bit version of GetHandleSize <v1.1>
|
|
BEq.S SHSNil
|
|
BSR a32GetSize ; <v1.1>
|
|
|
|
BrA.S SHSExit
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; 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
|
|
;
|
|
|
|
SetHandleSizeTrap
|
|
Move.L #JSetHandleSize,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24SetHandleSize ; 24 bit version of SetHandleSize <v1.1>
|
|
BEQ.S SHSNil
|
|
BSR a24SetSize ; Set size as specified in D0 <v1.1>
|
|
BrA.S SHSExit ; <v1.1>
|
|
|
|
|
|
v32SetHandleSize ; 32 bit version of SetHandleSize <v1.1>
|
|
BNE.S SHSOk
|
|
|
|
SHSNil
|
|
MoveQ #nilHandleErr,D0 ;return error code
|
|
BrA.S SHSExit
|
|
SHSOk
|
|
BSR a32SetSize ;Set size as specified in D0 <v1.1>
|
|
SHSExit
|
|
Move.L A1,A0 ;restore A0 for RecoverHandle
|
|
BrA.S NHExit
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION HandleZone(h: Handle): Zone;
|
|
;
|
|
; Returns the zone in which a memory block belongs.
|
|
;
|
|
; Argument:
|
|
; A0: h: handle for block
|
|
;
|
|
; Result:
|
|
; D0: ec: error code
|
|
; A0: zone: zone in which block belongs
|
|
;
|
|
|
|
HandleZoneTrap
|
|
Move.L #JHandleZone,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
vHandleZone ; 24/32 bit version of HandleZone <v1.1>
|
|
Move.L A6,A0
|
|
BrA.S NHOk
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
; FUNCTION RecoverHandle(ptr: Ptr): Handle;
|
|
; On entry: A0 - pointer
|
|
; On exit : A0 - handle
|
|
; Must preserve D0!!!!! (As per 64K ROM, that is.) <25Apr85>
|
|
; Given a ptr to an alleged handle data block, returns the handle. Because of
|
|
; 64K ROM permissiveness, no error can be returned in D0, just in MemErr.
|
|
; Uses own exit sequence to MMNoErrEpilogue and stuffs MemErr on its own. <25Apr85>
|
|
; To effect patch from Mac+, special MMRHPrologue (ÒRHÓ as in RecoverHandle) <A305>
|
|
; is used to ensure the the more reasonable of SysZone and ApplZone is used for A6
|
|
; when incoming TheZone is one of Sys or Appl.
|
|
; Put a bus error handler on the fetch of the relative handle, pinning to <C778>
|
|
; ROMBase, for safety. <C778>
|
|
|
|
RecoverHandleTrap
|
|
Move.L #JRecoverHandle,-(SP) ; get offset to routine onto stack <v1.9>
|
|
Move.L vMMRHPrologue,-(SP) ; get vector to MMRHPrologue <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24RecoverHandle ; 24 bit version of RecoverHandle <2>
|
|
MOVE.L Lo3Bytes,D2 ; mask to strip extraneous bits <2>
|
|
BRA.S vRecoverHandle ; <2>
|
|
|
|
v32RecoverHandle ; 32 bit version of RecoverHandle <2>
|
|
MOVEQ #-1,D2 ; no extraneous address bits in 32-bit mode <2>
|
|
|
|
vRecoverHandle
|
|
Move.L BusErrVct,A2 ; save current bus error handler <C778><1.7>
|
|
LEA RHBusErrHandler,A3 ; our RH version <C778>
|
|
Move.L A3,BusErrVct ; <C778><1.7>
|
|
|
|
Move.L -4(A0),A1 ; Handle, or ROMBase on bus error <C778>
|
|
ReturnFromBusErr ; <NK1><SM22> rb
|
|
Move.L A2,BusErrVct ;restore old one after danger <C778><1.7>
|
|
|
|
Add.L A6,A1
|
|
|
|
; The following code tests for bad mp, but can only set MemErr <25Apr85>
|
|
|
|
MOVE.L A0,D1 ; original pointer <2>
|
|
AND.L D2,D1 ; stripped to RAW address <2>
|
|
AND.L (A1),D2 ; master pointer value stripped to RAW address <2>
|
|
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).
|
|
; Also, it assumes that the ROZ is ALWAYS a 32-bit heap zone.
|
|
|
|
MOVEA.L A6,A3 ; save the previous zone start pointer <2>
|
|
MOVEA.L RomMapHndl,A6 ; get the handle to the ROM resource map <2>
|
|
MOVEA.L (A6),A2 ; and de-reference it <2>
|
|
SUBA.L -4(A2),A6 ; calculate heap start based on rel hdl in MP <2>
|
|
|
|
CMPA.L A3,A6 ; have we been here before? <2>
|
|
BNE.S v32RecoverHandle ; -> no, try again with the ROZ <2>
|
|
|
|
MoveQ #memBCErr,D2 ; block check failure <13Apr85>
|
|
RecoverExit
|
|
Move.W D2,MemErr ; stuff global error code (is anybody out there?)<25Apr85>
|
|
MoveA.L A1,A0 ; restore A0=handle <25Apr85>
|
|
|
|
Move.L vMMNoErrEpilogue,-(SP) ; get MMNoErrEpilogue vector <v1.9>
|
|
RTS
|
|
|
|
;----------------------------------------------------------------------
|
|
; RHBusErr -- bus error handler for RecoverHandle <NK1><SM22> rb, start
|
|
; A bus error in RecoverHandle causes ROMBase to be jammed for the relative handle. <NK1>
|
|
; Triggered by bus errors in Move.L -4(A0),A1 in RecoverHandle. <NK1>
|
|
; <NK1>
|
|
; Frame types 7 and B are handled. The PC is jammed with the instruction <NK1>
|
|
; following the bus error (it can't be incremented because the emulator <NK1>
|
|
; doesn't guarantee valid PC values) and A1 is jammed with ROMbase. The <NK1>
|
|
; frame is then mutated into a type 0 frame before RTE. This should work <NK1>
|
|
; with VM, NuKernel, and emulator environments. <NK1>
|
|
;
|
|
; Output: A1 = ROMBase <NK1>
|
|
; Regs: A3 <NK1>
|
|
;----------------------------------------------------------------------
|
|
RHBusErrHandler
|
|
movea.l ROMBase,a1 ; stuff fake Nil value into expected dest. reg <NK1>
|
|
movea.w (sp),a3 ; <NK1>
|
|
cmpi.w #$7008,6(SP) ; type 7 frame? <NK1>
|
|
beq.s RHBusErrType7 ; <NK1>
|
|
adda.w #(92-8),SP ; Leave space for type 0 frame <NK1>
|
|
bra.s RHBusErrCommon ; <NK1>
|
|
RHBusErrType7 ; <NK1>
|
|
adda.w #(60-8),SP ; Leave space for type 0 frame <NK1>
|
|
RHBusErrCommon ; <NK1>
|
|
clr.w 6(SP) ; Make it a type 0 frame <NK1>
|
|
move.w a3,(sp) ; Return SR <NK1>
|
|
lea ReturnFromBusErr,a3 ; <NK1>
|
|
move.l a3,2(SP) ; Return PC <NK1>
|
|
RTE ; <NK1><SM22> rb, end
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE EmptyHandle(h: Handle);
|
|
; - Deallocates data block pointed to by the given handle & zeroes the MP
|
|
; Note that EH will not deallocate the data block if it's a ROZ. <10Apr85>
|
|
; On entry, Registers:
|
|
; A0 - Handle
|
|
|
|
EmptyHandle
|
|
Move.L #JEmptyHandle,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24EmptyHandle ; 24 bit version of EmptyHandle <v1.1>
|
|
BNE.S @EHNotNIL
|
|
MoveQ #0,D0
|
|
BrA.S EHExit
|
|
@EHNotNIL
|
|
BSR a24EH ; Common code with ReAllocHandle <v1.1>
|
|
BrA.S EHExit
|
|
|
|
|
|
v32EmptyHandle ; 32 bit version of EmptyHandle <v1.1>
|
|
BNE.S EHNotNIL
|
|
MoveQ #0,D0
|
|
BrA.S EHExit
|
|
EHNotNIL
|
|
BSR a32EH ;Common code with ReAllocHandle
|
|
EHExit
|
|
BrA.S SHSExit
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE: ReAllocHandle
|
|
; Arguments: D0 (input) -- size to allocate (long)
|
|
; A0 (input) -- original handle
|
|
; D0 (output) -- result code (MemFullErr)
|
|
; A0 (output) -- original handle or nil if unsuccessful
|
|
; Function: Allocate a relocatable object given its original Handle.
|
|
; This is useful when a relocatable data block is purged: it can
|
|
; be retrieved without having to destroy the handle & having to
|
|
; update every copy of the handle.
|
|
;
|
|
; If the handle in A0 does not point to a purged block, the
|
|
; block is freed first.
|
|
;
|
|
; This sets up the GZ root semaphores for the grow zone proc
|
|
; so it doesn't mess with the handle in question.
|
|
;
|
|
; If ROZ and empty handle, just do it; if non-empty give warning or error <04Mar85>
|
|
; depending on whether no larger or growing. <04Mar85>
|
|
;
|
|
;----------------------------------------------------------------------
|
|
|
|
ReAllocHandleTrap
|
|
Move.L #JReAllocHandle,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24ReAllocHandle ; 24 bit version of ReAllocHandle <v1.1>
|
|
BEQ.S @RAHCont ; <SM29> kc
|
|
; from SetRovingPointerBeforeReallocHandle patch<SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap? <SM29> kc
|
|
BEQ.S @inSysZone ; yup <SM29> kc
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword? <SM29> kc
|
|
BEQ.S @notInSysZone ; nope <SM29> kc
|
|
@inSysZone ; <SM29> kc
|
|
CLR.L allocPtr(a6) ; clear the rover <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
BTST #ROZ,flags(A6) ; Bypass if ROZ <04Mar85>
|
|
BNE.S @RAHSkip ; <04Mar85>
|
|
|
|
BSR a24EH ; Empty the Handle for sure
|
|
BNE.S SHSExit ; EH failed, return error code
|
|
Move.L D2,D0 ; restore D0 (saved by MMHPrologue)
|
|
@RAHCont
|
|
Move.L A1,GZRootHnd ; record handle being reallocated
|
|
Clr.L GZRootPtr ; record lack of pointer
|
|
BSR a24AllocBk ; try to allocate block <v1.1>
|
|
BSR ClearGZStuff ; clean up gz safety stuff
|
|
Move.L A0,D2 ; see if alloc bk worked
|
|
BNE HCheckHandle ; If ok, take the NewHandle exit route.
|
|
MoveQ #memFullErr,D0 ; If we can't find the space, just report
|
|
Bra.S SHSExit ; the error, leaving handle alone . . .
|
|
@RAHSkip
|
|
BSR a24GetSize ; D0 := current data size <v1.1>
|
|
CMP.L D0,D2 ; requested - current ... unsigned <04Mar85>
|
|
BHI.S RAHError ; it's erroneous to grow <04Mar85>
|
|
MOVEQ #memROZWarn,D0 ; <04Mar85>
|
|
BRA.S SHSExit ; <04Mar85>
|
|
|
|
|
|
v32ReAllocHandle ; 32 bit version of ReAllocHandle <v1.1>
|
|
BEQ.S @RAHCont ; <SM29> kc
|
|
; from SetRovingPointerBeforeReallocHandle patch<SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle going in the System heap? <SM29> kc
|
|
BEQ.S @inSysZone ; yup <SM29> kc
|
|
BTST #tSysOrCurZone,d1 ; is the "sys" bit set in the trapword? <SM29> kc
|
|
BEQ.S @notInSysZone ; nope <SM29> kc
|
|
@inSysZone ; <SM29> kc
|
|
CLR.L allocPtr(a6) ; clear the rover <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
BTST #ROZ,flags(A6) ; Bypass if ROZ <04Mar85>
|
|
BNE.S RAHSkip ; <04Mar85>
|
|
|
|
BSR a32EH ; Empty the Handle for sure
|
|
BNE.S SHSExit ; EH failed, return error code
|
|
Move.L D2,D0 ; restore D0 (saved by MMHPrologue)
|
|
@RAHCont ; <SM29> kc
|
|
Move.L A1,GZRootHnd ; record handle being reallocated
|
|
Clr.L GZRootPtr ; record lack of pointer
|
|
BSR a32AllocBk ; try to allocate block <v1.1>
|
|
BSR ClearGZStuff ; clean up gz safety stuff
|
|
Move.L A0,D2 ; see if alloc bk worked
|
|
BNE HCheckHandle ; If ok, take the NewHandle exit route.
|
|
MoveQ #memFullErr,D0 ; If we can't find the space, just report
|
|
Bra.S SHSExit ; the error, leaving handle alone . . .
|
|
RAHSkip
|
|
BSR a32GetSize ; D0 := current data size <v1.1>
|
|
CMP.L D0,D2 ; requested - current ... unsigned <04Mar85>
|
|
BHI.S RAHError ; it's erroneous to grow <04Mar85>
|
|
MOVEQ #memROZWarn,D0 ; <04Mar85>
|
|
BRA.S SHSExit ; <04Mar85>
|
|
RAHError
|
|
MOVEQ #memROZError,D0 ; <04Mar85>
|
|
BRA.S SHSExit ; <04Mar85>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HLock(h: Handle);
|
|
;
|
|
; - Lock the relocatable block pointed to by the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HLock Move.L #JHLock,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
; <2> kc.start
|
|
v24Hlock BEQ HFail ; handle was NIL
|
|
BSET #Lock,(A1) ; set the lock bit
|
|
BRA HGood ; return success
|
|
|
|
v32Hlock BEQ HFail ; handle was NIL
|
|
bset #Lock,MPtag32-blkData32(A0) ; set the lock bit
|
|
BRA HGood ; return success
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HUnLock(h: Handle);
|
|
;
|
|
; - Unlock the relocatable block pointed to by the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HUnlock
|
|
Move.L #JHUnLock,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HUnlock
|
|
BEq HFail ; handle was NIL
|
|
|
|
BClr #Lock,(A1) ; reset the lock bit
|
|
BrA HGood ; return success
|
|
|
|
v32HUnlock
|
|
BEq.S HFail ; handle was NIL
|
|
|
|
BClr #Lock,MPtag32-blkData32(A0) ;reset the lock bit <v1.3>
|
|
BrA.S HGood ; return success
|
|
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HPurge(h: Handle);
|
|
;
|
|
; - Enable purging the relocatable block pointed to by the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HPurge
|
|
Move.L #JHPurge,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HPurge
|
|
BEq.S HFail ; handle was NIL
|
|
|
|
BSet #Purge,(A1) ; set the purge bit
|
|
BrA.S HGood ; return success
|
|
|
|
v32HPurge
|
|
BEq.S HFail ; handle was NIL
|
|
|
|
BSet #Purge,MPtag32-blkData32(A0);set the purge bit <v1.3>
|
|
BrA.S HGood ; return success
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HNoPurge(h: Handle);
|
|
;
|
|
; - Disable purging the relocatable block pointed to by the given handle.
|
|
; Stuff error code globally. <25Apr85>
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HNoPurge
|
|
Move.L #JHNoPurge,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HNoPurge
|
|
BEq.S HFail ; handle was NIL
|
|
|
|
BClr #Purge,(A1) ; reset the purge bit
|
|
BrA.S HGood ;return success <v1.1>
|
|
|
|
v32HNoPurge
|
|
BEq.S HFail ; handle was NIL
|
|
|
|
BClr #Purge,MPtag32-blkData32(A0);reset the purge bit <v1.3>
|
|
|
|
HGood
|
|
MoveQ #0,D0
|
|
BrA.S HExit
|
|
HFail
|
|
MoveQ #nilHandleErr,D0
|
|
HExit
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HRSRC(h: Handle); <13Jan86 JTC>
|
|
;
|
|
; - Set the resource bit of the relocatable block pointed to by the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HRSRC ; <13Jan86 JTC>
|
|
Move.L #JHRSRC,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HRSRC
|
|
BEq.S HFail ;handle was NIL <13Jan86 JTC>
|
|
|
|
BSet #Resource,(A1) ; <C206>
|
|
BrA.S HGood ;return success <13Jan86 JTC>
|
|
|
|
|
|
v32HRSRC
|
|
BEq.S HFail ;handle was NIL <13Jan86 JTC>
|
|
|
|
BSet #Resource,MPtag32-blkData32(A0) ;set the Resource bit <v1.3>
|
|
BrA.S HGood ;return success <13Jan86 JTC>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HNoRSRC(h: Handle); <13Jan86 JTC>
|
|
;
|
|
; - Clear the resource bit of the relocatable block pointed to by the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
;
|
|
; A0 - Handle
|
|
;
|
|
|
|
HNoRSRC ; <v1.1>
|
|
Move.L #jHNoRSRC,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HNoRSRC
|
|
BEq.S HFail ; handle was NIL <13Jan86 JTC>
|
|
|
|
BClr #Resource,(A1) ; reset the RSRC bit <C206>
|
|
BrA.S HGood ; return success <13Jan86 JTC>
|
|
|
|
v32HNoRSRC
|
|
BEq.S HFail ; handle was NIL <13Jan86 JTC>
|
|
|
|
BClr #Resource,MPtag32-blkData32(A0) ;reset the RSRC bit <v1.3>
|
|
BrA.S HGood ; return success <13Jan86 JTC>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION HGetFlags(h: Handle) : BYTE; <13Jan86 JTC>
|
|
;
|
|
; - Fetch the flag byte of the master pointer of the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
; A0 - Handle
|
|
; On exit, Registers:
|
|
; D0 - flag byte in D0.B, rest of bits clear
|
|
;
|
|
|
|
HGetFlags ; <13Jan86 JTC>
|
|
Move.L #JHGetFlags,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HGetFlags
|
|
BEq.S HFail ; if Nil, stuff D0 with error code <13Jan86 JTC>
|
|
|
|
MoveQ #0,D0 ; init high bits <13Jan86 JTC>
|
|
Move.B (A1),D0 ; stuff the flag byte <13Jan86 JTC>
|
|
|
|
Bra.S GetFlagsExit ; exit through common code <v.1.9>
|
|
|
|
v32HGetFlags
|
|
BEq.S HFail ; if Nil, stuff D0 with error code <13Jan86 JTC>
|
|
|
|
MoveQ #0,D0 ; init high bits <13Jan86 JTC>
|
|
|
|
|
|
Move.B MPtag32-BlkData32(A0),D0 ;stuff the flag byte <v1.3>
|
|
GetFlagsExit
|
|
Clr.W MemErr ; custom exit, clearing error bits <13Jan86 JTC>
|
|
|
|
MOVE.L vMMNoErrEpilogue,-(SP) ; get MMNoErrEpilogue vector <v1.9>
|
|
RtS ; <13Jan86 JTC>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE HSetFlags(h: Handle; f: BYTE); <13Jan86 JTC>
|
|
;
|
|
; - Set the flag byte of the master pointer of the given handle.
|
|
;
|
|
; On entry, Registers:
|
|
; A0 - Handle
|
|
; D0 - flag byte in D0.B, rest of bits don't care
|
|
;
|
|
|
|
HSetFlags ; <13Jan86 JTC>
|
|
Move.L #JHSetFlags,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24HSetFlags
|
|
BEq.S HFail ; if Nil, stuff D0 with error code <13Jan86 JTC>
|
|
|
|
Move.B D0,(A1) ; stuff flags <v1.3>
|
|
Bra.S HGood ; return OK <13Jan86 JTC>
|
|
|
|
|
|
v32HSetFlags
|
|
BEq.S HFail ; if Nil, stuff D0 with error code <13Jan86 JTC>
|
|
|
|
Move.B D0,MPtag32-BlkData32(A0) ;stuff the flag byte <v1.3>
|
|
Bra.S HGood ; return OK <13Jan86 JTC>
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; PROCEDURE MoreMasters();
|
|
;
|
|
; - Direct call from app, as opposed to demand requisition of MPs. <29jan85>
|
|
; Stuff error code globally, now. <25Apr85>
|
|
;
|
|
; No parameters.
|
|
;
|
|
MoreMasters
|
|
Move.L #jMoreMasters,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMMMPrologue,-(SP) ; get MMMMPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
|
|
v24MoreMasters ; 24 bit version of MoreMasters <v1.1>
|
|
MOVEQ #0,D0 ; assume success
|
|
BSR a24HMakeMoreMasters ; call ROM routine
|
|
BNE.S @1 ; br for success
|
|
MOVEQ #MemFullErr,D0 ; change D0 from 0 to error code
|
|
@1 MOVE.L (SP)+,A6 ; restore A6
|
|
Move.W D0,MemErr ; global error code <25Apr85>
|
|
RTS ; that's all
|
|
|
|
v32MoreMasters ; 32 bit version of MoreMasters <v1.1>
|
|
MOVEQ #0,D0 ; assume success
|
|
BSR a32HMakeMoreMasters ; call ROM routine
|
|
BNE.S @1 ; br for success
|
|
MOVEQ #MemFullErr,D0 ; change D0 from 0 to error code
|
|
@1 MOVE.L (SP)+,A6 ; restore A6
|
|
Move.W D0,MemErr ; global error code <25Apr85>
|
|
RTS ; that's all
|
|
|
|
|
|
|
|
;----------------------------------------------------------------------
|
|
;
|
|
; FUNCTION StripAddress(p: Ptr): Ptr;
|
|
;
|
|
; Make an address in D0 suitable for use in the current environment. Ultimately,
|
|
; this should test some flag and mask suitable bits.
|
|
;
|
|
; D0 <-> Address to be stripped.
|
|
;
|
|
__StripAddress
|
|
;Roll in FixStripAddress to not check the MMStartMode <SM12>
|
|
|
|
IF Supports24Bit THEN ; <SM17> CSS
|
|
BTST.B #Systemis24bit,SystemInfo ; is it 32 bit or 24 bit <SM12>
|
|
BEQ.S @1 ; no strip address for 32 bit mode <SM12>
|
|
AND.L Lo3Bytes,D0
|
|
@1
|
|
ENDIF
|
|
RTS
|
|
|
|
|
|
|
|
;-------------------- Moved in from RAM-based glue ----------------------------------------- <04May85>
|
|
;-------------------------------------------------------------------------------------------
|
|
; PROCEDURE MoveHHi( h : Handle );
|
|
;
|
|
; Moves h up to the top of its region (space between two locked/nonrelocatable objects or
|
|
; between a lockek/nonreloc and the heap end), but with neither purge nor heap growth, only
|
|
; compaction. The idea is to use a stack buffer to move slices of the rising block in steps,
|
|
; compacting along the way to free space at the top. Requires about 1K of stack space.
|
|
; If the block is already high, MoveHHi does nothing.
|
|
;
|
|
; Arguments: A0 = handle to be moved high
|
|
; Results: D0 = error code = 0 no error
|
|
; NilHandleErr
|
|
; LkdHandleErr
|
|
; StackErr
|
|
; Register conventions:
|
|
;
|
|
; A6 - theZone pointer
|
|
; A5 - stack locals, buffer
|
|
; A4 - handle of block we are moving
|
|
; A3 - BkLim(theZone)
|
|
;
|
|
; D6 - current BC of block moved so far (not including header)
|
|
; D7 - current BC of block left to move (not including header)
|
|
;-------------------------------------------------------------------------------------------
|
|
;
|
|
; Modification History:
|
|
; 11-Apr-84 LAK Mask dereferenced handle for end-of-heap compare (get rid of
|
|
; any lock/purge/rsrc bits set . . .
|
|
; 11-Jul-84 MDB, MPH Correctly save Master Pointer flag bits.
|
|
; 14-Nov-84 RS,SC nilHandleErr, already locked err, no compact if there is enough
|
|
; room in heap to make copy of block.
|
|
; 27-Nov-84 RS fix so that multiple heap compacts are not done if not needed.
|
|
; ie. if after a compact there is a large enough free block up high to
|
|
; move all the remaining
|
|
; 14-Mar-85 CC changed to proc that returns error in MemErr; check for new ROM
|
|
; 02 May 85 JTC Moved to ROM, with general scouring. Fixed subtle bug where a free block
|
|
; smaller than minFree would be 'allocated' after a slide.
|
|
; <01Nov85 JTC> Now take stack space bounded above by what is needed (if not smaller than minimum
|
|
; acceptable) and what is available based on HiHeapMark.
|
|
;-------------------------- Post V. 1.0 System Tools ---------------------------------------
|
|
; <13Jan86 JTC> Stuff TheZone across call, so that secondary calls such as _CompactMem will
|
|
; operate on same heap. The trick is to save incoming TheZone in the stack
|
|
; frame until exit, but to be careful about the various exit pts through MMEpilogue.
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
; ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ SuperMario roll in <SM 8> tcn ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
; MoveHHi - do not let it happen to a handle in the System Heap.
|
|
;
|
|
; MoveHHi is usually called because the caller is about to lock the handle down. This is
|
|
; the right thing to do in app heaps, but is bad for the System Heap. Immobile blocks in
|
|
; the System Heap limit the extendability of the Process Mgr heap.
|
|
;
|
|
; Assumptions:
|
|
; This assumes that the system heap is the lowest heap in memory. In particular, this check will
|
|
; disable MoveHHi in subzones residing in the system heap. This behavior has been around for a
|
|
; while in MultiFinder, but should go away when we rewrite this patch.
|
|
;
|
|
; To Do: Rewrite, using the relative handle to find the zone start, comparing that to
|
|
; SysZone. This will need a safe dereference, perhaps borrowed from the IIci ROM,
|
|
; which would include a bus-error handler.
|
|
;
|
|
;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
|
|
StkSlop EQU 3072 ; leave 3K for system
|
|
StackErr EQU -1 ; not enuf stack space for us
|
|
|
|
OldTagBC24 EQU 0 ; first long in original blk
|
|
MyBufCnt24 EQU 4 ; current buffer count
|
|
MyBufSize24 EQU 8 ; size of stack buffer
|
|
InRgnPtr24 EQU 12 ; region reference point
|
|
SvTagByte24 EQU 16 ; saved tag byte
|
|
SavedZone24 EQU 18 ; saved theZone across the call <13Jan86 JTC>
|
|
MyStkBuf24 EQU 22 ; start of stack buffer <13Jan86 JTC>
|
|
LclVarSize24 EQU MyStkBuf24
|
|
|
|
OldTagBC32 EQU 0 ; first long in original blk <v1.1>
|
|
SvBlkSize32 EQU 4 ; saved block size <v1.1>
|
|
MyBufCnt32 EQU 8 ; current buffer count <v1.1>
|
|
MyBufSize32 EQU 12 ; size of stack buffer <v1.1>
|
|
InRgnPtr32 EQU 16 ; region reference point <v1.1>
|
|
SavedZone32 EQU 20 ; saved theZone across the call <v1.1>
|
|
MyStkBuf32 EQU 24 ; start of stack buffer <v1.1>
|
|
LclVarSize32 EQU MyStkBuf32 ; <v1.1>
|
|
|
|
MoveHHi
|
|
Move.L #JMoveHHi,-(SP) ; get offset to vector onto stack <v1.9>
|
|
Move.L vMMHPrologue,-(SP) ; get MMHPrologue vector onto stack <v1.9>
|
|
RTS ; go to routine <v1.9>
|
|
|
|
v24MoveHHi ; 24 bit version of MoveHHi <v1.1>
|
|
;-------------------------------------------------------------------------------
|
|
; First check for the nasties: locked or purged.
|
|
;-------------------------------------------------------------------------------
|
|
BNE.S @goBubble ; NotEqual => not purged <31mar86>
|
|
; <SM29> kc
|
|
MOVEQ #NilHandleErr,D0 ; nil handle error <SM29> kc
|
|
BRA MHHExit ; go to common exit <SM29> kc
|
|
@goBubble ; <SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle in the System heap? <SM29> kc
|
|
BNE.S @notInSysZone ; nope <SM29> kc
|
|
; <SM29> kc
|
|
MOVEQ #noErr,D0 ; yup, don't do anything <SM29> kc
|
|
BRA MHHExit ; go to common exit <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
MOVE.L A1,A4 ; save handle to block for the duration <SM29> kc
|
|
TST.B (A4) ; make sure block is unlocked <SM29> kc
|
|
BPL.S @goAhead ; yup, all's cool <SM29> kc
|
|
; <SM29> kc
|
|
MOVEQ #memLockedErr,D0 ; locked handle error <SM29> kc
|
|
BRA MHHExit ; <SM29> kc
|
|
@goAhead
|
|
MOVE.L BkLim(A6),A3 ; last memory block, for the duration <04May85>
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; See if we aren't already packed at top, in which case a compact is avoided.
|
|
; Set up D7 for the duration.
|
|
;-------------------------------------------------------------------------------
|
|
|
|
SUBQ.L #BlkData24,A0 ; point to actual block
|
|
MOVE.L TagBC24(A0),D1 ; block header
|
|
AND.L Lo3Bytes,D1 ; isolate the block size
|
|
MOVE.L D1,D7 ; save in D7 for the duration
|
|
SUBQ.L #BlkData24,D7 ; adjust for data count only
|
|
|
|
MOVEQ #noErr,D0 ; assume no move necessary, ok exit
|
|
ADD.L D1,A0 ; point to next block
|
|
CMP.L A3,A0 ; is it already at the limit?
|
|
BCC MHHExit ; exit if so . . .
|
|
|
|
TST.B TagBC24(A0) ; ck for Free, Rel, or NRel
|
|
BEQ.S @1 ; br if Free
|
|
BPL MHHExit ; exit if NRel (we're at region top)
|
|
|
|
MOVE.L Handle24(A0),A0 ; handle offset
|
|
ADD.L A6,A0 ; master ptr
|
|
TST.B (A0) ; locked Rel?
|
|
BMI MHHExit ; exit if so
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Set up stack buffer and local vars, with frame ptr in A5 (for the variety's sake).
|
|
; Use as much stack as is available (knowing there will be no ApplZone growth),
|
|
; but no less than 128 bytes, permitting 768 bytes for intermediate stack activity.
|
|
; New intelligence: <01Nov85 JTC>
|
|
; The basic buffer is bounded first by the new HiHeapMark (rather than the old
|
|
; BKLim(ApplZone)). Then it is bounded by the size of the block to be moved plus
|
|
; a modicum of stack slop.
|
|
; Once the frame is set up, save theZone and stuff A6 there, for subsequent MM calls<13Jan86 JTC>
|
|
;-------------------------------------------------------------------------------
|
|
@1 MOVE.L SP,D1 ; figure out the size of our stack buffer
|
|
AND.L Lo3Bytes,D1 ; why not be safe with our pointer arithmetic? <01Nov85 JTC>
|
|
SUB.L HiHeapMark,D1 ; distance between stack and highest nonstack zone <01Nov85 JTC>
|
|
AND.L #$FFFFFFF0,D1 ; quad-word align the stack buffer for 040 <SM25>
|
|
|
|
**** MOVE.L ApplZone,A0 ; (make sure we use appl zone for this) <01Nov85 JTC>
|
|
**** SUB.L BkLim(A0),D1 ; distance between stack and appl heapzone <01Nov85 JTC>
|
|
|
|
MOVE.L #StkSlop,D2 ; we demand at least this much raw space
|
|
MOVE.L D2,D0 ; in case of much space, just take what is needed <01Nov85 JTC>
|
|
ADD.L D7,D0 ; block size + slop <01Nov85 JTC>
|
|
CMP.L D0,D1 ; available-(what's needed for this case) <01Nov85 JTC>
|
|
BLT.S @3 ; Less Than (signed OK) ==> no excess stack <01Nov85 JTC>
|
|
MOVE.L D0,D1 ; use more parsimonious stack block <01Nov85 JTC>
|
|
@3 ; bypass using smaller stack request <01Nov85 JTC>
|
|
MOVEQ #StackErr,D0 ; assume error
|
|
CMP.L D2,D1 ; make sure there's at least 1K
|
|
BLT MHHExit ; just exit if not
|
|
SUB.L #128,D2 ; just use a little of the slop (was SUB.W) <01Nov85 JTC>
|
|
SUB.L D2,D1 ; buffer size is what's left
|
|
|
|
SUB.L D1,SP ; buffer
|
|
SUB.W #LclVarSize24,SP ; and space for locals
|
|
MOVE.L SP,A5 ; save pointer in A5
|
|
MOVE.L D1,MyBufSize24(A5) ; save buffer size
|
|
MOVE.L theZone,SavedZone24(A5) ; preserve incoming theZone... <13Jan86 JTC>
|
|
MOVE.L A6,theZone ; ...to be replace by handle's zone <13Jan86 JTC>
|
|
|
|
MOVE.L (A4),A0 ; also keep TagBC of block we are moving
|
|
SUBQ #BlkData24,A0
|
|
MOVE.L TagBC24(A0),OldTagBC24(A5)
|
|
|
|
MOVE.L (A4),D0 ; deref handle
|
|
AND.L Lo3Bytes,D0 ; get rid of lock, purge, etc. bits
|
|
MOVE.L D0,InRgnPtr24(A5) ; save blk as region hint
|
|
Move.B (A4),SvTagByte24(A5) ; Save tag byte.
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Don't do compact if there is a free block at the top of the region that
|
|
; is large enough to hold the contents of the block to be moved up.
|
|
;-------------------------------------------------------------------------------
|
|
BSR a24FindRgnTop ;get A2 pointing to the last block before end of region
|
|
TST.B (A2) ;is it free?
|
|
BNE.S @goCont ;no, do the buffering stuff
|
|
MOVE.L TagBC24(A2),D1 ;get the size of the block
|
|
MOVE.L D1,D0
|
|
SUBQ.L #BlkData24,D0 ;last block data space
|
|
CMP.L D0,D7 ;is it enough space?
|
|
BHI.S @goCont ;no, do the buffering stuff
|
|
SUB.L D7,D0 ;must be nonnegative! <02May85>
|
|
BEQ.S @5 ;if just enough space, go do it! <02May85>
|
|
MOVEQ #minFree24,D2 ;need at least this much to have a free block <02May85><v1.2>
|
|
CMP.L D2,D0 ; <02May85>
|
|
BCS.S @goCont ;Carry Set => not minFree left, do it the hard way <02May85>
|
|
@5
|
|
MOVE.L (A4),A0 ;ptr to the block we are moving from
|
|
MOVE.L A2,A1 ;ptr to free block
|
|
ADD.L D1,A1 ;ptr to end of free block
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; One-shot move:
|
|
; D7 = amount of move
|
|
; A0 = ptr to migrating block data
|
|
; A1 = ptr to END of free block at top of region
|
|
;-------------------------------------------------------------------------------
|
|
@moveRest MOVE.L D7,D0 ;number of bytes of data to move
|
|
SUB.L D0,A1 ;grab what we need
|
|
SUB.L D0,ZCBFree(A6) ;
|
|
_BlockMove ;move the bits
|
|
MOVE.L A4,A0 ;get the handle
|
|
_EmptyHandle ;empty it out
|
|
BRA.S @MNSFinish
|
|
|
|
;-------------------------------------------------------------------------------------------
|
|
; Entry to fill/compact/stuff loop...
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
@goCont
|
|
MOVEQ #0,D6 ; D6 holds number of bytes moved up . . .
|
|
BSR a24FillStkBuf ; this does it and also adjusts handlesize
|
|
BSR DoCompact ; then move everything down
|
|
BSR a24FindRgnTop
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; The top of the MoveNxtSeg loop:
|
|
; The stack buffer is full, the heap is compacted, A2 points to the free block
|
|
; (at least as large as the buffer) at the region's top. This sequence loads
|
|
; the buffer's contents up high in the region.
|
|
;-------------------------------------------------------------------------------
|
|
|
|
@MoveNxtSeg
|
|
MOVE.L TagBC24(A2),D1 ; size of free block we have at top now
|
|
MOVE.L MyBufCnt24(A5),D0 ; stack buffer count
|
|
LEA MyStkBuf24(A5),A0 ; source
|
|
MOVE.L A2,A1 ; compute destination
|
|
ADD.L D1,A1 ; point to end of free block
|
|
SUB.L D0,A1 ; grab what we need
|
|
ADD.L D0,D6 ; update count moved
|
|
SUB.L D0,ZCBFree(A6) ; *** this is a fix ***
|
|
_BlockMove ; move it in
|
|
|
|
TST.L D7 ; are we done?
|
|
BEQ.S @MNSFinish ; br if so <27oct85> BBM
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; There's more to move than the bufferfull we just did. If there's just enough
|
|
; free space, or enough to move the block and have at least a minFree block,
|
|
; then exit via moveRest stub earlier. Otherwise, continue looping with the
|
|
; stack buffer...
|
|
;-------------------------------------------------------------------------------
|
|
MOVE.L A1,D0 ; ptr to dest data
|
|
SUB.L A2,D0 ; free bytes remaining
|
|
BEQ.S @MNSExact ; br if none
|
|
MOVE.L D0,D1 ; get the #free bytes remaining
|
|
SUBQ.L #BlkData24,D1 ; #free for real data
|
|
SUB.L D7,D1 ; enough for the remaining bytes?
|
|
BCS.S @MNSAdj ; Carry Set => not enough, keep buffering <02May85>
|
|
BEQ.S @MNSEscape ; EQual => just enough, so do it <02May85>
|
|
|
|
MOVEQ #minFree24,D2 ; if unequal, must be this much to be worthwhile <02May85><v1.2>
|
|
CMP.L D1,D2 ; extra VERSUS minFree <02May85>
|
|
BCC.S @MNSAdj ; not enough extra for a free block: the hard way <02Jun87>
|
|
; CC => D1 < minFree => fiddle the slack space <02Jun87>
|
|
@MNSEscape
|
|
MOVE.L (A4),A0 ; yes, move it. get source ptr
|
|
BRA.S @moveRest
|
|
@MNSAdj
|
|
MOVE.L D0,TagBC24(A2) ; adjust size of remaining free blk
|
|
@MNSExact
|
|
MOVEM.L (A1),D4-D5 ; save 2 longs at front of blk
|
|
MOVE.L A6,Handle24(A1)
|
|
MOVE.L D6,TagBC24(A1) ; data size is blk size in this case . . .
|
|
MOVE.B #$40,TagBC24(A1) ; make it an nrel blk
|
|
|
|
MOVE.L A1,-(SP)
|
|
BSR a24FillStkBuf ; this does it and also adjusts handlesize <27oct85> BBM
|
|
BSR DoCompact ; then move everything down
|
|
BSR a24FindRgnTop ; point A2 to free bk at top . . . <27oct85> BBM
|
|
MOVE.L (SP)+,A1
|
|
MOVEM.L D4-D5,(A1) ; restore data at front of blk
|
|
BRA @MoveNxtSeg ; then go again
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Clean-up code: set up any free block left over; set up header of relocated block;
|
|
; update master pointer; recover stack space and saved registers.
|
|
; Registers: A6 Zone
|
|
; A5 Local variables and buffer on stack
|
|
; A4 Handle of migrating block
|
|
; A2 Ptr to any free space
|
|
; A1 Ptr to moved block
|
|
;-------------------------------------------------------------------------------
|
|
|
|
@MNSFinish
|
|
SUBQ.L #8,A1 ; grab 8 more bytes for block header
|
|
SUBQ.L #8,ZCBFree(A6) ; *** this is a fix ***
|
|
MOVE.L A1,D0 ; figure free space left
|
|
SUB.L A2,D0
|
|
BEQ.S @11 ; br if we used it all
|
|
MOVE.L D0,TagBC24(A2) ; create a free block of remaining size
|
|
|
|
@11 MOVE.L A1,A2 ; point A2 to our new block
|
|
MOVE.L OldTagBC24(A5),(A2)+ ; TagBC
|
|
MOVE.L A4,A0
|
|
SUB.L A6,A0 ; form handle offset
|
|
MOVE.L A0,(A2)+ ; Handle
|
|
|
|
MOVE.L A2,(A4) ; update master pointer with ptr to text of block
|
|
MOVE.B SvTagByte24(A5),(A4) ; and restore flags . . .
|
|
MOVE.L SavedZone24(A5),theZone ; and theZone <13Jan86 JTC>
|
|
MOVEQ #noErr,D0 ; ok!
|
|
|
|
ADD.L MyBufSize24(A5),SP ; recover buffer space
|
|
ADD.W #LclVarSize24,SP ; and locals space
|
|
|
|
BRA MHHExit ; <v1.1>
|
|
|
|
|
|
;-------- 32 bit version ---
|
|
|
|
v32MoveHHi ; 32 bit version of MoveHHi <v1.1>
|
|
;-------------------------------------------------------------------------------
|
|
; First check for the nasties: locked or purged.
|
|
;-------------------------------------------------------------------------------
|
|
BNE.S @goBubble ; NotEqual => not purged <31mar86>
|
|
; <SM29> kc
|
|
MOVEQ #NilHandleErr,D0 ; nil handle error <SM29> kc
|
|
BRA MHHExit ; jump to common exit <SM29> kc
|
|
@goBubble ; <SM29> kc
|
|
CMPA.L SysZone,a6 ; is this handle in the System heap? <SM29> kc
|
|
BNE.S @notInSysZone ; nope <SM29> kc
|
|
; <SM29> kc
|
|
MOVEQ #noErr,D0 ; yup, don't do anything <SM29> kc
|
|
BRA MHHExit ; go to common exit <SM29> kc
|
|
@notInSysZone ; <SM29> kc
|
|
MOVE.L A1,A4 ; save handle to block for the duration <SM29> kc
|
|
TST.B MPtag32-BlkData32(A0) ; make sure block is unlocked <SM29> kc
|
|
BPL.S @goAhead ; yup, all's cool <SM29> kc
|
|
; <SM29> kc
|
|
MOVEQ #memLockedErr,D0 ; locked handle error <SM29> kc
|
|
BRA MHHExit ; jump to common exit <SM29> kc
|
|
@goAhead
|
|
MOVE.L BkLim(A6),A3 ; last memory block, for the duration <04May85>
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; See if we aren't already packed at top, in which case a compact is avoided.
|
|
; Set up D7 for the duration.
|
|
;-------------------------------------------------------------------------------
|
|
|
|
SUB.L #BlkData32,A0 ; point to actual block <v1.1>
|
|
Move.L BlkSize32(A0),D1 ; get block size <v1.1>
|
|
MOVE.L D1,D7 ; save in D7 for the duration <v1.1>
|
|
SUB.L #BlkData32,D7 ; adjust for data count only <v1.1>
|
|
|
|
MOVEQ #noErr,D0 ; assume no move necessary, ok exit
|
|
ADD.L D1,A0 ; point to next block
|
|
CMP.L A3,A0 ; is it already at the limit?
|
|
BCC MHHExit ; exit if so . . .
|
|
|
|
TST.B TagBC32(A0) ; ck for Free, Rel, or NRel
|
|
BEQ.S @1 ; br if Free
|
|
BPL MHHExit ; exit if NRel (we're at region top)
|
|
|
|
Tst.B MPtag32(A0) ; locked Rel? <v1.1>
|
|
BMI MHHExit ; exit if so
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Set up stack buffer and local vars, with frame ptr in A5 (for the variety's sake).
|
|
; Use as much stack as is available (knowing there will be no ApplZone growth),
|
|
; but no less than 128 bytes, permitting 768 bytes for intermediate stack activity.
|
|
; New intelligence: <01Nov85 JTC>
|
|
; The basic buffer is bounded first by the new HiHeapMark (rather than the old
|
|
; BKLim(ApplZone)). Then it is bounded by the size of the block to be moved plus
|
|
; a modicum of stack slop.
|
|
; Once the frame is set up, save theZone and stuff A6 there, for subsequent MM calls<13Jan86 JTC>
|
|
;-------------------------------------------------------------------------------
|
|
@1 MOVE.L SP,D1 ; figure out the size of our stack buffer
|
|
SUB.L HiHeapMark,D1 ; distance between stack and highest nonstack zone <01Nov85 JTC>
|
|
AND.L #$FFFFFFF0,D1 ; quad-word align the stack buffer for 040 <SM25>
|
|
|
|
**** MOVE.L ApplZone,A0 ; (make sure we use appl zone for this) <01Nov85 JTC>
|
|
**** SUB.L BkLim(A0),D1 ; distance between stack and appl heapzone <01Nov85 JTC>
|
|
|
|
MOVE.L #StkSlop,D2 ; we demand at least this much raw space
|
|
MOVE.L D2,D0 ; in case of much space, just take what is needed <01Nov85 JTC>
|
|
ADD.L D7,D0 ; block size + slop <01Nov85 JTC>
|
|
CMP.L D0,D1 ; available-(what's needed for this case) <01Nov85 JTC>
|
|
BLT.S @3 ; Less Than (signed OK) ==> no excess stack <01Nov85 JTC>
|
|
MOVE.L D0,D1 ; use more parsimonious stack block <01Nov85 JTC>
|
|
@3 ; bypass using smaller stack request <01Nov85 JTC>
|
|
MOVEQ #StackErr,D0 ; assume error
|
|
CMP.L D2,D1 ; make sure there's at least 1K
|
|
BLT MHHExit ; just exit if not
|
|
SUB.L #128,D2 ; just use a little of the slop (was SUB.W) <01Nov85 JTC>
|
|
SUB.L D2,D1 ; buffer size is what's left
|
|
|
|
SUB.L D1,SP ; buffer
|
|
SUB.W #LclVarSize32,SP ; and space for locals
|
|
MOVE.L SP,A5 ; save pointer in A5
|
|
MOVE.L D1,MyBufSize32(A5) ; save buffer size
|
|
MOVE.L theZone,SavedZone32(A5) ; preserve incoming theZone... <13Jan86 JTC>
|
|
MOVE.L A6,theZone ; ...to be replace by handle's zone <13Jan86 JTC>
|
|
|
|
MOVE.L (A4),A0 ; also keep TagBC of block we are moving
|
|
SUB.L #BlkData32,A0 ; <v1.1>
|
|
MOVE.L TagBC32(A0),OldTagBC32(A5); save 1st long word <v1.1>
|
|
Move.L (A4),InRgnPtr32(A5) ; save blk as region hint <v1.1>
|
|
Move.L Blksize32(A0),SvBlksize32(A5); save block size <v1.1>
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Don't do compact if there is a free block at the top of the region that
|
|
; is large enough to hold the contents of the block to be moved up.
|
|
;-------------------------------------------------------------------------------
|
|
BSR a32FindRgnTop ;get A2 pointing to the last block before end of region
|
|
TST.B (A2) ;is it free?
|
|
BNE.S @goCont ;no, do the buffering stuff
|
|
MOVE.L BlkSize32(A2),D1 ;get the size of the block <v1.1>
|
|
MOVE.L D1,D0 ; <v1.1>
|
|
SUB.L #BlkData32,D0 ;last block data space <v1.1>
|
|
CMP.L D0,D7 ;is it enough space?
|
|
BHI.S @goCont ;no, do the buffering stuff
|
|
SUB.L D7,D0 ;must be nonnegative! <02May85>
|
|
BEQ.S @5 ;if just enough space, go do it! <02May85>
|
|
MOVEQ #minFree32,D2 ;need at least this much to have a free block <02May85><v1.2>
|
|
CMP.L D2,D0 ; <02May85>
|
|
BCS.S @goCont ;Carry Set => not minFree left, do it the hard way <02May85>
|
|
@5
|
|
MOVE.L (A4),A0 ;ptr to the block we are moving from
|
|
MOVE.L A2,A1 ;ptr to free block
|
|
ADD.L D1,A1 ;ptr to end of free block
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; One-shot move:
|
|
; D7 = amount of move
|
|
; A0 = ptr to migrating block data
|
|
; A1 = ptr to END of free block at top of region
|
|
;-------------------------------------------------------------------------------
|
|
@moveRest MOVE.L D7,D0 ;number of bytes of data to move
|
|
SUB.L D0,A1 ;grab what we need
|
|
SUB.L D0,ZCBFree(A6) ;
|
|
_BlockMove ;move the bits
|
|
MOVE.L A4,A0 ;get the handle
|
|
_EmptyHandle ;empty it out
|
|
BRA @MNSFinish ; <SM17> CSS
|
|
|
|
;-------------------------------------------------------------------------------------------
|
|
; Entry to fill/compact/stuff loop...
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
@goCont
|
|
MOVEQ #0,D6 ; D6 holds number of bytes moved up . . .
|
|
BSR a32FillStkBuf ; this does it and also adjusts handlesize
|
|
BSR DoCompact ; then move everything down
|
|
BSR a32FindRgnTop
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; The top of the MoveNxtSeg loop:
|
|
; The stack buffer is full, the heap is compacted, A2 points to the free block
|
|
; (at least as large as the buffer) at the region's top. This sequence loads
|
|
; the buffer's contents up high in the region.
|
|
;-------------------------------------------------------------------------------
|
|
|
|
@MoveNxtSeg
|
|
MOVE.L BlkSize32(A2),D1 ; size of free block we have at top <v1.1>
|
|
MOVE.L MyBufCnt32(A5),D0 ; stack buffer count
|
|
LEA MyStkBuf32(A5),A0 ; source
|
|
MOVE.L A2,A1 ; compute destination
|
|
ADD.L D1,A1 ; point to end of free block
|
|
SUB.L D0,A1 ; grab what we need
|
|
ADD.L D0,D6 ; update count moved
|
|
SUB.L D0,ZCBFree(A6) ; *** this is a fix ***
|
|
_BlockMove ; move it in
|
|
|
|
TST.L D7 ; are we done?
|
|
BEQ.S @MNSFinish ; br if so <27oct85> BBM
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; There's more to move than the bufferfull we just did. If there's just enough
|
|
; free space, or enough to move the block and have at least a minFree block,
|
|
; then exit via moveRest stub earlier. Otherwise, continue looping with the
|
|
; stack buffer...
|
|
;-------------------------------------------------------------------------------
|
|
MOVE.L A1,D0 ; ptr to dest data
|
|
SUB.L A2,D0 ; free bytes remaining
|
|
BEQ.S @MNSExact ; br if none
|
|
MOVE.L D0,D1 ; get the #free bytes remaining
|
|
SUB.L #BlkData32,D1 ;#free for real data <v1.1>
|
|
SUB.L D7,D1 ; enough for the remaining bytes?
|
|
BCS.S @MNSAdj ; Carry Set => not enough, keep buffering <02May85>
|
|
BEQ.S @MNSEscape ; EQual => just enough, so do it <02May85>
|
|
|
|
MOVEQ #minFree32,D2 ; if unequal, must be this much to be worthwhile <02May85><v1.2>
|
|
CMP.L D1,D2 ; extra VERSUS minFree <02May85>
|
|
BCC.S @MNSAdj ; not enough extra for a free block: the hard way <02Jun87>
|
|
; CC => D1 < minFree => fiddle the slack space <02Jun87>
|
|
@MNSEscape
|
|
MOVE.L (A4),A0 ; yes, move it. get source ptr
|
|
BRA.S @moveRest
|
|
@MNSAdj
|
|
Clr.L TagBC32(A2) ; make it a free block <v1.1>
|
|
MOVE.L D0,BlkSize32(A2) ; adjust size of remaining free blk <v1.1>
|
|
@MNSExact
|
|
MOVEM.L (A1),D3-D5 ; save 3 longs at front of blk <v1.1>
|
|
MOVE.L A6,Handle32(A1) ; <v1.1>
|
|
MOVE.L D6,BlkSize32(A1) ; data size is blk size in this case <v1.1>
|
|
MOVE.B #$40,TagBC32(A1) ; make it an nrel blk
|
|
|
|
MOVE.L A1,-(SP)
|
|
BSR a32FillStkBuf ; this does it and also adjusts handlesize <27oct85> BBM
|
|
BSR DoCompact ; then move everything down
|
|
BSR a32FindRgnTop ; point A2 to free bk at top . . . <27oct85> BBM
|
|
MOVE.L (SP)+,A1
|
|
MOVEM.L D3-D5,(A1) ; restore data at front of blk <v1.1>
|
|
BRA @MoveNxtSeg ; then go again
|
|
|
|
;-------------------------------------------------------------------------------
|
|
; Clean-up code: set up any free block left over; set up header of relocated block;
|
|
; update master pointer; recover stack space and saved registers.
|
|
; Registers: A6 Zone
|
|
; A5 Local variables and buffer on stack
|
|
; A4 Handle of migrating block
|
|
; A2 Ptr to any free space
|
|
; A1 Ptr to moved block
|
|
;-------------------------------------------------------------------------------
|
|
|
|
@MNSFinish
|
|
SUB.L #blkData32,A1 ; grab 12 more bytes for block header <v1.1>
|
|
SUB.L #blkData32,ZCBFree(A6) ; *** this is a fix *** <v1.1>
|
|
MOVE.L A1,D0 ; figure free space left
|
|
SUB.L A2,D0
|
|
BEQ.S @11 ; br if we used it all
|
|
Clr.L TagBC32(A2) ; create a free block <v1.1>
|
|
MOVE.L D0,BlkSize32(A2) ; set remaining size <v1.1>
|
|
|
|
@11 MOVE.L A1,A2 ; point A2 to our new block
|
|
MOVE.L OldTagBC32(A5),(A2)+ ; TagBC
|
|
Move.L SvBlkSize32(A5),(A2)+ ; block size <v1.1>
|
|
MOVE.L A4,A0
|
|
SUB.L A6,A0 ; form handle offset
|
|
MOVE.L A0,(A2)+ ; Handle
|
|
|
|
MOVE.L A2,(A4) ; update master pointer with ptr to text of block
|
|
MOVE.L SavedZone32(A5),theZone ; and theZone <13Jan86 JTC>
|
|
MOVEQ #noErr,D0 ; ok!
|
|
|
|
ADD.L MyBufSize32(A5),SP ; recover buffer space
|
|
ADD.W #LclVarSize32,SP ; and locals space
|
|
MHHExit
|
|
Move.L vMMEpilogue,-(SP) ; get vector to MMEpilogue <v1.9>
|
|
RTS ; jump to it <v1.9>
|
|
|
|
;-------------------------------------------------------------------------------------------
|
|
; Utility FUNCTION FindRgnTop( zone : Ptr; localVars : Ptr; hand : Handle; EOZ : Ptr ) : Ptr;
|
|
;
|
|
; Return a pointer to the last block beyond the handle in question, but before the next locked
|
|
; or nonrelocatable or trailing block.
|
|
;
|
|
; Arguments: A6 = heap zone
|
|
; A5 = ptr to local vars and buffer on stack
|
|
; A4 = handle in migration
|
|
; A3 = last block in zone
|
|
;
|
|
; Results: A2 = ptr to 'last' block
|
|
;
|
|
; Registers: D0-D1/A0-A1
|
|
;
|
|
; Called at labels: moveRest (just before), goCont, moveNxtSeg
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
a24FindRgnTop MOVE.L (A4),D0 ; deref handle
|
|
BEQ.S @10 ; br if nil (disposed)
|
|
|
|
AND.L Lo3Bytes,D0 ; get rid of lock, purge, etc. bits
|
|
MOVE.L D0,A0
|
|
SUBQ #BlkData24,A0 ; start scan from here for speed
|
|
BRA.S @FRTLoop
|
|
|
|
@10 LEA HeapData(A6),A0 ; scan from start of zone
|
|
|
|
@FRTLoop MOVE.L TagBC24(A0),D1 ; block header
|
|
AND.L Lo3Bytes,D1 ; isolate the block size
|
|
TST.B TagBC24(A0) ; ck for Free, Rel, or NRel
|
|
BEQ.S @1 ; br if Free
|
|
BPL.S @2 ; br if NReloc (end of region)
|
|
MOVE.L Handle24(A0),A1 ; handle offset
|
|
ADD.L A6,A1 ; master ptr
|
|
TST.B (A1) ; locked Rel?
|
|
BMI.S @2 ; br if Locked (end of region)
|
|
|
|
@1 MOVE.L A0,A2 ; remember this as the last block
|
|
ADD.L D1,A0 ; point to next block
|
|
CMP.L A3,A0 ; reached the end of zone?
|
|
BCS.S @FRTLoop ; loop if not (fruitloops)
|
|
BRA.S @3 ; otherwise exit (must be in last rgn)
|
|
|
|
@2 CMP.L InRgnPtr24(A5),A0 ; are we in the correct region?
|
|
BLS.S @1 ; keep looking, if not
|
|
|
|
@3 RTS
|
|
|
|
;
|
|
; ------- 32 bit version
|
|
;
|
|
a32FindRgnTop MOVE.L (A4),D0 ; deref handle
|
|
BEQ.S @10 ; br if nil (disposed)
|
|
|
|
MOVE.L D0,A0 ; <v1.1>
|
|
SUB.L #BlkData32,A0 ; start scan from here for speed <v1.1>
|
|
BRA.S @FRTLoop
|
|
|
|
@10 LEA HeapData(A6),A0 ; scan from start of zone
|
|
|
|
@FRTLoop
|
|
MOVE.L BlkSize32(A0),D1 ; block size <v1.1>
|
|
TST.B TagBC32(A0) ; ck for Free, Rel, or NRel
|
|
BEQ.S @1 ; br if Free
|
|
BPL.S @2 ; br if NReloc (end of region)
|
|
Tst.B MPtag32(A0) ; locked Rel? <v1.1>
|
|
BMI.S @2 ; br if Locked (end of region)
|
|
|
|
@1 MOVE.L A0,A2 ; remember this as the last block
|
|
ADD.L D1,A0 ; point to next block
|
|
CMP.L A3,A0 ; reached the end of zone?
|
|
BCS.S @FRTLoop ; loop if not (fruitloops)
|
|
BRA.S @3 ; otherwise exit (must be in last rgn)
|
|
|
|
@2 CMP.L InRgnPtr32(A5),A0 ; are we in the correct region?
|
|
BLS.S @1 ; keep looking, if not
|
|
|
|
@3 RTS
|
|
|
|
;-------------------------------------------------------------------------------------------
|
|
; Utility PROCEDURE FilStkBuf( VAR curDataSize : LongInt; zone : Ptr; localVars : Ptr; hand : Handle; );
|
|
;
|
|
; Fill stack buffer with MIN(MyBufSize,HandleSize) bytes from block in migration. If moving entire
|
|
; handle, just empty that handle on completion. Otherwise, adjust the size of the depleted handle and
|
|
; create a new free block.
|
|
;
|
|
; Arguments: D7 curDataSize = amount left to move
|
|
; A6 zone = heap zone
|
|
; A5 localVars = ptr to stack vars and buffer
|
|
; A4 hand = handle to block being moved
|
|
;
|
|
; Results: D7 updated curDataSize
|
|
;
|
|
; Called after labels: goCont, moveNxtSeg
|
|
;
|
|
; Registers: D0-D2/A0-A1
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
a24FillStkBuf
|
|
MOVE.L MyBufSize24(A5),D0 ; figure MIN(MyBufSize,HandleSize)
|
|
CMP.L D7,D0 ; D7 is current data size of blk we're moving
|
|
BLT.S @1
|
|
MOVE.L D7,D0 ; use current size if <= buffer size
|
|
@1
|
|
LEA MyStkBuf24(A5),A1 ; destination
|
|
MOVE.L (A4),A0 ; BlkData pointer
|
|
ADD.L D7,A0 ; end of block pointer
|
|
SUB.L D0,A0 ; source (move last D0 bytes of blk)
|
|
SUB.L D0,D7 ; adjust current size
|
|
MOVE.L D0,MyBufCnt24(A5) ; set up buffer count
|
|
MOVE.L D0,D1 ; and save in D1 (size of new free blk)
|
|
_BlockMove
|
|
|
|
MOVE.L A4,A0 ; HBlk
|
|
MOVE.L D7,D0 ; remaining bytes in block
|
|
BNE.S @2 ; br if space left
|
|
_EmptyHandle ; get rid of entire blk when 0 (updates freemem)
|
|
RTS
|
|
|
|
; We still haven't emptied the block, so create a smaller block with data size
|
|
; D0=D7 and a free block of size D1 (the amount we moved). Note that by virtue
|
|
; of this style of block truncation, the physical/logical size fudge factor is
|
|
; guaranteed to be zero.
|
|
@2
|
|
MOVE.L (A0),A0 ; ptr to data portion of what's left
|
|
SUBQ #BlkData24,A0 ; ptr to block that's left
|
|
MOVE.B TagBC24(A0),D2 ; save Tag
|
|
ADDQ.L #BlkData24,D0 ; new physical block size
|
|
MOVE.L D0,TagBC24(A0) ; set BC
|
|
MOVE.B D2,TagBC24(A0) ; and restore tag
|
|
|
|
ADD.L D0,A0 ; point to new free block
|
|
MOVE.L D1,TagBC24(A0) ; set Tag=free, BC
|
|
ADD.L D1, ZCBFree(A6) ; update free count.
|
|
RTS
|
|
|
|
; 32 bit version
|
|
a32FillStkBuf
|
|
MOVE.L MyBufSize32(A5),D0 ; figure MIN(MyBufSize,HandleSize)
|
|
CMP.L D7,D0 ; D7 is current data size of blk we're moving
|
|
BLT.S @1
|
|
MOVE.L D7,D0 ; use current size if <= buffer size
|
|
@1
|
|
LEA MyStkBuf32(A5),A1 ; destination
|
|
MOVE.L (A4),A0 ; BlkData pointer
|
|
ADD.L D7,A0 ; end of block pointer
|
|
SUB.L D0,A0 ; source (move last D0 bytes of blk)
|
|
SUB.L D0,D7 ; adjust current size
|
|
MOVE.L D0,MyBufCnt32(A5) ; set up buffer count
|
|
MOVE.L D0,D1 ; and save in D1 (size of new free blk)
|
|
_BlockMove
|
|
|
|
MOVE.L A4,A0 ; HBlk
|
|
MOVE.L D7,D0 ; remaining bytes in block
|
|
BNE.S @2 ; br if space left
|
|
_EmptyHandle ; get rid of entire blk when 0 (updates freemem)
|
|
RTS
|
|
|
|
; We still haven't emptied the block, so create a smaller block with data size
|
|
; D0=D7 and a free block of size D1 (the amount we moved). Note that by virtue
|
|
; of this style of block truncation, the physical/logical size fudge factor is
|
|
; guaranteed to be zero.
|
|
@2
|
|
MOVE.L (A0),A0 ; ptr to data portion of what's left
|
|
SUB.L #BlkData32,A0 ; ptr to block that's left <v1.1>
|
|
ADD.L #BlkData32,D0 ; new physical block size <v1.1>
|
|
MOVE.L D0,BlkSize32(A0) ; set size <v1.1>
|
|
ADD.L D0,A0 ; point to new free block <v1.1>
|
|
Clr.L tagBC32(A0) ; set as free block <v1.1>
|
|
MOVE.L D1,BlkSize32(A0) ; set free block size <v1.1>
|
|
ADD.L D1, ZCBFree(A6) ; update free count.
|
|
RTS
|
|
|
|
;-------------------------------------------------------------------------------------------
|
|
; Utility PROCEDURE DoCompact;
|
|
;
|
|
; Arguments: none
|
|
; Results: thoroughly compacted heap
|
|
; Registers: D0
|
|
; Called at labels: goCont, moveNxtSeg
|
|
;-------------------------------------------------------------------------------------------
|
|
|
|
DoCompact
|
|
MOVE.L RealMemTop,D0 ; compact to push all the <v1.9>
|
|
_CompactMem ; free space to the top . . .
|
|
RTS
|
|
|
|
|
|
|
|
END ; *********** comment this out for RAMTest
|
|
|