* src/vm/exceptions.h (exceptionptr): Removed.
[cacao.git] / src / vm / jit / reg.c
1 /* src/vm/jit/reg.c - register allocator setup
2
3    Copyright (C) 1996-2005, 2006 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    Contact: cacao@cacaojvm.org
26
27    Authors: Andreas Krall
28
29    Changes: Stefan Ring
30             Christian Thalinger
31             Christian Ullrich
32             Michael Starzinger
33             Edwin Steiner
34
35    $Id: reg.c 7596 2007-03-28 21:05:53Z twisti $
36
37 */
38
39
40 #include "config.h"
41
42 #include <assert.h>
43
44 #include "vm/types.h"
45
46 #include "arch.h"
47 #include "md-abi.h"
48
49 #include "mm/memory.h"
50 #include "vm/jit/abi.h"
51 #include "vm/jit/reg.h"
52
53
54 /* reg_setup *******************************************************************
55
56    TODO
57
58 *******************************************************************************/
59
60 void reg_setup(jitdata *jd)
61 {
62         methodinfo   *m;
63         registerdata *rd;
64         s4            i;
65
66         /* get required compiler data */
67
68         m  = jd->m;
69         rd = jd->rd;
70
71         /* setup the integer register table */
72
73 #if defined(__ARM__)
74         /* On ARM longs can be split across argument regs and stack. This is
75          * signed by setting the HIGH_REG to INT_ARG_CNT in md_param_alloc().
76          * Here we make sure it resolves to a special dummy reg (REG_SPLIT). */
77         rd->argintregs = DMNEW(s4, INT_ARG_CNT + 1);
78         rd->argintregs[INT_ARG_CNT] = REG_SPLIT;
79 #else
80         rd->argintregs = DMNEW(s4, INT_ARG_CNT);
81 #endif
82         rd->tmpintregs = DMNEW(s4, INT_TMP_CNT);
83         rd->savintregs = DMNEW(s4, INT_SAV_CNT);
84         rd->freeargintregs = DMNEW(s4, INT_ARG_CNT);
85         rd->freetmpintregs = DMNEW(s4, INT_TMP_CNT);
86         rd->freesavintregs = DMNEW(s4, INT_SAV_CNT);
87
88         rd->argintreguse = 0;
89         rd->tmpintreguse = 0;
90         rd->savintreguse = 0;
91
92         for (i = 0; i < INT_REG_CNT; i++) {
93                 switch (nregdescint[i]) {
94                 case REG_RET:
95                         rd->intreg_ret = i; 
96                         break;
97                 case REG_SAV:
98                         rd->savintregs[rd->savintreguse++] = i;
99                         break;
100                 case REG_TMP:
101                         rd->tmpintregs[rd->tmpintreguse++] = i; 
102                         break;
103                 case REG_ARG:
104                         rd->argintregs[rd->argintreguse++] = i;
105                         break;
106                 }
107         }
108         assert(rd->savintreguse == INT_SAV_CNT);
109         assert(rd->tmpintreguse == INT_TMP_CNT);
110         assert(rd->argintreguse == INT_ARG_CNT);
111
112 #if defined(__X86_64__)
113         /* 
114          * on x86_64 the argument registers are not in ascending order 
115          * a00 (%rdi) <-> a03 (%rcx) and a01 (%rsi) <-> a02 (%rdx)
116          */
117         i = rd->argintregs[3];
118         rd->argintregs[3] = rd->argintregs[0];
119         rd->argintregs[0] = i;
120
121         i = rd->argintregs[2];
122         rd->argintregs[2] = rd->argintregs[1];
123         rd->argintregs[1] = i;
124 #endif
125                 
126 #ifdef HAS_ADDRESS_REGISTER_FILE
127         /* setup the address register table */
128
129         rd->argadrregs = DMNEW(s4, ADR_ARG_CNT);
130         rd->tmpadrregs = DMNEW(s4, ADR_TMP_CNT);
131         rd->savadrregs = DMNEW(s4, ADR_SAV_CNT);
132         rd->freeargadrregs = DMNEW(s4, ADR_ARG_CNT);
133         rd->freetmpadrregs = DMNEW(s4, ADR_TMP_CNT);
134         rd->freesavadrregs = DMNEW(s4, ADR_SAV_CNT);
135
136         /*rd->adrreg_argnum = 0; XXX ask twisti */
137         rd->argadrreguse = 0;
138         rd->tmpadrreguse = 0;
139         rd->savadrreguse = 0;
140
141         for (i = 0; i < ADR_REG_CNT; i++) {
142                 switch (nregdescadr[i]) {
143                 case REG_RET:
144                         rd->adrreg_ret = i; 
145                         break;
146                 case REG_SAV:
147                         rd->savadrregs[rd->savadrreguse++] = i;
148                         break;
149                 case REG_TMP:
150                         rd->tmpadrregs[rd->tmpadrreguse++] = i; 
151                         break;
152                 case REG_ARG:
153                         rd->argadrregs[rd->argadrreguse++] = i;
154                         break;
155                 }
156         }
157         assert(rd->savadrreguse == ADR_SAV_CNT);
158         assert(rd->tmpadrreguse == ADR_TMP_CNT);
159         assert(rd->argadrreguse == ADR_ARG_CNT);
160 #endif
161                 
162         /* setup the float register table */
163
164         rd->argfltregs = DMNEW(s4, FLT_ARG_CNT);
165         rd->tmpfltregs = DMNEW(s4, FLT_TMP_CNT);
166         rd->savfltregs = DMNEW(s4, FLT_SAV_CNT);
167         rd->freeargfltregs = DMNEW(s4, FLT_ARG_CNT);
168         rd->freetmpfltregs = DMNEW(s4, FLT_TMP_CNT);
169         rd->freesavfltregs = DMNEW(s4, FLT_SAV_CNT);
170
171         rd->argfltreguse = 0;
172         rd->tmpfltreguse = 0;
173         rd->savfltreguse = 0;
174
175         for (i = 0; i < FLT_REG_CNT; i++) {
176                 switch (nregdescfloat[i]) {
177                 case REG_RET:
178                         rd->fltreg_ret = i;
179                         break;
180                 case REG_SAV:
181                         rd->savfltregs[rd->savfltreguse++] = i;
182                         break;
183                 case REG_TMP:
184                         rd->tmpfltregs[rd->tmpfltreguse++] = i;
185                         break;
186                 case REG_ARG:
187                         rd->argfltregs[rd->argfltreguse++] = i;
188                         break;
189                 }
190         }
191         assert(rd->savfltreguse == FLT_SAV_CNT);
192         assert(rd->tmpfltreguse == FLT_TMP_CNT);
193         assert(rd->argfltreguse == FLT_ARG_CNT);
194
195
196         rd->freemem    = DMNEW(s4, m->maxstack);
197 #if defined(HAS_4BYTE_STACKSLOT)
198         rd->freemem_2  = DMNEW(s4, m->maxstack);
199 #endif
200
201 #if defined(SPECIALMEMUSE)
202 # if defined(__DARWIN__)
203         /* 6*4=24 byte linkage area + 8*4=32 byte minimum parameter Area */
204         rd->memuse = LA_SIZE_IN_POINTERS + INT_ARG_CNT; 
205 # else
206         rd->memuse = LA_SIZE_IN_POINTERS;
207 # endif
208 #else
209         rd->memuse = 0; /* init to zero -> analyse_stack will set it to a higher  */
210                         /* value, if appropriate */
211 #endif
212
213         /* Set rd->arg*reguse to *_ARG_CNBT to not use unused argument            */
214         /* registers as temp registers  */
215 #if defined(HAS_ADDRESS_REGISTER_FILE)
216         rd->argadrreguse = 0;
217 #endif /* defined(HAS_ADDRESS_REGISTER_FILE) */
218         rd->argintreguse = 0;
219         rd->argfltreguse = 0;
220 }
221
222
223 /*
224  * These are local overrides for various environment variables in Emacs.
225  * Please do not remove this and leave it at the end of the file, where
226  * Emacs will automagically detect them.
227  * ---------------------------------------------------------------------
228  * Local variables:
229  * mode: c
230  * indent-tabs-mode: t
231  * c-basic-offset: 4
232  * tab-width: 4
233  * End:
234  * vim:noexpandtab:sw=4:ts=4:
235  */