summaryrefslogtreecommitdiff
path: root/utils/vl/asm.c
diff options
context:
space:
mode:
Diffstat (limited to 'utils/vl/asm.c')
-rw-r--r--utils/vl/asm.c186
1 files changed, 136 insertions, 50 deletions
diff --git a/utils/vl/asm.c b/utils/vl/asm.c
index 28ec1a89..1e7039f1 100644
--- a/utils/vl/asm.c
+++ b/utils/vl/asm.c
@@ -13,7 +13,27 @@ long BADOFFSET = -1;
OFFSET++;\
*/
-#define LPUT(c)\
+#define LPUT(l) { \
+ if (little) { \
+ LLEPUT(l); \
+ } else { \
+ LBEPUT(l); \
+ } \
+ }
+
+#define LLEPUT(c)\
+ {\
+ cbp[0] = (c);\
+ cbp[1] = (c)>>8;\
+ cbp[2] = (c)>>16;\
+ cbp[3] = (c)>>24;\
+ cbp += 4;\
+ cbc -= 4;\
+ if(cbc <= 0)\
+ cflush();\
+ }
+
+#define LBEPUT(c)\
{\
cbp[0] = (c)>>24;\
cbp[1] = (c)>>16;\
@@ -25,6 +45,35 @@ long BADOFFSET = -1;
cflush();\
}
+#define HPUT(h) { \
+ if (little) { \
+ HLEPUT(h); \
+ } else { \
+ HBEPUT(h); \
+ } \
+ }
+
+#define HLEPUT(c)\
+ {\
+ cbp[0] = (c);\
+ cbp[1] = (c)>>8;\
+ cbp += 2;\
+ cbc -= 2;\
+ if(cbc <= 0)\
+ cflush();\
+ }
+
+#define HBEPUT(c)\
+ {\
+ cbp[0] = (c)>>8;\
+ cbp[1] = (c);\
+ cbp += 2;\
+ cbc -= 2;\
+ if(cbc <= 0)\
+ cflush();\
+ }
+
+
#define CPUT(c)\
{\
cbp[0] = (c);\
@@ -34,6 +83,24 @@ long BADOFFSET = -1;
cflush();\
}
+void
+objput(long l) /* emit long in byte order appropriate to object machine */
+{
+ LPUT(l);
+}
+
+void
+objhput(short s)
+{
+ HPUT(s);
+}
+
+void
+lput(long l) /* emit long in big-endian byte order */
+{
+ LBEPUT(l);
+}
+
long
entryvalue(void)
{
@@ -111,6 +178,7 @@ asmb(void)
case 2:
case 3:
case 5:
+ case 6:
OFFSET = HEADR+textsize;
seek(cout, OFFSET, 0);
break;
@@ -138,6 +206,7 @@ asmb(void)
case 2:
case 1:
case 5:
+ case 6:
OFFSET = HEADR+textsize+datsize;
seek(cout, OFFSET, 0);
break;
@@ -203,7 +272,11 @@ asmb(void)
lput(0L); /* complete mystery */
break;
case 2:
- lput(0x407); /* magic */
+ if (little)
+ t = 24;
+ else
+ t = 16;
+ lput(((((4*t)+0)*t)+7)); /* magic */
lput(textsize); /* sizes */
lput(datsize);
lput(bsssize);
@@ -318,47 +391,67 @@ asmb(void)
lput(0x80L); /* flags */
break;
case 5:
+ /* first part of ELF is byte-wide parts, thus no byte-order issues */
strnput("\177ELF", 4); /* e_ident */
CPUT(1); /* class = 32 bit */
- CPUT(2); /* data = MSB */
- CPUT(1); /* version = CURRENT */
- strnput("", 9);
- lput((2L<<16)|8L); /* type = EXEC; machine = MIPS */
- lput(1L); /* version = CURRENT */
- lput(entryvalue()); /* entry vaddr */
- lput(52L); /* offset to first phdr */
- lput(0L); /* offset to first shdr */
- lput(0L); /* flags = MIPS */
- lput((52L<<16)|32L); /* Ehdr & Phdr sizes*/
- lput((3L<<16)|0L); /* # Phdrs & Shdr size */
- lput((0L<<16)|0L); /* # Shdrs & shdr string size */
-
- lput(1L); /* text - type = PT_LOAD */
- lput(0L); /* file offset */
- lput(INITTEXT-HEADR); /* vaddr */
- lput(INITTEXT-HEADR); /* paddr */
- lput(HEADR+textsize); /* file size */
- lput(HEADR+textsize); /* memory size */
- lput(0x05L); /* protections = RX */
- lput(0x10000L); /* alignment code?? */
-
- lput(1L); /* data - type = PT_LOAD */
- lput(HEADR+textsize); /* file offset */
- lput(INITDAT); /* vaddr */
- lput(INITDAT); /* paddr */
- lput(datsize); /* file size */
- lput(datsize+bsssize); /* memory size */
- lput(0x06L); /* protections = RW */
- lput(0x10000L); /* alignment code?? */
-
- lput(0L); /* data - type = PT_NULL */
- lput(HEADR+textsize+datsize); /* file offset */
- lput(0L);
- lput(0L);
- lput(symsize); /* symbol table size */
- lput(lcsize); /* line number size */
- lput(0x04L); /* protections = R */
- lput(0x04L); /* alignment code?? */
+ CPUT(little? 1: 2); /* data: 1 = LSB, 2 = MSB */
+ CPUT(1); /* version = 1 */
+ strnput("", 9); /* reserved for expansion */
+ /* entire remainder of ELF file is in target byte order */
+
+ /* file header part of ELF header */
+ objhput(2); /* type = EXEC */
+ objhput(8); /* machine = MIPS */
+ objput(1L); /* version = CURRENT */
+ objput(entryvalue()); /* entry vaddr */
+ objput(52L); /* offset to first phdr */
+ objput(0L); /* offset to first shdr */
+ objput(0L); /* flags (no MIPS flags defined) */
+ objhput(52); /* Ehdr size */
+ objhput(32); /* Phdr size */
+ objhput(3); /* # of Phdrs */
+ objhput(0); /* Shdr size */
+ objhput(0); /* # of Shdrs */
+ objhput(0); /* Shdr string size */
+
+ /* "Program headers" - one per chunk of file to load */
+
+ /*
+ * include ELF headers in text -- 8l doesn't,
+ * but in theory it aids demand loading.
+ */
+ objput(1L); /* text: type = PT_LOAD */
+ objput(0L); /* file offset */
+ objput(INITTEXT-HEADR); /* vaddr */
+ objput(INITTEXT-HEADR); /* paddr */
+ objput(HEADR+textsize); /* file size */
+ objput(HEADR+textsize); /* memory size */
+ objput(0x05L); /* protections = RX */
+ objput(0x1000L); /* page-align text off's & vaddrs */
+
+ objput(1L); /* data: type = PT_LOAD */
+ objput(HEADR+textsize); /* file offset */
+ objput(INITDAT); /* vaddr */
+ objput(INITDAT); /* paddr */
+ objput(datsize); /* file size */
+ objput(datsize+bsssize); /* memory size */
+ objput(0x06L); /* protections = RW */
+ if(INITDAT % 4096 == 0 && (HEADR + textsize) % 4096 == 0)
+ objput(0x1000L); /* page-align data off's & vaddrs */
+ else
+ objput(0L); /* do not claim alignment */
+
+ objput(0L); /* P9 symbols: type = PT_NULL */
+ objput(HEADR+textsize+datsize); /* file offset */
+ objput(0L);
+ objput(0L);
+ objput(symsize); /* symbol table size */
+ objput(lcsize); /* line number size */
+ objput(0x04L); /* protections = R */
+ objput(0L); /* do not claim alignment */
+ break;
+ case 6:
+ break;
}
cflush();
}
@@ -375,13 +468,6 @@ strnput(char *s, int n)
}
void
-lput(long l)
-{
-
- LPUT(l);
-}
-
-void
cflush(void)
{
int n;
@@ -477,7 +563,7 @@ putsymb(char *s, int t, long v, int ver)
if(t == 'f')
s++;
- LPUT(v);
+ LBEPUT(v);
if(ver)
t += 'a' - 'A';
CPUT(t+0x80); /* 0x80 is variable length */
@@ -820,7 +906,7 @@ asmout(Prog *p, Optab *o, int aflag)
else
v = (p->cond->pc - pc-4) >> 2;
if(((v << 16) >> 16) != v)
- diag("short branch too far: %d\n%P", v, p);
+ diag("short branch too far: %ld\n%P", v, p);
o1 = OP_IRR(opirr(p->as), v, p->from.reg, p->reg);
break;