-
Notifications
You must be signed in to change notification settings - Fork 2.4k
SnippetS
This document describes the design and rationale for snippets code generator. Implementation of code functionality is located here. Proposal for CPU backend integration is here.
We believe that core CNN operators (convolution, gemm, fully connected) are limited by compute, the rest is memory bound. Math approximations (like transcendental functions) are rare in emerging workloads and could be treated with the same machinery. Snippets are designed to optimize topology for memory, while leaving compute intensive kernels for backend developers.
We believe potential speedup is proportional to shrink in memory-walked bytes. So we can transform the problem to a task to optimize for memory walks, whatever pattern snippet has and operations it contains. Number of memory walks should be less or equal to handcrafted optimizations. This guarantees performance improvements over the previous approach (excluding corner cases caused by cache effects). Shrinkage factor might be encoded to some cost function in future evolution of code generator. Snippets generator provides diagnostics to estimate this shrinkage factor with ngraph::snippets::op::Subgraph::print_statistics(bool verbose)
member.
We design SnippetS generator for back-end developers. The main purpose of inventing snippets code generator is an operator fusion, register allocation and target kernel generation decomposition. This allows modifications (like new fusion support) and feature extensions (like new operation support) to be done in a single point of modification and avoid combinatorial explosion for fusions/types/architectures etc.
We believe that creating a full-fledged compiler or usage of existing compiler infrastructure (like LLVM & MLIR) is superfluous at this point of evelition. We aim to provide a flexible and performant framework for operation fusions, leaving micro optimizations (e.g. instruction scheduling) to the backend H/W.
We do not aim to invent a DSL for SnippetS and would like to keep it this way. DSL gives users more flexibility to express uncommon operations. However, the shift towards an approach to encode topologies with elementary operations followed by smart enough fusions is already expressive and performant enough.
Snippet is a compiled compute kernel generated from a subgraph using SnippetS code generator for specific architecture with a scheduling domain. Using this scheduling domain and calling convention backend can execute generated compute kernels. For the first generation, snippets are statically scheduled towards the output domain. Multi-output snippets are supported if all outputs are broadcast-compatible in a sense that domains for all outputs can be broadcasted from one root domain which defines snippet schedule. It’s a subject of extension for future generations.
We use nGraph as the highest level IR for subgraph representation and lowering transformations. Opset1 is a base operation set for code generation. We aim to keep the minimal possible and sufficient operation set (or ISA) and keep it RISC-like (memory and compute decomposed).
One subgraph corresponds to one snippet. Operations which cannot be scheduled by a single schedule should not be placed in the same subgraph. Snippet somewhat conceptually close to OpenCL kernel without a restriction to express only embarrassingly parallel tasks. Subgraph once extracted from full topology IR is treated as an operation and data flow descriptor in scalar notation (similar to OpenCL/CUDA). Tensor sizes are used for defining scheduling domain and detecting broadcasts/reductions.
We split operations into 3 groups: layout-oblivious (LOO), layout-aware(-tolerant) and layout-dependent. Layout-oblivious operation semantics and implementation are completely agnostic to a specific layout in which tensors are placed in memory. For example, elements-wise math and ReLU does in this category. Implementation layout-aware operation depends on the layout of input/output tensors. For example, convolutions and other block-wise kernels or layout repaks. For layout-specific operation semantics and implementation depends on the layout. For example, the Yolo region. Patterns to fuse constructed in terms of taxonomy above.
Code generation is split into 2 phases, tokenization and lowering.
Tokenization runs on full topology nGraph function inside a specific plugin in a stage of common transformations. Input of tokenization is a topology graph. Output is a modified topology graph with ngraph::snippets::op::Subgraph
operations installed. Each subgraph contains nGraph function (called body) which holds a part of original topology legal for snippet generation (can be scheduled with a single schedule)
Procedure of finding subgraphs suitable for code generation is called tokenization, meaning that we split the topology tree into subgraphs in the same greedy approach which is used for parsing input stream of characters into the tokens. It also could be seen as and modified into a basic block construction problem, since we also find a leader and potentially terminators. Implementation can be found here.
Tokenization has an advantage over the pattern matching approach (used in traditional and MLIR-based compilers) since it can handle arbitrary patterns of operations. Pattern matching deduces specific configuration of operations to translate to another one, more suitable for target machine or further lowering. This means that relations between operations are fixed. Tokenization on the other hand has the only limitation on specific operation types which are suitable and profitable to fuse with respect to original topology correctness (keeping it as a direct acyclic graph).
The extracted body comes to a plug-in wrapped as a composite Subgraph
operation which is seen as a block box from a plugin standpoint and can participate in any plugin specific subroutines (e.g. layout assignment, memory allocation, etc.).
Subgraph accepts arbitrary numbers of inputs and outputs. There is 1:1 mapping for external (subgraph node’s) and internal (body) parameters indexes.
Pattern here is an exact subgraph configuration (nodes and edges between them). The first generation of snippets supports only layout-oblivious operations which may have broadcast on inputs and broadcast-compatible outputs. For example Shapes <1, 42, 17, 31>
, <1, 42, 17, 1>
and <1, 42, 1, 31>
are considered as broadcast-compatible. Layout-oblivious operation with multiple outputs as a snippet leader and forms a new subgraph. The most beneficial patterns are subgraphs with complex control flow but minimal number of inputs/and outputs. For example, GeLU has a 5x shrinkage factor from original unfused subgraph in number of bytes walked. Subgraph below could be considered as an example of such a subgraph. Leader detection procedure aims to find such subgraphs.
...
|
Add
|
+-------+-------+
| |
| Add
| |
| Clamp
| |
+---Multiply----+
|
...
Operations are greedily added to the subgraph until
- New operation doesn’t introduce a loop in a topology function.
- Number of inputs and outputs satisfies target criteria.
- Operation is not a predecessor of topology output.
- Resulting subgraph can be scheduled (all outputs are broadcast-compatible).
If a potential subgraph doesn’t meet any of criteria above, the procedure continues to find a new leader.
Lowering is a sequence of subgraph (snippet body) traversal passes to generate a compute kernel out of subgraphs of operations extracted by tokenization.
- Common optimizations
- Canonicalization
- Domain normalization
- Conversion to snippets dialect
- Target-specific optimizations
- Register allocation
- Schedule generation
- Target code emission
Constants are treated as inputs for a subgraph with an exception for scalar cases (since we don’t need to schedule them). snippets::op::Scalar
is used to represent this kind of constants.
If such Scalar comes as a second input of Power operation, it’s replaced with snippets::op::PowerStatic
.
The goal of this step is to apply target independent and schedule related optimizations and to make snippet schedulable.
All input and output shapes are normalized to 6D for future schedule generation. If shape propagation fails or leads to inconsistent output shapes an exception is raised.
Layout assigned by user code and passed to a generate
function is propagated through subgraph on this step as well. Layout is passed to a generate function as a BlockedShapeVector
which is a std::vector<BlockedShape>
, while BlockedShape
is std::tuple<ngraph::Shape, ngraph::AxisVector, ngraph::element::Type>
. For example, if backend supports NCHW16c
layout and tensor has size of <1, 42, 17, 31>
and hold single precision floating point this structure should be std::make_tuple(ngraph::Shape {1, 3, 17, 31, 16}, ngraph::AxisVector {0, 1, 2, 3, 1}, ngraph::element::f32);
. This allows generic layout representation.
The goal for this step is to transform a subgraph (body function) into a form possible to code generation. Input for this step is subgraph in a canonical form output is a subgraph in snippets dialect.
Snippet or kernel is formed around the subgraph body in a sequence of traversal steps. Let’s walk through these steps with the smallest possible subgraph which contains out of single [Add]
operation.
While we extract subgraphs with the tokenization part we explicitly insert Parameters and Results to its body to form a complete nGraph Function.
Parameter Parameter
| |
| |
+------Add------+
|
|
Result
This function represents operation dependencies in scalar (similar to OpenCL) notation while shapes of tensors are used to generate schedules. At this point kernel-schedule decomposition is made (similar to Halide/OpenCL/TVM)
As a next step explicit memory operations are placed for each input and output. InsertLoad
and InsertStore
passes derived from MatcherPass
.
Parameter Parameter
| |
| |
Load Load
| |
| |
+------Add------+
|
|
Store
|
|
Result
By default, memory operations assumes vector memory access, if scalar access is needed special passes ReplaceLoadsWithScalarLoads
and ReplaceStoresWithScalarStores
should be executed.
For each operation in body function inputs are checked against broadcasting. In case of parameters to be broadcasted explicit broadcast operation is generated. For example, if for the subgraph above we have <1, 42, 17, 31>
and <1, 42, 17, 1>
resulting subgraph is going to be
Parameter <1, 42, 17, 31> Parameter <1, 42, 17, 1>
| |
| |
Load <1, 42, 17, 31> Load <1, 42, 17, 1>
| |
| |
| BroadcastMove <1, 42, 17, 31>
| |
| |
+--------------Add--------------+
|
|
Store <1, 42, 17, 31>
|
|
Result <1, 42, 17, 31>
If load followed by broadcast is detected then this pair is replaced by a single Broadcast load instruction. Like the following
Parameter Parameter
| |
| |
Load BroadcastLoad
| |
| |
+------Add------+
|
|
Store
|
|
Result
Broadcast and regular streaming vector load is possible from the same pointer. Broadcast load should always go before streaming load. Broadcast load for non the most varying dimension is not generated, however it affects the generated schedule.
Target developers can plug in to the code generation pipeline some specific optimizations with passing ngraph::pass::Manager
into generate
function of subgraph
. Passes are executed on subgraph in canonical form converted to a snippet dialect.
It might be also extended to provide an interface for target independent optimizations in future
Canonicalized subgraph in a snippets dialect forms a basic block or region inside a snippet (kernel). Registers are allocated globally for the whole subgraph. Since all operations for a subgraph are assumed to be vector, only vector registers are allocated for the first generation of SnippetS. Linear scan register allocation algorithm is used. Register allocator is implemented as a function pass ngraph::snippets::pass::AssignRegisters
and store allocated registers for each node into rt_info
. rt_info
for a node holds a register for Node's output. However, this part should be refactored batter, either to become target independent or use target specific abstraction to acquire a new register
The goal of this step is to transform subgraphs in a scalar notation into kernel functions callable from user code. Kernel
and Tile
operations are introduced for this purpose. Each of this operation has a constructor from code region described as a collection of operation and operands pairs Kernel(const std::vector<std::pair<std::shared_ptr<ngraph::snippets::Emitter>, ngraph::snippets::RegInfo>>& region);
.
If we return to example above this comes to a following hierarchical IR. If we limit scope to layout oblivious operations with broadcasting support, tile could be generated as a single loop over the most warning dimension. The second Tile
is generated to handle tails and can be omitted if not needed. Special pass replaces memory operations on vector to scalar versions for tail subgraph.
+--------------------------------------------------+
| |
| Kernel |
| +-------------------------------------------+ |
| | | |
| | Tile | |
| | | |
| | Parameter Parameter | |
| | | | | |
| | | | | |
| | Load Load | |
| | | | | |
| | | | | |
| | +-----Add-----+ | |
| | | | |
| | | | |
| | Store | |
| | | | |
| | | | |
| | Result | |
| | | |
| +-------------------------------------------+ |
| |
| +-------------------------------------------+ |
| | | |
| | Tile | |
| | | |
| | Parameter Parameter | |
| | | | | |
| | | | | |
| | ScalarLoad ScalarLoad | |
| | | | | |
| | | | | |
| | +-----Add-----+ | |
| | | | |
| | | | |
| | ScalarStore | |
| | | | |
| | | | |
| | Result | |
| | | |
| +-------------------------------------------+ |
| |
+--------------------------------------------------+
+--------------------------------------------------+
| |
| Data |
| |
+--------------------------------------------------+
Where
-
Kernel
constants a collection of the tiles, corresponds to a Subgraph node and responsible for function signature generation, calls generators for all tiles and data sections -
Tile
contains single subgraph body, vector or scalar -
Data
corresponds to data section aggregated for all nodes in all Tile’s subgraphs
Target code emission is table based. Target is responsible for filling jitters
table field in Generator
class.
std::map<const ngraph::DiscreteTypeInfo, std::function<std::shared_ptr<Emitter>(std::shared_ptr<ngraph::Node>)>> jitters;
An OpenVINO plugin is treated as a target for snippets.
Each nGraph node is mapped to a convertor function which creates Emitter
form this node. Each specific emitter should extend from Emitter
. It is used to map this node to target code and has emit_code
and emit_data
methods. emit_data
is used during data section generation. All operations from snippets dialect which are legal for code generation should be expressed as operations derived from nGraph Op as well as Emitter derived snippets::Emitter class which knows how to translate this Op to Target specific ISA. (ex. xbyak is a jit backend for CPU plugin).
For minimal code generator support target should provide emitters for the following operations
Kernel
Tile
Data
Load
ScalarLoad
BroadcastLoad
Store
ScalarStore
Once a schedule is generated, target code is emitted from a kernel in Generator::generate method by executing Kernel::emit_code function. Since Kernel and Tile represents hierarchical
Target can potentially extend snippets dialect with target specific operation for code emission. It should implement:
- nGraph operation (ex.
class FMA : public ngraph::op::Op
) - Emitter for this operation (ex.
class FmaEmitter : public Emitter
) - register this pair in
jitters
map
Parameters for a generated snippet are split into schedule-invariant and schedule-dependent. Schedule-invariant parameters include pointers to input/output tensors and strides for each of them with the same rank as scheduling domain.
Subgraph can be executed with nGraph references if no generator is present.
© Copyright 2018-2024, OpenVINO team
- Home
- General resources
- How to build
-
Developer documentation
- Inference Engine architecture
- CPU plugin
- GPU plugin
- HETERO plugin architecture
- Snippets
- Sample for IE C++/C/Python API
- Proxy plugin (Concept)
- Tests