//+--------------------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1992 - 1994. // // File: heap.hxx // // Contents: Heap code headers // // Classes: CHeap // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // 10-Apr095 HenryLee Added global LUID // 10-May-95 KentCe Defer Heap Destruction to the last // process detach. // //---------------------------------------------------------------------------- #ifndef __HEAP_HXX__ #define __HEAP_HXX__ #include #include #include #include #include #ifdef COORD #include #endif //Space to reserve for heap. const ULONG MINHEAPGROWTH = 4096; const ULONG INITIALHEAPSIZE = 16384; #ifdef MULTIHEAP #include class CPerContext; #endif //+------------------------------------------------------------------------- // // Class: CLockDfMutex // // Purpose: Simple class to guarantee that a DfMutex is unlocked // // History: 29-Apr-95 DonnaLi Created // //-------------------------------------------------------------------------- class CLockDfMutex { public: CLockDfMutex(CDfMutex& dmtx); ~CLockDfMutex(void); private: CDfMutex& _dmtx; }; //+------------------------------------------------------------------------- // // Member: CLockDfMutex::CLockDfMutex // // Synopsis: Get mutex // // Arguments: [dmtx] -- mutex to get // // History: 29-Apr-95 DonnaLi Created // //-------------------------------------------------------------------------- inline CLockDfMutex::CLockDfMutex(CDfMutex& dmtx) : _dmtx(dmtx) { _dmtx.Take(DFM_TIMEOUT); } //+------------------------------------------------------------------------- // // Member: CLockDfMutex::~CLockDfMutex // // Synopsis: Release the mutex // // History: 29-Apr-95 DonnaLi Created // //-------------------------------------------------------------------------- inline CLockDfMutex::~CLockDfMutex(void) { _dmtx.Release(); } // // Take advantage of Windows 95 Shared Heap. // #if !defined(_CHICAGO_) //+--------------------------------------------------------------------------- // // Class: CBlockPreHeader // // Purpose: Required header fields for a block // // History: 29-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- class CBlockPreHeader { protected: ULONG _ulSize; //Size of block BOOL _fFree; //TRUE if block is free }; //+--------------------------------------------------------------------------- // // Class: CBlockHeader // // Purpose: Fields required for free blocks but overwritten for // allocated blocks. // // History: 29-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- class CBlockHeader: public CBlockPreHeader { public: inline ULONG GetSize(void) const; inline BOOL IsFree(void) const; inline ULONG GetNext(void) const; inline void SetSize(ULONG ulSize); inline void SetFree(void); inline void ResetFree(void); inline void SetNext(ULONG ulNext); private: ULONG _ulNext; //Pointer to next block }; //+--------------------------------------------------------------------------- // // Member: CBlockHeader::GetSize, public // // Synopsis: Returns the size of the block // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CBlockHeader::GetSize(void) const { return _ulSize; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::IsFree, public // // Synopsis: Returns free state of block // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline BOOL CBlockHeader::IsFree(void) const { return _fFree; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::GetNext, public // // Synopsis: Return next offset // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CBlockHeader::GetNext(void) const { return _ulNext; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::SetSize, public // // Synopsis: Set size of block // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CBlockHeader::SetSize(ULONG ulSize) { _ulSize = ulSize; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::SetFree, public // // Synopsis: Set this block to free // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CBlockHeader::SetFree(void) { _fFree = TRUE; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::ResetFree, public // // Synopsis: Set this block to !free // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CBlockHeader::ResetFree(void) { _fFree = FALSE; } //+--------------------------------------------------------------------------- // // Member: CBlockHeader::SetNext, public // // Synopsis: Set next offset // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CBlockHeader::SetNext(ULONG ulNext) { _ulNext = ulNext; } const ULONG CBLOCKMIN = ((sizeof(CBlockHeader) & 7) ? sizeof(CBlockHeader) + (8 - (sizeof(CBlockHeader) & 7)) : sizeof(CBlockHeader)); //+--------------------------------------------------------------------------- // // Class: CHeapHeader // // Purpose: Header information for shared memory heap // // Interface: // // History: 30-Mar-94 PhilipLa Created // // Notes: The size of this structure must be a multiple of 8 bytes. // //---------------------------------------------------------------------------- class CHeapHeader { public: inline ULONG GetFirstFree(void) const; inline void SetFirstFree(ULONG ulNew); inline BOOL IsCompacted(void) const; inline void SetCompacted(void); inline void ResetCompacted(void); inline void ResetAllocedBlocks(void); inline ULONG IncrementAllocedBlocks(void); inline ULONG DecrementAllocedBlocks(void); inline ULONG GetAllocedBlocks(void); inline DFLUID IncrementLuid(void); inline void ResetLuid(void); #if DBG == 1 ULONG _ulAllocedBytes; ULONG _ulFreeBytes; ULONG _ulFreeBlocks; #endif private: ULONG _ulFirstFree; ULONG _ulAllocedBlocks; BOOL _fIsCompacted; DFLUID _dfLuid; #if DBG == 1 ULONG ulPad; #endif }; //+--------------------------------------------------------------------------- // // Member: CHeapHeader::GetFirstFree, public // // Synopsis: Return first free information // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CHeapHeader::GetFirstFree(void) const { return _ulFirstFree; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::SetFirstFree, public // // Synopsis: Set first free information // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CHeapHeader::SetFirstFree(ULONG ulNew) { _ulFirstFree = ulNew; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::IsCompacted, public // // Synopsis: Return TRUE if heap is compacted // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline BOOL CHeapHeader::IsCompacted(void) const { return _fIsCompacted; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::SetCompacted, public // // Synopsis: Set compacted bit // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CHeapHeader::SetCompacted(void) { _fIsCompacted = TRUE; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::ResetCompacted, public // // Synopsis: Reset compacted bit // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CHeapHeader::ResetCompacted(void) { _fIsCompacted = FALSE; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::IncrementLuid, public // // Synopsis: Increment the global LUID // // History: 06-Apr-95 HenryLee Created // //---------------------------------------------------------------------------- inline ULONG CHeapHeader::IncrementLuid() { return ++_dfLuid; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::ResetLuid, public // // Synopsis: Increment the global LUID // // History: 06-Apr-95 HenryLee Created // //---------------------------------------------------------------------------- inline void CHeapHeader::ResetLuid() { _dfLuid = LUID_BASE; // some LUIDs are reserved } #else // define(_CHICAGO_) extern HANDLE gs_hSharedHeap; // hSharedHeap Handle for Win95. extern DFLUID gs_dfluid; // shared docfile LUID #endif // !define(_CHICAGO_) //+--------------------------------------------------------------------------- // // Class: CSmAllocator // // Purpose: Shared memory heap implementation // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // //---------------------------------------------------------------------------- class CSmAllocator: public IMalloc { public: inline CSmAllocator(); inline ~CSmAllocator(); STDMETHOD_(ULONG,AddRef) ( void ); STDMETHOD_(ULONG,Release) ( void ); STDMETHOD(QueryInterface) ( REFIID riid, void ** ppv ); STDMETHOD_(void*,Alloc) ( ULONG cb ); STDMETHOD_(void *,Realloc) ( void *pvCurrent, ULONG cbNeeded ); STDMETHOD_(void,Free) ( void *pvMemToFree ); STDMETHOD_(ULONG,GetSize) ( void * pv ); STDMETHOD_(void,HeapMinimize) ( void ); STDMETHOD_(int,DidAlloc) ( void * pv ); inline SCODE Sync(void); inline DFLUID IncrementLuid(void); #if !defined(MULTIHEAP) SCODE Init ( LPWSTR pszName ); #else SCODE Init ( ULONG ulHeapName, BOOL fUnmarshal ); #endif inline void * GetBase(void); // This function is equivalent to Free above, except that is does // not attempt to first acquire the mutex. It should be used ONLY // when the calling function guarantees to already have the mutex. void FreeNoMutex (void * pv); #if !defined(MULTIHEAP) inline CDfMutex * GetMutex (void); #endif #ifdef MULTIHEAP void SetState (CSharedMemoryBlock *psmb, BYTE * pbBase, ULONG ulHeapName, CPerContext ** ppcPrev, CPerContext *ppcOwner); void GetState (CSharedMemoryBlock **ppsmb, BYTE ** ppbBase, ULONG *pulHeapName); inline const ULONG GetHeapName (); SCODE Uninit (); inline const ULONG GetHeapSize () { return _cbSize; }; #endif private: inline void DoFree (void *pv); #if !defined(MULTIHEAP) CDfMutex _dmtx; #endif // // Take advantage of Windows 95 Shared Heap. // #if !defined(_CHICAGO_) CBlockHeader * FindBlock(ULONG cb, CBlockHeader **ppbhPrev); inline CHeapHeader *GetHeader(void); inline CBlockHeader * GetAddress(ULONG ulOffset) const; inline ULONG GetOffset(CBlockHeader *pbh) const; inline SCODE Reset(void); #if DBG == 1 void PrintFreeBlocks(void); #ifdef MULTIHEAP void PrintAllocatedBlocks(void); #endif #endif #ifdef MULTIHEAP CSharedMemoryBlock *_psmb; BYTE *_pbBase; ULONG _cbSize; CPerContext * _ppcOwner; ULONG _ulHeapName; ULONG _cRefs; // yes, this object has a lifetime now #else CSharedMemoryBlock _smb; BYTE *_pbBase; ULONG _cbSize; #endif // MULTIHEAP #else // defined(_CHICAGO_) HANDLE m_hSharedHeap; #endif // !defined(_CHICAGO_) }; #ifdef MULTIHEAP extern CSmAllocator g_SmAllocator; // single-threaded allocator extern CSharedMemoryBlock g_smb; //performance optimization extern ULONG g_ulHeapName; extern CSmAllocator& GetTlsSmAllocator(); // all other threads #define g_smAllocator (GetTlsSmAllocator()) //+--------------------------------------------------------------------------- // // Class: CErrorSmAllocator // // Synopsis: returned by GetTlsSmAllocator for out of memory failures // // History: 02-May-1996 HenryLee Created // //---------------------------------------------------------------------------- class CErrorSmAllocator : public CSmAllocator { public: STDMETHOD_(void*,Alloc) (ULONG cb) { return NULL; }; STDMETHOD_(void*,Realloc) (void* pv, ULONG cb) { return NULL; }; STDMETHOD_(void,Free) (void *pv) { return; }; STDMETHOD_(ULONG,GetSize) (void *pv) { return 0; }; STDMETHOD_(void,HeapMinimize) () { return; }; STDMETHOD_(int,DidAlloc) (void *pv) { return FALSE; }; SCODE Init (ULONG ul, BOOL f) { return STG_E_INSUFFICIENTMEMORY; }; SCODE Sync (void) { return STG_E_INSUFFICIENTMEMORY; }; }; extern CErrorSmAllocator g_ErrorSmAllocator; #else extern CSmAllocator g_smAllocator; #endif extern CRITICAL_SECTION g_csScratchBuffer; //+--------------------------------------------------------------------------- // // Member: CSmAllocator::CSmAllocator, public // // Synopsis: Constructor // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // //---------------------------------------------------------------------------- inline CSmAllocator::CSmAllocator(void) #if !defined(_CHICAGO_) #ifdef MULTIHEAP : _cbSize(0), _pbBase(NULL), _cRefs(1), _ulHeapName(0), _psmb(NULL), _ppcOwner(NULL) #else : _cbSize(0) #endif // MULTIHEAP #else : m_hSharedHeap(NULL) #endif { #if !defined(MULTIHEAP) InitializeCriticalSection(&g_csScratchBuffer); #ifdef COORD InitializeCriticalSection(&g_csResourceList); #endif #endif // MULTIHEAP } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::~CSmAllocator, public // // Synopsis: Destructor // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // 10-May-95 KentCe Defer Heap Destruction to the last // process detach. // //---------------------------------------------------------------------------- inline CSmAllocator::~CSmAllocator(void) { #if !defined(MULTIHEAP) DeleteCriticalSection(&g_csScratchBuffer); #ifdef COORD DeleteCriticalSection(&g_csResourceList); #endif #endif // MULTIHEAP } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::Sync, public // // Synopsis: Sync memory to global state. // // Arguments: None. // // Returns: Appropriate status code // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // //---------------------------------------------------------------------------- inline SCODE CSmAllocator::Sync(void) { SCODE sc = S_OK; #if !defined(_CHICAGO_) #if !defined(MULTIHEAP) if (!_smb.IsSynced()) { CLockDfMutex lckdmtx(_dmtx); sc = _smb.Sync(); _cbSize = _smb.GetSize(); } #else if (!_psmb->IsSynced()) { sc = _psmb->Sync(); } _cbSize = _psmb->GetSize(); #endif #endif return sc; } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::IncrementLuid, public // // Synopsis: Increments the global LUID // // Arguments: None. // // Returns: Appropriate status code // // History: 06-Apr-95 HenryLee Created //---------------------------------------------------------------------------- inline DFLUID CSmAllocator::IncrementLuid(void) { #if !defined(MULTIHEAP) CLockDfMutex lckdmx(_dmtx); #endif #ifdef _CHICAGO_ // // On Chicago, we merely increment the globally available // LUID to the next value. // return ++gs_dfluid; #else return GetHeader()->IncrementLuid(); #endif } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::GetBase, public // // Synopsis: Return pointer to base of heap // // History: 29-Mar-94 PhilipLa Created // 05-Feb-95 KentCe Use Win95 Shared Heap. // //---------------------------------------------------------------------------- inline void * CSmAllocator::GetBase(void) { #if defined(_CHICAGO_) return NULL; #else return _pbBase; #endif } #if !defined(MULTIHEAP) //+--------------------------------------------------------------------------- // // Member: CSmAllocator::GetMutex, public // // Synopsis: Return a pointer to the Mutex // // History: 19-Jul-95 SusiA Created // //---------------------------------------------------------------------------- inline CDfMutex * CSmAllocator::GetMutex(void) { return &_dmtx; } #endif // // Take advantage of Windows 95 Shared Heap. // #if !defined(_CHICAGO_) //+--------------------------------------------------------------------------- // // Member: CSmAllocator::GetAddress, private // // Synopsis: Returns an address given an offset from the base // // Arguments: [ulOffset] -- Offset to convert to address // // History: 29-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline CBlockHeader * CSmAllocator::GetAddress(ULONG ulOffset) const { return (ulOffset == 0) ? NULL : (CBlockHeader *)(_pbBase + ulOffset); } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::GetOffset // // Synopsis: Returns a byte offset from the base given a pointer // // Arguments: [pbh] -- Pointer to convert to offset // // History: 29-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CSmAllocator::GetOffset(CBlockHeader *pbh) const { memAssert((BYTE *)pbh >= _pbBase && (BYTE*)pbh < _pbBase + _cbSize); return (ULONG)pbh - (ULONG)_pbBase; } //+--------------------------------------------------------------------------- // // Member: CSmAllocator::GetHeader, private // // Synopsis: Return pointer to CHeapHeader for this heap // // History: 30-Mar-94 PhilipLa Created // //---------------------------------------------------------------------------- inline CHeapHeader * CSmAllocator::GetHeader(void) { return (CHeapHeader *)_pbBase; } #ifdef MULTIHEAP //+------------------------------------------------------------------------- // // Member: CSmAllocator::HeapName, public // // Synopsis: Return the luid part of the shared heap name // // History: 30-Nov-95 HenryLee Created // //-------------------------------------------------------------------------- inline const ULONG CSmAllocator::GetHeapName() { return _ulHeapName; } #endif //+--------------------------------------------------------------------------- // // Member: CHeapHeader::ResetAllocedBlocks, public // // Synopsis: Reset the allocated block counter // // History: 04-Apr-94 PhilipLa Created // //---------------------------------------------------------------------------- inline void CHeapHeader::ResetAllocedBlocks(void) { _ulAllocedBlocks = 0; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::IncrementAllocedBlocks, public // // Synopsis: Increment the allocated block count // // History: 04-Apr-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CHeapHeader::IncrementAllocedBlocks(void) { return ++_ulAllocedBlocks; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::DecrementAllocedBlocks, public // // Synopsis: Decrement the allocated block count // // History: 04-Apr-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CHeapHeader::DecrementAllocedBlocks(void) { return --_ulAllocedBlocks; } //+--------------------------------------------------------------------------- // // Member: CHeapHeader::GetAllocedBlocks, public // // Synopsis: Return the allocated block count // // History: 04-Apr-94 PhilipLa Created // //---------------------------------------------------------------------------- inline ULONG CHeapHeader::GetAllocedBlocks(void) { return _ulAllocedBlocks; } #endif // !defined(_CHICAGO_) #endif // #ifndef __HEAP_HXX__