2020-06-15 14:18:57 +00:00
|
|
|
#include "libc/mem/mem.h"
|
2020-11-25 16:19:00 +00:00
|
|
|
#include "third_party/dlmalloc/dlmalloc.internal.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
|
|
|
|
static void internal_inspect_all(mstate m,
|
|
|
|
void (*handler)(void* start, void* end,
|
|
|
|
size_t used_bytes,
|
|
|
|
void* callback_arg),
|
|
|
|
void* arg) {
|
|
|
|
if (is_initialized(m)) {
|
|
|
|
mchunkptr top = m->top;
|
|
|
|
msegmentptr s;
|
|
|
|
for (s = &m->seg; s != 0; s = s->next) {
|
|
|
|
mchunkptr q = align_as_chunk(s->base);
|
|
|
|
while (segment_holds(s, q) && q->head != FENCEPOST_HEAD) {
|
|
|
|
mchunkptr next = next_chunk(q);
|
|
|
|
size_t sz = chunksize(q);
|
|
|
|
size_t used;
|
|
|
|
void* start;
|
|
|
|
if (is_inuse(q)) {
|
|
|
|
used = sz - CHUNK_OVERHEAD; /* must not be mmapped */
|
|
|
|
start = chunk2mem(q);
|
|
|
|
} else {
|
|
|
|
used = 0;
|
|
|
|
if (is_small(sz)) { /* offset by possible bookkeeping */
|
|
|
|
start = (void*)((char*)q + sizeof(struct malloc_chunk));
|
|
|
|
} else {
|
|
|
|
start = (void*)((char*)q + sizeof(struct malloc_tree_chunk));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (start < (void*)next) /* skip if all space is bookkeeping */
|
|
|
|
handler(start, next, used, arg);
|
|
|
|
if (q == top) break;
|
|
|
|
q = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Traverses the heap and calls the given handler for each managed
|
|
|
|
* region, skipping all bytes that are (or may be) used for bookkeeping
|
|
|
|
* purposes. Traversal does not include include chunks that have been
|
|
|
|
* directly memory mapped. Each reported region begins at the start
|
|
|
|
* address, and continues up to but not including the end address. The
|
|
|
|
* first used_bytes of the region contain allocated data. If
|
|
|
|
* used_bytes is zero, the region is unallocated. The handler is
|
|
|
|
* invoked with the given callback argument. If locks are defined, they
|
|
|
|
* are held during the entire traversal. It is a bad idea to invoke
|
|
|
|
* other malloc functions from within the handler.
|
|
|
|
*
|
|
|
|
* For example, to count the number of in-use chunks with size greater
|
|
|
|
* than 1000, you could write:
|
|
|
|
*
|
|
|
|
* static int count = 0;
|
|
|
|
* void count_chunks(void* start, void* end, size_t used, void* arg) {
|
|
|
|
* if (used >= 1000) ++count;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* then,
|
|
|
|
*
|
|
|
|
* malloc_inspect_all(count_chunks, NULL);
|
|
|
|
*/
|
|
|
|
void malloc_inspect_all(void (*handler)(void* start, void* end,
|
|
|
|
size_t used_bytes, void* callback_arg),
|
|
|
|
void* arg) {
|
|
|
|
ensure_initialization();
|
2020-09-07 04:39:00 +00:00
|
|
|
if (!PREACTION(g_dlmalloc)) {
|
|
|
|
internal_inspect_all(g_dlmalloc, handler, arg);
|
|
|
|
POSTACTION(g_dlmalloc);
|
2020-06-15 14:18:57 +00:00
|
|
|
}
|
|
|
|
}
|