Implemented some functions: gc_get_free_bytes, gc_get_heap_size,
[cacao.git] / src / mm / boehm.c
1 /* mm/boehm.c - interface for boehm gc
2
3    Copyright (C) 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003
4    R. Grafl, A. Krall, C. Kruegel, C. Oates, R. Obermaisser,
5    M. Probst, S. Ring, E. Steiner, C. Thalinger, D. Thuernbeck,
6    P. Tomsich, J. Wenninger
7
8    This file is part of CACAO.
9
10    This program is free software; you can redistribute it and/or
11    modify it under the terms of the GNU General Public License as
12    published by the Free Software Foundation; either version 2, or (at
13    your option) any later version.
14
15    This program is distributed in the hope that it will be useful, but
16    WITHOUT ANY WARRANTY; without even the implied warranty of
17    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18    General Public License for more details.
19
20    You should have received a copy of the GNU General Public License
21    along with this program; if not, write to the Free Software
22    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
23    02111-1307, USA.
24
25    Contact: cacao@complang.tuwien.ac.at
26
27    Authors: Stefan Ring
28
29    $Id: boehm.c 984 2004-03-28 23:08:07Z twisti $
30
31 */
32
33 #if defined(USE_THREADS) && defined(NATIVE_THREADS) && defined(__LINUX__)
34 #define GC_LINUX_THREADS
35 #endif
36
37 #include "main.h"
38 #include "boehm.h"
39 #include "global.h"
40 #include "native.h"
41 #include "asmpart.h"
42 #include "builtin.h"
43 #include "threads/thread.h"
44 #include "toolbox/loging.h"
45 #include "gc.h"
46
47
48 static void *stackcall_twoargs(struct otherstackcall *p)
49 {
50         return (*p->p2)(p->p, p->l);
51 }
52
53
54 static void *stackcall_malloc(void *p, u4 bytelength)
55 {
56         return GC_MALLOC(bytelength);
57 }
58
59
60 static void *stackcall_malloc_atomic(void *p, u4 bytelength)
61 {
62         return GC_MALLOC_ATOMIC(bytelength);
63 }
64
65
66 static void *stackcall_malloc_uncollectable(void *p, u4 bytelength)
67 {
68         return GC_MALLOC_UNCOLLECTABLE(bytelength);
69 }
70
71
72 static void *stackcall_realloc(void *p, u4 bytelength)
73 {
74         return GC_REALLOC(p, bytelength);
75 }
76
77 static void *stackcall_free(void *p, u4 bytelength)
78 {
79         GC_FREE(p);
80         return NULL;
81 }
82
83
84 #if defined(USE_THREADS) && !defined(NATIVE_THREADS)
85 #define MAINTHREADCALL(r,m,pp,ll) \
86         if (currentThread == NULL || currentThread == mainThread) { \
87                 r = m(pp, ll); \
88         } else { \
89                 struct otherstackcall sc; \
90                 sc.p2 = m; \
91                 sc.p = pp; \
92                 sc.l = ll; \
93                 r = (*asm_switchstackandcall)(CONTEXT(mainThread).usedStackTop, \
94                                 stackcall_twoargs, \
95                                 (void**)&(CONTEXT(currentThread).usedStackTop), &sc); \
96         }
97 #else
98 #define MAINTHREADCALL(r,m,pp,ll) \
99         { r = m(pp, ll); }
100 #endif
101
102
103 void *heap_alloc_uncollectable(u4 bytelength)
104 {
105         void *result;
106         MAINTHREADCALL(result, stackcall_malloc_uncollectable, NULL, bytelength);
107         return result;
108 }
109
110
111 void runboehmfinalizer(void *o, void *p)
112 {
113         java_objectheader *ob = (java_objectheader *) o;
114         asm_calljavafunction(ob->vftbl->class->finalizer, ob, NULL, NULL, NULL);
115         
116         /* if we had an exception in the finalizer, ignore it */
117         *exceptionptr = NULL;
118 }
119
120
121 void *heap_allocate(u4 bytelength, bool references, methodinfo *finalizer)
122 {
123         void *result;
124
125         if (references) {
126                 MAINTHREADCALL(result, stackcall_malloc, NULL, bytelength);
127
128         } else {
129                 MAINTHREADCALL(result, stackcall_malloc_atomic, NULL, bytelength);
130         }
131
132         if (!result) {
133                 log_text("java_lang_OutOfMemoryError");
134                 *exceptionptr = new_exception(string_java_lang_OutOfMemoryError);
135                 return NULL;
136         }
137
138         if (finalizer)
139                 GC_REGISTER_FINALIZER(result, runboehmfinalizer, 0, 0, 0);
140
141         return (u1*) result;
142 }
143
144
145 void *heap_reallocate(void *p, u4 bytelength)
146 {
147         void *result;
148
149         MAINTHREADCALL(result, stackcall_realloc, p, bytelength);
150
151         return result;
152 }
153
154 void heap_free(void *p)
155 {
156         void *result;
157
158         MAINTHREADCALL(result, stackcall_free, p, 0);
159 }
160
161
162 void heap_init(u4 size, u4 startsize, void **stackbottom)
163 {
164         GC_INIT();
165 }
166
167
168 void heap_close()
169 {
170 }
171
172
173 void gc_init()
174 {
175         GC_init();
176 }
177
178
179 void gc_call()
180 {
181         if (collectverbose)
182                 dolog("Garbage Collection:  previous/now = %d / %d ",
183                           0, 0);
184
185         GC_gcollect();
186 }
187
188
189 s8 gc_get_heap_size()
190 {
191         return GC_get_heap_size();
192 }
193
194
195 s8 gc_get_free_bytes()
196 {
197         return GC_get_free_bytes();
198 }
199
200
201 void gc_finalize_all()
202 {
203         GC_finalize_all();
204 }
205
206
207 /*
208  * These are local overrides for various environment variables in Emacs.
209  * Please do not remove this and leave it at the end of the file, where
210  * Emacs will automagically detect them.
211  * ---------------------------------------------------------------------
212  * Local variables:
213  * mode: c
214  * indent-tabs-mode: t
215  * c-basic-offset: 4
216  * tab-width: 4
217  * End:
218  */