1
1
# Defining Dialect Attributes and Types
2
2
3
3
This document is a quickstart to defining dialect specific extensions to the
4
- [ attribute] ( LangRef.md#attributes ) and [ type system ] ( LangRef.md#type-system ) .
5
- The main part of the tutorial focuses on defining types, but the instructions
6
- are nearly identical for defining attributes.
4
+ [ attribute] ( LangRef.md#attributes ) and [ type] ( LangRef.md#type-system ) systems in
5
+ MLIR. The main part of this tutorial focuses on defining types, but the
6
+ instructions are nearly identical for defining attributes.
7
7
8
8
See [ MLIR specification] ( LangRef.md ) for more information about MLIR, the
9
9
structure of the IR, operations, etc.
10
10
11
11
## Types
12
12
13
13
Types in MLIR (like attributes, locations, and many other things) are
14
- value-typed. This means that instances of ` Type ` should be passed around
15
- by-value, as opposed to by-pointer or by-reference. The ` Type ` class in itself
16
- acts as a wrapper around an internal storage object that is uniqued within an
17
- instance of an ` MLIRContext ` .
18
-
19
- ### Reserving a range of type kinds
20
-
21
- Types in MLIR rely on having a unique ` kind ` value to ensure that casting checks
22
- remain extremely
23
- efficient ([ rationale] ( ../Rationale/Rationale.md#reserving-dialect-type-kinds ) ). For a dialect
24
- author, this means that a range of type ` kind ` values must be explicitly, and
25
- statically, reserved. A dialect can reserve a range of values by adding a new
26
- entry to the
27
- [ DialectSymbolRegistry] ( https://github.com/llvm/llvm-project/blob/master/mlir/include/mlir/IR/DialectSymbolRegistry.def ) .
28
- To support out-of-tree and experimental dialects, the registry predefines a set
29
- of privates ranges, ` PRIVATE_EXPERIMENTAL_[0-9] ` , that are free for immediate
30
- use.
31
-
32
- ``` c++
33
- DEFINE_SYM_KIND_RANGE (LINALG) // Linear Algebra Dialect
34
- DEFINE_SYM_KIND_RANGE(TOY) // Toy language (tutorial) Dialect
35
-
36
- // The following ranges are reserved for experimenting with MLIR dialects in a
37
- // private context without having to register them here.
38
- DEFINE_SYM_KIND_RANGE(PRIVATE_EXPERIMENTAL_0)
39
- ```
40
-
41
- For the sake of this tutorial, we will use the predefined
42
- `PRIVATE_EXPERIMENTAL_0` range. These definitions will provide a range in the
43
- Type::Kind enum to use when defining the derived types.
44
-
45
- ```c++
46
- namespace MyTypes {
47
- enum Kinds {
48
- // These kinds will be used in the examples below.
49
- Simple = Type::Kind::FIRST_PRIVATE_EXPERIMENTAL_0_TYPE,
50
- Complex,
51
- Recursive
52
- };
53
- }
54
- ```
14
+ value-typed. This means that instances of ` Type ` are passed around by-value, as
15
+ opposed to by-pointer or by-reference. The ` Type ` class in itself acts as a
16
+ wrapper around an internal storage object that is uniqued within an instance of
17
+ an ` MLIRContext ` .
55
18
56
19
### Defining the type class
57
20
58
21
As described above, ` Type ` objects in MLIR are value-typed and rely on having an
59
- implicitly internal storage object that holds the actual data for the type. When
22
+ implicit internal storage object that holds the actual data for the type. When
60
23
defining a new ` Type ` it isn't always necessary to define a new storage class.
61
24
So before defining the derived ` Type ` , it's important to know which of the two
62
- classes of ` Type ` we are defining. Some types are _ primitives_ meaning they do
63
- not have any parameters and are singletons uniqued by kind, like the
64
- [ ` index ` type] ( LangRef.md#index-type ) . Parametric types on the other hand, have
65
- additional information that differentiates different instances of the same
66
- ` Type ` kind. For example the [ ` integer ` type] ( LangRef.md#integer-type ) has a
67
- bitwidth, making ` i8 ` and ` i16 ` be different instances of
68
- [ ` integer ` type] ( LangRef.md#integer-type ) . Types can also have a mutable
69
- component, which can be used, for example, to construct self-referring recursive
70
- types. The mutable component _ cannot_ be used to differentiate types within the
71
- same kind, so usually such types are also parametric where the parameters serve
72
- to identify them.
73
-
74
- #### Simple non-parametric types
75
-
76
- For simple parameterless types, we can jump straight into defining the derived
77
- type class. Given that these types are uniqued solely on ` kind ` , we don't need
78
- to provide our own storage class.
25
+ classes of ` Type ` we are defining:
26
+
27
+ Some types are _ singleton_ in nature, meaning they have no parameters and only
28
+ ever have one instance, like the [ ` index ` type] ( LangRef.md#index-type ) .
29
+
30
+ Other types are _ parametric_ , and contain additional information that
31
+ differentiates different instances of the same ` Type ` . For example the
32
+ [ ` integer ` type] ( LangRef.md#integer-type ) contains a bitwidth, with ` i8 ` and
33
+ ` i16 ` representing different instances of
34
+ [ ` integer ` type] ( LangRef.md#integer-type ) . _ Parametric_ may also contain a
35
+ mutable component, which can be used, for example, to construct self-referring
36
+ recursive types. The mutable component _ cannot_ be used to differentiate
37
+ instances of a type class, so usually such types contain other parametric
38
+ components that serve to identify them.
39
+
40
+ #### Singleton types
41
+
42
+ For singleton types, we can jump straight into defining the derived type class.
43
+ Given that only one instance of such types may exist, there is no need to
44
+ provide our own storage class.
79
45
80
46
``` c++
81
- // / This class defines a simple parameterless type. All derived types must
82
- // / inherit from the CRTP class 'Type::TypeBase'. It takes as template
47
+ // / This class defines a simple parameterless singleton type. All derived types
48
+ // / must inherit from the CRTP class 'Type::TypeBase'. It takes as template
83
49
// / parameters the concrete type (SimpleType), the base class to use (Type),
84
- // / using the default storage class (TypeStorage) as the storage type.
85
- // / 'Type::TypeBase' also provides several utility methods to simplify type
86
- // / construction.
50
+ // / the internal storage class (the default TypeStorage here), and an optional
51
+ // / set of type traits and interfaces(detailed below).
87
52
class SimpleType : public Type ::TypeBase<SimpleType, Type, TypeStorage> {
88
53
public:
89
54
/// Inherit some necessary constructors from 'TypeBase'.
90
55
using Base::Base;
91
56
92
- /// This method is used to get an instance of the 'SimpleType'. Given that
93
- /// this is a parameterless type, it just needs to take the context for
94
- /// uniquing purposes.
95
- static SimpleType get(MLIRContext * context) {
96
- // Call into a helper 'get' method in 'TypeBase' to get a uniqued instance
97
- // of this type.
98
- return Base::get(context, MyTypes::Simple);
99
- }
57
+ /// The ` TypeBase ` class provides the following utility methods for
58
+ /// constructing instances of this type:
59
+ /// static SimpleType get(MLIRContext * ctx);
100
60
};
101
61
```
102
62
103
63
#### Parametric types
104
64
105
- Parametric types are those that have additional construction or uniquing
106
- constraints outside of the type `kind`. As such, these types require defining a
107
- type storage class.
65
+ Parametric types are those with additional construction or uniquing constraints,
66
+ that allow for representing multiple different instances of a single class. As
67
+ such, these types require defining a type storage class to contain the
68
+ parametric data.
108
69
109
70
##### Defining a type storage
110
71
@@ -113,10 +74,10 @@ parametric type instance. The storage classes must obey the following:
113
74
114
75
* Inherit from the base type storage class `TypeStorage`.
115
76
* Define a type alias, `KeyTy`, that maps to a type that uniquely identifies
116
- an instance of the parent type.
77
+ an instance of the derived type.
117
78
* Provide a construction method that is used to allocate a new instance of the
118
79
storage class.
119
- - `Storage *construct(TypeStorageAllocator &, const KeyTy &key)`
80
+ - `static Storage *construct(TypeStorageAllocator &, const KeyTy &key)`
120
81
* Provide a comparison method between the storage and `KeyTy`.
121
82
- `bool operator==(const KeyTy &) const`
122
83
* Provide a method to generate the `KeyTy` from a list of arguments passed to
@@ -165,6 +126,7 @@ struct ComplexTypeStorage : public TypeStorage {
165
126
ComplexTypeStorage(key.first, key.second);
166
127
}
167
128
129
+ /// The parametric data held by the storage class.
168
130
unsigned nonZeroParam;
169
131
Type integerType;
170
132
};
@@ -173,16 +135,16 @@ struct ComplexTypeStorage : public TypeStorage {
173
135
##### Type class definition
174
136
175
137
Now that the storage class has been created, the derived type class can be
176
- defined. This structure is similar to the
177
- [simple type](#simple-non-parametric-types), except for a bit more of the
178
- functionality of `Type::TypeBase` is put to use.
138
+ defined. This structure is similar to [ singleton types ] ( #singleton-types ) ,
139
+ except that a bit more of the functionality provided by ` Type::TypeBase ` is put
140
+ to use.
179
141
180
142
``` c++
181
143
// / This class defines a parametric type. All derived types must inherit from
182
144
// / the CRTP class 'Type::TypeBase'. It takes as template parameters the
183
- // / concrete type (ComplexType), the base class to use (Type), and the storage
184
- // / class (ComplexTypeStorage). 'Type::TypeBase' also provides several utility
185
- // / methods to simplify type construction and verification .
145
+ // / concrete type (ComplexType), the base class to use (Type), the storage
146
+ // / class (ComplexTypeStorage), and an optional set of traits and
147
+ // / interfaces(detailed below) .
186
148
class ComplexType : public Type ::TypeBase<ComplexType, Type,
187
149
ComplexTypeStorage> {
188
150
public:
@@ -195,8 +157,8 @@ public:
195
157
static ComplexType get(unsigned param, Type type) {
196
158
// Call into a helper 'get' method in 'TypeBase' to get a uniqued instance
197
159
// of this type. All parameters to the storage class are passed after the
198
- // type kind .
199
- return Base::get(type.getContext(), MyTypes::Complex, param, type);
160
+ // context .
161
+ return Base::get(type.getContext(), param, type);
200
162
}
201
163
202
164
/// This method is used to get an instance of the 'ComplexType', defined at
@@ -206,8 +168,8 @@ public:
206
168
static ComplexType getChecked(unsigned param, Type type, Location location) {
207
169
// Call into a helper 'getChecked' method in 'TypeBase' to get a uniqued
208
170
// instance of this type. All parameters to the storage class are passed
209
- // after the type kind .
210
- return Base::getChecked(location, MyTypes::Complex, param, type);
171
+ // after the location .
172
+ return Base::getChecked(location, param, type);
211
173
}
212
174
213
175
/// This method is used to verify the construction invariants passed into the
@@ -237,28 +199,26 @@ public:
237
199
};
238
200
```
239
201
240
- #### Types with a mutable component
202
+ #### Mutable types
241
203
242
- Types with a mutable component require defining a type storage class regardless
243
- of being parametric. The storage contains both the parameters and the mutable
244
- component and is accessed in a thread-safe way by the type support
245
- infrastructure.
204
+ Types with a mutable component are special instances of parametric types that
205
+ allow for mutating certain parameters after construction.
246
206
247
207
##### Defining a type storage
248
208
249
209
In addition to the requirements for the type storage class for parametric types,
250
210
the storage class for types with a mutable component must additionally obey the
251
211
following.
252
212
253
- * The mutable component must not participate in the storage key .
213
+ * The mutable component must not participate in the storage `KeyTy` .
254
214
* Provide a mutation method that is used to modify an existing instance of the
255
215
storage. This method modifies the mutable component based on arguments,
256
- using `allocator` for any new dynamically-allocated storage, and indicates
216
+ using `allocator` for any newly dynamically-allocated storage, and indicates
257
217
whether the modification was successful.
258
218
- `LogicalResult mutate(StorageAllocator &allocator, Args ...&& args)`
259
219
260
220
Let's define a simple storage for recursive types, where a type is identified by
261
- its name and can contain another type including itself.
221
+ its name and may contain another type including itself.
262
222
263
223
```c++
264
224
/// Here we define a storage class for a RecursiveType that is identified by its
@@ -307,10 +267,9 @@ struct RecursiveTypeStorage : public TypeStorage {
307
267
308
268
##### Type class definition
309
269
310
- Having defined the storage class, we can define the type class itself. This is
311
- similar to parametric types. ` Type::TypeBase ` provides a ` mutate ` method that
312
- forwards its arguments to the ` mutate ` method of the storage and ensures the
313
- modification happens under lock.
270
+ Having defined the storage class, we can define the type class itself.
271
+ ` Type::TypeBase ` provides a ` mutate ` method that forwards its arguments to the
272
+ ` mutate ` method of the storage and ensures the mutation happens safely.
314
273
315
274
``` c++
316
275
class RecursiveType : public Type ::TypeBase<RecursiveType, Type,
@@ -323,19 +282,21 @@ public:
323
282
/// and returns the type with uninitialized body.
324
283
static RecursiveType get(MLIRContext * ctx, StringRef name) {
325
284
// Call into the base to get a uniqued instance of this type. The parameter
326
- // (name) is passed after the kind .
327
- return Base::get(ctx, MyTypes::Recursive, name);
285
+ // (name) is passed after the context .
286
+ return Base::get(ctx, name);
328
287
}
329
288
330
289
/// Now we can change the mutable component of the type. This is an instance
331
290
/// method callable on an already existing RecursiveType.
332
291
void setBody(Type body) {
333
292
// Call into the base to mutate the type.
334
293
LogicalResult result = Base::mutate(body);
335
- // Most types expect mutation to always succeed, but types can implement
294
+
295
+ // Most types expect the mutation to always succeed, but types can implement
336
296
// custom logic for handling mutation failures.
337
297
assert(succeeded(result) &&
338
298
"attempting to change the body of an already-initialized type");
299
+
339
300
// Avoid unused-variable warning when building without assertions.
340
301
(void) result;
341
302
}
@@ -356,23 +317,55 @@ public:
356
317
### Registering types with a Dialect
357
318
358
319
Once the dialect types have been defined, they must then be registered with a
359
- `Dialect`. This is done via similar mechanism to
360
- [operations](LangRef.md#operations), `addTypes`.
320
+ `Dialect`. This is done via a similar mechanism to
321
+ [operations](LangRef.md#operations), with the `addTypes` method .
361
322
362
323
```c++
363
324
struct MyDialect : public Dialect {
364
325
MyDialect(MLIRContext *context) : Dialect(/*name=*/"mydialect", context) {
365
- /// Add these types to the dialect.
366
- addTypes<SimpleType, ComplexType>();
326
+ /// Add these defined types to the dialect.
327
+ addTypes<SimpleType, ComplexType, RecursiveType >();
367
328
}
368
329
};
369
330
```
370
331
371
332
### Parsing and Printing
372
333
373
334
As a final step after registration, a dialect must override the ` printType ` and
374
- ` parseType ` hooks. These enable native support for roundtripping the type in the
375
- textual IR.
335
+ ` parseType ` hooks. These enable native support for round-tripping the type in
336
+ the textual ` .mlir ` .
337
+
338
+ ``` c++
339
+ class MyDialect : public Dialect {
340
+ public:
341
+ /// Parse an instance of a type registered to the dialect.
342
+ Type parseType(DialectAsmParser &parser) const override;
343
+
344
+ /// Print an instance of a type registered to the dialect.
345
+ void printType(Type type, DialectAsmPrinter &printer) const override;
346
+ };
347
+ ```
348
+
349
+ These methods take an instance of a high-level parser or printer that allows for
350
+ easily implementing the necessary functionality. As described in the
351
+ [MLIR language reference](../../LangRef.md#dialect-types), dialect types are
352
+ generally represented as: `! dialect-namespace < type-data >`, with a pretty
353
+ form available under certain circumstances. The responsibility of our parser and
354
+ printer is to provide the `type-data` bits.
355
+
356
+ ### Traits
357
+
358
+ Similarly to operations, `Type` classes may attach `Traits` that provide
359
+ additional mixin methods and other data. `Trait` classes may be specified via
360
+ the trailing template argument of the `Type::TypeBase` class. See the main
361
+ [`Trait`](../Traits.md) documentation for more information on defining and using
362
+ traits.
363
+
364
+ ### Interfaces
365
+
366
+ Similarly to operations, `Type` classes may attach `Interfaces` to provide an
367
+ abstract interface into the type. See the main [`Interface`](../Interfaces.md)
368
+ documentation for more information on defining and using interfaces.
376
369
377
370
## Attributes
378
371
0 commit comments