blob: 3b71e5fc7d4d7925aec32b98115fa4c5c193e5e9 [file] [log] [blame]
Mike Kelly8c1701a2019-02-11 17:01:27 +00001//
2// Copyright © 2017 Arm Ltd. All rights reserved.
3// SPDX-License-Identifier: MIT
4//
5
6#include "Serializer.hpp"
Saoirse Stewart3166c3e2019-02-18 15:24:53 +00007
8#include "SerializerUtils.hpp"
9
Mike Kelly8c1701a2019-02-11 17:01:27 +000010#include <armnn/ArmNN.hpp>
Saoirse Stewart3166c3e2019-02-18 15:24:53 +000011
Mike Kelly8c1701a2019-02-11 17:01:27 +000012#include <iostream>
Saoirse Stewart3166c3e2019-02-18 15:24:53 +000013
Matthew Bentham268509a2019-02-25 13:58:24 +000014#include <ArmnnSchema_generated.h>
Saoirse Stewart3166c3e2019-02-18 15:24:53 +000015
Mike Kelly8c1701a2019-02-11 17:01:27 +000016#include <flatbuffers/util.h>
17
18using namespace armnn;
19namespace fb = flatbuffers;
Derek Lamberti0028d1b2019-02-20 13:57:42 +000020namespace serializer = armnnSerializer;
Mike Kelly8c1701a2019-02-11 17:01:27 +000021
22namespace armnnSerializer
23{
24
Mike Kellyaf484012019-02-20 16:53:11 +000025serializer::ActivationFunction GetFlatBufferActivationFunction(armnn::ActivationFunction function)
26{
27 switch (function)
28 {
29 case armnn::ActivationFunction::Sigmoid:
30 return serializer::ActivationFunction::ActivationFunction_Sigmoid;
31 case armnn::ActivationFunction::TanH:
32 return serializer::ActivationFunction::ActivationFunction_TanH;
33 case armnn::ActivationFunction::Linear:
34 return serializer::ActivationFunction::ActivationFunction_Linear;
35 case armnn::ActivationFunction::ReLu:
36 return serializer::ActivationFunction::ActivationFunction_ReLu;
37 case armnn::ActivationFunction::BoundedReLu:
38 return serializer::ActivationFunction::ActivationFunction_BoundedReLu;
39 case armnn::ActivationFunction::LeakyReLu:
40 return serializer::ActivationFunction::ActivationFunction_LeakyReLu;
41 case armnn::ActivationFunction::Abs:
42 return serializer::ActivationFunction::ActivationFunction_Abs;
43 case armnn::ActivationFunction::Sqrt:
44 return serializer::ActivationFunction::ActivationFunction_Sqrt;
45 case armnn::ActivationFunction::Square:
46 return serializer::ActivationFunction::ActivationFunction_Square;
47 default:
48 return serializer::ActivationFunction::ActivationFunction_Sigmoid;
49 }
50}
51
Saoirse Stewartcb8a3212019-02-14 15:46:10 +000052uint32_t SerializerVisitor::GetSerializedId(unsigned int guid)
53{
54 std::pair<unsigned int, uint32_t> guidPair(guid, m_layerId);
55
56 if (m_guidMap.empty())
57 {
58 m_guidMap.insert(guidPair);
59 }
60 else if (m_guidMap.find(guid) == m_guidMap.end())
61 {
62 guidPair.second = ++m_layerId;
63 m_guidMap.insert(guidPair);
64 return m_layerId;
65 }
Saoirse Stewart30211042019-02-18 17:19:16 +000066 return m_guidMap[guid];
Saoirse Stewartcb8a3212019-02-14 15:46:10 +000067}
68
Mike Kelly8c1701a2019-02-11 17:01:27 +000069// Build FlatBuffer for Input Layer
Derek Lamberti0028d1b2019-02-20 13:57:42 +000070void SerializerVisitor::VisitInputLayer(const armnn::IConnectableLayer* layer, LayerBindingId id, const char* name)
Mike Kelly8c1701a2019-02-11 17:01:27 +000071{
72 // Create FlatBuffer BaseLayer
73 auto flatBufferInputBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Input);
74
75 // Create FlatBuffer BindableBaseLayer
76 auto flatBufferInputBindableBaseLayer = serializer::CreateBindableLayerBase(m_flatBufferBuilder,
77 flatBufferInputBaseLayer,
78 id);
Mike Kelly8c1701a2019-02-11 17:01:27 +000079 // Push layer Guid to outputIds.
Saoirse Stewartcb8a3212019-02-14 15:46:10 +000080 m_inputIds.push_back(GetSerializedId(layer->GetGuid()));
Mike Kelly8c1701a2019-02-11 17:01:27 +000081
82 // Create the FlatBuffer InputLayer
83 auto flatBufferInputLayer = serializer::CreateInputLayer(m_flatBufferBuilder, flatBufferInputBindableBaseLayer);
84
85 // Add the AnyLayer to the FlatBufferLayers
86 CreateAnyLayer(flatBufferInputLayer.o, serializer::Layer::Layer_InputLayer);
87}
88
89// Build FlatBuffer for Output Layer
Derek Lamberti0028d1b2019-02-20 13:57:42 +000090void SerializerVisitor::VisitOutputLayer(const armnn::IConnectableLayer* layer, LayerBindingId id, const char* name)
Mike Kelly8c1701a2019-02-11 17:01:27 +000091{
92 // Create FlatBuffer BaseLayer
93 auto flatBufferOutputBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Output);
94
95 // Create FlatBuffer BindableBaseLayer
96 auto flatBufferOutputBindableBaseLayer = serializer::CreateBindableLayerBase(m_flatBufferBuilder,
97 flatBufferOutputBaseLayer,
98 id);
99 // Push layer Guid to outputIds.
Saoirse Stewartcb8a3212019-02-14 15:46:10 +0000100 m_outputIds.push_back(GetSerializedId(layer->GetGuid()));
Mike Kelly8c1701a2019-02-11 17:01:27 +0000101
102 // Create the FlatBuffer OutputLayer
103 auto flatBufferOutputLayer = serializer::CreateOutputLayer(m_flatBufferBuilder, flatBufferOutputBindableBaseLayer);
104 // Add the AnyLayer to the FlatBufferLayers
105 CreateAnyLayer(flatBufferOutputLayer.o, serializer::Layer::Layer_OutputLayer);
106}
107
Mike Kellyaf484012019-02-20 16:53:11 +0000108// Build FlatBuffer for Activation Layer
109void SerializerVisitor::VisitActivationLayer(const armnn::IConnectableLayer* layer,
110 const armnn::ActivationDescriptor& descriptor,
111 const char* name)
112{
113 // Create FlatBuffer BaseLayer
114 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Activation);
115
116 // Create the FlatBuffer ActivationDescriptor
117 auto flatBufferDescriptor = CreateActivationDescriptor(m_flatBufferBuilder,
118 GetFlatBufferActivationFunction(descriptor.m_Function),
119 descriptor.m_A,
120 descriptor.m_B);
121
122 // Create the FlatBuffer ActivationLayer
123 auto flatBufferAdditionLayer = CreateActivationLayer(m_flatBufferBuilder,
124 flatBufferBaseLayer,
125 flatBufferDescriptor);
126
127 // Add the AnyLayer to the FlatBufferLayers
128 CreateAnyLayer(flatBufferAdditionLayer.o, serializer::Layer::Layer_ActivationLayer);
129}
130
Mike Kelly8c1701a2019-02-11 17:01:27 +0000131// Build FlatBuffer for Addition Layer
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000132void SerializerVisitor::VisitAdditionLayer(const armnn::IConnectableLayer* layer, const char* name)
Mike Kelly8c1701a2019-02-11 17:01:27 +0000133{
134 // Create FlatBuffer BaseLayer
135 auto flatBufferAdditionBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Addition);
136
137 // Create the FlatBuffer AdditionLayer
138 auto flatBufferAdditionLayer = serializer::CreateAdditionLayer(m_flatBufferBuilder, flatBufferAdditionBaseLayer);
139
140 // Add the AnyLayer to the FlatBufferLayers
141 CreateAnyLayer(flatBufferAdditionLayer.o, serializer::Layer::Layer_AdditionLayer);
142}
143
Nattapat Chaimanowong6b4ed982019-02-26 17:24:13 +0000144// Build FlatBuffer for BatchToSpaceNd Layer
145void SerializerVisitor::VisitBatchToSpaceNdLayer(const armnn::IConnectableLayer* layer,
146 const armnn::BatchToSpaceNdDescriptor& descriptor,
147 const char* name)
148{
149 // Create FlatBuffer BaseLayer
150 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_BatchToSpaceNd);
151
152 std::vector<unsigned int> crops;
153 crops.reserve(descriptor.m_Crops.size() * 2);
154 for (auto& crop : descriptor.m_Crops)
155 {
156 crops.push_back(crop.first);
157 crops.push_back(crop.second);
158 }
159
160 auto flatBufferDescriptor =
161 CreateBatchToSpaceNdDescriptor(m_flatBufferBuilder,
162 m_flatBufferBuilder.CreateVector(descriptor.m_BlockShape),
163 m_flatBufferBuilder.CreateVector(crops),
164 GetFlatBufferDataLayout(descriptor.m_DataLayout));
165
166 auto flatBufferLayer = serializer::CreateBatchToSpaceNdLayer(m_flatBufferBuilder,
167 flatBufferBaseLayer,
168 flatBufferDescriptor);
169
170 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_BatchToSpaceNdLayer);
171}
172
ruoyan018e7fa232019-02-28 15:09:07 +0000173void SerializerVisitor::VisitBatchNormalizationLayer(const armnn::IConnectableLayer* layer,
174 const armnn::BatchNormalizationDescriptor& batchNormDescriptor,
175 const armnn::ConstTensor& mean,
176 const armnn::ConstTensor& variance,
177 const armnn::ConstTensor& beta,
178 const armnn::ConstTensor& gamma,
179 const char* name)
180{
181 auto fbBatchNormalizationBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_BatchNormalization);
182 auto fbBatchNormalizationDescriptor = serializer::CreateBatchNormalizationDescriptor(
183 m_flatBufferBuilder,
184 batchNormDescriptor.m_Eps,
185 GetFlatBufferDataLayout(batchNormDescriptor.m_DataLayout));
186
187 auto fbMeanConstTensorInfo = CreateConstTensorInfo(mean);
188 auto fbVarianceConstTensorInfo = CreateConstTensorInfo(variance);
189 auto fbBetaConstTensorInfo = CreateConstTensorInfo(beta);
190 auto fbGammaConstTensorInfo = CreateConstTensorInfo(gamma);
191 auto fbBatchNormalizationLayer = serializer::CreateBatchNormalizationLayer(m_flatBufferBuilder,
192 fbBatchNormalizationBaseLayer,
193 fbBatchNormalizationDescriptor,
194 fbMeanConstTensorInfo,
195 fbVarianceConstTensorInfo,
196 fbBetaConstTensorInfo,
197 fbGammaConstTensorInfo);
198
199 CreateAnyLayer(fbBatchNormalizationLayer.o, serializer::Layer::Layer_BatchNormalizationLayer);
200}
201
Conor Kennedy76277882019-02-26 08:29:54 +0000202// Build FlatBuffer for Constant Layer
203void SerializerVisitor::VisitConstantLayer(const armnn::IConnectableLayer* layer,
204 const armnn::ConstTensor& input,
205 const char* name)
206{
207 // Create FlatBuffer BaseLayer
208 auto flatBufferConstantBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Constant);
209
210 auto flatBufferConstTensorInfo = CreateConstTensorInfo(input);
211
212 // Create the FlatBuffer ConstantLayer
213 auto flatBufferLayer = CreateConstantLayer(m_flatBufferBuilder,
214 flatBufferConstantBaseLayer,
215 flatBufferConstTensorInfo);
216
217 // Add the AnyLayer to the FlatBufferLayers
218 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_ConstantLayer);
219}
220
Mike Kellya0766c32019-02-19 17:22:07 +0000221// Build FlatBuffer for Convolution2dLayer
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000222void SerializerVisitor::VisitConvolution2dLayer(const armnn::IConnectableLayer* layer,
223 const armnn::Convolution2dDescriptor& descriptor,
224 const armnn::ConstTensor& weights,
225 const armnn::Optional<armnn::ConstTensor>& biases,
Mike Kellya0766c32019-02-19 17:22:07 +0000226 const char* name)
227{
228 // Create FlatBuffer BaseLayer
229 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Convolution2d);
230
231 auto flatBufferDescriptor = CreateConvolution2dDescriptor(m_flatBufferBuilder,
232 descriptor.m_PadLeft,
233 descriptor.m_PadRight,
234 descriptor.m_PadTop,
235 descriptor.m_PadBottom,
236 descriptor.m_StrideX,
237 descriptor.m_StrideY,
238 descriptor.m_BiasEnabled,
239 GetFlatBufferDataLayout(descriptor.m_DataLayout));
240 auto flatBufferWeightsConstTensorInfo = CreateConstTensorInfo(weights);
241 flatbuffers::Offset<serializer::ConstTensor> flatBufferBiasesConstTensorInfo;
242
243 if (biases.has_value())
244 {
245 flatBufferBiasesConstTensorInfo = CreateConstTensorInfo(biases.value());
246 }
247
248 // Create the FlatBuffer Convolution2dLayer
249 auto flatBufferLayer = CreateConvolution2dLayer(m_flatBufferBuilder,
250 flatBufferBaseLayer,
251 flatBufferDescriptor,
252 flatBufferWeightsConstTensorInfo,
253 flatBufferBiasesConstTensorInfo);
254
255 // Add the AnyLayer to the FlatBufferLayers
256 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_Convolution2dLayer);
257}
258
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000259void SerializerVisitor::VisitDepthwiseConvolution2dLayer(const armnn::IConnectableLayer* layer,
260 const armnn::DepthwiseConvolution2dDescriptor& descriptor,
261 const armnn::ConstTensor& weights,
262 const armnn::Optional<armnn::ConstTensor>& biases,
Aron Virginas-Tarc04125f2019-02-19 16:31:08 +0000263 const char* name)
264{
265 auto fbBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_DepthwiseConvolution2d);
266 auto fbDescriptor = CreateDepthwiseConvolution2dDescriptor(m_flatBufferBuilder,
267 descriptor.m_PadLeft,
268 descriptor.m_PadRight,
269 descriptor.m_PadTop,
270 descriptor.m_PadBottom,
271 descriptor.m_StrideX,
272 descriptor.m_StrideY,
273 descriptor.m_BiasEnabled,
274 GetFlatBufferDataLayout(descriptor.m_DataLayout));
275
276 flatbuffers::Offset<serializer::ConstTensor> fbWeightsConstTensorInfo = CreateConstTensorInfo(weights);
277 flatbuffers::Offset<serializer::ConstTensor> fbBiasesConstTensorInfo;
278 if (biases.has_value())
279 {
280 fbBiasesConstTensorInfo = CreateConstTensorInfo(biases.value());
281 }
282
283 auto flatBufferLayer = CreateDepthwiseConvolution2dLayer(m_flatBufferBuilder,
284 fbBaseLayer,
285 fbDescriptor,
286 fbWeightsConstTensorInfo,
287 fbBiasesConstTensorInfo);
288
289 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_DepthwiseConvolution2dLayer);
290}
291
Éanna Ó Catháin58885892019-02-27 16:16:39 +0000292void SerializerVisitor::VisitDivisionLayer(const armnn::IConnectableLayer* layer, const char* name)
293{
Aron Virginas-Tar0fe32452019-02-28 13:12:47 +0000294 auto fbDivisionBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Division);
295 auto fbDivisionLayer = serializer::CreateDivisionLayer(m_flatBufferBuilder, fbDivisionBaseLayer);
Éanna Ó Catháin58885892019-02-27 16:16:39 +0000296
Aron Virginas-Tar0fe32452019-02-28 13:12:47 +0000297 CreateAnyLayer(fbDivisionLayer.o, serializer::Layer::Layer_DivisionLayer);
298}
Éanna Ó Catháin58885892019-02-27 16:16:39 +0000299
Aron Virginas-Tar377351e2019-02-27 14:42:31 +0000300void SerializerVisitor::VisitEqualLayer(const armnn::IConnectableLayer* layer, const char* name)
301{
302 auto fbBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Equal);
303 auto fbEqualLayer = serializer::CreateEqualLayer(m_flatBufferBuilder, fbBaseLayer);
304
305 CreateAnyLayer(fbEqualLayer.o, serializer::Layer::Layer_EqualLayer);
306}
307
Finn Williamsdd2ba7e2019-03-01 11:51:52 +0000308void SerializerVisitor::VisitFloorLayer(const armnn::IConnectableLayer *layer, const char *name)
309{
310 auto flatBufferFloorBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Floor);
311 auto flatBufferFloorLayer = serializer::CreateFloorLayer(m_flatBufferBuilder, flatBufferFloorBaseLayer);
312
313 CreateAnyLayer(flatBufferFloorLayer.o, serializer::Layer::Layer_FloorLayer);
314}
315
Saoirse Stewarta1ed73a2019-03-04 13:40:12 +0000316void SerializerVisitor::VisitGatherLayer(const armnn::IConnectableLayer* layer, const char* name)
317{
Matteo Martincighf81edaa2019-03-04 14:34:30 +0000318 auto fbGatherBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Gather);
319 auto flatBufferLayer = serializer::CreateGatherLayer(m_flatBufferBuilder, fbGatherBaseLayer);
Saoirse Stewarta1ed73a2019-03-04 13:40:12 +0000320
321 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_GatherLayer);
322}
323
Conor Kennedy79ffdf52019-03-01 14:24:54 +0000324void SerializerVisitor::VisitGreaterLayer(const armnn::IConnectableLayer* layer, const char* name)
325{
326 auto fbGreaterBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Greater);
Sadik Armaganac97c8c2019-03-04 17:44:21 +0000327 auto fbGreaterLayer = serializer::CreateGreaterLayer(m_flatBufferBuilder, fbGreaterBaseLayer);
Conor Kennedy79ffdf52019-03-01 14:24:54 +0000328
329 CreateAnyLayer(fbGreaterLayer.o, serializer::Layer::Layer_GreaterLayer);
330}
331
Sadik Armaganac97c8c2019-03-04 17:44:21 +0000332void SerializerVisitor::VisitMaximumLayer(const armnn::IConnectableLayer* layer, const char* name)
333{
334 auto fbMaximumBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Maximum);
335 auto fbMaximumLayer = serializer::CreateMaximumLayer(m_flatBufferBuilder, fbMaximumBaseLayer);
336
337 CreateAnyLayer(fbMaximumLayer.o, serializer::Layer::Layer_MaximumLayer);
338}
339
340void SerializerVisitor::VisitMeanLayer(const armnn::IConnectableLayer* layer,
341 const armnn::MeanDescriptor& descriptor,
342 const char* name)
343{
344 auto fbMeanBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Mean);
345 auto fbMeanDescriptor = serializer::CreateMeanDescriptor(m_flatBufferBuilder,
346 m_flatBufferBuilder.CreateVector(descriptor.m_Axis),
347 descriptor.m_KeepDims);
348
349 auto fbMeanLayer = serializer::CreateMeanLayer(m_flatBufferBuilder,
350 fbMeanBaseLayer,
351 fbMeanDescriptor);
352
353 CreateAnyLayer(fbMeanLayer.o, serializer::Layer::Layer_MeanLayer);
354}
355
356void SerializerVisitor::VisitMinimumLayer(const armnn::IConnectableLayer* layer, const char* name)
357{
358 auto fbMinimumBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Minimum);
359 auto fbMinimumLayer = serializer::CreateMinimumLayer(m_flatBufferBuilder, fbMinimumBaseLayer);
360
361 CreateAnyLayer(fbMinimumLayer.o, serializer::Layer::Layer_MinimumLayer);
362}
363
Jim Flynnac25a1b2019-02-28 10:40:49 +0000364void SerializerVisitor::VisitMergerLayer(const armnn::IConnectableLayer* layer,
365 const armnn::OriginsDescriptor& mergerDescriptor,
366 const char* name)
367{
368 auto flatBufferMergerBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Merger);
369
370 std::vector<flatbuffers::Offset<UintVector>> views;
371 for (unsigned int v = 0; v < mergerDescriptor.GetNumViews(); ++v)
372 {
373 const uint32_t* origin = mergerDescriptor.GetViewOrigin(v);
374 std::vector<uint32_t> origins;
375 for (unsigned int d = 0; d < mergerDescriptor.GetNumDimensions(); ++d)
376 {
377 origins.push_back(origin[d]);
378 }
379 auto view = m_flatBufferBuilder.CreateVector(origins);
380 auto uintVector = CreateUintVector(m_flatBufferBuilder, view);
381 views.push_back(uintVector);
382 }
383
384 auto flatBufferMergerDescriptor = CreateOriginsDescriptor(m_flatBufferBuilder,
385 mergerDescriptor.GetConcatAxis(),
386 mergerDescriptor.GetNumViews(),
387 mergerDescriptor.GetNumDimensions(),
388 m_flatBufferBuilder.CreateVector(views));
389
390 auto flatBufferLayer = CreateMergerLayer(m_flatBufferBuilder,
391 flatBufferMergerBaseLayer,
392 flatBufferMergerDescriptor);
393
394 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_MergerLayer);
395}
396
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000397void SerializerVisitor::VisitMultiplicationLayer(const armnn::IConnectableLayer* layer, const char* name)
Sadik Armagan5f450272019-02-12 14:31:45 +0000398{
Sadik Armaganac97c8c2019-03-04 17:44:21 +0000399 auto fbMultiplicationBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Multiplication);
400 auto fbMultiplicationLayer = serializer::CreateMultiplicationLayer(m_flatBufferBuilder,
401 fbMultiplicationBaseLayer);
Sadik Armagan5f450272019-02-12 14:31:45 +0000402
Sadik Armaganac97c8c2019-03-04 17:44:21 +0000403 CreateAnyLayer(fbMultiplicationLayer.o, serializer::Layer::Layer_MultiplicationLayer);
Sadik Armagan5f450272019-02-12 14:31:45 +0000404}
405
Nattapat Chaimanowongebb0f9c2019-03-01 12:14:06 +0000406void SerializerVisitor::VisitPadLayer(const armnn::IConnectableLayer* layer,
407 const armnn::PadDescriptor& padDescriptor,
408 const char* name)
409{
410 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Pad);
411
412 std::vector<unsigned int> padList;
413 for (auto& p: padDescriptor.m_PadList)
414 {
415 padList.push_back(p.first);
416 padList.push_back(p.second);
417 }
418
419 auto flatBufferPadDesc = serializer::CreatePadDescriptor(m_flatBufferBuilder,
420 m_flatBufferBuilder.CreateVector(padList));
421
422 auto flatBufferPadLayer = serializer::CreatePadLayer(m_flatBufferBuilder,
423 flatBufferBaseLayer,
424 flatBufferPadDesc);
425
426 CreateAnyLayer(flatBufferPadLayer.o, serializer::Layer::Layer_PadLayer);
427}
428
Nattapat Chaimanowong30b00202019-02-20 17:31:34 +0000429void SerializerVisitor::VisitPermuteLayer(const armnn::IConnectableLayer* layer,
430 const armnn::PermuteDescriptor& permuteDescriptor,
431 const char* name)
432{
433 // Create FlatBuffer BaseLayer
434 auto flatBufferPermuteBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Permute);
435
436 std::vector<unsigned int> dimMappings;
437 for (auto& v: permuteDescriptor.m_DimMappings)
438 {
439 dimMappings.push_back(v);
440 }
441
442 auto flatBufferPermuteDesc = serializer::CreatePermuteDescriptor(m_flatBufferBuilder,
443 m_flatBufferBuilder.CreateVector(dimMappings));
444
445 // Create the FlatBuffer PermuteLayer
446 auto flatBufferPermuteLayer = serializer::CreatePermuteLayer(m_flatBufferBuilder,
447 flatBufferPermuteBaseLayer,
448 flatBufferPermuteDesc);
449
450 // Add the AnyLayer to the FlatBufferLayers
451 CreateAnyLayer(flatBufferPermuteLayer.o, serializer::Layer::Layer_PermuteLayer);
452}
453
Saoirse Stewart263829c2019-02-19 15:54:14 +0000454// Build FlatBuffer for Reshape Layer
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000455void SerializerVisitor::VisitReshapeLayer(const armnn::IConnectableLayer* layer,
Saoirse Stewart263829c2019-02-19 15:54:14 +0000456 const armnn::ReshapeDescriptor& reshapeDescriptor,
457 const char* name)
458{
459 // Create FlatBuffer BaseLayer
460 auto flatBufferReshapeBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Reshape);
461
462 std::vector<unsigned int> targetShape;
463 for (unsigned int i =0; i < reshapeDescriptor.m_TargetShape.GetNumDimensions(); i++)
464 {
465 targetShape.push_back(reshapeDescriptor.m_TargetShape[i]);
466 }
467
468 auto flatBufferReshapeDesc = serializer::CreateReshapeDescriptor(m_flatBufferBuilder,
469 m_flatBufferBuilder.CreateVector(targetShape));
470
471 // Create the FlatBuffer ReshapeLayer
472 auto flatBufferReshapeLayer = serializer::CreateReshapeLayer(m_flatBufferBuilder, flatBufferReshapeBaseLayer,
473 flatBufferReshapeDesc);
474
475 // Add the AnyLayer to the FlatBufferLayers
476 CreateAnyLayer(flatBufferReshapeLayer.o, serializer::Layer::Layer_ReshapeLayer);
477}
478
Nattapat Chaimanowong6522cdc2019-03-01 16:14:13 +0000479void SerializerVisitor::VisitResizeBilinearLayer(const armnn::IConnectableLayer* layer,
480 const armnn::ResizeBilinearDescriptor& resizeDescriptor,
481 const char* name)
482{
483 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_ResizeBilinear);
484
485 auto flatBufferDescriptor =
486 CreateResizeBilinearDescriptor(m_flatBufferBuilder,
487 resizeDescriptor.m_TargetWidth,
488 resizeDescriptor.m_TargetHeight,
489 GetFlatBufferDataLayout(resizeDescriptor.m_DataLayout));
490
491 auto flatBufferLayer = serializer::CreateResizeBilinearLayer(m_flatBufferBuilder,
492 flatBufferBaseLayer,
493 flatBufferDescriptor);
494
495 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_ResizeBilinearLayer);
496}
497
Sadik Armagan8b42a382019-03-01 14:24:49 +0000498void SerializerVisitor::VisitRsqrtLayer(const armnn::IConnectableLayer* layer, const char* name)
499{
500 auto fbRsqrtBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Rsqrt);
501 auto fbRsqrtLayer = serializer::CreateRsqrtLayer(m_flatBufferBuilder, fbRsqrtBaseLayer);
502
503 CreateAnyLayer(fbRsqrtLayer.o, serializer::Layer::Layer_RsqrtLayer);
504}
505
Aron Virginas-Tarfc413c02019-02-13 15:41:52 +0000506// Build FlatBuffer for Softmax Layer
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000507void SerializerVisitor::VisitSoftmaxLayer(const armnn::IConnectableLayer* layer,
508 const armnn::SoftmaxDescriptor& softmaxDescriptor,
Aron Virginas-Tarfc413c02019-02-13 15:41:52 +0000509 const char* name)
510{
511 // Create FlatBuffer BaseLayer
512 auto flatBufferSoftmaxBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Softmax);
513
514 // Create the FlatBuffer SoftmaxDescriptor
515 auto flatBufferSoftmaxDesc =
516 serializer::CreateSoftmaxDescriptor(m_flatBufferBuilder, softmaxDescriptor.m_Beta);
517
518 // Create the FlatBuffer SoftmaxLayer
519 auto flatBufferSoftmaxLayer =
520 serializer::CreateSoftmaxLayer(m_flatBufferBuilder,
521 flatBufferSoftmaxBaseLayer,
522 flatBufferSoftmaxDesc);
523
524 CreateAnyLayer(flatBufferSoftmaxLayer.o, serializer::Layer::Layer_SoftmaxLayer);
525}
526
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000527void SerializerVisitor::VisitPooling2dLayer(const armnn::IConnectableLayer* layer,
528 const armnn::Pooling2dDescriptor& pooling2dDescriptor,
Saoirse Stewart3166c3e2019-02-18 15:24:53 +0000529 const char* name)
530{
531 auto fbPooling2dBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Pooling2d);
532 auto fbPooling2dDescriptor = serializer::CreatePooling2dDescriptor(
533 m_flatBufferBuilder,
534 GetFlatBufferPoolingAlgorithm(pooling2dDescriptor.m_PoolType),
535 pooling2dDescriptor.m_PadLeft,
536 pooling2dDescriptor.m_PadRight,
537 pooling2dDescriptor.m_PadTop,
538 pooling2dDescriptor.m_PadBottom,
539 pooling2dDescriptor.m_PoolWidth,
540 pooling2dDescriptor.m_PoolHeight,
541 pooling2dDescriptor.m_StrideX,
542 pooling2dDescriptor.m_StrideY,
543 GetFlatBufferOutputShapeRounding(pooling2dDescriptor.m_OutputShapeRounding),
544 GetFlatBufferPaddingMethod(pooling2dDescriptor.m_PaddingMethod),
545 GetFlatBufferDataLayout(pooling2dDescriptor.m_DataLayout));
546
547 auto fbPooling2dLayer = serializer::CreatePooling2dLayer(m_flatBufferBuilder,
548 fbPooling2dBaseLayer,
549 fbPooling2dDescriptor);
550
551 CreateAnyLayer(fbPooling2dLayer.o, serializer::Layer::Layer_Pooling2dLayer);
552}
553
Sadik Armagandbb0c0c2019-02-21 09:01:41 +0000554// Build FlatBuffer for FullyConnected Layer
555void SerializerVisitor::VisitFullyConnectedLayer(const armnn::IConnectableLayer* layer,
556 const armnn::FullyConnectedDescriptor& fullyConnectedDescriptor,
557 const armnn::ConstTensor& weights,
558 const armnn::Optional<armnn::ConstTensor>& biases,
559 const char* name)
560{
561 // Create FlatBuffer BaseLayer
562 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_FullyConnected);
563
564 // Create FlatBuffer FullyConnectedDescriptor
565 auto flatBufferDescriptor =
566 serializer::CreateFullyConnectedDescriptor(m_flatBufferBuilder,
567 fullyConnectedDescriptor.m_BiasEnabled,
568 fullyConnectedDescriptor.m_TransposeWeightMatrix);
569
570 // Create FlatBuffer weights data
571 auto flatBufferWeights = CreateConstTensorInfo(weights);
572
573 // Create FlatBuffer bias data
574 flatbuffers::Offset<serializer::ConstTensor> flatBufferBiases;
575 if (fullyConnectedDescriptor.m_BiasEnabled)
576 {
577 flatBufferBiases = CreateConstTensorInfo(biases.value());
578 }
579
580 // Create FlatBuffer FullyConnectedLayer
581 auto flatBufferLayer = serializer::CreateFullyConnectedLayer(m_flatBufferBuilder,
582 flatBufferBaseLayer,
583 flatBufferDescriptor,
584 flatBufferWeights,
585 flatBufferBiases);
586
587 // Add created FullyConnectedLayer to the FlatBufferLayers
588 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_FullyConnectedLayer);
589}
590
Nattapat Chaimanowong45286992019-02-26 15:53:02 +0000591// Build FlatBuffer for SpaceToBatchNd Layer
592void SerializerVisitor::VisitSpaceToBatchNdLayer(const armnn::IConnectableLayer* layer,
593 const armnn::SpaceToBatchNdDescriptor& spaceToBatchNdDescriptor,
594 const char* name)
595{
596 // Create FlatBuffer BaseLayer
597 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_SpaceToBatchNd);
598
599 std::vector<unsigned int> padList;
600 padList.reserve(spaceToBatchNdDescriptor.m_PadList.size()*2);
601 for (auto& pad : spaceToBatchNdDescriptor.m_PadList)
602 {
603 padList.push_back(pad.first);
604 padList.push_back(pad.second);
605 }
606
607 auto flatBufferDescriptor =
608 CreateSpaceToBatchNdDescriptor(m_flatBufferBuilder,
609 m_flatBufferBuilder.CreateVector(spaceToBatchNdDescriptor.m_BlockShape),
610 m_flatBufferBuilder.CreateVector(padList),
611 GetFlatBufferDataLayout(spaceToBatchNdDescriptor.m_DataLayout));
612
613 auto flatBufferLayer = serializer::CreateSpaceToBatchNdLayer(m_flatBufferBuilder,
614 flatBufferBaseLayer,
615 flatBufferDescriptor);
616
617 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_SpaceToBatchNdLayer);
618}
619
Nina Drozd57728782019-02-27 10:53:27 +0000620void SerializerVisitor::VisitNormalizationLayer(const armnn::IConnectableLayer* layer,
621 const armnn::NormalizationDescriptor& descriptor,
622 const char* name)
623{
624 auto fbNormalizationBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Normalization);
625
626 auto fbNormalizationDescriptor = serializer::CreateNormalizationDescriptor(
627 m_flatBufferBuilder,
628 GetFlatBufferNormalizationAlgorithmChannel(descriptor.m_NormChannelType),
629 GetFlatBufferNormalizationAlgorithmMethod(descriptor.m_NormMethodType),
630 descriptor.m_NormSize,
631 descriptor.m_Alpha,
632 descriptor.m_Beta,
633 descriptor.m_K,
634 GetFlatBufferDataLayout(descriptor.m_DataLayout));
635
636 auto flatBufferLayer = serializer::CreateNormalizationLayer(m_flatBufferBuilder,
637 fbNormalizationBaseLayer,
638 fbNormalizationDescriptor);
639
640 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_NormalizationLayer);
641}
642
Nattapat Chaimanowongb3485212019-03-04 12:35:39 +0000643void SerializerVisitor::VisitStridedSliceLayer(const armnn::IConnectableLayer* layer,
644 const armnn::StridedSliceDescriptor& stridedSliceDescriptor,
645 const char* name)
646{
647 auto flatBufferBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_StridedSlice);
648
649 auto flatBufferDescriptor =
650 CreateStridedSliceDescriptor(m_flatBufferBuilder,
651 m_flatBufferBuilder.CreateVector(stridedSliceDescriptor.m_Begin),
652 m_flatBufferBuilder.CreateVector(stridedSliceDescriptor.m_End),
653 m_flatBufferBuilder.CreateVector(stridedSliceDescriptor.m_Stride),
654 stridedSliceDescriptor.m_BeginMask,
655 stridedSliceDescriptor.m_EndMask,
656 stridedSliceDescriptor.m_ShrinkAxisMask,
657 stridedSliceDescriptor.m_EllipsisMask,
658 stridedSliceDescriptor.m_NewAxisMask,
659 GetFlatBufferDataLayout(stridedSliceDescriptor.m_DataLayout));
660
661 auto flatBufferLayer = serializer::CreateStridedSliceLayer(m_flatBufferBuilder,
662 flatBufferBaseLayer,
663 flatBufferDescriptor);
664
665 CreateAnyLayer(flatBufferLayer.o, serializer::Layer::Layer_StridedSliceLayer);
666}
667
Conor Kennedyda1f9752019-03-01 14:37:12 +0000668void SerializerVisitor::VisitSubtractionLayer(const armnn::IConnectableLayer* layer, const char* name)
669{
670 auto fbSubtractionBaseLayer = CreateLayerBase(layer, serializer::LayerType::LayerType_Subtraction);
671 auto fbSubtractionLayer = serializer::CreateSubtractionLayer(m_flatBufferBuilder, fbSubtractionBaseLayer);
672
673 CreateAnyLayer(fbSubtractionLayer.o, serializer::Layer::Layer_SubtractionLayer);
674}
675
Sadik Armagandbb0c0c2019-02-21 09:01:41 +0000676fb::Offset<serializer::LayerBase> SerializerVisitor::CreateLayerBase(const IConnectableLayer* layer,
Nattapat Chaimanowongac9cadc2019-02-13 15:52:41 +0000677 const serializer::LayerType layerType)
Mike Kelly8c1701a2019-02-11 17:01:27 +0000678{
679 std::vector<fb::Offset<serializer::InputSlot>> inputSlots = CreateInputSlots(layer);
680 std::vector<fb::Offset<serializer::OutputSlot>> outputSlots = CreateOutputSlots(layer);
681
682 return serializer::CreateLayerBase(m_flatBufferBuilder,
Saoirse Stewartcb8a3212019-02-14 15:46:10 +0000683 GetSerializedId(layer->GetGuid()),
Mike Kelly8c1701a2019-02-11 17:01:27 +0000684 m_flatBufferBuilder.CreateString(layer->GetName()),
685 layerType,
686 m_flatBufferBuilder.CreateVector(inputSlots),
687 m_flatBufferBuilder.CreateVector(outputSlots));
688}
689
Nattapat Chaimanowongac9cadc2019-02-13 15:52:41 +0000690void SerializerVisitor::CreateAnyLayer(const flatbuffers::Offset<void>& layer, const serializer::Layer serializerLayer)
Mike Kelly8c1701a2019-02-11 17:01:27 +0000691{
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000692 auto anyLayer = armnnSerializer::CreateAnyLayer(m_flatBufferBuilder, serializerLayer, layer);
Mike Kelly8c1701a2019-02-11 17:01:27 +0000693 m_serializedLayers.push_back(anyLayer);
694}
695
Mike Kellya0766c32019-02-19 17:22:07 +0000696template <typename T>
697flatbuffers::Offset<flatbuffers::Vector<T>> SerializerVisitor::CreateDataVector(const void* memory, unsigned int size)
698{
699 const T* buffer = reinterpret_cast<const T*>(memory);
700 std::vector<T> vector(buffer, buffer + (size / sizeof(T)));
701 auto fbVector = m_flatBufferBuilder.CreateVector(vector);
702 return fbVector;
703}
704
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000705flatbuffers::Offset<serializer::ConstTensor>
706 SerializerVisitor::CreateConstTensorInfo(const armnn::ConstTensor& constTensor)
Mike Kellya0766c32019-02-19 17:22:07 +0000707{
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000708 armnn::TensorInfo tensorInfo = constTensor.GetInfo();
Mike Kellya0766c32019-02-19 17:22:07 +0000709
710 // Get the dimensions
711 std::vector<unsigned int> shape;
712
713 for(unsigned int dim = 0; dim < tensorInfo.GetShape().GetNumDimensions(); ++dim)
714 {
715 shape.push_back(tensorInfo.GetShape()[dim]);
716 }
717
718 // Create FlatBuffer TensorInfo
719 auto flatBufferTensorInfo = serializer::CreateTensorInfo(m_flatBufferBuilder,
720 m_flatBufferBuilder.CreateVector(shape),
721 GetFlatBufferDataType(tensorInfo.GetDataType()),
722 tensorInfo.GetQuantizationScale(),
723 tensorInfo.GetQuantizationOffset());
724 flatbuffers::Offset<void> fbPayload;
725
726 switch (tensorInfo.GetDataType())
727 {
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000728 case armnn::DataType::Float32:
729 case armnn::DataType::Signed32:
Mike Kellya0766c32019-02-19 17:22:07 +0000730 {
731 auto fbVector = CreateDataVector<int32_t>(constTensor.GetMemoryArea(), constTensor.GetNumBytes());
732 flatbuffers::Offset<serializer::IntData> flatBuffersData = serializer::CreateIntData(
733 m_flatBufferBuilder,
734 fbVector);
735 fbPayload = flatBuffersData.o;
736 break;
737 }
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000738 case armnn::DataType::Float16:
Mike Kellya0766c32019-02-19 17:22:07 +0000739 {
740 auto fbVector = CreateDataVector<int16_t>(constTensor.GetMemoryArea(), constTensor.GetNumBytes());
741 flatbuffers::Offset<serializer::ShortData> flatBuffersData = serializer::CreateShortData(
742 m_flatBufferBuilder,
743 fbVector);
744 fbPayload = flatBuffersData.o;
745 break;
746 }
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000747 case armnn::DataType::QuantisedAsymm8:
748 case armnn::DataType::Boolean:
Mike Kellya0766c32019-02-19 17:22:07 +0000749 default:
750 {
751 auto fbVector = CreateDataVector<int8_t>(constTensor.GetMemoryArea(), constTensor.GetNumBytes());
752 flatbuffers::Offset<serializer::ByteData> flatBuffersData = serializer::CreateByteData(
753 m_flatBufferBuilder,
754 fbVector);
755 fbPayload = flatBuffersData.o;
756 }
757 }
758 flatbuffers::Offset<serializer::ConstTensor> flatBufferConstTensor = serializer::CreateConstTensor(
759 m_flatBufferBuilder,
760 flatBufferTensorInfo,
761 GetFlatBufferConstTensorData(tensorInfo.GetDataType()),
762 fbPayload);
763 return flatBufferConstTensor;
764}
765
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000766std::vector<fb::Offset<serializer::InputSlot>>
767 SerializerVisitor::CreateInputSlots(const armnn::IConnectableLayer* layer)
Mike Kelly8c1701a2019-02-11 17:01:27 +0000768{
Mike Kellya0766c32019-02-19 17:22:07 +0000769 std::vector<fb::Offset<serializer::InputSlot>> inputSlots;
Mike Kelly8c1701a2019-02-11 17:01:27 +0000770
771 // Get the InputSlots
772 for (unsigned int slotIndex = 0; slotIndex<layer->GetNumInputSlots(); ++slotIndex)
773 {
774 const IInputSlot& inputSlot = layer->GetInputSlot(slotIndex);
775
776 // Get the Connection for the InputSlot
777 const IOutputSlot* connection = inputSlot.GetConnection();
778
779 // Create FlatBuffer Connection
Saoirse Stewartcb8a3212019-02-14 15:46:10 +0000780 serializer::Connection conn(GetSerializedId(inputSlot.GetConnection()->GetOwningLayerGuid()),
781 connection->CalculateIndexOnOwner());
Mike Kelly8c1701a2019-02-11 17:01:27 +0000782 // Create FlatBuffer InputSlot
783 inputSlots.push_back(serializer::CreateInputSlot(m_flatBufferBuilder, slotIndex, &conn));
784 }
785 return inputSlots;
786}
787
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000788std::vector<fb::Offset<serializer::OutputSlot>>
789 SerializerVisitor::CreateOutputSlots(const armnn::IConnectableLayer* layer)
Mike Kelly8c1701a2019-02-11 17:01:27 +0000790{
791 std::vector<fb::Offset<serializer::OutputSlot>> outputSlots;
792
793 // Get the OutputSlots
794 for (unsigned int slotIndex = 0; slotIndex < layer->GetNumOutputSlots(); ++slotIndex)
795 {
796 const IOutputSlot& outputSlot = layer->GetOutputSlot(slotIndex);
Derek Lamberti0028d1b2019-02-20 13:57:42 +0000797 const armnn::TensorInfo& tensorInfo = outputSlot.GetTensorInfo();
Mike Kelly8c1701a2019-02-11 17:01:27 +0000798
799 // Get the dimensions
800 std::vector<unsigned int> shape;
801 for(unsigned int dim = 0; dim < tensorInfo.GetShape().GetNumDimensions(); ++dim)
802 {
803 shape.push_back(tensorInfo.GetShape()[dim]);
804 }
805
806 // Create FlatBuffer TensorInfo
807 auto flatBufferTensorInfo = serializer::CreateTensorInfo(m_flatBufferBuilder,
808 m_flatBufferBuilder.CreateVector(shape),
809 GetFlatBufferDataType(tensorInfo.GetDataType()),
810 tensorInfo.GetQuantizationScale(),
811 tensorInfo.GetQuantizationOffset());
812
813 // Create FlatBuffer Outputslot
814 outputSlots.push_back(serializer::CreateOutputSlot(m_flatBufferBuilder,
815 slotIndex,
816 flatBufferTensorInfo));
817 }
818 return outputSlots;
819}
820
Nattapat Chaimanowongac9cadc2019-02-13 15:52:41 +0000821
822ISerializer* ISerializer::CreateRaw()
823{
824 return new Serializer();
825}
826
827ISerializerPtr ISerializer::Create()
828{
829 return ISerializerPtr(CreateRaw(), &ISerializer::Destroy);
830}
831
832void ISerializer::Destroy(ISerializer* serializer)
833{
834 delete serializer;
835}
836
837void Serializer::Serialize(const INetwork& inNetwork)
838{
839 // Iterate through to network
840 inNetwork.Accept(m_SerializerVisitor);
841 flatbuffers::FlatBufferBuilder& fbBuilder = m_SerializerVisitor.GetFlatBufferBuilder();
842
843 // Create FlatBuffer SerializedGraph
844 auto serializedGraph = serializer::CreateSerializedGraph(
845 fbBuilder,
846 fbBuilder.CreateVector(m_SerializerVisitor.GetSerializedLayers()),
847 fbBuilder.CreateVector(m_SerializerVisitor.GetInputIds()),
848 fbBuilder.CreateVector(m_SerializerVisitor.GetOutputIds()));
849
850 // Serialize the graph
851 fbBuilder.Finish(serializedGraph);
852}
853
854bool Serializer::SaveSerializedToStream(std::ostream& stream)
855{
856 flatbuffers::FlatBufferBuilder& fbBuilder = m_SerializerVisitor.GetFlatBufferBuilder();
857
Nattapat Chaimanowong7b53b692019-02-12 14:38:31 +0000858 auto bytesToWrite = boost::numeric_cast<std::streamsize>(fbBuilder.GetSize());
859 stream.write(reinterpret_cast<const char*>(fbBuilder.GetBufferPointer()), bytesToWrite);
Nattapat Chaimanowongac9cadc2019-02-13 15:52:41 +0000860 return !stream.bad();
861}
862
Matteo Martincighec333912019-02-13 15:12:39 +0000863} // namespace armnnSerializer