* Enable ngraph python tests * Refactor and unify ngraph with onnx python tests * Revert deprecated test cases * Set ngraph and onnx python tests as a one test suite execution * Change unstrict Xfails to strict ones * Update after review: - add model zoo to onnx tests, - improvements of tests * Revert mounting zoo models dir Co-authored-by: Michał Karzyński <4430709+postrational@users.noreply.github.com>
418 lines
15 KiB
Python
418 lines
15 KiB
Python
# ******************************************************************************
|
|
# Copyright 2018-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.
|
|
# ******************************************************************************
|
|
import numpy as np
|
|
import onnx
|
|
import pytest
|
|
from onnx.helper import make_graph, make_model, make_node, make_tensor_value_info
|
|
|
|
from tests.runtime import get_runtime
|
|
from tests.test_onnx.utils import get_node_model, import_onnx_model, run_model, run_node
|
|
from tests import xfail_issue_35911, xfail_issue_35912
|
|
|
|
|
|
@pytest.fixture
|
|
def ndarray_1x1x4x4():
|
|
return np.array(
|
|
[[11, 12, 13, 14], [15, 16, 17, 18], [19, 20, 21, 22], [23, 24, 25, 26]], dtype=np.float32
|
|
).reshape([1, 1, 4, 4])
|
|
|
|
|
|
def make_onnx_model_for_conv_op(x_shape, weights_shape, transpose=False, **attributes):
|
|
output_shape = () # We don't need output shape to be accurate for these tests
|
|
|
|
if transpose:
|
|
node_op = "ConvTranspose"
|
|
else:
|
|
node_op = "Conv"
|
|
|
|
node = make_node(node_op, ["X", "weight"], ["Y"], name="test_node", **attributes)
|
|
graph = make_graph(
|
|
[node],
|
|
"test_graph",
|
|
[
|
|
make_tensor_value_info("X", onnx.TensorProto.FLOAT, x_shape),
|
|
make_tensor_value_info("weight", onnx.TensorProto.FLOAT, weights_shape),
|
|
],
|
|
[make_tensor_value_info("Y", onnx.TensorProto.FLOAT, output_shape)],
|
|
)
|
|
model = make_model(graph, producer_name="ngraph ONNXImporter")
|
|
return model
|
|
|
|
|
|
def import_and_compute_conv(x, weights, transpose=False, **attributes):
|
|
x, weights = np.array(x), np.array(weights)
|
|
onnx_model = make_onnx_model_for_conv_op(x.shape, weights.shape, transpose=transpose, **attributes)
|
|
ng_model_function = import_onnx_model(onnx_model)
|
|
computation = get_runtime().computation(ng_model_function)
|
|
return computation(x, weights)[0]
|
|
|
|
|
|
def test_2d_conv():
|
|
# x should have shape N(batch) x C x H x W
|
|
input_x = np.array(
|
|
[
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
],
|
|
dtype=np.float32,
|
|
).reshape(1, 1, 9, 9)
|
|
|
|
# filter weights should have shape M x C x kH x kW
|
|
input_filter = np.array([[1.0, 0.0, -1.0], [2.0, 0.0, -2.0], [1.0, 0.0, -1.0]], dtype=np.float32).reshape(
|
|
[1, 1, 3, 3]
|
|
)
|
|
|
|
# convolution with padding=1 should produce 9 x 9 output:
|
|
result = import_and_compute_conv(input_x, input_filter, pads=(1, 1, 1, 1), strides=(1, 1))
|
|
assert np.array_equal(
|
|
result,
|
|
np.array(
|
|
[
|
|
[
|
|
[
|
|
[0.0, -15.0, -15.0, 15.0, 15.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -15.0, -15.0, 15.0, 15.0, 0.0, 0.0, 0.0, 0.0],
|
|
]
|
|
]
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
# convolution with padding=0 should produce 7 x 7 output:
|
|
result = import_and_compute_conv(input_x, input_filter, pads=(0, 0, 0, 0), strides=(1, 1))
|
|
assert np.array_equal(
|
|
result,
|
|
np.array(
|
|
[
|
|
[
|
|
[
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
[-20, -20, 20, 20, 0, 0, 0],
|
|
]
|
|
]
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
# convolution with strides=2 should produce 4 x 4 output:
|
|
result = import_and_compute_conv(input_x, input_filter, pads=(0, 0, 0, 0), strides=(2, 2))
|
|
assert np.array_equal(
|
|
result,
|
|
np.array(
|
|
[
|
|
[
|
|
[
|
|
[-20.0, 20.0, 0.0, 0.0],
|
|
[-20.0, 20.0, 0.0, 0.0],
|
|
[-20.0, 20.0, 0.0, 0.0],
|
|
[-20.0, 20.0, 0.0, 0.0],
|
|
]
|
|
]
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
# convolution with dilations=2 should produce 5 x 5 output:
|
|
result = import_and_compute_conv(input_x, input_filter, dilations=(2, 2))
|
|
assert np.array_equal(
|
|
result,
|
|
np.array(
|
|
[
|
|
[
|
|
[
|
|
[0, 0, 20, 20, 0],
|
|
[0, 0, 20, 20, 0],
|
|
[0, 0, 20, 20, 0],
|
|
[0, 0, 20, 20, 0],
|
|
[0, 0, 20, 20, 0],
|
|
]
|
|
]
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
|
|
def test_3d_conv():
|
|
# x should have shape N(batch) x C x H x W x D
|
|
input_x = np.array(
|
|
[
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, 0.0, 5.0, 5.0, 0.0, 0.0, 0.0, 0.0, 0.0],
|
|
],
|
|
dtype=np.float32,
|
|
).reshape([1, 1, 9, 9, 1])
|
|
input_x = np.broadcast_to(input_x, (1, 1, 9, 9, 4))
|
|
|
|
# filter weights should have shape M x C x kH x kW x kD
|
|
input_filter = np.array([[1.0, 0.0, -1.0], [2.0, 0.0, -2.0], [1.0, 0.0, -1.0]], dtype=np.float32).reshape(
|
|
[1, 1, 3, 3, 1]
|
|
)
|
|
input_filter = np.broadcast_to(input_filter, (1, 1, 3, 3, 3))
|
|
|
|
# convolution with padding=0 should produce 7 x 7 x 2 output:
|
|
result = import_and_compute_conv(
|
|
input_x, input_filter, dilations=(1, 1, 1), pads=(0, 0, 0, 0, 0, 0), strides=(1, 1, 1)
|
|
)
|
|
|
|
assert np.array_equal(
|
|
np.moveaxis(result.squeeze(), (0, 1, 2), (1, 2, 0)),
|
|
np.array(
|
|
[
|
|
[
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
],
|
|
[
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
[-60.0, -60.0, 60.0, 60.0, 0.0, 0.0, 0.0],
|
|
],
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
|
|
def test_2d_conv_transpose():
|
|
# x should have shape N(batch) x C x H x W
|
|
input_x = np.array(
|
|
[
|
|
[0.0, -15.0, -15.0, 15.0, 15.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -20.0, -20.0, 20.0, 20.0, 0.0, 0.0, 0.0, 0.0],
|
|
[0.0, -15.0, -15.0, 15.0, 15.0, 0.0, 0.0, 0.0, 0.0],
|
|
],
|
|
dtype=np.float32,
|
|
).reshape([1, 1, 9, 9])
|
|
|
|
# filter weights should have shape M x C x kH x kW
|
|
input_filter = np.array([[1.0, 0.0, -1.0], [2.0, 0.0, -2.0], [1.0, 0.0, -1.0]], dtype=np.float32).reshape(
|
|
[1, 1, 3, 3]
|
|
)
|
|
|
|
# deconvolution with padding=1 should produce 9 x 9 output:
|
|
result = import_and_compute_conv(input_x, input_filter, transpose=True, pads=(1, 1, 1, 1), strides=(1, 1))
|
|
|
|
assert np.array_equal(
|
|
result.reshape([9, 9]),
|
|
np.array(
|
|
[
|
|
[-50.0, -50.0, 100.0, 100.0, -50.0, -50.0, 0.0, 0.0, 0.0],
|
|
[-75.0, -75.0, 150.0, 150.0, -75.0, -75.0, 0.0, 0.0, 0.0],
|
|
[-80.0, -80.0, 160.0, 160.0, -80.0, -80.0, 0.0, 0.0, 0.0],
|
|
[-80.0, -80.0, 160.0, 160.0, -80.0, -80.0, 0.0, 0.0, 0.0],
|
|
[-80.0, -80.0, 160.0, 160.0, -80.0, -80.0, 0.0, 0.0, 0.0],
|
|
[-80.0, -80.0, 160.0, 160.0, -80.0, -80.0, 0.0, 0.0, 0.0],
|
|
[-80.0, -80.0, 160.0, 160.0, -80.0, -80.0, 0.0, 0.0, 0.0],
|
|
[-75.0, -75.0, 150.0, 150.0, -75.0, -75.0, 0.0, 0.0, 0.0],
|
|
[-50.0, -50.0, 100.0, 100.0, -50.0, -50.0, 0.0, 0.0, 0.0],
|
|
],
|
|
dtype=np.float32,
|
|
),
|
|
)
|
|
|
|
|
|
@xfail_issue_35911
|
|
def test_pad_opset_1():
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
y = np.pad(x, pad_width=1, mode="constant")
|
|
|
|
model = get_node_model("Pad", x, paddings=[1, 1, 1, 1])
|
|
ng_results = run_model(model, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
x = np.random.randn(1, 3, 4, 5).astype(np.float32)
|
|
y = np.pad(x, pad_width=((0, 0), (0, 0), (1, 2), (3, 4)), mode="constant")
|
|
|
|
model = get_node_model("Pad", x, mode="constant", paddings=[0, 0, 1, 3, 0, 0, 2, 4])
|
|
ng_results = run_model(model, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
# incorrect paddings rank
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
model = get_node_model("Pad", x, paddings=[0, 1, 1, 3, 1, 2])
|
|
with pytest.raises(RuntimeError):
|
|
run_model(model, [x])
|
|
|
|
# no paddings arttribute
|
|
model = get_node_model("Pad", x)
|
|
with pytest.raises(RuntimeError):
|
|
import_onnx_model(model)
|
|
|
|
|
|
def test_pad_opset_2():
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
y = np.pad(x, pad_width=1, mode="constant")
|
|
|
|
model = get_node_model("Pad", x, opset=2, pads=[1, 1, 1, 1])
|
|
ng_results = run_model(model, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
x = np.random.randn(1, 3, 4, 5).astype(np.float32)
|
|
y = np.pad(x, pad_width=((0, 0), (0, 0), (1, 2), (3, 4)), mode="constant")
|
|
|
|
model = get_node_model("Pad", x, opset=2, mode="constant", pads=[0, 0, 1, 3, 0, 0, 2, 4])
|
|
ng_results = run_model(model, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
# incorrect pads rank
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
model = get_node_model("Pad", x, opset=2, pads=[0, 1, 1, 3, 1, 2])
|
|
with pytest.raises(RuntimeError):
|
|
run_model(model, [x])
|
|
|
|
|
|
@xfail_issue_35912
|
|
def test_pad_negative_values_begin():
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
|
|
# Axis 1 begin
|
|
model = get_node_model("Pad", x, opset=2, pads=[-1, 0, 0, 0])
|
|
ng_result = run_model(model, [x])[0]
|
|
assert np.array_equal(ng_result, np.array([[1, 1]]))
|
|
|
|
# Axis 2 begin
|
|
model = get_node_model("Pad", x, opset=2, pads=[0, -1, 0, 0])
|
|
ng_result = run_model(model, [x])[0]
|
|
assert np.array_equal(ng_result, np.array([[1], [1]]))
|
|
|
|
|
|
@xfail_issue_35912
|
|
def test_pad_negative_values_end():
|
|
x = np.ones((2, 2), dtype=np.float32)
|
|
|
|
# Axis 1 end
|
|
model = get_node_model("Pad", x, opset=2, pads=[0, 0, -1, 0])
|
|
ng_result = run_model(model, [x])[0]
|
|
assert np.array_equal(ng_result, np.array([[1.0, 1.0]]))
|
|
|
|
# Axis 2 end
|
|
model = get_node_model("Pad", x, opset=2, pads=[0, 0, 0, -1])
|
|
ng_result = run_model(model, [x])[0]
|
|
assert np.array_equal(ng_result, np.array([[1], [1]]))
|
|
|
|
|
|
def test_pool_average(ndarray_1x1x4x4):
|
|
x = ndarray_1x1x4x4
|
|
node = onnx.helper.make_node(
|
|
"AveragePool", inputs=["x"], outputs=["y"], kernel_shape=(2, 2), strides=(2, 2)
|
|
)
|
|
y = np.array([[13.5, 15.5], [21.5, 23.5]], dtype=np.float32).reshape([1, 1, 2, 2])
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
node = onnx.helper.make_node(
|
|
"AveragePool", inputs=["x"], outputs=["y"], kernel_shape=(2, 2), strides=(2, 2), pads=(1, 1, 1, 1)
|
|
)
|
|
y = np.array([[11, 12.5, 14], [17, 18.5, 20], [23, 24.5, 26]], dtype=np.float32).reshape([1, 1, 3, 3])
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
|
|
def test_pool_average_3d(ndarray_1x1x4x4):
|
|
x = np.broadcast_to(ndarray_1x1x4x4, (1, 1, 4, 4, 4))
|
|
node = onnx.helper.make_node(
|
|
"AveragePool", inputs=["x"], outputs=["y"], kernel_shape=(2, 2, 2), strides=(2, 2, 2)
|
|
)
|
|
y = np.array([[[13.5, 15.5], [21.5, 23.5]], [[13.5, 15.5], [21.5, 23.5]]], dtype=np.float32).reshape(
|
|
[1, 1, 2, 2, 2]
|
|
)
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
|
|
def test_pool_max(ndarray_1x1x4x4):
|
|
node = onnx.helper.make_node("MaxPool", inputs=["x"], outputs=["y"], kernel_shape=(2, 2), strides=(2, 2))
|
|
|
|
x = ndarray_1x1x4x4
|
|
y = np.array([[16, 18], [24, 26]], dtype=np.float32).reshape([1, 1, 2, 2])
|
|
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
|
|
def test_pool_global_max(ndarray_1x1x4x4):
|
|
node = onnx.helper.make_node("GlobalMaxPool", inputs=["x"], outputs=["y"])
|
|
|
|
x = ndarray_1x1x4x4
|
|
y = np.array([26], dtype=np.float32).reshape([1, 1, 1, 1])
|
|
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
|
|
def test_pool_global_average(ndarray_1x1x4x4):
|
|
node = onnx.helper.make_node("GlobalAveragePool", inputs=["x"], outputs=["y"])
|
|
|
|
x = ndarray_1x1x4x4
|
|
y = np.array([18.5], dtype=np.float32).reshape([1, 1, 1, 1])
|
|
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|
|
|
|
|
|
def test_pool_global_average_3d(ndarray_1x1x4x4):
|
|
x = np.broadcast_to(ndarray_1x1x4x4, (1, 1, 4, 4, 4))
|
|
|
|
node = onnx.helper.make_node("GlobalAveragePool", inputs=["x"], outputs=["y"])
|
|
y = np.array([18.5], dtype=np.float32).reshape([1, 1, 1, 1, 1])
|
|
ng_results = run_node(node, [x])
|
|
assert np.array_equal(ng_results, [y])
|