/************************************************************ * mm_realloc.c * * Copyright (C) 2007 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name Gregory Nutt nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ************************************************************/ /************************************************************ * Included Files ************************************************************/ #include #include "mm_environment.h" #include /* For NULL */ #include "mm_internal.h" /************************************************************ * Definitions ************************************************************/ /************************************************************ * Global Functions ************************************************************/ /************************************************************ * realloc * * Description: * If the reallocation is for less space, then: * (1) the current allocation is reduced in size * (2) the remainder at the end of the allocation is * returned to the free list. * * If the request is for more space and the current * allocation can be extended, it will be extended by: * (1) Taking the additional space from the following * free chunk, or * (2) Taking the additional space from the preceding * free chunk. * (3) Or both * * If the request is for more space but the current chunk * cannot be extended, then malloc a new buffer, copy the * data into the new buffer, and free the old buffer. * ************************************************************/ void *realloc(void *oldmem, size_t size) { struct mm_allocnode_s *oldnode; struct mm_freenode_s *prev; struct mm_freenode_s *next; uint32 oldsize; uint32 prevsize = 0; uint32 nextsize = 0; /* If oldmem is NULL, then realloc is equivalent to malloc */ if (!oldmem) { return malloc(size); } /* If size is zero, then realloc is equivalent to free */ if (size <= 0) { free(oldmem); return NULL; } /* Adjust the size to account for (1) the size of the allocated * node and (2) to make sure that it is an even multiple of * our granule size. */ size = MM_ALIGN_UP(size + SIZEOF_MM_ALLOCNODE); /* Map the memory chunk into an allocated node structure */ oldnode = (struct mm_allocnode_s *)((char*)oldmem - SIZEOF_MM_ALLOCNODE); /* We need to hold the MM semaphore while we muck with the * nodelist. */ mm_takesemaphore(); /* Check if this is a request to reduce the size of the allocation. */ oldsize = oldnode->size; if (size <= oldsize) { mm_shrinkchunk(oldnode, size); mm_givesemaphore(); return oldmem; } /* This is a request to increase the size of the allocation, Get the * available sizes before and after the oldnode so that we can make * the best decision */ next = (struct mm_freenode_s *)((char*)oldnode + oldnode->size); if ((next->preceding & MM_ALLOC_BIT) == 0) { nextsize = next->size; } prev = (struct mm_freenode_s *)((char*)oldnode - (oldnode->preceding & ~MM_ALLOC_BIT)); if ((prev->preceding & MM_ALLOC_BIT) == 0) { prevsize = prev->size; } /* Now, check if we can extend the current allocation or not */ if (nextsize + prevsize + oldsize >= size) { uint32 needed = size - oldsize; uint32 takeprev; uint32 takenext; /* Check if we can extend into the previous chunk and if the * previous chunk is smaller than the next chunk. */ if (prevsize > 0 && (nextsize >= prevsize || nextsize <= 0)) { /* Can we get everything we need from the previous chunk? */ if (needed > prevsize) { /* No, take the whole previous chunk and get the * rest that we need from the next chunk. */ takeprev = prevsize; takenext = needed - prevsize; } else { /* Yes, take what we need from the previous chunk */ takeprev = needed; takenext = 0; } needed = 0; } /* Check if we can extend into the next chunk and if we still * need more memory. */ if (nextsize > 0 && needed) { /* Can we get everything we need from the next chunk? */ if (needed > nextsize) { /* No, take the whole next chunk and get the * rest that we need from the previous chunk. */ takeprev = needed - nextsize; takenext = nextsize; } else { /* Yes, take what we need from the previous chunk */ takeprev = 0; takenext = needed; } } /* Extend into the previous free chunk */ if (takeprev) { struct mm_allocnode_s *newnode; /* Remove the previous node. There must be a predecessor, * but there may not be a successor node. */ DEBUGASSERT(prev->blink); prev->blink->flink = prev->flink; if (prev->flink) { prev->flink->blink = prev->blink; } /* Extend the node into the previous free chunk */ newnode = (struct mm_allocnode_s *)((char*)oldnode - takeprev); /* Did we consume the entire preceding chunk? */ if (takeprev < prevsize) { /* No, just take what we need from the previous chunk * and put it back into the free list */ prev->size -= takeprev; newnode->size = oldsize + takeprev; newnode->preceding = prev->size | MM_ALLOC_BIT; next->preceding = newnode->size | (next->preceding & MM_ALLOC_BIT); /* Return the previous free node to the nodelist (with the new size) */ mm_addfreechunk(prev); /* Now we want to return newnode */ oldnode = newnode; } else { /* Yes update its size (newnode->preceding is already set) */ newnode->size += oldsize; next->preceding = newnode->size; } oldnode = newnode; oldsize = newnode->size; } /* Extend into the next free chunk */ if (takenext) { struct mm_freenode_s *newnode; struct mm_allocnode_s *andbeyond; /* Get the chunk following the next node (which could be the tail chunk) */ andbeyond = (struct mm_allocnode_s*)((char*)next + nextsize); /* Remove the next node. There must be a predecessor, * but there may not be a successor node. */ DEBUGASSERT(next->blink); next->blink->flink = next->flink; if (next->flink) { next->flink->blink = next->blink; } /* Extend the node into the previous next chunk */ oldnode->size = oldsize + takenext; newnode = (struct mm_freenode_s *)((char*)oldnode + oldnode->size); /* Did we consume the entire preceding chunk? */ if (takenext < nextsize) { /* No, take what we need from the next chunk and return it * to the free nodelist. */ newnode->size = nextsize - takenext; newnode->preceding = oldnode->size; andbeyond->preceding = newnode->size | (andbeyond->preceding & MM_ALLOC_BIT); /* Add the new free node to the nodelist (with the new size) */ mm_addfreechunk(newnode); } else { /* Yes, just update some pointers. */ andbeyond->preceding = oldnode->size | (andbeyond->preceding & MM_ALLOC_BIT); } } mm_givesemaphore(); return (void*)((char*)oldnode + SIZEOF_MM_ALLOCNODE); } /* The current chunk cannot be extended. Just allocate a new chunk and copy */ else { /* Allocate a new block. On failure, realloc must return NULL but * leave the original memory in place. */ mm_givesemaphore(); char *newmem = (char*)malloc(size); if (newmem) { memcpy(newmem, oldmem, oldsize); free(oldmem); } return newmem; } }