diff --git a/mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td b/mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td --- a/mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td +++ b/mlir/include/mlir/Dialect/LLVMIR/LLVMOpBase.td @@ -226,36 +226,46 @@ [{convertType(opInst.getOperand($0).getType().cast())}]; string result = [{convertType(opInst.getResult($0).getType().cast())}]; + string struct_result = + [{convertType(opInst.getResult(0).getType().cast() + .getBody()[$0])}]; } // Base class for LLVM intrinsics operation. It is similar to LLVM_Op, but -// provides the "llvmBuilder" field for constructing the intrinsic. The builder -// relies on the contents on "overloadedResults" and "overloadedOperands" lists -// that contain the positions of intrinsic results and operands that are -// overloadable in the LLVM sense, that is their types must be passed in during -// the construction of the intrinsic declaration to differentiate between -// differently-typed versions of the intrinsic. "opName" contains the name of -// the operation to be associated with the intrinsic and "enumName" contains the -// name of the intrinsic as appears in `llvm::Intrinsic` enum; one usually wants -// these to be related. +// provides the "llvmBuilder" field for constructing the intrinsic. +// The builder relies on the contents of "overloadedResults" and +// "overloadedOperands" lists that contain the positions of intrinsic results +// and operands that are overloadable in the LLVM sense, that is their types +// must be passed in during the construction of the intrinsic declaration to +// differentiate between differently-typed versions of the intrinsic. +// If the intrinsic has multiple results, this will eventually be packed into a +// single struct result. In this case, the types of any overloaded results need +// to be accessed via the LLVMStructType, instead of directly via the result. +// "opName" contains the name of the operation to be associated with the +// intrinsic and "enumName" contains the name of the intrinsic as appears in +// `llvm::Intrinsic` enum; one usually wants these to be related. class LLVM_IntrOpBase overloadedResults, list overloadedOperands, - list traits, bit hasResult> + list traits, int numResults> : LLVM_OpBase, - Results { + Results { + string resultPattern = !if(!gt(numResults, 1), + LLVM_IntrPatterns.struct_result, + LLVM_IntrPatterns.result); let llvmBuilder = [{ llvm::Module *module = builder.GetInsertBlock()->getModule(); llvm::Function *fn = llvm::Intrinsic::getDeclaration( module, llvm::Intrinsic::}] # enumName # [{, { }] # StrJoin.lst, + ListIntSubst.lst, ListIntSubst.lst)>.result # [{ }); auto operands = lookupValues(opInst.getOperands()); - }] # !if(hasResult, "$res = ", "") # [{builder.CreateCall(fn, operands); + }] # !if(!gt(numResults, 0), "$res = ", "") + # [{builder.CreateCall(fn, operands); }]; } @@ -263,9 +273,10 @@ // the intrinsic into the LLVM dialect and prefixes its name with "intr.". class LLVM_IntrOp overloadedResults, list overloadedOperands, list traits, - bit hasResult> + int numResults> : LLVM_IntrOpBase; + overloadedResults, overloadedOperands, traits, + numResults>; // Base class for LLVM intrinsic operations returning no results. Places the // intrinsic into the LLVM dialect and prefixes its name with "intr.". diff --git a/mlir/include/mlir/Dialect/LLVMIR/NVVMOps.td b/mlir/include/mlir/Dialect/LLVMIR/NVVMOps.td --- a/mlir/include/mlir/Dialect/LLVMIR/NVVMOps.td +++ b/mlir/include/mlir/Dialect/LLVMIR/NVVMOps.td @@ -40,9 +40,9 @@ class NVVM_IntrOp overloadedResults, list overloadedOperands, list traits, - bit hasResult> + int numResults> : LLVM_IntrOpBase; + overloadedResults, overloadedOperands, traits, numResults>; //===----------------------------------------------------------------------===// diff --git a/mlir/tools/mlir-tblgen/LLVMIRIntrinsicGen.cpp b/mlir/tools/mlir-tblgen/LLVMIRIntrinsicGen.cpp --- a/mlir/tools/mlir-tblgen/LLVMIRIntrinsicGen.cpp +++ b/mlir/tools/mlir-tblgen/LLVMIRIntrinsicGen.cpp @@ -210,7 +210,7 @@ printBracketedRange(intr.getOverloadableOperandsIdxs().set_bits(), os); os << ", "; printBracketedRange(traits, os); - os << ", " << (intr.getNumResults() == 0 ? 0 : 1) << ">, Arguments<(ins" + os << ", " << intr.getNumResults() << ">, Arguments<(ins" << (operands.empty() ? "" : " "); llvm::interleaveComma(operands, os); os << ")>;\n\n";