1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
/* Stack allocation routines. This is intended for machines without support
for the `alloca' function.
Copyright 1996, 1997, 1999, 2000, 2001, 2006 Free Software Foundation, Inc.
This file is part of the GNU MP Library.
The GNU MP Library is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3 of the License, or (at your
option) any later version.
The GNU MP 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 Lesser General Public
License for more details.
You should have received a copy of the GNU Lesser General Public License
along with the GNU MP Library. If not, see http://www.gnu.org/licenses/. */
#include "gmp.h"
#include "gmp-impl.h"
struct tmp_stack
{
void *end;
void *alloc_point;
struct tmp_stack *prev;
};
typedef struct tmp_stack tmp_stack;
static unsigned long max_total_allocation = 0;
static unsigned long current_total_allocation = 0;
static tmp_stack xxx = {&xxx, &xxx, 0};
static tmp_stack *current = &xxx;
/* The rounded size of the header of each allocation block. */
#define HSIZ ROUND_UP_MULTIPLE (sizeof (tmp_stack), __TMP_ALIGN)
/* Allocate a block of exactly <size> bytes. This should only be called
through the TMP_ALLOC macro, which takes care of rounding/alignment. */
void *
__gmp_tmp_alloc (unsigned long size)
{
void *that;
ASSERT ((size % __TMP_ALIGN) == 0);
ASSERT (((unsigned) current->alloc_point % __TMP_ALIGN) == 0);
if (size > (char *) current->end - (char *) current->alloc_point)
{
void *chunk;
tmp_stack *header;
unsigned long chunk_size;
unsigned long now;
/* Allocate a chunk that makes the total current allocation somewhat
larger than the maximum allocation ever. If size is very large, we
allocate that much. */
now = current_total_allocation + size;
if (now > max_total_allocation)
{
/* We need more temporary memory than ever before. Increase
for future needs. */
now = (now * 3 / 2 + __TMP_ALIGN - 1) & -__TMP_ALIGN;
chunk_size = now - current_total_allocation + HSIZ;
current_total_allocation = now;
max_total_allocation = current_total_allocation;
}
else
{
chunk_size = max_total_allocation - current_total_allocation + HSIZ;
current_total_allocation = max_total_allocation;
}
chunk = (*__gmp_allocate_func) (chunk_size);
header = (tmp_stack *) chunk;
header->end = (char *) chunk + chunk_size;
header->alloc_point = (char *) chunk + HSIZ;
header->prev = current;
current = header;
}
that = current->alloc_point;
current->alloc_point = (char *) that + size;
ASSERT (((unsigned) that % __TMP_ALIGN) == 0);
return that;
}
/* Typically called at function entry. <mark> is assigned so that
__gmp_tmp_free can later be used to reclaim all subsequently allocated
storage. */
void
__gmp_tmp_mark (struct tmp_marker *mark)
{
mark->which_chunk = current;
mark->alloc_point = current->alloc_point;
}
/* Free everything allocated since <mark> was assigned by __gmp_tmp_mark */
void
__gmp_tmp_free (struct tmp_marker *mark)
{
while (mark->which_chunk != current)
{
tmp_stack *tmp;
tmp = current;
current = tmp->prev;
current_total_allocation -= (((char *) (tmp->end) - (char *) tmp) - HSIZ);
(*__gmp_free_func) (tmp, (char *) tmp->end - (char *) tmp);
}
current->alloc_point = mark->alloc_point;
}
|