aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--.gitignore4
-rw-r--r--inc/parse.h8
-rw-r--r--inc/term.h25
-rw-r--r--makefile40
-rw-r--r--readme.md3
-rw-r--r--src/main.c11
-rw-r--r--src/parse.c36
-rw-r--r--src/term.c57
8 files changed, 184 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..50660f9
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,4 @@
+build/
+
+tags
+compile_commands.json
diff --git a/inc/parse.h b/inc/parse.h
new file mode 100644
index 0000000..cfd9102
--- /dev/null
+++ b/inc/parse.h
@@ -0,0 +1,8 @@
+#ifndef PARSE_H
+#define PARSE_H
+
+#include <term.h>
+
+struct term *parse(const char *term);
+
+#endif
diff --git a/inc/term.h b/inc/term.h
new file mode 100644
index 0000000..e750beb
--- /dev/null
+++ b/inc/term.h
@@ -0,0 +1,25 @@
+#ifndef TERM_H
+#define TERM_H
+
+typedef int bruijn;
+typedef enum { INV, ABS, APP, VAR } term_type;
+
+struct term {
+ term_type type;
+ union {
+ struct {
+ struct term *term;
+ } abs;
+ struct {
+ struct term *lhs;
+ struct term *rhs;
+ } app;
+ bruijn var;
+ } u;
+};
+
+struct term *new_term(term_type type);
+void print_term(struct term *term);
+void free_term(struct term *term);
+
+#endif
diff --git a/makefile b/makefile
new file mode 100644
index 0000000..713b3de
--- /dev/null
+++ b/makefile
@@ -0,0 +1,40 @@
+# Copyright (c) 2022, Marvin Borner <dev@marvinborner.de>
+# SPDX-License-Identifier: MIT
+
+CC = gcc
+LD = ld
+TG = ctags
+
+BUILD = $(PWD)/build
+SRC = ./src
+INC = ./inc
+SRCS = $(shell find $(SRC) -name '*.c')
+OBJS = $(SRCS:%=$(BUILD)/%.o)
+
+# I need the following on my machine. Look it up though before using it.
+# export ASAN_OPTIONS=verify_asan_link_order=0
+CFLAGS_DEBUG = -Wno-error -ggdb3 -Og -Wno-unused -fsanitize=address -fsanitize=undefined -fstack-protector-all
+CFLAGS_WARNINGS = -Wall -Wextra -Werror -Wshadow -Wpointer-arith -Wwrite-strings -Wredundant-decls -Wnested-externs -Wformat=2 -Wmissing-declarations -Wstrict-prototypes -Wmissing-prototypes -Wcast-qual -Wswitch-default -Wswitch-enum -Wunreachable-code -Wundef -Wold-style-definition -Wvla -pedantic -Wno-switch-enum
+CFLAGS = $(CFLAGS_WARNINGS) -std=c99 -fno-profile-generate -fno-omit-frame-pointer -fno-common -fno-asynchronous-unwind-tables -mno-red-zone -Ofast -D_DEFAULT_SOURCE -I$(INC) $(CFLAGS_DEBUG)
+
+all: $(BUILD) $(OBJS) $(BUILD)/milcr
+
+compile: all sync
+
+clean:
+ @rm -rf $(BUILD)/*
+
+sync: # Ugly hack
+ @$(MAKE) --always-make --dry-run | grep -wE 'gcc' | jq -nR '[inputs|{directory: ".", command: ., file: match(" [^ ]+$$").string[1:]}]' >compile_commands.json &
+ @$(TG) -R --exclude=.git --exclude=build . &
+
+$(BUILD)/%.c.o: %.c
+ @$(CC) -c $(CFLAGS) $< -o $(patsubst $(BUILD)/$(SRC)/%.c.o,$(BUILD)/%.o,$@)
+
+$(BUILD)/milcr: $(OBJS)
+ $(CC) $(CFLAGS) $(patsubst $(BUILD)/$(SRC)/%.c.o,$(BUILD)/%.o,$^) -o $@
+
+.PHONY: all compile clean sync
+
+$(BUILD):
+ @mkdir -p $@
diff --git a/readme.md b/readme.md
new file mode 100644
index 0000000..18c38fe
--- /dev/null
+++ b/readme.md
@@ -0,0 +1,3 @@
+# MILCR
+
+> **M**ultithreaded **I**nteraction net **L**ambda **C**alculus **R**educer
diff --git a/src/main.c b/src/main.c
new file mode 100644
index 0000000..431d8a5
--- /dev/null
+++ b/src/main.c
@@ -0,0 +1,11 @@
+#include <stdio.h>
+#include <parse.h>
+#include <term.h>
+
+int main(void)
+{
+ struct term *term = parse("([[((0 1) [(1 0)])]] [0])");
+ print_term(term);
+ free_term(term);
+ return 0;
+}
diff --git a/src/parse.c b/src/parse.c
new file mode 100644
index 0000000..00fd498
--- /dev/null
+++ b/src/parse.c
@@ -0,0 +1,36 @@
+// Just for debugging purposes
+// -> parses custom bruijn syntax
+
+#include <parse.h>
+#include <term.h>
+#include <stdio.h>
+
+static struct term *rec(const char **term)
+{
+ struct term *res = 0;
+ if (!**term) {
+ fprintf(stderr, "invalid parsing state!\n");
+ } else if (**term == '[') {
+ (*term)++;
+ res = new_term(ABS);
+ res->u.abs.term = rec(term);
+ } else if (**term == '(') {
+ (*term)++;
+ res = new_term(APP);
+ res->u.app.lhs = rec(term);
+ res->u.app.rhs = rec(term);
+ } else if (**term >= '0' && **term <= '9') {
+ res = new_term(VAR);
+ res->u.var = **term - '0';
+ (*term)++;
+ } else {
+ (*term)++;
+ res = rec(term); // this is quite tolerant..
+ }
+ return res;
+}
+
+struct term *parse(const char *term)
+{
+ return rec(&term);
+}
diff --git a/src/term.c b/src/term.c
new file mode 100644
index 0000000..f288653
--- /dev/null
+++ b/src/term.c
@@ -0,0 +1,57 @@
+#include <term.h>
+#include <stdlib.h>
+#include <stdio.h>
+
+struct term *new_term(term_type type)
+{
+ struct term *term = calloc(1, sizeof(*term));
+ if (!term) {
+ fprintf(stderr, "Out of memory!\n");
+ abort();
+ }
+ term->type = type;
+ return term;
+}
+
+void free_term(struct term *term)
+{
+ switch (term->type) {
+ case ABS:
+ free_term(term->u.abs.term);
+ free(term);
+ break;
+ case APP:
+ free_term(term->u.app.lhs);
+ free_term(term->u.app.rhs);
+ free(term);
+ break;
+ case VAR:
+ free(term);
+ break;
+ default:
+ fprintf(stderr, "Invalid type %d\n", term->type);
+ }
+}
+
+void print_term(struct term *term)
+{
+ switch (term->type) {
+ case ABS:
+ printf("[");
+ print_term(term->u.abs.term);
+ printf("]");
+ break;
+ case APP:
+ printf("(");
+ print_term(term->u.app.lhs);
+ printf(" ");
+ print_term(term->u.app.rhs);
+ printf(")");
+ break;
+ case VAR:
+ printf("%d", term->u.var);
+ break;
+ default:
+ fprintf(stderr, "Invalid type %d\n", term->type);
+ }
+}