From 33b0050c77efade055c7e41fb0b630e9d77b09aa Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 6 Dec 2024 20:46:07 +0100 Subject: [PATCH 01/11] extend snitch support extend snitch support --- .gitmodules | 6 +- Deeploy/Targets/Generic/Layers.py | 21 +- Deeploy/Targets/Generic/Parsers.py | 89 ++++ .../Generic/Templates/iNoNormTemplate.py | 38 ++ .../iSoftmaxPreAllocatedBuffTemplate.py | 60 +++ .../TopologyOptimizationPasses/Passes.py | 134 ++++- Deeploy/Targets/Generic/TypeCheckers.py | 17 +- Deeploy/Targets/Snitch/Bindings.py | 68 +++ .../SnitchClusterSynch.py | 46 ++ .../SnitchClusterTiling.py | 49 ++ .../SnitchClusterTilingSB.py | 504 ++++++++++++++++++ .../SnitchCoreFilter.py | 46 ++ .../SnitchProfileExecutionBlock.py | 52 ++ .../CodeTransformationPasses/__init__.py | 29 + Deeploy/Targets/Snitch/DataTypes.py | 40 ++ Deeploy/Targets/Snitch/Deployer.py | 7 +- Deeploy/Targets/Snitch/Parser.py | 96 ++++ Deeploy/Targets/Snitch/Platform.py | 41 +- .../Targets/Snitch/Templates/AddTemplate.py | 57 ++ .../Targets/Snitch/Templates/GemmTemplate.py | 28 + .../Targets/Snitch/Templates/RQAddTemplate.py | 70 +++ .../Snitch/Templates/RqGemmTemplate.py | 32 ++ .../Snitch/Templates/iSoftmaxTemplate.py | 41 ++ .../TileConstraints/GemmTileConstraint.py | 159 ++++++ .../TileConstraints/RqGemmTileConstraint.py | 174 ++++++ .../Snitch/TileConstraints/__init__.py | 28 + .../TileConstraints/iNoNormTileConstraint.py | 112 ++++ .../TileConstraints/iSoftmaxTileConstraint.py | 118 ++++ Deeploy/Targets/Snitch/Tiler.py | 47 ++ Deeploy/Targets/Snitch/TypeCheckers.py | 50 ++ .../TilingCodeGeneration.py | 36 +- DeeployTest/Platforms/Snitch/main.c | 42 +- DeeployTest/Tests/TestAdderLarge/inputs.npz | Bin 0 -> 33274 bytes DeeployTest/Tests/TestAdderLarge/network.onnx | 22 + DeeployTest/Tests/TestAdderLarge/outputs.npz | Bin 0 -> 65800 bytes DeeployTest/Tests/TestRQAdd/activations.npz | Bin 0 -> 394392 bytes DeeployTest/Tests/TestRQAdd/inputs.npz | Bin 0 -> 33274 bytes DeeployTest/Tests/TestRQAdd/network.onnx | Bin 0 -> 2142 bytes DeeployTest/Tests/TestRQAdd/outputs.npz | Bin 0 -> 65800 bytes DeeployTest/Tests/TestiNoNorm/activations.npz | Bin 0 -> 22 bytes DeeployTest/Tests/TestiNoNorm/inputs.npz | Bin 0 -> 131340 bytes DeeployTest/Tests/TestiNoNorm/network.onnx | Bin 0 -> 66721 bytes DeeployTest/Tests/TestiNoNorm/outputs.npz | Bin 0 -> 131342 bytes .../Tests/TestiSoftmaxLarge/activations.npz | Bin 0 -> 22 bytes .../Tests/TestiSoftmaxLarge/inputs.npz | Bin 0 -> 65800 bytes .../Tests/TestiSoftmaxLarge/network.onnx | Bin 0 -> 543 bytes .../Tests/TestiSoftmaxLarge/outputs.npz | Bin 0 -> 262408 bytes DeeployTest/Tests/testGEMM/inputs.npz | Bin 5114 -> 2554 bytes DeeployTest/Tests/testGEMM/network.onnx | Bin 20717 -> 8428 bytes DeeployTest/Tests/testGEMM/outputs.npz | Bin 8456 -> 4360 bytes DeeployTest/Tests/testRQGEMMTransB/inputs.npz | Bin 0 -> 4934 bytes .../Tests/testRQGEMMTransB/network.onnx | Bin 0 -> 3256 bytes .../Tests/testRQGEMMTransB/outputs.npz | Bin 0 -> 776 bytes DeeployTest/testRunner_tiled_snitch.py | 10 +- Makefile | 3 +- TargetLibraries/Snitch/CMakeLists.txt | 2 + .../Snitch/inc/DeeploySnitchMath.h | 3 + TargetLibraries/Snitch/inc/dmaStruct.h | 37 ++ TargetLibraries/Snitch/inc/kernel/Gemm.h | 27 + TargetLibraries/Snitch/inc/kernel/RQGemm.h | 59 ++ TargetLibraries/Snitch/inc/kernel/iNoNorm.h | 30 ++ TargetLibraries/Snitch/inc/macros.h | 7 +- TargetLibraries/Snitch/src/Add.c | 49 ++ TargetLibraries/Snitch/src/Gemm_s8.c | 54 ++ TargetLibraries/Snitch/src/RQGemm_s8.c | 295 +++++++++- TargetLibraries/Snitch/src/iNoNorm.c | 102 ++++ .../Snitch/third_party/pulp-nn-mixed | 1 + 67 files changed, 3000 insertions(+), 38 deletions(-) create mode 100644 Deeploy/Targets/Generic/Templates/iNoNormTemplate.py create mode 100644 Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py create mode 100644 Deeploy/Targets/Snitch/Bindings.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterSynch.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTiling.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchCoreFilter.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchProfileExecutionBlock.py create mode 100644 Deeploy/Targets/Snitch/CodeTransformationPasses/__init__.py create mode 100644 Deeploy/Targets/Snitch/DataTypes.py create mode 100644 Deeploy/Targets/Snitch/Parser.py create mode 100644 Deeploy/Targets/Snitch/Templates/AddTemplate.py create mode 100644 Deeploy/Targets/Snitch/Templates/GemmTemplate.py create mode 100644 Deeploy/Targets/Snitch/Templates/RQAddTemplate.py create mode 100644 Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py create mode 100644 Deeploy/Targets/Snitch/Templates/iSoftmaxTemplate.py create mode 100644 Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py create mode 100644 Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py create mode 100644 Deeploy/Targets/Snitch/TileConstraints/__init__.py create mode 100644 Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py create mode 100644 Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py create mode 100644 Deeploy/Targets/Snitch/Tiler.py create mode 100644 Deeploy/Targets/Snitch/TypeCheckers.py create mode 100644 DeeployTest/Tests/TestAdderLarge/inputs.npz create mode 100644 DeeployTest/Tests/TestAdderLarge/network.onnx create mode 100644 DeeployTest/Tests/TestAdderLarge/outputs.npz create mode 100644 DeeployTest/Tests/TestRQAdd/activations.npz create mode 100644 DeeployTest/Tests/TestRQAdd/inputs.npz create mode 100644 DeeployTest/Tests/TestRQAdd/network.onnx create mode 100644 DeeployTest/Tests/TestRQAdd/outputs.npz create mode 100644 DeeployTest/Tests/TestiNoNorm/activations.npz create mode 100644 DeeployTest/Tests/TestiNoNorm/inputs.npz create mode 100644 DeeployTest/Tests/TestiNoNorm/network.onnx create mode 100644 DeeployTest/Tests/TestiNoNorm/outputs.npz create mode 100644 DeeployTest/Tests/TestiSoftmaxLarge/activations.npz create mode 100644 DeeployTest/Tests/TestiSoftmaxLarge/inputs.npz create mode 100644 DeeployTest/Tests/TestiSoftmaxLarge/network.onnx create mode 100644 DeeployTest/Tests/TestiSoftmaxLarge/outputs.npz create mode 100644 DeeployTest/Tests/testRQGEMMTransB/inputs.npz create mode 100644 DeeployTest/Tests/testRQGEMMTransB/network.onnx create mode 100644 DeeployTest/Tests/testRQGEMMTransB/outputs.npz create mode 100644 TargetLibraries/Snitch/inc/dmaStruct.h create mode 100644 TargetLibraries/Snitch/inc/kernel/iNoNorm.h create mode 100644 TargetLibraries/Snitch/src/Add.c create mode 100644 TargetLibraries/Snitch/src/Gemm_s8.c create mode 100644 TargetLibraries/Snitch/src/iNoNorm.c create mode 160000 TargetLibraries/Snitch/third_party/pulp-nn-mixed diff --git a/.gitmodules b/.gitmodules index 1c1506d..336972c 100644 --- a/.gitmodules +++ b/.gitmodules @@ -6,4 +6,8 @@ url = https://github.com/pulp-platform/pulp-nnx.git [submodule "CMSIS-NN"] path = TargetLibraries/CMSIS/third_party/CMSIS-NN - url = https://github.com/ARM-software/CMSIS-NN.git \ No newline at end of file + url = https://github.com/ARM-software/CMSIS-NN.git +[submodule "TargetLibraries/Snitch/third_party/pulp-nn-mixed"] + path = TargetLibraries/Snitch/third_party/pulp-nn-mixed + url = https://github.com/Victor-Jung/pulp-nn-mixed.git + branch = deeploySnitchTarget diff --git a/Deeploy/Targets/Generic/Layers.py b/Deeploy/Targets/Generic/Layers.py index 91e8107..c963916 100644 --- a/Deeploy/Targets/Generic/Layers.py +++ b/Deeploy/Targets/Generic/Layers.py @@ -25,11 +25,11 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import List, Tuple +from typing import Dict, List, Tuple import numpy as np -from Deeploy.DeeployTypes import NodeMapper, ONNXLayer, Shape +from Deeploy.DeeployTypes import NodeMapper, ONNXLayer, Shape, OperatorRepresentation class ConcatLayer(ONNXLayer): @@ -85,6 +85,23 @@ def __init__(self, maps: List[NodeMapper]): super().__init__(maps) +class iNoNormLayer(ONNXLayer): + + def __init__(self, maps: List[NodeMapper]): + super().__init__(maps) + + def computeOps(self): + return self.mapper.parser.operatorRepresentation['size'] * 4 # 2 mul, 1 add, 1 right shift + + def computeShapes(self, inputShapes: Shape, outputShapes: Shape, operatorRepresentation: OperatorRepresentation, + channels_first: bool) -> Tuple[Shape]: + + # JUNGVI: Broadcast the weights and bias to have as many dimensions as the inputs + inputShapes[1] = [1] * (len(inputShapes[0]) - len(inputShapes[1])) + list(inputShapes[1]) + inputShapes[2] = inputShapes[1] + return (inputShapes, outputShapes) + + class RQSiGELULayer(iGELULayer): def __init__(self, maps: List[NodeMapper]): diff --git a/Deeploy/Targets/Generic/Parsers.py b/Deeploy/Targets/Generic/Parsers.py index 7c9e7e7..0ab0dde 100644 --- a/Deeploy/Targets/Generic/Parsers.py +++ b/Deeploy/Targets/Generic/Parsers.py @@ -751,6 +751,40 @@ def parseNodeCtxt(self, return ctxt, True +class iNoNormParser(NodeParser): + + def __init__(self): + super().__init__() + + def parseNode(self, node: gs.Node) -> bool: + + ret = all(['D' in node.attrs, 'mul' in node.attrs, 'n_levels' in node.attrs]) + + if ret: + self.operatorRepresentation['D'] = node.attrs['D'] + self.operatorRepresentation['log2D'] = int(np.log2(node.attrs['D'].values).tolist()[0]) + self.operatorRepresentation['mul'] = int(node.attrs['mul'].values.tolist()[0]) + self.operatorRepresentation['n_levels'] = node.attrs['n_levels'] + + return ret + + def parseNodeCtxt(self, + ctxt: NetworkContext, + node: gs.Node, + channels_first: bool = True) -> Tuple[NetworkContext, bool]: + + data_in = ctxt.lookup(node.inputs[0].name) + weights = ctxt.lookup(node.inputs[1].name) + bias = ctxt.lookup(node.inputs[2].name) + data_out = ctxt.lookup(node.outputs[0].name) + self.operatorRepresentation['data_in'] = data_in.name + self.operatorRepresentation['weights'] = weights.name + self.operatorRepresentation['bias'] = bias.name + self.operatorRepresentation['data_out'] = data_out.name + self.operatorRepresentation['size'] = np.prod(data_in.shape) + + return ctxt, True + class RQSiHardswishParser(iHardswishParser): @@ -2080,3 +2114,58 @@ def parseNodeCtxt(self, return newCtxt, True return ctxt, False + +class RQAddParser(AddParser): + + def parseNode(self, node: gs.Node) -> bool: + + if not super().parseNode(node): + return False + + ret = all([ + 'rqs1_mul' in node.attrs, + 'rqs1_add' in node.attrs, + 'rqs1_div' in node.attrs, + 'rqs1_signed' in node.attrs, + any(['rqs1_n_levels' in node.attrs, 'rqs1_n_levels_out' in node.attrs]), + 'rqs2_mul' in node.attrs, + 'rqs2_add' in node.attrs, + 'rqs2_div' in node.attrs, + 'rqs2_signed' in node.attrs, + any(['rqs2_n_levels' in node.attrs, 'rqs2_n_levels_out' in node.attrs]), + 'rqsOut_mul' in node.attrs, + 'rqsOut_add' in node.attrs, + 'rqsOut_div' in node.attrs, + 'rqsOut_signed' in node.attrs, + any(['rqsOut_n_levels' in node.attrs, 'rqsOut_n_levels_out' in node.attrs]), + ]) + + if ret: + if 'rqs1_n_levels' in node.attrs: + self.operatorRepresentation['rqs1_n_levels'] = int(node.attrs['rqs1_n_levels'].values) + else: + self.operatorRepresentation['rqs1_n_levels'] = int(node.attrs['rqs1_n_levels_out'].values) + self.operatorRepresentation['rqs1_mul'] = int(node.attrs['rqs1_mul']) + self.operatorRepresentation['rqs1_add'] = int(node.attrs['rqs1_add']) + self.operatorRepresentation['rqs1_signed'] = int(node.attrs['rqs1_signed'].values) + self.operatorRepresentation['rqs1_log2D'] = int(math.log2(node.attrs['rqs1_div'].values)) + + if 'rqs2_n_levels' in node.attrs: + self.operatorRepresentation['rqs2_n_levels'] = int(node.attrs['rqs2_n_levels'].values) + else: + self.operatorRepresentation['rqs2_n_levels'] = int(node.attrs['rqs2_n_levels_out'].values) + self.operatorRepresentation['rqs2_mul'] = int(node.attrs['rqs2_mul']) + self.operatorRepresentation['rqs2_add'] = int(node.attrs['rqs2_add']) + self.operatorRepresentation['rqs2_signed'] = int(node.attrs['rqs2_signed'].values) + self.operatorRepresentation['rqs2_log2D'] = int(math.log2(node.attrs['rqs2_div'].values)) + + if 'rqsOut_n_levels' in node.attrs: + self.operatorRepresentation['rqsOut_n_levels'] = int(node.attrs['rqsOut_n_levels'].values) + else: + self.operatorRepresentation['rqsOut_n_levels'] = int(node.attrs['rqsOut_n_levels_out'].values) + self.operatorRepresentation['rqsOut_mul'] = int(node.attrs['rqsOut_mul']) + self.operatorRepresentation['rqsOut_add'] = int(node.attrs['rqsOut_add']) + self.operatorRepresentation['rqsOut_signed'] = int(node.attrs['rqsOut_signed'].values) + self.operatorRepresentation['rqsOut_log2D'] = int(math.log2(node.attrs['rqsOut_div'].values)) + + return ret diff --git a/Deeploy/Targets/Generic/Templates/iNoNormTemplate.py b/Deeploy/Targets/Generic/Templates/iNoNormTemplate.py new file mode 100644 index 0000000..242962e --- /dev/null +++ b/Deeploy/Targets/Generic/Templates/iNoNormTemplate.py @@ -0,0 +1,38 @@ +# ---------------------------------------------------------------------- +# +# File: iNoNormTemplate.py +# +# Last edited: 22.02.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 Deeploy.DeeployTypes import NodeTemplate + + +class _iNoNormTemplate(NodeTemplate): + + def __init__(self, templateStr): + super().__init__(templateStr) + + +referenceTemplate = _iNoNormTemplate(""" +// iNoNorm (Name: ${nodeName}, Op: ${nodeOp}) +SnitchiNoNorm_s${data_in_type.referencedType.typeWidth}_s${data_out_type.referencedType.typeWidth}(${data_in}, ${data_out}, ${weights}, ${bias}, ${size}, ${mul}, ${log2D}); +""") diff --git a/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py b/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py new file mode 100644 index 0000000..30ec181 --- /dev/null +++ b/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py @@ -0,0 +1,60 @@ +# ---------------------------------------------------------------------- +# +# File: iSoftmaxPreAllocatedBuffTemplate.py +# +# Last edited: 09.07.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Moritz Scherer, scheremo@iis.ee.ethz.ch, ETH Zurich +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Dict, List, Tuple, Union + +from ortools.constraint_solver.pywrapcp import IntVar + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class iSoftmaxPreAllocatedBuffTemplate(NodeTemplate): + + @staticmethod + def computeTransientBuffersSize(ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> List[Tuple[str, Union[int, IntVar]]]: + + lastDimBuffer_dim = 8 * 4 * operatorRepresentation['lastDimLength'] + lastDimBuffer_name = operatorRepresentation['nodeName'] + "_lastDimBuffer" + return [(lastDimBuffer_name, lastDimBuffer_dim)] + + def hoistTransientBuffers(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + lastDimBuffer_name, lastDimBuffer_dim = iSoftmaxPreAllocatedBuffTemplate.computeTransientBuffersSize( + ctxt, operatorRepresentation)[0] + ctxt.hoistTransientBuffer(lastDimBuffer_name, lastDimBuffer_dim) + + operatorRepresentation['lastDimBuffer'] = lastDimBuffer_name + operatorRepresentation['lastDimBufferSize'] = lastDimBuffer_dim + return ctxt, operatorRepresentation, [lastDimBuffer_name] + + def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + + signedI = ctxt.lookup(operatorRepresentation['data_in'])._type.referencedType.typeMin < 0 + signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 + + operatorRepresentation['input_signed'] = signedI + operatorRepresentation['output_signed'] = signedO + + return ctxt, operatorRepresentation, [] diff --git a/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py b/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py index 96eb222..064a34e 100644 --- a/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py +++ b/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py @@ -24,13 +24,14 @@ # limitations under the License. import copy +from collections import OrderedDict from functools import partial from typing import List import numpy as np import onnx_graphsurgeon as gs -from Deeploy.CommonExtensions.OptimizationPasses.Matchers import Match, NonBranchingMatcher +from Deeploy.CommonExtensions.OptimizationPasses.Matchers import BranchingMatcher, Match, NonBranchingMatcher from Deeploy.CommonExtensions.OptimizationPasses.PassClasses import ReplaceSequentialPatternPass, contextagnostic @@ -865,3 +866,134 @@ def __init__(self): name = "_SPLIT_RequantShift_PASS" super().__init__(graph, partial(_split_rqs_fun, splitSet = self.splitSet), name) + +def _merge_add_rq_fun(graph: gs.Graph, match: Match, name: str): + + nodes_map = match.nodes_map + addNode = nodes_map['add'] + + rqDict = OrderedDict([("rqs1", None), ("rqs2", None), ("rqsOut", None)]) + + for key, node in nodes_map.items(): + + if node.outputs[0].name == addNode.inputs[0].name: + rqDict['rqs1'] = node + elif node.outputs[0].name == addNode.inputs[1].name: + rqDict['rqs2'] = node + elif node.inputs[0].name == addNode.outputs[0].name: + rqDict['rqsOut'] = node + + newAttrs = copy.copy(addNode.attrs) + newInputs = [] + + if rqDict['rqsOut'] is not None: + newOutputs = rqDict['rqsOut'].outputs + else: + newOutputs = addNode.outputs + + defaultAttrs = { + "mul": 1, + "add": 0, + "div": gs.Constant('div', np.array(1)), + 'shift': gs.Constant('div', np.array(0)) + } + guessAttrs = {"n_levels_out": 256, "signed": np.array([True])} + for idx, (rqKey, node) in enumerate(rqDict.items()): + if node.op == "RequantShift": + for key, attr in node.attrs.items(): + newAttrs[f"{rqKey}_{key}"] = attr + + if np.prod(node.inputs[1].values.shape) != 1: + return graph + + if np.prod(node.inputs[2].values.shape) != 1: + return graph + + if rqKey != 'rqsOut': + newInputs.append(node.inputs[0]) + + newAttrs[f"{rqKey}_mul"] = int(node.inputs[1].values.item()) + newAttrs[f"{rqKey}_add"] = int(node.inputs[2].values.item() + newAttrs[f"{rqKey}_div"].values.item() // 2) + newAttrs[f"{rqKey}_shift"] = int(np.log2(newAttrs[f"{rqKey}_div"].values.item())) + + else: + for key, attr in defaultAttrs.items(): + newAttrs[f"{rqKey}_{key}"] = attr + + for key, attr in guessAttrs.items(): + if not key in node.attrs: + newAttrs[f"{rqKey}_{key}"] = attr + else: + newAttrs[f"{rqKey}_{key}"] = node.attrs[key] + if rqKey != 'rqsOut': + newInputs.append(addNode.inputs[idx]) + + rqAdd = gs.Node(op = "RequantizedAdd", name = name, attrs = newAttrs) + graph.replaceInsertNode(newInputs, newOutputs, rqAdd) + + return graph + + +@contextagnostic +class AddRequantMergePass(ReplaceSequentialPatternPass): + pass + + def __init__(self): + _input1 = gs.Variable(name = 'input_1') + _input2 = gs.Variable(name = 'input_2') + _addIn1 = gs.Variable(name = 'addIn1') + _addIn2 = gs.Variable(name = 'addIn2') + _addOut = gs.Variable(name = 'addOut') + _rqs = gs.Variable(name = 'rqs') + + anyIn1 = gs.Node(inputs = [_input1], outputs = [_addIn1], op = r'.*', name = 'any1') + anyIn2 = gs.Node(inputs = [_input2], outputs = [_addIn2], op = r'.*', name = 'any2') + + addOut = gs.Node(inputs = [_addIn1, _addIn2], outputs = [_addOut], op = 'Add', name = 'add') + output = gs.Node(inputs = [_addOut], outputs = [_rqs], op = r'RequantShift', name = 'rqsOut') + + graph = gs.Graph(nodes = [anyIn1, anyIn2, addOut, output], inputs = [_input1, _input2], outputs = [_rqs]) + + super().__init__(graph, + replacement_fn = _merge_add_rq_fun, + name = "_MERGE_ADDRQ_PASS", + matcher = BranchingMatcher(regex_op = True)) + + +def merge_gemm_rq_fun(graph: gs.Graph, match: Match, name: str): + matched_nodes = [m for k, m in match.nodes_map.items()] + gemm = matched_nodes[0] + rqs = matched_nodes[1] + + # WIESEP: Per element quantization is not supported for RQGemm + if len(rqs.inputs[2].shape) > 0 and rqs.inputs[2].shape[-1] != 1: + return graph + + # WIESEP: Per column quantization is not supported for RQGemm + if len(rqs.inputs[2].shape) > 2 and rqs.inputs[2].shape[-3] != 1: + return graph + + _inputs = list(gemm.inputs) + list(rqs.inputs[2:]) + list(rqs.inputs[1:2]) + _outputs = rqs.outputs + + attrs = {**gemm.attrs, **rqs.attrs} + rqsGemm = gs.Node(op = 'RQGemm', name = name, attrs = attrs) + graph.replaceInsertNode(_inputs, _outputs, rqsGemm) + + return graph + + +@contextagnostic +class GEMMRequantMergePass(ReplaceSequentialPatternPass): + + def __init__(self): + passes = [] + graph = gs.Graph() + _input = gs.Variable(name = 'input_1') + output = graph.layer(inputs = [_input], outputs = ['matmul_out'], op = 'Gemm', name = 'gemm') + output = graph.layer(inputs = output, outputs = ['rqs'], op = 'RequantShift', name = 'rqs') + graph.outputs.append(output) + graph.inputs.append(_input) + + name = f"_MERGE_GEMM_RQ_PASS" + super().__init__(graph, merge_gemm_rq_fun, name) diff --git a/Deeploy/Targets/Generic/TypeCheckers.py b/Deeploy/Targets/Generic/TypeCheckers.py index cb88602..bd850c2 100644 --- a/Deeploy/Targets/Generic/TypeCheckers.py +++ b/Deeploy/Targets/Generic/TypeCheckers.py @@ -25,7 +25,7 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import List, Optional, Sequence, Type +from typing import Dict, List, Optional, Sequence, Type import numpy as np @@ -392,6 +392,21 @@ def _inferSignedness(self, inputs: List[VariableBuffer], return [False] +class iNoNormChecker(SignPropTypeChecker): + + def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): + super().__init__(input_types, output_types) + + def _inferNumLevels(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[int]: + return [2**(4 * self.input_types[0].referencedType.typeWidth)] + + def _inferSignedness(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[bool]: + if inputs[0]._signed: + return [True] + else: + return [False] + + class GELUChecker(SignPropTypeChecker): def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): diff --git a/Deeploy/Targets/Snitch/Bindings.py b/Deeploy/Targets/Snitch/Bindings.py new file mode 100644 index 0000000..f38be6e --- /dev/null +++ b/Deeploy/Targets/Snitch/Bindings.py @@ -0,0 +1,68 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchBindings.py +# +# Last edited: 30.05.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 functools import partial + +from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.CommonExtensions.CodeTransformationPasses.MemoryAllocation import ArgumentStructGeneration, MemoryManagementGeneration +from Deeploy.CommonExtensions.CodeTransformationPasses.Closure import ClosureGeneration, MemoryAwareClosureGeneration +from Deeploy.TilingExtension.CodeTransformationPasses.TilingVariableReplacement import TilingVariableReplacement +from Deeploy.FutureExtension.CodeTransformationPasses.FutureCodeTransformation import FutureGeneration +from Deeploy.Targets.Snitch.CodeTransformationPasses import SnitchClusterTiling, SnitchCoreFilterPass, \ + SnitchSynchCoresPass, SnitchProfileExecutionBlockPass +from Deeploy.CommonExtensions.DataTypes import int8_t, int32_t, uint8_t +from Deeploy.DeeployTypes import CodeTransformation, NodeBinding +from Deeploy.Targets.Generic.Templates import iNoNormTemplate +from Deeploy.Targets.Snitch.Templates import AddTemplate, RQAddTemplate, iSoftmaxTemplate +from Deeploy.Targets.Snitch.Templates.GemmTemplate import SnitchGemm_Template +from Deeploy.Targets.Snitch.Templates.RqGemmTemplate import SnitchRqGemm_Template +from Deeploy.Targets.Generic.TypeCheckers import AddChecker, GEMMChecker, SoftmaxChecker, iNoNormChecker +from Deeploy.Targets.Snitch.TypeCheckers import SnitchRQAddChecker + + +TilingCallClosure = partial(ClosureGeneration, closureSuffix = "_tiling_closure") +MemoryAwareFunctionCallClosure = partial(MemoryAwareClosureGeneration, closureSuffix = "_closure", startRegion = "L2", endRegion = "L1") + +BasicTransformer = CodeTransformation([SnitchSynchCoresPass(), ArgumentStructGeneration(), MemoryManagementGeneration(), FutureGeneration()]) + +TiledTransformer = CodeTransformation([ + SnitchCoreFilterPass("compute"), + SnitchProfileExecutionBlockPass(), + TilingVariableReplacement("L1"), + TilingCallClosure(writeback = False), + SnitchSynchCoresPass(), + SnitchClusterTiling("L1"), + ArgumentStructGeneration(), + MemoryManagementGeneration("L1"), + MemoryAwareFunctionCallClosure(writeback = False, generateStruct = True), + MemoryManagementGeneration() +]) + +SnitchiSoftmaxBindings = [NodeBinding(SoftmaxChecker([PointerClass(_type)], [PointerClass(uint8_t)]), iSoftmaxTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t, uint8_t]] +SnitchiNoNormBindings = [NodeBinding(iNoNormChecker([PointerClass(_type), PointerClass(int8_t), PointerClass(int32_t)], [PointerClass(int8_t)]), iNoNormTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] +SnitchRQAddBindings = [NodeBinding(SnitchRQAddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(_type)]), RQAddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] +SnitchAddBindings = [NodeBinding(AddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(int32_t)]), AddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] +SnitchGemmBindings = [NodeBinding(GEMMChecker([PointerClass(int8_t), PointerClass(int8_t), PointerClass(int32_t)], [PointerClass(int32_t)]), SnitchGemm_Template, TiledTransformer)] +SnitchRqGemmBindings = [NodeBinding(GEMMChecker([PointerClass(int8_t), PointerClass(int8_t), PointerClass(int32_t), PointerClass(int32_t), PointerClass(int32_t)], [PointerClass(int8_t)]), SnitchRqGemm_Template, TiledTransformer)] diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterSynch.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterSynch.py new file mode 100644 index 0000000..362f6c4 --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterSynch.py @@ -0,0 +1,46 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchClusterSynch.py +# +# Last edited: 31.05.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Tuple + +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ExecutionBlock, NetworkContext, \ + NodeTemplate, _NoVerbosity + +_synchTemplate = NodeTemplate(""" + snrt_cluster_hw_barrier(); +""") + + +class SnitchSynchCoresPass(CodeTransformationPass): + + def apply(self, + ctxt: NetworkContext, + executionBlock: ExecutionBlock, + name: str, + verbose: CodeGenVerbosity = _NoVerbosity) -> Tuple[NetworkContext, ExecutionBlock]: + # TODO: JUNGVI: These have to be core only barriers + executionBlock.addLeft(_synchTemplate, {}) + executionBlock.addRight(_synchTemplate, {}) + return ctxt, executionBlock diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTiling.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTiling.py new file mode 100644 index 0000000..50273a2 --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTiling.py @@ -0,0 +1,49 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchClusterTiling.py +# +# Last edited: 31.05.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Tuple + +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ExecutionBlock, NetworkContext, _NoVerbosity + +from .SnitchClusterTilingSB import SnitchClusterTilingGenerationSB + + +class SnitchClusterTiling(CodeTransformationPass): + + def __init__(self, targetMemLevel: str): + self.SB = SnitchClusterTilingGenerationSB(targetMemLevel) + + def apply(self, + ctxt: NetworkContext, + executionBlock: ExecutionBlock, + name: str, + verbose: CodeGenVerbosity = _NoVerbosity) -> Tuple[NetworkContext, ExecutionBlock]: + + if verbose.tilingProfiling == "L2": + raise NotImplementedError("Profiling not implemented for L2") + # ctxt, executionBlock = self.profilingSB.apply(ctxt, executionBlock, name) + else: + ctxt, executionBlock = self.SB.apply(ctxt, executionBlock, name) + return ctxt, executionBlock diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py new file mode 100644 index 0000000..19a05fe --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py @@ -0,0 +1,504 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchClusterTilingSB.py +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 copy +from collections import namedtuple +from typing import Any, Dict, List, Literal, Tuple +from Deeploy.TilingExtension.CodeTransformationPasses.TilingCodeGeneration import TilingCodeGeneration +from Deeploy.TilingExtension.CodeTransformationPasses.TilingPrototypes import SingleBufferingTilingMixIn, TilingMetaInfo +from Deeploy.Targets.Snitch.DataTypes import Snitch_DMA_copy +from Deeploy.DeeployTypes import ExecutionBlock, NetworkContext, NodeTemplate, OperatorRepresentation, CodeSnippet +from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint +from Deeploy.TilingExtension.TilingCodegen import HyperRectangle, TilingSchedule, VariableReplacementScheme, \ + calculateRectangleOffset, minimizeRectangleDims + +_openTileLoopTemplate = NodeTemplate(""" + +// TILING LOOP +for (int TILING_I=${numTiles}[*${tileIdxPtr}]; TILING_I<${numTiles}[(*${tileIdxPtr})+1]; TILING_I++){ +""") + +_closeTileLoopTemplate = NodeTemplate(""" + +// CLOSE TILING LOOP +} +*${tileIdxPtr} += 1; + +""") + +_moveTileInTemplate = NodeTemplate(""" + +// IMPORT TILE ${innerTilePtr} from ${outerTilePtr} +if(snrt_is_dm_core()){ + ${stateReference}.tid = snrt_dma_start_2d(${stateReference}.dst, + ${stateReference}.src, + ${stateReference}.size, + ${stateReference}.dst_stride, + ${stateReference}.src_stride, + ${stateReference}.repeat); +} +""") + +_iteratedMoveTileInTemplate = NodeTemplate(""" + +""") + +_blockTileInTemplate = NodeTemplate(""" + +// BLOCKING IMPORT TILE ${innerTilePtr} +if(snrt_is_dm_core()){ + // snrt_dma_wait(${stateReference}.tid); + snrt_dma_wait_all(); +} +""") + +_moveTileOutTemplate = NodeTemplate(""" + +// EXPORT TILE ${innerTilePtr} to ${outerTilePtr} +if(snrt_is_dm_core()){ + ${stateReference}.tid = snrt_dma_start_2d(${stateReference}.dst, + ${stateReference}.src, + ${stateReference}.size, + ${stateReference}.dst_stride, + ${stateReference}.src_stride, + ${stateReference}.repeat); +} +""") + +_blockTileOutTemplate = NodeTemplate(""" + +// BLOCKING EXPORT TILE ${innerTilePtr} +if(snrt_is_dm_core()){ + //snrt_dma_wait(${stateReference}.tid); + snrt_dma_wait_all(); +} +""") + +_updateDMATransferStructTemplate = NodeTemplate(""" + +// UPDATE DMA STRUCT ${stateReference} +${stateReference}.dst = ((char*)${dstPtr}) + ${dstOffsetPtr}[${tileNum}]; +${stateReference}.src = ((char*)${srcPtr}) + ${srcOffsetPtr}[${tileNum}]; +${stateReference}.size = ${sizePtr}[${tileNum}]; +${stateReference}.dst_stride = ${dstStridePtr}[${tileNum}]; +${stateReference}.src_stride = ${srcStridePtr}[${tileNum}]; +${stateReference}.repeat = ${repeatPtr}[${tileNum}]; +""") + +_updateReferenceTemplate = NodeTemplate(""" + +// UPDATE VARIABLE ${reference} +*${reference} = ${baseReference}[${tileNum}]; +""") + +_DMAUpdate = namedtuple("_DMAUpdate", "dst src size dst_stride src_stride repeat tid direction") + + +class SnitchClusterTilingSB(TilingCodeGeneration): + + _prefix = "TILING_REPLACED_" + + _openTileLoopTemplate = _openTileLoopTemplate + _closeTileLoopTemplate = _closeTileLoopTemplate + + _moveTileInTemplate = _moveTileInTemplate + _iteratedMoveTileInTemplate = _iteratedMoveTileInTemplate + _blockTileInTemplate = _blockTileInTemplate + + _moveTileOutTemplate = _moveTileOutTemplate + _blockTileOutTemplate = _blockTileOutTemplate + + _updateDMATransferStructTemplate = _updateDMATransferStructTemplate + _updateReferenceTemplate = _updateReferenceTemplate + + @property + def prefix(self): + return self._prefix + self.targetMemLevel + "_" + + def _DMAStructName(self, tensorName: str, nodeName: str) -> str: + return f"{self.prefix}_DMA_{nodeName}_{tensorName}" + + @classmethod + def _generatePointerUpdates(cls, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation, + loadSchedule: List[Dict[str, + HyperRectangle]], nodeMemoryConstraint: NodeMemoryConstraint, + tilingSchedule: TilingSchedule) -> Dict[str, _DMAUpdate]: + updateDict = {} + deltaOffsets = {} + + for idx, loadStep in enumerate(loadSchedule): + for _, (key, rect) in enumerate(loadStep.items()): + + if key in tilingSchedule.outputBaseOffsets.keys(): + baseOffsets = tilingSchedule.outputBaseOffsets[key] + direction = "FromL1" + else: + baseOffsets = tilingSchedule.inputBaseOffsets[key] + direction = "ToL1" + + if key not in updateDict.keys(): + updateDict[key] = [] + if key not in deltaOffsets.keys(): + deltaOffsets[key] = 0 + + referenceBuffer = ctxt.lookup(ctxt.lookup(operatorRepresentation[key])._referenceName) + l1Buffer = ctxt.lookup(operatorRepresentation[key]) + + finalMemoryLevel = TilingCodeGeneration.isFinalMemoryLevel(nodeMemoryConstraint, l1Buffer) + + struct = cls._rectToDMAStruct(ctxt, rect, direction, l1Buffer.name, l1Buffer._referenceName, + finalMemoryLevel) + accOffset = calculateRectangleOffset(rect, referenceBuffer) + + lIdx = idx % len(baseOffsets) + + if direction == "ToL1": + src = accOffset + dst = baseOffsets[lIdx] + else: + src = baseOffsets[lIdx] + dst = accOffset + + size = struct.value['size'].value + dst_stride = struct.value['dst_stride'].value + src_stride = struct.value['src_stride'].value + repeat = struct.value['repeat'].value + tid = struct.value['tid'].value + + sol = _DMAUpdate(dst, src, size, dst_stride, src_stride, repeat, tid, direction) + + deltaOffsets[key] = accOffset + updateDict[key].append(sol) + + return updateDict + + @classmethod + def _rectToDMAStruct(cls, ctxt: NetworkContext, rectangle: HyperRectangle, direction: Literal["ToL1", "FromL1"], + L1Name: str, L2Name: str, finalMemoryLevel: bool) -> Snitch_DMA_copy: + + referenceBuffer = ctxt.lookup(L2Name) + + rect, referenceRect = minimizeRectangleDims(rectangle, referenceBuffer) + assert len(rect.dims) <= 3, "Snitch's iDMA only 2D transfers are supported!" + + if direction == "FromL1": + _src = L1Name + _dst = referenceBuffer.name + else: + _src = referenceBuffer.name + _dst = L1Name + + transfer_size = rect.dims[-1] * (referenceBuffer._type.referencedType.typeWidth // 8) + + src_stride = 0 + dst_stride = 0 + repeat = 1 + if len(rect.dims) > 1: + repeat = rect.dims[-2] + if direction == "ToL1": + dst_stride = rect.dims[-1] * (referenceBuffer._type.referencedType.typeWidth // 8) + src_stride = referenceRect.dims[-1] * (referenceBuffer._type.referencedType.typeWidth // 8) + else: + dst_stride = referenceRect.dims[-1] * (referenceBuffer._type.referencedType.typeWidth // 8) + src_stride = rect.dims[-1] * (referenceBuffer._type.referencedType.typeWidth // 8) + + struct = Snitch_DMA_copy( + { + "dst": _dst, + "src": _src, + "size": transfer_size, + "dst_stride": dst_stride, + "src_stride": src_stride, + "repeat": repeat, + "tid": 0 + }, ctxt) + + return struct + + def _hoistDMAUpdates(self, ctxt: NetworkContext, tensorName: str, updateList: List[_DMAUpdate], + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict]: + + operatorRepresentation = operatorRepresentation.copy() + + nodeName = operatorRepresentation['nodeName'] + + dstList = [] + srcList = [] + sizeList = [] + dstStrideList = [] + srcStideList = [] + repeatList = [] + for update in updateList: + dstList.append(int(update.dst)) + srcList.append(int(update.src)) + sizeList.append(int(update.size)) + dstStrideList.append(int(update.dst_stride)) + srcStideList.append(int(update.src_stride)) + repeatList.append(int(update.repeat)) + + dmaName = self._DMAStructName(tensorName, nodeName) + + operatorRepresentation['stateReference'] = dmaName + operatorRepresentation['tileNum'] = "TILING_I" + + if updateList[0].direction == "ToL1": + operatorRepresentation['dstPtr'] = ctxt.lookup(operatorRepresentation[tensorName]).name + operatorRepresentation['srcPtr'] = ctxt.lookup(operatorRepresentation[tensorName])._referenceName + + dstOffsetList = [0] * len(updateList) + srcOffsetList = [srcList[i] - srcList[0] for i in range(0, len(srcList))] + # srcOffsetList = [0] + [sum(sizeList[:i+1]) for i in range(0, len(sizeList)-1)] + else: + operatorRepresentation['dstPtr'] = ctxt.lookup(operatorRepresentation[tensorName])._referenceName + operatorRepresentation['srcPtr'] = ctxt.lookup(operatorRepresentation[tensorName]).name + + dstOffsetList = [dstList[i] - dstList[0] for i in range(0, len(dstList))] + # dstOffsetList = [0] + [sum(sizeList[:i+1]) for i in range(0, len(sizeList)-1)] + srcOffsetList = [0] * len(updateList) + + namePrefix = self.prefix + f"{nodeName}_{tensorName}" + + name = namePrefix + "_dst_offset" + cb = ctxt.ConstantBuffer(name, [len(updateList)], dstOffsetList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'dstOffsetPtr') + + name = namePrefix + "_src_offset" + cb = ctxt.ConstantBuffer(name, [len(updateList)], srcOffsetList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'srcOffsetPtr') + + name = namePrefix + "_size" + cb = ctxt.ConstantBuffer(name, [len(updateList)], sizeList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'sizePtr', + Snitch_DMA_copy.structTypeDict['size']) + + name = namePrefix + "_dst_stride" + cb = ctxt.ConstantBuffer(name, [len(updateList)], dstStrideList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'dstStridePtr', + Snitch_DMA_copy.structTypeDict['dst_stride']) + + name = namePrefix + "_src_stride" + cb = ctxt.ConstantBuffer(name, [len(updateList)], srcStideList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'srcStridePtr', + Snitch_DMA_copy.structTypeDict['src_stride']) + + name = namePrefix + "_repeat" + cb = ctxt.ConstantBuffer(name, [len(updateList)], repeatList) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'repeatPtr', + Snitch_DMA_copy.structTypeDict['repeat']) + + return ctxt, operatorRepresentation + + def _generateEgressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: + + updates = [] + newCtxt = ctxt.copy() + + updateDict = self._generatePointerUpdates(ctxt, operatorRepresentation, tilingSchedule.outputLoadSchedule, + nodeMemoryConstraint, tilingSchedule) + + for key, updateList in updateDict.items(): + + newCtxt, newNodeRep = self._hoistDMAUpdates(newCtxt, key, updateList, operatorRepresentation) + updates.append(CodeSnippet(self._updateDMATransferStructTemplate, newNodeRep)) + + return newCtxt, updates + + def _generateIngressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: + + updates = [] + newCtxt = ctxt.copy() + + updateDict = self._generatePointerUpdates(ctxt, operatorRepresentation, tilingSchedule.inputLoadSchedule, nodeMemoryConstraint, + tilingSchedule) + + for key, updateList in updateDict.items(): + + newCtxt, newNodeRep = self._hoistDMAUpdates(newCtxt, key, updateList, operatorRepresentation) + updates.append(CodeSnippet(self._updateDMATransferStructTemplate, newNodeRep)) + + return newCtxt, updates + + def _generateVariableUpdates(self, tilingSchedule: TilingSchedule, variableReplacement: VariableReplacementScheme, + ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> List[CodeSnippet]: + + updates = [] + + for key in variableReplacement.perTileReplacements.keys(): + + buf = ctxt.lookup(operatorRepresentation[key]) + reference = str(buf._instance) + + updates.append( + CodeSnippet(self._updateReferenceTemplate, { + "reference": reference, + "tileNum": "TILING_I", + "baseReference": buf._referenceName + })) + + return updates + + def _generateDMACode(self, nodeMemoryConstraint: NodeMemoryConstraint, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation, loadSchedule: List[Dict[str, HyperRectangle]], + direction: Literal["ToL1", "FromL1"]) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: + + DMATransferCalls = [] + DMAWaitStatements = [] + transferNodeRep = {} + + loadStep = loadSchedule[0] + + for idx, (key, rectangle) in enumerate(loadStep.items()): + + permName = f"in{idx}_perm" + + externalPtr = ctxt.lookup(ctxt.lookup(operatorRepresentation[key])._referenceName) + internalPtr = ctxt.lookup(operatorRepresentation[key]) + + tensorName = key + nodeName = operatorRepresentation['nodeName'] + dmaName = self._DMAStructName(tensorName, nodeName) + + transferNodeRep = { + **transferNodeRep, + **{ + 'innerTilePtr': str(internalPtr._instance), + "outerTilePtr": str(externalPtr._instance), + "stateReference": dmaName + } + } + + finalMemoryLevel = TilingCodeGeneration.isFinalMemoryLevel(nodeMemoryConstraint, internalPtr) + struct = self._rectToDMAStruct(ctxt, rectangle, direction, internalPtr.name, externalPtr.name, + finalMemoryLevel) + + transferNodeRep["stateStruct"] = struct + _ = ctxt.hoistStruct(struct, dmaName, Snitch_DMA_copy) + ctxt.lookup(dmaName)._users += [operatorRepresentation['nodeName']] + + if permName in operatorRepresentation and direction == "ToL1": + + DMATransferCalls.append(CodeSnippet(self._iteratedMoveTileInTemplate, transferNodeRep)) + else: + DMATransferCalls.append(CodeSnippet(self._moveTileInTemplate, transferNodeRep)) + + DMAWaitStatements.append(CodeSnippet(self._blockTileInTemplate, transferNodeRep)) + + return DMATransferCalls, DMAWaitStatements + + def _generateIngressDMACode(self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, + ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: + + importLoadStep = tilingSchedule.inputLoadSchedule + ingressDMATransferCalls, ingressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, operatorRepresentation, + importLoadStep, "ToL1") + return ingressDMATransferCalls, ingressDMAWaitStatements + + def _generateEgressDMACode(self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, + ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: + + exportLoadStep = tilingSchedule.outputLoadSchedule + egressDMATransferCalls, egressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, operatorRepresentation, + exportLoadStep, "FromL1") + + return egressDMATransferCalls, egressDMAWaitStatements + + def _tilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, + nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, + variableReplacement: VariableReplacementScheme, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, ExecutionBlock, bool]: + + tileIdxPtr = self._hoistTileIdxPtr(ctxt, operatorRepresentation) + + ingressDMATransferCalls, ingressDMAWaitStatements = self._generateIngressDMACode( + tilingSchedule, nodeMemoryConstraint, ctxt, operatorRepresentation) + + egressDMATransferCalls, egressDMAWaitStatements = self._generateEgressDMACode( + tilingSchedule, nodeMemoryConstraint, ctxt, operatorRepresentation) + + ctxt, ingressDMAUpdates = self._generateIngressPointerUpdates(nodeMemoryConstraint, tilingSchedule, ctxt, + operatorRepresentation) + ctxt, egressDMAUpdates = self._generateEgressPointerUpdates(nodeMemoryConstraint, tilingSchedule, ctxt, operatorRepresentation) + + openLoopStatement = [ + CodeSnippet(self._openTileLoopTemplate, { + "numTiles": operatorRepresentation["numTiles"], + "tileIdxPtr": tileIdxPtr + }) + ] + + closeLoopStatement = [ + CodeSnippet(self._closeTileLoopTemplate, { + "numTiles": operatorRepresentation["numTiles"], + "tileIdxPtr": tileIdxPtr + }) + ] + + variableUpdates = self._generateVariableUpdates(tilingSchedule, variableReplacement, ctxt, operatorRepresentation) + + metaInfo = TilingMetaInfo(nodeName = operatorRepresentation['nodeName'] + "_L2", + nodeOps = operatorRepresentation['nodeOps'], + numTiles = len(tilingSchedule.outputLoadSchedule), + tileIdxVar = "TILING_I") + + newExecutionBlock = self.generateAllTilingCode(executionBlock, metaInfo, ingressDMATransferCalls, + ingressDMAWaitStatements, ingressDMAUpdates, + egressDMATransferCalls, egressDMAWaitStatements, + egressDMAUpdates, variableUpdates, openLoopStatement, + closeLoopStatement, [], []) + + return ctxt, newExecutionBlock, True + + def generateTilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, + nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedules: List[TilingSchedule], + variableReplacement: VariableReplacementScheme, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, ExecutionBlock, bool]: + + flatTilingSchedule = copy.copy(tilingSchedules[0]) + for tilingSchedule in tilingSchedules[1:]: + flatTilingSchedule += tilingSchedule + + # SCHEREMO: hoist numTiles + + offsetLists = list({**flatTilingSchedule.inputBaseOffsets, **flatTilingSchedule.outputBaseOffsets}.values()) + + if len(offsetLists) == 0: + return ctxt, executionBlock, False + + for offsetList in offsetLists: + if not len(offsetList) == 1: + return ctxt, executionBlock, False + + operatorRepresentation["numTiles"] = self._hoistNumTiles(ctxt, operatorRepresentation['nodeName'], tilingSchedules) + + return self._tilingLoop(ctxt, executionBlock, nodeMemoryConstraint, flatTilingSchedule, variableReplacement, + operatorRepresentation) + + +class SnitchClusterTilingGenerationSB(SnitchClusterTilingSB, SingleBufferingTilingMixIn): + pass diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchCoreFilter.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchCoreFilter.py new file mode 100644 index 0000000..6ea21f4 --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchCoreFilter.py @@ -0,0 +1,46 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchCoreFilter.py +# +# Last edited: 04.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Luka Macan, luka.macan@unibo.it, University of Bologna +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Literal, Tuple + +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ExecutionBlock, NetworkContext, \ + NodeTemplate, _NoVerbosity + + +class SnitchCoreFilterPass(CodeTransformationPass): + + def __init__(self, coreType: Literal["dm", "compute"]): + super().__init__() + self.coreType = coreType + + def apply(self, + ctxt: NetworkContext, + executionBlock: ExecutionBlock, + name: str, + verbose: CodeGenVerbosity = _NoVerbosity) -> Tuple[NetworkContext, ExecutionBlock]: + executionBlock.addLeft(NodeTemplate(f"if (snrt_is_{self.coreType}_core()) {{\n"), {}) + executionBlock.addRight(NodeTemplate("}\n"), {}) + return ctxt, executionBlock diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchProfileExecutionBlock.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchProfileExecutionBlock.py new file mode 100644 index 0000000..c2cce33 --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchProfileExecutionBlock.py @@ -0,0 +1,52 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchProfileExecutionBlock.py +# +# Last edited: 05.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Tuple + +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ExecutionBlock, NetworkContext, \ + NodeTemplate, _NoVerbosity + +_dumpCycleCntTemplate = NodeTemplate(""" + snrt_cluster_hw_barrier(); + if (snrt_is_dm_core()) { + #ifndef BANSHEE_SIMULATION + DUMP(getCycles()); + #else + printf("${position} of ${nodeName} block at cycle %d \\n", getCycles()); + #endif + } +""") + + +class SnitchProfileExecutionBlockPass(CodeTransformationPass): + + def apply(self, + ctxt: NetworkContext, + executionBlock: ExecutionBlock, + name: str, + verbose: CodeGenVerbosity = _NoVerbosity) -> Tuple[NetworkContext, ExecutionBlock]: + executionBlock.addLeft(_dumpCycleCntTemplate, {"position": "Start", "nodeName": name}) + executionBlock.addRight(_dumpCycleCntTemplate, {"position": "End", "nodeName": name}) + return ctxt, executionBlock diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/__init__.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/__init__.py new file mode 100644 index 0000000..d3281dd --- /dev/null +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/__init__.py @@ -0,0 +1,29 @@ +# ---------------------------------------------------------------------- +# +# File: __init__.py +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 .SnitchClusterSynch import * +from .SnitchClusterTiling import * +from .SnitchCoreFilter import * +from .SnitchProfileExecutionBlock import * diff --git a/Deeploy/Targets/Snitch/DataTypes.py b/Deeploy/Targets/Snitch/DataTypes.py new file mode 100644 index 0000000..b1d3a92 --- /dev/null +++ b/Deeploy/Targets/Snitch/DataTypes.py @@ -0,0 +1,40 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchDataTypes.py +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 Deeploy.AbstractDataTypes import PointerClass, Struct, VoidType +from Deeploy.CommonExtensions.DataTypes import uint16_t + + +class Snitch_DMA_copy(Struct): + typeName = "DMA_copy" + structTypeDict = { + "dst": PointerClass(VoidType), + "src": PointerClass(VoidType), + "size": uint16_t, + "dst_stride": uint16_t, + "src_stride": uint16_t, + "repeat": uint16_t, + "tid": uint16_t + } diff --git a/Deeploy/Targets/Snitch/Deployer.py b/Deeploy/Targets/Snitch/Deployer.py index ff32066..870a984 100644 --- a/Deeploy/Targets/Snitch/Deployer.py +++ b/Deeploy/Targets/Snitch/Deployer.py @@ -29,13 +29,12 @@ import onnx_graphsurgeon as gs from Deeploy.AbstractDataTypes import Pointer -from Deeploy.CommonExtensions.NetworkDeployers.SignPropDeployer import SignPropDeployer -from Deeploy.CommonExtensions.OptimizationPasses.TopologyOptimizationPasses.LoweringOptimizationPasses import \ - NCHWtoNHWCPass, RemoveGlobalOutputReshapePass, TransposeMatmulInputsPass from Deeploy.DeeployTypes import DeploymentPlatform, TopologyOptimizer +from Deeploy.CommonExtensions.NetworkDeployers.SignPropDeployer import SignPropDeployer from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import ReshapeConstOptPass, TransposeConstOptPass, \ TransposeMergePass, TransposeSplitPass - +from Deeploy.CommonExtensions.OptimizationPasses.TopologyOptimizationPasses.LoweringOptimizationPasses import \ + NCHWtoNHWCPass, RemoveGlobalOutputReshapePass, TransposeMatmulInputsPass class SnitchDeployer(SignPropDeployer): diff --git a/Deeploy/Targets/Snitch/Parser.py b/Deeploy/Targets/Snitch/Parser.py new file mode 100644 index 0000000..57b91a8 --- /dev/null +++ b/Deeploy/Targets/Snitch/Parser.py @@ -0,0 +1,96 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchParsers.py +# +# Last edited: 07.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# - Luka Macan, luka.macan@unibo.it, University of Bologna +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Tuple + +import onnx_graphsurgeon as gs + +from Deeploy.DeeployTypes import NetworkContext +from Deeploy.Targets.Generic.Parsers import GEMMParser, RQGEMMParser + + +class SnitchGEMMParser(GEMMParser): + + def parseNode(self, node: gs.Node) -> bool: + ret = super().parseNode(node) + + if not ret: + return False + + if not all([ + self.operatorRepresentation['transA'] == 0, + ]): + return False + + return True + + def parseNodeCtxt(self, + ctxt: NetworkContext, + node: gs.Node, + channels_first: bool = True) -> Tuple[NetworkContext, bool]: + newCtxt, ret = super().parseNodeCtxt(ctxt, node, channels_first) + + if not ret: + return ctxt, False + + if not all([ + self.operatorRepresentation['batch'] == 1, + ]): + return ctxt, False + + return newCtxt, True + + +class SnitchRQGEMMParser(RQGEMMParser): + + def parseNode(self, node: gs.Node) -> bool: + ret = super().parseNode(node) + + if not ret: + return False + + if not all([ + self.operatorRepresentation['transA'] == 0, + ]): + return False + + return True + + def parseNodeCtxt(self, + ctxt: NetworkContext, + node: gs.Node, + channels_first: bool = True) -> Tuple[NetworkContext, bool]: + newCtxt, ret = super().parseNodeCtxt(ctxt, node, channels_first) + + if not ret: + return ctxt, False + + if not all([ + self.operatorRepresentation['batch'] == 1, + ]): + return ctxt, False + + return newCtxt, True diff --git a/Deeploy/Targets/Snitch/Platform.py b/Deeploy/Targets/Snitch/Platform.py index 89cb97e..06321d7 100644 --- a/Deeploy/Targets/Snitch/Platform.py +++ b/Deeploy/Targets/Snitch/Platform.py @@ -28,18 +28,27 @@ import numpy as np + +from Deeploy.Targets.Generic.Bindings import BasicGatherBindings,BasicMatMulBinding, BasicPad1DBindings, \ + BasicPad2DBindings, BasicReshapeBindings, BasicRQIntegerDivBinding, BasicLayerNormBinding from Deeploy.DeeployTypes import ConstantBuffer, DeploymentEngine, DeploymentPlatform, NodeMapper, NodeTemplate, \ StructBuffer, TopologyOptimizer, TransientBuffer, VariableBuffer -from Deeploy.Targets.Generic.Bindings import BasicGatherBindings, BasicMatMulBinding, BasicPad1DBindings, \ - BasicPad2DBindings, BasicReshapeBindings, BasicRQIntegerDivBinding -from Deeploy.Targets.Generic.Layers import GatherLayer, MatMulLayer, PadLayer, ReshapeLayer, RQIntegerDivLayer -from Deeploy.Targets.Generic.Parsers import GatherParser, MatMulParser, Pad1DParser, Pad2DParser, RQIntegerDivParser, \ - UnsqueezeParser +from Deeploy.Targets.Generic.Layers import AddLayer, GatherLayer, GEMMLayer, MatMulLayer, PadLayer, ReshapeLayer, \ + RQGEMMLayer, RQIntegerDivLayer, iNoNormLayer, iSoftmaxLayer, iLayerNormLayer +from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import AddRequantMergePass, \ + GEMMRequantMergePass, IntegerDivRequantMergePass, MergeConstAddAndRequantPass, \ + MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, SkipEmptyConcatPass, SkipUnityRequantPass, \ + iGELURequantMergePass, iHardswishRequantMergePass +from Deeploy.Targets.Generic.Parsers import AddParser, GatherParser, MatMulParser, Pad1DParser, Pad2DParser, RQAddParser, \ + RQIntegerDivParser, UnsqueezeParser, iNoNormParser, iSoftmaxParser, iLayerNormParser +from Deeploy.Targets.Snitch.Parser import SnitchGEMMParser, SnitchRQGEMMParser +from Deeploy.Targets.PULPOpen.Platform import RQAddMapper from Deeploy.Targets.Generic.Templates import AllocateTemplate as BasicAllocateTemplate -from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import IntegerDivRequantMergePass, \ - MergeConstAddAndRequantPass, MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, SkipEmptyConcatPass, \ - SkipUnityRequantPass, iGELURequantMergePass, iHardswishRequantMergePass from Deeploy.Targets.Snitch.Templates import AllocateTemplate, FreeTemplate +from Deeploy.Targets.Snitch.Tiler import SnitchAddTileReadyBindings, SnitchGemmTilingReadyBindings, \ + SnitchiNoNormTilingReadyBindings, SnitchiSoftmaxTilingReadyBindings, SnitchRQAddTilingReadyBindings, \ + SnitchRqGemmTilingReadyBindings + GatherMapper = NodeMapper(GatherParser(), BasicGatherBindings) Pad1DMapper = NodeMapper(Pad1DParser(), BasicPad1DBindings) @@ -49,6 +58,13 @@ RQIntegerDivMapper = NodeMapper(RQIntegerDivParser(), [BasicRQIntegerDivBinding]) MatMulMapper = NodeMapper(MatMulParser(), [BasicMatMulBinding]) +GemmMapper = NodeMapper(SnitchGEMMParser(), SnitchGemmTilingReadyBindings) +RqGemmMapper = NodeMapper(SnitchRQGEMMParser(), SnitchRqGemmTilingReadyBindings) +iSoftmaxMapper = NodeMapper(iSoftmaxParser(), SnitchiSoftmaxTilingReadyBindings) +iNoNormMapper = NodeMapper(iNoNormParser(), SnitchiNoNormTilingReadyBindings) +iLayerNormMapper = NodeMapper(iLayerNormParser(), [BasicLayerNormBinding]) +RQAddMapper = NodeMapper(RQAddParser(), SnitchRQAddTilingReadyBindings) +AddMapper = NodeMapper(AddParser(), SnitchAddTileReadyBindings) SnitchMapping = { 'RQIntegerDiv': RQIntegerDivLayer([RQIntegerDivMapper]), @@ -56,6 +72,13 @@ 'Pad': PadLayer([Pad1DMapper, Pad2DMapper]), 'Unsqueeze': ReshapeLayer([UnsqueezeMapper]), 'MatMul': MatMulLayer([MatMulMapper]), + 'Gemm': GEMMLayer([GemmMapper]), + 'RQGemm': RQGEMMLayer([RqGemmMapper]), + 'iSoftmax': iSoftmaxLayer([iSoftmaxMapper]), + 'iNoNorm': iNoNormLayer([iNoNormMapper]), + 'iLayerNorm': iLayerNormLayer([iLayerNormMapper]), + 'RequantizedAdd': AddLayer([RQAddMapper]), + 'Add': AddLayer([AddMapper]), } @@ -136,6 +159,8 @@ class SnitchStructBuffer(StructBuffer): iGELURequantMergePass(), iHardswishRequantMergePass(), MergeConstAddAndRequantPass(), + AddRequantMergePass(), + GEMMRequantMergePass(), ]) _includeList = [ diff --git a/Deeploy/Targets/Snitch/Templates/AddTemplate.py b/Deeploy/Targets/Snitch/Templates/AddTemplate.py new file mode 100644 index 0000000..8b95896 --- /dev/null +++ b/Deeploy/Targets/Snitch/Templates/AddTemplate.py @@ -0,0 +1,57 @@ +# ---------------------------------------------------------------------- +# +# File: AddTemplate.py +# +# Last edited: 11.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Dict, List, Tuple + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class _SnitchAddTemplate(NodeTemplate): + + def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + + data_in_1 = ctxt.lookup(operatorRepresentation['data_in_1']) + data_in_2 = ctxt.lookup(operatorRepresentation['data_in_2']) + data_out = ctxt.lookup(operatorRepresentation['data_out']) + + input_1_offset = 0 + if hasattr(data_in_1, "_signed") and hasattr(data_in_1, "nLevels"): + input_1_offset = (data_in_1._signed == 0) * int(data_in_1.nLevels / 2) + input_2_offset = 0 + if hasattr(data_in_2, "_signed") and hasattr(data_in_2, "nLevels"): + input_2_offset = (data_in_2._signed == 0) * int(data_in_2.nLevels / 2) + output_offset = 0 + if hasattr(data_out, "_signed") and hasattr(data_out, "nLevels"): + output_offset = -(data_out._signed == 0) * int(data_out.nLevels // 2) + + operatorRepresentation['offset'] = input_1_offset + input_2_offset + output_offset + + return ctxt, operatorRepresentation, [] + + +referenceTemplate = _SnitchAddTemplate(""" +// Snitch Add (Name: ${nodeName}, Op: ${nodeOp}) +SnitchAdd(${data_in_1}, ${data_in_2}, ${data_out}, ${size}, ${offset}); +""") diff --git a/Deeploy/Targets/Snitch/Templates/GemmTemplate.py b/Deeploy/Targets/Snitch/Templates/GemmTemplate.py new file mode 100644 index 0000000..16e2c6e --- /dev/null +++ b/Deeploy/Targets/Snitch/Templates/GemmTemplate.py @@ -0,0 +1,28 @@ +from typing import Dict, List, Tuple + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class SnitchGemmTemplate(NodeTemplate): + + def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + if isinstance(operatorRepresentation['alpha'], float): + assert operatorRepresentation['alpha'].is_integer(), f"Parameter alpha is not an integer: {operatorRepresentation['alpha']}" + operatorRepresentation['alpha'] = int(operatorRepresentation['alpha']) + if isinstance(operatorRepresentation['beta'], float): + assert operatorRepresentation['beta'].is_integer(), f"Parameter beta is not an integer: {operatorRepresentation['beta']}" + operatorRepresentation['beta'] = int(operatorRepresentation['beta']) + + if operatorRepresentation['transB']: + operatorRepresentation['kernelName'] = "Gemm_s8_transB_row_parallel" + else: + operatorRepresentation['kernelName'] = "Gemm_s8_row_parallel" + + return ctxt, operatorRepresentation, [] + + +SnitchGemmTemplateStr = r""" +${kernelName}(${A}, ${B}, ${C}, ${data_out}, ${M}, ${N}, ${O}, ${alpha}, ${beta}); +""" + +SnitchGemm_Template = SnitchGemmTemplate(SnitchGemmTemplateStr) diff --git a/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py new file mode 100644 index 0000000..0582818 --- /dev/null +++ b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py @@ -0,0 +1,70 @@ +# ---------------------------------------------------------------------- +# +# File: RQAddTemplate.py +# +# Last edited: 11.11.2023 +# +# Copyright (C) 2023, ETH Zurich and University of Bologna. +# +# Author: +# - Moritz Scherer, ETH Zurich +# - Victor Jung, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# 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 typing import Dict, List, Tuple + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class SnitchRQAddTemplate(NodeTemplate): + + def __init__(self, templateStr): + super().__init__(templateStr) + + def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + # Extract signedness information of input, weights and output + signedI2 = ctxt.lookup(operatorRepresentation['data_in_2'])._type.referencedType.typeMin < 0 + signedI = ctxt.lookup(operatorRepresentation['data_in_1'])._type.referencedType.typeMin < 0 + signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 + operatorRepresentation['input_2_signed'] = signedI2 + operatorRepresentation['input_signed'] = signedI + operatorRepresentation['output_signed'] = signedO + + return ctxt, operatorRepresentation, [] + + +referenceTemplate = SnitchRQAddTemplate(""" + +<% +signatureString = '' +if input_signed: + signatureString += '_i8' +else: + signatureString += '_u8' +if input_2_signed: + signatureString += '_i8' +else: + signatureString += '_u8' +if output_signed: + signatureString += '_i8' +else: + signatureString += '_u8' +%> + +// PULP NN RQADD +pulp_nn_add${signatureString}(${data_in_1}, ${data_in_2}, ${data_out}, ${rqs1_mul}, ${rqs1_add}, ${rqs1_log2D}, ${rqs2_mul}, ${rqs2_add}, ${rqs2_log2D}, ${rqsOut_mul}, ${rqsOut_add}, ${rqsOut_log2D}, 1, ${size}, 1, 1); +""") diff --git a/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py b/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py new file mode 100644 index 0000000..a51d31d --- /dev/null +++ b/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py @@ -0,0 +1,32 @@ +from typing import Dict, List, Tuple + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class SnitchRqGemmTemplate(NodeTemplate): + + def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + if isinstance(operatorRepresentation['alpha'], float): + assert operatorRepresentation['alpha'].is_integer() + operatorRepresentation['alpha'] = int(operatorRepresentation['alpha']) + if isinstance(operatorRepresentation['beta'], float): + assert operatorRepresentation['beta'].is_integer() + operatorRepresentation['beta'] = int(operatorRepresentation['beta']) + + #LMACAN: WARNING: Assumes rounding is expected + add = ctxt.lookup(operatorRepresentation['add']) + add.values += 2**(operatorRepresentation['log2D'] - 1) + + if operatorRepresentation['transB']: + operatorRepresentation['kernelName'] = 'RQGemm_s8_transB_row_parallel_unrolled' + else: + operatorRepresentation['kernelName'] = 'RQGemm_s8_row_parallel_unrolled' + + return ctxt, operatorRepresentation, [] + + +SnitchRqGemmTemplateStr = r""" +${kernelName}(${A}, ${B}, ${C}, ${data_out}, ${M}, ${N}, ${O}, ${alpha}, ${beta}, ${mul}, ${add}, ${log2D}); +""" + +SnitchRqGemm_Template = SnitchRqGemmTemplate(SnitchRqGemmTemplateStr) diff --git a/Deeploy/Targets/Snitch/Templates/iSoftmaxTemplate.py b/Deeploy/Targets/Snitch/Templates/iSoftmaxTemplate.py new file mode 100644 index 0000000..9a15d91 --- /dev/null +++ b/Deeploy/Targets/Snitch/Templates/iSoftmaxTemplate.py @@ -0,0 +1,41 @@ +# ---------------------------------------------------------------------- +# +# File: iSoftmaxTemplate.py +# +# Last edited: 30.05.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 Deeploy.Targets.Generic.Templates.iSoftmaxPreAllocatedBuffTemplate import iSoftmaxPreAllocatedBuffTemplate + +referenceTemplate = iSoftmaxPreAllocatedBuffTemplate(""" +<% +signatureString = '' +if input_signed: + signatureString += '_i8' +else: + signatureString += '_u8' +if output_signed: + signatureString += '_i8' +else: + signatureString += '_u8' +%> +SnitchSoftmax${signatureString}(${data_in}, ${data_out}, ${lastDimBuffer}, ${size}, ${lastDimLength}, ${coeffB}, ${coeffC}, ${log2}); +""") diff --git a/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py new file mode 100644 index 0000000..b95be32 --- /dev/null +++ b/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py @@ -0,0 +1,159 @@ +from typing import Dict, List, Tuple + +from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.CommonExtensions.DataTypes import uint32_t +from Deeploy.DeeployTypes import NetworkContext, OperatorRepresentation +from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint +from Deeploy.TilingExtension.TileConstraint import TileConstraint +from Deeploy.TilingExtension.TilerModel import PerformanceHint, TilerModel +from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, HyperRectangle, TilingSchedule, \ + VariableReplacementScheme + +class GemmTileConstraint(TileConstraint): + + @staticmethod + def addGeometricalConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + + # Get to-be-tiled tensor's buffers + bufferA = ctxt.lookup(name = parseDict['A']) + bufferB = ctxt.lookup(name = parseDict['B']) + bufferY = ctxt.lookup(name = parseDict['data_out']) + + # Add I/O dimensions to the model as variables + for bufferName in [bufferA.name, bufferB.name, bufferY.name]: + tilerModel.addTensorDimToModel(ctxt, bufferName) + + dimCountA = len(bufferA.shape) + if parseDict['transA'] == 0: + heightIdxA, widthIdxA = dimCountA - 2, dimCountA - 1 + else: + heightIdxA, widthIdxA = dimCountA - 1, dimCountA - 2 + AHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = heightIdxA) + AWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = widthIdxA) + + dimCountB = len(bufferB.shape) + if parseDict['transB'] == 0: + heightIdxB, widthIdxB = dimCountB - 2, dimCountB - 1 + else: + heightIdxB, widthIdxB = dimCountB - 1, dimCountB - 2 + BHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferB.name, dimIdx = heightIdxB) + BWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferB.name, dimIdx = widthIdxB) + + dimCountY = len(bufferY.shape) + heightIdxY, widthIdxY = dimCountY - 2, dimCountY - 1 + YHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = heightIdxY) + YWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = widthIdxY) + + tilerModel.addConstraint(YHeightDimVar == AHeightDimVar) + tilerModel.addConstraint(YWidthDimVar == BWidthDimVar) + tilerModel.addConstraint(AWidthDimVar == BHeightDimVar) + + if 'C' in parseDict: + bufferC = ctxt.lookup(name = parseDict['C']) + + tilerModel.addTensorDimToModel(ctxt, bufferC.name) + + dimCountC = len(bufferC.shape) + heightIdxC, widthIdxC = dimCountC - 2, dimCountC - 1 + CHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferC.name, dimIdx = heightIdxC) + CWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferC.name, dimIdx = widthIdxC) + + tilerModel.addConstraint(CHeightDimVar == YHeightDimVar) + tilerModel.addConstraint(CWidthDimVar == YWidthDimVar) + + return tilerModel + + @staticmethod + def addPolicyConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + bufferA = ctxt.lookup(name = parseDict['A']) + bufferY = ctxt.lookup(name = parseDict['data_out']) + + dimCountA = len(bufferA.shape) + if parseDict['transA'] == 0: + heightIdxA, widthIdxA = dimCountA - 2, dimCountA - 1 + else: + heightIdxA, widthIdxA = dimCountA - 1, dimCountA - 2 + AHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = heightIdxA) + AWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = widthIdxA) + + dimCountY = len(bufferY.shape) + heightIdxY, widthIdxY = dimCountY - 2, dimCountY - 1 + YHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = heightIdxY) + YWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = widthIdxY) + + # Full inner dimension + tilerModel.addConstraint(AWidthDimVar == AWidthDimVar.Max()) + + # We parallelize over the output height dimension so try to keep it divisible by the number of cores (8) + if parseDict["M"] > 8: + tilerModel.addTileSizeDivisibleConstraint(parseDict, + "M", + YHeightDimVar, + 8, + strategy = PerformanceHint(priority = 1)) + + return tilerModel + + @classmethod + def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, + absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + outputCubes = [cube.rectangle for cube in absoluteOutputCubes] + + addrNames = ['A', 'B', 'C', 'data_out'] + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + + NOffset = 0 + NSize = operatorRepresentation["N"] + + replacements = { + "M": [], + "O": [], + "batch": [], + } + + replacementTypes = { + "M": PointerClass(uint32_t), + "O": PointerClass(uint32_t), + "batch": PointerClass(uint32_t), + } + + inputLoadSchedule = [] + outputLoadSchedule = [] + + for YCube in outputCubes: + assert len(YCube.offset) >= 2 or len( + YCube.offset) <= 3, f"Unsupported YCube dimensionality: {len(YCube.offset)}" + + MOffset, OOffset = YCube.offset[-2:] + MSize, OSize = YCube.dims[-2:] + + replacements["M"].append(MSize) + replacements["O"].append(OSize) + + if len(YCube.offset) == 3: + BatchOffset = YCube.offset[0] + BatchSize = YCube.dims[0] + else: + BatchOffset = 0 + BatchSize = 1 + + replacements["batch"].append(BatchSize) + + if operatorRepresentation['transA'] == 0: + ACube = HyperRectangle((BatchOffset, MOffset, NOffset), (BatchSize, MSize, NSize)) + else: + ACube = HyperRectangle((BatchOffset, NOffset, MOffset), (BatchSize, NSize, MSize)) + + if operatorRepresentation['transB'] == 0: + BCube = HyperRectangle((BatchOffset, NOffset, OOffset), (BatchSize, NSize, OSize)) + else: + BCube = HyperRectangle((BatchOffset, OOffset, NOffset), (BatchSize, OSize, NSize)) + + inputLoadSchedule.append({"A": ACube, "B": BCube, "C": YCube}) + outputLoadSchedule.append({"data_out": YCube}) + + schedule = TilingSchedule(inputBaseOffsets, outputBaseOffsets, inputLoadSchedule, outputLoadSchedule) + + return VariableReplacementScheme(replacements, replacementTypes), schedule diff --git a/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py new file mode 100644 index 0000000..517393e --- /dev/null +++ b/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py @@ -0,0 +1,174 @@ +from typing import Dict, List, Tuple + +from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.CommonExtensions.DataTypes import uint32_t +from Deeploy.DeeployTypes import NetworkContext, OperatorRepresentation +from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint +from Deeploy.TilingExtension.TileConstraint import TileConstraint +from Deeploy.TilingExtension.TilerModel import TilerModel, PerformanceHint +from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, HyperRectangle, TilingSchedule, VariableReplacementScheme + +class RqGemmTileConstraint(TileConstraint): + + @staticmethod + def addGeometricalConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + + # Get to-be-tiled tensor's buffers + bufferA = ctxt.lookup(name = parseDict['A']) + bufferB = ctxt.lookup(name = parseDict['B']) + bufferY = ctxt.lookup(name = parseDict['data_out']) + + # Add I/O dimensions to the model as variables + for bufferName in [bufferA.name, bufferB.name, bufferY.name]: + tilerModel.addTensorDimToModel(ctxt, bufferName) + + dimCountA = len(bufferA.shape) + if parseDict['transA'] == 0: + heightIdxA, widthIdxA = dimCountA - 2, dimCountA - 1 + else: + heightIdxA, widthIdxA = dimCountA - 1, dimCountA - 2 + AHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = heightIdxA) + AWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = widthIdxA) + + dimCountB = len(bufferB.shape) + if parseDict['transB'] == 0: + heightIdxB, widthIdxB = dimCountB - 2, dimCountB - 1 + else: + heightIdxB, widthIdxB = dimCountB - 1, dimCountB - 2 + BHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferB.name, dimIdx = heightIdxB) + BWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferB.name, dimIdx = widthIdxB) + + dimCountY = len(bufferY.shape) + heightIdxY, widthIdxY = dimCountY - 2, dimCountY - 1 + YHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = heightIdxY) + YWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = widthIdxY) + + tilerModel.addConstraint(YHeightDimVar == AHeightDimVar) + tilerModel.addConstraint(YWidthDimVar == BWidthDimVar) + tilerModel.addConstraint(AWidthDimVar == BHeightDimVar) + + if 'C' in parseDict: + bufferC = ctxt.lookup(name = parseDict['C']) + + tilerModel.addTensorDimToModel(ctxt, bufferC.name) + + dimCountC = len(bufferC.shape) + heightIdxC, widthIdxC = dimCountC - 2, dimCountC - 1 + CHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferC.name, dimIdx = heightIdxC) + CWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferC.name, dimIdx = widthIdxC) + + tilerModel.addConstraint(CHeightDimVar == YHeightDimVar) + tilerModel.addConstraint(CWidthDimVar == YWidthDimVar) + + bufferMul = ctxt.lookup(name = parseDict['mul']) + bufferAdd = ctxt.lookup(name = parseDict['add']) + + # Add I/O dimensions to the model as variables + for bufferName in [bufferMul.name, bufferAdd.name]: + tilerModel.addTensorDimToModel(ctxt, bufferName) + + MulDimVar = tilerModel.getTensorDimVar(tensorName = bufferMul.name, dimIdx = 0) + AddDimVar = tilerModel.getTensorDimVar(tensorName = bufferAdd.name, dimIdx = 0) + + tilerModel.addConstraint(MulDimVar == YHeightDimVar) + tilerModel.addConstraint(MulDimVar == AddDimVar) + + return tilerModel + + @staticmethod + def addPolicyConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + bufferA = ctxt.lookup(name = parseDict['A']) + bufferY = ctxt.lookup(name = parseDict['data_out']) + + dimCountA = len(bufferA.shape) + if parseDict['transA'] == 0: + heightIdxA, widthIdxA = dimCountA - 2, dimCountA - 1 + else: + heightIdxA, widthIdxA = dimCountA - 1, dimCountA - 2 + AHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = heightIdxA) + AWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferA.name, dimIdx = widthIdxA) + + dimCountY = len(bufferY.shape) + heightIdxY, widthIdxY = dimCountY - 2, dimCountY - 1 + YHeightDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = heightIdxY) + YWidthDimVar = tilerModel.getTensorDimVar(tensorName = bufferY.name, dimIdx = widthIdxY) + + # Full inner dimension + tilerModel.addConstraint(AWidthDimVar == AWidthDimVar.Max()) + + # We parallelize over the output height dimension so try to keep it divisible by the number of cores (8) + if parseDict["M"] > 8: + tilerModel.addTileSizeDivisibleConstraint(parseDict, + "M", + YHeightDimVar, + 8, + strategy = PerformanceHint(priority = 1)) + + return tilerModel + + @classmethod + def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, + absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + outputCubes = [cube.rectangle for cube in absoluteOutputCubes] + + addrNames = ['A', 'B', 'C', 'mul', 'add', 'data_out'] + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + + NOffset = 0 + NSize = operatorRepresentation["N"] + + replacements = { + "M": [], + "O": [], + "batch": [], + } + + replacementTypes = { + "M": PointerClass(uint32_t), + "O": PointerClass(uint32_t), + "batch": PointerClass(uint32_t), + } + + inputLoadSchedule = [] + outputLoadSchedule = [] + + for YCube in outputCubes: + assert len(YCube.offset) >= 2 or len( + YCube.offset) <= 3, f"Unsupported YCube dimensionality: {len(YCube.offset)}" + + MOffset, OOffset = YCube.offset[-2:] + MSize, OSize = YCube.dims[-2:] + + replacements["M"].append(MSize) + replacements["O"].append(OSize) + + if len(YCube.offset) == 3: + BatchOffset = YCube.offset[0] + BatchSize = YCube.dims[0] + else: + BatchOffset = 0 + BatchSize = 1 + + replacements["batch"].append(BatchSize) + + if operatorRepresentation['transA'] == 0: + ACube = HyperRectangle((BatchOffset, MOffset, NOffset), (BatchSize, MSize, NSize)) + else: + ACube = HyperRectangle((BatchOffset, NOffset, MOffset), (BatchSize, NSize, MSize)) + + if operatorRepresentation['transB'] == 0: + BCube = HyperRectangle((BatchOffset, NOffset, OOffset), (BatchSize, NSize, OSize)) + else: + BCube = HyperRectangle((BatchOffset, OOffset, NOffset), (BatchSize, OSize, NSize)) + + MulCube = HyperRectangle((MOffset,), (MSize,)) + AddCube = HyperRectangle((MOffset,), (MSize,)) + + inputLoadSchedule.append({"A": ACube, "B": BCube, "C": YCube, "mul": MulCube, "add": AddCube}) + outputLoadSchedule.append({"data_out": YCube}) + + schedule = TilingSchedule(inputBaseOffsets, outputBaseOffsets, inputLoadSchedule, outputLoadSchedule) + + return VariableReplacementScheme(replacements, replacementTypes), schedule diff --git a/Deeploy/Targets/Snitch/TileConstraints/__init__.py b/Deeploy/Targets/Snitch/TileConstraints/__init__.py new file mode 100644 index 0000000..93b3563 --- /dev/null +++ b/Deeploy/Targets/Snitch/TileConstraints/__init__.py @@ -0,0 +1,28 @@ +# ---------------------------------------------------------------------- +# +# File: __init__.py +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 . import * +from .iNoNormTileConstraint import * +from .iSoftmaxTileConstraint import * diff --git a/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py new file mode 100644 index 0000000..8819669 --- /dev/null +++ b/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py @@ -0,0 +1,112 @@ +# ---------------------------------------------------------------------- +# +# File: iNoNormTileConstraint.py +# +# Last edited: 06.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import Dict, List, Tuple + +import numpy as np + +from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.CommonExtensions.DataTypes import uint32_t +from Deeploy.DeeployTypes import NetworkContext, OperatorRepresentation +from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint +from Deeploy.TilingExtension.TileConstraint import TileConstraint +from Deeploy.TilingExtension.TilerModel import TilerModel +from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, TilingSchedule, VariableReplacementScheme + +class iNoNormTileConstraint(TileConstraint): + + @staticmethod + def addGeometricalConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + + inputBufferName = parseDict['data_in'] + weightsBufferName = parseDict['weights'] + biasBufferName = parseDict['bias'] + outputBufferName = parseDict['data_out'] + + # Add I/O dimensions to the model as variables + for bufferName in [inputBufferName, weightsBufferName, biasBufferName, outputBufferName]: + tilerModel.addTensorDimToModel(ctxt, bufferName) + + inputShape = ctxt.lookup(inputBufferName).shape + + weigthsBufferShapeLen = len(ctxt.lookup(weightsBufferName).shape) + biasBufferShapeLen = len(ctxt.lookup(biasBufferName).shape) + + weightsLastDimVar = tilerModel.getTensorDimVar(tensorName = weightsBufferName, + dimIdx = weigthsBufferShapeLen - 1) + biasLastDimVar = tilerModel.getTensorDimVar(tensorName = biasBufferName, dimIdx = biasBufferShapeLen - 1) + + tilerModel.addConstraint(biasLastDimVar == weightsLastDimVar) + + for dim in range(len(inputShape)): + inputDimVar = tilerModel.getTensorDimVar(tensorName = inputBufferName, dimIdx = dim) + weightDimVar = tilerModel.getTensorDimVar(tensorName = weightsBufferName, dimIdx = dim) + outputDimVar = tilerModel.getTensorDimVar(tensorName = outputBufferName, dimIdx = dim) + tilerModel.addConstraint(inputDimVar == outputDimVar) + tilerModel.addConstraint(weightDimVar == outputDimVar) + + return tilerModel + + @classmethod + def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, + absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + outputCubes = [cube.rectangle for cube in absoluteOutputCubes] + + addrNames = ['data_in', 'weights', 'bias', 'data_out'] + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + + replacements = {"size": []} + replacementTypes = {"size": PointerClass(uint32_t)} + + inputCubes = [] + weightCubes = [] + biasCubes = [] + + for outputCube in outputCubes: + + size = np.prod(outputCube.dims[1:]) + lastDimLength = outputCube.dims[-1] + + replacements['size'].append(size) + + inputCubes.append(outputCube) + weightCubes.append(outputCube) + biasCubes.append(outputCube) + + inputLoadSchedule = [] + outputLoadSchedule = [] + + for inp, w, b in zip(inputCubes, weightCubes, biasCubes): + inputLoadSchedule.append({"data_in": inp, "weights": w, "bias": b}) + + for out in outputCubes: + outputLoadSchedule.append({"data_out": out}) + + tilingSchedule = TilingSchedule(inputBaseOffsets, outputBaseOffsets, inputLoadSchedule, outputLoadSchedule) + variableReplacementSchedule = VariableReplacementScheme(replacements, replacementTypes) + + return variableReplacementSchedule, tilingSchedule diff --git a/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py new file mode 100644 index 0000000..e8cdc49 --- /dev/null +++ b/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py @@ -0,0 +1,118 @@ +# ---------------------------------------------------------------------- +# +# File: iSoftmaxTileConstraint.py +# +# Last edited: 13.11.2023 +# +# Copyright (C) 2023, ETH Zurich and University of Bologna. +# +# Author: Moritz Scherer, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# 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 typing import Dict, List, Tuple, Union + +import numpy as np +from ortools.constraint_solver.pywrapcp import IntVar + +from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.CommonExtensions.DataTypes import uint32_t +from Deeploy.DeeployTypes import NetworkContext, OperatorRepresentation +from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint +from Deeploy.TilingExtension.TileConstraint import TileConstraint +from Deeploy.TilingExtension.TilerModel import TilerModel +from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, TilingSchedule, VariableReplacementScheme + + +class iSoftmaxTileConstraint(TileConstraint): + + @staticmethod + def addGeometricalConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + inputBufferName = parseDict['data_in'] + outputBufferName = parseDict['data_out'] + + shapeLen = len(ctxt.lookup(inputBufferName).shape) + + # Add I/O dimensions to the model as variables + for bufferName in [inputBufferName, outputBufferName]: + tilerModel.addTensorDimToModel(ctxt, bufferName) + + for idx in range(shapeLen): + outputDim = tilerModel.getTensorDimVar(tensorName = outputBufferName, dimIdx = idx) + inputDim = tilerModel.getTensorDimVar(tensorName = inputBufferName, dimIdx = idx) + tilerModel.addConstraint(outputDim == inputDim) + + return tilerModel + + @staticmethod + def addPolicyConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkContext) -> TilerModel: + inputBufferName = parseDict['data_in'] + inputBuffer = ctxt.lookup(inputBufferName) + + lastDimLength = inputBuffer.shape[-1] + lastDimIdx = len(inputBuffer.shape) - 1 + lastDimVar = tilerModel.getTensorDimVar(tensorName = inputBufferName, dimIdx = lastDimIdx) + + tilerModel.addConstraint(lastDimVar == lastDimLength) + + return tilerModel + + @staticmethod + def constructSymbolicNodeRep(tilerModel: TilerModel, parseDict: Dict, + ctxt: NetworkContext) -> Dict[str, Union[int, IntVar]]: + + inputBufferName = parseDict['data_in'] + inputBuffer = ctxt.lookup(inputBufferName) + + lastDimIdx = len(inputBuffer.shape) - 1 + + symbolicParseDict = parseDict.copy() + symbolicParseDict['lastDimLength'] = tilerModel.getTensorDimVar(inputBuffer.name, lastDimIdx) + + return symbolicParseDict + + @classmethod + def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, + absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + outputCubes = [cube.rectangle for cube in absoluteOutputCubes] + + addrNames = ['data_in', 'data_out'] + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + + replacements = {"lastDimLength": [], "size": []} + + replacementTypes = {"lastDimLength": PointerClass(uint32_t), "size": PointerClass(uint32_t)} + + for cube in outputCubes: + lastDimLength = cube.dims[-1] + size = np.prod(cube.dims) + + replacements['lastDimLength'].append(lastDimLength) + replacements['size'].append(size) + + inputLoadSchedule = [] + outputLoadSchedule = [] + + for out in outputCubes: + inputLoadSchedule.append({"data_in": out}) + outputLoadSchedule.append({"data_out": out}) + + tilingSchedule = TilingSchedule(inputBaseOffsets, outputBaseOffsets, inputLoadSchedule, outputLoadSchedule) + variableReplacementSchedule = VariableReplacementScheme(replacements, replacementTypes) + + return variableReplacementSchedule, tilingSchedule \ No newline at end of file diff --git a/Deeploy/Targets/Snitch/Tiler.py b/Deeploy/Targets/Snitch/Tiler.py new file mode 100644 index 0000000..b64748d --- /dev/null +++ b/Deeploy/Targets/Snitch/Tiler.py @@ -0,0 +1,47 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchTiler.py +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 Deeploy.Targets.Snitch.Bindings import SnitchAddBindings, SnitchGemmBindings, SnitchiNoNormBindings, \ + SnitchiSoftmaxBindings, SnitchRQAddBindings, SnitchRqGemmBindings +from Deeploy.Targets.Generic.TileConstraints.AddTileConstraint import AddTileConstraint +from Deeploy.Targets.Snitch.TileConstraints import iNoNormTileConstraint, iSoftmaxTileConstraint +from Deeploy.Targets.Snitch.TileConstraints.GemmTileConstraint import GemmTileConstraint +from Deeploy.Targets.Snitch.TileConstraints.RqGemmTileConstraint import RqGemmTileConstraint + +from Deeploy.TilingExtension.TilerExtension import TilingReadyNodeBindings + +SnitchiSoftmaxTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchiSoftmaxBindings, + tileConstraint = iSoftmaxTileConstraint()) +SnitchiNoNormTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchiNoNormBindings, + tileConstraint = iNoNormTileConstraint()) +SnitchRQAddTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchRQAddBindings, + tileConstraint = AddTileConstraint()) +SnitchGemmTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchGemmBindings, + tileConstraint = GemmTileConstraint()) +SnitchRqGemmTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchRqGemmBindings, + tileConstraint = RqGemmTileConstraint()) + +SnitchAddTileReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchAddBindings, + tileConstraint = AddTileConstraint()) diff --git a/Deeploy/Targets/Snitch/TypeCheckers.py b/Deeploy/Targets/Snitch/TypeCheckers.py new file mode 100644 index 0000000..f978b78 --- /dev/null +++ b/Deeploy/Targets/Snitch/TypeCheckers.py @@ -0,0 +1,50 @@ +# ---------------------------------------------------------------------- +# +# File: SnitchCheckers.py +# +# Last edited: 07.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 typing import List, Sequence, Type + +from Deeploy.AbstractDataTypes import Pointer +from Deeploy.DeeployTypes import VariableBuffer, OperatorRepresentation +from Deeploy.CommonExtensions.TypeCheckers.SignPropTypeChecker import SignPropTypeChecker + +class SnitchRQAddChecker(SignPropTypeChecker): + + def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): + super().__init__(input_types, output_types) + + def _inferNumLevels(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[int]: + return [operatorRepresentation['rqsOut_n_levels']] + + def _inferSignedness(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[bool]: + return [bool(operatorRepresentation["rqsOut_signed"])] + + # Override this. This should compute the signednes of each output node of the Layer + def checkOutputType(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> bool: + outputTypeSigned = self.output_types[0].referencedType.typeMin < 0 + if operatorRepresentation['rqsOut_signed'] and outputTypeSigned: + return True + if (not operatorRepresentation['rqsOut_signed']) and (not outputTypeSigned): + return True + return False diff --git a/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py b/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py index e19d1ca..a99c530 100644 --- a/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py +++ b/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py @@ -24,15 +24,15 @@ # limitations under the License. from abc import abstractmethod -from typing import List, Tuple +from typing import Dict, List, Optional, Tuple, Type import Deeploy.CommonExtensions.DataTypes as BasicDataTypes -from Deeploy.AbstractDataTypes import PointerClass +from Deeploy.AbstractDataTypes import Immediate, PointerClass from Deeploy.CommonExtensions.CodeTransformationPasses.Closure import ClosureExecutionBlock from Deeploy.CommonExtensions.CodeTransformationPasses.IntrospectiveCodeTransformation import \ IntrospectiveCodeTransformationMixIn from Deeploy.CommonExtensions.CodeTransformationPasses.MemoryAllocation import ArgumentStructGeneration -from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ExecutionBlock, NetworkContext, \ +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ConstantBuffer, ExecutionBlock, NetworkContext, \ NodeTemplate, OperatorRepresentation, VariableBuffer, _NoVerbosity from Deeploy.TilingExtension.CodeTransformationPasses.TilingPrototypes import PrototypeTilingMixIn from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint @@ -121,6 +121,36 @@ def _hoistNumTiles(self, return newPtrName + def _hoistConstantAndReference(self, + ctxt: NetworkContext, + constBuf: ConstantBuffer, + operatorRepresentation: OperatorRepresentation, + nodeName: str, + operatorRepresentationName: str, + immediateType: Optional[Type[Immediate]] = None) -> Tuple[NetworkContext, Dict]: + + if immediateType is None: + _type = PointerClass(BasicDataTypes.int32_t) + else: + _type = PointerClass(immediateType) + + name = constBuf.name + + ctxt.add(constBuf, "global") + constBuf._type = _type + constBuf._instance = constBuf._type(name, ctxt) + constBuf._users = [nodeName] + constBuf._memoryLevel = self.targetMemLevel + + refName = name + "_ref" + reference = ctxt.hoistReference(name, refName) + ctxt.lookup(reference)._memoryLevel = self.targetMemLevel + + operatorRepresentation[operatorRepresentationName] = refName + + return ctxt, operatorRepresentation + + def apply(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, diff --git a/DeeployTest/Platforms/Snitch/main.c b/DeeployTest/Platforms/Snitch/main.c index 9ed91aa..232081a 100644 --- a/DeeployTest/Platforms/Snitch/main.c +++ b/DeeployTest/Platforms/Snitch/main.c @@ -31,6 +31,10 @@ #include "testinputs.h" #include "testoutputs.h" +// #define NOPRINT +// #define NOTEST +// #define CI + int main(void) { uint32_t core_id = snrt_global_core_idx(); @@ -50,8 +54,9 @@ int main(void) { snrt_cluster_num() * snrt_cluster_dm_core_num(), snrt_cluster_num()); #endif +#ifndef NOPRINT printf("Initializing...\r\n"); - +#endif InitNetwork(core_id, 1); #ifndef CI @@ -69,7 +74,9 @@ int main(void) { printf("Initialized\r\n"); #endif +#ifndef NOPRINT printf("Copy inputs...\r\n"); +#endif // WIESEP: Copy inputs to allocated memory for (uint32_t buf = 0; buf < DeeployNetwork_num_inputs; buf++) { @@ -83,18 +90,31 @@ int main(void) { #endif } +#ifndef NOPRINT + if (snrt_is_dm_core()) { + printf("Running network...\r\n"); + } +#endif + snrt_cluster_hw_barrier(); +#ifndef BANSHEE_SIMULATION if (snrt_is_dm_core()) { - printf("Running network...\r\n"); + ResetTimer(); + StartTimer(); } +#endif // BANSHEE_SIMULATION - // ResetTimer(); - // StartTimer(); - if (snrt_is_compute_core()) { - RunNetwork(compute_core_id, num_compute_cores); + RunNetwork(compute_core_id, num_compute_cores); + + uint32_t runtimeCycles = 0; +#ifndef BANSHEE_SIMULATION + if (snrt_is_dm_core()) { + runtimeCycles = getCycles(); + DUMP(runtimeCycles); + StopTimer(); } - // StopTimer(); +#endif // BANSHEE_SIMULATION snrt_cluster_hw_barrier(); @@ -108,6 +128,8 @@ int main(void) { #ifndef CI printf("Output:\r\n"); #endif + +#ifndef NOTEST int32_t tot_err = 0; uint32_t tot = 0; int32_t diff; @@ -130,8 +152,12 @@ int main(void) { } } } - printf("Runtime: %u cycles\r\n", getCycles()); printf("Errors: %u out of %u \r\n", tot_err, tot); +#endif + +#ifndef NOPRINT + printf("Runtime: %u cycles\r\n", runtimeCycles); +#endif } snrt_cluster_hw_barrier(); diff --git a/DeeployTest/Tests/TestAdderLarge/inputs.npz b/DeeployTest/Tests/TestAdderLarge/inputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..0b45680e475fbcfb8f33eae985f22161eb685e74 GIT binary patch literal 33274 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 z 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 literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestAdderLarge/network.onnx b/DeeployTest/Tests/TestAdderLarge/network.onnx new file mode 100644 index 0000000..513e19e --- /dev/null +++ b/DeeployTest/Tests/TestAdderLarge/network.onnx @@ -0,0 +1,22 @@ + onnx1.14.0:  +& +input_0 +input_1output_0Add"AddAddNetZ# +input_0 + + + +€ +€Z# +input_1 + + + +€ +€b$ +output_0 + + + +€ +€B \ No newline at end of file diff --git a/DeeployTest/Tests/TestAdderLarge/outputs.npz b/DeeployTest/Tests/TestAdderLarge/outputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..f578ee2d8ea87bcb980c6d80ac5baa06b91b735b GIT binary patch literal 65800 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 literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestRQAdd/activations.npz b/DeeployTest/Tests/TestRQAdd/activations.npz new file mode 100644 index 0000000000000000000000000000000000000000..3bab8fb291a24e7ae3c8069c0c0308a94bf6a087 GIT binary patch literal 394392 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 Mkvl{VCrp+809Wu}DF6Tf literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestRQAdd/outputs.npz b/DeeployTest/Tests/TestRQAdd/outputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..f5aa6cf99a0e7ed7081e36bb856a808513ac3257 GIT binary patch literal 65800 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 WvEg6-NmU*C+Xw#~|5yCM2mb@)Y*J+a literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiNoNorm/activations.npz b/DeeployTest/Tests/TestiNoNorm/activations.npz new file mode 100644 index 0000000000000000000000000000000000000000..15cb0ecb3e219d1701294bfdf0fe3f5cb5d208e7 GIT binary patch literal 22 NcmWIWW@Tf*000g10H*)| literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiNoNorm/inputs.npz b/DeeployTest/Tests/TestiNoNorm/inputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..31169689b970b1c686cfb0a6fa2bcda6f2ddd799 GIT binary patch literal 131340 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 literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiNoNorm/network.onnx b/DeeployTest/Tests/TestiNoNorm/network.onnx new file mode 100644 index 0000000000000000000000000000000000000000..58550bc7353dfdafb0178e272ae30e1855d52521 GIT binary patch literal 66721 zcmb5XPiSoIo#*$M)J%lY;?QWJ3|bt57Q%?;3 z{g3~ox%u8hJKAZ#!+(nJO#Xk&{BPQp4mRTuj#nWBXq!T~oP_WN`7_k#VHv^;xJ5n& zYxojh^?eHASqeJ+V+i}JoTta4 zI{V=y1ozsmaxHUxg5L!xBD&LhU{74?!;oD+ALXP@E zU(kELEw-3AE96u3Zzj}*_j$p-fj57N{sI0FS|fMxqK7x<4zYdAJ(~L&T>w*Hto_(; zg`a)tUt-&UK%CE8bSJ=fqWHZycV?bG<=))c0@*w957_LZhL<)_@07jkXH1BB*H@EV zvq0}$EjI0~Ja=O~&b5gc-uDW94LmPx6<8zB!K-{g&K`Y27vSCT9sIAM=1t^R=Up%+ zBKHv;a*N)59z*Z2?Ln>SS-T^t{!{q(lP!Ux9B&-8&kly)G?p;LVW<* z;+Q?H@*SAM>*v`N*zF^MHGIOi!!H19dMmK+h;a}DqxCVA)9nt`Wn zj5$K$?2vo>dT5Q^T%P9(=&{?s^#eW^*u(;z`);7u$ojcIdz*vlGS|FBS0Xx6r@FFR4Rzjt-souOq+b z=e{Dk7I;6*SKwza4ON{Yz;5~YVx~os{9kvenZWY2mf_w0GlKbzm+1~Ob%q|%+q3(f zqc^ALW*%!*z<6yZ#CVoR&<)fchdR*DlRU#3zxAsRyfeJR-(6V4d=Y(tu7i3Wb9CN; zd#L?c+n;gFV_YD=JDj0&o_i2t{tUfD?;Y_@t8Z3$t_x^|&NGR`*VrO4^)i2!-C*mH zwRPIj4`&bcZZx2PKU(D-e?bmliq5_|V$JPdgfk^{DP^A)UZ05X49tP^1$5qfW1pdY z%CjF&@(fo{^^f?T;gg`3=n8b^Hg?xq_S~Y2*gW@Npfe`GkKm==xkeq()`NH2 z(3@`;-u|q)4{xppz5DB+5$bOLag}RN;UCoF=S=6=KVfTQ-o*~GJ`tNfHMSYDun&9K zK^LI^0lG%NhVSromLTOF8(}NZ*C3)d-v-*n>{Co~%?e$QuElO_fcFl|*IS$OmEa>W zm#bXQ-92HqmmZtBy=Nu#)!*Li!CVLUY4aYm&>o$A7F&IzFT6H$)=NHH@O_4+m}lsq z0ecJejs<1xLq2$S=Q(zXT+oEh&kH|q++T}60q0A=Giv3r-NG9ei7Amgs5w0UIW}jq zzXiI0-<>kNxf=8l{|echJvwD=wL0h=YOmIE7uWdNgFg1PfUdDEfHS-2_xSuZID_|o z*{ivG{EyI2$K;IYfwLFL-n9$(T@cat`lP&vy;bg|S>{|F_6oZ>pP?P_F|YX}`eW!& z7u_N8XUGM5&%TfNA{%S3!@XYR&pP)O(NFR1z#86O68vlZV|=V5)whH9d%nh}OSzAQ zdSv4RwjF%t@GZ5yCnvz)?INo<2grBWe7yJe*(Ty5{bK8ooqGW` z@b-CuzJLaNryhGD;H_f~_u>5$_G-^Qo@GGi zxf&N^zEcsoLrx%|JH+3a+`$#Rv)W4o{P|!$-u8pL`i>au7@x2m;S-VdA96y?gLsQnkf{IhZXx#47%-5GxFCpO^UqHp2X zOJw+cVm9zSx)XeRbk+#yX4stHI*4Z z&KIXde4La|mbGe5Fx@r=J+?%ca8)WmkhiiPN_}xG|@)YP=Y-vmHrP$3Y zP5A6ga{Wu_5wM>QKX+lhh&)9ffV4$t&YoDGC+yxeW1YudeLRyGGqopY{DRK~P(FcP z0c&Veb_XpnJ@x>-LFc(!J3@!t@UMNw#Ej^7(HXNQR|x_-@7Ess;a=e9?2qU=_&a#1 zzS`}zkGcPIV%&L+euLfnmOzW$dI2Q(xA=Vo6}EnhV`S~WTIJlvmhcs*;S+K8;lh%SLXQ6Ks^&NI7X z=e~h1&>QExCzE`>KZCbMz(@TC`!l?KT%+$}?)e^l2O@HTUilXK2sN*ADL>BhS5Rg5 zGly^BEBw3F_ML>+$3Az_Mf~=lAAvKP=NR6;uA!c1Imxq~Bd?d_LU#xC=LlnrzW_7z z#y6k<=5UAeW$xEryzlPBv+?nKYJ4`zG5dOU3Ep!pR_p?scKH@M)Wj#?bA-;vn(kz# zJ>~cwopJ?rPUH2np0$PcgXhW+~zN^MbM(R_6EKCui>RH(7^}&9K3)JtK7Rg zHMe_i@wH}!>>2muGgpk;{lHt_vv`d*YnOn!~KvI^sW}Z=rX{_7#D8 z-{ThE8O^89LYc=L&*MC&y`8P{-Q0(FhS$#*&&k>C?E${T&l+nG(f5(`^3XXnp&Mf1 zFUe)^&JjQfzX7-8(7y){=&gB(Zh>tCt$=r{0}=b-ggo%(YT@nMo;tAJ@}9*Bu+{-o zUw1TzcZU^x0WMd0PoK~S>MyYAH}niYRR$4TiH~Oyv72`e?YB1f7tG;<&c9JF&^b>F zHNSf)knPiV>K@s8bFhcqS{;}!^M3E30blp5-agFzEwNuff_D$rHGZu<=GYRwc{bp6 zr*HAGhIt$8?oIg#*&dAz&=2@U^fP#8R@Z4q8Jy=p3Y!->-vJKBvJ4 zdrxeS-MBMyx<~smUjwRT?$>jhA-~yD{j42~Px-UaUd_=%tyf6Lc|SWVdWKK<@8VAI@H=TjqOUFLQVwd+d>|zl(o_uYkRNK;8!d)abOWLAlI* zDtDm5Ztq8+gja9>=kPVUec-+55^*ut3D5*)*mi)nIdRUe-hArp`^8vv+5`Fn5RoJH zc9mypwm#aeXS|Ph=xmjD7U8Gl@MowJs?DF*diW{ScSL)Dub?G%_v4-FBJV4*ds62P z3gjOB@hab|7ySB^b4;}(7f^FI@Cg`q2b|5FJfUlWvG2*3$XDY37JiQI9(ewq*TX*4+=q9UbM$lg1+>E!f$@!c;G@6& zivnFj4)6hf0qk3PH_m(0{~rE?yfw5#zL1~fGq88}YyKLz;}+`YY(ReoJ&_;d7xTOy zkX!fwI(5ck|Fp`pTYrIk2X!v8EcMfW%wZmd&Kt&>YVmhLp%I0&{^NJ+(YM1tsNr3?wM#W&>81U z_GuntOZaE?z&^i5Uqfqb3upHe)KAd07kfpOKR0 zN6?Ga_6(4FY-?cc9c<<~hrTGocleBu?N2@d^L8npz2`B-*M6tq9$(`levil%I7D_1 z&%?)AjCY>3KIo6(J=fm=-@{=}{M^?YbbZV_4cNX#j!1Pq)0io6~*X!dqLN{YkAEW4?C@8YX!MHGb#t+CM{2fam7?*0g`mx+CVr zoX{Q`p)G!wlYD;uo%JWHthdiL<=@?`y*3Wnyy`uJIXZXJ#=Jl0c|f*?@pIq|+B;(Q z$zxoL&hywsZlD4CGrkcY^E;RC+!Vj}@aC{q!mkBA=&+kdy*q1=KjG`HYG_2)>6h}~ zoC5qBT@SU-1AK1587R=px7ZS}{|cWTJ^=T(0`@oLhK`Wki@mvT&rzLy=(7Xw<4!}$ zcgy?iJS}nso#JD>^_|Ck)KGW0hBnXy+?Dt99(oA%o<3pkL5Nwmhs~akp+45%fwwR7 zh~J)F%(j5d+1}yjcY!erbQ>V=nYbr+7~!AC^oL&GnLgOo=vsJZETP6ffC4C=;&VpK zV!XXC=)40R)Lw6~-GDiKk55Z)iCnwrDrEDhb7p%gfpWwrK%J?aE{fKH?7S%V+qWC3b_a^C-W?)~8&{Id{?j5}kT$2IO1N z5aTS?X(J`Et%37Qp+oG}KE&Z;KQ(xVeguu^BEAW#To4<<`6}KgtdA|2-iLS>GNyMQ0Cb znRgjt_O%ChaEQNW9MQY01pkPy^}m1(e2LE9&1Efj=pM}9qW9h00k6Gy9{&8O&U|8u z{4MeV{(+cV_@M{%HT-)Jumx--khhOMQUjj~`5M}${MoO~J!v<$Gq{fpI70S3dSgZn61&ecx{Fo*nb29y;nU;y06&LL!682C ze9T#ck{l_bz@~i{JnI9x#C+tle{0@C?bY6`X`M*y4*CtcdX;k!J$!a~4pXoL0)EQZ zz`7Ctp-=P$atG`+OmYo*&tJU0M<>MEkLTA&r05o_ z{I1(WAjZCpO+c!zGn(67?61(L922mAK=%>Yi|Ekly8%9pHhA;*Dc64<=UoJ-^H$JZ z=nU+x@@#F)ch&jshz-Erjny}xU&9ypyQfdk0-qLYPJLdm8}|rpzyUU?k8`@WHf6sd z7rIk?+?91BIn4X0AA4|*XUGY^8U$qTL@N&>Fuo^t=o#LfS*OJIdN(U{`(TF7o}a-K z{yqLvczf12BA4)ApzhQdW$XPEopBYg)&a7y8_>tR?%|XpdET^wAAW}>#-;J2t5BD+JOQ;&pyH1zczVmwovbj@c}gGZdThE_yjua-s1xO z;Uvd*@F(z*cz5&&TKI^*#ou1E&CqvHcks$D;LV#-7vHGu#XG=CY3t>c4@R z!hZ|(O!VpKOS9?o%IY+DnVAf_Gluy;>bKtny4(_&bj?*rR=& zU<=Te*g_uKrR;Cc0I%%s?9CaC>tgP?l*c|rzdy-yMRe{lfepAoZ=D`~7ivy>dyVsV zsbM~LRp4X1XHr6^AbS!ee?kI;XH_fFKvJutVkcuvh$j?^rX??Hub4%Xmn z{H&v&GnmKz+Hs!eh}_=A7vy))Db(E7@f^+dcgPn&z3;C41AcqzCis%?8ooo<#hlCi zSjXoPz4^`6##|#n&6Cj2(I@14s55o&8>r_R;hp;qxk0vHduY+$L5&%FuutL7@L5Ae zo3hVB9$XWvPfx6J0WQ!*WIvxOlGA-lBl;b5&iM|0hQ0>7 z@YZqe5_t-*+!ABW9^D35N1JmE&noYrg4(nBcF5sbhnQn(^v}rNxevh4o(A~N_Rgcc6nFPI3n;2j;JyNW8>n^dwMO6JXD)kqMsJOc zHuxI4fR4~bczxc${~XxQTd1?l;higxzl68H1yrAIoaZo?`-`$_mcqPv3z=moYl)aUag?_m#}HLdLo_SS%7_zJynz8d)w-x_`& ze!ir?n0qj2o(5xw|{$H@QFC@@UUOv>}><=qXl!~l2SKS8bI8MV}~-=Qyb#LUBU2)41QDzk?nlub~w-?~eI8{5`(_4d{)LTKhA+y_z?Gh<+cP=RfQvW!c)! z;$H6YkyhyJO<(8xLi_=-HL|(ws|TGi&>d{%x6k+R?kp1HZr$BGsJ=0V;cOCouMF(z z1Qht$tNsmn^qs@+VzZZk&N^>_{aV+4opv590ol3r_57@FOo=?g=QXav))VJk9X5OO z4q9u3&GQRmfKLac&V2=%34LR$@wLYVx;=1*&hxfc>qczSH|X3&3Ee>L(;hwR1S;gg zKFN9Q#aw6j2Vn07G||JyN#5lZR9y)T=%?@rS`b%(5&CE1-k~>7Ou1Hr?gaT9`#(VM zx3WFDYk$7($g_*=EYUEx8ASXRQ2Q;wKE4ZJo;mU!yyt3fFVGg$KwA&B zFMVe4=JUS!96}p>+L(WbQ)kZ|vb{*>_-{Z++3yhhdTVocVL}b$FX(G{`>;-jzx&x$ z7jvwAS<^dHz`tK*-#c{n=Q-QIyV5rh*MqASJhC}o_1>?cNA&LM0<_?4l5;~)??a--r2li^7i0?CWijOtlLXWn1Sdt6)Ib)t46!3*~ zobO4D%tnrY?7P+@+lS|n;Hy>M?aJ!W zVP8Y1=wJ6%!T0F3o396jJUVkYyZNj!hqehD)SNfi&E?*v$m)iEp`%sK%;%Xm=+qygpCP{o z{tmDMb=U4IVej-C=g+iAK4+=%F;9oAPCCQy_1v`&u`$nefUJ+G@jD{-ns{Ta+u)os~kJbiTr>)qO&*cyU>X44C4Rn8%<;{|s@WxzYlQzJ9XUg>9tja?#@Ybn_afSqH zWbO8(9Eq7hKN5d|UxB?x_YAGjyO$1X&pj~q6TZfl!2GQ?VwCmUBaib~OPD93b0_+= zz`oTd;tFUWE)rvY^V#1Wuy1po5jz*ld`BM9H^BXyOZfnMgoa4`B;R{KW1YW)2K=mV zjsxTu=rOi6G!Rp%M|R(yi9J|b+Ya&q`vTo(>=9k7jNW;iSGj_gP~)cPr|=0pflF-m zSp)CvTjYRWkF8$iv+c2+pf_iM-oN7|cypUq*vAGO<7Y1ScaC3(d8TW4bvsb)uQB%E zc{;-ecvl+e=_L1IF9k98=qn(92ue^R>+kIKB=fkFmr2*h*}Pygj-p z@O}6QBD}g5n?5bR_H0h;v>+jy{{`CP_lVBC9kP2&$ex!u)X&iMG0zjB!?~mLe@mc# z*cUP8Qf|Q>wgBvA8^ zz5AI$&0~!M*_k7LCAu2<9=XEb_dmuw*JkVQ*(7|Qp)J&!Ej}H*Ip@Iq&Sj75Wj>D< zzJXWgzY$u%FW{Z8gZAVwwj|EH=F)e*%y!SicgZ=dIa=kr)Drg=ei!{WP|w78bJ?S3 zeU3h$3&;=XJE-^HeeQr6I;pUy9{VYLi1`j$&$-pREAPh)ID`FdwlGfx>?INV2yfj8 zFSKvAI9uiKW#wv_?QhW4$iA1>49IJ27tnKbcj#L3cs4%H62TKb&(P24Yxt{G?)4t| z1$hc$%(*;sdzAL*N_hJ!@VT7iJNORz8N2s)=m%PWfWJE}fqh4G_HN&g6YM@Q&mXW` zXN2Aw?rhjEvN-}Uhq*k5lU3FoLm$!G^9yv)r@Z4sZ05Yfwgx`#(fY@eyz>I;*?C?K zzHi_|%=xUDQvTV_dEKv{_dRy!xI{L02|byh0}Z^sXYd`dUqg+#Lhs{D&f!@+;@5#G zynVd|`Z`CO@>xXe?ldC1qZG5eh7aWGvHym+8*qU9gx-Hc?+y!Kose>^a+&{aLjz6d z?cIJlc=PNLw}IN1we%lE8CIzuNCA>YQL{G#t@Gsa};9Z&{7eKqUK7uu}KbJYf6nYA*orw9>mY>4cz`YtDBL9K< zzWi*G`&wYP&z~Z9*!AB--UTgw#?H~%&ptl(`3&Df1Jv`^U%wAha-45<6+TnYCO~iQ z#QEmvtYf@B+5$H7`hGRQI)e`CZ1>CjZ$L`?A~0tSbw+o81aE&gV6@8pcleyc+ndiB zHtSjc0=;v!phLEH4ZT6XfJX2DCp53ka`fCPy+GfhZ&Ut! zdyDNoI?v_+-X0U^@rn2!ps&&U_rz8`I%gR4$Q`n??qZMRY>_MUQlAQ6<+D}33lY9S zcZBQ{CwZPO<-OXkI~ndI{2E{PW!xpc&RhcN8hjwn9C|;_cj5oU$9;I#_S41u4pit0 zd;%y}c^~G~&sg~ac>WF8fZ2HK2lcM#dqSQPERgr19lAUC2L5oB=X^mn_7vG(hrJ+o z^kN<5DbyY7`wUuS^Yr)xU@tv>_ThO|(1bi9XN9~*et_1{D|8L~1_b1f$Wy3y%Dm1T z(aAfjXZHrAn0xYGb>uFvNvF^TU+vcOtjuLU#@!;D|7>gfg3i5_z*;^v{LWT?ji2$4 z`k*sk)DP-A(18Vbi!DO0p);sHCD=vQ=Lx(gCPctv{sNtShm`#u;oF#ZNzolbBXWyv7v8fo#|gZ3 zj-Yeo^9g>D98f=FT5RXo%%$xZxC3J%ehsoY-#};R7QlI0Y&R?NAipf3=sp1ZGj@2F zp{Mw`gAzDH2i4|W8|WGK3O=Imk=6OwR}J>DkFYz_YfrOfevhnO!|!9aXZO-0_c7Cm zU0p)9hZkgPmtg1_o3YQ>T6~<(vvqe<^et4s&*5kps$zt-7u!Z-<`DhIa>p-ev01=#4XG@)W&9QbBy2~yGAad%Fbb~ z1=PI9+K`RgK`x-yyhZP04|Cu?F0fC5_89Y>wD*WDAP*Gw03YMhIG=&>oiaY^0=)Y_ zBGx_J!9Rc#9+g`oBroe3D)x|H9C zDg4*u*5`Sd^Qh~9{%dp<@Z16zt(Xnjd^K<%0oh&Hm-&sk2NB&{{QdpUxq57E%=e@~ z{%iE^#@G^hhW!A!M^>l30_y!NoZ=I~Fv~Vi#NMG>;Ijws(kfMJ!`r*F zRqz2^;cvh0BS5V)$KTH?&)PlK_}rm?gUPJ#a$i!Nrne=CjHTl^xrN9=>`_wpC`*t>hM z$6NeApo{ni@D|>k_3$P9Jvn|2)lXYT%rW#4y1@1U{Q^GX{}cEfFhw3spuqFJfOgn_ zBPI82m3M2M0(&>kpREzzIa}m_Zx1!D1#4_yqceWA$~%ncKcjoX&sg{2t~>*M?A1NH zM|-ZJ&&U<9=Mp+cZ#`{8Z}3;d?xWYt1ewvpRc1w}2Y=>s9WZ3d8$c zBKO1@y9VZT)&^d^^9ADVrG=`GAOLymTcd_o-{Vt36S_y>9A_(f0{VNmAE4eBa~9AS z=nFJJI}m~I&ON%Gd{gKlvOBhpb0qXT$kR!VUlWs3_G_^@*9@CEtYPi~oS`$=^`N?5)Sg``@BJg|4yP0eutDUV`%#vB>6q3qOT+z94*C6syeE0Ens2aPe;_pBqjDOCF%{&T3ay`Fi6>>Lfc5_I^i z(K)v<4fGDW0LrgtH_Xl1)XSW!r{=*V->DHX_xOgyxuSc-r-OI@9exX_bLwNBT~NcD z`wIRHbnrbf-oe=_{~l-D9N13<4ItwCj9kE1@Hb!tPC%rV{mrnSB5$DX`O_*>&qlrU z9C$8IpaBodyc7FsK>}fwd$aBpd?bGltZUvIbWPpKIi_wVzK2!si8@y{IqX?bA8!tAe+$1a)rZ9(`OP z!>f0u4(iXF8+iT9Iq=ML2d|&B25-zwEuXt_uDd{QFGtv% z5(Hqa7IFt^W5$K13rICoIa7h z?8&@7BlHO$doO@BjEnljoXcK!(W~=J&20~Rpe6QE9WkD}kLP!R?Ld2?9zOQe!$0E_ zp>wEbd5w=f+Rv~zWNSFb2HHpZgLa@sr%wq#(+~a}>K_Z=YhrZzL-E&etgtyNM z{smaqowGg zhuwM|sIlo2v8mH;Kch+RHR^-SUhS_ZuQ4succTUs@)LIX2I^f3*zLuf37nuar?m_8 zhw#4R?mu8R-r3y!40;cBcI67}uJXUT7-}K6uSduMn=vzBAMIBDjO_y1b9A;yoEY*!J^u|c_BQMezjOTDUk%^mQ?BxP zSi`*bvxm-mVT|^}3H>17A#b1ww8YMl*C2pK9n{)>UU_E&)H*Y8LymLgTi`wtDDi)f zY@Y$Uh{)#f>0>_6XZ)<)Ave%>`a##@TtA`rzS>8GKf>-j=4p`+u}zW9b&DDWx_&t#DF{d-Q zAL~~7K>cjGLMN;>+Vbx55j5C0`1(8}+kZ;<#%x<)i}L8bM>VuXcS~GAFV5h-x1P~2 z;Y;A0b1;H8hxmluo`#v>6R`#QJ$Uti9-0qt|Bxp3q%lcP{t51I$qZX-N5g zyThkt?$C@Gn2MM zZ01kceC*>6yE)Z4+Yxe?^7)#(!6vo7v)NC9TtWLt4fqH2z8~g4o8hS5IcgXf^oum$Ury5X#kHCJuftvSmnf+!^_2%#n%#qFU zQ+VY*=AVVl*P-u_hbQIp`xc)T-nnb|4b&a)Oy~_c#=IN%f^lL>lzrJv=i?={yS_D{s3Db<^{V?4V_`%0Po5fc4K<<+DdfVBYYoo zt#f@iqcuA3r#|-j04>qIU!jMdLG8u9Ju~-ckH+7@H(T7IOYrB=Q*4JIjB_u}>EGIX zhFzfVQ$7QA=CFqOT4;}5oxXpI{EYlrJ+k%ndjQ=k_qvde`Mv3(#W?Go-Fw+0yPFO2 z6l#y|)5lq~>7%WU`QBYl^3QhFGWX!_eENt_!WX+S?##Z9k=>p5_z7S8x1RCGs~p$E zclg@>e4O)?&}(AMugzH(@cNYKjai_31m=|QKpS&>fKUrUZ5+X*0T4A z9JC`}!gpJnwbq~`&VBjc&A8*!$@XU}Y+XoLk!!KDWp-Fhw`~ch2biec&yK=0Wdw;4?H( zt6Alo_F@mlcj&F-yv~`R`byVefzCa&Dd+S&7wGNTKJ6)CbDz#!<1ajeIn+J5pLf8= zooO?_xdQSO`iShEFi(x#pl{*N(dqLEU5a_X9kRRfF~<{rKK8Xc$u(Z!m5-1^%>DL> z9OLX$Lt9{-g#9yq%H1;W;e3_(4%=Eiu^;iV&jv^<^!6()@i%{kPl0Uj&gi^*=L7pNzr{`PZBO*B$=7fr`&_k_# zH_pB-^K|ePc!R#cuHWe@|IFnK4SL@j`;VYNwpX7WcxxLIQvNKx$LA4w2DP?*MbO2Z z-|r@S3&_4h-pLOB2)9Cwy}3*GCm+x`+ZuTvpI?DXaD^^m z>(IN)|A@}~5g2~}t+CDVaejSH&>7>bPtYC=lsew^7Q21A#}=yY7~cRcu-U^q=rAX` zJ$UcP0ld32_Y~foC6MmoQ*ZssW&TX9(5Fa#>K0oc`@WszUL!j5MdUlENXuNegO1Qw z&_s?0Y|f!Q84o|$kmsO59?lB>XV}+JbA%Q7vAHwPs)o0p5^7Fkrpx@^4mGIt0lz8w z6R79oxeVuo&06l*-Rkp%ybr&Cui@Xpe*^sYwSJDBtmuK~0^T_HbOgV^SJ=y(SnZ|2 z?s-_J0`BA*U4W(th1cJ@J+y$1&`qI{TGXmrEKgT|r z`-(R=<4{0v$kJF|IwkKLpDTfkRP@8)ps`1wvnY8<2U>?>sN_!K{T_UYko zu=O$Ln&b0+>uaCpeg+Bq8e4@;AL|5kFYqDdTBqpFf#+tA*HG)&hj|*Px$0HsBlZA4 zL$?dEk09p9lW*I;8)lqbOZG9v)A2^Cb{Mj)E;|mJMcX= zQE})Zlh8Xi_yt&@4@{SXFdkW~>?MHkMfch4HXDG3&TcGpKm^a~XOpC3B z-a{kQy?A$=*_aaa==_}j-{pTGkz(8T$#}Jot4^PO>XFh8c@TaT1uL9j1 zxmE{%fqp$E33xs8XJu^X*z7Nn;~d%ZyM#J-8*}~l_;knxFqd;gc;n@(W&Tb%hCe}P zZ%^8h&E?sY@b0N4r}G)_p49uUHOTg{M(4e@W(3}kgie3++W#&31#&VMyt~uZfx`(o zv3cL7$O~Yd9$)pJ@ee6~j|OzcIr}wn)}G?C2Kp!9p48nfbN)Fr;_p+SFOlD@@}3g1 zzAe}%pSy66VVTduo&xY3+mw53uw5YQ?<@;_;QN^C>2KUVwkfdw*U%Se2@Sw}B|Z`K z=sgqr@E+{JyC2U%y*`25=6Q?m6y59ld9fkU+*LCs&sN_;NBGX5J#oLlFJSliVhs9__>VfMv-jxSO$lF3vcAW*LH7Gl z`&;Z+@Xn}@w19Ro*RSzao`OgCLukZy3_V)qTmjks%xi7^*YFYEdOr3&MdzGt%D+E) zNB8kPCXcqCLw^ctV7>$ju)tOUchN%UTiLvED+g@Wa7T@Hcx#owX9vB#z2;q@H=p$R}G~pG4dpx&qi& zi>?5kL62=8Uuzg|Ujb~u3BD1%HGT%Jum}A0JwpzO9YYhobKu!W?AEjgVV|Cj`R3T= z-DiuRd+4DHWc?~&?YFCZZ)eDU-nYOzr`>)okgeU|TL5(dyLq2Mh*_V|-{AW-HqWXh zric2opfg0KuU|ku z1uyW<>=Pq4dFUflfBQ_pUZ0SyKL?+Qoq`cCmv=7WnsiB4KWQ_NKRU+RbKIbO@7>z8>x^L{|L zA!Z66;HBOT`}AC_-<#xjw1xN4=g}B)6wnUp&u`{l1NYZ}1+ahTO)=LOI{Vgth=-q|Ysu{{JwJ0;{~2837oghQ zVTAS|!Y3)V9{4!tIeKeWV<>zNwWfdX3Mt1(AJHA6y8w+oG0)f_9{}U5UqYR04YhAk zfXh|>&UBv(Z1!uP&ecKPy)g@T`>o-Hy%`_j3w+-Y?_;muBG0h(F~{wp6V|^28)7Oj z0}=Tg|Ck5{bND;-_rN}`kgXeHe*dNS=sjm=us?I|BU|%gmFM=H?9W*B!X1?8Z-M>3 zp3fZpPmq&7z`Th#^R}SEf5Q7aFz4we>lMgcHnP8L#*dy4tw(NOEt2ym(YhXdC{E!XR<~B?n3{b{O%*ddv4Nz zZHmwT1&!G4e>hikFYrCE-V^c>vGO(0w?}sme?_c&d4hND4jPfC_>aID*r3zjp6o$g zAV-UzXMBU~oSsjIzK33-GiOBa9xC|tIDZytZ_qol{YK;feFW~pGw8_U4E8AR{_o(O zvEAxD!&CgV*~{L9_;IH8+{RoJkKt?<=t_Ldc}d(`(80Syb3|x^9OL#3usiD;@Qf|u zW1sH7#;?QAd@Xd?Bh+~dV6V<)z6!j@?>Fe(*E4bp-GD>%1@au&gK{L#3BLB@jMhll zX7DZ4J7ew&yFTtwy?zZab^!bye2sk#c91<|d+I@ge*)&PzB8TU?>=Ur1D@*w>iyAX zf5wNHKd%#fq_#VCFFqyw6j(#w6t^?e_ebLO>r(z+4%qG89PX@L5kJneJt2F(9r1_I z9_l&tz&LxEp|9cXzXZ->{IFwk_P`!?@jC|YP~A2B2EREdfVET1f7k0K`ES}AbOBul z+_5#yy^Fnv_inh81$;;#=CkySz5h4Bm=@awe1b+~d$ETu=I?rYx0gq7fL{YW!l#0d z_^pwR>+w(U*6!fXvH!y=e-64Q&tM9y?~d+31v+5d6+ZeE@aAy8XUN9*KJHBNzem+p zs~hL<{}1RRvO7~BsBJFyAMr8Xy=t@e1Jt_W5#BuF1pgLlKOH%`gwHbn{O`QZAHWpY zn|nGU<}Emce+PX=KZSR;1oq)i(eDA@3u%GR3|j{^{>>`in?Z)RMn`?`;R2s$WP5hk zQQMZ^z}rh8#}I=)fH^wfE9*Ez3j%VA#N*pT-IaOVi+RoEZsy49t!LaX@mrvGo?;tw zJi&(?_T`?XJ@VZ$&p5>%kd52H?rzrD5-72~Lq3B#%LbjhweJF*cI(>LKD@sJd5$%* z`!2`W*-Hy8(RqFeeh-^5EpIv^L2w(&fF%cwu)HUY9cMf%L_8!q!=tisjUb^Rqz9+6l zudi{=U14j`sXw3KL*6O2q>QbHKY?1SN9UPXqXPHv-p4C!!&#XR-4I6}&*Tx=IBWFO zUqD~R)WaU&Kj71%e?aFQ2`hF1AJIKYfxB}*&hPHjwb+Ng2yyNJ_ORE>eEwbHT+r$3 zT+V7A&ie$l{xN)l*H_-Y4)7HP`uWyR`iMTDzM2DHffS@QzBIVfYs5ji%F+HO9 z9Ifw8+(m>&^yb?o<`6hzflrW+`Td@4V=DZu5v2IZ-(fS4GuW$r>*wq}bP7%Qx|4h4 zF6J2DPw$I)B3Q#eW0%^yIo-z&e){NZt`NZl9lUksK-oU*%Neb$-xS&SF6HkR@0>l% zv0uSkYiB~Pf0^?Y$o6%D&U*UXL*4mXU_ViiuY#s!p2zzh&^7oNdj@T`el@fM*7&F& z&jGXm_N<@zox}O9=X-I5y@6h0+lB8_p0|zpE>-Az_z~23YkX%DlH<35H{WoFq3X=t zkGFZCkI4450TKHgs(%Sh$P0LDtf9``L2L5cbBtlwxx2M?hkpUpCST!aj$)a=_j_o- zc8tGuJOl5}o2~o>{|3GHKE&M9F*8Y_H0k)v*%~xU%(oeqbIjJ zGN)%NRc9@GSpZ{i(05zDU*`QcQ19a@D6wAy=iEV#&_Evh^L%`2XotK8+Uzxe68&dl zoWVTy*<#n`ncJ7OwLd|<(S7}LLLBlrbPC+5XQFRMZ_a6qd%Oca&g)&UZlzyDhELG1@!6m|!N*(y-81qW zxX&KB1NJ!djE{bvX9IP|5xe^iiT|MU?m4&pbnyO7TiqqJ#&3v-#NVXN^H&@;14GG{d&Q6j?VLO7gz8%pulHm%iCWGya(p|8of2`$6Y7Tk8>X9 zD1kdL_k5Y(-H&A2T(;a+AjvhR!=lL7- z5%>;Fm&^&If3H4m_p#*zvU9Y^!kv78n#|bObaa>nr0}HCSKwo zMY2f_!(=JeMJ~&rg@2g$-s#@d9}%yDQYgql{G;%%t0EK#i=WSZZ@K0V@sjU8&+~nL zywCHz@2%?Y-K=33$P#I<8|c<8u-#W6wvYLa{8_L#C*_}I9dbmhdFFS-j=)3o#m^0R z@3RK>-%{rW{}_FDkYfWMXYgZ>C3*qu+g(`q1@JrK9exYB0<7r{*U`I}bM#+>N*{8F zY?1o!=65!G8^gB=9|-t&h*g#YzJtWL6Td^o%XuFOxHoOjuue~%RqRA=haT|o_R$4Z-AJnhrKe*^-J^y)CYNQp5-~`aL%FMzCI?`y+rD*l577L`r+&`!nt?v z(7eSCuqzPZcJaHvh_8XWM%uSEZ_qmssBMh(uJBne$s>L5=K9W$!8UQ8rCz{4MtTQx zboVmEUd5l#6LnkgZR~Do8|NKa#~L&EfPM;Ec;AY4AkKSyGF{#$z7W|d+!lWKu#Qh| zNNhyfvvn?r(f0_H==RkB--9!6z^?&+zZkbN$g|s*^_|Na&UJ{-ecgibB%c+~8~Egz z=%Pb+ZyWHDI$>!aVcTmv$=`wVDL&#VY~!u(xh3{0`W84OW*^w6+zPV8-vD#%$MbzN z_BF*{f(!ir6uEdmNbl2~8dqbVWA~sw$aU`ec23V3B3t|;q-XB}V-|Zruh9FLIcKt- zbBuuf-5lgT5;3>f?!uTByTRw~m7CLC=NW!;wy=${j-LYF9usyAwz1ulb3~;3jm8pV z-^vbLfgYc^C5YhJbopnFm~%S(1)Q}!uO{D|0^J$(S*s_$LCULF$RW0S+#BSagx$g$ z5ovPM>*CqKk||MnBnSEhO2BW!nGVF&nblK0af$G|;3#NVcT?*Yy?VqQ=DC3Zx5 z))BF{*uE>}Dt3p@o~&nHzUES@_oH@9JKS#FY zm*~knbZ0c@1-bTNtaUxZ`xydvSYsO($ScrW^pk0hEwK-YY4N$60_@_m598d!1u=mb zbJwt=HlJ0vGh%MFK?B@(hn!DxJ^NfE?h0EuLoRAl{~F!iDzKOb&U2hQ#XOTebkqDh z;}~g=9}zPmrU&{fP=hDO`sd+}#{Y-3l6sfbzJ0lK}K?;a~+0y)NP z>mz;ycZ6=w*TjUuGR{~(iFW~KTmWP6oOpM-hqSJJJ|o6AU!xC+G2WT&*H~k2zzCEe zM(Xj|p@zPvE~`|{+Y|Y_?G=n4-e=cd9Gx}6!Y)>7Jr+vPk#lxhy7{h zJL3El*mw7^iT=;DiFN*4>*oWv_+QezwYoGGZ zc5C&-S);;NPIBK3a)j+UA5ZhoH}}*L`+a!(s_`Z40D5$H?pgNj9*0Qb{NANCTY>J5 zX2?C_&bO-d!OKb2KaK4)oIR``Z zMJ>+ju)PQWw?`|Wg}anZ*u-9lcW2)37`_4S#`BE7>&2W6IQ!Dq zBTMic-9A0jJ%wexG5QT$g{+ZR$PQ`!O>#QUdw?8byKCQoa|ZYdE&yW+cw-6>5Au6> z3^&I1+#2ckbu3)0(Lom~(xv!FX(Iwn%q!NZj@y_uIpj=vT`)|9xjn zjgO}JyRwfo-dP%Ayi2(m@)lU%9<19^%e_>@9s*}*m(sqS=UY>HMEbU^>3eL+TSNDq zZ7T8qCUS&an`T?z3_B6q;B$}m+)Q$x3EMLo;2o_HbB+v1_vcK$=MC(|?ye}_55}0J@f)j9AR&QMC>J4Mfc-bi#s9KzTDFg{}6wT?!C0|clYid zR>7|EG0(d@&2P)TPSEXDMEpJ2Lbu=bL4LO_x<32&eEWY+%nj0gydX~BDV(tt+z=S| z37k7x?4I6?ZQv92uHZZ5HZi`x3SWh^{}z6P-;XhY81KY4(h?`v>Ie2~UEhfxcjC?x zFlLO;IQL+k#g6nRY!D-ok2`3H>wq)tV;6(mo3TAP1-!ex%d@8$Xk*_0Ht`*{|EBJl z0sdBh%5}YC_xcGa@gE_#z&>!Fi<#jQcnmk3=DTvw?#DVUHQn7VwzD?Gte~#}XFdbg zatAN*y?}4gchFDKoi9-10`3I+34BN0b#y=GX&dJbE{%sPf&CxDS#t}F@!bGp58>U7 zJ$#6-MtYwy<~_RiiWp~ePWw#wdSrwB7~c@R8|3pjb#c~5@aOQq1D}X*&{xrY3%(ua zxB?Zn^K1aWQ#Wvd_!hYfw<4e5rhwROZ1?E>bi}z|d#T}e&}Yc!NI%6ie^>1D8G47b zj(vrc`#-`ifis85Szv9D&+mzG-jcjtAN`r99+CPUPNC3Ok&!t2*Y|ReYZTat_=Fte zzd*WY_t9dTGe_@<>ByVQO~c|G-?92R+`mVf>z+Kfgo=OY?-dzV@PxJlybFL+42tC0dQWZdSJ;m|B)Iiy&S}x@eN!7*BF9K$6Yz|d9Q!GNzSTkY z`;O&?q z6Lk5wL>u$(zJP6C^6sSrHAs{EJ7bZ@yIb*mq;YHL?&ty1St~gE(RYc|FXy?1(oZ5E zRQP9b?yie)AfWrM0=|gt`#FNM7e5i+oyqGzK_8-9>)YrxGQdAXUk85d#hUA22V7#i zr#=GW_t3510Q>ST0=m8ibYRgVK4V_uFOg^PJLm~s|8uZE$UpB#`0P>XZr0$9TR~b^ z-aZ1{1GtFoE-r}m{+!7^D&!FR8aP8jdhX3Z=CqC9CH5u$Bc!$cccw^;XIl3REOv(M zV*cNJdWXe?9v#A)HzEuC z=GxN=ITd=XFXq~dT=bS$d$*3~JR`+;i5Z;v9hid%+_Bu+{CsYXIKDEy8?^*l!voV`*%n7 zbPbN-oihRD$Ji}>=&vXFzO=o2@6BBAqXEkOX`ZP^cP{6&=8pV9z8_`8KSla1=mINX|9t}?#H|pP{5CgiNvc6n0HoU&%oV1=syQ5#Ds%A3HnLi z#{>L5HO}FTZIIP~81zU?Pagp!c%{8p=*~YzKSvI+H<0!g;9|_XyjBO^N5FSWT!rl& zJFhi7$2)X3dvkwtY((Hk?~(t&IP?x|p&!H9i#gu0G1lG1H$?Y5 zV|uuUaBa%ig6r_@z!m8BZQVBJ??^zm2W@jIVmoA^gmZr%4zjJ!eEmc8fM3}F?^$~l zX`OR8^@Q$TKAmRH{4u_78%MnN;!f;65*xrX^dq=!Axnee~!NzjxRY?|b#F+qdW0lf5Nk+|897&^HGg zoAUx$k-v`g9-OTL?nOJ2Dzq`I8b^;wW6MjF=c1(lMbZb=TC(~>XvBi;oIc4JiKnvf2 zRk+~<8~ElMPQN`h@V<*Vc7-p+ ze7_yOA)KFc_zL|gaUIfK*u%y&*WQA658J?gD`ZKG=erxv>z8`MR_|l3uxs!Xy(i9jkBO7}7|uWAI-9;$EOq1Dy*%iw<+h{o3l=?^WVZp zVE+|xuU*Xd<~+tW2l?+wJu+gqV4t{6Fk9LId40erV2(aSx`TokzXRqbczgC->)3;{ z7w9$UI5%RK@SbVD`;!a!+@p2v%YHj_Yg)s%wmwCNv-UQ+HQZH?-cGF;_Sy6cHq3XA$k(%4e;KE$O~W(R|m`m+?#!c zn0MwG6}Ec_=;m$^-{Nnv&*b@>z@1~iT*k^fYlJheLf^#aJz1*&@-gP$Ir7KE1f<__ zb8qk)XP?F%!B|yDF*TUOx5TOIuaFU2Y5zNL-pNd0d+){?vx?8T3*^fwb>QvG z^R_^Yxo2Y=e4e|g58uP-cZNjH5ZT~&@7CzB_m=UVQ(zxqKS%e^r)%i;mZ)QY`kmQ% zXXpi-HEU#rbeCtupMy=X2kdh+%{%el&A%ev^K0T0@ND;AqFc**wO$V*{Rhy23w&M5 zzmM%Z;4hIIgUq=%b9}c?;neLh5Yv%=i(LwA&rIm2$bj@sJ%(5JjfzLuExdR349wtW z*sJo?)BgZIpJ;%3)9g#1v-Lr4PypB2};{zhJ@*RES=0l$$}ZA} zwsF97?8knagCD+)jcM*D!tdfgLz;I9yqA~weOt!)wgPp|kov4`j(?BqzzOi2;WYm| zJ4W`@sSk*Uk1@~L!uj?(r2n0vXOBQ4#=H6s+*9N^((_v6CGjn=zVnWNJ^DFA|CE?H zb{li-HJopvndUwA=;qtI{gp`fTBFN(ewQ-8hn=WnJV|~ER;H>F+ z7wFb;Uhl--5_*B|{z^~*Yafx@lebTd{)mkD0=zYqkKxSmUhUU6a*faPuCT}G_T-#9 z_)27h9OLuN`8NH|d7gc@G56e(UlP~D*I<(v_iw#9u%`Xakc&5fzHK~t1$s{%dvM>* z89)JciE-YDF97#s?=?Q_%+M=ft~tB-J}raSoQQq}+~Gd)H%Q-!d#s819dHIt@HvycnByFq*fnthPJRft*eTN5?sJFSmY6Mb zk>qfZ^L%^qZ11B(zeIYE_TM6% zr$BGO7J2&RSI`@E>bv(RgM7{yuE(}dcV&z@6`Xgu3ST47fwpIUP z-x@P?dpO1C=MvtU?(Z6(`@3IFyX1LC&J`lyW8YoOcWdkjdp^m$XQq_*9S8Xx+iPde z67I4hu7-w?VB{rmWwGhuhwejXF^4E_+=;;Yb8%)Lcqf(t3< zyRQ!WlvsJ=BNzkkab*BU>`WiIi#ZQ+oO^W6RbqO45qyf?5x<4Bj|f^~ULfZa;=$eQ zHTo8Og})>B7~Q_c@H0?>9t3;=pS=#T{q9EMI^@eiey_f@fbAZ=r#X5@UV+{d>kb02 zruPxy?XxAu85+2R-cIv54Y3WnGxi`6>l;eM&9FOSJ)=gSA*~(39x>K)pOKg&ID2hD z2Ord9-oYk5dFx!`v+oXHK>8*-dc|t;eLBtaKZT#;Z;-yXYvB16@)llB%s>R5VN7$d>=oTJ zzDs_KT)`j6kDL>c1$;+L2^vAn3EUQOo@?EN?1?YojPtuGAK)(Vxx*;nTR3;oA??H6 zerK9vF0cdkIr0jb67lHP?9snZegXeI_ z+ynGEdIb`3Eu8bT=+z{1vng|0BgJ35cBuMofzl7 z3h>D^_Zr~s#hLw{9)lTr4R36X?w$73eui{+)|%rRftLDL#({K@|BP86?aLYLV*`X~ zwlCrA)&D-eLjQ2-_bhk(P#$=nt>AZSOT1@=nBU*%fEvVl#^Ro_-Lo;Cbtw<5xe4!? z5g2Q4BVx))o{1`pnQQo(*fV^4$W`FZg}w&N!N0(+kTc*;0%&7?yYmS)7~$){3b6s- zSRPxMfb%DKb$jg(a?UN%+0EHS+QXs-F>>qJ#gcnAfJ*si(GSpj{4bHte+0Zs`+I;M zshiNv*#YjfK_+tS$^9scW!w#Vi_e}L^b)wsNRDz1DQAp57wW(}o2$crYnuB#Kgf5H z;M$b+2F@LMp0(_Ei}(Qic{rTpw?4+7$m@`EAiqk?zOjS6M{8UIYg^a9vySmQpSyj8 zw6`8SgC8UPq=S4$K*pH=eDs*uRZs%=iCU}3TYM`>ztb1MUbhE%ra(+c zc^B#pzKH!@{MOZ{t=y$9awHk>)`+4I|&d#lim8-bYeedr6=&L2QI$UjRW(%Bv%A5XI{ z#4Md%uGi*cANH&N8a;yjLGIxO-I_gq-=cTw$6chDe|I~(sNs4z=YNbH(5*KI5iTWW z;ky;#?ajVNpa-7Wf-CTRn$NDNafog`eS6r>A3z79I{pS(9OOIqd$@QHaO(DagU_5b z^4pl}^vEIH3jTn;=p*Ix>~#k8+pl{yZ)bpx^p1=lFa6H$T|J%VzI+4j(>liA^{IXi zw?j49@KFvi}+-xC+nOK^+vj4Z=!tVKu&r0&@8JTRc_TQ_D?o#;&$~3|3Rz>fNNf9sd*s?6 z$4Bgss9~*3_!`WV@N)VufSkIYHs<^FZYy9<9X@;hLuwhDz)Q~Xt=NNo2J$?k19I-u zev}=)5xTqZ4O*)J;UNFaZ_wpR?75Pd#h;_FW6JxzKtF`{bBk`S_uj^gE-m+I&Lq$1 z3=JrhaEVxZG}qqUb%}HrPX-92{rd4Oj)+@uk=pn@!}nZ)9?s7e`ahoL9a-}ky5~EK z?@~pKQ;U6s?j5w)*7d$l;OxtO*3pM>#wrW!(UkL* z*xf-`{0tf@iCM*0z?^w-!uu-14fjk}%X`F(eOtnJ-5 zAfne(ICSsBe%)b@NN5NcXMZe*LV%ouSJGOE3W@mB_0qLHGN-zTY z>~k^O!SefqY>79=Ijuhev>Am`Kj!o>DzTbyf@;8&5(@gWvJ#*KV zHHOGG=3aZ^tP#Nw{StJ>@2u|2`6^;ou-m0=Z&&cY32z@;$R2wPtnbGf z=I`J$?*e#cm-0Q^XMt^>cRTj3XT%w=&%OCE&%VY;=kcA)C!7g9PhS5BJt7xch3J6xY;-Mv)CBRBASW=oEL4=AVX3SVVD*u)-T`$^a(G4^1- z{1|K#a{}+%aHct217_IHpJM)<9Fgu#oC9n4hV3U{`#swoP!qiY_Fz5FJRwHg`O1U* zuI=?PK6z)f*InQn9)eoH2RM6QA=a~3u{|$fJO3En9m@MQjWfr2+?n;;nD=pv&-*Fi z6S4pi-#UDcbS8VgMD8H{m@ij>0=+=CG0(d;B_G?G0d5;x>Ha=KcP>!_W80MPeDU`$ z^o32VbCqCqKs@=LFz?fBV27jfGye<3%obe@k;tW5Y)4<&#?fDuQ=T5D+N31;s^bvYOcm5~%8)SsH zPj`O_XASi(5{G>O?899T;r(t{*Lr?t6aE9FwEVYeq&+_x+F5 zJD($4YVM$WRtFwV!=kq*aEaJMY<(q&pqUT{8vJ*CYx_2Cki|j%ZYe$A_tJ5O=d{@0 zCU1yu20H4m7$4#>sl_q&BNzJ)WVg`Z(Z;j@OjIRit`#XP6n_B8)|@LcnhzE9%;sZ2KnzeGh$m} zR<-f_4qE)yh{z6KM@~ep;lH~VZD$TC#~kC^7~~n8I{|&xu}}BD1E+6>9V7E7i7Ami z{($sn;qf59bGZWByglGdM}wT>+|Fu_`DyYOzPGa4ey;f3VEe7Vb^O*>@7Ko%Z{9zA z>#e`}<&*n=O~?%Ut|xppPQ<}kEhy1!_&ef?Re zfAW5){`~zAf9q9`x$)S{w#$5 z@n(oWdl=GhQ*YzVFj{>b+TSJqXI_Q&&m4y4lUE`A1N1BE{OId2`*Zh0^Os+T^pB`< z{#i)d#B)af+tm0wuNPmu^x#!E`}NnM{Hgn)y&(7H>rkWDf9-W>zEAD{$=Uy%v%g0T zzUc9@!_fU};y(N=biaoG3-r47Dl{$gJfesHjPIYn8R{MC{VDqRz1JZ=f&bmZ5KlS( zN5C%;^KaP22d_i-?_P!e2d_fGEa~^C_fI|xvxl!k{|DrLks81CW(Z&UEDXPh|IZ#` z6T_vq?;!uxo1tXZ=EFlSp18Nc|N1P9*XjLJxGw`PF&?tV4d(rr_#e>6U!;e>&K!UE zDqQ?ScKK`6eZ@Tg3EzLC?tkGt`sw}(=l@6g`~vYm23qQW{Vx*t_oyfBE$D>-Rpt`nk{l#JwL`{oK7D zc=O~d!*B99f8yS~x8Hr^{rBE` z`o=e&6f1-Ed&Ql7`oY`pzW45XUngqw-o_m}{NSCx_wGBt^2Yz={nPh;{A=%j-)mA>egFUf literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiNoNorm/outputs.npz b/DeeployTest/Tests/TestiNoNorm/outputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..bb5e96c1833aff19aabc2ecb3668de7420983847 GIT binary patch literal 131342 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 EUw1*Xwg3PC literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiSoftmaxLarge/activations.npz b/DeeployTest/Tests/TestiSoftmaxLarge/activations.npz new file mode 100644 index 0000000000000000000000000000000000000000..15cb0ecb3e219d1701294bfdf0fe3f5cb5d208e7 GIT binary patch literal 22 NcmWIWW@Tf*000g10H*)| literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/TestiSoftmaxLarge/inputs.npz b/DeeployTest/Tests/TestiSoftmaxLarge/inputs.npz new file mode 100644 index 0000000000000000000000000000000000000000..e7a20ce3b3a3d1976420bf6a65e18de4cb9dc039 GIT binary patch literal 65800 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 literal 0 HcmV?d00001 diff --git a/DeeployTest/Tests/testGEMM/inputs.npz b/DeeployTest/Tests/testGEMM/inputs.npz index d94a87ef5b90fca66d976d43aa90614b150a237c..fed9cbdf5965aeb241d9c259fb1a446d14420c2f 100644 GIT binary patch delta 2292 zcmZvecQ_P|1IABxXI>e1c4U)PTxUh$IwR}snQRiWDrKA%amg0Blf73-$%-iBYn@YP z9&u)gjyo&)J^jWX-{13npXdGOecr!6&*xD35Q;z>0YNape)y?vH)NJ z*Wlor7Zn2nA20wo{yqLVMuRf98Ig=|WqA>}a(a{&M!tcj-4l*nO|r}mE+u`kn}oFL zcvX-TCkc{y!~IFH9drmPYV?Bk_>!Gc2p$SOBVDMIq<&KP;(92rc$5)U6;})ZNHl}X zeL~YmLF?5TFZN@)z0bcZS=qxXj?VzjEHy(8WG9Sz6RtI!gaVBzmwZOpVeT5w%udrj zH@?JtJTI<2k0M9^R4x^3oC7Xan{0b?8=ge7Qfo9$o?Sk5u-GZ?e1sfVbcdP05`0?K zs#BW$rDhEq+cZ|F0w~W^tlm-6^cgP5J&uhdyM0U9uI$g8?-$SwyVT>?@pK^`>CYDN z#pGb~FSRs2A?_b}Xw3sUO^YbHCgKque((un{UW7Ljc>Fnqf*5TE44NEfZ-;+*6ujB z$Kp<$ud!E&O_UbOwx$rfX2q&2py26M@m{U(bW44B z1C$(_+}AT`#ZAJ9R=Kwk5g7_3olh9w=0* z$LQM&YzF%IK)AfqXY z>wWGLwh1>xphTID$^G{xn2^?v-+NXj`iM89!86~44J{M7J)M~4>Z8}- z&4!0-09K>ZedGj%yTYmGr7+EM<4jK7=RZ(qM?4+xQO?~^j!>MViav3$I|bgxVaSs0?$5pCgp`hdgAC%F9;pwBg_n z*pobA0P8`f2g6lHI{WqOzzvsWk*vY*9H)TCrsF6LP&(==nGg6;1|!j`q{=7C_0=?p z330nVF!V%}AoPS@PcJOqA^qZ44F2}|){m|s`T$paBCv6&Y}dbW(*pjoZ(1n;A2&GR z+r5{Ct`5ctkS;aqt|Z0oe!n&Ki$i_umd6{)|dpMO)8n_`EAl6DNhyNKI5HKUzF z4MW`f8B;0FvwrK=36#c40bwxvRnM}vpYscW=)z7`$Xu7uv^NhFTuMCgcrFwrxxHB2tz!zdc?sNByEOL<)h?USTpk1)1@%jlm#BhL6ra0lf>0*)Z8d zbwI0p)TDoTQyQ;B+_d7MP%QXBKXpX>(keX>zVZMr zr23N$X>#~>GCU%zm$qZ0KDWYVI*zgTi<4x+k6;b4Y3)a=S1&VgA#mEC?FN`F*qmr^ zf@&&o!_FGlYCW>mJ%r6$#ZLQ+F+_31q!ks(dsX(EY!HJ*cze(Bba}oQC8jn#fG8w~ zu8)1nc1DhMF}zy%7H%=F&XdZH>&v%$*J@{+!Zl)=)u&o}`|eC_Y9z^M7!xHl|HA@= z(|M^xdWWuZ2-Llho+Z`A=*KfUC?+FykGq&AB70K1RP|t0Fud`EwKqL(nQV(+ITAL~ zz0uAD4;`@84&;;x$tDZ12c{=;0=)RTxyy&#qB61 zCOq~7H1wgdaE0}-((@fikamnj@t!32TlpY3k7P+fqTS>jc2?-HcZ;VtmwtP$$oGofwclI29=>Up z9DLCAF64=4?DE9Dw~K;xZUkGw;fuCp@PW7MPd8j>cdxZU1XYyo-neA{QD>Lq4`t?AnDbxP1eE&W(+X)cz=xCR*U&}`r(nQ; fa+S<9sKVL5g+CZ68(6eBcKij_mBH=#b delta 4856 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 Lc+^;A|JwcsuZBDE diff --git a/DeeployTest/Tests/testGEMM/network.onnx b/DeeployTest/Tests/testGEMM/network.onnx index 23adceffc50591ca9c12135d713e71d617e82767..2ce63976402001d016be7a421014fbdc995df33a 100644 GIT binary patch literal 8428 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 diff --git a/DeeployTest/Tests/testGEMM/outputs.npz b/DeeployTest/Tests/testGEMM/outputs.npz index c2cb4e2a95257a8150f78b22fc2fc3afcd7c43b3..a0898021dd3821eb7c219cd2338ee50f9a502f7d 100644 GIT binary patch literal 4360 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

Pd$JuY*=* zux(%O=SjM@ItSTGUQ$1de++oeuHYAsTX9tLRQP~1oN3q~XLpq2GsP&8znuCi!yRM8 zNG|S%(WaMprmOw#ZpOTOWfZA4#p4^x7b%_(atT~?8(s?iV2+;`dcPEWw8r~Wr6qFmZW7;2ZbvmMNQ6rOJbq$8XPWFzE;ECDkw0af+s6925g~RUV zDzgudFX8#;$}v&$_8fePahZW88LfCV#}o?QDr?Q@Uph5kMJqdW)}t)6OK{i{mmrzoDTMs%I`a6&-Mv!FY=8RTqxccMMh@UruJR~8?v`)(+ zCmlp8S4pbilHExj{S#*uZuTRM&9X8uI*@avWfm-TCzDs_atmxPehZ14)EvAxHWzB5 z*^Bd_fUsbgBo&K8GD!&xzvz#~6X1clILa8&TGG{{&2+&qXSKJ!&^I?J8f5l2GqxV> zKr?C&p{L7P`elT@sumvQsm5Jzb?&g8FI}=GIj#ir3o7P|ZO6au6gsSSEKu96)~uk~ z+PngXcbHKP5ijtZ%EO`mh6b`+6VzQF008&@UuZz&UufXoyw|_-{{{^h!ruHF8F=#l z1p}!Xf~+e@W^*)TINz1yvtyS8O(|- z7?iElZArKF3f_x5Tmhy1VOhycovICp>Q>JZV%0D+PqL=HK_HHF)@MPT9GYT7Hd3!G zBatY7am7|gMuY=od0S|o^)f;N02f<*k(20Uu|t0P%Wp9(C>hqlYQ^1+jtCsNOTRqX zh)oKskrzD3cc7EUDVlv>iwfb$d5U@BOv{3^5FWmA%E^0><^(dCkgyA%psPbOo8t45 zM0kR=Fd9L8;_gJ=l~Jl84eU#q3aZR0`C;5~?YBX_;iR`_KzkKD>gF&bp(o7;@^Ez8aZzdy};$ciApK6sK}j4`e+Nu^}Bu?tWR@&wcu4L%8TwcA=_5`+KFR^&osP z?eUA`@Z0ybmGa?L18?d-5%KAoZ?xHrziT_=n0h2@HLDRCK*K%Xu4 j`+eV!{-Y2aF!?`g|9S0DO%UFF87&fExSXrRzqW>*JbCnmKHj9wY8U(s7Dp&^_ zU^#paCGZ7&27cHI0ayc@K;zy854;BN%-R)bAuI#!@nTi9i>lV`NM(+`>@Y{3{F!6aR?w}QLAz)0BToc-Vjd8@pA($pMNi0p? zj~@m#DT$82IJq4BVR(XHi;giiO?;-A$U$O*&_q6sD(=FLUS%xf`mp2jxvW#@4Y(Cx zUDP5!j;@$xlC>>3M&1wi!Te;*P4ai(0o22V0Bc&toI$%;ql;JwZsUic8v7(vVNalS zpjt-JDCk_h0b`zGpMhE^W?zkYO5BfM1pUO8;`c!}{%KV8#Nx)3V|Nfc56772L=Ad~ zc$~cIJ;?ln#2T?ni92BsbYB$^CwDqATN7ioPA72>b}6V&ns=qy6}wfcmF% z51>bh*I++Flk7jj{4x9x{?BNPF)_=&Z>{$o@g5k05&R<18ro+Vy8~5xka=3~A@&W> zUX$e1*94kIwf-ZB;E%!}c0TOI9-t2O*@Jx(>R^!Ac~o=NpIU6)LoY)uq)pHk(GTw>a!&hIQfO@VP%CU8iMcArQ=hyd$>Rv`J#va$D_(|*}b`RWv zRD$mavg9BT+93>8P%|6TSt0?uuQ{mh zM{jzPxN-ydPZRd{{xWRUnMT!T-HjnPiCuuLxbFKfI!#>nZNV-4A~eZ*+AECKKrOLT z@Q|23_fbgUSECcCa{X`ubnbFgH6|bhdJpQs0F>j$E&E>dUOdE}?U#>TORNGOfzG4v zz8+L(mvPht7#w80=BS@~4}FQi><>QQ5huTQoctRe z)h{Z>rYr5f^fES$uUhoxMz9Z{hWuV)7H8OkY7f;n^FHurY<%5qoH32qW5oAE4z|v4 z2rgjj{8MN>WBu5JtUZ93Vv{6R%b926Jgm6V$>OresZ;!azP#of421Q~K7F=mE48ia(ib?x50 zKOl&;aV6NoT4+%IpyEccECNN*L>-@3r*>}qE)VZ_-t(QGd(Lq~zD&QljY{K=pS2}3 zj1jrUZ^E9}`y?2Rzq8DP8&ieqRE)joeC~Ex=B*ia`XY&!p3`JIE&cUQz;<56yg@G< z?TvYT5&7-mH;D*-;&nJ4q5Y_?HelD+1nl?reIsu@l}cT}N0^0Oc*eeh5QAgdC-4J) z!4O^#9FiYFJ9!lhKr=jqGknD4!(O_sg1iLgX-~rk_zi`4=cv|OKvlDkeu5d$x~+Jt z=u^}M=@ji8RPmF%6!pP3a(ScZUyyf&s`hKTFNvzY3?|7Zpn!ZC-2$z-2z7Xwut7ct z+WTiv?PJ&>*R|{1y3sGF=KeQEb9CVAOiszyK>xN0I+qo=1kI;A(0-TT4;15NgU(Io zQjLB`i%`v{_4N-2cq6FBHlpe)2i3{j0oCfC)VGc5ycD`WH)Lh_eRn-x2BaftDf$1D R-B6ZxXDq%3bpGZ!UjwX(MhgG{ literal 0 HcmV?d00001 diff --git a/DeeployTest/testRunner_tiled_snitch.py b/DeeployTest/testRunner_tiled_snitch.py index 34ebefb..4cd5324 100644 --- a/DeeployTest/testRunner_tiled_snitch.py +++ b/DeeployTest/testRunner_tiled_snitch.py @@ -37,9 +37,17 @@ default = 9, help = 'Set number of cluster cores') parser.set_defaults(toolchain_install_dir = "/usr/pack/riscv-1.0-kgf/pulp-llvm-0.12.0") + parser.add_argument('--simulator', + metavar = "", + dest = "simulator", + type = str, + choices = ["banshee", "vsim", "vsim.gui"], + default = "banshee", + help = "Select the simulator to use") + args = parser.parse_args() - testRunner = TestRunner(platform = "Snitch", simulator = "banshee", tiling = True, argument_parser = parser) + testRunner = TestRunner(platform = "Snitch", simulator = args.simulator, tiling = True, argument_parser = parser) testRunner.cmake_args += f" -D NUM_CORES={args.cores}" diff --git a/Makefile b/Makefile index b2b33c9..5ce77e7 100644 --- a/Makefile +++ b/Makefile @@ -265,7 +265,8 @@ pulp-sdk: ${PULP_SDK_INSTALL_DIR} ${TOOLCHAIN_DIR}/snitch_cluster: cd ${TOOLCHAIN_DIR} && \ git clone https://github.com/pulp-platform/snitch_cluster.git && \ - cd ${TOOLCHAIN_DIR}/snitch_cluster && git submodule update --init --recursive && \ + cd ${TOOLCHAIN_DIR}/snitch_cluster && git checkout ${SNITCH_COMMIT_HASH} && \ + git submodule update --init --recursive && \ git checkout ${SNITCH_COMMIT_HASH} && git apply ${TOOLCHAIN_DIR}/snitch_cluster.patch ${SNITCH_INSTALL_DIR}: ${TOOLCHAIN_DIR}/snitch_cluster diff --git a/TargetLibraries/Snitch/CMakeLists.txt b/TargetLibraries/Snitch/CMakeLists.txt index 634aa3c..49130ee 100644 --- a/TargetLibraries/Snitch/CMakeLists.txt +++ b/TargetLibraries/Snitch/CMakeLists.txt @@ -1,5 +1,6 @@ file(GLOB_RECURSE SOURCES "src/**" + "third_party/pulp-nn-mixed/DeeploySnitch/src/**" ) include(cmake/snitch-runtime-precompiled.cmake) @@ -8,6 +9,7 @@ add_deeploy_library(deeploysnitch STATIC ${SOURCES}) target_include_directories(deeploysnitch PUBLIC ${CMAKE_CURRENT_LIST_DIR}/inc + ${CMAKE_CURRENT_LIST_DIR}/inc/kernel ) diff --git a/TargetLibraries/Snitch/inc/DeeploySnitchMath.h b/TargetLibraries/Snitch/inc/DeeploySnitchMath.h index e64124a..193df11 100644 --- a/TargetLibraries/Snitch/inc/DeeploySnitchMath.h +++ b/TargetLibraries/Snitch/inc/DeeploySnitchMath.h @@ -54,5 +54,8 @@ #include "kernel/RQMatMul.h" #include "kernel/Softmax.h" #include "kernel/UniformRequantShift.h" +#include "kernel/iNoNorm.h" + +#include "dmaStruct.h" #endif //__DEEPLOY_MATH_HEADER_ diff --git a/TargetLibraries/Snitch/inc/dmaStruct.h b/TargetLibraries/Snitch/inc/dmaStruct.h new file mode 100644 index 0000000..bf36074 --- /dev/null +++ b/TargetLibraries/Snitch/inc/dmaStruct.h @@ -0,0 +1,37 @@ +/* ---------------------------------------------------------------------- +# +# File: dmaStruct.h +# +# Last edited: 03.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 "snrt.h" + +typedef struct { + void *dst; + void *src; + size_t size; + size_t dst_stride; + size_t src_stride; + size_t repeat; + snrt_dma_txid_t tid; +} DMA_copy; \ No newline at end of file diff --git a/TargetLibraries/Snitch/inc/kernel/Gemm.h b/TargetLibraries/Snitch/inc/kernel/Gemm.h index f320942..e845e15 100644 --- a/TargetLibraries/Snitch/inc/kernel/Gemm.h +++ b/TargetLibraries/Snitch/inc/kernel/Gemm.h @@ -48,6 +48,33 @@ /* General Matrix Multiplication (8bit) */ /******************************************************************************/ +/* + * General Matrix Multiplication + * transposed A = no + * transposed B = no + * multi-core = yes + * unrolling = no + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta); + +/* + * General Matrix Multiplication + * transposed A = no + * transposed B = yes + * multi-core = yes + * unrolling = no + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta); /* * General Matrix Multiplication ---------------------------------- * kernel = Gemm_parallel_s8_rv32im diff --git a/TargetLibraries/Snitch/inc/kernel/RQGemm.h b/TargetLibraries/Snitch/inc/kernel/RQGemm.h index e86b579..0fe5cbf 100644 --- a/TargetLibraries/Snitch/inc/kernel/RQGemm.h +++ b/TargetLibraries/Snitch/inc/kernel/RQGemm.h @@ -48,6 +48,65 @@ /* General Requantized Matrix Multiplication (8bit) */ /******************************************************************************/ +/* + * General Requantized Matrix Multiplication ---------------------------------- + * transposed A = no + * transposed B = no + * multi-core = yes + * unrolling = no + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, int32_t log2D); + +/* + * General Requantized Matrix Multiplication ---------------------------------- + * transposed A = no + * transposed B = no + * multi-core = yes + * unrolling = yes + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, + int32_t log2D); + +/* + * General Requantized Matrix Multiplication ---------------------------------- + * transposed A = no + * transposed B = yes + * multi-core = yes + * unrolling = no + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, + int32_t log2D); + +/* + * General Requantized Matrix Multiplication ---------------------------------- + * transposed A = no + * transposed B = yes + * multi-core = yes + * unrolling = yes + * simd = no + * parallelization = row-wise + * bias pushing = no + */ +void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, + int32_t *add, int32_t log2D); + /* * General Requantized Matrix Multiplication ---------------------------------- * kernel = RQGemm_parallel_s8_rv32im diff --git a/TargetLibraries/Snitch/inc/kernel/iNoNorm.h b/TargetLibraries/Snitch/inc/kernel/iNoNorm.h new file mode 100644 index 0000000..70ec15f --- /dev/null +++ b/TargetLibraries/Snitch/inc/kernel/iNoNorm.h @@ -0,0 +1,30 @@ +/* ---------------------------------------------------------------------- +# +# File: iNoNorm.h +# +# Last edited: 06.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 "DeeploySnitchMath.h" + +void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int32_t *bias, uint32_t size, int32_t mul, + int32_t log2D); diff --git a/TargetLibraries/Snitch/inc/macros.h b/TargetLibraries/Snitch/inc/macros.h index 44c708e..2756d83 100644 --- a/TargetLibraries/Snitch/inc/macros.h +++ b/TargetLibraries/Snitch/inc/macros.h @@ -29,7 +29,12 @@ #ifndef __DEEPLOY_MATH_MACROS_HEADER_ #define __DEEPLOY_MATH_MACROS_HEADER_ -// #define log2(x) __builtin_pulp_fl1(x) #define INT_LOG2(x) __builtin_ctz(x) +#define CLAMP(x, low, high) (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x))) +#define MIN(a, b) ((a) < (b) ? (a) : (b)) +#define MAX(a, b) ((a) > (b) ? (a) : (b)) + +// JUNGVI: The following macros are here to ensure compatibility with some PULP-NN kernels +#define clips8(x) CLAMP(x, -128, 127) #endif //__DEEPLOY_MATH_MACROS_HEADER_ diff --git a/TargetLibraries/Snitch/src/Add.c b/TargetLibraries/Snitch/src/Add.c new file mode 100644 index 0000000..2b62efb --- /dev/null +++ b/TargetLibraries/Snitch/src/Add.c @@ -0,0 +1,49 @@ +/* ---------------------------------------------------------------------- +# +# File: Add.c +# +# Last edited: 11.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 "DeeploySnitchMath.h" + +void SnitchAdd(int8_t *pIn1, int8_t *pIn2, int32_t *pOut, uint32_t size, int32_t offset) { + + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + uint32_t chunk, chunkSize, start, stop; + + chunkSize = size / numThreads; + if (core_id < (numThreads - 1)) { + chunk = chunkSize * core_id; + stop = chunk + chunkSize; + } else { + chunk = (chunkSize * core_id - 1) + (size - chunk); + stop = size; + } + start = chunk; + +#pragma loopunroll 2 + for (int i = start; i < stop; i++) { + pOut[i] = pIn1[i] + pIn2[i] + offset; + } +} diff --git a/TargetLibraries/Snitch/src/Gemm_s8.c b/TargetLibraries/Snitch/src/Gemm_s8.c new file mode 100644 index 0000000..63d7914 --- /dev/null +++ b/TargetLibraries/Snitch/src/Gemm_s8.c @@ -0,0 +1,54 @@ +#include "DeeploySnitchMath.h" +#include "Gemm.h" + +void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + for (uint32_t m = MStart; m < MEnd; m++) { + for (uint32_t o = 0; o < O; o++) { + int32_t sum = 0; + for (uint32_t n = 0; n < N; ++n) { + sum += (int32_t)pSrcA[m * N + n] * pSrcB[n * O + o]; + } + sum = alpha * sum + beta * pSrcC[m * O + o]; + + pDstY[m * O + o] = sum; + } + } +} + +void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + for (uint32_t m = MStart; m < MEnd; m++) { + for (uint32_t o = 0; o < O; o++) { + int32_t sum = 0; + for (uint32_t n = 0; n < N; ++n) { + sum += (int32_t)pSrcA[m * N + n] * pSrcB[o * N + n]; + } + sum = alpha * sum + beta * pSrcC[m * O + o]; + + pDstY[m * O + o] = sum; + } + } +} diff --git a/TargetLibraries/Snitch/src/RQGemm_s8.c b/TargetLibraries/Snitch/src/RQGemm_s8.c index 1850bf0..fc2576a 100644 --- a/TargetLibraries/Snitch/src/RQGemm_s8.c +++ b/TargetLibraries/Snitch/src/RQGemm_s8.c @@ -28,13 +28,294 @@ */ #include "DeeploySnitchMath.h" -void RQGemm_parallel_s8_rv32im( - int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t P, int32_t alpha, int32_t beta, int32_t transA, - int32_t transB, int32_t *mul, int32_t *add, int32_t log2D, bool rounding, - bool per_row_quant, int32_t A_offset, int32_t B_offset, int32_t C_offset, - int32_t Y_offset, int8_t output_min, int8_t output_max) { +#include "RQGemm.h" + +// Assumptions: +// - per-row requantization +// - single batch +void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, int32_t log2D) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + if (core_id < numThreads) { + for (uint32_t m = MStart; m < MEnd; m++) { + for (uint32_t o = 0; o < O; o++) { + int32_t sum = 0; + for (uint32_t n = 0; n < N; ++n) { + sum += (int32_t)pSrcA[m * N + n] * pSrcB[n * O + o]; + } + sum = alpha * sum + beta * pSrcC[m * O + o]; + + // Requantize value + sum = (sum * mul[m] + add[m]) >> log2D; + pDstY[m * O + o] = (int8_t)CLAMP(sum, -128, 127); + } + } + } +} + +// Assumptions: +// - per-row requantization +// - transposed input B +// - single batch +void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, + int32_t log2D) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + if (core_id < numThreads) { + for (uint32_t m = MStart; m + 1 < MEnd; m += 2) { + for (uint32_t o = 0; o + 1 < O; o += 2) { + int32_t sum0 = 0; + int32_t sum1 = 0; + int32_t sum2 = 0; + int32_t sum3 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (o + 0)]; + sum1 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (o + 1)]; + sum2 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[n * O + (o + 0)]; + sum3 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[n * O + (o + 1)]; + } + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (o + 0)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 0) * O + (o + 1)]; + sum2 = alpha * sum2 + beta * pSrcC[(m + 1) * O + (o + 0)]; + sum3 = alpha * sum3 + beta * pSrcC[(m + 1) * O + (o + 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 0] + add[m + 0]) >> log2D; + sum2 = (sum2 * mul[m + 1] + add[m + 1]) >> log2D; + sum3 = (sum3 * mul[m + 1] + add[m + 1]) >> log2D; + pDstY[(m + 0) * O + (o + 0)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 0) * O + (o + 1)] = (int8_t)CLAMP(sum1, -128, 127); + pDstY[(m + 1) * O + (o + 0)] = (int8_t)CLAMP(sum2, -128, 127); + pDstY[(m + 1) * O + (o + 1)] = (int8_t)CLAMP(sum3, -128, 127); + } + + if (O % 2 == 1) { + int32_t sum0 = 0; + int32_t sum1 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (O - 1)]; + sum1 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[n * O + (O - 1)]; + } + + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (O - 1)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 1) * O + (O - 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 1] + add[m + 1]) >> log2D; + pDstY[(m + 0) * O + (O - 1)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 1) * O + (O - 1)] = (int8_t)CLAMP(sum1, -128, 127); + } + } + + if (MSize % 2 == 1) { + uint32_t m = MEnd - 1; + + for (uint32_t o = 0; o + 1 < O; o += 2) { + int32_t sum0 = 0; + int32_t sum1 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (o + 0)]; + sum1 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (o + 1)]; + } + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (o + 0)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 0) * O + (o + 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 0] + add[m + 0]) >> log2D; + pDstY[(m + 0) * O + (o + 0)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 0) * O + (o + 1)] = (int8_t)CLAMP(sum1, -128, 127); + } + + if (O % 2 == 1) { + int32_t sum0 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[n * O + (O - 1)]; + } + + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (O - 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + pDstY[(m + 0) * O + (O - 1)] = (int8_t)CLAMP(sum0, -128, 127); + } + } + } +} + +// Assumptions: +// - per-row requantization +// - transposed input B +// - single batch +void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, + int32_t log2D) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + if (core_id < numThreads) { + for (uint32_t m = MStart; m < MEnd; m++) { + for (uint32_t o = 0; o < O; o++) { + int32_t sum = 0; + for (uint32_t n = 0; n < N; ++n) { + sum += (int32_t)pSrcA[m * N + n] * pSrcB[o * N + n]; + } + sum = alpha * sum + beta * pSrcC[m * O + o]; + + // Requantize value + sum = (sum * mul[m] + add[m]) >> log2D; + pDstY[m * O + o] = (int8_t)CLAMP(sum, -128, 127); + } + } + } +} + +// Assumptions: +// - per-row requantization +// - transposed input B +// - single batch +void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, + int32_t *add, int32_t log2D) { + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + + // Parallelize by assigning each core a row tile + uint32_t const MQuotient = M / numThreads; + uint32_t const MRemainder = M % numThreads; + uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); + uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MEnd = MStart + MSize; + + if (core_id < numThreads) { + for (uint32_t m = MStart; m + 1 < MEnd; m += 2) { + for (uint32_t o = 0; o + 1 < O; o += 2) { + int32_t sum0 = 0; + int32_t sum1 = 0; + int32_t sum2 = 0; + int32_t sum3 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(o + 0) * N + n]; + sum1 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(o + 1) * N + n]; + sum2 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[(o + 0) * N + n]; + sum3 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[(o + 1) * N + n]; + } + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (o + 0)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 0) * O + (o + 1)]; + sum2 = alpha * sum2 + beta * pSrcC[(m + 1) * O + (o + 0)]; + sum3 = alpha * sum3 + beta * pSrcC[(m + 1) * O + (o + 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 0] + add[m + 0]) >> log2D; + sum2 = (sum2 * mul[m + 1] + add[m + 1]) >> log2D; + sum3 = (sum3 * mul[m + 1] + add[m + 1]) >> log2D; + pDstY[(m + 0) * O + (o + 0)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 0) * O + (o + 1)] = (int8_t)CLAMP(sum1, -128, 127); + pDstY[(m + 1) * O + (o + 0)] = (int8_t)CLAMP(sum2, -128, 127); + pDstY[(m + 1) * O + (o + 1)] = (int8_t)CLAMP(sum3, -128, 127); + } + + if (O % 2 == 1) { + int32_t sum0 = 0; + int32_t sum1 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(O - 1) * N + n]; + sum1 += (int32_t)pSrcA[(m + 1) * N + n] * pSrcB[(O - 1) * N + n]; + } + + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (O - 1)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 1) * O + (O - 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 1] + add[m + 1]) >> log2D; + pDstY[(m + 0) * O + (O - 1)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 1) * O + (O - 1)] = (int8_t)CLAMP(sum1, -128, 127); + } + } + + if (MSize % 2 == 1) { + uint32_t m = MEnd - 1; + + for (uint32_t o = 0; o + 1 < O; o += 2) { + int32_t sum0 = 0; + int32_t sum1 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(o + 0) * N + n]; + sum1 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(o + 1) * N + n]; + } + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (o + 0)]; + sum1 = alpha * sum1 + beta * pSrcC[(m + 0) * O + (o + 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + sum1 = (sum1 * mul[m + 0] + add[m + 0]) >> log2D; + pDstY[(m + 0) * O + (o + 0)] = (int8_t)CLAMP(sum0, -128, 127); + pDstY[(m + 0) * O + (o + 1)] = (int8_t)CLAMP(sum1, -128, 127); + } + + if (O % 2 == 1) { + int32_t sum0 = 0; +#pragma unroll 2 + for (uint32_t n = 0; n < N; ++n) { + sum0 += (int32_t)pSrcA[(m + 0) * N + n] * pSrcB[(O - 1) * N + n]; + } + + sum0 = alpha * sum0 + beta * pSrcC[(m + 0) * O + (O - 1)]; + + // Requantize value + sum0 = (sum0 * mul[m + 0] + add[m + 0]) >> log2D; + pDstY[(m + 0) * O + (O - 1)] = (int8_t)CLAMP(sum0, -128, 127); + } + } + } +} + +void RQGemm_parallel_s8_rv32im(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t P, int32_t alpha, int32_t beta, int32_t transA, int32_t transB, int32_t *mul, + int32_t *add, int32_t log2D, bool rounding, bool per_row_quant, int32_t A_offset, + int32_t B_offset, int32_t C_offset, int32_t Y_offset, int8_t output_min, + int8_t output_max) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); diff --git a/TargetLibraries/Snitch/src/iNoNorm.c b/TargetLibraries/Snitch/src/iNoNorm.c new file mode 100644 index 0000000..f18f19b --- /dev/null +++ b/TargetLibraries/Snitch/src/iNoNorm.c @@ -0,0 +1,102 @@ +/* ---------------------------------------------------------------------- +# +# File: iNoNorm.c +# +# Last edited: 06.06.2024 +# +# Copyright (C) 2024, ETH Zurich and University of Bologna. +# +# Author: +# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# +# 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 "DeeploySnitchMath.h" + +void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int32_t *bias, uint32_t size, int32_t mul, + int32_t log2D) { + + uint32_t core_id = snrt_global_compute_core_idx(); + uint32_t numThreads = snrt_global_compute_core_num(); + uint32_t chunk, chunkSize, start, stop; + + chunkSize = size / numThreads; + if (core_id < (numThreads - 1)) { + chunk = chunkSize * core_id; + stop = chunk + chunkSize; + } else { + chunk = (chunkSize * core_id - 1) + (size - chunk); + stop = size; + } + start = chunk; + + uint32_t packedIn, packedWeights; + int8_t unpackedIn1, unpackedIn2, unpackedIn3, unpackedIn4; + int8_t unpackedWeights1, unpackedWeights2, unpackedWeights3, unpackedWeights4; + int16_t partialProduct1, partialProduct2, partialProduct3, partialProduct4; + + int32_t *dataInPtr = (int32_t *)(data_in); + int32_t *weightsPtr = (int32_t *)(weights); + int32_t *outputPtr = (int32_t *)(data_out); + + uint32_t firstReminderLoopSize = start % 4; + uint32_t lastReminderLoopSize = stop % 4; + uint32_t firstReminderLoopIdx = start; + uint32_t lastReminderLoopIdx = stop - lastReminderLoopSize; + start = (start + firstReminderLoopSize) >> 2; + stop = (stop - lastReminderLoopSize) >> 2; + uint32_t biasIdx = start * 4; + + // JUNGVI: Compute sequentially the first elements not aligned to a word (32b) + for (uint32_t i = firstReminderLoopIdx; i < firstReminderLoopIdx + firstReminderLoopSize; i++) { + data_out[i] = ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; + } + + for (uint32_t i = start; i < stop; i++) { + + packedIn = dataInPtr[i]; + packedWeights = weightsPtr[i]; + + unpackedIn1 = (packedIn & 0x000000FF); + unpackedIn2 = (packedIn & 0x0000FF00) >> 8; + unpackedIn3 = (packedIn & 0x00FF0000) >> 16; + unpackedIn4 = packedIn >> 24; + + unpackedWeights1 = (packedWeights & 0x000000FF); + unpackedWeights2 = (packedWeights & 0x0000FF00) >> 8; + unpackedWeights3 = (packedWeights & 0x00FF0000) >> 16; + unpackedWeights4 = packedWeights >> 24; + + partialProduct1 = (int16_t)(unpackedIn1 * unpackedWeights1); + partialProduct2 = (int16_t)(unpackedIn2 * unpackedWeights2); + partialProduct3 = (int16_t)(unpackedIn3 * unpackedWeights3); + partialProduct4 = (int16_t)(unpackedIn4 * unpackedWeights4); + + uint8_t outBuf1 = ((partialProduct1 + bias[biasIdx + 0]) * mul) >> log2D; + uint8_t outBuf2 = ((partialProduct2 + bias[biasIdx + 1]) * mul) >> log2D; + uint8_t outBuf3 = ((partialProduct3 + bias[biasIdx + 2]) * mul) >> log2D; + uint8_t outBuf4 = ((partialProduct4 + bias[biasIdx + 3]) * mul) >> log2D; + + uint32_t outPacked = (outBuf1 << 0) | (outBuf2 << 8) | (outBuf3 << 16) | (outBuf4 << 24); + outputPtr[i] = outPacked; + biasIdx += 4; + } + + // JUNGVI: Compute sequentially the last elements not aligned to a word (32b) + for (uint32_t i = lastReminderLoopIdx; i < lastReminderLoopIdx + lastReminderLoopSize; i++) { + data_out[i] = ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; + } +} diff --git a/TargetLibraries/Snitch/third_party/pulp-nn-mixed b/TargetLibraries/Snitch/third_party/pulp-nn-mixed new file mode 160000 index 0000000..4f0902b --- /dev/null +++ b/TargetLibraries/Snitch/third_party/pulp-nn-mixed @@ -0,0 +1 @@ +Subproject commit 4f0902bbdab3265b4f56e156cb4a98f52ac6e067 From a0263424986fa01fa26c4fa5778904b31ecae303 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 6 Dec 2024 21:50:53 +0100 Subject: [PATCH 02/11] CI flow and linting --- .github/workflows/BuildDocker.yml | 2 +- .github/workflows/CI.yml | 51 +++++++++-- .github/workflows/TestRunnerCortexM.yml | 2 +- .github/workflows/TestRunnerGeneric.yml | 2 +- .github/workflows/TestRunnerMempool.yml | 2 +- .github/workflows/TestRunnerSiracusa.yml | 2 +- .github/workflows/TestRunnerSnitch.yml | 4 +- .github/workflows/TestRunnerTiledSiracusa.yml | 2 +- .../TestRunnerTiledSiracusaSequential.yml | 2 +- .../TestRunnerTiledSiracusaWithNeureka.yml | 2 +- ...nnerTiledSiracusaWithNeurekaSequential.yml | 2 +- .github/workflows/TestRunnerTiledSnitch.yml | 47 ++++++++++ Deeploy/Targets/Generic/Layers.py | 4 +- Deeploy/Targets/Generic/Parsers.py | 2 + .../iSoftmaxPreAllocatedBuffTemplate.py | 10 ++- .../TopologyOptimizationPasses/Passes.py | 1 + Deeploy/Targets/Generic/TypeCheckers.py | 8 +- Deeploy/Targets/Snitch/Bindings.py | 64 ++++++++++---- .../SnitchClusterTilingSB.py | 87 +++++++++++-------- Deeploy/Targets/Snitch/Deployer.py | 7 +- Deeploy/Targets/Snitch/Platform.py | 23 +++-- .../Targets/Snitch/Templates/AddTemplate.py | 3 +- .../Targets/Snitch/Templates/GemmTemplate.py | 9 +- .../Targets/Snitch/Templates/RQAddTemplate.py | 3 +- .../Snitch/Templates/RqGemmTemplate.py | 3 +- .../TileConstraints/GemmTileConstraint.py | 12 +-- .../TileConstraints/RqGemmTileConstraint.py | 17 ++-- .../TileConstraints/iNoNormTileConstraint.py | 12 +-- .../TileConstraints/iSoftmaxTileConstraint.py | 13 +-- Deeploy/Targets/Snitch/Tiler.py | 3 +- Deeploy/Targets/Snitch/TypeCheckers.py | 9 +- .../TilingCodeGeneration.py | 5 +- DeeployTest/Platforms/Snitch/main.c | 2 +- TargetLibraries/Snitch/inc/kernel/Gemm.h | 15 ++-- TargetLibraries/Snitch/inc/kernel/RQGemm.h | 36 +++++--- TargetLibraries/Snitch/inc/kernel/iNoNorm.h | 3 +- TargetLibraries/Snitch/inc/macros.h | 6 +- TargetLibraries/Snitch/src/Add.c | 3 +- TargetLibraries/Snitch/src/Gemm_s8.c | 21 +++-- TargetLibraries/Snitch/src/RQGemm_s8.c | 61 ++++++++----- TargetLibraries/Snitch/src/iNoNorm.c | 18 ++-- 41 files changed, 390 insertions(+), 190 deletions(-) create mode 100644 .github/workflows/TestRunnerTiledSnitch.yml diff --git a/.github/workflows/BuildDocker.yml b/.github/workflows/BuildDocker.yml index 47f631e..d07e90a 100644 --- a/.github/workflows/BuildDocker.yml +++ b/.github/workflows/BuildDocker.yml @@ -38,4 +38,4 @@ jobs: file: Container/Dockerfile push: true # JUNGVI: If you operate from a fork and want to build a new docker make sure to replace 'pulp-platform' by your uname. - tags: ghcr.io/pulp-platform/deeploy:main + tags: ghcr.io/tahaelbayad/deeploy:main diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 35a0a56..aae30f5 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -13,7 +13,7 @@ jobs: build-deeploy: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -94,7 +94,42 @@ jobs: uses: ./.github/workflows/TestRunnerSnitch.yml with: test-names: | + Adder + iSoftmax + TestiNoNorm + TestAdderLarge + TestiSoftmaxLarge testMatMul + testRQGEMM + TestRQAdd + testRQGEMMTransB + + snitch-kernels-tiled-singlebuffer-L2: + uses: ./.github/workflows/TestRunnerTiledSnitch.yml + with: + tests-config: | + [ + { + "name": "TestiNoNorm", + "L1": [5000, 10000] + }, + { + "name": "TestAdderLarge", + "L1": [5000, 10000] + }, + { + "name": "TestiSoftmaxLarge", + "L1": [5000, 10000] + }, + { + "name": "testRQGEMM", + "L1": [2000, 5000] + }, + { + "name": "TestRQAdd", + "L1": [5000, 10000] + } + ] ### Mempool Tests ### mempool-kernels: @@ -507,7 +542,7 @@ jobs: deeploy-state-serialization: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -527,7 +562,7 @@ jobs: deeploy-memory-level-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -547,7 +582,7 @@ jobs: deeploy-tiler-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -571,7 +606,7 @@ jobs: deeploy-memory-allocation-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -593,7 +628,7 @@ jobs: deeploy-typing: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -610,7 +645,7 @@ jobs: deeploy-regex-matching: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -627,7 +662,7 @@ jobs: linting: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerCortexM.yml b/.github/workflows/TestRunnerCortexM.yml index 5f0e523..9d80f55 100644 --- a/.github/workflows/TestRunnerCortexM.yml +++ b/.github/workflows/TestRunnerCortexM.yml @@ -11,7 +11,7 @@ jobs: test-runner-cortexm: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerGeneric.yml b/.github/workflows/TestRunnerGeneric.yml index ce20fda..c123ca5 100644 --- a/.github/workflows/TestRunnerGeneric.yml +++ b/.github/workflows/TestRunnerGeneric.yml @@ -11,7 +11,7 @@ jobs: test-runner-generic: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerMempool.yml b/.github/workflows/TestRunnerMempool.yml index e8414ff..a179667 100644 --- a/.github/workflows/TestRunnerMempool.yml +++ b/.github/workflows/TestRunnerMempool.yml @@ -11,7 +11,7 @@ jobs: test-runner-mempool: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerSiracusa.yml b/.github/workflows/TestRunnerSiracusa.yml index 0b8b8cc..0143396 100644 --- a/.github/workflows/TestRunnerSiracusa.yml +++ b/.github/workflows/TestRunnerSiracusa.yml @@ -14,7 +14,7 @@ jobs: test-runner-siracusa: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerSnitch.yml b/.github/workflows/TestRunnerSnitch.yml index 9f2acd2..d2c35ef 100644 --- a/.github/workflows/TestRunnerSnitch.yml +++ b/.github/workflows/TestRunnerSnitch.yml @@ -11,7 +11,7 @@ jobs: test-runner-snitch: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -26,7 +26,7 @@ jobs: echo "$testNames" | while IFS= read -r testName; do if [[ -n "$testName" ]]; then echo "Running test: $testName" - python testRunner_snitch.py -t Tests/$testName --toolchain_install_dir /app/install/riscv-llvm/ + python testRunner_snitch.py -t Tests/$testName --simulator=banshee --cores=9 --toolchain_install_dir /app/install/riscv-llvm/ fi done shell: bash \ No newline at end of file diff --git a/.github/workflows/TestRunnerTiledSiracusa.yml b/.github/workflows/TestRunnerTiledSiracusa.yml index 48a8cb3..9f101c6 100644 --- a/.github/workflows/TestRunnerTiledSiracusa.yml +++ b/.github/workflows/TestRunnerTiledSiracusa.yml @@ -32,7 +32,7 @@ jobs: L1: ${{ fromJSON(inputs.L1) }} runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaSequential.yml b/.github/workflows/TestRunnerTiledSiracusaSequential.yml index 684036a..5e9e275 100644 --- a/.github/workflows/TestRunnerTiledSiracusaSequential.yml +++ b/.github/workflows/TestRunnerTiledSiracusaSequential.yml @@ -24,7 +24,7 @@ jobs: test-runner-siracusa-tiled: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml b/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml index 143c989..d16699f 100644 --- a/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml +++ b/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml @@ -36,7 +36,7 @@ jobs: L1: ${{ fromJSON(inputs.L1) }} runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml b/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml index 3271598..c362d86 100644 --- a/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml +++ b/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml @@ -28,7 +28,7 @@ jobs: test-runner-siracusa-neureka-tiled: runs-on: ubuntu-22.04 container: - image: ghcr.io/pulp-platform/deeploy:main + image: ghcr.io/tahaelbayad/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSnitch.yml b/.github/workflows/TestRunnerTiledSnitch.yml new file mode 100644 index 0000000..82f16be --- /dev/null +++ b/.github/workflows/TestRunnerTiledSnitch.yml @@ -0,0 +1,47 @@ +name: TestRunnerTiledSiracusaSequential + +on: + workflow_call: + inputs: + tests-config: + required: true + type: string + +jobs: + + test-runner-snitch-tiled: + runs-on: ubuntu-22.04 + container: + image: ghcr.io/tahaelbayad/deeploy:main + steps: + - name: Checkout Repo + uses: actions/checkout@v4 + with: + submodules: recursive + - name: Build Deeploy + run: pip install -e . + - name: Install jq + run: apt-get install -y jq + - name: Cache ccache + id: ccache-cache + uses: actions/cache@v4 + with: + path: /app/.ccache + key: ${{ runner.os }}-ccache + - name: Run Tests + run: | + cd DeeployTest + echo '${{ inputs.tests-config }}' > tests.json + mkdir -p /app/.ccache + export CCACHE_DIR=/app/.ccache + + jq -c '.[]' tests.json | while read test; do + testName=$(echo "$test" | jq -r '.name') + L1_values=$(echo "$test" | jq -r '.L1[]') + for L1_value in $L1_values; do + echo "Running test: $testName with L1: $L1_value" + python testRunner_tiled_snitch.py -t Tests/$testName --cores=9 --simulator=banshee --l1 $L1_value --defaultMemLevel=L2 --toolchain_install_dir /app/install/riscv-llvm/ + done + done + shell: bash + \ No newline at end of file diff --git a/Deeploy/Targets/Generic/Layers.py b/Deeploy/Targets/Generic/Layers.py index c963916..a714e6d 100644 --- a/Deeploy/Targets/Generic/Layers.py +++ b/Deeploy/Targets/Generic/Layers.py @@ -25,11 +25,11 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import Dict, List, Tuple +from typing import List, Tuple import numpy as np -from Deeploy.DeeployTypes import NodeMapper, ONNXLayer, Shape, OperatorRepresentation +from Deeploy.DeeployTypes import NodeMapper, ONNXLayer, OperatorRepresentation, Shape class ConcatLayer(ONNXLayer): diff --git a/Deeploy/Targets/Generic/Parsers.py b/Deeploy/Targets/Generic/Parsers.py index 0ab0dde..852e323 100644 --- a/Deeploy/Targets/Generic/Parsers.py +++ b/Deeploy/Targets/Generic/Parsers.py @@ -751,6 +751,7 @@ def parseNodeCtxt(self, return ctxt, True + class iNoNormParser(NodeParser): def __init__(self): @@ -2115,6 +2116,7 @@ def parseNodeCtxt(self, return ctxt, False + class RQAddParser(AddParser): def parseNode(self, node: gs.Node) -> bool: diff --git a/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py b/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py index 30ec181..45b80a7 100644 --- a/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py +++ b/Deeploy/Targets/Generic/Templates/iSoftmaxPreAllocatedBuffTemplate.py @@ -34,13 +34,16 @@ class iSoftmaxPreAllocatedBuffTemplate(NodeTemplate): @staticmethod - def computeTransientBuffersSize(ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> List[Tuple[str, Union[int, IntVar]]]: + def computeTransientBuffersSize( + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> List[Tuple[str, Union[int, IntVar]]]: lastDimBuffer_dim = 8 * 4 * operatorRepresentation['lastDimLength'] lastDimBuffer_name = operatorRepresentation['nodeName'] + "_lastDimBuffer" return [(lastDimBuffer_name, lastDimBuffer_dim)] - def hoistTransientBuffers(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def hoistTransientBuffers(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: lastDimBuffer_name, lastDimBuffer_dim = iSoftmaxPreAllocatedBuffTemplate.computeTransientBuffersSize( ctxt, operatorRepresentation)[0] ctxt.hoistTransientBuffer(lastDimBuffer_name, lastDimBuffer_dim) @@ -49,7 +52,8 @@ def hoistTransientBuffers(self, ctxt: NetworkContext, operatorRepresentation: Op operatorRepresentation['lastDimBufferSize'] = lastDimBuffer_dim return ctxt, operatorRepresentation, [lastDimBuffer_name] - def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: signedI = ctxt.lookup(operatorRepresentation['data_in'])._type.referencedType.typeMin < 0 signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 diff --git a/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py b/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py index 064a34e..3a83f4e 100644 --- a/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py +++ b/Deeploy/Targets/Generic/TopologyOptimizationPasses/Passes.py @@ -867,6 +867,7 @@ def __init__(self): name = "_SPLIT_RequantShift_PASS" super().__init__(graph, partial(_split_rqs_fun, splitSet = self.splitSet), name) + def _merge_add_rq_fun(graph: gs.Graph, match: Match, name: str): nodes_map = match.nodes_map diff --git a/Deeploy/Targets/Generic/TypeCheckers.py b/Deeploy/Targets/Generic/TypeCheckers.py index bd850c2..e1de80e 100644 --- a/Deeploy/Targets/Generic/TypeCheckers.py +++ b/Deeploy/Targets/Generic/TypeCheckers.py @@ -25,7 +25,7 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import Dict, List, Optional, Sequence, Type +from typing import List, Optional, Sequence, Type import numpy as np @@ -397,10 +397,12 @@ class iNoNormChecker(SignPropTypeChecker): def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): super().__init__(input_types, output_types) - def _inferNumLevels(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[int]: + def _inferNumLevels(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[int]: return [2**(4 * self.input_types[0].referencedType.typeWidth)] - def _inferSignedness(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[bool]: + def _inferSignedness(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[bool]: if inputs[0]._signed: return [True] else: diff --git a/Deeploy/Targets/Snitch/Bindings.py b/Deeploy/Targets/Snitch/Bindings.py index f38be6e..b4cb92c 100644 --- a/Deeploy/Targets/Snitch/Bindings.py +++ b/Deeploy/Targets/Snitch/Bindings.py @@ -26,26 +26,33 @@ from functools import partial from Deeploy.AbstractDataTypes import PointerClass -from Deeploy.CommonExtensions.CodeTransformationPasses.MemoryAllocation import ArgumentStructGeneration, MemoryManagementGeneration from Deeploy.CommonExtensions.CodeTransformationPasses.Closure import ClosureGeneration, MemoryAwareClosureGeneration -from Deeploy.TilingExtension.CodeTransformationPasses.TilingVariableReplacement import TilingVariableReplacement -from Deeploy.FutureExtension.CodeTransformationPasses.FutureCodeTransformation import FutureGeneration -from Deeploy.Targets.Snitch.CodeTransformationPasses import SnitchClusterTiling, SnitchCoreFilterPass, \ - SnitchSynchCoresPass, SnitchProfileExecutionBlockPass +from Deeploy.CommonExtensions.CodeTransformationPasses.MemoryAllocation import ArgumentStructGeneration, \ + MemoryManagementGeneration from Deeploy.CommonExtensions.DataTypes import int8_t, int32_t, uint8_t from Deeploy.DeeployTypes import CodeTransformation, NodeBinding +from Deeploy.FutureExtension.CodeTransformationPasses.FutureCodeTransformation import FutureGeneration from Deeploy.Targets.Generic.Templates import iNoNormTemplate +from Deeploy.Targets.Generic.TypeCheckers import AddChecker, GEMMChecker, SoftmaxChecker, iNoNormChecker +from Deeploy.Targets.Snitch.CodeTransformationPasses import SnitchClusterTiling, SnitchCoreFilterPass, \ + SnitchProfileExecutionBlockPass, SnitchSynchCoresPass from Deeploy.Targets.Snitch.Templates import AddTemplate, RQAddTemplate, iSoftmaxTemplate from Deeploy.Targets.Snitch.Templates.GemmTemplate import SnitchGemm_Template from Deeploy.Targets.Snitch.Templates.RqGemmTemplate import SnitchRqGemm_Template -from Deeploy.Targets.Generic.TypeCheckers import AddChecker, GEMMChecker, SoftmaxChecker, iNoNormChecker from Deeploy.Targets.Snitch.TypeCheckers import SnitchRQAddChecker - +from Deeploy.TilingExtension.CodeTransformationPasses.TilingVariableReplacement import TilingVariableReplacement TilingCallClosure = partial(ClosureGeneration, closureSuffix = "_tiling_closure") -MemoryAwareFunctionCallClosure = partial(MemoryAwareClosureGeneration, closureSuffix = "_closure", startRegion = "L2", endRegion = "L1") +MemoryAwareFunctionCallClosure = partial(MemoryAwareClosureGeneration, + closureSuffix = "_closure", + startRegion = "L2", + endRegion = "L1") -BasicTransformer = CodeTransformation([SnitchSynchCoresPass(), ArgumentStructGeneration(), MemoryManagementGeneration(), FutureGeneration()]) +BasicTransformer = CodeTransformation( + [SnitchSynchCoresPass(), + ArgumentStructGeneration(), + MemoryManagementGeneration(), + FutureGeneration()]) TiledTransformer = CodeTransformation([ SnitchCoreFilterPass("compute"), @@ -60,9 +67,36 @@ MemoryManagementGeneration() ]) -SnitchiSoftmaxBindings = [NodeBinding(SoftmaxChecker([PointerClass(_type)], [PointerClass(uint8_t)]), iSoftmaxTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t, uint8_t]] -SnitchiNoNormBindings = [NodeBinding(iNoNormChecker([PointerClass(_type), PointerClass(int8_t), PointerClass(int32_t)], [PointerClass(int8_t)]), iNoNormTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] -SnitchRQAddBindings = [NodeBinding(SnitchRQAddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(_type)]), RQAddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] -SnitchAddBindings = [NodeBinding(AddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(int32_t)]), AddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t]] -SnitchGemmBindings = [NodeBinding(GEMMChecker([PointerClass(int8_t), PointerClass(int8_t), PointerClass(int32_t)], [PointerClass(int32_t)]), SnitchGemm_Template, TiledTransformer)] -SnitchRqGemmBindings = [NodeBinding(GEMMChecker([PointerClass(int8_t), PointerClass(int8_t), PointerClass(int32_t), PointerClass(int32_t), PointerClass(int32_t)], [PointerClass(int8_t)]), SnitchRqGemm_Template, TiledTransformer)] +SnitchiSoftmaxBindings = [ + NodeBinding(SoftmaxChecker([PointerClass(_type)], [PointerClass(uint8_t)]), iSoftmaxTemplate.referenceTemplate, + TiledTransformer) for _type in [int8_t, uint8_t] +] +SnitchiNoNormBindings = [ + NodeBinding( + iNoNormChecker([PointerClass(_type), PointerClass(int8_t), + PointerClass(int32_t)], [PointerClass(int8_t)]), iNoNormTemplate.referenceTemplate, + TiledTransformer) for _type in [int8_t] +] +SnitchRQAddBindings = [ + NodeBinding(SnitchRQAddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(_type)]), + RQAddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t] +] +SnitchAddBindings = [ + NodeBinding(AddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(int32_t)]), + AddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t] +] +SnitchGemmBindings = [ + NodeBinding( + GEMMChecker([PointerClass(int8_t), PointerClass(int8_t), + PointerClass(int32_t)], [PointerClass(int32_t)]), SnitchGemm_Template, TiledTransformer) +] +SnitchRqGemmBindings = [ + NodeBinding( + GEMMChecker([ + PointerClass(int8_t), + PointerClass(int8_t), + PointerClass(int32_t), + PointerClass(int32_t), + PointerClass(int32_t) + ], [PointerClass(int8_t)]), SnitchRqGemm_Template, TiledTransformer) +] diff --git a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py index 19a05fe..f1c221f 100644 --- a/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py +++ b/Deeploy/Targets/Snitch/CodeTransformationPasses/SnitchClusterTilingSB.py @@ -25,11 +25,12 @@ import copy from collections import namedtuple -from typing import Any, Dict, List, Literal, Tuple +from typing import Dict, List, Literal, Tuple + +from Deeploy.DeeployTypes import CodeSnippet, ExecutionBlock, NetworkContext, NodeTemplate, OperatorRepresentation +from Deeploy.Targets.Snitch.DataTypes import Snitch_DMA_copy from Deeploy.TilingExtension.CodeTransformationPasses.TilingCodeGeneration import TilingCodeGeneration from Deeploy.TilingExtension.CodeTransformationPasses.TilingPrototypes import SingleBufferingTilingMixIn, TilingMetaInfo -from Deeploy.Targets.Snitch.DataTypes import Snitch_DMA_copy -from Deeploy.DeeployTypes import ExecutionBlock, NetworkContext, NodeTemplate, OperatorRepresentation, CodeSnippet from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint from Deeploy.TilingExtension.TilingCodegen import HyperRectangle, TilingSchedule, VariableReplacementScheme, \ calculateRectangleOffset, minimizeRectangleDims @@ -282,37 +283,43 @@ def _hoistDMAUpdates(self, ctxt: NetworkContext, tensorName: str, updateList: Li name = namePrefix + "_dst_offset" cb = ctxt.ConstantBuffer(name, [len(updateList)], dstOffsetList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'dstOffsetPtr') + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'dstOffsetPtr') name = namePrefix + "_src_offset" cb = ctxt.ConstantBuffer(name, [len(updateList)], srcOffsetList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'srcOffsetPtr') + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'srcOffsetPtr') name = namePrefix + "_size" cb = ctxt.ConstantBuffer(name, [len(updateList)], sizeList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'sizePtr', - Snitch_DMA_copy.structTypeDict['size']) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'sizePtr', + Snitch_DMA_copy.structTypeDict['size']) name = namePrefix + "_dst_stride" cb = ctxt.ConstantBuffer(name, [len(updateList)], dstStrideList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'dstStridePtr', - Snitch_DMA_copy.structTypeDict['dst_stride']) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'dstStridePtr', + Snitch_DMA_copy.structTypeDict['dst_stride']) name = namePrefix + "_src_stride" cb = ctxt.ConstantBuffer(name, [len(updateList)], srcStideList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'srcStridePtr', - Snitch_DMA_copy.structTypeDict['src_stride']) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'srcStridePtr', + Snitch_DMA_copy.structTypeDict['src_stride']) name = namePrefix + "_repeat" cb = ctxt.ConstantBuffer(name, [len(updateList)], repeatList) - ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, 'repeatPtr', - Snitch_DMA_copy.structTypeDict['repeat']) + ctxt, operatorRepresentation = self._hoistConstantAndReference(ctxt, cb, operatorRepresentation, nodeName, + 'repeatPtr', + Snitch_DMA_copy.structTypeDict['repeat']) return ctxt, operatorRepresentation - def _generateEgressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: + def _generateEgressPointerUpdates( + self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: updates = [] newCtxt = ctxt.copy() @@ -327,15 +334,15 @@ def _generateEgressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstrai return newCtxt, updates - def _generateIngressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: + def _generateIngressPointerUpdates( + self, nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedule: TilingSchedule, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, List[CodeSnippet]]: updates = [] newCtxt = ctxt.copy() - updateDict = self._generatePointerUpdates(ctxt, operatorRepresentation, tilingSchedule.inputLoadSchedule, nodeMemoryConstraint, - tilingSchedule) + updateDict = self._generatePointerUpdates(ctxt, operatorRepresentation, tilingSchedule.inputLoadSchedule, + nodeMemoryConstraint, tilingSchedule) for key, updateList in updateDict.items(): @@ -345,7 +352,8 @@ def _generateIngressPointerUpdates(self, nodeMemoryConstraint: NodeMemoryConstra return newCtxt, updates def _generateVariableUpdates(self, tilingSchedule: TilingSchedule, variableReplacement: VariableReplacementScheme, - ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> List[CodeSnippet]: + ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> List[CodeSnippet]: updates = [] @@ -411,20 +419,24 @@ def _generateDMACode(self, nodeMemoryConstraint: NodeMemoryConstraint, ctxt: Net return DMATransferCalls, DMAWaitStatements - def _generateIngressDMACode(self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, - ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: + def _generateIngressDMACode( + self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: importLoadStep = tilingSchedule.inputLoadSchedule - ingressDMATransferCalls, ingressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, operatorRepresentation, + ingressDMATransferCalls, ingressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, + operatorRepresentation, importLoadStep, "ToL1") return ingressDMATransferCalls, ingressDMAWaitStatements - def _generateEgressDMACode(self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, - ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: + def _generateEgressDMACode( + self, tilingSchedule: TilingSchedule, nodeMemoryConstraint: NodeMemoryConstraint, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[List[CodeSnippet], List[CodeSnippet]]: exportLoadStep = tilingSchedule.outputLoadSchedule - egressDMATransferCalls, egressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, operatorRepresentation, - exportLoadStep, "FromL1") + egressDMATransferCalls, egressDMAWaitStatements = self._generateDMACode(nodeMemoryConstraint, ctxt, + operatorRepresentation, exportLoadStep, + "FromL1") return egressDMATransferCalls, egressDMAWaitStatements @@ -443,7 +455,8 @@ def _tilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, ctxt, ingressDMAUpdates = self._generateIngressPointerUpdates(nodeMemoryConstraint, tilingSchedule, ctxt, operatorRepresentation) - ctxt, egressDMAUpdates = self._generateEgressPointerUpdates(nodeMemoryConstraint, tilingSchedule, ctxt, operatorRepresentation) + ctxt, egressDMAUpdates = self._generateEgressPointerUpdates(nodeMemoryConstraint, tilingSchedule, ctxt, + operatorRepresentation) openLoopStatement = [ CodeSnippet(self._openTileLoopTemplate, { @@ -459,7 +472,8 @@ def _tilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, }) ] - variableUpdates = self._generateVariableUpdates(tilingSchedule, variableReplacement, ctxt, operatorRepresentation) + variableUpdates = self._generateVariableUpdates(tilingSchedule, variableReplacement, ctxt, + operatorRepresentation) metaInfo = TilingMetaInfo(nodeName = operatorRepresentation['nodeName'] + "_L2", nodeOps = operatorRepresentation['nodeOps'], @@ -474,10 +488,10 @@ def _tilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, return ctxt, newExecutionBlock, True - def generateTilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, - nodeMemoryConstraint: NodeMemoryConstraint, tilingSchedules: List[TilingSchedule], - variableReplacement: VariableReplacementScheme, - operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, ExecutionBlock, bool]: + def generateTilingLoop( + self, ctxt: NetworkContext, executionBlock: ExecutionBlock, nodeMemoryConstraint: NodeMemoryConstraint, + tilingSchedules: List[TilingSchedule], variableReplacement: VariableReplacementScheme, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, ExecutionBlock, bool]: flatTilingSchedule = copy.copy(tilingSchedules[0]) for tilingSchedule in tilingSchedules[1:]: @@ -494,7 +508,8 @@ def generateTilingLoop(self, ctxt: NetworkContext, executionBlock: ExecutionBloc if not len(offsetList) == 1: return ctxt, executionBlock, False - operatorRepresentation["numTiles"] = self._hoistNumTiles(ctxt, operatorRepresentation['nodeName'], tilingSchedules) + operatorRepresentation["numTiles"] = self._hoistNumTiles(ctxt, operatorRepresentation['nodeName'], + tilingSchedules) return self._tilingLoop(ctxt, executionBlock, nodeMemoryConstraint, flatTilingSchedule, variableReplacement, operatorRepresentation) diff --git a/Deeploy/Targets/Snitch/Deployer.py b/Deeploy/Targets/Snitch/Deployer.py index 870a984..ff32066 100644 --- a/Deeploy/Targets/Snitch/Deployer.py +++ b/Deeploy/Targets/Snitch/Deployer.py @@ -29,12 +29,13 @@ import onnx_graphsurgeon as gs from Deeploy.AbstractDataTypes import Pointer -from Deeploy.DeeployTypes import DeploymentPlatform, TopologyOptimizer from Deeploy.CommonExtensions.NetworkDeployers.SignPropDeployer import SignPropDeployer -from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import ReshapeConstOptPass, TransposeConstOptPass, \ - TransposeMergePass, TransposeSplitPass from Deeploy.CommonExtensions.OptimizationPasses.TopologyOptimizationPasses.LoweringOptimizationPasses import \ NCHWtoNHWCPass, RemoveGlobalOutputReshapePass, TransposeMatmulInputsPass +from Deeploy.DeeployTypes import DeploymentPlatform, TopologyOptimizer +from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import ReshapeConstOptPass, TransposeConstOptPass, \ + TransposeMergePass, TransposeSplitPass + class SnitchDeployer(SignPropDeployer): diff --git a/Deeploy/Targets/Snitch/Platform.py b/Deeploy/Targets/Snitch/Platform.py index 06321d7..aa2893a 100644 --- a/Deeploy/Targets/Snitch/Platform.py +++ b/Deeploy/Targets/Snitch/Platform.py @@ -28,28 +28,25 @@ import numpy as np - -from Deeploy.Targets.Generic.Bindings import BasicGatherBindings,BasicMatMulBinding, BasicPad1DBindings, \ - BasicPad2DBindings, BasicReshapeBindings, BasicRQIntegerDivBinding, BasicLayerNormBinding from Deeploy.DeeployTypes import ConstantBuffer, DeploymentEngine, DeploymentPlatform, NodeMapper, NodeTemplate, \ StructBuffer, TopologyOptimizer, TransientBuffer, VariableBuffer +from Deeploy.Targets.Generic.Bindings import BasicGatherBindings, BasicLayerNormBinding, BasicMatMulBinding, \ + BasicPad1DBindings, BasicPad2DBindings, BasicReshapeBindings, BasicRQIntegerDivBinding from Deeploy.Targets.Generic.Layers import AddLayer, GatherLayer, GEMMLayer, MatMulLayer, PadLayer, ReshapeLayer, \ - RQGEMMLayer, RQIntegerDivLayer, iNoNormLayer, iSoftmaxLayer, iLayerNormLayer -from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import AddRequantMergePass, \ - GEMMRequantMergePass, IntegerDivRequantMergePass, MergeConstAddAndRequantPass, \ - MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, SkipEmptyConcatPass, SkipUnityRequantPass, \ - iGELURequantMergePass, iHardswishRequantMergePass -from Deeploy.Targets.Generic.Parsers import AddParser, GatherParser, MatMulParser, Pad1DParser, Pad2DParser, RQAddParser, \ - RQIntegerDivParser, UnsqueezeParser, iNoNormParser, iSoftmaxParser, iLayerNormParser -from Deeploy.Targets.Snitch.Parser import SnitchGEMMParser, SnitchRQGEMMParser -from Deeploy.Targets.PULPOpen.Platform import RQAddMapper + RQGEMMLayer, RQIntegerDivLayer, iLayerNormLayer, iNoNormLayer, iSoftmaxLayer +from Deeploy.Targets.Generic.Parsers import AddParser, GatherParser, MatMulParser, Pad1DParser, Pad2DParser, \ + RQAddParser, RQIntegerDivParser, UnsqueezeParser, iLayerNormParser, iNoNormParser, iSoftmaxParser from Deeploy.Targets.Generic.Templates import AllocateTemplate as BasicAllocateTemplate +from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import AddRequantMergePass, GEMMRequantMergePass, \ + IntegerDivRequantMergePass, MergeConstAddAndRequantPass, MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, \ + SkipEmptyConcatPass, SkipUnityRequantPass, iGELURequantMergePass, iHardswishRequantMergePass +from Deeploy.Targets.PULPOpen.Platform import RQAddMapper +from Deeploy.Targets.Snitch.Parser import SnitchGEMMParser, SnitchRQGEMMParser from Deeploy.Targets.Snitch.Templates import AllocateTemplate, FreeTemplate from Deeploy.Targets.Snitch.Tiler import SnitchAddTileReadyBindings, SnitchGemmTilingReadyBindings, \ SnitchiNoNormTilingReadyBindings, SnitchiSoftmaxTilingReadyBindings, SnitchRQAddTilingReadyBindings, \ SnitchRqGemmTilingReadyBindings - GatherMapper = NodeMapper(GatherParser(), BasicGatherBindings) Pad1DMapper = NodeMapper(Pad1DParser(), BasicPad1DBindings) Pad2DMapper = NodeMapper(Pad2DParser(), BasicPad2DBindings) diff --git a/Deeploy/Targets/Snitch/Templates/AddTemplate.py b/Deeploy/Targets/Snitch/Templates/AddTemplate.py index 8b95896..f604625 100644 --- a/Deeploy/Targets/Snitch/Templates/AddTemplate.py +++ b/Deeploy/Targets/Snitch/Templates/AddTemplate.py @@ -30,7 +30,8 @@ class _SnitchAddTemplate(NodeTemplate): - def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: data_in_1 = ctxt.lookup(operatorRepresentation['data_in_1']) data_in_2 = ctxt.lookup(operatorRepresentation['data_in_2']) diff --git a/Deeploy/Targets/Snitch/Templates/GemmTemplate.py b/Deeploy/Targets/Snitch/Templates/GemmTemplate.py index 16e2c6e..8bc0fee 100644 --- a/Deeploy/Targets/Snitch/Templates/GemmTemplate.py +++ b/Deeploy/Targets/Snitch/Templates/GemmTemplate.py @@ -5,12 +5,15 @@ class SnitchGemmTemplate(NodeTemplate): - def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: if isinstance(operatorRepresentation['alpha'], float): - assert operatorRepresentation['alpha'].is_integer(), f"Parameter alpha is not an integer: {operatorRepresentation['alpha']}" + assert operatorRepresentation['alpha'].is_integer( + ), f"Parameter alpha is not an integer: {operatorRepresentation['alpha']}" operatorRepresentation['alpha'] = int(operatorRepresentation['alpha']) if isinstance(operatorRepresentation['beta'], float): - assert operatorRepresentation['beta'].is_integer(), f"Parameter beta is not an integer: {operatorRepresentation['beta']}" + assert operatorRepresentation['beta'].is_integer( + ), f"Parameter beta is not an integer: {operatorRepresentation['beta']}" operatorRepresentation['beta'] = int(operatorRepresentation['beta']) if operatorRepresentation['transB']: diff --git a/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py index 0582818..2a49a3b 100644 --- a/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py +++ b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py @@ -35,7 +35,8 @@ class SnitchRQAddTemplate(NodeTemplate): def __init__(self, templateStr): super().__init__(templateStr) - def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: # Extract signedness information of input, weights and output signedI2 = ctxt.lookup(operatorRepresentation['data_in_2'])._type.referencedType.typeMin < 0 signedI = ctxt.lookup(operatorRepresentation['data_in_1'])._type.referencedType.typeMin < 0 diff --git a/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py b/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py index a51d31d..918690e 100644 --- a/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py +++ b/Deeploy/Targets/Snitch/Templates/RqGemmTemplate.py @@ -5,7 +5,8 @@ class SnitchRqGemmTemplate(NodeTemplate): - def alignToContext(self, ctxt: NetworkContext, operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: if isinstance(operatorRepresentation['alpha'], float): assert operatorRepresentation['alpha'].is_integer() operatorRepresentation['alpha'] = int(operatorRepresentation['alpha']) diff --git a/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py index b95be32..99fdddd 100644 --- a/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py +++ b/Deeploy/Targets/Snitch/TileConstraints/GemmTileConstraint.py @@ -9,6 +9,7 @@ from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, HyperRectangle, TilingSchedule, \ VariableReplacementScheme + class GemmTileConstraint(TileConstraint): @staticmethod @@ -95,14 +96,15 @@ def addPolicyConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkCo return tilerModel @classmethod - def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, - absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + def serializeTilingSolution( + cls, tilingSolution: NodeMemoryConstraint, absoluteOutputCubes: List[AbsoluteHyperRectangle], + targetMemLevel: str, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: outputCubes = [cube.rectangle for cube in absoluteOutputCubes] addrNames = ['A', 'B', 'C', 'data_out'] - inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, + operatorRepresentation, addrNames) NOffset = 0 NSize = operatorRepresentation["N"] diff --git a/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py index 517393e..5feae3b 100644 --- a/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py +++ b/Deeploy/Targets/Snitch/TileConstraints/RqGemmTileConstraint.py @@ -5,8 +5,10 @@ from Deeploy.DeeployTypes import NetworkContext, OperatorRepresentation from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint from Deeploy.TilingExtension.TileConstraint import TileConstraint -from Deeploy.TilingExtension.TilerModel import TilerModel, PerformanceHint -from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, HyperRectangle, TilingSchedule, VariableReplacementScheme +from Deeploy.TilingExtension.TilerModel import PerformanceHint, TilerModel +from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, HyperRectangle, TilingSchedule, \ + VariableReplacementScheme + class RqGemmTileConstraint(TileConstraint): @@ -107,14 +109,15 @@ def addPolicyConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: NetworkCo return tilerModel @classmethod - def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, - absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + def serializeTilingSolution( + cls, tilingSolution: NodeMemoryConstraint, absoluteOutputCubes: List[AbsoluteHyperRectangle], + targetMemLevel: str, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: outputCubes = [cube.rectangle for cube in absoluteOutputCubes] addrNames = ['A', 'B', 'C', 'mul', 'add', 'data_out'] - inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, + operatorRepresentation, addrNames) NOffset = 0 NSize = operatorRepresentation["N"] diff --git a/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py index 8819669..ab7b0cf 100644 --- a/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py +++ b/Deeploy/Targets/Snitch/TileConstraints/iNoNormTileConstraint.py @@ -35,6 +35,7 @@ from Deeploy.TilingExtension.TilerModel import TilerModel from Deeploy.TilingExtension.TilingCodegen import AbsoluteHyperRectangle, TilingSchedule, VariableReplacementScheme + class iNoNormTileConstraint(TileConstraint): @staticmethod @@ -70,14 +71,15 @@ def addGeometricalConstraint(tilerModel: TilerModel, parseDict: Dict, ctxt: Netw return tilerModel @classmethod - def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, - absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + def serializeTilingSolution( + cls, tilingSolution: NodeMemoryConstraint, absoluteOutputCubes: List[AbsoluteHyperRectangle], + targetMemLevel: str, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: outputCubes = [cube.rectangle for cube in absoluteOutputCubes] addrNames = ['data_in', 'weights', 'bias', 'data_out'] - inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, + operatorRepresentation, addrNames) replacements = {"size": []} replacementTypes = {"size": PointerClass(uint32_t)} diff --git a/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py b/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py index e8cdc49..5528491 100644 --- a/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py +++ b/Deeploy/Targets/Snitch/TileConstraints/iSoftmaxTileConstraint.py @@ -85,14 +85,15 @@ def constructSymbolicNodeRep(tilerModel: TilerModel, parseDict: Dict, return symbolicParseDict @classmethod - def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, - absoluteOutputCubes: List[AbsoluteHyperRectangle], targetMemLevel: str, - ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: + def serializeTilingSolution( + cls, tilingSolution: NodeMemoryConstraint, absoluteOutputCubes: List[AbsoluteHyperRectangle], + targetMemLevel: str, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[VariableReplacementScheme, TilingSchedule]: outputCubes = [cube.rectangle for cube in absoluteOutputCubes] addrNames = ['data_in', 'data_out'] - inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, operatorRepresentation, addrNames) + inputBaseOffsets, outputBaseOffsets = cls.extractBaseAddr(tilingSolution, targetMemLevel, + operatorRepresentation, addrNames) replacements = {"lastDimLength": [], "size": []} @@ -115,4 +116,4 @@ def serializeTilingSolution(cls, tilingSolution: NodeMemoryConstraint, tilingSchedule = TilingSchedule(inputBaseOffsets, outputBaseOffsets, inputLoadSchedule, outputLoadSchedule) variableReplacementSchedule = VariableReplacementScheme(replacements, replacementTypes) - return variableReplacementSchedule, tilingSchedule \ No newline at end of file + return variableReplacementSchedule, tilingSchedule diff --git a/Deeploy/Targets/Snitch/Tiler.py b/Deeploy/Targets/Snitch/Tiler.py index b64748d..38ba29f 100644 --- a/Deeploy/Targets/Snitch/Tiler.py +++ b/Deeploy/Targets/Snitch/Tiler.py @@ -23,13 +23,12 @@ # See the License for the specific language governing permissions and # limitations under the License. +from Deeploy.Targets.Generic.TileConstraints.AddTileConstraint import AddTileConstraint from Deeploy.Targets.Snitch.Bindings import SnitchAddBindings, SnitchGemmBindings, SnitchiNoNormBindings, \ SnitchiSoftmaxBindings, SnitchRQAddBindings, SnitchRqGemmBindings -from Deeploy.Targets.Generic.TileConstraints.AddTileConstraint import AddTileConstraint from Deeploy.Targets.Snitch.TileConstraints import iNoNormTileConstraint, iSoftmaxTileConstraint from Deeploy.Targets.Snitch.TileConstraints.GemmTileConstraint import GemmTileConstraint from Deeploy.Targets.Snitch.TileConstraints.RqGemmTileConstraint import RqGemmTileConstraint - from Deeploy.TilingExtension.TilerExtension import TilingReadyNodeBindings SnitchiSoftmaxTilingReadyBindings = TilingReadyNodeBindings(nodeBindings = SnitchiSoftmaxBindings, diff --git a/Deeploy/Targets/Snitch/TypeCheckers.py b/Deeploy/Targets/Snitch/TypeCheckers.py index f978b78..fa428af 100644 --- a/Deeploy/Targets/Snitch/TypeCheckers.py +++ b/Deeploy/Targets/Snitch/TypeCheckers.py @@ -26,18 +26,21 @@ from typing import List, Sequence, Type from Deeploy.AbstractDataTypes import Pointer -from Deeploy.DeeployTypes import VariableBuffer, OperatorRepresentation from Deeploy.CommonExtensions.TypeCheckers.SignPropTypeChecker import SignPropTypeChecker +from Deeploy.DeeployTypes import OperatorRepresentation, VariableBuffer + class SnitchRQAddChecker(SignPropTypeChecker): def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): super().__init__(input_types, output_types) - def _inferNumLevels(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[int]: + def _inferNumLevels(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[int]: return [operatorRepresentation['rqsOut_n_levels']] - def _inferSignedness(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> List[bool]: + def _inferSignedness(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[bool]: return [bool(operatorRepresentation["rqsOut_signed"])] # Override this. This should compute the signednes of each output node of the Layer diff --git a/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py b/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py index a99c530..f4e4d9a 100644 --- a/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py +++ b/Deeploy/TilingExtension/CodeTransformationPasses/TilingCodeGeneration.py @@ -32,8 +32,8 @@ from Deeploy.CommonExtensions.CodeTransformationPasses.IntrospectiveCodeTransformation import \ IntrospectiveCodeTransformationMixIn from Deeploy.CommonExtensions.CodeTransformationPasses.MemoryAllocation import ArgumentStructGeneration -from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ConstantBuffer, ExecutionBlock, NetworkContext, \ - NodeTemplate, OperatorRepresentation, VariableBuffer, _NoVerbosity +from Deeploy.DeeployTypes import CodeGenVerbosity, CodeTransformationPass, ConstantBuffer, ExecutionBlock, \ + NetworkContext, NodeTemplate, OperatorRepresentation, VariableBuffer, _NoVerbosity from Deeploy.TilingExtension.CodeTransformationPasses.TilingPrototypes import PrototypeTilingMixIn from Deeploy.TilingExtension.MemoryConstraints import NodeMemoryConstraint from Deeploy.TilingExtension.TilingCodegen import TilingSchedule, VariableReplacementScheme, minimizeVariableReplacement @@ -150,7 +150,6 @@ def _hoistConstantAndReference(self, return ctxt, operatorRepresentation - def apply(self, ctxt: NetworkContext, executionBlock: ExecutionBlock, diff --git a/DeeployTest/Platforms/Snitch/main.c b/DeeployTest/Platforms/Snitch/main.c index 232081a..38cfa96 100644 --- a/DeeployTest/Platforms/Snitch/main.c +++ b/DeeployTest/Platforms/Snitch/main.c @@ -66,7 +66,7 @@ int main(void) { DeeployNetwork_inputs[buf], DeeployNetwork_inputs_bytes[buf]); } for (uint32_t buf = 0; buf < DeeployNetwork_num_outputs; buf++) { - printf("testInputVector%d @ %p\r\n", buf, testOutputVector[buf]); + printf("testOutputVector%d @ %p\r\n", buf, testOutputVector[buf]); printf("DeeployNetwork_output_%d @ %p and %u elements\r\n", buf, DeeployNetwork_outputs[buf], DeeployNetwork_outputs_bytes[buf]); } diff --git a/TargetLibraries/Snitch/inc/kernel/Gemm.h b/TargetLibraries/Snitch/inc/kernel/Gemm.h index e845e15..b1197b7 100644 --- a/TargetLibraries/Snitch/inc/kernel/Gemm.h +++ b/TargetLibraries/Snitch/inc/kernel/Gemm.h @@ -58,8 +58,10 @@ * parallelization = row-wise * bias pushing = no */ -void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, +void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, uint32_t O, int32_t alpha, int32_t beta); /* @@ -72,9 +74,12 @@ void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__rest * parallelization = row-wise * bias pushing = no */ -void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, - uint32_t O, int32_t alpha, int32_t beta); +void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int32_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta); /* * General Matrix Multiplication ---------------------------------- * kernel = Gemm_parallel_s8_rv32im diff --git a/TargetLibraries/Snitch/inc/kernel/RQGemm.h b/TargetLibraries/Snitch/inc/kernel/RQGemm.h index 0fe5cbf..b1d77c1 100644 --- a/TargetLibraries/Snitch/inc/kernel/RQGemm.h +++ b/TargetLibraries/Snitch/inc/kernel/RQGemm.h @@ -58,9 +58,12 @@ * parallelization = row-wise * bias pushing = no */ -void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, - uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, int32_t log2D); +void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta, + int32_t *mul, int32_t *add, int32_t log2D); /* * General Requantized Matrix Multiplication ---------------------------------- @@ -72,9 +75,12 @@ void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__re * parallelization = row-wise * bias pushing = no */ -void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, +void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta, int32_t *mul, int32_t *add, int32_t log2D); /* @@ -87,9 +93,12 @@ void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t co * parallelization = row-wise * bias pushing = no */ -void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, +void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta, int32_t *mul, int32_t *add, int32_t log2D); /* @@ -102,10 +111,11 @@ void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t cons * parallelization = row-wise * bias pushing = no */ -void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, - int32_t *add, int32_t log2D); +void RQGemm_s8_transB_row_parallel_unrolled( + int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, + int32_t *add, int32_t log2D); /* * General Requantized Matrix Multiplication ---------------------------------- diff --git a/TargetLibraries/Snitch/inc/kernel/iNoNorm.h b/TargetLibraries/Snitch/inc/kernel/iNoNorm.h index 70ec15f..56b58e0 100644 --- a/TargetLibraries/Snitch/inc/kernel/iNoNorm.h +++ b/TargetLibraries/Snitch/inc/kernel/iNoNorm.h @@ -26,5 +26,6 @@ #include "DeeploySnitchMath.h" -void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int32_t *bias, uint32_t size, int32_t mul, +void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, + int32_t *bias, uint32_t size, int32_t mul, int32_t log2D); diff --git a/TargetLibraries/Snitch/inc/macros.h b/TargetLibraries/Snitch/inc/macros.h index 2756d83..a54bc24 100644 --- a/TargetLibraries/Snitch/inc/macros.h +++ b/TargetLibraries/Snitch/inc/macros.h @@ -30,11 +30,13 @@ #define __DEEPLOY_MATH_MACROS_HEADER_ #define INT_LOG2(x) __builtin_ctz(x) -#define CLAMP(x, low, high) (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x))) +#define CLAMP(x, low, high) \ + (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x))) #define MIN(a, b) ((a) < (b) ? (a) : (b)) #define MAX(a, b) ((a) > (b) ? (a) : (b)) -// JUNGVI: The following macros are here to ensure compatibility with some PULP-NN kernels +// JUNGVI: The following macros are here to ensure compatibility with some +// PULP-NN kernels #define clips8(x) CLAMP(x, -128, 127) #endif //__DEEPLOY_MATH_MACROS_HEADER_ diff --git a/TargetLibraries/Snitch/src/Add.c b/TargetLibraries/Snitch/src/Add.c index 2b62efb..094739c 100644 --- a/TargetLibraries/Snitch/src/Add.c +++ b/TargetLibraries/Snitch/src/Add.c @@ -26,7 +26,8 @@ #include "DeeploySnitchMath.h" -void SnitchAdd(int8_t *pIn1, int8_t *pIn2, int32_t *pOut, uint32_t size, int32_t offset) { +void SnitchAdd(int8_t *pIn1, int8_t *pIn2, int32_t *pOut, uint32_t size, + int32_t offset) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); diff --git a/TargetLibraries/Snitch/src/Gemm_s8.c b/TargetLibraries/Snitch/src/Gemm_s8.c index 63d7914..eefd407 100644 --- a/TargetLibraries/Snitch/src/Gemm_s8.c +++ b/TargetLibraries/Snitch/src/Gemm_s8.c @@ -1,8 +1,10 @@ #include "DeeploySnitchMath.h" #include "Gemm.h" -void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, +void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, uint32_t O, int32_t alpha, int32_t beta) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -11,7 +13,8 @@ void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__rest uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; for (uint32_t m = MStart; m < MEnd; m++) { @@ -27,9 +30,12 @@ void Gemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__rest } } -void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int32_t *__restrict__ pDstY, uint32_t M, uint32_t N, - uint32_t O, int32_t alpha, int32_t beta) { +void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int32_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -37,7 +43,8 @@ void Gemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; for (uint32_t m = MStart; m < MEnd; m++) { diff --git a/TargetLibraries/Snitch/src/RQGemm_s8.c b/TargetLibraries/Snitch/src/RQGemm_s8.c index fc2576a..cfbc867 100644 --- a/TargetLibraries/Snitch/src/RQGemm_s8.c +++ b/TargetLibraries/Snitch/src/RQGemm_s8.c @@ -33,9 +33,12 @@ // Assumptions: // - per-row requantization // - single batch -void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, - uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, int32_t log2D) { +void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, + uint32_t O, int32_t alpha, int32_t beta, + int32_t *mul, int32_t *add, int32_t log2D) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -43,7 +46,8 @@ void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__re uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; if (core_id < numThreads) { @@ -67,9 +71,12 @@ void RQGemm_s8_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__re // - per-row requantization // - transposed input B // - single batch -void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, +void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta, int32_t *mul, int32_t *add, int32_t log2D) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -78,7 +85,8 @@ void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t co uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; if (core_id < numThreads) { @@ -173,9 +181,12 @@ void RQGemm_s8_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t co // - per-row requantization // - transposed input B // - single batch -void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, int32_t *add, +void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, + int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, + int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, + int32_t beta, int32_t *mul, int32_t *add, int32_t log2D) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -184,7 +195,8 @@ void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t cons uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; if (core_id < numThreads) { @@ -208,10 +220,11 @@ void RQGemm_s8_transB_row_parallel(int8_t const *__restrict__ pSrcA, int8_t cons // - per-row requantization // - transposed input B // - single batch -void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, - uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, - int32_t *add, int32_t log2D) { +void RQGemm_s8_transB_row_parallel_unrolled( + int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t O, int32_t alpha, int32_t beta, int32_t *mul, + int32_t *add, int32_t log2D) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); @@ -219,7 +232,8 @@ void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, in uint32_t const MQuotient = M / numThreads; uint32_t const MRemainder = M % numThreads; uint32_t const MSize = MQuotient + (core_id < MRemainder ? 1 : 0); - uint32_t const MStart = core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); + uint32_t const MStart = + core_id * MQuotient + (core_id < MRemainder ? core_id : MRemainder); uint32_t const MEnd = MStart + MSize; if (core_id < numThreads) { @@ -310,12 +324,13 @@ void RQGemm_s8_transB_row_parallel_unrolled(int8_t const *__restrict__ pSrcA, in } } -void RQGemm_parallel_s8_rv32im(int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, - int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, uint32_t N, - uint32_t P, int32_t alpha, int32_t beta, int32_t transA, int32_t transB, int32_t *mul, - int32_t *add, int32_t log2D, bool rounding, bool per_row_quant, int32_t A_offset, - int32_t B_offset, int32_t C_offset, int32_t Y_offset, int8_t output_min, - int8_t output_max) { +void RQGemm_parallel_s8_rv32im( + int8_t const *__restrict__ pSrcA, int8_t const *__restrict__ pSrcB, + int32_t const *__restrict__ pSrcC, int8_t *__restrict__ pDstY, uint32_t M, + uint32_t N, uint32_t P, int32_t alpha, int32_t beta, int32_t transA, + int32_t transB, int32_t *mul, int32_t *add, int32_t log2D, bool rounding, + bool per_row_quant, int32_t A_offset, int32_t B_offset, int32_t C_offset, + int32_t Y_offset, int8_t output_min, int8_t output_max) { uint32_t core_id = snrt_global_compute_core_idx(); uint32_t numThreads = snrt_global_compute_core_num(); diff --git a/TargetLibraries/Snitch/src/iNoNorm.c b/TargetLibraries/Snitch/src/iNoNorm.c index f18f19b..30b3c68 100644 --- a/TargetLibraries/Snitch/src/iNoNorm.c +++ b/TargetLibraries/Snitch/src/iNoNorm.c @@ -26,7 +26,8 @@ #include "DeeploySnitchMath.h" -void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int32_t *bias, uint32_t size, int32_t mul, +void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, + int32_t *bias, uint32_t size, int32_t mul, int32_t log2D) { uint32_t core_id = snrt_global_compute_core_idx(); @@ -61,8 +62,10 @@ void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int uint32_t biasIdx = start * 4; // JUNGVI: Compute sequentially the first elements not aligned to a word (32b) - for (uint32_t i = firstReminderLoopIdx; i < firstReminderLoopIdx + firstReminderLoopSize; i++) { - data_out[i] = ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; + for (uint32_t i = firstReminderLoopIdx; + i < firstReminderLoopIdx + firstReminderLoopSize; i++) { + data_out[i] = + ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; } for (uint32_t i = start; i < stop; i++) { @@ -90,13 +93,16 @@ void SnitchiNoNorm_s8_s8(int8_t *data_in, int8_t *data_out, int8_t *weights, int uint8_t outBuf3 = ((partialProduct3 + bias[biasIdx + 2]) * mul) >> log2D; uint8_t outBuf4 = ((partialProduct4 + bias[biasIdx + 3]) * mul) >> log2D; - uint32_t outPacked = (outBuf1 << 0) | (outBuf2 << 8) | (outBuf3 << 16) | (outBuf4 << 24); + uint32_t outPacked = + (outBuf1 << 0) | (outBuf2 << 8) | (outBuf3 << 16) | (outBuf4 << 24); outputPtr[i] = outPacked; biasIdx += 4; } // JUNGVI: Compute sequentially the last elements not aligned to a word (32b) - for (uint32_t i = lastReminderLoopIdx; i < lastReminderLoopIdx + lastReminderLoopSize; i++) { - data_out[i] = ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; + for (uint32_t i = lastReminderLoopIdx; + i < lastReminderLoopIdx + lastReminderLoopSize; i++) { + data_out[i] = + ((((int32_t)data_in[i] * weights[i]) + bias[i]) * mul) >> log2D; } } From 3acf9341f972d41bb8afd93b39c26bda66d932d1 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 02:41:13 +0100 Subject: [PATCH 03/11] fix typo --- .github/workflows/TestRunnerTiledSnitch.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/TestRunnerTiledSnitch.yml b/.github/workflows/TestRunnerTiledSnitch.yml index 82f16be..10f3b26 100644 --- a/.github/workflows/TestRunnerTiledSnitch.yml +++ b/.github/workflows/TestRunnerTiledSnitch.yml @@ -1,4 +1,4 @@ -name: TestRunnerTiledSiracusaSequential +name: TestRunnerTiledSnitch on: workflow_call: From bc2c6f88474dc2bf4b14cbc56cc0e64172b9e2b8 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 03:44:27 +0100 Subject: [PATCH 04/11] change snitch patch --- toolchain/snitch_cluster.patch | 25 +++++++++++++++++++------ 1 file changed, 19 insertions(+), 6 deletions(-) diff --git a/toolchain/snitch_cluster.patch b/toolchain/snitch_cluster.patch index c39c525..b50b33c 100644 --- a/toolchain/snitch_cluster.patch +++ b/toolchain/snitch_cluster.patch @@ -15,21 +15,34 @@ index d0979b7..171921d 100644 + KEEP(*(.cbss .cbss.*)) __cbss_end = .; } >L3 - + +diff --git a/sw/snRuntime/src/alloc.h b/sw/snRuntime/src/alloc.h +index ba1dee9..f79769f 100644 +--- a/sw/snRuntime/src/alloc.h ++++ b/sw/snRuntime/src/alloc.h +@@ -69,6 +69,8 @@ inline void *snrt_l3alloc(size_t size) { + + // TODO: L3 alloc size check + ++ size = ALIGN_UP(size, MIN_CHUNK_SIZE); ++ + void *ret = (void *)alloc->next; + alloc->next += size; + return ret; diff --git a/sw/snRuntime/src/team.c b/sw/snRuntime/src/team.c index a9eb840..5290e1d 100644 --- a/sw/snRuntime/src/team.c +++ b/sw/snRuntime/src/team.c @@ -10,6 +10,10 @@ extern uint32_t snrt_global_core_idx(); - + extern uint32_t snrt_global_core_num(); - + +extern uint32_t snrt_global_compute_core_num(); + +extern uint32_t snrt_global_compute_core_idx(); + extern uint32_t snrt_cluster_idx(); - + extern uint32_t snrt_cluster_num(); diff --git a/target/snitch_cluster/sw/runtime/rtl/src/putchar.c b/target/snitch_cluster/sw/runtime/rtl/src/putchar.c index 0ad9500..215c8b1 100644 @@ -37,7 +50,7 @@ index 0ad9500..215c8b1 100644 +++ b/target/snitch_cluster/sw/runtime/rtl/src/putchar.c @@ -5,16 +5,19 @@ extern uintptr_t volatile tohost, fromhost; - + // Rudimentary string buffer for putc calls. -extern uint32_t _edram; #define PUTC_BUFFER_LEN (1024 - sizeof(size_t)) @@ -58,6 +71,6 @@ index 0ad9500..215c8b1 100644 +} putc_buffer_t; + +static volatile putc_buffer_t putc_buffer[SNRT_CLUSTER_NUM*SNRT_CLUSTER_CORE_NUM] __attribute__((section(".dram"))); - + // Provide an implementation for putchar. void _putchar(char character) { From 45cd6ec60b0c09ec7bb18cd783318a2064487882 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 17:13:36 +0100 Subject: [PATCH 05/11] add victor's comment --- .github/workflows/BuildDocker.yml | 2 +- .github/workflows/CI.yml | 16 ++++++++-------- .github/workflows/TestRunnerCortexM.yml | 2 +- .github/workflows/TestRunnerGeneric.yml | 2 +- .github/workflows/TestRunnerMempool.yml | 2 +- .github/workflows/TestRunnerSiracusa.yml | 2 +- .github/workflows/TestRunnerSnitch.yml | 2 +- .github/workflows/TestRunnerTiledSiracusa.yml | 2 +- .../TestRunnerTiledSiracusaSequential.yml | 2 +- .../TestRunnerTiledSiracusaWithNeureka.yml | 2 +- ...tRunnerTiledSiracusaWithNeurekaSequential.yml | 2 +- .github/workflows/TestRunnerTiledSnitch.yml | 2 +- 12 files changed, 19 insertions(+), 19 deletions(-) diff --git a/.github/workflows/BuildDocker.yml b/.github/workflows/BuildDocker.yml index d07e90a..47f631e 100644 --- a/.github/workflows/BuildDocker.yml +++ b/.github/workflows/BuildDocker.yml @@ -38,4 +38,4 @@ jobs: file: Container/Dockerfile push: true # JUNGVI: If you operate from a fork and want to build a new docker make sure to replace 'pulp-platform' by your uname. - tags: ghcr.io/tahaelbayad/deeploy:main + tags: ghcr.io/pulp-platform/deeploy:main diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index aae30f5..2cad974 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -13,7 +13,7 @@ jobs: build-deeploy: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -542,7 +542,7 @@ jobs: deeploy-state-serialization: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -562,7 +562,7 @@ jobs: deeploy-memory-level-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -582,7 +582,7 @@ jobs: deeploy-tiler-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -606,7 +606,7 @@ jobs: deeploy-memory-allocation-extension: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -628,7 +628,7 @@ jobs: deeploy-typing: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -645,7 +645,7 @@ jobs: deeploy-regex-matching: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 @@ -662,7 +662,7 @@ jobs: linting: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerCortexM.yml b/.github/workflows/TestRunnerCortexM.yml index 9d80f55..5f0e523 100644 --- a/.github/workflows/TestRunnerCortexM.yml +++ b/.github/workflows/TestRunnerCortexM.yml @@ -11,7 +11,7 @@ jobs: test-runner-cortexm: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerGeneric.yml b/.github/workflows/TestRunnerGeneric.yml index c123ca5..ce20fda 100644 --- a/.github/workflows/TestRunnerGeneric.yml +++ b/.github/workflows/TestRunnerGeneric.yml @@ -11,7 +11,7 @@ jobs: test-runner-generic: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerMempool.yml b/.github/workflows/TestRunnerMempool.yml index a179667..e8414ff 100644 --- a/.github/workflows/TestRunnerMempool.yml +++ b/.github/workflows/TestRunnerMempool.yml @@ -11,7 +11,7 @@ jobs: test-runner-mempool: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerSiracusa.yml b/.github/workflows/TestRunnerSiracusa.yml index 0143396..0b8b8cc 100644 --- a/.github/workflows/TestRunnerSiracusa.yml +++ b/.github/workflows/TestRunnerSiracusa.yml @@ -14,7 +14,7 @@ jobs: test-runner-siracusa: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerSnitch.yml b/.github/workflows/TestRunnerSnitch.yml index d2c35ef..3a0f9a0 100644 --- a/.github/workflows/TestRunnerSnitch.yml +++ b/.github/workflows/TestRunnerSnitch.yml @@ -11,7 +11,7 @@ jobs: test-runner-snitch: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusa.yml b/.github/workflows/TestRunnerTiledSiracusa.yml index 9f101c6..48a8cb3 100644 --- a/.github/workflows/TestRunnerTiledSiracusa.yml +++ b/.github/workflows/TestRunnerTiledSiracusa.yml @@ -32,7 +32,7 @@ jobs: L1: ${{ fromJSON(inputs.L1) }} runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaSequential.yml b/.github/workflows/TestRunnerTiledSiracusaSequential.yml index 5e9e275..684036a 100644 --- a/.github/workflows/TestRunnerTiledSiracusaSequential.yml +++ b/.github/workflows/TestRunnerTiledSiracusaSequential.yml @@ -24,7 +24,7 @@ jobs: test-runner-siracusa-tiled: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml b/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml index d16699f..143c989 100644 --- a/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml +++ b/.github/workflows/TestRunnerTiledSiracusaWithNeureka.yml @@ -36,7 +36,7 @@ jobs: L1: ${{ fromJSON(inputs.L1) }} runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml b/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml index c362d86..3271598 100644 --- a/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml +++ b/.github/workflows/TestRunnerTiledSiracusaWithNeurekaSequential.yml @@ -28,7 +28,7 @@ jobs: test-runner-siracusa-neureka-tiled: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 diff --git a/.github/workflows/TestRunnerTiledSnitch.yml b/.github/workflows/TestRunnerTiledSnitch.yml index 10f3b26..c109958 100644 --- a/.github/workflows/TestRunnerTiledSnitch.yml +++ b/.github/workflows/TestRunnerTiledSnitch.yml @@ -12,7 +12,7 @@ jobs: test-runner-snitch-tiled: runs-on: ubuntu-22.04 container: - image: ghcr.io/tahaelbayad/deeploy:main + image: ghcr.io/pulp-platform/deeploy:main steps: - name: Checkout Repo uses: actions/checkout@v4 From 44ed707c73e633042e102530b3651ab889efc2f4 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 19:22:34 +0100 Subject: [PATCH 06/11] add victor's comment --- .github/workflows/CI.yml | 2 +- ....yml => TestRunnerTiledSnitchSequential.yml} | 17 +++++++++++++++-- 2 files changed, 16 insertions(+), 3 deletions(-) rename .github/workflows/{TestRunnerTiledSnitch.yml => TestRunnerTiledSnitchSequential.yml} (71%) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 2cad974..43380af 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -105,7 +105,7 @@ jobs: testRQGEMMTransB snitch-kernels-tiled-singlebuffer-L2: - uses: ./.github/workflows/TestRunnerTiledSnitch.yml + uses: ./.github/workflows/TestRunnerTiledSnitchSequential.yml with: tests-config: | [ diff --git a/.github/workflows/TestRunnerTiledSnitch.yml b/.github/workflows/TestRunnerTiledSnitchSequential.yml similarity index 71% rename from .github/workflows/TestRunnerTiledSnitch.yml rename to .github/workflows/TestRunnerTiledSnitchSequential.yml index c109958..8060777 100644 --- a/.github/workflows/TestRunnerTiledSnitch.yml +++ b/.github/workflows/TestRunnerTiledSnitchSequential.yml @@ -1,4 +1,4 @@ -name: TestRunnerTiledSnitch +name: TestRunnerTiledSnitchSequential on: workflow_call: @@ -6,6 +6,19 @@ on: tests-config: required: true type: string + num-cores: + required: false + default: 9 + type: number + default-memory-level: + required: false + default: "L2" + type: string + simulator: + required: false + default: "banshee" + type: string + jobs: @@ -40,7 +53,7 @@ jobs: L1_values=$(echo "$test" | jq -r '.L1[]') for L1_value in $L1_values; do echo "Running test: $testName with L1: $L1_value" - python testRunner_tiled_snitch.py -t Tests/$testName --cores=9 --simulator=banshee --l1 $L1_value --defaultMemLevel=L2 --toolchain_install_dir /app/install/riscv-llvm/ + python testRunner_tiled_snitch.py -t Tests/$testName --cores=${{ inputs.num-cores }} --simulator=${{ inputs.simulator }} --l1 $L1_value --defaultMemLevel=${{ inputs.default-memory-level }} --toolchain_install_dir /app/install/riscv-llvm/ done done shell: bash From 27dff8ca355e13614aa1f0dae60dfedbbd0a0661 Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 19:29:18 +0100 Subject: [PATCH 07/11] Revert "add victor's comment" This reverts commit 44ed707c73e633042e102530b3651ab889efc2f4. --- .github/workflows/CI.yml | 2 +- ...Sequential.yml => TestRunnerTiledSnitch.yml} | 17 ++--------------- 2 files changed, 3 insertions(+), 16 deletions(-) rename .github/workflows/{TestRunnerTiledSnitchSequential.yml => TestRunnerTiledSnitch.yml} (71%) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 43380af..2cad974 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -105,7 +105,7 @@ jobs: testRQGEMMTransB snitch-kernels-tiled-singlebuffer-L2: - uses: ./.github/workflows/TestRunnerTiledSnitchSequential.yml + uses: ./.github/workflows/TestRunnerTiledSnitch.yml with: tests-config: | [ diff --git a/.github/workflows/TestRunnerTiledSnitchSequential.yml b/.github/workflows/TestRunnerTiledSnitch.yml similarity index 71% rename from .github/workflows/TestRunnerTiledSnitchSequential.yml rename to .github/workflows/TestRunnerTiledSnitch.yml index 8060777..c109958 100644 --- a/.github/workflows/TestRunnerTiledSnitchSequential.yml +++ b/.github/workflows/TestRunnerTiledSnitch.yml @@ -1,4 +1,4 @@ -name: TestRunnerTiledSnitchSequential +name: TestRunnerTiledSnitch on: workflow_call: @@ -6,19 +6,6 @@ on: tests-config: required: true type: string - num-cores: - required: false - default: 9 - type: number - default-memory-level: - required: false - default: "L2" - type: string - simulator: - required: false - default: "banshee" - type: string - jobs: @@ -53,7 +40,7 @@ jobs: L1_values=$(echo "$test" | jq -r '.L1[]') for L1_value in $L1_values; do echo "Running test: $testName with L1: $L1_value" - python testRunner_tiled_snitch.py -t Tests/$testName --cores=${{ inputs.num-cores }} --simulator=${{ inputs.simulator }} --l1 $L1_value --defaultMemLevel=${{ inputs.default-memory-level }} --toolchain_install_dir /app/install/riscv-llvm/ + python testRunner_tiled_snitch.py -t Tests/$testName --cores=9 --simulator=banshee --l1 $L1_value --defaultMemLevel=L2 --toolchain_install_dir /app/install/riscv-llvm/ done done shell: bash From 0d4baedb1ccbe54c898957ae5b08bd504b44408c Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 19:45:43 +0100 Subject: [PATCH 08/11] add victor's comment --- .github/workflows/CI.yml | 3 ++- .github/workflows/TestRunnerSnitch.yml | 9 ++++++++- ....yml => TestRunnerTiledSnitchSequential.yml} | 17 +++++++++++++++-- 3 files changed, 25 insertions(+), 4 deletions(-) rename .github/workflows/{TestRunnerTiledSnitch.yml => TestRunnerTiledSnitchSequential.yml} (71%) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 2cad974..e359aac 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -103,9 +103,10 @@ jobs: testRQGEMM TestRQAdd testRQGEMMTransB + num-cores: 9 snitch-kernels-tiled-singlebuffer-L2: - uses: ./.github/workflows/TestRunnerTiledSnitch.yml + uses: ./.github/workflows/TestRunnerTiledSnitchSequential.yml with: tests-config: | [ diff --git a/.github/workflows/TestRunnerSnitch.yml b/.github/workflows/TestRunnerSnitch.yml index 3a0f9a0..c7ae8c0 100644 --- a/.github/workflows/TestRunnerSnitch.yml +++ b/.github/workflows/TestRunnerSnitch.yml @@ -6,6 +6,13 @@ on: test-names: required: true type: string + num-cores: + required: true + type: number + simulator: + required: false + default: "banshee" + type: string jobs: test-runner-snitch: @@ -26,7 +33,7 @@ jobs: echo "$testNames" | while IFS= read -r testName; do if [[ -n "$testName" ]]; then echo "Running test: $testName" - python testRunner_snitch.py -t Tests/$testName --simulator=banshee --cores=9 --toolchain_install_dir /app/install/riscv-llvm/ + python testRunner_snitch.py -t Tests/$testName --simulator=${{ inputs.simulator }} --cores=${{ inputs.num-cores }} --toolchain_install_dir /app/install/riscv-llvm/ fi done shell: bash \ No newline at end of file diff --git a/.github/workflows/TestRunnerTiledSnitch.yml b/.github/workflows/TestRunnerTiledSnitchSequential.yml similarity index 71% rename from .github/workflows/TestRunnerTiledSnitch.yml rename to .github/workflows/TestRunnerTiledSnitchSequential.yml index c109958..10a101a 100644 --- a/.github/workflows/TestRunnerTiledSnitch.yml +++ b/.github/workflows/TestRunnerTiledSnitchSequential.yml @@ -1,4 +1,4 @@ -name: TestRunnerTiledSnitch +name: TestRunnerTiledSnitchSequential on: workflow_call: @@ -6,6 +6,19 @@ on: tests-config: required: true type: string + num-cores: + required: false + default: 9 + type: number + default-memory-level: + required: false + default: "L2" + type: string + simulator: + required: false + default: "banshee" + type: string + jobs: @@ -40,7 +53,7 @@ jobs: L1_values=$(echo "$test" | jq -r '.L1[]') for L1_value in $L1_values; do echo "Running test: $testName with L1: $L1_value" - python testRunner_tiled_snitch.py -t Tests/$testName --cores=9 --simulator=banshee --l1 $L1_value --defaultMemLevel=L2 --toolchain_install_dir /app/install/riscv-llvm/ + python testRunner_tiled_snitch.py -t Tests/$testName --cores=${{ inputs.num-cores }} --simulator=${{ inputs.simulator }} --l1 $L1_value --defaultMemLevel=${{ inputs.default-memory-level }} --toolchain_install_dir /app/install/riscv-llvm/ done done shell: bash From 6fe7c42f9a4b4a1455563a68c18d49a1d00204cc Mon Sep 17 00:00:00 2001 From: tahaelbayad Date: Fri, 13 Dec 2024 20:21:21 +0100 Subject: [PATCH 09/11] add victor's comment --- .gitmodules | 4 - TargetLibraries/Snitch/CMakeLists.txt | 1 - .../Snitch/src/pulp_nn_add_i8_i8_i8.c | 202 ++++++++++++++++++ .../Snitch/third_party/pulp-nn-mixed | 1 - 4 files changed, 202 insertions(+), 6 deletions(-) create mode 100644 TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c delete mode 160000 TargetLibraries/Snitch/third_party/pulp-nn-mixed diff --git a/.gitmodules b/.gitmodules index 336972c..def05e2 100644 --- a/.gitmodules +++ b/.gitmodules @@ -7,7 +7,3 @@ [submodule "CMSIS-NN"] path = TargetLibraries/CMSIS/third_party/CMSIS-NN url = https://github.com/ARM-software/CMSIS-NN.git -[submodule "TargetLibraries/Snitch/third_party/pulp-nn-mixed"] - path = TargetLibraries/Snitch/third_party/pulp-nn-mixed - url = https://github.com/Victor-Jung/pulp-nn-mixed.git - branch = deeploySnitchTarget diff --git a/TargetLibraries/Snitch/CMakeLists.txt b/TargetLibraries/Snitch/CMakeLists.txt index 49130ee..78a214f 100644 --- a/TargetLibraries/Snitch/CMakeLists.txt +++ b/TargetLibraries/Snitch/CMakeLists.txt @@ -1,6 +1,5 @@ file(GLOB_RECURSE SOURCES "src/**" - "third_party/pulp-nn-mixed/DeeploySnitch/src/**" ) include(cmake/snitch-runtime-precompiled.cmake) diff --git a/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c b/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c new file mode 100644 index 0000000..61a4044 --- /dev/null +++ b/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c @@ -0,0 +1,202 @@ +/* + * pulp_nn_add_i8_i8_i8.c + * Georg Rutishauser + * Victor Jung + * + * Copyright (C) 2018-2020 University of Bologna + * + * 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 "DeeploySnitchMath.h" + + + +void __attribute__ ((noinline)) pulp_nn_add_i8_i8_i8( + int8_t * pIn1, + int8_t * pIn2, + int8_t * pOut, + int32_t in1_mul, + int32_t in1_add, + uint16_t in1_shift, + int32_t in2_mul, + int32_t in2_add, + uint16_t in2_shift, + int32_t out_mul, + int32_t out_add, + uint16_t out_shift, + uint16_t dim_im_in_x, + uint16_t dim_im_in_y, + uint16_t ch_im_in, + int out_requant_flag) +{ + int core_id = snrt_global_compute_core_idx(); + int n_cores = snrt_global_compute_core_num(); + + if (dim_im_in_y < n_cores){ + n_cores = dim_im_in_y; + } + + int Log2Core = INT_LOG2(n_cores); + int chunck = (dim_im_in_y >> Log2Core) + ((dim_im_in_y & (n_cores - 1)) != 0); + + int32_t in1_rq1, in1_rq2, in1_rq3, in1_rq4, + in2_rq1, in2_rq2, in2_rq3, in2_rq4; + int32_t sum1, sum2, sum3, sum4; + int32_t sum_out1, sum_out2, sum_out3, sum_out4; + int32_t out1, out2, out3, out4, + sum_int1, sum_int2, sum_int3, sum_int4; + + + + int ch_im_in1_r = ch_im_in >> 0; + int ch_im_in2_r = ch_im_in >> 0; + int ch_im_out_r = ch_im_in >> 0; + + int start = MIN(chunck * core_id, dim_im_in_y); + int stop = MIN(start + chunck, dim_im_in_y); + + int8_t *target1 = pIn1 + start * ch_im_in1_r * dim_im_in_x; + int8_t *target2 = pIn2 + start * ch_im_in2_r * dim_im_in_x; + int8_t *pOutBuffer = pOut + start * ch_im_out_r * dim_im_in_x; + + int a = 0; + int b = 0; + + int8_t *target1_ext = &a; + int8_t *target2_ext = &b; + + for (int i=0; i<(((stop-start) * ch_im_out_r * dim_im_in_x) >> 2); i++) + { + target1_ext = target1; + target1+=4; + + target2_ext = target2; + target2+=4; +#ifdef ADD_VERBOSE + printf("core %d - in1 it0 before requant: %d\n", core_id, *(target1_ext)); + printf("core %d - in2 it0 before requant: %d\n", core_id, *(target2_ext)); +#endif + in1_rq1 = ((*(target1_ext)) * in1_mul + in1_add) >> in1_shift; + in2_rq1 = ((*(target2_ext)) * in2_mul + in2_add) >> in2_shift; + sum1 = clips8(in1_rq1) + clips8(in2_rq1); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq1 it0 after requant: %d\nclipped in1_rq1: %d\n", core_id, in1_rq1, clips8(in1_rq1)); + printf("core %d - in2_rq1 it0 after requant: %d\nclipped in2_rq1: %d\n", core_id, in2_rq1), clips8(in2_rq1); + printf("core %d - sum1: %d\n", core_id, sum1); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it1 before requant: %d\n", core_id, *(target1_ext + 1 )); + printf("core %d - in2 it1 before requant: %d\n", core_id, *(target2_ext + 1 )); +#endif + in1_rq2 = ((*(target1_ext + 1 )) * in1_mul + in1_add) >> in1_shift; + in2_rq2 = ((*(target2_ext + 1 )) * in2_mul + in2_add) >> in2_shift; + sum2 = clips8(in1_rq2) + clips8(in2_rq2); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq2 it1 after requant: %d\nclipped in1_rq2: %d\n", core_id, in1_rq2, clips8(in1_rq2)); + printf("core %d - in2_rq2 it1 after requant: %d\nclipped in2_rq2: %d\n", core_id, in2_rq2), clips8(in2_rq2); + printf("core %d - sum2: %d\n", core_id, sum2); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it2 before requant: %d\n", core_id, *(target1_ext + 2 )); + printf("core %d - in2 it2 before requant: %d\n", core_id, *(target2_ext + 2 )); +#endif + in1_rq3 = ((*(target1_ext + 2 )) * in1_mul + in1_add) >> in1_shift; + in2_rq3 = ((*(target2_ext + 2 )) * in2_mul + in2_add) >> in2_shift; + sum3 = clips8(in1_rq3) + clips8(in2_rq3); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq3 it2 after requant: %d\nclipped in1_rq3: %d\n", core_id, in1_rq3, clips8(in1_rq3)); + printf("core %d - in2_rq3 it2 after requant: %d\nclipped in2_rq3: %d\n", core_id, in2_rq3), clips8(in2_rq3); + printf("core %d - sum3: %d\n", core_id, sum3); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it3 before requant: %d\n", core_id, *(target1_ext + 3 )); + printf("core %d - in2 it3 before requant: %d\n", core_id, *(target2_ext + 3 )); +#endif + in1_rq4 = ((*(target1_ext + 3 )) * in1_mul + in1_add) >> in1_shift; + in2_rq4 = ((*(target2_ext + 3 )) * in2_mul + in2_add) >> in2_shift; + sum4 = clips8(in1_rq4) + clips8(in2_rq4); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq4 it3 after requant: %d\nclipped in1_rq4: %d\n", core_id, in1_rq4, clips8(in1_rq4)); + printf("core %d - in2_rq4 it3 after requant: %d\nclipped in2_rq4: %d\n", core_id, in2_rq4), clips8(in2_rq4); + printf("core %d - sum4: %d\n", core_id, sum4); +#endif + + if (out_requant_flag) { + sum1 = (sum1 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum1: %d\n", core_id, sum1); +#endif + sum2 = (sum2 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum2: %d\n", core_id, sum2); +#endif + sum3 = (sum3 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum3: %d\n", core_id, sum3); +#endif + sum4 = (sum4 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum4: %d\n", core_id, sum4); +#endif + } + out1 = clips8(sum1); +#ifdef ADD_VERBOSE + printf("core %d - out1 clipped: %d\n", core_id, out1); +#endif + out2 = clips8(sum2); +#ifdef ADD_VERBOSE + printf("core %d - out2 clipped: %d\n", core_id, out2); +#endif + out3 = clips8(sum3); +#ifdef ADD_VERBOSE + printf("core %d - out3 clipped: %d\n", core_id, out3); +#endif + out4 = clips8(sum4); +#ifdef ADD_VERBOSE + printf("core %d - out4 clipped: %d\n", core_id, out4); +#endif + + + *pOutBuffer = (int8_t) out1; + pOutBuffer++; + *pOutBuffer = (int8_t) out2; + pOutBuffer++; + *pOutBuffer = (int8_t) out3; + pOutBuffer++; + *pOutBuffer = (int8_t) out4; + pOutBuffer++; + } + // SCHEREMO: Cleanup leftovers, not doing it with this codebase for sub-byte formats + for (int i=0; i<(((stop-start) * ch_im_out_r * dim_im_in_x) % 4); i++){ + in1_rq1 = ((*(target1)) * in1_mul + in1_add) >> in1_shift; + in2_rq1 = ((*(target2)) * in2_mul + in2_add) >> in2_shift; + + // SCHEREMO: Maybe it's just LLVM, but unless I hack 3 non-unrolled nops in here, stuff fails + #pragma nounroll + for (int j = 0; j < 3; j++) { + asm volatile("nop" ::); + } + + target1++; + target2++; + sum1 = clips8(in1_rq1) + clips8(in2_rq1); + if (out_requant_flag) { + sum1 = (sum1 * out_mul + out_add) >> out_shift; + } + + out1 = clips8(sum1); + *pOutBuffer = (int8_t)out1; + pOutBuffer++; + } +} diff --git a/TargetLibraries/Snitch/third_party/pulp-nn-mixed b/TargetLibraries/Snitch/third_party/pulp-nn-mixed deleted file mode 160000 index 4f0902b..0000000 --- a/TargetLibraries/Snitch/third_party/pulp-nn-mixed +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 4f0902bbdab3265b4f56e156cb4a98f52ac6e067 From 41dcc0f660e4646bbc5b917e8f2ba8f2f9a2aa90 Mon Sep 17 00:00:00 2001 From: telbayad Date: Wed, 8 Jan 2025 18:33:41 +0100 Subject: [PATCH 10/11] apply Victor's comment --- .../Generic/Templates/RQAddTemplate.py | 48 ++++ Deeploy/Targets/PULPOpen/Bindings.py | 2 +- Deeploy/Targets/PULPOpen/Parsers.py | 58 +---- Deeploy/Targets/PULPOpen/Platform.py | 10 +- .../PULPOpen/Templates/RQAddTemplate.py | 25 +-- .../Targets/Snitch/{Parser.py => Parsers.py} | 3 +- Deeploy/Targets/Snitch/Platform.py | 2 +- .../Targets/Snitch/Templates/RQAddTemplate.py | 27 +-- .../Snitch/src/pulp_nn_add_i8_i8_i8.c | 202 ----------------- .../Snitch/src/snitch_nn_add_i8_i8_i8.c | 205 ++++++++++++++++++ 10 files changed, 268 insertions(+), 314 deletions(-) create mode 100644 Deeploy/Targets/Generic/Templates/RQAddTemplate.py rename Deeploy/Targets/Snitch/{Parser.py => Parsers.py} (97%) delete mode 100644 TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c create mode 100644 TargetLibraries/Snitch/src/snitch_nn_add_i8_i8_i8.c diff --git a/Deeploy/Targets/Generic/Templates/RQAddTemplate.py b/Deeploy/Targets/Generic/Templates/RQAddTemplate.py new file mode 100644 index 0000000..dacc9ac --- /dev/null +++ b/Deeploy/Targets/Generic/Templates/RQAddTemplate.py @@ -0,0 +1,48 @@ +# ---------------------------------------------------------------------- +# +# File: RQAddTemplate.py +# +# Last edited: 11.11.2023 +# +# Copyright (C) 2023, ETH Zurich and University of Bologna. +# +# Author: +# - Moritz Scherer, ETH Zurich +# - Victor Jung, ETH Zurich +# +# ---------------------------------------------------------------------- +# SPDX-License-Identifier: Apache-2.0 +# +# 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 +# +# 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 typing import Dict, List, Tuple + +from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation + + +class RQAddTemplate(NodeTemplate): + + def __init__(self, templateStr): + super().__init__(templateStr) + + def alignToContext(self, ctxt: NetworkContext, + operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: + # Extract signedness information of input, weights and output + signedI2 = ctxt.lookup(operatorRepresentation['data_in_2'])._type.referencedType.typeMin < 0 + signedI = ctxt.lookup(operatorRepresentation['data_in_1'])._type.referencedType.typeMin < 0 + signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 + operatorRepresentation['input_2_signed'] = signedI2 + operatorRepresentation['input_signed'] = signedI + operatorRepresentation['output_signed'] = signedO + + return ctxt, operatorRepresentation, [] diff --git a/Deeploy/Targets/PULPOpen/Bindings.py b/Deeploy/Targets/PULPOpen/Bindings.py index 5d23620..c8c8e38 100644 --- a/Deeploy/Targets/PULPOpen/Bindings.py +++ b/Deeploy/Targets/PULPOpen/Bindings.py @@ -157,7 +157,7 @@ PULPRQAddBindings = [ NodeBinding(PULPRQAddChecker([PointerClass(_type), PointerClass(_type2)], [PointerClass(_type3)]), - RQAddTemplate.RQAddTemplate, ForkTransformer) + RQAddTemplate.referenceTemplate, ForkTransformer) for _type in [int8_t, uint8_t] for _type2 in [int8_t, uint8_t] for _type3 in [int8_t, uint8_t] diff --git a/Deeploy/Targets/PULPOpen/Parsers.py b/Deeploy/Targets/PULPOpen/Parsers.py index 6878237..33b4abe 100644 --- a/Deeploy/Targets/PULPOpen/Parsers.py +++ b/Deeploy/Targets/PULPOpen/Parsers.py @@ -29,63 +29,7 @@ import onnx_graphsurgeon as gs from Deeploy.DeeployTypes import NetworkContext -from Deeploy.Targets.Generic.Parsers import AddParser, GEMMParser, RQSConv1DParser, RQSConv2DParser, RQSParserInterface - - -class PULPRQAddParser(AddParser): - - def parseNode(self, node: gs.Node) -> bool: - - if not super().parseNode(node): - return False - - ret = all([ - 'rqs1_mul' in node.attrs, - 'rqs1_add' in node.attrs, - 'rqs1_div' in node.attrs, - 'rqs1_signed' in node.attrs, - any(['rqs1_n_levels' in node.attrs, 'rqs1_n_levels_out' in node.attrs]), - 'rqs2_mul' in node.attrs, - 'rqs2_add' in node.attrs, - 'rqs2_div' in node.attrs, - 'rqs2_signed' in node.attrs, - any(['rqs2_n_levels' in node.attrs, 'rqs2_n_levels_out' in node.attrs]), - 'rqsOut_mul' in node.attrs, - 'rqsOut_add' in node.attrs, - 'rqsOut_div' in node.attrs, - 'rqsOut_signed' in node.attrs, - any(['rqsOut_n_levels' in node.attrs, 'rqsOut_n_levels_out' in node.attrs]), - ]) - - if ret: - if 'rqs1_n_levels' in node.attrs: - self.operatorRepresentation['rqs1_n_levels'] = int(node.attrs['rqs1_n_levels'].values) - else: - self.operatorRepresentation['rqs1_n_levels'] = int(node.attrs['rqs1_n_levels_out'].values) - self.operatorRepresentation['rqs1_mul'] = int(node.attrs['rqs1_mul']) - self.operatorRepresentation['rqs1_add'] = int(node.attrs['rqs1_add']) - self.operatorRepresentation['rqs1_signed'] = int(node.attrs['rqs1_signed'].values) - self.operatorRepresentation['rqs1_log2D'] = int(math.log2(node.attrs['rqs1_div'].values)) - - if 'rqs2_n_levels' in node.attrs: - self.operatorRepresentation['rqs2_n_levels'] = int(node.attrs['rqs2_n_levels'].values) - else: - self.operatorRepresentation['rqs2_n_levels'] = int(node.attrs['rqs2_n_levels_out'].values) - self.operatorRepresentation['rqs2_mul'] = int(node.attrs['rqs2_mul']) - self.operatorRepresentation['rqs2_add'] = int(node.attrs['rqs2_add']) - self.operatorRepresentation['rqs2_signed'] = int(node.attrs['rqs2_signed'].values) - self.operatorRepresentation['rqs2_log2D'] = int(math.log2(node.attrs['rqs2_div'].values)) - - if 'rqsOut_n_levels' in node.attrs: - self.operatorRepresentation['rqsOut_n_levels'] = int(node.attrs['rqsOut_n_levels'].values) - else: - self.operatorRepresentation['rqsOut_n_levels'] = int(node.attrs['rqsOut_n_levels_out'].values) - self.operatorRepresentation['rqsOut_mul'] = int(node.attrs['rqsOut_mul']) - self.operatorRepresentation['rqsOut_add'] = int(node.attrs['rqsOut_add']) - self.operatorRepresentation['rqsOut_signed'] = int(node.attrs['rqsOut_signed'].values) - self.operatorRepresentation['rqsOut_log2D'] = int(math.log2(node.attrs['rqsOut_div'].values)) - - return ret +from Deeploy.Targets.Generic.Parsers import GEMMParser, RQSConv1DParser, RQSConv2DParser, RQSParserInterface class PULPConv2DParser(RQSConv2DParser): diff --git a/Deeploy/Targets/PULPOpen/Platform.py b/Deeploy/Targets/PULPOpen/Platform.py index c32f29d..bac2d82 100644 --- a/Deeploy/Targets/PULPOpen/Platform.py +++ b/Deeploy/Targets/PULPOpen/Platform.py @@ -39,9 +39,9 @@ PadLayer, ReduceMeanLayer, RequantShiftLayer, ReshapeLayer, RQIntegerDivLayer, RQSiGELULayer, RQSiHardswishLayer, \ SliceLayer, TransposeLayer, iHardswishLayer, iRMSNormLayer, iSoftmaxLayer from Deeploy.Targets.Generic.Parsers import AddParser, ConcatParser, FlattenParser, GatherParser, MatMulParser, \ - MulParser, Pad1DParser, Pad2DParser, ReduceMeanParser, RequantShiftParser, ReshapeParser, RQIntegerDivParser, \ - RQSiGELUParser, RQSiHardswishParser, SliceParser, TransposeParser, UniformRequantShiftParser, UnsqueezeParser, \ - iHardswishParser, iRMSNormParser, iSoftmaxParser + MulParser, Pad1DParser, Pad2DParser, ReduceMeanParser, RequantShiftParser, ReshapeParser, RQAddParser, \ + RQIntegerDivParser, RQSiGELUParser, RQSiHardswishParser, SliceParser, TransposeParser, UniformRequantShiftParser, \ + UnsqueezeParser, iHardswishParser, iRMSNormParser, iSoftmaxParser from Deeploy.Targets.Generic.Templates import AllocateTemplate as BasicAllocateTemplate from Deeploy.Targets.Generic.TopologyOptimizationPasses.Passes import IntegerDivRequantMergePass, \ MergeConstAddAndRequantPass, MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, SkipEmptyConcatPass, \ @@ -50,7 +50,7 @@ PULPReduceMeanBindings from Deeploy.Targets.PULPOpen.Layers import PULPRQSConvLayer, PULPRQSGEMMLayer from Deeploy.Targets.PULPOpen.Parsers import PULPConv1DParser, PULPConv2DParser, PULPDWConv1DParser, \ - PULPDWConv2DParser, PULPGEMMParser, PULPMatrixVecParser, PULPRQAddParser, PULPTallGEMMParser + PULPDWConv2DParser, PULPGEMMParser, PULPMatrixVecParser, PULPTallGEMMParser from Deeploy.Targets.PULPOpen.Templates import AllocateTemplate, FreeTemplate from Deeploy.Targets.PULPOpen.Tiler import PULPAddTilingReadyBindings, PULPConcatTilingReadyBindings, \ PULPFlattenTilingReadyBindings, PULPiHardswishTilingReadyBindings, PULPiRMSNormTilingReadyBindings, \ @@ -62,7 +62,7 @@ from Deeploy.Targets.PULPOpen.TopologyOptimizationPasses.Passes import PULPAddRequantMergePass, \ PULPConvRequantMergePass, PULPGEMMRequantMergePass, PULPMatMulRequantMergePass -RQAddMapper = NodeMapper(PULPRQAddParser(), PULPRQAddTilingReadyBindings) +RQAddMapper = NodeMapper(RQAddParser(), PULPRQAddTilingReadyBindings) AddMapper = NodeMapper(AddParser(), PULPAddTilingReadyBindings) FlattenMapper = NodeMapper(FlattenParser(), PULPFlattenTilingReadyBindings) GatherMapper = NodeMapper(GatherParser(), BasicGatherBindings) diff --git a/Deeploy/Targets/PULPOpen/Templates/RQAddTemplate.py b/Deeploy/Targets/PULPOpen/Templates/RQAddTemplate.py index 49ede2b..f88b2db 100644 --- a/Deeploy/Targets/PULPOpen/Templates/RQAddTemplate.py +++ b/Deeploy/Targets/PULPOpen/Templates/RQAddTemplate.py @@ -23,30 +23,9 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import Dict, List, Tuple +from Deeploy.Targets.Generic.Templates.RQAddTemplate import RQAddTemplate -from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation - - -class PULPRQAddTemplate(NodeTemplate): - - def __init__(self, templateStr): - super().__init__(templateStr) - - def alignToContext(self, ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: - # Extract signedness information of input, weights and output - signedI2 = ctxt.lookup(operatorRepresentation['data_in_2'])._type.referencedType.typeMin < 0 - signedI = ctxt.lookup(operatorRepresentation['data_in_1'])._type.referencedType.typeMin < 0 - signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 - operatorRepresentation['input_2_signed'] = signedI2 - operatorRepresentation['input_signed'] = signedI - operatorRepresentation['output_signed'] = signedO - - return ctxt, operatorRepresentation, [] - - -RQAddTemplate = PULPRQAddTemplate(""" +referenceTemplate = RQAddTemplate(""" <% signatureString = '' diff --git a/Deeploy/Targets/Snitch/Parser.py b/Deeploy/Targets/Snitch/Parsers.py similarity index 97% rename from Deeploy/Targets/Snitch/Parser.py rename to Deeploy/Targets/Snitch/Parsers.py index 57b91a8..dfd3248 100644 --- a/Deeploy/Targets/Snitch/Parser.py +++ b/Deeploy/Targets/Snitch/Parsers.py @@ -19,7 +19,8 @@ # # # Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an AS IS BASIS, WITHOUT +# distributed under the Lic +# ense 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. diff --git a/Deeploy/Targets/Snitch/Platform.py b/Deeploy/Targets/Snitch/Platform.py index aa2893a..3b45d9e 100644 --- a/Deeploy/Targets/Snitch/Platform.py +++ b/Deeploy/Targets/Snitch/Platform.py @@ -41,7 +41,7 @@ IntegerDivRequantMergePass, MergeConstAddAndRequantPass, MergeTrueIntegerDivRequantShiftPass, RQSSplitPass, \ SkipEmptyConcatPass, SkipUnityRequantPass, iGELURequantMergePass, iHardswishRequantMergePass from Deeploy.Targets.PULPOpen.Platform import RQAddMapper -from Deeploy.Targets.Snitch.Parser import SnitchGEMMParser, SnitchRQGEMMParser +from Deeploy.Targets.Snitch.Parsers import SnitchGEMMParser, SnitchRQGEMMParser from Deeploy.Targets.Snitch.Templates import AllocateTemplate, FreeTemplate from Deeploy.Targets.Snitch.Tiler import SnitchAddTileReadyBindings, SnitchGemmTilingReadyBindings, \ SnitchiNoNormTilingReadyBindings, SnitchiSoftmaxTilingReadyBindings, SnitchRQAddTilingReadyBindings, \ diff --git a/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py index 2a49a3b..afc637c 100644 --- a/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py +++ b/Deeploy/Targets/Snitch/Templates/RQAddTemplate.py @@ -25,30 +25,9 @@ # See the License for the specific language governing permissions and # limitations under the License. -from typing import Dict, List, Tuple +from Deeploy.Targets.Generic.Templates.RQAddTemplate import RQAddTemplate -from Deeploy.DeeployTypes import NetworkContext, NodeTemplate, OperatorRepresentation - - -class SnitchRQAddTemplate(NodeTemplate): - - def __init__(self, templateStr): - super().__init__(templateStr) - - def alignToContext(self, ctxt: NetworkContext, - operatorRepresentation: OperatorRepresentation) -> Tuple[NetworkContext, Dict, List[str]]: - # Extract signedness information of input, weights and output - signedI2 = ctxt.lookup(operatorRepresentation['data_in_2'])._type.referencedType.typeMin < 0 - signedI = ctxt.lookup(operatorRepresentation['data_in_1'])._type.referencedType.typeMin < 0 - signedO = ctxt.lookup(operatorRepresentation['data_out'])._type.referencedType.typeMin < 0 - operatorRepresentation['input_2_signed'] = signedI2 - operatorRepresentation['input_signed'] = signedI - operatorRepresentation['output_signed'] = signedO - - return ctxt, operatorRepresentation, [] - - -referenceTemplate = SnitchRQAddTemplate(""" +referenceTemplate = RQAddTemplate(""" <% signatureString = '' @@ -67,5 +46,5 @@ def alignToContext(self, ctxt: NetworkContext, %> // PULP NN RQADD -pulp_nn_add${signatureString}(${data_in_1}, ${data_in_2}, ${data_out}, ${rqs1_mul}, ${rqs1_add}, ${rqs1_log2D}, ${rqs2_mul}, ${rqs2_add}, ${rqs2_log2D}, ${rqsOut_mul}, ${rqsOut_add}, ${rqsOut_log2D}, 1, ${size}, 1, 1); +snitch_nn_add${signatureString}(${data_in_1}, ${data_in_2}, ${data_out}, ${rqs1_mul}, ${rqs1_add}, ${rqs1_log2D}, ${rqs2_mul}, ${rqs2_add}, ${rqs2_log2D}, ${rqsOut_mul}, ${rqsOut_add}, ${rqsOut_log2D}, 1, ${size}, 1, 1); """) diff --git a/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c b/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c deleted file mode 100644 index 61a4044..0000000 --- a/TargetLibraries/Snitch/src/pulp_nn_add_i8_i8_i8.c +++ /dev/null @@ -1,202 +0,0 @@ -/* - * pulp_nn_add_i8_i8_i8.c - * Georg Rutishauser - * Victor Jung - * - * Copyright (C) 2018-2020 University of Bologna - * - * 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 "DeeploySnitchMath.h" - - - -void __attribute__ ((noinline)) pulp_nn_add_i8_i8_i8( - int8_t * pIn1, - int8_t * pIn2, - int8_t * pOut, - int32_t in1_mul, - int32_t in1_add, - uint16_t in1_shift, - int32_t in2_mul, - int32_t in2_add, - uint16_t in2_shift, - int32_t out_mul, - int32_t out_add, - uint16_t out_shift, - uint16_t dim_im_in_x, - uint16_t dim_im_in_y, - uint16_t ch_im_in, - int out_requant_flag) -{ - int core_id = snrt_global_compute_core_idx(); - int n_cores = snrt_global_compute_core_num(); - - if (dim_im_in_y < n_cores){ - n_cores = dim_im_in_y; - } - - int Log2Core = INT_LOG2(n_cores); - int chunck = (dim_im_in_y >> Log2Core) + ((dim_im_in_y & (n_cores - 1)) != 0); - - int32_t in1_rq1, in1_rq2, in1_rq3, in1_rq4, - in2_rq1, in2_rq2, in2_rq3, in2_rq4; - int32_t sum1, sum2, sum3, sum4; - int32_t sum_out1, sum_out2, sum_out3, sum_out4; - int32_t out1, out2, out3, out4, - sum_int1, sum_int2, sum_int3, sum_int4; - - - - int ch_im_in1_r = ch_im_in >> 0; - int ch_im_in2_r = ch_im_in >> 0; - int ch_im_out_r = ch_im_in >> 0; - - int start = MIN(chunck * core_id, dim_im_in_y); - int stop = MIN(start + chunck, dim_im_in_y); - - int8_t *target1 = pIn1 + start * ch_im_in1_r * dim_im_in_x; - int8_t *target2 = pIn2 + start * ch_im_in2_r * dim_im_in_x; - int8_t *pOutBuffer = pOut + start * ch_im_out_r * dim_im_in_x; - - int a = 0; - int b = 0; - - int8_t *target1_ext = &a; - int8_t *target2_ext = &b; - - for (int i=0; i<(((stop-start) * ch_im_out_r * dim_im_in_x) >> 2); i++) - { - target1_ext = target1; - target1+=4; - - target2_ext = target2; - target2+=4; -#ifdef ADD_VERBOSE - printf("core %d - in1 it0 before requant: %d\n", core_id, *(target1_ext)); - printf("core %d - in2 it0 before requant: %d\n", core_id, *(target2_ext)); -#endif - in1_rq1 = ((*(target1_ext)) * in1_mul + in1_add) >> in1_shift; - in2_rq1 = ((*(target2_ext)) * in2_mul + in2_add) >> in2_shift; - sum1 = clips8(in1_rq1) + clips8(in2_rq1); -#ifdef ADD_VERBOSE - printf("core %d - in1_rq1 it0 after requant: %d\nclipped in1_rq1: %d\n", core_id, in1_rq1, clips8(in1_rq1)); - printf("core %d - in2_rq1 it0 after requant: %d\nclipped in2_rq1: %d\n", core_id, in2_rq1), clips8(in2_rq1); - printf("core %d - sum1: %d\n", core_id, sum1); -#endif -#ifdef ADD_VERBOSE - printf("core %d - in1 it1 before requant: %d\n", core_id, *(target1_ext + 1 )); - printf("core %d - in2 it1 before requant: %d\n", core_id, *(target2_ext + 1 )); -#endif - in1_rq2 = ((*(target1_ext + 1 )) * in1_mul + in1_add) >> in1_shift; - in2_rq2 = ((*(target2_ext + 1 )) * in2_mul + in2_add) >> in2_shift; - sum2 = clips8(in1_rq2) + clips8(in2_rq2); -#ifdef ADD_VERBOSE - printf("core %d - in1_rq2 it1 after requant: %d\nclipped in1_rq2: %d\n", core_id, in1_rq2, clips8(in1_rq2)); - printf("core %d - in2_rq2 it1 after requant: %d\nclipped in2_rq2: %d\n", core_id, in2_rq2), clips8(in2_rq2); - printf("core %d - sum2: %d\n", core_id, sum2); -#endif -#ifdef ADD_VERBOSE - printf("core %d - in1 it2 before requant: %d\n", core_id, *(target1_ext + 2 )); - printf("core %d - in2 it2 before requant: %d\n", core_id, *(target2_ext + 2 )); -#endif - in1_rq3 = ((*(target1_ext + 2 )) * in1_mul + in1_add) >> in1_shift; - in2_rq3 = ((*(target2_ext + 2 )) * in2_mul + in2_add) >> in2_shift; - sum3 = clips8(in1_rq3) + clips8(in2_rq3); -#ifdef ADD_VERBOSE - printf("core %d - in1_rq3 it2 after requant: %d\nclipped in1_rq3: %d\n", core_id, in1_rq3, clips8(in1_rq3)); - printf("core %d - in2_rq3 it2 after requant: %d\nclipped in2_rq3: %d\n", core_id, in2_rq3), clips8(in2_rq3); - printf("core %d - sum3: %d\n", core_id, sum3); -#endif -#ifdef ADD_VERBOSE - printf("core %d - in1 it3 before requant: %d\n", core_id, *(target1_ext + 3 )); - printf("core %d - in2 it3 before requant: %d\n", core_id, *(target2_ext + 3 )); -#endif - in1_rq4 = ((*(target1_ext + 3 )) * in1_mul + in1_add) >> in1_shift; - in2_rq4 = ((*(target2_ext + 3 )) * in2_mul + in2_add) >> in2_shift; - sum4 = clips8(in1_rq4) + clips8(in2_rq4); -#ifdef ADD_VERBOSE - printf("core %d - in1_rq4 it3 after requant: %d\nclipped in1_rq4: %d\n", core_id, in1_rq4, clips8(in1_rq4)); - printf("core %d - in2_rq4 it3 after requant: %d\nclipped in2_rq4: %d\n", core_id, in2_rq4), clips8(in2_rq4); - printf("core %d - sum4: %d\n", core_id, sum4); -#endif - - if (out_requant_flag) { - sum1 = (sum1 * out_mul + out_add) >> out_shift; -#ifdef ADD_VERBOSE - printf("core %d - requantized sum1: %d\n", core_id, sum1); -#endif - sum2 = (sum2 * out_mul + out_add) >> out_shift; -#ifdef ADD_VERBOSE - printf("core %d - requantized sum2: %d\n", core_id, sum2); -#endif - sum3 = (sum3 * out_mul + out_add) >> out_shift; -#ifdef ADD_VERBOSE - printf("core %d - requantized sum3: %d\n", core_id, sum3); -#endif - sum4 = (sum4 * out_mul + out_add) >> out_shift; -#ifdef ADD_VERBOSE - printf("core %d - requantized sum4: %d\n", core_id, sum4); -#endif - } - out1 = clips8(sum1); -#ifdef ADD_VERBOSE - printf("core %d - out1 clipped: %d\n", core_id, out1); -#endif - out2 = clips8(sum2); -#ifdef ADD_VERBOSE - printf("core %d - out2 clipped: %d\n", core_id, out2); -#endif - out3 = clips8(sum3); -#ifdef ADD_VERBOSE - printf("core %d - out3 clipped: %d\n", core_id, out3); -#endif - out4 = clips8(sum4); -#ifdef ADD_VERBOSE - printf("core %d - out4 clipped: %d\n", core_id, out4); -#endif - - - *pOutBuffer = (int8_t) out1; - pOutBuffer++; - *pOutBuffer = (int8_t) out2; - pOutBuffer++; - *pOutBuffer = (int8_t) out3; - pOutBuffer++; - *pOutBuffer = (int8_t) out4; - pOutBuffer++; - } - // SCHEREMO: Cleanup leftovers, not doing it with this codebase for sub-byte formats - for (int i=0; i<(((stop-start) * ch_im_out_r * dim_im_in_x) % 4); i++){ - in1_rq1 = ((*(target1)) * in1_mul + in1_add) >> in1_shift; - in2_rq1 = ((*(target2)) * in2_mul + in2_add) >> in2_shift; - - // SCHEREMO: Maybe it's just LLVM, but unless I hack 3 non-unrolled nops in here, stuff fails - #pragma nounroll - for (int j = 0; j < 3; j++) { - asm volatile("nop" ::); - } - - target1++; - target2++; - sum1 = clips8(in1_rq1) + clips8(in2_rq1); - if (out_requant_flag) { - sum1 = (sum1 * out_mul + out_add) >> out_shift; - } - - out1 = clips8(sum1); - *pOutBuffer = (int8_t)out1; - pOutBuffer++; - } -} diff --git a/TargetLibraries/Snitch/src/snitch_nn_add_i8_i8_i8.c b/TargetLibraries/Snitch/src/snitch_nn_add_i8_i8_i8.c new file mode 100644 index 0000000..f83e3ba --- /dev/null +++ b/TargetLibraries/Snitch/src/snitch_nn_add_i8_i8_i8.c @@ -0,0 +1,205 @@ +/* + * pulp_nn_add_i8_i8_i8.c + * Georg Rutishauser + * Victor Jung + * + * Copyright (C) 2018-2020 University of Bologna + * + * 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 "DeeploySnitchMath.h" + +void __attribute__((noinline)) +snitch_nn_add_i8_i8_i8(int8_t *pIn1, int8_t *pIn2, int8_t *pOut, + int32_t in1_mul, int32_t in1_add, uint16_t in1_shift, + int32_t in2_mul, int32_t in2_add, uint16_t in2_shift, + int32_t out_mul, int32_t out_add, uint16_t out_shift, + uint16_t dim_im_in_x, uint16_t dim_im_in_y, + uint16_t ch_im_in, int out_requant_flag) { + int core_id = snrt_global_compute_core_idx(); + int n_cores = snrt_global_compute_core_num(); + + if (dim_im_in_y < n_cores) { + n_cores = dim_im_in_y; + } + + int Log2Core = INT_LOG2(n_cores); + int chunck = (dim_im_in_y >> Log2Core) + ((dim_im_in_y & (n_cores - 1)) != 0); + + int32_t in1_rq1, in1_rq2, in1_rq3, in1_rq4, in2_rq1, in2_rq2, in2_rq3, + in2_rq4; + int32_t sum1, sum2, sum3, sum4; + int32_t sum_out1, sum_out2, sum_out3, sum_out4; + int32_t out1, out2, out3, out4, sum_int1, sum_int2, sum_int3, sum_int4; + + int ch_im_in1_r = ch_im_in >> 0; + int ch_im_in2_r = ch_im_in >> 0; + int ch_im_out_r = ch_im_in >> 0; + + int start = MIN(chunck * core_id, dim_im_in_y); + int stop = MIN(start + chunck, dim_im_in_y); + + int8_t *target1 = pIn1 + start * ch_im_in1_r * dim_im_in_x; + int8_t *target2 = pIn2 + start * ch_im_in2_r * dim_im_in_x; + int8_t *pOutBuffer = pOut + start * ch_im_out_r * dim_im_in_x; + + int a = 0; + int b = 0; + + int8_t *target1_ext = &a; + int8_t *target2_ext = &b; + + for (int i = 0; i < (((stop - start) * ch_im_out_r * dim_im_in_x) >> 2); + i++) { + target1_ext = target1; + target1 += 4; + + target2_ext = target2; + target2 += 4; +#ifdef ADD_VERBOSE + printf("core %d - in1 it0 before requant: %d\n", core_id, *(target1_ext)); + printf("core %d - in2 it0 before requant: %d\n", core_id, *(target2_ext)); +#endif + in1_rq1 = ((*(target1_ext)) * in1_mul + in1_add) >> in1_shift; + in2_rq1 = ((*(target2_ext)) * in2_mul + in2_add) >> in2_shift; + sum1 = clips8(in1_rq1) + clips8(in2_rq1); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq1 it0 after requant: %d\nclipped in1_rq1: %d\n", + core_id, in1_rq1, clips8(in1_rq1)); + printf("core %d - in2_rq1 it0 after requant: %d\nclipped in2_rq1: %d\n", + core_id, in2_rq1), + clips8(in2_rq1); + printf("core %d - sum1: %d\n", core_id, sum1); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it1 before requant: %d\n", core_id, + *(target1_ext + 1)); + printf("core %d - in2 it1 before requant: %d\n", core_id, + *(target2_ext + 1)); +#endif + in1_rq2 = ((*(target1_ext + 1)) * in1_mul + in1_add) >> in1_shift; + in2_rq2 = ((*(target2_ext + 1)) * in2_mul + in2_add) >> in2_shift; + sum2 = clips8(in1_rq2) + clips8(in2_rq2); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq2 it1 after requant: %d\nclipped in1_rq2: %d\n", + core_id, in1_rq2, clips8(in1_rq2)); + printf("core %d - in2_rq2 it1 after requant: %d\nclipped in2_rq2: %d\n", + core_id, in2_rq2), + clips8(in2_rq2); + printf("core %d - sum2: %d\n", core_id, sum2); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it2 before requant: %d\n", core_id, + *(target1_ext + 2)); + printf("core %d - in2 it2 before requant: %d\n", core_id, + *(target2_ext + 2)); +#endif + in1_rq3 = ((*(target1_ext + 2)) * in1_mul + in1_add) >> in1_shift; + in2_rq3 = ((*(target2_ext + 2)) * in2_mul + in2_add) >> in2_shift; + sum3 = clips8(in1_rq3) + clips8(in2_rq3); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq3 it2 after requant: %d\nclipped in1_rq3: %d\n", + core_id, in1_rq3, clips8(in1_rq3)); + printf("core %d - in2_rq3 it2 after requant: %d\nclipped in2_rq3: %d\n", + core_id, in2_rq3), + clips8(in2_rq3); + printf("core %d - sum3: %d\n", core_id, sum3); +#endif +#ifdef ADD_VERBOSE + printf("core %d - in1 it3 before requant: %d\n", core_id, + *(target1_ext + 3)); + printf("core %d - in2 it3 before requant: %d\n", core_id, + *(target2_ext + 3)); +#endif + in1_rq4 = ((*(target1_ext + 3)) * in1_mul + in1_add) >> in1_shift; + in2_rq4 = ((*(target2_ext + 3)) * in2_mul + in2_add) >> in2_shift; + sum4 = clips8(in1_rq4) + clips8(in2_rq4); +#ifdef ADD_VERBOSE + printf("core %d - in1_rq4 it3 after requant: %d\nclipped in1_rq4: %d\n", + core_id, in1_rq4, clips8(in1_rq4)); + printf("core %d - in2_rq4 it3 after requant: %d\nclipped in2_rq4: %d\n", + core_id, in2_rq4), + clips8(in2_rq4); + printf("core %d - sum4: %d\n", core_id, sum4); +#endif + + if (out_requant_flag) { + sum1 = (sum1 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum1: %d\n", core_id, sum1); +#endif + sum2 = (sum2 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum2: %d\n", core_id, sum2); +#endif + sum3 = (sum3 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum3: %d\n", core_id, sum3); +#endif + sum4 = (sum4 * out_mul + out_add) >> out_shift; +#ifdef ADD_VERBOSE + printf("core %d - requantized sum4: %d\n", core_id, sum4); +#endif + } + out1 = clips8(sum1); +#ifdef ADD_VERBOSE + printf("core %d - out1 clipped: %d\n", core_id, out1); +#endif + out2 = clips8(sum2); +#ifdef ADD_VERBOSE + printf("core %d - out2 clipped: %d\n", core_id, out2); +#endif + out3 = clips8(sum3); +#ifdef ADD_VERBOSE + printf("core %d - out3 clipped: %d\n", core_id, out3); +#endif + out4 = clips8(sum4); +#ifdef ADD_VERBOSE + printf("core %d - out4 clipped: %d\n", core_id, out4); +#endif + + *pOutBuffer = (int8_t)out1; + pOutBuffer++; + *pOutBuffer = (int8_t)out2; + pOutBuffer++; + *pOutBuffer = (int8_t)out3; + pOutBuffer++; + *pOutBuffer = (int8_t)out4; + pOutBuffer++; + } + // SCHEREMO: Cleanup leftovers, not doing it with this codebase for sub-byte + // formats + for (int i = 0; i < (((stop - start) * ch_im_out_r * dim_im_in_x) % 4); i++) { + in1_rq1 = ((*(target1)) * in1_mul + in1_add) >> in1_shift; + in2_rq1 = ((*(target2)) * in2_mul + in2_add) >> in2_shift; + +// SCHEREMO: Maybe it's just LLVM, but unless I hack 3 non-unrolled nops in +// here, stuff fails +#pragma nounroll + for (int j = 0; j < 3; j++) { + asm volatile("nop" ::); + } + + target1++; + target2++; + sum1 = clips8(in1_rq1) + clips8(in2_rq1); + if (out_requant_flag) { + sum1 = (sum1 * out_mul + out_add) >> out_shift; + } + + out1 = clips8(sum1); + *pOutBuffer = (int8_t)out1; + pOutBuffer++; + } +} From 478d618fd67a1dd3b05a92ac82bc2583b3b70cff Mon Sep 17 00:00:00 2001 From: telbayad Date: Wed, 8 Jan 2025 19:07:55 +0100 Subject: [PATCH 11/11] apply Victor's comment --- Deeploy/Targets/Generic/TypeCheckers.py | 23 +++++++++++ Deeploy/Targets/PULPOpen/Bindings.py | 6 +-- Deeploy/Targets/Snitch/Bindings.py | 5 +-- Deeploy/Targets/Snitch/TypeCheckers.py | 53 ------------------------- 4 files changed, 28 insertions(+), 59 deletions(-) delete mode 100644 Deeploy/Targets/Snitch/TypeCheckers.py diff --git a/Deeploy/Targets/Generic/TypeCheckers.py b/Deeploy/Targets/Generic/TypeCheckers.py index e1de80e..475cd20 100644 --- a/Deeploy/Targets/Generic/TypeCheckers.py +++ b/Deeploy/Targets/Generic/TypeCheckers.py @@ -537,3 +537,26 @@ def _inferSignedness(self, inputs: List[VariableBuffer], return [True] else: return [False] + + +class RQAddChecker(SignPropTypeChecker): + + def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): + super().__init__(input_types, output_types) + + def _inferNumLevels(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[int]: + return [operatorRepresentation['rqsOut_n_levels']] + + def _inferSignedness(self, inputs: List[VariableBuffer], + operatorRepresentation: OperatorRepresentation) -> List[bool]: + return [bool(operatorRepresentation["rqsOut_signed"])] + + # Override this. This should compute the signednes of each output node of the Layer + def checkOutputType(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> bool: + outputTypeSigned = self.output_types[0].referencedType.typeMin < 0 + if operatorRepresentation['rqsOut_signed'] and outputTypeSigned: + return True + if (not operatorRepresentation['rqsOut_signed']) and (not outputTypeSigned): + return True + return False diff --git a/Deeploy/Targets/PULPOpen/Bindings.py b/Deeploy/Targets/PULPOpen/Bindings.py index c8c8e38..cb7515b 100644 --- a/Deeploy/Targets/PULPOpen/Bindings.py +++ b/Deeploy/Targets/PULPOpen/Bindings.py @@ -38,7 +38,7 @@ from Deeploy.FutureExtension.CodeTransformationPasses.FutureCodeTransformation import FutureGeneration from Deeploy.Targets.Generic.Templates import ConcatTemplate, RQSiGELUTemplate, iHardswishTemplate from Deeploy.Targets.Generic.TypeCheckers import ConcatChecker, GELUChecker, HardswishChecker, MatMulChecker, \ - MulChecker, ReduceMeanChecker, RQHardswishChecker, SliceChecker, SoftmaxChecker, TransposeChecker, \ + MulChecker, ReduceMeanChecker, RQAddChecker, RQHardswishChecker, SliceChecker, SoftmaxChecker, TransposeChecker, \ iLayerNormChecker from Deeploy.Targets.PULPOpen.CodeTransformationPasses.PULPClusterSynch import PULPSynchCoresPass from Deeploy.Targets.PULPOpen.CodeTransformationPasses.PULPClusterTiling import PULPClusterTiling @@ -48,7 +48,7 @@ MulTemplate, ReduceMeanTemplate, RequantShiftTemplate, RQAddTemplate, RQSiHardswishTemplate, SliceTemplate, \ TallGEMMTemplate, TransposeTemplate, UniformRequantShiftTemplate, iRMSNormTemplate, iSoftmaxTemplate from Deeploy.Targets.PULPOpen.TypeCheckers import PULPConvChecker, PULPLinearChecker, PULPMaxPoolChecker, \ - PULPRequantShiftChecker, PULPRQAddChecker + PULPRequantShiftChecker from Deeploy.TilingExtension.CodeTransformationPasses.TilingVariableReplacement import TilingVariableReplacement _clusterEntryClosureCallTemplate = NodeTemplate(""" @@ -156,7 +156,7 @@ ] PULPRQAddBindings = [ - NodeBinding(PULPRQAddChecker([PointerClass(_type), PointerClass(_type2)], [PointerClass(_type3)]), + NodeBinding(RQAddChecker([PointerClass(_type), PointerClass(_type2)], [PointerClass(_type3)]), RQAddTemplate.referenceTemplate, ForkTransformer) for _type in [int8_t, uint8_t] for _type2 in [int8_t, uint8_t] diff --git a/Deeploy/Targets/Snitch/Bindings.py b/Deeploy/Targets/Snitch/Bindings.py index b4cb92c..0b319b7 100644 --- a/Deeploy/Targets/Snitch/Bindings.py +++ b/Deeploy/Targets/Snitch/Bindings.py @@ -33,13 +33,12 @@ from Deeploy.DeeployTypes import CodeTransformation, NodeBinding from Deeploy.FutureExtension.CodeTransformationPasses.FutureCodeTransformation import FutureGeneration from Deeploy.Targets.Generic.Templates import iNoNormTemplate -from Deeploy.Targets.Generic.TypeCheckers import AddChecker, GEMMChecker, SoftmaxChecker, iNoNormChecker +from Deeploy.Targets.Generic.TypeCheckers import AddChecker, GEMMChecker, RQAddChecker, SoftmaxChecker, iNoNormChecker from Deeploy.Targets.Snitch.CodeTransformationPasses import SnitchClusterTiling, SnitchCoreFilterPass, \ SnitchProfileExecutionBlockPass, SnitchSynchCoresPass from Deeploy.Targets.Snitch.Templates import AddTemplate, RQAddTemplate, iSoftmaxTemplate from Deeploy.Targets.Snitch.Templates.GemmTemplate import SnitchGemm_Template from Deeploy.Targets.Snitch.Templates.RqGemmTemplate import SnitchRqGemm_Template -from Deeploy.Targets.Snitch.TypeCheckers import SnitchRQAddChecker from Deeploy.TilingExtension.CodeTransformationPasses.TilingVariableReplacement import TilingVariableReplacement TilingCallClosure = partial(ClosureGeneration, closureSuffix = "_tiling_closure") @@ -78,7 +77,7 @@ TiledTransformer) for _type in [int8_t] ] SnitchRQAddBindings = [ - NodeBinding(SnitchRQAddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(_type)]), + NodeBinding(RQAddChecker([PointerClass(_type), PointerClass(_type)], [PointerClass(_type)]), RQAddTemplate.referenceTemplate, TiledTransformer) for _type in [int8_t] ] SnitchAddBindings = [ diff --git a/Deeploy/Targets/Snitch/TypeCheckers.py b/Deeploy/Targets/Snitch/TypeCheckers.py deleted file mode 100644 index fa428af..0000000 --- a/Deeploy/Targets/Snitch/TypeCheckers.py +++ /dev/null @@ -1,53 +0,0 @@ -# ---------------------------------------------------------------------- -# -# File: SnitchCheckers.py -# -# Last edited: 07.06.2024 -# -# Copyright (C) 2024, ETH Zurich and University of Bologna. -# -# Author: -# - Victor Jung, jungvi@iis.ee.ethz.ch, ETH Zurich -# -# ---------------------------------------------------------------------- -# SPDX-License-Identifier: Apache-2.0 -# -# 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 -# -# -# 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 typing import List, Sequence, Type - -from Deeploy.AbstractDataTypes import Pointer -from Deeploy.CommonExtensions.TypeCheckers.SignPropTypeChecker import SignPropTypeChecker -from Deeploy.DeeployTypes import OperatorRepresentation, VariableBuffer - - -class SnitchRQAddChecker(SignPropTypeChecker): - - def __init__(self, input_types: Sequence[Type[Pointer]], output_types: Sequence[Type[Pointer]]): - super().__init__(input_types, output_types) - - def _inferNumLevels(self, inputs: List[VariableBuffer], - operatorRepresentation: OperatorRepresentation) -> List[int]: - return [operatorRepresentation['rqsOut_n_levels']] - - def _inferSignedness(self, inputs: List[VariableBuffer], - operatorRepresentation: OperatorRepresentation) -> List[bool]: - return [bool(operatorRepresentation["rqsOut_signed"])] - - # Override this. This should compute the signednes of each output node of the Layer - def checkOutputType(self, inputs: List[VariableBuffer], operatorRepresentation: OperatorRepresentation) -> bool: - outputTypeSigned = self.output_types[0].referencedType.typeMin < 0 - if operatorRepresentation['rqsOut_signed'] and outputTypeSigned: - return True - if (not operatorRepresentation['rqsOut_signed']) and (not outputTypeSigned): - return True - return False