os/ossrv/ssl/libcrypto/src/crypto/mem_dbg.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* crypto/mem_dbg.c */
sl@0
     2
/* Copyright (C) 1995-1998 Eric Young (eay@cryptsoft.com)
sl@0
     3
 * All rights reserved.
sl@0
     4
 *
sl@0
     5
 * This package is an SSL implementation written
sl@0
     6
 * by Eric Young (eay@cryptsoft.com).
sl@0
     7
 * The implementation was written so as to conform with Netscapes SSL.
sl@0
     8
 * 
sl@0
     9
 * This library is free for commercial and non-commercial use as long as
sl@0
    10
 * the following conditions are aheared to.  The following conditions
sl@0
    11
 * apply to all code found in this distribution, be it the RC4, RSA,
sl@0
    12
 * lhash, DES, etc., code; not just the SSL code.  The SSL documentation
sl@0
    13
 * included with this distribution is covered by the same copyright terms
sl@0
    14
 * except that the holder is Tim Hudson (tjh@cryptsoft.com).
sl@0
    15
 * 
sl@0
    16
 * Copyright remains Eric Young's, and as such any Copyright notices in
sl@0
    17
 * the code are not to be removed.
sl@0
    18
 * If this package is used in a product, Eric Young should be given attribution
sl@0
    19
 * as the author of the parts of the library used.
sl@0
    20
 * This can be in the form of a textual message at program startup or
sl@0
    21
 * in documentation (online or textual) provided with the package.
sl@0
    22
 * 
sl@0
    23
 * Redistribution and use in source and binary forms, with or without
sl@0
    24
 * modification, are permitted provided that the following conditions
sl@0
    25
 * are met:
sl@0
    26
 * 1. Redistributions of source code must retain the copyright
sl@0
    27
 *    notice, this list of conditions and the following disclaimer.
sl@0
    28
 * 2. Redistributions in binary form must reproduce the above copyright
sl@0
    29
 *    notice, this list of conditions and the following disclaimer in the
sl@0
    30
 *    documentation and/or other materials provided with the distribution.
sl@0
    31
 * 3. All advertising materials mentioning features or use of this software
sl@0
    32
 *    must display the following acknowledgement:
sl@0
    33
 *    "This product includes cryptographic software written by
sl@0
    34
 *     Eric Young (eay@cryptsoft.com)"
sl@0
    35
 *    The word 'cryptographic' can be left out if the rouines from the library
sl@0
    36
 *    being used are not cryptographic related :-).
sl@0
    37
 * 4. If you include any Windows specific code (or a derivative thereof) from 
sl@0
    38
 *    the apps directory (application code) you must include an acknowledgement:
sl@0
    39
 *    "This product includes software written by Tim Hudson (tjh@cryptsoft.com)"
sl@0
    40
 * 
sl@0
    41
 * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND
sl@0
    42
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
sl@0
    43
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
sl@0
    44
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
sl@0
    45
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
sl@0
    46
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
sl@0
    47
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
sl@0
    48
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
sl@0
    49
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
sl@0
    50
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
sl@0
    51
 * SUCH DAMAGE.
sl@0
    52
 * 
sl@0
    53
 * The licence and distribution terms for any publically available version or
sl@0
    54
 * derivative of this code cannot be changed.  i.e. this code cannot simply be
sl@0
    55
 * copied and put under another distribution licence
sl@0
    56
 * [including the GNU Public Licence.]
sl@0
    57
 */
sl@0
    58
/*
sl@0
    59
 © Portions copyright (c) 2006 Nokia Corporation.  All rights reserved.
sl@0
    60
 */
sl@0
    61
sl@0
    62
#include <stdio.h>
sl@0
    63
#include <stdlib.h>
sl@0
    64
#include <time.h>	
sl@0
    65
#include "cryptlib.h"
sl@0
    66
#include <openssl/crypto.h>
sl@0
    67
#include <openssl/buffer.h>
sl@0
    68
#include <openssl/bio.h>
sl@0
    69
#include <openssl/lhash.h>
sl@0
    70
sl@0
    71
#if (defined(SYMBIAN) && (defined(__WINSCW__) || defined(__WINS__)))
sl@0
    72
#include "libcrypto_wsd_macros.h"
sl@0
    73
#include "libcrypto_wsd.h"
sl@0
    74
#endif
sl@0
    75
sl@0
    76
#ifndef EMULATOR
sl@0
    77
static int mh_mode=CRYPTO_MEM_CHECK_OFF;
sl@0
    78
/* The state changes to CRYPTO_MEM_CHECK_ON | CRYPTO_MEM_CHECK_ENABLE
sl@0
    79
 * when the application asks for it (usually after library initialisation
sl@0
    80
 * for which no book-keeping is desired).
sl@0
    81
 *
sl@0
    82
 * State CRYPTO_MEM_CHECK_ON exists only temporarily when the library
sl@0
    83
 * thinks that certain allocations should not be checked (e.g. the data
sl@0
    84
 * structures used for memory checking).  It is not suitable as an initial
sl@0
    85
 * state: the library will unexpectedly enable memory checking when it
sl@0
    86
 * executes one of those sections that want to disable checking
sl@0
    87
 * temporarily.
sl@0
    88
 *
sl@0
    89
 * State CRYPTO_MEM_CHECK_ENABLE without ..._ON makes no sense whatsoever.
sl@0
    90
 */
sl@0
    91
sl@0
    92
static unsigned long order = 0; /* number of memory requests */
sl@0
    93
static LHASH *mh=NULL; /* hash-table of memory requests (address as key);
sl@0
    94
                       * access requires MALLOC2 lock */
sl@0
    95
                   
sl@0
    96
sl@0
    97
#endif                        
sl@0
    98
sl@0
    99
sl@0
   100
typedef struct app_mem_info_st
sl@0
   101
/* For application-defined information (static C-string `info')
sl@0
   102
 * to be displayed in memory leak list.
sl@0
   103
 * Each thread has its own stack.  For applications, there is
sl@0
   104
 *   CRYPTO_push_info("...")     to push an entry,
sl@0
   105
 *   CRYPTO_pop_info()           to pop an entry,
sl@0
   106
 *   CRYPTO_remove_all_info()    to pop all entries.
sl@0
   107
 */
sl@0
   108
	{	
sl@0
   109
	unsigned long thread;
sl@0
   110
	const char *file;
sl@0
   111
	int line;
sl@0
   112
	const char *info;
sl@0
   113
	struct app_mem_info_st *next; /* tail of thread's stack */
sl@0
   114
	int references;
sl@0
   115
	} APP_INFO;
sl@0
   116
sl@0
   117
static void app_info_free(APP_INFO *);
sl@0
   118
sl@0
   119
#ifndef EMULATOR
sl@0
   120
static LHASH *amih=NULL; /* hash-table with those app_mem_info_st's
sl@0
   121
                          * that are at the top of their thread's stack
sl@0
   122
                          * (with `thread' as key);
sl@0
   123
                          * access requires MALLOC2 lock */
sl@0
   124
#else
sl@0
   125
 GET_STATIC_VAR_FROM_TLS(amih,mem_dbg,LHASH *)
sl@0
   126
 #define amih (*GET_WSD_VAR_NAME(amih,mem_dbg,s)())
sl@0
   127
#endif                           
sl@0
   128
sl@0
   129
typedef struct mem_st
sl@0
   130
/* memory-block description */
sl@0
   131
	{
sl@0
   132
	void *addr;
sl@0
   133
	int num;
sl@0
   134
	const char *file;
sl@0
   135
	int line;
sl@0
   136
	unsigned long thread;
sl@0
   137
	unsigned long iorder;
sl@0
   138
	time_t time;
sl@0
   139
	APP_INFO *app_info;
sl@0
   140
	} MEM;
sl@0
   141
sl@0
   142
sl@0
   143
sl@0
   144
#ifdef EMULATOR
sl@0
   145
 GET_STATIC_VAR_FROM_TLS(mh_mode,mem_dbg,int)
sl@0
   146
 #define mh_mode (*GET_WSD_VAR_NAME(mh_mode,mem_dbg,s)())
sl@0
   147
 GET_STATIC_VAR_FROM_TLS(order,mem_dbg,unsigned long)
sl@0
   148
 #define order (*GET_WSD_VAR_NAME(order,mem_dbg,s)())
sl@0
   149
 GET_STATIC_VAR_FROM_TLS(mh,mem_dbg,LHASH*)
sl@0
   150
 #define mh (*GET_WSD_VAR_NAME(mh,mem_dbg,s)())
sl@0
   151
#endif 
sl@0
   152
sl@0
   153
#ifndef EMULATOR
sl@0
   154
static long options =             /* extra information to be recorded */
sl@0
   155
#if defined(CRYPTO_MDEBUG_TIME) || defined(CRYPTO_MDEBUG_ALL)
sl@0
   156
	V_CRYPTO_MDEBUG_TIME |
sl@0
   157
#endif
sl@0
   158
#if defined(CRYPTO_MDEBUG_THREAD) || defined(CRYPTO_MDEBUG_ALL)
sl@0
   159
	V_CRYPTO_MDEBUG_THREAD |
sl@0
   160
#endif
sl@0
   161
	0;
sl@0
   162
static unsigned int num_disable = 0; /* num_disable > 0
sl@0
   163
                                      *     iff
sl@0
   164
                                      * mh_mode == CRYPTO_MEM_CHECK_ON (w/o ..._ENABLE)
sl@0
   165
                                      */
sl@0
   166
static unsigned long disabling_thread = 0; /* Valid iff num_disable > 0.
sl@0
   167
                                            * CRYPTO_LOCK_MALLOC2 is locked
sl@0
   168
                                            * exactly in this case (by the
sl@0
   169
                                            * thread named in disabling_thread).
sl@0
   170
                                            */
sl@0
   171
sl@0
   172
#else
sl@0
   173
 GET_STATIC_VAR_FROM_TLS(options,mem_dbg,unsigned long)
sl@0
   174
 #define options (*GET_WSD_VAR_NAME(options,mem_dbg,s)())
sl@0
   175
 GET_STATIC_VAR_FROM_TLS(num_disable,mem_dbg,unsigned int)
sl@0
   176
 #define num_disable (*GET_WSD_VAR_NAME(num_disable,mem_dbg,s)())
sl@0
   177
 GET_STATIC_VAR_FROM_TLS(disabling_thread,mem_dbg,unsigned long)
sl@0
   178
 #define disabling_thread (*GET_WSD_VAR_NAME(disabling_thread,mem_dbg,s)())
sl@0
   179
#endif	
sl@0
   180
sl@0
   181
sl@0
   182
sl@0
   183
static void app_info_free(APP_INFO *inf)
sl@0
   184
	{
sl@0
   185
	if (--(inf->references) <= 0)
sl@0
   186
		{
sl@0
   187
		if (inf->next != NULL)
sl@0
   188
			{
sl@0
   189
			app_info_free(inf->next);
sl@0
   190
			}
sl@0
   191
		OPENSSL_free(inf);
sl@0
   192
		}
sl@0
   193
	}
sl@0
   194
sl@0
   195
EXPORT_C int CRYPTO_mem_ctrl(int mode)
sl@0
   196
	{
sl@0
   197
	int ret=mh_mode;
sl@0
   198
sl@0
   199
	CRYPTO_w_lock(CRYPTO_LOCK_MALLOC);
sl@0
   200
	switch (mode)
sl@0
   201
		{
sl@0
   202
	/* for applications (not to be called while multiple threads
sl@0
   203
	 * use the library): */
sl@0
   204
	case CRYPTO_MEM_CHECK_ON: /* aka MemCheck_start() */
sl@0
   205
		mh_mode = CRYPTO_MEM_CHECK_ON|CRYPTO_MEM_CHECK_ENABLE;
sl@0
   206
		num_disable = 0;
sl@0
   207
		break;
sl@0
   208
	case CRYPTO_MEM_CHECK_OFF: /* aka MemCheck_stop() */
sl@0
   209
		mh_mode = 0;
sl@0
   210
		num_disable = 0; /* should be true *before* MemCheck_stop is used,
sl@0
   211
		                    or there'll be a lot of confusion */
sl@0
   212
		break;
sl@0
   213
sl@0
   214
	/* switch off temporarily (for library-internal use): */
sl@0
   215
	case CRYPTO_MEM_CHECK_DISABLE: /* aka MemCheck_off() */
sl@0
   216
		if (mh_mode & CRYPTO_MEM_CHECK_ON)
sl@0
   217
			{
sl@0
   218
			if (!num_disable || (disabling_thread != CRYPTO_thread_id())) /* otherwise we already have the MALLOC2 lock */
sl@0
   219
				{
sl@0
   220
				/* Long-time lock CRYPTO_LOCK_MALLOC2 must not be claimed while
sl@0
   221
				 * we're holding CRYPTO_LOCK_MALLOC, or we'll deadlock if
sl@0
   222
				 * somebody else holds CRYPTO_LOCK_MALLOC2 (and cannot release
sl@0
   223
				 * it because we block entry to this function).
sl@0
   224
				 * Give them a chance, first, and then claim the locks in
sl@0
   225
				 * appropriate order (long-time lock first).
sl@0
   226
				 */
sl@0
   227
				CRYPTO_w_unlock(CRYPTO_LOCK_MALLOC);
sl@0
   228
				/* Note that after we have waited for CRYPTO_LOCK_MALLOC2
sl@0
   229
				 * and CRYPTO_LOCK_MALLOC, we'll still be in the right
sl@0
   230
				 * "case" and "if" branch because MemCheck_start and
sl@0
   231
				 * MemCheck_stop may never be used while there are multiple
sl@0
   232
				 * OpenSSL threads. */
sl@0
   233
				CRYPTO_w_lock(CRYPTO_LOCK_MALLOC2);
sl@0
   234
				CRYPTO_w_lock(CRYPTO_LOCK_MALLOC);
sl@0
   235
				mh_mode &= ~CRYPTO_MEM_CHECK_ENABLE;
sl@0
   236
				disabling_thread=CRYPTO_thread_id();
sl@0
   237
				}
sl@0
   238
			num_disable++;
sl@0
   239
			}
sl@0
   240
		break;
sl@0
   241
	case CRYPTO_MEM_CHECK_ENABLE: /* aka MemCheck_on() */
sl@0
   242
		if (mh_mode & CRYPTO_MEM_CHECK_ON)
sl@0
   243
			{
sl@0
   244
			if (num_disable) /* always true, or something is going wrong */
sl@0
   245
				{
sl@0
   246
				num_disable--;
sl@0
   247
				if (num_disable == 0)
sl@0
   248
					{
sl@0
   249
					mh_mode|=CRYPTO_MEM_CHECK_ENABLE;
sl@0
   250
					CRYPTO_w_unlock(CRYPTO_LOCK_MALLOC2);
sl@0
   251
					}
sl@0
   252
				}
sl@0
   253
			}
sl@0
   254
		break;
sl@0
   255
sl@0
   256
	default:
sl@0
   257
		break;
sl@0
   258
		}
sl@0
   259
	CRYPTO_w_unlock(CRYPTO_LOCK_MALLOC);
sl@0
   260
	return(ret);
sl@0
   261
	}
sl@0
   262
sl@0
   263
EXPORT_C int CRYPTO_is_mem_check_on(void)
sl@0
   264
	{
sl@0
   265
	int ret = 0;
sl@0
   266
sl@0
   267
	if (mh_mode & CRYPTO_MEM_CHECK_ON)
sl@0
   268
		{
sl@0
   269
		CRYPTO_r_lock(CRYPTO_LOCK_MALLOC);
sl@0
   270
sl@0
   271
		ret = (mh_mode & CRYPTO_MEM_CHECK_ENABLE)
sl@0
   272
			|| (disabling_thread != CRYPTO_thread_id());
sl@0
   273
sl@0
   274
		CRYPTO_r_unlock(CRYPTO_LOCK_MALLOC);
sl@0
   275
		}
sl@0
   276
	return(ret);
sl@0
   277
	}	
sl@0
   278
sl@0
   279
sl@0
   280
EXPORT_C void CRYPTO_dbg_set_options(long bits)
sl@0
   281
	{
sl@0
   282
	options = bits;
sl@0
   283
	}
sl@0
   284
sl@0
   285
EXPORT_C long CRYPTO_dbg_get_options(void)
sl@0
   286
	{
sl@0
   287
	return options;
sl@0
   288
	}
sl@0
   289
sl@0
   290
/* static int mem_cmp(MEM *a, MEM *b) */
sl@0
   291
static int mem_cmp(const void *a_void, const void *b_void)
sl@0
   292
	{
sl@0
   293
#ifdef _WIN64
sl@0
   294
	const char *a=(const char *)((const MEM *)a_void)->addr,
sl@0
   295
		   *b=(const char *)((const MEM *)b_void)->addr;
sl@0
   296
	if (a==b)	return 0;
sl@0
   297
	else if (a>b)	return 1;
sl@0
   298
	else		return -1;
sl@0
   299
#else
sl@0
   300
	return((const char *)((const MEM *)a_void)->addr
sl@0
   301
		- (const char *)((const MEM *)b_void)->addr);
sl@0
   302
#endif
sl@0
   303
	}
sl@0
   304
sl@0
   305
/* static unsigned long mem_hash(MEM *a) */
sl@0
   306
static unsigned long mem_hash(const void *a_void)
sl@0
   307
	{
sl@0
   308
	unsigned long ret;
sl@0
   309
sl@0
   310
	ret=(unsigned long)((const MEM *)a_void)->addr;
sl@0
   311
sl@0
   312
	ret=ret*17851+(ret>>14)*7+(ret>>4)*251;
sl@0
   313
	return(ret);
sl@0
   314
	}
sl@0
   315
sl@0
   316
/* static int app_info_cmp(APP_INFO *a, APP_INFO *b) */
sl@0
   317
static int app_info_cmp(const void *a_void, const void *b_void)
sl@0
   318
	{
sl@0
   319
	return(((const APP_INFO *)a_void)->thread
sl@0
   320
		!= ((const APP_INFO *)b_void)->thread);
sl@0
   321
	}
sl@0
   322
sl@0
   323
/* static unsigned long app_info_hash(APP_INFO *a) */
sl@0
   324
static unsigned long app_info_hash(const void *a_void)
sl@0
   325
	{
sl@0
   326
	unsigned long ret;
sl@0
   327
sl@0
   328
	ret=(unsigned long)((const APP_INFO *)a_void)->thread;
sl@0
   329
sl@0
   330
	ret=ret*17851+(ret>>14)*7+(ret>>4)*251;
sl@0
   331
	return(ret);
sl@0
   332
	}
sl@0
   333
sl@0
   334
static APP_INFO *pop_info(void)
sl@0
   335
	{
sl@0
   336
	APP_INFO tmp;
sl@0
   337
	APP_INFO *ret = NULL;
sl@0
   338
sl@0
   339
	if (amih != NULL)
sl@0
   340
		{
sl@0
   341
		tmp.thread=CRYPTO_thread_id();
sl@0
   342
		if ((ret=(APP_INFO *)lh_delete(amih,&tmp)) != NULL)
sl@0
   343
			{
sl@0
   344
			APP_INFO *next=ret->next;
sl@0
   345
sl@0
   346
			if (next != NULL)
sl@0
   347
				{
sl@0
   348
				next->references++;
sl@0
   349
				lh_insert(amih,(char *)next);
sl@0
   350
				}
sl@0
   351
#ifdef LEVITTE_DEBUG_MEM
sl@0
   352
			if (ret->thread != tmp.thread)
sl@0
   353
				{
sl@0
   354
				fprintf(stderr, "pop_info(): deleted info has other thread ID (%lu) than the current thread (%lu)!!!!\n",
sl@0
   355
					ret->thread, tmp.thread);
sl@0
   356
				abort();
sl@0
   357
				}
sl@0
   358
#endif
sl@0
   359
			if (--(ret->references) <= 0)
sl@0
   360
				{
sl@0
   361
				ret->next = NULL;
sl@0
   362
				if (next != NULL)
sl@0
   363
					next->references--;
sl@0
   364
				OPENSSL_free(ret);
sl@0
   365
				}
sl@0
   366
			}
sl@0
   367
		}
sl@0
   368
	return(ret);
sl@0
   369
	}
sl@0
   370
sl@0
   371
EXPORT_C int CRYPTO_push_info_(const char *info, const char *file, int line)
sl@0
   372
	{
sl@0
   373
	APP_INFO *ami=NULL, *amim=NULL;
sl@0
   374
	int ret=0;
sl@0
   375
sl@0
   376
	if (is_MemCheck_on())
sl@0
   377
		{
sl@0
   378
		MemCheck_off(); /* obtain MALLOC2 lock */
sl@0
   379
sl@0
   380
		if ((ami = (APP_INFO *)OPENSSL_malloc(sizeof(APP_INFO))) == NULL)
sl@0
   381
			{
sl@0
   382
			ret=0;
sl@0
   383
			goto err;
sl@0
   384
			}
sl@0
   385
		if (amih == NULL)
sl@0
   386
			{
sl@0
   387
			if ((amih=lh_new(app_info_hash, app_info_cmp)) == NULL)
sl@0
   388
				{
sl@0
   389
				OPENSSL_free(ami);
sl@0
   390
				ret=0;
sl@0
   391
				goto err;
sl@0
   392
				}
sl@0
   393
			}
sl@0
   394
sl@0
   395
		ami->thread=CRYPTO_thread_id();
sl@0
   396
		ami->file=file;
sl@0
   397
		ami->line=line;
sl@0
   398
		ami->info=info;
sl@0
   399
		ami->references=1;
sl@0
   400
		ami->next=NULL;
sl@0
   401
sl@0
   402
		if ((amim=(APP_INFO *)lh_insert(amih,(char *)ami)) != NULL)
sl@0
   403
			{
sl@0
   404
#ifdef LEVITTE_DEBUG_MEM
sl@0
   405
			if (ami->thread != amim->thread)
sl@0
   406
				{
sl@0
   407
				fprintf(stderr, "CRYPTO_push_info(): previous info has other thread ID (%lu) than the current thread (%lu)!!!!\n",
sl@0
   408
					amim->thread, ami->thread);
sl@0
   409
				abort();
sl@0
   410
				}
sl@0
   411
#endif
sl@0
   412
			ami->next=amim;
sl@0
   413
			}
sl@0
   414
 err:
sl@0
   415
		MemCheck_on(); /* release MALLOC2 lock */
sl@0
   416
		}
sl@0
   417
sl@0
   418
	return(ret);
sl@0
   419
	}
sl@0
   420
sl@0
   421
EXPORT_C int CRYPTO_pop_info(void)
sl@0
   422
	{
sl@0
   423
	int ret=0;
sl@0
   424
sl@0
   425
	if (is_MemCheck_on()) /* _must_ be true, or something went severely wrong */
sl@0
   426
		{
sl@0
   427
		MemCheck_off(); /* obtain MALLOC2 lock */
sl@0
   428
sl@0
   429
		ret=(pop_info() != NULL);
sl@0
   430
sl@0
   431
		MemCheck_on(); /* release MALLOC2 lock */
sl@0
   432
		}
sl@0
   433
	return(ret);
sl@0
   434
	}
sl@0
   435
sl@0
   436
EXPORT_C int CRYPTO_remove_all_info(void)
sl@0
   437
	{
sl@0
   438
	int ret=0;
sl@0
   439
sl@0
   440
	if (is_MemCheck_on()) /* _must_ be true */
sl@0
   441
		{
sl@0
   442
		MemCheck_off(); /* obtain MALLOC2 lock */
sl@0
   443
sl@0
   444
		while(pop_info() != NULL)
sl@0
   445
			ret++;
sl@0
   446
sl@0
   447
		MemCheck_on(); /* release MALLOC2 lock */
sl@0
   448
		}
sl@0
   449
	return(ret);
sl@0
   450
	}
sl@0
   451
sl@0
   452
#ifndef EMULATOR
sl@0
   453
static unsigned long break_order_num=0;
sl@0
   454
#else
sl@0
   455
static const unsigned long break_order_num=0;
sl@0
   456
#endif
sl@0
   457
EXPORT_C void CRYPTO_dbg_malloc(void *addr, int num, const char *file, int line,
sl@0
   458
	int before_p)
sl@0
   459
	{
sl@0
   460
	MEM *m=NULL,*mm=NULL;
sl@0
   461
	APP_INFO tmp,*amim=NULL;
sl@0
   462
sl@0
   463
	switch(before_p & 127)
sl@0
   464
		{
sl@0
   465
	case 0:
sl@0
   466
		break;
sl@0
   467
	case 1:
sl@0
   468
		if (addr == NULL)
sl@0
   469
			break;
sl@0
   470
sl@0
   471
		if (is_MemCheck_on())
sl@0
   472
			{
sl@0
   473
			MemCheck_off(); /* make sure we hold MALLOC2 lock */
sl@0
   474
			if ((m=(MEM *)OPENSSL_malloc(sizeof(MEM))) == NULL)
sl@0
   475
				{
sl@0
   476
				OPENSSL_free(addr);
sl@0
   477
				MemCheck_on(); /* release MALLOC2 lock
sl@0
   478
				                * if num_disabled drops to 0 */
sl@0
   479
				return;
sl@0
   480
				}
sl@0
   481
			if (mh == NULL)
sl@0
   482
				{
sl@0
   483
				if ((mh=lh_new(mem_hash, mem_cmp)) == NULL)
sl@0
   484
					{
sl@0
   485
					OPENSSL_free(addr);
sl@0
   486
					OPENSSL_free(m);
sl@0
   487
					addr=NULL;
sl@0
   488
					goto err;
sl@0
   489
					}
sl@0
   490
				}
sl@0
   491
sl@0
   492
			m->addr=addr;
sl@0
   493
			m->file=file;
sl@0
   494
			m->line=line;
sl@0
   495
			m->num=num;
sl@0
   496
			if (options & V_CRYPTO_MDEBUG_THREAD)
sl@0
   497
				m->thread=CRYPTO_thread_id();
sl@0
   498
			else
sl@0
   499
				m->thread=0;
sl@0
   500
sl@0
   501
			if (order == break_order_num)
sl@0
   502
				{
sl@0
   503
				/* BREAK HERE */
sl@0
   504
				m->iorder=order;
sl@0
   505
				}
sl@0
   506
			m->iorder=order++;
sl@0
   507
#ifdef LEVITTE_DEBUG_MEM
sl@0
   508
			fprintf(stderr, "LEVITTE_DEBUG_MEM: [%5d] %c 0x%p (%d)\n",
sl@0
   509
				m->iorder,
sl@0
   510
				(before_p & 128) ? '*' : '+',
sl@0
   511
				m->addr, m->num);
sl@0
   512
#endif
sl@0
   513
			if (options & V_CRYPTO_MDEBUG_TIME)
sl@0
   514
				m->time=time(NULL);
sl@0
   515
			else
sl@0
   516
				m->time=0;
sl@0
   517
sl@0
   518
			tmp.thread=CRYPTO_thread_id();
sl@0
   519
			m->app_info=NULL;
sl@0
   520
			if (amih != NULL
sl@0
   521
				&& (amim=(APP_INFO *)lh_retrieve(amih,(char *)&tmp)) != NULL)
sl@0
   522
				{
sl@0
   523
				m->app_info = amim;
sl@0
   524
				amim->references++;
sl@0
   525
				}
sl@0
   526
sl@0
   527
			if ((mm=(MEM *)lh_insert(mh,(char *)m)) != NULL)
sl@0
   528
				{
sl@0
   529
				/* Not good, but don't sweat it */
sl@0
   530
				if (mm->app_info != NULL)
sl@0
   531
					{
sl@0
   532
					mm->app_info->references--;
sl@0
   533
					}
sl@0
   534
				OPENSSL_free(mm);
sl@0
   535
				}
sl@0
   536
		err:
sl@0
   537
			MemCheck_on(); /* release MALLOC2 lock
sl@0
   538
			                * if num_disabled drops to 0 */
sl@0
   539
			}
sl@0
   540
		break;
sl@0
   541
		}
sl@0
   542
	return;
sl@0
   543
	}
sl@0
   544
sl@0
   545
EXPORT_C void CRYPTO_dbg_free(void *addr, int before_p)
sl@0
   546
	{
sl@0
   547
	MEM m,*mp=NULL;
sl@0
   548
sl@0
   549
	switch(before_p)
sl@0
   550
		{
sl@0
   551
	case 0:
sl@0
   552
		if (addr == NULL)
sl@0
   553
			break;
sl@0
   554
sl@0
   555
		if (is_MemCheck_on() && (mh != NULL))
sl@0
   556
			{
sl@0
   557
			MemCheck_off(); /* make sure we hold MALLOC2 lock */
sl@0
   558
sl@0
   559
			m.addr=addr;
sl@0
   560
			mp=(MEM *)lh_delete(mh,(char *)&m);
sl@0
   561
			if (mp != NULL)
sl@0
   562
				{
sl@0
   563
#ifdef LEVITTE_DEBUG_MEM
sl@0
   564
			fprintf(stderr, "LEVITTE_DEBUG_MEM: [%5d] - 0x%p (%d)\n",
sl@0
   565
				mp->order, mp->addr, mp->num);
sl@0
   566
#endif
sl@0
   567
				if (mp->app_info != NULL)
sl@0
   568
					app_info_free(mp->app_info);
sl@0
   569
				OPENSSL_free(mp);
sl@0
   570
				}
sl@0
   571
sl@0
   572
			MemCheck_on(); /* release MALLOC2 lock
sl@0
   573
			                * if num_disabled drops to 0 */
sl@0
   574
			}
sl@0
   575
		break;
sl@0
   576
	case 1:
sl@0
   577
		break;
sl@0
   578
		}
sl@0
   579
	}
sl@0
   580
sl@0
   581
EXPORT_C void CRYPTO_dbg_realloc(void *addr1, void *addr2, int num,
sl@0
   582
	const char *file, int line, int before_p)
sl@0
   583
	{
sl@0
   584
	MEM m,*mp;
sl@0
   585
sl@0
   586
#ifdef LEVITTE_DEBUG_MEM
sl@0
   587
	fprintf(stderr, "LEVITTE_DEBUG_MEM: --> CRYPTO_dbg_malloc(addr1 = %p, addr2 = %p, num = %d, file = \"%s\", line = %d, before_p = %d)\n",
sl@0
   588
		addr1, addr2, num, file, line, before_p);
sl@0
   589
#endif
sl@0
   590
sl@0
   591
	switch(before_p)
sl@0
   592
		{
sl@0
   593
	case 0:
sl@0
   594
		break;
sl@0
   595
	case 1:
sl@0
   596
		if (addr2 == NULL)
sl@0
   597
			break;
sl@0
   598
sl@0
   599
		if (addr1 == NULL)
sl@0
   600
			{
sl@0
   601
			CRYPTO_dbg_malloc(addr2, num, file, line, 128 | before_p);
sl@0
   602
			break;
sl@0
   603
			}
sl@0
   604
sl@0
   605
		if (is_MemCheck_on())
sl@0
   606
			{
sl@0
   607
			MemCheck_off(); /* make sure we hold MALLOC2 lock */
sl@0
   608
sl@0
   609
			m.addr=addr1;
sl@0
   610
			mp=(MEM *)lh_delete(mh,(char *)&m);
sl@0
   611
			if (mp != NULL)
sl@0
   612
				{
sl@0
   613
#ifdef LEVITTE_DEBUG_MEM
sl@0
   614
				fprintf(stderr, "LEVITTE_DEBUG_MEM: [%5d] * 0x%p (%d) -> 0x%p (%d)\n",
sl@0
   615
					mp->order,
sl@0
   616
					mp->addr, mp->num,
sl@0
   617
					addr2, num);
sl@0
   618
#endif
sl@0
   619
				mp->addr=addr2;
sl@0
   620
				mp->num=num;
sl@0
   621
				lh_insert(mh,(char *)mp);
sl@0
   622
				}
sl@0
   623
sl@0
   624
			MemCheck_on(); /* release MALLOC2 lock
sl@0
   625
			                * if num_disabled drops to 0 */
sl@0
   626
			}
sl@0
   627
		break;
sl@0
   628
		}
sl@0
   629
	return;
sl@0
   630
	}
sl@0
   631
sl@0
   632
sl@0
   633
typedef struct mem_leak_st
sl@0
   634
	{
sl@0
   635
	BIO *bio;
sl@0
   636
	int chunks;
sl@0
   637
	long bytes;
sl@0
   638
	} MEM_LEAK;
sl@0
   639
sl@0
   640
static void print_leak(const MEM *m, MEM_LEAK *l)
sl@0
   641
	{
sl@0
   642
#ifndef SYMBIAN		
sl@0
   643
	char buf[1024];
sl@0
   644
#else
sl@0
   645
  char buf[512];
sl@0
   646
#endif	
sl@0
   647
	char *bufp = buf;
sl@0
   648
	APP_INFO *amip;
sl@0
   649
	int ami_cnt;
sl@0
   650
	struct tm *lcl = NULL;
sl@0
   651
	unsigned long ti;
sl@0
   652
sl@0
   653
#define BUF_REMAIN (sizeof buf - (size_t)(bufp - buf))
sl@0
   654
sl@0
   655
	if(m->addr == (char *)l->bio)
sl@0
   656
	    return;
sl@0
   657
sl@0
   658
	if (options & V_CRYPTO_MDEBUG_TIME)
sl@0
   659
		{
sl@0
   660
		lcl = localtime(&m->time);
sl@0
   661
	
sl@0
   662
		BIO_snprintf(bufp, BUF_REMAIN, "[%02d:%02d:%02d] ",
sl@0
   663
			lcl->tm_hour,lcl->tm_min,lcl->tm_sec);
sl@0
   664
		bufp += strlen(bufp);
sl@0
   665
		}
sl@0
   666
sl@0
   667
	BIO_snprintf(bufp, BUF_REMAIN, "%5lu file=%s, line=%d, ",
sl@0
   668
		m->iorder,m->file,m->line);
sl@0
   669
	bufp += strlen(bufp);
sl@0
   670
sl@0
   671
	if (options & V_CRYPTO_MDEBUG_THREAD)
sl@0
   672
		{
sl@0
   673
		BIO_snprintf(bufp, BUF_REMAIN, "thread=%lu, ", m->thread);
sl@0
   674
		bufp += strlen(bufp);
sl@0
   675
		}
sl@0
   676
sl@0
   677
	BIO_snprintf(bufp, BUF_REMAIN, "number=%d, address=%08lX\n",
sl@0
   678
		m->num,(unsigned long)m->addr);
sl@0
   679
	bufp += strlen(bufp);
sl@0
   680
sl@0
   681
	BIO_puts(l->bio,buf);
sl@0
   682
	
sl@0
   683
	l->chunks++;
sl@0
   684
	l->bytes+=m->num;
sl@0
   685
sl@0
   686
	amip=m->app_info;
sl@0
   687
	ami_cnt=0;
sl@0
   688
	if (!amip)
sl@0
   689
		return;
sl@0
   690
	ti=amip->thread;
sl@0
   691
	
sl@0
   692
	do
sl@0
   693
		{
sl@0
   694
		int buf_len;
sl@0
   695
		int info_len;
sl@0
   696
sl@0
   697
		ami_cnt++;
sl@0
   698
		memset(buf,'>',ami_cnt);
sl@0
   699
		BIO_snprintf(buf + ami_cnt, sizeof buf - ami_cnt,
sl@0
   700
			" thread=%lu, file=%s, line=%d, info=\"",
sl@0
   701
			amip->thread, amip->file, amip->line);
sl@0
   702
		buf_len=strlen(buf);
sl@0
   703
		info_len=strlen(amip->info);
sl@0
   704
		if (128 - buf_len - 3 < info_len)
sl@0
   705
			{
sl@0
   706
			memcpy(buf + buf_len, amip->info, 128 - buf_len - 3);
sl@0
   707
			buf_len = 128 - 3;
sl@0
   708
			}
sl@0
   709
		else
sl@0
   710
			{
sl@0
   711
			BUF_strlcpy(buf + buf_len, amip->info,
sl@0
   712
				    sizeof buf - buf_len);
sl@0
   713
			buf_len = strlen(buf);
sl@0
   714
			}
sl@0
   715
		BIO_snprintf(buf + buf_len, sizeof buf - buf_len, "\"\n");
sl@0
   716
		
sl@0
   717
		BIO_puts(l->bio,buf);
sl@0
   718
sl@0
   719
		amip = amip->next;
sl@0
   720
		}
sl@0
   721
	while(amip && amip->thread == ti);
sl@0
   722
		
sl@0
   723
#ifdef LEVITTE_DEBUG_MEM
sl@0
   724
	if (amip)
sl@0
   725
		{
sl@0
   726
		fprintf(stderr, "Thread switch detected in backtrace!!!!\n");
sl@0
   727
		abort();
sl@0
   728
		}
sl@0
   729
#endif
sl@0
   730
	}
sl@0
   731
sl@0
   732
static IMPLEMENT_LHASH_DOALL_ARG_FN(print_leak, const MEM *, MEM_LEAK *)
sl@0
   733
sl@0
   734
EXPORT_C void CRYPTO_mem_leaks(BIO *b)
sl@0
   735
	{
sl@0
   736
	MEM_LEAK ml;
sl@0
   737
sl@0
   738
	if (mh == NULL && amih == NULL)
sl@0
   739
		return;
sl@0
   740
sl@0
   741
	MemCheck_off(); /* obtain MALLOC2 lock */
sl@0
   742
sl@0
   743
	ml.bio=b;
sl@0
   744
	ml.bytes=0;
sl@0
   745
	ml.chunks=0;
sl@0
   746
	if (mh != NULL)
sl@0
   747
		lh_doall_arg(mh, LHASH_DOALL_ARG_FN(print_leak),
sl@0
   748
				(char *)&ml);
sl@0
   749
	if (ml.chunks != 0)
sl@0
   750
		{
sl@0
   751
		BIO_printf(b,"%ld bytes leaked in %d chunks\n",
sl@0
   752
			   ml.bytes,ml.chunks);
sl@0
   753
		}
sl@0
   754
	else
sl@0
   755
		{
sl@0
   756
		/* Make sure that, if we found no leaks, memory-leak debugging itself
sl@0
   757
		 * does not introduce memory leaks (which might irritate
sl@0
   758
		 * external debugging tools).
sl@0
   759
		 * (When someone enables leak checking, but does not call
sl@0
   760
		 * this function, we declare it to be their fault.)
sl@0
   761
		 *
sl@0
   762
		 * XXX    This should be in CRYPTO_mem_leaks_cb,
sl@0
   763
		 * and CRYPTO_mem_leaks should be implemented by
sl@0
   764
		 * using CRYPTO_mem_leaks_cb.
sl@0
   765
		 * (Also their should be a variant of lh_doall_arg
sl@0
   766
		 * that takes a function pointer instead of a void *;
sl@0
   767
		 * this would obviate the ugly and illegal
sl@0
   768
		 * void_fn_to_char kludge in CRYPTO_mem_leaks_cb.
sl@0
   769
		 * Otherwise the code police will come and get us.)
sl@0
   770
		 */
sl@0
   771
		int old_mh_mode;
sl@0
   772
sl@0
   773
		CRYPTO_w_lock(CRYPTO_LOCK_MALLOC);
sl@0
   774
sl@0
   775
		/* avoid deadlock when lh_free() uses CRYPTO_dbg_free(),
sl@0
   776
		 * which uses CRYPTO_is_mem_check_on */
sl@0
   777
		old_mh_mode = mh_mode;
sl@0
   778
		mh_mode = CRYPTO_MEM_CHECK_OFF;
sl@0
   779
sl@0
   780
		if (mh != NULL)
sl@0
   781
			{
sl@0
   782
			lh_free(mh);
sl@0
   783
			mh = NULL;
sl@0
   784
			}
sl@0
   785
		if (amih != NULL)
sl@0
   786
			{
sl@0
   787
			if (lh_num_items(amih) == 0) 
sl@0
   788
				{
sl@0
   789
				lh_free(amih);
sl@0
   790
				amih = NULL;
sl@0
   791
				}
sl@0
   792
			}
sl@0
   793
sl@0
   794
		mh_mode = old_mh_mode;
sl@0
   795
		CRYPTO_w_unlock(CRYPTO_LOCK_MALLOC);
sl@0
   796
		}
sl@0
   797
	MemCheck_on(); /* release MALLOC2 lock */
sl@0
   798
	}
sl@0
   799
sl@0
   800
#ifndef OPENSSL_NO_FP_API
sl@0
   801
EXPORT_C void CRYPTO_mem_leaks_fp(FILE *fp)
sl@0
   802
	{
sl@0
   803
	BIO *b=NULL;
sl@0
   804
sl@0
   805
	if (mh == NULL) return;
sl@0
   806
	/* Need to turn off memory checking when allocated BIOs ... especially
sl@0
   807
	 * as we're creating them at a time when we're trying to check we've not
sl@0
   808
	 * left anything un-free()'d!! */
sl@0
   809
	MemCheck_off();
sl@0
   810
	b = BIO_new(BIO_s_file());
sl@0
   811
	MemCheck_on();
sl@0
   812
	if(!b) return;
sl@0
   813
	BIO_set_fp(b,fp,BIO_NOCLOSE);
sl@0
   814
	CRYPTO_mem_leaks(b);
sl@0
   815
	BIO_free(b);
sl@0
   816
	}
sl@0
   817
#endif
sl@0
   818
sl@0
   819
sl@0
   820
sl@0
   821
/* FIXME: We really don't allow much to the callback.  For example, it has
sl@0
   822
   no chance of reaching the info stack for the item it processes.  Should
sl@0
   823
   it really be this way?  -- Richard Levitte */
sl@0
   824
/* NB: The prototypes have been typedef'd to CRYPTO_MEM_LEAK_CB inside crypto.h
sl@0
   825
 * If this code is restructured, remove the callback type if it is no longer
sl@0
   826
 * needed. -- Geoff Thorpe */
sl@0
   827
static void cb_leak(const MEM *m, CRYPTO_MEM_LEAK_CB **cb)
sl@0
   828
	{
sl@0
   829
	(**cb)(m->iorder,m->file,m->line,m->num,m->addr);
sl@0
   830
	}
sl@0
   831
sl@0
   832
static IMPLEMENT_LHASH_DOALL_ARG_FN(cb_leak, const MEM *, CRYPTO_MEM_LEAK_CB **)
sl@0
   833
sl@0
   834
EXPORT_C void CRYPTO_mem_leaks_cb(CRYPTO_MEM_LEAK_CB *cb)
sl@0
   835
	{
sl@0
   836
	if (mh == NULL) return;
sl@0
   837
	CRYPTO_w_lock(CRYPTO_LOCK_MALLOC2);
sl@0
   838
	lh_doall_arg(mh, LHASH_DOALL_ARG_FN(cb_leak), &cb);
sl@0
   839
	CRYPTO_w_unlock(CRYPTO_LOCK_MALLOC2);
sl@0
   840
	}