1 9c635587 2004-06-09 devnull #include <u.h>
2 9c635587 2004-06-09 devnull #define NOPLAN9DEFINES
3 9c635587 2004-06-09 devnull #include <libc.h>
6 9c635587 2004-06-09 devnull * The Unix libc routines cannot be trusted to do their own locking.
7 9c635587 2004-06-09 devnull * Sad but apparently true.
9 9c635587 2004-06-09 devnull static int mallocpid;
12 9c635587 2004-06-09 devnull * The Unix mallocs don't do nearly enough error checking
13 9c635587 2004-06-09 devnull * for my tastes. We'll waste another 24 bytes per guy so that
14 9c635587 2004-06-09 devnull * we can. This is severely antisocial, since now free and p9free
15 9c635587 2004-06-09 devnull * are not interchangeable.
17 9c635587 2004-06-09 devnull int debugmalloc;
19 9c635587 2004-06-09 devnull #define Overhead (debugmalloc ? (6*sizeof(ulong)) : 0)
20 9c635587 2004-06-09 devnull #define MallocMagic 0xA110C09
21 9c635587 2004-06-09 devnull #define ReallocMagic 0xB110C09
22 9c635587 2004-06-09 devnull #define CallocMagic 0xC110C09
23 9c635587 2004-06-09 devnull #define FreeMagic 0xF533F533
24 9c635587 2004-06-09 devnull #define CheckMagic 0
25 9c635587 2004-06-09 devnull #define END "\x7F\x2E\x55\x23"
27 9c635587 2004-06-09 devnull static void
28 9c635587 2004-06-09 devnull whoops(void *v)
30 9c635587 2004-06-09 devnull fprint(2, "bad malloc block %p\n", v);
34 9c635587 2004-06-09 devnull static void*
35 9c635587 2004-06-09 devnull mark(void *v, ulong pc, ulong n, ulong magic)
37 9c635587 2004-06-09 devnull ulong *u;
40 9c635587 2004-06-09 devnull if(!debugmalloc)
41 9c635587 2004-06-09 devnull return v;
43 9c635587 2004-06-09 devnull if(v == nil)
44 9c635587 2004-06-09 devnull return nil;
46 9c635587 2004-06-09 devnull if(magic == FreeMagic || magic == CheckMagic){
47 9c635587 2004-06-09 devnull u = (ulong*)((char*)v-4*sizeof(ulong));
48 9c635587 2004-06-09 devnull if(u[0] != MallocMagic && u[0] != ReallocMagic && u[0] != CallocMagic)
49 9c635587 2004-06-09 devnull whoops(v);
50 9c635587 2004-06-09 devnull n = u[1];
51 9c635587 2004-06-09 devnull p = (char*)v+n;
52 9c635587 2004-06-09 devnull if(memcmp(p, END, 4) != 0)
53 9c635587 2004-06-09 devnull whoops(v);
54 9c635587 2004-06-09 devnull if(magic != CheckMagic){
55 9c635587 2004-06-09 devnull u[0] = FreeMagic;
56 9c635587 2004-06-09 devnull u[1] = u[2] = u[3] = pc;
57 9c635587 2004-06-09 devnull if(n > 16){
58 9c635587 2004-06-09 devnull u[4] = u[5] = u[6] = u[7] = pc;
59 9c635587 2004-06-09 devnull memset((char*)v+16, 0xFB, n-16);
62 9c635587 2004-06-09 devnull return u;
65 9c635587 2004-06-09 devnull u[0] = magic;
66 9c635587 2004-06-09 devnull u[1] = n;
67 9c635587 2004-06-09 devnull u[2] = 0;
68 9c635587 2004-06-09 devnull u[3] = 0;
69 9c635587 2004-06-09 devnull if(magic == ReallocMagic)
70 9c635587 2004-06-09 devnull u[3] = pc;
72 9c635587 2004-06-09 devnull u[2] = pc;
73 9c635587 2004-06-09 devnull p = (char*)(u+4)+n;
74 9c635587 2004-06-09 devnull memmove(p, END, 4);
75 9c635587 2004-06-09 devnull return u+4;
80 9c635587 2004-06-09 devnull setmalloctag(void *v, ulong t)
82 9c635587 2004-06-09 devnull ulong *u;
84 9c635587 2004-06-09 devnull if(!debugmalloc)
87 9c635587 2004-06-09 devnull if(v == nil)
89 9c635587 2004-06-09 devnull u = mark(v, 0, 0, 0);
90 9c635587 2004-06-09 devnull u[2] = t;
94 9c635587 2004-06-09 devnull setrealloctag(void *v, ulong t)
96 9c635587 2004-06-09 devnull ulong *u;
98 9c635587 2004-06-09 devnull if(!debugmalloc)
101 9c635587 2004-06-09 devnull if(v == nil)
103 9c635587 2004-06-09 devnull u = mark(v, 0, 0, 0);
104 9c635587 2004-06-09 devnull u[3] = t;
108 9c635587 2004-06-09 devnull p9malloc(ulong n)
110 9c635587 2004-06-09 devnull void *v;
111 9c635587 2004-06-09 devnull if(n == 0)
113 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d malloc\n", argv0, getpid()); */
114 9c635587 2004-06-09 devnull mallocpid = getpid();
115 9c635587 2004-06-09 devnull v = malloc(n+Overhead);
116 9c635587 2004-06-09 devnull v = mark(v, getcallerpc(&n), n, MallocMagic);
117 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d donemalloc\n", argv0, getpid()); */
118 9c635587 2004-06-09 devnull return v;
122 9c635587 2004-06-09 devnull p9free(void *v)
124 9c635587 2004-06-09 devnull if(v == nil)
127 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d free\n", argv0, getpid()); */
128 9c635587 2004-06-09 devnull mallocpid = getpid();
129 9c635587 2004-06-09 devnull v = mark(v, getcallerpc(&v), 0, FreeMagic);
130 9c635587 2004-06-09 devnull free(v);
131 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d donefree\n", argv0, getpid()); */
135 9c635587 2004-06-09 devnull p9calloc(ulong a, ulong b)
137 9c635587 2004-06-09 devnull void *v;
139 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d calloc\n", argv0, getpid()); */
140 9c635587 2004-06-09 devnull mallocpid = getpid();
141 9c635587 2004-06-09 devnull v = calloc(a*b+Overhead, 1);
142 9c635587 2004-06-09 devnull v = mark(v, getcallerpc(&a), a*b, CallocMagic);
143 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d donecalloc\n", argv0, getpid()); */
144 9c635587 2004-06-09 devnull return v;
148 9c635587 2004-06-09 devnull p9realloc(void *v, ulong n)
150 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d realloc\n", argv0, getpid()); */
151 9c635587 2004-06-09 devnull mallocpid = getpid();
152 9c635587 2004-06-09 devnull v = mark(v, getcallerpc(&v), 0, CheckMagic);
153 9c635587 2004-06-09 devnull v = realloc(v, n+Overhead);
154 9c635587 2004-06-09 devnull v = mark(v, getcallerpc(&v), n, ReallocMagic);
155 cbeb0b26 2006-04-01 devnull /*fprint(2, "%s %d donerealloc\n", argv0, getpid()); */
156 9c635587 2004-06-09 devnull return v;