2
/*****************************************************************************
2
/***************************************************************************
4
4
* Project ___| | | | _ \| |
5
5
* / __| | | | |_) | |
6
6
* | (__| |_| | _ <| |___
7
7
* \___|\___/|_| \_\_____|
9
* Copyright (C) 2000, Daniel Stenberg, <daniel@haxx.se>, et al.
11
* In order to be useful for every potential user, curl and libcurl are
12
* dual-licensed under the MPL and the MIT/X-derivate licenses.
9
* Copyright (C) 1998 - 2004, Daniel Stenberg, <daniel@haxx.se>, et al.
11
* This software is licensed as described in the file COPYING, which
12
* you should have received as part of this distribution. The terms
13
* are also available at http://curl.haxx.se/docs/copyright.html.
14
15
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
15
16
* copies of the Software, and permit persons to whom the Software is
16
* furnished to do so, under the terms of the MPL or the MIT/X-derivate
17
* licenses. You may pick one of these licenses.
17
* furnished to do so, under the terms of the COPYING file.
19
19
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
20
20
* KIND, either express or implied.
22
* $Id: memdebug.c,v 1.22 2002/02/28 12:37:05 bagder Exp $
23
*****************************************************************************/
22
* $Id: memdebug.c,v 1.41 2004/03/08 11:33:49 bagder Exp $
23
***************************************************************************/
27
27
#include <curl/curl.h>
29
#if defined(WIN32) && !defined(__GNUC__) || defined(__MINGW32__)
31
#else /* some kind of unix */
32
29
#ifdef HAVE_SYS_SOCKET_H
33
30
#include <sys/socket.h>
37
33
#define _MPRINTF_REPLACE
38
34
#include <curl/mprintf.h>
45
41
#include <unistd.h>
48
/* DONT include memdebug.h here! */
44
#define MEMDEBUG_NODEFINES /* don't redefine the standard functions */
50
/* I'm hoping this is the thing with the strictest alignment
51
* requirements. That also means we waste some space :-( */
60
59
* Don't use these with multithreaded test programs!
62
#define logfile curl_debuglogfile
63
FILE *curl_debuglogfile;
64
static bool memlimit; /* enable memory limit */
65
static long memsize; /* set number of mallocs allowed */
65
67
/* this sets the log file name */
66
68
void curl_memdebug(const char *logname)
76
/* This function sets the number of malloc() calls that should return
78
void curl_memlimit(long limit)
84
/* returns TRUE if this isn't allowed! */
85
static bool countcheck(const char *func, int line, const char *source)
87
/* if source is NULL, then the call is made internally and this check
89
if(memlimit && source) {
92
fprintf(logfile, "LIMIT %s:%d %s reached memlimit\n",
94
return TRUE; /* RETURN ERROR! */
97
memsize--; /* countdown */
99
/* log the countdown */
100
if(logfile && source)
101
fprintf(logfile, "LIMIT %s:%d %ld ALLOCS left\n",
102
source, line, memsize);
106
return FALSE; /* allow this */
75
109
void *curl_domalloc(size_t wantedsize, int line, const char *source)
77
111
struct memdebug *mem;
114
if(countcheck("malloc", line, source))
80
117
/* alloc at least 64 bytes */
81
118
size = sizeof(struct memdebug)+wantedsize;
90
127
if(logfile && source)
91
fprintf(logfile, "MEM %s:%d malloc(%d) = %p\n",
128
fprintf(logfile, "MEM %s:%d malloc(%zd) = %p\n",
92
129
source, line, wantedsize, mem->mem);
133
void *curl_docalloc(size_t wanted_elements, size_t wanted_size,
134
int line, const char *source)
136
struct memdebug *mem;
137
size_t size, user_size;
139
if(countcheck("calloc", line, source))
142
/* alloc at least 64 bytes */
143
user_size = wanted_size * wanted_elements;
144
size = sizeof(struct memdebug) + user_size;
146
mem = (struct memdebug *)(malloc)(size);
148
/* fill memory with zeroes */
149
memset(mem->mem, 0, user_size);
150
mem->size = user_size;
153
if(logfile && source)
154
fprintf(logfile, "MEM %s:%d calloc(%u,%u) = %p\n",
155
source, line, wanted_elements, wanted_size, mem->mem);
96
159
char *curl_dostrdup(const char *str, int line, const char *source)
164
curlassert(str != NULL);
102
fprintf(stderr, "ILLEGAL strdup() on NULL at %s:%d\n",
166
if(countcheck("strdup", line, source))
107
169
len=strlen(str)+1;
110
172
memcpy(mem, str, len);
113
fprintf(logfile, "MEM %s:%d strdup(%p) (%d) = %p\n",
175
fprintf(logfile, "MEM %s:%d strdup(%p) (%zd) = %p\n",
114
176
source, line, str, len, mem);
181
/* We provide a realloc() that accepts a NULL as pointer, which then
182
performs a malloc(). In order to work with ares. */
119
183
void *curl_dorealloc(void *ptr, size_t wantedsize,
120
184
int line, const char *source)
122
struct memdebug *mem;
186
struct memdebug *mem=NULL;
124
188
size_t size = sizeof(struct memdebug)+wantedsize;
126
mem = (struct memdebug *)((char *)ptr - offsetof(struct memdebug, mem));
190
if(countcheck("realloc", line, source))
194
mem = (struct memdebug *)((char *)ptr - offsetof(struct memdebug, mem));
128
196
mem=(struct memdebug *)(realloc)(mem, size);
130
fprintf(logfile, "MEM %s:%d realloc(%p, %d) = %p\n",
198
fprintf(logfile, "MEM %s:%d realloc(0x%x, %zd) = %p\n",
131
199
source, line, ptr, wantedsize, mem?mem->mem:NULL);
159
224
fprintf(logfile, "MEM %s:%d free(%p)\n", source, line, ptr);
162
int curl_socket(int domain, int type, int protocol, int line, char *source)
227
int curl_socket(int domain, int type, int protocol, int line,
164
230
int sockfd=(socket)(domain, type, protocol);
231
if(logfile && (sockfd!=-1))
166
232
fprintf(logfile, "FD %s:%d socket() = %d\n",
167
233
source, line, sockfd);
171
int curl_accept(int s, struct sockaddr *addr, socklen_t *addrlen,
237
int curl_accept(int s, void *saddr, void *saddrlen,
172
238
int line, const char *source)
240
struct sockaddr *addr = (struct sockaddr *)saddr;
241
socklen_t *addrlen = (socklen_t *)saddrlen;
174
242
int sockfd=(accept)(s, addr, addrlen);
176
244
fprintf(logfile, "FD %s:%d accept() = %d\n",
181
249
/* this is our own defined way to close sockets on *ALL* platforms */
182
int curl_sclose(int sockfd, int line, char *source)
250
int curl_sclose(int sockfd, int line, const char *source)
184
252
int res=sclose(sockfd);
194
262
FILE *res=(fopen)(file, mode);
196
fprintf(logfile, "FILE %s:%d fopen(\"%s\") = %p\n",
197
source, line, file, res);
264
fprintf(logfile, "FILE %s:%d fopen(\"%s\",\"%s\") = %p\n",
265
source, line, file, mode, res);
201
269
int curl_fclose(FILE *file, int line, const char *source)
203
int res=(fclose)(file);
273
curlassert(file != NULL);
205
277
fprintf(logfile, "FILE %s:%d fclose(%p)\n",
206
278
source, line, file);