summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--05/constants.b5
-rw-r--r--05/main.b5
-rw-r--r--05/main.c19
-rw-r--r--05/parse.b37
4 files changed, 48 insertions, 18 deletions
diff --git a/05/constants.b b/05/constants.b
index e454942..8e2f20c 100644
--- a/05/constants.b
+++ b/05/constants.b
@@ -144,6 +144,7 @@
; ushort (padding)
; uint type
; immediately following the header in memory are the arguments of the expression
+; - for functions, a pointer to the name of the function (we don't know where it is yet)
; - for local variables, the 64-bit rbp offset (number to be subtracted from rbp)
; - for global variables, the 64-bit runtime address
; - for constant ints, the 64-bit integral value
@@ -157,6 +158,7 @@
; - for function calls, the function, followed by each of the arguments to the function — info indicates the number of arguments
; File/line number are not stored in expressions.
; Note that string literals are stored as constant integers (you can check the type to know what it is)
+#define EXPRESSION_FUNCTION 198
#define EXPRESSION_LOCAL_VARIABLE 199
#define EXPRESSION_GLOBAL_VARIABLE 200
#define EXPRESSION_CONSTANT_INT 201
@@ -241,8 +243,9 @@
; reading the first 16 bits of type data as a word will give this if the type refers to a function pointer.
#define TYPE2_FUNCTION_POINTER 0x100d
-; types willl be initialized (in main) so that this refers to the type char*
+; types willl be initialized (in main) so that these will refer to the proper types
#define TYPE_POINTER_TO_CHAR 20
+#define TYPE_POINTER_TO_VOID 22
; STATEMENTS
; In C, note that `if', `while', etc. always have a single statement as their body:
diff --git a/05/main.b b/05/main.b
index 64585d0..94ef526 100644
--- a/05/main.b
+++ b/05/main.b
@@ -99,6 +99,11 @@ function types_init
p += 1
*1p = TYPE_CHAR
p += 1
+ p = _types + TYPE_POINTER_TO_VOID
+ *1p = TYPE_POINTER
+ p += 1
+ *1p = TYPE_VOID
+ p += 1
*8ptypes_bytes_used = p - types
return
diff --git a/05/main.c b/05/main.c
index 37b2279..57eba24 100644
--- a/05/main.c
+++ b/05/main.c
@@ -1,12 +1,19 @@
-static int g;
+/* static int g; */
-int f(int x, int y[3]) {
- int z = 17 +x;
- int g[]={1,2,3,4,5};
- int h;
- return *(g + x + z);
+int main() {
+ int a = exit;
+ int b[] = {1,2,3};
+ exit(1);
}
+/* int f(int x, int y[3]) { */
+/* int z = 17 +x; */
+/* int g[]={1,2,3,4,5}; */
+/* int h; */
+/* funciton(h,g,z); */
+/* return *(g + x + z); */
+/* } */
+
/* typedef int AA[sizeof x]; */
diff --git a/05/parse.b b/05/parse.b
index 091be9c..b1f8b90 100644
--- a/05/parse.b
+++ b/05/parse.b
@@ -654,10 +654,10 @@ function parse_statement
token += 16
if *1token == SYMBOL_LBRACE goto local_init_lbrace
n = token_next_semicolon_or_comma_not_in_brackets(token)
- out += 16
+ out += 24
p = expressions_end
*8out = p
- out -= 16
+ out -= 24
expressions_end = parse_expression(token, n, p)
p += 4
type_decay_array_to_pointer(*4p) ; fix typing for `int[] x = {5,6}; int *y = x;`
@@ -1764,16 +1764,17 @@ function parse_type_declarators
types_bytes_used += 1
p = suffix + 16
- if *1p != KEYWORD_VOID goto ftype_has_parameters
if *1p == SYMBOL_RPAREN goto ftype_no_parameters ; e.g. int f() { return 17; }
+ if *1p != KEYWORD_VOID goto ftype_has_parameters
n = p + 16
+ suffix += 16
if *1n != SYMBOL_RPAREN goto ftype_has_parameters
:ftype_no_parameters
; special handling of function type with no parameters
out = types + types_bytes_used
*1out = 0
types_bytes_used += 1
- suffix += 48
+ suffix += 32
goto type_declarators_loop
@@ -2402,7 +2403,6 @@ function parse_expression
goto expr_find_operator_loop
:expr_find_operator_loop_end
-
if best == 0 goto unrecognized_expression
n = best - tokens
@@ -2423,6 +2423,7 @@ function parse_expression
a = out + 4 ; type of first operand
out = parse_expression(tokens, best, out) ; first operand
p = best + 16
+ if c == EXPRESSION_CALL goto parse_call
b = out + 4 ; type of second operand
if c != EXPRESSION_SUBSCRIPT goto binary_not_subscript
tokens_end -= 16
@@ -2545,7 +2546,9 @@ function parse_expression
:expr_binary_bad_types
bad_types_to_operator(tokens, *4a, *4b)
-
+ :parse_call
+ byte 0xcc ; @TODO
+
:parse_expr_unary
if c == KEYWORD_SIZEOF goto parse_sizeof
*1out = unary_op_to_expression_type(c)
@@ -2849,11 +2852,15 @@ function parse_expression
return out
:not_global
- in -= 16
- token_error(in, .str_undeclared_variable)
- :str_undeclared_variable
- string Undeclared variable.
- byte 0
+ ; it must be a function
+ *1out = EXPRESSION_FUNCTION
+ out += 4
+ *4out = TYPE_POINTER_TO_VOID
+ out += 4
+ *8out = a
+ out += 8
+ return out
+
:found_local_variable
; it's a local variable
*1out = EXPRESSION_LOCAL_VARIABLE
@@ -3716,6 +3723,8 @@ function operator_right_associative
byte EXPRESSION_DOT
byte SYMBOL_LSQUARE
byte EXPRESSION_SUBSCRIPT
+ byte SYMBOL_LPAREN
+ byte EXPRESSION_CALL
byte 0
byte 0
@@ -3791,6 +3800,7 @@ function print_expression
:print_expr_skip_type
c = *1expression
+ if c == EXPRESSION_FUNCTION goto print_expr_function
if c == EXPRESSION_LOCAL_VARIABLE goto print_local_variable
if c == EXPRESSION_GLOBAL_VARIABLE goto print_global_variable
if c == EXPRESSION_CONSTANT_INT goto print_expr_int
@@ -3832,6 +3842,11 @@ function print_expression
:str_local_prefix
string [rbp-
byte 0
+ :print_expr_function
+ expression += 8
+ puts(*8expression)
+ expression += 8
+ return expression
:print_global_variable
puts(.str_global_at)
expression += 8