Merge remote-tracking branch 'upstream/master' into d6/binary
This commit is contained in:
commit
52ae7406b5
|
@ -14,6 +14,9 @@ cc src/modal.c -o bin/modal
|
||||||
|
|
||||||
```
|
```
|
||||||
bin/modal examples/hello.modal
|
bin/modal examples/hello.modal
|
||||||
|
-v Print version
|
||||||
|
-q Quiet mode, no step printing
|
||||||
|
-n Infinite mode, no rewrites limit
|
||||||
```
|
```
|
||||||
|
|
||||||
## Credits
|
## Credits
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
?(?-) (This example demonstrates how to keep the runtime active between prompts.)
|
|
||||||
|
|
||||||
<> ((You said: quit\n) send) ((You quit.) print ')
|
|
||||||
<> (?: print ') (?:)
|
|
||||||
<> (?: send) (?: wait stdin)
|
|
||||||
<> (wait ?~) ((You said: ?~\n) send)
|
|
||||||
<> (' ?x) (?x ')
|
|
||||||
|
|
||||||
(Say something, or type "quit": \n) print '
|
|
||||||
|
|
||||||
wait stdin
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
|
||||||
|
<> (what is ?x) (is ?x a (programming language))
|
||||||
|
<> (is ?x a ?y) (or, is ?x (a virtual machine))
|
||||||
|
<> (or, is ?x ?y) (?x is ?y <> ?y (a meta-language))
|
||||||
|
<> ((a meta-language)) modal
|
||||||
|
<> (?x is ?x) (?(?: ?:) ?x)
|
||||||
|
|
||||||
|
what is modal
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
?(?-) (This is a fully functional Modal REPL.)
|
||||||
|
|
||||||
|
<> quit EOF
|
||||||
|
<> wait?~ (?~ wait!)
|
||||||
|
<> (?: print) ?:
|
||||||
|
<> (EOF wait!) (Bye.\n print)
|
||||||
|
|
||||||
|
(Say something, or type "quit":\n) print wait!
|
|
@ -0,0 +1,21 @@
|
||||||
|
?(?-) (Rules)
|
||||||
|
|
||||||
|
<> (* (. > (. ?x))) (* (. (. > ?x)))
|
||||||
|
<> (. (. > (* ?x))) (* (. (* > ?x)))
|
||||||
|
|
||||||
|
?(?-) (Physics)
|
||||||
|
|
||||||
|
<> (Tri > (?x ?y)) (Tri (?x > ?y))
|
||||||
|
<> (Tri (?x > (?y ?z))) (Tri (?x (?y > ?z)))
|
||||||
|
<> (?x (?y > (?z ?n))) (. (?y (?z > ?n)))
|
||||||
|
<> ((?x > ())) (< ())
|
||||||
|
<> (Tri < (* ?^)) (?(?: ?:) (*?^ \n))
|
||||||
|
<> ((?x < ?y)) (< (?x ?y))
|
||||||
|
|
||||||
|
?(?-) (Print)
|
||||||
|
|
||||||
|
<> (Tri.join ?x ?:) (Tri > ?x ?:)
|
||||||
|
<> (Tri.dup ?x ?^) (Tri.join ?x ?^)
|
||||||
|
<> (Tri < ?x) (Tri.dup (. ?x) (?x \n))
|
||||||
|
|
||||||
|
?(?* (Tri < (?*))) ...............*...............
|
|
@ -1,74 +1,82 @@
|
||||||
?(?-) (This example tests various aspects of the implementation.)
|
?(?-) (This example tests various aspects of the implementation.)
|
||||||
|
|
||||||
?(?-) (Inline rules)
|
?(?-) (Early Test Primitives)
|
||||||
|
|
||||||
<> ((?x -> ?y)) (<> ?x ?y)
|
<> (?x = ?x ?n test) (?(?: ?:) (#pass ?n\n))
|
||||||
(nap -> (tap =))
|
|
||||||
|
|
||||||
nap tap test
|
<> (ghost) ()
|
||||||
|
|
||||||
|
?(?-) (Formatter)
|
||||||
|
|
||||||
|
?((?x ?y) one) aaa(bbb) = one (formatter 1) test
|
||||||
|
?((?x ?y) one) (bbb)aaa = one (formatter 2) test
|
||||||
|
(a b c ) = (a b c) (formatter 3) test
|
||||||
|
( a b c) = (a b c) (formatter 4) test
|
||||||
|
( a b c ) = (a b c) (formatter 5) test
|
||||||
|
(a b c ( a b c ) ) = (a b c (a b c)) (formatter 6) test
|
||||||
|
|
||||||
?(?-) (Empty replacements)
|
?(?-) (Empty replacements)
|
||||||
|
|
||||||
<> (?x pop-plain)
|
(ab cd () ghost) = (ab cd ()) (empty 1) test
|
||||||
<> (?x pop) ()
|
(ab cd ghost ()) = (ab cd ()) (empty 2) test
|
||||||
<> (ghost) ()
|
(ab ghost cd ()) = (ab cd ()) (empty 3) test
|
||||||
|
(ghost ab cd ()) = (ab cd ()) (empty 4) test
|
||||||
|
(ghost) = () (empty 5) test
|
||||||
|
|
||||||
abc def pop-plain = abc test
|
(q ?((?x ?y ?z) (?x ?y)) a b c) = (q a b) (empty 6) test
|
||||||
abc def pop = abc test
|
(q ?((?x ?y ?z) (?x ?z)) a b c) = (q a c) (empty 7) test
|
||||||
(ghost) = () test
|
(q ?((?x ?y ?z) (?y ?z)) a b c) = (q b c) (empty 8) test
|
||||||
|
|
||||||
?(?-) (Basic replacements)
|
<> (prefix/pop ?x) ()
|
||||||
|
<> (?x suffix/pop) ()
|
||||||
|
|
||||||
<> (replace-name (foo)) ((bar) =)
|
(ab prefix/pop cd ef) = (ab ef) (empty 9) test
|
||||||
|
(ab cd suffix/pop ef) = (ab ef) (empty 10) test
|
||||||
replace-name (foo) (bar) test
|
|
||||||
|
|
||||||
?(?-) (Basic register setups)
|
?(?-) (Basic register setups)
|
||||||
|
|
||||||
<> (dup (?x)) ((?x ?x) =)
|
<> (dup (?x)) (?x ?x)
|
||||||
<> (swap (?x ?y)) ((?y ?x) =)
|
<> (swap (?x ?y)) (?y ?x)
|
||||||
<> (compare (?x ?x ?x)) ((#t) =)
|
<> (compare (?x ?x ?x)) (#t)
|
||||||
|
<> (rotate (?x (?y (?z)))) (?y (?z (?x)))
|
||||||
|
<> (unused ?x) ?y
|
||||||
|
|
||||||
dup (abc) (abc abc) test
|
(dup (abc)) = (abc abc) (basic 1) test
|
||||||
swap (abc def) (def abc) test
|
(swap (abc def)) = (def abc) (basic 2) test
|
||||||
compare (abc abc abc) (#t) test
|
(compare (abc abc abc)) = (#t) (basic 3) test
|
||||||
|
(rotate (abc (def (ghi)))) = (def (ghi (abc))) (basic 4) test
|
||||||
|
(unused hey) = (?y) (basic 5) test
|
||||||
|
|
||||||
?(?-) (Empty register replacement)
|
?(?-) (Substring registers)
|
||||||
|
|
||||||
<> (replace-empty ?x) (?y)
|
<> (connect ?x ?y ?z) (?x-?y?z)
|
||||||
|
<> (prefix-?x) (?x-suffix)
|
||||||
|
|
||||||
replace-empty abc = ?y test
|
connect foo bar baz = foo-barbaz (substring 1) test
|
||||||
|
prefix-anything = anything-suffix (substring 2) test
|
||||||
?(?-) (Guards setups)
|
?(foo QQQ) foobar = foobar (substring 3) test
|
||||||
|
|
||||||
<> (join (String ?x) (String ?y)) ((?x ?y) =)
|
|
||||||
|
|
||||||
join (String abc) (String def) (abc def) test
|
|
||||||
|
|
||||||
?(?-) (Lambdas)
|
?(?-) (Lambdas)
|
||||||
|
|
||||||
?((?x) ((?x ?x) =)) abc (abc abc) test
|
(?(?x (?x ?x)) abc) = (abc abc) (lambda 1) test
|
||||||
abc ?(?x) def = abc test
|
abc ?(?x) def = abc (lambda 2) test
|
||||||
|
|
||||||
?(?-) (Explode)
|
?(?-) (op: explode)
|
||||||
|
|
||||||
<> (explode ?*) ((?*) =)
|
(?(?* ?*) cow) = (c (o (w ()))) (explode word) test
|
||||||
|
(?(?* ?*) (12 34 45)) = (12 (34 (45 ()))) (explode tuple) test
|
||||||
|
(?(?* ?*) ()) = () (explode empty) test
|
||||||
|
|
||||||
explode cow (c (o (w ()))) test
|
?(?-) (op: join)
|
||||||
explode (12 34 45) (12 (34 (45 ()))) test
|
|
||||||
|
|
||||||
?(?-) (Implode)
|
?(?^ ?^) (b (a (t ()))) = bat (join 1) test
|
||||||
|
?(?^ ?^) (12 (34 (56 ()))) = 123456 (join 2) test
|
||||||
|
(?(?^ ?^) ()) = () (join empty) test
|
||||||
|
|
||||||
<> (implode ?^) (?^ =)
|
?(?-) (op: unwrap)
|
||||||
|
|
||||||
implode (b (a (t ()))) bat test
|
?(?. ?.) (abcd) = abcd (unwrap 1) test
|
||||||
implode (12 (34 (56 ()))) 123456 test
|
(?(?. ?.) ()) = () (unwrap empty) test
|
||||||
|
|
||||||
?(?-) (Test Primitives)
|
|
||||||
|
|
||||||
<> (?: print) (?:)
|
|
||||||
<> (?x = ?x test) (#ok)
|
|
||||||
<> (?x = ?y test) (#fail)
|
|
||||||
|
|
||||||
?(?-) (List reversal)
|
?(?-) (List reversal)
|
||||||
|
|
||||||
|
@ -76,5 +84,15 @@ implode (12 (34 (56 ()))) 123456 test
|
||||||
<> (reverse (?*)) (reverse List (?*) ())
|
<> (reverse (?*)) (reverse List (?*) ())
|
||||||
<> (reverse List (?x ?y) ?z) (reverse List ?y (?x ?z))
|
<> (reverse List (?x ?y) ?z) (reverse List ?y (?x ?z))
|
||||||
|
|
||||||
reverse (modal) = ladom test
|
reverse (modal) = ladom (reverse 1) test
|
||||||
|
|
||||||
|
?(?-) (Inline rules)
|
||||||
|
|
||||||
|
<> ((?x -> ?y)) (<> ?x ?y)
|
||||||
|
(nap -> (tap =))
|
||||||
|
|
||||||
|
nap tap (inline 1) test
|
||||||
|
|
||||||
|
?(?-) (Late Test Primitives)
|
||||||
|
|
||||||
|
<> (?x = ?y ?n test) (?(?: ?:) (#fail ?n found: ?x expect: ?y\n))
|
10
makefile
10
makefile
|
@ -8,12 +8,14 @@ all: dest
|
||||||
dest:
|
dest:
|
||||||
@ mkdir -p bin
|
@ mkdir -p bin
|
||||||
run: all bin/modal
|
run: all bin/modal
|
||||||
@ bin/modal examples/hello.modal 2> /dev/null
|
@ bin/modal -q examples/hello.modal
|
||||||
debug: bin/modal-debug
|
debug: all bin/modal-debug
|
||||||
@ bin/modal-debug examples/hello.modal
|
@ bin/modal-debug examples/hello.modal
|
||||||
test: bin/modal-debug bin/modal
|
test: all bin/modal-debug bin/modal
|
||||||
@ bin/modal -v
|
@ bin/modal -v
|
||||||
@ bin/modal-debug examples/tests.modal
|
@ bin/modal-debug -q examples/fizzbuzz.modal
|
||||||
|
@ bin/modal-debug -q examples/sierpinski.modal
|
||||||
|
@ bin/modal-debug -q examples/tests.modal
|
||||||
install: bin/modal
|
install: bin/modal
|
||||||
cp bin/modal ~/bin/
|
cp bin/modal ~/bin/
|
||||||
uninstall:
|
uninstall:
|
||||||
|
|
306
src/modal.c
306
src/modal.c
|
@ -1,16 +1,15 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
int id;
|
unsigned int id, refs, ptr;
|
||||||
char *a, *b;
|
char *a, *b, reg[0x10];
|
||||||
} Rule;
|
} Rule;
|
||||||
|
|
||||||
static int flip, rmin = 0xff, rmax = 0x00, cycles = 0x10000;
|
static int flip, quiet, cycles = 0x10000;
|
||||||
static Rule rules[0x1000], lambda, *rules_ = rules;
|
static Rule rules[0x1000], *rules_ = rules, lambda;
|
||||||
static char dict[0x8000], *dict_ = dict;
|
static char dict[0x8000], *dict_ = dict, empty;
|
||||||
static char bank_a[0x4000], *src_ = bank_a;
|
static char bank_a[0x4000], *src_ = bank_a;
|
||||||
static char bank_b[0x4000], *dst_ = bank_b;
|
static char bank_b[0x4000], *dst_ = bank_b;
|
||||||
static char *regs[0x100];
|
|
||||||
|
|
||||||
#define spacer(c) (c <= ' ' || c == '(' || c == ')')
|
#define spacer(c) (c <= ' ' || c == '(' || c == ')')
|
||||||
|
|
||||||
|
@ -30,62 +29,17 @@ walk(char *s)
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
|
||||||
set_reg(int r, char *b)
|
|
||||||
{
|
|
||||||
if(regs[r]) {
|
|
||||||
char *a = regs[r], *aa = walk(a), *bb = walk(b);
|
|
||||||
while(a < aa && b < bb)
|
|
||||||
if(*a++ != *b++) return 0;
|
|
||||||
} else {
|
|
||||||
regs[r] = b;
|
|
||||||
if(r < rmin) rmin = r;
|
|
||||||
if(r > rmax) rmax = r;
|
|
||||||
}
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
put_reg(char r)
|
write_reg(char r, char *reg)
|
||||||
{
|
{
|
||||||
char c, *s = regs[(int)r], *ss;
|
char c, *cap = walk(reg);
|
||||||
if(!s) {
|
switch(r) {
|
||||||
*dst_++ = '?', *dst_++ = r;
|
case ':': /* op: output */
|
||||||
return;
|
if(*reg == '(') reg++, --cap;
|
||||||
}
|
while(reg < cap) {
|
||||||
ss = walk(s);
|
c = *reg++;
|
||||||
if(r == '*') {
|
|
||||||
int i, depth = 0;
|
|
||||||
if(*s == '(') { /* special explode tuple */
|
|
||||||
s++;
|
|
||||||
while(s < ss) {
|
|
||||||
while((c = *s) && !spacer(c))
|
|
||||||
*dst_++ = c, s++;
|
|
||||||
*dst_++ = ' ';
|
|
||||||
*dst_++ = '(', s++, depth++;
|
|
||||||
}
|
|
||||||
} else { /* special explode token */
|
|
||||||
while((c = *s++) && !spacer(c))
|
|
||||||
*dst_++ = c, *dst_++ = ' ', *dst_++ = '(', depth++;
|
|
||||||
}
|
|
||||||
for(i = 0; i < depth; i++)
|
|
||||||
*dst_++ = ')';
|
|
||||||
} else if(r == '.') { /* special unpack */
|
|
||||||
if(*s == '(') s++, --ss;
|
|
||||||
while(s < ss) *dst_++ = *s++;
|
|
||||||
} else if(r == '^') { /* special join */
|
|
||||||
if(*s == '(') s++, --ss;
|
|
||||||
while(s < ss && (c = *s++))
|
|
||||||
if(!spacer(c)) *dst_++ = c;
|
|
||||||
} else if(r == '~') { /* special stdin */
|
|
||||||
while(fread(&c, 1, 1, stdin) && c >= ' ')
|
|
||||||
*dst_++ = c;
|
|
||||||
} else if(r == ':') { /* special stdout */
|
|
||||||
if(*s == '(') s++, --ss;
|
|
||||||
while(s < ss) {
|
|
||||||
c = *s++;
|
|
||||||
if(c == '\\') {
|
if(c == '\\') {
|
||||||
switch(*s++) {
|
switch(*reg++) {
|
||||||
case 't': putc(0x09, stdout); break;
|
case 't': putc(0x09, stdout); break;
|
||||||
case 'n': putc(0x0a, stdout); break;
|
case 'n': putc(0x0a, stdout); break;
|
||||||
case 's': putc(0x20, stdout); break;
|
case 's': putc(0x20, stdout); break;
|
||||||
|
@ -93,117 +47,178 @@ put_reg(char r)
|
||||||
} else
|
} else
|
||||||
putc(c, stdout);
|
putc(c, stdout);
|
||||||
}
|
}
|
||||||
} else
|
return;
|
||||||
while(s < ss) *dst_++ = *s++;
|
case '~': { /* op: input */
|
||||||
|
while(fread(&c, 1, 1, stdin) && c >= ' ')
|
||||||
|
*dst_++ = c;
|
||||||
|
if(feof(stdin))
|
||||||
|
*dst_++ = 'E', *dst_++ = 'O', *dst_++ = 'F';
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
case '^': /* op: join */
|
||||||
static char *
|
if(*reg == '(') reg++, --cap;
|
||||||
match_rule(Rule *r, char *p)
|
while(reg < cap && (c = *reg++))
|
||||||
{
|
if(!spacer(c)) *dst_++ = c;
|
||||||
int i;
|
return;
|
||||||
char c, last = 0, *a = r->a, *b = p;
|
case '.': /* op: unwrap */
|
||||||
if(rmax) {
|
if(*reg == '(') reg++, --cap;
|
||||||
for(i = rmin; i <= rmax; i++)
|
while(reg < cap) *dst_++ = *reg++;
|
||||||
regs[i] = 0;
|
return;
|
||||||
rmin = 0xff, rmax = 0x00;
|
case '*': { /* op: explode */
|
||||||
|
int i, depth = 0;
|
||||||
|
if(*reg == '(' && reg[1] != ')') { /* tuple */
|
||||||
|
reg++;
|
||||||
|
while(reg < cap) {
|
||||||
|
while((c = *reg) && !spacer(c))
|
||||||
|
*dst_++ = c, reg++;
|
||||||
|
*dst_++ = ' ';
|
||||||
|
*dst_++ = '(', reg++, depth++;
|
||||||
}
|
}
|
||||||
while((c = *a)) {
|
} else /* token */
|
||||||
if(spacer(last) && c == '?') {
|
while((c = *reg++) && !spacer(c))
|
||||||
if(!set_reg(*(++a), b)) return NULL;
|
*dst_++ = c, *dst_++ = ' ', *dst_++ = '(', depth++;
|
||||||
a++, b = walk(b);
|
for(i = 0; i < depth; i++) *dst_++ = ')';
|
||||||
continue;
|
return;
|
||||||
}
|
}
|
||||||
if(c != *b) return NULL;
|
default:
|
||||||
a++, b++, last = c;
|
while(reg < cap) *dst_++ = *reg++;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
c = *b;
|
|
||||||
return spacer(c) ? b : NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
commit_rule(Rule *r, char *s, int create)
|
write_rule(Rule *r, char *s, int create)
|
||||||
{
|
{
|
||||||
while((*dst_++ = *s++))
|
while((*dst_++ = *s++))
|
||||||
;
|
;
|
||||||
*dst_++ = 0;
|
*dst_ = 0;
|
||||||
if((flip = !flip))
|
if((flip = !flip))
|
||||||
src_ = bank_b, dst_ = bank_a;
|
src_ = bank_b, dst_ = bank_a;
|
||||||
else
|
else
|
||||||
src_ = bank_a, dst_ = bank_b;
|
src_ = bank_a, dst_ = bank_b;
|
||||||
|
if(!quiet) {
|
||||||
if(create)
|
if(create)
|
||||||
fprintf(stderr, "<> (%s) (%s)\n", r->a, r->b);
|
fprintf(stderr, "<> (%s) (%s)\n", r->a, r->b);
|
||||||
else
|
else
|
||||||
fprintf(stderr, "%02d %s\n", r->id, src_);
|
fprintf(stderr, "%02d %s\n", r->id, src_), ++r->refs;
|
||||||
|
}
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
write_rule(Rule *r, char last, char *res)
|
apply_rule(Rule *r, char *s)
|
||||||
{
|
{
|
||||||
char c, *b = r->b, *origin = dst_;
|
unsigned int i, id;
|
||||||
while((c = *b++))
|
char c, *a = r->a, *b = r->b, *origin = dst_, *reg, *regs[0x08];
|
||||||
if(spacer(last) && c == '?')
|
/* phase: clean registers */
|
||||||
put_reg(*b++);
|
for(i = 0; i < r->ptr; i++)
|
||||||
|
regs[i] = NULL;
|
||||||
|
/* phase: match rule */
|
||||||
|
while((c = *a++)) {
|
||||||
|
if(c == '?') {
|
||||||
|
char *pcap = walk(s);
|
||||||
|
id = *a++ - '0';
|
||||||
|
if((reg = regs[id])) { /* reg cmp */
|
||||||
|
char *rcap = walk(reg), *pp = s;
|
||||||
|
while(reg < rcap || pp < pcap)
|
||||||
|
if(*reg++ != *pp++) return 0;
|
||||||
|
} else /* reg set */
|
||||||
|
regs[id] = s;
|
||||||
|
s = pcap;
|
||||||
|
} else if(c != *s++)
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
c = *s;
|
||||||
|
if(!spacer(c)) return 0;
|
||||||
|
/* phase: write rule */
|
||||||
|
while((c = *b++)) {
|
||||||
|
if(c == '?') {
|
||||||
|
id = *b - '0';
|
||||||
|
if(id < 9 && (reg = regs[id]))
|
||||||
|
b++, write_reg(r->reg[id], reg);
|
||||||
else
|
else
|
||||||
*dst_++ = c, last = c;
|
*dst_++ = c;
|
||||||
if(dst_ == origin) {
|
|
||||||
while(*res == ' ') res++;
|
|
||||||
if(*res == ')' && *(dst_ - 1) == ' ') dst_--;
|
|
||||||
}
|
|
||||||
return commit_rule(r, res, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static char *
|
|
||||||
parse_frag(char *s)
|
|
||||||
{
|
|
||||||
char c, *ss;
|
|
||||||
while((c = *s) && c <= ' ') s++;
|
|
||||||
if(*s != ')' && !(*s == '<' && s[1] == '>')) {
|
|
||||||
ss = walk(s);
|
|
||||||
if(*s == '(') {
|
|
||||||
s++;
|
|
||||||
while(s < ss - 1) *dict_++ = *s++;
|
|
||||||
s++;
|
|
||||||
} else
|
} else
|
||||||
while(s < ss) *dict_++ = *s++;
|
*dst_++ = c;
|
||||||
}
|
}
|
||||||
*dict_++ = 0;
|
if(dst_ == origin) {
|
||||||
return s;
|
while(*s == ' ') s++;
|
||||||
|
if(*s == ')' && *(dst_ - 1) == ' ') dst_--;
|
||||||
|
}
|
||||||
|
return write_rule(r, s, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
find_register(Rule *r, char reg)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
for(i = 0; i < (int)r->ptr; i++)
|
||||||
|
if(r->reg[i] == reg) return i;
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static char *
|
static char *
|
||||||
create_rule(Rule *r, int id, char *s)
|
compile_rule(Rule *r, int id, char *src)
|
||||||
{
|
{
|
||||||
r->id = id, s += 2;
|
char c, *cap;
|
||||||
r->a = dict_, s = parse_frag(s);
|
int wrapped, reg;
|
||||||
r->b = dict_, s = parse_frag(s);
|
r->id = id, r->ptr = 0, r->a = &empty, r->b = ∅
|
||||||
return s;
|
/* phase: compile left */
|
||||||
|
while((c = *src) && c == ' ') src++;
|
||||||
|
if(c == ')' || (c == '<' && src[1] == '>')) return src;
|
||||||
|
r->a = dict_, cap = walk(src), wrapped = c == '(';
|
||||||
|
if(wrapped) src++, cap--;
|
||||||
|
while(src < cap) {
|
||||||
|
c = *src, *dict_++ = *src++;
|
||||||
|
if(c == '?') {
|
||||||
|
c = *src++, reg = find_register(r, c);
|
||||||
|
if(reg == -1 && c != '(')
|
||||||
|
r->reg[r->ptr] = c, reg = r->ptr++;
|
||||||
|
*dict_++ = '0' + reg;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
src += wrapped, *dict_++ = 0;
|
||||||
|
/* phase: compile right */
|
||||||
|
while((c = *src) && c == ' ') src++;
|
||||||
|
if(c == ')' || (c == '<' && src[1] == '>')) return src;
|
||||||
|
r->b = dict_, cap = walk(src), wrapped = c == '(';
|
||||||
|
if(wrapped) src++, cap--;
|
||||||
|
while(src < cap) {
|
||||||
|
c = *src, *dict_++ = *src++;
|
||||||
|
if(c == '?') {
|
||||||
|
c = *src++, reg = find_register(r, c);
|
||||||
|
*dict_++ = reg != -1 ? '0' + reg : c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
src += wrapped, *dict_++ = 0;
|
||||||
|
return src;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
rewrite(void)
|
rewrite(void)
|
||||||
{
|
{
|
||||||
char c, last = 0, *cap, *s = src_, *res;
|
char c, last = 0, *cap, *s = src_;
|
||||||
while((c = *s) && c <= ' ') s++;
|
while(*s == ' ') s++;
|
||||||
while((c = *s)) {
|
while((c = *s)) {
|
||||||
if(spacer(last)) {
|
if(c == '(' || spacer(last)) {
|
||||||
Rule *r;
|
Rule *r = NULL;
|
||||||
if(*s == '<' && s[1] == '>') {
|
/* phase: rule */
|
||||||
|
if(c == '<' && s[1] == '>') {
|
||||||
r = rules_++;
|
r = rules_++;
|
||||||
s = create_rule(r, rules_ - rules - 1, s);
|
s = compile_rule(r, rules_ - rules - 1, s + 2);
|
||||||
while((c = *s) && c <= ' ') s++;
|
while(*s == ' ') s++;
|
||||||
return commit_rule(r, s, 1);
|
return write_rule(r, s, 1);
|
||||||
}
|
}
|
||||||
if(*s == '?' && s[1] == '(') {
|
/* phase: lambda */
|
||||||
r = &lambda, cap = walk(s + 1);
|
if(c == '?' && s[1] == '(') {
|
||||||
create_rule(&lambda, -1, s), s = cap;
|
cap = walk(s + 1), compile_rule(&lambda, -1, s + 2), s = cap;
|
||||||
while((c = *s) && c <= ' ') s++;
|
while(*s == ' ') s++;
|
||||||
if((res = match_rule(&lambda, s)) != NULL)
|
if(!apply_rule(&lambda, s)) write_rule(&lambda, s, 0);
|
||||||
return write_rule(&lambda, last, res);
|
return 1;
|
||||||
}
|
}
|
||||||
|
/* phase: match */
|
||||||
for(r = rules; r < rules_; r++)
|
for(r = rules; r < rules_; r++)
|
||||||
if((res = match_rule(r, s)) != NULL)
|
if(apply_rule(r, s)) return 1;
|
||||||
return write_rule(r, last, res);
|
|
||||||
}
|
}
|
||||||
*dst_++ = last = c;
|
*dst_++ = last = c;
|
||||||
s++;
|
s++;
|
||||||
|
@ -216,31 +231,38 @@ int
|
||||||
main(int argc, char **argv)
|
main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
FILE *f;
|
FILE *f;
|
||||||
int i;
|
int i, pl = 0, pr = 0;
|
||||||
char c, *w = bank_a;
|
char c, last = 0, *w = bank_a;
|
||||||
if(argc < 2)
|
if(argc < 2)
|
||||||
return !printf("usage: modal [-vqn] source.modal\n");
|
return !printf("usage: modal [-vqn] source.modal\n");
|
||||||
for(i = 1; i < argc && *argv[i] == '-'; i++) {
|
for(i = 1; i < argc && *argv[i] == '-'; i++) {
|
||||||
switch(argv[i][1]) {
|
switch(argv[i][1]) {
|
||||||
case 'v': /* version */ return !printf("Modal Interpreter, 18 Apr 2024.\n");
|
case 'v': /* version */ return !printf("Modal Interpreter, 26 Apr 2024.\n");
|
||||||
case 'q': /* quiet */ fclose(stderr); break;
|
case 'q': /* quiet */ quiet = 1; break;
|
||||||
case 'n': /* infinite */ cycles = 0xffffffff; break;
|
case 'n': /* infinite */ cycles = 0xffffffff; break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(!(f = fopen(argv[i], "r")))
|
if(!(f = fopen(argv[i], "r")))
|
||||||
return !fprintf(stdout, "Invalid Modal file: %s.\n", argv[i]);
|
return !fprintf(stdout, "Modal file invalid: %s.\n", argv[i]);
|
||||||
while(fread(&c, 1, 1, f)) {
|
while(fread(&c, 1, 1, f)) {
|
||||||
c = c <= 0x20 ? 0x20 : c;
|
c = c <= 0x20 ? 0x20 : c;
|
||||||
if(w > bank_a) {
|
if(c == ' ' && last == '(') continue;
|
||||||
if(c == ' ' && *(w - 1) == '(') continue;
|
if(c == ')' && last == ' ') w--;
|
||||||
if(c == ')' && *(w - 1) == ' ') w--;
|
if(c == ' ' && last == ' ') w--;
|
||||||
if(c == ' ' && *(w - 1) == ' ') w--;
|
if(c == '(') pl++;
|
||||||
}
|
if(c == ')') pr++;
|
||||||
*w++ = c;
|
if(c == '(' && last != '?' && !spacer(last)) *w++ = ' ';
|
||||||
|
if(last == ')' && !spacer(c)) *w++ = ' ';
|
||||||
|
*w++ = last = c;
|
||||||
}
|
}
|
||||||
while(*(--w) <= ' ') *w = 0;
|
while(*(--w) <= ' ') *w = 0;
|
||||||
fclose(f);
|
fclose(f);
|
||||||
|
if(pr != pl)
|
||||||
|
return !fprintf(stdout, "Modal program imbalanced.\n");
|
||||||
while(rewrite())
|
while(rewrite())
|
||||||
if(!cycles--) return !fprintf(stdout, "Modal rewrites exceeded.\n");
|
if(!cycles--) return !fprintf(stdout, "Modal rewrites exceeded.\n");
|
||||||
|
while(rules_-- > rules && !quiet)
|
||||||
|
if(!rules_->refs) printf("-- Unused rule: %d <> (%s) (%s)\n", rules_->refs, rules_->a, rules_->b);
|
||||||
|
if(!quiet) printf(".. %s\n", src_);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
Loading…
Reference in New Issue