]> de.git.xonotic.org Git - xonotic/gmqcc.git/blob - util.c
Fix potential bug with derefrencing a NULL pointer
[xonotic/gmqcc.git] / util.c
1 /*
2  * Copyright (C) 2012 
3  *      Dale Weiler
4  *
5  * Permission is hereby granted, free of charge, to any person obtaining a copy of
6  * this software and associated documentation files (the "Software"), to deal in
7  * the Software without restriction, including without limitation the rights to
8  * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
9  * of the Software, and to permit persons to whom the Software is furnished to do
10  * so, subject to the following conditions:
11  *
12  * The above copyright notice and this permission notice shall be included in all
13  * copies or substantial portions of the Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21  * SOFTWARE.
22  */
23 #include <stdarg.h>
24 #include <errno.h>
25 #include "gmqcc.h"
26  
27 struct memblock_t {
28         const char  *file;
29         unsigned int line;
30         unsigned int byte;
31 };
32
33 void *util_memory_a(unsigned int byte, unsigned int line, const char *file) {
34         struct memblock_t *data = malloc(sizeof(struct memblock_t) + byte);
35         if (!data) return NULL;
36         data->line = line;
37         data->byte = byte;
38         data->file = file;
39         
40         util_debug("MEM", "allocation: %08u (bytes) at %s:%u\n", byte, file, line);
41         return (void*)((uintptr_t)data+sizeof(struct memblock_t));
42 }
43
44 void util_memory_d(void *ptrn, unsigned int line, const char *file) {
45         if (!ptrn) return;
46         void              *data = (void*)((uintptr_t)ptrn-sizeof(struct memblock_t));
47         struct memblock_t *info = (struct memblock_t*)data;
48         
49         util_debug("MEM", "released:   %08u (bytes) at %s:%u\n", info->byte, file, line);
50         free(data);
51 }
52
53 #ifndef mem_d
54 #define mem_d(x) util_memory_d((x), __LINE__, __FILE__)
55 #endif
56 #ifndef mem_a
57 #define mem_a(x) util_memory_a((x), __LINE__, __FILE__)
58 #endif
59
60 /*
61  * Some string utility functions, because strdup uses malloc, and we want
62  * to track all memory (without replacing malloc).
63  */
64 char *util_strdup(const char *s) {
65         size_t  len;
66         char   *ptr;
67         
68         if (!s)
69                 return NULL;
70                 
71         len = strlen(s);
72         ptr = mem_a (len+1);
73         
74         if (ptr && len) {
75                 memcpy(ptr, s, len);
76                 ptr[len] = '\0';
77         }
78         
79         return ptr;
80 }
81
82 /*
83  * Remove quotes from a string, escapes from \ in string
84  * as well.  This function shouldn't be used to create a
85  * char array that is later freed (it uses pointer arith)
86  */
87 char *util_strrq(char *s) {
88         char *dst = s;
89         char *src = s;
90         char  chr;
91         while ((chr = *src++) != '\0') {
92                 if (chr == '\\') {
93                         *dst++ = chr;
94                         if ((chr = *src++) == '\0')
95                                 break;
96                         *dst++ = chr;
97                 } else if (chr != '"')
98                         *dst++ = chr;
99         }
100         *dst = '\0';
101         return dst;
102 }
103
104 /*
105  * Remove newline from a string (if it exists).  This is
106  * done pointer wise instead of strlen(), and an array
107  * access.
108  */
109 char *util_strrnl(char *src) {
110         if (!src) return NULL;
111         char   *cpy = src;
112         while (*cpy && *cpy != '\n')
113                 cpy++;
114                 
115         *cpy = '\0';
116         return src;
117 }
118
119 void util_debug(const char *area, const char *ms, ...) {
120         va_list  va;
121         va_start(va, ms);
122         fprintf (stdout, "DEBUG: ");
123         fputc   ('[',  stdout);
124         fprintf (stdout, area);
125         fputs   ("] ", stdout);
126         vfprintf(stdout, ms, va);
127         va_end  (va);
128 }
129
130 /*
131  * Implements libc getline for systems that don't have it, which is 
132  * assmed all.  This works the same as getline().
133  */
134 int util_getline(char **lineptr, size_t *n, FILE *stream) {
135         int   chr;
136         int   ret;
137         char *pos;      
138
139         if (!lineptr || !n || !stream)
140                 return -1;
141         if (!*lineptr) {
142                 if (!(*lineptr = mem_a((*n = 64))))
143                         return -1;
144         }
145
146         chr = *n;
147         pos = *lineptr;
148
149         for (;;) {
150                 int c = getc(stream);
151                 
152                 if (chr < 2) {
153                         char *tmp = mem_a((*n+=(*n>16)?*n:64));
154                         if  (!tmp)
155                                 return -1;
156                         
157                         chr = *n + *lineptr - pos;
158                         strcpy(tmp,*lineptr);
159                         
160                         if (!(*lineptr = tmp))
161                                 return -1;
162                                 
163                         pos = *n - chr + *lineptr;
164                 }
165
166                 if (ferror(stream)) 
167                         return -1;
168                 if (c == EOF) {
169                         if (pos == *lineptr)
170                                 return -1;
171                         else
172                                 break;
173                 }
174
175                 *pos++ = c;
176                  chr--;
177                 if (c == '\n')
178                         break;
179         }
180         *pos = '\0';
181         return (ret = pos - *lineptr);
182 }