diff options
Diffstat (limited to 'libc/stdlib/malloc/realloc.c')
| -rw-r--r-- | libc/stdlib/malloc/realloc.c | 76 | 
1 files changed, 76 insertions, 0 deletions
diff --git a/libc/stdlib/malloc/realloc.c b/libc/stdlib/malloc/realloc.c new file mode 100644 index 000000000..faf7ac2f8 --- /dev/null +++ b/libc/stdlib/malloc/realloc.c @@ -0,0 +1,76 @@ +/* + * libc/stdlib/malloc-zarg/realloc.c -- realloc function + * + *  Copyright (C) 2002  NEC Corporation + *  Copyright (C) 2002  Miles Bader <miles@gnu.org> + * + * This file is subject to the terms and conditions of the GNU Lesser + * General Public License.  See the file COPYING.LIB in the main + * directory of this archive for more details. + *  + * Written by Miles Bader <miles@gnu.org> + */ + +#include <stdlib.h> +#include <string.h> +#include <sys/mman.h> + +#include "malloc.h" +#include "heap.h" + + +void *realloc (void *mem, size_t new_size) +{ +  if (! mem) +    return malloc (new_size); +  else +    { +      void *base_mem = (size_t *)mem - 1; +      size_t size = *(size_t *)base_mem; + +      MALLOC_DEBUG ("realloc: 0x%lx, %d (base = 0x%lx, total_size = %d)\n", +		    (long)mem, new_size, (long)base_mem, size); + +      if (new_size <= size) +	return mem; +      else +	{ +	  void *new_mem = 0; +	  size_t ext_size = new_size - size; +	  void *ext_addr = (char *)base_mem + ext_size; + +	  if (size >= MALLOC_MMAP_THRESHOLD) +	    /* Try to extend this block in place using mmap.  */ +	    { +	      ext_size += MALLOC_ROUND_UP_TO_PAGE_SIZE (ext_size); + +	      new_mem = mmap (ext_addr, ext_size, PROT_READ | PROT_WRITE, +			      MAP_FIXED | MAP_SHARED | MAP_ANONYMOUS, 0, 0); +	      if (new_mem == MAP_FAILED) +		/* Can't do it.  */ +		ext_size = 0; +	    } +	  else +	    ext_size = __heap_alloc_at (&__malloc_heap, ext_addr, ext_size); + +	  if (! ext_size) +	    /* Our attempts to extend MEM in place failed, just +	       allocate-and-copy.  */ +	    { +	      new_mem = malloc (new_size); +	      if (new_mem) +		{ +		  memcpy (new_mem, mem, size); +		  free (mem); +		} +	    } + +	  if (new_mem) +	    MALLOC_DEBUG ("  realloc: returning 0x%lx" +			  " (base:0x%lx, total_size:%d)\n", +			  (long)new_mem, (long)new_mem - sizeof(size_t), size); + +	  return new_mem; +	} +    } +}  | 
