Mercurial > hg > CbC > old > device
annotate mc-nop-386.c @ 58:727c280bdd25
*** empty log message ***
author | kono |
---|---|
date | Wed, 19 Feb 2003 18:48:37 +0900 |
parents | 3d7f199e99d0 |
children | 8ffb8ca3fe34 |
rev | line source |
---|---|
0 | 1 /* Micro-C Code Generatation Part for intel386 */ |
1 | 2 /* $Id$ */ |
0 | 3 |
4 #define EXTERN extern | |
5 #include "mc.h" | |
6 | |
18 | 7 extern int get_register_var(void); |
8 extern void bexpr(int e1, char cond, int l1); | |
9 extern void closing(void); | |
10 extern void cmpdimm(int e, int csreg); | |
11 extern void code_enter(char *name) ; | |
50 | 12 extern void code_enter1(int args); |
18 | 13 extern void code_init(void); |
50 | 14 extern void code_leave(char *name) ; |
15 extern void leave(int control,char *name) ; | |
18 | 16 extern void def_label(int cslabel, int dlabel) ; |
17 extern void emit_data(int e, int t, NMTBL *n); | |
18 extern void emit_init(void); | |
19 extern void enter(char *name) ; | |
50 | 20 extern void enter1() ; |
18 | 21 extern void g_expr(int e1); |
22 extern void gen_comment(char *s); | |
23 extern void gen_gdecl(char *n, int gpc) ; | |
24 extern void gen_source(char *s); | |
25 extern void gexpr(int e1); | |
26 extern void jcond(int l, char cond); | |
27 extern void jmp(int l); | |
28 extern void opening(char *filename); | |
29 extern void ret(void); | |
49 | 30 extern int assign_expr0(int e,int e1,int type,int type1); |
31 extern int integral(int t); | |
50 | 32 extern void tree_print_t(int e,int t); |
33 extern int rvalue_t(int e,int t); | |
0 | 34 |
42 | 35 static int edx_setup() ; |
18 | 36 static int lvar(int l); |
37 static void jump(int e1, int env); | |
38 static void data_mode(char *name); | |
39 static void text_mode(void); | |
40 static void assign(int e1); | |
41 static void assop(int e1); | |
55 | 42 static void b_expr(int e1, char cond, int l1,int err); |
18 | 43 static void emit_push(void); |
44 static void free_register(int i); | |
45 static void function(int e1); | |
46 static void global_table(void) ; | |
47 /* static void indexy(char *op, char *gvar); */ | |
48 /* static void jmp_eq_label(int l); */ | |
49 /* static void jmp_label(int l); */ | |
50 static void ld_indexx(int byte, int n, int xreg); | |
51 static void local_table(void) ; | |
52 static void machinop(int e1); | |
53 static void rexpr(int e1, int l1, char *s); | |
54 static void rindirect(int e1); | |
55 static void sassign(int e1); | |
56 static void shift(char *op, int xreg) ; | |
57 /* static void st_indexx(int byte, int n, int xreg); */ | |
58 static void string(int e1); | |
43 | 59 static void tosop(int op,int reg); |
42 | 60 static void edx_cleanup(); |
26 | 61 static void use_register(int virt, int real, int move); |
57 | 62 static void emit_copy(int from,int to,int length,int offset,int value,int det); |
51 | 63 static int is_same_type(int s,int t); |
18 | 64 |
65 extern int error(int n); | |
66 extern int fwdlabel(void); | |
67 extern void fwddef(int l); | |
68 extern int backdef(void); | |
69 extern int size(int t); | |
49 | 70 extern int list4(int e1, int e2, int e3,int e4); |
18 | 71 extern int list3(int e1, int e2, int e3); |
49 | 72 extern int list2(int e1, int e2); |
2 | 73 extern int scalar(int); |
74 extern int reverse0(int); | |
49 | 75 extern void remove0(int *list,int element); |
55 | 76 extern int append4(int target,int t,int ty,int e1); |
0 | 77 |
78 #define TEXT_EMIT_MODE 0 | |
79 #define DATA_EMIT_MODE 1 | |
80 #define RODATA_EMIT_MODE 2 | |
81 | |
18 | 82 static int output_mode = TEXT_EMIT_MODE; |
83 static int data_alignment = 0; | |
0 | 84 |
36 | 85 static int code_disp_label; |
86 static int func_disp_label; | |
87 | |
0 | 88 /* |
53 | 89 -16 -8 local2 |
90 -12 -4 local1 | |
91 -8 8 arg3 | |
92 -4 4 arg2 | |
93 0 0 arg1 | |
94 local2 -20 4 0 (%edi) | |
95 local1 <-- -16 0 local variable 0 (%esi) | |
50 | 96 %edi -12 <- disp_offset %ebp |
97 %esi -8 | |
0 | 98 %ebx -4 |
99 %ebp = %esp 0 | |
100 %eip 4 <- arg_offset | |
53 | 101 arg1 8 0 |
102 arg2 12 4 | |
50 | 103 see enter/enter1/leave see code_enter |
0 | 104 */ |
105 int arg_offset = 8; | |
3 | 106 int disp_offset = -12; |
107 int func_disp_offset = -12; | |
53 | 108 int code_disp_offset = 0; |
109 int jump_offset = 0; | |
110 | |
3 | 111 int size_of_int = 4; |
0 | 112 int endian = 0; |
113 int MAX_REGISTER=6; /* intel386¤Î¥ì¥¸¥¹¥¿¤ò4¤Ä¤Þ¤Ç»È¤¦*/ | |
114 int REAL_MAX_REGISTER=8; /* intel386¤Î¥ì¥¸¥¹¥¿¤¬8¤Ä¤È¤¤¤¦¤³¤È*/ | |
115 int MAX_DATA_REG=4; | |
116 int MAX_POINTER=3; | |
117 int MAX_REGISTGER_VAR=2; | |
118 | |
119 static int creg; /* current register */ | |
120 static int dreg; /* temporary register */ | |
121 static int reg_sp; /* REGister Stack-Pointer */ | |
122 | |
123 | |
124 #define REG_EAX 0 | |
125 #define REG_EBX 1 | |
126 #define REG_ECX 2 | |
127 #define REG_EDX 3 | |
128 #define REG_ESI 4 | |
129 #define REG_EDI 5 | |
130 #define REG_EBP 6 | |
131 #define REG_ESP 7 | |
132 | |
133 | |
134 #define DATA_REG 0 | |
135 #define POINTER_REG 3 | |
136 static char *reg_name[8]; | |
137 static char *reg_name_l[4]; | |
13 | 138 static char *reg_name_w[4]; |
0 | 139 |
140 /* | |
141 creg currrent virtual register | |
142 dreg spare virtual register | |
143 | |
144 rname[creg] currrent real register | |
145 rname[dreg] spare real register | |
146 | |
147 regs[] virtual register usage | |
40 | 148 regv[] value in virtual register flag |
0 | 149 |
150 reg_name[rname[creg]] | |
151 */ | |
152 | |
153 #define MAX_MAX 10 | |
154 static int rname[MAX_MAX]; | |
155 static int regs[MAX_MAX]; /* »È¤ï¤ì¤Æ¤¤¤ë¥ì¥¸¥¹¥¿¤ò¼¨¤¹¥Õ¥é¥° */ | |
156 static int reg_stack[MAX_MAX]; /* ¼ÂºÝ¤Î¥ì¥¸¥¹¥¿¤ÎÎΰè */ | |
40 | 157 static int regv[MAX_MAX]; /* Ãͤ¬Æþ¤Ã¤Æ¤¤¤ë¤«¤É¤¦¤« */ |
0 | 158 |
159 void | |
160 code_init(void) | |
161 { | |
24 | 162 arg_offset = 8; |
163 func_disp_offset = -12; | |
164 disp_offset = -12; | |
165 size_of_int = 4; | |
166 endian = 0; | |
167 MAX_REGISTER=6; | |
168 MAX_DATA_REG=4; | |
169 MAX_POINTER=3; | |
0 | 170 MAX_REGISTER_VAR=2; |
171 | |
172 reg_name[REG_EAX] = "%eax"; | |
173 reg_name[REG_EBX] = "%ebx"; | |
174 reg_name[REG_ECX] = "%ecx"; | |
175 reg_name[REG_EDX] = "%edx"; | |
176 reg_name[REG_ESI] = "%esi"; | |
177 reg_name[REG_EDI] = "%edi"; | |
178 reg_name[REG_EBP] = "%ebp"; | |
179 reg_name[REG_ESP] = "%esp"; | |
180 reg_name_l[REG_EAX] = "%al"; | |
181 reg_name_l[REG_EBX] = "%bl"; | |
182 reg_name_l[REG_ECX] = "%cl"; | |
183 reg_name_l[REG_EDX] = "%dl"; | |
13 | 184 reg_name_w[REG_EAX] = "%ax"; |
185 reg_name_w[REG_EBX] = "%bx"; | |
186 reg_name_w[REG_ECX] = "%cx"; | |
187 reg_name_w[REG_EDX] = "%dx"; | |
0 | 188 |
189 } | |
190 | |
191 char * | |
18 | 192 register_name(int i,int byte) |
0 | 193 { |
38 | 194 if (i<0) { |
195 error(REG_ERR); | |
196 return "%eax"; | |
197 } | |
24 | 198 if (byte && rname[i] <= REG_EDX) { |
18 | 199 return reg_name_l[rname[i]]; |
200 } else { | |
24 | 201 return reg_name[rname[i]]; /* should be error */ |
18 | 202 } |
0 | 203 } |
204 | |
205 int | |
206 get_register(void) | |
207 { /* »È¤ï¤ì¤Æ¤¤¤Ê¤¤¥ì¥¸¥¹¥¿¤òÄ´¤Ù¤ë */ | |
208 int i; | |
209 for(i=0;i<MAX_REGISTER;i++) { | |
210 if (! regs[i]) { /* »È¤ï¤ì¤Æ¤¤¤Ê¤¤¤Ê¤é */ | |
211 regs[i]=1; /* ¤½¤Î¥ì¥¸¥¹¥¿¤ò»È¤¦¤³¤È¤òÀë¸À¤· */ | |
212 return i; /* ¤½¤Î¾ì½ê¤òɽ¤¹ÈÖ¹æ¤òÊÖ¤¹ */ | |
213 } | |
214 } | |
215 return -1; /* ¶õ¤¤¤Æ¤¤¤ë¾ì½ê¤¬¤Ê¤¤¤Ê¤é¡¢¤½¤ì¤òɽ¤¹ -1 ¤òÊÖ¤¹ */ | |
216 } | |
217 | |
218 void | |
219 free_register(int i) { /* ¤¤¤é¤Ê¤¯¤Ê¤Ã¤¿¥ì¥¸¥¹¥¿¤ò³«Êü */ | |
40 | 220 regv[i]=regs[i]=0; |
0 | 221 } |
222 | |
223 int | |
224 register_full(void) | |
225 { | |
226 int i; | |
227 for(i=0;i<MAX_REGISTER;i++) { | |
228 if (! regs[i]) { | |
229 return 0; | |
230 } | |
231 } | |
232 return 1; | |
233 } | |
234 | |
23 | 235 static |
236 int | |
237 free_register_count(void) | |
238 { | |
239 int i,count; | |
240 count = 0; | |
241 for(i=0;i<MAX_REGISTER;i++) { | |
42 | 242 if (! regs[i] && ! regv[i]) count++; |
23 | 243 } |
244 return count; | |
245 } | |
246 | |
42 | 247 static void |
248 free_all_register(void) | |
249 { | |
250 int i; | |
251 for(i=0;i<MAX_REGISTER;i++) { | |
252 regs[i]=regv[i]=0; | |
253 } | |
254 creg = get_register(); | |
255 dreg = get_register(); | |
256 return; | |
257 } | |
258 | |
40 | 259 void |
260 use_register_var(int i) { | |
261 regv[i]=1; | |
262 } | |
263 | |
264 static int creg_regvar = -1; | |
265 static int creg_regvar_back; | |
266 static int creg_back; | |
267 | |
268 void | |
269 creg_destroy() { | |
270 creg_back = creg; creg_regvar_back = creg_regvar; | |
271 if (creg_regvar>=0) | |
272 creg = creg_regvar; | |
273 creg_regvar=-1; | |
274 } | |
275 | |
276 void | |
277 creg_un_destroy() { | |
278 creg = creg_back; creg_regvar = creg_regvar_back; | |
279 } | |
280 | |
42 | 281 void |
282 register_usage(char *s) | |
283 { | |
284 int i; | |
285 printf("# %d: %s:",lineno,s); | |
286 printf(" creg=%s dreg=%s ",register_name(creg,0),register_name(dreg,0)); | |
287 for(i=0;i<MAX_REGISTER;i++) { | |
288 printf("%d",regs[i]); | |
289 } | |
290 printf(":"); | |
291 for(i=0;i<MAX_REGISTER;i++) { | |
292 printf("%d",regv[i]); | |
293 } | |
294 #if 0 | |
295 printf(" regs_stack",register_name(creg,0),register_name(dreg,0)); | |
296 for(i=reg_sp;i>=0;i--) { | |
297 if(reg_stack[i]>=0) | |
298 printf(" %s",register_name(reg_stack[i],0)); | |
299 } | |
300 #endif | |
301 printf("\n"); | |
302 } | |
23 | 303 |
0 | 304 void |
305 gexpr_init(void) | |
306 { | |
307 while(reg_sp > 0) { | |
308 free_register(reg_stack[--reg_sp]); | |
309 } | |
310 text_mode(); | |
26 | 311 use_register(creg,REG_EAX,0); |
42 | 312 regv[dreg]=0; |
40 | 313 creg_regvar = -1; |
42 | 314 register_usage("gexpr_init"); |
0 | 315 } |
316 | |
42 | 317 |
0 | 318 void |
319 emit_init(void) | |
320 { | |
321 int i; | |
40 | 322 for(i=0;i<REAL_MAX_REGISTER;i++) { regs[i]=0; regv[i]=0;rname[i]=i;} |
42 | 323 free_all_register(); |
0 | 324 reg_sp = 0; |
325 text_mode(); | |
326 } | |
327 | |
328 int | |
329 virtual(int real) | |
330 { | |
331 int real_v,i; | |
332 real_v = -1; | |
333 for(i=0;i<MAX_REGISTER;i++) { | |
334 if (rname[i]==real) { | |
335 real_v=i; | |
336 break; | |
337 } | |
338 } | |
23 | 339 /* if (real_v == -1) |
340 error(-1); */ | |
0 | 341 return real_v; |
342 } | |
343 | |
344 void | |
345 use_register(int virt, int real, int move) | |
346 { | |
347 int real_v; | |
348 char *move_op; | |
349 if (rname[virt]==real) | |
350 return; | |
351 real_v = virtual(real); | |
352 move_op = regs[real_v]?"\txchg %s,%s\n":"\tmovl %s,%s\n"; | |
41 | 353 if (move || (regv[real_v])) { |
0 | 354 printf(move_op,reg_name[rname[virt]],reg_name[real]); |
355 } | |
356 rname[real_v] = rname[virt]; | |
357 rname[virt] = real; | |
358 } | |
359 | |
360 void | |
361 use_pointer(int virt, int move) | |
362 { | |
363 int i; | |
364 if (rname[virt]>=POINTER_REG) | |
365 return; | |
366 for(i=POINTER_REG;i<MAX_REGISTER;i++) { | |
367 if (!regs[virtual(i)]) { | |
368 use_register(virt,i,move); | |
369 return; | |
370 } | |
371 } | |
372 /* we prefer EBX */ | |
373 use_register(virt,REG_EBX,move); | |
374 } | |
375 | |
376 void | |
377 use_data_reg(int virt, int move) | |
378 { | |
379 int i; | |
380 if (rname[virt]<MAX_DATA_REG) | |
381 return; | |
382 for(i=0;i<MAX_DATA_REG;i++) { | |
383 if (!regs[virtual(i)]) { | |
384 use_register(virt,i,move); | |
385 return; | |
386 } | |
387 } | |
388 /* we prefer EBX */ | |
389 use_register(virt,REG_EBX,move); | |
390 } | |
391 | |
392 int | |
393 pop_register(void) | |
394 { /* ¥ì¥¸¥¹¥¿¤«¤éÃͤò¼è¤ê½Ð¤¹ */ | |
42 | 395 return reg_stack[--reg_sp]; |
0 | 396 } |
397 | |
398 int | |
399 stack_used(void) { | |
400 return reg_stack[--reg_sp]<0; | |
401 } | |
402 | |
403 void | |
404 emit_push(void) | |
405 { | |
406 int new_reg; | |
407 new_reg = get_register(); | |
408 if(new_reg<0) { /* ¤â¤¦¥ì¥¸¥¹¥¿¤¬¤Ê¤¤ */ | |
409 reg_stack[reg_sp++] = -1; | |
18 | 410 printf("\tpushl %s\n",register_name(creg,0)); |
42 | 411 /* creg is used soon, don't regv[creg]=0 */ |
0 | 412 } else { |
413 reg_stack[reg_sp++] = creg; /* push ¤¹¤ë¤«¤ï¤ê¤Ë¥ì¥¸¥¹¥¿¤ò»È¤¦ */ | |
18 | 414 creg = new_reg; |
42 | 415 regv[creg]=1; |
0 | 416 } |
42 | 417 } |
418 | |
419 void | |
420 emit_push_x(int xreg) | |
421 { | |
422 int new_reg; | |
423 new_reg = get_register(); | |
424 if(new_reg<0) { /* ¤â¤¦¥ì¥¸¥¹¥¿¤¬¤Ê¤¤ */ | |
425 reg_stack[reg_sp++] = -1; | |
426 printf("\tpushl %s\n",register_name(xreg,0)); | |
427 /* creg is used soon, don't regv[xreg]=0 */ | |
428 } else { | |
429 reg_stack[reg_sp++] = xreg; /* push ¤¹¤ë¤«¤ï¤ê¤Ë¥ì¥¸¥¹¥¿¤ò»È¤¦ */ | |
430 xreg = new_reg; | |
431 regv[xreg]=1; | |
432 } | |
0 | 433 } |
434 | |
18 | 435 int |
0 | 436 emit_pop(int type) |
437 { | |
18 | 438 int xreg; |
42 | 439 if ((xreg=pop_register())==-1) { |
0 | 440 if (type==POINTER_REG) |
441 use_pointer(dreg,0); | |
442 else if (type==DATA_REG) | |
443 use_data_reg(dreg,0); | |
42 | 444 if (regv[dreg]) { |
445 printf("# emit_pop dreg conflict\n"); | |
446 } | |
18 | 447 printf("\tpopl %s\n",register_name(dreg,0)); |
0 | 448 xreg = dreg; |
42 | 449 regv[xreg]=1; |
450 } | |
451 return xreg; | |
452 } | |
453 | |
454 static void | |
455 emit_pop_free(int xreg) | |
456 { | |
457 if (xreg==dreg) { | |
458 regv[dreg]=0; | |
43 | 459 } else if (xreg!=-1) { |
42 | 460 free_register(xreg); |
0 | 461 } |
462 } | |
463 | |
464 int | |
465 get_register_var(void) | |
466 { | |
467 int i; | |
468 for(i=REG_ESI;i<REG_ESP;i++) { | |
469 if (! regs[i]) { /* »È¤ï¤ì¤Æ¤¤¤Ê¤¤¤Ê¤é */ | |
470 regs[i]=1; /* ¤½¤Î¥ì¥¸¥¹¥¿¤ò»È¤¦¤³¤È¤òÀë¸À¤· */ | |
41 | 471 regv[i]=0; |
0 | 472 return i; /* ¤½¤Î¾ì½ê¤òɽ¤¹ÈÖ¹æ¤òÊÖ¤¹ */ |
473 } | |
474 } | |
475 return -1; | |
476 } | |
477 | |
478 void | |
479 gexpr(int e1) | |
480 { | |
481 gexpr_init(); | |
24 | 482 #if 0 |
32 | 483 if(lineno==2862) { |
24 | 484 g_expr(e1); /*break here*/ |
485 return; | |
486 } | |
487 #endif | |
0 | 488 g_expr(e1); |
42 | 489 } |
490 | |
491 int | |
492 csvalue() | |
493 { | |
494 return rname[creg]; /* for siwtch value */ | |
0 | 495 } |
496 | |
497 void | |
498 g_expr(int e1) | |
499 { | |
22 | 500 int e2,e3/*,e4*/; |
0 | 501 char *xrn; |
502 NMTBL *n; | |
503 | |
504 if (chk) return; | |
505 e2 = cadr(e1); | |
506 switch (car(e1)){ | |
507 case GVAR: | |
508 /* use_pointer(creg,0); */ | |
18 | 509 printf("\tmovl $%s,%s\n",(char *)caddr(e1),register_name(creg,0)); |
42 | 510 regv[creg]=1; |
0 | 511 return; |
512 case RGVAR: | |
513 /* use_pointer(creg,0); */ | |
18 | 514 printf("\tmovl %s,%s\n",(char *)caddr(e1),register_name(creg,0)); |
42 | 515 regv[creg]=1; |
0 | 516 return; |
517 case CRGVAR: | |
27 | 518 printf("\tmovsbl %s,%s\n",(char *)caddr(e1),register_name(creg,0)); |
42 | 519 regv[creg]=1; |
0 | 520 return; |
521 case LVAR: | |
522 /* use_pointer(creg,0); */ | |
18 | 523 printf("\tlea %d(%%ebp),%s\n",lvar(e2),register_name(creg,0)); |
42 | 524 regv[creg]=1; |
0 | 525 return; |
526 case REGISTER: | |
527 /* this is of course redundant... */ | |
18 | 528 /* we can use rname for this? */ |
529 /* or why not creg=e2? */ | |
530 printf("\tmovl %s,%s\n",register_name(e2,0),register_name(creg,0)); | |
42 | 531 regv[creg]=1; |
0 | 532 return; |
533 case RLVAR: | |
18 | 534 printf("\tmovl %d(%%ebp),%s\n",lvar(e2),register_name(creg,0)); |
42 | 535 regv[creg]=1; |
0 | 536 return; |
537 case CRLVAR: | |
21 | 538 printf("\tmovsbl %d(%%ebp),%s\n",lvar(e2),register_name(creg,0)); |
42 | 539 regv[creg]=1; |
0 | 540 return; |
541 case FNAME: | |
18 | 542 printf("\tmovl $%s,%s\n",((NMTBL *)e2)->nm,register_name(creg,0)); |
42 | 543 regv[creg]=1; |
0 | 544 return; |
545 case CONST: /* ÂåÆþ¤¹¤ëÃͤ¬0¤Ç¤âÆÃÊ̤ʽèÍý¤Ï¤·¤Ê¤¤ */ | |
18 | 546 printf("\tmovl $%d,%s\n",e2,register_name(creg,0)); |
42 | 547 regv[creg]=1; |
0 | 548 return; |
549 case STRING: | |
550 string(e1); | |
42 | 551 regv[creg]=1; |
0 | 552 return; |
553 case FUNCTION: | |
554 function(e1); | |
42 | 555 regv[creg]=1; |
0 | 556 return; |
557 case CODE: | |
558 jump(e2,caddr(e1)); | |
559 return; | |
560 case INDIRECT: | |
561 g_expr(e2); | |
562 return; | |
563 case RINDIRECT: case CRINDIRECT: | |
564 rindirect(e1); | |
565 return; | |
566 case ADDRESS: | |
567 g_expr(e2); | |
568 return; | |
569 case MINUS: /* ¥ì¥¸¥¹¥¿¤ËÂФ·¡¢negl¤ò¼Â¹Ô¤¹¤ì¤Ð¼Â¸½²Äǽ */ | |
570 g_expr(e2); | |
571 /* use_data_reg(creg,1); */ | |
18 | 572 printf("\tnegl %s\n", register_name(creg,0)); |
0 | 573 return; |
574 case BNOT: /* ~ */ | |
575 g_expr(e2); | |
576 /* use_data_reg(creg,1); */ | |
18 | 577 printf("\tnotl %s\n", register_name(creg,0)); |
0 | 578 return; |
579 case LNOT: /* ! */ | |
580 g_expr(e2); | |
581 use_data_reg(creg,1); | |
18 | 582 xrn = register_name(creg,1); |
583 printf("\tcmpl $0,%s\n", register_name(creg,0)); | |
584 printf("\tsete %s\n", xrn); | |
585 printf("\tmovzbl %s,%s\n", xrn,register_name(creg,0)); | |
0 | 586 return; |
587 case PREINC: | |
588 if (car(e2)==REGISTER) { | |
18 | 589 printf("\taddl $%d,%s\n",caddr(e1),register_name(cadr(e2),0)); |
590 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); | |
0 | 591 return; |
592 } | |
593 g_expr(e2); | |
18 | 594 xrn = register_name(creg,0); |
595 printf("\taddl $%d,(%s)\n",caddr(e1),xrn); | |
596 printf("\tmovl (%s),%s\n",xrn,xrn); | |
0 | 597 return; |
598 case POSTINC: | |
599 if (car(e2)==REGISTER) { | |
18 | 600 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); |
601 printf("\taddl $%d,%s\n",caddr(e1),register_name(cadr(e2),0)); | |
0 | 602 return; |
603 } | |
604 g_expr(e2); | |
605 emit_push(); | |
41 | 606 xrn = register_name((e2=emit_pop(0)),0); |
18 | 607 printf("\tmovl (%s),%s\n",xrn,register_name(creg,0)); |
0 | 608 printf("\taddl $%d,(%s)\n",caddr(e1),xrn); |
42 | 609 emit_pop_free(e2); |
0 | 610 return; |
611 case CPOSTINC: | |
612 /* char *p; *p++ */ | |
613 if (car(e2)==REGISTER) { | |
18 | 614 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); |
615 printf("\taddl $%d,%s\n",caddr(e1),register_name(cadr(e2),0)); | |
0 | 616 return; |
617 } | |
618 g_expr(e2); | |
42 | 619 emit_push(); |
41 | 620 xrn = register_name((e2=emit_pop(0)),1); |
18 | 621 printf("\tmovsbl (%s),%s\n",xrn,register_name(creg,0)); |
0 | 622 printf("\tincl (%s)\n",xrn); |
42 | 623 emit_pop_free(e2); |
0 | 624 return; |
625 case CPREINC: | |
626 if (car(e2)==REGISTER) { | |
18 | 627 printf("\taddl $%d,%s\n",caddr(e1),register_name(cadr(e2),0)); |
628 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); | |
0 | 629 return; |
630 } | |
631 g_expr(e2); | |
18 | 632 printf("\tincl (%s)\n",register_name(creg,0)); |
21 | 633 printf("\tmovsbl (%s),%s\n",register_name(creg,0),register_name(creg,0)); |
0 | 634 return; |
635 case CPOSTDEC: | |
636 if (car(e2)==REGISTER) { | |
18 | 637 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); |
638 printf("\tdecl %s\n",register_name(cadr(e2),0)); | |
0 | 639 return; |
640 } | |
641 g_expr(e2); | |
21 | 642 printf("\tmovsbl (%s),%s\n",register_name(creg,0),register_name(creg,0)); |
18 | 643 printf("\tdecl (%s)\n",register_name(creg,0)); |
0 | 644 return; |
645 case CPREDEC: | |
646 if (car(e2)==REGISTER) { | |
18 | 647 printf("\tdecl %s\n",register_name(cadr(e2),0)); |
648 printf("\tmovl %s,%s\n",register_name(cadr(e2),0),register_name(creg,0)); | |
0 | 649 return; |
650 } | |
651 g_expr(e2); | |
652 emit_push(); | |
18 | 653 e2 = emit_pop(0); |
654 printf("\tdecl (%s)\n",register_name(e2,0)); | |
21 | 655 printf("\tmovsbl (%s),%s\n",register_name(e2,0),register_name(creg,0)); |
42 | 656 emit_pop_free(e2); |
0 | 657 return; |
658 case MUL: case UMUL: | |
659 case DIV: case UDIV: | |
660 case MOD: case UMOD: | |
661 case LSHIFT: case ULSHIFT: case RSHIFT: case URSHIFT: | |
662 case ADD: case SUB: case BAND: case EOR: case BOR: | |
663 machinop(e1); | |
664 return; | |
665 case COND: | |
666 e2=fwdlabel(); | |
55 | 667 b_expr(cadr(e1),0,e2,0); |
22 | 668 use_register(creg,REG_EAX,0); |
0 | 669 g_expr(caddr(e1)); |
22 | 670 /* e4 = rname[creg]; this is a bad idea */ |
671 use_register(creg,REG_EAX,1); | |
0 | 672 jmp(e3=fwdlabel()); |
673 fwddef(e2); | |
22 | 674 use_register(creg,REG_EAX,0); |
0 | 675 g_expr(cadddr(e1)); |
22 | 676 /* use_register(creg,e4,1); */ |
677 use_register(creg,REG_EAX,1); | |
0 | 678 fwddef(e3); |
679 return; | |
680 case SASS: | |
681 sassign(e1); | |
682 return; | |
683 case ASS: case CASS: | |
684 assign(e1); | |
685 return; | |
686 case ASSOP: case CASSOP: | |
687 assop(e1); | |
688 return; | |
57 | 689 case RSTRUCT: |
690 g_expr(e2); | |
691 return; | |
0 | 692 case COMMA: |
693 g_expr(e2); | |
694 g_expr(caddr(e1)); | |
695 return; | |
696 case RETURN: | |
697 n = (NMTBL *)e2; | |
698 if (retcont==0) | |
699 retcont=fwdlabel(); | |
18 | 700 printf("\tleal _%d,%s\n",retcont,register_name(creg,0)); |
42 | 701 regv[creg]=1; |
0 | 702 return; |
703 case ENVIRONMENT: | |
18 | 704 printf("\tmovl %%ebp,%s\n",register_name(creg,0)); |
42 | 705 regv[creg]=1; |
0 | 706 return; |
707 default: | |
55 | 708 b_expr(e1,1,e2=fwdlabel(),1); /* including > < ... */ |
18 | 709 xrn = register_name(creg,0); |
710 printf("\txorl %s,%s\n",xrn,xrn); | |
0 | 711 jmp(e3=fwdlabel()); |
712 fwddef(e2); | |
18 | 713 printf("\tmovl $1,%s\n",xrn); |
0 | 714 fwddef(e3); |
42 | 715 regv[creg]=1; |
0 | 716 } |
717 } | |
718 | |
719 void | |
720 bexpr(int e1, char cond, int l1) | |
721 { | |
722 gexpr_init(); | |
55 | 723 b_expr(e1,cond,l1,0); |
0 | 724 } |
725 | |
726 void | |
55 | 727 b_expr(int e1, char cond, int l1,int err) |
0 | 728 { |
729 int e2,l2; | |
730 if (chk) return; | |
731 e2=cadr(e1); | |
732 switch(car(e1)) { | |
733 case LNOT: | |
55 | 734 b_expr(e2,!cond,l1,0); |
0 | 735 return; |
736 case GT: | |
737 rexpr(e1,l1,cond?"g":"le"); | |
738 return; | |
739 case UGT: | |
740 rexpr(e1,l1,cond?"a":"be"); | |
741 return; | |
742 case GE: | |
743 rexpr(e1,l1,cond?"ge":"l"); | |
744 return; | |
745 case UGE: | |
746 rexpr(e1,l1,cond?"ae":"b"); | |
747 return; | |
748 case LT: | |
749 rexpr(e1,l1,cond?"l":"ge"); | |
750 return; | |
751 case ULT: | |
752 rexpr(e1,l1,cond?"b":"ae"); | |
753 return; | |
754 case LE: | |
755 rexpr(e1,l1,cond?"le":"g"); | |
756 return; | |
757 case ULE: | |
758 rexpr(e1,l1,cond?"be":"a"); | |
759 return; | |
760 case EQ: | |
761 rexpr(e1,l1,cond?"e":"ne"); | |
762 return; | |
763 case NEQ: | |
764 rexpr(e1,l1,cond?"ne":"e"); | |
765 return; | |
766 case LAND: | |
55 | 767 b_expr(e2,0,cond?(l2=fwdlabel()):l1,0); |
768 b_expr(caddr(e1),cond,l1,0); | |
0 | 769 if(cond) fwddef(l2); |
770 return; | |
771 case LOR: | |
55 | 772 b_expr(e2,1,cond?l1:(l2=fwdlabel()),0); |
773 b_expr(caddr(e1),cond,l1,0); | |
0 | 774 if(!cond) fwddef(l2); |
775 return; | |
776 case CRGVAR: | |
18 | 777 printf("\tcmpb $0,%s\n",(char *)caddr(e1)); |
0 | 778 jcond(l1,cond); |
779 return; | |
780 case CRLVAR: | |
781 printf("\tcmpb $0,%d(%%ebp)\n",lvar(e2)); | |
782 jcond(l1,cond); | |
783 return; | |
784 case RGVAR: | |
18 | 785 printf("\tcmpl $0,%s\n",(char *)caddr(e1)); |
0 | 786 jcond(l1,cond); |
787 return; | |
788 case RLVAR: | |
789 printf("\tcmpl $0,%d(%%ebp)\n",lvar(e2)); | |
790 jcond(l1,cond); | |
791 return; | |
792 case REGISTER: | |
18 | 793 printf("\tcmpl $0,%s\n",register_name(e2,0)); |
0 | 794 jcond(l1,cond); |
795 return; | |
796 case CONST: | |
797 if((cond&&e2)||(!cond&&!e2)) jmp(l1); | |
798 return; | |
799 default: | |
55 | 800 if(err) { |
801 error(-1); return; /* recursice g_expr/b_expr */ | |
802 } | |
0 | 803 g_expr(e1); |
18 | 804 printf("\tcmpl $0,%s\n",register_name(creg,0)); |
0 | 805 jcond(l1,cond); |
806 return; | |
807 } | |
808 } | |
809 | |
810 void | |
811 ascii(char *s) | |
812 { | |
813 printf("\t.string \""); | |
814 while(*s) { | |
815 if (*s=='\n') | |
816 printf("%cn",92); | |
817 else if (*s<' ') | |
818 printf("%c%03o",92,*s); | |
819 else if (*s==34) | |
820 printf("%c%c",92,34); | |
821 else | |
822 printf("%c",*s); | |
823 s++; | |
824 } | |
825 printf("%c\n",34); | |
826 } | |
827 | |
828 void | |
829 string(int e1) | |
830 { | |
831 char *s; | |
832 int i,lb; | |
833 | |
834 if (0) { | |
835 s=(char *)cadr(e1); | |
836 lb=fwdlabel(); | |
837 printf("\tjmp _%d\n",lb); | |
838 i=backdef(); | |
839 ascii(s); | |
840 printf("\t.align 2\n"); | |
841 fwddef(lb); | |
18 | 842 printf("\tlea _%d,%s\n",i,register_name(creg,0)); |
0 | 843 } else { |
844 s=(char *)cadr(e1); | |
845 printf(".section\t.rodata\n"); | |
846 lb=fwdlabel(); | |
847 printf("_%d:\n",lb); | |
848 ascii(s); | |
849 if (output_mode==TEXT_EMIT_MODE) { | |
850 printf(".text\n"); | |
851 } else { | |
852 text_mode(); | |
853 } | |
18 | 854 printf("\tlea _%d,%s\n",lb,register_name(creg,0)); |
0 | 855 } |
856 } | |
857 | |
13 | 858 #define MAX_COPY_LEN 20 |
859 | |
860 void | |
57 | 861 emit_copy(int from,int to,int length,int offset,int value,int det) |
13 | 862 { |
39 | 863 int fix = 0; |
57 | 864 /* length <0 means upward direction copy */ |
13 | 865 switch (length) { |
866 case 0: break; | |
57 | 867 case 1: case -1: |
38 | 868 printf("\tmovb %d(%s),%s\n",offset,register_name(from,0), reg_name_l[rname[dreg]] ); |
869 printf("\tmovb %s,%d(%s)\n",reg_name_l[rname[dreg]] ,offset,register_name(to,0)); | |
13 | 870 break; |
57 | 871 case 2: case -2: |
38 | 872 printf("\tmovw %d(%s),%s\n",offset,register_name(from,0), reg_name_w[rname[dreg]] ); |
873 printf("\tmovw %s,%d(%s)\n",reg_name_w[rname[dreg]] ,offset,register_name(to,0)); | |
13 | 874 break; |
57 | 875 case 4: case -4: |
38 | 876 printf("\tmovl %d(%s),%s\n",offset,register_name(from,0), register_name(dreg,0)); |
877 printf("\tmovl %s,%d(%s)\n",register_name(dreg,0), offset,register_name(to,0)); | |
13 | 878 break; |
879 default: | |
57 | 880 if (-MAX_COPY_LEN<length && length <0) { |
881 for(;length<=4;length+=4,offset-=4) | |
882 emit_copy(from,to,4,offset,0,det); | |
883 for(;length<=2;length+=2,offset-=2) | |
884 emit_copy(from,to,2,offset,0,det); | |
885 if(length>0) | |
886 emit_copy(from,to,length,offset,0,det); | |
887 break; | |
888 } else if (length <=MAX_COPY_LEN) { | |
14 | 889 for(;length>=4;length-=4,offset+=4) |
57 | 890 emit_copy(from,to,4,offset,0,det); |
14 | 891 for(;length>=2;length-=2,offset+=2) |
57 | 892 emit_copy(from,to,2,offset,0,det); |
13 | 893 if(length>0) |
57 | 894 emit_copy(from,to,length,offset,0,det); |
39 | 895 break; |
13 | 896 } |
58 | 897 if (det) { |
898 /* | |
899 call bcopy | |
900 g_expr(list3(FUNCTION,,); | |
901 break; | |
902 */ | |
903 } | |
39 | 904 use_register(from,REG_ESI,1); |
905 use_register(to, REG_EDI,1); | |
906 use_register(dreg,REG_ECX,0); | |
57 | 907 if (length<0) { |
908 printf("\tmovl $%d,%%ecx\n",-length/4); | |
909 printf("\taddl $%d,%%esi\n",-length); | |
910 printf("\taddl $%d,%%edi\n",-length); | |
911 printf("\tstd\n\trep\n\tmovsl\n"); | |
912 if(length%4) { | |
913 emit_copy(from,to,length,offset+length/4,0,det); | |
914 } | |
915 } else { | |
916 printf("\tmovl $%d,%%ecx\n",length/4); | |
917 fix = (length/4)*4; | |
918 printf("\tcld\n\trep\n\tmovsl\n"); | |
919 if(length%4) { | |
920 emit_copy(from,to,length,offset+length/4,0,det); | |
921 } | |
14 | 922 } |
13 | 923 } |
47 | 924 if (value) { |
57 | 925 /* creg must point top of the destination data */ |
39 | 926 /* this code is necessary for the value of assignment or function call */ |
927 /* otherwise we don't need this */ | |
47 | 928 if (fix) printf("\tsubl $%d,%s\n",fix,register_name(to,0)); |
929 if(creg!=to) { | |
930 if (to==dreg) | |
931 printf("\tmovl %s,%s\n",register_name(to,0),register_name(creg,0)); | |
932 else { | |
933 free_register(creg); creg=to; | |
934 } | |
45 | 935 } |
39 | 936 } |
41 | 937 regv[from]=regv[to]=regv[dreg]=0; |
42 | 938 regv[creg]=1; |
13 | 939 } |
940 | |
941 int | |
942 struct_push(int e4,int t) | |
943 { | |
42 | 944 int length,xreg,save,lreg; |
13 | 945 g_expr(e4); |
14 | 946 length=size(t); |
947 if(length%size_of_int) { | |
948 length += size_of_int - (length%size_of_int); | |
13 | 949 } |
40 | 950 if (length==4) { |
951 printf("\tpushl (%s)\n",register_name(creg,0)); return 1; | |
952 } | |
953 if (length==8) { | |
954 printf("\tpushl 4(%s)\n",register_name(creg,0)); | |
955 printf("\tpushl (%s)\n",register_name(creg,0)); return 2; | |
956 } | |
38 | 957 /* I think, after g_expr, at least two registers are free */ |
14 | 958 printf("\tsubl $%d,%%esp\n",length); |
38 | 959 if (register_full()) { |
960 /* this is wrong assumption */ | |
961 save = 1; | |
46
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
962 for(lreg=0;lreg==creg||lreg==dreg;lreg++); |
38 | 963 printf("\tpushl %s\n",register_name(lreg,0)); |
46
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
964 xreg = lreg; regv[xreg]=0; |
38 | 965 } else { |
966 save=0; | |
967 xreg = get_register(); | |
13 | 968 } |
46
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
969 if (save) |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
970 printf("\tlea %d(%%esp),%s\n",size_of_int,register_name(xreg,0)); |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
971 else |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
972 printf("\tmovl %%esp,%s\n",register_name(xreg,0)); |
41 | 973 regv[xreg]=1; |
57 | 974 /* downward direction copy */ |
975 emit_copy(creg,xreg,length,0,0,1); | |
46
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
976 /* we have value in creg, it may be changed */ |
38 | 977 if (save) { |
46
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
978 if(creg==xreg) { |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
979 creg = get_register(); /* creg is freed in emit_copy */ |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
980 } |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
981 printf("\tpopl %s\n",register_name(xreg,0)); |
b1c8ac8c308d
fix cascading struct retrun. Now it should be compatible with gcc
kono
parents:
45
diff
changeset
|
982 regv[xreg]=1; |
38 | 983 } else |
984 free_register(xreg); | |
14 | 985 return length/size_of_int; |
13 | 986 } |
987 | |
0 | 988 void |
989 function(int e1) | |
990 { | |
13 | 991 int e2,e3,e4,e5,nargs,t; |
0 | 992 NMTBL *n; |
23 | 993 int save,saved; |
994 if (free_register_count()<1) { | |
995 for(save = 0;save==dreg||save==creg;save++); | |
996 printf("\tpushl %s\n",register_name(save,0)); | |
997 saved = 1; | |
998 } else { | |
999 save = get_register(); | |
1000 saved = 0; | |
1001 } | |
41 | 1002 regv[save]=0; |
0 | 1003 e2 = cadr(e1); |
1004 nargs = 0; | |
1005 for (e3 = caddr(e1); e3; e3 = cadr(e3)) { | |
13 | 1006 t=caddr(e3); |
0 | 1007 n=(NMTBL *)(e5=(cadr(e4 = car(e3)))); |
58 | 1008 if(scalar(t)) { |
1009 g_expr(e4); | |
18 | 1010 printf("\tpushl %s\n",register_name(creg,0)); |
58 | 1011 } else if (car(t)==STRUCT||car(t)==UNION) { |
1012 nargs += struct_push(e4,t); | |
1013 continue; | |
1014 } else { | |
1015 error(TYERR); | |
0 | 1016 } |
1017 ++nargs; | |
1018 } | |
1019 if (car(e2) == FNAME) { | |
1020 n=(NMTBL *)cadr(e2); | |
23 | 1021 } else { |
44 | 1022 use_register(creg,REG_EAX,0); |
23 | 1023 g_expr(e2); |
1024 } | |
1025 | |
26 | 1026 /* we don't have to save creg nor dreg */ |
41 | 1027 regs[creg]=0; regs[dreg]=0; |
1028 regv[creg]= regv[dreg]= regv[save]= 0; | |
23 | 1029 use_register(creg,REG_EAX,0); |
1030 use_register(dreg,REG_EDX,0); /* will be destroyed */ | |
1031 use_register(save,REG_ECX,0); /* will be destroyed */ | |
41 | 1032 regs[creg]=1; regs[dreg]=1; |
23 | 1033 |
1034 if (car(e2) == FNAME) { | |
0 | 1035 printf("\tcall\t%s\n",n->nm); |
23 | 1036 } else { |
18 | 1037 printf("\tcall\t*%s\n",register_name(creg,0)); |
0 | 1038 } |
6 | 1039 if (nargs) printf("\taddl $%d,%%esp\n",size_of_int*nargs); |
23 | 1040 if (saved) { |
1041 printf("\tpopl %s\n",register_name(save,0)); | |
1042 } else { | |
1043 free_register(save); | |
1044 } | |
41 | 1045 regv[save]=0; |
1046 regv[creg]=1; | |
0 | 1047 } |
1048 | |
51 | 1049 /* goto arguments list */ |
52 | 1050 /* target list4(list2(tag,disp),cdr,ty,source_expr) */ |
51 | 1051 /* source expr=listn(tag,...) */ |
1052 /* source (after) list2(tag,disp) */ | |
52 | 1053 /* source list list3(e,cdr,sz) */ |
48 | 1054 |
57 | 1055 #define DEBUG_PARALLEL_ASSIGN 1 |
56 | 1056 |
48 | 1057 int |
52 | 1058 overrap(int t,int sz,int source) |
48 | 1059 { |
52 | 1060 int s,s0,s1; |
1061 int t0=cadr(t); | |
1062 int t1=t0+sz; | |
48 | 1063 for(;source;source=cadr(source)) { |
52 | 1064 s=car(source); s0=cadr(s); |
1065 if(car(s)==REGISTER && car(t)==REGISTER) { | |
56 | 1066 if(s0==t0) return s; |
52 | 1067 } else if (is_same_type(s,t)) { |
1068 s1=s0+caddr(source); | |
56 | 1069 #if DEBUG_PARALLEL_ASSIGN>1 |
51 | 1070 printf("# ovedrrap source %d t0 %d t1 %d\n",car(car(t)),t0,t1); |
1071 printf("# ovedrrap target %d s0 %d s1 %d\n",car(car(source)),s0,s1); | |
54 | 1072 printf("# ovedrrap equal = %d\n",((t0<=s0&&s0<t1)||(t0<s1&&s1<=t1))); |
52 | 1073 #endif |
55 | 1074 if((t0<=s0&&s0<t1)||(t0<s1&&s1<=t1)) return s; |
48 | 1075 } |
1076 } | |
1077 return 0; | |
1078 } | |
1079 | |
51 | 1080 void |
55 | 1081 remove_target(int *target,int t,int *use) |
1082 { | |
1083 int use0=*use; | |
1084 while(use0) { | |
1085 if (car(use0)==t) { | |
1086 free_register(caddr(use0)); | |
1087 break; | |
51 | 1088 } |
55 | 1089 use0 = cadr(use0); |
1090 } | |
1091 remove0(target,t); | |
1092 } | |
1093 | |
1094 void | |
1095 save_target(int t,int s,int *target,int *use,int sz,int ty) | |
1096 { | |
1097 int e1; | |
1098 /*¿·¤·¤¤¥ì¥¸¥¹¥¿(or ¥¹¥¿¥Ã¥¯)¤ò¼èÆÀ¤¹¤ë*/ | |
1099 if (sz==size_of_int && (e1=get_register())!=-1) { | |
1100 *use=list3(t,*use,e1); | |
1101 e1=list2(REGISTER,e1); | |
1102 g_expr(assign_expr0(e1,s,ty,ty)); | |
1103 *target = append4(*target,t,ty,e1); | |
1104 } else { | |
1105 disp-=sz; | |
1106 g_expr(assign_expr0((e1=list2(LVAR,disp)),s,ty,ty)); | |
1107 *target = append4(*target,t,ty,e1); | |
51 | 1108 } |
1109 } | |
1110 | |
56 | 1111 int |
1112 circular_dependency(int t,int s,int *target,int *source) | |
1113 { | |
1114 int target0=*target; | |
1115 int t1,sz,ty,s1; | |
1116 while(target0) { | |
1117 if (cadddr(target0)==s) { | |
1118 t1=car(target0); | |
1119 s=cadddr(target0); | |
1120 sz=size(ty=caddr(target0)); | |
1121 if(t==t1) { | |
1122 #if DEBUG_PARALLEL_ASSIGN | |
1123 printf("# circular dependency %d ty %d+%d sz %d\n",car(t1),ty,cadr(t1),sz); | |
1124 #endif | |
1125 return 1; | |
1126 } | |
1127 if ((s1=overrap(t1,sz,*source))) { | |
1128 /* another overrap start over */ | |
1129 return circular_dependency(t,s1,target,source); | |
1130 } | |
1131 } | |
1132 target0=cadr(target0); | |
1133 } | |
1134 return 0; | |
1135 } | |
1136 | |
55 | 1137 void |
1138 parallel_assign(int *target,int *source,int *processing,int *use) | |
48 | 1139 { |
56 | 1140 int t,s,sz,ty,target0,s1; |
48 | 1141 while(*target) { |
56 | 1142 target0=*target; |
1143 while(target0) { | |
1144 t=car(target0); s=cadddr(target0); | |
1145 sz=size(ty=caddr(target0)); | |
1146 if(car(t)==car(s) && cadr(t)==cadr(s)) { | |
1147 /*½ñ¤¹þ¤ßÀ褬¼«Ê¬¼«¿È*/ | |
1148 #if DEBUG_PARALLEL_ASSIGN | |
1149 printf("# remove same %d ty %d+%d sz %d\n",car(t),ty,cadr(t),sz); | |
52 | 1150 #endif |
56 | 1151 remove_target(target,t,use); |
1152 /* Ç˲õ¤µ¤ì¤Æ¤Ïº¤¤ë¤Î¤Ç¡¢source list¤«¤é¤Ï½ü¤«¤Ê¤¤ */ | |
1153 } else if (!(s1=overrap(t,sz,*source))) { | |
1154 /* ½Å¤Ê¤Ã¤Æ¤Ê¤¤¤Î¤Ç°Â¿´¤·¤Æ½ñ¤¹þ¤á¤ë */ | |
1155 #if DEBUG_PARALLEL_ASSIGN | |
52 | 1156 printf("# normal assign %d ty %d+%d sz %d\n",car(t),ty,cadr(t),sz); |
1157 #endif | |
56 | 1158 g_expr(assign_expr0(t,s,ty,ty)); |
1159 remove_target(target,t,use); remove0(source,s); | |
1160 } else { | |
1161 if(circular_dependency(t,s1,target,source)) { | |
1162 #if DEBUG_PARALLEL_ASSIGN | |
1163 printf("# saving %d ty %d+%d sz %d\n",car(t),ty,cadr(t),sz); | |
55 | 1164 #endif |
56 | 1165 remove_target(target,t,use); remove0(source,s); |
1166 save_target(t,s,target,use,sz,ty); | |
1167 } | |
1168 } | |
1169 target0=cadr(target0); | |
48 | 1170 } |
1171 } | |
1172 } | |
1173 | |
49 | 1174 void |
1175 remove0(int *parent,int e) | |
48 | 1176 { |
1177 int list; | |
49 | 1178 while ((list=*parent)) { |
48 | 1179 if (car(list)==e) { |
51 | 1180 *parent= cadr(list); return; |
1181 } else { | |
1182 parent=&cadr(list); | |
1183 } | |
1184 } | |
1185 } | |
1186 | |
1187 void | |
1188 remove0_all(int *parent,int e) | |
1189 { | |
1190 int list; | |
1191 while ((list=*parent)) { | |
1192 if (car(list)==e) { | |
48 | 1193 *parent= cadr(list); |
1194 } else { | |
1195 parent=&cadr(list); | |
1196 } | |
1197 } | |
1198 } | |
1199 | |
49 | 1200 int |
1201 is_simple(int e1) | |
1202 { | |
1203 return ( | |
1204 e1==CONST || e1==FNAME || e1==LVAR || e1==REGISTER || | |
57 | 1205 e1==GVAR || e1==RGVAR || e1==RLVAR || e1==CRLVAR || e1==CRGVAR |
49 | 1206 ); |
1207 } | |
1208 | |
1209 int | |
51 | 1210 is_same_type(int e1,int e2) |
49 | 1211 { |
51 | 1212 int ce1=car(e1); |
1213 int ce2=car(e2); | |
1214 return ( | |
1215 (ce1==LVAR && (ce2==RLVAR||ce2==CRLVAR)) | |
1216 || (ce2==LVAR && (ce1==RLVAR||ce1==CRLVAR)) | |
1217 || (ce1==GVAR && (ce2==RGVAR||ce2==CRGVAR)) | |
1218 || (ce2==GVAR && (ce1==RGVAR||ce1==CRGVAR)) | |
1219 ); | |
49 | 1220 } |
1221 | |
52 | 1222 int |
1223 is_memory(int e1) | |
1224 { | |
1225 int ce1=car(e1); | |
1226 return ( | |
1227 ce1==LVAR ||ce1==RLVAR||ce1==CRLVAR || | |
1228 ce1==GVAR ||ce1==RGVAR||ce1==CRGVAR || | |
1229 ce1==REGISTER | |
1230 ); | |
1231 } | |
1232 | |
48 | 1233 void |
1234 jump(int e1, int env) | |
1235 { | |
51 | 1236 int e2,e3,e4,sz,arg_size,ty,max_regs,regs; |
1237 int t0,s0; | |
48 | 1238 NMTBL *code0; |
1239 int target = 0; | |
1240 int source = 0; | |
1241 int processing = 0; | |
55 | 1242 int use = 0; |
48 | 1243 |
1244 /* ¤Þ¤º¡¢¥µ¥¤¥º¤ò·×»»¤·¤Ê¤¬¤é¡¢·è¤Þ¤Ã¤¿·Á¤ËÍ¡£ */ | |
51 | 1245 |
50 | 1246 arg_size = 0; regs = 0; max_regs = MAX_REGISTER_VAR-1; |
1247 for (e3 = reverse0(caddr(e1)); e3; e3 = cadr(e3)) { | |
49 | 1248 e2 = car(e3); sz = size(ty=caddr(e3)); |
53 | 1249 if (regs <= max_regs&&scalar(ty)) { |
51 | 1250 target=list4(list2(REGISTER,virtual((regs++)+REG_ESI)), |
1251 target,ty,e2); | |
48 | 1252 } else { |
53 | 1253 target=list4(list2(LVAR,0), |
51 | 1254 target,ty,e2); |
49 | 1255 arg_size += sz; |
48 | 1256 } |
56 | 1257 #if DEBUG_PARALLEL_ASSIGN |
52 | 1258 printf("# target %d ty %d+%d sz %d\n",car(car(target)),ty,cadr(car(target)),sz); |
1259 #endif | |
48 | 1260 } |
1261 | |
1262 /* disp ¤òÈô¤ÓÀè»÷¹ç¤ï¤»¤Æ½¤Àµ */ | |
50 | 1263 if (fnptr->sc==CODE) { |
1264 if (-arg_size<disp) disp = -arg_size; | |
53 | 1265 jump_offset = 0; |
50 | 1266 } else { |
1267 if (disp_offset-arg_size<disp) disp = disp_offset-arg_size; | |
53 | 1268 jump_offset = 0; |
50 | 1269 } |
48 | 1270 |
1271 /* Ê£»¨¤Ê¼°¤òÁ°¤â¤Ã¤Æ·×»»¤·¤Æ¤ª¤¯ */ | |
1272 /* ɬÍפʤé¶É½êÊÑ¿ô¤òÍѤ¤¤ë¡£ */ | |
1273 /* ¶É½êÊÑ¿ô¤Ø¤Î¥ª¥Õ¥»¥Ã¥È¤ò³Ð¤¨¤Æ¤ª¤¯ */ | |
1274 | |
51 | 1275 for (e2 = target; e2; e2 = cadr(e2)) { |
1276 t0=car(e2); s0=cadddr(e2); | |
52 | 1277 sz=size(ty=caddr(e2)); |
53 | 1278 if(car(t0)==LVAR) { |
1279 /* ¤³¤³¤Ç¡¢½ñ¹þÀ襢¥É¥ì¥¹¤ò·è¤á¤ë */ | |
1280 cadr(t0)=-arg_size+jump_offset; | |
1281 arg_size-=sz; | |
1282 } | |
51 | 1283 if (!is_simple(car(s0))) { |
52 | 1284 disp-=sz; |
53 | 1285 g_expr(assign_expr0((e4=list2(LVAR,disp)),s0,ty,ty)); |
51 | 1286 cadddr(e2)=e4; |
1287 s0=e4; | |
52 | 1288 } else if (is_same_type(t0,s0)) { |
1289 if(cadr(t0)==cadr(s0)) { | |
56 | 1290 #if DEBUG_PARALLEL_ASSIGN |
1291 printf("# remove same memory %d ty %d+%d sz %d\n",car(t0),ty,cadr(t0),sz); | |
1292 #endif | |
52 | 1293 /* we should check size also (but currently useless */ |
1294 remove0(&target,t0); | |
1295 /* still we have source to avoid overwrite */ | |
51 | 1296 } |
52 | 1297 } |
1298 if(is_memory(s0)) { | |
1299 source=list3(s0,source,sz); | |
56 | 1300 #if DEBUG_PARALLEL_ASSIGN |
52 | 1301 printf("# source %d ty %d+%d sz %d\n",car(car(source)),ty,cadr(car(source)),sz); |
51 | 1302 #endif |
48 | 1303 } |
1304 } | |
1305 | |
1306 /* compute jump address */ | |
1307 e2 = cadr(e1); | |
1308 if (car(e2) == FNAME) { | |
1309 code0=(NMTBL *)cadr(e2); | |
1310 if (code0->sc!=CODE) { | |
1311 error(TYERR); return; | |
1312 } | |
1313 } else { /* indirect */ | |
1314 g_expr(e2); | |
1315 emit_push(); | |
1316 } | |
50 | 1317 if (env) { |
1318 g_expr(env); | |
1319 emit_push(); | |
1320 } | |
48 | 1321 |
1322 /* ÊÂÎóÂåÆþ¤ò¼Â¹Ô */ | |
1323 | |
55 | 1324 parallel_assign(&target,&source,&processing,&use); |
1325 while (use) { | |
1326 free_register(caddr(use)); use=cadr(use); | |
1327 } | |
1328 if(target) error(-1); | |
48 | 1329 |
50 | 1330 if (env) { |
1331 /* change the frame pointer */ | |
1332 e3 = emit_pop(0); | |
1333 printf("\tmovl %s,%%ebp\n",register_name(e3,0)); | |
1334 emit_pop_free(e3); | |
1335 } else if (fnptr->sc==FUNCTION) { | |
1336 printf("\tlea %d(%%ebp),%%ebp\n",disp_offset); | |
1337 } | |
1338 | |
48 | 1339 /* shrink stack if necessary */ |
50 | 1340 /* |
1341 if (!env && -arg_size<disp) { | |
1342 printf("\tleal %d(%%ebp),%%esp\n",-arg_size); | |
1343 } Why we need this? | |
1344 */ | |
1345 | |
48 | 1346 if (car(e2) == FNAME) { |
1347 printf("\tjmp %s\n",code0->nm); | |
1348 } else { | |
1349 e2 = emit_pop(0); | |
1350 printf("\tjmp *%s\n",register_name(e2,0)); | |
1351 emit_pop_free(e2); | |
1352 } | |
1353 } | |
1354 | |
0 | 1355 void |
1356 machinop(int e1) | |
1357 { | |
1358 int e2,e3,op; | |
1359 | |
1360 e2 = cadr(e1); | |
1361 op = car(e1); | |
1362 e3 = caddr(e1); | |
1363 g_expr(e3); | |
1364 emit_push(); | |
1365 g_expr(e2); | |
43 | 1366 tosop(car(e1),(e2=pop_register())); |
1367 emit_pop_free(e2); | |
42 | 1368 regv[creg]=1; |
0 | 1369 return; |
1370 } | |
1371 | |
1372 | |
1373 void | |
1374 rindirect(int e1) /* *(p +5 ) */ | |
1375 | |
1376 { | |
1377 char *op; | |
1378 int e2,e3,byte; | |
1379 | |
1380 op = ((byte = (car(e1) == CRINDIRECT)) ? "movsbl" : "movl"); | |
1381 e3 = cadr(e2 = cadr(e1)); | |
1382 g_expr(e2); | |
21 | 1383 printf("\t%s (%s),%s\n",op,register_name(creg,0),register_name(creg,0)); |
0 | 1384 } |
1385 | |
1386 void | |
1387 sassign(int e1) | |
1388 { | |
57 | 1389 int e2,e3,e4,sz,xreg,det; |
0 | 1390 |
18 | 1391 /* structure assignment */ |
37 | 1392 e2 = cadr(e1); /* pointer variable to the struct */ |
1393 e3 = cadr(e2); /* offset of the variable (distination) */ | |
39 | 1394 e4 = caddr(e1); /* right value (source) */ |
37 | 1395 sz = cadddr(e1); /* size of struct or union */ |
0 | 1396 g_expr(e4); |
1397 emit_push(); | |
1398 g_expr(e2); | |
38 | 1399 xreg = emit_pop(0); |
57 | 1400 /* °ìÈÌŪ¤Ë¤Ï¥³¥Ô¡¼¤Î¥ª¡¼¥Ð¥é¥Ã¥×¤Î¾õ¶·¤Ï¼Â¹Ô»þ¤Ë¤·¤«¤ï¤«¤é¤Ê¤¤ */ |
1401 /* ¤·¤«¤·¡¢¤ï¤«¤ë¾ì¹ç¤â¤¢¤ë */ | |
1402 if (car(e4)==RSTRUCT) e4=cadr(e4); | |
1403 if (is_same_type(e2,e4)) { | |
1404 if(cadr(e2)<cadr(e4)) sz=-sz; | |
1405 det=1; | |
1406 } else { | |
1407 det = 0; | |
1408 } | |
1409 emit_copy(xreg,creg,sz,0,1,det); | |
42 | 1410 emit_pop_free(xreg); |
0 | 1411 return; |
1412 } | |
1413 | |
1414 void | |
1415 assign(int e1) | |
1416 { | |
1417 char *op; | |
1418 int e2,e3,e4,byte; | |
1419 | |
1420 op = ((byte=(car(e1) == CASS))? "movb" : "movl"); | |
1421 /* e2=e4 */ | |
1422 e2 = cadr(e1); | |
1423 e3 = cadr(e2); | |
1424 e4 = caddr(e1); | |
1425 switch(car(e2)) { | |
1426 case GVAR: /* i=3 */ | |
1427 g_expr(e4); | |
1428 if (byte) use_data_reg(creg,1); | |
18 | 1429 printf("\t%s %s,%s\n",op,register_name(creg,byte),(char *)caddr(e2)); |
0 | 1430 return; |
1431 case LVAR: | |
1432 g_expr(e4); | |
1433 if (byte) use_data_reg(creg,1); | |
18 | 1434 printf("\t%s %s,%d(%%ebp)\n",op,register_name(creg,byte),lvar(cadr(e2))); |
0 | 1435 return; |
1436 case REGISTER: | |
1437 g_expr(e4); | |
1438 if (creg!=cadr(e2)) | |
23 | 1439 printf("\tmovl %s,%s\n",register_name(creg,0),register_name(cadr(e2),0)); |
0 | 1440 return; |
1441 } | |
23 | 1442 g_expr(e2); |
0 | 1443 emit_push(); |
23 | 1444 use_data_reg(creg,0); |
1445 g_expr(e4); | |
0 | 1446 if (byte) use_data_reg(creg,1); |
23 | 1447 e2 = emit_pop(0); |
1448 printf("\t%s %s,(%s)\n",op,register_name(creg,byte),register_name(e2,0)); | |
42 | 1449 emit_pop_free(e2); |
1450 regv[creg]=1; | |
0 | 1451 return; |
1452 } | |
1453 | |
1454 void | |
1455 assop(int e1) | |
1456 { | |
43 | 1457 int e2,e3,byte,op,reg; |
0 | 1458 char *xrn; |
42 | 1459 int xreg,edx; |
0 | 1460 |
1461 /* e2 op= e3 */ | |
1462 byte = (car(e1) == CASSOP); | |
1463 e2 = cadr(e1); | |
1464 if (car(e2)==INDIRECT) e2=cadr(e2); | |
1465 e3 = caddr(e1); | |
1466 op = cadddr(e1); | |
1467 | |
1468 g_expr(e3); | |
1469 if (car(e2)==REGISTER) { | |
43 | 1470 xreg = creg; |
1471 creg = reg = cadr(e2); | |
1472 tosop(op,xreg); | |
1473 creg = xreg; | |
1474 printf("\tmovl %s,%s\n",register_name(reg,0),register_name(creg,0)); | |
42 | 1475 regv[creg]=1; |
0 | 1476 return; |
1477 } | |
43 | 1478 emit_push(); |
0 | 1479 g_expr(e2); |
42 | 1480 edx = edx_setup(); |
18 | 1481 xrn = register_name(xreg = emit_pop(0),0); /* pop e3 value */ |
42 | 1482 regv[xreg]=regs[xreg]=1; |
1483 printf("\tmovl %s,%s # assop \n",register_name(creg,0),register_name(edx,0)); | |
1484 regv[edx]=1; | |
1485 ld_indexx(byte,0,edx); | |
43 | 1486 tosop(op,xreg); |
42 | 1487 printf("\t%s %s,(%s)\n",byte ? "movb" : "movl",register_name(creg,byte),register_name(edx,0)); |
1488 edx_cleanup(); | |
1489 emit_pop_free(xreg); | |
1490 regv[creg]=1; | |
0 | 1491 return; |
1492 } | |
1493 | |
1494 | |
1495 void | |
43 | 1496 tosop(int op,int oreg) |
0 | 1497 { |
43 | 1498 int dx; |
18 | 1499 char *orn,*crn; |
0 | 1500 |
1501 switch(op) { | |
1502 case LSHIFT: | |
1503 case ULSHIFT: | |
1504 shift("sall",oreg); | |
1505 return; | |
1506 case RSHIFT: | |
1507 shift("sarl",oreg); | |
1508 return; | |
1509 case URSHIFT: | |
1510 shift("shrl",oreg); | |
1511 return; | |
1512 } | |
1513 if(oreg==-1) { | |
18 | 1514 printf("\tpopl %s\n",register_name(dreg,0)); |
0 | 1515 oreg = dreg; |
42 | 1516 regv[dreg]=1; |
0 | 1517 } |
42 | 1518 regv[oreg]=1; regs[oreg]=1; |
18 | 1519 orn = register_name(oreg,0); |
1520 crn = register_name(creg,0); | |
0 | 1521 switch(op) { |
1522 case ADD: | |
1523 printf("\taddl %s,%s\n",orn,crn); | |
41 | 1524 break; |
0 | 1525 case SUB: |
1526 printf("\tsubl %s,%s\n",orn,crn); | |
41 | 1527 break; |
0 | 1528 case BAND: |
1529 printf("\tandl %s,%s\n",orn,crn); | |
41 | 1530 break; |
0 | 1531 case EOR: |
1532 printf("\txorl %s,%s\n",orn,crn); | |
41 | 1533 break; |
0 | 1534 case BOR: |
1535 printf("\torl %s,%s\n",orn,crn); | |
41 | 1536 break; |
0 | 1537 case MUL: |
1538 case UMUL: | |
1539 printf("\t%s %s,%s\n","imull",orn,crn); | |
41 | 1540 break; |
0 | 1541 case DIV: |
1542 case UDIV: | |
42 | 1543 use_register(creg,REG_EAX,1); |
1544 edx_setup(); | |
1545 orn = register_name(oreg,0); | |
0 | 1546 if (op==DIV) |
1547 printf("\tcltd\n\tdivl %s\n",orn); | |
1548 else | |
1549 printf("\txor %%edx,%%edx\n\tidivl %s\n",orn); | |
42 | 1550 edx_cleanup(); |
41 | 1551 break; |
0 | 1552 case MOD: |
1553 case UMOD: | |
42 | 1554 use_register(creg,REG_EAX,1); |
1555 edx_setup(); | |
1556 orn = register_name(oreg,0); | |
0 | 1557 if (op==DIV) |
1558 printf("\tcltd\n\tdivl %s\n",orn); | |
1559 else | |
1560 printf("\txor %%edx,%%edx\n\tidivl %s\n",orn); | |
42 | 1561 dx = virtual(REG_EDX); |
1562 if (dx!=creg) { | |
1563 rname[dx]=rname[creg]; | |
1564 rname[creg]=REG_EDX; | |
1565 } | |
1566 edx_cleanup(); | |
41 | 1567 break; |
0 | 1568 } |
42 | 1569 if (oreg!=dreg&&oreg>=0) |
1570 free_register(oreg); | |
0 | 1571 } |
1572 | |
42 | 1573 static int edx_stack=0; |
0 | 1574 |
42 | 1575 int |
1576 edx_setup() | |
0 | 1577 { |
42 | 1578 int edx_save; |
1579 /* make real EDX register empty */ | |
1580 if (free_register_count()<1) { | |
1581 for(edx_save = 0;edx_save==dreg||edx_save==creg;edx_save++); | |
1582 printf("\tpushl %s\n",register_name(edx_save,0)); | |
1583 edx_stack = list3(edx_save,edx_stack,0); | |
1584 } else { | |
1585 edx_save = get_register(); | |
1586 edx_stack = list3(edx_save,edx_stack,1); | |
0 | 1587 } |
42 | 1588 regv[edx_save]=0; |
1589 use_register(edx_save,REG_EDX,0); | |
1590 return edx_save; | |
0 | 1591 } |
1592 | |
18 | 1593 |
1594 void | |
42 | 1595 edx_cleanup() |
0 | 1596 { |
42 | 1597 if (caddr(edx_stack)==0) { |
1598 printf("\tpopl %s\n",register_name(car(edx_stack),0)); | |
1599 } else | |
1600 free_register(car(edx_stack)); | |
1601 edx_stack = cadr(edx_stack); | |
0 | 1602 } |
1603 | |
1604 void | |
1605 shift(char *op, int reg) | |
1606 { | |
1607 if (reg>=0) { | |
42 | 1608 use_register(reg,REG_ECX,1); |
0 | 1609 } else { |
1610 use_register(dreg,REG_ECX,0); | |
1611 printf("\tpopl %%ecx\n"); | |
1612 } | |
18 | 1613 printf("\t%s %%cl,%s\n",op,register_name(creg,0)); |
0 | 1614 } |
1615 | |
1616 void | |
18 | 1617 ld_indexx(int byte, int n, int xreg) |
0 | 1618 { |
1619 char *op; | |
1620 | |
1621 op = byte ? "movsbl" : "movl"; | |
1622 if (n) | |
18 | 1623 printf("\t%s %d(%s),%s\n",op,n,register_name(xreg,0),register_name(creg,byte)); |
0 | 1624 else |
18 | 1625 printf("\t%s (%s),%s\n",op,register_name(xreg,0),register_name(creg,byte)); |
0 | 1626 } |
1627 | |
18 | 1628 /* |
0 | 1629 void |
18 | 1630 st_indexx(int byte, int n, int xreg) |
0 | 1631 { |
1632 char *op; | |
1633 | |
1634 op = byte ? "movb" : "movl"; | |
1635 if (n) | |
18 | 1636 printf("\t%s %s,%d(%s)\n",op,register_name(creg,byte),n,register_name(xreg,0)); |
0 | 1637 else |
18 | 1638 printf("\t%s %s,(%s)\n",op,register_name(creg,byte),register_name(xreg,0)); |
0 | 1639 } |
1640 | |
1641 void | |
1642 indexy(char *op, char *gvar) | |
1643 { | |
18 | 1644 printf("\t%s %s,%s\n",op,register_name(creg,0),gvar); |
0 | 1645 } |
18 | 1646 */ |
0 | 1647 |
1648 void | |
1649 cmpdimm(int e, int csreg) | |
1650 { | |
1651 /* used in dosiwtch() */ | |
1652 use_register(creg,csreg,0); | |
18 | 1653 printf("\tcmpl $%d,%s\n",e,register_name(creg,0)); |
0 | 1654 } |
1655 | |
1656 void | |
1657 opening(char *filename) | |
1658 { | |
1659 printf("\t.file \"%s\"\n",filename); | |
1660 printf("\t.version\t\"01.01\"\n"); | |
1661 printf("gcc2_compiled.:\n"); | |
1662 printf(".text\n"); | |
1663 emit_init(); | |
1664 } | |
1665 | |
1666 void | |
1667 closing(void) | |
1668 { | |
1669 global_table(); | |
1670 printf("\t.ident \"Micro-C compiled\"\n"); | |
1671 } | |
1672 | |
1673 void | |
1674 rexpr(int e1, int l1, char *s) | |
1675 { | |
1676 g_expr(list3(SUB,cadr(e1),caddr(e1))); | |
1677 printf("\tj%s\t_%d\n",s,l1); | |
1678 } | |
1679 | |
1680 void | |
1681 jcond(int l, char cond) | |
1682 { | |
1683 printf("\tj%s\t_%d\n",cond?"ne":"e",l); | |
1684 } | |
1685 | |
1686 void | |
1687 jmp(int l) | |
1688 { | |
1689 control=0; | |
1690 printf("\tjmp\t_%d\n",l); | |
1691 /* align? */ | |
42 | 1692 /* |
1693 this is not allowed because of ? operator | |
1694 regv[creg]=regv[dreg]=0; | |
1695 use_register(creg,REG_EAX,0); | |
1696 use_register(dreg,REG_EBX,0); | |
1697 */ | |
0 | 1698 } |
1699 | |
1700 int | |
1701 fwdlabel(void) | |
1702 { | |
1703 return labelno++; | |
1704 } | |
1705 | |
1706 void | |
1707 fwddef(int l) | |
1708 { | |
1709 control=1; | |
1710 printf("_%d:\n",l); | |
1711 } | |
1712 | |
1713 int | |
1714 backdef(void) | |
1715 { | |
1716 control=1; | |
1717 printf("_%d:\n",labelno); | |
1718 return labelno++; | |
1719 } | |
1720 | |
1721 void | |
1722 def_label(int cslabel, int dlabel) | |
1723 { | |
1724 int fl; | |
1725 | |
1726 fl = 0; | |
1727 if (control) { | |
1728 jmp(fl=fwdlabel()); | |
1729 } | |
1730 fwddef(cslabel); | |
1731 if (dlabel) | |
1732 jmp(dlabel); | |
1733 if (fl) { | |
1734 fwddef(fl); | |
1735 } | |
1736 } | |
1737 | |
1738 void | |
1739 gen_comment(char *s) | |
1740 { | |
21 | 1741 printf("## %s",s); |
0 | 1742 } |
1743 | |
1744 void | |
1745 gen_source(char *s) | |
1746 { | |
1747 printf("%s",s); | |
1748 } | |
1749 | |
1750 void | |
1751 code_enter(char *name) | |
1752 { | |
1753 printf("\t.align 4\n"); | |
1754 if (stmode!=STATIC) | |
1755 printf(".globl %s\n",name); | |
1756 printf("\t.type\t%s,@function\n",name); | |
1757 printf("%s:\n",name); | |
1758 } | |
1759 | |
1760 void | |
50 | 1761 code_enter1(int args) |
0 | 1762 { |
36 | 1763 code_disp_label=fwdlabel(); |
50 | 1764 printf("\tlea _%d(%%ebp),%%esp\n",code_disp_label); |
53 | 1765 |
21 | 1766 printf("## args %d disp %d code_arg_offset=%d code_disp_offset=%d\n",args,disp,code_arg_offset,code_disp_offset); |
0 | 1767 } |
1768 | |
1769 void | |
50 | 1770 code_leave(char *name) |
0 | 1771 { |
53 | 1772 printf("\t.set _%d,%d\n",code_disp_label,disp+code_disp_offset); |
0 | 1773 printf("_%d:\n",labelno); |
1774 printf("\t.size\t%s,_%d-%s\n",name,labelno,name); | |
12 | 1775 local_table(); |
0 | 1776 labelno++; |
55 | 1777 free_all_register(); |
0 | 1778 } |
1779 | |
1780 void | |
1781 enter(char *name) | |
1782 { | |
1783 printf("\t.align 2\n"); | |
1784 if (stmode!=STATIC) | |
1785 printf(".globl %s\n",name); | |
1786 printf("%s:\n",name); | |
1787 printf("\t.type\t%s,@function\n",name); | |
1788 printf("\tpushl %%ebp\n"); | |
1789 printf("\tmovl %%esp,%%ebp\n"); | |
1790 printf("\tpushl %%ebx\n"); | |
24 | 1791 printf("\tpushl %%esi\n"); |
1792 printf("\tpushl %%edi\n"); | |
0 | 1793 } |
1794 | |
1795 void | |
50 | 1796 enter1() |
0 | 1797 { |
36 | 1798 func_disp_label=fwdlabel(); |
50 | 1799 printf("\tlea _%d(%%ebp),%%esp\n",func_disp_label); |
36 | 1800 /* if(disp) printf("\tsubl $%d,%%esp\n",-disp); */ |
0 | 1801 } |
1802 | |
1803 void | |
50 | 1804 leave(int control, char *name) |
0 | 1805 { |
1806 if (control) | |
1807 use_register(creg,REG_EAX,1); | |
1808 if (retcont) { | |
1809 if (control) | |
1810 jmp(retlabel); | |
1811 fwddef(retcont); | |
1812 use_register(creg,REG_EAX,0); | |
18 | 1813 printf("\tmovl %s,%s\n",reg_name[REG_ESI],register_name(creg,0)); |
0 | 1814 /* printf("\tleave\n"); */ |
1815 } | |
1816 fwddef(retlabel); | |
42 | 1817 /* use_register(creg,REG_EAX,0); too late */ |
52 | 1818 /* if(disp) printf("\taddl $%d,%%esp\n",-disp); */ |
0 | 1819 printf("\tlea %d(%%ebp),%%esp\n",disp_offset); |
24 | 1820 printf("\tpopl %%edi\n"); |
1821 printf("\tpopl %%esi\n"); | |
0 | 1822 printf("\tpopl %%ebx\n"); |
1823 printf("\tleave\n"); | |
1824 printf("\tret\n"); | |
52 | 1825 printf("\t.set _%d,%d\n",func_disp_label,disp+disp_offset); |
0 | 1826 printf("_%d:\n",labelno); |
1827 printf("\t.size\t%s,_%d-%s\n",name,labelno,name); | |
12 | 1828 local_table(); |
0 | 1829 labelno++; |
55 | 1830 free_all_register(); |
0 | 1831 } |
1832 | |
1833 | |
1834 void | |
1835 ret(void) | |
1836 { | |
1837 use_register(creg,REG_EAX,1); | |
1838 jmp(retlabel); | |
1839 } | |
1840 | |
1841 void | |
1842 gen_gdecl(char *n, int gpc) | |
1843 { | |
1844 /* | |
1845 if (stmode!=STATIC) | |
1846 printf(".globl %s\n",n); | |
1847 */ | |
1848 } | |
1849 | |
1850 void | |
1851 align(int t) | |
1852 { | |
1853 if (t!=CHAR) { | |
1854 if (data_alignment & 1) | |
1855 printf("\t.align 2\n"); | |
1856 data_alignment = 0; | |
1857 } | |
1858 } | |
1859 | |
1860 void | |
1861 emit_data(int e, int t, NMTBL *n) | |
1862 { | |
1863 int l; | |
1864 char *name; | |
1865 name = n->nm; | |
13 | 1866 if(mode!=GDECL) { |
1867 error(-1); return; | |
1868 } | |
1869 if (n->dsp != -1) { | |
1870 n->dsp = -1; /* initiallized flag */ | |
1871 printf(".globl\t%s\n",name); | |
1872 data_mode(name); | |
1873 align(t); | |
1874 printf("%s:\n",name); | |
1875 } else { | |
1876 data_mode(0); | |
1877 } | |
1878 if(car(e)==CONST) { | |
1879 if (t==CHAR) { | |
1880 printf("\t.byte %d\n",cadr(e)); | |
1881 if (data_alignment>0) | |
1882 data_alignment++; | |
1883 gpc += 1; | |
0 | 1884 } else { |
13 | 1885 printf("\t.long %d\n",cadr(e)); |
3 | 1886 gpc += size_of_int; |
13 | 1887 } |
1888 } else if(t!=CHAR) { | |
1889 gpc += size_of_int; | |
1890 if(car(e)==ADDRESS&&car(cadr(e))==GVAR) { | |
18 | 1891 printf("\t.long %s\n",(char *)caddr(cadr(e))); |
13 | 1892 } else if(car(e)==FNAME) { |
1893 printf("\t.long %s\n",((NMTBL *)cadr(e))->nm); | |
1894 } else if(car(e)==STRING) { | |
1895 if (car(n->ty)!=ARRAY || cadr(n->ty)!=CHAR) { | |
1896 l = fwdlabel(); | |
1897 printf("\t.long _%d\n",l); | |
1898 printf(".section\t.rodata\n"); | |
1899 printf("_%d:\n",l); | |
1900 output_mode = RODATA_EMIT_MODE; | |
1901 } | |
1902 ascii((char *)cadr(e)); | |
1903 } else error(TYERR); | |
0 | 1904 } |
1905 } | |
1906 | |
1907 void | |
1908 emit_data_closing(NMTBL *n) | |
1909 { | |
1910 int lb; | |
18 | 1911 if (mode==GDECL) { |
0 | 1912 data_mode(0); |
1913 lb=fwdlabel(); | |
1914 printf("_%d:\n",lb); | |
1915 printf("\t.size\t%s,_%d-%s\n",n->nm,lb,n->nm); | |
1916 } | |
1917 } | |
1918 | |
1919 void | |
1920 global_table(void) | |
1921 { | |
1922 NMTBL *n; | |
12 | 1923 int init; |
1924 init=0; | |
0 | 1925 for(n=ntable;n < &ntable[GSYMS];n++) { |
1926 if (n->sc == GVAR && n->dsp != -1) { | |
1927 /* n->dsp = -1 means initialized global */ | |
12 | 1928 if (init==0) { |
1929 data_mode(0); | |
1930 init=1; | |
1931 } | |
0 | 1932 printf(".comm %s,%d\n",n->nm,size(n->ty)); |
1933 } | |
1934 } | |
1935 } | |
1936 | |
1937 void | |
1938 local_table(void) | |
1939 { | |
1940 NMTBL *n; | |
12 | 1941 int init; |
1942 init=0; | |
0 | 1943 /* static local variables */ |
1944 for(n=ntable+GSYMS;n < &ntable[GSYMS+LSYMS];n++) { | |
1945 if (n->sc == GVAR) { | |
12 | 1946 if (init==0) { |
1947 data_mode(0); | |
1948 init=1; | |
1949 } | |
0 | 1950 printf(".lcomm %s,%d\n",n->nm,size(n->ty)); |
1951 } | |
1952 } | |
1953 } | |
1954 | |
1955 void | |
1956 text_mode(void) | |
1957 { | |
1958 if (output_mode!=TEXT_EMIT_MODE) { | |
1959 printf(".text\n"); | |
1960 printf("\t.align 2\n"); | |
1961 output_mode = TEXT_EMIT_MODE; | |
1962 } | |
1963 } | |
1964 | |
1965 void | |
1966 data_mode(char *name) | |
1967 { | |
1968 if (output_mode!=DATA_EMIT_MODE) { | |
1969 printf(".data\n"); | |
1970 output_mode = DATA_EMIT_MODE; | |
1971 } | |
1972 if (name) | |
1973 printf("\t.type\t%s,@object\n",name); | |
1974 } | |
1975 | |
1976 int | |
1977 lvar(int l) | |
1978 { | |
50 | 1979 if (fnptr->sc==CODE) { |
53 | 1980 return l+code_disp_offset; |
50 | 1981 } else if (l<0) { |
1982 return l+disp_offset; | |
0 | 1983 } else { |
50 | 1984 return l+arg_offset; |
0 | 1985 } |
1986 } | |
1987 | |
50 | 1988 |
0 | 1989 /* end */ |