* Fixed extractor for ONNX ReduceXXX operations and fixed ReduceAxisNormalizer transformation * Unit test for ReduceAxisNormalizer transformation
67 lines
3.1 KiB
Python
67 lines
3.1 KiB
Python
"""
|
|
Copyright (C) 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.
|
|
"""
|
|
|
|
from extensions.ops.ReduceOps import reduce_map
|
|
from extensions.ops.range import Range
|
|
from extensions.ops.rank import Rank
|
|
from mo.front.common.partial_infer.utils import int64_array
|
|
from mo.front.common.replacement import FrontReplacementSubgraph
|
|
from mo.front.subgraph_matcher import SubgraphMatch
|
|
from mo.front.tf.graph_utils import create_op_with_const_inputs
|
|
from mo.graph.graph import Graph
|
|
from mo.ops.const import Const
|
|
|
|
|
|
class ReduceAxisNormalizer(FrontReplacementSubgraph):
|
|
"""
|
|
Reduce operation requires information about axis, that is represented in original frameworks differently: as an
|
|
operation attribute or as a 1-st input port value. ReduceAxisNormalizer adds second input to Reduce operations with
|
|
axes to normalize if axes are specified as an attribute.
|
|
"""
|
|
enabled = True
|
|
|
|
def pattern(self):
|
|
return dict(
|
|
nodes=[
|
|
('reduce', dict(kind='op', op=lambda op: op in reduce_map))
|
|
],
|
|
edges=[]
|
|
)
|
|
|
|
def replace_sub_graph(self, graph: Graph, match: [dict, SubgraphMatch]):
|
|
node = match['reduce']
|
|
connected_in_ports = [port for port in node.in_ports().values() if not port.disconnected()]
|
|
if len(connected_in_ports) == 1:
|
|
node_name = node.soft_get('name', node.id)
|
|
|
|
# if the 'axis' is None then we still add a second input to the layer with a 1D array with 1 element equal
|
|
# to None. The infer function handles this case because the input shape is known at this stage only
|
|
if node.has_valid('axis'):
|
|
const = Const(graph, {'name': node_name + '/axis', 'value': node.axis}).create_node()
|
|
node.add_input_port(1, skip_if_exist=True)
|
|
const.out_port(0).connect(node.in_port(1))
|
|
del graph.node[node.id]['axis']
|
|
else:
|
|
# The default (if there is no 'axis') is to reduce over all the dimensions of the input tensor.
|
|
axes = create_op_with_const_inputs(graph, Range, {0: int64_array(0), 2: int64_array(1)},
|
|
dict(name=node_name + '/axes'))
|
|
end_of_range = Rank(graph, dict(name=node_name + '/range_end')).create_node()
|
|
node.in_port(0).get_connection().get_source().connect(end_of_range.in_port(0))
|
|
end_of_range.out_port(0).connect(axes.in_port(1))
|
|
|
|
node.add_input_port(1, skip_if_exist=True)
|
|
axes.out_port(0).connect(node.in_port(1))
|