[EmitC] Remove the forked emitter and generate all the code in the conversion pass (#16357)
This resolves the longstanding painpoint of having a custom fork of the
EmitC dialect to C emitter infrastructure in IREE. This is done by
generating `emitc.verbatim` ops for the missing pieces in the
conversion.
Things that need to be addressed:
- Move `createModuleStructure` into a separate file. Skipped for now as
it introduces hundreds lines of diff noise
- The creation of the descriptor arrays are a 1:1 copy from the old
CModuleTarget. Helper structs/functions around these would help
readability.
- Convert the `vm.module` to a `builtin.module` in the conversion.
(Would need to make the conversion pass run on the outer
`builtin.module` and nest the pass manager correctly, I think?)
- The `FuncAnalysis` became kind of a sink where different pieces of
information are mashed together with subtle bugs around uninitialized
variables from the different constructos.
---------
Co-authored-by: Marius Brehler <marius.brehler@iml.fraunhofer.de>
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/ConvertVMToEmitC.cpp b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/ConvertVMToEmitC.cpp
index ca276f5..a4cb660 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/ConvertVMToEmitC.cpp
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/ConvertVMToEmitC.cpp
@@ -14,11 +14,9 @@
#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h"
#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/VMAnalysis.h"
#include "iree/compiler/Dialect/VM/IR/VMOps.h"
-#include "iree/compiler/Dialect/VM/Utils/CallingConvention.h"
#include "llvm/ADT/TypeSwitch.h"
#include "mlir/Dialect/ControlFlow/IR/ControlFlowOps.h"
#include "mlir/Dialect/EmitC/IR/EmitC.h"
-#include "mlir/Dialect/Func/IR/FuncOps.h"
#include "mlir/IR/Builders.h"
#include "mlir/IR/BuiltinDialect.h"
#include "mlir/IR/BuiltinOps.h"
@@ -46,15 +44,6 @@
CCONV_ARGUMENT_MODULE_STATE,
};
-/// The EmitC dialect is currently missing operations to cleanly represent some
-/// constructs we need for the C target. This includes storage class specifiers
-/// on functions, forward declarations of functions, globals and arrays.
-/// As a workaround the conversion currently adds bits of information via
-/// attributes that later get used by the CModuleTarget.
-void attachAttribute(Operation *op, StringRef name, Attribute value) {
- op->setAttr(name, value);
-}
-
/// Create a call to memset to clear a struct
LogicalResult clearStruct(OpBuilder builder, Value structValue) {
auto loc = structValue.getLoc();
@@ -82,10 +71,11 @@
auto moduleOp = funcOp.getOperation()->getParentOfType<IREE::VM::ModuleOp>();
FunctionType funcType = funcOp.getFunctionType();
- std::string name =
- std::string(moduleOp.getName()) + "_" + std::string(funcOp.getName());
- std::string moduleTypeName = (moduleOp.getName() + "_t").str();
- std::string moduleStateTypeName = (moduleOp.getName() + "_state_t").str();
+ std::string name = moduleOp.getName().str() + "_" + funcOp.getName().str();
+ std::string moduleTypeName =
+ std::string("struct ") + moduleOp.getName().str() + "_t";
+ std::string moduleStateTypeName =
+ std::string("struct ") + moduleOp.getName().str() + "_state_t";
Type stackType =
emitc::PointerType::get(emitc::OpaqueType::get(ctx, "iree_vm_stack_t"));
@@ -113,17 +103,10 @@
auto newFuncType = mlir::FunctionType::get(
ctx, {inputTypes}, {emitc::OpaqueType::get(ctx, "iree_status_t")});
- auto newFuncOp = builder.create<mlir::func::FuncOp>(loc, name, newFuncType);
-
- attachAttribute(newFuncOp, "emitc.static", UnitAttr::get(ctx));
-
- std::optional<std::string> callingConvention =
- makeCallingConventionString(funcOp);
-
- // Annotate new function with calling convention string which gets used in
- // the CModuleTarget.
- attachAttribute(newFuncOp, "vm.calling_convention",
- StringAttr::get(ctx, callingConvention.value()));
+ auto newFuncOp = builder.create<mlir::emitc::FuncOp>(loc, name, newFuncType);
+ newFuncOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ newFuncOp.setPrivate();
// This call shold be equivalent to rewriter.inlineRegionBefore()
newFuncOp.getFunctionBody().getBlocks().splice(
@@ -373,20 +356,23 @@
/// Releases refs which are local to the function as well as ref arguments.
void releaseRefs(OpBuilder &builder, Location location,
- mlir::func::FuncOp funcOp,
+ mlir::emitc::FuncOp funcOp,
IREE::VM::ModuleAnalysis &moduleAnalysis) {
auto ctx = builder.getContext();
auto &funcAnalysis = moduleAnalysis.lookupFunction(funcOp);
- for (auto pair : funcAnalysis.localRefs()) {
- Operation *op = pair.second;
+ if (funcAnalysis.hasLocalRefs()) {
- assert(isa<emitc::ApplyOp>(op));
+ for (auto pair : funcAnalysis.localRefs()) {
+ Operation *op = pair.second;
- Value localRef = cast<emitc::ApplyOp>(op).getResult();
+ assert(isa<emitc::ApplyOp>(op));
- emitc_builders::ireeVmRefRelease(builder, location, localRef);
+ Value localRef = cast<emitc::ApplyOp>(op).getResult();
+
+ emitc_builders::ireeVmRefRelease(builder, location, localRef);
+ }
}
// We only release the original arguments not the results which were appended
@@ -403,8 +389,8 @@
}
}
-/// Generate an emitc.call op with one result and split the current block into a
-/// continuation and failure block based on the truthiness of the result
+/// Generate an emitc.call_opaque op with one result and split the current block
+/// into a continuation and failure block based on the truthiness of the result
/// value, i.e. a truthy value branches to the continuation block when
/// `negateCondition` is false.
emitc::CallOpaqueOp failableCall(
@@ -444,7 +430,7 @@
}
builder.setInsertionPointToEnd(condBlock);
- builder.create<IREE::VM::CondBranchOp>(
+ builder.create<mlir::cf::CondBranchOp>(
location, conditionI1.getResult(),
negateCondition ? failureBlock : continuationBlock,
negateCondition ? continuationBlock : failureBlock);
@@ -461,11 +447,12 @@
auto blockBuilder = [&builder, &location,
&moduleAnalysis](emitc::CallOpaqueOp &callOp) {
Block *block = builder.getBlock();
- mlir::func::FuncOp funcOp = cast<mlir::func::FuncOp>(block->getParentOp());
+ mlir::emitc::FuncOp funcOp =
+ cast<mlir::emitc::FuncOp>(block->getParentOp());
releaseRefs(builder, location, funcOp, moduleAnalysis);
- builder.create<mlir::func::ReturnOp>(location, callOp.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(location, callOp.getResult(0));
};
auto ctx = builder.getContext();
@@ -483,7 +470,8 @@
auto ctx = builder.getContext();
Block *block = builder.getBlock();
- mlir::func::FuncOp funcOp = cast<mlir::func::FuncOp>(block->getParentOp());
+ mlir::emitc::FuncOp funcOp =
+ cast<mlir::emitc::FuncOp>(block->getParentOp());
releaseRefs(builder, location, funcOp, moduleAnalysis);
@@ -497,23 +485,23 @@
/*templateArgs=*/ArrayAttr{},
/*operands=*/ArrayRef<Value>{});
- builder.create<mlir::func::ReturnOp>(location, statusOp.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(location, statusOp.getResult(0));
};
return failableCall(builder, location, type, callee, args, operands,
blockBuilder);
}
-/// Generate a mlir.call op with one result and split the current block into a
+/// Generate a emitc.call op with one result and split the current block into a
/// continuation and failure block based on the truthiness of the result
/// value, i.e. a truthy value branches to the continuation block when
/// `negateCondition` is false.
-mlir::func::CallOp failableCall(
- OpBuilder &builder, Location location, mlir::func::FuncOp &callee,
+mlir::emitc::CallOp failableCall(
+ OpBuilder &builder, Location location, mlir::emitc::FuncOp &callee,
ArrayRef<Value> operands,
- const std::function<void(mlir::func::CallOp &)> &failureBlockBuilder,
+ const std::function<void(mlir::emitc::CallOp &)> &failureBlockBuilder,
bool negateCondition = false) {
- auto callOp = builder.create<mlir::func::CallOp>(
+ auto callOp = builder.create<mlir::emitc::CallOp>(
/*location=*/location,
/*callee=*/callee,
/*operands=*/operands);
@@ -542,7 +530,7 @@
}
builder.setInsertionPointToEnd(condBlock);
- builder.create<IREE::VM::CondBranchOp>(
+ builder.create<mlir::cf::CondBranchOp>(
location, conditionI1.getResult(),
negateCondition ? failureBlock : continuationBlock,
negateCondition ? continuationBlock : failureBlock);
@@ -552,18 +540,19 @@
return callOp;
}
-mlir::func::CallOp returnIfError(OpBuilder &builder, Location location,
- mlir::func::FuncOp &callee,
- ArrayRef<Value> operands,
- IREE::VM::ModuleAnalysis &moduleAnalysis) {
+mlir::emitc::CallOp returnIfError(OpBuilder &builder, Location location,
+ mlir::emitc::FuncOp &callee,
+ ArrayRef<Value> operands,
+ IREE::VM::ModuleAnalysis &moduleAnalysis) {
auto blockBuilder = [&builder, &location,
- &moduleAnalysis](mlir::func::CallOp &callOp) {
+ &moduleAnalysis](mlir::emitc::CallOp &callOp) {
Block *block = builder.getBlock();
- mlir::func::FuncOp funcOp = cast<mlir::func::FuncOp>(block->getParentOp());
+ mlir::emitc::FuncOp funcOp =
+ cast<mlir::emitc::FuncOp>(block->getParentOp());
releaseRefs(builder, location, funcOp, moduleAnalysis);
- builder.create<mlir::func::ReturnOp>(location, callOp.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(location, callOp.getResult(0));
};
return failableCall(builder, location, callee, operands, blockBuilder,
@@ -590,19 +579,20 @@
ctx, {emitc::PointerType::get(emitc::OpaqueType::get(ctx, "void"))},
{});
- auto funcOp = builder.create<mlir::func::FuncOp>(
+ auto funcOp = builder.create<mlir::emitc::FuncOp>(
loc, moduleName + "_destroy", funcType);
+ funcOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ funcOp.setPrivate();
- moduleAnalysis.addDummy(funcOp);
-
- attachAttribute(funcOp, "emitc.static", UnitAttr::get(ctx));
+ moduleAnalysis.addDummy(funcOp, /*emitAtEnd=*/false);
Block *entryBlock = funcOp.addEntryBlock();
const BlockArgument moduleArg = funcOp.getArgument(moduleArgIndex);
builder.setInsertionPointToStart(entryBlock);
- std::string moduleTypeName = moduleName + "_t";
+ std::string moduleTypeName = std::string("struct ") + moduleName + "_t";
auto castedModuleOp = builder.create<emitc::CastOp>(
/*location=*/loc,
@@ -624,7 +614,7 @@
/*templateArgs=*/ArrayAttr{},
/*operands=*/ArrayRef<Value>{allocatorOp, castedModuleOp.getResult()});
- builder.create<mlir::func::ReturnOp>(loc);
+ builder.create<mlir::emitc::ReturnOp>(loc, nullptr);
}
// iree_status_t alloc_state(void*, iree_allocator_t,
@@ -643,12 +633,13 @@
emitc::OpaqueType::get(ctx, "iree_vm_module_state_t")))},
{emitc::OpaqueType::get(ctx, "iree_status_t")});
- auto funcOp = builder.create<mlir::func::FuncOp>(
+ auto funcOp = builder.create<mlir::emitc::FuncOp>(
loc, moduleName + "_alloc_state", funcType);
+ funcOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ funcOp.setPrivate();
- moduleAnalysis.addDummy(funcOp);
-
- attachAttribute(funcOp, "emitc.static", UnitAttr::get(ctx));
+ moduleAnalysis.addDummy(funcOp, /*emitAtEnd=*/false);
Block *entryBlock = funcOp.addEntryBlock();
@@ -658,7 +649,8 @@
builder.setInsertionPointToStart(entryBlock);
- std::string moduleStateTypeName = moduleName + "_state_t";
+ std::string moduleStateTypeName =
+ std::string("struct ") + moduleName + "_state_t";
Value state = emitc_builders::allocateVariable(
builder, loc,
@@ -807,7 +799,7 @@
auto status = emitc_builders::ireeOkStatus(builder, loc);
- builder.create<mlir::func::ReturnOp>(loc, status);
+ builder.create<mlir::emitc::ReturnOp>(loc, status);
}
// void free_state(void*, iree_vm_module_state_t*)
@@ -823,12 +815,13 @@
emitc::OpaqueType::get(ctx, "iree_vm_module_state_t"))},
{});
- auto funcOp = builder.create<mlir::func::FuncOp>(
+ auto funcOp = builder.create<mlir::emitc::FuncOp>(
loc, moduleName + "_free_state", funcType);
+ funcOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ funcOp.setPrivate();
- moduleAnalysis.addDummy(funcOp);
-
- attachAttribute(funcOp, "emitc.static", UnitAttr::get(ctx));
+ moduleAnalysis.addDummy(funcOp, /*emitAtEnd=*/false);
Block *entryBlock = funcOp.addEntryBlock();
@@ -837,7 +830,8 @@
builder.setInsertionPointToStart(entryBlock);
- std::string moduleStateTypeName = moduleName + "_state_t";
+ std::string moduleStateTypeName =
+ std::string("struct ") + moduleName + "_state_t";
auto stateOp = builder.create<emitc::CastOp>(
/*location=*/loc,
@@ -891,7 +885,7 @@
/*operands=*/
ArrayRef<Value>{allocatorOp, stateOp.getResult()});
- builder.create<mlir::func::ReturnOp>(loc);
+ builder.create<mlir::emitc::ReturnOp>(loc, nullptr);
}
// iree_status_t resolve_import(
@@ -922,12 +916,13 @@
},
{emitc::OpaqueType::get(ctx, "iree_status_t")});
- auto funcOp = builder.create<mlir::func::FuncOp>(
+ auto funcOp = builder.create<mlir::emitc::FuncOp>(
loc, moduleName + "_resolve_import", funcType);
+ funcOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ funcOp.setPrivate();
- moduleAnalysis.addDummy(funcOp);
-
- attachAttribute(funcOp, "emitc.static", UnitAttr::get(ctx));
+ moduleAnalysis.addDummy(funcOp, /*emitAtEnd=*/false);
Block *entryBlock = funcOp.addEntryBlock();
@@ -938,7 +933,8 @@
builder.setInsertionPointToStart(entryBlock);
- std::string moduleStateTypeName = moduleName + "_state_t";
+ std::string moduleStateTypeName =
+ std::string("struct ") + moduleName + "_state_t";
auto stateOp = builder.create<emitc::CastOp>(
/*location=*/loc,
@@ -972,7 +968,7 @@
auto status = emitc_builders::ireeOkStatus(builder, loc);
- builder.create<mlir::func::ReturnOp>(loc, status);
+ builder.create<mlir::emitc::ReturnOp>(loc, status);
}
// iree_status_t create(
@@ -1000,19 +996,14 @@
emitc::OpaqueType::get(ctx, "iree_status_t"),
});
- auto funcOp = builder.create<mlir::func::FuncOp>(
+ auto funcOp = builder.create<mlir::emitc::FuncOp>(
loc, moduleName + "_create", funcType);
-
- moduleAnalysis.addDummy(funcOp);
+ funcOp.setPublic();
// This function needs an iree_vm_native_module_descriptor_t that is emitted
// by the CModuleTarget at the moment. So we add a marker to this function
// and delay the printing of it.
- attachAttribute(funcOp, "vm.emit_at_end", UnitAttr::get(ctx));
-
- // This functions is the only one users need and it is therefore declared
- // separatly from all other functions.
- attachAttribute(funcOp, "vm.module.constructor", UnitAttr::get(ctx));
+ moduleAnalysis.addDummy(funcOp, /*emitAtEnd=*/true);
Block *entryBlock = funcOp.addEntryBlock();
@@ -1022,7 +1013,7 @@
builder.setInsertionPointToStart(entryBlock);
- std::string moduleTypeName = moduleName + "_t";
+ std::string moduleTypeName = std::string("struct ") + moduleName + "_t";
Value module = emitc_builders::allocateVariable(
builder, loc,
@@ -1051,9 +1042,8 @@
/*memberName=*/"allocator",
/*operand=*/module,
/*value=*/allocatorArg);
+
auto &typeTable = moduleAnalysis.typeTable;
- attachAttribute(moduleOp, "vm.num_types",
- builder.getI32IntegerAttr(typeTable.size()));
if (!typeTable.empty()) {
Type typeRefType = emitc::OpaqueType::get(ctx, "iree_vm_ref_type_t");
Type typeRefArrayType = emitc::PointerType::get(typeRefType);
@@ -1063,7 +1053,6 @@
std::string listType = "!vm.list";
for (auto [index, typeDef] : llvm::enumerate(typeTable)) {
std::string typeName = typeDef.full_name;
- moduleAnalysis.mapType(typeDef.type, index);
std::string listPrefix = typeName.substr(0, listType.size());
if (listType == listPrefix) {
typeName = listPrefix;
@@ -1131,13 +1120,13 @@
/*operands=*/
ArrayRef<Value>{allocatorArg, module});
- builder.create<mlir::func::ReturnOp>(loc,
- vmInitializeStatus.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(loc,
+ vmInitializeStatus.getResult(0));
}
builder.setInsertionPointToEnd(condBlock);
- builder.create<IREE::VM::CondBranchOp>(loc, vmInitializeIsOk.getResult(0),
+ builder.create<mlir::cf::CondBranchOp>(loc, vmInitializeIsOk.getResult(0),
continuationBlock, failureBlock);
builder.setInsertionPointToStart(continuationBlock);
@@ -1166,7 +1155,331 @@
/*operands=*/
ArrayRef<Value>{vmModulePtr, instanceArg, allocatorArg, moduleArg});
- builder.create<mlir::func::ReturnOp>(loc, status.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(loc, status.getResult(0));
+ }
+
+ return success();
+}
+
+/// Generate boilerplate code like includes for the IREE C API, include guards,
+/// structures to hold the module state, functions and global variables to
+/// create a module instance etc.
+LogicalResult
+createModuleStructure(IREE::VM::ModuleOp moduleOp,
+ IREE::VM::EmitCTypeConverter &typeConverter) {
+ if (failed(createAPIFunctions(moduleOp, typeConverter.analysis))) {
+ return failure();
+ }
+
+ auto loc = moduleOp.getLoc();
+
+ OpBuilder builder(moduleOp);
+
+ SmallVector<Operation *> opsToRemove;
+ {
+ OpBuilder::InsertionGuard guard(builder);
+ builder.setInsertionPointToStart(&moduleOp.getBlock());
+
+ std::string includeGuard = moduleOp.getName().upper() + "_H_";
+
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::IFNDEF,
+ includeGuard);
+
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::DEFINE,
+ includeGuard);
+
+ builder.create<emitc::IncludeOp>(loc, "iree/vm/api.h");
+
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::IFDEF,
+ "__cplusplus");
+ builder.create<emitc::VerbatimOp>(loc, "extern \"C\" {");
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::ENDIF,
+ "// __cplusplus");
+
+ // Emit declarations for public functions.
+ for (auto funcOp : moduleOp.getOps<mlir::emitc::FuncOp>()) {
+ if (funcOp.isPublic()) {
+ builder.create<emitc::DeclareFuncOp>(loc, funcOp.getName());
+ }
+ }
+
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::IFDEF,
+ "__cplusplus");
+ builder.create<emitc::VerbatimOp>(loc, "} // extern \"C\"");
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::ENDIF,
+ "// __cplusplus");
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::ENDIF,
+ std::string("// ") + includeGuard);
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::IF,
+ "defined(EMITC_IMPLEMENTATION)");
+ builder.create<emitc::IncludeOp>(loc, "iree/vm/ops.h");
+ builder.create<emitc::IncludeOp>(loc, "iree/vm/ops_emitc.h");
+ builder.create<emitc::IncludeOp>(loc, "iree/vm/shims_emitc.h");
+
+ // Rodata ops.
+ for (auto rodataOp : moduleOp.getOps<IREE::VM::RodataOp>()) {
+ auto value = llvm::dyn_cast<IREE::Util::SerializableAttrInterface>(
+ rodataOp.getValue());
+ assert(value && "expected a serializable rodata value");
+ SmallVector<char> byteBuffer;
+ if (failed(value.serializeToVector(
+ rodataOp.getLoc(), llvm::endianness::little, byteBuffer))) {
+ return rodataOp.emitError() << "error during serialization";
+ }
+
+ constexpr size_t kDefaultRodataAlignment = 16;
+ size_t alignment =
+ rodataOp.getAlignment()
+ ? static_cast<size_t>(rodataOp.getAlignment().value())
+ : 0;
+ if (alignment == 0)
+ alignment = kDefaultRodataAlignment;
+
+ std::string bufferName =
+ moduleOp.getName().str() + "_" + rodataOp.getName().str();
+
+ std::string stmt = "iree_alignas(" + std::to_string(alignment) +
+ ") static const uint8_t " + bufferName + "[] = {";
+ size_t index = 0;
+ for (char value : byteBuffer) {
+ if (index++ > 0)
+ stmt += ", ";
+ stmt += std::to_string(
+ static_cast<unsigned int>(static_cast<unsigned char>(value)));
+ }
+ stmt += "};";
+ builder.create<emitc::VerbatimOp>(loc, stmt);
+ opsToRemove.push_back(rodataOp.getOperation());
+ }
+
+ // structs
+ // Returns |count| or 1 if |count| == 0.
+ // Some compilers (MSVC) don't support zero-length struct fields on the
+ // interior of structs (just VLA at the tail).
+ auto countOrEmpty = [](uint32_t count) { return count ? count : 1; };
+
+ const int64_t numTypes = typeConverter.analysis.typeTable.size();
+
+ std::string moduleStructName = moduleOp.getName().str() + "_t";
+ SmallVector<emitc_builders::StructField> moduleStructFields{
+ {"iree_allocator_t", "allocator"},
+ {"iree_vm_ref_type_t", "types", countOrEmpty(numTypes)}};
+
+ emitc_builders::structDefinition(builder, loc, moduleStructName,
+ moduleStructFields);
+
+ auto ordinalCounts = moduleOp.getOrdinalCountsAttr();
+
+ std::string moduleStructStateName = moduleOp.getName().str() + "_state_t";
+ SmallVector<emitc_builders::StructField> moduleStructStateFields{
+ {"iree_allocator_t", "allocator"},
+ {"uint8_t", "rwdata", countOrEmpty(ordinalCounts.getGlobalBytes())},
+ {"iree_vm_ref_t", "refs", countOrEmpty(ordinalCounts.getGlobalRefs())},
+ {"iree_vm_buffer_t", "rodata_buffers",
+ countOrEmpty(ordinalCounts.getRodatas())},
+ {"iree_vm_function_t", "imports",
+ countOrEmpty(ordinalCounts.getImportFuncs())},
+ };
+
+ emitc_builders::structDefinition(builder, loc, moduleStructStateName,
+ moduleStructStateFields);
+
+ // Emit declarations for private functions.
+ for (auto funcOp : moduleOp.getOps<mlir::emitc::FuncOp>()) {
+ if (funcOp.isPrivate()) {
+ builder.create<emitc::DeclareFuncOp>(loc, funcOp.getName());
+ }
+ }
+
+ // TODO(simon-camp): Move these to a structured helper
+ // global descriptors
+ // - define structs for each entity etc.
+ auto printStringView = [](StringRef s) -> std::string {
+ // We can't use iree_make_string_view because function calls are not
+ // allowed for constant expressions in C.
+ // TODO(#7605): Switch to IREE_SVL. We can't use IREE_SVL today because it
+ // uses designated initializers, which cause issues when compiled as C++.
+ return ("{\"" + s + "\", " + std::to_string(s.size()) + "}").str();
+ };
+
+ // dependencies
+ std::string dependenciesName = moduleOp.getName().str() + "_dependencies_";
+ std::string deps;
+ deps += "static const iree_vm_module_dependency_t " + dependenciesName +
+ "[] = {";
+ auto dependencies = moduleOp.getDependencies();
+ if (dependencies.empty()) {
+ // Empty list placeholder.
+ deps += "{{0}},";
+ } else {
+ for (auto &dependency : dependencies) {
+ deps += "{" + printStringView(dependency.name) + ", " +
+ std::to_string(dependency.minimumVersion) + ", " +
+ (dependency.isOptional
+ ? "IREE_VM_MODULE_DEPENDENCY_FLAG_OPTIONAL"
+ : "IREE_VM_MODULE_DEPENDENCY_FLAG_REQUIRED") +
+ "},";
+ }
+ }
+ deps += "};";
+ builder.create<emitc::VerbatimOp>(loc, deps);
+
+ // Imports.
+ SmallVector<IREE::VM::ImportOp> importOps(
+ moduleOp.getOps<IREE::VM::ImportOp>());
+ std::string importName = moduleOp.getName().str() + "_imports_";
+ std::string imports;
+ imports += "static const iree_vm_native_import_descriptor_t " + importName +
+ "[] = {";
+ if (importOps.empty()) {
+ // Empty list placeholder.
+ imports += "{0},";
+ } else {
+ // Sort import ops by ordinal.
+ llvm::sort(importOps, [](auto &lhs, auto &rhs) {
+ return lhs.getOrdinal()->getZExtValue() <
+ rhs.getOrdinal()->getZExtValue();
+ });
+ for (auto importOp : importOps) {
+ imports +=
+ std::string("{") +
+ (importOp.getIsOptional() ? "IREE_VM_NATIVE_IMPORT_OPTIONAL"
+ : "IREE_VM_NATIVE_IMPORT_REQUIRED") +
+ ", " + printStringView(importOp.getName()) + "},";
+ }
+ }
+ imports += "};";
+ builder.create<emitc::VerbatimOp>(loc, imports);
+
+ for (auto op : moduleOp.getOps<IREE::VM::ImportOp>()) {
+ opsToRemove.push_back(op);
+ }
+
+ // Exports.
+ SmallVector<emitc::FuncOp> exportedFunctions;
+ for (auto func : moduleOp.getOps<emitc::FuncOp>()) {
+ if (typeConverter.analysis.lookupFunction(func).isExported()) {
+ exportedFunctions.push_back(func);
+ }
+ }
+ auto extractExportName = [&typeConverter](emitc::FuncOp funcOp) {
+ return typeConverter.analysis.lookupFunction(funcOp).getExportName();
+ };
+ std::string exportName = moduleOp.getName().str() + "_exports_";
+ std::string exports;
+ exports += "static const iree_vm_native_export_descriptor_t " + exportName +
+ "[] = {";
+ if (exportedFunctions.empty()) {
+ // Empty list placeholder.
+ exports += "{{0}},";
+ } else {
+ // Sort export ops.
+ llvm::sort(exportedFunctions, [&extractExportName](auto &lhs, auto &rhs) {
+ return extractExportName(lhs).compare(extractExportName(rhs)) < 0;
+ });
+ for (auto funcOp : exportedFunctions) {
+ StringRef exportName = extractExportName(funcOp);
+ StringRef callingConvention =
+ typeConverter.analysis.lookupFunction(funcOp)
+ .getCallingConvention();
+
+ // TODO(simon-camp): support function-level reflection attributes
+ exports += "{" + printStringView(exportName) + ", " +
+ printStringView(callingConvention) + ", 0, NULL},";
+ }
+ }
+ exports += "};";
+ builder.create<emitc::VerbatimOp>(loc, exports);
+
+ // Functions.
+ std::string functionName = moduleOp.getName().str() + "_funcs_";
+ std::string functions;
+ functions +=
+ "static const iree_vm_native_function_ptr_t " + functionName + "[] = {";
+ if (exportedFunctions.empty()) {
+ // Empty list placeholder.
+ functions += "{0},";
+ } else {
+ // We only add exported functions to the table, as calls to internal
+ // functions are directly mapped to C function calls of the generated
+ // implementation.
+ for (auto funcOp : exportedFunctions) {
+ auto funcName = funcOp.getName();
+ functions += std::string("{") +
+ "(iree_vm_native_function_shim_t)iree_emitc_shim, " +
+ "(iree_vm_native_function_target_t)" + funcName.str() +
+ "},";
+ }
+ }
+ functions += "};";
+ builder.create<emitc::VerbatimOp>(loc, functions);
+
+ // Module descriptor.
+ // TODO(simon-camp): support module-level reflection attributes
+ std::string descriptorName = moduleOp.getName().str() + "_descriptor_";
+ std::string descriptor;
+ descriptor +=
+ "static const iree_vm_native_module_descriptor_t " + descriptorName +
+ " = {"
+ // name:
+ + printStringView(moduleOp.getName()) +
+ ","
+ // version:
+ + std::to_string(moduleOp.getVersion().value_or(0u)) +
+ ","
+ // attrs:
+ + "0," +
+ "NULL,"
+ // dependencies:
+ + std::to_string(dependencies.size()) + "," + dependenciesName +
+ ","
+ // imports:
+ + std::to_string(importOps.size()) + "," + importName +
+ ","
+ // exports:
+ + std::to_string(exportedFunctions.size()) + "," + exportName +
+ ","
+ // functions:
+ + std::to_string(exportedFunctions.size()) + "," + functionName + "," +
+ "};";
+
+ builder.create<emitc::VerbatimOp>(loc, descriptor);
+
+ // Move functions marked as `emitAtEnd` to the end of the module.
+ auto funcs =
+ SmallVector<emitc::FuncOp>(moduleOp.getOps<mlir::emitc::FuncOp>());
+ for (auto func : funcs) {
+ if (typeConverter.analysis.lookupFunction(func).shouldEmitAtEnd()) {
+ func->moveBefore(moduleOp.getBlock().getTerminator());
+ }
+ }
+
+ builder.setInsertionPoint(moduleOp.getBlock().getTerminator());
+ emitc_builders::preprocessorDirective(builder, loc, emitc_builders::ENDIF,
+ " // EMITC_IMPLEMENTATION");
+ }
+
+ for (auto op : opsToRemove) {
+ op->erase();
+ }
+
+ // TODO(simon-camp): The Cpp Emitter expects a builtin.module as the
+ // outer container of the supported operations. Instead of nesting a
+ // builtin.module inside the vm.module as in the current implementation,
+ // the conversion pass should be changed to replace the vm.module
+ // with a builtin.module.
+ builder.setInsertionPointToStart(&moduleOp.getBlock());
+ auto innerModule = builder.create<mlir::ModuleOp>(loc);
+
+ IRRewriter rewriter(moduleOp.getContext());
+
+ for (Operation &op : llvm::make_early_inc_range(moduleOp.getBlock())) {
+ if (isa<IREE::VM::ModuleTerminatorOp>(op) ||
+ &op == innerModule.getOperation()) {
+ continue;
+ }
+ rewriter.moveOpBefore(&op, innerModule.getBody(),
+ innerModule.getBody()->end());
}
return success();
@@ -1216,8 +1529,8 @@
}
};
-// Convert vm operations to emitc calls. The resultiong call has the ops
-// operands as arguments followed by an argument for every attribute.
+// Convert vm operations to emitc opaque_calls. The resultiong opaque_call has
+// the ops operands as arguments followed by an argument for every attribute.
template <typename OpTy>
class GenericOpConversion : public EmitCConversionPattern<OpTy> {
using Adaptor = typename OpTy::Adaptor;
@@ -1244,8 +1557,8 @@
ArrayAttr templateArgs;
// If the operation has attributes, we need to explicitely build the args
- // attribute of the emitc call op. This consists of index attributes for
- // the operands, followed by the source op attributes themselves.
+ // attribute of the emitc opaque_call op. This consists of index attributes
+ // for the operands, followed by the source op attributes themselves.
if (op->getAttrs().size() > 0) {
SmallVector<Attribute> args_ =
indexSequence(adaptor.getOperands().size(), op.getContext());
@@ -1279,12 +1592,12 @@
}
};
-class FuncOpConversion : public EmitCConversionPattern<mlir::func::FuncOp> {
- using Adaptor = mlir::func::FuncOp::Adaptor;
- using EmitCConversionPattern<mlir::func::FuncOp>::EmitCConversionPattern;
+class FuncOpConversion : public EmitCConversionPattern<mlir::emitc::FuncOp> {
+ using Adaptor = mlir::emitc::FuncOp::Adaptor;
+ using EmitCConversionPattern<mlir::emitc::FuncOp>::EmitCConversionPattern;
LogicalResult
- matchAndRewrite(mlir::func::FuncOp funcOp, Adaptor adaptor,
+ matchAndRewrite(mlir::emitc::FuncOp funcOp, Adaptor adaptor,
ConversionPatternRewriter &rewriter) const override {
TypeConverter::SignatureConversion signatureConverter(
funcOp.getFunctionType().getNumInputs());
@@ -1323,12 +1636,10 @@
auto ctx = exportOp.getContext();
auto loc = exportOp.getLoc();
- mlir::func::FuncOp funcOp = lookupSymbolRef<mlir::func::FuncOp>(
+ mlir::emitc::FuncOp funcOp = lookupSymbolRef<mlir::emitc::FuncOp>(
exportOp.getOperation(), "function_ref");
- auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
-
- std::string newFuncName = (funcOp.getName() + "_export_shim").str();
+ std::string newFuncName = funcOp.getName().str() + "_export_shim";
Type stackType =
emitc::PointerType::get(emitc::OpaqueType::get(ctx, "iree_vm_stack_t"));
@@ -1352,16 +1663,12 @@
ctx, {inputTypes}, {emitc::OpaqueType::get(ctx, "iree_status_t")});
auto newFuncOp =
- rewriter.create<mlir::func::FuncOp>(loc, newFuncName, newFuncType);
+ rewriter.create<mlir::emitc::FuncOp>(loc, newFuncName, newFuncType);
+ newFuncOp.setSpecifiersAttr(
+ rewriter.getArrayAttr({rewriter.getStringAttr("static")}));
+ newFuncOp.setPrivate();
- FunctionType functionType = funcAnalysis.getFunctionType();
-
- getModuleAnalysis().add(newFuncOp, functionType);
-
- attachAttribute(newFuncOp, "emitc.static", UnitAttr::get(ctx));
- attachAttribute(newFuncOp, "vm.calling_convention",
- funcOp.getOperation()->getAttr("vm.calling_convention"));
- attachAttribute(newFuncOp, "vm.export_name", exportOp.getExportNameAttr());
+ getModuleAnalysis().addFromExport(newFuncOp, exportOp);
// Populate newly generated function.
{
@@ -1438,7 +1745,7 @@
auto status = emitc_builders::ireeOkStatus(rewriter, loc);
- rewriter.create<mlir::func::ReturnOp>(loc, status);
+ rewriter.create<mlir::emitc::ReturnOp>(loc, status);
}
rewriter.eraseOp(exportOp);
@@ -1449,7 +1756,7 @@
FailureOr<std::pair<Value, Value>>
castModuleAndStateStructs(ConversionPatternRewriter &rewriter,
IREE::VM::ExportOp &exportOp,
- mlir::func::FuncOp &newFuncOp) const {
+ mlir::emitc::FuncOp &newFuncOp) const {
auto ctx = exportOp.getContext();
auto loc = exportOp.getLoc();
@@ -1459,8 +1766,10 @@
auto moduleOp =
newFuncOp.getOperation()->getParentOfType<IREE::VM::ModuleOp>();
- std::string moduleTypeName = (moduleOp.getName() + "_t").str();
- std::string moduleStateTypeName = (moduleOp.getName() + "_state_t").str();
+ std::string moduleTypeName =
+ std::string("struct ") + moduleOp.getName().str() + "_t";
+ std::string moduleStateTypeName =
+ std::string("struct ") + moduleOp.getName().str() + "_state_t";
auto moduleCasted = rewriter.create<emitc::CastOp>(
/*location=*/loc,
@@ -1481,10 +1790,10 @@
FailureOr<std::pair<GeneratedStruct, GeneratedStruct>>
typedefArgumentAndResultStructs(ConversionPatternRewriter &rewriter,
IREE::VM::ExportOp &exportOp,
- mlir::func::FuncOp &newFuncOp) const {
+ mlir::emitc::FuncOp &newFuncOp) const {
auto loc = exportOp.getLoc();
- mlir::func::FuncOp funcOp = lookupSymbolRef<mlir::func::FuncOp>(
+ mlir::emitc::FuncOp funcOp = lookupSymbolRef<mlir::emitc::FuncOp>(
exportOp.getOperation(), "function_ref");
auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
@@ -1540,7 +1849,7 @@
<< "failed to emit C type for struct definition";
}
- std::string structName = (funcOp.getName() + "_args_t").str();
+ std::string structName = funcOp.getName().str() + "_args_t";
argumentStruct.name = structName;
typedefStruct(structName, structBody.value());
}
@@ -1554,7 +1863,7 @@
return failure();
}
- std::string structName = (funcOp.getName() + "_result_t").str();
+ std::string structName = funcOp.getName().str() + "_result_t";
resultStruct.name = structName;
typedefStruct(structName, structBody.value());
}
@@ -1564,7 +1873,7 @@
void castArgumentAndResultStructs(ConversionPatternRewriter &rewriter,
IREE::VM::ExportOp &exportOp,
- mlir::func::FuncOp &newFuncOp,
+ mlir::emitc::FuncOp &newFuncOp,
GeneratedStruct &argumentStruct,
GeneratedStruct &resultStruct) const {
auto ctx = exportOp.getContext();
@@ -1583,7 +1892,8 @@
/*operand=*/newFuncOp.getArgument(SHIM_ARGUMENT_ARGS_STORAGE));
// cast
- std::string argumentsType = argumentStruct.name.value();
+ std::string argumentsType =
+ std::string("struct ") + argumentStruct.name.value();
auto arguments = rewriter.create<emitc::CastOp>(
/*location=*/loc,
/*type=*/
@@ -1605,7 +1915,8 @@
/*operand=*/newFuncOp.getArgument(SHIM_ARGUMENT_RETS_STORAGE));
// cast
- std::string resultType = resultStruct.name.value();
+ std::string resultType =
+ std::string("struct ") + resultStruct.name.value();
auto results = rewriter.create<emitc::CastOp>(
/*location=*/loc,
/*type=*/
@@ -1627,7 +1938,7 @@
return success();
}
- mlir::func::FuncOp funcOp = lookupSymbolRef<mlir::func::FuncOp>(
+ mlir::emitc::FuncOp funcOp = lookupSymbolRef<mlir::emitc::FuncOp>(
exportOp.getOperation(), "function_ref");
auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
@@ -1687,7 +1998,7 @@
const auto typeConverter = getTypeConverter<IREE::VM::EmitCTypeConverter>();
- mlir::func::FuncOp funcOp = lookupSymbolRef<mlir::func::FuncOp>(
+ mlir::emitc::FuncOp funcOp = lookupSymbolRef<mlir::emitc::FuncOp>(
exportOp.getOperation(), "function_ref");
auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
@@ -1799,8 +2110,8 @@
Region *parentRegion = condBlock->getParent();
failureBlock = builder.createBlock(parentRegion, parentRegion->end());
- mlir::func::FuncOp funcOp =
- cast<mlir::func::FuncOp>(failureBlock->getParentOp());
+ mlir::emitc::FuncOp funcOp =
+ cast<mlir::emitc::FuncOp>(failureBlock->getParentOp());
releaseRefs(builder, location, funcOp, typeConverter.analysis);
auto statusOp = builder.create<emitc::CallOpaqueOp>(
@@ -1812,7 +2123,7 @@
ctx, {emitc::OpaqueAttr::get(ctx, "IREE_STATUS_NOT_FOUND")}),
/*templateArgs=*/ArrayAttr{},
/*operands=*/ArrayRef<Value>{});
- builder.create<mlir::func::ReturnOp>(location, statusOp.getResult(0));
+ builder.create<mlir::emitc::ReturnOp>(location, statusOp.getResult(0));
}
builder.setInsertionPointToEnd(condBlock);
@@ -1825,7 +2136,6 @@
LogicalResult createImportShim(IREE::VM::ImportOp &importOp,
DenseIntElementsAttr segmentSizes,
OpBuilder &builder) const {
- auto ctx = importOp.getContext();
auto loc = importOp.getLoc();
auto moduleOp =
@@ -1847,12 +2157,13 @@
<< "Failed to build function type for wrapper";
}
- auto newFuncOp = builder.create<mlir::func::FuncOp>(
+ auto newFuncOp = builder.create<mlir::emitc::FuncOp>(
loc, newFuncName.value(), newFuncType.value());
+ newFuncOp.setSpecifiersAttr(
+ builder.getArrayAttr({builder.getStringAttr("static")}));
+ newFuncOp.setPrivate();
- typeConverter.analysis.add(newFuncOp, importOp.getFunctionType());
-
- attachAttribute(newFuncOp, "emitc.static", UnitAttr::get(ctx));
+ typeConverter.analysis.addFromImport(newFuncOp, importOp);
// Populate newly generated function.
{
@@ -1906,7 +2217,7 @@
auto status = emitc_builders::ireeOkStatus(builder, loc);
- builder.create<mlir::func::ReturnOp>(loc, status);
+ builder.create<mlir::emitc::ReturnOp>(loc, status);
}
return success();
@@ -2060,7 +2371,7 @@
}
LogicalResult packArgumentBuffer(ArrayRef<Type> inputTypes,
- mlir::func::FuncOp &funcOp, Value call,
+ mlir::emitc::FuncOp &funcOp, Value call,
OpBuilder &builder, Location loc) const {
if (inputTypes.empty()) {
return success();
@@ -2130,7 +2441,7 @@
}
LogicalResult unpackResultBuffer(ArrayRef<Type> resultTypes,
- mlir::func::FuncOp &funcOp, Value call,
+ mlir::emitc::FuncOp &funcOp, Value call,
OpBuilder &builder, Location loc) const {
if (resultTypes.empty()) {
return success();
@@ -2301,8 +2612,8 @@
LogicalResult
matchAndRewrite(OpTy op, Adaptor adaptor,
ConversionPatternRewriter &rewriter) const override {
- mlir::func::FuncOp funcOp =
- lookupSymbolRef<mlir::func::FuncOp>(op.getOperation(), "callee");
+ mlir::emitc::FuncOp funcOp =
+ lookupSymbolRef<mlir::emitc::FuncOp>(op.getOperation(), "callee");
IREE::VM::ImportOp importOp =
lookupSymbolRef<IREE::VM::ImportOp>(op.getOperation(), "callee");
@@ -2322,13 +2633,13 @@
LogicalResult rewriteInternalCall(Operation *op, Adaptor adaptor,
ConversionPatternRewriter &rewriter,
- mlir::func::FuncOp funcOp) const {
+ mlir::emitc::FuncOp funcOp) const {
auto loc = op->getLoc();
SmallVector<Value> updatedOperands;
SmallVector<Value> resultOperands;
- auto parentFuncOp = op->getParentOfType<mlir::func::FuncOp>();
+ auto parentFuncOp = op->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stackArg =
parentFuncOp.getArgument(CCONV_ARGUMENT_STACK);
@@ -2369,7 +2680,7 @@
int importOrdinal = importOp.getOrdinal()->getZExtValue();
- auto funcOp = op->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stackArg = funcOp.getArgument(CCONV_ARGUMENT_STACK);
const BlockArgument stateArg =
@@ -2409,7 +2720,7 @@
if (!funcName.has_value())
return op->emitError() << "Couldn't build name to imported function";
- auto callee = moduleOp.lookupSymbol<mlir::func::FuncOp>(funcName.value());
+ auto callee = moduleOp.lookupSymbol<mlir::emitc::FuncOp>(funcName.value());
if (callee == nullptr) {
return op->emitError()
<< "Couldn't find function with name `" << funcName.value() << "`";
@@ -2551,7 +2862,7 @@
auto loc = cmpOp.getLoc();
auto funcOp =
- cmpOp.getOperation()->template getParentOfType<mlir::func::FuncOp>();
+ cmpOp.getOperation()->template getParentOfType<mlir::emitc::FuncOp>();
const auto typeConverter =
this->template getTypeConverter<IREE::VM::EmitCTypeConverter>();
@@ -2604,7 +2915,7 @@
auto ctx = cmpOp.getContext();
auto loc = cmpOp.getLoc();
- auto funcOp = cmpOp.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = cmpOp.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
@@ -2699,7 +3010,7 @@
}
auto funcOp =
- constRefRodataOp.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ constRefRodataOp.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stateArg =
funcOp.getArgument(CCONV_ARGUMENT_MODULE_STATE);
@@ -2954,7 +3265,7 @@
auto ctx = op.getContext();
auto loc = op.getLoc();
- auto funcOp = op.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
// The result variables are the last N arguments of the function.
unsigned int firstOutputArgumentIndex =
@@ -2994,7 +3305,7 @@
auto status = emitc_builders::ireeOkStatus(rewriter, loc);
- rewriter.replaceOpWithNewOp<mlir::func::ReturnOp>(op, status);
+ rewriter.replaceOpWithNewOp<mlir::emitc::ReturnOp>(op, status);
return success();
}
@@ -3017,7 +3328,7 @@
lookupSymbolRef<IREE::VM::ImportOp>(op.getOperation(), "import");
int importOrdinal = importOp.getOrdinal()->getZExtValue();
- auto funcOp = op->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stateArg =
funcOp.getArgument(CCONV_ARGUMENT_MODULE_STATE);
@@ -3085,20 +3396,20 @@
passthroughBlock =
rewriter.createBlock(parentRegion, parentRegion->end());
- auto funcOp = op.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
releaseRefs(rewriter, loc, funcOp, getModuleAnalysis());
auto status = emitc_builders::ireeOkStatus(rewriter, loc);
- rewriter.create<mlir::func::ReturnOp>(loc, status);
+ rewriter.create<mlir::emitc::ReturnOp>(loc, status);
}
Block *failureBlock;
{
OpBuilder::InsertionGuard guard(rewriter);
failureBlock = rewriter.createBlock(parentRegion, parentRegion->end());
- auto funcOp = op.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
releaseRefs(rewriter, loc, funcOp, getModuleAnalysis());
@@ -3143,7 +3454,7 @@
/*operands=*/
ArrayRef<Value>{messageSizeIntOp.getResult(), messageDataOp});
- rewriter.create<mlir::func::ReturnOp>(loc, status.getResult(0));
+ rewriter.create<mlir::emitc::ReturnOp>(loc, status.getResult(0));
}
Type boolType = rewriter.getIntegerType(1);
@@ -3184,7 +3495,7 @@
}
auto funcOp =
- loadOp.getOperation()->template getParentOfType<mlir::func::FuncOp>();
+ loadOp.getOperation()->template getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stateArg =
funcOp.getArgument(CCONV_ARGUMENT_MODULE_STATE);
@@ -3245,7 +3556,7 @@
auto globalOrdinal = globalOp.getOrdinal()->getZExtValue();
- auto funcOp = op->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = op->getParentOfType<mlir::emitc::FuncOp>();
auto &funcAnalysis = this->getModuleAnalysis().lookupFunction(funcOp);
@@ -3271,7 +3582,7 @@
/*operand=*/refs);
auto moduleOp = op->getParentOfType<IREE::VM::ModuleOp>();
- auto parentFuncOp = op->getParentOfType<mlir::func::FuncOp>();
+ auto parentFuncOp = op->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument moduleArg =
parentFuncOp.getArgument(CCONV_ARGUMENT_MODULE);
Type elementType = localValue.getType();
@@ -3345,7 +3656,7 @@
}
auto funcOp =
- storeOp.getOperation()->template getParentOfType<mlir::func::FuncOp>();
+ storeOp.getOperation()->template getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stateArg =
funcOp.getArgument(CCONV_ARGUMENT_MODULE_STATE);
@@ -3376,9 +3687,9 @@
StringRef funcName;
};
-// Convert vm operations with wrapped containers to multiple emitc calls. The
-// wrapping ref pointers are first dereferenced and the results are used as the
-// arguments of the specified function name.
+// Convert vm operations with wrapped containers to multiple emitc opaque_calls.
+// The wrapping ref pointers are first dereferenced and the results are used as
+// the arguments of the specified function name.
template <typename OpTy>
class ContainerOpConversion : public EmitCConversionPattern<OpTy> {
using Adaptor = typename OpTy::Adaptor;
@@ -3559,7 +3870,7 @@
Value containerPtr = emitc_builders::addressOf(rewriter, loc, container);
auto funcOp =
- op.getOperation()->template getParentOfType<mlir::func::FuncOp>();
+ op.getOperation()->template getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument stateArg =
funcOp.getArgument(CCONV_ARGUMENT_MODULE_STATE);
@@ -3642,7 +3953,7 @@
auto moduleOp = op.getOperation()->getParentOfType<IREE::VM::ModuleOp>();
auto parentFuncOp =
- op.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ op.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument moduleArg =
parentFuncOp.getArgument(CCONV_ARGUMENT_MODULE);
auto elementTypePtr =
@@ -3858,7 +4169,7 @@
auto moduleOp = getOp.getOperation()->getParentOfType<IREE::VM::ModuleOp>();
auto parentFuncOp =
- getOp.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ getOp.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
const BlockArgument moduleArg =
parentFuncOp.getArgument(CCONV_ARGUMENT_MODULE);
@@ -4052,7 +4363,7 @@
/*args=*/ArrayAttr{},
/*operands=*/ArrayRef<Value>{refValue}, getModuleAnalysis());
- auto funcOp = setOp.getOperation()->getParentOfType<mlir::func::FuncOp>();
+ auto funcOp = setOp.getOperation()->getParentOfType<mlir::emitc::FuncOp>();
auto &funcAnalysis = getModuleAnalysis().lookupFunction(funcOp);
@@ -4333,7 +4644,7 @@
namespace {
// A pass converting IREE VM operations into the EmitC dialect.
-// vm.func ops get converted to std.func with the calling convention used by
+// vm.func ops get converted to emitc.func with the calling convention used by
// EmitC. Each function gets three additional arguments a `iree_vm_stack_t*` as
// well as two module specific struct pointers (`{module_name}_t*` and
// `{module_name}_state_t`). These are followed by the original function
@@ -4362,8 +4673,8 @@
MLIR_DEFINE_EXPLICIT_INTERNAL_INLINE_TYPE_ID(ConvertVMToEmitCPass)
void getDependentDialects(DialectRegistry ®istry) const override {
- registry.insert<mlir::emitc::EmitCDialect, mlir::BuiltinDialect,
- mlir::func::FuncDialect, IREE::Util::UtilDialect>();
+ registry.insert<mlir::BuiltinDialect, mlir::cf::ControlFlowDialect,
+ mlir::emitc::EmitCDialect, IREE::Util::UtilDialect>();
}
StringRef getArgument() const override { return "iree-convert-vm-to-emitc"; }
@@ -4378,9 +4689,9 @@
ConversionTarget target(getContext());
EmitCTypeConverter typeConverter(module);
- // Convert vm.func ops to std.func with the calling convention used by
+ // Convert vm.func ops to emitc.func with the calling convention used by
// EmitC. We convert these upfront to make sure vm.call ops always
- // reference std.func ops with the correct calling convention during the
+ // reference emitc.func ops with the correct calling convention during the
// conversion.
SmallVector<IREE::VM::FuncOp> funcsToRemove;
SmallVector<BlockArgument> blockArgsToRemove;
@@ -4395,11 +4706,6 @@
funcOp.erase();
}
- // Generate func ops that implement the C API.
- if (failed(createAPIFunctions(module, typeConverter.analysis))) {
- return signalPassFailure();
- }
-
SmallVector<std::string> importShims;
// The conversion of `call/call.variadic` ops on imported functions expects
@@ -4416,11 +4722,10 @@
populateVMToEmitCPatterns(target, typeConverter, patterns);
target.addLegalDialect<emitc::EmitCDialect, mlir::BuiltinDialect,
- mlir::cf::ControlFlowDialect,
- mlir::func::FuncDialect>();
+ mlir::cf::ControlFlowDialect>();
- target.addDynamicallyLegalOp<mlir::func::FuncOp>(
- [&](mlir::func::FuncOp op) {
+ target.addDynamicallyLegalOp<mlir::emitc::FuncOp>(
+ [&](mlir::emitc::FuncOp op) {
return typeConverter.isSignatureLegal(op.getFunctionType());
});
@@ -4455,6 +4760,10 @@
return;
}
});
+
+ if (failed(createModuleStructure(module, typeConverter))) {
+ return signalPassFailure();
+ }
}
};
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.cpp b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.cpp
index f8b770d..84babe5 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.cpp
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.cpp
@@ -7,11 +7,42 @@
#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h"
#include "llvm/ADT/ArrayRef.h"
-#include "mlir/Dialect/EmitC/IR/EmitC.h"
namespace mlir::iree_compiler::emitc_builders {
namespace {
+std::string mapPreprocessorDirective(PreprocessorDirective directive) {
+ switch (directive) {
+ case PreprocessorDirective::DEFINE:
+ return "#define";
+ case PreprocessorDirective::UNDEF:
+ return "#undef";
+ case PreprocessorDirective::IFDEF:
+ return "#ifdef";
+ case PreprocessorDirective::IFNDEF:
+ return "#ifndef";
+ case PreprocessorDirective::IF:
+ return "#if";
+ case PreprocessorDirective::ENDIF:
+ return "#endif";
+ case PreprocessorDirective::ELSE:
+ return "#else";
+ case PreprocessorDirective::ELIF:
+ return "#elif";
+ case PreprocessorDirective::LINE:
+ return "#line";
+ case PreprocessorDirective::ERROR:
+ return "#error";
+ case PreprocessorDirective::INCLUDE:
+ return "#include";
+ case PreprocessorDirective::PRAGMA:
+ return "#pragma";
+ default:
+ llvm_unreachable("unsupported unary operator");
+ return "XXX";
+ }
+}
+
std::string mapUnaryOperator(UnaryOperator op) {
switch (op) {
case UnaryOperator::PLUS:
@@ -272,20 +303,17 @@
void structDefinition(OpBuilder builder, Location location,
StringRef structName, ArrayRef<StructField> fields) {
- std::string structBody;
-
- for (auto &field : fields) {
- structBody += field.type + " " + field.name + ";";
- }
-
auto ctx = builder.getContext();
+ std::string decl = std::string("struct ") + structName.str() + " {";
+ for (auto &field : fields) {
+ decl += field.type + " " + field.name;
+ if (field.isArray())
+ decl += "[" + std::to_string(field.arraySize.value()) + "]";
+ decl += ";";
+ }
+ decl += "};";
- builder.create<emitc::CallOpaqueOp>(
- /*location=*/location, /*type=*/TypeRange{},
- /*callee=*/StringAttr::get(ctx, "EMITC_TYPEDEF_STRUCT"), /*args=*/
- ArrayAttr::get(ctx, {emitc::OpaqueAttr::get(ctx, structName),
- emitc::OpaqueAttr::get(ctx, structBody)}),
- /*templateArgs=*/ArrayAttr{}, /*operands=*/ArrayRef<Value>{});
+ builder.create<emitc::VerbatimOp>(location, StringAttr::get(ctx, decl));
}
Value structMember(OpBuilder builder, Location location, Type type,
@@ -419,4 +447,17 @@
/*operands=*/ArrayRef<Value>{operand});
}
+emitc::VerbatimOp preprocessorDirective(OpBuilder builder, Location location,
+ PreprocessorDirective directive,
+ StringRef value) {
+
+ auto t = mapPreprocessorDirective(directive);
+ if (!value.empty()) {
+ t += " ";
+ t += value;
+ }
+
+ return builder.create<emitc::VerbatimOp>(location, t);
+}
+
} // namespace mlir::iree_compiler::emitc_builders
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h
index cc8ec72..ab107bc 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCBuilders.h
@@ -11,17 +11,38 @@
#include "llvm/ADT/ArrayRef.h"
#include "llvm/ADT/StringRef.h"
+#include "mlir/Dialect/EmitC/IR/EmitC.h"
#include "mlir/IR/Builders.h"
#include "mlir/IR/BuiltinAttributes.h"
#include "mlir/IR/Location.h"
#include "mlir/IR/Types.h"
#include "mlir/IR/Value.h"
+#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCTypeConverter.h"
+
namespace mlir::iree_compiler::emitc_builders {
struct StructField {
std::string type;
std::string name;
+ std::optional<size_t> arraySize = std::nullopt;
+
+ bool isArray() const { return arraySize.has_value(); }
+};
+
+enum PreprocessorDirective {
+ DEFINE = 0,
+ UNDEF,
+ IFDEF,
+ IFNDEF,
+ IF,
+ ENDIF,
+ ELSE,
+ ELIF,
+ LINE,
+ ERROR,
+ INCLUDE,
+ PRAGMA
};
enum UnaryOperator {
@@ -123,6 +144,10 @@
void ireeVmRefRelease(OpBuilder builder, Location location, Value operand);
+emitc::VerbatimOp preprocessorDirective(OpBuilder builder, Location location,
+ PreprocessorDirective directive,
+ StringRef value);
+
} // namespace mlir::iree_compiler::emitc_builders
#endif // IREE_COMPILER_DIALECT_VM_CONVERSION_VMTOEMITC_EMITCBUILDERS_H_
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCTypeConverter.cpp b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCTypeConverter.cpp
index ca29cb1..b2ae62b 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCTypeConverter.cpp
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/EmitCTypeConverter.cpp
@@ -108,7 +108,6 @@
}
return emitc::OpaqueType::get(type.getContext(), typeLiteral);
}
-
return {};
}
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/VMAnalysis.h b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/VMAnalysis.h
index 0230ab8..e905920 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/VMAnalysis.h
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/VMAnalysis.h
@@ -7,26 +7,48 @@
#ifndef IREE_COMPILER_DIALECT_VM_CONVERSION_VMTOEMITC_VMANALYSIS_H_
#define IREE_COMPILER_DIALECT_VM_CONVERSION_VMTOEMITC_VMANALYSIS_H_
+#include <optional>
+
#include "iree/compiler/Dialect/VM/Analysis/RegisterAllocation.h"
#include "iree/compiler/Dialect/VM/Analysis/ValueLiveness.h"
#include "iree/compiler/Dialect/VM/IR/VMTypes.h"
+#include "iree/compiler/Dialect/VM/Utils/CallingConvention.h"
#include "iree/compiler/Dialect/VM/Utils/TypeTable.h"
#include "mlir/Dialect/EmitC/IR/EmitC.h"
-#include "mlir/Dialect/Func/IR/FuncOps.h"
namespace mlir::iree_compiler::IREE::VM {
+/// TODO(simon-camp): This struct grew from being a wrapper around the
+/// RegisterAllocation and ValueLiveness analyses to also cache other things
+/// needed throughout the conversion. This led to hard to locate failures
+/// when only part of this struct was correctly initialized. This
+/// should be split into multiple structs each with a single responsibility.
struct FuncAnalysis {
-public:
FuncAnalysis() = default;
+ FuncAnalysis(bool emitAtEnd) : emitAtEnd(emitAtEnd) {}
FuncAnalysis(IREE::VM::FuncOp funcOp) {
Operation *op = funcOp.getOperation();
registerAllocation = RegisterAllocation(op);
valueLiveness = ValueLiveness(op);
originalFunctionType = funcOp.getFunctionType();
+ callingConvention = makeCallingConventionString(funcOp).value();
+ refs = DenseMap<int64_t, Operation *>{};
}
- FuncAnalysis(FunctionType functionType) {
+ FuncAnalysis(mlir::emitc::FuncOp funcOp) {
+ originalFunctionType = funcOp.getFunctionType();
+ }
+ FuncAnalysis(IREE::VM::ImportOp importOp) {
+ originalFunctionType = importOp.getFunctionType();
+ callingConvention = makeImportCallingConventionString(importOp).value();
+ }
+ FuncAnalysis(FunctionType functionType, StringRef cconv) {
originalFunctionType = functionType;
+ callingConvention = cconv.str();
+ }
+ FuncAnalysis(FuncAnalysis &analysis, StringRef exportName_) {
+ originalFunctionType = analysis.getFunctionType();
+ callingConvention = analysis.getCallingConvention().str();
+ exportName = exportName_.str();
}
FuncAnalysis(FuncAnalysis &&) = default;
@@ -34,57 +56,94 @@
FuncAnalysis(const FuncAnalysis &) = delete;
FuncAnalysis &operator=(const FuncAnalysis &) = delete;
- FunctionType getFunctionType() { return originalFunctionType; }
+ StringRef getCallingConvention() {
+ assert(callingConvention.has_value());
+ return callingConvention.value();
+ }
+
+ StringRef getExportName() {
+ assert(exportName.has_value());
+ return exportName.value();
+ }
+
+ bool isExported() { return exportName.has_value(); }
+
+ bool shouldEmitAtEnd() { return emitAtEnd.value_or(false); }
+
+ FunctionType getFunctionType() {
+ assert(originalFunctionType.has_value());
+ return originalFunctionType.value();
+ }
int getNumRefRegisters() {
- return registerAllocation.getMaxRefRegisterOrdinal() + 1;
+ assert(registerAllocation.has_value());
+ return registerAllocation.value().getMaxRefRegisterOrdinal() + 1;
}
int getNumRefArguments() {
- assert(originalFunctionType);
- return llvm::count_if(originalFunctionType.getInputs(), [](Type inputType) {
- return isa<IREE::VM::RefType>(inputType);
- });
+ assert(originalFunctionType.has_value());
+ return llvm::count_if(
+ originalFunctionType.value().getInputs(),
+ [](Type inputType) { return isa<IREE::VM::RefType>(inputType); });
}
int getNumLocalRefs() { return getNumRefRegisters() - getNumRefArguments(); }
uint16_t getRefRegisterOrdinal(TypedValue<IREE::VM::RefType> ref) {
- return registerAllocation.mapToRegister(ref).ordinal();
+ assert(registerAllocation.has_value());
+ return registerAllocation.value().mapToRegister(ref).ordinal();
}
bool isMove(Value ref, Operation *op) {
assert(isa<IREE::VM::RefType>(ref.getType()));
- bool lastUse = valueLiveness.isLastValueUse(ref, op);
+ assert(valueLiveness.has_value());
+ bool lastUse = valueLiveness.value().isLastValueUse(ref, op);
return lastUse && false;
}
void cacheLocalRef(int64_t ordinal, emitc::ApplyOp applyOp) {
- assert(!refs.count(ordinal) && "ref was already cached");
- refs[ordinal] = applyOp.getOperation();
+ assert(refs.has_value());
+ assert(!refs.value().count(ordinal) && "ref was already cached");
+ refs.value()[ordinal] = applyOp.getOperation();
}
emitc::ApplyOp lookupLocalRef(int64_t ordinal) {
- assert(refs.count(ordinal) && "ref not found in cache");
- Operation *op = refs[ordinal];
+ assert(refs.has_value());
+ assert(refs.value().count(ordinal) && "ref not found in cache");
+ Operation *op = refs.value()[ordinal];
return cast<emitc::ApplyOp>(op);
}
- DenseMap<int64_t, Operation *> &localRefs() { return refs; }
+ bool hasLocalRefs() { return refs.has_value(); }
+
+ DenseMap<int64_t, Operation *> &localRefs() {
+ assert(refs.has_value());
+ return refs.value();
+ }
private:
- RegisterAllocation registerAllocation;
- ValueLiveness valueLiveness;
- DenseMap<int64_t, Operation *> refs;
- FunctionType originalFunctionType;
+ std::optional<RegisterAllocation> registerAllocation;
+ std::optional<ValueLiveness> valueLiveness;
+ std::optional<DenseMap<int64_t, Operation *>> refs;
+ std::optional<FunctionType> originalFunctionType;
+ std::optional<std::string> callingConvention;
+ std::optional<std::string> exportName;
+ std::optional<bool> emitAtEnd;
};
struct ModuleAnalysis {
ModuleAnalysis(IREE::VM::ModuleOp module) {
typeTable = buildTypeTable(module);
+ for (auto [index, typeDef] : llvm::enumerate(typeTable)) {
+ mapType(typeDef.type, index);
+ }
+
for (auto func : module.getOps<IREE::VM::FuncOp>()) {
functions[func.getOperation()] = FuncAnalysis(func);
}
+ for (auto func : module.getOps<mlir::emitc::FuncOp>()) {
+ functions[func.getOperation()] = FuncAnalysis(func);
+ }
}
ModuleAnalysis(ModuleAnalysis &&) = default;
@@ -92,15 +151,25 @@
ModuleAnalysis(const ModuleAnalysis &) = delete;
ModuleAnalysis &operator=(const ModuleAnalysis &) = delete;
- void addDummy(mlir::func::FuncOp func) {
- functions[func.getOperation()] = FuncAnalysis();
+ void addDummy(mlir::emitc::FuncOp func, bool emitAtEnd) {
+ functions[func.getOperation()] = FuncAnalysis(emitAtEnd);
}
- void add(mlir::func::FuncOp func, FunctionType type) {
- functions[func.getOperation()] = FuncAnalysis(type);
+ void addFromExport(mlir::emitc::FuncOp func, IREE::VM::ExportOp exportOp) {
+ mlir::emitc::FuncOp funcOp =
+ exportOp->getParentOfType<IREE::VM::ModuleOp>()
+ .lookupSymbol<mlir::emitc::FuncOp>(exportOp.getFunctionRefAttr());
+
+ auto &funcAnalysis = lookupFunction(funcOp);
+ functions[func.getOperation()] =
+ FuncAnalysis(funcAnalysis, exportOp.getExportName());
}
- void move(mlir::func::FuncOp newFunc, IREE::VM::FuncOp oldFunc) {
+ void addFromImport(mlir::emitc::FuncOp func, IREE::VM::ImportOp import) {
+ functions[func.getOperation()] = FuncAnalysis(import);
+ }
+
+ void move(mlir::emitc::FuncOp newFunc, IREE::VM::FuncOp oldFunc) {
auto &analysis = lookupFunction(oldFunc.getOperation());
functions[newFunc.getOperation()] = std::move(analysis);
@@ -125,12 +194,12 @@
Value lookupRef(Value ref) {
auto refValue = cast<TypedValue<IREE::VM::RefType>>(ref);
- mlir::func::FuncOp funcOp;
+ mlir::emitc::FuncOp funcOp;
if (auto definingOp = ref.getDefiningOp()) {
- funcOp = definingOp->getParentOfType<mlir::func::FuncOp>();
+ funcOp = definingOp->getParentOfType<mlir::emitc::FuncOp>();
} else {
Operation *op = llvm::cast<BlockArgument>(ref).getOwner()->getParentOp();
- funcOp = cast<mlir::func::FuncOp>(op);
+ funcOp = cast<mlir::emitc::FuncOp>(op);
}
auto &analysis = lookupFunction(funcOp);
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops.mlir
index 48dd304..3f0bdda 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_add_i32
+// CHECK-LABEL: emitc.func private @my_module_add_i32
vm.module @my_module {
vm.func @add_i32(%arg0: i32, %arg1: i32) {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_add_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_sub_i32
+// CHECK-LABEL: emitc.func private @my_module_sub_i32
vm.module @my_module {
vm.func @sub_i32(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_sub_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -22,7 +22,7 @@
// -----
-// CHECK-LABEL: @my_module_mul_i32
+// CHECK-LABEL: emitc.func private @my_module_mul_i32
vm.module @my_module {
vm.func @mul_i32(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_mul_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -33,7 +33,7 @@
// -----
-// CHECK-LABEL: @my_module_div_i32_s
+// CHECK-LABEL: emitc.func private @my_module_div_i32_s
vm.module @my_module {
vm.func @div_i32_s(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_div_i32s"(%arg3, %arg4) : (i32, i32) -> i32
@@ -44,7 +44,7 @@
// -----
-// CHECK-LABEL: @my_module_div_i32_u
+// CHECK-LABEL: emitc.func private @my_module_div_i32_u
vm.module @my_module {
vm.func @div_i32_u(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_div_i32u"(%arg3, %arg4) : (i32, i32) -> i32
@@ -55,7 +55,7 @@
// -----
-// CHECK-LABEL: @my_module_rem_i32_s
+// CHECK-LABEL: emitc.func private @my_module_rem_i32_s
vm.module @my_module {
vm.func @rem_i32_s(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_rem_i32s"(%arg3, %arg4) : (i32, i32) -> i32
@@ -66,7 +66,7 @@
// -----
-// CHECK-LABEL: @my_module_rem_i32_u
+// CHECK-LABEL: emitc.func private @my_module_rem_i32_u
vm.module @my_module {
vm.func @rem_i32_u(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_rem_i32u"(%arg3, %arg4) : (i32, i32) -> i32
@@ -77,7 +77,7 @@
// -----
-// CHECK-LABEL: @my_module_fma_i32
+// CHECK-LABEL: emitc.func private @my_module_fma_i32
vm.module @my_module {
vm.func @fma_i32(%arg0: i32, %arg1: i32, %arg2: i32) {
// CHECK: %0 = emitc.call_opaque "vm_fma_i32"(%arg3, %arg4, %arg5) : (i32, i32, i32) -> i32
@@ -88,7 +88,7 @@
// -----
-// CHECK-LABEL: @my_module_abs_i32
+// CHECK-LABEL: emitc.func private @my_module_abs_i32
vm.module @my_module {
vm.func @abs_i32(%arg0 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_abs_i32"(%arg3) : (i32) -> i32
@@ -99,7 +99,7 @@
// -----
-// CHECK-LABEL: @my_module_min_i32_s
+// CHECK-LABEL: emitc.func private @my_module_min_i32_s
vm.module @my_module {
vm.func @min_i32_s(%arg0: i32, %arg1: i32) {
// CHECK: %0 = emitc.call_opaque "vm_min_i32s"(%arg3, %arg4) : (i32, i32) -> i32
@@ -110,7 +110,7 @@
// -----
-// CHECK-LABEL: @my_module_not_i32
+// CHECK-LABEL: emitc.func private @my_module_not_i32
vm.module @my_module {
vm.func @not_i32(%arg0 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_not_i32"(%arg3) : (i32) -> i32
@@ -121,7 +121,7 @@
// -----
-// CHECK-LABEL: @my_module_and_i32
+// CHECK-LABEL: emitc.func private @my_module_and_i32
vm.module @my_module {
vm.func @and_i32(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_and_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -132,7 +132,7 @@
// -----
-// CHECK-LABEL: @my_module_or_i32
+// CHECK-LABEL: emitc.func private @my_module_or_i32
vm.module @my_module {
vm.func @or_i32(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_or_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -143,7 +143,7 @@
// -----
-// CHECK-LABEL: @my_module_xor_i32
+// CHECK-LABEL: emitc.func private @my_module_xor_i32
vm.module @my_module {
vm.func @xor_i32(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_xor_i32"(%arg3, %arg4) : (i32, i32) -> i32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_f32.mlir
index 9518a2b..ce95840 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_f32.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_add_f32
+// CHECK-LABEL: emitc.func private @my_module_add_f32
vm.module @my_module {
vm.func @add_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_add_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_sub_f32
+// CHECK-LABEL: emitc.func private @my_module_sub_f32
vm.module @my_module {
vm.func @sub_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_sub_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -22,7 +22,7 @@
// -----
-// CHECK-LABEL: @my_module_mul_f32
+// CHECK-LABEL: emitc.func private @my_module_mul_f32
vm.module @my_module {
vm.func @mul_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_mul_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -33,7 +33,7 @@
// -----
-// CHECK-LABEL: @my_module_div_f32
+// CHECK-LABEL: emitc.func private @my_module_div_f32
vm.module @my_module {
vm.func @div_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_div_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -44,7 +44,7 @@
// -----
-// CHECK-LABEL: @my_module_rem_f32
+// CHECK-LABEL: emitc.func private @my_module_rem_f32
vm.module @my_module {
vm.func @rem_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_rem_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -55,7 +55,7 @@
// -----
-// CHECK-LABEL: @my_module_fma_f32
+// CHECK-LABEL: emitc.func private @my_module_fma_f32
vm.module @my_module {
vm.func @fma_f32(%arg0: f32, %arg1: f32, %arg2: f32) {
// CHECK: %0 = emitc.call_opaque "vm_fma_f32"(%arg3, %arg4, %arg5) : (f32, f32, f32) -> f32
@@ -66,7 +66,7 @@
// -----
-// CHECK-LABEL: @my_module_abs_f32
+// CHECK-LABEL: emitc.func private @my_module_abs_f32
vm.module @my_module {
vm.func @abs_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_abs_f32"(%arg3) : (f32) -> f32
@@ -77,7 +77,7 @@
// -----
-// CHECK-LABEL: @my_module_neg_f32
+// CHECK-LABEL: emitc.func private @my_module_neg_f32
vm.module @my_module {
vm.func @neg_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_neg_f32"(%arg3) : (f32) -> f32
@@ -88,7 +88,7 @@
// -----
-// CHECK-LABEL: @my_module_ceil_f32
+// CHECK-LABEL: emitc.func private @my_module_ceil_f32
vm.module @my_module {
vm.func @ceil_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_ceil_f32"(%arg3) : (f32) -> f32
@@ -99,7 +99,7 @@
// -----
-// CHECK-LABEL: @my_module_floor_f32
+// CHECK-LABEL: emitc.func private @my_module_floor_f32
vm.module @my_module {
vm.func @floor_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_floor_f32"(%arg3) : (f32) -> f32
@@ -110,7 +110,7 @@
// -----
-// CHECK-LABEL: @my_module_min_f32
+// CHECK-LABEL: emitc.func private @my_module_min_f32
vm.module @my_module {
vm.func @min_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_min_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -121,7 +121,7 @@
// -----
-// CHECK-LABEL: @my_module_max_f32
+// CHECK-LABEL: emitc.func private @my_module_max_f32
vm.module @my_module {
vm.func @max_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_max_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -132,7 +132,7 @@
// -----
-// CHECK-LABEL: @my_module_atan_f32
+// CHECK-LABEL: emitc.func private @my_module_atan_f32
vm.module @my_module {
vm.func @atan_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_atan_f32"(%arg3) : (f32) -> f32
@@ -143,7 +143,7 @@
// -----
-// CHECK-LABEL: @my_module_atan2_f32
+// CHECK-LABEL: emitc.func private @my_module_atan2_f32
vm.module @my_module {
vm.func @atan2_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_atan2_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -154,7 +154,7 @@
// -----
-// CHECK-LABEL: @my_module_cos_f32
+// CHECK-LABEL: emitc.func private @my_module_cos_f32
vm.module @my_module {
vm.func @cos_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cos_f32"(%arg3) : (f32) -> f32
@@ -165,7 +165,7 @@
// -----
-// CHECK-LABEL: @my_module_sin_f32
+// CHECK-LABEL: emitc.func private @my_module_sin_f32
vm.module @my_module {
vm.func @sin_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_sin_f32"(%arg3) : (f32) -> f32
@@ -176,7 +176,7 @@
// -----
-// CHECK-LABEL: @my_module_exp_f32
+// CHECK-LABEL: emitc.func private @my_module_exp_f32
vm.module @my_module {
vm.func @exp_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_exp_f32"(%arg3) : (f32) -> f32
@@ -187,7 +187,7 @@
// -----
-// CHECK-LABEL: @my_module_exp2_f32
+// CHECK-LABEL: emitc.func private @my_module_exp2_f32
vm.module @my_module {
vm.func @exp2_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_exp2_f32"(%arg3) : (f32) -> f32
@@ -198,7 +198,7 @@
// -----
-// CHECK-LABEL: @my_module_expm1_f32
+// CHECK-LABEL: emitc.func private @my_module_expm1_f32
vm.module @my_module {
vm.func @expm1_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_expm1_f32"(%arg3) : (f32) -> f32
@@ -209,7 +209,7 @@
// -----
-// CHECK-LABEL: @my_module_log_f32
+// CHECK-LABEL: emitc.func private @my_module_log_f32
vm.module @my_module {
vm.func @log_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_log_f32"(%arg3) : (f32) -> f32
@@ -220,7 +220,7 @@
// -----
-// CHECK-LABEL: @my_module_log10_f32
+// CHECK-LABEL: emitc.func private @my_module_log10_f32
vm.module @my_module {
vm.func @log10_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_log10_f32"(%arg3) : (f32) -> f32
@@ -231,7 +231,7 @@
// -----
-// CHECK-LABEL: @my_module_log1p_f32
+// CHECK-LABEL: emitc.func private @my_module_log1p_f32
vm.module @my_module {
vm.func @log1p_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_log1p_f32"(%arg3) : (f32) -> f32
@@ -242,7 +242,7 @@
// -----
-// CHECK-LABEL: @my_module_log2_f32
+// CHECK-LABEL: emitc.func private @my_module_log2_f32
vm.module @my_module {
vm.func @log2_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_log2_f32"(%arg3) : (f32) -> f32
@@ -253,7 +253,7 @@
// -----
-// CHECK-LABEL: @my_module_pow_f32
+// CHECK-LABEL: emitc.func private @my_module_pow_f32
vm.module @my_module {
vm.func @pow_f32(%arg0 : f32, %arg1 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_pow_f32"(%arg3, %arg4) : (f32, f32) -> f32
@@ -264,7 +264,7 @@
// -----
-// CHECK-LABEL: @my_module_rsqrt_f32
+// CHECK-LABEL: emitc.func private @my_module_rsqrt_f32
vm.module @my_module {
vm.func @rsqrt_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_rsqrt_f32"(%arg3) : (f32) -> f32
@@ -275,7 +275,7 @@
// -----
-// CHECK-LABEL: @my_module_sqrt_f32
+// CHECK-LABEL: emitc.func private @my_module_sqrt_f32
vm.module @my_module {
vm.func @sqrt_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_sqrt_f32"(%arg3) : (f32) -> f32
@@ -286,7 +286,7 @@
// -----
-// CHECK-LABEL: @my_module_tanh_f32
+// CHECK-LABEL: emitc.func private @my_module_tanh_f32
vm.module @my_module {
vm.func @tanh_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_tanh_f32"(%arg3) : (f32) -> f32
@@ -297,7 +297,7 @@
// -----
-// CHECK-LABEL: @my_module_erf_f32
+// CHECK-LABEL: emitc.func private @my_module_erf_f32
vm.module @my_module {
vm.func @erf_f32(%arg0 : f32) -> f32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_erf_f32"(%arg3) : (f32) -> f32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_i64.mlir
index 63f6546..a5bb597 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/arithmetic_ops_i64.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_add_i64
+// CHECK-LABEL: emitc.func private @my_module_add_i64
vm.module @my_module {
vm.func @add_i64(%arg0: i64, %arg1: i64) {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_add_i64"(%arg3, %arg4) : (i64, i64) -> i64
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_sub_i64
+// CHECK-LABEL: emitc.func private @my_module_sub_i64
vm.module @my_module {
vm.func @sub_i64(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_sub_i64"(%arg3, %arg4) : (i64, i64) -> i64
@@ -22,7 +22,7 @@
// -----
-// CHECK-LABEL: @my_module_mul_i64
+// CHECK-LABEL: emitc.func private @my_module_mul_i64
vm.module @my_module {
vm.func @mul_i64(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_mul_i64"(%arg3, %arg4) : (i64, i64) -> i64
@@ -33,7 +33,7 @@
// -----
-// CHECK-LABEL: @my_module_div_i64_s
+// CHECK-LABEL: emitc.func private @my_module_div_i64_s
vm.module @my_module {
vm.func @div_i64_s(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_div_i64s"(%arg3, %arg4) : (i64, i64) -> i64
@@ -44,7 +44,7 @@
// -----
-// CHECK-LABEL: @my_module_div_i64_u
+// CHECK-LABEL: emitc.func private @my_module_div_i64_u
vm.module @my_module {
vm.func @div_i64_u(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_div_i64u"(%arg3, %arg4) : (i64, i64) -> i64
@@ -55,7 +55,7 @@
// -----
-// CHECK-LABEL: @my_module_rem_i64_s
+// CHECK-LABEL: emitc.func private @my_module_rem_i64_s
vm.module @my_module {
vm.func @rem_i64_s(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_rem_i64s"(%arg3, %arg4) : (i64, i64) -> i64
@@ -66,7 +66,7 @@
// -----
-// CHECK-LABEL: @my_module_rem_i64_u
+// CHECK-LABEL: emitc.func private @my_module_rem_i64_u
vm.module @my_module {
vm.func @rem_i64_u(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_rem_i64u"(%arg3, %arg4) : (i64, i64) -> i64
@@ -77,7 +77,7 @@
// -----
-// CHECK-LABEL: @my_module_fma_i64
+// CHECK-LABEL: emitc.func private @my_module_fma_i64
vm.module @my_module {
vm.func @fma_i64(%arg0: i64, %arg1: i64, %arg2: i64) {
// CHECK: %0 = emitc.call_opaque "vm_fma_i64"(%arg3, %arg4, %arg5) : (i64, i64, i64) -> i64
@@ -88,7 +88,7 @@
// -----
-// CHECK-LABEL: @my_module_abs_i64
+// CHECK-LABEL: emitc.func private @my_module_abs_i64
vm.module @my_module {
vm.func @abs_i64(%arg0 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_abs_i64"(%arg3) : (i64) -> i64
@@ -99,7 +99,7 @@
// -----
-// CHECK-LABEL: @my_module_min_i64_s
+// CHECK-LABEL: emitc.func private @my_module_min_i64_s
vm.module @my_module {
vm.func @min_i64_s(%arg0: i64, %arg1: i64) {
// CHECK: %0 = emitc.call_opaque "vm_min_i64s"(%arg3, %arg4) : (i64, i64) -> i64
@@ -110,7 +110,7 @@
// -----
-// CHECK-LABEL: @my_module_not_i64
+// CHECK-LABEL: emitc.func private @my_module_not_i64
vm.module @my_module {
vm.func @not_i64(%arg0 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_not_i64"(%arg3) : (i64) -> i64
@@ -121,7 +121,7 @@
// -----
-// CHECK-LABEL: @my_module_and_i64
+// CHECK-LABEL: emitc.func private @my_module_and_i64
vm.module @my_module {
vm.func @and_i64(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_and_i64"(%arg3, %arg4) : (i64, i64) -> i64
@@ -132,7 +132,7 @@
// -----
-// CHECK-LABEL: @my_module_or_i64
+// CHECK-LABEL: emitc.func private @my_module_or_i64
vm.module @my_module {
vm.func @or_i64(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_or_i64"(%arg3, %arg4) : (i64, i64) -> i64
@@ -143,7 +143,7 @@
// -----
-// CHECK-LABEL: @my_module_xor_i64
+// CHECK-LABEL: emitc.func private @my_module_xor_i64
vm.module @my_module {
vm.func @xor_i64(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_xor_i64"(%arg3, %arg4) : (i64, i64) -> i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops.mlir
index 32ce78c..2dc2274 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_select_i32
+// CHECK-LABEL: emitc.func private @my_module_select_i32
vm.module @my_module {
vm.func @select_i32(%arg0 : i32, %arg1 : i32, %arg2 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_select_i32"(%arg3, %arg4, %arg5) : (i32, i32, i32) -> i32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_f32.mlir
index 4d279e6..4c63698 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_f32.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_select_f32
+// CHECK-LABEL: emitc.func private @my_module_select_f32
vm.module @my_module {
vm.func @select_f32(%arg0 : i32, %arg1 : f32, %arg2 : f32) -> f32 {
// CHECK: %0 = emitc.call_opaque "vm_select_f32"(%arg3, %arg4, %arg5) : (i32, f32, f32) -> f32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_i64.mlir
index ba1a5db..a1afb27 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/assignment_ops_i64.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_select_i64
+// CHECK-LABEL: emitc.func private @my_module_select_i64
vm.module @my_module {
vm.func @select_i64(%arg0 : i32, %arg1 : i64, %arg2 : i64) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_select_i64"(%arg3, %arg4, %arg5) : (i32, i64, i64) -> i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops.mlir
index 0517fe0..33a7db8 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops.mlir
@@ -1,13 +1,13 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_buffer_alloc
+// CHECK-LABEL: emitc.func private @my_module_buffer_alloc
vm.module @my_module {
vm.func @buffer_alloc() {
// CHECK: %[[SIZE:.+]] = "emitc.constant"() <{value = 128 : i64}> : () -> i64
// CHECK-DAG: %[[ALIGNMENT:.+]] = "emitc.constant"() <{value = 32 : i32}> : () -> i32
// CHECK-DAG: %[[BUFFER:.+]] = "emitc.variable"() <{value = #emitc.opaque<"NULL">}> : () -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK-DAG: %[[BUFFER_PTR:.+]] = emitc.apply "&"(%[[BUFFER]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>) -> !emitc.ptr<!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>>
- // CHECK-DAG: %[[ALLOCTOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
+ // CHECK-DAG: %[[ALLOCTOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
// CHECK-DAG: %[[BUFFER_ACCESS:.+]] = "emitc.constant"() <{value = #emitc.opaque<"IREE_VM_BUFFER_ACCESS_MUTABLE | IREE_VM_BUFFER_ACCESS_ORIGIN_GUEST">}> : () -> !emitc.opaque<"iree_vm_buffer_access_t">
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "iree_vm_buffer_create"(%[[BUFFER_ACCESS]], %[[SIZE]], %[[ALIGNMENT]], %[[ALLOCTOR]], %[[BUFFER_PTR]]) : (!emitc.opaque<"iree_vm_buffer_access_t">, i64, i32, !emitc.opaque<"iree_allocator_t">, !emitc.ptr<!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>>) -> !emitc.opaque<"iree_status_t">
@@ -23,7 +23,7 @@
// -----
-// CHECK-LABEL: @my_module_buffer_clone
+// CHECK-LABEL: emitc.func private @my_module_buffer_clone
vm.module @my_module {
vm.func @buffer_clone(%buf : !vm.buffer) {
// CHECK-DAG: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -32,7 +32,7 @@
// CHECK: %[[BUFFER:.+]] = "emitc.variable"() <{value = #emitc.opaque<"NULL">}> : () -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK-DAG: %[[BUFFER_PTR:.+]] = emitc.apply "&"(%[[BUFFER]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>) -> !emitc.ptr<!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>>
- // CHECK-DAG: %[[ALLOCATOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
+ // CHECK-DAG: %[[ALLOCATOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
// CHECK-DAG: %[[BUFFER_ACCESS:.+]] = "emitc.constant"() <{value = #emitc.opaque<"IREE_VM_BUFFER_ACCESS_MUTABLE | IREE_VM_BUFFER_ACCESS_ORIGIN_GUEST">}> : () -> !emitc.opaque<"iree_vm_buffer_access_t">
// CHECK-DAG: %[[BUFFER_REF2:.+]] = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-DAG: %[[BUFFER_PTR2:.+]] = emitc.call_opaque "iree_vm_buffer_deref"(%[[BUFFER_REF2]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
@@ -48,7 +48,7 @@
// -----
-// CHECK-LABEL: @my_module_buffer_length
+// CHECK-LABEL: emitc.func private @my_module_buffer_length
vm.module @my_module {
vm.func @buffer_length(%buf : !vm.buffer) {
// CHECK: %[[BUFFER_REF:.+]] = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
@@ -63,7 +63,7 @@
// -----
-// CHECK-LABEL: @my_module_buffer_compare
+// CHECK-LABEL: emitc.func private @my_module_buffer_compare
vm.module @my_module {
vm.func @buffer_compare(%buf : !vm.buffer, %buf2 : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -87,7 +87,7 @@
// -----
-// CHECK-LABEL: @my_module_buffer_copy
+// CHECK-LABEL: emitc.func private @my_module_buffer_copy
vm.module @my_module {
vm.func @buffer_copy(%buf : !vm.buffer, %buf2 : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -110,7 +110,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_fill_i8
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_i8
vm.func @buffer_fill_i8(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C16:.+]] = "emitc.constant"() <{value = 16 : i64}> : () -> i64
@@ -127,7 +127,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_fill_i16
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_i16
vm.func @buffer_fill_i16(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_fill_i16"
%c0 = vm.const.i64 0
@@ -137,7 +137,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_fill_i32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_i32
vm.func @buffer_fill_i32(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_fill_i32"
%c0 = vm.const.i64 0
@@ -151,7 +151,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_load_i8s
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i8s
vm.func @buffer_load_i8s(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -159,7 +159,7 @@
// CHECK-NEXT: %[[BUFFER_PTR:.+]] = emitc.call_opaque "iree_vm_buffer_deref"(%[[BUFFER_REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK: %[[RESULT:.+]] = "emitc.variable"() <{value = 0 : i32}> : () -> i32
- // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%6) : (i32) -> !emitc.ptr<i32>
+ // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i8s"(%[[BUFFER_PTR]], %[[C0]], %[[RESULT_PTR]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>, i64, !emitc.ptr<i32>) -> !emitc.opaque<"iree_status_t">
%c0 = vm.const.i64 0
@@ -167,7 +167,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_load_i8u
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i8u
vm.func @buffer_load_i8u(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i8u"
%c0 = vm.const.i64 0
@@ -175,7 +175,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_load_i16s
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i16s
vm.func @buffer_load_i16s(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i16s"
%c0 = vm.const.i64 0
@@ -183,7 +183,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_load_i16u
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i16u
vm.func @buffer_load_i16u(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i16u"
%c0 = vm.const.i64 0
@@ -191,7 +191,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_load_i32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i32
vm.func @buffer_load_i32(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i32"
%c0 = vm.const.i64 0
@@ -203,7 +203,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_store_i8
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_i8
vm.func @buffer_store_i8(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C102:.+]] = "emitc.constant"() <{value = 102 : i32}> : () -> i32
@@ -218,7 +218,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_store_i16
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_i16
vm.func @buffer_store_i16(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_store_i16"
%c0 = vm.const.i64 0
@@ -227,7 +227,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_buffer_store_i32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_i32
vm.func @buffer_store_i32(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_store_i32"
%c0 = vm.const.i64 0
@@ -240,7 +240,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_hash
+ // CHECK-LABEL: emitc.func private @my_module_buffer_hash
vm.func @buffer_hash(%buf : !vm.buffer) {
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "iree_vm_buffer_hash"
%c0 = vm.const.i64 0
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f32.mlir
index 1ada44e..4b598ac 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f32.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_fill_f32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_f32
vm.func @buffer_fill_f32(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C16:.+]] = "emitc.constant"() <{value = 16 : i64}> : () -> i64
@@ -22,7 +22,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_load_f32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_f32
vm.func @buffer_load_f32(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -30,7 +30,7 @@
// CHECK-NEXT: %[[BUFFER_PTR:.+]] = emitc.call_opaque "iree_vm_buffer_deref"(%[[BUFFER_REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK: %[[RESULT:.+]] = "emitc.variable"() <{value = 0.000000e+00 : f32}> : () -> f32
- // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%6) : (f32) -> !emitc.ptr<f32>
+ // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (f32) -> !emitc.ptr<f32>
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_f32"(%[[BUFFER_PTR]], %[[C0]], %[[RESULT_PTR]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>, i64, !emitc.ptr<f32>) -> !emitc.opaque<"iree_status_t">
%c0 = vm.const.i64 0
@@ -42,7 +42,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_store_f32
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_f32
vm.func @buffer_store_f32(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C102:.+]] = "emitc.constant"() <{value = 1.020000e+02 : f32}> : () -> f32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f64.mlir
index 697cc66..2184c8c 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_f64.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_fill_f64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_f64
vm.func @buffer_fill_f64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C16:.+]] = "emitc.constant"() <{value = 16 : i64}> : () -> i64
@@ -22,7 +22,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_load_f64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_f64
vm.func @buffer_load_f64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -30,7 +30,7 @@
// CHECK-NEXT: %[[BUFFER_PTR:.+]] = emitc.call_opaque "iree_vm_buffer_deref"(%[[BUFFER_REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK: %[[RESULT:.+]] = "emitc.variable"() <{value = 0.000000e+00 : f64}> : () -> f64
- // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%6) : (f64) -> !emitc.ptr<f64>
+ // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (f64) -> !emitc.ptr<f64>
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_f64"(%[[BUFFER_PTR]], %[[C0]], %[[RESULT_PTR]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>, i64, !emitc.ptr<f64>) -> !emitc.opaque<"iree_status_t">
%c0 = vm.const.i64 0
@@ -42,7 +42,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_store_f64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_f64
vm.func @buffer_store_f64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C102:.+]] = "emitc.constant"() <{value = 1.020000e+02 : f64}> : () -> f64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_i64.mlir
index 431dced..394f414 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/buffer_ops_i64.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_fill_i64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_fill_i64
vm.func @buffer_fill_i64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C16:.+]] = "emitc.constant"() <{value = 16 : i64}> : () -> i64
@@ -22,7 +22,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_load_i64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_load_i64
vm.func @buffer_load_i64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
@@ -30,7 +30,7 @@
// CHECK-NEXT: %[[BUFFER_PTR:.+]] = emitc.call_opaque "iree_vm_buffer_deref"(%[[BUFFER_REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>
// CHECK: %[[RESULT:.+]] = "emitc.variable"() <{value = 0 : i64}> : () -> i64
- // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%6) : (i64) -> !emitc.ptr<i64>
+ // CHECK-NEXT: %[[RESULT_PTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i64) -> !emitc.ptr<i64>
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "vm_buffer_load_i64"(%[[BUFFER_PTR]], %[[C0]], %[[RESULT_PTR]]) : (!emitc.ptr<!emitc.opaque<"iree_vm_buffer_t">>, i64, !emitc.ptr<i64>) -> !emitc.opaque<"iree_status_t">
%c0 = vm.const.i64 0
@@ -42,7 +42,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_buffer_store_i64
+ // CHECK-LABEL: emitc.func private @my_module_buffer_store_i64
vm.func @buffer_store_i64(%buf : !vm.buffer) {
// CHECK: %[[C0:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
// CHECK: %[[C102:.+]] = "emitc.constant"() <{value = 102 : i64}> : () -> i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops.mlir
index 244c37e..31c5ece 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @module {
- // CHECK-LABEL: @module_cmp_eq_i32
+ // CHECK-LABEL: emitc.func private @module_cmp_eq_i32
vm.func @cmp_eq_i32(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_eq_i32"(%arg3, %arg4) : (i32, i32) -> i32
%0 = vm.cmp.eq.i32 %arg0, %arg1 : i32
@@ -12,7 +12,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_ne_i32
+ // CHECK-LABEL: emitc.func private @module_cmp_ne_i32
vm.func @cmp_ne_i32(%arg0 : i32, %arg1 : i32) {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_ne_i32"(%arg3, %arg4) : (i32, i32) -> i32
%0 = vm.cmp.ne.i32 %arg0, %arg1 : i32
@@ -23,7 +23,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_i32_s
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_i32_s
vm.func @cmp_lt_i32_s(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_i32s"(%arg3, %arg4) : (i32, i32) -> i32
%0 = vm.cmp.lt.i32.s %arg0, %arg1 : i32
@@ -34,7 +34,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_i32_u
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_i32_u
vm.func @cmp_lt_i32_u(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_i32u"(%arg3, %arg4) : (i32, i32) -> i32
%0 = vm.cmp.lt.i32.u %arg0, %arg1 : i32
@@ -45,7 +45,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_nz_i32
+ // CHECK-LABEL: emitc.func private @module_cmp_nz_i32
vm.func @cmp_nz_i32(%arg0 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_nz_i32"(%arg3) : (i32) -> i32
%0 = vm.cmp.nz.i32 %arg0 : i32
@@ -56,7 +56,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_eq_ref
+ // CHECK-LABEL: emitc.func private @module_cmp_eq_ref
vm.func @cmp_eq_ref(%arg0 : !vm.ref<?>, %arg1 : !vm.ref<?>) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_eq_ref"(%arg3, %arg4) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> i32
%0 = vm.cmp.eq.ref %arg0, %arg1 : !vm.ref<?>
@@ -67,7 +67,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_ne_ref
+ // CHECK-LABEL: emitc.func private @module_cmp_ne_ref
vm.func @cmp_ne_ref(%arg0 : !vm.ref<?>, %arg1 : !vm.ref<?>) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_ne_ref"(%arg3, %arg4) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> i32
%0 = vm.cmp.ne.ref %arg0, %arg1 : !vm.ref<?>
@@ -78,7 +78,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_nz_ref
+ // CHECK-LABEL: emitc.func private @module_cmp_nz_ref
vm.func @cmp_nz_ref(%arg0 : !vm.ref<?>) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_nz_ref"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> i32
%0 = vm.cmp.nz.ref %arg0 : !vm.ref<?>
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_f32.mlir
index 2755935..825fe8b 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_f32.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @module {
- // CHECK-LABEL: @module_cmp_eq_f32o
+ // CHECK-LABEL: emitc.func private @module_cmp_eq_f32o
vm.func @cmp_eq_f32o(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_eq_f32o"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.eq.f32.o %arg0, %arg1 : f32
@@ -12,7 +12,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_eq_f32u
+ // CHECK-LABEL: emitc.func private @module_cmp_eq_f32u
vm.func @cmp_eq_f32u(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_eq_f32u"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.eq.f32.u %arg0, %arg1 : f32
@@ -23,7 +23,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_ne_f32o
+ // CHECK-LABEL: emitc.func private @module_cmp_ne_f32o
vm.func @cmp_ne_f32o(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_ne_f32o"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.ne.f32.o %arg0, %arg1 : f32
@@ -34,7 +34,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_ne_f32u
+ // CHECK-LABEL: emitc.func private @module_cmp_ne_f32u
vm.func @cmp_ne_f32u(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_ne_f32u"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.ne.f32.u %arg0, %arg1 : f32
@@ -45,7 +45,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_f32o
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_f32o
vm.func @cmp_lt_f32o(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_f32o"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.lt.f32.o %arg0, %arg1 : f32
@@ -56,7 +56,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_f32u
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_f32u
vm.func @cmp_lt_f32u(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_f32u"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.lt.f32.u %arg0, %arg1 : f32
@@ -67,7 +67,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lte_f32o
+ // CHECK-LABEL: emitc.func private @module_cmp_lte_f32o
vm.func @cmp_lte_f32o(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lte_f32o"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.lte.f32.o %arg0, %arg1 : f32
@@ -78,7 +78,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lte_f32u
+ // CHECK-LABEL: emitc.func private @module_cmp_lte_f32u
vm.func @cmp_lte_f32u(%arg0 : f32, %arg1 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lte_f32u"(%arg3, %arg4) : (f32, f32) -> i32
%0 = vm.cmp.lte.f32.u %arg0, %arg1 : f32
@@ -89,7 +89,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_nan_f32
+ // CHECK-LABEL: emitc.func private @module_cmp_nan_f32
vm.func @cmp_nan_f32(%arg0 : f32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_nan_f32"(%arg3) : (f32) -> i32
%0 = vm.cmp.nan.f32 %arg0 : f32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_i64.mlir
index 357efcb..dcf6ee3 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/comparison_ops_i64.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @module {
- // CHECK-LABEL: @module_cmp_eq_i64
+ // CHECK-LABEL: emitc.func private @module_cmp_eq_i64
vm.func @cmp_eq_i64(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_eq_i64"(%arg3, %arg4) : (i64, i64) -> i32
%0 = vm.cmp.eq.i64 %arg0, %arg1 : i64
@@ -12,7 +12,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_ne_i64
+ // CHECK-LABEL: emitc.func private @module_cmp_ne_i64
vm.func @cmp_ne_i64(%arg0 : i64, %arg1 : i64) {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_ne_i64"(%arg3, %arg4) : (i64, i64) -> i32
%0 = vm.cmp.ne.i64 %arg0, %arg1 : i64
@@ -23,7 +23,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_i64_s
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_i64_s
vm.func @cmp_lt_i64_s(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_i64s"(%arg3, %arg4) : (i64, i64) -> i32
%0 = vm.cmp.lt.i64.s %arg0, %arg1 : i64
@@ -34,7 +34,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_lt_i64_u
+ // CHECK-LABEL: emitc.func private @module_cmp_lt_i64_u
vm.func @cmp_lt_i64_u(%arg0 : i64, %arg1 : i64) -> i64 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_lt_i64u"(%arg3, %arg4) : (i64, i64) -> i32
%0 = vm.cmp.lt.i64.u %arg0, %arg1 : i64
@@ -45,7 +45,7 @@
// -----
vm.module @module {
- // CHECK-LABEL: @module_cmp_nz_i64
+ // CHECK-LABEL: emitc.func private @module_cmp_nz_i64
vm.func @cmp_nz_i64(%arg0 : i64) -> i64 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cmp_nz_i64"(%arg3) : (i64) -> i32
%0 = vm.cmp.nz.i64 %arg0 : i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops.mlir
index 0a28e03..eaf8656 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops.mlir
@@ -2,7 +2,7 @@
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_i32_zero
+ // CHECK-LABEL: emitc.func private @my_module_const_i32_zero
vm.func @const_i32_zero() -> i32 {
// CHECK: %[[ZERO:.+]] = "emitc.constant"() <{value = 0 : i32}> : () -> i32
%zero = vm.const.i32.zero
@@ -13,7 +13,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_i32
+ // CHECK-LABEL: emitc.func private @my_module_const_i32
vm.func @const_i32() {
// CHECK-NEXT: %0 = "emitc.constant"() <{value = 0 : i32}> : () -> i32
%0 = vm.const.i32 0
@@ -28,7 +28,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_ref_zero
+ // CHECK-LABEL: emitc.func private @my_module_const_ref_zero
vm.func @const_ref_zero() {
// CHECK: %[[REF:.+]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %[[REFPTR:.+]] = emitc.apply "&"(%[[REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_f32.mlir
index 8dcd455..7622fd1 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_f32.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_f32_zero
+ // CHECK-LABEL: emitc.func private @my_module_const_f32_zero
vm.func @const_f32_zero() -> f32 {
// CHECK: %[[ZERO:.+]] = "emitc.constant"() <{value = 0.000000e+00 : f32}> : () -> f32
%zero = vm.const.f32.zero
@@ -12,7 +12,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_f32
+ // CHECK-LABEL: emitc.func private @my_module_const_f32
vm.func @const_f32() {
// CHECK-NEXT: %0 = "emitc.constant"() <{value = 5.000000e-01 : f32}> : () -> f32
%0 = vm.const.f32 0.5
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_i64.mlir
index 5b4ede3..f0c689b 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/const_ops_i64.mlir
@@ -2,7 +2,7 @@
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_i64_zero
+ // CHECK-LABEL: emitc.func private @my_module_const_i64_zero
vm.func @const_i64_zero() -> i64 {
// CHECK: %[[ZERO:.+]] = "emitc.constant"() <{value = 0 : i64}> : () -> i64
%zero = vm.const.i64.zero
@@ -13,7 +13,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_const_i64
+ // CHECK-LABEL: emitc.func private @my_module_const_i64
vm.func @const_i64() {
// CHECK-NEXT: %0 = "emitc.constant"() <{value = 0 : i64}> : () -> i64
%0 = vm.const.i64 0
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/control_flow_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/control_flow_ops.mlir
index fb58ef9..2f4b938 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/control_flow_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/control_flow_ops.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_branch_empty
+ // CHECK-LABEL: emitc.func private @my_module_branch_empty
vm.func @branch_empty() {
// CHECK: cf.br ^bb1
vm.br ^bb1
@@ -15,7 +15,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_branch_int_args
+ // CHECK-LABEL: emitc.func private @my_module_branch_int_args
vm.func @branch_int_args(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: cf.br ^bb1(%arg3, %arg4 : i32, i32)
vm.br ^bb1(%arg0, %arg1 : i32, i32)
@@ -29,7 +29,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_branch_ref_args
+ // CHECK-LABEL: emitc.func private @my_module_branch_ref_args
vm.func @branch_ref_args(%arg0 : !vm.ref<?>) -> !vm.ref<?> {
// CHECK: cf.br ^bb1
// CHECK: cf.br ^bb2
@@ -44,7 +44,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_branch_mixed_args
+ // CHECK-LABEL: emitc.func private @my_module_branch_mixed_args
vm.func @branch_mixed_args(%arg0 : !vm.ref<?>, %arg1: i32, %arg2 : !vm.ref<?>, %arg3: i32) -> !vm.ref<?> {
// CHECK: cf.br ^bb1
// CHECK: cf.br ^bb2(%arg4, %arg6 : i32, i32)
@@ -60,15 +60,15 @@
// Test vm.call conversion on an imported function.
vm.module @my_module {
- // CHECK: func.func @my_module_call_[[IMPORTFN:[^\(]+]]
+ // CHECK: emitc.func private @my_module_call_[[IMPORTFN:[^\(]+]]
vm.import private @imported_fn(%arg0 : i32) -> i32
- // CHECK: func.func @my_module_call_imported_fn
+ // CHECK: emitc.func private @my_module_call_imported_fn
vm.func @call_imported_fn(%arg0 : i32) -> i32 {
// Lookup import from module struct.
// CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[IMPORT:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[IMPORTS]]) {args = [0 : index, 0 : ui32]}
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
@@ -77,7 +77,7 @@
// CHECK-NEXT: %[[RESPTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// Call the function created by the vm.import conversion.
- // CHECK-NEXT: %{{.+}} = call @my_module_call_[[IMPORTFN]](%arg0, %[[IMPORT]], %arg3, %[[RESPTR]])
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_call_[[IMPORTFN]](%arg0, %[[IMPORT]], %arg3, %[[RESPTR]])
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>, i32, !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
%0 = vm.call @imported_fn(%arg0) : (i32) -> i32
@@ -89,12 +89,12 @@
// Test that the order of imports and calls doesn't matter.
vm.module @my_module {
- // CHECK: func.func @my_module_call_imported_fn
+ // CHECK: emitc.func private @my_module_call_imported_fn
vm.func @call_imported_fn(%arg0 : i32) -> i32 {
// Lookup import from module struct.
// CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[IMPORT:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[IMPORTS]]) {args = [0 : index, 0 : ui32]}
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
@@ -103,14 +103,14 @@
// CHECK-NEXT: %[[RESPTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// Call the function created by the vm.import conversion.
- // CHECK-NEXT: %{{.+}} = call @my_module_call_[[IMPORTFN:[^\(]+]](%arg0, %[[IMPORT]], %arg3, %[[RESPTR]])
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_call_[[IMPORTFN:[^\(]+]](%arg0, %[[IMPORT]], %arg3, %[[RESPTR]])
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>, i32, !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
%0 = vm.call @imported_fn(%arg0) : (i32) -> i32
vm.return %0 : i32
}
- // CHECK: func.func @my_module_call_[[IMPORTFN]]
+ // CHECK: emitc.func private @my_module_call_[[IMPORTFN]]
vm.import private @imported_fn(%arg0 : i32) -> i32
}
@@ -122,7 +122,7 @@
vm.return %arg0 : i32
}
- // CHECK-LABEL: @my_module_call_internal_fn
+ // CHECK-LABEL: emitc.func private @my_module_call_internal_fn
vm.func @call_internal_fn(%arg0 : i32) -> i32 {
// Create a variable for the result.
@@ -130,9 +130,9 @@
// CHECK-NEXT: %[[RESPTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// Call the function created by the vm.import conversion.
- // CHECK-NEXT: %{{.+}} = call @my_module_internal_fn(%arg0, %arg1, %arg2, %arg3, %[[RESPTR]])
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"my_module_t">>,
- // CHECK-SAME: !emitc.ptr<!emitc.opaque<"my_module_state_t">>, i32, !emitc.ptr<i32>)
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_internal_fn(%arg0, %arg1, %arg2, %arg3, %[[RESPTR]])
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"struct my_module_t">>,
+ // CHECK-SAME: !emitc.ptr<!emitc.opaque<"struct my_module_state_t">>, i32, !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
%0 = vm.call @internal_fn(%arg0) : (i32) -> i32
vm.return %0 : i32
@@ -143,15 +143,15 @@
// Test vm.call.variadic conversion on an imported function.
vm.module @my_module {
- // CHECK: func.func @my_module_call_[[VARIADICFN:[^\(]+]]
+ // CHECK: emitc.func private @my_module_call_[[VARIADICFN:[^\(]+]]
vm.import private @variadic_fn(%arg0 : i32 ...) -> i32
- // CHECK: func.func @my_module_call_variadic
+ // CHECK: emitc.func private @my_module_call_variadic
vm.func @call_variadic(%arg0 : i32, %arg1 : i32) -> i32 {
// Lookup import from module struct.
// CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[IMPORT:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[IMPORTS]]) {args = [0 : index, 0 : ui32]}
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
@@ -163,7 +163,7 @@
// CHECK-NEXT: %[[RESPTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// Call the function created by the vm.import conversion.
- // CHECK-NEXT: %{{.+}} = call @my_module_call_[[VARIADICFN]](%arg0, %[[IMPORT]], %[[NARGS]], %arg3, %arg4, %[[RESPTR]])
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_call_[[VARIADICFN]](%arg0, %[[IMPORT]], %[[NARGS]], %arg3, %arg4, %[[RESPTR]])
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
// CHECK-SAME: i32, i32, i32, !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
@@ -176,15 +176,15 @@
// Test vm.call.variadic with zero arguments.
vm.module @my_module {
- // CHECK: func.func @my_module_call_[[VARIADICFN:[^\(]+]]
+ // CHECK: emitc.func private @my_module_call_[[VARIADICFN:[^\(]+]]
vm.import private @variadic_fn(%arg0 : i32 ...) -> i32
- // CHECK: func.func @my_module_call_variadic
+ // CHECK: emitc.func private @my_module_call_variadic
vm.func @call_variadic() -> i32 {
// Lookup import from module struct.
// CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[IMPORT:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[IMPORTS]]) {args = [0 : index, 0 : ui32]}
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
@@ -196,7 +196,7 @@
// CHECK-NEXT: %[[RESPTR:.+]] = emitc.apply "&"(%[[RESULT]]) : (i32) -> !emitc.ptr<i32>
// Call the function created by the vm.import conversion.
- // CHECK-NEXT: %{{.+}} = call @my_module_call_[[VARIADICFN]](%arg0, %[[IMPORT]], %[[NARGS]], %[[RESPTR]])
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_call_[[VARIADICFN]](%arg0, %[[IMPORT]], %[[NARGS]], %[[RESPTR]])
// CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
// CHECK-SAME: i32, !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
@@ -210,10 +210,10 @@
// TODO(simon-camp): add check statements
// Test vm.call.variadic with multiple variadic packs.
vm.module @my_module {
- // CHECK: func.func @my_module_call_[[VARIADICFN:[^\(]+]]
+ // CHECK: emitc.func private @my_module_call_[[VARIADICFN:[^\(]+]]
vm.import private @variadic_fn(%is : i32 ..., %fs : f32 ...) -> i32
- // CHECK: func.func @my_module_call_variadic
+ // CHECK: emitc.func private @my_module_call_variadic
vm.func @call_variadic(%i : i32, %f : f32) -> i32 {
%0 = vm.call.variadic @variadic_fn([%i, %i], [%f, %f, %f]) : (i32 ..., f32 ...) -> i32
@@ -224,7 +224,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_cond_branch_empty
+ // CHECK-LABEL: emitc.func private @my_module_cond_branch_empty
vm.func @cond_branch_empty(%arg0 : i32, %arg1 : i32, %arg2 : i32) -> i32 {
// CHECK: cf.cond_br %{{.}}, ^bb1, ^bb2
vm.cond_br %arg0, ^bb1, ^bb2
@@ -242,7 +242,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_cond_branch_int_args
+ // CHECK-LABEL: emitc.func private @my_module_cond_branch_int_args
vm.func @cond_branch_int_args(%arg0 : i32, %arg1 : i32, %arg2 : i32) -> i32 {
// CHECK: cf.cond_br {{%.}}, ^bb1(%arg4 : i32), ^bb2(%arg5 : i32)
vm.cond_br %arg0, ^bb1(%arg1 : i32), ^bb2(%arg2 : i32)
@@ -260,7 +260,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_cond_branch_ref_args
+ // CHECK-LABEL: emitc.func private @my_module_cond_branch_ref_args
vm.func @cond_branch_ref_args(%arg0 : i32, %arg1 : !vm.ref<?>, %arg2 : !vm.ref<?>) -> !vm.ref<?> {
// CHECK: cf.cond_br {{%.}}, ^bb1, ^bb4
// CHECK: cf.br ^bb2
@@ -279,7 +279,7 @@
// -----
-// CHECK-LABEL: @my_module_br_table_empty
+// CHECK-LABEL: emitc.func private @my_module_br_table_empty
vm.module @my_module {
vm.func @br_table_empty(%arg0: i32, %arg1: i32) -> i32 {
// CHECK-NOT: vm.br_table
@@ -300,7 +300,7 @@
// -----
-// CHECK-LABEL: @my_module_br_table
+// CHECK-LABEL: emitc.func private @my_module_br_table
vm.module @my_module {
vm.func @br_table(%arg0: i32, %arg1: i32, %arg2: i32) -> i32 {
// CHECK-NOT: vm.br_table
@@ -330,7 +330,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_fail
+ // CHECK-LABEL: emitc.func private @my_module_fail
vm.func @fail(%arg0 : i32) {
// Typecast the argument to fail and branch respectively.
@@ -364,8 +364,8 @@
// Test vm.import conversion on a void function.
vm.module @my_module {
- // CHECK-LABEL: func.func @my_module_call_0v_v_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>)
- // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {emitc.static} {
+ // CHECK-LABEL: emitc.func private @my_module_call_0v_v_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>)
+ // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {specifiers = ["static"]} {
// Calculate the size of the arguments. To avoid empty structs we insert a dummy value.
// CHECK-NEXT: %[[ARGSIZE:.+]] = "emitc.constant"() <{value = #emitc.opaque<"0">}> : () -> !emitc.opaque<"iree_host_size_t">
@@ -426,9 +426,9 @@
// Test vm.import conversion on a variadic function.
vm.module @my_module {
- // CHECK-LABEL: func.func @my_module_call_0iCiD_i_2_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
+ // CHECK-LABEL: emitc.func private @my_module_call_0iCiD_i_2_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
// CHECK-SAME: %arg2: i32, %arg3: i32, %arg4: i32, %arg5: i32, %arg6: !emitc.ptr<i32>)
- // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {emitc.static} {
+ // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {specifiers = ["static"]} {
// Calculate the size of the arguments.
// CHECK-NEXT: %[[ARGSIZE0:.+]] = "emitc.constant"() <{value = #emitc.opaque<"0">}> : () -> !emitc.opaque<"iree_host_size_t">
@@ -529,9 +529,9 @@
// Test vm.call.variadic with zero variadic arguments.
vm.module @my_module {
- // CHECK-LABEL: func.func @my_module_call_0iCiD_i_0_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
+ // CHECK-LABEL: emitc.func private @my_module_call_0iCiD_i_0_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
// CHECK-SAME: %arg2: i32, %arg3: i32, %arg4: !emitc.ptr<i32>)
- // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {emitc.static} {
+ // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {specifiers = ["static"]} {
// Calculate the size of the arguments.
// CHECK-NEXT: %[[ARGSIZE0:.+]] = "emitc.constant"() <{value = #emitc.opaque<"0">}> : () -> !emitc.opaque<"iree_host_size_t">
@@ -616,9 +616,9 @@
// Test vm.import conversion on a function with vm.ref arguments.
vm.module @my_module {
- // CHECK-LABEL: func.func @my_module_call_0r_r_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
+ // CHECK-LABEL: emitc.func private @my_module_call_0r_r_import_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>,
// CHECK-SAME: %arg2: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, %arg3: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>)
- // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {emitc.static} {
+ // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {specifiers = ["static"]} {
// Calculate the size of the arguments.
// CHECK-NEXT: %[[ARGSIZE0:.+]] = "emitc.constant"() <{value = #emitc.opaque<"0">}> : () -> !emitc.opaque<"iree_host_size_t">
@@ -690,39 +690,39 @@
vm.module @my_module {
- // Typedef structs for arguments and results
- // CHECK: emitc.call_opaque "EMITC_TYPEDEF_STRUCT"() {args = [#emitc.opaque<"my_module_fn_args_t">, #emitc.opaque<"int32_t arg0;">]} : () -> ()
- // CHECK: emitc.call_opaque "EMITC_TYPEDEF_STRUCT"() {args = [#emitc.opaque<"my_module_fn_result_t">, #emitc.opaque<"int32_t res0;">]} : () -> ()
-
+ // Define structs for arguments and results
+ // CHECK: emitc.verbatim "struct my_module_fn_args_t {int32_t arg0;};"
+ // CHECK-NEXT: emitc.verbatim "struct my_module_fn_result_t {int32_t res0;};"
+
// Create a new function to export with the adapted signature.
- // CHECK: func.func @my_module_fn_export_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.opaque<"uint32_t">, %arg2: !emitc.opaque<"iree_byte_span_t">, %arg3: !emitc.opaque<"iree_byte_span_t">,
+ // CHECK: emitc.func private @my_module_fn_export_shim(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, %arg1: !emitc.opaque<"uint32_t">, %arg2: !emitc.opaque<"iree_byte_span_t">, %arg3: !emitc.opaque<"iree_byte_span_t">,
// CHECK-SAME: %arg4: !emitc.ptr<!emitc.opaque<"void">>, %arg5: !emitc.ptr<!emitc.opaque<"void">>)
- // CHECK-SAME: -> !emitc.opaque<"iree_status_t"> attributes {emitc.static, vm.calling_convention = "0i_i", vm.export_name = "fn"}
+ // CHECK-SAME: -> !emitc.opaque<"iree_status_t">
// Cast module and module state structs.
- // CHECK-NEXT: %[[MODULECASTED:.+]] = emitc.cast %arg4 : !emitc.ptr<!emitc.opaque<"void">> to !emitc.ptr<!emitc.opaque<"my_module_t">>
- // CHECK-NEXT: %[[MODSTATECASTED:.+]] = emitc.cast %arg5 : !emitc.ptr<!emitc.opaque<"void">> to !emitc.ptr<!emitc.opaque<"my_module_state_t">>
+ // CHECK-NEXT: %[[MODULECASTED:.+]] = emitc.cast %arg4 : !emitc.ptr<!emitc.opaque<"void">> to !emitc.ptr<!emitc.opaque<"struct my_module_t">>
+ // CHECK-NEXT: %[[MODSTATECASTED:.+]] = emitc.cast %arg5 : !emitc.ptr<!emitc.opaque<"void">> to !emitc.ptr<!emitc.opaque<"struct my_module_state_t">>
// Cast argument and result structs.
// CHECK-NEXT: %[[ARGDATA:.+]] = emitc.call_opaque "EMITC_STRUCT_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"data">]}
// CHECK-SAME: : (!emitc.opaque<"iree_byte_span_t">) -> !emitc.ptr<ui8>
- // CHECK-NEXT: %[[ARGS:.+]] = emitc.cast %[[ARGDATA]] : !emitc.ptr<ui8> to !emitc.ptr<!emitc.opaque<"my_module_fn_args_t">>
+ // CHECK-NEXT: %[[ARGS:.+]] = emitc.cast %[[ARGDATA]] : !emitc.ptr<ui8> to !emitc.ptr<!emitc.opaque<"struct my_module_fn_args_t">>
// CHECK-NEXT: %[[RESULTDATA:.+]] = emitc.call_opaque "EMITC_STRUCT_MEMBER"(%arg3) {args = [0 : index, #emitc.opaque<"data">]}
// CHECK-SAME: : (!emitc.opaque<"iree_byte_span_t">) -> !emitc.ptr<ui8>
- // CHECK-NEXT: %[[RESULTS:.+]] = emitc.cast %[[RESULTDATA]] : !emitc.ptr<ui8> to !emitc.ptr<!emitc.opaque<"my_module_fn_result_t">>
+ // CHECK-NEXT: %[[RESULTS:.+]] = emitc.cast %[[RESULTDATA]] : !emitc.ptr<ui8> to !emitc.ptr<!emitc.opaque<"struct my_module_fn_result_t">>
// Unpack the argument from the struct.
// CHECK-NEXT: %[[MARG:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%[[ARGS]]) {args = [0 : index, #emitc.opaque<"arg0">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_fn_args_t">>) -> i32
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_fn_args_t">>) -> i32
// Unpack the result pointer from the struct.
// CHECK-NEXT: %[[MRES:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER_ADDRESS"(%[[RESULTS]]) {args = [0 : index, #emitc.opaque<"res0">]}
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"my_module_fn_result_t">>) -> !emitc.ptr<i32>
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"struct my_module_fn_result_t">>) -> !emitc.ptr<i32>
// Call the internal function.
- // CHECK-NEXT: %{{.+}} = call @my_module_fn(%arg0, %[[MODULECASTED]], %[[MODSTATECASTED]], %[[MARG]], %[[MRES]])
- // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"my_module_t">>,
- // CHECK-SAME: !emitc.ptr<!emitc.opaque<"my_module_state_t">>, i32, !emitc.ptr<i32>) -> !emitc.opaque<"iree_status_t">
+ // CHECK-NEXT: %{{.+}} = emitc.call @my_module_fn(%arg0, %[[MODULECASTED]], %[[MODSTATECASTED]], %[[MARG]], %[[MRES]])
+ // CHECK-SAME: : (!emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>, !emitc.ptr<!emitc.opaque<"struct my_module_t">>,
+ // CHECK-SAME: !emitc.ptr<!emitc.opaque<"struct my_module_state_t">>, i32, !emitc.ptr<i32>) -> !emitc.opaque<"iree_status_t">
// Return ok status.
// CHECK: %[[STATUS:.+]] = emitc.call_opaque "iree_ok_status"() : () -> !emitc.opaque<"iree_status_t">
@@ -738,7 +738,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_return
+ // CHECK-LABEL: emitc.func private @my_module_return
vm.func @return(%arg0 : i32, %arg1 : !vm.ref<?>) -> (i32, !vm.ref<?>) {
// Move the i32 value and ref into the result function arguments.
@@ -766,9 +766,9 @@
vm.module @my_module {
vm.import private optional @optional_import_fn(%arg0 : i32) -> i32
- // CHECK-LABEL: @my_module_call_fn
+ // CHECK-LABEL: emitc.func private @my_module_call_fn
vm.func @call_fn() -> i32 {
- // CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
+ // CHECK-NEXT: %[[IMPORTS:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"imports">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[IMPORT:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[IMPORTS]]) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_function_t">>
// CHECK-NEXT: %[[MODULE:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%[[IMPORT]]) {args = [0 : index, #emitc.opaque<"module">]} : (!emitc.ptr<!emitc.opaque<"iree_vm_function_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_module_t">>
// CHECK-NEXT: %[[CONDITION0:.+]] = emitc.call_opaque "EMITC_UNARY"(%[[MODULE]]) {args = [#emitc.opaque<"!">, 0 : index]} : (!emitc.ptr<!emitc.opaque<"iree_vm_module_t">>) -> i1
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops.mlir
index 7c6409a..871cb09 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_trunc
+// CHECK-LABEL: emitc.func private @my_module_trunc
vm.module @my_module {
vm.func @trunc(%arg0 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_trunc_i32i8"(%arg3) : (i32) -> i32
@@ -13,7 +13,7 @@
// -----
-// CHECK-LABEL: @my_module_ext
+// CHECK-LABEL: emitc.func private @my_module_ext
vm.module @my_module {
vm.func @ext(%arg0 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_ext_i8i32s"(%arg3) : (i32) -> i32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_f32.mlir
index e4bea00..2c1c11e 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_f32.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_bitcast
+// CHECK-LABEL: emitc.func private @my_module_bitcast
vm.module @my_module {
vm.func @bitcast(%arg0 : i32) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_bitcast_i32f32"(%arg3) : (i32) -> f32
@@ -13,7 +13,7 @@
// -----
-// CHECK-LABEL: @my_module_cast
+// CHECK-LABEL: emitc.func private @my_module_cast
vm.module @my_module {
vm.func @cast(%arg0 : i32) -> (i32, i32) {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_cast_si32f32"(%arg3) : (i32) -> f32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_i64.mlir
index b4c1ed1..6d17f59 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/conversion_ops_i64.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_trunc_i64
+// CHECK-LABEL: emitc.func private @my_module_trunc_i64
vm.module @my_module {
vm.func @trunc_i64(%arg0 : i64) -> i32 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_trunc_i64i32"(%arg3) : (i64) -> i32
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_ext_i64
+// CHECK-LABEL: emitc.func private @my_module_ext_i64
vm.module @my_module {
vm.func @ext_i64(%arg0 : i32) -> i64 {
// CHECK-NEXT: %0 = emitc.call_opaque "vm_ext_i32i64s"(%arg3) : (i32) -> i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/func_op.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/func_op.mlir
index 47a06db..85295045 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/func_op.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/func_op.mlir
@@ -4,15 +4,14 @@
// some arguments are getting added. For more details see comments on the
// `ConvertVMToEmitCPass` class in ConvertVMToEmitC.cpp.
vm.module @my_module {
- // CHECK: func.func @my_module_fn(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>
- // CHECK-SAME: %arg1: !emitc.ptr<!emitc.opaque<"my_module_t">>,
- // CHECK-SAME: %arg2: !emitc.ptr<!emitc.opaque<"my_module_state_t">>,
+ // CHECK: emitc.func private @my_module_fn(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_stack_t">>
+ // CHECK-SAME: %arg1: !emitc.ptr<!emitc.opaque<"struct my_module_t">>,
+ // CHECK-SAME: %arg2: !emitc.ptr<!emitc.opaque<"struct my_module_state_t">>,
// CHECK-SAME: %arg3: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>,
// CHECK-SAME: %arg4: i32,
// CHECK-SAME: %arg5: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>,
// CHECK-SAME: %arg6: !emitc.ptr<i32>)
// CHECK-SAME: -> !emitc.opaque<"iree_status_t">
- // CHECK-SAME: attributes {emitc.static, vm.calling_convention = "0ri_ri"}
vm.func @fn(%arg0 : !vm.ref<?>, %arg1 : i32) -> (!vm.ref<?>, i32) {
vm.return %arg0, %arg1 : !vm.ref<?>, i32
}
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops.mlir
index 3dc77e0..2bc89b7 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops.mlir
@@ -3,9 +3,9 @@
vm.module @my_module {
vm.global.i32 private @c42 = 42 : i32
- // CHECK-LABEL: @my_module_global_load_i32
+ // CHECK-LABEL: emitc.func private @my_module_global_load_i32
vm.func @global_load_i32() -> i32 {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: %1 = emitc.call_opaque "vm_global_load_i32"(%0) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<ui8>) -> i32
%0 = vm.global.load.i32 @c42 : i32
vm.return %0 : i32
@@ -17,9 +17,9 @@
vm.module @my_module {
vm.global.i32 private mutable @c107_mut = 107 : i32
- // CHECK-LABEL: @my_module_global_store_i32
+ // CHECK-LABEL: emitc.func private @my_module_global_store_i32
vm.func @global_store_i32(%arg0 : i32) {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: emitc.call_opaque "vm_global_store_i32"(%0, %arg3) {args = [0 : index, 0 : ui32, 1 : index]} : (!emitc.ptr<ui8>, i32) -> ()
vm.global.store.i32 %arg0, @c107_mut : i32
vm.return
@@ -31,9 +31,9 @@
vm.module @my_module {
vm.global.ref private @g0 : !vm.buffer
- // CHECK-LABEL: @my_module_global_load_ref
+ // CHECK-LABEL: emitc.func private @my_module_global_load_ref
vm.func @global_load_ref() -> !vm.buffer {
- // CHECK: %[[A:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"refs">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
+ // CHECK: %[[A:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"refs">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
// CHECK: %[[B:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[A]]) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
// CHECK: %[[C:.+]] = emitc.call_opaque "iree_vm_type_def_as_ref"(%{{.+}}) : (!emitc.opaque<"iree_vm_type_def_t">) -> !emitc.opaque<"iree_vm_ref_type_t">
// CHECK: %{{.+}} = emitc.call_opaque "iree_vm_ref_retain_or_move_checked"(%[[B]], %[[C]], %arg3) {args = [false, 0 : index, 1 : index, 2 : index]} : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, !emitc.opaque<"iree_vm_ref_type_t">, !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_status_t">
@@ -47,9 +47,9 @@
vm.module @my_module {
vm.global.ref private mutable @g0_mut : !vm.buffer
- // CHECK-LABEL: @my_module_global_store_ref
+ // CHECK-LABEL: emitc.func private @my_module_global_store_ref
vm.func @global_store_ref(%arg0 : !vm.buffer) {
- // CHECK: %[[A:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"refs">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
+ // CHECK: %[[A:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"refs">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
// CHECK: %[[B:.+]] = emitc.call_opaque "EMITC_ARRAY_ELEMENT_ADDRESS"(%[[A]]) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
// CHECK: %[[C:.+]] = emitc.call_opaque "iree_vm_type_def_as_ref"(%{{.+}}) : (!emitc.opaque<"iree_vm_type_def_t">) -> !emitc.opaque<"iree_vm_ref_type_t">
// CHECK: %{{.+}} = emitc.call_opaque "iree_vm_ref_retain_or_move_checked"(%arg3, %[[C]], %[[B]]) {args = [false, 0 : index, 1 : index, 2 : index]} : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, !emitc.opaque<"iree_vm_ref_type_t">, !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_status_t">
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_f32.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_f32.mlir
index 5a0b6cc..cdfef0d 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_f32.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_f32.mlir
@@ -3,9 +3,9 @@
vm.module @my_module {
vm.global.f32 private @c42 = 42.5 : f32
- // CHECK-LABEL: @my_module_global_load_f32
+ // CHECK-LABEL: emitc.func private @my_module_global_load_f32
vm.func @global_load_f32() -> f32 {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: %1 = emitc.call_opaque "vm_global_load_f32"(%0) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<ui8>) -> f32
%0 = vm.global.load.f32 @c42 : f32
vm.return %0 : f32
@@ -17,9 +17,9 @@
vm.module @my_module {
vm.global.f32 private mutable @c107_mut = 107.5 : f32
- // CHECK-LABEL: @my_module_global_store_f32
+ // CHECK-LABEL: emitc.func private @my_module_global_store_f32
vm.func @global_store_f32(%arg0 : f32) {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: emitc.call_opaque "vm_global_store_f32"(%0, %arg3) {args = [0 : index, 0 : ui32, 1 : index]} : (!emitc.ptr<ui8>, f32) -> ()
vm.global.store.f32 %arg0, @c107_mut : f32
vm.return
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_i64.mlir
index 49ada1d..16b6199 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/global_ops_i64.mlir
@@ -3,9 +3,9 @@
vm.module @my_module {
vm.global.i64 private @c42 = 42 : i64
- // CHECK-LABEL: @my_module_global_load_i64
+ // CHECK-LABEL: emitc.func private @my_module_global_load_i64
vm.func @global_load_i64() -> i64 {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: %1 = emitc.call_opaque "vm_global_load_i64"(%0) {args = [0 : index, 0 : ui32]} : (!emitc.ptr<ui8>) -> i64
%0 = vm.global.load.i64 @c42 : i64
vm.return %0 : i64
@@ -17,9 +17,9 @@
vm.module @my_module {
vm.global.i64 private mutable @c107_mut = 107 : i64
- // CHECK-LABEL: @my_module_global_store_i64
+ // CHECK-LABEL: emitc.func private @my_module_global_store_i64
vm.func @global_store_i64(%arg0 : i64) {
- // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.ptr<ui8>
+ // CHECK-NEXT: %0 = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"rwdata">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.ptr<ui8>
// CHECK-NEXT: emitc.call_opaque "vm_global_store_i64"(%0, %arg3) {args = [0 : index, 0 : ui32, 1 : index]} : (!emitc.ptr<ui8>, i64) -> ()
vm.global.store.i64 %arg0, @c107_mut : i64
vm.return
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops.mlir
index 07f3440..92526fd 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops.mlir
@@ -1,11 +1,11 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_alloc
+ // CHECK-LABEL: emitc.func private @my_module_list_alloc
vm.func @list_alloc(%arg0: i32) -> !vm.list<i32> {
// CHECK: %[[LIST:.+]] = "emitc.variable"() <{value = #emitc.opaque<"NULL">}> : () -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
// CHECK: %[[LIST_PTR:.+]] = emitc.apply "&"(%3) : (!emitc.ptr<!emitc.opaque<"iree_vm_list_t">>) -> !emitc.ptr<!emitc.ptr<!emitc.opaque<"iree_vm_list_t">>>
- // CHECK: %[[ALLOCATOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
+ // CHECK: %[[ALLOCATOR:.+]] = emitc.call_opaque "EMITC_STRUCT_PTR_MEMBER"(%arg2) {args = [0 : index, #emitc.opaque<"allocator">]} : (!emitc.ptr<!emitc.opaque<"struct my_module_state_t">>) -> !emitc.opaque<"iree_allocator_t">
// CHECK: %[[TYPE_DEF:.+]] = emitc.call_opaque "iree_vm_make_value_type_def"() {args = [#emitc.opaque<"IREE_VM_VALUE_TYPE_I32">]} : () -> !emitc.opaque<"iree_vm_type_def_t">
// CHECK-NEXT: %[[STATUS:.+]] = emitc.call_opaque "iree_vm_list_create"(%[[TYPE_DEF]], %arg3, %[[ALLOCATOR]], %[[LIST_PTR]]) : (!emitc.opaque<"iree_vm_type_def_t">, i32, !emitc.opaque<"iree_allocator_t">, !emitc.ptr<!emitc.ptr<!emitc.opaque<"iree_vm_list_t">>>) -> !emitc.opaque<"iree_status_t">
@@ -21,7 +21,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_reserve
+ // CHECK-LABEL: emitc.func private @my_module_list_reserve
vm.func @list_reserve(%arg0: !vm.list<i32>, %arg1: i32) {
// CHECK-NEXT: %0 = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %1 = emitc.call_opaque "iree_vm_list_deref"(%0) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
@@ -34,7 +34,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_resize
+ // CHECK-LABEL: emitc.func private @my_module_list_resize
vm.func @list_resize(%arg0: !vm.list<i32>, %arg1: i32) {
// CHECK-NEXT: %0 = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %1 = emitc.call_opaque "iree_vm_list_deref"(%0) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
@@ -47,7 +47,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_size
+ // CHECK-LABEL: emitc.func private @my_module_list_size
vm.func @list_size(%arg0: !vm.list<i32>) -> i32 {
// CHECK-NEXT: %0 = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %1 = emitc.call_opaque "iree_vm_list_deref"(%0) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
@@ -60,7 +60,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_get_i32
+ // CHECK-LABEL: emitc.func private @my_module_list_get_i32
vm.func @list_get_i32(%arg0: !vm.list<i32>, %arg1: i32) -> i32 {
// CHECK-NEXT: %0 = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_value_t">
// CHECK-NEXT: %1 = emitc.apply "&"(%0) : (!emitc.opaque<"iree_vm_value_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_value_t">>
@@ -75,7 +75,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_get_ref
+ // CHECK-LABEL: emitc.func private @my_module_list_get_ref
vm.func @list_get_ref(%arg0: !vm.list<!vm.ref<?>>, %arg1: i32) -> !vm.buffer {
// CHECK-NEXT: %0 = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %1 = emitc.call_opaque "iree_vm_list_deref"(%0) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
@@ -100,7 +100,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_set_i32
+ // CHECK-LABEL: emitc.func private @my_module_list_set_i32
vm.func @list_set_i32(%arg0: !vm.list<i32>, %arg1: i32, %arg2: i32) {
// CHECK-NEXT: %0 = emitc.call_opaque "iree_vm_value_make_i32"(%arg5) : (i32) -> !emitc.opaque<"iree_vm_value_t">
// CHECK-NEXT: %1 = emitc.apply "&"(%0) : (!emitc.opaque<"iree_vm_value_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_value_t">>
@@ -115,7 +115,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_set_ref
+ // CHECK-LABEL: emitc.func private @my_module_list_set_ref
vm.func @list_set_ref(%arg0: !vm.list<!vm.ref<?>>, %arg1: i32, %arg2: !vm.buffer) {
// CHECK-NEXT: %0 = emitc.apply "*"(%arg3) : (!emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>) -> !emitc.opaque<"iree_vm_ref_t">
// CHECK-NEXT: %1 = emitc.call_opaque "iree_vm_list_deref"(%0) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_list_t">>
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops_i64.mlir
index 2dc4997..37847f5 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/list_ops_i64.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_get_i64
+ // CHECK-LABEL: emitc.func private @my_module_list_get_i64
vm.func @list_get_i64(%arg0: !vm.list<i64>, %arg1: i32) -> i64 {
// CHECK-NEXT: %0 = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_value_t">
// CHECK-NEXT: %1 = emitc.apply "&"(%0) : (!emitc.opaque<"iree_vm_value_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_value_t">>
@@ -16,7 +16,7 @@
// -----
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_set_i64
+ // CHECK-LABEL: emitc.func private @my_module_list_set_i64
vm.func @list_set_i64(%arg0: !vm.list<i64>, %arg1: i32, %arg2: i64) {
// CHECK-NEXT: %0 = emitc.call_opaque "iree_vm_value_make_i64"(%arg5) : (i64) -> !emitc.opaque<"iree_vm_value_t">
// CHECK-NEXT: %1 = emitc.apply "&"(%0) : (!emitc.opaque<"iree_vm_value_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_value_t">>
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops.mlir
index fb0c42d..ae54f6e 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_shl_i32
+// CHECK-LABEL: emitc.func private @my_module_shl_i32
vm.module @my_module {
vm.func @shl_i32(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_shl_i32"(%arg3, %arg4) : (i32, i32) -> i32
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_shr_i32_s
+// CHECK-LABEL: emitc.func private @my_module_shr_i32_s
vm.module @my_module {
vm.func @shr_i32_s(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_shr_i32s"(%arg3, %arg4) : (i32, i32) -> i32
@@ -22,7 +22,7 @@
// -----
-// CHECK-LABEL: @my_module_shr_i32_u
+// CHECK-LABEL: emitc.func private @my_module_shr_i32_u
vm.module @my_module {
vm.func @shr_i32_u(%arg0 : i32, %arg1 : i32) -> i32 {
// CHECK: %0 = emitc.call_opaque "vm_shr_i32u"(%arg3, %arg4) : (i32, i32) -> i32
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops_i64.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops_i64.mlir
index ed130dc..75192f2 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops_i64.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/shift_ops_i64.mlir
@@ -1,6 +1,6 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
-// CHECK-LABEL: @my_module_shl_i64
+// CHECK-LABEL: emitc.func private @my_module_shl_i64
vm.module @my_module {
vm.func @shl_i64(%arg0 : i64, %arg1 : i32) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_shl_i64"(%arg3, %arg4) : (i64, i32) -> i64
@@ -11,7 +11,7 @@
// -----
-// CHECK-LABEL: @my_module_shr_i64_s
+// CHECK-LABEL: emitc.func private @my_module_shr_i64_s
vm.module @my_module {
vm.func @shr_i64_s(%arg0 : i64, %arg1 : i32) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_shr_i64s"(%arg3, %arg4) : (i64, i32) -> i64
@@ -22,7 +22,7 @@
// -----
-// CHECK-LABEL: @my_module_shr_i64_u
+// CHECK-LABEL: emitc.func private @my_module_shr_i64_u
vm.module @my_module {
vm.func @shr_i64_u(%arg0 : i64, %arg1 : i32) -> i64 {
// CHECK: %0 = emitc.call_opaque "vm_shr_i64u"(%arg3, %arg4) : (i64, i32) -> i64
diff --git a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/type_conversion.mlir b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/type_conversion.mlir
index d63b717..2329053 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/type_conversion.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Conversion/VMToEmitC/test/type_conversion.mlir
@@ -1,7 +1,7 @@
// RUN: iree-opt --split-input-file --pass-pipeline="builtin.module(vm.module(iree-vm-ordinal-allocation),vm.module(iree-convert-vm-to-emitc))" %s | FileCheck %s
vm.module @my_module {
- // CHECK-LABEL: @my_module_list_alloc
+ // CHECK-LABEL: emitc.func private @my_module_list_alloc
vm.func @list_alloc(%arg0: i32) {
// CHECK: %[[REF:.+]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_ref_t">
// CHECK: %[[REFPTR:.+]] = emitc.apply "&"(%[[REF]]) : (!emitc.opaque<"iree_vm_ref_t">) -> !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>
@@ -11,7 +11,7 @@
vm.return
}
- // CHECK-LABEL: @my_module_list_size
+ // CHECK-LABEL: emitc.func private @my_module_list_size
vm.func @list_size(%arg0: i32) {
%list = vm.list.alloc %arg0 : (i32) -> !vm.list<i32>
// CHECK: %[[REF:.+]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_ref_t">
@@ -28,7 +28,7 @@
vm.module @my_module {
vm.rodata private @byte_buffer dense<[1, 2, 3]> : tensor<3xi32>
- // CHECK-LABEL: @my_module_ref
+ // CHECK-LABEL: emitc.func private @my_module_ref
vm.export @ref
vm.func @ref(%arg0: i32) {
// CHECK: %[[REF:.+]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.opaque<"iree_vm_ref_t">
@@ -42,12 +42,12 @@
// -----
-// Test the func.func conversion, which is needed as a second step after the
+// Test the emitc.func conversion, which is needed as a second step after the
// vm.func conversion. All references in the signature should be converted to
// emitc pointers.
vm.module @my_module {
- // CHECK: func.func @fn(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, %arg1: i32)
- func.func @fn(%arg0 : !vm.ref<?>, %arg1 : i32) -> () {
- return
+ // CHECK: emitc.func @fn(%arg0: !emitc.ptr<!emitc.opaque<"iree_vm_ref_t">>, %arg1: i32)
+ emitc.func @fn(%arg0 : !vm.ref<?>, %arg1 : i32) -> () {
+ emitc.return
}
}
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/BUILD.bazel b/compiler/src/iree/compiler/Dialect/VM/Target/C/BUILD.bazel
index 264974f..071ff09 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/BUILD.bazel
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/BUILD.bazel
@@ -13,29 +13,6 @@
)
iree_compiler_cc_library(
- name = "TranslateToCpp",
- srcs = [
- "TranslateToCpp.cpp",
- ],
- hdrs = [
- "CppEmitter.h",
- ],
- defines = [
- "IREE_HAVE_C_OUTPUT_FORMAT",
- ],
- deps = [
- "@llvm-project//llvm:Support",
- "@llvm-project//mlir:ControlFlowDialect",
- "@llvm-project//mlir:EmitCDialect",
- "@llvm-project//mlir:FuncDialect",
- "@llvm-project//mlir:FunctionInterfaces",
- "@llvm-project//mlir:IR",
- "@llvm-project//mlir:SCFDialect",
- "@llvm-project//mlir:Support",
- ],
-)
-
-iree_compiler_cc_library(
name = "C",
srcs = [
"CModuleTarget.cpp",
@@ -46,8 +23,10 @@
"CModuleTarget.h",
"TranslationFlags.h",
],
+ defines = [
+ "IREE_HAVE_C_OUTPUT_FORMAT",
+ ],
deps = [
- ":TranslateToCpp",
"//compiler/src/iree/compiler/Dialect/Util/IR",
"//compiler/src/iree/compiler/Dialect/Util/Transforms",
"//compiler/src/iree/compiler/Dialect/VM/Analysis",
@@ -55,9 +34,11 @@
"//compiler/src/iree/compiler/Dialect/VM/IR",
"//compiler/src/iree/compiler/Dialect/VM/Transforms",
"@llvm-project//llvm:Support",
+ "@llvm-project//mlir:ControlFlowDialect",
"@llvm-project//mlir:IR",
"@llvm-project//mlir:Pass",
"@llvm-project//mlir:Support",
+ "@llvm-project//mlir:TargetCpp",
"@llvm-project//mlir:Transforms",
"@llvm-project//mlir:TranslateLib",
],
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/CMakeLists.txt b/compiler/src/iree/compiler/Dialect/VM/Target/C/CMakeLists.txt
index 98dd734..695de19 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/CMakeLists.txt
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/CMakeLists.txt
@@ -25,27 +25,6 @@
iree_cc_library(
NAME
- TranslateToCpp
- HDRS
- "CppEmitter.h"
- SRCS
- "TranslateToCpp.cpp"
- DEPS
- LLVMSupport
- MLIRControlFlowDialect
- MLIREmitCDialect
- MLIRFuncDialect
- MLIRFunctionInterfaces
- MLIRIR
- MLIRSCFDialect
- MLIRSupport
- DEFINES
- "IREE_HAVE_C_OUTPUT_FORMAT"
- PUBLIC
-)
-
-iree_cc_library(
- NAME
C
HDRS
"CModuleTarget.h"
@@ -55,11 +34,12 @@
"TranslationFlags.cpp"
"TranslationRegistration.cpp"
DEPS
- ::TranslateToCpp
LLVMSupport
+ MLIRControlFlowDialect
MLIRIR
MLIRPass
MLIRSupport
+ MLIRTargetCpp
MLIRTransforms
MLIRTranslateLib
iree::compiler::Dialect::Util::IR
@@ -68,6 +48,8 @@
iree::compiler::Dialect::VM::Conversion::VMToEmitC
iree::compiler::Dialect::VM::IR
iree::compiler::Dialect::VM::Transforms
+ DEFINES
+ "IREE_HAVE_C_OUTPUT_FORMAT"
PUBLIC
)
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/CModuleTarget.cpp b/compiler/src/iree/compiler/Dialect/VM/Target/C/CModuleTarget.cpp
index 0617e80..1f01cd0 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/CModuleTarget.cpp
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/CModuleTarget.cpp
@@ -12,291 +12,15 @@
#include "iree/compiler/Dialect/VM/Analysis/RegisterAllocation.h"
#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/ConvertVMToEmitC.h"
#include "iree/compiler/Dialect/VM/Conversion/VMToEmitC/DropExcludedExports.h"
-#include "iree/compiler/Dialect/VM/Target/C/CppEmitter.h"
#include "iree/compiler/Dialect/VM/Transforms/Passes.h"
#include "llvm/ADT/TypeSwitch.h"
+#include "mlir/Dialect/ControlFlow/IR/ControlFlow.h"
#include "mlir/Pass/PassManager.h"
+#include "mlir/Target/Cpp/CppEmitter.h"
#include "mlir/Transforms/Passes.h"
namespace mlir::iree_compiler::IREE::VM {
-static void printCompilerConfigurationBlock(llvm::raw_ostream &output) {
- output << "//" << std::string(77, '=') << "\n"
- << "// compiler configuration\n"
- << "//" << std::string(77, '=') << "\n\n";
-}
-
-static void printModuleComment(IREE::VM::ModuleOp &moduleOp,
- llvm::raw_ostream &output) {
- output << "//" << std::string(77, '=') << "\n"
- << "// module \"" << moduleOp.getName()
- << "\"\n"
- "//"
- << std::string(77, '=') << "\n";
-}
-
-static LogicalResult
-printFunctionDeclaration(mlir::func::FuncOp funcOp, llvm::raw_ostream &output,
- mlir::emitc::CppEmitter &emitter) {
- Operation *op = funcOp.getOperation();
- if (op->hasAttr("emitc.static"))
- output << "static ";
-
- if (failed(emitter.emitTypes(funcOp.getLoc(),
- funcOp.getFunctionType().getResults())))
- return failure();
- output << " " << funcOp.getName();
-
- output << "(";
-
- bool error = false;
- llvm::interleaveComma(funcOp.getArguments(), output, [&](BlockArgument arg) {
- if (failed(emitter.emitType(funcOp.getLoc(), arg.getType())))
- error = true;
- });
- if (error)
- return failure();
- output << ");\n";
-
- return success();
-}
-
-static LogicalResult printRodataBuffers(IREE::VM::ModuleOp &moduleOp,
- mlir::emitc::CppEmitter &emitter) {
- llvm::raw_ostream &output = emitter.ostream();
- std::string moduleName = moduleOp.getName().str();
-
- for (auto rodataOp : moduleOp.getOps<IREE::VM::RodataOp>()) {
- auto value = llvm::dyn_cast<IREE::Util::SerializableAttrInterface>(
- rodataOp.getValue());
- assert(value && "expected a serializable rodata value");
- SmallVector<char> byteBuffer;
- if (failed(value.serializeToVector(rodataOp.getLoc(),
- llvm::endianness::little, byteBuffer))) {
- return rodataOp.emitError() << "error during serialization";
- }
-
- constexpr size_t kDefaultRodataAlignment = 16;
- size_t alignment =
- rodataOp.getAlignment()
- ? static_cast<size_t>(rodataOp.getAlignment().value())
- : 0;
- if (alignment == 0)
- alignment = kDefaultRodataAlignment;
-
- std::string bufferName =
- moduleOp.getName().str() + "_" + rodataOp.getName().str();
-
- output << "iree_alignas(" << alignment << ") static const uint8_t "
- << bufferName << "[] = {";
- llvm::interleaveComma(byteBuffer, output, [&](char value) {
- output << static_cast<unsigned int>(static_cast<unsigned char>(value));
- });
- output << "};\n";
- }
-
- output << "\n";
-
- return success();
-}
-
-static LogicalResult printStructDefinitions(IREE::VM::ModuleOp &moduleOp,
- mlir::emitc::CppEmitter &emitter) {
- llvm::raw_ostream &output = emitter.ostream();
- std::string moduleName = moduleOp.getName().str();
-
- // Returns |count| or 1 if |count| == 0.
- // Some compilers (MSVC) don't support zero-length struct fields on the
- // interior of structs (just VLA at the tail).
- auto countOrEmpty = [](uint32_t count) { return count ? count : 1; };
-
- const int64_t numTypes =
- llvm::cast<IntegerAttr>(moduleOp.getOperation()->getAttr("vm.num_types"))
- .getInt();
-
- output << "struct " << moduleName << "_t {\n";
- output << "iree_allocator_t allocator;\n";
- output << "iree_vm_ref_type_t types[" << countOrEmpty(numTypes) << "];\n";
- output << "};\n";
-
- output << "struct " << moduleName << "_state_t {\n";
-
- auto ordinalCounts = moduleOp.getOrdinalCountsAttr();
- output << "iree_allocator_t allocator;\n";
- output << "uint8_t rwdata[" << countOrEmpty(ordinalCounts.getGlobalBytes())
- << "];\n";
- output << "iree_vm_ref_t refs[" << countOrEmpty(ordinalCounts.getGlobalRefs())
- << "];\n";
- output << "iree_vm_buffer_t rodata_buffers["
- << countOrEmpty(ordinalCounts.getRodatas()) << "];\n";
- output << "iree_vm_function_t imports["
- << countOrEmpty(ordinalCounts.getImportFuncs()) << "];\n";
- output << "};\n";
-
- output << "typedef struct " << moduleName << "_t " << moduleName << "_t;\n";
- output << "typedef struct " << moduleName << "_state_t " << moduleName
- << "_state_t;\n";
-
- output << "\n";
-
- return success();
-}
-
-static LogicalResult buildModuleDescriptors(IREE::VM::ModuleOp &moduleOp,
- mlir::emitc::CppEmitter &emitter) {
- SymbolTable symbolTable(moduleOp);
- std::string moduleName = moduleOp.getName().str();
- llvm::raw_ostream &output = emitter.ostream();
-
- auto printStringView = [](StringRef s) -> std::string {
- // We can't use iree_make_string_view because function calls are not allowed
- // for constant expressions in C.
- // TODO(#7605): Switch to IREE_SVL. We can't use IREE_SVL today because it
- // uses designated initializers, which cause issues when compiled as C++.
- return ("{\"" + s + "\", " + std::to_string(s.size()) + "}").str();
- };
-
- // dependencies
- std::string dependenciesName = moduleName + "_dependencies_";
- output << "static const iree_vm_module_dependency_t " << dependenciesName
- << "[] = {\n";
- auto dependencies = moduleOp.getDependencies();
- if (dependencies.empty()) {
- // Empty list placeholder.
- output << " {{0}},\n";
- } else {
- for (auto &dependency : dependencies) {
- output << "{" << printStringView(dependency.name) << ", "
- << dependency.minimumVersion << ", "
- << (dependency.isOptional
- ? "IREE_VM_MODULE_DEPENDENCY_FLAG_OPTIONAL"
- : "IREE_VM_MODULE_DEPENDENCY_FLAG_REQUIRED")
- << "},\n";
- }
- }
- output << "};\n";
- output << "\n";
-
- // imports
- SmallVector<IREE::VM::ImportOp> importOps(
- moduleOp.getOps<IREE::VM::ImportOp>());
- std::string importName = moduleName + "_imports_";
- output << "static const iree_vm_native_import_descriptor_t " << importName
- << "[] = {\n";
- if (importOps.empty()) {
- // Empty list placeholder.
- output << " {0},\n";
- } else {
- // sort import ops by ordinal
- llvm::sort(importOps, [](auto &lhs, auto &rhs) {
- return lhs.getOrdinal()->getZExtValue() <
- rhs.getOrdinal()->getZExtValue();
- });
- for (auto importOp : importOps) {
- output << "{"
- << (importOp.getIsOptional() ? "IREE_VM_NATIVE_IMPORT_OPTIONAL"
- : "IREE_VM_NATIVE_IMPORT_REQUIRED")
- << ", " << printStringView(importOp.getName()) << "},\n";
- }
- }
- output << "};\n";
- output << "\n";
-
- // exports
- SmallVector<func::FuncOp> exportedFunctions;
- for (auto func : moduleOp.getOps<func::FuncOp>()) {
- if (func.getOperation()->hasAttr("vm.export_name")) {
- exportedFunctions.push_back(func);
- }
- }
- auto extractExportName = [](func::FuncOp funcOp) {
- return llvm::cast<StringAttr>(
- funcOp.getOperation()->getAttr("vm.export_name"));
- };
- std::string exportName = moduleName + "_exports_";
- output << "static const iree_vm_native_export_descriptor_t " << exportName
- << "[] = {\n";
- if (exportedFunctions.empty()) {
- // Empty list placeholder.
- output << " {{0}},\n";
- } else {
- // sort export ops
- llvm::sort(exportedFunctions, [&extractExportName](auto &lhs, auto &rhs) {
- return extractExportName(lhs).compare(extractExportName(rhs)) < 0;
- });
- for (auto funcOp : exportedFunctions) {
- StringAttr exportName = extractExportName(funcOp);
- StringAttr callingConvention = llvm::cast<StringAttr>(
- funcOp.getOperation()->getAttr("vm.calling_convention"));
- if (!callingConvention) {
- return funcOp.emitError("Couldn't find calling convention attribute");
- }
-
- // TODO(simon-camp): support function-level reflection attributes
- output << "{" << printStringView(exportName) << ", "
- << printStringView(callingConvention.getValue())
- << ", 0, NULL},\n";
- }
- }
- output << "};\n";
- output << "\n";
-
- // functions
- std::string functionName = moduleName + "_funcs_";
- output << "static const iree_vm_native_function_ptr_t " << functionName
- << "[] = {\n";
- if (exportedFunctions.empty()) {
- // Empty list placeholder.
- output << " {0},\n";
- } else {
- // We only add exported functions to the table, as calls to internal
- // functions are directly mapped to C function calls of the generated
- // implementation.
- for (auto funcOp : exportedFunctions) {
- auto funcName = funcOp.getName();
- output << "{"
- << "(iree_vm_native_function_shim_t)iree_emitc_shim, "
- << "(iree_vm_native_function_target_t)" << funcName << "},\n";
- }
- }
- output << "};\n";
- output << "\n";
-
- // module descriptor
- // TODO(simon-camp): support module-level reflection attributes
- std::string descriptorName = moduleName + "_descriptor_";
- output << "static const iree_vm_native_module_descriptor_t " << descriptorName
- << " = {\n"
- // name:
- << printStringView(moduleName)
- << ",\n"
- // version:
- << moduleOp.getVersion().value_or(0u)
- << ",\n"
- // attrs:
- << "0,\n"
- << "NULL,\n"
- // dependencies:
- << dependencies.size() << ",\n"
- << dependenciesName
- << ",\n"
- // imports:
- << importOps.size() << ",\n"
- << importName
- << ",\n"
- // exports:
- << exportedFunctions.size() << ",\n"
- << exportName
- << ",\n"
- // functions:
- << exportedFunctions.size() << ",\n"
- << functionName << ",\n"
- << "};\n";
-
- output << "\n";
- return success();
-}
-
/// Adapted from BytecodeModuleTarget and extended by C specific passes
static LogicalResult
canonicalizeModule(IREE::VM::ModuleOp moduleOp,
@@ -385,126 +109,28 @@
LogicalResult translateModuleToC(IREE::VM::ModuleOp moduleOp,
CTargetOptions targetOptions,
llvm::raw_ostream &output) {
- moduleOp.getContext()->getOrLoadDialect<IREE::Util::UtilDialect>();
+ moduleOp.getContext()
+ ->loadDialect<IREE::Util::UtilDialect, mlir::cf::ControlFlowDialect>();
if (failed(canonicalizeModule(moduleOp, targetOptions))) {
return moduleOp.emitError()
<< "failed to canonicalize vm.module to a serializable form";
}
+ auto innerModules = moduleOp.getOps<mlir::ModuleOp>();
+ if (innerModules.empty()) {
+ return moduleOp.emitError()
+ << "vm module does not contain an inner builtin.module op";
+ }
+ mlir::ModuleOp mlirModule = *innerModules.begin();
if (targetOptions.outputFormat == COutputFormat::kMlirText) {
// Use the standard MLIR text printer.
- moduleOp.getOperation()->print(output);
+ mlirModule.getOperation()->print(output);
output << "\n";
return success();
}
- std::string includeGuard = moduleOp.getName().upper();
- output << "#ifndef " << includeGuard << "_H_\n";
- output << "#define " << includeGuard << "_H_\n";
-
- auto printInclude = [&output](std::string include) {
- output << "#include \"" << include << "\"\n";
- };
-
- printInclude("iree/vm/api.h");
- output << "\n";
-
- output << "#ifdef __cplusplus\n";
- output << "extern \"C\" {\n";
- output << "#endif // __cplusplus\n";
- output << "\n";
-
- mlir::emitc::CppEmitter emitter(output, /*declareVariablesAtTop=*/true);
- for (auto funcOp : moduleOp.getOps<mlir::func::FuncOp>()) {
- Operation *op = funcOp.getOperation();
- if (!op->hasAttr("vm.module.constructor"))
- continue;
- if (failed(printFunctionDeclaration(funcOp, output, emitter)))
- return failure();
- }
-
- output << "\n";
- output << "#ifdef __cplusplus\n";
- output << "} // extern \"C\"\n";
- output << "#endif // __cplusplus\n";
- output << "\n";
-
- output << "#endif // " << includeGuard << "_H_\n\n";
- output << "#if defined(EMITC_IMPLEMENTATION)\n";
-
- printInclude("iree/vm/ops.h");
- printInclude("iree/vm/ops_emitc.h");
- printInclude("iree/vm/shims_emitc.h");
- output << "\n";
-
- printCompilerConfigurationBlock(output);
- output << "\n";
-
- printModuleComment(moduleOp, output);
- output << "\n";
-
- mlir::emitc::CppEmitter::Scope scope(emitter);
-
- if (failed(printRodataBuffers(moduleOp, emitter))) {
- return failure();
- }
-
- // build struct definitions
- if (failed(printStructDefinitions(moduleOp, emitter))) {
- return failure();
- }
-
- // translate functions
- output << "// DECLARE FUNCTIONS\n";
-
- for (auto funcOp : moduleOp.getOps<mlir::func::FuncOp>()) {
- Operation *op = funcOp.getOperation();
- if (op->hasAttr("vm.module.constructor"))
- continue;
- if (failed(printFunctionDeclaration(funcOp, output, emitter)))
- return failure();
- }
-
- output << "// DEFINE FUNCTIONS\n";
-
- // Emit code for functions skipping those marked with `vm.emit_at_end`.
- for (Operation &op : moduleOp.getOps()) {
- // TODO(simon-camp): Clean up. We generate calls to a macro that defines a
- // struct. As we declare all variables at the start of the function, the
- // macro call cannot be inlined into the function.
- if (!isa<mlir::func::FuncOp, emitc::CallOpaqueOp>(op))
- continue;
- if (op.hasAttr("vm.emit_at_end"))
- continue;
- if (op.hasAttr("emitc.static"))
- output << "static ";
- if (failed(emitter.emitOperation(op,
- /*trailingSemicolon=*/false)))
- return failure();
- }
-
- output << "\n";
-
- // generate module descriptors
- if (failed(buildModuleDescriptors(moduleOp, emitter))) {
- return failure();
- }
-
- // Emit code for functions marked with `vm.emit_at_end`.
- for (auto funcOp : moduleOp.getOps<mlir::func::FuncOp>()) {
- Operation *op = funcOp.getOperation();
- if (!op->hasAttr("vm.emit_at_end"))
- continue;
- if (op->hasAttr("emitc.static"))
- output << "static ";
- if (failed(emitter.emitOperation(*funcOp.getOperation(),
- /*trailingSemicolon=*/false)))
- return failure();
- }
-
- output << "#endif // EMITC_IMPLEMENTATION\n";
- return success();
+ return mlir::emitc::translateToCpp(mlirModule.getOperation(), output, true);
}
LogicalResult translateModuleToC(mlir::ModuleOp outerModuleOp,
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/TranslateToCpp.cpp b/compiler/src/iree/compiler/Dialect/VM/Target/C/TranslateToCpp.cpp
deleted file mode 100644
index a5163c2..0000000
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/TranslateToCpp.cpp
+++ /dev/null
@@ -1,861 +0,0 @@
-// Copyright 2020 The IREE Authors
-//
-// Licensed 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
-
-// Formated in LLVM style. Avoid reformatting for upcoming upstreaming.
-// clang-format off
-
-#include "iree/compiler/Dialect/VM/Target/C/CppEmitter.h"
-
-#include "mlir/Dialect/EmitC/IR/EmitC.h"
-#include "mlir/Dialect/SCF/IR/SCF.h"
-#include "mlir/Dialect/ControlFlow/IR/ControlFlowOps.h"
-#include "mlir/Dialect/Func/IR/FuncOps.h"
-#include "mlir/IR/BuiltinOps.h"
-#include "mlir/IR/BuiltinTypes.h"
-#include "mlir/IR/Dialect.h"
-#include "mlir/IR/Operation.h"
-#include "mlir/Support/IndentedOstream.h"
-#include "llvm/ADT/DenseMap.h"
-#include "llvm/ADT/StringExtras.h"
-#include "llvm/ADT/StringMap.h"
-#include "llvm/ADT/TypeSwitch.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;
-
-static LogicalResult printConstantOp(CppEmitter &emitter, Operation *operation,
- Attribute value) {
- OpResult result = operation->getResult(0);
-
- // Only emit an assignment as the variable was already declared when printing
- // the FuncOp.
- if (emitter.shouldDeclareVariablesAtTop()) {
- // Skip the assignment if the emitc.constant has no value.
- if (auto oAttr = llvm::dyn_cast<emitc::OpaqueAttr>(value)) {
- if (oAttr.getValue().empty())
- return success();
- }
-
- if (failed(emitter.emitVariableAssignment(result)))
- return failure();
- return emitter.emitAttribute(operation->getLoc(), value);
- }
-
- // Emit a variable declaration for an emitc.constant op without value.
- if (auto oAttr = llvm::dyn_cast<emitc::OpaqueAttr>(value)) {
- if (oAttr.getValue().empty())
- // The semicolon gets printed by the emitOperation function.
- return emitter.emitVariableDeclaration(result,
- /*trailingSemicolon=*/false);
- }
-
- // Emit a variable declaration.
- if (failed(emitter.emitAssignPrefix(*operation)))
- return failure();
- return emitter.emitAttribute(operation->getLoc(), value);
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- emitc::ConstantOp constantOp) {
- Operation *operation = constantOp.getOperation();
- Attribute value = constantOp.getValue();
-
- return printConstantOp(emitter, operation, value);
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- emitc::VariableOp variableOp) {
- Operation *operation = variableOp.getOperation();
- Attribute value = variableOp.getValue();
-
- return printConstantOp(emitter, operation, value);
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- mlir::func::ConstantOp constantOp) {
- Operation *operation = constantOp.getOperation();
- Attribute value = constantOp.getValueAttr();
-
- return printConstantOp(emitter, operation, value);
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- cf::BranchOp branchOp) {
- raw_ostream &os = emitter.ostream();
- Block &successor = *branchOp.getSuccessor();
-
- for (auto [operand, arg] :
- llvm::zip_equal(branchOp.getOperands(), successor.getArguments())) {
- os << emitter.getOrCreateName(arg) << " = "
- << 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 printOperation(CppEmitter &emitter,
- cf::CondBranchOp condBranchOp) {
- raw_ostream &os = emitter.ostream();
- Block &trueSuccessor = *condBranchOp.getTrueDest();
- Block &falseSuccessor = *condBranchOp.getFalseDest();
-
- os << "if (" << emitter.getOrCreateName(condBranchOp.getCondition())
- << ") {\n";
-
- // If condition is true.
- for (auto [operand, arg] : llvm::zip_equal(condBranchOp.getTrueOperands(),
- trueSuccessor.getArguments())) {
- os << emitter.getOrCreateName(arg) << " = "
- << 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 [operand, arg] : llvm::zip_equal(condBranchOp.getFalseOperands(),
- falseSuccessor.getArguments())) {
- os << emitter.getOrCreateName(arg) << " = "
- << 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 printOperation(CppEmitter &emitter, mlir::func::CallOp callOp) {
- if (failed(emitter.emitAssignPrefix(*callOp.getOperation())))
- return failure();
-
- raw_ostream &os = emitter.ostream();
- os << callOp.getCallee() << "(";
- if (failed(emitter.emitOperands(*callOp.getOperation())))
- return failure();
- os << ")";
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, emitc::CallOpaqueOp callOp) {
- raw_ostream &os = emitter.ostream();
- Operation &op = *callOp.getOperation();
-
- if (failed(emitter.emitAssignPrefix(op)))
- return failure();
- os << callOp.getCallee();
-
- auto emitArgs = [&](Attribute attr) -> LogicalResult {
- if (auto t = llvm::dyn_cast<IntegerAttr>(attr)) {
- // Index attributes are treated specially as operand index.
- if (t.getType().isIndex()) {
- int64_t 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.getLoc(), attr)))
- return failure();
-
- return success();
- };
-
- if (callOp.getTemplateArgs()) {
- os << "<";
- if (failed(interleaveCommaWithError(*callOp.getTemplateArgs(), os, emitArgs)))
- return failure();
- os << ">";
- }
-
- os << "(";
-
- LogicalResult emittedArgs =
- callOp.getArgs() ? interleaveCommaWithError(*callOp.getArgs(), os, emitArgs)
- : emitter.emitOperands(op);
- if (failed(emittedArgs))
- return failure();
- os << ")";
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- emitc::ApplyOp applyOp) {
- raw_ostream &os = emitter.ostream();
- Operation &op = *applyOp.getOperation();
-
- if (failed(emitter.emitAssignPrefix(op)))
- return failure();
- os << applyOp.getApplicableOperator();
- os << emitter.getOrCreateName(applyOp.getOperand());
-
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, emitc::CastOp castOp) {
- raw_ostream &os = emitter.ostream();
- Operation &op = *castOp.getOperation();
-
- if (failed(emitter.emitAssignPrefix(op)))
- return failure();
- os << "(";
- if (failed(emitter.emitType(op.getLoc(), op.getResult(0).getType())))
- return failure();
- os << ") ";
- os << emitter.getOrCreateName(castOp.getOperand());
-
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter,
- emitc::IncludeOp includeOp) {
- raw_ostream &os = emitter.ostream();
-
- os << "#include ";
- if (includeOp.getIsStandardInclude())
- os << "<" << includeOp.getInclude() << ">";
- else
- os << "\"" << includeOp.getInclude() << "\"";
-
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, scf::ForOp forOp) {
-
- raw_indented_ostream &os = emitter.ostream();
-
- OperandRange operands = forOp.getInitArgs();
- Block::BlockArgListType iterArgs = forOp.getRegionIterArgs();
- Operation::result_range results = forOp.getResults();
-
- if (!emitter.shouldDeclareVariablesAtTop()) {
- for (OpResult result : results) {
- if (failed(emitter.emitVariableDeclaration(result,
- /*trailingSemicolon=*/true)))
- return failure();
- }
- }
-
- for (auto [iterArg, operand] : llvm::zip_equal(iterArgs, operands)) {
- if (failed(emitter.emitType(forOp.getLoc(), iterArg.getType())))
- return failure();
- os << " " << emitter.getOrCreateName(iterArg) << " = ";
- os << emitter.getOrCreateName(operand) << ";";
- os << "\n";
- }
-
- os << "for (";
- if (failed(
- emitter.emitType(forOp.getLoc(), forOp.getInductionVar().getType())))
- return failure();
- os << " ";
- os << emitter.getOrCreateName(forOp.getInductionVar());
- os << " = ";
- os << emitter.getOrCreateName(forOp.getLowerBound());
- os << "; ";
- os << emitter.getOrCreateName(forOp.getInductionVar());
- os << " < ";
- os << emitter.getOrCreateName(forOp.getUpperBound());
- os << "; ";
- os << emitter.getOrCreateName(forOp.getInductionVar());
- os << " += ";
- os << emitter.getOrCreateName(forOp.getStep());
- os << ") {\n";
- os.indent();
-
- Region &forRegion = forOp.getRegion();
- 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();
- }
-
- Operation *yieldOp = forRegion.getBlocks().front().getTerminator();
- // Copy yield operands into iterArgs at the end of a loop iteration.
- for (auto [iterArg, operand] : llvm::zip_equal(iterArgs, yieldOp->getOperands())) {
- os << emitter.getOrCreateName(iterArg) << " = "
- << emitter.getOrCreateName(operand) << ";\n";
- }
-
- os.unindent() << "}";
-
- // Copy iterArgs into results after the for loop.
- for (auto [result, iterArg] : llvm::zip_equal(results, iterArgs)) {
- os << "\n"
- << emitter.getOrCreateName(result) << " = "
- << emitter.getOrCreateName(iterArg) << ";";
- }
-
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, scf::IfOp ifOp) {
- raw_indented_ostream &os = emitter.ostream();
-
- if (!emitter.shouldDeclareVariablesAtTop()) {
- for (OpResult result : ifOp.getResults()) {
- if (failed(emitter.emitVariableDeclaration(result,
- /*trailingSemicolon=*/true)))
- return failure();
- }
- }
-
- os << "if (";
- if (failed(emitter.emitOperands(*ifOp.getOperation())))
- return failure();
- os << ") {\n";
- os.indent();
-
- Region &thenRegion = ifOp.getThenRegion();
- for (Operation &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.unindent() << "}";
-
- Region &elseRegion = ifOp.getElseRegion();
- if (!elseRegion.empty()) {
- os << " else {\n";
- os.indent();
-
- for (Operation &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.unindent() << "}";
- }
-
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, scf::YieldOp yieldOp) {
- raw_ostream &os = emitter.ostream();
- Operation &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_equal(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 printOperation(CppEmitter &emitter, func::ReturnOp returnOp) {
- raw_ostream &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 printOperation(CppEmitter &emitter, ModuleOp moduleOp) {
- CppEmitter::Scope scope(emitter);
-
- for (Operation &op : moduleOp) {
- if (failed(emitter.emitOperation(op, /*trailingSemicolon=*/false)))
- return failure();
- }
- return success();
-}
-
-static LogicalResult printOperation(CppEmitter &emitter, func::FuncOp functionOp) {
- // We need to declare variables at top if the function has multiple blocks.
- if (!emitter.shouldDeclareVariablesAtTop() &&
- functionOp.getBlocks().size() > 1) {
- return functionOp.emitOpError(
- "with multiple blocks needs variables declared at top");
- }
-
- CppEmitter::Scope scope(emitter);
- raw_indented_ostream &os = emitter.ostream();
- if (failed(emitter.emitTypes(functionOp.getLoc(),
- functionOp.getFunctionType().getResults())))
- return failure();
- os << " " << functionOp.getName();
-
- os << "(";
- if (failed(interleaveCommaWithError(
- functionOp.getArguments(), os,
- [&](BlockArgument arg) -> LogicalResult {
- if (failed(emitter.emitType(functionOp.getLoc(), arg.getType())))
- return failure();
- os << " " << emitter.getOrCreateName(arg);
- return success();
- })))
- return failure();
- os << ") {\n";
- os.indent();
- if (emitter.shouldDeclareVariablesAtTop()) {
- // Declare all variables that hold op results including those from nested
- // regions.
- WalkResult result =
- functionOp.walk<WalkOrder::PreOrder>([&](Operation *op) -> WalkResult {
- for (OpResult 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();
- }
-
- Region::BlockListType &blocks = functionOp.getBlocks();
- // Create label names for basic blocks.
- for (Block &block : blocks) {
- emitter.getOrCreateName(block);
- }
-
- // Declare variables for basic block arguments.
- for (auto it = std::next(blocks.begin()); it != blocks.end(); ++it) {
- Block &block = *it;
- for (BlockArgument &arg : block.getArguments()) {
- if (emitter.hasValueInScope(arg))
- return functionOp.emitOpError(" block argument #")
- << arg.getArgNumber() << " is out of scope";
- if (failed(
- emitter.emitType(block.getParentOp()->getLoc(), arg.getType()))) {
- return failure();
- }
- os << " " << emitter.getOrCreateName(arg) << ";\n";
- }
- }
-
- for (Block &block : blocks) {
- // Only print a label if there is more than one block.
- if (blocks.size() > 1 && !block.hasNoPredecessors()) {
- if (failed(emitter.emitLabel(block)))
- return failure();
- }
- for (Operation &op : block.getOperations()) {
- // When generating code for an scf.if or cf.cond_br op no semicolon needs
- // to be printed after the closing brace.
- // When generating code for an scf.for op, printing a trailing semicolon
- // is handled within the printOperation function.
- bool trailingSemicolon = !isa<scf::IfOp, scf::ForOp, cf::CondBranchOp>(op);
-
- if (failed(emitter.emitOperation(
- op, /*trailingSemicolon=*/trailingSemicolon)))
- return failure();
- }
- }
- os.unindent() << "}\n";
- return success();
-}
-
-CppEmitter::CppEmitter(raw_ostream &os, bool declareVariablesAtTop)
- : os(os), declareVariablesAtTop(declareVariablesAtTop) {
- valueInScopeCount.push(0);
- labelInScopeCount.push(0);
-}
-
-/// Return the existing or a new name for a Value.
-StringRef CppEmitter::getOrCreateName(Value val) {
- if (!valueMapper.count(val))
- valueMapper.insert(val, formatv("v{0}", ++valueInScopeCount.top()));
- return *valueMapper.begin(val);
-}
-
-/// Return the existing or a new label 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::shouldMapToUnsigned(IntegerType::SignednessSemantics val) {
- switch (val) {
- case IntegerType::Signless:
- return false;
- case IntegerType::Signed:
- return false;
- case IntegerType::Unsigned:
- return true;
- default:
- assert(false && "unsupported IntegerType");
- return false;
- }
-}
-
-bool CppEmitter::hasValueInScope(Value val) { return valueMapper.count(val); }
-
-bool CppEmitter::hasBlockLabel(Block &block) {
- return blockMapper.count(&block);
-}
-
-LogicalResult CppEmitter::emitAttribute(Location loc, Attribute attr) {
- auto printInt = [&](APInt val, bool isUnsigned) {
- if (val.getBitWidth() == 1) {
- if (val.getBoolValue())
- os << "true";
- else
- os << "false";
- } else {
- SmallString<128> strValue;
- val.toString(strValue, 10, !isUnsigned, false);
- os << strValue;
- }
- };
-
- 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 = llvm::dyn_cast<FloatAttr>(attr)) {
- printFloat(fAttr.getValue());
- return success();
- }
- if (auto dense = llvm::dyn_cast<DenseFPElementsAttr>(attr)) {
- os << '{';
- interleaveComma(dense, os, [&](APFloat val) { printFloat(val); });
- os << '}';
- return success();
- }
-
- // Print integer attributes.
- if (auto iAttr = llvm::dyn_cast<IntegerAttr>(attr)) {
- if (auto iType = llvm::dyn_cast<IntegerType>(iAttr.getType())) {
- printInt(iAttr.getValue(), shouldMapToUnsigned(iType.getSignedness()));
- return success();
- }
- if (auto iType = llvm::dyn_cast<IndexType>(iAttr.getType())) {
- printInt(iAttr.getValue(), false);
- return success();
- }
- }
- if (auto dense = llvm::dyn_cast<DenseIntElementsAttr>(attr)) {
- if (auto iType = llvm::dyn_cast<IntegerType>(dense.getType()
- .cast<TensorType>()
- .getElementType()
- )) {
- os << '{';
- interleaveComma(dense, os, [&](APInt val) {
- printInt(val, shouldMapToUnsigned(iType.getSignedness()));
- });
- os << '}';
- return success();
- }
- if (auto iType = llvm::dyn_cast<IndexType>(dense.getType()
- .cast<TensorType>()
- .getElementType()
- )) {
- os << '{';
- interleaveComma(dense, os, [&](APInt val) { printInt(val, false); });
- os << '}';
- return success();
- }
- }
-
- // Print opaque attributes.
- if (auto oAttr = llvm::dyn_cast<emitc::OpaqueAttr>(attr)) {
- os << oAttr.getValue();
- return success();
- }
-
- // Print symbolic reference attributes.
- if (auto sAttr = llvm::dyn_cast<SymbolRefAttr>(attr)) {
- if (sAttr.getNestedReferences().size() > 1)
- return emitError(loc, "attribute has more than 1 nested reference");
- os << sAttr.getRootReference().getValue();
- return success();
- }
-
- // Print type attributes.
- if (auto type = llvm::dyn_cast<TypeAttr>(attr))
- return emitType(loc, type.getValue());
-
- return emitError(loc, "cannot emit attribute: ") << attr;
-}
-
-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<StringRef> exclude) {
- if (failed(emitOperands(op)))
- return failure();
- // Insert comma in between operands and non-filtered attributes if needed.
- if (op.getNumOperands() > 0) {
- for (NamedAttribute attr : op.getAttrs()) {
- if (!llvm::is_contained(exclude, attr.getName().strref())) {
- os << ", ";
- break;
- }
- }
- }
- // Emit attributes.
- auto emitNamedAttribute = [&](NamedAttribute attr) -> LogicalResult {
- if (llvm::is_contained(exclude, attr.getName().strref()))
- return success();
- os << "/* " << attr.getName() << " */";
- if (failed(emitAttribute(op.getLoc(), attr.getValue())))
- 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()->getLoc(), 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: {
- OpResult result = op.getResult(0);
- if (shouldDeclareVariablesAtTop()) {
- if (failed(emitVariableAssignment(result)))
- return failure();
- } else {
- if (failed(emitVariableDeclaration(result, /*trailingSemicolon=*/false)))
- return failure();
- os << " = ";
- }
- break;
- }
- default:
- if (!shouldDeclareVariablesAtTop()) {
- for (OpResult 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();
-}
-
-LogicalResult CppEmitter::emitOperation(Operation &op, bool trailingSemicolon) {
- LogicalResult status =
- llvm::TypeSwitch<Operation *, LogicalResult>(&op)
- // EmitC ops.
- .Case<emitc::ApplyOp, emitc::CallOpaqueOp, emitc::CastOp, emitc::ConstantOp,
- emitc::IncludeOp, emitc::VariableOp>(
- [&](auto op) { return printOperation(*this, op); })
- // SCF ops.
- .Case<scf::ForOp, scf::IfOp, scf::YieldOp>(
- [&](auto op) { return printOperation(*this, op); })
- // Standard ops.
- .Case<cf::BranchOp, mlir::func::CallOp, cf::CondBranchOp, mlir::func::ConstantOp,
- func::FuncOp, ModuleOp, func::ReturnOp>(
- [&](auto op) { return printOperation(*this, op); })
- .Default([&](Operation *) {
- return op.emitOpError("unable to find printer for op");
- });
-
- if (failed(status))
- return failure();
- os << (trailingSemicolon ? ";\n" : "\n");
- return success();
-}
-
-LogicalResult CppEmitter::emitType(Location loc, Type type) {
- if (auto iType = llvm::dyn_cast<IntegerType>(type)) {
- switch (iType.getWidth()) {
- case 1:
- return (os << "bool"), success();
- case 8:
- case 16:
- case 32:
- case 64:
- if (shouldMapToUnsigned(iType.getSignedness()))
- return (os << "uint" << iType.getWidth() << "_t"), success();
- else
- return (os << "int" << iType.getWidth() << "_t"), success();
- default:
- return emitError(loc, "cannot emit integer type ") << type;
- }
- }
- if (auto fType = llvm::dyn_cast<FloatType>(type)) {
- switch (fType.getWidth()) {
- case 32:
- return (os << "float"), success();
- case 64:
- return (os << "double"), success();
- default:
- return emitError(loc, "cannot emit float type ") << type;
- }
- }
- if (auto iType = llvm::dyn_cast<IndexType>(type))
- return (os << "size_t"), success();
- if (auto tType = llvm::dyn_cast<TensorType>(type)) {
- if (!tType.hasRank())
- return emitError(loc, "cannot emit unranked tensor type");
- if (!tType.hasStaticShape())
- return emitError(loc, "cannot emit tensor type with non static shape");
- os << "Tensor<";
- if (failed(emitType(loc, tType.getElementType())))
- return failure();
- auto shape = tType.getShape();
- for (auto dimSize : shape) {
- os << ", ";
- os << dimSize;
- }
- os << ">";
- return success();
- }
- if (auto tType = llvm::dyn_cast<TupleType>(type))
- return emitTupleType(loc, tType.getTypes());
- if (auto oType = llvm::dyn_cast<emitc::OpaqueType>(type)) {
- os << oType.getValue();
- return success();
- }
- if (auto pType = llvm::dyn_cast<emitc::PointerType>(type)) {
- if (failed(emitType(loc, pType.getPointee())))
- return failure();
- os << "*";
- return success();
- }
- return emitError(loc, "cannot emit type ") << type;
-}
-
-LogicalResult CppEmitter::emitTypes(Location loc, ArrayRef<Type> types) {
- switch (types.size()) {
- case 0:
- os << "void";
- return success();
- case 1:
- return emitType(loc, types.front());
- default:
- return emitTupleType(loc, types);
- }
-}
-
-LogicalResult CppEmitter::emitTupleType(Location loc, ArrayRef<Type> types) {
- os << "std::tuple<";
- if (failed(interleaveCommaWithError(
- types, os, [&](Type type) { return emitType(loc, type); })))
- return failure();
- os << ">";
- return success();
-}
-
-LogicalResult emitc::translateToCpp(Operation *op, raw_ostream &os,
- bool declareVariablesAtTop) {
- CppEmitter emitter(os, declareVariablesAtTop);
- return emitter.emitOperation(*op, /*trailingSemicolon=*/false);
-}
-// clang-format on
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/add.mlir b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/add.mlir
index d70a583..20aa594 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/add.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/add.mlir
@@ -2,7 +2,8 @@
// CHECK: #include "iree/vm/ops.h"
vm.module @add_module {
- // CHECK: static iree_status_t add_module_add_1(iree_vm_stack_t* v1, add_module_t* v2, add_module_state_t* v3, int32_t v4, int32_t v5, int32_t* v6, int32_t* v7) {
+ // TODO(simon-camp): Add back check for static modifier
+ // CHECK: iree_status_t add_module_add_1(iree_vm_stack_t* v1, struct add_module_t* v2, struct add_module_state_t* v3, int32_t v4, int32_t v5, int32_t* v6, int32_t* v7) {
vm.func @add_1(%arg0 : i32, %arg1 : i32) -> (i32, i32) {
// CHECK-NEXT: int32_t v8;
// CHECK-NEXT: int32_t v9;
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/calling_convention.mlir b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/calling_convention.mlir
index 54881da..d8e8861 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/calling_convention.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/calling_convention.mlir
@@ -1,8 +1,10 @@
// RUN: iree-compile --compile-mode=vm --output-format=vm-c %s | FileCheck %s
+// TODO(simon-camp): Add back check for static modifiers
+
// CHECK: #include "iree/vm/ops.h"
vm.module @calling_convention_test {
- // CHECK: static iree_status_t calling_convention_test_no_in_no_return(iree_vm_stack_t* v1, calling_convention_test_t* v2, calling_convention_test_state_t* v3) {
+ // CHECK: iree_status_t calling_convention_test_no_in_no_return(iree_vm_stack_t* v1, struct calling_convention_test_t* v2, struct calling_convention_test_state_t* v3) {
vm.func @no_in_no_return() -> () {
// CHECK-NEXT: iree_status_t v4;
// CHECK-NEXT: v4 = iree_ok_status();
@@ -10,7 +12,7 @@
vm.return
}
- // CHECK: static iree_status_t calling_convention_test_i32_in_no_return(iree_vm_stack_t* v1, calling_convention_test_t* v2, calling_convention_test_state_t* v3, int32_t v4) {
+ // CHECK: iree_status_t calling_convention_test_i32_in_no_return(iree_vm_stack_t* v1, struct calling_convention_test_t* v2, struct calling_convention_test_state_t* v3, int32_t v4) {
vm.func @i32_in_no_return(%arg0 : i32) -> () {
// CHECK-NEXT: iree_status_t v5;
// CHECK-NEXT: v5 = iree_ok_status();
@@ -18,7 +20,7 @@
vm.return
}
- // CHECK: static iree_status_t calling_convention_test_no_in_i32_return(iree_vm_stack_t* v1, calling_convention_test_t* v2, calling_convention_test_state_t* v3, int32_t* v4) {
+ // CHECK: iree_status_t calling_convention_test_no_in_i32_return(iree_vm_stack_t* v1, struct calling_convention_test_t* v2, struct calling_convention_test_state_t* v3, int32_t* v4) {
vm.func @no_in_i32_return() -> (i32) {
// CHECK-NEXT: int32_t v5;
// CHECK-NEXT: iree_status_t v6;
@@ -30,7 +32,7 @@
vm.return %0 : i32
}
- // CHECK: static iree_status_t calling_convention_test_i32_in_i32_return(iree_vm_stack_t* v1, calling_convention_test_t* v2, calling_convention_test_state_t* v3, int32_t v4, int32_t* v5) {
+ // CHECK: iree_status_t calling_convention_test_i32_in_i32_return(iree_vm_stack_t* v1, struct calling_convention_test_t* v2, struct calling_convention_test_state_t* v3, int32_t v4, int32_t* v5) {
vm.func @i32_in_i32_return(%arg0 : i32) -> (i32) {
// CHECK-NEXT: int32_t v6;
// CHECK-NEXT: iree_status_t v7;
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/constant_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/constant_ops.mlir
index f79cf0c..3f58fe8 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/constant_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/constant_ops.mlir
@@ -8,12 +8,12 @@
// Check the generated state struct
// CHECK-LABEL: struct rodata_ops_state_t {
- // CHECK-NEXT: iree_allocator_t allocator;
- // CHECK-NEXT: uint8_t rwdata[1];
- // CHECK-NEXT: iree_vm_ref_t refs[1];
- // CHECK-NEXT: iree_vm_buffer_t rodata_buffers[2];
- // CHECK-NEXT: iree_vm_function_t imports[1];
- // CHECK-NEXT: };
+ // CHECK-SAME: iree_allocator_t allocator;
+ // CHECK-SAME: uint8_t rwdata[1];
+ // CHECK-SAME: iree_vm_ref_t refs[1];
+ // CHECK-SAME: iree_vm_buffer_t rodata_buffers[2];
+ // CHECK-SAME: iree_vm_function_t imports[1];
+ // CHECK-SAME: };
// We mark the rodata ops public in this test to explicitly prevent DCE from
// deleting them.
@@ -46,7 +46,7 @@
// -----
vm.module @constant_ops {
- // CHECK-LABEL: constant_ops_neg_constant
+ // CHECK: constant_ops_neg_constant([[ARGS:[^)]*]]) {
vm.func @neg_constant() -> i32 {
// CHECK: int32_t [[CONST:[^ ]*]];
// CHECK-NOT: [[CONST]] = 4294967292;
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/control_flow.mlir b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/control_flow.mlir
index 3e76198..c00e50d 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/control_flow.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/control_flow.mlir
@@ -11,7 +11,8 @@
vm.return %e : i32
}
}
-// CHECK: static iree_status_t control_flow_module_control_flow_test(iree_vm_stack_t* v1, control_flow_module_t* v2, control_flow_module_state_t* v3, int32_t [[A:[^ ]*]], int32_t [[COND:[^ ]*]], int32_t* [[RESULT:[^ ]*]]) {
+// TODO(simon-camp): Add back check for static modifier
+// CHECK: iree_status_t control_flow_module_control_flow_test(iree_vm_stack_t* v1, struct control_flow_module_t* v2, struct control_flow_module_state_t* v3, int32_t [[A:[^ ]*]], int32_t [[COND:[^ ]*]], int32_t* [[RESULT:[^ ]*]]) {
// CHECK-NEXT: int32_t [[COND_NZ:[^ ]*]];
// CHECK-NEXT: bool [[COND_BOOL:[^ ]*]];
// CHECK-NEXT: int32_t [[B:[^ ]*]];
diff --git a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/global_ops.mlir b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/global_ops.mlir
index ddb731d..fd56741 100644
--- a/compiler/src/iree/compiler/Dialect/VM/Target/C/test/global_ops.mlir
+++ b/compiler/src/iree/compiler/Dialect/VM/Target/C/test/global_ops.mlir
@@ -1,23 +1,22 @@
// RUN: iree-compile --compile-mode=vm --output-format=vm-c --iree-vm-c-module-optimize=false %s | FileCheck %s
+// TODO(simon-camp): Add back check for static modifiers
+
vm.module @global_ops {
// check the generated state struct
// CHECK-LABEL: struct global_ops_state_t {
- // CHECK-NEXT: iree_allocator_t allocator;
- // CHECK-NEXT: uint8_t rwdata[8];
- // CHECK-NEXT: iree_vm_ref_t refs[1];
- // CHECK-NEXT: iree_vm_buffer_t rodata_buffers[1];
- // CHECK-NEXT: iree_vm_function_t imports[1];
- // CHECK-NEXT: };
+ // CHECK-SAME: iree_allocator_t allocator;
+ // CHECK-SAME: uint8_t rwdata[8];
+ // CHECK-SAME: iree_vm_ref_t refs[1];
+ // CHECK-SAME: iree_vm_buffer_t rodata_buffers[1];
+ // CHECK-SAME: iree_vm_function_t imports[1];
+ // CHECK-SAME: };
vm.global.i32 mutable @c42 = 42 : i32
vm.global.i32 mutable @c107_mut = 107 : i32
- // Skip forward declarations
- // CHECK: DEFINE FUNCTIONS
-
vm.export @test_global_load_i32
- // CHECK: static iree_status_t global_ops_test_global_load_i32(
+ // CHECK: iree_status_t global_ops_test_global_load_i32([[ARGS:[^)]*]]) {
vm.func @test_global_load_i32() -> i32 {
// CHECK-NEXT: uint8_t* v5;
// CHECK-NEXT: int32_t v6;
@@ -29,7 +28,7 @@
}
vm.export @test_global_store_i32
- // CHECK: static iree_status_t global_ops_test_global_store_i32(
+ // CHECK: iree_status_t global_ops_test_global_store_i32([[ARGS:[^)]*]]) {
vm.func @test_global_store_i32() -> i32 {
// CHECK-NEXT: int32_t v5;
// CHECK-NEXT: uint8_t* v6;
diff --git a/runtime/src/iree/vm/ops_emitc.h b/runtime/src/iree/vm/ops_emitc.h
index 4ae8c8e..1bacddd 100644
--- a/runtime/src/iree/vm/ops_emitc.h
+++ b/runtime/src/iree/vm/ops_emitc.h
@@ -40,12 +40,6 @@
#define EMITC_STRUCT_PTR_MEMBER_ASSIGN(struct, member, value) \
(struct)->member = (value)
-// Create a typdef struct
-#define EMITC_TYPEDEF_STRUCT(typename, body) \
- typedef struct { \
- body \
- } typename;
-
// Get an array element
#define EMITC_ARRAY_ELEMENT(array, index) (array)[index]