diff options
author | Robin Watts <robin.watts@artifex.com> | 2011-12-16 16:18:16 +0000 |
---|---|---|
committer | Robin Watts <robin.watts@artifex.com> | 2011-12-16 16:23:31 +0000 |
commit | 81d80fed5412aed37046d23c0919875f5742e1e0 (patch) | |
tree | 9bebb4d5e228ebc4ffa1a80f42c7c53f5d7bb6c4 /fitz | |
parent | b63f97dc832e4085ffc0e4e477f5d86fd44360eb (diff) | |
download | mupdf-81d80fed5412aed37046d23c0919875f5742e1e0.tar.xz |
New Memento features; nesting of listed blocks, labelling of blocks
When listing blocks from Memento, look for internal pointers to
other blocks, and use this to generate a nested list - much
easier to read when looking to see which stray reference is
causing blocks to leak.
Add new Memento_label feature to allow blocks to be labelled. Makes
block displaying much easier to read.
Diffstat (limited to 'fitz')
-rw-r--r-- | fitz/memento.c | 213 | ||||
-rw-r--r-- | fitz/memento.h | 2 |
2 files changed, 183 insertions, 32 deletions
diff --git a/fitz/memento.c b/fitz/memento.c index 493b9d1c..daac3dfa 100644 --- a/fitz/memento.c +++ b/fitz/memento.c @@ -67,7 +67,8 @@ enum { }; enum { - Memento_Flag_OldBlock = 1 + Memento_Flag_OldBlock = 1, + Memento_Flag_HasParent = 2 }; typedef struct Memento_BlkHeader Memento_BlkHeader; @@ -79,6 +80,13 @@ struct Memento_BlkHeader int lastCheckedOK; int flags; Memento_BlkHeader *next; + + const char *label; + + /* Entries for nesting display calculations */ + Memento_BlkHeader *child; + Memento_BlkHeader *sibling; + char preblk[Memento_PreSize]; }; @@ -399,25 +407,144 @@ static int Memento_appBlock(Memento_Blocks *blks, return 0; } +static void showBlock(Memento_BlkHeader *b, int space) +{ + fprintf(stderr, "0x%p:(size=%d,num=%d)", + MEMBLK_TOBLK(b), (int)b->rawsize, b->sequence); + if (b->label) + fprintf(stderr, "%c(%s)", space, b->label); +} + +static void blockDisplay(Memento_BlkHeader *b, int n) +{ + int i = 0; + n++; + while(i < n) + { + fprintf(stderr, &" "[32-((n-i)&31)]); + i += ((n-i)&31); + } + showBlock(b, '\t'); + fprintf(stderr, "\n"); +} + static int Memento_listBlock(Memento_BlkHeader *b, void *arg) { int *counts = (int *)arg; - fprintf(stderr, " 0x%p:(size=%d,num=%d)\n", - MEMBLK_TOBLK(b), (int)b->rawsize, b->sequence); + blockDisplay(b, 0); counts[0]++; counts[1]+= b->rawsize; return 0; } -void Memento_listBlocks(void) { - int counts[2]; - counts[0] = 0; - counts[1] = 0; +static void doNestedDisplay(Memento_BlkHeader *b, + int depth) +{ + blockDisplay(b, depth); + for (b = b->child; b; b = b->sibling) + doNestedDisplay(b, depth+1); +} + +static int ptrcmp(const void *a_, const void *b_) +{ + const char **a = (const char **)a_; + const char **b = (const char **)b_; + return (int)(*a-*b); +} + +int Memento_listBlocksNested(void) +{ + int count, size, i; + Memento_BlkHeader *b; + void **blocks, *minptr, *maxptr; + int mask; + + /* Count the blocks */ + count = 0; + size = 0; + for (b = globals.used.head; b; b = b->next) { + size += b->rawsize; + count++; + } + + /* Make our block list */ + blocks = MEMENTO_UNDERLYING_MALLOC(sizeof(void *) * count); + if (blocks == NULL) + return 1; + + /* Populate our block list */ + b = globals.used.head; + minptr = maxptr = MEMBLK_TOBLK(b); + mask = (int)minptr; + for (i = 0; b; b = b->next, i++) { + void *p = MEMBLK_TOBLK(b); + mask &= (int)p; + if (p < minptr) + minptr = p; + if (p > maxptr) + maxptr = p; + blocks[i] = p; + b->flags &= ~Memento_Flag_HasParent; + b->child = NULL; + b->sibling = NULL; + } + qsort(blocks, count, sizeof(void *), ptrcmp); + + /* Now, calculate tree */ + for (b = globals.used.head; b; b = b->next) { + char *p = MEMBLK_TOBLK(b); + int end = (b->rawsize < 1024 ? b->rawsize : 1024); + for (i = 0; i < end; i += sizeof(void *)) { + void *q = *(void **)(&p[i]); + void **r; + + /* Do trivial checks on pointer */ + if ((mask & (int)q) != mask || q < minptr || q > maxptr) + continue; + + /* Search for pointer */ + r = bsearch(&q, blocks, count, sizeof(void *), ptrcmp); + if (r) { + /* Found child */ + Memento_BlkHeader *child = MEMBLK_FROMBLK(*r); + + /* We're assuming tree structure, not graph - ignore second + * and subsequent pointers. */ + if (child->flags & Memento_Flag_HasParent) + continue; + + child->sibling = b->child; + b->child = child; + child->flags |= Memento_Flag_HasParent; + } + } + } + + /* Now display with nesting */ + for (b = globals.used.head; b; b = b->next) { + if ((b->flags & Memento_Flag_HasParent) == 0) + doNestedDisplay(b, 0); + } + fprintf(stderr, " Total number of blocks = %d\n", count); + fprintf(stderr, " Total size of blocks = %d\n", size); + + MEMENTO_UNDERLYING_FREE(blocks); + return 0; +} + +void Memento_listBlocks(void) +{ fprintf(stderr, "Allocated blocks:\n"); - Memento_appBlocks(&globals.used, Memento_listBlock, &counts[0]); - fprintf(stderr, " Total number of blocks = %d\n", counts[0]); - fprintf(stderr, " Total size of blocks = %d\n", counts[1]); + if (Memento_listBlocksNested()) + { + int counts[2]; + counts[0] = 0; + counts[1] = 0; + Memento_appBlocks(&globals.used, Memento_listBlock, &counts[0]); + fprintf(stderr, " Total number of blocks = %d\n", counts[0]); + fprintf(stderr, " Total size of blocks = %d\n", counts[1]); + } } static int Memento_listNewBlock(Memento_BlkHeader *b, @@ -538,6 +665,17 @@ int Memento_breakAt(int event) return event; } +void *Memento_label(void *ptr, const char *label) +{ + Memento_BlkHeader *block; + + if (ptr == NULL) + return NULL; + block = MEMBLK_FROMBLK(ptr); + block->label = label; + return ptr; +} + #ifdef MEMENTO_HAS_FORK #include <unistd.h> #include <sys/wait.h> @@ -667,6 +805,9 @@ void *Memento_malloc(size_t s) memblk->sequence = globals.sequence; memblk->lastCheckedOK = memblk->sequence; memblk->flags = 0; + memblk->label = 0; + memblk->child = NULL; + memblk->sibling = NULL; Memento_addBlockHead(&globals.used, memblk, 0); return MEMBLK_TOBLK(memblk); } @@ -690,13 +831,14 @@ static int checkBlock(Memento_BlkHeader *memblk, const char *action) &data, memblk); if (!data.found) { /* Failure! */ - fprintf(stderr, "Attempt to %s block 0x%p(size=%d,num=%d) not on allocated list!\n", - action, memblk, memblk->rawsize, memblk->sequence); + fprintf(stderr, "Attempt to %s block ", action); + showBlock(memblk, 32); Memento_breakpoint(); return 1; } else if (data.preCorrupt || data.postCorrupt) { - fprintf(stderr, "Block 0x%p(size=%d,num=%d) found to be corrupted on %s!\n", - action, memblk->rawsize, memblk->sequence, action); + fprintf(stderr, "Block "); + showBlock(memblk, ' '); + fprintf(stderr, " found to be corrupted on %s!\n", action); if (data.preCorrupt) { fprintf(stderr, "Preguard corrupted\n"); } @@ -821,8 +963,8 @@ static int Memento_Internal_checkAllAlloced(Memento_BlkHeader *memblk, void *arg fprintf(stderr, "Allocated blocks:\n"); data->found |= 2; } - fprintf(stderr, " Block 0x%p(size=%d,num=%d)", - memblk, memblk->rawsize, memblk->sequence); + fprintf(stderr, " Block "); + showBlock(memblk, ' '); if (data->preCorrupt) { fprintf(stderr, " Preguard "); } @@ -850,28 +992,28 @@ static int Memento_Internal_checkAllFreed(Memento_BlkHeader *memblk, void *arg) fprintf(stderr, "Freed blocks:\n"); data->found |= 4; } - fprintf(stderr, " 0x%p(size=%d,num=%d) ", - MEMBLK_TOBLK(memblk), memblk->rawsize, memblk->sequence); + fprintf(stderr, " "); + showBlock(memblk, ' '); if (data->freeCorrupt) { - fprintf(stderr, "index %d (address 0x%p) onwards ", data->index, + fprintf(stderr, " index %d (address 0x%p) onwards", data->index, &((char *)MEMBLK_TOBLK(memblk))[data->index]); if (data->preCorrupt) { - fprintf(stderr, "+ preguard "); + fprintf(stderr, "+ preguard"); } if (data->postCorrupt) { - fprintf(stderr, "+ postguard "); + fprintf(stderr, "+ postguard"); } } else { if (data->preCorrupt) { - fprintf(stderr, " preguard "); + fprintf(stderr, " preguard"); } if (data->postCorrupt) { - fprintf(stderr, "%s Postguard ", + fprintf(stderr, "%s Postguard", (data->preCorrupt ? "+" : "")); } } - fprintf(stderr, "corrupted.\n " - "Block last checked OK at allocation %d. Now %d.\n", + fprintf(stderr, " corrupted.\n" + " Block last checked OK at allocation %d. Now %d.\n", memblk->lastCheckedOK, globals.sequence); data->preCorrupt = 0; data->postCorrupt = 0; @@ -969,22 +1111,24 @@ int Memento_find(void *a) data.flags = 0; Memento_appBlocks(&globals.used, Memento_containsAddr, &data); if (data.blk != NULL) { - fprintf(stderr, "Address 0x%p is in %sallocated block 0x%p(size=%d,num=%d)\n", + fprintf(stderr, "Address 0x%p is in %sallocated block ", data.addr, (data.flags == 1 ? "" : (data.flags == 2 ? - "preguard of " : "postguard of ")), - MEMBLK_TOBLK(data.blk), data.blk->rawsize, data.blk->sequence); + "preguard of " : "postguard of "))); + showBlock(data.blk, ' '); + fprintf(stderr, "\n"); return data.blk->sequence; } data.blk = NULL; data.flags = 0; Memento_appBlocks(&globals.free, Memento_containsAddr, &data); if (data.blk != NULL) { - fprintf(stderr, "Address 0x%p is in %sfreed block 0x%p(size=%d,num=%d)\n", + fprintf(stderr, "Address 0x%p is in %sfreed block ", data.addr, (data.flags == 1 ? "" : (data.flags == 2 ? - "preguard of " : "postguard of ")), - MEMBLK_TOBLK(data.blk), data.blk->rawsize, data.blk->sequence); + "preguard of " : "postguard of "))); + showBlock(data.blk, ' '); + fprintf(stderr, "\n"); return data.blk->sequence; } return 0; @@ -1095,8 +1239,13 @@ size_t (Memento_setMax)(size_t max) return 0; } -void Memento_stats(void) +void (Memento_stats)(void) +{ +} + +void *(Memento_label)(void *ptr, const char *label) { + return ptr; } #endif diff --git a/fitz/memento.h b/fitz/memento.h index c46b5823..b5953fe8 100644 --- a/fitz/memento.h +++ b/fitz/memento.h @@ -178,6 +178,7 @@ void Memento_listBlocks(void); void Memento_listNewBlocks(void); size_t Memento_setMax(size_t); void Memento_stats(void); +void *Memento_label(void *, const char *); void *Memento_malloc(size_t s); void *Memento_realloc(void *, size_t s); @@ -215,6 +216,7 @@ void *Memento_calloc(size_t, size_t); #define Memento_listNewBlocks() do {} while (0) #define Memento_setMax(A) 0 #define Memento_stats() do {} while (0) +#define Memento_label(A,B) (A) #endif /* MEMENTO */ |