mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-30 04:26:45 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			459 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			459 lines
		
	
	
		
			9.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-
 | |
|  * See the file LICENSE for redistribution information.
 | |
|  *
 | |
|  * Copyright (c) 1997-2002
 | |
|  *	Sleepycat Software.  All rights reserved.
 | |
|  */
 | |
| 
 | |
| #include "db_config.h"
 | |
| 
 | |
| #ifndef lint
 | |
| static const char revid[] = "$Id: os_alloc.c,v 11.32 2002/08/06 04:57:07 bostic Exp $";
 | |
| #endif /* not lint */
 | |
| 
 | |
| #ifndef NO_SYSTEM_INCLUDES
 | |
| #include <sys/types.h>
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #endif
 | |
| 
 | |
| #include "db_int.h"
 | |
| 
 | |
| #ifdef DIAGNOSTIC
 | |
| static void __os_guard __P((DB_ENV *));
 | |
| 
 | |
| union __db_alloc {
 | |
| 	size_t size;
 | |
| 	double align;
 | |
| };
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * !!!
 | |
|  * Correct for systems that return NULL when you allocate 0 bytes of memory.
 | |
|  * There are several places in DB where we allocate the number of bytes held
 | |
|  * by the key/data item, and it can be 0.  Correct here so that malloc never
 | |
|  * returns a NULL for that reason (which behavior is permitted by ANSI).  We
 | |
|  * could make these calls macros on non-Alpha architectures (that's where we
 | |
|  * saw the problem), but it's probably not worth the autoconf complexity.
 | |
|  *
 | |
|  * !!!
 | |
|  * Correct for systems that don't set errno when malloc and friends fail.
 | |
|  *
 | |
|  * !!!
 | |
|  * There is no circumstance in which we can call __os_umalloc, __os_urealloc
 | |
|  * or __os_ufree without an environment handle, as we need one to determine
 | |
|  * whether or not to use an application-specified malloc function.  If we
 | |
|  * don't have an environment handle, we should be calling __os_XXX instead.
 | |
|  * Make DIAGNOSTIC blow up if we get this wrong.
 | |
|  *
 | |
|  *	Out of memory.
 | |
|  *	We wish to hold the whole sky,
 | |
|  *	But we never will.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * __os_umalloc --
 | |
|  *	A malloc(3) function that will use, in order of preference,
 | |
|  *	the allocation function specified to the DB handle, the DB_ENV
 | |
|  *	handle, or __os_malloc.
 | |
|  *
 | |
|  * PUBLIC: int __os_umalloc __P((DB_ENV *, size_t, void *));
 | |
|  */
 | |
| int
 | |
| __os_umalloc(dbenv, size, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	size_t size;
 | |
| 	void *storep;
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	/* Require an environment handle. */
 | |
| 	DB_ASSERT(dbenv != NULL);
 | |
| 
 | |
| 	/* Never allocate 0 bytes -- some C libraries don't like it. */
 | |
| 	if (size == 0)
 | |
| 		++size;
 | |
| 
 | |
| 	if (dbenv == NULL || dbenv->db_malloc == NULL) {
 | |
| 		if (DB_GLOBAL(j_malloc) != NULL)
 | |
| 			*(void **)storep = DB_GLOBAL(j_malloc)(size);
 | |
| 		else
 | |
| 			*(void **)storep = malloc(size);
 | |
| 		if (*(void **)storep == NULL) {
 | |
| 			/*
 | |
| 			 *  Correct error return, see __os_malloc.
 | |
| 			 */
 | |
| 			if ((ret = __os_get_errno()) == 0) {
 | |
| 				ret = ENOMEM;
 | |
| 				__os_set_errno(ENOMEM);
 | |
| 			}
 | |
| 			__db_err(dbenv,
 | |
| 			    "malloc: %s: %lu", strerror(ret), (u_long)size);
 | |
| 			return (ret);
 | |
| 		}
 | |
| 		return (0);
 | |
| 	}
 | |
| 
 | |
| 	if ((*(void **)storep = dbenv->db_malloc(size)) == NULL) {
 | |
| 		__db_err(dbenv, "User-specified malloc function returned NULL");
 | |
| 		return (ENOMEM);
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_urealloc --
 | |
|  *	realloc(3) counterpart to __os_umalloc.
 | |
|  *
 | |
|  * PUBLIC: int __os_urealloc __P((DB_ENV *, size_t, void *));
 | |
|  */
 | |
| int
 | |
| __os_urealloc(dbenv, size, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	size_t size;
 | |
| 	void *storep;
 | |
| {
 | |
| 	int ret;
 | |
| 	void *ptr;
 | |
| 
 | |
| 	ptr = *(void **)storep;
 | |
| 
 | |
| 	/* Require an environment handle. */
 | |
| 	DB_ASSERT(dbenv != NULL);
 | |
| 
 | |
| 	/* Never allocate 0 bytes -- some C libraries don't like it. */
 | |
| 	if (size == 0)
 | |
| 		++size;
 | |
| 
 | |
| 	if (dbenv == NULL || dbenv->db_realloc == NULL) {
 | |
| 		if (ptr == NULL)
 | |
| 			return (__os_umalloc(dbenv, size, storep));
 | |
| 
 | |
| 		if (DB_GLOBAL(j_realloc) != NULL)
 | |
| 			*(void **)storep = DB_GLOBAL(j_realloc)(ptr, size);
 | |
| 		else
 | |
| 			*(void **)storep = realloc(ptr, size);
 | |
| 		if (*(void **)storep == NULL) {
 | |
| 			/*
 | |
| 			 * Correct errno, see __os_realloc.
 | |
| 			 */
 | |
| 			if ((ret = __os_get_errno()) == 0) {
 | |
| 				ret = ENOMEM;
 | |
| 				__os_set_errno(ENOMEM);
 | |
| 			}
 | |
| 			__db_err(dbenv,
 | |
| 			    "realloc: %s: %lu", strerror(ret), (u_long)size);
 | |
| 			return (ret);
 | |
| 		}
 | |
| 		return (0);
 | |
| 	}
 | |
| 
 | |
| 	if ((*(void **)storep = dbenv->db_realloc(ptr, size)) == NULL) {
 | |
| 		__db_err(dbenv,
 | |
| 		    "User-specified realloc function returned NULL");
 | |
| 		return (ENOMEM);
 | |
| 	}
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_ufree --
 | |
|  *	free(3) counterpart to __os_umalloc.
 | |
|  *
 | |
|  * PUBLIC: int __os_ufree __P((DB_ENV *, void *));
 | |
|  */
 | |
| int
 | |
| __os_ufree(dbenv, ptr)
 | |
| 	DB_ENV *dbenv;
 | |
| 	void *ptr;
 | |
| {
 | |
| 	/* Require an environment handle. */
 | |
| 	DB_ASSERT(dbenv != NULL);
 | |
| 
 | |
| 	if (dbenv != NULL && dbenv->db_free != NULL)
 | |
| 		dbenv->db_free(ptr);
 | |
| 	else if (DB_GLOBAL(j_free) != NULL)
 | |
| 		DB_GLOBAL(j_free)(ptr);
 | |
| 	else
 | |
| 		free(ptr);
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_strdup --
 | |
|  *	The strdup(3) function for DB.
 | |
|  *
 | |
|  * PUBLIC: int __os_strdup __P((DB_ENV *, const char *, void *));
 | |
|  */
 | |
| int
 | |
| __os_strdup(dbenv, str, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	const char *str;
 | |
| 	void *storep;
 | |
| {
 | |
| 	size_t size;
 | |
| 	int ret;
 | |
| 	void *p;
 | |
| 
 | |
| 	*(void **)storep = NULL;
 | |
| 
 | |
| 	size = strlen(str) + 1;
 | |
| 	if ((ret = __os_malloc(dbenv, size, &p)) != 0)
 | |
| 		return (ret);
 | |
| 
 | |
| 	memcpy(p, str, size);
 | |
| 
 | |
| 	*(void **)storep = p;
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_calloc --
 | |
|  *	The calloc(3) function for DB.
 | |
|  *
 | |
|  * PUBLIC: int __os_calloc __P((DB_ENV *, size_t, size_t, void *));
 | |
|  */
 | |
| int
 | |
| __os_calloc(dbenv, num, size, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	size_t num, size;
 | |
| 	void *storep;
 | |
| {
 | |
| 	void *p;
 | |
| 	int ret;
 | |
| 
 | |
| 	size *= num;
 | |
| 	if ((ret = __os_malloc(dbenv, size, &p)) != 0)
 | |
| 		return (ret);
 | |
| 
 | |
| 	memset(p, 0, size);
 | |
| 
 | |
| 	*(void **)storep = p;
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_malloc --
 | |
|  *	The malloc(3) function for DB.
 | |
|  *
 | |
|  * PUBLIC: int __os_malloc __P((DB_ENV *, size_t, void *));
 | |
|  */
 | |
| int
 | |
| __os_malloc(dbenv, size, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	size_t size;
 | |
| 	void *storep;
 | |
| {
 | |
| 	int ret;
 | |
| 	void *p;
 | |
| 
 | |
| 	*(void **)storep = NULL;
 | |
| 
 | |
| 	/* Never allocate 0 bytes -- some C libraries don't like it. */
 | |
| 	if (size == 0)
 | |
| 		++size;
 | |
| 
 | |
| #ifdef DIAGNOSTIC
 | |
| 	/* Add room for size and a guard byte. */
 | |
| 	size += sizeof(union __db_alloc) + 1;
 | |
| #endif
 | |
| 
 | |
| 	if (DB_GLOBAL(j_malloc) != NULL)
 | |
| 		p = DB_GLOBAL(j_malloc)(size);
 | |
| 	else
 | |
| 		p = malloc(size);
 | |
| 	if (p == NULL) {
 | |
| 		/*
 | |
| 		 * Some C libraries don't correctly set errno when malloc(3)
 | |
| 		 * fails.  We'd like to 0 out errno before calling malloc,
 | |
| 		 * but it turns out that setting errno is quite expensive on
 | |
| 		 * Windows/NT in an MT environment.
 | |
| 		 */
 | |
| 		if ((ret = __os_get_errno()) == 0) {
 | |
| 			ret = ENOMEM;
 | |
| 			__os_set_errno(ENOMEM);
 | |
| 		}
 | |
| 		__db_err(dbenv,
 | |
| 		    "malloc: %s: %lu", strerror(ret), (u_long)size);
 | |
| 		return (ret);
 | |
| 	}
 | |
| 
 | |
| #ifdef DIAGNOSTIC
 | |
| 	/*
 | |
| 	 * Guard bytes: if #DIAGNOSTIC is defined, we allocate an additional
 | |
| 	 * byte after the memory and set it to a special value that we check
 | |
| 	 * for when the memory is free'd.
 | |
| 	 */
 | |
| 	((u_int8_t *)p)[size - 1] = CLEAR_BYTE;
 | |
| 
 | |
| 	((union __db_alloc *)p)->size = size;
 | |
| 	p = &((union __db_alloc *)p)[1];
 | |
| #endif
 | |
| 	*(void **)storep = p;
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_realloc --
 | |
|  *	The realloc(3) function for DB.
 | |
|  *
 | |
|  * PUBLIC: int __os_realloc __P((DB_ENV *, size_t, void *));
 | |
|  */
 | |
| int
 | |
| __os_realloc(dbenv, size, storep)
 | |
| 	DB_ENV *dbenv;
 | |
| 	size_t size;
 | |
| 	void *storep;
 | |
| {
 | |
| 	int ret;
 | |
| 	void *p, *ptr;
 | |
| 
 | |
| 	ptr = *(void **)storep;
 | |
| 
 | |
| 	/* Never allocate 0 bytes -- some C libraries don't like it. */
 | |
| 	if (size == 0)
 | |
| 		++size;
 | |
| 
 | |
| 	/* If we haven't yet allocated anything yet, simply call malloc. */
 | |
| 	if (ptr == NULL)
 | |
| 		return (__os_malloc(dbenv, size, storep));
 | |
| 
 | |
| #ifdef DIAGNOSTIC
 | |
| 	/* Add room for size and a guard byte. */
 | |
| 	size += sizeof(union __db_alloc) + 1;
 | |
| 
 | |
| 	/* Back up to the real begining */
 | |
| 	ptr = &((union __db_alloc *)ptr)[-1];
 | |
| #endif
 | |
| 
 | |
| 	/*
 | |
| 	 * Don't overwrite the original pointer, there are places in DB we
 | |
| 	 * try to continue after realloc fails.
 | |
| 	 */
 | |
| 	if (DB_GLOBAL(j_realloc) != NULL)
 | |
| 		p = DB_GLOBAL(j_realloc)(ptr, size);
 | |
| 	else
 | |
| 		p = realloc(ptr, size);
 | |
| 	if (p == NULL) {
 | |
| 		/*
 | |
| 		 * Some C libraries don't correctly set errno when malloc(3)
 | |
| 		 * fails.  We'd like to 0 out errno before calling malloc,
 | |
| 		 * but it turns out that setting errno is quite expensive on
 | |
| 		 * Windows/NT in an MT environment.
 | |
| 		 */
 | |
| 		if ((ret = __os_get_errno()) == 0) {
 | |
| 			ret = ENOMEM;
 | |
| 			__os_set_errno(ENOMEM);
 | |
| 		}
 | |
| 		__db_err(dbenv,
 | |
| 		    "realloc: %s: %lu", strerror(ret), (u_long)size);
 | |
| 		return (ret);
 | |
| 	}
 | |
| #ifdef DIAGNOSTIC
 | |
| 	((u_int8_t *)p)[size - 1] = CLEAR_BYTE;	/* Initialize guard byte. */
 | |
| 
 | |
| 	((union __db_alloc *)p)->size = size;
 | |
| 	p = &((union __db_alloc *)p)[1];
 | |
| #endif
 | |
| 
 | |
| 	*(void **)storep = p;
 | |
| 
 | |
| 	return (0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * __os_free --
 | |
|  *	The free(3) function for DB.
 | |
|  *
 | |
|  * PUBLIC: void __os_free __P((DB_ENV *, void *));
 | |
|  */
 | |
| void
 | |
| __os_free(dbenv, ptr)
 | |
| 	DB_ENV *dbenv;
 | |
| 	void *ptr;
 | |
| {
 | |
| #ifdef DIAGNOSTIC
 | |
| 	int size;
 | |
| 	/*
 | |
| 	 * Check that the guard byte (one past the end of the memory) is
 | |
| 	 * still CLEAR_BYTE.
 | |
| 	 */
 | |
| 	if (ptr == NULL)
 | |
| 		return;
 | |
| 
 | |
| 	ptr = &((union __db_alloc *)ptr)[-1];
 | |
| 	size = ((union __db_alloc *)ptr)->size;
 | |
| 	if (((u_int8_t *)ptr)[size - 1] != CLEAR_BYTE)
 | |
| 		 __os_guard(dbenv);
 | |
| 
 | |
| 	/* Clear memory. */
 | |
| 	if (size != 0)
 | |
| 		memset(ptr, CLEAR_BYTE, size);
 | |
| #endif
 | |
| 	COMPQUIET(dbenv, NULL);
 | |
| 
 | |
| 	if (DB_GLOBAL(j_free) != NULL)
 | |
| 		DB_GLOBAL(j_free)(ptr);
 | |
| 	else
 | |
| 		free(ptr);
 | |
| }
 | |
| 
 | |
| #ifdef DIAGNOSTIC
 | |
| /*
 | |
|  * __os_guard --
 | |
|  *	Complain and abort.
 | |
|  */
 | |
| static void
 | |
| __os_guard(dbenv)
 | |
| 	DB_ENV *dbenv;
 | |
| {
 | |
| 	__db_err(dbenv, "Guard byte incorrect during free");
 | |
| 	abort();
 | |
| 	/* NOTREACHED */
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * __ua_memcpy --
 | |
|  *	Copy memory to memory without relying on any kind of alignment.
 | |
|  *
 | |
|  *	There are places in DB that we have unaligned data, for example,
 | |
|  *	when we've stored a structure in a log record as a DBT, and now
 | |
|  *	we want to look at it.  Unfortunately, if you have code like:
 | |
|  *
 | |
|  *		struct a {
 | |
|  *			int x;
 | |
|  *		} *p;
 | |
|  *
 | |
|  *		void *func_argument;
 | |
|  *		int local;
 | |
|  *
 | |
|  *		p = (struct a *)func_argument;
 | |
|  *		memcpy(&local, p->x, sizeof(local));
 | |
|  *
 | |
|  *	compilers optimize to use inline instructions requiring alignment,
 | |
|  *	and records in the log don't have any particular alignment.  (This
 | |
|  *	isn't a compiler bug, because it's a structure they're allowed to
 | |
|  *	assume alignment.)
 | |
|  *
 | |
|  *	Casting the memcpy arguments to (u_int8_t *) appears to work most
 | |
|  *	of the time, but we've seen examples where it wasn't sufficient
 | |
|  *	and there's nothing in ANSI C that requires that work.
 | |
|  *
 | |
|  * PUBLIC: void *__ua_memcpy __P((void *, const void *, size_t));
 | |
|  */
 | |
| void *
 | |
| __ua_memcpy(dst, src, len)
 | |
| 	void *dst;
 | |
| 	const void *src;
 | |
| 	size_t len;
 | |
| {
 | |
| 	return ((void *)memcpy(dst, src, len));
 | |
| }
 | 
