summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPeter Mikkelsen <peter@pmikkelsen.com>2021-07-07 16:32:02 +0000
committerPeter Mikkelsen <peter@pmikkelsen.com>2021-07-07 16:32:02 +0000
commit0f958749e189e4dacd7a1f70cfc33460e1228d3b (patch)
tree5679ea79743a736cc3179c29ea603ae0f39bb51e
parent73b9bad83e6405809e80809c10c8917f4bd4e341 (diff)
Make '=..'/2 work according to spec. Introduce types.c for functions which tells us something about term types. Should be used a lot more instead of explicitly looking into terms->tag everywhere
-rw-r--r--builtins.c17
-rw-r--r--fns.h10
-rw-r--r--mkfile3
-rw-r--r--types.c58
4 files changed, 86 insertions, 2 deletions
diff --git a/builtins.c b/builtins.c
index 35db61a..5751502 100644
--- a/builtins.c
+++ b/builtins.c
@@ -409,6 +409,23 @@ builtinuniv(Term *goal, Binding **bindings, Module *module)
Term *term = goal->children;
Term *list = term->next;
+ if(term->tag == VariableTerm && ispartiallist(list))
+ Throw(instantiationerror());
+ if(!(ispartiallist(list) || islist(list)))
+ Throw(typeerror(L"list", list));
+
+ Term *head = listhead(list);
+ Term *tail = listtail(list);
+
+ if(term->tag == VariableTerm && head->tag == VariableTerm)
+ Throw(instantiationerror());
+ if(islist(list) && !(head->tag == AtomTerm || head->tag == VariableTerm) && !isemptylist(tail))
+ Throw(typeerror(L"atom", head));
+ if(islist(list) && head->tag == CompoundTerm && isemptylist(tail))
+ Throw(typeerror(L"atomic", head));
+ if(term->tag == VariableTerm && isemptylist(list))
+ Throw(domainerror(L"non_empty_list", list));
+
int len;
if(term->tag == VariableTerm){
Rune *name;
diff --git a/fns.h b/fns.h
index 7255166..d548d6e 100644
--- a/fns.h
+++ b/fns.h
@@ -63,4 +63,12 @@ void writeterm(Term *, Term *, Term *);
/* module.c */
void initmodules(void);
Module *parsemodule(char *);
-Module *getmodule(Rune *); \ No newline at end of file
+Module *getmodule(Rune *);
+
+/* types.c */
+int islist(Term *);
+int ispartiallist(Term *t);
+int isemptylist(Term *);
+int isnonemptylist(Term *);
+Term *listhead(Term *);
+Term *listtail(Term *);
diff --git a/mkfile b/mkfile
index ef5d2b0..8f66c02 100644
--- a/mkfile
+++ b/mkfile
@@ -13,7 +13,8 @@ OFILES=\
flags.$O\
error.$O\
streams.$O\
- module.$O
+ module.$O\
+ types.$O\
HFILES=dat.h fns.h
diff --git a/types.c b/types.c
new file mode 100644
index 0000000..6f2b33d
--- /dev/null
+++ b/types.c
@@ -0,0 +1,58 @@
+#include <u.h>
+#include <libc.h>
+#include <bio.h>
+
+#include "dat.h"
+#include "fns.h"
+
+/* Type tests */
+int
+islist(Term *t)
+{
+ return (isemptylist(t) || isnonemptylist(t));
+}
+
+int
+ispartiallist(Term *t)
+{
+ if(t->tag == VariableTerm)
+ return 1;
+ else if(t->tag == CompoundTerm && runestrcmp(t->text, L".") == 0 && t->arity == 2)
+ return ispartiallist(listtail(t));
+ else
+ return 0;
+}
+
+int
+isemptylist(Term *t)
+{
+ return (t->tag == AtomTerm && runestrcmp(t->text, L"[]") == 0);
+}
+
+int
+isnonemptylist(Term *t)
+{
+ if(t->tag == CompoundTerm && runestrcmp(t->text, L".") == 0 && t->arity == 2)
+ return islist(listtail(t));
+ else
+ return 0;
+}
+
+/* Other functions */
+Term *
+listhead(Term *t)
+{
+ if(t->tag == CompoundTerm && runestrcmp(t->text, L".") == 0 && t->arity == 2)
+ return t->children;
+ else
+ return nil;
+}
+
+Term *
+listtail(Term *t)
+{
+ if(t->tag == CompoundTerm && runestrcmp(t->text, L".") == 0 && t->arity == 2)
+ return t->children->next;
+ else
+ return nil;
+} \ No newline at end of file