diff --git a/mlir/include/mlir/Dialect/CMakeLists.txt b/mlir/include/mlir/Dialect/CMakeLists.txt --- a/mlir/include/mlir/Dialect/CMakeLists.txt +++ b/mlir/include/mlir/Dialect/CMakeLists.txt @@ -5,6 +5,7 @@ add_subdirectory(AMX) add_subdirectory(Complex) add_subdirectory(DLTI) +add_subdirectory(EmitC) add_subdirectory(GPU) add_subdirectory(Math) add_subdirectory(Linalg) diff --git a/mlir/include/mlir/Dialect/EmitC/CMakeLists.txt b/mlir/include/mlir/Dialect/EmitC/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/EmitC/CMakeLists.txt @@ -0,0 +1 @@ +add_subdirectory(IR) diff --git a/mlir/include/mlir/Dialect/EmitC/IR/CMakeLists.txt b/mlir/include/mlir/Dialect/EmitC/IR/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/EmitC/IR/CMakeLists.txt @@ -0,0 +1,7 @@ +add_mlir_dialect(EmitC emitc) +add_mlir_doc(EmitC EmitC Dialects/ -gen-dialect-doc) + +set(LLVM_TARGET_DEFINITIONS EmitC.td) +mlir_tablegen(EmitCAttrDefs.h.inc -gen-attrdef-decls) +mlir_tablegen(EmitCAttrDefs.cpp.inc -gen-attrdef-defs) +add_public_tablegen_target(MLIREmitCAttrDefsIncGen) diff --git a/mlir/include/mlir/Dialect/EmitC/IR/EmitC.h b/mlir/include/mlir/Dialect/EmitC/IR/EmitC.h new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/EmitC/IR/EmitC.h @@ -0,0 +1,33 @@ +//===- EmitC.h - EmitC Dialect ----------------------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// This file declares EmitC in MLIR. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_DIALECT_EMITC_IR_EMITC_H +#define MLIR_DIALECT_EMITC_IR_EMITC_H + +#include "mlir/IR/BuiltinOps.h" +#include "mlir/IR/BuiltinTypes.h" +#include "mlir/IR/Dialect.h" +#include "mlir/IR/OpDefinition.h" +#include "mlir/Interfaces/SideEffectInterfaces.h" + +#include "mlir/Dialect/EmitC/IR/EmitCDialect.h.inc" + +#define GET_ATTRDEF_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitCAttrDefs.h.inc" + +#define GET_TYPEDEF_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitCTypes.h.inc" + +#define GET_OP_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitC.h.inc" + +#endif // MLIR_DIALECT_EMITC_IR_EMITC_H diff --git a/mlir/include/mlir/Dialect/EmitC/IR/EmitC.td b/mlir/include/mlir/Dialect/EmitC/IR/EmitC.td new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/EmitC/IR/EmitC.td @@ -0,0 +1,193 @@ +//===- EmitC.td - EmitC dialect ----------------------------*- tablegen -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Defines the MLIR EmitC dialect, its attribute, type and operations. +// +//===----------------------------------------------------------------------===// + +#ifndef EMITC_DIALECT_EMITC_IR_EMITC +#define EMITC_DIALECT_EMITC_IR_EMITC + +include "mlir/IR/OpBase.td" +include "mlir/Interfaces/SideEffectInterfaces.td" + +//===----------------------------------------------------------------------===// +// EmitC dialect definition +//===----------------------------------------------------------------------===// + +def EmitC_Dialect : Dialect { + let name = "emitc"; + let cppNamespace = "::mlir::emitc"; + let hasConstantMaterializer = 1; +} + +//===----------------------------------------------------------------------===// +// EmitC attribute definitions +//===----------------------------------------------------------------------===// + +class EmitC_Attr : AttrDef { } + +def OpaqueAttr : EmitC_Attr<"Opaque"> { + let mnemonic = "opaque"; + + let summary = "An opaque attribute"; + + let description = [{ + An opaque attribute of which the value gets emitted as is. + + Examples: + + ```mlir + #emitc.opaque<""> + #emitc.opaque<"NULL"> + #emitc.opaque<"nullptr"> + ``` + }]; + + let parameters = (ins StringRefParameter<"the opaque value">:$value); + + let printer = [{ + $_printer << "opaque<\"" << getValue() << "\">"; + }]; + + let parser = [{ + if ($_parser.parseLess()) + return Attribute(); + StringRef value; + auto loc = $_parser.getCurrentLocation(); + if ($_parser.parseOptionalString(&value)) { + $_parser.emitError(loc) << "expected string"; + return Attribute(); + } + if ($_parser.parseGreater()) + return Attribute(); + return get($_ctxt, value); + }]; +} + +//===----------------------------------------------------------------------===// +// EmitC type definitions +//===----------------------------------------------------------------------===// + +class EmitC_Type : TypeDef { } + +def OpaqueType : EmitC_Type<"Opaque"> { + let mnemonic = "opaque"; + + let summary = "An opaque type"; + + let description = [{ + An opaque data type of which the value gets emitted as is. + + Examples: + + ```mlir + !emitc.opaque<"int"> + !emitc.opaque<"float *"> + !emitc.opaque<"std::vector"> + ``` + }]; + + let parameters = (ins StringRefParameter<"the opaque value">:$value); + + let printer = [{ + $_printer << "opaque<\"" << getValue() << "\">"; + }]; + + let parser = [{ + if ($_parser.parseLess()) + return Type(); + StringRef value; + auto loc = $_parser.getCurrentLocation(); + if ($_parser.parseOptionalString(&value) || value.empty()) { + $_parser.emitError(loc) << "expected non empty string"; + return Type(); + } + if ($_parser.parseGreater()) + return Type(); + return get($_ctxt, value); + }]; +} + +//===----------------------------------------------------------------------===// +// EmitC op definitions +//===----------------------------------------------------------------------===// + +// Base class for EmitC dialect ops. +class EmitC_Op traits = []> : + Op; + +def EmitC_ApplyOp : EmitC_Op<"apply", []> { + let summary = "apply operation"; + let description = [{ + With the "apply" operation the operators & (address of) and * (contents of) + can be applied to a single operand. + }]; + let arguments = (ins + Arg:$applicableOperator, + AnyType:$operand); + let results = (outs AnyType:$result); + let assemblyFormat = [{ + $applicableOperator `(` $operand `)` attr-dict `:` functional-type($operand, results) + }]; + let verifier = [{ return ::verify(*this); }]; +} + +def EmitC_CallOp : EmitC_Op<"call", []> { + let summary = "call operation"; + let description = [{ + The "call" operation represents a C++ function call. The call allows + specifying order of operands and attributes in the call as follows: + + - integer value of index type refers to an operand; + - attribute which will get lowered to constant value in call; + }]; + let arguments = (ins + Arg:$callee, + Arg, "the order of operands and attributes">:$args, + Arg, "template arguments">:$template_args, + Variadic:$operands); + let results = (outs Variadic); + let assemblyFormat = [{ + $callee `(` $operands `)` attr-dict `:` functional-type($operands, results) + }]; + let verifier = [{ return ::verify(*this); }]; +} + +def EmitC_ConstOp : EmitC_Op<"const", [ConstantLike, NoSideEffect]> { + let summary = "Constant op"; + let description = [{ + The `const` operation produces an SSA value equal to some constant + specified by an attribute, similar to `std.constant`. + In addition to the `std.constant` operation, the `emitc.const` operation + is intended to support opaque attributes and EmitC's opaque type. + }]; + + let arguments = (ins AnyAttr:$value); + let results = (outs AnyType); + + let verifier = [{ return ::verify(*this); }]; + + let hasFolder = 1; +} + +def EmitC_IncludeOp : EmitC_Op<"include", [NoSideEffect, HasParent<"ModuleOp">]> { + let summary = "include operation"; + let description = [{ + The `include` operation allows to define a source file inclusion via the + `#include` directive. + }]; + let arguments = (ins + Arg:$include, + UnitAttr:$is_standard_include); + let assemblyFormat = [{ + $include attr-dict (`is_standard_include` $is_standard_include^)? + }]; +} + +#endif // EMITC_DIALECT_EMITC_IR_EMITC diff --git a/mlir/include/mlir/InitAllDialects.h b/mlir/include/mlir/InitAllDialects.h --- a/mlir/include/mlir/InitAllDialects.h +++ b/mlir/include/mlir/InitAllDialects.h @@ -21,6 +21,7 @@ #include "mlir/Dialect/Async/IR/Async.h" #include "mlir/Dialect/Complex/IR/Complex.h" #include "mlir/Dialect/DLTI/DLTI.h" +#include "mlir/Dialect/EmitC/IR/EmitC.h" #include "mlir/Dialect/GPU/GPUDialect.h" #include "mlir/Dialect/LLVMIR/LLVMDialect.h" #include "mlir/Dialect/LLVMIR/NVVMDialect.h" @@ -57,6 +58,7 @@ async::AsyncDialect, complex::ComplexDialect, DLTIDialect, + emitc::EmitCDialect, gpu::GPUDialect, LLVM::LLVMDialect, linalg::LinalgDialect, diff --git a/mlir/include/mlir/InitAllTranslations.h b/mlir/include/mlir/InitAllTranslations.h --- a/mlir/include/mlir/InitAllTranslations.h +++ b/mlir/include/mlir/InitAllTranslations.h @@ -18,6 +18,8 @@ void registerFromLLVMIRTranslation(); void registerFromSPIRVTranslation(); +void registerToCppTranslation(); +void registerToCTranslation(); void registerToLLVMIRTranslation(); void registerToSPIRVTranslation(); @@ -28,6 +30,8 @@ static bool initOnce = []() { registerFromLLVMIRTranslation(); registerFromSPIRVTranslation(); + registerToCppTranslation(); + registerToCTranslation(); registerToLLVMIRTranslation(); registerToSPIRVTranslation(); return true; diff --git a/mlir/include/mlir/Target/Cpp/Cpp.h b/mlir/include/mlir/Target/Cpp/Cpp.h new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Target/Cpp/Cpp.h @@ -0,0 +1,196 @@ +//===- Cpp.h - Helpers to create C++ emitter --------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// This file defines helpers to emit C++ code using the EmitC dialect. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_TARGET_CPP_CPP_H +#define MLIR_TARGET_CPP_CPP_H + +#include "mlir/IR/BuiltinTypes.h" +#include "mlir/IR/Value.h" +#include "llvm/ADT/ScopedHashTable.h" +#include "llvm/Support/raw_ostream.h" +#include + +namespace mlir { +namespace emitc { + +struct TargetOptions { + bool forwardDeclareVariables; +}; + +/// Convenience functions to produce interleaved output with functions returning +/// a LogicalResult. This is different than those in STL as functions used on +/// each element doesn't return a string. +template +inline LogicalResult +interleaveWithError(ForwardIterator begin, ForwardIterator end, + UnaryFunctor each_fn, NullaryFunctor between_fn) { + if (begin == end) + return success(); + if (failed(each_fn(*begin))) + return failure(); + ++begin; + for (; begin != end; ++begin) { + between_fn(); + if (failed(each_fn(*begin))) + return failure(); + } + return success(); +} + +template +inline LogicalResult interleaveWithError(const Container &c, + UnaryFunctor each_fn, + NullaryFunctor between_fn) { + return interleaveWithError(c.begin(), c.end(), each_fn, between_fn); +} + +template +inline LogicalResult interleaveCommaWithError(const Container &c, + raw_ostream &os, + UnaryFunctor each_fn) { + return interleaveWithError(c.begin(), c.end(), each_fn, + [&]() { os << ", "; }); +} + +/// Emitter that uses dialect specific emitters to emit C++ code. +struct CppEmitter { + explicit CppEmitter(raw_ostream &os, bool restrictToC, + bool forwardDeclareVariables); + + /// Emits attribute or returns failure. + LogicalResult emitAttribute(Operation &op, Attribute attr); + + /// Emits operation 'op' with/without training semicolon or returns failure. + LogicalResult emitOperation(Operation &op, bool trailingSemicolon); + + /// Emits type 'type' or returns failure. + LogicalResult emitType(Operation &op, Type type); + + /// Emits array of types as a std::tuple of the emitted types. + /// - emits void for an empty array; + /// - emits the type of the only element for arrays of size one; + /// - emits a std::tuple otherwise; + LogicalResult emitTypes(Operation &op, ArrayRef types); + + /// Emits array of types as a std::tuple of the emitted types independently of + /// the array size. + LogicalResult emitTupleType(Operation &op, ArrayRef types); + + /// Emits an assignment for a variable which has been declared previously. + LogicalResult emitVariableAssignment(OpResult result); + + /// Emits a variable declaration for a result of an operation. + LogicalResult emitVariableDeclaration(OpResult result, + bool trailingSemicolon); + + /// Emits the variable declaration and assignment prefix for 'op'. + /// - emits separate variable followed by std::tie for multi-valued operation; + /// - emits single type followed by variable for single result; + /// - emits nothing if no value produced by op; + /// Emits final '=' operator where a type is produced. Returns failure if + /// any result type could not be converted. + LogicalResult emitAssignPrefix(Operation &op); + + /// Emits a label for the block. + LogicalResult emitLabel(Block &block); + + /// Emits the operands and atttributes of the operation. All operands are + /// emitted first and then all attributes in alphabetical order. + LogicalResult emitOperandsAndAttributes(Operation &op, + ArrayRef exclude = {}); + + /// Emits the operands of the operation. All operands are emitted in order. + LogicalResult emitOperands(Operation &op); + + /// Return the existing or a new name for a Value. + StringRef getOrCreateName(Value val); + + /// Return the existing or a new label of a Block. + StringRef getOrCreateName(Block &block); + + /// Whether to map an mlir integer to a signed integer in C++. + bool mapToSigned(IntegerType::SignednessSemantics val); + + /// RAII helper function to manage entering/exiting C++ scopes. + struct Scope { + Scope(CppEmitter &emitter) + : valMapperScope(emitter.valMapper), + blockMapperScope(emitter.blockMapper), emitter(emitter) { + emitter.valueInScopeCount.push(emitter.valueInScopeCount.top()); + emitter.labelInScopeCount.push(emitter.labelInScopeCount.top()); + } + ~Scope() { + emitter.valueInScopeCount.pop(); + emitter.labelInScopeCount.pop(); + } + + private: + llvm::ScopedHashTableScope valMapperScope; + llvm::ScopedHashTableScope blockMapperScope; + CppEmitter &emitter; + }; + + /// Returns wether the Value is assigned to a C++ variable in the scope. + bool hasValueInScope(Value val); + + // Returns whether a label is assigned to the block. + bool hasBlockLabel(Block &block); + + /// Returns the output stream. + raw_ostream &ostream() { return os; }; + + /// Returns if to emitc C. + bool restrictedToC() { return restrictToC; }; + + /// Returns if all variables need to be forward declared. + bool forwardDeclaredVariables() { return forwardDeclareVariables; }; + +private: + using ValMapper = llvm::ScopedHashTable; + using BlockMapper = llvm::ScopedHashTable; + + /// Output stream to emit to. + raw_ostream &os; + + /// Boolean that restricts the emitter to C. + bool restrictToC; + + /// Boolean to enforce a forward declaration of all variables. + bool forwardDeclareVariables; + + /// Map from value to name of C++ variable that contain the name. + ValMapper valMapper; + + /// Map from block to name of C++ label. + BlockMapper blockMapper; + + /// The number of values in the current scope. This is used to declare the + /// names of values in a scope. + std::stack valueInScopeCount; + std::stack labelInScopeCount; +}; + +/// Translates the given operation to C++ code. The operation or operations in +/// the region of 'op' need almost all be in EmitC dialect. +LogicalResult TranslateToCpp(Operation &op, raw_ostream &os, + bool forwardDeclareVariables = false, + bool trailingSemicolon = false); + +/// Similar to `TranslateToCpp`, but translates the given operation to C code. +LogicalResult TranslateToC(Operation &op, raw_ostream &os, + bool forwardDeclareVariables = false, + bool trailingSemicolon = false); +} // namespace emitc +} // namespace mlir + +#endif // MLIR_TARGET_CPP_CPP_H diff --git a/mlir/lib/Dialect/CMakeLists.txt b/mlir/lib/Dialect/CMakeLists.txt --- a/mlir/lib/Dialect/CMakeLists.txt +++ b/mlir/lib/Dialect/CMakeLists.txt @@ -5,6 +5,7 @@ add_subdirectory(AMX) add_subdirectory(Complex) add_subdirectory(DLTI) +add_subdirectory(EmitC) add_subdirectory(GPU) add_subdirectory(Linalg) add_subdirectory(LLVMIR) diff --git a/mlir/lib/Dialect/EmitC/CMakeLists.txt b/mlir/lib/Dialect/EmitC/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/EmitC/CMakeLists.txt @@ -0,0 +1 @@ +add_subdirectory(IR) diff --git a/mlir/lib/Dialect/EmitC/IR/CMakeLists.txt b/mlir/lib/Dialect/EmitC/IR/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/EmitC/IR/CMakeLists.txt @@ -0,0 +1,14 @@ +add_mlir_dialect_library(MLIREmitC + EmitC.cpp + + ADDITIONAL_HEADER_DIRS + ${MLIR_MAIN_INCLUDE_DIR}/mlir/Dialect/EmitC + + DEPENDS + MLIREmitCIncGen + MLIREmitCAttrDefsIncGen + + LINK_LIBS PUBLIC + MLIRIR + MLIRSideEffectInterfaces + ) diff --git a/mlir/lib/Dialect/EmitC/IR/EmitC.cpp b/mlir/lib/Dialect/EmitC/IR/EmitC.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/EmitC/IR/EmitC.cpp @@ -0,0 +1,201 @@ +//===- EmitC.cpp - EmitC Dialect --------------------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Dialect/EmitC/IR/EmitC.h" +#include "mlir/Dialect/StandardOps/IR/Ops.h" +#include "mlir/IR/Builders.h" +#include "mlir/IR/DialectImplementation.h" +#include "mlir/IR/OpImplementation.h" +#include "mlir/IR/TypeUtilities.h" +#include "mlir/IR/Types.h" +#include "llvm/ADT/TypeSwitch.h" + +using namespace mlir; +using namespace mlir::emitc; + +//===----------------------------------------------------------------------===// +// EmitCDialect +//===----------------------------------------------------------------------===// + +void emitc::EmitCDialect::initialize() { + addOperations< +#define GET_OP_LIST +#include "mlir/Dialect/EmitC/IR/EmitC.cpp.inc" + >(); + addTypes< +#define GET_TYPEDEF_LIST +#include "mlir/Dialect/EmitC/IR/EmitCTypes.cpp.inc" + >(); + addAttributes< +#define GET_ATTRDEF_LIST +#include "mlir/Dialect/EmitC/IR/EmitCAttrDefs.cpp.inc" + >(); +} + +/// Materialize a single constant operation from a given attribute value with +/// the desired resultant type. +Operation *emitc::EmitCDialect::materializeConstant(OpBuilder &builder, + Attribute value, Type type, + Location loc) { + return builder.create(loc, type, value); +} + +//===----------------------------------------------------------------------===// +// ApplyOp +//===----------------------------------------------------------------------===// + +static LogicalResult verify(mlir::emitc::ApplyOp op) { + StringRef applicableOperator = op.applicableOperator(); + + // Applicable operator must not be empty. + if (applicableOperator.empty()) { + return op.emitOpError("applicable operator must not be empty"); + } + + // Only `*` and `&` are supported. + if (!applicableOperator.equals("&") && !applicableOperator.equals("*")) + return op.emitOpError("applicable operator is illegal"); + + return success(); +} + +//===----------------------------------------------------------------------===// +// CallOp +//===----------------------------------------------------------------------===// + +static LogicalResult verify(mlir::emitc::CallOp op) { + // Callee must not be empty. + if (op.callee().empty()) { + return op.emitOpError("callee must not be empty"); + } + + auto argsAttr = op.args(); + if (argsAttr.hasValue()) { + for (auto &arg : argsAttr.getValue()) { + if (auto iArg = arg.dyn_cast()) { + if (iArg.getType().isIndex()) { + int64_t index = iArg.getInt(); + // Args with elements of type index must be in range + // [0..operands.size). + if ((index < 0) || + (index >= static_cast(op.getNumOperands()))) { + return op.emitOpError("index argument is out of range"); + } + } + } + // Args with elements of type ArrayAttr must have a type. + else if (auto aArg = arg.dyn_cast()) { + if (aArg.getType().isa()) { + return op.emitOpError("array argument has no type"); + } + } + } + } + + auto templateArgsAttr = op.template_args(); + if (templateArgsAttr.hasValue()) { + for (auto &tArg : templateArgsAttr.getValue()) { + // C++ forbids float literals as template arguments. + if (auto iArg = tArg.dyn_cast()) { + return op.emitOpError("float literal as template argument is invalid"); + } + // Template args with elements of type ArrayAttr are not allowed. + else if (auto aArg = tArg.dyn_cast()) { + return op.emitOpError("array as template arguments is invalid"); + } + // Template args with elements of type DenseElementsAttr are not + // allowed. + else if (auto dArg = tArg.dyn_cast()) { + return op.emitOpError("dense elements as template " + "argument are invalid"); + } + } + } + return success(); +} + +//===----------------------------------------------------------------------===// +// ConstOp +//===----------------------------------------------------------------------===// + +/// The constant op requires that the attribute's type matches the return type. +static LogicalResult verify(ConstOp &op) { + auto value = op.value(); + Type type = op.getType(); + if (!value.getType().isa() && type != value.getType()) + return op.emitOpError() << "requires attribute's type (" << value.getType() + << ") to match op's return type (" << type << ")"; + return success(); +} + +// Folder. +OpFoldResult ConstOp::fold(ArrayRef operands) { + assert(operands.empty() && "constant has no operands"); + return value(); +} + +//===----------------------------------------------------------------------===// +// TableGen'd op method definitions +//===----------------------------------------------------------------------===// + +#define GET_OP_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitC.cpp.inc" + +//===----------------------------------------------------------------------===// +// EmitC Attributes +//===----------------------------------------------------------------------===// + +#define GET_ATTRDEF_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitCAttrDefs.cpp.inc" + +Attribute emitc::EmitCDialect::parseAttribute(DialectAsmParser &parser, + Type type) const { + llvm::SMLoc typeLoc = parser.getCurrentLocation(); + StringRef mnemonic; + if (parser.parseKeyword(&mnemonic)) + return Attribute(); + Attribute genAttr; + auto parseResult = + generatedAttributeParser(getContext(), parser, mnemonic, type, genAttr); + if (parseResult.hasValue()) + return genAttr; + parser.emitError(typeLoc, "unknown attribute in EmitC dialect"); + return Attribute(); +} + +void emitc::EmitCDialect::printAttribute(Attribute attr, + DialectAsmPrinter &os) const { + if (failed(generatedAttributePrinter(attr, os))) + llvm_unreachable("unexpected 'EmitC' attribute kind"); +} + +//===----------------------------------------------------------------------===// +// EmitC Types +//===----------------------------------------------------------------------===// + +#define GET_TYPEDEF_CLASSES +#include "mlir/Dialect/EmitC/IR/EmitCTypes.cpp.inc" + +Type emitc::EmitCDialect::parseType(DialectAsmParser &parser) const { + llvm::SMLoc typeLoc = parser.getCurrentLocation(); + StringRef mnemonic; + if (parser.parseKeyword(&mnemonic)) + return Type(); + Type genType; + auto parseResult = + generatedTypeParser(getContext(), parser, mnemonic, genType); + if (parseResult.hasValue()) + return genType; + parser.emitError(typeLoc, "unknown type in EmitC dialect"); + return Type(); +} + +void emitc::EmitCDialect::printType(Type type, DialectAsmPrinter &os) const { + if (failed(generatedTypePrinter(type, os))) + llvm_unreachable("unexpected 'EmitC' type kind"); +} diff --git a/mlir/lib/Target/CMakeLists.txt b/mlir/lib/Target/CMakeLists.txt --- a/mlir/lib/Target/CMakeLists.txt +++ b/mlir/lib/Target/CMakeLists.txt @@ -1,2 +1,3 @@ +add_subdirectory(Cpp) add_subdirectory(SPIRV) add_subdirectory(LLVMIR) diff --git a/mlir/lib/Target/Cpp/CMakeLists.txt b/mlir/lib/Target/Cpp/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/lib/Target/Cpp/CMakeLists.txt @@ -0,0 +1,13 @@ +add_mlir_translation_library(MLIRTargetCpp + TranslateRegistration.cpp + TranslateToCpp.cpp + + ADDITIONAL_HEADER_DIRS + ${MLIR_MAIN_INCLUDE_DIR}/mlir/Target/Cpp + + LINK_LIBS PUBLIC + MLIREmitC + MLIRIR + MLIRSCF + MLIRStandard + ) diff --git a/mlir/lib/Target/Cpp/TranslateRegistration.cpp b/mlir/lib/Target/Cpp/TranslateRegistration.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Target/Cpp/TranslateRegistration.cpp @@ -0,0 +1,93 @@ +//===- TranslateRegistration.cpp - Register translation ----------- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Dialect/EmitC/IR/EmitC.h" +#include "mlir/Dialect/SCF/SCF.h" +#include "mlir/Dialect/StandardOps/IR/Ops.h" +#include "mlir/IR/BuiltinOps.h" +#include "mlir/IR/Dialect.h" +#include "mlir/Target/Cpp/Cpp.h" +#include "mlir/Translation.h" + +using namespace mlir; + +namespace mlir { + +//===----------------------------------------------------------------------===// +// Cpp registration +//===----------------------------------------------------------------------===// + +void registerToCppTranslation() { + TranslateFromMLIRRegistration reg( + "mlir-to-cpp", + [](ModuleOp module, llvm::raw_ostream &output) { + return emitc::TranslateToCpp(*module.getOperation(), output, + /*forwardDeclareVariables=*/false, + /*trailingSemiColon=*/false); + }, + [](DialectRegistry ®istry) { + // clang-format off + registry.insert(); + // clang-format on + }); + + TranslateFromMLIRRegistration regFordwardDeclared( + "mlir-to-cpp-forward-declared", + [](ModuleOp module, llvm::raw_ostream &output) { + return emitc::TranslateToCpp(*module.getOperation(), output, + /*forwardDeclareVariables=*/true, + /*trailingSemiColon=*/false); + }, + [](DialectRegistry ®istry) { + // clang-format off + registry.insert(); + // clang-format on + }); +} + +//===----------------------------------------------------------------------===// +// C registration +//===----------------------------------------------------------------------===// + +void registerToCTranslation() { + TranslateFromMLIRRegistration reg( + "mlir-to-c", + [](ModuleOp module, llvm::raw_ostream &output) { + return emitc::TranslateToC(*module.getOperation(), output, + /*forwardDeclareVariables=*/false, + /*trailingSemiColon=*/false); + }, + [](DialectRegistry ®istry) { + // clang-format off + registry.insert(); + // clang-format on + }); + + TranslateFromMLIRRegistration regFordwardDeclared( + "mlir-to-c-forward-declared", + [](ModuleOp module, llvm::raw_ostream &output) { + return emitc::TranslateToC(*module.getOperation(), output, + /*forwardDeclareVariables=*/true, + /*trailingSemiColon=*/false); + }, + [](DialectRegistry ®istry) { + // clang-format off + registry.insert(); + // clang-format on + }); +} + +} // namespace mlir diff --git a/mlir/lib/Target/Cpp/TranslateToCpp.cpp b/mlir/lib/Target/Cpp/TranslateToCpp.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Target/Cpp/TranslateToCpp.cpp @@ -0,0 +1,926 @@ +//===- TranslateToCpp.cpp - Translating to C++ calls ------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Dialect/EmitC/IR/EmitC.h" +#include "mlir/Dialect/SCF/SCF.h" +#include "mlir/Dialect/StandardOps/IR/Ops.h" +#include "mlir/IR/BuiltinOps.h" +#include "mlir/IR/BuiltinTypes.h" +#include "mlir/IR/Dialect.h" +#include "mlir/IR/Operation.h" +#include "mlir/Target/Cpp/Cpp.h" +#include "llvm/ADT/DenseMap.h" +#include "llvm/ADT/StringExtras.h" +#include "llvm/ADT/StringMap.h" +#include "llvm/Support/Debug.h" +#include "llvm/Support/FormatVariadic.h" + +#define DEBUG_TYPE "translate-to-cpp" + +using namespace mlir; +using namespace mlir::emitc; +using llvm::formatv; + +template +static LogicalResult printConstantOp(CppEmitter &emitter, + ConstOpTy constantOp) { + auto &os = emitter.ostream(); + auto result = constantOp.getOperation()->getResult(0); + auto value = constantOp.value(); + + bool isScalar = !result.getType().template isa(); + + // Add braces only if + // - cpp code is emitted, + // - variables are not forward declared + // - and the emitted type is a scalar (to prevent double brace + // initialization). + bool braceInitialization = + !emitter.restrictedToC() && !emitter.forwardDeclaredVariables(); + bool emitBraces = braceInitialization && isScalar; + + // Emit an assignment if variables are forward declared. + if (emitter.forwardDeclaredVariables()) { + // Special case for emitc.const. + if (auto oAttr = value.template dyn_cast()) { + if (oAttr.getValue().empty()) + return success(); + } + + if (failed(emitter.emitVariableAssignment(result))) + return failure(); + if (failed(emitter.emitAttribute(*constantOp.getOperation(), value))) + return failure(); + return success(); + } + + // Special case for emitc.const. + if (auto oAttr = value.template dyn_cast()) { + if (oAttr.getValue().empty()) { + // The semicolon gets printed by the emitOperation function. + if (failed(emitter.emitVariableDeclaration(result, + /*trailingSemicolon=*/false))) + return failure(); + return success(); + } + } + + // We have to emit a variable declaration. + if (!braceInitialization) { + // If brace initialization is not used, we have to emit an assignment. + if (failed(emitter.emitAssignPrefix(*constantOp.getOperation()))) { + return failure(); + } + if (failed(emitter.emitAttribute(*constantOp.getOperation(), value))) + return failure(); + return success(); + } + + if (failed(emitter.emitVariableDeclaration(result, + /*trailingSemicolon=*/false))) + return failure(); + + if (emitBraces) + os << "{"; + if (failed(emitter.emitAttribute(*constantOp.getOperation(), value))) + return failure(); + if (emitBraces) + os << "}"; + + return success(); +} + +static LogicalResult printBranchOp(CppEmitter &emitter, + mlir::BranchOp branchOp) { + auto &os = emitter.ostream(); + Block &successor = *branchOp.getSuccessor(); + + for (auto pair : + llvm::zip(branchOp.getOperands(), successor.getArguments())) { + auto &operand = std::get<0>(pair); + auto &argument = std::get<1>(pair); + os << emitter.getOrCreateName(argument) << " = " + << emitter.getOrCreateName(operand) << ";\n"; + } + + os << "goto "; + if (!(emitter.hasBlockLabel(successor))) { + return branchOp.emitOpError() << "Unable to find label for successor block"; + } + os << emitter.getOrCreateName(successor); + return success(); +} + +static LogicalResult printCondBranchOp(CppEmitter &emitter, + mlir::CondBranchOp condBranchOp) { + auto &os = emitter.ostream(); + Block &trueSuccessor = *condBranchOp.getTrueDest(); + Block &falseSuccessor = *condBranchOp.getFalseDest(); + + os << "if (" << emitter.getOrCreateName(condBranchOp.getCondition()) + << ") {\n"; + + // If condition is true. + for (auto pair : llvm::zip(condBranchOp.getTrueOperands(), + trueSuccessor.getArguments())) { + auto &operand = std::get<0>(pair); + auto &argument = std::get<1>(pair); + os << emitter.getOrCreateName(argument) << " = " + << emitter.getOrCreateName(operand) << ";\n"; + } + + os << "goto "; + if (!(emitter.hasBlockLabel(trueSuccessor))) { + return condBranchOp.emitOpError() + << "Unable to find label for successor block"; + } + os << emitter.getOrCreateName(trueSuccessor) << ";\n"; + os << "} else {\n"; + // If condition is false. + for (auto pair : llvm::zip(condBranchOp.getFalseOperands(), + falseSuccessor.getArguments())) { + auto &operand = std::get<0>(pair); + auto &argument = std::get<1>(pair); + os << emitter.getOrCreateName(argument) << " = " + << emitter.getOrCreateName(operand) << ";\n"; + } + + os << "goto "; + if (!(emitter.hasBlockLabel(falseSuccessor))) { + return condBranchOp.emitOpError() + << "Unable to find label for successor block"; + } + os << emitter.getOrCreateName(falseSuccessor) << ";\n"; + os << "}"; + return success(); +} + +static LogicalResult printCallOp(CppEmitter &emitter, mlir::CallOp callOp) { + if (failed(emitter.emitAssignPrefix(*callOp.getOperation()))) + return failure(); + + auto &os = emitter.ostream(); + os << callOp.getCallee() << "("; + if (failed(emitter.emitOperands(*callOp.getOperation()))) + return failure(); + os << ")"; + return success(); +} + +static LogicalResult printCallOp(CppEmitter &emitter, emitc::CallOp callOp) { + auto &os = emitter.ostream(); + auto &op = *callOp.getOperation(); + + if (failed(emitter.emitAssignPrefix(op))) + return failure(); + os << callOp.callee(); + + auto emitArgs = [&](Attribute attr) -> LogicalResult { + if (auto t = attr.dyn_cast()) { + // Index attributes are treated specially as operand index. + if (t.getType().isIndex()) { + auto idx = t.getInt(); + if ((idx < 0) || (idx >= op.getNumOperands())) + return op.emitOpError() << "invalid operand index"; + if (!emitter.hasValueInScope(op.getOperand(idx))) + return op.emitOpError() + << "operand " << idx << "'s value not defined in scope"; + os << emitter.getOrCreateName(op.getOperand(idx)); + return success(); + } + } + if (failed(emitter.emitAttribute(op, attr))) { + return failure(); + } + return success(); + }; + + if (callOp.template_args()) { + if (emitter.restrictedToC()) { + return op.emitOpError() + << "template arguments are not supported if emitting C"; + } + os << "<"; + if (failed(interleaveCommaWithError(*callOp.template_args(), os, emitArgs))) + return failure(); + os << ">"; + } + + os << "("; + + auto emittedArgs = + callOp.args() ? interleaveCommaWithError(*callOp.args(), os, emitArgs) + : emitter.emitOperands(op); + if (failed(emittedArgs)) + return failure(); + os << ")"; + return success(); +} + +static LogicalResult printApplyOp(CppEmitter &emitter, emitc::ApplyOp applyOp) { + auto &os = emitter.ostream(); + auto &op = *applyOp.getOperation(); + + if (failed(emitter.emitAssignPrefix(op))) + return failure(); + os << applyOp.applicableOperator(); + os << emitter.getOrCreateName(applyOp.getOperand()); + + return success(); +} + +static LogicalResult printForOp(CppEmitter &emitter, scf::ForOp forOp) { + + auto &os = emitter.ostream(); + + auto operands = forOp.getIterOperands(); + auto iterArgs = forOp.getRegionIterArgs(); + auto results = forOp.getResults(); + + if (!emitter.forwardDeclaredVariables()) { + for (auto result : forOp.getResults()) { + if (failed(emitter.emitVariableDeclaration(result, + /*trailingSemicolon=*/true))) + return failure(); + } + } + + for (auto pair : llvm::zip(iterArgs, operands)) { + if (failed(emitter.emitType(*forOp.getOperation(), + std::get<0>(pair).getType()))) + return failure(); + os << " " << emitter.getOrCreateName(std::get<0>(pair)) << " = "; + os << emitter.getOrCreateName(std::get<1>(pair)) << ";"; + os << "\n"; + } + + os << "for ("; + if (failed(emitter.emitType(*forOp.getOperation(), + forOp.getInductionVar().getType()))) + return failure(); + os << " "; + os << emitter.getOrCreateName(forOp.getInductionVar()); + + if (emitter.restrictedToC()) { + os << " = "; + os << emitter.getOrCreateName(forOp.lowerBound()); + } else { + os << "{"; + os << emitter.getOrCreateName(forOp.lowerBound()); + os << "}"; + } + + os << "; "; + os << emitter.getOrCreateName(forOp.getInductionVar()); + os << " < "; + os << emitter.getOrCreateName(forOp.upperBound()); + os << "; "; + os << emitter.getOrCreateName(forOp.getInductionVar()); + os << " += "; + os << emitter.getOrCreateName(forOp.step()); + os << ") {\n"; + + auto &forRegion = forOp.region(); + auto regionOps = forRegion.getOps(); + + // We skip the trailing yield op because this updates the result variables + // of the for op in the generated code. Instead we update the iterArgs at + // the end of a loop iteration and set the result variables after the for + // loop. + for (auto it = regionOps.begin(); std::next(it) != regionOps.end(); ++it) { + if (failed(emitter.emitOperation(*it, /*trailingSemicolon=*/true))) + return failure(); + } + + auto yieldOp = forRegion.getBlocks().front().getTerminator(); + // Copy yield operands into iterArgs at the end of a loop iteration. + for (auto pair : llvm::zip(iterArgs, yieldOp->getOperands())) { + auto iterArg = std::get<0>(pair); + auto operand = std::get<1>(pair); + os << emitter.getOrCreateName(iterArg) << " = " + << emitter.getOrCreateName(operand) << ";\n"; + } + + os << "}\n"; + + // Copy iterArgs into results after the for loop. + for (auto pair : llvm::zip(results, iterArgs)) { + auto result = std::get<0>(pair); + auto iterArg = std::get<1>(pair); + os << emitter.getOrCreateName(result) << " = " + << emitter.getOrCreateName(iterArg) << ";\n"; + } + + return success(); +} + +static LogicalResult printIfOp(CppEmitter &emitter, scf::IfOp ifOp) { + auto &os = emitter.ostream(); + + if (!emitter.forwardDeclaredVariables()) { + for (auto result : ifOp.getResults()) { + if (failed(emitter.emitVariableDeclaration(result, + /*trailingSemicolon=*/true))) + return failure(); + } + } + + os << "if ("; + if (failed(emitter.emitOperands(*ifOp.getOperation()))) + return failure(); + os << ") {\n"; + + auto &thenRegion = ifOp.thenRegion(); + for (auto &op : thenRegion.getOps()) { + // Note: This prints a superfluous semicolon if the terminating yield op has + // zero results. + if (failed(emitter.emitOperation(op, /*trailingSemicolon=*/true))) + return failure(); + } + + os << "}\n"; + + auto &elseRegion = ifOp.elseRegion(); + if (!elseRegion.empty()) { + os << "else {\n"; + + for (auto &op : elseRegion.getOps()) { + // Note: This prints a superfluous semicolon if the terminating yield op + // has zero results. + if (failed(emitter.emitOperation(op, /*trailingSemicolon=*/true))) + return failure(); + } + + os << "}\n"; + } + + return success(); +} + +static LogicalResult printYieldOp(CppEmitter &emitter, scf::YieldOp yieldOp) { + auto &os = emitter.ostream(); + auto &parentOp = *yieldOp.getOperation()->getParentOp(); + + if (yieldOp.getNumOperands() != parentOp.getNumResults()) { + return yieldOp.emitError("number of operands does not to match the number " + "of the parent op's results"); + } + + if (failed(interleaveWithError( + llvm::zip(parentOp.getResults(), yieldOp.getOperands()), + [&](auto pair) -> LogicalResult { + auto result = std::get<0>(pair); + auto operand = std::get<1>(pair); + os << emitter.getOrCreateName(result) << " = "; + + if (!emitter.hasValueInScope(operand)) + return yieldOp.emitError() << "operand value not in scope"; + os << emitter.getOrCreateName(operand); + return success(); + }, + [&]() { os << ";\n"; }))) + return failure(); + + return success(); +} + +static LogicalResult printReturnOp(CppEmitter &emitter, ReturnOp returnOp) { + auto &os = emitter.ostream(); + os << "return"; + switch (returnOp.getNumOperands()) { + case 0: + return success(); + case 1: + os << " " << emitter.getOrCreateName(returnOp.getOperand(0)); + return success(emitter.hasValueInScope(returnOp.getOperand(0))); + default: + os << " std::make_tuple("; + if (failed(emitter.emitOperandsAndAttributes(*returnOp.getOperation()))) + return failure(); + os << ")"; + return success(); + } +} + +static LogicalResult printModule(CppEmitter &emitter, ModuleOp moduleOp) { + CppEmitter::Scope scope(emitter); + auto &os = emitter.ostream(); + + if (emitter.restrictedToC()) { + os << "#include \n"; + os << "#include \n"; + os << "#include \n\n"; + } else { + os << "#include \n\n"; + } + + for (emitc::IncludeOp includeOp : moduleOp.getOps()) { + os << "#include "; + if (includeOp.is_standard_include()) { + os << "<" << includeOp.include() << ">\n"; + } else { + os << "\"" << includeOp.include() << "\"\n"; + } + } + os << "\n"; + + os << "// Forward declare functions.\n"; + for (FuncOp funcOp : moduleOp.getOps()) { + if (failed(emitter.emitTypes(*funcOp.getOperation(), + funcOp.getType().getResults()))) + return failure(); + os << " " << funcOp.getName() << "("; + if (failed(interleaveCommaWithError( + funcOp.getArguments(), os, [&](BlockArgument arg) { + return emitter.emitType(*funcOp.getOperation(), arg.getType()); + }))) + return failure(); + os << ");\n"; + } + os << "\n"; + + for (Operation &op : moduleOp) { + if (failed(emitter.emitOperation(op, /*trailingSemicolon=*/false))) + return failure(); + } + return success(); +} + +static LogicalResult printFunction(CppEmitter &emitter, FuncOp functionOp) { + // We need to forward-declare variables if the function has multiple blocks. + if (!emitter.forwardDeclaredVariables() && + functionOp.getBlocks().size() > 1) { + return functionOp.emitOpError() + << "with multiple blocks needs forward declared variables"; + } + + CppEmitter::Scope scope(emitter); + auto &os = emitter.ostream(); + if (failed(emitter.emitTypes(*functionOp.getOperation(), + functionOp.getType().getResults()))) + return failure(); + os << " " << functionOp.getName(); + + os << "("; + if (failed(interleaveCommaWithError( + functionOp.getArguments(), os, + [&](BlockArgument arg) -> LogicalResult { + if (failed(emitter.emitType(*functionOp.getOperation(), + arg.getType()))) + return failure(); + os << " " << emitter.getOrCreateName(arg); + return success(); + }))) + return failure(); + os << ") {\n"; + + if (emitter.forwardDeclaredVariables()) { + // We forward declare all result variables including results from ops + // inside of regions. + auto result = + functionOp.walk([&](Operation *op) -> WalkResult { + for (auto result : op->getResults()) { + if (failed(emitter.emitVariableDeclaration( + result, /*trailingSemicolon=*/true))) { + return WalkResult( + op->emitError("Unable to declare result variable for op")); + } + } + return WalkResult::advance(); + }); + if (result.wasInterrupted()) + return failure(); + } + + auto &blocks = functionOp.getBlocks(); + // Create label names for basic blocks. + for (auto &block : blocks) { + emitter.getOrCreateName(block); + } + + // Emit variables for basic block arguments. + for (auto it = std::next(blocks.begin()); it != blocks.end(); ++it) { + Block &block = *it; + for (auto &arg : block.getArguments()) { + if (emitter.hasValueInScope(arg)) + return functionOp.emitOpError(" block argument #") + << arg.getArgNumber() << " is out of scope"; + if (failed(emitter.emitType(*block.getParentOp(), arg.getType()))) { + return failure(); + } + os << " " << emitter.getOrCreateName(arg) << ";\n"; + } + } + + for (auto &block : blocks) { + // Only print a label if there is more than one block. + if (blocks.size() > 1) { + if (failed(emitter.emitLabel(block))) { + return failure(); + } + } + for (Operation &op : block.getOperations()) { + // Don't print additional semicolons after these operations. + bool trailingSemicolon = !isa(op); + + if (failed(emitter.emitOperation( + op, /*trailingSemicolon=*/trailingSemicolon))) + return failure(); + } + } + os << "}\n"; + return success(); +} + +CppEmitter::CppEmitter(raw_ostream &os, bool restrictToC, + bool forwardDeclareVariables) + : os(os), restrictToC(restrictToC), + forwardDeclareVariables(forwardDeclareVariables) { + valueInScopeCount.push(0); + labelInScopeCount.push(0); +} + +/// Return the existing or a new name for a Value. +StringRef CppEmitter::getOrCreateName(Value val) { + if (!valMapper.count(val)) + valMapper.insert(val, formatv("v{0}", ++valueInScopeCount.top())); + return *valMapper.begin(val); +} + +/// Return the existing or a new name for a Block. +StringRef CppEmitter::getOrCreateName(Block &block) { + if (!blockMapper.count(&block)) + blockMapper.insert(&block, formatv("label{0}", ++labelInScopeCount.top())); + return *blockMapper.begin(&block); +} + +bool CppEmitter::mapToSigned(IntegerType::SignednessSemantics val) { + switch (val) { + case IntegerType::Signless: + return true; + case IntegerType::Signed: + return true; + case IntegerType::Unsigned: + return false; + } +} + +bool CppEmitter::hasValueInScope(Value val) { return valMapper.count(val); } + +bool CppEmitter::hasBlockLabel(Block &block) { + return blockMapper.count(&block); +} + +LogicalResult CppEmitter::emitAttribute(Operation &op, Attribute attr) { + auto printInt = [&](APInt val, bool isSigned) { + if (val.getBitWidth() == 1) { + if (val.getBoolValue()) + os << "true"; + else + os << "false"; + } else { + val.print(os, isSigned); + } + }; + + auto printFloat = [&](APFloat val) { + if (val.isFinite()) { + SmallString<128> strValue; + // Use default values of toString except don't truncate zeros. + val.toString(strValue, 0, 0, false); + switch (llvm::APFloatBase::SemanticsToEnum(val.getSemantics())) { + case llvm::APFloatBase::S_IEEEsingle: + os << "(float)"; + break; + case llvm::APFloatBase::S_IEEEdouble: + os << "(double)"; + break; + default: + break; + }; + os << strValue; + } else if (val.isNaN()) { + os << "NAN"; + } else if (val.isInfinity()) { + if (val.isNegative()) + os << "-"; + os << "INFINITY"; + } + }; + + // Print floating point attributes. + if (auto fAttr = attr.dyn_cast()) { + printFloat(fAttr.getValue()); + return success(); + } + auto denseErrorMessage = "dense attributes are not supported if emitting C"; + if (auto dense = attr.dyn_cast()) { + // Dense attributes are not supported if emitting C. + if (restrictedToC()) + return op.emitError(denseErrorMessage); + os << '{'; + interleaveComma(dense, os, [&](APFloat val) { printFloat(val); }); + os << '}'; + return success(); + } + + // Print int attributes. + if (auto iAttr = attr.dyn_cast()) { + if (auto iType = iAttr.getType().dyn_cast()) { + printInt(iAttr.getValue(), mapToSigned(iType.getSignedness())); + return success(); + } + if (auto iType = iAttr.getType().dyn_cast()) { + printInt(iAttr.getValue(), false); + return success(); + } + } + if (auto dense = attr.dyn_cast()) { + // Dense attributes are not supported if emitting C. + if (restrictedToC()) + return op.emitError(denseErrorMessage); + if (auto iType = dense.getType() + .cast() + .getElementType() + .dyn_cast()) { + os << '{'; + interleaveComma(dense, os, [&](APInt val) { + printInt(val, mapToSigned(iType.getSignedness())); + }); + os << '}'; + return success(); + } + if (auto iType = dense.getType() + .cast() + .getElementType() + .dyn_cast()) { + os << '{'; + interleaveComma(dense, os, [&](APInt val) { printInt(val, false); }); + os << '}'; + return success(); + } + } + if (auto oAttr = attr.dyn_cast()) { + os << oAttr.getValue(); + return success(); + } + if (auto sAttr = attr.dyn_cast()) { + if (sAttr.getNestedReferences().size() > 1) { + return op.emitError(" attribute has more than 1 nested reference"); + } + os << sAttr.getRootReference(); + return success(); + } + if (auto type = attr.dyn_cast()) { + return emitType(op, type.getValue()); + } + return op.emitError("cannot emit attribute of type ") << attr.getType(); +} + +LogicalResult CppEmitter::emitOperands(Operation &op) { + auto emitOperandName = [&](Value result) -> LogicalResult { + if (!hasValueInScope(result)) + return op.emitOpError() << "operand value not in scope"; + os << getOrCreateName(result); + return success(); + }; + return interleaveCommaWithError(op.getOperands(), os, emitOperandName); +} + +LogicalResult +CppEmitter::emitOperandsAndAttributes(Operation &op, + ArrayRef exclude) { + if (failed(emitOperands(op))) + return failure(); + // Insert comma in between operands and non-filtered attributes if needed. + if (op.getNumOperands() > 0) { + for (auto attr : op.getAttrs()) { + if (!llvm::is_contained(exclude, attr.first.strref())) { + os << ", "; + break; + } + } + } + // Emit attributes. + auto emitNamedAttribute = [&](NamedAttribute attr) -> LogicalResult { + if (llvm::is_contained(exclude, attr.first.strref())) + return success(); + os << "/* " << attr.first << " */"; + if (failed(emitAttribute(op, attr.second))) + return failure(); + return success(); + }; + return interleaveCommaWithError(op.getAttrs(), os, emitNamedAttribute); +} + +LogicalResult CppEmitter::emitVariableAssignment(OpResult result) { + if (!hasValueInScope(result)) { + return result.getDefiningOp()->emitOpError( + "result variable for the operation has not been declared."); + } + os << getOrCreateName(result) << " = "; + return success(); +} + +LogicalResult CppEmitter::emitVariableDeclaration(OpResult result, + bool trailingSemicolon) { + if (hasValueInScope(result)) { + return result.getDefiningOp()->emitError( + "result variable for the operation already declared."); + } + if (failed(emitType(*result.getOwner(), result.getType()))) { + return failure(); + } + os << " " << getOrCreateName(result); + if (trailingSemicolon) { + os << ";\n"; + } + return success(); +} + +LogicalResult CppEmitter::emitAssignPrefix(Operation &op) { + switch (op.getNumResults()) { + case 0: + break; + case 1: { + auto result = op.getResult(0); + if (forwardDeclaredVariables()) { + if (failed(emitVariableAssignment(result))) + return failure(); + } else { + if (failed(emitVariableDeclaration(result, /*trailingSemicolon=*/false))) + return failure(); + os << " = "; + } + break; + } + default: + if (!forwardDeclaredVariables()) { + for (auto result : op.getResults()) { + if (failed(emitVariableDeclaration(result, /*trailingSemicolon=*/true))) + return failure(); + } + } + os << "std::tie("; + interleaveComma(op.getResults(), os, + [&](Value result) { os << getOrCreateName(result); }); + os << ") = "; + } + return success(); +} + +LogicalResult CppEmitter::emitLabel(Block &block) { + if (!hasBlockLabel(block)) { + return block.getParentOp()->emitError("Label for block not found."); + } + os << getOrCreateName(block) << ":\n"; + return success(); +} + +static LogicalResult printOperation(CppEmitter &emitter, Operation &op) { + // EmitC ops. + if (auto applyOp = dyn_cast(op)) + return printApplyOp(emitter, applyOp); + if (auto callOp = dyn_cast(op)) + return printCallOp(emitter, callOp); + if (auto constOp = dyn_cast(op)) + return printConstantOp(emitter, constOp); + if (auto includeOp = dyn_cast(op)) + // IncludeOp were already printed via printModule. + return success(); + + // SCF ops. + if (auto forOp = dyn_cast(op)) + return printForOp(emitter, forOp); + if (auto ifOp = dyn_cast(op)) + return printIfOp(emitter, ifOp); + if (auto yieldOp = dyn_cast(op)) + return printYieldOp(emitter, yieldOp); + + // Standard ops. + if (auto branchOp = dyn_cast(op)) + return printBranchOp(emitter, branchOp); + if (auto callOp = dyn_cast(op)) + return printCallOp(emitter, callOp); + if (auto branchOp = dyn_cast(op)) + return printCondBranchOp(emitter, branchOp); + if (auto constantOp = dyn_cast(op)) + return printConstantOp(emitter, constantOp); + if (auto funcOp = dyn_cast(op)) + return printFunction(emitter, funcOp); + if (auto moduleOp = dyn_cast(op)) + return printModule(emitter, moduleOp); + if (auto returnOp = dyn_cast(op)) + return printReturnOp(emitter, returnOp); + + return op.emitOpError() << "unable to find printer for op"; +} + +LogicalResult CppEmitter::emitOperation(Operation &op, bool trailingSemicolon) { + if (failed(printOperation(*this, op))) + return failure(); + os << (trailingSemicolon ? ";\n" : "\n"); + return success(); +} + +LogicalResult CppEmitter::emitType(Operation &op, Type type) { + if (auto iType = type.dyn_cast()) { + switch (iType.getWidth()) { + case 1: + return (os << "bool"), success(); + case 8: + case 16: + case 32: + case 64: + if (mapToSigned(iType.getSignedness())) { + return (os << "int" << iType.getWidth() << "_t"), success(); + } else { + return (os << "uint" << iType.getWidth() << "_t"), success(); + } + default: + return op.emitError("cannot emit integer type ") << type; + } + } + if (auto fType = type.dyn_cast()) { + switch (fType.getWidth()) { + case 32: + return (os << "float"), success(); + case 64: + return (os << "double"), success(); + default: + return op.emitError("cannot emit float type ") << type; + } + } + if (auto iType = type.dyn_cast()) { + return (os << "size_t"), success(); + } + if (auto tType = type.dyn_cast()) { + // TensorType is not supported if emitting C. + if (restrictedToC()) + return op.emitError("cannot emit tensor type if emitting C"); + if (!tType.hasRank()) + return op.emitError("cannot emit unranked tensor type"); + if (!tType.hasStaticShape()) + return op.emitError("cannot emit tensor type with non static shape"); + os << "Tensor<"; + if (failed(emitType(op, tType.getElementType()))) + return failure(); + auto shape = tType.getShape(); + for (auto dimSize : shape) { + os << ", "; + os << dimSize; + } + os << ">"; + return success(); + } + if (auto tType = type.dyn_cast()) { + return emitTupleType(op, tType.getTypes()); + } + if (auto oType = type.dyn_cast()) { + os << oType.getValue(); + return success(); + } + return op.emitError("cannot emit type ") << type; +} + +LogicalResult CppEmitter::emitTypes(Operation &op, ArrayRef types) { + switch (types.size()) { + case 0: + os << "void"; + return success(); + case 1: + return emitType(op, types.front()); + default: + return emitTupleType(op, types); + } +} + +LogicalResult CppEmitter::emitTupleType(Operation &op, ArrayRef types) { + if (restrictedToC()) + return op.emitError("cannot emit tuple type if emitting C"); + os << "std::tuple<"; + if (failed(interleaveCommaWithError( + types, os, [&](Type type) { return emitType(op, type); }))) + return failure(); + os << ">"; + return success(); +} + +LogicalResult emitc::TranslateToCpp(Operation &op, raw_ostream &os, + bool forwardDeclareVariables, + bool trailingSemicolon) { + CppEmitter emitter(os, /*restrictToC=*/false, forwardDeclareVariables); + return emitter.emitOperation(op, trailingSemicolon); +} + +LogicalResult emitc::TranslateToC(Operation &op, raw_ostream &os, + bool forwardDeclareVariables, + bool trailingSemicolon) { + CppEmitter emitter(os, /*restrictToC=*/true, forwardDeclareVariables); + return emitter.emitOperation(op, trailingSemicolon); +} diff --git a/mlir/test/Dialect/EmitC/invalid_ops.mlir b/mlir/test/Dialect/EmitC/invalid_ops.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/EmitC/invalid_ops.mlir @@ -0,0 +1,87 @@ +// RUN: mlir-opt %s -split-input-file -verify-diagnostics + +func @const_attribute_return_type_1() { + // expected-error @+1 {{'emitc.const' op requires attribute's type ('i64') to match op's return type ('i32')}} + %c0 = "emitc.const"(){value = 42: i64} : () -> i32 + return +} + +// ----- + +func @const_attribute_return_type_2() { + // expected-error @+1 {{'emitc.const' op requires attribute's type ('!emitc.opaque<"int32_t*">') to match op's return type ('!emitc.opaque<"int32_t">')}} + %c0 = "emitc.const"(){value = "nullptr" : !emitc.opaque<"int32_t*">} : () -> !emitc.opaque<"int32_t"> + return +} + +// ----- + +func @index_args_out_of_range_1() { + // expected-error @+1 {{'emitc.call' op index argument is out of range}} + emitc.call "test" () {args = [0 : index]} : () -> () + return +} + +// ----- + +func @index_args_out_of_range_2(%arg : i32) { + // expected-error @+1 {{'emitc.call' op index argument is out of range}} + emitc.call "test" (%arg, %arg) {args = [2 : index]} : (i32, i32) -> () + return +} + +// ----- + +func @empty_callee() { + // expected-error @+1 {{'emitc.call' op callee must not be empty}} + emitc.call "" () : () -> () + return +} + +// ----- + +func @nonetype_arg(%arg : i32) { + // expected-error @+1 {{'emitc.call' op array argument has no type}} + emitc.call "nonetype_arg"(%arg) {args = [0 : index, [0, 1, 2]]} : (i32) -> i32 + return +} + +// ----- + +func @float_template_argument(%arg : i32) { + // expected-error @+1 {{'emitc.call' op float literal as template argument is invalid}} + emitc.call "float_template_argument"(%arg) {template_args = [0.5 : f32]} : (i32) -> i32 + return +} + +// ----- + +func @array_template_arg(%arg : i32) { + // expected-error @+1 {{'emitc.call' op array as template arguments is invalid}} + emitc.call "nonetype_template_arg"(%arg) {template_args = [[0, 1, 2]]} : (i32) -> i32 + return +} + +// ----- + +func @dense_template_argument(%arg : i32) { + // expected-error @+1 {{'emitc.call' op dense elements as template argument are invalid}} + emitc.call "dense_template_argument"(%arg) {template_args = [dense<[1.0, 1.0]> : tensor<2xf32>]} : (i32) -> i32 + return +} + +// ----- + +func @empty_operator(%arg : i32) { + // expected-error @+1 {{'emitc.apply' op applicable operator must not be empty}} + %2 = emitc.apply ""(%arg) : (i32) -> !emitc.opaque<"int32_t*"> + return +} + +// ----- + +func @illegal_operator(%arg : i32) { + // expected-error @+1 {{'emitc.apply' op applicable operator is illegal}} + %2 = emitc.apply "+"(%arg) : (i32) -> !emitc.opaque<"int32_t*"> + return +} diff --git a/mlir/test/Dialect/EmitC/ops.mlir b/mlir/test/Dialect/EmitC/ops.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/EmitC/ops.mlir @@ -0,0 +1,25 @@ +// RUN: mlir-opt -allow-unregistered-dialect -verify-diagnostics %s | FileCheck %s + +"emitc.include" (){include = "test.h", is_standard_include} : () -> () +emitc.include "test.h" is_standard_include + +// CHECK-LABEL: func @f(%{{.*}}: i32, %{{.*}}: !custom.int32_t) -> i1 { +func @f(%arg0: i32, %f: !custom<"int32_t">) -> i1 { + %1 = "emitc.call"() {callee = "blah"} : () -> i64 + emitc.call "foo" (%1) {args = [ + 0 : index, dense<[0, 1]> : tensor<2xi32>, 0 : index + ]} : (i64) -> () + %2:3 = "bar"(%1) : (i64) -> (i1,i1,i1) + return %2#1 : i1 +} + +func @c(%arg0: i32) { + %1 = "emitc.const"(){value = 42 : i32} : () -> i32 + return +} + +func @a(%arg0: i32, %arg1: i32) { + %1 = "emitc.apply"(%arg0) {applicableOperator = "&"} : (i32) -> !emitc.opaque<"int32_t*"> + %2 = emitc.apply "&"(%arg1) : (i32) -> !emitc.opaque<"int32_t*"> + return +} diff --git a/mlir/test/Dialect/EmitC/types.mlir b/mlir/test/Dialect/EmitC/types.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/EmitC/types.mlir @@ -0,0 +1,18 @@ +// RUN: mlir-opt -allow-unregistered-dialect -verify-diagnostics %s | FileCheck %s +// check parser +// RUN: mlir-opt -allow-unregistered-dialect -verify-diagnostics %s | mlir-opt -allow-unregistered-dialect -verify-diagnostics | FileCheck %s + +// CHECK-LABEL: func @opaque_types() { +func @opaque_types() { + // CHECK-NEXT: !emitc.opaque<"int"> + emitc.call "f"() {args = [!emitc<"opaque<\"int\">">]} : () -> () + // CHECK-NEXT: !emitc.opaque<"byte"> + emitc.call "f"() {args = [!emitc<"opaque<\"byte\">">]} : () -> () + // CHECK-NEXT: !emitc.opaque<"unsigned"> + emitc.call "f"() {args = [!emitc<"opaque<\"unsigned\">">]} : () -> () + // CHECK-NEXT: !emitc.opaque<"status_t"> + emitc.call "f"() {args = [!emitc<"opaque<\"status_t\">">]} : () -> () + // CHECK-NEXT: !emitc.opaque<"std::vector"> + emitc.call "f"() {args = [!emitc.opaque<"std::vector">]} : () -> () + return +} diff --git a/mlir/test/Target/Cpp/call-cpp.mlir b/mlir/test/Target/Cpp/call-cpp.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/call-cpp.mlir @@ -0,0 +1,22 @@ +// This file contains tests for emitc call ops which are only supported if cpp code is emitted. + +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @emitc_call() { + %0 = constant 0 : index + %1:2 = emitc.call "two_results" () : () -> (i32, i32) + return +} +// CPP-DEFAULT: void emitc_call() { +// CPP-DEFAULT-NEXT: size_t [[V1:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: int32_t [[V2:[^ ]*]]; +// CPP-DEFAULT-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-DEFAULT-NEXT: std::tie([[V2]], [[V3]]) = two_results(); + +// CPP-FWDDECL: void emitc_call() { +// CPP-FWDDECL-NEXT: size_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V1]] = 0; +// CPP-FWDDECL-NEXT: std::tie([[V2]], [[V3]]) = two_results(); diff --git a/mlir/test/Target/Cpp/call.mlir b/mlir/test/Target/Cpp/call.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/call.mlir @@ -0,0 +1,29 @@ +// RUN: mlir-translate -mlir-to-c %s | FileCheck %s -check-prefix=C-DEFAULT +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @emitc_call() { + %0 = emitc.call "func_a" () : () -> i32 + %1 = emitc.call "func_b" () : () -> i32 + return +} +// C-DEFAULT: void emitc_call() { +// C-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = func_a(); +// C-DEFAULT-NEXT: int32_t [[V1:[^ ]*]] = func_b(); + +// CPP-DEFAULT: void emitc_call() { +// CPP-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = func_a(); +// CPP-DEFAULT-NEXT: int32_t [[V1:[^ ]*]] = func_b(); + +// C-FWDDECL: void emitc_call() { +// C-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// C-FWDDECL-NEXT: [[V0:]] = func_a(); +// C-FWDDECL-NEXT: [[V1:]] = func_b(); + +// CPP-FWDDECL: void emitc_call() { +// CPP-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0:]] = func_a(); +// CPP-FWDDECL-NEXT: [[V1:]] = func_b(); diff --git a/mlir/test/Target/Cpp/common-cpp.mlir b/mlir/test/Target/Cpp/common-cpp.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/common-cpp.mlir @@ -0,0 +1,104 @@ +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s + +// CHECK: #include "myheader.h" +emitc.include "myheader.h" +// CHECK: #include +emitc.include "myheader.h" is_standard_include + +// CHECK: void test_foo_print() { +func @test_foo_print() { + // CHECK: [[V1:[^ ]*]] = foo::constant({0, 1}); + %0 = emitc.call "foo::constant"() {args = [dense<[0, 1]> : tensor<2xi32>]} : () -> (i32) + // CHECK: [[V2:[^ ]*]] = foo::op_and_attr({0, 1}, [[V1]]); + %1 = emitc.call "foo::op_and_attr"(%0) {args = [dense<[0, 1]> : tensor<2xi32>, 0 : index]} : (i32) -> (i32) + // CHECK: [[V3:[^ ]*]] = foo::op_and_attr([[V2]], {0, 1}); + %2 = emitc.call "foo::op_and_attr"(%1) {args = [0 : index, dense<[0, 1]> : tensor<2xi32>]} : (i32) -> (i32) + // CHECK: foo::print([[V3]]); + emitc.call "foo::print"(%2): (i32) -> () + return +} + +// CHECK: int32_t test_single_return(int32_t [[V2:.*]]) +func @test_single_return(%arg0 : i32) -> i32 { + // CHECK: return [[V2]] + return %arg0 : i32 +} + +// CHECK: std::tuple test_multiple_return() +func @test_multiple_return() -> (i32, i32) { + // CHECK: std::tie([[V3:.*]], [[V4:.*]]) = foo::blah(); + %0:2 = emitc.call "foo::blah"() : () -> (i32, i32) + // CHECK: [[V5:[^ ]*]] = test_single_return([[V3]]); + %1 = call @test_single_return(%0#0) : (i32) -> i32 + // CHECK: return std::make_tuple([[V5]], [[V4]]); + return %1, %0#1 : i32, i32 +} + +// CHECK: test_float +func @test_float() { + // CHECK: foo::constant({(float)0.0e+00, (float)1.000000000e+00}) + %0 = emitc.call "foo::constant"() {args = [dense<[0.000000e+00, 1.000000e+00]> : tensor<2xf32>]} : () -> f32 + return +} + +// CHECK: test_uint +func @test_uint() { + // CHECK: uint32_t + %0 = emitc.call "foo::constant"() {args = [dense<[0, 1]> : tensor<2xui32>]} : () -> ui32 + // CHECK: uint64_t + %1 = emitc.call "foo::constant"() {args = [dense<[0, 1]> : tensor<2xui64>]} : () -> ui64 + return +} + +// CHECK: int64_t test_plus_int(int64_t [[V1]]) +func @test_plus_int(%arg0 : i64) -> i64 { + // CHECK: mhlo::add([[V1]], [[V1]]) + %0 = emitc.call "mhlo::add"(%arg0, %arg0) {args = [0 : index, 1 : index]} : (i64, i64) -> i64 + return %0 : i64 +} + +// CHECK: Tensor mixed_types(Tensor [[V1]]) +func @mixed_types(%arg0: tensor<2xf64>) -> tensor<2xf32> { + // CHECK: foo::mixed_types([[V1]]); + %0 = emitc.call "foo::mixed_types"(%arg0) {args = [0 : index]} : (tensor<2xf64>) -> tensor<2xf32> + return %0 : tensor<2xf32> +} + +// CHECK: Tensor mhlo_convert(Tensor [[V1]]) +func @mhlo_convert(%arg0: tensor) -> tensor { + // CHECK: mhlo::convert([[V1]]); + %0 = emitc.call "mhlo::convert"(%arg0) {args = [0 : index]} : (tensor) -> tensor + return %0 : tensor +} + +// CHECK: status_t opaque_types(bool [[V1:[^ ]*]], char [[V2:[^ ]*]]) { +func @opaque_types(%arg0: !emitc.opaque<"bool">, %arg1: !emitc.opaque<"char">) -> !emitc.opaque<"status_t"> { + // CHECK: int [[V3:[^ ]*]] = a([[V1]], [[V2]]); + %0 = emitc.call "a"(%arg0, %arg1) : (!emitc.opaque<"bool">, !emitc.opaque<"char">) -> (!emitc.opaque<"int">) + // CHECK: char [[V4:[^ ]*]] = b([[V3]]); + %1 = emitc.call "b"(%0): (!emitc.opaque<"int">) -> (!emitc.opaque<"char">) + // CHECK: status_t [[V5:[^ ]*]] = c([[V3]], [[V4]]); + %2 = emitc.call "c"(%0, %1): (!emitc.opaque<"int">, !emitc.opaque<"char">) -> (!emitc.opaque<"status_t">) + return %2 : !emitc.opaque<"status_t"> +} + +func @apply(%arg0: i32) -> !emitc.opaque<"int32_t*"> { + // CHECK: int32_t* [[V2]] = &[[V1]]; + %0 = emitc.apply "&"(%arg0) : (i32) -> !emitc.opaque<"int32_t*"> + // CHECK: int32_t [[V3]] = *[[V2]]; + %1 = emitc.apply "*"(%0) : (!emitc.opaque<"int32_t*">) -> (i32) + return %0 : !emitc.opaque<"int32_t*"> +} + +// CHECK: void emitc_constant() +func @emitc_constant() { + // CHECK: int32_t [[V0:[^ ]*]]; + %c0 = "emitc.const"(){value = #emitc.opaque<""> : i32} : () -> i32 + // CHECK: int32_t [[V1:[^ ]*]]{42}; + %c1 = "emitc.const"(){value = 42 : i32} : () -> i32 + // CHECK: int32_t* [[V2:[^ ]*]]; + %c2 = "emitc.const"(){value = #emitc.opaque<""> : !emitc.opaque<"int32_t*">} : () -> !emitc.opaque<"int32_t*"> + // CHECK: int32_t* [[V3:[^ ]*]]{nullptr}; + %c3 = "emitc.const"(){value = #emitc.opaque<"nullptr"> : !emitc.opaque<"int32_t*">} : () -> !emitc.opaque<"int32_t*"> + return +} diff --git a/mlir/test/Target/Cpp/const.mlir b/mlir/test/Target/Cpp/const.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/const.mlir @@ -0,0 +1,44 @@ +// RUN: mlir-translate -mlir-to-c %s | FileCheck %s -check-prefix=C-DEFAULT +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + + +func @emitc_constant() { + %c0 = "emitc.const"(){value = #emitc.opaque<""> : i32} : () -> i32 + %c1 = "emitc.const"(){value = 42 : i32} : () -> i32 + %c2 = "emitc.const"(){value = #emitc.opaque<""> : !emitc.opaque<"int32_t*">} : () -> !emitc.opaque<"int32_t*"> + %c3 = "emitc.const"(){value = #emitc.opaque<"NULL"> : !emitc.opaque<"int32_t*">} : () -> !emitc.opaque<"int32_t*"> + return +} +// C-DEFAULT: void emitc_constant() { +// C-DEFAULT-NEXT: int32_t [[V0:[^ ]*]]; +// C-DEFAULT-NEXT: int32_t [[V1:[^ ]*]] = 42; +// C-DEFAULT-NEXT: int32_t* [[V2:[^ ]*]]; +// C-DEFAULT-NEXT: int32_t* [[V3:[^ ]*]] = NULL; + +// CPP-DEFAULT: void emitc_constant() { +// CPP-DEFAULT-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-DEFAULT-NEXT: int32_t [[V1:[^ ]*]]{42}; +// CPP-DEFAULT-NEXT: int32_t* [[V2:[^ ]*]]; +// CPP-DEFAULT-NEXT: int32_t* [[V3:[^ ]*]]{NULL}; + +// C-FWDDECL: void emitc_constant() { +// C-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t* [[V2:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t* [[V3:[^ ]*]]; +// C-FWDDECL-NEXT: ; +// C-FWDDECL-NEXT: [[V1]] = 42; +// C-FWDDECL-NEXT: ; +// C-FWDDECL-NEXT: [[V3]] = NULL; + +// CPP-FWDDECL: void emitc_constant() { +// CPP-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t* [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t* [[V3:[^ ]*]]; +// CPP-FWDDECL-NEXT: ; +// CPP-FWDDECL-NEXT: [[V1]] = 42; +// CPP-FWDDECL-NEXT: ; +// CPP-FWDDECL-NEXT: [[V3]] = NULL; diff --git a/mlir/test/Target/Cpp/control_flow.mlir b/mlir/test/Target/Cpp/control_flow.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/control_flow.mlir @@ -0,0 +1,115 @@ +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +// simple(10, true) -> 20 +// simple(10, false) -> 30 +func @simple(i64, i1) -> i64 { +^bb0(%a: i64, %cond: i1): + cond_br %cond, ^bb1, ^bb2 +^bb1: + br ^bb3(%a: i64) +^bb2: + %b = emitc.call "add"(%a, %a) : (i64, i64) -> i64 + br ^bb3(%b: i64) +^bb3(%c: i64): + br ^bb4(%c, %a : i64, i64) +^bb4(%d : i64, %e : i64): + %0 = emitc.call "add"(%d, %e) : (i64, i64) -> i64 + return %0 : i64 +} +// C-FWDDECL: int64_t simple(int64_t [[A:[^ ]*]], bool [[COND:[^ ]*]]) { + // C-FWDDECL-NEXT: int64_t [[B:[^ ]*]]; + // C-FWDDECL-NEXT: int64_t [[V0:[^ ]*]]; + // C-FWDDECL-NEXT: int64_t [[C:[^ ]*]]; + // C-FWDDECL-NEXT: int64_t [[D:[^ ]*]]; + // C-FWDDECL-NEXT: int64_t [[E:[^ ]*]]; + // C-FWDDECL-NEXT: [[BB0:[^ ]*]]: + // C-FWDDECL-NEXT: if ([[COND]]) { + // C-FWDDECL-NEXT: goto [[BB1:[^ ]*]]; + // C-FWDDECL-NEXT: } else { + // C-FWDDECL-NEXT: goto [[BB2:[^ ]*]]; + // C-FWDDECL-NEXT: } + // C-FWDDECL-NEXT: [[BB1]]: + // C-FWDDECL-NEXT: [[C]] = [[A]]; + // C-FWDDECL-NEXT: goto [[BB3:[^ ]*]]; + // C-FWDDECL-NEXT: [[BB2]]: + // C-FWDDECL-NEXT: [[B]] = add([[A]], [[A]]); + // C-FWDDECL-NEXT: [[C]] = [[B]]; + // C-FWDDECL-NEXT: goto [[BB3]]; + // C-FWDDECL-NEXT: [[BB3]]: + // C-FWDDECL-NEXT: [[D]] = [[C]]; + // C-FWDDECL-NEXT: [[E]] = [[A]]; + // C-FWDDECL-NEXT: goto [[BB4:[^ ]*]]; + // C-FWDDECL-NEXT: [[BB4]]: + // C-FWDDECL-NEXT: [[V0]] = add([[D]], [[E]]); + // C-FWDDECL-NEXT: return [[V0]]; + + // CPP-FWDDECL: int64_t simple(int64_t [[A:[^ ]*]], bool [[COND:[^ ]*]]) { + // CPP-FWDDECL-NEXT: int64_t [[B:[^ ]*]]; + // CPP-FWDDECL-NEXT: int64_t [[V0:[^ ]*]]; + // CPP-FWDDECL-NEXT: int64_t [[C:[^ ]*]]; + // CPP-FWDDECL-NEXT: int64_t [[D:[^ ]*]]; + // CPP-FWDDECL-NEXT: int64_t [[E:[^ ]*]]; + // CPP-FWDDECL-NEXT: [[BB0:[^ ]*]]: + // CPP-FWDDECL-NEXT: if ([[COND]]) { + // CPP-FWDDECL-NEXT: goto [[BB1:[^ ]*]]; + // CPP-FWDDECL-NEXT: } else { + // CPP-FWDDECL-NEXT: goto [[BB2:[^ ]*]]; + // CPP-FWDDECL-NEXT: } + // CPP-FWDDECL-NEXT: [[BB1]]: + // CPP-FWDDECL-NEXT: [[C]] = [[A]]; + // CPP-FWDDECL-NEXT: goto [[BB3:[^ ]*]]; + // CPP-FWDDECL-NEXT: [[BB2]]: + // CPP-FWDDECL-NEXT: [[B]] = add([[A]], [[A]]); + // CPP-FWDDECL-NEXT: [[C]] = [[B]]; + // CPP-FWDDECL-NEXT: goto [[BB3]]; + // CPP-FWDDECL-NEXT: [[BB3]]: + // CPP-FWDDECL-NEXT: [[D]] = [[C]]; + // CPP-FWDDECL-NEXT: [[E]] = [[A]]; + // CPP-FWDDECL-NEXT: goto [[BB4:[^ ]*]]; + // CPP-FWDDECL-NEXT: [[BB4]]: + // CPP-FWDDECL-NEXT: [[V0]] = add([[D]], [[E]]); + // CPP-FWDDECL-NEXT: return [[V0]]; + + +func @block_labels0() { +^bb1: + br ^bb2 +^bb2: + return +} +// C-FWDDECL: void block_labels0() { + // C-FWDDECL-NEXT: label1: + // C-FWDDECL-NEXT: goto label2; + // C-FWDDECL-NEXT: label2: + // C-FWDDECL-NEXT: return; + // C-FWDDECL-NEXT: } + +// CPP-FWDDECL: void block_labels0() { + // CPP-FWDDECL-NEXT: label1: + // CPP-FWDDECL-NEXT: goto label2; + // CPP-FWDDECL-NEXT: label2: + // CPP-FWDDECL-NEXT: return; + // CPP-FWDDECL-NEXT: } + + +// Repeat the same function to make sure the names of the block labels get reset. +func @block_labels1() { +^bb1: + br ^bb2 +^bb2: + return +} +// C-FWDDECL: void block_labels1() { + // C-FWDDECL-NEXT: label1: + // C-FWDDECL-NEXT: goto label2; + // C-FWDDECL-NEXT: label2: + // C-FWDDECL-NEXT: return; + // C-FWDDECL-NEXT: } + +// CPP-FWDDECL: void block_labels1() { + // CPP-FWDDECL-NEXT: label1: + // CPP-FWDDECL-NEXT: goto label2; + // CPP-FWDDECL-NEXT: label2: + // CPP-FWDDECL-NEXT: return; + // CPP-FWDDECL-NEXT: } diff --git a/mlir/test/Target/Cpp/for.mlir b/mlir/test/Target/Cpp/for.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/for.mlir @@ -0,0 +1,154 @@ +// RUN: mlir-translate -mlir-to-c %s | FileCheck %s -check-prefix=C-DEFAULT +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @test_for(%arg0 : index, %arg1 : index, %arg2 : index) { + scf.for %i0 = %arg0 to %arg1 step %arg2 { + %0 = emitc.call "f"() : () -> i32 + } + return +} +// C-DEFAULT: void test_for(size_t [[START:[^ ]*]], size_t [[STOP:[^ ]*]], size_t [[STEP:[^ ]*]]) { +// C-DEFAULT-NEXT: for (size_t [[ITER:[^ ]*]] = [[START]]; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// C-DEFAULT-NEXT: int32_t [[V4:[^ ]*]] = f(); +// C-DEFAULT-NEXT: } +// C-DEFAULT-EMPTY: +// C-DEFAULT-NEXT: return; + +// CPP-DEFAULT: void test_for(size_t [[START:[^ ]*]], size_t [[STOP:[^ ]*]], size_t [[STEP:[^ ]*]]) { +// CPP-DEFAULT-NEXT: for (size_t [[ITER:[^ ]*]]{[[START]]}; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// CPP-DEFAULT-NEXT: int32_t [[V4:[^ ]*]] = f(); +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-EMPTY: +// CPP-DEFAULT-NEXT: return; + +// C-FWDDECL: void test_for(size_t [[START:[^ ]*]], size_t [[STOP:[^ ]*]], size_t [[STEP:[^ ]*]]) { +// C-FWDDECL-NEXT: int32_t [[V4:[^ ]*]]; +// C-FWDDECL-NEXT: for (size_t [[ITER:[^ ]*]] = [[START]]; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// C-FWDDECL-NEXT: [[V4]] = f(); +// C-FWDDECL-NEXT: } +// C-FWDDECL-EMPTY: +// C-FWDDECL-NEXT: return; + +// CPP-FWDDECL: void test_for(size_t [[START:[^ ]*]], size_t [[STOP:[^ ]*]], size_t [[STEP:[^ ]*]]) { +// CPP-FWDDECL-NEXT: int32_t [[V4:[^ ]*]]; +// CPP-FWDDECL-NEXT: for (size_t [[ITER:[^ ]*]]{[[START]]}; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// CPP-FWDDECL-NEXT: [[V4]] = f(); +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-EMPTY: +// CPP-FWDDECL-NEXT: return; + +func @test_for_yield() { + %start = constant 0 : index + %stop = constant 10 : index + %step = constant 1 : index + + %s0 = constant 0 : i32 + %p0 = constant 1.0 : f32 + + %result:2 = scf.for %iter = %start to %stop step %step iter_args(%si = %s0, %pi = %p0) -> (i32, f32) { + %sn = emitc.call "add"(%si, %iter) : (i32, index) -> i32 + %pn = emitc.call "mul"(%pi, %iter) : (f32, index) -> f32 + scf.yield %sn, %pn : i32, f32 + } + + return +} +// C-DEFAULT: void test_for_yield() { +// C-DEFAULT-NEXT: size_t [[START:[^ ]*]] = 0; +// C-DEFAULT-NEXT: size_t [[STOP:[^ ]*]] = 10; +// C-DEFAULT-NEXT: size_t [[STEP:[^ ]*]] = 1; +// C-DEFAULT-NEXT: int32_t [[S0:[^ ]*]] = 0; +// C-DEFAULT-NEXT: float [[P0:[^ ]*]] = (float)1.000000000e+00; +// C-DEFAULT-NEXT: int32_t [[SE:[^ ]*]]; +// C-DEFAULT-NEXT: float [[PE:[^ ]*]]; +// C-DEFAULT-NEXT: int32_t [[SI:[^ ]*]] = [[S0]]; +// C-DEFAULT-NEXT: float [[PI:[^ ]*]] = [[P0]]; +// C-DEFAULT-NEXT: for (size_t [[ITER:[^ ]*]] = [[START]]; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// C-DEFAULT-NEXT: int32_t [[SN:[^ ]*]] = add([[SI]], [[ITER]]); +// C-DEFAULT-NEXT: float [[PN:[^ ]*]] = mul([[PI]], [[ITER]]); +// C-DEFAULT-NEXT: [[SI]] = [[SN]]; +// C-DEFAULT-NEXT: [[PI]] = [[PN]]; +// C-DEFAULT-NEXT: } +// C-DEFAULT-NEXT: [[SE]] = [[SI]]; +// C-DEFAULT-NEXT: [[PE]] = [[PI]]; +// C-DEFAULT-EMPTY: +// C-DEFAULT-NEXT: return; + +// CPP-DEFAULT: void test_for_yield() { +// CPP-DEFAULT-NEXT: size_t [[START:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: size_t [[STOP:[^ ]*]]{10}; +// CPP-DEFAULT-NEXT: size_t [[STEP:[^ ]*]]{1}; +// CPP-DEFAULT-NEXT: int32_t [[S0:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: float [[P0:[^ ]*]]{(float)1.000000000e+00}; +// CPP-DEFAULT-NEXT: int32_t [[SE:[^ ]*]]; +// CPP-DEFAULT-NEXT: float [[PE:[^ ]*]]; +// CPP-DEFAULT-NEXT: int32_t [[SI:[^ ]*]] = [[S0]]; +// CPP-DEFAULT-NEXT: float [[PI:[^ ]*]] = [[P0]]; +// CPP-DEFAULT-NEXT: for (size_t [[ITER:[^ ]*]]{[[START]]}; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// CPP-DEFAULT-NEXT: int32_t [[SN:[^ ]*]] = add([[SI]], [[ITER]]); +// CPP-DEFAULT-NEXT: float [[PN:[^ ]*]] = mul([[PI]], [[ITER]]); +// CPP-DEFAULT-NEXT: [[SI]] = [[SN]]; +// CPP-DEFAULT-NEXT: [[PI]] = [[PN]]; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-NEXT: [[SE]] = [[SI]]; +// CPP-DEFAULT-NEXT: [[PE]] = [[PI]]; +// CPP-DEFAULT-EMPTY: +// CPP-DEFAULT-NEXT: return; + +// C-FWDDECL: void test_for_yield() { +// C-FWDDECL-NEXT: size_t [[START:[^ ]*]]; +// C-FWDDECL-NEXT: size_t [[STOP:[^ ]*]]; +// C-FWDDECL-NEXT: size_t [[STEP:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[S0:[^ ]*]]; +// C-FWDDECL-NEXT: float [[P0:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[SE:[^ ]*]]; +// C-FWDDECL-NEXT: float [[PE:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[SN:[^ ]*]]; +// C-FWDDECL-NEXT: float [[PN:[^ ]*]]; +// C-FWDDECL-NEXT: [[START]] = 0; +// C-FWDDECL-NEXT: [[STOP]] = 10; +// C-FWDDECL-NEXT: [[STEP]] = 1; +// C-FWDDECL-NEXT: [[S0]] = 0; +// C-FWDDECL-NEXT: [[P0]] = (float)1.000000000e+00; +// C-FWDDECL-NEXT: int32_t [[SI:[^ ]*]] = [[S0]]; +// C-FWDDECL-NEXT: float [[PI:[^ ]*]] = [[P0]]; +// C-FWDDECL-NEXT: for (size_t [[ITER:[^ ]*]] = [[START]]; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// C-FWDDECL-NEXT: [[SN]] = add([[SI]], [[ITER]]); +// C-FWDDECL-NEXT: [[PN]] = mul([[PI]], [[ITER]]); +// C-FWDDECL-NEXT: [[SI]] = [[SN]]; +// C-FWDDECL-NEXT: [[PI]] = [[PN]]; +// C-FWDDECL-NEXT: } +// C-FWDDECL-NEXT: [[SE]] = [[SI]]; +// C-FWDDECL-NEXT: [[PE]] = [[PI]]; +// C-FWDDECL-EMPTY: +// C-FWDDECL-NEXT: return; + +// CPP-FWDDECL: void test_for_yield() { +// CPP-FWDDECL-NEXT: size_t [[START:[^ ]*]]; +// CPP-FWDDECL-NEXT: size_t [[STOP:[^ ]*]]; +// CPP-FWDDECL-NEXT: size_t [[STEP:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[S0:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[P0:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[SE:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[PE:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[SN:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[PN:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[START]] = 0; +// CPP-FWDDECL-NEXT: [[STOP]] = 10; +// CPP-FWDDECL-NEXT: [[STEP]] = 1; +// CPP-FWDDECL-NEXT: [[S0]] = 0; +// CPP-FWDDECL-NEXT: [[P0]] = (float)1.000000000e+00; +// CPP-FWDDECL-NEXT: int32_t [[SI:[^ ]*]] = [[S0]]; +// CPP-FWDDECL-NEXT: float [[PI:[^ ]*]] = [[P0]]; +// CPP-FWDDECL-NEXT: for (size_t [[ITER:[^ ]*]]{[[START]]}; [[ITER]] < [[STOP]]; [[ITER]] += [[STEP]]) { +// CPP-FWDDECL-NEXT: [[SN]] = add([[SI]], [[ITER]]); +// CPP-FWDDECL-NEXT: [[PN]] = mul([[PI]], [[ITER]]); +// CPP-FWDDECL-NEXT: [[SI]] = [[SN]]; +// CPP-FWDDECL-NEXT: [[PI]] = [[PN]]; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-NEXT: [[SE]] = [[SI]]; +// CPP-FWDDECL-NEXT: [[PE]] = [[PI]]; +// CPP-FWDDECL-EMPTY: +// CPP-FWDDECL-NEXT: return; diff --git a/mlir/test/Target/Cpp/if.mlir b/mlir/test/Target/Cpp/if.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/if.mlir @@ -0,0 +1,205 @@ +// RUN: mlir-translate -mlir-to-c %s | FileCheck %s -check-prefix=C-DEFAULT +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @test_if(%arg0: i1, %arg1: f32) { + scf.if %arg0 { + %0 = emitc.call "func_const"(%arg1) : (f32) -> i32 + } + return +} +// C-DEFAULT: void test_if(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-DEFAULT-NEXT: if ([[V0]]) { +// C-DEFAULT-NEXT: int32_t [[V2:[^ ]*]] = func_const([[V1]]); +// C-DEFAULT-NEXT: ; +// C-DEFAULT-NEXT: } +// C-DEFAULT-EMPTY: +// C-DEFAULT-NEXT: return; + +// CPP-DEFAULT: void test_if(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-DEFAULT-NEXT: if ([[V0]]) { +// CPP-DEFAULT-NEXT: int32_t [[V2:[^ ]*]] = func_const([[V1]]); +// CPP-DEFAULT-NEXT: ; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-EMPTY: +// CPP-DEFAULT-NEXT: return; + +// C-FWDDECL: void test_if(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-FWDDECL-NEXT: int32_t [[V2:[^ ]*]]; +// C-FWDDECL-NEXT: if ([[V0]]) { +// C-FWDDECL-NEXT: [[V2]] = func_const([[V1]]); +// C-FWDDECL-NEXT: ; +// C-FWDDECL-NEXT: } +// C-FWDDECL-EMPTY: +// C-FWDDECL-NEXT: return; + +// CPP-FWDDECL: void test_if(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-FWDDECL-NEXT: int32_t [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: if ([[V0]]) { +// CPP-FWDDECL-NEXT: [[V2]] = func_const([[V1]]); +// CPP-FWDDECL-NEXT: ; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-EMPTY: +// CPP-FWDDECL-NEXT: return; + + +func @test_if_else(%arg0: i1, %arg1: f32) { + scf.if %arg0 { + %0 = emitc.call "func_true"(%arg1) : (f32) -> i32 + } else { + %0 = emitc.call "func_false"(%arg1) : (f32) -> i32 + } + return +} +// C-DEFAULT: void test_if_else(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-DEFAULT-NEXT: if ([[V0]]) { +// C-DEFAULT-NEXT: int32_t [[V2:[^ ]*]] = func_true([[V1]]); +// C-DEFAULT-NEXT: ; +// C-DEFAULT-NEXT: } +// C-DEFAULT-NEXT: else { +// C-DEFAULT-NEXT: int32_t [[V3:[^ ]*]] = func_false([[V1]]); +// C-DEFAULT-NEXT: ; +// C-DEFAULT-NEXT: } +// C-DEFAULT-EMPTY: +// C-DEFAULT-NEXT: return; + +// CPP-DEFAULT: void test_if_else(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-DEFAULT-NEXT: if ([[V0]]) { +// CPP-DEFAULT-NEXT: int32_t [[V2:[^ ]*]] = func_true([[V1]]); +// CPP-DEFAULT-NEXT: ; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-NEXT: else { +// CPP-DEFAULT-NEXT: int32_t [[V3:[^ ]*]] = func_false([[V1]]); +// CPP-DEFAULT-NEXT: ; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-EMPTY: +// CPP-DEFAULT-NEXT: return; + +// C-FWDDECL: void test_if_else(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-FWDDECL-NEXT: int32_t [[V2:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// C-FWDDECL-NEXT: if ([[V0]]) { +// C-FWDDECL-NEXT: [[V2]] = func_true([[V1]]); +// C-FWDDECL-NEXT: ; +// C-FWDDECL-NEXT: } +// C-FWDDECL-NEXT: else { +// C-FWDDECL-NEXT: [[V3]] = func_false([[V1]]); +// C-FWDDECL-NEXT: ; +// C-FWDDECL-NEXT: } +// C-FWDDECL-EMPTY: +// C-FWDDECL-NEXT: return; + +// CPP-FWDDECL: void test_if_else(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-FWDDECL-NEXT: int32_t [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-FWDDECL-NEXT: if ([[V0]]) { +// CPP-FWDDECL-NEXT: [[V2]] = func_true([[V1]]); +// CPP-FWDDECL-NEXT: ; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-NEXT: else { +// CPP-FWDDECL-NEXT: [[V3]] = func_false([[V1]]); +// CPP-FWDDECL-NEXT: ; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-EMPTY: +// CPP-FWDDECL-NEXT: return; + + +func @test_if_yield(%arg0: i1, %arg1: f32) { + %0 = constant 0 : i8 + %x, %y = scf.if %arg0 -> (i32, f64) { + %1 = emitc.call "func_true_1"(%arg1) : (f32) -> i32 + %2 = emitc.call "func_true_2"(%arg1) : (f32) -> f64 + scf.yield %1, %2 : i32, f64 + } else { + %1 = emitc.call "func_false_1"(%arg1) : (f32) -> i32 + %2 = emitc.call "func_false_2"(%arg1) : (f32) -> f64 + scf.yield %1, %2 : i32, f64 + } + return +} +// C-DEFAULT: void test_if_yield(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-DEFAULT-NEXT: int8_t [[V2:[^ ]*]] = 0; +// C-DEFAULT-NEXT: int32_t [[V3:[^ ]*]]; +// C-DEFAULT-NEXT: double [[V4:[^ ]*]]; +// C-DEFAULT-NEXT: if ([[V0]]) { +// C-DEFAULT-NEXT: int32_t [[V5:[^ ]*]] = func_true_1([[V1]]); +// C-DEFAULT-NEXT: double [[V6:[^ ]*]] = func_true_2([[V1]]); +// C-DEFAULT-NEXT: [[V3]] = [[V5]]; +// C-DEFAULT-NEXT: [[V4]] = [[V6]]; +// C-DEFAULT-NEXT: } +// C-DEFAULT-NEXT: else { +// C-DEFAULT-NEXT: int32_t [[V7:[^ ]*]] = func_false_1([[V1]]); +// C-DEFAULT-NEXT: double [[V8:[^ ]*]] = func_false_2([[V1]]); +// C-DEFAULT-NEXT: [[V3]] = [[V7]]; +// C-DEFAULT-NEXT: [[V4]] = [[V8]]; +// C-DEFAULT-NEXT: } +// C-DEFAULT-EMPTY: +// C-DEFAULT-NEXT: return; + +// CPP-DEFAULT: void test_if_yield(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-DEFAULT-NEXT: int8_t [[V2:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-DEFAULT-NEXT: double [[V4:[^ ]*]]; +// CPP-DEFAULT-NEXT: if ([[V0]]) { +// CPP-DEFAULT-NEXT: int32_t [[V5:[^ ]*]] = func_true_1([[V1]]); +// CPP-DEFAULT-NEXT: double [[V6:[^ ]*]] = func_true_2([[V1]]); +// CPP-DEFAULT-NEXT: [[V3]] = [[V5]]; +// CPP-DEFAULT-NEXT: [[V4]] = [[V6]]; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-NEXT: else { +// CPP-DEFAULT-NEXT: int32_t [[V7:[^ ]*]] = func_false_1([[V1]]); +// CPP-DEFAULT-NEXT: double [[V8:[^ ]*]] = func_false_2([[V1]]); +// CPP-DEFAULT-NEXT: [[V3]] = [[V7]]; +// CPP-DEFAULT-NEXT: [[V4]] = [[V8]]; +// CPP-DEFAULT-NEXT: } +// CPP-DEFAULT-EMPTY: +// CPP-DEFAULT-NEXT: return; + +// C-FWDDECL: void test_if_yield(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// C-FWDDECL-NEXT: int8_t [[V2:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// C-FWDDECL-NEXT: double [[V4:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V5:[^ ]*]]; +// C-FWDDECL-NEXT: double [[V6:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V7:[^ ]*]]; +// C-FWDDECL-NEXT: double [[V8:[^ ]*]]; +// C-FWDDECL-NEXT: [[V2]] = 0; +// C-FWDDECL-NEXT: if ([[V0]]) { +// C-FWDDECL-NEXT: [[V5]] = func_true_1([[V1]]); +// C-FWDDECL-NEXT: [[V6]] = func_true_2([[V1]]); +// C-FWDDECL-NEXT: [[V3]] = [[V5]]; +// C-FWDDECL-NEXT: [[V4]] = [[V6]]; +// C-FWDDECL-NEXT: } +// C-FWDDECL-NEXT: else { +// C-FWDDECL-NEXT: [[V7]] = func_false_1([[V1]]); +// C-FWDDECL-NEXT: [[V8]] = func_false_2([[V1]]); +// C-FWDDECL-NEXT: [[V3]] = [[V7]]; +// C-FWDDECL-NEXT: [[V4]] = [[V8]]; +// C-FWDDECL-NEXT: } +// C-FWDDECL-EMPTY: +// C-FWDDECL-NEXT: return; + +// CPP-FWDDECL: void test_if_yield(bool [[V0:[^ ]*]], float [[V1:[^ ]*]]) { +// CPP-FWDDECL-NEXT: int8_t [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-FWDDECL-NEXT: double [[V4:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V5:[^ ]*]]; +// CPP-FWDDECL-NEXT: double [[V6:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V7:[^ ]*]]; +// CPP-FWDDECL-NEXT: double [[V8:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V2]] = 0; +// CPP-FWDDECL-NEXT: if ([[V0]]) { +// CPP-FWDDECL-NEXT: [[V5]] = func_true_1([[V1]]); +// CPP-FWDDECL-NEXT: [[V6]] = func_true_2([[V1]]); +// CPP-FWDDECL-NEXT: [[V3]] = [[V5]]; +// CPP-FWDDECL-NEXT: [[V4]] = [[V6]]; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-NEXT: else { +// CPP-FWDDECL-NEXT: [[V7]] = func_false_1([[V1]]); +// CPP-FWDDECL-NEXT: [[V8]] = func_false_2([[V1]]); +// CPP-FWDDECL-NEXT: [[V3]] = [[V7]]; +// CPP-FWDDECL-NEXT: [[V4]] = [[V8]]; +// CPP-FWDDECL-NEXT: } +// CPP-FWDDECL-EMPTY: +// CPP-FWDDECL-NEXT: return; diff --git a/mlir/test/Target/Cpp/invalid-c.mlir b/mlir/test/Target/Cpp/invalid-c.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/invalid-c.mlir @@ -0,0 +1,25 @@ +// RUN: mlir-translate -split-input-file -mlir-to-c -verify-diagnostics %s + +func @func_template() { + // expected-error @+1 {{'emitc.call' op template arguments are not supported if emitting C}} + %0 = emitc.call "func_template" () {template_args = [i32]} : () -> i32 + return +} + +// ----- + +func @func_tuple() { + // expected-error @+1 {{cannot emit tuple type if emitting C}} + %cst = "emitc.const"(){value = tuple} : () -> i32 + return +} + +// ----- + +func @func_tensor() { + // expected-error @+1 {{cannot emit tensor type if emitting C}} + %cst = "emitc.const"(){value = dense<1> : tensor<24xi32>} : () -> tensor<24xi32> + return +} + +// ----- diff --git a/mlir/test/Target/Cpp/invalid-cpp.mlir b/mlir/test/Target/Cpp/invalid-cpp.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/invalid-cpp.mlir @@ -0,0 +1,13 @@ +// RUN: mlir-translate -split-input-file -mlir-to-cpp -verify-diagnostics %s + +// expected-error@+1 {{cannot emit tensor type with non static shape}} +func @non_static_shape(%arg0 : tensor) { + return +} + +// ----- + +// expected-error@+1 {{cannot emit unranked tensor type}} +func @unranked_tensor(%arg0 : tensor<*xf32>) { + return +} diff --git a/mlir/test/Target/Cpp/invalid.mlir b/mlir/test/Target/Cpp/invalid.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/invalid.mlir @@ -0,0 +1,46 @@ +// RUN: mlir-translate -split-input-file -mlir-to-cpp -verify-diagnostics %s +// RUN: mlir-translate -split-input-file -mlir-to-c -verify-diagnostics %s + +// expected-error@+1 {{'func' op with multiple blocks needs forward declared variables}} +func @multiple_blocks() { +^bb1: + br ^bb2 +^bb2: + return +} + +// ----- + +func @unsupported_std_op(%arg0: f64) -> f64 { + // expected-error@+1 {{'std.absf' op unable to find printer for op}} + %0 = absf %arg0 : f64 + return %0 : f64 +} + +// ----- + +// expected-error@+1 {{cannot emit integer type 'i80'}} +func @unsupported_integer_type(%arg0 : i80) { + return +} + +// ----- + +// expected-error@+1 {{cannot emit float type 'f80'}} +func @unsupported_float_type(%arg0 : f80) { + return +} + +// ----- + +// expected-error@+1 {{cannot emit type 'memref<100xf32>'}} +func @memref_type(%arg0 : memref<100xf32>) { + return +} + +// ----- + +// expected-error@+1 {{cannot emit type 'vector<100xf32>'}} +func @vector_type(%arg0 : vector<100xf32>) { + return +} diff --git a/mlir/test/Target/Cpp/stdops-cpp.mlir b/mlir/test/Target/Cpp/stdops-cpp.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/stdops-cpp.mlir @@ -0,0 +1,66 @@ +// This file contains tests for std ops which are only supported if cpp code is emitted. + +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @std_constant() { + %c0 = constant dense<0> : tensor + %c1 = constant dense<[0, 1]> : tensor<2xindex> + %c2 = constant dense<[[0.0, 1.0], [2.0, 3.0]]> : tensor<2x2xf32> + return +} +// CPP-DEFAULT: void std_constant() { +// CPP-DEFAULT-NEXT: Tensor [[V0:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: Tensor [[V1:[^ ]*]]{0, 1}; +// CPP-DEFAULT-NEXT: Tensor [[V2:[^ ]*]]{(float)0.0e+00, (float)1.000000000e+00, (float)2.000000000e+00, (float)3.000000000e+00}; + +// CPP-FWDDECL: void std_constant() { +// CPP-FWDDECL-NEXT: Tensor [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: Tensor [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: Tensor [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0]] = {0}; +// CPP-FWDDECL-NEXT: [[V1]] = {0, 1}; +// CPP-FWDDECL-NEXT: [[V2]] = {(float)0.0e+00, (float)1.000000000e+00, (float)2.000000000e+00, (float)3.000000000e+00}; + +func @std_call() { + %c = constant 0 : i8 + %0:2 = call @two_results () : () -> (i32, f32) + %1:2 = call @two_results () : () -> (i32, f32) + return +} +// CPP-DEFAULT: void std_call() { +// CPP-DEFAULT-NEXT: int8_t [[V0:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: int32_t [[V1:[^ ]*]]; +// CPP-DEFAULT-NEXT: float [[V2:[^ ]*]]; +// CPP-DEFAULT-NEXT: std::tie([[V1]], [[V2]]) = two_results(); +// CPP-DEFAULT-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-DEFAULT-NEXT: float [[V4:[^ ]*]]; +// CPP-DEFAULT-NEXT: std::tie([[V3]], [[V4]]) = two_results(); + +// CPP-FWDDECL: void std_call() { +// CPP-FWDDECL-NEXT: int8_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V3:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[V4:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0]] = 0; +// CPP-FWDDECL-NEXT: std::tie([[V1]], [[V2]]) = two_results(); +// CPP-FWDDECL-NEXT: std::tie([[V3]], [[V4]]) = two_results(); + + +func @two_results() -> (i32, f32) { + %0 = constant 0 : i32 + %1 = constant 1.0 : f32 + return %0, %1 : i32, f32 +} +// CPP-DEFAULT: std::tuple two_results() { +// CPP-DEFAULT: int32_t [[V0:[^ ]*]]{0}; +// CPP-DEFAULT: float [[V1:[^ ]*]]{(float)1.000000000e+00}; +// CPP-DEFAULT: return std::make_tuple([[V0]], [[V1]]); + +// CPP-FWDDECL: std::tuple two_results() { +// CPP-FWDDECL: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL: float [[V1:[^ ]*]]; +// CPP-FWDDECL: [[V0]] = 0; +// CPP-FWDDECL: [[V1]] = (float)1.000000000e+00; +// CPP-FWDDECL: return std::make_tuple([[V0]], [[V1]]); diff --git a/mlir/test/Target/Cpp/stdops.mlir b/mlir/test/Target/Cpp/stdops.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Target/Cpp/stdops.mlir @@ -0,0 +1,100 @@ +// RUN: mlir-translate -mlir-to-c %s | FileCheck %s -check-prefix=C-DEFAULT +// RUN: mlir-translate -mlir-to-cpp %s | FileCheck %s -check-prefix=CPP-DEFAULT +// RUN: mlir-translate -mlir-to-c-forward-declared %s | FileCheck %s -check-prefix=C-FWDDECL +// RUN: mlir-translate -mlir-to-cpp-forward-declared %s | FileCheck %s -check-prefix=CPP-FWDDECL + +func @std_constant() { + %c0 = constant 0 : i32 + %c1 = constant 2 : index + %c2 = constant 2.0 : f32 + return +} +// C-DEFAULT: void std_constant() { +// C-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = 0; +// C-DEFAULT-NEXT: size_t [[V1:[^ ]*]] = 2; +// C-DEFAULT-NEXT: float [[V2:[^ ]*]] = (float)2.000000000e+00; + +// CPP-DEFAULT: void std_constant() { +// CPP-DEFAULT-NEXT: int32_t [[V0:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: size_t [[V1:[^ ]*]]{2}; +// CPP-DEFAULT-NEXT: float [[V2:[^ ]*]]{(float)2.000000000e+00}; + +// C-FWDDECL: void std_constant() { +// C-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// C-FWDDECL-NEXT: size_t [[V1:[^ ]*]]; +// C-FWDDECL-NEXT: float [[V2:[^ ]*]]; +// C-FWDDECL-NEXT: [[V0]] = 0; +// C-FWDDECL-NEXT: [[V1]] = 2; +// C-FWDDECL-NEXT: [[V2]] = (float)2.000000000e+00; + +// CPP-FWDDECL: void std_constant() { +// CPP-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: size_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: float [[V2:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0]] = 0; +// CPP-FWDDECL-NEXT: [[V1]] = 2; +// CPP-FWDDECL-NEXT: [[V2]] = (float)2.000000000e+00; + +func @std_call() { + %0 = call @one_result () : () -> i32 + %1 = call @one_result () : () -> i32 + return +} +// C-DEFAULT: void std_call() { +// C-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = one_result(); +// C-DEFAULT-NEXT: int32_t [[V1:[^ ]*]] = one_result(); + +// CPP-DEFAULT: void std_call() { +// CPP-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = one_result(); +// CPP-DEFAULT-NEXT: int32_t [[V1:[^ ]*]] = one_result(); + +// C-FWDDECL: void std_call() { +// C-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// C-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// C-FWDDECL-NEXT: [[V0]] = one_result(); +// C-FWDDECL-NEXT: [[V1]] = one_result(); + +// CPP-FWDDECL: void std_call() { +// CPP-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: int32_t [[V1:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0]] = one_result(); +// CPP-FWDDECL-NEXT: [[V1]] = one_result(); + + +func @one_result() -> i32 { + %0 = constant 0 : i32 + return %0 : i32 +} +// C-DEFAULT: int32_t one_result() { +// C-DEFAULT-NEXT: int32_t [[V0:[^ ]*]] = 0; +// C-DEFAULT-NEXT: return [[V0]]; + +// CPP-DEFAULT: int32_t one_result() { +// CPP-DEFAULT-NEXT: int32_t [[V0:[^ ]*]]{0}; +// CPP-DEFAULT-NEXT: return [[V0]]; + +// C-FWDDECL: int32_t one_result() { +// C-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// C-FWDDECL-NEXT: [[V0]] = 0; +// C-FWDDECL-NEXT: return [[V0]]; + +// CPP-FWDDECL: int32_t one_result() { +// CPP-FWDDECL-NEXT: int32_t [[V0:[^ ]*]]; +// CPP-FWDDECL-NEXT: [[V0]] = 0; +// CPP-FWDDECL-NEXT: return [[V0]]; + + +func @single_return_statement(%arg0 : i32) -> i32 { + return %arg0 : i32 +} +// C-DEFAULT: int32_t single_return_statement(int32_t [[V0:[^ ]*]]) { +// C-DEFAULT-NEXT: return [[V0]]; + +// CPP-DEFAULT: int32_t single_return_statement(int32_t [[V0:[^ ]*]]) { +// CPP-DEFAULT-NEXT: return [[V0]]; + +// C-FWDDECL: int32_t single_return_statement(int32_t [[V0:[^ ]*]]) { +// C-FWDDECL-NEXT: return [[V0]]; + +// CPP-FWDDECL: int32_t single_return_statement(int32_t [[V0:[^ ]*]]) { +// CPP-FWDDECL-NEXT: return [[V0]]; diff --git a/mlir/test/mlir-opt/commandline.mlir b/mlir/test/mlir-opt/commandline.mlir --- a/mlir/test/mlir-opt/commandline.mlir +++ b/mlir/test/mlir-opt/commandline.mlir @@ -8,6 +8,7 @@ // CHECK-NEXT: async // CHECK-NEXT: complex // CHECK-NEXT: dlti +// CHECK-NEXT: emitc // CHECK-NEXT: gpu // CHECK-NEXT: linalg // CHECK-NEXT: llvm