Commit 310fcf07 authored by mbencer's avatar mbencer

Merge remote-tracking branch 'origin/master' into mbencer/BuilderSplitV1

parents b4b352d5 6c0cf85a
...@@ -18,12 +18,12 @@ include(ExternalProject) ...@@ -18,12 +18,12 @@ include(ExternalProject)
# Includes blas 3.8.0 in mkldnn # Includes blas 3.8.0 in mkldnn
set(NGRAPH_MKLDNN_SHORT_VERSION 1) set(NGRAPH_MKLDNN_SHORT_VERSION 1)
set(NGRAPH_MKLDNN_FULL_VERSION 1.1.1.0) set(NGRAPH_MKLDNN_FULL_VERSION 1.2.0.0)
set(NGRAPH_MKLDNN_MKLML_ASSET_VERSION "v0.21") set(NGRAPH_MKLDNN_MKLML_ASSET_VERSION "v0.21")
set(NGRAPH_MKLDNN_VERSION "v1.1.1") set(NGRAPH_MKLDNN_VERSION "v1.2")
set(NGRAPH_MKLDNN_MKLML_VERSION "2019.0.5.20190502") set(NGRAPH_MKLDNN_MKLML_VERSION "2019.0.5.20190502")
set(NGRAPH_MKLDNN_MKLML_WIN32_VERSION "2020.0.20190813") set(NGRAPH_MKLDNN_MKLML_WIN32_VERSION "2020.0.20190813")
set(NGRAPH_MKLDNN_GIT_TAG "v1.1.1") set(NGRAPH_MKLDNN_GIT_TAG "v1.2")
#------------------------------------------------------------------------------ #------------------------------------------------------------------------------
# Fetch and install MKL-DNN # Fetch and install MKL-DNN
......
...@@ -19,7 +19,7 @@ include(ExternalProject) ...@@ -19,7 +19,7 @@ include(ExternalProject)
set(MLIR_LLVM_REPO_URL https://github.com/llvm/llvm-project.git) set(MLIR_LLVM_REPO_URL https://github.com/llvm/llvm-project.git)
# Change these commit IDs to move to latest stable versions # Change these commit IDs to move to latest stable versions
set(MLIR_LLVM_COMMIT_ID 96400ae) set(MLIR_LLVM_COMMIT_ID 376c6853)
# MLIR environment variables. Some of them are used by LIT tool. # MLIR environment variables. Some of them are used by LIT tool.
......
...@@ -26,7 +26,6 @@ ...@@ -26,7 +26,6 @@
#include <llvm/ADT/DenseSet.h> #include <llvm/ADT/DenseSet.h>
#include <map> #include <map>
#include <mlir/EDSC/Builders.h> #include <mlir/EDSC/Builders.h>
#include <mlir/EDSC/Helpers.h>
#include <mlir/EDSC/Intrinsics.h> #include <mlir/EDSC/Intrinsics.h>
#include <mlir/IR/AffineExpr.h> #include <mlir/IR/AffineExpr.h>
#include <mlir/IR/IntegerSet.h> #include <mlir/IR/IntegerSet.h>
......
...@@ -194,7 +194,8 @@ void MLIRCPUBackend::lowerNgDialect() ...@@ -194,7 +194,8 @@ void MLIRCPUBackend::lowerNgDialect()
void MLIRCPUBackend::lowerStandardDialect() void MLIRCPUBackend::lowerStandardDialect()
{ {
mlir::PassManager pm(&m_context); mlir::PassManager pm(&m_context);
pm.addPass(mlir::createLowerToLLVMPass()); pm.addPass(mlir::createLowerToLLVMPass(
/*useAlloca=*/false, /*useBarePtrCallConv=*/false, /*emitCWrappers=*/true));
// Apply any generic pass manager command line options. // Apply any generic pass manager command line options.
mlir::applyPassManagerCLOptions(pm); mlir::applyPassManagerCLOptions(pm);
......
...@@ -23,9 +23,7 @@ ...@@ -23,9 +23,7 @@
#include "contrib/mlir/core/ngraph_dialect/type.hpp" #include "contrib/mlir/core/ngraph_dialect/type.hpp"
#include <llvm/IR/Module.h> #include <llvm/IR/Module.h>
#include <mlir/EDSC/Builders.h> #include <mlir/Dialect/AffineOps/EDSC/Builders.h>
#include <mlir/EDSC/Helpers.h>
#include <mlir/EDSC/Intrinsics.h>
#include <mlir/IR/IntegerSet.h> #include <mlir/IR/IntegerSet.h>
#include <mlir/IR/MLIRContext.h> #include <mlir/IR/MLIRContext.h>
#include <mlir/IR/StandardTypes.h> #include <mlir/IR/StandardTypes.h>
......
...@@ -719,7 +719,7 @@ static void __mlir_cblas_sgemm_with_bias(StaticMemRef* memRefmatA, ...@@ -719,7 +719,7 @@ static void __mlir_cblas_sgemm_with_bias(StaticMemRef* memRefmatA,
} }
} }
extern "C" void __mlir_callback_1_input(void* input, void* output, size_t index, OpType type) extern "C" void _mlir_ciface_callback_1_input(void* input, void* output, size_t index, OpType type)
{ {
auto unrankedMemRefInput = reinterpret_cast<UnrankedMemRef*>(input); auto unrankedMemRefInput = reinterpret_cast<UnrankedMemRef*>(input);
auto unrankedMemRefOutput = reinterpret_cast<UnrankedMemRef*>(output); auto unrankedMemRefOutput = reinterpret_cast<UnrankedMemRef*>(output);
...@@ -752,8 +752,8 @@ extern "C" void __mlir_callback_1_input(void* input, void* output, size_t index, ...@@ -752,8 +752,8 @@ extern "C" void __mlir_callback_1_input(void* input, void* output, size_t index,
} }
} }
extern "C" void extern "C" void _mlir_ciface_callback_2_inputs(
__mlir_callback_2_inputs(void* input0, void* input1, void* output, size_t index, OpType type) void* input0, void* input1, void* output, size_t index, OpType type)
{ {
auto unrankedMemRefInput0 = reinterpret_cast<UnrankedMemRef*>(input0); auto unrankedMemRefInput0 = reinterpret_cast<UnrankedMemRef*>(input0);
auto unrankedMemRefInput1 = reinterpret_cast<UnrankedMemRef*>(input1); auto unrankedMemRefInput1 = reinterpret_cast<UnrankedMemRef*>(input1);
...@@ -780,7 +780,7 @@ extern "C" void ...@@ -780,7 +780,7 @@ extern "C" void
} }
} }
extern "C" void __mlir_callback_3_inputs( extern "C" void _mlir_ciface_callback_3_inputs(
void* input0, void* input1, void* input2, void* output, size_t index, OpType type) void* input0, void* input1, void* input2, void* output, size_t index, OpType type)
{ {
auto unrankedMemRefInput0 = reinterpret_cast<UnrankedMemRef*>(input0); auto unrankedMemRefInput0 = reinterpret_cast<UnrankedMemRef*>(input0);
......
...@@ -83,7 +83,7 @@ void MLIRCPURuntime::bindArguments(const std::vector<MemRefArg>& args) ...@@ -83,7 +83,7 @@ void MLIRCPURuntime::bindArguments(const std::vector<MemRefArg>& args)
{ {
NGRAPH_CHECK(m_module, "MLIR module is not ready."); NGRAPH_CHECK(m_module, "MLIR module is not ready.");
auto func = m_module->lookupSymbol<mlir::LLVM::LLVMFuncOp>("main"); auto func = m_module->lookupSymbol<mlir::LLVM::LLVMFuncOp>("_mlir_ciface_main");
NGRAPH_CHECK(func && !func.getBlocks().empty(), "Function not found"); NGRAPH_CHECK(func && !func.getBlocks().empty(), "Function not found");
// Set external arguments // Set external arguments
...@@ -127,14 +127,15 @@ void MLIRCPURuntime::execute() ...@@ -127,14 +127,15 @@ void MLIRCPURuntime::execute()
// uniformity reasons, it takes a list of type-erased pointers to arguments. // uniformity reasons, it takes a list of type-erased pointers to arguments.
// Please, note that 'invoke' method is overloaded with a parameter pack version. // Please, note that 'invoke' method is overloaded with a parameter pack version.
// Make sure the MutableArrayRef version is invoked. // Make sure the MutableArrayRef version is invoked.
auto invocationResult = m_engine->invoke("main", llvm::MutableArrayRef<void*>(m_invokeArgs)); auto invocationResult =
m_engine->invoke("_mlir_ciface_main", llvm::MutableArrayRef<void*>(m_invokeArgs));
if (clDumpObjectFile) if (clDumpObjectFile)
{ {
m_engine->dumpToObjectFile(clObjectFilename.empty() ? "jitted_mlir.o" m_engine->dumpToObjectFile(clObjectFilename.empty() ? "jitted_mlir.o"
: clObjectFilename.getValue()); : clObjectFilename.getValue());
} }
NGRAPH_CHECK(!invocationResult, "JIT invocation of 'main' failed\n"); NGRAPH_CHECK(!invocationResult, "JIT invocation of '_mlir_ciface_main' failed\n");
} }
void MLIRCPURuntime::cleanup() void MLIRCPURuntime::cleanup()
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
set(LIBS set(LIBS
mlir_backend mlir_backend
MLIROptMain MLIROptLib
MLIRPass MLIRPass
MLIRParser MLIRParser
LLVMSupport LLVMSupport
......
...@@ -21,10 +21,21 @@ ...@@ -21,10 +21,21 @@
#include "contrib/mlir/core/ngraph_dialect/dialect.hpp" #include "contrib/mlir/core/ngraph_dialect/dialect.hpp"
#include <llvm/Support/CommandLine.h> #include <mlir/Dialect/AffineOps/AffineOps.h>
#include <llvm/Support/Debug.h> #include <mlir/Dialect/LLVMIR/LLVMDialect.h>
#include <mlir/Dialect/LoopOps/LoopOps.h>
#include <mlir/Dialect/StandardOps/Ops.h>
#include <mlir/Dialect/VectorOps/VectorOps.h>
#include <mlir/IR/Dialect.h> #include <mlir/IR/Dialect.h>
#include <mlir/IR/MLIRContext.h> #include <mlir/IR/MLIRContext.h>
#include <mlir/Pass/Pass.h>
#include <mlir/Transforms/LocationSnapshot.h>
#include <mlir/Transforms/Passes.h>
#include <llvm/Support/CommandLine.h>
#include <llvm/Support/Debug.h>
using namespace mlir;
static llvm::cl::opt<bool> clPrintIRAfterAll( static llvm::cl::opt<bool> clPrintIRAfterAll(
"ngraph-print-ir-after-all", "ngraph-print-ir-after-all",
...@@ -35,15 +46,47 @@ static llvm::cl::opt<bool> clPrintIRAfterAll( ...@@ -35,15 +46,47 @@ static llvm::cl::opt<bool> clPrintIRAfterAll(
void ngraph::runtime::ngmlir::initializeNGraphMLIR() void ngraph::runtime::ngmlir::initializeNGraphMLIR()
{ {
// Initialize a dialect only once. // Initialize MLIR dialects and passes only once.
// We currently have no way to query if a dialect is previously static bool init_once = []() {
// registered. So using a global flag instead. // In-tree Dialects.
static bool init = false; registerDialect<AffineOpsDialect>();
if (!init) registerDialect<LLVM::LLVMDialect>();
{ registerDialect<loop::LoopOpsDialect>();
mlir::registerDialect<mlir::NGraphOpsDialect>(); registerDialect<StandardOpsDialect>();
init = true; registerDialect<vector::VectorOpsDialect>();
}
// nGraph dialects.
registerDialect<mlir::NGraphOpsDialect>();
// In-tree passes.
// No-op to avoid DCE on the following pass initializations.
if (std::getenv("bar") != (char*)-1)
return false;
createCanonicalizerPass();
createCSEPass();
createVectorizePass({});
createLoopUnrollPass();
createLoopUnrollAndJamPass();
createSimplifyAffineStructuresPass();
createLoopFusionPass();
createLoopInvariantCodeMotionPass();
createAffineLoopInvariantCodeMotionPass();
createPipelineDataTransferPass();
createLowerAffinePass();
createLoopTilingPass(0);
createLoopCoalescingPass();
createAffineDataCopyGenerationPass(0, 0);
createMemRefDataFlowOptPass();
createStripDebugInfoPass();
createPrintOpStatsPass();
createInlinerPass();
createSymbolDCEPass();
createLocationSnapshotPass({});
return true;
}();
(void)init_once;
} }
void ngraph::runtime::ngmlir::dumpMlirModule(const std::string msg, mlir::ModuleOp module) void ngraph::runtime::ngmlir::dumpMlirModule(const std::string msg, mlir::ModuleOp module)
......
...@@ -171,6 +171,8 @@ add_library(onnx_import STATIC ...@@ -171,6 +171,8 @@ add_library(onnx_import STATIC
op/reshape.hpp op/reshape.hpp
op/reverse_sequence.cpp op/reverse_sequence.cpp
op/reverse_sequence.hpp op/reverse_sequence.hpp
op/round.cpp
op/round.hpp
op/scatter_nd.cpp op/scatter_nd.cpp
op/scatter_nd.hpp op/scatter_nd.hpp
op/selu.cpp op/selu.cpp
......
...@@ -16,7 +16,6 @@ ...@@ -16,7 +16,6 @@
#include "average_pool.hpp" #include "average_pool.hpp"
#include "ngraph/node.hpp" #include "ngraph/node.hpp"
#include "ngraph/op/avg_pool.hpp"
#include "utils/pooling_factory.hpp" #include "utils/pooling_factory.hpp"
namespace ngraph namespace ngraph
...@@ -29,7 +28,7 @@ namespace ngraph ...@@ -29,7 +28,7 @@ namespace ngraph
{ {
NodeVector average_pool(const Node& node) NodeVector average_pool(const Node& node)
{ {
return pooling::PoolingFactory(node).make_avg_pool(); return pooling::LocalPoolingFactory(node).make_avg_pool();
} }
} // namespace set_1 } // namespace set_1
......
...@@ -31,7 +31,7 @@ namespace ngraph ...@@ -31,7 +31,7 @@ namespace ngraph
{ {
NodeVector max_pool(const Node& node) NodeVector max_pool(const Node& node)
{ {
auto max_pool = pooling::PoolingFactory(node).make_max_pool(); auto max_pool = pooling::LocalPoolingFactory(node).make_max_pool();
max_pool.emplace_back(std::make_shared<NullNode>()); // Indices (optional) max_pool.emplace_back(std::make_shared<NullNode>()); // Indices (optional)
return max_pool; return max_pool;
} }
......
...@@ -42,9 +42,9 @@ namespace ngraph ...@@ -42,9 +42,9 @@ namespace ngraph
auto off_on_values = auto off_on_values =
std::make_shared<default_opset::Split>(values, split_axis, 2); std::make_shared<default_opset::Split>(values, split_axis, 2);
auto off_value = auto off_value =
reshape::interpret_as_scalar(get_output_element(off_on_values, 0ul)); reshape::interpret_as_scalar(get_output_element(off_on_values, size_t{0}));
auto on_value = auto on_value =
reshape::interpret_as_scalar(get_output_element(off_on_values, 1ul)); reshape::interpret_as_scalar(get_output_element(off_on_values, size_t{1}));
auto axis = node.get_attribute_value<std::int64_t>("axis", -1); auto axis = node.get_attribute_value<std::int64_t>("axis", -1);
......
...@@ -65,14 +65,19 @@ namespace ngraph ...@@ -65,14 +65,19 @@ namespace ngraph
NodeVector pad(const Node& node) NodeVector pad(const Node& node)
{ {
auto data = node.get_ng_inputs().at(0); auto data = node.get_ng_inputs().at(0);
const Shape& data_shape = data->get_shape();
const auto data_rank =
node.get_ng_inputs().at(0)->get_output_partial_shape(0).rank();
CHECK_VALID_NODE(
node, data_rank.is_static(), "Data rank must be static for pad op");
const auto data_rank_value = static_cast<size_t>(data_rank);
double value = node.get_attribute_value<double>("value", 0); double value = node.get_attribute_value<double>("value", 0);
const std::string mode = const std::string mode =
node.get_attribute_value<std::string>("mode", "constant"); node.get_attribute_value<std::string>("mode", "constant");
ngraph::op::PadMode pad_mode = get_pad_mode(mode); ngraph::op::PadMode pad_mode = get_pad_mode(mode);
auto paddings = convpool::get_pads(node, data_shape); const auto paddings = convpool::get_pads(node, data_rank_value);
ngraph::CoordinateDiff padding_below = paddings.first; ngraph::CoordinateDiff padding_below = paddings.first;
ngraph::CoordinateDiff padding_above = paddings.second; ngraph::CoordinateDiff padding_above = paddings.second;
......
//*****************************************************************************
// Copyright 2017-2020 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//*****************************************************************************
#include <memory>
#include "ngraph/opsets/opset0.hpp"
#include "round.hpp"
namespace ngraph
{
namespace onnx_import
{
namespace op
{
namespace set_1
{
NodeVector round(const Node& node)
{
const std::shared_ptr<ngraph::Node> data{node.get_ng_inputs().at(0)};
return {std::make_shared<ngraph::opset0::Round>(data)};
}
} // namespace set_1
} // namespace op
} // namespace onnx_import
} // namespace ngraph
//*****************************************************************************
// Copyright 2017-2020 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//*****************************************************************************
#pragma once
#include "core/node.hpp"
#include "ngraph/node.hpp"
namespace ngraph
{
namespace onnx_import
{
namespace op
{
namespace set_1
{
NodeVector round(const Node& node);
} // namespace set_1
} // namespace op
} // namespace onnx_import
} // namespace ngraph
...@@ -101,6 +101,7 @@ ...@@ -101,6 +101,7 @@
#include "op/relu.hpp" #include "op/relu.hpp"
#include "op/reshape.hpp" #include "op/reshape.hpp"
#include "op/reverse_sequence.hpp" #include "op/reverse_sequence.hpp"
#include "op/round.hpp"
#include "op/scatter_nd.hpp" #include "op/scatter_nd.hpp"
#include "op/selu.hpp" #include "op/selu.hpp"
#include "op/shape.hpp" #include "op/shape.hpp"
...@@ -334,6 +335,7 @@ namespace ngraph ...@@ -334,6 +335,7 @@ namespace ngraph
REGISTER_OPERATOR("Relu", 1, relu); REGISTER_OPERATOR("Relu", 1, relu);
REGISTER_OPERATOR("Reshape", 1, reshape); REGISTER_OPERATOR("Reshape", 1, reshape);
REGISTER_OPERATOR("ReverseSequence", 1, reverse_sequence); REGISTER_OPERATOR("ReverseSequence", 1, reverse_sequence);
REGISTER_OPERATOR("Round", 1, round);
REGISTER_OPERATOR("ScatterND", 1, scatter_nd); REGISTER_OPERATOR("ScatterND", 1, scatter_nd);
REGISTER_OPERATOR("Selu", 1, selu); REGISTER_OPERATOR("Selu", 1, selu);
REGISTER_OPERATOR("Shape", 1, shape); REGISTER_OPERATOR("Shape", 1, shape);
......
...@@ -38,28 +38,41 @@ namespace ngraph ...@@ -38,28 +38,41 @@ namespace ngraph
namespace detail namespace detail
{ {
Strides get_strides_helper(const Node& node, /// \brief Helper method used to read vector attribute
const std::string& name, /// \note Default value is vector of size spatial dims filled with
const Shape& kernel_shape) /// ones
{ ///
return node.get_attribute_value<std::vector<std::size_t>>( /// \param node Node from which attribute is read
name, std::vector<std::size_t>(kernel_shape.size(), 1UL)); /// \param attr_name Attribute name (such as `strides`, `dilations`)
///
/// \return Read vector attribute if available or default value
std::vector<std::size_t> get_attribute_value(const Node& node,
const std::string& attr_name)
{
if (node.has_attribute(attr_name))
{
return node.get_attribute_value<std::vector<std::size_t>>(attr_name);
} }
} // namespace detail const auto data_rank =
node.get_ng_inputs().at(0)->get_output_partial_shape(0).rank();
Strides get_strides(const Node& node, const Shape& kernel_shape) CHECK_VALID_NODE(node,
{ data_rank.is_static(),
return detail::get_strides_helper(node, "strides", kernel_shape); "If '",
attr_name,
"' is not provided data rank must be static");
const auto data_spatial_dims = static_cast<size_t>(data_rank) - 2;
return std::vector<std::size_t>(data_spatial_dims, 1UL);
} }
} // namespace detail
Strides get_strides(const Node& node) Strides get_strides(const Node& node)
{ {
return get_strides(node, get_kernel_shape(node)); return detail::get_attribute_value(node, "strides");
} }
Strides get_dilations(const Node& node) Strides get_dilations(const Node& node)
{ {
return detail::get_strides_helper(node, "dilations", get_kernel_shape(node)); return detail::get_attribute_value(node, "dilations");
} }
ngraph::op::PadType get_auto_pad(const Node& node) ngraph::op::PadType get_auto_pad(const Node& node)
...@@ -90,16 +103,16 @@ namespace ngraph ...@@ -90,16 +103,16 @@ namespace ngraph
} }
std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node, std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node,
const Shape& kernel_shape) const size_t kernel_rank)
{ {
CoordinateDiff pads(kernel_shape.size(), 0); CoordinateDiff pads(kernel_rank, 0);
if (node.has_attribute("pads")) if (node.has_attribute("pads"))
{ {
auto pads_int64 = node.get_attribute_value<std::vector<int64_t>>("pads"); auto pads_int64 = node.get_attribute_value<std::vector<int64_t>>("pads");
pads = CoordinateDiff{std::begin(pads_int64), std::end(pads_int64)}; pads = CoordinateDiff{std::begin(pads_int64), std::end(pads_int64)};
} }
if (pads.size() == kernel_shape.size() * 2) if (pads.size() == kernel_rank * 2)
{ {
return {{std::begin(pads), std::begin(pads) + pads.size() / 2}, return {{std::begin(pads), std::begin(pads) + pads.size() / 2},
{std::begin(pads) + pads.size() / 2, std::end(pads)}}; {std::begin(pads) + pads.size() / 2, std::end(pads)}};
...@@ -112,6 +125,18 @@ namespace ngraph ...@@ -112,6 +125,18 @@ namespace ngraph
} }
} }
std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node)
{
const auto data_rank =
node.get_ng_inputs().at(0)->get_output_partial_shape(0).rank();
CHECK_VALID_NODE(node,
data_rank.is_static(),
"The rank of node must be static in order to calculate pads");
const auto data_spatial_dims = static_cast<size_t>(data_rank) - 2;
return get_pads(node, data_spatial_dims);
}
void calculate_auto_pads(const Shape& data_shape, void calculate_auto_pads(const Shape& data_shape,
const Shape& filter_shape, const Shape& filter_shape,
const Strides& strides, const Strides& strides,
......
...@@ -33,13 +33,6 @@ namespace ngraph ...@@ -33,13 +33,6 @@ namespace ngraph
/// \return The kernel Shape object representing its dimensions (height, width, depth). /// \return The kernel Shape object representing its dimensions (height, width, depth).
Shape get_kernel_shape(const Node& node); Shape get_kernel_shape(const Node& node);
/// \brief Get number of pixels to stride operation by in each direction.
///
/// \param node The Node ptr representing Conv or Pool operation.
/// \param kernel_shape The shape of the kernel which we retrieve strides for.
/// \return The kernel Shape object representing its dimensions (height, width, depth).
Strides get_strides(const Node& node, const Shape& kernel_shape);
/// \brief Get number of pixels to stride operation by in each direction. /// \brief Get number of pixels to stride operation by in each direction.
/// ///
/// \param node The Node ptr representing Conv or Pool operation. /// \param node The Node ptr representing Conv or Pool operation.
...@@ -59,12 +52,12 @@ namespace ngraph ...@@ -59,12 +52,12 @@ namespace ngraph
/// `pads` value should follow [x1_begin, x2_begin..., x1_end, x2_end,...]. /// `pads` value should follow [x1_begin, x2_begin..., x1_end, x2_end,...].
/// ///
/// \param node The Node ptr representing ONNX operation. /// \param node The Node ptr representing ONNX operation.
/// \param kernel_shape The shape of the kernel which we retrieve pads for. /// \param kernel_rank The rank of the kernel which we retrieve pads for.
/// ///
/// \return A pair of (padding_above, padding_below), which elements contains number of /// \return A pair of (padding_above, padding_below), which elements contains number of
/// pixels to pad in respective dimensions (height, width, depth). /// pixels to pad in respective dimensions (height, width, depth).
std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node, std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node,
const Shape& kernel_shape); const size_t kernel_rank);
/// \brief Get padding values for the operation described by an ONNX node. /// \brief Get padding values for the operation described by an ONNX node.
/// \details Values are taken from the `pads` attribute. /// \details Values are taken from the `pads` attribute.
...@@ -75,11 +68,7 @@ namespace ngraph ...@@ -75,11 +68,7 @@ namespace ngraph
/// ///
/// \return A pair of (padding_above, padding_below), which elements contains number of /// \return A pair of (padding_above, padding_below), which elements contains number of
/// pixels to pad in respective dimensions (height, width, depth). /// pixels to pad in respective dimensions (height, width, depth).
std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node);
inline std::pair<CoordinateDiff, CoordinateDiff> get_pads(const Node& node)
{
return get_pads(node, get_kernel_shape(node));
}
/// ///
/// \brief Calculate paddings with respect to auto_pad value. /// \brief Calculate paddings with respect to auto_pad value.
......
...@@ -17,6 +17,7 @@ ...@@ -17,6 +17,7 @@
#include <iterator> #include <iterator>
#include "default_opset.hpp" #include "default_opset.hpp"
#include "exceptions.hpp"
#include "ngraph/coordinate_diff.hpp" #include "ngraph/coordinate_diff.hpp"
#include "utils/convpool.hpp" #include "utils/convpool.hpp"
#include "utils/pooling_factory.hpp" #include "utils/pooling_factory.hpp"
...@@ -30,12 +31,11 @@ namespace ngraph ...@@ -30,12 +31,11 @@ namespace ngraph
PoolingFactory::PoolingFactory(const Node& node) PoolingFactory::PoolingFactory(const Node& node)
: m_onnx_node{node} : m_onnx_node{node}
, m_inputs{node.get_ng_inputs()} , m_inputs{node.get_ng_inputs()}
, m_kernel_shape{convpool::get_kernel_shape(node)}
, m_strides{convpool::get_strides(node)} , m_strides{convpool::get_strides(node)}
, m_dilations{convpool::get_dilations(node)} , m_dilations{convpool::get_dilations(node)}
, m_auto_pad{convpool::get_auto_pad(node)} , m_auto_pad{convpool::get_auto_pad(node)}
{ {
auto paddings = convpool::get_pads(node); const auto paddings = convpool::get_pads(node);
const CoordinateDiff& padding_above{paddings.second}; const CoordinateDiff& padding_above{paddings.second};
const CoordinateDiff& padding_below{paddings.first}; const CoordinateDiff& padding_below{paddings.first};
m_padding_below = Shape{std::begin(padding_below), std::end(padding_below)}; m_padding_below = Shape{std::begin(padding_below), std::end(padding_below)};
...@@ -44,7 +44,7 @@ namespace ngraph ...@@ -44,7 +44,7 @@ namespace ngraph
NodeVector PoolingFactory::make_avg_pool() const NodeVector PoolingFactory::make_avg_pool() const
{ {
bool count_include_pad = const bool count_include_pad =
m_onnx_node.get_attribute_value<std::int64_t>("count_include_pad", 0); m_onnx_node.get_attribute_value<std::int64_t>("count_include_pad", 0);
return {std::make_shared<default_opset::AvgPool>(m_inputs.at(0), return {std::make_shared<default_opset::AvgPool>(m_inputs.at(0),
m_strides, m_strides,
...@@ -67,13 +67,31 @@ namespace ngraph ...@@ -67,13 +67,31 @@ namespace ngraph
m_auto_pad)}; m_auto_pad)};
} }
LocalPoolingFactory::LocalPoolingFactory(const Node& node)
: PoolingFactory(node)
{
// Kernel shape is required
m_kernel_shape = node.get_attribute_value<std::vector<std::size_t>>("kernel_shape");
}
GlobalPoolingFactory::GlobalPoolingFactory(const Node& node) GlobalPoolingFactory::GlobalPoolingFactory(const Node& node)
: PoolingFactory(node) : PoolingFactory(node)
{ {
// Correct the kernel shape. const auto data_shape = node.get_ng_inputs().at(0)->get_output_partial_shape(0);
const Shape& data_shape{m_inputs.at(0)->get_shape()}; const auto data_rank = data_shape.rank();
CHECK_VALID_NODE(
node, data_rank.is_static(), "Data rank must be static for global pooling ops");
Shape kernel_shape;
for (auto i = 2; i < static_cast<size_t>(data_rank); ++i)
{
CHECK_VALID_NODE(node,
data_shape[i].is_static(),
"All spatial dimensions must be known for global pooling ops");
kernel_shape.emplace_back(static_cast<size_t>(data_shape[i]));
}
// Set shape to all but {N,C} axes. // Set shape to all but {N,C} axes.
m_kernel_shape = Shape{std::next(std::begin(data_shape), 2), std::end(data_shape)}; m_kernel_shape = kernel_shape;
} }
} // namespace pooling } // namespace pooling
} // namespace onnx_import } // namespace onnx_import
......
...@@ -48,7 +48,6 @@ namespace ngraph ...@@ -48,7 +48,6 @@ namespace ngraph
class PoolingFactory class PoolingFactory
{ {
public: public:
explicit PoolingFactory(const Node& node);
virtual ~PoolingFactory() = default; virtual ~PoolingFactory() = default;
/// ///
...@@ -64,6 +63,8 @@ namespace ngraph ...@@ -64,6 +63,8 @@ namespace ngraph
NodeVector make_max_pool() const; NodeVector make_max_pool() const;
protected: protected:
explicit PoolingFactory(const Node& node);
Node m_onnx_node; Node m_onnx_node;
const NodeVector m_inputs; const NodeVector m_inputs;
Shape m_kernel_shape; Shape m_kernel_shape;
...@@ -75,9 +76,20 @@ namespace ngraph ...@@ -75,9 +76,20 @@ namespace ngraph
}; };
/// ///
/// \brief Factory class which generates sub-graphs for ONNX 'global' pooling /// \brief Factory class which generates sub-graphs for ONNX 'local' pooling
/// operators. /// operators.
/// \note Kernel shape attribute is required
class LocalPoolingFactory : public PoolingFactory
{
public:
explicit LocalPoolingFactory(const Node& node);
virtual ~LocalPoolingFactory() = default;
};
/// ///
/// \brief Factory class which generates sub-graphs for ONNX 'global' pooling
/// operators.
/// \note Kernel shape is calculated based on spatial dims
class GlobalPoolingFactory : public PoolingFactory class GlobalPoolingFactory : public PoolingFactory
{ {
public: public:
......
...@@ -130,7 +130,7 @@ void op::v1::Gather::validate_and_infer_types() ...@@ -130,7 +130,7 @@ void op::v1::Gather::validate_and_infer_types()
")."); ").");
} }
auto axis = get_axis(); int64_t axis = get_axis();
if (input_rank.is_static() && axis != AXIS_NOT_SET_VALUE) if (input_rank.is_static() && axis != AXIS_NOT_SET_VALUE)
{ {
NODE_VALIDATION_CHECK(this, NODE_VALIDATION_CHECK(this,
......
...@@ -39,11 +39,19 @@ namespace ngraph ...@@ -39,11 +39,19 @@ namespace ngraph
MatcherState::~MatcherState() MatcherState::~MatcherState()
{ {
if (m_restore) if (m_restore)
{
if (!m_matcher->m_matched_list.empty())
{ {
m_matcher->m_matched_list.erase(m_matcher->m_matched_list.begin() + m_watermark, m_matcher->m_matched_list.erase(m_matcher->m_matched_list.begin() + m_watermark,
m_matcher->m_matched_list.end()); m_matcher->m_matched_list.end());
m_matcher->m_pattern_value_maps.erase(m_pattern_value_maps.begin() + m_capture_size, }
m_pattern_value_maps.end());
if (!m_pattern_value_maps.empty())
{
m_matcher->m_pattern_value_maps.erase(
m_pattern_value_maps.begin() + m_capture_size, m_pattern_value_maps.end());
}
m_matcher->m_pattern_map = m_pattern_value_map; m_matcher->m_pattern_map = m_pattern_value_map;
} }
} }
......
...@@ -36,7 +36,7 @@ ...@@ -36,7 +36,7 @@
namespace mkldnn namespace mkldnn
{ {
class primitive; struct primitive;
} }
namespace ngraph namespace ngraph
......
...@@ -35,7 +35,7 @@ ...@@ -35,7 +35,7 @@
namespace mkldnn namespace mkldnn
{ {
class primitive; struct primitive;
} }
namespace ngraph namespace ngraph
......
...@@ -453,6 +453,7 @@ model_gatherND_int32 ...@@ -453,6 +453,7 @@ model_gatherND_int32
model_gatherND_float model_gatherND_float
model_pad_constant model_pad_constant
model_reciprocal model_reciprocal
model_round
tile_3d_small_data_rank tile_3d_small_data_rank
tile_3d_few_repeats tile_3d_few_repeats
select_v1 select_v1
......
...@@ -282,6 +282,7 @@ model_argmax_int32 ...@@ -282,6 +282,7 @@ model_argmax_int32
model_argmin_int32 model_argmin_int32
model_lp_norm_default model_lp_norm_default
model_instance_normalization model_instance_normalization
model_round
# passing locally, fails closeness checks in CI which may be too strict # passing locally, fails closeness checks in CI which may be too strict
elu elu
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
// CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64 // CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64
// CHECK: %0 = memref_cast %arg0 : memref<2x3xf32> to memref<*xf32> // CHECK: %0 = memref_cast %arg0 : memref<2x3xf32> to memref<*xf32>
// CHECK: %1 = memref_cast %arg2 : memref<2x3xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg2 : memref<2x3xf32> to memref<*xf32>
// CHECK: call @__mlir_callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_softmax(%arg0: !ng.tensor<2x3xf32>, %arg1: !ng.tensor<1x!ng.i64>) -> !ng.tensor<2x3xf32> { func @simple_softmax(%arg0: !ng.tensor<2x3xf32>, %arg1: !ng.tensor<1x!ng.i64>) -> !ng.tensor<2x3xf32> {
%0 = "ng.softmax"(%arg0) {axes = [0]} : (!ng.tensor<2x3xf32>) -> !ng.tensor<2x3xf32> %0 = "ng.softmax"(%arg0) {axes = [0]} : (!ng.tensor<2x3xf32>) -> !ng.tensor<2x3xf32>
"ng.return"(%0) : (!ng.tensor<2x3xf32>) -> () "ng.return"(%0) : (!ng.tensor<2x3xf32>) -> ()
...@@ -26,7 +26,7 @@ func @simple_softmax(%arg0: !ng.tensor<2x3xf32>, %arg1: !ng.tensor<1x!ng.i64>) - ...@@ -26,7 +26,7 @@ func @simple_softmax(%arg0: !ng.tensor<2x3xf32>, %arg1: !ng.tensor<1x!ng.i64>) -
// CHECK: %1 = memref_cast %arg1 : memref<6x4xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg1 : memref<6x4xf32> to memref<*xf32>
// CHECK: %2 = memref_cast %arg2 : memref<3x4xf32> to memref<*xf32> // CHECK: %2 = memref_cast %arg2 : memref<3x4xf32> to memref<*xf32>
// CHECK: %3 = memref_cast %arg3 : memref<3x4xf32> to memref<*xf32> // CHECK: %3 = memref_cast %arg3 : memref<3x4xf32> to memref<*xf32>
// CHECK: call @__mlir_callback_3_inputs(%0, %1, %2, %3, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_3_inputs(%0, %1, %2, %3, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_gemm(%arg0: !ng.tensor<3x6xf32>, %arg1: !ng.tensor<6x4xf32>, %arg2: !ng.tensor<3x4xf32>) -> !ng.tensor<3x4xf32> { func @simple_gemm(%arg0: !ng.tensor<3x6xf32>, %arg1: !ng.tensor<6x4xf32>, %arg2: !ng.tensor<3x4xf32>) -> !ng.tensor<3x4xf32> {
%0 = "ng.gemm"(%arg0, %arg1, %arg2) {alpha = 1.000000e+00 : f32, beta = 1.000000e+00 : f32, transA = false, transB = false} : (!ng.tensor<3x6xf32>, !ng.tensor<6x4xf32>, !ng.tensor<3x4xf32>) -> !ng.tensor<3x4xf32> %0 = "ng.gemm"(%arg0, %arg1, %arg2) {alpha = 1.000000e+00 : f32, beta = 1.000000e+00 : f32, transA = false, transB = false} : (!ng.tensor<3x6xf32>, !ng.tensor<6x4xf32>, !ng.tensor<3x4xf32>) -> !ng.tensor<3x4xf32>
"ng.return"(%0) : (!ng.tensor<3x4xf32>) -> () "ng.return"(%0) : (!ng.tensor<3x4xf32>) -> ()
...@@ -41,7 +41,7 @@ func @simple_gemm(%arg0: !ng.tensor<3x6xf32>, %arg1: !ng.tensor<6x4xf32>, %arg2: ...@@ -41,7 +41,7 @@ func @simple_gemm(%arg0: !ng.tensor<3x6xf32>, %arg1: !ng.tensor<6x4xf32>, %arg2:
// CHECK: %0 = memref_cast %arg0 : memref<3x2xf32> to memref<*xf32> // CHECK: %0 = memref_cast %arg0 : memref<3x2xf32> to memref<*xf32>
// CHECK: %1 = memref_cast %arg1 : memref<2x3xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg1 : memref<2x3xf32> to memref<*xf32>
// CHECK: %2 = memref_cast %arg2 : memref<2x2xf32> to memref<*xf32> // CHECK: %2 = memref_cast %arg2 : memref<2x2xf32> to memref<*xf32>
// CHECK: call @__mlir_callback_2_inputs(%0, %1, %2, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_2_inputs(%0, %1, %2, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_matmul(%arg0: !ng.tensor<3x2xf32>, %arg1: !ng.tensor<2x3xf32>) -> !ng.tensor<2x2xf32> { func @simple_matmul(%arg0: !ng.tensor<3x2xf32>, %arg1: !ng.tensor<2x3xf32>) -> !ng.tensor<2x2xf32> {
%0 = "ng.matmul"(%arg0, %arg1) {transposeA = true, transposeB = true} : (!ng.tensor<3x2xf32>, !ng.tensor<2x3xf32>) -> !ng.tensor<2x2xf32> %0 = "ng.matmul"(%arg0, %arg1) {transposeA = true, transposeB = true} : (!ng.tensor<3x2xf32>, !ng.tensor<2x3xf32>) -> !ng.tensor<2x2xf32>
"ng.return"(%0) : (!ng.tensor<2x2xf32>) -> () "ng.return"(%0) : (!ng.tensor<2x2xf32>) -> ()
...@@ -55,7 +55,7 @@ func @simple_matmul(%arg0: !ng.tensor<3x2xf32>, %arg1: !ng.tensor<2x3xf32>) -> ! ...@@ -55,7 +55,7 @@ func @simple_matmul(%arg0: !ng.tensor<3x2xf32>, %arg1: !ng.tensor<2x3xf32>) -> !
// CHECK: %1 = memref_cast %arg1 : memref<2x1x3x3xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg1 : memref<2x1x3x3xf32> to memref<*xf32>
// CHECK: %[[C1:.*]] = constant 0 : i64 // CHECK: %[[C1:.*]] = constant 0 : i64
// CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64 // CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64
// CHECK: call @__mlir_callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_avgpool(%arg0: !ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32> { func @simple_avgpool(%arg0: !ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32> {
%0 = "ng.avgPool"(%arg0) {includePadding = true, padAbove = [1, 1], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 2]} : (!ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32> %0 = "ng.avgPool"(%arg0) {includePadding = true, padAbove = [1, 1], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 2]} : (!ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32>
"ng.return"(%0) : (!ng.tensor<2x1x3x3xf32>) -> () "ng.return"(%0) : (!ng.tensor<2x1x3x3xf32>) -> ()
...@@ -69,7 +69,7 @@ func @simple_avgpool(%arg0: !ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32> ...@@ -69,7 +69,7 @@ func @simple_avgpool(%arg0: !ng.tensor<2x1x3x3xf32>) -> !ng.tensor<2x1x3x3xf32>
// CHECK: %1 = memref_cast %arg1 : memref<2x2x3x3xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg1 : memref<2x2x3x3xf32> to memref<*xf32>
// CHECK: %[[C1:.*]] = constant 0 : i64 // CHECK: %[[C1:.*]] = constant 0 : i64
// CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64 // CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64
// CHECK: call @__mlir_callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_avgpoolbackprop(%arg0: !ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3x3xf32> { func @simple_avgpoolbackprop(%arg0: !ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3x3xf32> {
%0 = "ng.avgPoolBackprop"(%arg0) {forwardArgShape = [2, 2, 3, 3], includePadding = false, padAbove = [0, 0], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 2]} : (!ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3x3xf32> %0 = "ng.avgPoolBackprop"(%arg0) {forwardArgShape = [2, 2, 3, 3], includePadding = false, padAbove = [0, 0], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 2]} : (!ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3x3xf32>
"ng.return"(%0) : (!ng.tensor<2x2x3x3xf32>) -> () "ng.return"(%0) : (!ng.tensor<2x2x3x3xf32>) -> ()
...@@ -83,7 +83,7 @@ func @simple_avgpoolbackprop(%arg0: !ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3 ...@@ -83,7 +83,7 @@ func @simple_avgpoolbackprop(%arg0: !ng.tensor<2x2x2x2xf32>) -> !ng.tensor<2x2x3
// CHECK: %1 = memref_cast %arg1 : memref<64x3x9x6x5xf32> to memref<*xf32> // CHECK: %1 = memref_cast %arg1 : memref<64x3x9x6x5xf32> to memref<*xf32>
// CHECK: %[[C1:.*]] = constant 0 : i64 // CHECK: %[[C1:.*]] = constant 0 : i64
// CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64 // CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64
// CHECK: call @__mlir_callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_1_input(%0, %1, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_maxpool(%arg0: !ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x5xf32> { func @simple_maxpool(%arg0: !ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x5xf32> {
%0 = "ng.maxPool"(%arg0) {padAbove = [6, 4, 5], padBelow = [5, 6, 4], windowMovementStrides = [2, 3, 4], windowShape = [2, 3, 2]} : (!ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x5xf32> %0 = "ng.maxPool"(%arg0) {padAbove = [6, 4, 5], padBelow = [5, 6, 4], windowMovementStrides = [2, 3, 4], windowShape = [2, 3, 2]} : (!ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x5xf32>
"ng.return"(%0) : (!ng.tensor<64x3x9x6x5xf32>) -> () "ng.return"(%0) : (!ng.tensor<64x3x9x6x5xf32>) -> ()
...@@ -98,7 +98,7 @@ func @simple_maxpool(%arg0: !ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x ...@@ -98,7 +98,7 @@ func @simple_maxpool(%arg0: !ng.tensor<64x3x7x8x10xf32>) -> !ng.tensor<64x3x9x6x
// CHECK: %2 = memref_cast %arg2 : memref<2x2x5x5xf32> to memref<*xf32> // CHECK: %2 = memref_cast %arg2 : memref<2x2x5x5xf32> to memref<*xf32>
// CHECK: %[[C1:.*]] = constant 0 : i64 // CHECK: %[[C1:.*]] = constant 0 : i64
// CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64 // CHECK: %[[C2:.*]] = constant {{[0-9]+}} : i64
// CHECK: call @__mlir_callback_2_inputs(%0, %1, %2, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> () // CHECK: call @callback_2_inputs(%0, %1, %2, %[[C1]], %[[C2]]) : (memref<*xf32>, memref<*xf32>, memref<*xf32>, i64, i64) -> ()
func @simple_maxpoolbackprop(%arg0: !ng.tensor<2x2x5x5xf32>, %arg1: !ng.tensor<2x2x4x3xf32>) -> !ng.tensor<2x2x5x5xf32> { func @simple_maxpoolbackprop(%arg0: !ng.tensor<2x2x5x5xf32>, %arg1: !ng.tensor<2x2x4x3xf32>) -> !ng.tensor<2x2x5x5xf32> {
%0 = "ng.maxPoolBackprop"(%arg0, %arg1) {padAbove = [0, 0], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 3]} : (!ng.tensor<2x2x5x5xf32>, !ng.tensor<2x2x4x3xf32>) -> !ng.tensor<2x2x5x5xf32> %0 = "ng.maxPoolBackprop"(%arg0, %arg1) {padAbove = [0, 0], padBelow = [0, 0], windowMovementStrides = [1, 1], windowShape = [2, 3]} : (!ng.tensor<2x2x5x5xf32>, !ng.tensor<2x2x4x3xf32>) -> !ng.tensor<2x2x5x5xf32>
"ng.return"(%0) : (!ng.tensor<2x2x5x5xf32>) -> () "ng.return"(%0) : (!ng.tensor<2x2x5x5xf32>) -> ()
......
ir_version: 3
producer_name: "nGraph ONNX Importer"
graph {
node {
input: "x"
output: "y"
op_type: "AveragePool"
attribute {
name: "kernel_shape"
ints: 2
ints: 2
type: INTS
}
attribute {
name: "strides"
ints: 2
ints: 2
type: INTS
}
}
name: "compute_graph"
input {
name: "x"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
}
}
}
}
output {
name: "y"
type {
tensor_type {
elem_type: 1
shape {
}
}
}
}
}
opset_import {
version: 7
}
ir_version: 3
producer_name: "nGraph ONNX Importer"
graph {
node {
input: "x"
output: "y"
op_type: "GlobalAveragePool"
attribute {
name: "strides"
ints: 2
ints: 2
type: INTS
}
}
name: "compute_graph"
input {
name: "x"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_value: 5
}
dim {
dim_value: 5
}
}
}
}
}
output {
name: "y"
type {
tensor_type {
elem_type: 1
shape {
}
}
}
}
}
opset_import {
version: 7
}
ir_version: 3
producer_name: "nGraph ONNX Importer"
graph {
node {
input: "x"
output: "y"
op_type: "GlobalMaxPool"
attribute {
name: "strides"
ints: 2
ints: 2
type: INTS
}
}
name: "compute_graph"
input {
name: "x"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_value: 5
}
dim {
dim_value: 5
}
}
}
}
}
output {
name: "y"
type {
tensor_type {
elem_type: 1
shape {
}
}
}
}
}
opset_import {
version: 7
}
ir_version: 3
producer_name: "nGraph ONNX Importer"
graph {
node {
input: "x"
output: "y"
op_type: "MaxPool"
attribute {
name: "kernel_shape"
ints: 2
ints: 2
type: INTS
}
attribute {
name: "strides"
ints: 2
ints: 2
type: INTS
}
attribute {
name: "pads"
ints: 1
ints: 1
ints: 1
ints: 1
type: INTS
}
}
name: "compute_graph"
input {
name: "x"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
dim {
dim_param: "batch"
}
}
}
}
}
output {
name: "y"
type {
tensor_type {
elem_type: 1
shape {
}
}
}
}
}
opset_import {
version: 7
}
ir_version: 3
producer_name: "backend-test"
graph {
node {
input: "x"
output: "y"
op_type: "Round"
}
name: "test_round"
input {
name: "x"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_value: 15
}
}
}
}
}
output {
name: "y"
type {
tensor_type {
elem_type: 1
shape {
dim {
dim_value: 15
}
}
}
}
}
}
opset_import {
version: 11
}
...@@ -1963,3 +1963,30 @@ NGRAPH_TEST(onnx_${BACKEND_NAME}, model_reciprocal) ...@@ -1963,3 +1963,30 @@ NGRAPH_TEST(onnx_${BACKEND_NAME}, model_reciprocal)
test_case.run(); test_case.run();
} }
NGRAPH_TEST(onnx_${BACKEND_NAME}, model_round)
{
const auto round_fn =
onnx_import::import_onnx_model(file_util::path_join(SERIALIZED_ZOO, "onnx/round.prototxt"));
auto test_case = ngraph::test::NgraphTestCase(round_fn, "${BACKEND_NAME}");
test_case.add_input<float>({0.1f,
0.5f,
0.9f,
1.2f,
1.5f,
1.8f,
2.3f,
2.5f,
2.7f,
-1.1f,
-1.5f,
-1.9f,
-2.2f,
-2.5f,
-2.8f});
test_case.add_expected_output<float>(
{0.f, 0.f, 1.f, 1.f, 2.f, 2.f, 2.f, 2.f, 3.f, -1.f, -2.f, -2.f, -2.f, -2.f, -3.f});
test_case.run();
}
...@@ -282,3 +282,83 @@ NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, model_conv_with_dynamic_batch) ...@@ -282,3 +282,83 @@ NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, model_conv_with_dynamic_batch)
test_case.run(); test_case.run();
} }
NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, avg_pool_dyn_shape)
{
const auto function = onnx_import::import_onnx_model(
file_util::path_join(SERIALIZED_ZOO, "onnx/dynamic_shapes/average_pool_2d_dyn.prototxt"));
auto test_case = NgraphTestCase(function, "${BACKEND_NAME}", BackendMode::DYNAMIC);
const Shape shape{1, 1, 4, 4};
const auto elems_in_tensor = shape_size(shape);
std::vector<float> input_values(elems_in_tensor);
std::iota(input_values.begin(), input_values.end(), 0.f);
test_case.add_input<float>(shape, input_values);
std::vector<float> expected_values{2.5f, 4.5f, 10.5f, 12.5f};
test_case.add_expected_output<float>(Shape{1, 1, 2, 2}, expected_values);
test_case.run();
}
NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, max_pool_dyn_shape)
{
const auto function = onnx_import::import_onnx_model(
file_util::path_join(SERIALIZED_ZOO, "onnx/dynamic_shapes/max_pool_2d_dyn.prototxt"));
auto test_case = NgraphTestCase(function, "${BACKEND_NAME}", BackendMode::DYNAMIC);
const Shape shape{1, 1, 4, 4};
const auto elems_in_tensor = shape_size(shape);
std::vector<float> input_values(elems_in_tensor);
std::iota(input_values.begin(), input_values.end(), 0.f);
test_case.add_input<float>(shape, input_values);
std::vector<float> expected_values{0.f, 2.f, 3.f, 8.f, 10.f, 11.f, 12.f, 14.f, 15.f};
test_case.add_expected_output<float>(Shape{1, 1, 3, 3}, expected_values);
test_case.run();
}
NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, global_avg_pool_dyn_shape)
{
const auto function = onnx_import::import_onnx_model(file_util::path_join(
SERIALIZED_ZOO, "onnx/dynamic_shapes/global_average_pool_dyn.prototxt"));
auto test_case = NgraphTestCase(function, "${BACKEND_NAME}", BackendMode::DYNAMIC);
const Shape shape{1, 3, 5, 5};
const auto elems_in_tensor = shape_size(shape);
std::vector<float> input_values(elems_in_tensor);
std::iota(input_values.begin(), input_values.end(), 0.f);
test_case.add_input<float>(shape, input_values);
std::vector<float> expected_values{12.f, 37.f, 62.f};
test_case.add_expected_output<float>(Shape{1, 3, 1, 1}, expected_values);
test_case.run();
}
NGRAPH_TEST(onnx_dyn_shapes_${BACKEND_NAME}, global_max_pool_dyn_shape)
{
const auto function = onnx_import::import_onnx_model(
file_util::path_join(SERIALIZED_ZOO, "onnx/dynamic_shapes/global_max_pool_dyn.prototxt"));
auto test_case = NgraphTestCase(function, "${BACKEND_NAME}", BackendMode::DYNAMIC);
const Shape shape{1, 3, 5, 5};
const auto elems_in_tensor = shape_size(shape);
std::vector<float> input_values(elems_in_tensor);
std::iota(input_values.begin(), input_values.end(), 0.f);
test_case.add_input<float>(shape, input_values);
std::vector<float> expected_values{24.f, 49.f, 74.f};
test_case.add_expected_output<float>(Shape{1, 3, 1, 1}, expected_values);
test_case.run();
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment