aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--inc/context.h2
-rw-r--r--inc/tokenize.h2
-rw-r--r--inc/treeify.h27
-rw-r--r--src/tokenize.c14
-rw-r--r--src/treeify.c100
5 files changed, 100 insertions, 45 deletions
diff --git a/inc/context.h b/inc/context.h
index 3f7b686..b11e23d 100644
--- a/inc/context.h
+++ b/inc/context.h
@@ -20,7 +20,7 @@ struct ctx {
struct token *tokens;
struct {
- struct tree *head;
+ struct node *head;
struct node *current;
} tree;
};
diff --git a/inc/tokenize.h b/inc/tokenize.h
index 0142bb9..d1a2931 100644
--- a/inc/tokenize.h
+++ b/inc/tokenize.h
@@ -22,6 +22,8 @@ enum token_type {
NEWLINE,
EOL,
END,
+
+ SOMETHING,
};
struct token {
diff --git a/inc/treeify.h b/inc/treeify.h
index 3af67f8..bc8a64d 100644
--- a/inc/treeify.h
+++ b/inc/treeify.h
@@ -15,13 +15,20 @@ enum node_type {
// (*f* x y)
struct node_expression_identifier {
ctx_string name; // f
- ctx_string type; // u32
};
-// (f *x* *y*)
+enum node_expression_parameter_type {
+ PARAM_TYPE_IDENT,
+ PARAM_TYPE_EXPRESSION,
+};
+
+// (f *x* *y* *(expr)*)
struct node_expression_parameter {
- ctx_string name; // x or y
- ctx_string type; // u32
+ enum node_expression_parameter_type type;
+ union {
+ ctx_string name; // x or y
+ struct node_expression *expression; // (expr)
+ } data;
};
// *(f x y)*
@@ -55,16 +62,14 @@ struct node_declaration {
struct node {
enum node_type type;
- /* struct node *next; */
+ struct node *prev;
+ struct node *next;
void *data;
};
-struct tree {
- struct node *node;
-};
-
-struct tree *tree_create(void);
-void tree_destroy(struct tree *tree);
+struct node *tree_create(void);
+void tree_destroy(struct node *tree);
+void tree_add(struct ctx *ctx, enum node_type type, void *data);
void treeify(struct ctx *ctx);
diff --git a/src/tokenize.c b/src/tokenize.c
index 179b251..02059a8 100644
--- a/src/tokenize.c
+++ b/src/tokenize.c
@@ -46,7 +46,7 @@ static size_t peek_alnum_to(struct ctx *ctx, size_t start, char ch)
errln(ctx, "Unexpected end of buffer");
}
-static size_t peek_special_to(struct ctx *ctx, size_t start, char ch)
+static size_t peek_identifier(struct ctx *ctx, size_t start, char ch)
{
for (size_t i = start; i < ctx->size; i++) {
char cur = ctx->data[i];
@@ -54,8 +54,8 @@ static size_t peek_special_to(struct ctx *ctx, size_t start, char ch)
if (cur == ch || cur == ';' || cur == ')')
return i;
- if (isalnum(cur) || cur < '!' || cur > '~')
- errln(ctx, "'%c' is not special", cur);
+ if (!isalnum(cur) && (cur < '!' || cur > '~'))
+ errln(ctx, "'%c' is not an identifier", cur);
}
errln(ctx, "Unexpected end of buffer");
@@ -156,14 +156,10 @@ void tokenize(struct ctx *ctx)
continue;
}
- if (peek_to_is_alnum(ctx, i, ' ')) { // General identifier
- size_t end_ident = peek_alnum_to(ctx, i, ' ');
+ if (peek_identifier(ctx, i, ' ')) { // General identifier
+ size_t end_ident = peek_to(ctx, i, ' ');
token_add(ctx, IDENT, i, end_ident);
i = end_ident - 1;
- } else { // Special/custom operator
- size_t end_operator = peek_special_to(ctx, i, ' ');
- token_add(ctx, OPERATOR, i, end_operator);
- i = end_operator - 1;
}
}
diff --git a/src/treeify.c b/src/treeify.c
index 32a12c2..8da6481 100644
--- a/src/treeify.c
+++ b/src/treeify.c
@@ -1,32 +1,62 @@
#include <assert.h>
+#include <stdio.h>
#include <stdlib.h>
#include <log.h>
#include <tokenize.h>
#include <treeify.h>
-static void expected(enum token_type type)
+static void __expect(struct ctx *ctx, struct token *token, enum token_type type, const char *file,
+ int line, const char *func, const char *type_enum)
{
- err("Expected token of type %d while creating tree!\n", type);
+ if (token->type != type) {
+ printf("[DBG] %s:%d: %s\n", file, line, func);
+ token_print(ctx, token);
+ err("Expected token of type %s!\n", type_enum);
+ }
}
-static void unexpected(struct ctx *ctx, struct token *token)
-{
- token_print(ctx, token);
- err("Unexpected token while creating tree!\n");
-}
+#define expect(token, type) __expect(ctx, token, type, __FILE__, __LINE__, __func__, #type)
static struct token *next(struct token *token, size_t i)
{
return token + i;
}
+static struct token *parse_declaration(struct ctx *ctx, struct token *token);
+static struct token *parse_expression(struct ctx *ctx, struct token *token)
+{
+ struct node_expression *node = malloc(sizeof(*node));
+ // TODO: Push expressions into tree (using subtrees)
+
+ struct token *iterator = token;
+ while ((iterator = next(iterator, 1))) {
+ if (iterator->type == LPAREN) {
+ iterator = parse_expression(ctx, iterator);
+ continue;
+ }
+
+ if (iterator->type == TYPE) {
+ iterator = parse_declaration(ctx, iterator);
+ continue;
+ }
+
+ if (iterator->type == RPAREN)
+ break;
+
+ if (iterator->type == EOL || iterator->type == END)
+ expect(iterator, RPAREN);
+
+ expect(iterator, IDENT);
+ }
+
+ return iterator;
+}
+
static struct token *parse_declaration(struct ctx *ctx, struct token *token)
{
- if (next(token, 1)->type != TYPEDELIM)
- unexpected(ctx, next(token, 1));
- if (next(token, 2)->type != PARAM)
- unexpected(ctx, next(token, 2));
+ expect(next(token, 1), TYPEDELIM);
+ expect(next(token, 2), PARAM);
// Search for equal sign
struct token *iterator = token;
@@ -34,7 +64,7 @@ static struct token *parse_declaration(struct ctx *ctx, struct token *token)
if (iterator->type == EQUAL)
break;
if (iterator->type == EOL || iterator->type == END)
- expected(EQUAL);
+ expect(iterator, EQUAL);
}
struct node_declaration *node = malloc(sizeof(*node));
@@ -47,46 +77,69 @@ static struct token *parse_declaration(struct ctx *ctx, struct token *token)
node->parameters = malloc((diff / 3 + 1) * sizeof(*node->parameters));
for (size_t i = 0; i < diff / 3; i++) {
struct token *param = token + (i + 1) * 3;
- assert(param->type == TYPE);
- assert(next(param, 2)->type == PARAM);
+ expect(param, TYPE);
+ expect(next(param, 2), PARAM);
node->parameters[i].type = param->string;
node->parameters[i].name = next(param, 2)->string;
}
+ tree_add(ctx, DECLARATION, node);
+
return next(iterator, 1);
}
static struct token *parse(struct ctx *ctx, struct token *token)
{
switch (token->type) {
+ case LPAREN:
+ return parse_expression(ctx, token);
case TYPE:
return parse_declaration(ctx, token);
+ case RPAREN:
+ return next(token, 1);
+ case END:
+ return NULL;
case UNKNOWN:
case TYPEDELIM:
case PARAM:
case IDENT:
case OPERATOR:
- case LPAREN:
- case RPAREN:
case EQUAL:
case NEWLINE:
+ case SOMETHING:
case EOL:
- case END:
default:
- unexpected(ctx, token);
+ expect(token, SOMETHING);
}
return NULL;
}
-struct tree *tree_create(void)
+struct node *tree_create(void)
{
- struct tree *tree = malloc(sizeof(*tree));
- tree->node = NULL;
+ struct node *tree = malloc(sizeof(*tree));
+ tree->prev = NULL;
+ tree->next = NULL;
return tree;
}
-void tree_destroy(struct tree *tree)
+void tree_add(struct ctx *ctx, enum node_type type, void *data)
+{
+ assert(ctx->tree.head);
+ struct node *node = malloc(sizeof(*node));
+ node->type = type;
+ node->data = data;
+ if (!ctx->tree.current) {
+ ctx->tree.head->next = node;
+ node->prev = ctx->tree.head;
+ } else {
+ ctx->tree.current->next = node;
+ node->prev = ctx->tree.current;
+ }
+ ctx->tree.current = node;
+}
+
+void tree_destroy(struct node *tree)
{
// TODO: Destroy nodes
free(tree);
@@ -96,12 +149,11 @@ void treeify(struct ctx *ctx)
{
struct token *token = &ctx->tokens[1];
while (token) {
- assert(token->type != UNKNOWN);
if (token->type == NEWLINE || token->type == EOL) {
token = next(token, 1);
continue;
}
- if (token->type == END)
+ if (token->type == END || token->type == UNKNOWN)
break;
token = parse(ctx, token);