shithub: mc

Download patch

ref: 08b1eacc62abb4f77fd7403c692cdc4b84c7dd76
parent: 92d743d5e5eaf030196e52bdcc94b8c1735c1219
parent: 30234b14a10a94f21b6f84559fce135f43cb2e71
author: Ori Bernstein <[email protected]>
date: Thu Oct 4 10:51:44 EDT 2012

Merge branch 'master' of git+ssh://git.eigenstate.org/git/ori/mc

--- a/libstd/Makefile
+++ b/libstd/Makefile
@@ -12,7 +12,8 @@
 
 ASMSRC= \
     start.s \
-    syscall.s
+    syscall.s \
+    util.s
 
 include ../config.mk
 
--- a/libstd/sys-linux.myr
+++ b/libstd/sys-linux.myr
@@ -369,6 +369,7 @@
 	const Sysprocess_vm_writev	: scno = 311
 
 	extern const syscall : (sc:scno, args:... -> int64)
+	extern const cstring : (str : byte[:] -> byte*)
 
 	const exit	: (status:int64 -> void)
 	const getpid	: ( -> int64)
@@ -387,9 +388,9 @@
 const exit	= {status;		syscall(Sysexit, 1);}
 const getpid	= {;			-> syscall(Sysgetpid, 1);}
 const kill	= {pid, sig;		-> syscall(Syskill, pid, sig);}
-const open	= {path, opts, mode;	-> syscall(Sysopen, path castto(char*), opts);}
+const open	= {path, opts, mode;	-> syscall(Sysopen, cstring(path), opts, mode);}
 const close	= {fd;			-> syscall(Sysclose, fd);}
-const creat	= {path, mode;		-> syscall(Syscreat, path castto(char*), mode);}
+const creat	= {path, mode;		-> syscall(Syscreat, cstring(path), mode);}
 const read	= {fd, buf;		-> syscall(Sysread, fd, buf castto(char*), buf.len);}
 const write	= {fd, buf;		-> syscall(Syswrite, fd, buf castto(char*), buf.len castto(size));}
 const lseek	= {fd, off, whence;	-> syscall(Syslseek, fd, off, whence);}
--- a/libstd/sys-osx.myr
+++ b/libstd/sys-osx.myr
@@ -415,6 +415,7 @@
 	const Sysfileport_makefd	: scno = 0x20001b1
 
 	extern const syscall : (sc:scno, args:... -> int64)
+	extern const cstring : (str : byte[:] -> byte*)
 
 	const exit	: (status:int64 -> void)
 	const getpid	: ( -> int64)
@@ -433,7 +434,7 @@
 const exit	= {status;		syscall(Sysexit, 1);}
 const getpid	= {;			-> syscall(Sysgetpid, 1);}
 const kill	= {pid, sig;		-> syscall(Syskill, pid, sig);}
-const open	= {path, opts, mode;	-> syscall(Sysopen, path castto(char*), opts);}
+const open	= {path, opts, mode;	-> syscall(Sysopen, cstring(path), opts, mode);}
 const close	= {fd;			-> syscall(Sysclose, fd);}
 const creat	= {path, mode;		-> open(path, Ocreat | Otrunc | Owronly, mode);}
 const read	= {fd, buf;		-> syscall(Sysread, fd, buf castto(char*), buf.len);}
--- a/libstd/syscall-linux.s
+++ b/libstd/syscall-linux.s
@@ -20,3 +20,4 @@
 
 	popq %rbp
 	ret
+
--- /dev/null
+++ b/libstd/util.s
@@ -1,0 +1,29 @@
+/*
+ * Allocates a C string on the stack, for
+ * use within system calls, which is the only
+ * place the Myrddin stack should need nul-terminated
+ * strings.
+ *
+ * This is in assembly, because for efficiency we
+ * allocate the C strings on the stack, and don't adjust
+ * %rsp when returning.
+ */
+.globl std$cstring
+std$cstring:
+	movq (%rsp),%r15	/* ret addr */
+	movq 8(%rsp),%rsi	/* src */
+        movq %rsp,%rdi          /* dest */
+	movq %rsp,%rax          /* ret val */
+	movq 16(%rsp),%rcx	/* len */
+
+	subq %rcx,%rsp          /* get stack */
+        subq $1,%rsp            /* nul */
+        andq $(~15),%rsp        /* align */
+
+	cld
+	rep movsb
+        movb $0,(%rdi)          /* terminate */
+
+	pushq %r15              /* ret addr */
+	ret
+
--- a/myrbuild/myrbuild.c
+++ b/myrbuild/myrbuild.c
@@ -162,7 +162,7 @@
     char *localdep;
     char *deps[512];
     char buf[1024];
-    char *extra[2] = {"-o", "" /* filename */};
+    char *extra[] = {"-g", "-o", "" /* filename */};
 
     if (hassuffix(file, ".myr")) {
         getdeps(file, deps, 512, &ndeps);
@@ -189,8 +189,8 @@
     } else if (hassuffix(file, ".s")) {
         swapsuffix(buf, sizeof buf, file, ".s", ".o");
         if (isfresh(file, buf)) {
-            extra[1] = buf;
-            gencmd(&cmd, &ncmd, as, file, extra, 2);
+            extra[2] = buf;
+            gencmd(&cmd, &ncmd, as, file, extra, 3);
             run(cmd);
         }
     }
--- a/parse/dump.c
+++ b/parse/dump.c
@@ -132,7 +132,7 @@
                 outnode(n->file.stmts[i], fd, depth + 1);
             break;
         case Ndecl:
-            fprintf(fd, "(did = %zd, isconst = %d, isgeneric = %d, isextern = %d\n, isexport = %d)",
+            fprintf(fd, "(did = %zd, isconst = %d, isgeneric = %d, isextern = %d, isexport = %d)\n",
                     n->decl.did, n->decl.isconst, n->decl.isgeneric, n->decl.isextern, n->decl.isexport);
             outsym(n, fd, depth + 1);
             outnode(n->decl.init, fd, depth + 1);
--- a/parse/infer.c
+++ b/parse/infer.c
@@ -130,12 +130,29 @@
         ret = mktyvar(t->line);
         htput(ht, t->pname, ret);
         return ret;
+    } else if (t->type == Tygeneric) {
+        for (i = 0; i < t->nparam; i++)
+            if (!hthas(ht, t->param[i]->pname))
+                htput(ht, t->param[i]->pname, mktyvar(t->param[i]->line));
+        return mktyname(t->line, t->name, tyfreshen(st, ht, t->sub[0]));
+    } else {
+        ret = tydup(t);
+        for (i = 0; i < t->nsub; i++)
+            ret->sub[i] = tyfreshen(st, ht, t->sub[i]);
+        return ret;
     }
+}
 
-    ret = tydup(t);
-    for (i = 0; i < t->nsub; i++)
-        ret->sub[i] = tyfreshen(st, ht, t->sub[i]);
-    return ret;
+static Type *tyspecialize(Inferstate *st, Type *t)
+{
+    Htab *ht;
+
+    assert(t->type == Tygeneric);
+    ht = mkht(strhash, streq);
+    t = tyfreshen(st, ht, t);
+    htfree(ht);
+
+    return t;
 }
 
 /* Freshens the type of a declaration. */
@@ -484,11 +501,17 @@
         b = t;
     }
 
+    if (a->type == Tygeneric)
+        a = tyspecialize(st, a);
+    if (b->type == Tygeneric)
+        b = tyspecialize(st, b);
+
     r = NULL;
     if (a->type == Tyvar) {
         tytab[a->tid] = b;
         r = b;
     }
+
     /* Disallow recursive types */
     if (a->type == Tyvar && b->type != Tyvar) 
         if (occurs(a, b))
--- a/parse/type.c
+++ b/parse/type.c
@@ -53,12 +53,24 @@
 
     r = mktype(t->line, t->type);
     r->resolved = 0; /* re-resolving doesn't hurt */
+    r->fixed = 0; /* re-resolving doesn't hurt */
+
     r->cstrs = bsdup(t->cstrs);
+    r->cstrlist = memdup(t->cstrlist, t->ncstrlist * sizeof(Node*));
+    r->ncstrlist = t->ncstrlist;
+
+    r->param = memdup(t->param, t->nparam * sizeof(Type*));
+    r->nparam = t->nparam;
+    r->inst = memdup(t->param, t->nparam * sizeof(Type*));
+    r->ninst = t->ninst;
+
+    r->sub = memdup(t->sub, t->nsub * sizeof(Type*));
     r->nsub = t->nsub;
     r->nmemb = t->nmemb;
-    r->sub = memdup(t->sub, t->nsub * sizeof(Type*));
     switch (t->type) {
-        case Tyunres:    r->name = t->name;              break;
+        case Tygeneric:   r->name = t->name;              break;
+        case Tyname:   r->name = t->name;              break;
+        case Tyunres:   r->name = t->name;              break;
         case Tyarray:   r->asize = t->asize;            break;
         case Typaram:   r->pname = strdup(t->pname);    break;
         case Tystruct:  r->sdecls = memdup(t->sdecls, t->nmemb*sizeof(Node*));   break;