2024-11-24 11:10:42 +00:00
|
|
|
#include "node.h"
|
|
|
|
|
|
2024-12-06 21:33:53 +00:00
|
|
|
#include <blue/object/string.h>
|
2024-11-24 11:10:42 +00:00
|
|
|
#include <ivy/lang/ast.h>
|
2024-11-24 21:31:49 +00:00
|
|
|
#include <stdio.h>
|
2024-11-24 11:10:42 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
extern struct ast_node_type unit_node_ops;
|
|
|
|
|
extern struct ast_node_type unit_package_node_ops;
|
2024-11-25 16:50:42 +00:00
|
|
|
extern struct ast_node_type unit_import_node_ops;
|
|
|
|
|
extern struct ast_node_type class_node_ops;
|
2024-11-26 21:30:40 +00:00
|
|
|
extern struct ast_node_type msgh_node_ops;
|
|
|
|
|
extern struct ast_node_type selector_node_ops;
|
2024-11-27 22:29:29 +00:00
|
|
|
extern struct ast_node_type expr_node_ops;
|
2024-11-28 16:59:04 +00:00
|
|
|
extern struct ast_node_type block_node_ops;
|
2024-11-28 22:05:37 +00:00
|
|
|
extern struct ast_node_type msg_node_ops;
|
|
|
|
|
extern struct ast_node_type op_node_ops;
|
|
|
|
|
extern struct ast_node_type ident_node_ops;
|
|
|
|
|
extern struct ast_node_type int_node_ops;
|
|
|
|
|
extern struct ast_node_type double_node_ops;
|
2024-12-06 13:46:41 +00:00
|
|
|
extern struct ast_node_type atom_node_ops;
|
2024-11-28 22:05:37 +00:00
|
|
|
extern struct ast_node_type string_node_ops;
|
2024-12-06 13:46:58 +00:00
|
|
|
extern struct ast_node_type fstring_node_ops;
|
2024-12-03 21:57:44 +00:00
|
|
|
extern struct ast_node_type cascade_node_ops;
|
2024-12-04 16:35:19 +00:00
|
|
|
extern struct ast_node_type cond_group_node_ops;
|
|
|
|
|
extern struct ast_node_type cond_node_ops;
|
2024-12-04 22:22:25 +00:00
|
|
|
extern struct ast_node_type match_node_ops;
|
2024-12-05 19:29:21 +00:00
|
|
|
extern struct ast_node_type while_loop_node_ops;
|
2024-12-06 10:02:31 +00:00
|
|
|
extern struct ast_node_type for_loop_node_ops;
|
2024-12-06 10:01:56 +00:00
|
|
|
extern struct ast_node_type return_node_ops;
|
2024-12-06 19:47:27 +00:00
|
|
|
extern struct ast_node_type property_node_ops;
|
2024-12-04 22:22:25 +00:00
|
|
|
extern struct ast_node_type discard_node_ops;
|
2024-11-24 11:10:42 +00:00
|
|
|
|
|
|
|
|
static const struct ast_node_type *node_ops[] = {
|
|
|
|
|
[IVY_AST_UNIT] = &unit_node_ops,
|
|
|
|
|
[IVY_AST_UNIT_PACKAGE] = &unit_package_node_ops,
|
2024-11-25 16:50:42 +00:00
|
|
|
[IVY_AST_UNIT_IMPORT] = &unit_import_node_ops,
|
|
|
|
|
[IVY_AST_CLASS] = &class_node_ops,
|
2024-11-27 12:56:10 +00:00
|
|
|
[IVY_AST_MSGH] = &msgh_node_ops,
|
|
|
|
|
[IVY_AST_SELECTOR] = &selector_node_ops,
|
2024-11-27 22:29:29 +00:00
|
|
|
[IVY_AST_EXPR] = &expr_node_ops,
|
2024-11-28 16:59:04 +00:00
|
|
|
[IVY_AST_BLOCK] = &block_node_ops,
|
2024-11-28 22:05:37 +00:00
|
|
|
[IVY_AST_MSG] = &msg_node_ops,
|
|
|
|
|
[IVY_AST_OP] = &op_node_ops,
|
|
|
|
|
[IVY_AST_IDENT] = &ident_node_ops,
|
|
|
|
|
[IVY_AST_INT] = &int_node_ops,
|
|
|
|
|
[IVY_AST_DOUBLE] = &double_node_ops,
|
2024-12-06 13:46:41 +00:00
|
|
|
[IVY_AST_ATOM] = &atom_node_ops,
|
2024-11-28 22:05:37 +00:00
|
|
|
[IVY_AST_STRING] = &string_node_ops,
|
2024-12-06 13:46:58 +00:00
|
|
|
[IVY_AST_FSTRING] = &fstring_node_ops,
|
2024-12-03 21:57:44 +00:00
|
|
|
[IVY_AST_CASCADE] = &cascade_node_ops,
|
2024-12-04 16:35:19 +00:00
|
|
|
[IVY_AST_COND_GROUP] = &cond_group_node_ops,
|
|
|
|
|
[IVY_AST_COND] = &cond_node_ops,
|
2024-12-04 22:22:25 +00:00
|
|
|
[IVY_AST_MATCH] = &match_node_ops,
|
2024-12-05 19:29:21 +00:00
|
|
|
[IVY_AST_WHILE_LOOP] = &while_loop_node_ops,
|
2024-12-06 10:02:31 +00:00
|
|
|
[IVY_AST_FOR_LOOP] = &for_loop_node_ops,
|
2024-12-06 10:01:56 +00:00
|
|
|
[IVY_AST_RETURN] = &return_node_ops,
|
2024-12-06 19:47:27 +00:00
|
|
|
[IVY_AST_PROPERTY] = &property_node_ops,
|
2024-12-04 22:22:25 +00:00
|
|
|
[IVY_AST_DISCARD] = &discard_node_ops,
|
2024-11-24 11:10:42 +00:00
|
|
|
};
|
|
|
|
|
static const size_t nr_node_ops = sizeof node_ops / sizeof node_ops[0];
|
|
|
|
|
|
|
|
|
|
const struct ast_node_type *get_ast_node_type(enum ivy_ast_node_type type)
|
|
|
|
|
{
|
|
|
|
|
if (type >= nr_node_ops) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return node_ops[type];
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-27 22:45:34 +00:00
|
|
|
enum token_expr_type get_token_expr_type(struct ivy_token *tok)
|
2024-11-27 12:56:10 +00:00
|
|
|
{
|
2024-11-28 16:59:04 +00:00
|
|
|
const struct ivy_operator *op = NULL;
|
|
|
|
|
|
2024-11-27 12:56:10 +00:00
|
|
|
switch (tok->t_type) {
|
|
|
|
|
case IVY_TOK_IDENT:
|
|
|
|
|
case IVY_TOK_INT:
|
|
|
|
|
case IVY_TOK_DOUBLE:
|
|
|
|
|
case IVY_TOK_STRING:
|
|
|
|
|
case IVY_TOK_STR_START:
|
|
|
|
|
case IVY_TOK_ATOM:
|
|
|
|
|
return TOK_EXPR_BEGIN;
|
|
|
|
|
case IVY_TOK_SYMBOL:
|
|
|
|
|
switch (tok->t_symbol) {
|
|
|
|
|
case IVY_SYM_LEFT_PAREN:
|
|
|
|
|
case IVY_SYM_CARET:
|
|
|
|
|
return TOK_EXPR_BEGIN;
|
|
|
|
|
case IVY_SYM_COMMA:
|
|
|
|
|
return TOK_EXPR_ANY;
|
|
|
|
|
default:
|
2024-12-01 13:25:36 +00:00
|
|
|
op = ivy_operator_get_by_token(tok->t_symbol);
|
2024-11-28 16:59:04 +00:00
|
|
|
return op ? TOK_EXPR_ANY : TOK_EXPR_NONE;
|
2024-11-27 12:56:10 +00:00
|
|
|
}
|
|
|
|
|
case IVY_TOK_KEYWORD:
|
|
|
|
|
switch (tok->t_keyword) {
|
|
|
|
|
case IVY_KW_IF:
|
|
|
|
|
case IVY_KW_ELSE:
|
|
|
|
|
case IVY_KW_MATCH:
|
|
|
|
|
case IVY_KW_FOR:
|
|
|
|
|
case IVY_KW_WHILE:
|
|
|
|
|
case IVY_KW_TRY:
|
|
|
|
|
case IVY_KW_CATCH:
|
|
|
|
|
case IVY_KW_THROW:
|
|
|
|
|
return TOK_EXPR_BEGIN;
|
|
|
|
|
default:
|
2024-12-01 13:25:36 +00:00
|
|
|
op = ivy_operator_get_by_token(tok->t_keyword);
|
2024-11-28 16:59:04 +00:00
|
|
|
return op ? TOK_EXPR_ANY : TOK_EXPR_NONE;
|
2024-11-27 12:56:10 +00:00
|
|
|
return TOK_EXPR_NONE;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return TOK_EXPR_NONE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-24 11:10:42 +00:00
|
|
|
token_parse_function get_token_parser(
|
|
|
|
|
struct ivy_ast_node *context, struct ivy_token *tok)
|
|
|
|
|
{
|
|
|
|
|
const struct ast_node_type *type = get_ast_node_type(context->n_type);
|
|
|
|
|
if (!type) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2024-12-01 13:25:36 +00:00
|
|
|
token_parse_function generic_parser
|
|
|
|
|
= type->n_token_parsers[__TOK_PARSER_INDEX(tok->t_type)];
|
2024-11-27 22:29:29 +00:00
|
|
|
|
|
|
|
|
if (!generic_parser) {
|
2024-12-01 13:25:36 +00:00
|
|
|
generic_parser
|
|
|
|
|
= type->n_token_parsers[__TOK_PARSER_FALLBACK_INDEX];
|
2024-11-27 22:29:29 +00:00
|
|
|
}
|
|
|
|
|
|
2024-11-24 11:10:42 +00:00
|
|
|
token_parse_function better_parser = NULL;
|
|
|
|
|
|
|
|
|
|
switch (tok->t_type) {
|
|
|
|
|
case IVY_TOK_KEYWORD:
|
2024-12-01 13:25:36 +00:00
|
|
|
better_parser
|
|
|
|
|
= type->n_keyword_parsers[__KW_PARSER_INDEX(tok->t_keyword)];
|
2024-11-28 10:56:43 +00:00
|
|
|
if (type->n_keyword_parsers[__KW_PARSER_FALLBACK_INDEX]) {
|
2024-12-01 13:25:36 +00:00
|
|
|
generic_parser
|
|
|
|
|
= type->n_keyword_parsers[__KW_PARSER_FALLBACK_INDEX];
|
2024-11-27 22:29:29 +00:00
|
|
|
}
|
2024-11-24 11:10:42 +00:00
|
|
|
break;
|
|
|
|
|
case IVY_TOK_SYMBOL:
|
2024-12-01 13:25:36 +00:00
|
|
|
better_parser
|
|
|
|
|
= type->n_symbol_parsers[__SYM_PARSER_INDEX(tok->t_symbol)];
|
2024-11-28 10:56:43 +00:00
|
|
|
if (type->n_symbol_parsers[__SYM_PARSER_FALLBACK_INDEX]) {
|
2024-12-01 13:25:36 +00:00
|
|
|
generic_parser
|
|
|
|
|
= type->n_symbol_parsers[__SYM_PARSER_FALLBACK_INDEX];
|
2024-11-27 22:29:29 +00:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (better_parser) {
|
|
|
|
|
return better_parser;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-27 22:45:34 +00:00
|
|
|
enum token_expr_type expr_type = get_token_expr_type(tok);
|
2024-11-27 22:29:29 +00:00
|
|
|
switch (expr_type) {
|
|
|
|
|
case TOK_EXPR_BEGIN:
|
|
|
|
|
better_parser = type->n_expr_parser.expr_begin
|
|
|
|
|
? type->n_expr_parser.expr_begin
|
|
|
|
|
: type->n_expr_parser.expr_all;
|
|
|
|
|
break;
|
|
|
|
|
case TOK_EXPR_ANY:
|
|
|
|
|
better_parser = type->n_expr_parser.expr_other
|
|
|
|
|
? type->n_expr_parser.expr_other
|
|
|
|
|
: type->n_expr_parser.expr_all;
|
2024-11-24 11:10:42 +00:00
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return better_parser ? better_parser : generic_parser;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-28 22:06:25 +00:00
|
|
|
struct ivy_ast_node *ast_node_create(enum ivy_ast_node_type type)
|
2024-11-24 11:10:42 +00:00
|
|
|
{
|
2024-11-28 22:06:25 +00:00
|
|
|
const struct ast_node_type *type_info = get_ast_node_type(type);
|
|
|
|
|
if (!type_info) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct ivy_ast_node *node = malloc(type_info->n_node_size);
|
2024-11-24 11:10:42 +00:00
|
|
|
if (!node) {
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-28 22:06:25 +00:00
|
|
|
memset(node, 0x0, type_info->n_node_size);
|
2024-11-24 11:10:42 +00:00
|
|
|
|
|
|
|
|
node->n_type = type;
|
|
|
|
|
return node;
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-06 20:24:08 +00:00
|
|
|
void ivy_ast_node_to_string(struct ivy_ast_node *node, struct b_string *out)
|
2024-11-24 20:50:12 +00:00
|
|
|
{
|
2024-12-06 20:24:08 +00:00
|
|
|
const struct ast_node_type *type_info = get_ast_node_type(node->n_type);
|
|
|
|
|
if (!type_info) {
|
|
|
|
|
return;
|
2024-11-24 20:50:12 +00:00
|
|
|
}
|
|
|
|
|
|
2024-12-06 20:24:08 +00:00
|
|
|
if (type_info->n_to_string) {
|
|
|
|
|
type_info->n_to_string(node, out);
|
2024-11-24 20:50:12 +00:00
|
|
|
} else {
|
2024-12-06 21:33:53 +00:00
|
|
|
b_string_append_cstr(
|
|
|
|
|
out, ivy_ast_node_type_to_string(node->n_type));
|
2024-11-24 20:50:12 +00:00
|
|
|
}
|
2024-11-24 16:12:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ivy_ast_node_destroy(struct ivy_ast_node *node)
|
|
|
|
|
{
|
2024-11-24 20:50:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define ENUM_STR(x) \
|
|
|
|
|
case x: \
|
|
|
|
|
return #x
|
|
|
|
|
|
|
|
|
|
const char *ivy_ast_node_type_to_string(enum ivy_ast_node_type v)
|
|
|
|
|
{
|
|
|
|
|
switch (v) {
|
|
|
|
|
ENUM_STR(IVY_AST_NONE);
|
|
|
|
|
ENUM_STR(IVY_AST_UNIT);
|
|
|
|
|
ENUM_STR(IVY_AST_OP);
|
|
|
|
|
ENUM_STR(IVY_AST_MSG);
|
|
|
|
|
ENUM_STR(IVY_AST_CLASS);
|
|
|
|
|
ENUM_STR(IVY_AST_MSGH);
|
|
|
|
|
ENUM_STR(IVY_AST_PROPERTY);
|
2024-11-28 16:59:04 +00:00
|
|
|
ENUM_STR(IVY_AST_SELECTOR);
|
|
|
|
|
ENUM_STR(IVY_AST_EXPR);
|
2024-11-24 20:50:12 +00:00
|
|
|
ENUM_STR(IVY_AST_LAMBDA);
|
|
|
|
|
ENUM_STR(IVY_AST_UNIT_PACKAGE);
|
|
|
|
|
ENUM_STR(IVY_AST_UNIT_IMPORT);
|
2024-12-04 22:22:25 +00:00
|
|
|
ENUM_STR(IVY_AST_DISCARD);
|
2024-11-24 20:50:12 +00:00
|
|
|
ENUM_STR(IVY_AST_INT);
|
|
|
|
|
ENUM_STR(IVY_AST_DOUBLE);
|
|
|
|
|
ENUM_STR(IVY_AST_STRING);
|
|
|
|
|
ENUM_STR(IVY_AST_FSTRING);
|
|
|
|
|
ENUM_STR(IVY_AST_ATOM);
|
|
|
|
|
ENUM_STR(IVY_AST_IDENT);
|
|
|
|
|
ENUM_STR(IVY_AST_FOR_LOOP);
|
|
|
|
|
ENUM_STR(IVY_AST_WHILE_LOOP);
|
2024-12-03 21:57:44 +00:00
|
|
|
ENUM_STR(IVY_AST_CASCADE);
|
2024-11-24 20:50:12 +00:00
|
|
|
ENUM_STR(IVY_AST_COND_GROUP);
|
2024-12-04 22:22:25 +00:00
|
|
|
ENUM_STR(IVY_AST_MATCH);
|
2024-11-24 20:50:12 +00:00
|
|
|
ENUM_STR(IVY_AST_COND);
|
|
|
|
|
ENUM_STR(IVY_AST_TUPLE);
|
2024-11-28 10:26:53 +00:00
|
|
|
ENUM_STR(IVY_AST_BLOCK);
|
2024-12-06 10:01:56 +00:00
|
|
|
ENUM_STR(IVY_AST_RETURN);
|
2024-11-24 20:50:12 +00:00
|
|
|
ENUM_STR(IVY_AST_TYPE_COUNT);
|
|
|
|
|
default:
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
}
|