]> git.sur5r.net Git - cc65/blob - libsrc/common/realloc.c
Don't pass mode argument to open() from _fopen().
[cc65] / libsrc / common / realloc.c
1 /*****************************************************************************/
2 /*                                                                           */
3 /*                                 realloc.c                                 */
4 /*                                                                           */
5 /*               Change the size of an allocated memory block                */
6 /*                                                                           */
7 /*                                                                           */
8 /*                                                                           */
9 /* (C) 1998-2004 Ullrich von Bassewitz                                       */
10 /*               Wacholderweg 14                                             */
11 /*               D-70597 Stuttgart                                           */
12 /* EMail:        uz@musoftware.de                                            */
13 /*                                                                           */
14 /*                                                                           */
15 /* This software is provided 'as-is', without any expressed or implied       */
16 /* warranty.  In no event will the authors be held liable for any damages    */
17 /* arising from the use of this software.                                    */
18 /*                                                                           */
19 /* Permission is granted to anyone to use this software for any purpose,     */
20 /* including commercial applications, and to alter it and redistribute it    */
21 /* freely, subject to the following restrictions:                            */
22 /*                                                                           */
23 /* 1. The origin of this software must not be misrepresented; you must not   */
24 /*    claim that you wrote the original software. If you use this software   */
25 /*    in a product, an acknowledgment in the product documentation would be  */
26 /*    appreciated but is not required.                                       */
27 /* 2. Altered source versions must be plainly marked as such, and must not   */
28 /*    be misrepresented as being the original software.                      */
29 /* 3. This notice may not be removed or altered from any source              */
30 /*    distribution.                                                          */
31 /*                                                                           */
32 /*****************************************************************************/
33
34
35
36 #include <stdlib.h>
37 #include <string.h>
38 #include <_heap.h>
39
40
41
42 void* __fastcall__ realloc (void* block, register size_t size)
43 {
44     register struct usedblock* b;
45     struct usedblock* newblock;
46     unsigned oldsize;
47     unsigned newhptr;
48
49     /* Check the block parameter */
50     if (!block) {
51         /* Block is NULL, same as malloc */
52         return malloc (size);
53     }
54
55     /* Check the size parameter */
56     if (size == 0) {
57         /* Block is not NULL, but size is: free the block */
58         free (block);
59         return 0;
60     }
61
62     /* Make the internal used size from the given size */
63     size += HEAP_ADMIN_SPACE;
64     if (size < sizeof (struct freeblock)) {
65         size = sizeof (struct freeblock);
66     }
67
68     /* The word below the user block contains a pointer to the start of the
69     ** raw memory block. The first word of this raw memory block is the full
70     ** size of the block. Get a pointer to the real block, get the old block
71     ** size.
72     */
73     b = (((struct usedblock*) block) - 1)->start;
74     oldsize = b->size;
75
76     /* Is the block at the current heap top? */
77     if (((unsigned) b) + oldsize == ((unsigned) _heapptr)) {
78         /* Check if we've enough memory at the heap top */
79         newhptr = ((unsigned) _heapptr) - oldsize + size;
80         if (newhptr <= ((unsigned) _heapend)) {
81             /* Ok, there's space enough */
82             _heapptr = (unsigned*) newhptr;
83             b->size = size;
84             b->start = b;
85             return block;
86         }
87     }
88
89     /* The given block was not located on top of the heap, or there's no
90     ** room left. Try to allocate a new block and copy the data.
91     */
92     if (newblock = malloc (size)) {
93
94         /* Adjust the old size to the user visible portion */
95         oldsize -= HEAP_ADMIN_SPACE;
96
97         /* If the new block is larger than the old one, copy the old
98         ** data only
99         */
100         if (size > oldsize) {
101             size = oldsize;
102         }
103
104         /* Copy the block data */
105         memcpy (newblock, block, size);
106         free (block);
107     }
108     return newblock;
109 }
110
111
112