windows-nt/Source/XPSP1/NT/admin/wmi/wbem/winmgmt/wbemcomn/arena.cpp
2020-09-26 16:20:57 +08:00

300 lines
6.8 KiB
C++

/*++
Copyright (C) 1996-2001 Microsoft Corporation
Module Name:
Abstract:
History:
--*/
#include "precomp.h"
#include "genutils.h"
#include "arena.h"
#include "sync.h"
#include "reg.h"
#include "arrtempl.h"
#define STARTUP_HEAP_HINT_REGVAL_W L"Startup Heap Preallocation Size"
static HANDLE g_hHeap = NULL;
static class DefaultInitializer
{
public:
DefaultInitializer()
{
CWin32DefaultArena::WbemHeapInitialize( GetProcessHeap() );
}
} g_hDefaultInitializer;
BOOL CWin32DefaultArena::WbemHeapInitialize( HANDLE hHeap )
{
if ( g_hHeap != NULL )
{
return FALSE;
}
g_hHeap = hHeap;
return TRUE;
}
void CWin32DefaultArena::WbemHeapFree()
{
if ( g_hHeap == NULL )
{
return;
}
if (g_hHeap != GetProcessHeap())
HeapDestroy(g_hHeap);
g_hHeap = NULL;
return;
}
//
//***************************************************************************
LPVOID CWin32DefaultArena::WbemMemAlloc(SIZE_T dwBytes)
{
if ( g_hHeap == NULL )
return NULL;
return HeapAlloc( g_hHeap, 0, dwBytes);
}
//***************************************************************************
//
//***************************************************************************
LPVOID CWin32DefaultArena::WbemMemReAlloc(LPVOID pOriginal, SIZE_T dwNewSize)
{
if ( g_hHeap == NULL )
return NULL;
return HeapReAlloc( g_hHeap, 0, pOriginal, dwNewSize);
}
//***************************************************************************
//
//***************************************************************************
BOOL CWin32DefaultArena::WbemMemFree(LPVOID pBlock)
{
if ( g_hHeap == NULL )
return FALSE;
if (pBlock==0)
return TRUE;
return HeapFree( g_hHeap, 0, pBlock);
}
//***************************************************************************
//
//***************************************************************************
SIZE_T CWin32DefaultArena::WbemMemSize(LPVOID pBlock)
{
if ( g_hHeap == NULL )
return 0;
return HeapSize( g_hHeap, 0, pBlock);
}
//***************************************************************************
//
//***************************************************************************
BSTR CWin32DefaultArena::WbemSysAllocString(const wchar_t *wszString)
{
if ( g_hHeap == NULL )
return NULL;
BSTR pBuffer = SysAllocString(wszString);
return pBuffer;
}
//***************************************************************************
//
//***************************************************************************
BSTR CWin32DefaultArena::WbemSysAllocStringByteLen(const char *szString, UINT len)
{
if ( g_hHeap == NULL )
return NULL;
BSTR pBuffer = SysAllocStringByteLen(szString, len);
return pBuffer;
}
//***************************************************************************
//
//****************************************************************************
INT CWin32DefaultArena::WbemSysReAllocString(BSTR *bszString, const wchar_t *wszString)
{
if ( g_hHeap == NULL )
return FALSE;
INT nRet = SysReAllocString(bszString, wszString);
return nRet;
}
//***************************************************************************
//
//***************************************************************************
BSTR CWin32DefaultArena::WbemSysAllocStringLen(const wchar_t *wszString, UINT len)
{
if ( g_hHeap == NULL )
return NULL;
BSTR pBuffer = SysAllocStringLen(wszString, len);
return pBuffer;
}
//***************************************************************************
//
//***************************************************************************
int CWin32DefaultArena::WbemSysReAllocStringLen( BSTR *bszString,
const wchar_t *wszString,
UINT nLen)
{
if ( g_hHeap == NULL )
return FALSE;
INT nRet = SysReAllocStringLen(bszString, wszString, nLen);
return nRet;
}
//***************************************************************************
//
//***************************************************************************
BOOL CWin32DefaultArena::WbemOutOfMemory()
{
return FALSE;
}
BOOL CWin32DefaultArena::ValidateMemSize(BOOL bLargeValidation)
{
if ( g_hHeap == NULL )
return FALSE;
MEMORYSTATUS memBuffer;
memset(&memBuffer, 0, sizeof(MEMORYSTATUS));
memBuffer.dwLength = sizeof(MEMORYSTATUS);
DWORD dwMemReq = 0;
if (bLargeValidation)
dwMemReq = 0x400000; //4MB
else
dwMemReq = 0x200000; //2MB
GlobalMemoryStatus(&memBuffer);
if (memBuffer.dwAvailPageFile >= dwMemReq)
{
return TRUE;
}
static CCritSec cs;
try
{
cs.Enter();
}
catch(...)
{
return FALSE;
}
//THIS ABSOLUTELY HAS TO BE HeapAlloc, and not the WBEM Allocator!!!
LPVOID pBuff = HeapAlloc( g_hHeap, 0, dwMemReq);
//THIS ABSOLUTELY HAS TO BE HeapAlloc, and not the WBEM Allocator!!!
if (pBuff == NULL)
{
cs.Leave();
return FALSE;
}
HeapFree( g_hHeap, 0, pBuff);
GlobalMemoryStatus(&memBuffer);
cs.Leave();
if (memBuffer.dwAvailPageFile >= dwMemReq)
{
return TRUE;
}
return FALSE;
}
HANDLE CWin32DefaultArena::GetArenaHeap()
{
return g_hHeap;
}
BOOL CWin32DefaultArena::WriteHeapHint()
{
if ( g_hHeap == NULL )
return FALSE;
//
// Don't bother if not on NT. We will use internal NT APIs
//
if(!IsNT())
return FALSE;
//
// Don't bother if running in a client --- only WinMgmt uses hints
//
if(!IsWinMgmt())
return FALSE;
//
// In WinMgmt. Walk the heap to calculate total size
//
PROCESS_HEAP_ENTRY entry;
entry.lpData = NULL;
DWORD dwTotalAllocations = 0;
while(HeapWalk( g_hHeap, &entry))
{
if(entry.wFlags & PROCESS_HEAP_ENTRY_BUSY)
{
//
// Allocated block. Add both it's size and its overhead to the total
// We want the overhead since it figures into the total required
// commitment.
//
dwTotalAllocations += entry.cbData + entry.cbOverhead;
}
}
//
// Write the total to the registry. Note that we write this data even if
// startup preallocations are disabled, since they may be enabled before
// WinMgmt starts up the next time
//
HKEY hKey;
long lRes = RegOpenKeyEx(HKEY_LOCAL_MACHINE, WBEM_REG_WINMGMT, 0,
KEY_SET_VALUE, &hKey);
if(lRes)
return FALSE;
CRegCloseMe cm1(hKey);
lRes = RegSetValueExW(hKey, STARTUP_HEAP_HINT_REGVAL_W, 0,
REG_DWORD, (const BYTE*)&dwTotalAllocations, sizeof(DWORD));
if(lRes)
return FALSE;
return TRUE;
}