aboutsummaryrefslogtreecommitdiff
path: root/src/check_expr.cpp
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2019-09-17 19:47:04 +0100
committergingerBill <bill@gingerbill.org>2019-09-17 19:47:04 +0100
commit68582c5ad1b2bf562242b9d2f40c89efad343b66 (patch)
tree17de290db1fd5ea483f1c06a4ddf776c78b9bcfb /src/check_expr.cpp
parentda3467c25f72bdd04a03e7db0924ba29b33cc593 (diff)
Add suggestions to errors on casts and assignments.
Diffstat (limited to 'src/check_expr.cpp')
-rw-r--r--src/check_expr.cpp93
1 files changed, 81 insertions, 12 deletions
diff --git a/src/check_expr.cpp b/src/check_expr.cpp
index 8a5ed9c1c..5ef42a3fc 100644
--- a/src/check_expr.cpp
+++ b/src/check_expr.cpp
@@ -90,7 +90,7 @@ Type *type_to_abi_compat_param_type(gbAllocator a, Type *original_type, ProcCall
Type *type_to_abi_compat_result_type(gbAllocator a, Type *original_type, ProcCallingConvention cc);
bool abi_compat_return_by_pointer(gbAllocator a, ProcCallingConvention cc, Type *abi_return_type);
void set_procedure_abi_types(CheckerContext *c, Type *type);
-
+void check_assignment_error_suggestion(CheckerContext *c, Operand *o, Type *type);
Entity *entity_from_expr(Ast *expr) {
expr = unparen_expr(expr);
@@ -786,6 +786,7 @@ void check_assignment(CheckerContext *c, Operand *operand, Type *type, String co
op_type_str,
type_str,
LIT(context_name));
+ check_assignment_error_suggestion(c, operand, type);
break;
}
operand->mode = Addressing_Invalid;
@@ -1509,32 +1510,98 @@ bool check_representable_as_constant(CheckerContext *c, ExactValue in_value, Typ
return false;
}
+
+void check_assignment_error_suggestion(CheckerContext *c, Operand *o, Type *type) {
+ gbString a = expr_to_string(o->expr);
+ gbString b = type_to_string(type);
+ defer(
+ gb_string_free(b);
+ gb_string_free(a);
+ );
+
+ Type *src = base_type(o->type);
+ Type *dst = base_type(type);
+
+ if (is_type_array(src) && is_type_slice(dst)) {
+ Type *s = src->Array.elem;
+ Type *d = dst->Slice.elem;
+ if (are_types_identical(s, d)) {
+ error_line("\tSuggestion: the array expression may be sliced with %s[:]\n", a);
+ }
+ } else if (are_types_identical(src, dst)) {
+ error_line("\tSuggestion: the expression may be directly casted to type %s\n", b);
+ } else if (are_types_identical(src, t_string) && is_type_u8_slice(dst)) {
+ error_line("\tSuggestion: a string may be casted to %s\n", a, b);
+ } else if (is_type_u8_slice(src) && are_types_identical(dst, t_string)) {
+ error_line("\tSuggestion: the expression may be casted to %s\n", b);
+ }
+}
+
+void check_cast_error_suggestion(CheckerContext *c, Operand *o, Type *type) {
+ gbString a = expr_to_string(o->expr);
+ gbString b = type_to_string(type);
+ defer(
+ gb_string_free(b);
+ gb_string_free(a);
+ );
+
+ Type *src = base_type(o->type);
+ Type *dst = base_type(type);
+
+ if (is_type_array(src) && is_type_slice(dst)) {
+ Type *s = src->Array.elem;
+ Type *d = dst->Slice.elem;
+ if (are_types_identical(s, d)) {
+ error_line("\tSuggestion: the array expression may be sliced with %s[:]\n", a);
+ }
+ } else if (is_type_pointer(o->type) && is_type_integer(type)) {
+ if (is_type_uintptr(type)) {
+ error_line("\tSuggestion: a pointer may be directly casted to %s\n", b);
+ } else {
+ error_line("\tSuggestion: for a pointer to be casted to an integer, it must be converted to 'uintptr' first\n");
+ i64 x = type_size_of(o->type);
+ i64 y = type_size_of(type);
+ if (x != y) {
+ error_line("\tNote: the type of expression and the type of the cast have a different size in bytes, %lld vs %lld\n", x, y);
+ }
+ }
+ } else if (is_type_integer(o->type) && is_type_pointer(type)) {
+ if (is_type_uintptr(o->type)) {
+ error_line("\tSuggestion: %a may be directly casted to %s\n", a, b);
+ } else {
+ error_line("\tSuggestion: for an integer to be casted to a pointer, it must be converted to 'uintptr' first\n");
+ }
+ } else if (are_types_identical(src, t_string) && is_type_u8_slice(dst)) {
+ error_line("\tSuggestion: a string may be casted to %s\n", a, b);
+ } else if (is_type_u8_slice(src) && are_types_identical(dst, t_string)) {
+ error_line("\tSuggestion: the expression may be casted to %s\n", b);
+ }
+}
+
+
void check_is_expressible(CheckerContext *c, Operand *o, Type *type) {
GB_ASSERT(is_type_constant_type(type));
GB_ASSERT(o->mode == Addressing_Constant);
if (!check_representable_as_constant(c, o->value, type, &o->value)) {
gbString a = expr_to_string(o->expr);
gbString b = type_to_string(type);
+ defer(
+ gb_string_free(b);
+ gb_string_free(a);
+ o->mode = Addressing_Invalid;
+ );
+
if (is_type_numeric(o->type) && is_type_numeric(type)) {
if (!is_type_integer(o->type) && is_type_integer(type)) {
error(o->expr, "'%s' truncated to '%s'", a, b);
} else {
- #if 0
- gbAllocator ha = heap_allocator();
- String str = big_int_to_string(ha, &o->value.value_integer);
- defer (gb_free(ha, str.text));
- error(o->expr, "'%s = %.*s' overflows '%s'", a, LIT(str), b);
- #else
error(o->expr, "Cannot convert '%s' to '%s'", a, b);
- #endif
+ check_assignment_error_suggestion(c, o, type);
}
} else {
error(o->expr, "Cannot convert '%s' to '%s'", a, b);
+ check_assignment_error_suggestion(c, o, type);
}
-
- gb_string_free(b);
- gb_string_free(a);
- o->mode = Addressing_Invalid;
}
}
@@ -2165,6 +2232,8 @@ void check_cast(CheckerContext *c, Operand *x, Type *type) {
gb_string_free(to_type);
gb_string_free(expr_str);
+ check_cast_error_suggestion(c, x, type);
+
x->mode = Addressing_Invalid;
return;
}