2019-02-22 17:00:25 +08:00
|
|
|
# LLVM IR Dialect
|
|
|
|
|
|
|
|
This dialect wraps the LLVM IR types and instructions into MLIR types and
|
2019-03-30 04:15:06 +08:00
|
|
|
operations. It provides several additional operations that are necessary to
|
2019-02-22 17:00:25 +08:00
|
|
|
cover for the differences in the IR structure (e.g., MLIR does not have `phi`
|
|
|
|
operations and LLVM IR does not have a `constant` operation).
|
|
|
|
|
|
|
|
In this document, we use "LLVM IR" to designate the
|
|
|
|
[intermediate representation of LLVM](https://llvm.org/docs/LangRef.html) and
|
|
|
|
"LLVM IR _dialect_" to refer to the MLIR dialect reflecting LLVM instructions
|
|
|
|
and types.
|
|
|
|
|
|
|
|
[TOC]
|
|
|
|
|
|
|
|
## Context and Module Association
|
|
|
|
|
|
|
|
The LLVM IR dialect object _contains_ an LLVM Context and an LLVM Module that it
|
|
|
|
uses to define, print, parse and manage LLVM IR types. These objects can be
|
|
|
|
obtained from the dialect object using `.getLLVMContext()` and
|
|
|
|
`getLLVMModule()`. All LLVM IR objects that interact with the LLVM IR dialect
|
|
|
|
must exist in the dialect's context.
|
|
|
|
|
2019-04-05 23:19:42 +08:00
|
|
|
## Types
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
The LLVM IR dialect defines a single MLIR type, `LLVM::LLVMType`, that can wrap
|
|
|
|
any existing LLVM IR type. Its syntax is as follows
|
|
|
|
|
|
|
|
``` {.ebnf}
|
2019-03-07 17:27:09 +08:00
|
|
|
type ::= `!llvm<"` llvm-canonical-type `">
|
2019-02-22 17:00:25 +08:00
|
|
|
llvm-canonical-type ::= <canonical textual representation defined by LLVM>
|
|
|
|
```
|
|
|
|
|
2019-04-06 14:56:49 +08:00
|
|
|
For example, one can use primitive types `!llvm.i32`, pointer types
|
2019-03-07 17:27:09 +08:00
|
|
|
`!llvm<"i8*">`, vector types `!llvm<"<4 x float>">` or structure types
|
|
|
|
`!llvm<"{i32, float}">`. The parsing and printing of the canonical form is
|
2019-02-22 17:00:25 +08:00
|
|
|
delegated to the LLVM assembly parser and printer.
|
|
|
|
|
|
|
|
LLVM IR dialect types contain an `llvm::Type*` object that can be obtained by
|
|
|
|
calling `.getUnderlyingType()` and used in LLVM API calls directly. These
|
|
|
|
objects are allocated within the LLVM context associated with the LLVM IR
|
|
|
|
dialect and may be linked to the properties of the associated LLVM module.
|
|
|
|
|
|
|
|
LLVM IR dialect type can be constructed from any `llvm::Type*` that is
|
|
|
|
associated with the LLVM context of the dialect. In this document, we use the
|
|
|
|
term "wrapped LLVM IR type" to refer to the LLVM IR dialect type containing a
|
|
|
|
specific LLVM IR type.
|
|
|
|
|
2019-04-05 23:19:42 +08:00
|
|
|
## Operations
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
All operations in the LLVM IR dialect have a custom form in MLIR. The mnemonic
|
|
|
|
of an operation is that used in LLVM IR prefixed with "`llvm.`".
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
### LLVM IR operations
|
|
|
|
|
|
|
|
The following operations are currently supported. The semantics of these
|
|
|
|
operations corresponds to the semantics of the similarly-named LLVM IR
|
|
|
|
instructions.
|
|
|
|
|
|
|
|
#### Integer binary arithmetic operations
|
|
|
|
|
|
|
|
Take two arguments of wrapped LLVM IR integer type, produce one value of the
|
|
|
|
same type.
|
|
|
|
|
|
|
|
- `add`
|
|
|
|
- `sub`
|
|
|
|
- `mul`
|
|
|
|
- `udiv`
|
|
|
|
- `sdiv`
|
|
|
|
- `urem`
|
|
|
|
- `srem`
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Integer addition.
|
2019-04-06 14:56:49 +08:00
|
|
|
%0 = llvm.add %a, %b : !llvm.i32
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Unsigned integer division.
|
2019-04-06 14:56:49 +08:00
|
|
|
%1 = llvm.udiv %a, %b : !llvm.i32
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
#### Floating point binary arithmetic operations
|
|
|
|
|
|
|
|
Take two arguments of wrapped LLVM IR floating point type, produce one value of
|
|
|
|
the same type.
|
|
|
|
|
|
|
|
- `fadd`
|
|
|
|
- `fsub`
|
|
|
|
- `fmul`
|
|
|
|
- `fdiv`
|
|
|
|
- `frem`
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Float addition.
|
2019-04-06 14:56:49 +08:00
|
|
|
%0 = llvm.fadd %a, %b : !llvm.float
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Float division.
|
2019-04-06 14:56:49 +08:00
|
|
|
%1 = llvm.fdiv %a, %b : !llvm.float
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
2019-03-30 04:15:06 +08:00
|
|
|
#### Memory-related operations
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
- `<r> = alloca <size> x <type>`
|
|
|
|
- `<r> = getelementptr <address>[<index> (, <index>)+]`
|
2019-02-22 17:00:25 +08:00
|
|
|
- `<r> = load <address>`
|
|
|
|
- `store <value>, <address>`
|
|
|
|
|
|
|
|
In these operations, `<size>` must be a value of wrapped LLVM IR integer type,
|
|
|
|
`<address>` must be a value of wrapped LLVM IR pointer type, and `<value>` must
|
|
|
|
be a value of wrapped LLVM IR type that corresponds to the pointee type of
|
|
|
|
`<address>`.
|
|
|
|
|
2019-02-26 17:27:27 +08:00
|
|
|
The `index` operands are integer values whose semantics is identical to the
|
|
|
|
non-pointer arguments of LLVM IR's `getelementptr`.
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Allocate an array of 4 floats on stack
|
2019-04-06 14:56:49 +08:00
|
|
|
%c4 = llvm.constant(4) : !llvm.i64
|
|
|
|
%0 = llvm.alloca %c4 x !llvm.float : (!llvm.i64) -> !llvm<"float*">
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Get the second element of the array (note 0-based indexing).
|
2019-04-06 14:56:49 +08:00
|
|
|
%c1 = llvm.constant(1) : !llvm.i64
|
|
|
|
%1 = llvm.getelementptr %0[%c1] : (!llvm<"float*">, !llvm.i64)
|
2019-03-07 17:27:09 +08:00
|
|
|
-> !llvm<"float*">
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Store a constant into this element.
|
2019-04-06 14:56:49 +08:00
|
|
|
%cf = llvm.constant(42.0 : f32) : !llvm.float
|
2019-04-03 06:33:54 +08:00
|
|
|
llvm.store %cf, %1 : !llvm<"float*">
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Load the value from this element.
|
2019-04-03 06:33:54 +08:00
|
|
|
%3 = llvm.load %1 : !llvm<"float*">
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
#### Operations on values of aggregate type.
|
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
- `<value> = extractvalue <struct>[<index> (, <index>)+]`
|
|
|
|
- `<struct> = insertvalue <value>, <struct>[<index> (, <index>)+]`
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
In these operations, `<struct>` must be a value of wrapped LLVM IR structure
|
|
|
|
type and `<value>` must be a value that corresponds to one of the (nested)
|
|
|
|
structure element types.
|
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
Note the use of integer literals to designate subscripts, which is made possbile
|
|
|
|
by `extractvalue` and `insertvalue` must have constant subscripts. Internally,
|
|
|
|
they are modeled as array attributes.
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Get the value third element of the second element of a structure.
|
2019-04-03 06:33:54 +08:00
|
|
|
%0 = llvm.extractvalue %s[1, 2] : !llvm<"{i32, {i1, i8, i16}">
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Insert the value to the third element of the second element of a structure.
|
|
|
|
// Note that this returns a new structure-typed value.
|
2019-04-03 06:33:54 +08:00
|
|
|
%1 = llvm.insertvalue %0, %s[1, 2] : !llvm<"{i32, {i1, i8, i16}">
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
#### Terminator operations.
|
|
|
|
|
|
|
|
Branch operations:
|
|
|
|
|
|
|
|
- `br [<successor>(<operands>)]`
|
|
|
|
- `cond_br <condition> [<true-successor>(<true-operands>),`
|
|
|
|
`<false-successor>(<false-operands>)]`
|
|
|
|
|
|
|
|
In order to comply with MLIR design, branch operations in the LLVM IR dialect
|
|
|
|
pass arguments to basic blocks. Successors must be valid block MLIR identifiers
|
|
|
|
and operand lists for each of them must have the same types as the arguments of
|
|
|
|
the respective blocks. `<condition>` must be a wrapped LLVM IR `i1` type.
|
|
|
|
|
2019-02-26 18:02:26 +08:00
|
|
|
Since LLVM IR uses the name of the predecessor basic block to identify the
|
|
|
|
sources of a PHI node, it is invalid for two entries of the PHI node to indicate
|
|
|
|
different values coming from the same block. Therefore, `cond_br` in the LLVM IR
|
|
|
|
dialect disallows its successors to be the same block _if_ this block has
|
|
|
|
arguments.
|
|
|
|
|
2019-02-22 17:00:25 +08:00
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Branch without arguments.
|
|
|
|
^bb0:
|
2019-04-03 06:33:54 +08:00
|
|
|
llvm.br ^bb0
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Branch and pass arguments.
|
2019-04-06 14:56:49 +08:00
|
|
|
^bb1(%arg: !llvm.i32):
|
|
|
|
llvm.br ^bb1(%arg : !llvm.i32)
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
// Conditionally branch and pass arguments to one of the blocks.
|
2019-04-06 14:56:49 +08:00
|
|
|
llvm.cond_br %cond, ^bb0, %bb1(%arg : !llvm.i32)
|
2019-02-26 18:02:26 +08:00
|
|
|
|
|
|
|
// It's okay to use the same block without arguments, but probably useless.
|
2019-04-03 06:33:54 +08:00
|
|
|
llvm.cond_br %cond, ^bb0, ^bb0
|
2019-02-26 18:02:26 +08:00
|
|
|
|
|
|
|
// ERROR: Passing different arguments to the same block in a conditional branch.
|
2019-04-06 14:56:49 +08:00
|
|
|
llvm.cond_br %cond, ^bb1(%0 : !llvm.i32), ^bb1(%1 : !llvm.i32)
|
2019-02-26 18:02:26 +08:00
|
|
|
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
Call operations:
|
|
|
|
|
|
|
|
- `<r> = call(<operands>)`
|
2019-03-07 17:27:09 +08:00
|
|
|
- `call(<operands>)`
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-03-07 17:27:09 +08:00
|
|
|
In LLVM IR, functions may return either 0 or 1 value. LLVM IR dialect implements
|
|
|
|
this behavior by providing a variadic `call` operation for 0- and 1-result
|
|
|
|
functions. Even though MLIR supports multi-result functions, LLVM IR dialect
|
|
|
|
disallows them.
|
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
The `call` instruction supports both direct and indirect calls. Direct calls
|
|
|
|
start with a function name (`@`-prefixed) and indirect calls start with an SSA
|
|
|
|
value (`%`-prefixed). The direct callee, if present, is stored as a function
|
|
|
|
attribute `callee`. The trailing type of the instruction is always the MLIR
|
|
|
|
function type, which may be different from the indirect callee that has the
|
|
|
|
wrapped LLVM IR function type.
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
2019-03-07 17:27:09 +08:00
|
|
|
// Direct call without arguments and with one result.
|
2019-04-06 14:56:49 +08:00
|
|
|
%0 = llvm.call @foo() : () -> (!llvm.float)
|
2019-03-07 17:27:09 +08:00
|
|
|
|
|
|
|
// Direct call with arguments and without a result.
|
2019-04-06 14:56:49 +08:00
|
|
|
llvm.call @bar(%0) : (!llvm.float) -> ()
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-03-07 17:27:09 +08:00
|
|
|
// Indirect call with an argument and without a result.
|
2019-04-06 14:56:49 +08:00
|
|
|
llvm.call %1(%0) : (!llvm.float) -> ()
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
|
|
|
#### Miscellaneous operations.
|
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
Integer comparisons: `icmp "predicate" <lhs>, <rhs>`. The following predicate
|
|
|
|
values are supported:
|
2019-02-26 17:27:27 +08:00
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
- `eq` - equality comparison;
|
|
|
|
- `ne` - inequality comparison;
|
|
|
|
- `slt` - signed less-than comparison
|
|
|
|
- `sle` - signed less-than-or-equal comparison
|
|
|
|
- `sgt` - signed greater-than comparison
|
|
|
|
- `sge` - signed greater-than-or-equal comparison
|
|
|
|
- `ult` - unsigned less-than comparison
|
|
|
|
- `ule` - unsigned less-than-or-equal comparison
|
|
|
|
- `ugt` - unsigned greater-than comparison
|
|
|
|
- `uge` - unsigned greater-than-or-equal comparison
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Bitwise reinterpretation: `bitcast <value>`.
|
|
|
|
|
|
|
|
Selection: `select <condition>, <lhs>, <rhs>`.
|
|
|
|
|
2019-04-05 23:19:42 +08:00
|
|
|
### Pseudo-operations
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
These operations do not have LLVM IR counterparts but are necessary to map LLVM
|
|
|
|
IR into MLIR.
|
|
|
|
|
2019-04-05 23:19:42 +08:00
|
|
|
#### `llvm.constant`
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Unlike LLVM IR, MLIR does not have first-class constant values. Therefore, all
|
|
|
|
constants must be created as SSA values before being used in other operations.
|
|
|
|
`llvm.constant` creates such values for scalars and vectors. It has a mandatory
|
|
|
|
`value` attribute, which may be an integer, floating point attribute; splat,
|
|
|
|
dense or sparse attribute containing integers or floats. The type of the
|
2019-04-03 06:33:54 +08:00
|
|
|
attribute is one the corresponding MLIR standard types. It may be omitted for
|
|
|
|
`i64` and `f64` types that are implied. The operation produces a new SSA value
|
|
|
|
of the specified LLVM IR dialect type. The type of that value _must_ correspond
|
|
|
|
to the attribute type converted to LLVM IR.
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Examples:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
2019-04-03 06:33:54 +08:00
|
|
|
// Integer constant, internal i32 is mandatory
|
2019-04-06 14:56:49 +08:00
|
|
|
%0 = llvm.constant(42 : i32) : !llvm.i32
|
2019-04-03 06:33:54 +08:00
|
|
|
|
|
|
|
// It's okay to omit i64.
|
2019-04-06 14:56:49 +08:00
|
|
|
%1 = llvm.constant(42) : !llvm.i64
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
// Floating point constant.
|
2019-04-06 14:56:49 +08:00
|
|
|
%2 = llvm.constant(42.0 : f32) : !llvm.float
|
2019-02-22 17:00:25 +08:00
|
|
|
|
2019-04-03 06:33:54 +08:00
|
|
|
// Splat vector constant,.
|
|
|
|
%3 = llvm.constant(splat<vector<4xf32>, 1.0>) : !llvm<"<4 x float>">
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|
|
|
|
|
2019-04-05 23:19:42 +08:00
|
|
|
#### `llvm.undef`
|
2019-02-22 17:00:25 +08:00
|
|
|
|
|
|
|
Unlike LLVM IR, MLIR does not have first-class undefined values. Such values
|
|
|
|
must be created as SSA values using `llvm.undef`. This operation has no operands
|
|
|
|
or attributes. It creates an undefined value of the specified LLVM IR dialect
|
|
|
|
type wrapping an LLVM IR structure type.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
```mlir {.mlir}
|
|
|
|
// Create a structure with a 32-bit integer followed by a float.
|
2019-04-03 06:33:54 +08:00
|
|
|
%0 = llvm.undef : !llvm<"{i32, float}">
|
2019-02-22 17:00:25 +08:00
|
|
|
```
|