aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorGinger Bill <bill@gingerbill.org>2017-05-07 20:52:20 +0100
committerGinger Bill <bill@gingerbill.org>2017-05-07 20:52:20 +0100
commit7692061eef48765ac51b8a8f3024491381fdf182 (patch)
tree9cf1f7c42d6d415cc170fdacb519118458660a39 /src
parentf7f2272c50f2eb1f9e200de05cdb6b4965ec8104 (diff)
Add XOR for booleans
Diffstat (limited to 'src')
-rw-r--r--src/check_expr.c33
-rw-r--r--src/parser.c60
2 files changed, 71 insertions, 22 deletions
diff --git a/src/check_expr.c b/src/check_expr.c
index 56899bff4..414b15f06 100644
--- a/src/check_expr.c
+++ b/src/check_expr.c
@@ -30,7 +30,7 @@ gb_inline Type *check_type(Checker *c, AstNode *expression) {
void error_operand_not_expression(Operand *o) {
if (o->mode == Addressing_Type) {
gbString err = expr_to_string(o->expr);
- error_node(o->expr, "`%s` is not an expression", err);
+ error_node(o->expr, "`%s` is not an expression but a type", err);
gb_string_free(err);
o->mode = Addressing_Invalid;
}
@@ -1787,8 +1787,8 @@ bool check_unary_op(Checker *c, Operand *o, Token op) {
break;
case Token_Xor:
- if (!is_type_integer(type)) {
- error(op, "Operator `%.*s` is only allowed with integers", LIT(op.string));
+ if (!is_type_integer(type) && !is_type_boolean(type)) {
+ error(op, "Operator `%.*s` is only allowed with integers or booleans", LIT(op.string));
}
break;
@@ -1845,6 +1845,8 @@ bool check_binary_op(Checker *c, Operand *o, Token op) {
case Token_Or:
case Token_AndEq:
case Token_OrEq:
+ case Token_Xor:
+ case Token_XorEq:
if (!is_type_integer(type) && !is_type_boolean(type)) {
error(op, "Operator `%.*s` is only allowed with integers or booleans", LIT(op.string));
return false;
@@ -1852,10 +1854,8 @@ bool check_binary_op(Checker *c, Operand *o, Token op) {
break;
case Token_Mod:
- case Token_Xor:
case Token_AndNot:
case Token_ModEq:
- case Token_XorEq:
case Token_AndNotEq:
if (!is_type_integer(type)) {
error(op, "Operator `%.*s` is only allowed with integers", LIT(op.string));
@@ -4649,13 +4649,19 @@ typedef enum CallArgumentError {
CallArgumentError_TooManyArguments,
} CallArgumentError;
+typedef enum CallArgumentErrorMode {
+ CallArgumentMode_NoErrors,
+ CallArgumentMode_ShowErrors,
+} CallArgumentErrorMode;
+
CallArgumentError check_call_arguments_internal(Checker *c, AstNode *call, Type *proc_type, Operand *operands, isize operand_count,
- bool show_error, i64 *score_) {
+ CallArgumentErrorMode show_error_mode, i64 *score_) {
ast_node(ce, CallExpr, call);
isize param_count = 0;
bool variadic = proc_type->Proc.variadic;
bool vari_expand = (ce->ellipsis.pos.line != 0);
i64 score = 0;
+ bool show_error = show_error_mode == CallArgumentMode_ShowErrors;
if (proc_type->Proc.params != NULL) {
param_count = proc_type->Proc.params->Tuple.variable_count;
@@ -4839,7 +4845,7 @@ Type *check_call_arguments(Checker *c, Operand *operand, Type *proc_type, AstNod
Type *proc_type = base_type(p->type);
if (proc_type != NULL && is_type_proc(proc_type)) {
i64 score = 0;
- CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, false, &score);
+ CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, CallArgumentMode_NoErrors, &score);
if (err == CallArgumentError_None) {
valids[valid_count].index = i;
valids[valid_count].score = score;
@@ -4884,14 +4890,14 @@ Type *check_call_arguments(Checker *c, Operand *operand, Type *proc_type, AstNod
add_entity_use(c, expr, e);
proc_type = e->type;
i64 score = 0;
- CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, true, &score);
+ CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, CallArgumentMode_ShowErrors, &score);
}
gb_free(heap_allocator(), valids);
gb_free(heap_allocator(), procs);
} else {
i64 score = 0;
- CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, true, &score);
+ CallArgumentError err = check_call_arguments_internal(c, call, proc_type, operands.e, operands.count, CallArgumentMode_ShowErrors, &score);
array_free(&operands);
}
return proc_type;
@@ -4936,14 +4942,6 @@ ExprKind check_call_expr(Checker *c, Operand *operand, AstNode *call) {
}
if (operand->mode == Addressing_Type) {
- #if 0
- gbString str = type_to_string(operand->type);
- error_node(call, "Expected a procedure, got a type `%s`", str);
- gb_string_free(str);
- operand->mode = Addressing_Invalid;
- operand->expr = call;
- return Expr_Stmt;
- #else
Type *t = operand->type;
gbString str = type_to_string(t);
operand->mode = Addressing_Invalid;
@@ -4961,7 +4959,6 @@ ExprKind check_call_expr(Checker *c, Operand *operand, AstNode *call) {
gb_string_free(str);
return Expr_Expr;
- #endif
}
if (operand->mode == Addressing_Builtin) {
diff --git a/src/parser.c b/src/parser.c
index 09234a538..7054adfa0 100644
--- a/src/parser.c
+++ b/src/parser.c
@@ -32,8 +32,7 @@ typedef struct AstFile {
// < 0: In Control Clause
// NOTE(bill): Used to prevent type literals in control clauses
isize expr_level;
- bool allow_range;
- bool ignore_operand;
+ bool allow_range; // NOTE(bill): Ranges are only allowed in certain cases
AstNodeArray decls;
bool is_global_scope;
@@ -3774,7 +3773,48 @@ void parse_setup_file_decls(Parser *p, AstFile *f, String base_dir, AstNodeArray
syntax_error_node(node, "Only declarations are allowed at file scope %.*s", LIT(ast_node_strings[node->kind]));
} else if (node->kind == AstNode_ImportDecl) {
ast_node(id, ImportDecl, node);
+ String collection_name = {0};
+ String oirignal_string = id->relpath.string;
String file_str = id->relpath.string;
+ gbAllocator allocator = heap_allocator(); // TODO(bill): Change this allocator
+ String import_file = {0};
+
+ #if 0
+ isize colon_pos = -1;
+ for (isize j = 0; j < file_str.len; j++) {
+ if (file_str.text[j] == ':') {
+ colon_pos = j;
+ break;
+ }
+ }
+ if (colon_pos > 0) {
+ collection_name = make_string(file_str.text, colon_pos);
+ file_str.text += colon_pos+1;
+ file_str.len -= colon_pos+1;
+ }
+
+ if (collection_name.len == 0) {
+ syntax_error_node(node, "Missing import collection for path: `%.*s`", LIT(oirignal_string));
+ decls.e[i] = ast_bad_decl(f, id->relpath, id->relpath);
+ continue;
+ }
+
+
+ if (str_eq(collection_name, str_lit("core"))) {
+ String abs_path = get_fullpath_core(allocator, file_str);
+ if (gb_file_exists(cast(char *)abs_path.text)) { // NOTE(bill): This should be null terminated
+ import_file = abs_path;
+ }
+ } else if (str_eq(collection_name, str_lit("local"))) {
+ String rel_path = get_fullpath_relative(allocator, base_dir, file_str);
+ if (gb_file_exists(cast(char *)rel_path.text)) { // NOTE(bill): This should be null terminated
+ import_file = rel_path;
+ }
+ } else {
+ syntax_error_node(node, "Unknown import collection: `%.*s`", LIT(collection_name));
+ decls.e[i] = ast_bad_decl(f, id->relpath, id->relpath);
+ continue;
+ }
if (!is_import_path_valid(file_str)) {
if (id->is_import) {
@@ -3787,16 +3827,28 @@ void parse_setup_file_decls(Parser *p, AstFile *f, String base_dir, AstNodeArray
continue;
}
- gbAllocator allocator = heap_allocator(); // TODO(bill): Change this allocator
+ #else
+ if (!is_import_path_valid(file_str)) {
+ if (id->is_import) {
+ syntax_error_node(node, "Invalid import path: `%.*s`", LIT(file_str));
+ } else {
+ syntax_error_node(node, "Invalid include path: `%.*s`", LIT(file_str));
+ }
+ // NOTE(bill): It's a naughty name
+ decls.e[i] = ast_bad_decl(f, id->relpath, id->relpath);
+ continue;
+ }
+
String rel_path = get_fullpath_relative(allocator, base_dir, file_str);
- String import_file = rel_path;
+ import_file = rel_path;
if (!gb_file_exists(cast(char *)rel_path.text)) { // NOTE(bill): This should be null terminated
String abs_path = get_fullpath_core(allocator, file_str);
if (gb_file_exists(cast(char *)abs_path.text)) {
import_file = abs_path;
}
}
+ #endif
id->fullpath = import_file;
try_add_import_path(p, import_file, file_str, ast_node_token(node).pos);