[CIR] Use destination type when emitting constant function ptrs (#189741)

This updates the CIR constant emitter to use the correct destination
type when emitting a constant initializer for a structure that might be
initialized with non-prototyped function pointers. We were previously
using the type from whatever function declaration we had, but this may
not be the correct type.

This change also updates the `replaceUsesOfNonProtoTypeWithRealFunction`
to ignore global initializer uses, which do not need to be updated after
this change.
This commit is contained in:
Andy Kaylor 2026-04-03 10:58:46 -07:00 committed by GitHub
parent ffd29734cc
commit 68b6a27771
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 85 additions and 3 deletions

View File

@ -1383,8 +1383,15 @@ ConstantLValueEmitter::tryEmitBase(const APValue::LValueBase &base) {
cir::FuncOp fop = cgm.getAddrOfFunction(fd);
CIRGenBuilderTy &builder = cgm.getBuilder();
mlir::MLIRContext *mlirContext = builder.getContext();
// Use the destination pointer type (e.g. struct field type), not
// fop.getFunctionType(), so initializers stay valid when a no-prototype
// FuncOp is later replaced by a prototyped definition with the same
// symbol. CIR allows the view type to differ from the symbol's type.
mlir::Type ptrTy = cgm.getTypes().convertTypeForMem(destType);
assert(mlir::isa<cir::PointerType>(ptrTy) &&
"function address in constant must be a pointer");
return cir::GlobalViewAttr::get(
builder.getPointerTo(fop.getFunctionType()),
ptrTy,
mlir::FlatSymbolRefAttr::get(mlirContext, fop.getSymNameAttr()));
}

View File

@ -1735,9 +1735,13 @@ void CIRGenModule::replaceUsesOfNonProtoTypeWithRealFunction(
// Replace type
getGlobalOp.getAddr().setType(
cir::PointerType::get(newFn.getFunctionType()));
} else if (mlir::isa<cir::GlobalOp>(use.getUser())) {
// Function addresses in global initializers use GlobalViewAttrs typed to
// the initializer context (e.g. struct field type), not the FuncOp type,
// so no update is required when the no-proto FuncOp is replaced.
} else {
errorNYI(use.getUser()->getLoc(),
"replaceUsesOfNonProtoTypeWithRealFunction: unexpected use");
llvm_unreachable(
"replaceUsesOfNonProtoTypeWithRealFunction: unexpected use type");
}
}
}

View File

@ -0,0 +1,71 @@
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -w -fclangir -emit-cir %s -o %t.cir
// RUN: FileCheck --input-file=%t.cir %s
// Because LLVM IR uses opaque pointers, nothing in this test would be worth
// checking there, so LLVM and OGCG checks are omitted.
// Exercise replacing a no-prototype declaration with a prototyped definition
// when a global struct initializer already took the address of that function.
// The initializer must use the struct field's function-pointer type (not the
// FuncOp's no-prototype type) so CIR stays valid after the replacement.
// CHECK: !{{[^ ]+}} = !cir.record<struct "S1" {!cir.ptr<!cir.func<()>>, !s32i, !s32i}>
// CHECK: !{{[^ ]+}} = !cir.record<struct "S2" {!cir.ptr<!cir.func<(...)>>, !s32i, !s32i}>
// CHECK: !{{[^ ]+}} = !cir.record<struct "S3" {!cir.ptr<!cir.func<(...)>>, !s32i, !s32i}>
// CHECK: !{{[^ ]+}} = !cir.record<struct "S4" {!cir.ptr<!cir.func<(...)>>, !s32i, !s32i}>
struct S1 {
void (*fn_ptr)(void);
int a;
int b;
};
void f1();
struct S1 s1 = {f1, 0, 1};
void f1(void) {}
// CHECK: cir.global {{.*}} @s1 = #cir.const_record<{#cir.global_view<@f1> : !cir.ptr<!cir.func<()>>, #cir.int<0> : !s32i, #cir.int<1> : !s32i}>
struct S2 {
void (*fn_ptr)();
int a;
int b;
};
void f2();
struct S2 s2 = {f2, 0, 1};
void f2(void) {}
// CHECK: cir.global {{.*}} @s2 = #cir.const_record<{#cir.global_view<@f2> : !cir.ptr<!cir.func<(...)>>, #cir.int<0> : !s32i, #cir.int<1> : !s32i}>
struct S3 {
void (*fn_ptr)();
int a;
int b;
};
void f3();
struct S3 s3 = {f3, 0, 1};
void f3(int x) {}
// CHECK: cir.global {{.*}} @s3 = #cir.const_record<{#cir.global_view<@f3> : !cir.ptr<!cir.func<(...)>>, #cir.int<0> : !s32i, #cir.int<1> : !s32i}>
struct S4 {
void (*fn_ptr)();
int a;
int b;
};
void f4(int x);
// In this case we are initializing with a fully prototyped function. The
// initializer should still match the struct field's function-pointer type.
struct S4 s4 = {f4, 0, 1};
// CHECK: cir.global {{.*}} @s4 = #cir.const_record<{#cir.global_view<@f4> : !cir.ptr<!cir.func<(...)>>, #cir.int<0> : !s32i, #cir.int<1> : !s32i}>