Skip to content

Commit c996d49

Browse files
committed
[mlir] Update the documentation for defining types
The documentation needs a refresh now that "kinds" are no longer a concept. This revision also adds mentions to a few other new concepts, e.g. traits and interfaces. Differential Revision: https://reviews.llvm.org/D86182
1 parent d9ff48d commit c996d49

File tree

4 files changed

+111
-155
lines changed

4 files changed

+111
-155
lines changed

mlir/docs/Rationale/Rationale.md

-7
Original file line numberDiff line numberDiff line change
@@ -458,13 +458,6 @@ experience. When we do, we should chat with benoitjacob@ and
458458
This section describes the design decisions that shaped the dialect extensible
459459
type system present in MLIR.
460460

461-
#### Reserving dialect type kinds
462-
463-
Dialects that wish to define type extensions must reserve a range of type kinds
464-
within a '.def' file within the core IR library. This means that every dialect
465-
wishing to define custom types must modify this file, but it guarantees that all
466-
type casting checkings are performed in O(1) time.
467-
468461
#### Interactions between dialects
469462

470463
There are two different interactions between dialects that are important to

mlir/docs/Tutorials/DefiningAttributesAndTypes.md

+104-111
Original file line numberDiff line numberDiff line change
@@ -1,110 +1,71 @@
11
# Defining Dialect Attributes and Types
22

33
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.
77

88
See [MLIR specification](LangRef.md) for more information about MLIR, the
99
structure of the IR, operations, etc.
1010

1111
## Types
1212

1313
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`.
5518

5619
### Defining the type class
5720

5821
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
6023
defining a new `Type` it isn't always necessary to define a new storage class.
6124
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.
7945

8046
```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
8349
/// 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).
8752
class SimpleType : public Type::TypeBase<SimpleType, Type, TypeStorage> {
8853
public:
8954
/// Inherit some necessary constructors from 'TypeBase'.
9055
using Base::Base;
9156

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);
10060
};
10161
```
10262
10363
#### Parametric types
10464
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.
10869
10970
##### Defining a type storage
11071
@@ -113,10 +74,10 @@ parametric type instance. The storage classes must obey the following:
11374
11475
* Inherit from the base type storage class `TypeStorage`.
11576
* 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.
11778
* Provide a construction method that is used to allocate a new instance of the
11879
storage class.
119-
- `Storage *construct(TypeStorageAllocator &, const KeyTy &key)`
80+
- `static Storage *construct(TypeStorageAllocator &, const KeyTy &key)`
12081
* Provide a comparison method between the storage and `KeyTy`.
12182
- `bool operator==(const KeyTy &) const`
12283
* Provide a method to generate the `KeyTy` from a list of arguments passed to
@@ -165,6 +126,7 @@ struct ComplexTypeStorage : public TypeStorage {
165126
ComplexTypeStorage(key.first, key.second);
166127
}
167128
129+
/// The parametric data held by the storage class.
168130
unsigned nonZeroParam;
169131
Type integerType;
170132
};
@@ -173,16 +135,16 @@ struct ComplexTypeStorage : public TypeStorage {
173135
##### Type class definition
174136

175137
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.
179141

180142
```c++
181143
/// This class defines a parametric type. All derived types must inherit from
182144
/// 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).
186148
class ComplexType : public Type::TypeBase<ComplexType, Type,
187149
ComplexTypeStorage> {
188150
public:
@@ -195,8 +157,8 @@ public:
195157
static ComplexType get(unsigned param, Type type) {
196158
// Call into a helper 'get' method in 'TypeBase' to get a uniqued instance
197159
// 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);
200162
}
201163

202164
/// This method is used to get an instance of the 'ComplexType', defined at
@@ -206,8 +168,8 @@ public:
206168
static ComplexType getChecked(unsigned param, Type type, Location location) {
207169
// Call into a helper 'getChecked' method in 'TypeBase' to get a uniqued
208170
// 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);
211173
}
212174

213175
/// This method is used to verify the construction invariants passed into the
@@ -237,28 +199,26 @@ public:
237199
};
238200
```
239201
240-
#### Types with a mutable component
202+
#### Mutable types
241203
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.
246206
247207
##### Defining a type storage
248208
249209
In addition to the requirements for the type storage class for parametric types,
250210
the storage class for types with a mutable component must additionally obey the
251211
following.
252212
253-
* The mutable component must not participate in the storage key.
213+
* The mutable component must not participate in the storage `KeyTy`.
254214
* Provide a mutation method that is used to modify an existing instance of the
255215
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
257217
whether the modification was successful.
258218
- `LogicalResult mutate(StorageAllocator &allocator, Args ...&& args)`
259219
260220
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.
262222
263223
```c++
264224
/// Here we define a storage class for a RecursiveType that is identified by its
@@ -307,10 +267,9 @@ struct RecursiveTypeStorage : public TypeStorage {
307267

308268
##### Type class definition
309269

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.
314273

315274
```c++
316275
class RecursiveType : public Type::TypeBase<RecursiveType, Type,
@@ -323,19 +282,21 @@ public:
323282
/// and returns the type with uninitialized body.
324283
static RecursiveType get(MLIRContext *ctx, StringRef name) {
325284
// 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);
328287
}
329288

330289
/// Now we can change the mutable component of the type. This is an instance
331290
/// method callable on an already existing RecursiveType.
332291
void setBody(Type body) {
333292
// Call into the base to mutate the type.
334293
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
336296
// custom logic for handling mutation failures.
337297
assert(succeeded(result) &&
338298
"attempting to change the body of an already-initialized type");
299+
339300
// Avoid unused-variable warning when building without assertions.
340301
(void) result;
341302
}
@@ -356,23 +317,55 @@ public:
356317
### Registering types with a Dialect
357318
358319
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.
361322
362323
```c++
363324
struct MyDialect : public Dialect {
364325
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>();
367328
}
368329
};
369330
```
370331

371332
### Parsing and Printing
372333

373334
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.
376369
377370
## Attributes
378371

0 commit comments

Comments
 (0)