Inside CRT: Debug Heap Management

转自www.codeguru.com

When you compile a debug build of your program with Visual Studio and run it in debugger, you can see that the memory allocated or deallocated has funny values, such as 0xCDCDCDCD or 0xDDDDDDDD. This is the result of the work Microsoft has put in to detect memory corruption and leaks in the Win32 platform. In this article, I will explain how memory allocation/deallocation is done via new/delete or malloc/free.

First, I will explain what all these values that you see, like CD, DD, and so forth, mean.

ValueNameDescription
0xCDClean MemoryAllocated memory via malloc or new but never written by the application.
0xDDDead MemoryMemory that has been released with delete or free. It is used to detect writing through dangling pointers.
0xFDFence MemoryAlso known as "no mans land." This is used to wrap the allocated memory (like surrounding it with fences) and is used to detect indexing arrays out of bounds.
0xAB(Allocated Block?)Memory allocated by LocalAlloc().
0xBAADF00DBad FoodMemory allocated by LocalAlloc() with LMEM_FIXED, but not yet written to.
0xCC When the code is compiled with the /GZ option, uninitialized variables are automatically assigned to this value (at byte level).

If you take a look at DBGHEAP.C, you can see how some of these values are defined:

static unsigned char _bNoMansLandFill = 0xFD;   /* fill no-man's land with this */
static unsigned char _bDeadLandFill   = 0xDD;   /* fill free objects with this */
static unsigned char _bCleanLandFill  = 0xCD;   /* fill new objects with this */

Before going any further, take a look at the memory management function that I will refer in this article.

FunctionDescription
mallocC/C++ function that allocates a block of memory from the heap. The implementation of the C++ operator new is based on malloc.
_malloc_dbgDebug version of malloc; only available in the debug versions of the run-time libraries. _malloc_dbg is a debug version of the malloc function. When _DEBUG is not defined, each call to _malloc_dbg is reduced to a call to malloc. Both malloc and _malloc_dbg allocate a block of memory in the base heap, but _malloc_dbg offers several debugging features: buffers on either side of the user portion of the block to test for leaks, a block type parameter to track specific allocation types, and filename/linenumber information to determine the origin of allocation requests.
freeC/C++ function that frees an allocated block. The implementation of C++ operator delete is based on free.
_free_dbgDebug version of free; only available in the debug versions of the run-time libraries. The _free_dbg function is a debug version of the free function. When _DEBUG is not defined, each call to _free_dbg is reduced to a call to free. Both free and _free_dbg free a memory block in the base heap, but _free_dbg accommodates two debugging features: the ability to keep freed blocks in the heap's linked list to simulate low memory conditions and a block type parameter to free specific allocation types.
LocalAlloc
GlobalAlloc
Win32 API to allocate the specified number of bytes from the heap. Windows memory management does not provide a separate local heap and global heap.
LocalFree
GlobalFree
Win32 API free the specified local memory object and invalidates its handle.
HeapAllocWin32 API allocates a block of memory from a heap. The allocated memory is not movable.
HeapFreeWin32 API frees a memory block allocated from a heap by the HeapAlloc or HeapReAlloc function.

There are many other functions that deal with memory management. For a complete view please refer to MSDN.

Note: Because this article is about memory management in a debug build, all the references to malloc and free in the following are actually references to their debug versions, _malloc_dbg and _free_dbg.

Compile the following code and run it in the debugger, walking step by step into it to see how memory is allocated and deallocated.

int main(int argc, char* argv[])
{
   char *buffer = new char[12];

   delete [] buffer;

   return 0;
}

Here, 12 bytes are dynamically allocated, but the CRT allocates more than that by wrapping the allocated block with bookkeeping information. For each allocated block, the CRT keeps information in a structure called _CrtMemBlockHeader, which is declared in DBGINT.H:

#define nNoMansLandSize 4

typedef struct _CrtMemBlockHeader
{
        struct _CrtMemBlockHeader * pBlockHeaderNext;
        struct _CrtMemBlockHeader * pBlockHeaderPrev;
        char *                      szFileName;
        int                         nLine;
        size_t                      nDataSize;
        int                         nBlockUse;
        long                        lRequest;
        unsigned char               gap[nNoMansLandSize];
        /* followed by:
         *  unsigned char           data[nDataSize];
         *  unsigned char           anotherGap[nNoMansLandSize];
         */

} _CrtMemBlockHeader;

It stores the following information:

FieldDescription
pBlockHeaderNextA pointer to the next block allocated, but next means the previous allocated block because the list is seen as a stack, with the latest allocated block at the top.
pBlockHeaderPrevA pointer to the previous block allocated; this means the block that was allocated after the current block.
szFileNameA pointer to the name of the file in which the call to malloc was made, if known.
nLineThe line in the source file indicated by szFileName at which the call to malloc was made, if known.
nDataSizeNumber of bytes requested
nBlockUse0 - Freed block, but not released back to the Win32 heap
1 - Normal block (allocated with new/malloc)
2 - CRT blocks, allocated by CRT for its own use
lRequestCounter incremented with each allocation
gapA zone of 4 bytes (in the current implementation) filled with 0xFD, fencing the data block, of nDataSize bytes. Another block filled with 0xFD of the same size follows the data.

Most of the work of heap block allocation and deallocation are made by HeapAlloc() and HeapFree(). When you request 12 bytes to be allocated on the heap, malloc() will call HeapAlloc(), requesting 36 more bytes.

blockSize = sizeof(_CrtMemBlockHeader) + nSize + nNoMansLandSize;

malloc requests space for the 12 bytes we need (nSize), plus 32 bytes for the _CrtMemBlockHeader structure and another nNoMansLandSize bytes (4 bytes) to fence the data zone and close the gap.

But, HeapAlloc() will allocate even more bytes: 8 bytes below the requested block (that is, at a lower address) and 32 above it (that is, at a bigger address). It also initializes the requested block to 0xBAADF00D (bad food).

Then, malloc() fills the _CrtMemBlockHeader block with information and initializes the data block with 0xCD and no mans land with 0xFD.

Here is a table that shows how memory looks after the call to HeapAlloc() and after malloc() returns. For a complete situation, see the last table. (Note: All values are in hex.)

Addressafter HeapAlloc()after malloc()
00320FD8
00320FDC
00320FE0
00320FE4
00320FE8
00320FEC
00320FF0
00320FF4
00320FF8
00320FFC
00321000
00321004
00321008
0032100C
00321010
00321014
00321018
0032101C
00321020
00321024
00321028
0032102C
09 00 09 01
E8 07 18 00
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
0D F0 AD BA
AB AB AB AB
AB AB AB AB
00 00 00 00
00 00 00 00
79 00 09 00
EE 04 EE 00
40 05 32 00
40 05 32 00
09 00 09 01
E8 07 18 00
98 07 32 00
00 00 00 00
00 00 00 00
00 00 00 00
0C 00 00 00
01 00 00 00
2E 00 00 00
FD FD FD FD
CD CD CD CD
CD CD CD CD
CD CD CD CD
FD FD FD FD
AB AB AB AB
AB AB AB AB
00 00 00 00
00 00 00 00
79 00 09 00
EE 04 EE 00
40 05 32 00
40 05 32 00

Colors:

  • Green: win32 bookkeeping info
  • Blue: block size requested by malloc and filled with bad food
  • Magenta: _CrtMemBlockHeader block
  • Red: no mans land
  • Black: requested data block

In this example, after the call to malloc() returns, buffer will point to memory address 0x00321000.

When you call delete/free, the CRT will set the block it requested from HeapAlloc() to 0xDD, indicating this is a free zone. Normally after this, free() will call HeapFree() to give back the block to the Win32 heap, in which case the block will be overwritten with 0xFEEEEEEE, to indicate Win32 heap free memory.

You can avoid this by using the CRTDBG_DELAY_FREE_MEM_DF flag to _CrtSetDbgFlag(). It prevents memory from actually being freed, as for simulating low-memory conditions. When this bit is on, freed blocks are kept in the debug heap's linked list but are marked as _FREE_BLOCK. This is useful if you want to detect dangling pointers errors, which can be done by verifying if the freed block is written with 0xDD pattern or something else. Use _CrtCheckMemory() to verify the heap.s integrity.

The next table shows how the memory looks during the free(), before HeapFree() is called and afterwards.

AddressBefore HeapFree()After HeapFree()
00320FD8
00320FDC
00320FE0
00320FE4
00320FE8
00320FEC
00320FF0
00320FF4
00320FF8
00320FFC
00321000
00321004
00321008
0032100C
00321010
00321014
00321018
0032101C
00321020
00321024
00321028
0032102C
09 00 09 01
5E 07 18 00
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
DD DD DD DD
AB AB AB AB
AB AB AB AB
00 00 00 00
00 00 00 00
79 00 09 00
EE 04 EE 00
40 05 32 00
40 05 32 00
82 00 09 01
5E 04 18 00
E0 2B 32 00
78 01 32 00
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE
EE FE EE FE

Colors:

  • Green: win32 bookkeeping info
  • Blue: CRT block filled with dead memory
  • Gray: memory given back to win32 heap

The two tables above are put in a single, more detailed, table below:

Address (hex)OffsetHeapAllocmallocFree before HeapFreeFree after HeapFreeDescription
00320FD8-4001090009010900090109000901090082Win32 Heap info
00320FDC-36001807E8001807E80018075E0018045EWin32 Heap info
00320FE0-32BAADF00D00320798DDDDDDDD00322BE0pBlockHeaderNext
00320FE4-28BAADF00D00000000DDDDDDDD00320178pBlockHeaderPrev
00320FE8-24BAADF00D00000000DDDDDDDDFEEEEEEEszFileName
00320FEC-20BAADF00D00000000DDDDDDDDFEEEEEEEnLine
00320FF0-16BAADF00D0000000CDDDDDDDDFEEEEEEEnDataSize
00320FF4-12BAADF00D00000001DDDDDDDDFEEEEEEEnBlockUse
00320FF8-8BAADF00D0000002EDDDDDDDDFEEEEEEElRequest
00320FFC-4BAADF00DFDFDFDFDDDDDDDDDFEEEEEEEgap (no mans land)
003210000BAADF00DCDCDCDCDDDDDDDDDFEEEEEEEData requested
00321004+4BAADF00DCDCDCDCDDDDDDDDDFEEEEEEEData requested
00321008+8BAADF00DCDCDCDCDDDDDDDDDFEEEEEEEData requested
0032100C+12BAADF00DFDFDFDFDDDDDDDDDFEEEEEEENo mans land
00321010+16ABABABABABABABABABABABABFEEEEEEEWin32 Heap info
00321014+20ABABABABABABABABABABABABFEEEEEEEWin32 Heap info
00321018+24000000000000000000000000FEEEEEEEWin32 Heap info
0032101C+28000000000000000000000000FEEEEEEEWin32 Heap info
00321020+32000900790009007900090079FEEEEEEEWin32 Heap info
00321024+3600EE04EE00EE04EE00EE04EEFEEEEEEEWin32 Heap info
00321028+40003205400032054000320540FEEEEEEEWin32 Heap info
0032102C+44003205400032054000320540FEEEEEEEWin32 Heap info

About the Author
Marius Bancila is a software developer working for a company that provides industrial automation solutions, but he is mainly focused on GUI design with MFC. When he discovered the .NET framework quickly fell inlove. He considers that CodeGuru is the best place on internet to spend time on.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java heap space是一种常见的OutOfMemoryError错误,它表示Java堆内存不足。当Java应用程序需要分配更多的内存给对象,但是堆内存已经被完全使用时,就会抛出这个错误。 解决Java heap space错误的方法有以下几种: 1. 增加堆内存大小:可以通过在启动应用程序时使用-Xmx参数来增加堆内存的大小。例如,使用命令`java -Xmx2g MyApp`将堆内存大小设置为2GB。 2. 优化代码和内存使用:检查代码中是否存在内存泄漏或者不必要的对象引用。确保及时释放不再使用的对象,避免创建过多的临时对象。 3. 使用垃圾回收器:Java提供了不同的垃圾回收器,可以根据应用程序的需求选择合适的垃圾回收器。例如,使用G1垃圾回收器可以更好地处理大内存和多核处理器的情况。 4. 分析内存使用情况:使用工具如VisualVM或JConsole来监视和分析应用程序的内存使用情况,找出内存使用过多的地方,并进行优化。 5. 减少对象的创建和销毁:避免频繁地创建和销毁对象,可以使用对象池或缓存来重复使用对象,减少内存的使用。 6. 使用更高效的数据结构和算法:选择合适的数据结构和算法可以减少内存的使用。例如,使用HashMap代替ArrayList可以减少内存的占用。 范例:<<引用:java.lang.OutOfMemoryError: Java heap space。主要有3种比较常见的OutOfMemory Error: Java.lang.OutOfMemoryError: Java heap space java.lang.OutOfMemoryError: PermGen space java.lang.OutOfMemoryError: GC overhead limit exceeded 。 引用:java.lang.OutOfMemoryError: Java heap space。3. java.lang.OutOfMemoryError: GC overhead limit exceeded 。 请回答我或者给我介绍或演示一下:OutOfMemoryError: Java heap space? 当Java应用程序需要分配更多的内存给对象,但是堆内存已经被完全使用时,就会抛出Java heap space错误。解决这个错误的方法包括增加堆内存大小、优化代码和内存使用、使用垃圾回收器、分析内存使用情况、减少对象的创建和销毁以及使用更高效的数据结构和算法。具体的方法可以根据应用程序的需求和情况来选择。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值