ref: 14aeadb41737f1765b6f3489f3b8da330aeccf39
parent: 0bd2678722f5f70a063190d54d12e62420f02b05
author: Ori Bernstein <[email protected]>
date: Fri Aug 21 14:54:53 EDT 2015
Get closer to the C ABI
--- a/6/asm.h
+++ b/6/asm.h
@@ -130,10 +130,14 @@
struct Func {
char *name; /* function name */
int isexport; /* is this exported from the asm? */
- size_t stksz; /* stack size */
Type *type; /* type of function */
+
+ Node **args; /* argument list */
+ size_t nargs; /* number of args, including hidden ones */
Htab *stkoff; /* Loc* -> int stackoff map */
+ size_t stksz; /* stack size */
Node *ret; /* return value */
+
Cfg *cfg; /* flow graph */
};
--- a/6/insns.def
+++ b/6/insns.def
@@ -304,7 +304,11 @@
Insn(Icall,
"\tcall %v\n",
"\tCALL %V\n",
- Use(.l={1}),
+ Use(.l={1}, .r={
+ Rrdi, Rrsi, Rrdx, Rrcx, Rr8, Rr9,
+ Rxmm0d, Rxmm1d, Rxmm2d, Rxmm3d,
+ Rxmm4d, Rxmm5d, Rxmm6d, Rxmm7d,
+ }),
Def(.r={Rrax, Rrbx, Rrcx, Rrdx, Rrsi,
Rrdi, Rr8,Rr9,Rr10,Rr11,
Rxmm0d, Rxmm1d, Rxmm2d, Rxmm3d,
@@ -315,7 +319,11 @@
Insn(Icallind,
"\tcall *%v\n",
"\tCALL *%V\n",
- Use(.l={1}),
+ Use(.l={1}, .r={
+ Rrdi, Rrsi, Rrdx, Rrcx, Rr8, Rr9,
+ Rxmm0d, Rxmm1d, Rxmm2d, Rxmm3d,
+ Rxmm4d, Rxmm5d, Rxmm6d, Rxmm7d,
+ }),
Def(.r={Rrax, Rrbx, Rrcx, Rrdx, Rrsi,
Rrdi, Rr8,Rr9,Rr10,Rr11,
Rxmm0d, Rxmm1d, Rxmm2d, Rxmm3d,
--- a/6/isel.c
+++ b/6/isel.c
@@ -19,6 +19,14 @@
/* forward decls */
Loc *selexpr(Isel *s, Node *n);
+#define Nfloatregargs 8
+#define Nintregargs 6
+regid floatargregs[] = {
+ Rxmm0d, Rxmm1d, Rxmm2d, Rxmm3d,
+ Rxmm4d, Rxmm5d, Rxmm6d, Rxmm7d,
+};
+regid intargregs[] = {Rrdi, Rrsi, Rrdx, Rrcx, Rr8, Rr9};
+
/* used to decide which operator is appropriate
* for implementing various conditional operators */
struct {
@@ -50,14 +58,12 @@
[Ofle] = {Icomis, Ijbe, Isetbe},
};
-static Mode mode(Node *n)
+static Mode tymode(Type *t)
{
- Type *t;
-
- t = tybase(exprtype(n));
/* FIXME: What should the mode for, say, structs be when we have no
* intention of loading /through/ the pointer? For now, we'll just say it's
* the pointer mode, since we expect to address through the pointer */
+ t = tybase(t);
switch (t->type) {
case Tyflt32: return ModeF; break;
case Tyflt64: return ModeD; break;
@@ -64,7 +70,7 @@
default:
if (stacktype(t))
return ModeQ;
- switch (size(n)) {
+ switch (tysize(t)) {
case 1: return ModeB; break;
case 2: return ModeW; break;
case 4: return ModeL; break;
@@ -75,6 +81,16 @@
return ModeQ;
}
+static Mode mode(Node *n)
+{
+ if (n->type == Nexpr)
+ return tymode(exprtype(n));
+ else if (n->type == Ndecl)
+ return tymode(n->decl.type);
+ else
+ die("invalid node type");
+}
+
static Loc *loc(Isel *s, Node *n)
{
ssize_t stkoff;
@@ -90,9 +106,11 @@
stkoff = ptoi(htget(s->stkoff, n));
l = locmem(-stkoff, locphysreg(Rrbp), NULL, mode(n));
} else {
- if (!hthas(s->reglocs, n))
- htput(s->reglocs, n, locreg(mode(n)));
- return htget(s->reglocs, n);
+ l = htget(s->reglocs, n);
+ if (!l) {
+ l = locreg(mode(n));
+ htput(s->reglocs, n, l);
+ }
}
break;
case Olit:
@@ -471,14 +489,31 @@
g(s, op, f, NULL);
}
+static size_t countargs(Type *t)
+{
+ size_t nargs;
+
+ t = tybase(t);
+ nargs = t->nsub - 1;
+ if (stacktype(t->sub[0]))
+ nargs++;
+ /* valists are replaced with hidden type parameter,
+ * which we want on the stack for ease of ABI */
+ if (tybase(t->sub[t->nsub - 1])->type == Tyvalist)
+ nargs--;
+ return nargs;
+}
+
static Loc *gencall(Isel *s, Node *n)
{
Loc *src, *dst, *arg; /* values we reduced */
+ size_t argsz, argoff, nargs;
+ size_t nfloats, nints;
Loc *retloc, *rsp, *ret; /* hard-coded registers */
Loc *stkbump; /* calculated stack offset */
- int argsz, argoff;
size_t i, a;
- Type *t;
+ int vararg;
+ Type *t, *fn;
rsp = locphysreg(Rrsp);
t = exprtype(n);
@@ -492,6 +527,10 @@
retloc = coreg(Rrax, mode(n));
ret = locreg(mode(n));
}
+ fn = tybase(exprtype(n->expr.args[0]));
+ /* calculate the number of args we expect to see, adjust
+ * for a hidden return argument. */
+ nargs = countargs(fn);
argsz = 0;
/* Have to calculate the amount to bump the stack
* pointer by in one pass first, otherwise if we push
@@ -510,20 +549,36 @@
/* Now, we can evaluate the arguments */
argoff = 0;
+ nfloats = 0;
+ nints = 0;
+ vararg = 0;
for (i = 1; i < n->expr.nargs; i++) {
arg = selexpr(s, n->expr.args[i]);
argoff = align(argoff, min(size(n->expr.args[i]), Ptrsz));
+ if (i > nargs)
+ vararg = 1;
if (stacknode(n->expr.args[i])) {
src = locreg(ModeQ);
g(s, Ilea, arg, src, NULL);
a = alignto(1, n->expr.args[i]->expr.type);
blit(s, rsp, src, argoff, 0, size(n->expr.args[i]), a);
+ argoff += size(n->expr.args[i]);
+ } else if (!vararg && isfloatmode(arg->mode) && nfloats < Nfloatregargs) {
+ dst = coreg(floatargregs[nfloats], arg->mode);
+ arg = inri(s, arg);
+ g(s, Imovs, arg, dst, NULL);
+ nfloats++;
+ } else if (!vararg && isintmode(arg->mode) && nints < Nintregargs) {
+ dst = coreg(intargregs[nints], arg->mode);
+ arg = inri(s, arg);
+ g(s, Imov, arg, dst, NULL);
+ nints++;
} else {
dst = locmem(argoff, rsp, NULL, arg->mode);
arg = inri(s, arg);
stor(s, arg, dst);
+ argoff += size(n->expr.args[i]);
}
- argoff += size(n->expr.args[i]);
}
call(s, n->expr.args[0]);
if (argsz)
@@ -842,8 +897,48 @@
Rnone
};
-static void prologue(Isel *s, size_t sz)
+void addarglocs(Isel *s, Func *fn)
{
+ size_t i, nints, nfloats, nargs;
+ size_t argoff;
+ int vararg;
+ Node *arg;
+ Loc *a, *l;
+
+ argoff = 0;
+ nfloats = 0;
+ nints = 0;
+ vararg = 0;
+ nargs = countargs(fn->type);
+ for (i = 0; i < fn->nargs; i++) {
+ arg = fn->args[i];
+ argoff = align(argoff, min(size(arg), Ptrsz));
+ if (i >= nargs)
+ vararg = 1;
+ if (stacknode(arg)) {
+ htput(s->stkoff, arg, itop(-(argoff + 2*Ptrsz)));
+ argoff += size(arg);
+ } else if (!vararg && isfloatmode(mode(arg)) && nfloats < Nfloatregargs) {
+ a = coreg(floatargregs[nfloats], mode(arg));
+ l = locreg(mode(arg));
+ g(s, Imovs, a, l, NULL);
+ htput(s->reglocs, arg, l);
+ nfloats++;
+ } else if (!vararg && isintmode(mode(arg)) && nints < Nintregargs) {
+ a = coreg(intargregs[nints], mode(arg));
+ l = locreg(mode(arg));
+ g(s, Imov, a, l, NULL);
+ htput(s->reglocs, arg, l);
+ nints++;
+ } else {
+ htput(s->stkoff, arg, itop(-(argoff + 2*Ptrsz)));
+ argoff += size(arg);
+ }
+ }
+}
+
+static void prologue(Isel *s, Func *fn, size_t sz)
+{
Loc *rsp;
Loc *rbp;
Loc *stksz;
@@ -867,6 +962,7 @@
g(s, Imov, phys, s->calleesave[i], NULL);
}
}
+ addarglocs(s, fn);
s->nsaved = i;
s->stksz = stksz; /* need to update if we spill */
}
@@ -927,7 +1023,7 @@
lappend(&is->bb, &is->nbb, mkasmbb(fn->cfg->bb[i]));
is->curbb = is->bb[0];
- prologue(is, fn->stksz);
+ prologue(is, fn, fn->stksz);
for (j = 0; j < fn->cfg->nbb - 1; j++) {
is->curbb = is->bb[j];
if (!is->bb[j])
@@ -945,4 +1041,5 @@
is->curbb = is->bb[is->nbb - 1];
epilogue(is);
regalloc(is);
+ is->stksz->lit = align(is->stksz->lit, 16);
}
--- a/6/ra.c
+++ b/6/ra.c
@@ -919,6 +919,8 @@
{
if (s->nuses[r] == 0)
return -1;
+ if (bshas(s->neverspill, r))
+ return -10000;
return s->degree[r] * 100 / s->nuses[r];
}
@@ -953,7 +955,6 @@
if (!m) {
for (i = 0; i < s->nwlspill; i++) {
if (bshas(s->neverspill, s->wlspill[i]->reg.id)) {
- printf("Not spilling %zd\n", s->wlspill[i]->reg.id);
continue;
}
m = s->wlspill[i];
--- a/6/simp.c
+++ b/6/simp.c
@@ -47,10 +47,13 @@
/* location handling */
Node **blobs;
size_t nblobs;
- size_t stksz;
- size_t argsz;
Htab *globls;
- Htab *stkoff;
+
+ Htab *_stkoff;
+ size_t stksz;
+
+ Node **args;
+ size_t nargs;
};
static Node *simp(Simp *s, Node *n);
@@ -166,7 +169,7 @@
static int addressable(Simp *s, Node *a)
{
if (a->type == Ndecl || (a->type == Nexpr && exprop(a) == Ovar))
- return hthas(s->stkoff, a) || hthas(s->globls, a);
+ return hthas(s->_stkoff, a) || hthas(s->globls, a);
else
return stacknode(a);
}
@@ -209,7 +212,7 @@
dump(n, stdout);
printf("declared at %zd, size = %zd\n", s->stksz, size(n));
}
- htput(s->stkoff, n, itop(s->stksz));
+ htput(s->_stkoff, n, itop(s->stksz));
}
static void declarelocal(Simp *s, Node *n)
@@ -1565,13 +1568,7 @@
static void declarearg(Simp *s, Node *n)
{
assert(n->type == Ndecl || (n->type == Nexpr && exprop(n) == Ovar));
- s->argsz = align(s->argsz, min(size(n), Ptrsz));
- htput(s->stkoff, n, itop(-(s->argsz + 2*Ptrsz)));
- if (debugopt['i']) {
- dump(n, stdout);
- printf("declared at %zd\n", -(s->argsz + 2*Ptrsz));
- }
- s->argsz += size(n);
+ lappend(&s->args, &s->nargs, n);
}
static int islbl(Node *n)
@@ -1724,10 +1721,13 @@
fn = zalloc(sizeof(Func));
fn->name = strdup(name);
+ fn->type = dcl->decl.type;
fn->isexport = isexport(dcl);
fn->stksz = align(s->stksz, 8);
- fn->stkoff = s->stkoff;
+ fn->stkoff = s->_stkoff;
fn->ret = s->ret;
+ fn->args = s->args;
+ fn->nargs = s->nargs;
fn->cfg = cfg;
return fn;
}
@@ -1800,7 +1800,7 @@
if (ismain(dcl))
dcl->decl.vis = Vishidden;
- s.stkoff = mkht(varhash, vareq);
+ s._stkoff = mkht(varhash, vareq);
s.globls = globls;
s.blobs = *blob;
s.nblobs = *nblob;
--- a/libstd/syscall+freebsd-x64.s
+++ b/libstd/syscall+freebsd-x64.s
@@ -1,14 +1,5 @@
.globl sys$syscall
sys$syscall:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
/*
hack: We load 6 args regardless of
how many we actually have. This may
@@ -16,24 +7,15 @@
doesn't use them, it's going to be
harmless.
*/
- movq 80 (%rsp),%rax
- movq 88 (%rsp),%rdi
- movq 96 (%rsp),%rsi
- movq 104(%rsp),%rdx
- movq 112(%rsp),%r10
- movq 120(%rsp),%r8
- movq 128(%rsp),%r9
+ movq %rdi,%rax
+ movq %rsi,%rdi
+ movq %rdx,%rsi
+ movq %rcx,%rdx
+ movq %r8,%r10
+ movq %r9,%r8
+ movq 8(%rsp),%r9
syscall
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
--- a/libstd/syscall+linux-x64.s
+++ b/libstd/syscall+linux-x64.s
@@ -1,15 +1,5 @@
.globl sys$syscall
sys$syscall:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
-
/*
hack: We load 6 args regardless of
how many we actually have. This may
@@ -17,25 +7,16 @@
doesn't use them, it's going to be
harmless.
*/
- movq 80 (%rsp),%rax
- /* 88: hidden type arg */
- movq 96 (%rsp),%rdi
- movq 104(%rsp),%rsi
- movq 112(%rsp),%rdx
- movq 120(%rsp),%r10
- movq 128(%rsp),%r8
- movq 136(%rsp),%r9
+ movq %rdi,%rax
+ /* 8(%rsp): hidden type arg */
+ movq 16(%rsp),%rdi
+ movq 24(%rsp),%rsi
+ movq 32(%rsp),%rdx
+ movq 40(%rsp),%r10
+ movq 48(%rsp),%r8
+ movq 56(%rsp),%r9
syscall
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
--- a/libstd/syscall+osx-x64.s
+++ b/libstd/syscall+osx-x64.s
@@ -1,14 +1,6 @@
.globl _sys$syscall
_sys$syscall:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
+
/*
hack: We load 6 args regardless of
how many we actually have. This may
@@ -16,14 +8,14 @@
doesn't use them, it's going to be
harmless.
*/
- movq 80 (%rsp),%rax
- /* 88: hidden type arg */
- movq 96 (%rsp),%rdi
- movq 104(%rsp),%rsi
- movq 112(%rsp),%rdx
- movq 120(%rsp),%r10
- movq 128(%rsp),%r8
- movq 136(%rsp),%r9
+ movq %rdi,%rax
+ /* 8(%rsp): hidden type arg */
+ movq 16(%rsp),%rdi
+ movq 24(%rsp),%rsi
+ movq 32(%rsp),%rdx
+ movq 40(%rsp),%r10
+ movq 48(%rsp),%r8
+ movq 56(%rsp),%r9
syscall
jae .success
@@ -30,15 +22,6 @@
negq %rax
.success:
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
/*
@@ -51,16 +34,6 @@
*/
.globl _sys$__osx_fork
_sys$__osx_fork:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
-
movq $0x2000002,%rax
syscall
@@ -72,16 +45,6 @@
jz .isparent
xorq %rax,%rax
.isparent:
-
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
/*
@@ -91,16 +54,6 @@
*/
.globl _sys$__osx_pipe
_sys$__osx_pipe:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
-
movq $0x200002a,%rax
syscall
@@ -112,30 +65,10 @@
movl %eax,(%rdi)
movl %edx,4(%rdi)
xorq %rax,%rax
-
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
.globl _sys$__osx_lseek
_sys$__osx_lseek:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
-
movq $0x20000C7,%rax
syscall
@@ -146,31 +79,11 @@
movq 80(%rsp),%rdi
shlq $32,%rdx
orq %rdx,%rax
-
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
.globl _sys$__osx_gettimeofday
_sys$__osx_gettimeofday:
- pushq %rbp
- pushq %rdi
- pushq %rsi
- pushq %rdx
- pushq %r10
- pushq %r8
- pushq %r9
- pushq %rcx
- pushq %r11
-
movq $0x2000074,%rax
syscall
@@ -182,15 +95,5 @@
movq %rax, (%rdi)
movl %edx,8(%rdi)
xorq %rax,%rax
-
- popq %r11
- popq %rcx
- popq %r9
- popq %r8
- popq %r10
- popq %rdx
- popq %rsi
- popq %rdi
- popq %rbp
ret
--- a/libstd/util+posixy-x64.s
+++ b/libstd/util+posixy-x64.s
@@ -56,10 +56,9 @@
pushq %rbx
movq 8(%rbp),%r15 /* ret addr */
- movq 16(%rbp),%rbx /* len */
/* get stack space */
- subq %rbx,%rsp /* get stack space */
+ subq %rdi,%rsp /* get stack space */
movq %rsp,%rax /* top of stack (return value) */
subq $31,%rsp /* "unpop" the args for return */
andq $(~15),%rsp /* align */
--- a/libstd/varargs.myr
+++ b/libstd/varargs.myr
@@ -77,6 +77,9 @@
-> sl
}
+const inspectarg = {x
+}
+
generic vanext = {ap -> @a
var v : @a
var ti
@@ -87,6 +90,7 @@
/* apply the alignment to the arg pointer */
align = ti.align castto(intptr)
+ inspectarg(align)
p = ap.args castto(intptr)
p = (p + align - 1) & ~(align - 1)
ap.args = p castto(byte#)
--- a/mbld/deps.myr
+++ b/mbld/deps.myr
@@ -14,6 +14,7 @@
var usepat : regex.regex#
;;
+const Abiversion = 5
var usepat : regex.regex#
type dep = union
@@ -234,12 +235,13 @@
| `std.None: std.fatal("library {}: could not read usefile\n", lib)
;;
match bio.getbe32(f)
- | `std.Some 4: /* nothing: version matches. */
- | `std.Some 3: std.fput(1, "library {}: warning: old usefile version\n", lib)
- | `std.Some 2: std.fput(1, "library {}: warning: old usefile version\n", lib)
- | `std.Some 1: std.fput(1, "library {}: warning: old usefile version\n", lib)
- | `std.Some 0: std.fput(1, "library {}: warning: old usefile version\n", lib)
- | `std.Some _: std.fatal("library {}: usefile version unknown\n", lib)
+ | `std.Some Abiversion: /* nothing: version matches. */
+ | `std.Some v:
+ if v < Abiversion
+ std.fput(1, "library {}: warning: old abi version\n", lib)
+ else
+ std.fatal("library {}: usefile version unknown\n", lib)
+ ;;
| `std.None: std.fatal("library {}: corrutpt or invalid usefile\n", lib)
;;
std.slfree(rdstr(f))
--- a/muse/Makefile
+++ b/muse/Makefile
@@ -5,3 +5,4 @@
DEPS=../parse/libparse.a
include ../mk/c.mk
+include ../config.mk
--- a/muse/muse.c
+++ b/muse/muse.c
@@ -81,7 +81,7 @@
lappend(&incpaths, &nincpaths, Instroot "/lib/myr");
if (!outfile) {
- fprintf(stderr, "Output file needed when merging usefiles.");
+ fprintf(stderr, "output file needed when merging usefiles.\n");
exit(1);
}
--- a/parse/parse.h
+++ b/parse/parse.h
@@ -4,7 +4,7 @@
# define FATAL
#endif
-#define Abiversion 4
+#define Abiversion 5
typedef uint8_t byte;
typedef unsigned int uint;
--- a/test/runtest.sh
+++ b/test/runtest.sh
@@ -10,7 +10,7 @@
function build {
rm -f $1 $1.o $1.s $1.use
- ../mbld/mbld -b $1 -C../6/6m -M../muse/muse -I../libstd -r../rt/_myrrt.o $1.myr
+ mbld -b $1 -C../6/6m -M../muse/muse -I../libstd -r../rt/_myrrt.o $1.myr
}
function pass {