diff mc-nop-386.c @ 38:d48d952da354

struct copy continue
author kono
date Tue, 11 Feb 2003 11:26:51 +0900
parents 412ad2e6c2a2
children c63c4fdeb9a7
line wrap: on
line diff
--- a/mc-nop-386.c	Mon Feb 10 22:29:12 2003 +0900
+++ b/mc-nop-386.c	Tue Feb 11 11:26:51 2003 +0900
@@ -55,7 +55,7 @@
 static void tosop(int op);
 static void div_cleanup(char *orn);
 static void use_register(int virt, int real, int move);
-static void emit_copy(char *from,char *to,int length,int offset);
+static void emit_copy(int from,int to,int length,int offset);
 
 extern int error(int n);
 extern int fwdlabel(void);
@@ -177,6 +177,10 @@
 char *
 register_name(int i,int byte)
 {
+    if (i<0) {
+	error(REG_ERR);
+	return "%eax";
+    }
     if (byte && rname[i] <= REG_EDX) {
 	return reg_name_l[rname[i]];
     } else {
@@ -740,22 +744,22 @@
 #define MAX_COPY_LEN 20
 
 void 
-emit_copy(char *from,char *to,int length,int offset)
+emit_copy(int from,int  to,int length,int offset)
 {
     if (length<0) return;
     switch (length) {
     case 0:	break;
     case 1:
-	printf("\tmovb %d(%s),%s\n",offset,from, reg_name_l[rname[dreg]] );
-	printf("\tmovb %s,%d(%s)\n",reg_name_l[rname[dreg]] ,offset,to);
+	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:
-	printf("\tmovw %d(%s),%s\n",offset,from, reg_name_w[rname[dreg]] );
-	printf("\tmovw %s,%d(%s)\n",reg_name_w[rname[dreg]] ,offset,to);
+	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:
-	printf("\tmovl %d(%s),%s\n",offset,from, register_name(dreg,0));
-	printf("\tmovl %s,%d(%s)\n",register_name(dreg,0), offset,to);
+	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) {
@@ -767,16 +771,13 @@
 		emit_copy(from,to,length,offset);
 	    return;
 	}
-	printf("\txchg %%edi,%s\n",from);
-	printf("\txchg %%esi,%s\n",to);
-	printf("\txchg %%ecx,%s\n",register_name(dreg,0));
+	use_register(from,REG_EDI,1);
+	use_register(to,  REG_ESI,1);
+	use_register(dreg,REG_ECX,1);
 	printf("\tmovl $%d,%%ecx\n",length/4);
 	printf("\tcld\n\trep\n\tmovsl\n");
-	printf("\txchg %%ecx,%s\n",register_name(dreg,0));
-	printf("\txchg %%esi,%s\n",to);
-	printf("\txchg %%edi,%s\n",from);
 	if(length%4) {
-	    emit_copy("%esi","%edi",length,offset+length/4);
+	    emit_copy(from,to,length,offset+length/4);
 	}
     }
 }
@@ -784,28 +785,29 @@
 int
 struct_push(int e4,int t) 
 {
-    int length;
+    int length,xreg,save;
     g_expr(e4);
     length=size(t); 
     if(length%size_of_int) {
 	length += size_of_int - (length%size_of_int);
     }
+    /* I think, after g_expr, at least two registers are free */
     printf("\tsubl $%d,%%esp\n",length);
-    if (length<=MAX_COPY_LEN)
-	emit_copy(register_name(creg,0),"%esp",length,0);
-    else {
-	printf("\tpushl %%edi\n");
-	printf("\tpushl %%esi\n");
-	printf("\tpushl %%ecx\n");
-	printf("\tleal 12(%%esp),%%edi\n");
-	if (rname[creg]!=REG_ESI)
-		printf("\tmovl %s,%%esi\n",register_name(creg,0));
-	printf("\tmovl $%d,%%ecx\n",length/size_of_int);
-	printf("\tcld\n\trep\n\tmovsl\n");
-	printf("\tpopl %%ecx\n");
-	printf("\tpopl %%esi\n");
-	printf("\tpopl %%edi\n");
+    if (register_full()) {
+	/* this is wrong assumption */
+	save = 1;
+	printf("\tpushl %s\n",register_name(lreg,0));
+	xreg = lreg;
+    } else {
+	save=0;
+	xreg = get_register();
     }
+    printf("\tmovl %%esp,%s\n",register_name(xreg,0));
+    emit_copy(creg,xreg,length,0);
+    if (save) {
+	printf("\tpopl %s\n",register_name(lreg,0));
+    } else
+	free_register(xreg);
     return length/size_of_int;
 }
 
@@ -1107,8 +1109,7 @@
 void
 sassign(int e1)
 {
-    int e2,e3,e4,sz;
-    char *xrn;
+    int e2,e3,e4,sz,xreg;
 
     /* structure assignment */
     e2 = cadr(e1);  /* pointer variable to the struct */
@@ -1118,8 +1119,8 @@
     g_expr(e4);
     emit_push();
     g_expr(e2);
-    xrn = register_name(emit_pop(0),0);
-    emit_copy(xrn,register_name(creg,0),sz,0);
+    xreg = emit_pop(0);
+    emit_copy(xreg,creg,sz,0);
     return;
 }