summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorpommicket <pommicket@gmail.com>2022-01-21 18:57:13 -0500
committerpommicket <pommicket@gmail.com>2022-01-21 18:57:13 -0500
commit1b4c6b04dd0f2813b4db47dc7efacd61256c81d2 (patch)
treefc0eac0176be2399e5d4e5fab65c611ccdde56dc
parent3641ac98dc7747b7c9ebe2b7ea4007ecab033d5a (diff)
start array types (const expr evaluation)
-rw-r--r--05/main.c2
-rw-r--r--05/parse.b104
2 files changed, 102 insertions, 4 deletions
diff --git a/05/main.c b/05/main.c
index 36f5511..22b84e3 100644
--- a/05/main.c
+++ b/05/main.c
@@ -3,4 +3,4 @@
long double d;
} (*x)(void);
*/
-typedef long int unsigned (*Foo(int *,int,int,unsigned,void (*)(int)))(int x);
+typedef long int unsigned Foo[-18446744073709551610];
diff --git a/05/parse.b b/05/parse.b
index 7ca80f8..a875619 100644
--- a/05/parse.b
+++ b/05/parse.b
@@ -22,6 +22,7 @@ function parse_tokens
:parse_tokens_eof
return
+
; *p_token should be pointing to a {, this will advance it to point to the matching }
function token_skip_to_matching_rbrace
argument p_token
@@ -53,6 +54,38 @@ function token_skip_to_matching_rbrace
string Unmatched {
byte 0
+; *p_token should be pointing to a [, this will advance it to point to the matching ]
+function token_skip_to_matching_rsquare
+ argument p_token
+ local token
+ local depth
+ token = *8p_token
+ depth = 0
+ :skip_square_loop
+ if *1token == SYMBOL_LSQUARE goto skip_square_incdepth
+ if *1token == SYMBOL_RSQUARE goto skip_square_decdepth
+ if *1token == TOKEN_EOF goto skip_square_eof
+ :skip_square_next
+ token += 16
+ goto skip_square_loop
+ :skip_square_incdepth
+ depth += 1
+ goto skip_square_next
+ :skip_square_decdepth
+ depth -= 1
+ if depth == 0 goto skip_square_ret
+ goto skip_square_next
+ :skip_square_ret
+ *8p_token = token
+ return
+
+ :skip_square_eof
+ token_error(*8p_token, .str_skip_square_eof)
+ :str_skip_square_eof
+ string Unmatched [
+ byte 0
+
+
; parse things like `int x` or `int f(void, int, char *)`
; advances *p_token and sets *p_ident to a pointer to the identifier (or 0 if this is just a type)
; returns type ID
@@ -213,7 +246,26 @@ function parse_type_to
prefix_end = p
goto parse_type_loop
:parse_array_type
- byte 0xcc ; @TODO
+ local expr
+ expr = malloc(4000)
+ *1out = TYPE_ARRAY
+ out += 1
+ p = suffix
+ token_skip_to_matching_rsquare(&p)
+ suffix += 16 ; skip [
+ parse_expression(suffix, p, expr)
+ evaluate_constant_expression(expr, &n)
+ if n < 0 goto bad_array_size
+ *8out = n
+ out += 8
+ free(expr)
+ suffix = p + 16
+ goto parse_type_loop
+ :bad_array_size
+ token_error(*8p_token, .str_bad_array_size)
+ :str_bad_array_size
+ string Very large or negative array size.
+ byte 0
:parse_function_type
p = suffix + 16
local RRR
@@ -705,6 +757,8 @@ function parse_expression
if c == EXPRESSION_LOGICAL_NOT goto unary_type_logical_not
if c == EXPRESSION_ADDRESS_OF goto unary_address_of
if c == EXPRESSION_DEREFERENCE goto unary_dereference
+ if c == EXPRESSION_PRE_INCREMENT goto unary_type_arithmetic_nopromote
+ if c == EXPRESSION_PRE_DECREMENT goto unary_type_arithmetic_nopromote
fputs(2, .str_unop_this_shouldnt_happen)
exit(1)
:str_unop_this_shouldnt_happen
@@ -729,7 +783,10 @@ function parse_expression
if *1p > TYPE_DOUBLE goto unary_bad_type
*4type = type_promotion(*4a)
return out
-
+ :unary_type_arithmetic_nopromote
+ if *1p > TYPE_DOUBLE goto unary_bad_type
+ *4type = *4a
+ return out
:unary_bad_type
fprint_token_location(1, tokens)
puts(.str_unary_bad_type)
@@ -874,6 +931,46 @@ function parse_expression
:return_type_double
return TYPE_DOUBLE
+; evaluate an expression which can be the size of an array, e.g.
+; enum { A, B, C };
+; int x[A * sizeof(float) + 3 << 5];
+; @NONSTANDARD: doesn't handle floats, but really why would you use floats in an array size
+; e.g. SomeType x[(int)3.3];
+; this is also used for #if evaluation
+; NOTE: this returns the end of the expression, not the value (which is stored in *8p_value)
+function evaluate_constant_expression
+ argument expr
+ argument p_value
+
+ if *1expr == EXPRESSION_IDENTIFIER goto eval_constant_identifier
+ if *1expr == EXPRESSION_CONSTANT_INT goto eval_constant_int
+ if *1expr == EXPRESSION_UNARY_PLUS goto eval_unary_plus
+ if *1expr == EXPRESSION_UNARY_MINUS goto eval_unary_minus
+ byte 0xcc
+
+ :eval_constant_identifier
+ ; @TODO: enum values
+ fputs(2, .str_constant_identifier)
+ exit(1)
+ :str_constant_identifier
+ string Constant identifiers not handled (see @TODO).
+ byte 10
+ byte 0
+ :eval_constant_int
+ expr += 8
+ *8p_value = *8expr
+ expr += 8
+ return expr
+ :eval_unary_plus
+ expr += 8
+ expr = evaluate_constant_expression(expr, p_value)
+ return expr
+ :eval_unary_minus
+ expr += 8
+ expr = evaluate_constant_expression(expr, p_value)
+ *8p_value = 0 - *8p_value
+ return expr
+
; the "usual conversions" for binary operators, as the C standard calls it
function expr_binary_type_usual_conversions
argument token ; for errors
@@ -1406,7 +1503,8 @@ function print_type
:print_type_array
putc('[)
type += 1
- putn(*8type) ; UNALIGNED
+ c = types + type
+ putn(*8c) ; UNALIGNED
putc('])
type += 8
goto print_type_top