This document outlines how memory is managed internally by TensorFlow Lite Micro (TFLM) today. It outlines the "online" allocation strategy used by the default TFLM APIs for loading a model into a shared tensor arena.
The main "working" space for TFLM allocations is inside a single char
or
int8_t
buffer. This buffer can be managed by passing it directly into a
tflite::MicroInterpreter
constructor or through a tflite::MicroAllocator
instance that can be passed into a tflite::MicroInterpreter
constructor.
Internally, the tflite::MicroAllocator
classifies allocations into 3 different
sections:
- Head - non-persistent allocations.
- Temporary - short term "scoped" allocations.
- Tail - persistent allocations.
The illustration below represents typical allocations in TFLM:
--------------------------------------------------------------------------------
| | | |
| HEAD |<-- TEMPORARY -->| TAIL |
| | | |
--------------------------------------------------------------------------------
* Lowest Address Highest Address *
This non-persistent section typically holds shared Tensor buffers. This section does not allocate small iterative chunks, it can only be set by a specific length for the entire section.
This allocation length of this section is managed by the
tflite::GreedyMemoryPlanner
. That memory planner looks at the entire graph of
a model and tries to reuse as many buffers as possible to create the smallest
length for the head. The Tensor buffers for this section can be accessed via a
TfLiteEvalTensor
or TfLiteTensor
instance on the tflite::MicroInterpreter
.
All, or a subset of, tensors can be allocated using an offline planner. An offline planner performs tensor allocation on e.g. a host PC. The offline tensor allocation plan is added to model metadata. See format below.
For each non-constant tensor in the tensors:[Tensor]
list of the subgraph, a
byte offset to the start of the head section of the memory arena is given. -1
indicates that the tensor will be allocated at runtime by the
tflite::GreedyMemoryPlanner
. The offline plan is permitted to overlap buffers
if it knows that the data will not be used at the same time.
The offline tensor allocation plan will be encoded in the metadata:[Metadata]
field of the model, using the following encoding:
Metadata component | Value |
---|---|
name:string | “OfflineMemoryAllocation” |
buffer:unit | Index of buffer containing offline tensor allocation data |
The buffer contents for the offline tensor allocation is a list of 32-bit integers of the following format:
Offset | Value |
---|---|
0 | Offline allocation format version |
1 | Number of subgraphs |
2 | Number offsets following: n |
3 | Byte offset of tensor #0 or -1 to allocate at runtime |
4 | Byte offset of tensor #1 or -1 to allocate at runtime |
... | ... |
3+(n-1) | Byte offset of tensor #(n-1) or -1 to allocate at runtime |
Note that offsets 0 (the version) and 1 (the number of subgraphs) are currently ignored by the micro memory allocator. In case of multiple subgraphs, it assumes all tensors for all subgraphs are concatenated: all tensors for the first subgraph are first, followed by those of the second subgraph, etc.
The tflite::GreedyMemoryPlanner
treats the provided offline tensor allocation
plan as constant fixed offset to the start of the head section and will attempt
to fit any other tensors (such as scratch tensors added a runtime using the
RequestScratchBufferInArena
API of TfLiteContext
) around those fixed
offsets.
This section is used to allocate "scoped" or short-term, non-guaranteed buffers. Allocations from this section start from the current end address of the head section and grow towards the tail section. An allocation chain can be reset (and must be reset before adjusting the head) and moves the current allocation start address back to the end of the head section.
TFLM currently uses these allocations for a scope allocation of large C structs or scratch memory that is expected to be valid for at least the lifetime of a method call. This section.
This section holds all persistent allocations used by TFLM. This section contains many random sized allocations and grows towards the end of the head section. Allocations in this section come from a variety of areas inside of TFLM. TFLM provides a recording API to assist with auditing the contents of this section.
TFLM provides simple APIs for auditing memory usage in the shared tensor arena. These APIs are opt-in and require some additional memory overhead and a working debug logging implementation (reference implementation).
A typical bare-bones TFLM interpreter setup looks as such:
// Buffer for the tensor arena:
size_t tensor_arena_size = 2048;
uint8_t tensor_arena[tensor_arena_size];
// Interpreter using the shared tensor arena above:
tflite::MicroInterpreter interpreter(
tflite::GetModel(my_model_data), ops_resolver,
tensor_arena, tensor_arena_size);
// Invoke one time which will allocate internals:
if (interpreter.Invoke() != kTfLiteOk) {
MicroPrintf("Exception during invoke()!");
}
Recording API can simply be used by including the RecordingMicroInterpreter
class (recording_micro_interpreter.h
) and replace tflite::MicroInterpreter
with tflite::RecordingMicroInterpreter
. The same call to invoke()
is
performed, but another call is made to PrintAllocations()
which will output
detailed allocation logging:
// Add an include to the recording API:
#include "recording_micro_interpreter.h"
// Simply change the class name from 'MicroInterpreter' to 'RecordingMicroInterpreter':
tflite::RecordingMicroInterpreter interpreter(
tflite::GetModel(my_model_data), ops_resolver,
tensor_arena, tensor_arena_size);
// Invoke one time which will allocate internals:
if (interpreter.Invoke() != kTfLiteOk) {
MicroPrintf("Exception during invoke()!");
}
// Print out detailed allocation information:
interpreter.GetMicroAllocator().PrintAllocations();
The output of this call will look something similar to this (output from the memory_arena_threshold_test):
[RecordingMicroAllocator] Arena allocation total 9568 bytes
[RecordingMicroAllocator] Arena allocation head 7744 bytes
[RecordingMicroAllocator] Arena allocation tail 1824 bytes
[RecordingMicroAllocator] 'TfLiteEvalTensor data' used 360 bytes with alignment overhead (requested 360 bytes for 15 allocations)
[RecordingMicroAllocator] 'Persistent TfLiteTensor data' used 0 bytes with alignment overhead (requested 0 bytes for 0 tensors)
[RecordingMicroAllocator] 'Persistent TfLiteTensor quantization data' used 0 bytes with alignment overhead (requested 0 bytes for 0 allocations)
[RecordingMicroAllocator] 'TfLiteTensor variable buffer data' used 0 bytes with alignment overhead (requested 0 bytes for 0 allocations)
[RecordingMicroAllocator] 'NodeAndRegistration struct' used 392 bytes with alignment overhead (requested 392 bytes for 7 NodeAndRegistration structs)
[RecordingMicroAllocator] 'Operator runtime data' used 136 bytes with alignment overhead (requested 136 bytes for 5 OpData structs)
More information about each recorded allocation section:
- 'TfLiteEvalTensor data'
- C struct that holds the data type, dimension, and a pointer to the buffer representing the Tensor.
- 'Persistent TfLiteTensor data'
- C struct that holds more information than a
TfLiteEvalTensor
struct in the graph. - Allocations in this bucket will only show up when accessing tensors from
the accessors on
tflite::MicroInterpreter
.
- C struct that holds more information than a
- 'Persistent TfLiteTensor quantization data'
- Length of persistent quantization data assigned to persistent
TfLiteTensor
structs. - Allocations in this bucket will only show up when accessing tensors from
the accessors on
tflite::MicroInterpreter
.
- Length of persistent quantization data assigned to persistent
- 'TfLiteTensor variable buffer data'
- Length of buffer data from a variable tensor (retains data throughout
calls to
invoke()
).
- Length of buffer data from a variable tensor (retains data throughout
calls to
- 'NodeAndRegistration struct'
- C struct that holds a
TfLiteRegistration
andTfLiteNode
struct instance. - Each operator in a model will contain one
NodeAndRegistration
struct.
- C struct that holds a
- 'Operator runtime data'
- Persistent allocations of data cached by TFLM kernels (e.g. quantization params, multipliers, etc).