[CIR] Upstream __builtin_creal for ComplexType (#146927)
Upstream `__builtin_creal` support for ComplexType https://github.com/llvm/llvm-project/issues/141365
This commit is contained in:
parent
9b5959dd9a
commit
ddfc13c191
@ -83,13 +83,6 @@ RValue CIRGenFunction::emitBuiltinExpr(const GlobalDecl &gd, unsigned builtinID,
|
|||||||
|
|
||||||
const FunctionDecl *fd = gd.getDecl()->getAsFunction();
|
const FunctionDecl *fd = gd.getDecl()->getAsFunction();
|
||||||
|
|
||||||
// If this is an alias for a lib function (e.g. __builtin_sin), emit
|
|
||||||
// the call using the normal call path, but using the unmangled
|
|
||||||
// version of the function name.
|
|
||||||
if (getContext().BuiltinInfo.isLibFunction(builtinID))
|
|
||||||
return emitLibraryCall(*this, fd, e,
|
|
||||||
cgm.getBuiltinLibFunction(fd, builtinID));
|
|
||||||
|
|
||||||
assert(!cir::MissingFeatures::builtinCallF128());
|
assert(!cir::MissingFeatures::builtinCallF128());
|
||||||
|
|
||||||
// If the builtin has been declared explicitly with an assembler label,
|
// If the builtin has been declared explicitly with an assembler label,
|
||||||
@ -124,6 +117,17 @@ RValue CIRGenFunction::emitBuiltinExpr(const GlobalDecl &gd, unsigned builtinID,
|
|||||||
return RValue::get(complex);
|
return RValue::get(complex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case Builtin::BI__builtin_creal:
|
||||||
|
case Builtin::BI__builtin_crealf:
|
||||||
|
case Builtin::BI__builtin_creall:
|
||||||
|
case Builtin::BIcreal:
|
||||||
|
case Builtin::BIcrealf:
|
||||||
|
case Builtin::BIcreall: {
|
||||||
|
mlir::Value complex = emitComplexExpr(e->getArg(0));
|
||||||
|
mlir::Value real = builder.createComplexReal(loc, complex);
|
||||||
|
return RValue::get(real);
|
||||||
|
}
|
||||||
|
|
||||||
case Builtin::BI__builtin_clrsb:
|
case Builtin::BI__builtin_clrsb:
|
||||||
case Builtin::BI__builtin_clrsbl:
|
case Builtin::BI__builtin_clrsbl:
|
||||||
case Builtin::BI__builtin_clrsbll:
|
case Builtin::BI__builtin_clrsbll:
|
||||||
@ -192,6 +196,13 @@ RValue CIRGenFunction::emitBuiltinExpr(const GlobalDecl &gd, unsigned builtinID,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If this is an alias for a lib function (e.g. __builtin_sin), emit
|
||||||
|
// the call using the normal call path, but using the unmangled
|
||||||
|
// version of the function name.
|
||||||
|
if (getContext().BuiltinInfo.isLibFunction(builtinID))
|
||||||
|
return emitLibraryCall(*this, fd, e,
|
||||||
|
cgm.getBuiltinLibFunction(fd, builtinID));
|
||||||
|
|
||||||
cgm.errorNYI(e->getSourceRange(), "unimplemented builtin call");
|
cgm.errorNYI(e->getSourceRange(), "unimplemented builtin call");
|
||||||
return getUndefRValue(e->getType());
|
return getUndefRValue(e->getType());
|
||||||
}
|
}
|
||||||
|
@ -1094,8 +1094,7 @@ RValue CIRGenFunction::emitAnyExpr(const Expr *e, AggValueSlot aggSlot) {
|
|||||||
case cir::TEK_Scalar:
|
case cir::TEK_Scalar:
|
||||||
return RValue::get(emitScalarExpr(e));
|
return RValue::get(emitScalarExpr(e));
|
||||||
case cir::TEK_Complex:
|
case cir::TEK_Complex:
|
||||||
cgm.errorNYI(e->getSourceRange(), "emitAnyExpr: complex type");
|
return RValue::getComplex(emitComplexExpr(e));
|
||||||
return RValue::get(nullptr);
|
|
||||||
case cir::TEK_Aggregate: {
|
case cir::TEK_Aggregate: {
|
||||||
if (aggSlot.isIgnored())
|
if (aggSlot.isIgnored())
|
||||||
aggSlot = createAggTemp(e->getType(), getLoc(e->getSourceRange()),
|
aggSlot = createAggTemp(e->getType(), getLoc(e->getSourceRange()),
|
||||||
|
@ -33,3 +33,28 @@ void foo() {
|
|||||||
// OGCG: %[[R_IMAG_PTR:.*]] = getelementptr inbounds nuw { i32, i32 }, ptr %[[COMPLEX_R]], i32 0, i32 1
|
// OGCG: %[[R_IMAG_PTR:.*]] = getelementptr inbounds nuw { i32, i32 }, ptr %[[COMPLEX_R]], i32 0, i32 1
|
||||||
// OGCG: store i32 %[[A_REAL]], ptr %[[R_REAL_PTR]], align 4
|
// OGCG: store i32 %[[A_REAL]], ptr %[[R_REAL_PTR]], align 4
|
||||||
// OGCG: store i32 %[[A_IMAG]], ptr %[[R_IMAG_PTR]], align 4
|
// OGCG: store i32 %[[A_IMAG]], ptr %[[R_IMAG_PTR]], align 4
|
||||||
|
|
||||||
|
void foo2() {
|
||||||
|
double _Complex a;
|
||||||
|
double real = __builtin_creal(a);
|
||||||
|
}
|
||||||
|
|
||||||
|
// CIR: %[[COMPLEX:.*]] = cir.alloca !cir.complex<!cir.double>, !cir.ptr<!cir.complex<!cir.double>>, ["a"]
|
||||||
|
// CIR: %[[INIT:.*]] = cir.alloca !cir.double, !cir.ptr<!cir.double>, ["real", init]
|
||||||
|
// CIR: %[[TMP:.*]] = cir.load{{.*}} %[[COMPLEX]] : !cir.ptr<!cir.complex<!cir.double>>, !cir.complex<!cir.double>
|
||||||
|
// CIR: %[[REAL:.*]] = cir.complex.real %[[TMP]] : !cir.complex<!cir.double> -> !cir.double
|
||||||
|
// CIR: cir.store{{.*}} %[[REAL]], %[[INIT]] : !cir.double, !cir.ptr<!cir.double>
|
||||||
|
|
||||||
|
// LLVM: %[[COMPLEX:.*]] = alloca { double, double }, i64 1, align 8
|
||||||
|
// LLVM: %[[INIT:.*]] = alloca double, i64 1, align 8
|
||||||
|
// LLVM: %[[TMP:.*]] = load { double, double }, ptr %[[COMPLEX]], align 8
|
||||||
|
// LLVM: %[[REAL:.*]] = extractvalue { double, double } %[[TMP]], 0
|
||||||
|
// LLVM: store double %[[REAL]], ptr %[[INIT]], align 8
|
||||||
|
|
||||||
|
// OGCG: %[[COMPLEX:.*]] = alloca { double, double }, align 8
|
||||||
|
// OGCG: %[[INIT:.*]] = alloca double, align 8
|
||||||
|
// OGCG: %[[A_REAL_PTR:.*]] = getelementptr inbounds nuw { double, double }, ptr %[[COMPLEX]], i32 0, i32 0
|
||||||
|
// OGCG: %[[A_REAL:.*]] = load double, ptr %[[A_REAL_PTR]], align 8
|
||||||
|
// OGCG: %[[A_IMAG_PTR:.*]] = getelementptr inbounds nuw { double, double }, ptr %[[COMPLEX]], i32 0, i32 1
|
||||||
|
// OGCG: %[[A_IMAG:.*]] = load double, ptr %[[A_IMAG_PTR]], align 8
|
||||||
|
// OGCG: store double %[[A_REAL]], ptr %[[INIT]], align 8
|
||||||
|
Loading…
x
Reference in New Issue
Block a user