* Initial version of ReduceL1, ReduceL2 and ReduceLp enabling in the MO * Added operations ReduceL1 and ReduceL2 to nGraph * Removed ReduceLp. Added ReduceL1 and ReduceL2 * Separated specification of ReduceLp into ReduceL1 and ReduceL2 * Updated ReduceL1 and ReduceL2 specification * Fixed ReduceL1 and ReduceL2 type prop tests * Implemented nGraph transformation to decompose ReduceL1 and ReduceL2. Disabled them for CPU and GPU plugins * Updated supported framework layers * Added unit tests for ReduceL1 and ReduceL2 reference implementation * Fixed ReduceXXX operations reference implementation by adding support for a new parameter 'keep_dims' * Fixed constant folding for v0::Any * Added ReduceL1 and ReduceL2 to Python API * Implemented ReduceL1 and ReduceL2 decomposition tests and fixed ReduceL2 decomposition * Added specific creator for ReduceXXX operations instead of NodeBuilders * Fixed conversion ReduceXXX to CNNLayer * Fixed parser for ReduceLogicalXXX operations
73 lines
3.1 KiB
Python
73 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.replacement import FrontReplacementSubgraph
|
|
from mo.front.subgraph_matcher import SubgraphMatch
|
|
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:
|
|
- by layer parameter
|
|
- by 1-port input value
|
|
|
|
ReduceAxisNormalizer reforms Reduce operations to store axis info in 1-port input.
|
|
"""
|
|
enabled = True
|
|
force_shape_inference = 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('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.
|
|
|
|
begin_of_range = Const(graph, dict(name=node_name + '/range_begin_', value=0)).create_node()
|
|
step = Const(graph, dict(name=node_name + '/range_step_', value=1)).create_node()
|
|
end_of_range = Rank(graph, dict(name=node_name + '/range_end_')).create_node()
|
|
axes = Range(graph, dict(name=node_name + '/axes_')).create_node()
|
|
|
|
begin_of_range.out_port(0).connect(axes.in_port(0))
|
|
end_of_range.out_port(0).connect(axes.in_port(1))
|
|
step.out_port(0).connect(axes.in_port(2))
|
|
|
|
node.add_input_port(1, skip_if_exist=True)
|
|
axes.out_port(0).connect(node.in_port(1))
|
|
node.in_port(0).get_connection().get_source().connect(end_of_range.in_port(0))
|