annotate src/fftw-3.3.3/kernel/alloc.c @ 23:619f715526df sv_v2.1

Update Vamp plugin SDK to 2.5
author Chris Cannam
date Thu, 09 May 2013 10:52:46 +0100
parents 37bf6b4a2645
children
rev   line source
Chris@10 1 /*
Chris@10 2 * Copyright (c) 2003, 2007-11 Matteo Frigo
Chris@10 3 * Copyright (c) 2003, 2007-11 Massachusetts Institute of Technology
Chris@10 4 *
Chris@10 5 * This program is free software; you can redistribute it and/or modify
Chris@10 6 * it under the terms of the GNU General Public License as published by
Chris@10 7 * the Free Software Foundation; either version 2 of the License, or
Chris@10 8 * (at your option) any later version.
Chris@10 9 *
Chris@10 10 * This program is distributed in the hope that it will be useful,
Chris@10 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Chris@10 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Chris@10 13 * GNU General Public License for more details.
Chris@10 14 *
Chris@10 15 * You should have received a copy of the GNU General Public License
Chris@10 16 * along with this program; if not, write to the Free Software
Chris@10 17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Chris@10 18 *
Chris@10 19 */
Chris@10 20
Chris@10 21 #include "ifftw.h"
Chris@10 22
Chris@10 23 /**********************************************************
Chris@10 24 * DEBUGGING CODE
Chris@10 25 **********************************************************/
Chris@10 26 #if defined(FFTW_DEBUG_MALLOC)
Chris@10 27
Chris@10 28 #include <stdio.h>
Chris@10 29
Chris@10 30 /*
Chris@10 31 debugging malloc/free.
Chris@10 32
Chris@10 33 1) Initialize every malloced and freed area to random values, just
Chris@10 34 to make sure we are not using uninitialized pointers.
Chris@10 35
Chris@10 36 2) check for blocks freed twice.
Chris@10 37
Chris@10 38 3) Check for writes past the ends of allocated blocks
Chris@10 39
Chris@10 40 4) destroy contents of freed blocks in order to detect incorrect reuse.
Chris@10 41
Chris@10 42 5) keep track of who allocates what and report memory leaks
Chris@10 43
Chris@10 44 This code is a quick and dirty hack. May be nonportable.
Chris@10 45 Use at your own risk.
Chris@10 46
Chris@10 47 */
Chris@10 48
Chris@10 49 #define MAGIC ((size_t)0xABadCafe)
Chris@10 50 #define PAD_FACTOR 2
Chris@10 51 #define SZ_HEADER (4 * sizeof(size_t))
Chris@10 52 #define HASHSZ 1031
Chris@10 53
Chris@10 54 static unsigned int hashaddr(void *p)
Chris@10 55 {
Chris@10 56 return ((unsigned long)p) % HASHSZ;
Chris@10 57 }
Chris@10 58
Chris@10 59 struct mstat {
Chris@10 60 int siz;
Chris@10 61 int maxsiz;
Chris@10 62 int cnt;
Chris@10 63 int maxcnt;
Chris@10 64 };
Chris@10 65
Chris@10 66 static struct mstat mstat[MALLOC_WHAT_LAST];
Chris@10 67
Chris@10 68 struct minfo {
Chris@10 69 const char *file;
Chris@10 70 int line;
Chris@10 71 size_t n;
Chris@10 72 void *p;
Chris@10 73 struct minfo *next;
Chris@10 74 };
Chris@10 75
Chris@10 76 static struct minfo *minfo[HASHSZ] = {0};
Chris@10 77
Chris@10 78 #if defined(HAVE_THREADS) || defined(HAVE_OPENMP)
Chris@10 79 int X(in_thread) = 0;
Chris@10 80 #endif
Chris@10 81
Chris@10 82 void *X(malloc_debug)(size_t n, enum malloc_tag what,
Chris@10 83 const char *file, int line)
Chris@10 84 {
Chris@10 85 char *p;
Chris@10 86 size_t i;
Chris@10 87 struct minfo *info;
Chris@10 88 struct mstat *stat = mstat + what;
Chris@10 89 struct mstat *estat = mstat + EVERYTHING;
Chris@10 90
Chris@10 91 if (n == 0)
Chris@10 92 n = 1;
Chris@10 93
Chris@10 94 if (!IN_THREAD) {
Chris@10 95 stat->siz += n;
Chris@10 96 if (stat->siz > stat->maxsiz)
Chris@10 97 stat->maxsiz = stat->siz;
Chris@10 98 estat->siz += n;
Chris@10 99 if (estat->siz > estat->maxsiz)
Chris@10 100 estat->maxsiz = estat->siz;
Chris@10 101 }
Chris@10 102
Chris@10 103 p = (char *) X(kernel_malloc)(PAD_FACTOR * n + SZ_HEADER);
Chris@10 104 A(p);
Chris@10 105
Chris@10 106 /* store the sz in a known position */
Chris@10 107 ((size_t *) p)[0] = n;
Chris@10 108 ((size_t *) p)[1] = MAGIC;
Chris@10 109 ((size_t *) p)[2] = what;
Chris@10 110
Chris@10 111 /* fill with junk */
Chris@10 112 for (i = 0; i < PAD_FACTOR * n; i++)
Chris@10 113 p[i + SZ_HEADER] = (char) (i ^ 0xEF);
Chris@10 114
Chris@10 115 if (!IN_THREAD) {
Chris@10 116 ++stat->cnt;
Chris@10 117 ++estat->cnt;
Chris@10 118
Chris@10 119 if (stat->cnt > stat->maxcnt)
Chris@10 120 stat->maxcnt = stat->cnt;
Chris@10 121 if (estat->cnt > estat->maxcnt)
Chris@10 122 estat->maxcnt = estat->cnt;
Chris@10 123 }
Chris@10 124
Chris@10 125 /* skip the info we stored previously */
Chris@10 126 p = p + SZ_HEADER;
Chris@10 127
Chris@10 128 if (!IN_THREAD) {
Chris@10 129 unsigned int h = hashaddr(p);
Chris@10 130 /* record allocation in allocation list */
Chris@10 131 info = (struct minfo *) malloc(sizeof(struct minfo));
Chris@10 132 info->n = n;
Chris@10 133 info->file = file;
Chris@10 134 info->line = line;
Chris@10 135 info->p = p;
Chris@10 136 info->next = minfo[h];
Chris@10 137 minfo[h] = info;
Chris@10 138 }
Chris@10 139
Chris@10 140 return (void *) p;
Chris@10 141 }
Chris@10 142
Chris@10 143 void X(ifree)(void *p)
Chris@10 144 {
Chris@10 145 char *q;
Chris@10 146
Chris@10 147 A(p);
Chris@10 148
Chris@10 149 q = ((char *) p) - SZ_HEADER;
Chris@10 150 A(q);
Chris@10 151
Chris@10 152 {
Chris@10 153 size_t n = ((size_t *) q)[0];
Chris@10 154 size_t magic = ((size_t *) q)[1];
Chris@10 155 int what = ((size_t *) q)[2];
Chris@10 156 size_t i;
Chris@10 157 struct mstat *stat = mstat + what;
Chris@10 158 struct mstat *estat = mstat + EVERYTHING;
Chris@10 159
Chris@10 160 /* set to zero to detect duplicate free's */
Chris@10 161 ((size_t *) q)[0] = 0;
Chris@10 162
Chris@10 163 A(magic == MAGIC);
Chris@10 164 ((size_t *) q)[1] = ~MAGIC;
Chris@10 165
Chris@10 166 if (!IN_THREAD) {
Chris@10 167 stat->siz -= n;
Chris@10 168 A(stat->siz >= 0);
Chris@10 169 estat->siz -= n;
Chris@10 170 A(estat->siz >= 0);
Chris@10 171 }
Chris@10 172
Chris@10 173 /* check for writing past end of array: */
Chris@10 174 for (i = n; i < PAD_FACTOR * n; ++i)
Chris@10 175 if (q[i + SZ_HEADER] != (char) (i ^ 0xEF)) {
Chris@10 176 A(0 /* array bounds overwritten */ );
Chris@10 177 }
Chris@10 178 for (i = 0; i < PAD_FACTOR * n; ++i)
Chris@10 179 q[i + SZ_HEADER] = (char) (i ^ 0xAD);
Chris@10 180
Chris@10 181 if (!IN_THREAD) {
Chris@10 182 --stat->cnt;
Chris@10 183 --estat->cnt;
Chris@10 184
Chris@10 185 A(stat->cnt >= 0);
Chris@10 186 A((stat->cnt == 0 && stat->siz == 0) ||
Chris@10 187 (stat->cnt > 0 && stat->siz > 0));
Chris@10 188 A(estat->cnt >= 0);
Chris@10 189 A((estat->cnt == 0 && estat->siz == 0) ||
Chris@10 190 (estat->cnt > 0 && estat->siz > 0));
Chris@10 191 }
Chris@10 192
Chris@10 193 X(kernel_free)(q);
Chris@10 194 }
Chris@10 195
Chris@10 196 if (!IN_THREAD) {
Chris@10 197 /* delete minfo entry */
Chris@10 198 unsigned int h = hashaddr(p);
Chris@10 199 struct minfo **i;
Chris@10 200
Chris@10 201 for (i = minfo + h; *i; i = &((*i)->next)) {
Chris@10 202 if ((*i)->p == p) {
Chris@10 203 struct minfo *i0 = (*i)->next;
Chris@10 204 free(*i);
Chris@10 205 *i = i0;
Chris@10 206 return;
Chris@10 207 }
Chris@10 208 }
Chris@10 209
Chris@10 210 A(0 /* no entry in minfo list */ );
Chris@10 211 }
Chris@10 212 }
Chris@10 213
Chris@10 214 void X(malloc_print_minfo)(int verbose)
Chris@10 215 {
Chris@10 216 struct minfo *info;
Chris@10 217 int what;
Chris@10 218 unsigned int h;
Chris@10 219 int leak = 0;
Chris@10 220
Chris@10 221 if (verbose > 2) {
Chris@10 222 static const char *names[MALLOC_WHAT_LAST] = {
Chris@10 223 "EVERYTHING",
Chris@10 224 "PLANS", "SOLVERS", "PROBLEMS", "BUFFERS",
Chris@10 225 "HASHT", "TENSORS", "PLANNERS", "SLVDSC", "TWIDDLES",
Chris@10 226 "STRIDES", "OTHER"
Chris@10 227 };
Chris@10 228
Chris@10 229 printf("%12s %8s %8s %10s %10s\n",
Chris@10 230 "what", "cnt", "maxcnt", "siz", "maxsiz");
Chris@10 231
Chris@10 232 for (what = 0; what < MALLOC_WHAT_LAST; ++what) {
Chris@10 233 struct mstat *stat = mstat + what;
Chris@10 234 printf("%12s %8d %8d %10d %10d\n",
Chris@10 235 names[what], stat->cnt, stat->maxcnt,
Chris@10 236 stat->siz, stat->maxsiz);
Chris@10 237 }
Chris@10 238 }
Chris@10 239
Chris@10 240 for (h = 0; h < HASHSZ; ++h)
Chris@10 241 if (minfo[h]) {
Chris@10 242 printf("\nUnfreed allocations:\n");
Chris@10 243 break;
Chris@10 244 }
Chris@10 245
Chris@10 246 for (h = 0; h < HASHSZ; ++h)
Chris@10 247 for (info = minfo[h]; info; info = info->next) {
Chris@10 248 leak = 1;
Chris@10 249 printf("%s:%d: %zd bytes at %p\n",
Chris@10 250 info->file, info->line, info->n, info->p);
Chris@10 251 }
Chris@10 252
Chris@10 253 if (leak)
Chris@10 254 abort();
Chris@10 255 }
Chris@10 256
Chris@10 257 #else
Chris@10 258 /**********************************************************
Chris@10 259 * NON DEBUGGING CODE
Chris@10 260 **********************************************************/
Chris@10 261 /* production version, no hacks */
Chris@10 262
Chris@10 263 void *X(malloc_plain)(size_t n)
Chris@10 264 {
Chris@10 265 void *p;
Chris@10 266 if (n == 0)
Chris@10 267 n = 1;
Chris@10 268 p = X(kernel_malloc)(n);
Chris@10 269 CK(p);
Chris@10 270
Chris@10 271 #ifdef MIN_ALIGNMENT
Chris@10 272 A((((uintptr_t)p) % MIN_ALIGNMENT) == 0);
Chris@10 273 #endif
Chris@10 274
Chris@10 275 return p;
Chris@10 276 }
Chris@10 277
Chris@10 278 void X(ifree)(void *p)
Chris@10 279 {
Chris@10 280 X(kernel_free)(p);
Chris@10 281 }
Chris@10 282
Chris@10 283 #endif
Chris@10 284
Chris@10 285 void X(ifree0)(void *p)
Chris@10 286 {
Chris@10 287 /* common pattern */
Chris@10 288 if (p) X(ifree)(p);
Chris@10 289 }