//+------------------------------------------------------------------------- // // Microsoft Windows // Copyright (C) Microsoft Corporation, 1992 - 1993. // // File: Mem.CXX // // Contents: Memory tracking code // // Classes: CMemAlloc // // Functions: DfCreateSharedAllocator // DfPrintAllocs // DfGetMemAlloced // // History: 17-May-93 AlexT Created // //-------------------------------------------------------------------------- #include #pragma hdrstop #if defined(WIN32) #include #include #include // This global variable holds the base address of the shared memory region. // It is required by the based pointer stuff and is accessed directly. // Initialization is a side effect of calling DfCreateSharedAllocator or // DfSyncSharedMemory #ifdef MULTIHEAP //__declspec(thread) void *DFBASEPTR = NULL; #else void *DFBASEPTR = NULL; #endif #endif #if DBG == 1 #include #ifdef MEMTRACK #ifdef WIN32 // Multithread protection for allocation list CStaticDfMutex _sdmtxAllocs(TSTR("DfAllocList")); #define TAKE_ALLOCS_MUTEX olVerSucc(_sdmtxAllocs.Take(INFINITE)) #define RELEASE_ALLOCS_MUTEX _sdmtxAllocs.Release() #else #define TAKE_ALLOCS_MUTEX #define RELEASE_ALLOCS_MUTEX #endif #define GET_ALLOC_LIST_HEAD ((CMemAlloc *)DfGetResLimit(DBRI_ALLOC_LIST)) #define SET_ALLOC_LIST_HEAD(pma) DfSetResLimit(DBRI_ALLOC_LIST, (LONG)(pma)) #endif // MEMTRACK #define DEB_MEMORY 0x01000000 #define DEB_LEAKS 0x01100000 const int NEWMEM = 0xDEDE; const int OLDMEM = 0xEDED; //+-------------------------------------------------------------- // // Class: CMemAlloc (ma) // // Purpose: Tracks memory allocations // // Interface: See below // // History: 08-Jul-92 DrewB Created // 17-May-93 AlexT Add idContext // //--------------------------------------------------------------- class CMemAlloc { public: void *pvCaller; void *pvMem; ULONG cbSize; ContextId idContext; #ifdef MEMTRACK CMemAlloc *pmaPrev, *pmaNext; #endif }; //+--------------------------------------------------------------------------- // // Function: AddAlloc, private // // Synopsis: Puts an allocation into the allocation list // // Arguments: [pma] - Allocation descriptor // [pvCaller] - Allocator // [cbSize] - Real size // [pvMem] - Memory block // [cid] - Context ID // // History: 11-Jan-94 DrewB Created // //---------------------------------------------------------------------------- #if DBG == 1 static void AddAlloc(CMemAlloc *pma, void *pvCaller, ULONG cbSize, void *pvMem, ContextId cid) { pma->pvCaller = pvCaller; pma->cbSize = cbSize; pma->pvMem = pvMem; pma->idContext = cid; #ifdef MEMTRACK TAKE_ALLOCS_MUTEX; pma->pmaNext = GET_ALLOC_LIST_HEAD; if (pma->pmaNext) pma->pmaNext->pmaPrev = pma; pma->pmaPrev = NULL; SET_ALLOC_LIST_HEAD(pma); RELEASE_ALLOCS_MUTEX; ModifyResLimit(DBRQ_MEMORY_ALLOCATED, (LONG)cbSize); olDebugOut((DEB_MEMORY, "%s alloced %p:%lu, total %ld\n", cid != 0 ? "Task" : "Shrd", pvMem, pma->cbSize, DfGetResLimit(DBRQ_MEMORY_ALLOCATED))); #endif } #endif //+--------------------------------------------------------------------------- // // Function: RemoveAlloc, private // // Synopsis: Takes an allocation out of the allocation list // // Arguments: [pma] - Allocation descriptor // [pvMem] - Real allocation // [cid] - Context ID // // History: 11-Jan-94 DrewB Created // //---------------------------------------------------------------------------- #if DBG == 1 static void RemoveAlloc(CMemAlloc *pma, void *pvMem, ContextId cid) { #ifdef MEMTRACK olAssert(pma->pvMem == pvMem && aMsg("Address mismatch")); olAssert(pma->idContext == cid && aMsg("Context mismatch")); TAKE_ALLOCS_MUTEX; if (pma->pmaPrev) { pma->pmaPrev->pmaNext = pma->pmaNext; } else { SET_ALLOC_LIST_HEAD(pma->pmaNext); } if (pma->pmaNext) { pma->pmaNext->pmaPrev = pma->pmaPrev; } RELEASE_ALLOCS_MUTEX; ModifyResLimit(DBR_MEMORY, (LONG)pma->cbSize); ModifyResLimit(DBRQ_MEMORY_ALLOCATED, -(LONG)pma->cbSize); olAssert(DfGetResLimit(DBRQ_MEMORY_ALLOCATED) >= 0); olDebugOut((DEB_MEMORY, "%s freed %p:%lu, total %ld\n", cid != 0 ? "Task" : "Shrd", pma->pvMem, pma->cbSize, DfGetResLimit(DBRQ_MEMORY_ALLOCATED))); #endif } #endif //+--------------------------------------------------------------------------- // // Function: DfGetMemAlloced, private // // Synopsis: Returns the amount of memory currently allocated // // History: 08-Jul-92 DrewB Created // //---------------------------------------------------------------------------- STDAPI_(LONG) DfGetMemAlloced(void) { return DfGetResLimit(DBRQ_MEMORY_ALLOCATED); } //+-------------------------------------------------------------- // // Function: DfPrintAllocs, private // // Synopsis: Walks the allocation list and prints out their info // // History: 08-Jul-92 DrewB Created // //--------------------------------------------------------------- STDAPI_(void) DfPrintAllocs(void) { #ifdef MEMTRACK CMemAlloc *pma; olDebugOut((DEB_ITRACE, "In DfPrintAllocs()\n")); TAKE_ALLOCS_MUTEX; pma = GET_ALLOC_LIST_HEAD; while (pma != NULL) { olDebugOut((DEB_LEAKS, "Alloc %s: %p alloc %p:%4lu bytes\n", pma->idContext ? "LOCAL" : "SHARED", pma->pvCaller, pma->pvMem, pma->cbSize)); pma = pma->pmaNext; } RELEASE_ALLOCS_MUTEX; olDebugOut((DEB_ITRACE, "Out DfPrintAllocs\n")); #endif } #endif #ifdef MULTIHEAP CSmAllocator g_SmAllocator; // optimization for single threaded case CErrorSmAllocator g_ErrorSmAllocator; IMalloc *g_pTaskAllocator = 0; CSharedMemoryBlock g_smb; // optimize single threaded case ULONG g_ulHeapName = 0; // name of the above heap TEB * g_pteb = NtCurrentTeb(); CSmAllocator& GetTlsSmAllocator() { HRESULT hr; COleTls otls(hr); // even for main thread, we need to initialize TLS memAssert (SUCCEEDED(hr) && "Error initializing TLS"); if (g_pteb == NtCurrentTeb()) { // return the static global allocator for main thread // DoThreadSpecificCleanup does not deallocate this allocator return g_SmAllocator; } if (otls->pSmAllocator == NULL) { if ((otls->pSmAllocator = new CSmAllocator()) == NULL) otls->pSmAllocator = &g_ErrorSmAllocator; // DoThreadSpecificCleanup will deallocate this when thread goes away } return *(otls->pSmAllocator); } // // This initialization used to be done in the global allocator // which is now a per thread allocator // class CResourceCriticalSection { public: CResourceCriticalSection () { #if WIN32 == 100 _nts = RtlInitializeCriticalSection(&g_csScratchBuffer); #else InitializeCriticalSection(&g_csScratchBuffer); #endif } ~CResourceCriticalSection () { #if WIN32 == 100 if (NT_SUCCESS(_nts)) #endif DeleteCriticalSection(&g_csScratchBuffer); } #if WIN32 == 100 HRESULT GetInitializationError () { return NtStatusToScode(_nts); } private: NTSTATUS _nts; #endif }; CResourceCriticalSection g_ResourceCriticalSection; #else CSmAllocator g_smAllocator; #endif // MULTIHEAP //+------------------------------------------------------------------------- // // Member: CMallocBased::operator new, public // // Synopsis: Overridden allocator // // Effects: Allocates memory from given allocator // // Arguments: [size] -- byte count to allocate // [pMalloc] -- allocator // // Returns: memory block address // // Algorithm: // // History: 21-May-93 AlexT Created // // Notes: // //-------------------------------------------------------------------------- void *CMallocBased::operator new(size_t size, IMalloc * const pMalloc) { #ifndef WIN32 olAssert(DfGetResLimit(DBRQ_MEMORY_ALLOCATED) >= 0); #endif olAssert(size > 0); #if DBG==1 if (SimulateFailure(DBF_MEMORY)) { return(NULL); } if (!HaveResource(DBR_MEMORY, (LONG)size)) { // Artificial limit exceeded so force failure return NULL; } #endif void *pv = g_smAllocator.Alloc( #if DBG==1 sizeof(CMemAlloc) + #endif size); if (pv == NULL) { #if DBG==1 ModifyResLimit(DBR_MEMORY, (LONG)size); #endif return NULL; } #if DBG==1 memset(pv, NEWMEM, sizeof(CMemAlloc) + size); #endif #if DBG==1 // Put debug info in buffer // Note: This assumes CMemAlloc will end up properly aligned CMemAlloc *pma = (CMemAlloc *) pv; pv = (void *) ((CMemAlloc *) pv + 1); #if defined(_X86_) PVOID ra = *(((void **)&size)-1); #else PVOID ra = _ReturnAddress(); #endif AddAlloc(pma, ra, (ULONG) size, pv, 0); #endif return pv; } //+------------------------------------------------------------------------- // // Member: CMallocBased::operator delete // // Synopsis: Overridden deallocator // // Effects: Frees memory block // // Arguments: [pv] -- memory block address // // Algorithm: // // History: 21-May-93 AlexT Created // // Notes: // //-------------------------------------------------------------------------- void CMallocBased::operator delete(void *pv) { if (pv == NULL) return; #if DBG==1 // Assumes ma ends up properly aligned CMemAlloc *pma = (CMemAlloc *) pv; pma--; RemoveAlloc(pma, pv, 0); pv = (void *) pma; memset(pv, OLDMEM, (size_t) pma->cbSize); #endif g_smAllocator.Free(pv); } //+------------------------------------------------------------------------- // // Member: CMallocBased::deleteNoMutex // // Synopsis: deallocator function without Mutex // // Effects: Frees memory block // // Arguments: [pv] -- memory block address // Algorithm: // // History: 19-Jul-95 SusiA Created // // Notes: // //-------------------------------------------------------------------------- void CMallocBased::deleteNoMutex(void *pv) { if (pv == NULL) return; #if DBG==1 // Assumes ma ends up properly aligned CMemAlloc *pma = (CMemAlloc *) pv; pma--; RemoveAlloc(pma, pv, 0); pv = (void *) pma; memset(pv, OLDMEM, (size_t) pma->cbSize); #endif g_smAllocator.FreeNoMutex(pv); } #if !defined(WIN32) && DBG==1 //+------------------------------------------------------------------------- // // Class: CSharedMalloc // // Purpose: Track shared allocators // // Interface: IMalloc // // History: 28-May-93 AlexT Created // // Notes: This is only for builds that use CoCreateStandardMalloc // (which is non-WIN32 builds for now). We inherit from // CMallocBased to pick up memory tracking. // //-------------------------------------------------------------------------- class CSharedMalloc : public IMalloc, public CMallocBased { public: CSharedMalloc(IMalloc *pMalloc); // *** IUnknown methods *** STDMETHOD(QueryInterface) (THIS_ REFIID riid, LPVOID FAR* ppvObj); STDMETHOD_(ULONG,AddRef) (THIS); STDMETHOD_(ULONG,Release) (THIS); // *** IMalloc methods *** STDMETHOD_(void FAR*, Alloc) (THIS_ SIZE_T cb); STDMETHOD_(void FAR*, Realloc) (THIS_ void FAR* pv, SIZE_T cb); STDMETHOD_(void, Free) (THIS_ void FAR* pv); STDMETHOD_(SIZE_T, GetSize) (THIS_ void FAR* pv); STDMETHOD_(int, DidAlloc) (THIS_ void FAR* pv); STDMETHOD_(void, HeapMinimize) (THIS); private: IMalloc * const _pMalloc; }; CSharedMalloc::CSharedMalloc(IMalloc *pMalloc) : _pMalloc(pMalloc) { _pMalloc->AddRef(); } STDMETHODIMP CSharedMalloc::QueryInterface(THIS_ REFIID riid, LPVOID FAR* ppvObj) { olAssert(!aMsg("CSharedMalloc::QueryInterface unsupported")); return(ResultFromScode(E_UNEXPECTED)); } STDMETHODIMP_(ULONG) CSharedMalloc::AddRef (THIS) { return(_pMalloc->AddRef()); } STDMETHODIMP_(ULONG) CSharedMalloc::Release (THIS) { ULONG cRef = _pMalloc->Release(); if (cRef == 0) delete this; return (ULONG) cRef; } STDMETHODIMP_(void FAR*) CSharedMalloc::Alloc (THIS_ SIZE_T cb) { return _pMalloc->Alloc(cb); } STDMETHODIMP_(void FAR*) CSharedMalloc::Realloc (THIS_ void FAR* pv, SIZE_T cb) { olAssert(!aMsg("CSharedMalloc::Realloc unsupported")); return(NULL); } STDMETHODIMP_(void) CSharedMalloc::Free (THIS_ void FAR* pv) { _pMalloc->Free(pv); } STDMETHODIMP_(SIZE_T) CSharedMalloc::GetSize (THIS_ void FAR* pv) { olAssert(!aMsg("CSharedMalloc::GetSize unsupported")); return(0); } STDMETHODIMP_(int) CSharedMalloc::DidAlloc (THIS_ void FAR* pv) { olAssert(!aMsg("CSharedMalloc::DidAlloc unsupported")); return(TRUE); } STDMETHODIMP_(void) CSharedMalloc::HeapMinimize (THIS) { olAssert(!aMsg("CSharedMalloc::HeapMinimize unsupported")); } #endif #if !defined(MULTIHEAP) COleStaticMutexSem mxsInitSm; BOOL fInitialisedSm = FALSE; //+------------------------------------------------------------------------- // // Function: DfCreateSharedAllocator // // Synopsis: Initialises the shared memory region for this process. // // Arguments: [ppm] -- return address for shared memory IMalloc* // // Returns: status code // // History: 27-May-94 MikeSe Created // // Notes: This routine is called indirectly through DfCreateSharedAllocator, in // such a way that in most circumstances it will be executed // exactly once per process. (Because we overwrite the contents // of DfCreateSharedAllocator if successful). // // However, in order to be multi-thread safe, we need a process // mutex to prevent execution of the initialisation code twice. // // Furthermore, since we are initialising system-wide state // (namely the shared memory region) we must provide cross-process // mutual exclusion over this activity. // //-------------------------------------------------------------------------- HRESULT DfCreateSharedAllocator (IMalloc ** ppm, BOOL fTaskMemory ) { // multi-thread safety mxsInitSm.Request(); HRESULT hr = S_OK; if ( !fInitialisedSm ) { // We need to do the initialisation. Obtain exclusion against // other processes. SECURITY_ATTRIBUTES secattr; secattr.nLength = sizeof(SECURITY_ATTRIBUTES); #ifndef _CHICAGO_ CWorldSecurityDescriptor wsd; secattr.lpSecurityDescriptor = &wsd; #else secattr.lpSecurityDescriptor = NULL; #endif // !_CHICAGO_ secattr.bInheritHandle = FALSE; HANDLE hMutex = CreateMutex ( &secattr, FALSE, TEXT("OleDfSharedMemoryMutex")); if ( hMutex == NULL && GetLastError() == ERROR_ACCESS_DENIED ) hMutex = OpenMutex(SYNCHRONIZE, FALSE, TEXT("OleDfSharedMemoryMutex")); if ( hMutex != NULL ) { WaitForSingleObject ( hMutex, INFINITE ); hr = g_smAllocator.Init ( DOCFILE_SM_NAME ); if ( SUCCEEDED(hr) ) { *ppm = &g_smAllocator; // Also set up base address #ifdef USEBASED DFBASEPTR = g_smAllocator.GetBase(); #endif fInitialisedSm = TRUE; } ReleaseMutex ( hMutex ); } else hr = HRESULT_FROM_WIN32(GetLastError()); } else // fInitialisedSm is TRUE { *ppm = &g_smAllocator; } mxsInitSm.Release(); return hr; } #endif // !defined(MULTIHEAP) #ifdef MULTIHEAP //+------------------------------------------------------------------------- // // Function: DfCreateSharedAllocator // // Synopsis: Initialises a shared memory region for this process. // // Arguments: [ppm] -- return address for shared memory IMalloc* // // Returns: status code // // History: 20-Nov-95 HenryLee Created // // Notes: This routine is called indirectly by DfCreateSharedAllocator // such a way that in most circumstances it will be executed // exactly once per docfile open. // //-------------------------------------------------------------------------- HRESULT DfCreateSharedAllocator (IMalloc ** ppm, BOOL fTaskMemory ) { HRESULT hr = S_OK; #if WIN32 == 100 hr = g_ResourceCriticalSection.GetInitializationError(); if (!SUCCEEDED(hr)) return hr; #endif COleTls otls(hr); // check if TLS is initialized if (!SUCCEEDED(hr)) return hr; CSmAllocator *pMalloc = &g_smAllocator; *ppm = NULL; if ((gs_iSharedHeaps > (DOCFILE_SM_LIMIT / DOCFILE_SM_SIZE)) || fTaskMemory) { // use task allocator if we eat up 1Gig of address space if (g_pTaskAllocator == NULL) { hr = CoGetMalloc (MEMCTX_TASK, &g_pTaskAllocator); if (!SUCCEEDED(hr)) return hr; } // reset the allocator state to initialize it properly pMalloc->SetState (NULL, NULL, NULL, NULL, 0); DFBASEPTR = 0; *ppm = g_pTaskAllocator; } else { LUID luid; // generate a unique name if (AllocateLocallyUniqueId (&luid) == FALSE) return HRESULT_FROM_WIN32(GetLastError()); // reset the allocator state to initialize it properly pMalloc->SetState (NULL, NULL, NULL, NULL, 0); hr = pMalloc->Init ( luid.LowPart, FALSE ); if ( SUCCEEDED(hr) ) { *ppm = pMalloc; DFBASEPTR = pMalloc->GetBase(); pMalloc->AddRef(); } } return hr; } #endif //+--------------------------------------------------------------------------- // // Function: DfSyncSharedMemory, public // // Synopsis: Sync up shared memory // // Returns: Appropriate status code // // History: 08-Apr-94 PhilipLa Created // // Notes: // //---------------------------------------------------------------------------- #ifdef MULTIHEAP SCODE DfSyncSharedMemory(ULONG ulHeapName) #else SCODE DfSyncSharedMemory(void) #endif { // make sure we have initialised HRESULT hr = S_OK; #ifdef MULTIHEAP CSmAllocator *pMalloc = &g_smAllocator; if (ulHeapName != 0) // reopen a shared heap for unmarshaling { // reset the allocator state to initialize it properly pMalloc->SetState (NULL, NULL, NULL, NULL, 0); hr = pMalloc->Init ( ulHeapName, TRUE); DFBASEPTR = pMalloc->GetBase(); } else // try to create a new shared heap #else if (!fInitialisedSm) #endif // MULTIHEAP { IMalloc * pm; hr = DfCreateSharedAllocator ( &pm, FALSE ); } if ( SUCCEEDED(hr) ) #ifdef MULTIHEAP hr = pMalloc->Sync(); #else hr = g_smAllocator.Sync(); #endif return hr; } //+--------------------------------------------------------------------------- // // Function: DfInitSharedMemBase, public // // Synopsis: Set up the base of shared memory // // History: 31-May-94 MikeSe Created // // Notes: // //---------------------------------------------------------------------------- void DfInitSharedMemBase() { #if !defined MULTIHEAP // make sure we have initialised. This is sufficient to ensure // that DFBASEPTR is set up if (!fInitialisedSm) { IMalloc * pm; HRESULT hr = DfCreateSharedAllocator ( &pm ); } #endif } #if DBG==1 && !defined(MULTIHEAP) //+------------------------------------------------------------------------- // // Method: CLocalAlloc::operator new, public // // Synopsis: Overloaded new operator to allocate objects from // task local space. // // Arguments: [size] -- Size of block to allocate // // Returns: Pointer to memory allocated. // // History: 17-Aug-92 PhilipLa Created. // 18-May-93 AlexT Switch to task IMalloc // //-------------------------------------------------------------------------- void *CLocalAlloc::operator new(size_t size) { #ifndef WIN32 olAssert(DfGetResLimit(DBRQ_MEMORY_ALLOCATED) >= 0); #endif olAssert(size > 0); if (SimulateFailure(DBF_MEMORY)) { return(NULL); } if (!HaveResource(DBR_MEMORY, (LONG)size)) { // Artificial limit exceeded so force failure return NULL; } void *pv = TaskMemAlloc(sizeof(CMemAlloc *) + size); if (pv != NULL) { void * const pvOrig = pv; memset(pv, NEWMEM, sizeof(CMemAlloc *) + size); CMemAlloc *pma = NULL; // Allocate tracking block (with shared memory) IMalloc *pMalloc; #ifdef WIN32 #ifdef MULTIHEAP pMalloc = &g_smAllocator; #else if ( FAILED(DfCreateSharedAllocator ( &pMalloc )) ) pMalloc = NULL; #endif // MULTIHEAP #else if (FAILED(DfGetScode(CoGetMalloc(MEMCTX_SHARED, &pMalloc)))) pMalloc = NULL; #endif if (pMalloc) { pma = (CMemAlloc *) pMalloc->Alloc(sizeof(CMemAlloc)); if (pma != NULL) { memset(pma, NEWMEM, sizeof(CMemAlloc)); CMemAlloc **ppma = (CMemAlloc **) pv; pv = (void *) ((CMemAlloc **)pv + 1); *ppma = pma; AddAlloc(pma, *(((void **)&size)-1), size, pv, GetCurrentContextId()); } pMalloc->Release(); } if (pma == NULL) { // Couldn't allocate tracking block - fail allocation TaskMemFree(pvOrig); pv = NULL; } } if (pv == NULL) { ModifyResLimit(DBR_MEMORY, (LONG)size); } return(pv); } //+------------------------------------------------------------------------- // // Method: CLocalAlloc::operator delete, public // // Synopsis: Free memory from task local space // // Arguments: [pv] -- Pointer to memory to free // // History: 17-Aug-92 PhilipLa Created. // 18-May-93 AlexT Switch to task IMalloc // //-------------------------------------------------------------------------- void CLocalAlloc::operator delete(void *pv) { if (pv == NULL) return; CMemAlloc **ppma = (CMemAlloc **)pv; ppma--; CMemAlloc *pma = *ppma; RemoveAlloc(pma, pv, GetCurrentContextId()); pv = (void *) ppma; memset(pv, OLDMEM, (size_t) pma->cbSize); // Free tracking block IMalloc *pMalloc = NULL; #ifdef WIN32 #ifdef MULTIHEAP pMalloc = &g_smAllocator; #else if ( FAILED(DfCreateSharedAllocator ( &pMalloc )) ) pMalloc = NULL; #endif // MULTIHEAP #else if (FAILED(DfGetScode(CoGetMalloc(MEMCTX_SHARED, &pMalloc)))) pMalloc = NULL; #endif if (pMalloc != NULL) { memset(pma, OLDMEM, sizeof(CMemAlloc)); pMalloc->Free(pma); pMalloc->Release(); } else olAssert(!aMsg("Unable to get shared allocator\n")); TaskMemFree(pv); } #endif // DBG==1 && !defined(MULTIHEAP)