Unity Barracuda
Unity Barracuda is a lightweight and cross-platform Neural Net inference library for Unity. Barracuda can run Neural Nets both on GPU and CPU. Currently Barracuda is in the preview development stage, so adventures are expected.
Getting Unity Barracuda
- via
Unity Package Manager
: open Package Manager window in Unity Editor, enablepreview
packages, select Barracuda and install it. - from
GitHub
: edit your Unity Project'sPackages/manifest.json
and add dependency to the Barracuda GitHub repo:"com.unity.barracuda" : "https://github.com/Unity-Technologies/barracuda-release.git"
Using Barracuda
Typically the following steps are needed to use Barracuda in application:
- add .onnx file to your project - it will behave like any other regular asset,
- load model from the asset,
- create inference engine (the worker),
- execute model and
- fetch results.
You can import ONNX models simply by adding .onnx file directly to your project, however Tensorflow models require additional attention by running python script for now. See Converting Tensorflow models to Barracuda format paragraph below for more information.
Load Model into Barracuda
To load ONNX model, first add its .onnx file to your project. It will be imported and show up as an asset of type NNModel
.
Next, add public NNModel
field to your C# script and assign reference to your asset via editor UI. Load model with ModelLoader
:
public NNModel modelSource;
<..>
var model = ModelLoader.Load(modelSource);
To load Tensorflow models that were manually converted with tensorflow_to_barracuda.py
python script, use the following code:
var model = ModelLoader.LoadFromStreamingAssets(modelName + ".nn");
Create inference engine (Worker)
Inference engine in Barracuda is called Worker. Worker is responsible for breaking down model into executable tasks and scheduling them on GPU or CPU.
var worker = BarracudaWorkerFactory.CreateWorker(BarracudaWorkerFactory.Type.ComputePrecompiled, model)
Execute the model
Inputs can be provided both as sole Tensor
object (assuming Model has only one input) or as a dictionary of name and Tensor
pairs.
var inputs = new Dictionary<string, Tensor>();
inputs[name1] = new Tensor(...);
inputs[name2] = new Tensor(...);
worker.Execute(inputs);
Execution is asynchronous for GPU backends. Currently implementation is synchronous for CPU backends, however it is good to assume that execution will be async for all backends in the future.
Fetch outputs
If model has only single output, then simple worker.Peek()
can be used, otherwise output names should be provided.
var O = worker.Peek(outputName);
Note: worker.Peek()
does not transfer ownership of the tensor to you and tensor will still be owned by the worker
. Calling worker.Peek()
is preferable way and allows to reduce memory allocations. However, if you expect to use tensor for longer time, call worker.Fetch()
- otherwise tensor values will be lost after the next call to worker.Execute()
or after call to worker.Dispose()
Cleanup
As a Barracuda client you are responsible to Dispose
worker, inputs and outputs you created, received via worker.Fetch()
or taken ownership by calling tensor.TakeOwnership()
. This is necessary to properly free GPU resources.
O.Dispose();
worker.Dispose();
Note: It is not necessary to Dispose
tensor that you received via worker.Peek()
call.
Working with data
Tensor
Tensor values in Barracuda are accessed via batch
,height
,width
,channels
layout also known as NHWC or channels-last. You can interact with Tensor
data via multi-dimensional array operators:
var tensor = new Tensor(batchCount, height, width, channelCount);
tensor[n, y, x, c] = 1.0f; // as N batches of 3 dimensional data: N x {X, Y, C}
tensor[n, c] = 2.0f; // as N batches of 1 dimensional data: N x {C}
tensor[ i] = 3.0f; // as flat array
There are number of Tensor
constructors that cover a variety of scenarios. By default tensors are initialized with 0
upon construction, unless initialization Array
is provided.
tensor = new Tensor(batchCount, height, width, channelCount); // batch of 3 dimensional data, 0 initialized: batchCount x {height, width, channelCount}
tensor = new Tensor(batchCount, elementCount); // batch of 1 dimensional data, 0 initialized: batchCount x {elementCount}
var stridedArray = new float[batchCount * elementCount] { ... };
tensor = new Tensor(batchCount, elementCount, stridedArray); // batch of 1 dimensional data, initialized from strided array
var jaggedArray = new float[batchCount][elementCount] { ... };
tensor = new Tensor(batchCount, elementCount, jaggedArray); // batch of 1 dimensional data, initialized from jagged array
Texture2D texture = ...;
tensor = new Tensor(texture); // tensor initialized with texture data: 1 x { texture.width, texture.height, 3}
You can query shape of the Tensor
object, but you can not change it. Shape of the Tensor
is immutable. If you want to have different shape of Tensor
, you have to construct a new instance of Tensor
object.
var shape = tensor.shape;
Debug.Log(shape + " or " + shape.batch + shape.height + shape.width + shape.channels);
Texture as input
You can directly pass Texture2D
, Texture2DArray
, Texture3D
or RenderTexture
to Barracuda without accessing individual pixels on CPU:
var channelCount = 3; // you can treat input pixels as 1 (grayscale), 3 (color) or 4 (color with alpha) channels
var tensor = new Tensor(texture, channelCount);
You can batch multiple textures into the single Tensor
object:
var textures = new [] { texture0, texture1, texture2, texture3 }; // these textures will form a batch
var tensor = new Tensor(textures, channelCount);
Note that to form a batch all textures must have the same width and height dimensions.
Texture as output
If you want to use Barracuda execution results further in the graphics pipeline, you can copy data from Tensor
into RenderTexture
without stalling CPU or GPU:
var tensor = worker.Peek();
var texture = BarracudaTextureUtils.TensorToRenderTexture(tensor);
If you wish, you can reuse the same RenderTexture
multiple times:
var texture = new RenderTexture(width, height, 0);
// ...
var tensor = worker.Peek();
BarracudaTextureUtils.TensorToRenderTexture(tensor, texture);
Introspecting Barracuda models
Barracuda model has very simple memory representation. Once model is loaded you can query for inputs and outputs:
string[] inputNames = model.inputs; // query model inputs
string[] outputNames = model.outputs; // query model outputs
Or you can directly iterate through the layers and investigate what model is going to do:
foreach (var layer in model.layers)
Debug.Log(layer.name + " does " + layer.type);
Verbose mode
You can turn on verbose mode for different parts of Barracuda:
bool verbose = true;
var model = ModelLoader.LoadModel(onnxAsset, verbose); // verbose loader
var worker = BarracudaWorkerFactory.CreateWorker(BarracudaWorkerFactory.Type.ComputePrecompiled, model, verbose); // verbose execution
Converting TensorFlow models to Barracuda format
Barracuda comes with dedicated python scripts to convert pre-trained constant .pb
TensorFlow graph.
To produce a constant graph from a trained model:
- In TF1.x use freeze_graph. Example from a
SavedModel
(more examples).
from tensorflow.python.tools import freeze_graph
freeze_graph.freeze_graph(None, None, <input_is_binary>, None, None
<output_name>, None, None, <output_path>, False,
clear_devices, None, None, None, False, False,
<saved_model_dir>, tag_constants.SERVING)
- In TF2.x use convert_to_constants. Example from a Keras
Model
(more examples):
from tensorflow.python.framework import convert_to_constants
@tf.function(input_signature=[tf.TensorSpec(shape=[<input_shape>], dtype=tf.float32)])
def to_save(x):
return model(x)
f = to_save.get_concrete_function()
constantGraph = convert_to_constants.convert_variables_to_constants_v2(f)
tf.io.write_graph(constantGraph.graph.as_graph_def(), <output_dir>, <output_file>)
Convert constant graph to Barracuda:
python tensorflow_to_barracuda.py Models/3DBall-tf-model.pb Destination/3DBall-bc.nn
There is legacy converter from ONNX:
python onnx_to_barracuda.py Models/mnist/model.onnx Destination/mnist-bc.nn
If network has multiple outputs, but you need only particular ones during the inference, there is an optional -trim
flag to remove unused outputs and calculations.
For example:
python tensorflow_to_barracuda.py Models/3DBall-tf-model.pb Destination/3DBall-bc.bytes -trim action$
First, trim will remove outputs from the graph that do not match regexp pattern. Second trim will strip all nodes that do not participate in the evaluation of the output.
In the example above only outputs that end with action
will be left.
You could pass --print-supported-ops
to get approximate list of supported operations/activations for specific converter.
P.S. Python 3.5 or 3.6 is recommended P.P.S. We plan to migrate Tensorflow converter from Python to C# in the future.
Approximate list of ONNX operations supported by Barracuda
Operations
Add
Sum
Sub
Mul
Div
Pow
Min
Max
Mean
AveragePool
MaxPool
GlobalAveragePool
GlobalMaxPool
Upsample
Gemm
MatMul
Conv
ConvTranspose
BatchNormalization
InstanceNormalization
Greater
Less
Equal
Or
And
Not
Xor
Pad
Constant
Identity
Cast
Dropout
Reshape
Unsqueeze
Squeeze
Flatten
Concat
Slice
P.S. some of these operations are under limited support
Activations
Softmax
Tanh
Sigmoid
Elu
LeakyRelu
Selu
Approximate list of TensorFlow nodes supported by Barracuda script converter
Operations
Add
AvgPool
BatchNormalization
BatchNormalizationRuntime
BiasAdd
Ceil
Concat
Conv2D
Conv2DBackpropInput
Dense
DepthwiseConv2dNative
Exp
Flatten
Floor
FusedBatchNorm
GlobalAveragePool
GlobalAvgPool
InstanceNormalization
MatMul
Max
MaxPool
Maximum
Mean
Min
Minimum
Mul
Multinomial
Nop
Neg
OneHot
Pad
Pow
Prod
RandomStandardNormal
RandomUniform
RealDiv
Reshape
ResizeBicubic
ResizeBilinear
ResizeNearestNeighbor
StridedSlice
Sqrt
Sub
Sum
Activations
Elu
LeakyRelu
Linear
Log
LogSoftmax
Relu
Relu6
Selu
Sigmoid
Softmax
Softplus
Softsign
Swish
P.S. some of these nodes are under limited support
Reporting issues
If you encounter issues running Barracuda in your Unity project, please report them on our GitHub repo.