-
Notifications
You must be signed in to change notification settings - Fork 402
/
Copy pathvldheap.cpp
212 lines (194 loc) · 6.98 KB
/
vldheap.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
////////////////////////////////////////////////////////////////////////////////
//
// Visual Leak Detector - Internal C++ Heap Management
// Copyright (c) 2006 Dan Moulding
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
//
// See COPYING.txt for the full terms of the GNU Lesser General Public License.
//
////////////////////////////////////////////////////////////////////////////////
#include <cassert>
#define VLDBUILD // Declares that we are building Visual Leak Detector.
#include "ntapi.h" // Provides access to NT APIs.
#include "vldheap.h" // Provides access to VLD's internal heap data structures.
#undef new // Do not map "new" to VLD's new operator in this file
// Global variables.
vldblockheader_t *vldblocklist = NULL; // List of internally allocated blocks on VLD's private heap.
HANDLE vldheap; // VLD's private heap.
CRITICAL_SECTION vldheaplock; // Serializes access to VLD's private heap.
// Local helper functions.
static inline void vlddelete (void *block);
static inline void* vldnew (size_t size, const char *file, int line);
// scalar delete operator - Delete operator used to free internally used memory
// back to VLD's private heap.
//
// - block (IN): Pointer to the scalar memory block to free.
//
// Return Value:
//
// None.
//
void operator delete (void *block)
{
vlddelete(block);
}
// vector delete operator - Delete operator used to free internally used memory
// back to VLD's private heap.
//
// - block (IN): Pointer to the vector memory block to free.
//
// Return Value:
//
// None.
//
void operator delete [] (void *block)
{
vlddelete(block);
}
// scalar delete operator - Delete operator used to free memory partially
// allocated by new in the event that the corresponding new operator throws
// an exception.
//
// Note: This version of the delete operator should never be called directly.
// The compiler automatically generates calls to this function as needed.
//
void operator delete (void *block, const char *, int)
{
vlddelete(block);
}
// vector delete operator - Delete operator used to free memory partially
// allocated by new in the event that the corresponding new operator throws
// an exception.
//
// Note: This version of the delete operator should never be called directly.
// The compiler automatically generates calls to this function as needed.
//
void operator delete [] (void *block, const char *, int)
{
vlddelete(block);
}
// scalar new operator - New operator used to allocate a scalar memory block
// from VLD's private heap.
//
// - size (IN): Size of the memory block to be allocated.
//
// - file (IN): The name of the file from which this function is being
// called.
//
// - line (IN): The line number, in the above file, at which this function is
// being called.
//
// Return Value:
//
// If the allocation succeeds, a pointer to the allocated memory block is
// returned. If the allocation fails, NULL is returned.
//
void* operator new (size_t size, const char *file, int line)
{
return vldnew(size, file, line);
}
// vector new operator - New operator used to allocate a vector memory block
// from VLD's private heap.
//
// - size (IN): Size of the memory block to be allocated.
//
// - file (IN): The name of the file from which this function is being
// called.
//
// - line (IN): The line number, in the above file, at which this function is
// being called.
//
// Return Value:
//
// If the allocation succeeds, a pointer to the allocated memory block is
// returned. If the allocation fails, NULL is returned.
//
void* operator new [] (size_t size, const char *file, int line)
{
return vldnew(size, file, line);
}
// vlddelete - Local helper function that actually frees memory back to VLD's
// private heap.
//
// - block (IN): Pointer to a memory block being freed.
//
// Return Value:
//
// None.
//
void vlddelete (void *block)
{
BOOL freed;
vldblockheader_t *header = VLDBLOCKHEADER((LPVOID)block);
// Unlink the block from the block list.
EnterCriticalSection(&vldheaplock);
if (header->prev) {
header->prev->next = header->next;
}
else {
vldblocklist = header->next;
}
if (header->next) {
header->next->prev = header->prev;
}
LeaveCriticalSection(&vldheaplock);
// Free the block.
freed = RtlFreeHeap(vldheap, 0x0, header);
assert(freed != FALSE);
}
// vldnew - Local helper function that actually allocates memory from VLD's
// private heap. Prepends a header, which is used for bookeeping information
// that allows VLD to detect and report internal memory leaks, to the returned
// block, but the header is transparent to the caller because the returned
// pointer points to the useable section of memory requested by the caller, it
// does not point to the block header.
//
// - size (IN): Size of the memory block to be allocated.
//
// - file (IN): Name of the file that called the new operator.
//
// - line (IN): Line, in the above file, at which the new operator was called.
//
// Return Value:
//
// If the memory allocation succeeds, a pointer to the allocated memory
// block is returned. If the allocation fails, NULL is returned.
//
void* vldnew (size_t size, const char *file, int line)
{
vldblockheader_t *header = (vldblockheader_t*)RtlAllocateHeap(vldheap, 0x0, size + sizeof(vldblockheader_t));
static SIZE_T serialnumber = 0;
if (header == NULL) {
// Out of memory.
return NULL;
}
// Fill in the block's header information.
header->file = file;
header->line = line;
header->serialnumber = serialnumber++;
header->size = size;
// Link the block into the block list.
EnterCriticalSection(&vldheaplock);
header->next = vldblocklist;
if (header->next != NULL) {
header->next->prev = header;
}
header->prev = NULL;
vldblocklist = header;
LeaveCriticalSection(&vldheaplock);
// Return a pointer to the beginning of the data section of the block.
return (void*)VLDBLOCKDATA(header);
}