mirror of
https://github.com/ziglang/zig.git
synced 2024-11-27 07:32:44 +00:00
parseh: fix __cdecl causing a fn ptr to be double ptr
This commit is contained in:
parent
87922bfae0
commit
baf889c879
@ -3886,17 +3886,19 @@ static TypeTableEntry *analyze_cast_expr(CodeGen *g, ImportTableEntry *import, B
|
||||
}
|
||||
|
||||
// explicit cast from pointer to another pointer
|
||||
if (actual_type->id == TypeTableEntryIdPointer &&
|
||||
wanted_type->id == TypeTableEntryIdPointer)
|
||||
if ((actual_type->id == TypeTableEntryIdPointer || actual_type->id == TypeTableEntryIdFn) &&
|
||||
(wanted_type->id == TypeTableEntryIdPointer || wanted_type->id == TypeTableEntryIdFn))
|
||||
{
|
||||
return resolve_cast(g, context, node, expr_node, wanted_type, CastOpPointerReinterpret, false);
|
||||
}
|
||||
|
||||
// explicit cast from maybe pointer to another maybe pointer
|
||||
if (actual_type->id == TypeTableEntryIdMaybe &&
|
||||
actual_type->data.maybe.child_type->id == TypeTableEntryIdPointer &&
|
||||
(actual_type->data.maybe.child_type->id == TypeTableEntryIdPointer ||
|
||||
actual_type->data.maybe.child_type->id == TypeTableEntryIdFn) &&
|
||||
wanted_type->id == TypeTableEntryIdMaybe &&
|
||||
wanted_type->data.maybe.child_type->id == TypeTableEntryIdPointer)
|
||||
(wanted_type->data.maybe.child_type->id == TypeTableEntryIdPointer ||
|
||||
wanted_type->data.maybe.child_type->id == TypeTableEntryIdFn))
|
||||
{
|
||||
return resolve_cast(g, context, node, expr_node, wanted_type, CastOpPointerReinterpret, false);
|
||||
}
|
||||
|
@ -314,6 +314,19 @@ static bool is_c_void_type(Context *c, TypeTableEntry *type_entry) {
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool qual_type_child_is_fn_proto(const QualType &qt) {
|
||||
if (qt.getTypePtr()->getTypeClass() == Type::Paren) {
|
||||
const ParenType *paren_type = static_cast<const ParenType *>(qt.getTypePtr());
|
||||
if (paren_type->getInnerType()->getTypeClass() == Type::FunctionProto) {
|
||||
return true;
|
||||
}
|
||||
} else if (qt.getTypePtr()->getTypeClass() == Type::Attributed) {
|
||||
const AttributedType *attr_type = static_cast<const AttributedType *>(qt.getTypePtr());
|
||||
return qual_type_child_is_fn_proto(attr_type->getEquivalentType());
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static TypeTableEntry *resolve_type_with_table(Context *c, const Type *ty, const Decl *decl,
|
||||
HashMap<Buf *, TypeTableEntry *, buf_hash, buf_eql_buf> *type_table)
|
||||
{
|
||||
@ -395,12 +408,9 @@ static TypeTableEntry *resolve_type_with_table(Context *c, const Type *ty, const
|
||||
return c->codegen->builtin_types.entry_invalid;
|
||||
}
|
||||
|
||||
if (child_qt.getTypePtr()->getTypeClass() == Type::Paren) {
|
||||
const ParenType *paren_type = static_cast<const ParenType *>(child_qt.getTypePtr());
|
||||
if (paren_type->getInnerType()->getTypeClass() == Type::FunctionProto) {
|
||||
if (qual_type_child_is_fn_proto(child_qt)) {
|
||||
return get_maybe_type(c->codegen, child_type);
|
||||
}
|
||||
}
|
||||
bool is_const = child_qt.isConstQualified();
|
||||
|
||||
TypeTableEntry *non_null_pointer_type = get_pointer_to_type(c->codegen, child_type, is_const);
|
||||
@ -1639,7 +1649,12 @@ int parse_h_file(ImportTableEntry *import, ZigList<ErrorMsg *> *errors, const ch
|
||||
unsigned offset = fsl.getManager().getFileOffset(fsl);
|
||||
const char *source = (const char *)fsl.getManager().getBufferData(file_id).bytes_begin();
|
||||
Buf *msg = buf_create_from_str((const char *)msg_str_ref.bytes_begin());
|
||||
Buf *path = buf_create_from_str((const char *)filename.bytes_begin());
|
||||
Buf *path;
|
||||
if (filename.empty()) {
|
||||
path = buf_alloc();
|
||||
} else {
|
||||
path = buf_create_from_mem((const char *)filename.bytes_begin(), filename.size());
|
||||
}
|
||||
|
||||
ErrorMsg *err_msg = err_msg_create_with_offset(path, line, column, offset, source, msg);
|
||||
|
||||
|
@ -1928,6 +1928,10 @@ extern void (*fn_ptr)(void);
|
||||
add_parseh_case("#define string", R"SOURCE(
|
||||
#define foo "a string"
|
||||
)SOURCE", 1, "pub const foo = c\"a string\";");
|
||||
|
||||
add_parseh_case("__cdecl doesn't mess up function pointers", R"SOURCE(
|
||||
void foo(void (__cdecl *fn_ptr)(void));
|
||||
)SOURCE", 1, "pub extern fn foo(fn_ptr: ?extern fn());");
|
||||
}
|
||||
|
||||
static void run_self_hosted_test(void) {
|
||||
|
Loading…
Reference in New Issue
Block a user