IREE TFLite C API Compatibility Shim

EXPERIMENTAL: we are working towards making this a stable API but it has a ways to go still. Progress is being tracked in https://github.com/iree-org/iree/projects/17.

Provides a (mostly) tflite-compatible API that allows loading compiled IREE modules, managing tensors, and invoking functions with the same conventions as TFLite's C API.

The intent is not to have a faithful reproduction of all tflite features as IREE obviates the need for many of them (such as builtin ops). This is meant as a way for applications that may currently be using the tflite API to quickly onboard with IREE. We still expect applications to later migrate to proper IREE APIs to gain the most from IREE as many of the features in IREE are not possible to expse from simple single-shot invocation APIs like tflite.

Quickstart

Compiling TFLite FlatBuffers to IREE Modules

# TODO(#3970): example command line converting the fb (iree-compile-tflite).

TODO(benvanik): tosa flow, iree-compile (whenever ready)

Retargeting to the IREE TFLite Binding Library

# TODO(benvanik): example command line with include path and IREE static lib.

The bindings supply copies of the tflite include files that expose only the public API and portions that IREE supports. All unsupported features are guarded by a IREE_BINDINGS_TFLITE_INCLUDE_UNSUPPORTED_APIS define which can be enabled if compatibility with code that may reference the calls or types is required. Needing that is a strong indication, though, that the application is not following the public API and will corrupt its memory touching random structure fields.

Always prefer static linkage when possible: because the IREE runtime is so small the overhead of linking it in its own library can double the size. Instead, ensure you are linking it into your main application so the use of things like the C runtime are amortized across both your code and IREE's. You can force static linkage via the TFL_COMPILE_LIBRARY define (as with normal tflite).

Loading and Executing Models

// TODO(benvanik): tflite usage example.

Use TfLiteModelCreateFromFile if loading your model from the filesystem so that IREE can directly map it into memory. Even if the model is fetched dynamically it is still worth it to write it to disk and then map it so that the infrequently used mapped pages can be discarded by the OS to reclaim memory. Only use TfLiteModelCreate if you have the model embedded in your binary or are sure you can accept the wired pages for the convenience like in a REPL/notebook. When used in conjunction with the IREE C API the compiler arranges the module memory to be able to perform things like data prefetching (for constants prior to when they are used), page eviction (for initializer constants that won't be needed again), and when compiling for native targets the ability to directly execute pages from the memory.

TODO(benvanik): stock example

Support

Glossary
โœ”๏ธSupported and expected to match tflite semantics
โš ๏ธSupported with compatibility caveats (avoid if possible)
๐ŸขSupported with performance caveats (prefer the IREE C API)
๐ŸšซUnimplemented but supportable if needed
โ›”Unsupported and unlikely to ever be (see notes below)
๐Ÿ”’Not part of the tflite public API
โ”Unknown; not yet studied

Op Coverage

TODO(benvanik): note about TOSA, link to tflite->tosa documentation

API

Only the public C API functions are supported. The contents of internal structures like TfLiteTensor and TfLiteContext are undefined.

TFLite APINotes
โœ”๏ธTfLiteVersionreturns an IREE version string
๐Ÿ”’TfLiteModel structimplementation detail
โœ”๏ธTfLiteModelCreate
โœ”๏ธTfLiteModelCreateFromFile
โœ”๏ธTfLiteModelDelete
๐Ÿ”’TfLiteInterpreterOptions structimplementation detail
โœ”๏ธTfLiteInterpreterOptionsCreate
โœ”๏ธTfLiteInterpreterOptionsDelete
๐ŸขTfLiteInterpreterOptionsSetNumThreadsinterpreters will not share thread pools; see external contexts
โœ”๏ธTfLiteInterpreterOptionsSetErrorReporter
โ›”TfLiteInterpreterOptionsAddBuiltinOpIREE's compiler generates code
๐ŸšซTfLiteInterpreterOptionsAddCustomOpnot yet implemented
๐ŸšซTfLiteInterpreterOptionsSetOpResolvernot yet implemented
โš ๏ธTfLiteInterpreterOptionsAddDelegateavailable but a no-op; not needed in IREE
โš ๏ธTfLiteInterpreterOptionsSetUseNNAPIavailable but a no-op; NNAPI not supported
๐Ÿ”’TfLiteInterpreter structimplementation detail
โœ”๏ธTfLiteInterpreterCreate
โœ”๏ธTfLiteInterpreterCreateWithSelectedOpsalias to TfLiteInterpreterCreate
โœ”๏ธTfLiteInterpreterDelete
โœ”๏ธTfLiteInterpreterResetVariableTensors
โœ”๏ธTfLiteInterpreterGetInputTensorCount
โœ”๏ธTfLiteInterpreterGetInputTensor
โœ”๏ธTfLiteInterpreterResizeInputTensor
โœ”๏ธTfLiteInterpreterAllocateTensors
โœ”๏ธTfLiteInterpreterInvoke
โœ”๏ธTfLiteInterpreterGetOutputTensorCount
โœ”๏ธTfLiteInterpreterGetOutputTensor
๐ŸšซTfLiteTensor structcurrently opaque; could be exposed with caveats
โœ”๏ธTfLiteTensorType
โœ”๏ธTfLiteTensorNumDims
โœ”๏ธTfLiteTensorDim
โœ”๏ธTfLiteTensorByteSize
โœ”๏ธTfLiteTensorData
โœ”๏ธTfLiteTensorName
โœ”๏ธTfLiteTensorQuantizationParams
โœ”๏ธTfLiteTensorCopyFromBuffer
โœ”๏ธTfLiteTensorCopyToBuffer

Features

TFLite FeatureNotes
๐Ÿ”’SparsityAPI not public; likely possible
๐Ÿ”’Complex NumbersAPI not public; likely possible
๐Ÿ”’External ContextsAPI not public; support possible but API inadequate for performance sensitive applications
๐ŸšซCustom Opsnot yet implemented; can be supported with performance caveats
๐ŸšซDynamic Model Creationavoid doing this and use a compiler; almost all use cases besides specialized tools like REPLs can compile their models offline
โ›”Delegatesconcept mismatch; not needed in IREE due to its hardware abstraction layer (HAL)
โ›”TFLite Microconcept mismatch; compilers are much better at this scale

๐Ÿงช External Contexts

CURRENTLY UNSUPPORTED: tflite has experimental support for “external contexts” but they are not exposed via the public API yet.

Though it's possible to use multiple TfLiteInterpreter instances in the same process in the real tflite it is strongly discouraged: each interpreter will create its own thread and memory pools and device handles to accelerators and assume it owns all resources exclusively. The experimental external contexts API is present to try to allow for something better than that and IREE would be able to make use of it to the extent the feature allows.

But IREE is designed to fully support large constellations of models all running concurrently and passing data between both each other and the application efficiently pipelined cross-device and cross-process. Though external contexts would allow IREE to at least share some resources such as the thread pool it would still require applications to cooperatively schedule model execution to ensure that predictable latencies and memory consumption would be fixed at the sum of all models peak memory use regardless of scheduling.

When using more than one simultaneously loaded and execution model it is much better to use the IREE C API instead.

๐Ÿคท๐Ÿฟโ€โ™‚๏ธ Custom Ops

CURRENTLY UNSUPPORTED: possible to implement if needed; it seems as if there barely any usage of the custom op C API outside of Google though so we recommend avoiding it for now. (1, 2).

Custom ops in tflite map to functions imported into compiled IREE modules. The IREE tflite API shim could provide a wrapper implemented as an iree_vm_module_t that resolves and executes the functions as they are called by the VM. Having real IREE modules, though, provides significant benefits in representation such as the ability to have asynchronous custom behavior that interacts well with hardware accelerators and IREE's concurrency and pipelining model. It also allows for a majority of the kind of operations that previously would have necessitated custom ops at runtime to instead be done in the compiler as MLIR dialects and lowered right to native code, SPIR-V, or WebAssembly without the need for expensive interop and opportunities for the compiler to tightly optimize the custom behavior with the rest of the model.

Relevant unsupported APIs:

๐Ÿ™…โ€โ™€๏ธ Dynamic Model Creation

ACTIVELY DISCOURAGED: IREE separates compilation and execution; for nearly all models that are capable of running on tflite there is no benefit (and often extreme downsides) to constructing them at runtime. As in any other domain if you don‘t need a JIT you should never use a JIT. Dynamic model creation in tflite is most frequently used to work around the lack of dynamism that has been inherent in the tflite interchange format and interpreter, neither issues of which IREE suffers from. Though it’s possible to ship MLIR to target devices and construct models on-demand there are almost no situations in which one should do so beyond tools like REPLs and it is not a supported IREE use case.

Relevant unsupported APIs:

๐Ÿ™…โ€โ™€๏ธ Delegates

SUPERFLUOUS: The concept of delegates - something that dissects a model at runtime and attempts to slice off random parts of it to steal for itself - is unnecessary in IREE. The extensible hardware abstraction layer (HAL) that IREE provides achieves cross-device portability in a way that allows for predictable high-performance heterogeneity and efficient co-optimization with the compiler. The act of mutating a user‘s input graph happens in the compiler where significantly more metadata and resources are available to optimize the model and the the model is deployed as multi-architecture binaries containing one or more formats of native machine code or low-level intermediate representations like SPIR-V or WebAssembly. It’s akin to transmitting JPEGs and WebPs to web browsers and allowing for the client to select the appropriate decoder vs. shipping source uncompressed PNGs and having to transcode them on the fly. The problem of distribution for deployable IREE artifacts matches that of the kind apps must already deal with: split APKs, universal binaries, etc and it's easy to map anything you can do with IREE artifacts to that mental model.

Relevant unsupported APIs:

๐Ÿ™…โ€โ™€๏ธ TFLite Micro

ACTIVELY DISCOURAGED: in situations where memory and compute are at a premium one should always use an ahead-of-time compiler and not waste precious resources (memory, power, thermals, etc) on things like memory planning (especially when only static shapes are supported), data conversion like endianness swapping, executing unfused elementwise ops, including any executable bytes for code-paths that will never be reached in your model, or perform a single superfluous m e m c p y.

In it‘s most size-optimized form the IREE VM bytecode is lowered to C or LLVM IR and compiled along with application code. All tensor operations are aggressively fused to avoid the need for transient memory, runtime memory planning (even when using dynamic shapes), or copies. Just as there’s an expectation that compiler toolchains perform basic optimizations to application code (dead code removal, code deduplication, etc) so too should there be an expectation for ML models and even more so in environments where every byte (and joule) matters.

IREE will likely never have a shim for the tflite micro API or the tflite TF_LITE_STATIC_MEMORY mode; when operating at that scale the entire solution needs to change.