* Merged with default branch at rev 16f3633aaa5a.
[cacao.git] / src / vm / jit / code.c
1 /* src/vm/jit/code.c - codeinfo struct for representing compiled code
2
3    Copyright (C) 1996-2005, 2006, 2007 R. Grafl, A. Krall, C. Kruegel,
4    C. Oates, R. Obermaisser, M. Platter, M. Probst, S. Ring,
5    E. Steiner, C. Thalinger, D. Thuernbeck, P. Tomsich, C. Ullrich,
6    J. Wenninger, Institut f. Computersprachen - TU Wien
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., 51 Franklin Street, Fifth Floor, Boston, MA
23    02110-1301, USA.
24
25 */
26
27
28 #include "config.h"
29
30 #include <assert.h>
31
32 #include "vm/types.h"
33
34 #include "arch.h"
35
36 #include "mm/memory.h"
37
38 #if defined(ENABLE_THREADS)
39 # include "threads/native/lock.h"
40 #endif
41
42 #include "vm/jit/code.h"
43 #include "vm/jit/codegen-common.h"
44 #include "vm/jit/methodheader.h"
45 #include "vm/jit/patcher-common.h"
46
47 #include "vmcore/options.h"
48
49
50 /* code_init *******************************************************************
51
52    Initialize the code-subsystem.
53
54 *******************************************************************************/
55
56 bool code_init(void)
57 {
58         /* check for offset of code->m == 0 (see comment in code.h) */
59
60         assert(OFFSET(codeinfo, m) == 0);
61
62         /* everything's ok */
63
64         return true;
65 }
66
67
68 /* code_codeinfo_new ***********************************************************
69
70    Create a new codeinfo for the given method.
71    
72    IN:
73        m................method to create a new codeinfo for
74
75    The following fields are set in codeinfo:
76        m
77        patchers
78
79    RETURN VALUE:
80        a new, initialized codeinfo, or
81            NULL if an exception occurred.
82   
83 *******************************************************************************/
84
85 codeinfo *code_codeinfo_new(methodinfo *m)
86 {
87         codeinfo *code;
88
89         code = NEW(codeinfo);
90
91         code->m = m;
92
93         patcher_list_create(code);
94
95 #if defined(ENABLE_STATISTICS)
96         if (opt_stat)
97                 size_codeinfo += sizeof(codeinfo);
98 #endif
99
100         return code;
101 }
102
103
104 /* code_find_codeinfo_for_pc ***************************************************
105
106    Return the codeinfo for the compilation unit that contains the
107    given PC.
108
109    IN:
110        pc...............machine code position
111
112    RETURN VALUE:
113        the codeinfo * for the given PC
114
115 *******************************************************************************/
116
117 codeinfo *code_find_codeinfo_for_pc(u1 *pc)
118 {
119         u1 *pv;
120
121         pv = codegen_get_pv_from_pc(pc);
122         assert(pv);
123
124         return *(codeinfo **)(pv + CodeinfoPointer);
125 }
126
127
128 /* code_find_codeinfo_for_pc ***************************************************
129
130    Return the codeinfo for the compilation unit that contains the
131    given PC. This method does not check the return value and is used
132    by the GC.
133
134    IN:
135        pc...............machine code position
136
137    RETURN VALUE:
138        the codeinfo * for the given PC, or NULL
139
140 *******************************************************************************/
141
142 codeinfo *code_find_codeinfo_for_pc_nocheck(u1 *pc)
143 {
144         codeinfo *code;
145         u1 *pv;
146
147         pv = codegen_get_pv_from_pc_nocheck(pc);
148
149         if (pv == NULL)
150                 return NULL;
151
152         code = *(codeinfo **)(pv + CodeinfoPointer);
153
154         return code;
155 }
156
157
158 /* code_get_methodinfo_for_pv **************************************************
159
160    Return the methodinfo for the given PV.
161
162    IN:
163        pv...............PV
164
165    RETURN VALUE:
166        the methodinfo *
167
168 *******************************************************************************/
169
170 methodinfo *code_get_methodinfo_for_pv(u1 *pv)
171 {
172         codeinfo *code;
173
174         code = *((codeinfo **) (pv + CodeinfoPointer));
175
176         return code->m;
177 }
178
179
180 /* code_get_sync_slot_count ****************************************************
181
182    Return the number of stack slots used for storing the synchronized object
183    (and the return value around lock_monitor_exit calls) by the given code.
184    
185    IN:
186        code.............the codeinfo of the code in question
187                             (must be != NULL)
188
189    RETURN VALUE:
190        the number of stack slots used for synchronization
191   
192 *******************************************************************************/
193
194 #if defined(ENABLE_REPLACEMENT)
195 int code_get_sync_slot_count(codeinfo *code)
196 {
197 #ifdef ENABLE_THREADS
198         int count;
199         
200         assert(code);
201
202         if (!checksync)
203                 return 0;
204
205         if (!(code->m->flags & ACC_SYNCHRONIZED))
206                 return 0;
207
208         count = 1;
209
210 #ifdef HAS_4BYTE_STACKSLOT
211         /* long and double need 2 4-byte slots */
212         if (IS_2_WORD_TYPE(code->m->parseddesc->returntype.type))
213                 count++;
214 #endif
215
216 #if defined(__POWERPC__)
217         /* powerpc needs an extra slot */
218         count++;
219 #endif
220
221         return count;
222
223 #else /* !ENABLE_THREADS */
224         
225         return 0;
226
227 #endif /* ENABLE_THREADS */
228 }
229 #endif /* defined(ENABLE_REPLACEMENT) */
230
231
232 /* code_codeinfo_free **********************************************************
233
234    Free the memory used by a codeinfo.
235    
236    IN:
237        code.............the codeinfo to free
238
239 *******************************************************************************/
240
241 void code_codeinfo_free(codeinfo *code)
242 {
243         if (code == NULL)
244                 return;
245
246         if (code->mcode != NULL)
247                 CFREE((void *) (ptrint) code->mcode, code->mcodelength);
248
249         patcher_list_free(code);
250
251 #if defined(ENABLE_REPLACEMENT)
252         replace_free_replacement_points(code);
253 #endif
254
255         FREE(code, codeinfo);
256
257 #if defined(ENABLE_STATISTICS)
258         if (opt_stat)
259                 size_codeinfo -= sizeof(codeinfo);
260 #endif
261 }
262
263
264 /* code_free_code_of_method ****************************************************
265
266    Free all codeinfos of the given method
267    
268    IN:
269        m................the method of which the codeinfos are to be freed
270
271 *******************************************************************************/
272
273 void code_free_code_of_method(methodinfo *m)
274 {
275         codeinfo *nextcode;
276         codeinfo *code;
277
278         if (!m)
279                 return;
280         
281         nextcode = m->code;
282         while (nextcode) {
283                 code = nextcode;
284                 nextcode = code->prev;
285                 code_codeinfo_free(code);
286         }
287
288         m->code = NULL;
289 }
290
291 /*
292  * These are local overrides for various environment variables in Emacs.
293  * Please do not remove this and leave it at the end of the file, where
294  * Emacs will automagically detect them.
295  * ---------------------------------------------------------------------
296  * Local variables:
297  * mode: c
298  * indent-tabs-mode: t
299  * c-basic-offset: 4
300  * tab-width: 4
301  * End:
302  * vim:noexpandtab:sw=4:ts=4:
303  */