Mercurial > hg > CbC > old > device
diff mc-nop-386.c @ 57:3d7f199e99d0
struct handling
author | kono |
---|---|
date | Wed, 19 Feb 2003 17:19:55 +0900 |
parents | 5aa4528b6983 |
children | 727c280bdd25 |
line wrap: on
line diff
--- a/mc-nop-386.c Wed Feb 19 12:15:05 2003 +0900 +++ b/mc-nop-386.c Wed Feb 19 17:19:55 2003 +0900 @@ -59,7 +59,7 @@ static void tosop(int op,int reg); static void edx_cleanup(); static void use_register(int virt, int real, int move); -static void emit_copy(int from,int to,int length,int offset,int value); +static void emit_copy(int from,int to,int length,int offset,int value,int det); static int is_same_type(int s,int t); extern int error(int n); @@ -677,9 +677,6 @@ use_register(creg,REG_EAX,1); fwddef(e3); return; - case RSTRUCT: - g_expr(e2); - return; case SASS: sassign(e1); return; @@ -689,6 +686,9 @@ case ASSOP: case CASSOP: assop(e1); return; + case RSTRUCT: + g_expr(e2); + return; case COMMA: g_expr(e2); g_expr(caddr(e1)); @@ -858,45 +858,64 @@ #define MAX_COPY_LEN 20 void -emit_copy(int from,int to,int length,int offset,int value) +emit_copy(int from,int to,int length,int offset,int value,int det) { int fix = 0; - if (length<=0) return; + /* length <0 means upward direction copy */ switch (length) { case 0: break; - case 1: + case 1: case -1: printf("\tmovb %d(%s),%s\n",offset,register_name(from,0), reg_name_l[rname[dreg]] ); printf("\tmovb %s,%d(%s)\n",reg_name_l[rname[dreg]] ,offset,register_name(to,0)); break; - case 2: + case 2: case -2: printf("\tmovw %d(%s),%s\n",offset,register_name(from,0), reg_name_w[rname[dreg]] ); printf("\tmovw %s,%d(%s)\n",reg_name_w[rname[dreg]] ,offset,register_name(to,0)); break; - case 4: + case 4: case -4: printf("\tmovl %d(%s),%s\n",offset,register_name(from,0), register_name(dreg,0)); printf("\tmovl %s,%d(%s)\n",register_name(dreg,0), offset,register_name(to,0)); break; default: - if (length <=MAX_COPY_LEN) { + if (-MAX_COPY_LEN<length && length <0) { + for(;length<=4;length+=4,offset-=4) + emit_copy(from,to,4,offset,0,det); + for(;length<=2;length+=2,offset-=2) + emit_copy(from,to,2,offset,0,det); + if(length>0) + emit_copy(from,to,length,offset,0,det); + break; + } else if (length <=MAX_COPY_LEN) { for(;length>=4;length-=4,offset+=4) - emit_copy(from,to,4,offset,0); + emit_copy(from,to,4,offset,0,det); for(;length>=2;length-=2,offset+=2) - emit_copy(from,to,2,offset,0); + emit_copy(from,to,2,offset,0,det); if(length>0) - emit_copy(from,to,length,offset,0); + emit_copy(from,to,length,offset,0,det); break; } use_register(from,REG_ESI,1); use_register(to, REG_EDI,1); use_register(dreg,REG_ECX,0); - printf("\tmovl $%d,%%ecx\n",length/4); - fix = (length/4)*4; - printf("\tcld\n\trep\n\tmovsl\n"); - if(length%4) { - emit_copy(from,to,length,offset+length/4,0); + if (length<0) { + printf("\tmovl $%d,%%ecx\n",-length/4); + printf("\taddl $%d,%%esi\n",-length); + printf("\taddl $%d,%%edi\n",-length); + printf("\tstd\n\trep\n\tmovsl\n"); + if(length%4) { + emit_copy(from,to,length,offset+length/4,0,det); + } + } else { + printf("\tmovl $%d,%%ecx\n",length/4); + fix = (length/4)*4; + printf("\tcld\n\trep\n\tmovsl\n"); + if(length%4) { + emit_copy(from,to,length,offset+length/4,0,det); + } } } if (value) { + /* creg must point top of the destination data */ /* this code is necessary for the value of assignment or function call */ /* otherwise we don't need this */ if (fix) printf("\tsubl $%d,%s\n",fix,register_name(to,0)); @@ -945,7 +964,8 @@ else printf("\tmovl %%esp,%s\n",register_name(xreg,0)); regv[xreg]=1; - emit_copy(creg,xreg,length,0,0); + /* downward direction copy */ + emit_copy(creg,xreg,length,0,0,1); /* we have value in creg, it may be changed */ if (save) { if(creg==xreg) { @@ -1038,7 +1058,7 @@ /* source (after) list2(tag,disp) */ /* source list list3(e,cdr,sz) */ -#define DEBUG_PARALLEL_ASSIGN 0 +#define DEBUG_PARALLEL_ASSIGN 1 int overrap(int t,int sz,int source) @@ -1188,7 +1208,7 @@ { return ( e1==CONST || e1==FNAME || e1==LVAR || e1==REGISTER || - e1==GVAR || RGVAR || RLVAR || CRLVAR || CRGVAR + e1==GVAR || e1==RGVAR || e1==RLVAR || e1==CRLVAR || e1==CRGVAR ); } @@ -1372,7 +1392,7 @@ void sassign(int e1) { - int e2,e3,e4,sz,xreg; + int e2,e3,e4,sz,xreg,det; /* structure assignment */ e2 = cadr(e1); /* pointer variable to the struct */ @@ -1383,7 +1403,16 @@ emit_push(); g_expr(e2); xreg = emit_pop(0); - emit_copy(xreg,creg,sz,0,1); + /* 一般的にはコピーのオーバラップの状況は実行時にしかわからない */ + /* しかし、わかる場合もある */ + if (car(e4)==RSTRUCT) e4=cadr(e4); + if (is_same_type(e2,e4)) { + if(cadr(e2)<cadr(e4)) sz=-sz; + det=1; + } else { + det = 0; + } + emit_copy(xreg,creg,sz,0,1,det); emit_pop_free(xreg); return; }