torch-mlir/docs/importers/onnx_importer.md

6.1 KiB

TorchOnnx To Torch Conversions

We enable the direct representation of many ONNX features directly in the torch dialect as torch.operator custom ops with names like onnx.{OperatorName}. The majority of ONNX operators are represented with a systematic transformation. torch_mlir.extras.onnx_importer for the reference importer which complies with the rules below.

Adding new ONNX operators

With the exception of certain special or complicated ONNX operators, most are relatively straight-forward to map, following this general procedure:

  • Plan the ops you wish to support by consulting the ONNX operator database.
    • This database has detailed diffs wrt different support versions but at the level of detail we operate, most version diffs are inconsequential and just require a bit more pattern support.
    • This typically applies to generalization of broadcasting semantics, expanded type support, and other things of the like.
  • Prerequisite: Add support for the op to torch-mlir if it does not already exist.
  • Open the corresponding implementation file DefaultDomainXtoY.cpp corresponding with the alphabetic sort of the op and add a conversion.
  • Generate successful test cases:
    • All onnx_importer.py tests are dumped to the test temp dir (success or failure). This is typically located under tools/torch-mlir/test/python/onnx_importer/Output. The .mlir files under there should provide good variants to drive lit test coverage of conversion.
      • (Optionally) If there is an Onnx file that uses the op of interest, convert that file to Onnx MLIR form using the following Python command, python -m torch_mlir.tools.import_onnx my_model.onnx.
    • There are often many variants of tests for checking conformance of different historic ONNX encodings, but these are often not load bearing at the MLIR level.
    • Pick a handful of test cases and add them to test/Conversion/TorchOnnxToTorch/simple_ops_x_to_y.mlir corresponding to an alphabetic breakdown. At this time, ignore tests that are not exercising useful differences in the pattern implementations.
      • (Optionally) Use torch-mlir-opt to validate the outputs of the new op. First, build the project using cmake --build build --target tools/torch-mlir/all. This will generate the conversion binary, torch-mlir-opt. Then call torch-mlir-opt with the MLIR pass convert-torch-onnx-to-torch:
        build/bin/torch-mlir-opt -convert-torch-onnx-to-torch \
        -split-input-file [DESIRED_ONNX_FILE].mlir
        
  • Generate failure test cases:
    • Some ops have forms that do not (easily) map to torch-mlir. If you leave an op under-implemented, add a failing test case to test/Conversion/TorchOnnxToTorch/unsupported_simple_ops.mlir.
  • Optional but recommended: Use your test case files to fuzz against the torch-mlir backend of your choice by running a backend conversion pipeline and fixing any crashes/issues.
  • Send a patch with your changes.

ONNX proto to torch dialect mapping

Type Conversion

  • Tensors: ONNX tensor types are converted to torch.vtensor with static and dynamic dimensions. We require that shape inference has run to produce ranked tensors.
  • Tensor element types are directly converted to corresponding MLIR types as used by the rest of torch-mlir.
  • String, sequence and sparse tensor types are presently not mapped.

Attributes

A subset of attributes types are converted directly to an attribute dict on the op with a name like torch.onnx.{AttributeName}. The following attribute type mappings are made:

  • FLOAT: FloatAttr
  • INT: Signed IntegerAttr of width 64
  • STRING: StringAttr
  • TENSOR: Converted to one of:
    • DenseResourceElementsAttr for inlined raw_data
    • DenseElementsAttr for splats
    • DenseElementsAttr for inlined typed proto initialization
  • FLOATS: ArrayAttr of FloatAttr
  • INTS: ArrayAttr of signed IntegerAttr of width 64
  • STRINGS: ArrayAttr of StringAttr
  • TENSORS: ArrayAttr of corresponding TENSOR conversion

The following attribute types have no present, systematic conversion. Their presence on an op indicates that the op is a special form, which must be handled specially:

  • GRAPH
  • SPARSE_TENSOR (TBD: it is possible to handle this systematically if useful).
  • TYPE_PROTO (TBD: it may be possible to handle this systematically if useful).
  • Plural equivalents of the above.

Default operation conversion

Operations are converted to a torch.operator with name onnx.{OperatorName}. The constraint that the ONNX graph is topologically sorted and free of cycles matches the SSA form. Operands and results are mapped directly.

This conversion only applies to the default (empty) domain.

Quantization information

Quantization parameters are carried out of line in the ONNX protobuf and will be repatriated upon import to torch. The exact mechanism is not yet implemented.

Version and metadata

The IsolatedFromAbove parent of the ops can contain the following metadata:

  • torch.onnx_meta.ir_version: 64bit IntegerAttr corresponding to ModelProto.ir_version.
  • torch.onnx_meta.producer_name: StringAttr corresponding to ModelProto.producer_name.
  • torch.onnx_meta.producer_version: StringAttr corresponding to ModelProto.producer_version.
  • torch.onnx_meta.opset_version: 64bit IntegerAttr corresponding to ModelProto.opset_import.version for the domain "" (empty). Will be ommitted if the default opset is not included.
  • torch.onnx_meta.opset_versions: DictAttr of 64bit IntegerAttr for each non default domain.

Generally, the importer handles variations in ir_version whereas the transformations here handle opset version differences. Version independent transformations are encouraged where possible if there are only minor variations of an op. Major variations should use since_version sensitive patterns.

Special op forms

Certain ONNX operators map to different structural components of torch-mlir's representation:

  • ConstantOfShape: Mapped to torch.vtensor.literal with a corresponding value attribute.