/* * Copyright (C) 2012 * Dale Weiler * * Permission is hereby granted, free of charge, to any person obtaining a copy of * this software and associated documentation files (the "Software"), to deal in * the Software without restriction, including without limitation the rights to * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies * of the Software, and to permit persons to whom the Software is furnished to do * so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include #include "gmqcc.h" unsigned long long mem_ab = 0; unsigned long long mem_db = 0; unsigned long long mem_at = 0; unsigned long long mem_dt = 0; struct memblock_t { const char *file; unsigned int line; unsigned int byte; }; void *util_memory_a(unsigned int byte, unsigned int line, const char *file) { struct memblock_t *info = malloc(sizeof(struct memblock_t) + byte); void *data =(void*)((uintptr_t)info+sizeof(struct memblock_t)); if (!data) return NULL; info->line = line; info->byte = byte; info->file = file; util_debug("MEM", "allocation: % 8u (bytes) address 0x%08X @ %s:%u\n", byte, data, file, line); mem_at++; mem_ab += info->byte; return data; } void util_memory_d(void *ptrn, unsigned int line, const char *file) { if (!ptrn) return; void *data = (void*)((uintptr_t)ptrn-sizeof(struct memblock_t)); struct memblock_t *info = (struct memblock_t*)data; util_debug("MEM", "released: % 8u (bytes) address 0x%08X @ %s:%u\n", info->byte, data, file, line); mem_db += info->byte; mem_dt++; free(data); } void util_meminfo() { util_debug("MEM", "Memory information:\n\ Total allocations: %llu\n\ Total deallocations: %llu\n\ Total allocated: %llu (bytes)\n\ Total deallocated: %llu (bytes)\n", mem_at, mem_dt, mem_ab, mem_db ); } //#ifndef mem_d //#define mem_d(x) util_memory_d((x), __LINE__, __FILE__) //#endif //#ifndef mem_a //#define mem_a(x) util_memory_a((x), __LINE__, __FILE__) //#endif /* * Some string utility functions, because strdup uses malloc, and we want * to track all memory (without replacing malloc). */ char *util_strdup(const char *s) { size_t len = 0; char *ptr = NULL; if (!s) return NULL; if ((len = strlen(s)) && (ptr = mem_a(len+1))) { memcpy(ptr, s, len); ptr[len] = '\0'; } return ptr; } /* * Remove quotes from a string, escapes from \ in string * as well. This function shouldn't be used to create a * char array that is later freed (it uses pointer arith) */ char *util_strrq(char *s) { char *dst = s; char *src = s; char chr; while ((chr = *src++) != '\0') { if (chr == '\\') { *dst++ = chr; if ((chr = *src++) == '\0') break; *dst++ = chr; } else if (chr != '"') *dst++ = chr; } *dst = '\0'; return dst; } /* * Remove newline from a string (if it exists). This is * done pointer wise instead of strlen(), and an array * access. */ char *util_strrnl(char *src) { if (!src) return NULL; char *cpy = src; while (*cpy && *cpy != '\n') cpy++; *cpy = '\0'; return src; } void util_debug(const char *area, const char *ms, ...) { va_list va; va_start(va, ms); fprintf (stdout, "DEBUG: "); fputc ('[', stdout); fprintf(stdout, "%s", area); fputs ("] ", stdout); vfprintf(stdout, ms, va); va_end (va); } /* * Endianess swapping, all data must be stored little-endian. This * reorders by stride and length, much nicer than other functions for * certian-sized types like short or int. */ void util_endianswap(void *m, int s, int l) { size_t w = 0; size_t i = 0; /* ignore if we're already LE */ if(*((char *)&s)) return; for(; w < l; w++) { for(; i < s << 1; i++) { unsigned char *p = (unsigned char *)m+w*s; unsigned char t = p[i]; p[i] = p[s-i-1]; p[s-i-1] = t; } } } /* * Implements libc getline for systems that don't have it, which is * assmed all. This works the same as getline(). */ int util_getline(char **lineptr, size_t *n, FILE *stream) { int chr; int ret; char *pos; if (!lineptr || !n || !stream) return -1; if (!*lineptr) { if (!(*lineptr = mem_a((*n = 64)))) return -1; } chr = *n; pos = *lineptr; for (;;) { int c = getc(stream); if (chr < 2) { char *tmp = mem_a((*n+=(*n>16)?*n:64)); if (!tmp) return -1; chr = *n + *lineptr - pos; strcpy(tmp,*lineptr); if (!(*lineptr = tmp)) { mem_d (tmp); return -1; } pos = *n - chr + *lineptr; } if (ferror(stream)) return -1; if (c == EOF) { if (pos == *lineptr) return -1; else break; } *pos++ = c; chr--; if (c == '\n') break; } *pos = '\0'; return (ret = pos - *lineptr); }