21 for (
size_t i = 0; i < search.
count; i++)
35 CTASSERTF(lhs != NULL && rhs != NULL,
"(lhs=%p, rhs=%p)", (
void *)lhs, (
void *)rhs);
48 if (lhs_kind != rhs_kind)
58 case eTreeTypeBool:
return true;
62 case eTreeTypeReference:
65 default:
return false;
82 if (lhs_kind == rhs_kind)
104 static bool can_cast_length(
size_t dst,
size_t src)
118 static tree_t *cast_check_length(
const tree_t *dst,
tree_t *expr,
size_t dstlen,
size_t srclen)
120 if (!can_cast_length(dstlen, srclen))
124 "creating an array with a length greater than its backing memory is unwise. (%s < %s)",
139 case eTreeTypePointer:
157 case eTreeTypePointer:
161 "cannot cast unrelated pointer types `%s` to `%s`",
165 return cast_check_length(dst, expr, dst->
length, src->
length);
204 case eTreeTypeBool:
return expr;
214 CTASSERTF(dst != NULL && expr != NULL,
"(dst=%p, expr=%p)", (
void *)dst, (
void *)expr);
227 case eTreeTypeOpaque:
return cast_to_opaque(dst, expr);
229 case eTreeTypePointer:
return cast_to_pointer(dst, expr);
233 case eTreeTypeDigit:
return cast_to_digit(dst, expr);
235 case eTreeTypeBool:
return cast_to_bool(dst, expr);
243 static bool eval_binary(mpz_t value,
const tree_t *expr)
263 case eBinaryAdd: mpz_add(value, lhs, rhs);
break;
264 case eBinarySub: mpz_sub(value, lhs, rhs);
break;
265 case eBinaryMul: mpz_mul(value, lhs, rhs);
break;
266 default:
return false;
272 static bool eval_cast(mpz_t value,
const tree_t *expr)
286 case eCastSignExtend: mpz_set(value, src);
break;
287 default:
return false;
302 case eTreeExprBinary:
303 return eval_binary(value, expr);
306 return eval_cast(value, expr);
308 default:
return false;
314 return length != SIZE_MAX;
326 return tree_is(type, eTreeTypeUnion) ||
tree_is(type, eTreeTypeStruct);
332 return tree_is(type, eTreeTypePointer);
338 return tree_is(type, eTreeTypeArray);
344 return tree_is(type, eTreeTypeOpaque);
350 return tree_is(type, eTreeTypeReference);
356 return tree_is(type, eTreeTypeDigit);
CT_PUREFN CT_TREE_API const node_t * tree_get_node(const tree_t *tree)
CT_PUREFN CT_TREE_API tree_kind_t tree_get_kind(const tree_t *tree)
CT_PUREFN CT_TREE_API const tree_t * tree_get_type(const tree_t *tree)
CT_PUREFN CT_TREE_API bool tree_is(const tree_t *self, tree_kind_t kind)
CT_TREE_API const char * tree_to_string(const tree_t *self)
bool util_type_is_aggregate(const tree_t *type)
bool util_type_is_digit(const tree_t *type)
const char * util_length_name(size_t length)
get the pretty name of a length return either the length as a string or "unbounded" if the length is ...
bool util_type_is_reference(const tree_t *type)
bool util_type_is_pointer(const tree_t *type)
bool util_length_bounded(size_t length)
check if the length of an array is bounded
bool util_type_is_array(const tree_t *type)
bool util_types_comparable(tree_cookie_t *cookie, const tree_t *lhs, const tree_t *rhs)
query two types for comparability in binary logic operations
bool util_type_is_opaque(const tree_t *type)
tree_t * util_type_cast(const tree_t *dst, tree_t *expr)
attempt to cast an expression to a type
CT_MEMORY_API arena_t * get_global_arena(void)
get the global memory arena
#define CTASSERT(expr)
assert a condition, prints the condition as a message
#define CTASSERTF(expr,...)
assert a condition with a message and optional format arguments
void * util_select_decl(tree_t *sema, search_t search, const char *name)
search for a declaration by name in a set of tags
bool util_eval_digit(mpz_t value, const tree_t *expr)
evaluate a digit expression
bool util_types_equal(const tree_t *lhs, const tree_t *rhs)
compare two types for strict equality compares two types for exact equality, does not follow typedefs
CT_TREE_API tree_t * tree_expr_cast(const node_t *node, const tree_t *type, const tree_t *expr, tree_cast_t cast)
create a cast expression
CT_TREE_API tree_t * tree_error(const node_t *node, const diagnostic_t *diagnostic, const char *message,...)
CT_TREE_API tree_t * tree_resolve_type(tree_cookie_t *cookie, const tree_t *decl)
tree_kind_t
all tree node types
CT_TREE_API const tree_t * tree_follow_type(const tree_t *type)
CT_TREE_API void * tree_module_get(tree_t *self, size_t tag, const char *name)
recursively search for a declaration in a module
STA_DECL char * str_format(arena_t *arena, const char *fmt,...)