From efe42f0bf8d5d24f13f2016142ad78d21e0e8e23 Mon Sep 17 00:00:00 2001 From: Yaossg Date: Tue, 24 Dec 2024 15:00:51 +0800 Subject: [PATCH] 100% coverage --- README.md | 1 + boot.c | 13 +-------- cov-test.sh | 4 +-- run.sh | 2 +- test/error/addressof_rvalue.c | 3 ++ test/error/deref_non_ptr.c | 4 +++ test/error/deref_void_ptr.c | 4 +++ test/error/endless_char.c | 3 ++ test/error/endless_comment.c | 1 + test/error/endless_string.c | 3 ++ test/error/invalid_break.c | 3 ++ test/error/invalid_continue.c | 3 ++ test/error/invalid_dots.c | 1 + test/error/invalid_escape.c | 3 ++ test/error/negative_array_size.c | 1 + test/error/unexpected_char.c | 1 + test/error/unexpected_global.c | 1 + test/error/unexpected_token.c | 3 ++ test/error/variable_array_size.c | 2 ++ test/error/void_arg.c | 1 + test/error/void_global.c | 1 + test/error/void_local.c | 3 ++ test/operator/assign.c | 28 +++++++++++++++++++ test/{op => operator}/binary.c | 0 test/{op => operator}/binary.out | 0 test/{op => operator}/inc.c | 0 test/{op => operator}/inc.out | 0 test/{op => operator}/short.c | 0 test/{op => operator}/short.in | 0 test/{op => operator}/short.out | 0 test/unsupported/address_of_pointer.c | 4 +++ test/unsupported/array_of_pointer_arg.c | 1 + test/unsupported/array_of_pointer_global.c | 1 + .../array_of_pointer_local.c} | 2 +- 34 files changed, 81 insertions(+), 16 deletions(-) create mode 100644 test/error/addressof_rvalue.c create mode 100644 test/error/deref_non_ptr.c create mode 100644 test/error/deref_void_ptr.c create mode 100644 test/error/endless_char.c create mode 100644 test/error/endless_comment.c create mode 100644 test/error/endless_string.c create mode 100644 test/error/invalid_break.c create mode 100644 test/error/invalid_continue.c create mode 100644 test/error/invalid_dots.c create mode 100644 test/error/invalid_escape.c create mode 100644 test/error/negative_array_size.c create mode 100644 test/error/unexpected_char.c create mode 100644 test/error/unexpected_global.c create mode 100644 test/error/unexpected_token.c create mode 100644 test/error/variable_array_size.c create mode 100644 test/error/void_arg.c create mode 100644 test/error/void_global.c create mode 100644 test/error/void_local.c create mode 100644 test/operator/assign.c rename test/{op => operator}/binary.c (100%) rename test/{op => operator}/binary.out (100%) rename test/{op => operator}/inc.c (100%) rename test/{op => operator}/inc.out (100%) rename test/{op => operator}/short.c (100%) rename test/{op => operator}/short.in (100%) rename test/{op => operator}/short.out (100%) create mode 100644 test/unsupported/address_of_pointer.c create mode 100644 test/unsupported/array_of_pointer_arg.c create mode 100644 test/unsupported/array_of_pointer_global.c rename test/{error/array_pointer.c => unsupported/array_of_pointer_local.c} (53%) diff --git a/README.md b/README.md index 78b3f1d..35d031d 100644 --- a/README.md +++ b/README.md @@ -138,6 +138,7 @@ $ sh boot.sh - 函数只支持最多八个参数。函数声明中支持可变参数,仅用于兼容 C 语言库。 - 类型检查有遗漏,若 C 编译器报错,而本语言编译通过,就可以认为是 UB。 - 例如函数调用的参数和 `return` 语句不会检查类型。 + - 赋值和比较时没有类型检查,这是为了方便 `0` 成为空指针常量。 ## 限制 diff --git a/boot.c b/boot.c index bd2bdb0..db14a1c 100644 --- a/boot.c +++ b/boot.c @@ -1095,10 +1095,6 @@ int asm_add(int lhs, int rhs) { idx = lhs; } int ptr_type = reg_type[ptr]; - if (ptr_type == TYPE_VOID_PTR) { - fprintf(stderr, "void pointer cannot be arithmetically operated\n"); - exit(1); - } int offset = next_reg(TYPE_INT); _asm_ri("slli", offset, idx, log_step_of(ptr_type)); return asm_rr(ptr_type, "add", ptr, offset); @@ -1120,10 +1116,6 @@ int asm_sub(int lhs, int rhs) { fprintf(stderr, "pointer type mismatch\n"); exit(1); } - if (lhs_type == TYPE_VOID_PTR) { - fprintf(stderr, "void pointer cannot be arithmetically operated\n"); - exit(1); - } int diff = asm_rr(TYPE_INT, "sub", lhs, rhs); _asm_ri("srai", diff, diff, log_step_of(lhs_type)); return diff; @@ -1151,6 +1143,7 @@ int addressof(int reg) { indirection[reg] = 0; } else { printf("cannot take address of this expression"); + exit(1); } return reg; } @@ -1651,10 +1644,6 @@ void parse_local_variable(int type) { unget_token(); expect_token(TOKEN_ASSIGN); int reg = parse_expr(); - if (type != reg_type[reg]) { - fprintf(stderr, "type mismatch in assignment\n"); - exit(1); - } store_into_local(reg, slot); } diff --git a/cov-test.sh b/cov-test.sh index a155324..f37caa7 100644 --- a/cov-test.sh +++ b/cov-test.sh @@ -2,8 +2,8 @@ mkdir -p cov && cd cov && rm * gcc --coverage -g -O0 ../boot.c -o boot.elf for i in ../test/**/*.c; do - echo "Running coverage for test '$(basename $i .c)'" - ./boot.elf < $i > /dev/null + echo "Running coverage for test '$i'" + ./boot.elf < $i > /dev/null 2>/dev/null gcov boot.elf-boot.c done diff --git a/run.sh b/run.sh index 983bab7..268461c 100644 --- a/run.sh +++ b/run.sh @@ -3,4 +3,4 @@ gcc boot.c -o boot.elf && riscv64-linux-gnu-gcc-12 -static $1.s -o $1.elf && qemu-riscv64 $1.elf echo $? -rm $1.s $1.elf boot.elf +rm $1.s $1.elf boot.elf 2> /dev/null diff --git a/test/error/addressof_rvalue.c b/test/error/addressof_rvalue.c new file mode 100644 index 0000000..007fe9b --- /dev/null +++ b/test/error/addressof_rvalue.c @@ -0,0 +1,3 @@ +int main() { + &0; +} \ No newline at end of file diff --git a/test/error/deref_non_ptr.c b/test/error/deref_non_ptr.c new file mode 100644 index 0000000..ef0a4fe --- /dev/null +++ b/test/error/deref_non_ptr.c @@ -0,0 +1,4 @@ +int main() { + int a; + *a; +} \ No newline at end of file diff --git a/test/error/deref_void_ptr.c b/test/error/deref_void_ptr.c new file mode 100644 index 0000000..c6451de --- /dev/null +++ b/test/error/deref_void_ptr.c @@ -0,0 +1,4 @@ +int main() { + void* p; + *p; +} \ No newline at end of file diff --git a/test/error/endless_char.c b/test/error/endless_char.c new file mode 100644 index 0000000..436b024 --- /dev/null +++ b/test/error/endless_char.c @@ -0,0 +1,3 @@ +int main() { + ' +} \ No newline at end of file diff --git a/test/error/endless_comment.c b/test/error/endless_comment.c new file mode 100644 index 0000000..22e8364 --- /dev/null +++ b/test/error/endless_comment.c @@ -0,0 +1 @@ +/* \ No newline at end of file diff --git a/test/error/endless_string.c b/test/error/endless_string.c new file mode 100644 index 0000000..e107bc9 --- /dev/null +++ b/test/error/endless_string.c @@ -0,0 +1,3 @@ +int main() { + " +} \ No newline at end of file diff --git a/test/error/invalid_break.c b/test/error/invalid_break.c new file mode 100644 index 0000000..f2b16d6 --- /dev/null +++ b/test/error/invalid_break.c @@ -0,0 +1,3 @@ +int main() { + break; +} \ No newline at end of file diff --git a/test/error/invalid_continue.c b/test/error/invalid_continue.c new file mode 100644 index 0000000..4f268a6 --- /dev/null +++ b/test/error/invalid_continue.c @@ -0,0 +1,3 @@ +int main() { + continue; +} \ No newline at end of file diff --git a/test/error/invalid_dots.c b/test/error/invalid_dots.c new file mode 100644 index 0000000..945c9b4 --- /dev/null +++ b/test/error/invalid_dots.c @@ -0,0 +1 @@ +. \ No newline at end of file diff --git a/test/error/invalid_escape.c b/test/error/invalid_escape.c new file mode 100644 index 0000000..3cafa9d --- /dev/null +++ b/test/error/invalid_escape.c @@ -0,0 +1,3 @@ +int main() { + return '\i'; +} \ No newline at end of file diff --git a/test/error/negative_array_size.c b/test/error/negative_array_size.c new file mode 100644 index 0000000..75bac71 --- /dev/null +++ b/test/error/negative_array_size.c @@ -0,0 +1 @@ +int a[-1]; \ No newline at end of file diff --git a/test/error/unexpected_char.c b/test/error/unexpected_char.c new file mode 100644 index 0000000..64845fb --- /dev/null +++ b/test/error/unexpected_char.c @@ -0,0 +1 @@ +` \ No newline at end of file diff --git a/test/error/unexpected_global.c b/test/error/unexpected_global.c new file mode 100644 index 0000000..c227083 --- /dev/null +++ b/test/error/unexpected_global.c @@ -0,0 +1 @@ +0 \ No newline at end of file diff --git a/test/error/unexpected_token.c b/test/error/unexpected_token.c new file mode 100644 index 0000000..e629775 --- /dev/null +++ b/test/error/unexpected_token.c @@ -0,0 +1,3 @@ +int main() { + if 1 +} \ No newline at end of file diff --git a/test/error/variable_array_size.c b/test/error/variable_array_size.c new file mode 100644 index 0000000..a63cd6b --- /dev/null +++ b/test/error/variable_array_size.c @@ -0,0 +1,2 @@ +int size = 10; +int a[size]; \ No newline at end of file diff --git a/test/error/void_arg.c b/test/error/void_arg.c new file mode 100644 index 0000000..ab2394c --- /dev/null +++ b/test/error/void_arg.c @@ -0,0 +1 @@ +void f(void a) {} \ No newline at end of file diff --git a/test/error/void_global.c b/test/error/void_global.c new file mode 100644 index 0000000..877e6d2 --- /dev/null +++ b/test/error/void_global.c @@ -0,0 +1 @@ +void a; \ No newline at end of file diff --git a/test/error/void_local.c b/test/error/void_local.c new file mode 100644 index 0000000..b439b08 --- /dev/null +++ b/test/error/void_local.c @@ -0,0 +1,3 @@ +void f() { + void a; +} \ No newline at end of file diff --git a/test/operator/assign.c b/test/operator/assign.c new file mode 100644 index 0000000..4932b7b --- /dev/null +++ b/test/operator/assign.c @@ -0,0 +1,28 @@ +int printf(char* format, ...); + +int main() { + int a = 5; + int b = 4; + a += b; + printf("%d\n", a); + a -= b; + printf("%d\n", a); + a *= b; + printf("%d\n", a); + a /= b; + printf("%d\n", a); + a %= b; + printf("%d\n", a); + a &= b; + printf("%d\n", a); + a |= b; + printf("%d\n", a); + a ^= b; + printf("%d\n", a); + a = b; + printf("%d\n", a); + a <<= b; + printf("%d\n", a); + a >>= b; + printf("%d\n", a); +} \ No newline at end of file diff --git a/test/op/binary.c b/test/operator/binary.c similarity index 100% rename from test/op/binary.c rename to test/operator/binary.c diff --git a/test/op/binary.out b/test/operator/binary.out similarity index 100% rename from test/op/binary.out rename to test/operator/binary.out diff --git a/test/op/inc.c b/test/operator/inc.c similarity index 100% rename from test/op/inc.c rename to test/operator/inc.c diff --git a/test/op/inc.out b/test/operator/inc.out similarity index 100% rename from test/op/inc.out rename to test/operator/inc.out diff --git a/test/op/short.c b/test/operator/short.c similarity index 100% rename from test/op/short.c rename to test/operator/short.c diff --git a/test/op/short.in b/test/operator/short.in similarity index 100% rename from test/op/short.in rename to test/operator/short.in diff --git a/test/op/short.out b/test/operator/short.out similarity index 100% rename from test/op/short.out rename to test/operator/short.out diff --git a/test/unsupported/address_of_pointer.c b/test/unsupported/address_of_pointer.c new file mode 100644 index 0000000..59fc6ce --- /dev/null +++ b/test/unsupported/address_of_pointer.c @@ -0,0 +1,4 @@ +int main() { + int* p; + &p; +} \ No newline at end of file diff --git a/test/unsupported/array_of_pointer_arg.c b/test/unsupported/array_of_pointer_arg.c new file mode 100644 index 0000000..7513795 --- /dev/null +++ b/test/unsupported/array_of_pointer_arg.c @@ -0,0 +1 @@ +void f(int* a[]) {} \ No newline at end of file diff --git a/test/unsupported/array_of_pointer_global.c b/test/unsupported/array_of_pointer_global.c new file mode 100644 index 0000000..5aeb229 --- /dev/null +++ b/test/unsupported/array_of_pointer_global.c @@ -0,0 +1 @@ +int* a[10]; \ No newline at end of file diff --git a/test/error/array_pointer.c b/test/unsupported/array_of_pointer_local.c similarity index 53% rename from test/error/array_pointer.c rename to test/unsupported/array_of_pointer_local.c index 1e83dbd..cf9adce 100644 --- a/test/error/array_pointer.c +++ b/test/unsupported/array_of_pointer_local.c @@ -1,3 +1,3 @@ -int main() { +void f() { int* a[10]; } \ No newline at end of file