/************************************************************************ * Copyright (C) 2002-2009, Xiph.org Foundation * Copyright (C) 2010, Robin Watts for Pinknoise Productions Ltd * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the names of the Xiph.org Foundation nor Pinknoise * Productions Ltd nor the names of its contributors may be used to * endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ************************************************************************/ #define HEAD_ALIGN 64 #include <stdlib.h> #include <string.h> #include <stdio.h> #define MISC_C #include "misc.h" //#include <sys/time.h> static void **pointers=NULL; static long *insertlist=NULL; /* We can't embed this in the pointer list; a pointer can have any value... */ static char **files=NULL; static long *file_bytes=NULL; static int filecount=0; static int ptop=0; static int palloced=0; static int pinsert=0; typedef struct { char *file; long line; long ptr; long bytes; } head; long global_bytes=0; long start_time=-1; static void *_insert(void *ptr,long bytes,char *file,long line){ ((head *)ptr)->file=file; ((head *)ptr)->line=line; ((head *)ptr)->ptr=pinsert; ((head *)ptr)->bytes=bytes-HEAD_ALIGN; if(pinsert>=palloced){ palloced+=64; if(pointers){ pointers=(void **)realloc(pointers,sizeof(void **)*palloced); insertlist=(long *)realloc(insertlist,sizeof(long *)*palloced); }else{ pointers=(void **)malloc(sizeof(void **)*palloced); insertlist=(long *)malloc(sizeof(long *)*palloced); } } pointers[pinsert]=ptr; if(pinsert==ptop) pinsert=++ptop; else pinsert=insertlist[pinsert]; #ifdef _VDBG_GRAPHFILE { FILE *out; struct timeval tv; static struct timezone tz; int i; char buffer[80]; gettimeofday(&tv,&tz); for(i=0;i<filecount;i++) if(!strcmp(file,files[i]))break; if(i==filecount){ filecount++; if(!files){ files=malloc(filecount*sizeof(*files)); file_bytes=malloc(filecount*sizeof(*file_bytes)); }else{ files=realloc(files,filecount*sizeof(*files)); file_bytes=realloc(file_bytes,filecount*sizeof(*file_bytes)); } files[i]=strdup(file); file_bytes[i]=0; } file_bytes[i]+=bytes-HEAD_ALIGN; if(start_time==-1)start_time=(tv.tv_sec*1000)+(tv.tv_usec/1000); snprintf(buffer,80,"%s",file); if(strchr(buffer,'.'))strchr(buffer,'.')[0]=0; strcat(buffer,_VDBG_GRAPHFILE); out=fopen(buffer,"a"); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), file_bytes[i]-(bytes-HEAD_ALIGN)); fprintf(out,"%ld, %ld # FILE %s LINE %ld\n", -start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), file_bytes[i],file,line); fclose(out); out=fopen("total"_VDBG_GRAPHFILE,"a"); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), global_bytes); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), global_bytes+(bytes-HEAD_ALIGN)); fclose(out); } #endif global_bytes+=(bytes-HEAD_ALIGN); return(void *)(((char *)ptr)+HEAD_ALIGN); } static void _ripremove(void *ptr){ int insert; #ifdef _VDBG_GRAPHFILE { FILE *out=fopen("total"_VDBG_GRAPHFILE,"a"); struct timeval tv; static struct timezone tz; char buffer[80]; char *file =((head *)ptr)->file; long bytes =((head *)ptr)->bytes; int i; gettimeofday(&tv,&tz); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), global_bytes); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), global_bytes-((head *)ptr)->bytes); fclose(out); for(i=0;i<filecount;i++) if(!strcmp(file,files[i]))break; snprintf(buffer,80,"%s",file); if(strchr(buffer,'.'))strchr(buffer,'.')[0]=0; strcat(buffer,_VDBG_GRAPHFILE); out=fopen(buffer,"a"); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), file_bytes[i]); fprintf(out,"%ld, %ld\n",-start_time+(tv.tv_sec*1000)+(tv.tv_usec/1000), file_bytes[i]-bytes); fclose(out); file_bytes[i]-=bytes; } #endif global_bytes-=((head *)ptr)->bytes; insert=((head *)ptr)->ptr; insertlist[insert]=pinsert; pinsert=insert; if(pointers[insert]==NULL){ fprintf(stderr,"DEBUGGING MALLOC ERROR: freeing previously freed memory\n"); fprintf(stderr,"\t%s %ld\n",((head *)ptr)->file,((head *)ptr)->line); } if(global_bytes<0){ fprintf(stderr,"DEBUGGING MALLOC ERROR: freeing unmalloced memory\n"); } pointers[insert]=NULL; } void _VDBG_dump(void){ int i; for(i=0;i<ptop;i++){ head *ptr=pointers[i]; if(ptr) fprintf(stderr,"unfreed bytes from %s:%ld\n", ptr->file,ptr->line); } } extern void *_VDBG_malloc(void *ptr,long bytes,char *file,long line){ bytes+=HEAD_ALIGN; if(ptr){ ptr=(void *)(((char *)ptr)-HEAD_ALIGN); _ripremove(ptr); ptr=realloc(ptr,bytes); }else{ ptr=malloc(bytes); memset(ptr,0,bytes); } return _insert(ptr,bytes,file,line); } extern void _VDBG_free(void *ptr,char *file,long line){ if(ptr){ ptr=(void *)(((char *)ptr)-HEAD_ALIGN); _ripremove(ptr); free(ptr); } }