heapinit.c
来自「c语言编程软件vc6.0中文绿色版_vc6.0官方下载」· C语言 代码 · 共 790 行 · 第 1/2 页
C
790 行
/***
*heapinit.c - Initialze the heap
*
* Copyright (c) 1989-1998, Microsoft Corporation. All rights reserved.
*
*Purpose:
*
*******************************************************************************/
#ifdef WINHEAP
#include <cruntime.h>
#include <malloc.h>
#include <stdlib.h>
#include <winheap.h>
HANDLE _crtheap;
/*
* Dummy definition of _amblksiz. Included primarily so the dll will build
* without having to change crtlib.c (there is an access function for _amblksiz
* defined in crtlib.c).
*/
unsigned int _amblksiz = BYTES_PER_PARA;
int __active_heap;
void __cdecl _GetLinkerVersion(LinkerVersion * plv)
{
PIMAGE_DOS_HEADER pidh;
PIMAGE_NT_HEADERS pinh;
plv->dw = 0;
pidh = (PIMAGE_DOS_HEADER) GetModuleHandle(NULL);
if ( pidh->e_magic != IMAGE_DOS_SIGNATURE || pidh->e_lfanew == 0)
return;
pinh = (PIMAGE_NT_HEADERS)(((PBYTE)pidh) + pidh->e_lfanew);
plv->bverMajor = pinh->OptionalHeader.MajorLinkerVersion;
plv->bverMinor = pinh->OptionalHeader.MinorLinkerVersion;
}
/***
*__heap_select() - Choose from the V6, V5 or system heaps
*
*Purpose:
* Check OS, environment and build bits to determine appropriate
* small-block heap for the app.
*
*Entry:
* <void>
*Exit:
* Returns __V6_HEAP, __V5_HEAP or __SYSTEM_HEAP
*
*Exceptions:
* none
*
*******************************************************************************/
int __cdecl __heap_select(void)
{
char env_heap_select_string[(_MAX_PATH+5)*16];
char env_app_name[_MAX_PATH];
char *env_heap_type = NULL;
char *cp;
int heap_choice;
LinkerVersion lv;
OSVERSIONINFO osvi;
// First, check the OS for NT >= 5.0
osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
if ( GetVersionEx(&osvi) )
if ( (osvi.dwPlatformId == VER_PLATFORM_WIN32_NT) &&
(osvi.dwMajorVersion >= 5) )
return __SYSTEM_HEAP;
// Second, check the envrionment variable override
if (GetEnvironmentVariableA(__HEAP_ENV_STRING, env_heap_select_string, sizeof(env_heap_select_string)))
{
for (cp = env_heap_select_string; *cp; ++cp)
if ('a' <= *cp && *cp <= 'z')
*cp += 'A' - 'a';
if (!strncmp(__GLOBAL_HEAP_SELECTOR,env_heap_select_string,sizeof(__GLOBAL_HEAP_SELECTOR)-1))
env_heap_type = env_heap_select_string;
else
{
GetModuleFileName(NULL,env_app_name,sizeof(env_app_name));
for (cp = env_app_name; *cp; ++cp)
if ('a' <= *cp && *cp <= 'z')
*cp += 'A' - 'a';
env_heap_type = strstr(env_heap_select_string,env_app_name);
}
if (env_heap_type)
{
if (env_heap_type = strchr(env_heap_type,','))
{
cp = ++env_heap_type;
while (*cp)
{
if (*cp == ';')
*cp = 0;
else cp++;
}
heap_choice = (int)strtol(env_heap_type, NULL, 10);
if ( (heap_choice == __V5_HEAP) ||
(heap_choice == __V6_HEAP) ||
(heap_choice == __SYSTEM_HEAP) )
return heap_choice;
}
}
}
// Third, check the build bits in the app; apps built with tools >= VC++ 6.0
// will get the V6 heap, apps built with older tools will get the V5 heap
_GetLinkerVersion(&lv);
if (lv.bverMajor >= 6)
return __V6_HEAP;
else
return __V5_HEAP;
}
/***
*_heap_init() - Initialize the heap
*
*Purpose:
* Setup the initial C library heap.
*
* NOTES:
* (1) This routine should only be called once!
* (2) This routine must be called before any other heap requests.
*
*Entry:
* <void>
*Exit:
* Returns 1 if successful, 0 otherwise.
*
*Exceptions:
* If heap cannot be initialized, the program will be terminated
* with a fatal runtime error.
*
*******************************************************************************/
int __cdecl _heap_init (
int mtflag
)
{
// Initialize the "big-block" heap first.
if ( (_crtheap = HeapCreate( mtflag ? 0 : HEAP_NO_SERIALIZE,
BYTES_PER_PAGE, 0 )) == NULL )
return 0;
// Pick a heap, any heap
__active_heap = __heap_select();
if ( __active_heap == __V6_HEAP )
{
// Initialize the small-block heap
if (__sbh_heap_init(MAX_ALLOC_DATA_SIZE) == 0)
{
HeapDestroy(_crtheap);
return 0;
}
}
else if ( __active_heap == __V5_HEAP )
{
if ( __old_sbh_new_region() == NULL )
{
HeapDestroy( _crtheap );
return 0;
}
}
return 1;
}
/***
*_heap_term() - return the heap to the OS
*
*Purpose:
*
* NOTES:
* (1) This routine should only be called once!
* (2) This routine must be called AFTER any other heap requests.
*
*Entry:
* <void>
*Exit:
* <void>
*
*Exceptions:
*
*******************************************************************************/
void __cdecl _heap_term (void)
{
// if it has been initialized, destroy the small-block heap
if ( __active_heap == __V6_HEAP )
{
PHEADER pHeader = __sbh_pHeaderList;
int cntHeader;
// scan through all the headers
for (cntHeader = 0; cntHeader < __sbh_cntHeaderList; cntHeader++)
{
// decommit and release the address space for the region
VirtualFree(pHeader->pHeapData, BYTES_PER_REGION, MEM_DECOMMIT);
VirtualFree(pHeader->pHeapData, 0, MEM_RELEASE);
// free the region data structure
HeapFree(_crtheap, 0, pHeader->pRegion);
pHeader++;
}
// free the header list
HeapFree(_crtheap, 0, __sbh_pHeaderList);
}
else if ( __active_heap == __V5_HEAP )
{
__old_sbh_region_t *preg = &__old_small_block_heap;
// Release the regions of the small-block heap
do
{
if ( preg->p_pages_begin != NULL )
VirtualFree( preg->p_pages_begin, 0, MEM_RELEASE );
}
while ( (preg = preg->p_next_region) != &__old_small_block_heap );
}
// destroy the large-block heap
HeapDestroy(_crtheap);
}
#else /* WINHEAP */
#ifdef _WIN32
#include <cruntime.h>
#include <oscalls.h>
#include <dos.h>
#include <heap.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
/*
* Heap descriptor
*/
struct _heap_desc_ _heap_desc = {
&_heap_desc.sentinel, /* pfirstdesc */
&_heap_desc.sentinel, /* proverdesc */
NULL, /* emptylist */
NULL, /* sentinel.pnextdesc */
NULL /* sentinel.pblock */
};
/*
* Array of region structures
* [Note: We count on the fact that this is always initialized to zero
* by the compiler.]
*/
struct _heap_region_ _heap_regions[_HEAP_REGIONMAX];
void ** _heap_descpages; /* linked list of pages used for descriptors */
/*
* Control parameter locations
*/
unsigned int _heap_resetsize = 0xffffffff;
/* NOTE: Currenlty, _heap_growsize is a #define to _amblksiz */
unsigned int _heap_growsize = _HEAP_GROWSIZE; /* region inc size */
unsigned int _heap_regionsize = _HEAP_REGIONSIZE_L; /* region size */
unsigned int _heap_maxregsize = _HEAP_MAXREGSIZE_L; /* max region size */
/***
*_heap_init() - Initialize the heap
*
*Purpose:
* Setup the initial C library heap. All necessary memory and
* data bases are init'd appropriately so future requests work
* correctly.
*
* NOTES:
* (1) This routine should only be called once!
* (2) This routine must be called before any other heap requests.
*
*
*Entry:
* <void>
*Exit:
* <void>
*
*Exceptions:
* If heap cannot be initialized, the program will be terminated
* with a fatal runtime error.
*
*******************************************************************************/
void __cdecl _heap_init (
void
)
{
/*
* Test for Win32S or Phar Lap TNT environment
* which cannot allocate uncommitted memory
* without actually allocating physical memory
*
* High bit of _osver is set for both of those environments
* -AND- the Windows version will be less than 4.0.
*/
if ( ( _osver & 0x8000 ) && ( _winmajor < 4 ) )
{
_heap_regionsize = _HEAP_REGIONSIZE_S;
_heap_maxregsize = _HEAP_MAXREGSIZE_S;
}
}
/***
*_heap_term() - Clean-up the heap
*
*Purpose:
* This routine will decommit and release ALL of the CRT heap.
* All memory malloc-ed by the CRT will then be invalid.
*
* NOTES:
* (1) This routine should only be called once!
* (2) This routine must be called AFTER any other heap requests.
*
*Entry:
* <void>
*Exit:
* <void>
*
*Exceptions:
*
*******************************************************************************/
void __cdecl _heap_term (
void
)
{
int index;
void **pageptr;
/*
* Loop through the region descriptor table, decommitting
* and releasing (freeing up) each region that is in use.
*/
for ( index=0 ; index < _HEAP_REGIONMAX ; index++ ) {
void * regbase ;
if ( (regbase = _heap_regions[index]._regbase)
&& VirtualFree(regbase, _heap_regions[index]._currsize, MEM_DECOMMIT)
&& VirtualFree(regbase, 0, MEM_RELEASE) )
regbase = _heap_regions[index]._regbase = NULL ;
}
/*
* Now we need to decommit and release the pages used for descriptors
* _heap_descpages points to the head of a singly-linked list of the pages.
*/
pageptr = _heap_descpages;
while ( pageptr ) {
void **nextpage;
nextpage = *pageptr;
if(!VirtualFree(pageptr, 0, MEM_RELEASE))
break; /* if the linked list is corrupted, give up */
pageptr = nextpage;
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?