ClamAV is an open source (GPLv2) anti-virus toolkit.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
clamav/clamav-devel/libclamav/js/alloc.c

318 lines
5.4 KiB

/*
* Memory allocation routines.
* Copyright (c) 1998 New Generation Software (NGS) Oy
*
* Author: Markku Rossi <mtr@ngs.fi>
*/
/*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the Free
* Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
* MA 02111-1307, USA
*/
#if HAVE_CONFIG_H
#include "clamav-config.h"
#endif
#ifdef CL_EXPERIMENTAL
/*
* $Source: /tmp/cvsroot-15-2-2007/clamav-devel/libclamav/js/alloc.c,v $
* $Id: alloc.c,v 1.3 2006/10/28 11:27:44 njh Exp $
*/
#include "js/jsint.h"
/*
* Global functions.
*/
#if JS_DEBUG_MEMORY_LEAKS
struct mem_debug_st
{
struct mem_debug_st *next;
struct mem_debug_st *prev;
char *file;
int line;
size_t size;
};
typedef struct mem_debug_st MemDebug;
static MemDebug *mem_debug_blocks = NULL;
int mem_debug_balance = 0;
unsigned int alloc_fail = 0;
unsigned int alloc_count = 0;
static void
register_block (MemDebug *b, size_t size, char *file, int line)
{
b->next = NULL;
b->prev = NULL;
b->file = file;
b->line = line;
b->size = size;
if (mem_debug_blocks)
{
b->next = mem_debug_blocks;
mem_debug_blocks->prev = b;
}
mem_debug_blocks = b;
mem_debug_balance++;
}
static void
unregister_block (MemDebug *b)
{
if (b->file == NULL)
{
fprintf (stderr, "freeing the same block twise\n");
abort ();
}
if (b->next)
b->next->prev = b->prev;
if (b->prev)
b->prev->next = b->next;
else
mem_debug_blocks = b->next;
b->file = NULL;
mem_debug_balance--;
}
static inline int
check_fail ()
{
return alloc_fail != 0 && ++alloc_count >= alloc_fail;
}
void
js_alloc_dump_blocks ()
{
MemDebug *b;
unsigned int bytes = 0;
fprintf (stderr, "js_alloc_dump_blocks(): #blocks=%d\n", mem_debug_balance);
for (b = mem_debug_blocks; b; b = b->next)
{
fprintf (stderr, "%s:%d: %lu\n", b->file, b->line, b->size);
bytes += b->size;
}
fprintf (stderr, "leaks=%u\n", bytes);
}
void *
js_malloc_i (JSVirtualMachine *vm, size_t size, char *file, int line)
{
MemDebug *ptr;
ptr = malloc (sizeof (MemDebug) + size);
if (check_fail () || ptr == NULL)
{
if (vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return NULL;
}
register_block (ptr, size, file, line);
return (unsigned char *) ptr + sizeof (MemDebug);
}
void *
js_calloc_i (JSVirtualMachine *vm, size_t num, size_t size, char *file,
int line)
{
MemDebug *ptr;
ptr = malloc (sizeof (MemDebug) + num * size);
if (check_fail () || ptr == NULL)
{
if (vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return NULL;
}
memset (ptr, 0, sizeof (MemDebug) + num * size);
register_block (ptr, num * size, file, line);
return (unsigned char *) ptr + sizeof (MemDebug);
}
void *
js_realloc_i (JSVirtualMachine *vm, void *ptr, size_t size, char *file,
int line)
{
void *nptr;
MemDebug *b;
if (ptr == NULL)
return js_malloc_i (vm, size, file, line);
nptr = js_malloc_i (vm, size, file, line);
if (nptr == NULL)
{
if (vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return NULL;
}
b = (MemDebug *) ((unsigned char *) ptr - sizeof (MemDebug));
memcpy (nptr, ptr, size < b->size ? size : b->size);
js_free (ptr);
return nptr;
}
void
js_free (void *ptr)
{
MemDebug *b;
if (ptr == NULL)
return;
b = (MemDebug *) ((unsigned char *) ptr - sizeof (MemDebug));
unregister_block (b);
free (b);
}
char *
js_strdup_i (JSVirtualMachine *vm, const char *str, char *file, int line)
{
char *tmp;
tmp = js_malloc_i (vm, strlen (str) + 1, file, line);
if (tmp == NULL)
return NULL;
strcpy (tmp, str);
return tmp;
}
#else /* not JS_DEBUG_MEMORY_LEAKS */
void *
js_malloc (JSVirtualMachine *vm, size_t size)
{
void *ptr;
ptr = malloc (size);
if (ptr == NULL && vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return ptr;
}
void *
js_calloc (JSVirtualMachine *vm, size_t num, size_t size)
{
void *ptr;
ptr = calloc (num, size);
if (ptr == NULL && vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return ptr;
}
void *
js_realloc (JSVirtualMachine *vm, void *ptr, size_t size)
{
void *nptr;
if (ptr == NULL)
return js_malloc (vm, size);
nptr = realloc (ptr, size);
if (nptr == NULL && vm != NULL)
{
sprintf (vm->error, "VM: memory exhausted");
js_vm_error (vm);
}
return nptr;
}
void
js_free (void *ptr)
{
if (ptr == NULL)
return;
free (ptr);
}
char *
js_strdup (JSVirtualMachine *vm, const char *str)
{
char *tmp;
tmp = js_malloc (vm, strlen (str) + 1);
if (tmp == NULL)
return NULL;
strcpy (tmp, str);
return tmp;
}
#endif /* not JS_DEBUG_MEMORY_LEAKS */
#endif /*CL_EXPERIMENTAL*/