From caedb995230f084b0c217b5ba85659c6dd8a071a Mon Sep 17 00:00:00 2001 From: Sirui Mu Date: Tue, 31 Dec 2024 17:45:50 +0800 Subject: [PATCH] [CIR] Add support for GCC function attribute "const" and "pure" This patch adds support for the following GCC function attributes: - __attribute__((const)) - __attribute__((pure)) --- .../CIR/Dialect/Builder/CIRBaseBuilder.h | 25 +++++---- clang/include/clang/CIR/Dialect/IR/CIROps.td | 52 +++++++++++++++++-- .../clang/CIR/Interfaces/CIROpInterfaces.td | 19 ++++--- clang/lib/CIR/CodeGen/CIRGenCall.cpp | 45 +++++++++------- clang/lib/CIR/CodeGen/CIRGenModule.cpp | 3 +- clang/lib/CIR/CodeGen/CIRGenModule.h | 3 +- clang/lib/CIR/Dialect/IR/CIRDialect.cpp | 40 +++++++++++--- .../CIR/Lowering/DirectToLLVM/LowerToLLVM.cpp | 47 +++++++++++++++++ .../CIR/Lowering/DirectToLLVM/LowerToLLVM.h | 5 ++ clang/test/CIR/CodeGen/call-side-effect.cpp | 25 +++++++++ clang/test/CIR/IR/side-effect.cir | 21 ++++++++ clang/test/CIR/Lowering/call.cir | 19 +++++++ 12 files changed, 255 insertions(+), 49 deletions(-) create mode 100644 clang/test/CIR/CodeGen/call-side-effect.cpp create mode 100644 clang/test/CIR/IR/side-effect.cir diff --git a/clang/include/clang/CIR/Dialect/Builder/CIRBaseBuilder.h b/clang/include/clang/CIR/Dialect/Builder/CIRBaseBuilder.h index 771b7dd33cd4..502fd0d52524 100644 --- a/clang/include/clang/CIR/Dialect/Builder/CIRBaseBuilder.h +++ b/clang/include/clang/CIR/Dialect/Builder/CIRBaseBuilder.h @@ -652,10 +652,11 @@ class CIRBaseBuilderTy : public mlir::OpBuilder { mlir::Type returnType = cir::VoidType(), mlir::ValueRange operands = mlir::ValueRange(), cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { - cir::CallOp callOp = - create(loc, callee, returnType, operands, callingConv); + cir::CallOp callOp = create(loc, callee, returnType, operands, + callingConv, sideEffect); if (extraFnAttr) { callOp->setAttr("extra_attrs", extraFnAttr); @@ -671,10 +672,11 @@ class CIRBaseBuilderTy : public mlir::OpBuilder { cir::CallOp createCallOp(mlir::Location loc, cir::FuncOp callee, mlir::ValueRange operands = mlir::ValueRange(), cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { return createCallOp(loc, mlir::SymbolRefAttr::get(callee), callee.getFunctionType().getReturnType(), operands, - callingConv, extraFnAttr); + callingConv, sideEffect, extraFnAttr); } cir::CallOp @@ -682,21 +684,23 @@ class CIRBaseBuilderTy : public mlir::OpBuilder { cir::FuncType fn_type, mlir::ValueRange operands = mlir::ValueRange(), cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { llvm::SmallVector resOperands({ind_target}); resOperands.append(operands.begin(), operands.end()); return createCallOp(loc, mlir::SymbolRefAttr(), fn_type.getReturnType(), - resOperands, callingConv, extraFnAttr); + resOperands, callingConv, sideEffect, extraFnAttr); } cir::CallOp createCallOp(mlir::Location loc, mlir::SymbolRefAttr callee, mlir::ValueRange operands = mlir::ValueRange(), cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { return createCallOp(loc, callee, cir::VoidType(), operands, callingConv, - extraFnAttr); + sideEffect, extraFnAttr); } cir::CallOp @@ -705,10 +709,11 @@ class CIRBaseBuilderTy : public mlir::OpBuilder { mlir::Type returnType = cir::VoidType(), mlir::ValueRange operands = mlir::ValueRange(), cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { cir::CallOp tryCallOp = create(loc, callee, returnType, operands, callingConv, - /*exception=*/getUnitAttr()); + sideEffect, /*exception=*/getUnitAttr()); if (extraFnAttr) { tryCallOp->setAttr("extra_attrs", extraFnAttr); } else { @@ -724,20 +729,22 @@ class CIRBaseBuilderTy : public mlir::OpBuilder { createTryCallOp(mlir::Location loc, cir::FuncOp callee, mlir::ValueRange operands, cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All, cir::ExtraFuncAttributesAttr extraFnAttr = {}) { return createTryCallOp(loc, mlir::SymbolRefAttr::get(callee), callee.getFunctionType().getReturnType(), operands, - callingConv, extraFnAttr); + callingConv, sideEffect, extraFnAttr); } cir::CallOp createIndirectTryCallOp(mlir::Location loc, mlir::Value ind_target, cir::FuncType fn_type, mlir::ValueRange operands, - cir::CallingConv callingConv = cir::CallingConv::C) { + cir::CallingConv callingConv = cir::CallingConv::C, + cir::SideEffect sideEffect = cir::SideEffect::All) { llvm::SmallVector resOperands({ind_target}); resOperands.append(operands.begin(), operands.end()); return createTryCallOp(loc, mlir::SymbolRefAttr(), fn_type.getReturnType(), - resOperands, callingConv); + resOperands, callingConv, sideEffect); } struct GetMethodResults { diff --git a/clang/include/clang/CIR/Dialect/IR/CIROps.td b/clang/include/clang/CIR/Dialect/IR/CIROps.td index e02194ad15e0..08390eb65691 100644 --- a/clang/include/clang/CIR/Dialect/IR/CIROps.td +++ b/clang/include/clang/CIR/Dialect/IR/CIROps.td @@ -3574,6 +3574,39 @@ def DeleteArrayOp : CIR_Op<"delete.array">, // CallOp and TryCallOp //===----------------------------------------------------------------------===// +def SE_All : I32EnumAttrCase<"All", 1, "all">; +def SE_Pure : I32EnumAttrCase<"Pure", 2, "pure">; +def SE_Const : I32EnumAttrCase<"Const", 3, "const">; + +def SideEffect : I32EnumAttr< + "SideEffect", "allowed side effects of a function", + [SE_All, SE_Pure, SE_Const]> { + let description = [{ + The `#cir.SideEffect` attribute specifies the possible side effects of the + callee of a call operation. This is an enumeration attribute and all + possible enumerators are: + + - all: The callee can have any side effects. This is the default if no side + effects are explicitly listed. + - pure: The callee may read data from memory, but it cannot write data to + memory. This has the same effect as the GNU C/C++ attribute + `__attribute__((pure))`. + - const: The callee may not read or write data from memory. This has the + same effect as the GNU C/C++ attribute `__attribute__((const))`. + + Examples: + + ```mlir + %0 = cir.const #cir.int<0> : !s32i + %1 = cir.const #cir.int<1> : !s32i + %2 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(all) + %2 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(pure) + %2 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(const) + ``` + }]; + let cppNamespace = "::cir"; +} + class CIR_CallOp extra_traits = []> : Op extra_traits = []> : OptionalAttr:$callee, Variadic:$arg_ops, DefaultValuedAttr:$calling_conv, + DefaultValuedAttr:$side_effect, ExtraFuncAttr:$extra_attrs, OptionalAttr:$ast ); @@ -3676,12 +3710,15 @@ def CallOp : CIR_CallOp<"call", [NoRegionArguments]> { OpBuilder<(ins "mlir::SymbolRefAttr":$callee, "mlir::Type":$resType, CArg<"mlir::ValueRange", "{}">:$operands, CArg<"CallingConv", "CallingConv::C">:$callingConv, + CArg<"SideEffect", "SideEffect::All">:$sideEffect, CArg<"mlir::UnitAttr", "{}">:$exception), [{ $_state.addOperands(operands); if (callee) $_state.addAttribute("callee", callee); $_state.addAttribute("calling_conv", CallingConvAttr::get($_builder.getContext(), callingConv)); + $_state.addAttribute("side_effect", + SideEffectAttr::get($_builder.getContext(), sideEffect)); if (exception) $_state.addAttribute("exception", exception); if (resType && !isa(resType)) @@ -3693,6 +3730,7 @@ def CallOp : CIR_CallOp<"call", [NoRegionArguments]> { "FuncType":$fn_type, CArg<"mlir::ValueRange", "{}">:$operands, CArg<"CallingConv", "CallingConv::C">:$callingConv, + CArg<"SideEffect", "SideEffect::All">:$sideEffect, CArg<"mlir::UnitAttr", "{}">:$exception), [{ $_state.addOperands(ValueRange{ind_target}); $_state.addOperands(operands); @@ -3700,6 +3738,8 @@ def CallOp : CIR_CallOp<"call", [NoRegionArguments]> { $_state.addTypes(fn_type.getReturnType()); $_state.addAttribute("calling_conv", CallingConvAttr::get($_builder.getContext(), callingConv)); + $_state.addAttribute("side_effect", + SideEffectAttr::get($_builder.getContext(), sideEffect)); if (exception) $_state.addAttribute("exception", exception); // Create region placeholder for potential cleanups. @@ -3742,7 +3782,8 @@ def TryCallOp : CIR_CallOp<"try_call", CArg<"mlir::ValueRange", "{}">:$operands, CArg<"mlir::ValueRange", "{}">:$contOperands, CArg<"mlir::ValueRange", "{}">:$landingPadOperands, - CArg<"CallingConv", "CallingConv::C">:$callingConv), [{ + CArg<"CallingConv", "CallingConv::C">:$callingConv, + CArg<"SideEffect", "SideEffect::All">:$sideEffect), [{ $_state.addOperands(operands); if (callee) $_state.addAttribute("callee", callee); @@ -3751,6 +3792,8 @@ def TryCallOp : CIR_CallOp<"try_call", $_state.addAttribute("calling_conv", CallingConvAttr::get($_builder.getContext(), callingConv)); + $_state.addAttribute("side_effect", + SideEffectAttr::get($_builder.getContext(), sideEffect)); // Handle branches $_state.addOperands(contOperands); @@ -3771,7 +3814,8 @@ def TryCallOp : CIR_CallOp<"try_call", CArg<"mlir::ValueRange", "{}">:$operands, CArg<"mlir::ValueRange", "{}">:$contOperands, CArg<"mlir::ValueRange", "{}">:$landingPadOperands, - CArg<"CallingConv", "CallingConv::C">:$callingConv), [{ + CArg<"CallingConv", "CallingConv::C">:$callingConv, + CArg<"SideEffect", "SideEffect::All">:$sideEffect), [{ ::llvm::SmallVector finalCallOperands({ind_target}); finalCallOperands.append(operands.begin(), operands.end()); $_state.addOperands(finalCallOperands); @@ -3781,6 +3825,8 @@ def TryCallOp : CIR_CallOp<"try_call", $_state.addAttribute("calling_conv", CallingConvAttr::get($_builder.getContext(), callingConv)); + $_state.addAttribute("side_effect", + SideEffectAttr::get($_builder.getContext(), sideEffect)); // Handle branches $_state.addOperands(contOperands); @@ -4187,7 +4233,7 @@ def MemCpyInlineOp : CIR_MemOp<"memcpy_inline"> { Given two CIR pointers, `src` and `dst`, `memcpy_inline` will copy `len` bytes from the memory pointed by `src` to the memory pointed by `dst`. - Unlike `cir.libc.memcpy`, this Op guarantees that no external functions + Unlike `cir.libc.memcpy`, this Op guarantees that no external functions are called, and length of copied bytes is a constant. Examples: diff --git a/clang/include/clang/CIR/Interfaces/CIROpInterfaces.td b/clang/include/clang/CIR/Interfaces/CIROpInterfaces.td index 445a558debda..6437112cd451 100644 --- a/clang/include/clang/CIR/Interfaces/CIROpInterfaces.td +++ b/clang/include/clang/CIR/Interfaces/CIROpInterfaces.td @@ -37,6 +37,9 @@ let cppNamespace = "::cir" in { InterfaceMethod< "Return the calling convention of the call operation", "cir::CallingConv", "getCallingConv", (ins)>, + InterfaceMethod< + "Return the side effects of the call operation", + "cir::SideEffect", "getSideEffect", (ins)>, ]; } @@ -50,20 +53,20 @@ let cppNamespace = "::cir" in { >, InterfaceMethod<"", "bool", "hasLocalLinkage", (ins), [{}], - /*defaultImplementation=*/[{ - return cir::isLocalLinkage($_op.getLinkage()); + /*defaultImplementation=*/[{ + return cir::isLocalLinkage($_op.getLinkage()); }] >, InterfaceMethod<"", "bool", "hasExternalWeakLinkage", (ins), [{}], - /*defaultImplementation=*/[{ - return cir::isExternalWeakLinkage($_op.getLinkage()); + /*defaultImplementation=*/[{ + return cir::isExternalWeakLinkage($_op.getLinkage()); }] >, InterfaceMethod<"", "bool", "isExternalLinkage", (ins), [{}], - /*defaultImplementation=*/[{ - return cir::isExternalLinkage($_op.getLinkage()); + /*defaultImplementation=*/[{ + return cir::isExternalLinkage($_op.getLinkage()); }] >, InterfaceMethod<"", @@ -87,8 +90,8 @@ let cppNamespace = "::cir" in { }] >, ]; - let extraClassDeclaration = [{ - bool hasDefaultVisibility(); + let extraClassDeclaration = [{ + bool hasDefaultVisibility(); bool canBenefitFromLocalAlias(); }]; } diff --git a/clang/lib/CIR/CodeGen/CIRGenCall.cpp b/clang/lib/CIR/CodeGen/CIRGenCall.cpp index 45bcdbf40cee..dd625573c79b 100644 --- a/clang/lib/CIR/CodeGen/CIRGenCall.cpp +++ b/clang/lib/CIR/CodeGen/CIRGenCall.cpp @@ -345,12 +345,10 @@ static void AddAttributesFromFunctionProtoType(CIRGenBuilderTy &builder, /// attributes that restrict how the frontend generates code must be /// added here rather than getDefaultFunctionAttributes. /// -void CIRGenModule::constructAttributeList(StringRef Name, - const CIRGenFunctionInfo &FI, - CIRGenCalleeInfo CalleeInfo, - mlir::NamedAttrList &funcAttrs, - cir::CallingConv &callingConv, - bool AttrOnCallSite, bool IsThunk) { +void CIRGenModule::constructAttributeList( + StringRef Name, const CIRGenFunctionInfo &FI, CIRGenCalleeInfo CalleeInfo, + mlir::NamedAttrList &funcAttrs, cir::CallingConv &callingConv, + cir::SideEffect &sideEffect, bool AttrOnCallSite, bool IsThunk) { // Implementation Disclaimer // // UnimplementedFeature and asserts are used throughout the code to track @@ -364,6 +362,7 @@ void CIRGenModule::constructAttributeList(StringRef Name, // Collect function CIR attributes from the CC lowering. callingConv = FI.getEffectiveCallingConvention(); + sideEffect = cir::SideEffect::All; // TODO: NoReturn, cmse_nonsecure_call // Collect function CIR attributes from the callee prototype if we have one. @@ -421,8 +420,10 @@ void CIRGenModule::constructAttributeList(StringRef Name, if (TargetDecl->hasAttr()) { // gcc specifies that 'const' functions have greater restrictions than // 'pure' functions, so they also cannot have infinite loops. + sideEffect = cir::SideEffect::Const; } else if (TargetDecl->hasAttr()) { // gcc specifies that 'pure' functions cannot have infinite loops. + sideEffect = cir::SideEffect::Pure; } else if (TargetDecl->hasAttr()) { } @@ -466,11 +467,13 @@ void CIRGenModule::constructAttributeList(StringRef Name, getDefaultFunctionAttributes(Name, HasOptnone, AttrOnCallSite, funcAttrs); } -static cir::CIRCallOpInterface emitCallLikeOp( - CIRGenFunction &CGF, mlir::Location callLoc, cir::FuncType indirectFuncTy, - mlir::Value indirectFuncVal, cir::FuncOp directFuncOp, - SmallVectorImpl &CIRCallArgs, bool isInvoke, - cir::CallingConv callingConv, cir::ExtraFuncAttributesAttr extraFnAttrs) { +static cir::CIRCallOpInterface +emitCallLikeOp(CIRGenFunction &CGF, mlir::Location callLoc, + cir::FuncType indirectFuncTy, mlir::Value indirectFuncVal, + cir::FuncOp directFuncOp, + SmallVectorImpl &CIRCallArgs, bool isInvoke, + cir::CallingConv callingConv, cir::SideEffect sideEffect, + cir::ExtraFuncAttributesAttr extraFnAttrs) { auto &builder = CGF.getBuilder(); auto getOrCreateSurroundingTryOp = [&]() { // In OG, we build the landing pad for this scope. In CIR, we emit a @@ -521,10 +524,11 @@ static cir::CIRCallOpInterface emitCallLikeOp( assert(callingConv == cir::CallingConv::C && "NYI"); if (indirectFuncTy) { callOpWithExceptions = builder.createIndirectTryCallOp( - callLoc, indirectFuncVal, indirectFuncTy, CIRCallArgs); + callLoc, indirectFuncVal, indirectFuncTy, CIRCallArgs, callingConv, + sideEffect); } else { - callOpWithExceptions = - builder.createTryCallOp(callLoc, directFuncOp, CIRCallArgs); + callOpWithExceptions = builder.createTryCallOp( + callLoc, directFuncOp, CIRCallArgs, callingConv, sideEffect); } callOpWithExceptions->setAttr("extra_attrs", extraFnAttrs); @@ -544,12 +548,12 @@ static cir::CIRCallOpInterface emitCallLikeOp( if (indirectFuncTy) { // TODO(cir): Set calling convention for indirect calls. assert(callingConv == cir::CallingConv::C && "NYI"); - return builder.createIndirectCallOp(callLoc, indirectFuncVal, - indirectFuncTy, CIRCallArgs, - cir::CallingConv::C, extraFnAttrs); + return builder.createIndirectCallOp( + callLoc, indirectFuncVal, indirectFuncTy, CIRCallArgs, + cir::CallingConv::C, sideEffect, extraFnAttrs); } return builder.createCallOp(callLoc, directFuncOp, CIRCallArgs, callingConv, - extraFnAttrs); + sideEffect, extraFnAttrs); } static RValue getRValueThroughMemory(mlir::Location loc, @@ -755,8 +759,9 @@ RValue CIRGenFunction::emitCall(const CIRGenFunctionInfo &CallInfo, FnName = calleeFnOp.getName(); cir::CallingConv callingConv; + cir::SideEffect sideEffect; CGM.constructAttributeList(FnName, CallInfo, Callee.getAbstractInfo(), Attrs, - callingConv, + callingConv, sideEffect, /*AttrOnCallSite=*/true, /*IsThunk=*/false); @@ -837,7 +842,7 @@ RValue CIRGenFunction::emitCall(const CIRGenFunctionInfo &CallInfo, cir::CIRCallOpInterface callLikeOp = emitCallLikeOp( *this, callLoc, indirectFuncTy, indirectFuncVal, directFuncOp, - CIRCallArgs, isInvoke, callingConv, extraFnAttrs); + CIRCallArgs, isInvoke, callingConv, sideEffect, extraFnAttrs); if (E) callLikeOp->setAttr("ast", diff --git a/clang/lib/CIR/CodeGen/CIRGenModule.cpp b/clang/lib/CIR/CodeGen/CIRGenModule.cpp index b35ef11c7782..a6458a8a82bd 100644 --- a/clang/lib/CIR/CodeGen/CIRGenModule.cpp +++ b/clang/lib/CIR/CodeGen/CIRGenModule.cpp @@ -2682,10 +2682,11 @@ void CIRGenModule::setCIRFunctionAttributes(GlobalDecl GD, cir::FuncOp func, bool isThunk) { // TODO(cir): More logic of constructAttributeList is needed. cir::CallingConv callingConv; + cir::SideEffect sideEffect; // Initialize PAL with existing attributes to merge attributes. mlir::NamedAttrList PAL{func.getExtraAttrs().getElements().getValue()}; - constructAttributeList(func.getName(), info, GD, PAL, callingConv, + constructAttributeList(func.getName(), info, GD, PAL, callingConv, sideEffect, /*AttrOnCallSite=*/false, isThunk); func.setExtraAttrsAttr(cir::ExtraFuncAttributesAttr::get( &getMLIRContext(), PAL.getDictionary(&getMLIRContext()))); diff --git a/clang/lib/CIR/CodeGen/CIRGenModule.h b/clang/lib/CIR/CodeGen/CIRGenModule.h index 905754a4ad3a..74ede04a53f4 100644 --- a/clang/lib/CIR/CodeGen/CIRGenModule.h +++ b/clang/lib/CIR/CodeGen/CIRGenModule.h @@ -317,7 +317,8 @@ class CIRGenModule : public CIRGenTypeCache { CIRGenCalleeInfo CalleeInfo, mlir::NamedAttrList &Attrs, cir::CallingConv &callingConv, - bool AttrOnCallSite, bool IsThunk); + cir::SideEffect &sideEffect, bool AttrOnCallSite, + bool IsThunk); /// Helper function for getDefaultFunctionAttributes. Builds a set of function /// attributes which can be simply added to a function. diff --git a/clang/lib/CIR/Dialect/IR/CIRDialect.cpp b/clang/lib/CIR/Dialect/IR/CIRDialect.cpp index 1c5a6467f538..b4f79cafdea1 100644 --- a/clang/lib/CIR/Dialect/IR/CIRDialect.cpp +++ b/clang/lib/CIR/Dialect/IR/CIRDialect.cpp @@ -172,6 +172,7 @@ template struct EnumTraits {}; REGISTER_ENUM_TYPE(GlobalLinkageKind); REGISTER_ENUM_TYPE(CallingConv); +REGISTER_ENUM_TYPE(SideEffect); REGISTER_ENUM_TYPE_WITH_NS(cir::sob, SignedOverflowBehavior); } // namespace @@ -2893,6 +2894,18 @@ static ::mlir::ParseResult parseCallCommon(::mlir::OpAsmParser &parser, builder.getContext(), callingConv)); } + if (parser.parseOptionalKeyword("side_effect").succeeded()) { + if (parser.parseLParen().failed()) + return failure(); + cir::SideEffect sideEffect; + if (parseCIRKeyword(parser, sideEffect).failed()) + return failure(); + if (parser.parseRParen().failed()) + return failure(); + result.addAttribute("side_effect", cir::SideEffectAttr::get( + builder.getContext(), sideEffect)); + } + Attribute extraAttrs; if (::mlir::succeeded(parser.parseOptionalKeyword("extra"))) { if (parser.parseLParen().failed()) @@ -2923,11 +2936,14 @@ static ::mlir::ParseResult parseCallCommon(::mlir::OpAsmParser &parser, return ::mlir::success(); } -void printCallCommon( - Operation *op, mlir::Value indirectCallee, mlir::FlatSymbolRefAttr flatSym, - ::mlir::OpAsmPrinter &state, cir::ExtraFuncAttributesAttr extraAttrs, - cir::CallingConv callingConv, ::mlir::UnitAttr exception = {}, - mlir::Block *cont = nullptr, mlir::Block *landingPad = nullptr) { +void printCallCommon(Operation *op, mlir::Value indirectCallee, + mlir::FlatSymbolRefAttr flatSym, + ::mlir::OpAsmPrinter &state, + cir::ExtraFuncAttributesAttr extraAttrs, + cir::CallingConv callingConv, cir::SideEffect sideEffect, + ::mlir::UnitAttr exception = {}, + mlir::Block *cont = nullptr, + mlir::Block *landingPad = nullptr) { state << ' '; auto callLikeOp = mlir::cast(op); @@ -2977,6 +2993,7 @@ void printCallCommon( elidedAttrs.push_back("ast"); elidedAttrs.push_back("extra_attrs"); elidedAttrs.push_back("calling_conv"); + elidedAttrs.push_back("side_effect"); elidedAttrs.push_back("exception"); elidedAttrs.push_back("operandSegmentSizes"); @@ -2991,6 +3008,12 @@ void printCallCommon( state << ")"; } + if (sideEffect != cir::SideEffect::All) { + state << " side_effect("; + state << stringifySideEffect(sideEffect); + state << ")"; + } + if (!extraAttrs.getElements().empty()) { state << " extra("; state.printAttributeWithoutType(extraAttrs); @@ -3023,9 +3046,10 @@ ::mlir::ParseResult cir::CallOp::parse(::mlir::OpAsmParser &parser, void cir::CallOp::print(::mlir::OpAsmPrinter &state) { mlir::Value indirectCallee = isIndirect() ? getIndirectCall() : nullptr; cir::CallingConv callingConv = getCallingConv(); + cir::SideEffect sideEffect = getSideEffect(); mlir::UnitAttr exception = getExceptionAttr(); printCallCommon(*this, indirectCallee, getCalleeAttr(), state, - getExtraAttrs(), callingConv, exception); + getExtraAttrs(), callingConv, sideEffect, exception); } //===----------------------------------------------------------------------===// @@ -3075,8 +3099,10 @@ ::mlir::ParseResult cir::TryCallOp::parse(::mlir::OpAsmParser &parser, void cir::TryCallOp::print(::mlir::OpAsmPrinter &state) { mlir::Value indirectCallee = isIndirect() ? getIndirectCall() : nullptr; cir::CallingConv callingConv = getCallingConv(); + cir::SideEffect sideEffect = getSideEffect(); printCallCommon(*this, indirectCallee, getCalleeAttr(), state, - getExtraAttrs(), callingConv, {}, getCont(), getLandingPad()); + getExtraAttrs(), callingConv, sideEffect, {}, getCont(), + getLandingPad()); } mlir::SuccessorOperands cir::TryCallOp::getSuccessorOperands(unsigned index) { diff --git a/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.cpp b/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.cpp index c933035cd850..43dbed3da715 100644 --- a/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.cpp +++ b/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.cpp @@ -711,6 +711,43 @@ mlir::LLVM::CConv convertCallingConv(cir::CallingConv callinvConv) { llvm_unreachable("Unknown calling convention"); } +void convertSideEffectForCall(mlir::Operation *callOp, + cir::SideEffect sideEffect, + mlir::LLVM::MemoryEffectsAttr &memoryEffect, + bool &noUnwind, bool &willReturn) { + using mlir::LLVM::ModRefInfo; + + switch (sideEffect) { + case cir::SideEffect::All: + memoryEffect = {}; + noUnwind = false; + willReturn = false; + break; + + case cir::SideEffect::Pure: + memoryEffect = mlir::LLVM::MemoryEffectsAttr::get( + callOp->getContext(), /*other=*/ModRefInfo::Ref, + /*argMem=*/ModRefInfo::Ref, + /*inaccessibleMem=*/ModRefInfo::Ref); + noUnwind = true; + willReturn = true; + break; + + case cir::SideEffect::Const: + memoryEffect = mlir::LLVM::MemoryEffectsAttr::get( + callOp->getContext(), /*other=*/ModRefInfo::NoModRef, + /*argMem=*/ModRefInfo::NoModRef, + /*inaccessibleMem=*/ModRefInfo::NoModRef); + noUnwind = true; + willReturn = true; + break; + + default: + callOp->emitError("unknown side effect"); + break; + } +} + mlir::LogicalResult CIRToLLVMCopyOpLowering::matchAndRewrite( cir::CopyOp op, OpAdaptor adaptor, mlir::ConversionPatternRewriter &rewriter) const { @@ -1255,6 +1292,12 @@ rewriteToCallOrInvoke(mlir::Operation *op, mlir::ValueRange callOperands, auto cconv = convertCallingConv(callIf.getCallingConv()); + mlir::LLVM::MemoryEffectsAttr memoryEffects; + bool noUnwind = false; + bool willReturn = false; + convertSideEffectForCall(op, callIf.getSideEffect(), memoryEffects, noUnwind, + willReturn); + mlir::LLVM::LLVMFunctionType llvmFnTy; if (calleeAttr) { // direct call auto fn = @@ -1283,6 +1326,10 @@ rewriteToCallOrInvoke(mlir::Operation *op, mlir::ValueRange callOperands, auto newOp = rewriter.replaceOpWithNewOp( op, llvmFnTy, calleeAttr, callOperands); newOp.setCConv(cconv); + if (memoryEffects) + newOp.setMemoryEffectsAttr(memoryEffects); + newOp.setNoUnwind(noUnwind); + newOp.setWillReturn(willReturn); } return mlir::success(); } diff --git a/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.h b/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.h index 48baae2ae799..441d18ac15fe 100644 --- a/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.h +++ b/clang/lib/CIR/Lowering/DirectToLLVM/LowerToLLVM.h @@ -26,6 +26,11 @@ mlir::LLVM::Linkage convertLinkage(cir::GlobalLinkageKind linkage); mlir::LLVM::CConv convertCallingConv(cir::CallingConv callinvConv); +void convertSideEffectForCall(mlir::Operation *callOp, + cir::SideEffect sideEffect, + mlir::LLVM::MemoryEffectsAttr &memoryEffect, + bool &noUnwind, bool &willReturn); + void buildCtorDtorList( mlir::ModuleOp module, mlir::StringRef globalXtorName, mlir::StringRef llvmXtorName, diff --git a/clang/test/CIR/CodeGen/call-side-effect.cpp b/clang/test/CIR/CodeGen/call-side-effect.cpp new file mode 100644 index 000000000000..564e2c115b79 --- /dev/null +++ b/clang/test/CIR/CodeGen/call-side-effect.cpp @@ -0,0 +1,25 @@ +// RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-cir %s -o %t.cir +// RUN: FileCheck --input-file=%t.cir --check-prefix=CIR %s +// RUN: %clang_cc1 -std=c++20 -triple x86_64-unknown-linux-gnu -fclangir -emit-llvm %s -o %t.ll +// RUN: FileCheck --input-file=%t.ll --check-prefix=LLVM %s + +[[gnu::pure]] int pure_func(int x); +[[gnu::const]] int const_func(int x); + +int test(int x) { + int y1 = pure_func(x); + int y2 = const_func(x); + return y1 + y2; +} + +// CIR-LABEL: @_Z4testi +// CIR: %{{.+}} = cir.call @_Z9pure_funci(%{{.+}}) : (!s32i) -> !s32i side_effect(pure) +// CIR: %{{.+}} = cir.call @_Z10const_funci(%{{.+}}) : (!s32i) -> !s32i side_effect(const) +// CIR: } + +// LLVM-LABEL: @_Z4testi(i32 %0) +// LLVM: %{{.+}} = call i32 @_Z9pure_funci(i32 %{{.+}}) #[[#meta_pure:]] +// LLVM: %{{.+}} = call i32 @_Z10const_funci(i32 %{{.+}}) #[[#meta_const:]] +// LLVM: } +// LLVM: attributes #[[#meta_pure]] = { nounwind willreturn memory(read) } +// LLVM: attributes #[[#meta_const]] = { nounwind willreturn memory(none) } diff --git a/clang/test/CIR/IR/side-effect.cir b/clang/test/CIR/IR/side-effect.cir new file mode 100644 index 000000000000..76659f5c6ef3 --- /dev/null +++ b/clang/test/CIR/IR/side-effect.cir @@ -0,0 +1,21 @@ +// RUN: cir-opt %s -o %t.cir +// RUN: FileCheck %s --input-file=%t.cir + +!s32i = !cir.int + +module { + cir.func private @add(%arg0: !s32i, %arg1: !s32i) -> !s32i + cir.func @call_with_side_effect() { + %0 = cir.const #cir.int<0> : !s32i + %1 = cir.const #cir.int<1> : !s32i + %2 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(all) + %3 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(pure) + %4 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(const) + cir.return + } + // CHECK-LABEL: @call_with_side_effect() + // CHECK: %{{.+}} = cir.call @add(%{{.+}}, %{{.+}}) : (!s32i, !s32i) -> !s32i + // CHECK-NEXT: %{{.+}} = cir.call @add(%{{.+}}, %{{.+}}) : (!s32i, !s32i) -> !s32i side_effect(pure) + // CHECK-NEXT: %{{.+}} = cir.call @add(%{{.+}}, %{{.+}}) : (!s32i, !s32i) -> !s32i side_effect(const) + // CHECK: } +} diff --git a/clang/test/CIR/Lowering/call.cir b/clang/test/CIR/Lowering/call.cir index ade54037b76b..ed4916d55e14 100644 --- a/clang/test/CIR/Lowering/call.cir +++ b/clang/test/CIR/Lowering/call.cir @@ -99,4 +99,23 @@ module { // LLVM-NEXT: %2 = call i32 (i32, ...) %1(i32 0, i32 0) // LLVM-NEXT: ret i32 %2 + cir.func private @add(%arg0: !s32i, %arg1: !s32i) -> !s32i + + cir.func @call_with_side_effect() { + %0 = cir.const #cir.int<0> : !s32i + %1 = cir.const #cir.int<1> : !s32i + %2 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(all) + %3 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(pure) + %4 = cir.call @add(%0, %1) : (!s32i, !s32i) -> !s32i side_effect(const) + cir.return + } + + // LLVM: @call_with_side_effect + // LLVM: %{{.+}} = call i32 @add(i32 0, i32 1) + // LLVM: %{{.+}} = call i32 @add(i32 0, i32 1) #[[#pure:]] + // LLVM: %{{.+}} = call i32 @add(i32 0, i32 1) #[[#const:]] + // LLVM: } + // LLVM: attributes #[[#pure]] = { nounwind willreturn memory(read) } + // LLVM: attributes #[[#const]] = { nounwind willreturn memory(none) } + } // end module