diff --git a/mlir/include/mlir/Dialect/CMakeLists.txt b/mlir/include/mlir/Dialect/CMakeLists.txt --- a/mlir/include/mlir/Dialect/CMakeLists.txt +++ b/mlir/include/mlir/Dialect/CMakeLists.txt @@ -17,6 +17,7 @@ add_subdirectory(Shape) add_subdirectory(SPIRV) add_subdirectory(StandardOps) +add_subdirectory(Target) add_subdirectory(Tensor) add_subdirectory(Tosa) add_subdirectory(Vector) diff --git a/mlir/include/mlir/Dialect/Target/CMakeLists.txt b/mlir/include/mlir/Dialect/Target/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/Target/CMakeLists.txt @@ -0,0 +1,2 @@ +add_mlir_dialect(Target target) +add_mlir_doc(Target -gen-dialect-doc TargetDialect Dialects/) diff --git a/mlir/include/mlir/Dialect/Target/Target.h b/mlir/include/mlir/Dialect/Target/Target.h new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/Target/Target.h @@ -0,0 +1,112 @@ +//===- Target.h - Target MLIR Dialect ---------------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Defines the dialect containing the objects pertaining to target information. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_DIALECT_TARGET_TARGET_H +#define MLIR_DIALECT_TARGET_TARGET_H + +#include "mlir/IR/Attributes.h" +#include "mlir/IR/Dialect.h" +#include "mlir/Interfaces/DataLayoutInterfaces.h" + +namespace mlir { +namespace impl { +class DataLayoutEntryStorage; +class DataLayoutSpecStorage; +} // namespace impl + +//===----------------------------------------------------------------------===// +// DataLayoutEntryAttr +//===----------------------------------------------------------------------===// + +/// A data layout entry attribute is a key-value pair where the key is a type or +/// an identifier and the value is another attribute. These entries form a data +/// layout specification. +class DataLayoutEntryAttr + : public Attribute::AttrBase { +public: + using Base::Base; + + /// The keyword used for this attribute in custom syntax. + constexpr static const char *kAttrKeyword = "dl_entry"; + + /// Returns the entry with the given key and value. + static DataLayoutEntryAttr get(Identifier key, Attribute value); + static DataLayoutEntryAttr get(Type key, Attribute value); + + /// Returns the key of this entry. + DataLayoutEntryKey getKey() const; + + /// Returns the value of this entry. + Attribute getValue() const; + + /// Parses an instance of this attribute. + static DataLayoutEntryAttr parse(DialectAsmParser &parser); + + /// Prints this attribute. + void print(DialectAsmPrinter &os) const; +}; + +//===----------------------------------------------------------------------===// +// DataLayoutSpecAttr +//===----------------------------------------------------------------------===// + +/// A data layout specification is a list of entries that specify (partial) data +/// layout information. It is expected to be attached to operations that serve +/// as scopes for data layout requests. +class DataLayoutSpecAttr + : public Attribute::AttrBase { +public: + using Base::Base; + + /// The keyword used for this attribute in custom syntax. + constexpr static const char *kAttrKeyword = "dl_spec"; + + /// Returns the specification containing the given list of keys. + static DataLayoutSpecAttr get(MLIRContext *ctx, + ArrayRef entries); + + /// Returns the specification containing the given list of keys. If the list + /// contains duplicate keys or is otherwise invalid, reports errors at the + /// specified location and returns null. + static DataLayoutSpecAttr + getChecked(Location loc, ArrayRef entries); + + /// Checks that the given list of entries does not contain duplicate keys. + static LogicalResult + verifyConstructionInvariants(Location loc, + ArrayRef entries); + + /// Combines this specification with other, enclosing specifications listed + /// from outermost to innermost. This overwrites the older entries with the + /// same key as the newer entries if the entries are compatible. Returns null + /// if the specifications are not compatible. + DataLayoutSpecAttr combineWith(ArrayRef specs) const; + + /// Returns the list of entries. + DataLayoutEntryListRef getEntries() const; + + /// Parses an instance of this attribute. + static DataLayoutSpecAttr parse(DialectAsmParser &parser); + + /// Prints this attribute. + void print(DialectAsmPrinter &os) const; +}; + +} // namespace mlir + +#include "mlir/Dialect/Target/TargetDialect.h.inc" + +#endif // MLIR_DIALECT_TARGET_TARGET_H diff --git a/mlir/include/mlir/Dialect/Target/Target.td b/mlir/include/mlir/Dialect/Target/Target.td new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/Target/Target.td @@ -0,0 +1,14 @@ +//===- Target.td - Target information dialect ---------------*- tablegen -*-==// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef TARGET_TD +#define TARGET_TD + +include "mlir/Dialect/Target/TargetBase.td" + +#endif // TARGET_TD diff --git a/mlir/include/mlir/Dialect/Target/TargetBase.td b/mlir/include/mlir/Dialect/Target/TargetBase.td new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/Target/TargetBase.td @@ -0,0 +1,54 @@ +//===- TargetBase.td - Target information dialect common ----*- tablegen -*-==// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef TARGET_BASE +#define TARGET_BASE + +include "mlir/IR/OpBase.td" + +def Target_Dialect : Dialect { + let name = "target"; + let cppNamespace = "::mlir"; + let hasOperationAttrVerify = 1; + + let description = [{ + The target dialect is intended to hold attributes and other components + pertaining to descriptions of compilation targets. + }]; + + let extraClassDeclaration = [{ + constexpr static const char *kDataLayoutAttrName = "target.dl_spec"; + constexpr static const char *kDataLayoutEndiannessKey = "target.endianness"; + constexpr static const char *kDataLayoutEndiannessBig = "big"; + constexpr static const char *kDataLayoutEndiannessLittle = "little"; + }]; +} + +def Target_DataLayoutEntryAttr : DialectAttr< + Target_Dialect, + CPred<"$_self.isa<::mlir::DataLayoutEntryAttr>()">, + "Target data layout entry"> { + let storageType = "::mlir::DataLayoutEntryAttr"; + let returnType = "::mlir::DataLayoutEntryAttr"; + let convertFromStorage = "$_self"; +} + +def Target_DataLayoutSpecAttr : DialectAttr< + Target_Dialect, + CPred<"$_self.isa<::mlir::DataLayoutSpecAttr>()">, + "Target data layout specification"> { + let storageType = "::mlir::DataLayoutSpecAttr"; + let returnType = "::mlir::DataLayoutSpecAttr"; + let convertFromStorage = "$_self"; +} + +def HasTargetDataLayout : NativeOpTrait<"HasTargetDataLayout"> { + let cppNamespace = "::mlir"; +} + +#endif // TARGET_BASE diff --git a/mlir/include/mlir/Dialect/Target/Traits.h b/mlir/include/mlir/Dialect/Target/Traits.h new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Dialect/Target/Traits.h @@ -0,0 +1,47 @@ +//===- Traits.h - Trait Declaration for MLIR Target dialect -----*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Defines the dialect containing the objects pertaining to target information. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_DIALECT_TARGET_TRAITS_H +#define MLIR_DIALECT_TARGET_TRAITS_H + +#include "mlir/IR/OpDefinition.h" +#include "mlir/Interfaces/DataLayoutInterfaces.h" + +namespace mlir { +class DataLayoutSpecAttr; + +namespace impl { +LogicalResult verifyHasTargetDataLayoutTrait(Operation *op); +DataLayoutSpecInterface getDataLayoutSpec(Operation *op); +} // namespace impl + +/// Trait to be used by operations willing to use the implementation of the +/// data layout interfaces provided by the Target dialect. +template +class HasTargetDataLayout + : public OpTrait::TraitBase { +public: + /// Verifies that the operation to which this trait is attached is valid for + /// the trait, i.e., that it implements the data layout operation interface. + static LogicalResult verifyTrait(Operation *op) { + return impl::verifyHasTargetDataLayoutTrait(op); + } + + /// Returns the data layout specification as provided by the Target dialect + /// specification attribute. + DataLayoutSpecInterface getDataLayoutSpec() { + return impl::getDataLayoutSpec(this->getOperation()); + } +}; +} // namespace mlir + +#endif // MLIR_DIALECT_TARGET_TRAITS_H diff --git a/mlir/include/mlir/IR/OpBase.td b/mlir/include/mlir/IR/OpBase.td --- a/mlir/include/mlir/IR/OpBase.td +++ b/mlir/include/mlir/IR/OpBase.td @@ -240,7 +240,7 @@ string description = ?; // A list of dialects this dialect will load on construction as dependencies. - // These are dialects that this dialect may involved in canonicalization + // These are dialects that this dialect may involve in canonicalization // pattern or interfaces. list dependentDialects = []; diff --git a/mlir/include/mlir/InitAllDialects.h b/mlir/include/mlir/InitAllDialects.h --- a/mlir/include/mlir/InitAllDialects.h +++ b/mlir/include/mlir/InitAllDialects.h @@ -39,6 +39,7 @@ #include "mlir/Dialect/SPIRV/IR/SPIRVDialect.h" #include "mlir/Dialect/Shape/IR/Shape.h" #include "mlir/Dialect/StandardOps/IR/Ops.h" +#include "mlir/Dialect/Target/Target.h" #include "mlir/Dialect/Tensor/IR/Tensor.h" #include "mlir/Dialect/Tosa/IR/TosaOps.h" #include "mlir/Dialect/Vector/VectorOps.h" @@ -75,6 +76,7 @@ ROCDL::ROCDLDialect, SDBMDialect, shape::ShapeDialect, + TargetDialect, tensor::TensorDialect, tosa::TosaDialect>(); // clang-format on diff --git a/mlir/include/mlir/Interfaces/CMakeLists.txt b/mlir/include/mlir/Interfaces/CMakeLists.txt --- a/mlir/include/mlir/Interfaces/CMakeLists.txt +++ b/mlir/include/mlir/Interfaces/CMakeLists.txt @@ -9,3 +9,13 @@ add_mlir_interface(VectorInterfaces) add_mlir_interface(ViewLikeInterface) +set(LLVM_TARGET_DEFINITIONS DataLayoutInterfaces.td) +mlir_tablegen(DataLayoutAttrInterface.h.inc -gen-attr-interface-decls) +mlir_tablegen(DataLayoutAttrInterface.cpp.inc -gen-attr-interface-defs) +mlir_tablegen(DataLayoutOpInterface.h.inc -gen-op-interface-decls) +mlir_tablegen(DataLayoutOpInterface.cpp.inc -gen-op-interface-defs) +mlir_tablegen(DataLayoutTypeInterface.h.inc -gen-type-interface-decls) +mlir_tablegen(DataLayoutTypeInterface.cpp.inc -gen-type-interface-defs) +add_public_tablegen_target(MLIRDataLayoutInterfacesIncGen) +add_dependencies(mlir-generic-headers MLIRDataLayoutInterfacesIncGen) + diff --git a/mlir/include/mlir/Interfaces/DataLayoutInterfaces.h b/mlir/include/mlir/Interfaces/DataLayoutInterfaces.h new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Interfaces/DataLayoutInterfaces.h @@ -0,0 +1,172 @@ +//===- CopyOpInterface.h - copy operations interface ------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Defines the interfaces for the data layout specification, operations to which +// they can be attached, types subject to data layout and dialects containing +// data layout entries. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_INTERFACES_DATALAYOUTINTERFACES_H +#define MLIR_INTERFACES_DATALAYOUTINTERFACES_H + +#include "mlir/IR/DialectInterface.h" +#include "mlir/IR/OpDefinition.h" +#include "llvm/ADT/DenseMap.h" + +namespace mlir { +class DataLayout; +class DataLayoutEntryInterface; +using DataLayoutEntryKey = llvm::PointerUnion; +// Using explicit SmallVector size because we cannot infer the size from the +// forward declaration, and we need the typedef in the actual declaration. +using DataLayoutEntryList = llvm::SmallVector; +using DataLayoutEntryListRef = llvm::ArrayRef; +class DataLayoutOpInterface; +class DataLayoutSpecInterface; + +namespace detail { +/// Default handler for the type size request. Computes results for built-in +/// types and dispatches to the DataLayoutTypeInterface for other types. +unsigned getDefaultTypeSize(Type type, const DataLayout &dataLayout, + ArrayRef params); + +/// Default handler for the required alignemnt request. Computes results for +/// built-in types and dispatches to the DataLayoutTypeInterface for other +/// types. +unsigned getDefaultABIAlignment(Type type, const DataLayout &dataLayout, + ArrayRef params); + +/// Default handler for the preferred alignemnt request. Computes results for +/// built-in types and dispatches to the DataLayoutTypeInterface for other +/// types. +unsigned +getDefaultPreferredAlignment(Type type, const DataLayout &dataLayout, + ArrayRef params); + +/// Given a list of data layout entries, returns a new list containing the +/// entries with keys having the given type ID, i.e. belonging to the same type +/// class. +DataLayoutEntryList filterEntriesForType(DataLayoutEntryListRef entries, + TypeID typeID); + +/// Given a list of data layout entries, returns the entry that has the given +/// identifier as key, if such an entry exists in the list. +DataLayoutEntryInterface +filterEntryForIdentifier(DataLayoutEntryListRef entries, Identifier id); + +/// Verifies that the operation implementing the data layout interface is valid. +/// This calls the verifier of the spec attribute and checks if the layout is +/// compatible with specs attached to the enclosing operations. +LogicalResult verifyDataLayoutOp(DataLayoutOpInterface op); + +/// Verifies that a data layout spec is valid. This dispatches to individual +/// entry verifiers, and then to the verifiers implemented by the relevant type +/// and dialect interfaces for type and identifier keys respectively. +LogicalResult verifyDataLayoutSpec(DataLayoutSpecInterface spec, Location loc); +} // namespace detail +} // namespace mlir + +#include "mlir/Interfaces/DataLayoutAttrInterface.h.inc" +#include "mlir/Interfaces/DataLayoutOpInterface.h.inc" +#include "mlir/Interfaces/DataLayoutTypeInterface.h.inc" + +namespace mlir { + +//===----------------------------------------------------------------------===// +// DataLayoutDialectInterface +//===----------------------------------------------------------------------===// + +/// An interface to be implemented by dialects that can have identifiers in the +/// data layout specification entries. Provides hooks for verifying the entry +/// validity and combining two entries. +class DataLayoutDialectInterface + : public DialectInterface::Base { +public: + DataLayoutDialectInterface(Dialect *dialect) : Base(dialect) {} + + /// Checks whether the given data layout entry is valid and reports any errors + /// at the provided location. Derived classes should override this. + virtual LogicalResult verifyEntry(DataLayoutEntryInterface entry, + Location loc) const { + return success(); + } + + /// Default implementation of entry combination that combines identical + /// entries and returns null otherwise. + static DataLayoutEntryInterface + defaultCombine(DataLayoutEntryInterface outer, + DataLayoutEntryInterface inner) { + if (!outer || outer == inner) + return inner; + return {}; + } + + /// Combines two entries with identifies that belong to this dialect. Returns + /// the combined entry or null if the entries are not compatible. Derived + /// classes likely need to reimplement this. + virtual DataLayoutEntryInterface + combine(DataLayoutEntryInterface outer, + DataLayoutEntryInterface inner) const { + return defaultCombine(outer, inner); + } +}; + +//===----------------------------------------------------------------------===// +// DataLayout +//===----------------------------------------------------------------------===// + +/// The main mechanism for performing data layout queries. Instances of this +/// class can be created for an operation implementing DataLayoutOpInterface. +/// Upon construction, a layout spec combining that of the given operation with +/// all its ancestors will be computed and used to handle further requests. For +/// efficiency, results to all requests will be cached in this object. +/// Therefore, if the data layout spec for the scoping operation, or any of the +/// enclosing operations, changes, the cache is no longer valid. The user is +/// responsible creating a new DataLayout object after any spec change. In debug +/// mode, the cache validity is being checked in every request. +class DataLayout { +public: + explicit DataLayout(DataLayoutOpInterface op); + + /// Returns the size of the given type in the current scope. + unsigned getTypeSize(Type t) const; + + /// Returns the required alignment of the given type in the current scope. + unsigned getTypeABIAlignment(Type t) const; + + /// Returns the preferred of the given type in the current scope. + unsigned getTypePreferredAlignment(Type t) const; + +private: + /// Combined layout spec at the given scope. + const DataLayoutSpecInterface originalLayout; + +#ifndef NDEBUG + /// List of enclosing layout specs. + SmallVector layoutStack; +#endif + + /// Asserts that the cache is still valid. Expensive in debug mode. No-op in + /// release mode. + void checkValid() const; + + /// Operation defining the scope of requests. + // TODO: this is mutable because the generated interface method are not const. + // Update the generator to support const methods and change this to const. + mutable DataLayoutOpInterface scope; + + /// Caches for individual requests. + mutable DenseMap sizes; + mutable DenseMap abiAlignments; + mutable DenseMap preferredAlignments; +}; + +} // namespace mlir + +#endif // MLIR_INTERFACES_DATALAYOUTINTERFACES_H diff --git a/mlir/include/mlir/Interfaces/DataLayoutInterfaces.td b/mlir/include/mlir/Interfaces/DataLayoutInterfaces.td new file mode 100644 --- /dev/null +++ b/mlir/include/mlir/Interfaces/DataLayoutInterfaces.td @@ -0,0 +1,323 @@ +//===- DataLayoutInterfaces.td - Data layout interfaces ----*- tablegen -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// Defines the interfaces for the data layout specification, operations to which +// they can be attached, and types that are subject to data layout. +// +//===----------------------------------------------------------------------===// + +#ifndef MLIR_DATALAYOUTINTERFACES +#define MLIR_DATALAYOUTINTERFACES + +include "mlir/IR/OpBase.td" + +//===----------------------------------------------------------------------===// +// Attribute interfaces +//===----------------------------------------------------------------------===// + +def DataLayoutEntryInterface : AttrInterface<"DataLayoutEntryInterface"> { + let cppNamespace = "::mlir"; + + let description = [{ + Attribute interface describing an entry in a data layout specification. + + A data layout specification entry is a key-value pair. Its key is either a + type, when the entry is related to a type or a class of types, or an + identifier, when it is not. `DataLayoutEntryKey` is an alias allowing one + to use both key types. Its value is an arbitrary attribute that is + interpreted either by the type for type keys or by the dialect containing + the identifier for identifier keys. The interface provides a hook that + can be used by specific implementations to delegate the verification of + attribute fitness for a particular key to the relevant type or dialect. + }]; + + let methods = [ + InterfaceMethod< + /*description=*/"Returns the key of the this layout entry.", + /*retTy=*/"::mlir::DataLayoutEntryKey", + /*methodName=*/"getKey", + /*args=*/(ins) + >, + InterfaceMethod< + /*description=*/"Returns the value of this layout entry.", + /*retTy=*/"::mlir::Attribute", + /*methodName=*/"getValue", + /*args=*/(ins) + >, + InterfaceMethod< + /*description=*/"Checks that the entry is well-formed, reports errors " + "at the provided location.", + /*retTy=*/"::mlir::LogicalResult", + /*methodName=*/"verify", + /*args=*/(ins "::mlir::Location":$loc), + /*methodBody=*/"", + /*defaultImplementation=*/[{ return ::mlir::success(); }] + > + ]; + + let extraClassDeclaration = [{ + /// Returns `true` if the key of this entry is a type. + bool isTypeEntry() { + return getKey().is<::mlir::Type>(); + } + }]; +} + +def DataLayoutSpecInterface : AttrInterface<"DataLayoutSpecInterface"> { + let cppNamespace = "::mlir"; + + let description = [{ + Attribute interface describing a data layout specification. + + A data layout specification is seen as a sequence of entries, each of which + is an attribute implementing the data layout entry interface. It assumes + a contiguous underlying storage for entries. The interface provides a hook + for implementations to verify the well-formedness of the specification, + with a default implementation that verifies the absence of entries with + duplicate keys and the well-formedness of each individual entry before + dispatching to the type or dialect the entry is associated with. + + Data layout specifications may need to be combined in case they appear on + nested operations subject to layout, or to ensure the validity of layout + modification. Concerete specification attributes must implement the + corresponding hook. + }]; + // The underlying storage being contiguous may be revised in the future, but + // care must be taken to avoid materializing or copying the entire list of + // entries. + + let methods = [ + InterfaceMethod< + /*description=*/"Combines the current layout with the given list of " + "layouts, provided from the outermost (oldest) to the " + "innermost (newest). Returns null on failure.", + /*retTy=*/"::mlir::DataLayoutSpecInterface", + /*methodName=*/"combineWith", + /*args=*/(ins "::llvm::ArrayRef":$specs) + >, + InterfaceMethod< + /*description=*/"Returns the list of layout entries.", + /*retTy=*/"::mlir::DataLayoutEntryListRef", + /*methodName=*/"getEntries", + /*args=*/(ins) + >, + // Implementations may override this if they have an efficient lookup + // mechanism. + InterfaceMethod< + /*description=*/"Returns a copy of the entries related to a specific " + "type class regardles of type parameters. ", + /*retTy=*/"::mlir::DataLayoutEntryList", + /*methodName=*/"getSpecForType", + /*args=*/(ins "::mlir::TypeID":$type), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::filterEntriesForType($_attr.getEntries(), type); + }] + >, + // Implementations may override this if they have an efficient lookup + // mechanism. + InterfaceMethod< + /*description=*/"Returns the entry related to the given identifier, if " + "present.", + /*retTy=*/"::mlir::DataLayoutEntryInterface", + /*methodName=*/"getSpecForIdentifier", + /*args=*/(ins "::mlir::Identifier":$identifier), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::filterEntryForIdentifier($_attr.getEntries(), + identifier); + }] + >, + InterfaceMethod< + /*description=*/"Verifies the validity of the specification and reports " + "any errors at the given location.", + /*retTy=*/"::mlir::LogicalResult", + /*methodName=*/"verify", + /*args=*/(ins "::mlir::Location":$loc), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::verifyDataLayoutSpec($_attr, loc); + }] + >, + ]; + + let extraClassDeclaration = [{ + /// Returns a copy of the entries related to the type specified as template + /// parameter. + template + DataLayoutEntryList getSpecForType() { + return getSpecForType(TypeID::get()); + } + + /// Populates the given maps with lists of entries grouped by the type or + /// identifier they are associated with. Users are not expected to call this + /// method directly. + void bucketEntriesByType( + ::llvm::DenseMap<::mlir::TypeID, ::mlir::DataLayoutEntryList> &types, + ::llvm::DenseMap<::mlir::Identifier, + ::mlir::DataLayoutEntryInterface> &ids); + }]; +} + +//===----------------------------------------------------------------------===// +// Operation interface +//===----------------------------------------------------------------------===// + +def DataLayoutOpInterface : OpInterface<"DataLayoutOpInterface"> { + let cppNamespace = "::mlir"; + + let description = [{ + Interface for operations that can have a data layout specification attached. + + The `DataLayout` object, which can be used for data layout queries, can be + constructed for such operations. The absence of a data layout specification + must be handled without failing. + + Concrete operations must implement the hook returning the data layout + specification. They may optionally override the methods used in data layout + queries, default implementations of which provide predefined answers for + built-in types and dispatch to the type interface for all other types. These + methods must be idempotent, that is return the same result on repeated + queries with the same parameters. They are declared static and therefore + have no access to the operation or its attributes. Instead, they receive a + list of data layout entries relevant to the request. The entries are known + to have passed the spec and entry verifier. + }]; + + let methods = [ + InterfaceMethod< + /*description=*/"Returns the data layout specification for this op, or " + "null if it does not exist.", + /*retTy=*/"DataLayoutSpecInterface", + /*methodName=*/"getDataLayoutSpec", + /*args=*/(ins) + >, + StaticInterfaceMethod< + /*description=*/"Returns the size of the given type computed using the " + "relevant entries. The data layout object can be used " + "for recursive queries.", + /*retTy=*/"unsigned", + /*methodName=*/"getTypeSize", + /*args=*/(ins "::mlir::Type":$type, + "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::getDefaultTypeSize(type, dataLayout, params); + }] + >, + StaticInterfaceMethod< + /*description=*/"Returns the alignment required by the ABI for the given " + "type computed using the relevant entries. The data " + "layout object can be used for recursive queries.", + /*retTy=*/"unsigned", + /*methodName=*/"getTypeABIAlignment", + /*args=*/(ins "::mlir::Type":$type, + "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::getDefaultABIAlignment(type, dataLayout, params); + }] + >, + StaticInterfaceMethod< + /*description=*/"Returns the alignment preferred by the given type " + "computed using the relevant entries. The data layout" + "object can be used for recursive queries.", + /*retTy=*/"unsigned", + /*methodName=*/"getTypePreferredAlignment", + /*args=*/(ins "::mlir::Type":$type, + "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params), + /*methodBody=*/"", + /*defaultImplementation=*/[{ + return ::mlir::detail::getDefaultPreferredAlignment(type, dataLayout, + params); + }] + >, + ]; + + let verify = [{ return ::mlir::detail::verifyDataLayoutOp($_op); }]; +} + +//===----------------------------------------------------------------------===// +// Type interface +//===----------------------------------------------------------------------===// + +def DataLayoutTypeInterface : TypeInterface<"DataLayoutTypeInterface"> { + let cppNamespace = "::mlir"; + + let description = [{ + Interface for types subject to data layout. + + Types willing to be supported by the data layout subsystem should implement + this interface by providing implementations of functions querying their + size, required and preferred alignment. Each of this functions accepts as + arguments a data layout object that can be used to perform recursive queries + in the same scope, and a list of data layout entries relevant to this type. + Specifically, the entries are those that have as key _any instance_ of the + same type class as the current type. For example, if IntegerType had + implemented this interface, it would have received the entries with keys i1, + i2, i8, etc. regardless of the bitwidth of this type. This mechanism allows + types to "interpolate" the results in a type-specific way instead of listing + all possible types in the specification. + + The list of entries may be empty, in which case the type must provide a + reasonable default value. The entries in the list are known to have passed + the spec and the entry verifiers, as well as the type-specifid verifier if + provided. + + In case of nested layout specs or spec changes, the type can override a hook + indicating whether the outer (old) and the inner (new) spec are compatible. + }]; + + let methods = [ + InterfaceMethod< + /*description=*/"Returns the size of this type in bytes.", + /*retTy=*/"unsigned", + /*methodName=*/"getTypeSize", + /*args=*/(ins "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params) + >, + InterfaceMethod< + /*description=*/"Returns the ABI-required alignment for this type.", + /*retTy=*/"unsigned", + /*methodName=*/"getABIAlignment", + /*args=*/(ins "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params) + >, + InterfaceMethod< + /*description=*/"Returns the preferred alignemnt for this type.", + /*retTy=*/"unsigned", + /*methodName=*/"getPreferredAlignment", + /*args=*/(ins "const ::mlir::DataLayout &":$dataLayout, + "::mlir::DataLayoutEntryListRef":$params) + >, + InterfaceMethod< + /*desc=*/"Returns true if the two lists of entries are compatible.", + /*retTy=*/"bool", + /*methodName=*/"areCompatible", + /*args=*/(ins "::mlir::DataLayoutEntryListRef":$oldLayout, + "::mlir::DataLayoutEntryListRef":$newLayout), + /*methodBody=*/"", + /*defaultImplementation=*/[{ return true; }] + >, + InterfaceMethod< + /*description=*/"Verifies that the given list of entries is valid for " + "this type", + /*retTy=*/"::mlir::LogicalResult", + /*methodName=*/"verifyEntries", + /*args=*/(ins "::mlir::DataLayoutEntryListRef":$entries, + "::mlir::Location":$loc), + /*methodBody=*/"", + /*defaultImplementation=*/[{ return ::mlir::success(); }] + >, + ]; +} + +#endif // MLIR_DATALAYOUTINTERFACES diff --git a/mlir/lib/Dialect/CMakeLists.txt b/mlir/lib/Dialect/CMakeLists.txt --- a/mlir/lib/Dialect/CMakeLists.txt +++ b/mlir/lib/Dialect/CMakeLists.txt @@ -18,6 +18,7 @@ add_subdirectory(Shape) add_subdirectory(SPIRV) add_subdirectory(StandardOps) +add_subdirectory(Target) add_subdirectory(Tensor) add_subdirectory(Tosa) add_subdirectory(Vector) diff --git a/mlir/lib/Dialect/Target/CMakeLists.txt b/mlir/lib/Dialect/Target/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/Target/CMakeLists.txt @@ -0,0 +1,11 @@ +add_mlir_dialect_library(MLIRTarget + Target.cpp + Traits.cpp + + DEPENDS + MLIRTargetIncGen + + LINK_LIBS PUBLIC + MLIRIR + MLIRDataLayoutInterfaces + ) diff --git a/mlir/lib/Dialect/Target/Target.cpp b/mlir/lib/Dialect/Target/Target.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/Target/Target.cpp @@ -0,0 +1,374 @@ +//===- Target.cpp - Target MLIR Dialect Implementation --------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Dialect/Target/Target.h" +#include "mlir/IR/Builders.h" +#include "mlir/IR/BuiltinDialect.h" +#include "mlir/IR/Dialect.h" +#include "mlir/IR/DialectImplementation.h" +#include "llvm/ADT/TypeSwitch.h" + +using namespace mlir; + +//===----------------------------------------------------------------------===// +// DataLayoutEntryAttr +//===----------------------------------------------------------------------===// + +namespace mlir { +namespace impl { +class DataLayoutEntryStorage : public AttributeStorage { +public: + using KeyTy = std::pair; + + DataLayoutEntryStorage(DataLayoutEntryKey entryKey, Attribute value) + : entryKey(entryKey), value(value) {} + + static DataLayoutEntryStorage *construct(AttributeStorageAllocator &allocator, + const KeyTy &key) { + return new (allocator.allocate()) + DataLayoutEntryStorage(key.first, key.second); + } + + bool operator==(const KeyTy &other) const { + return other.first == entryKey && other.second == value; + } + + DataLayoutEntryKey entryKey; + Attribute value; +}; +} // namespace impl +} // namespace mlir + +DataLayoutEntryAttr DataLayoutEntryAttr::get(Identifier key, Attribute value) { + return Base::get(key.getContext(), key, value); +} + +DataLayoutEntryAttr DataLayoutEntryAttr::get(Type key, Attribute value) { + return Base::get(key.getContext(), key, value); +} + +DataLayoutEntryKey DataLayoutEntryAttr::getKey() const { + return getImpl()->entryKey; +} + +Attribute DataLayoutEntryAttr::getValue() const { return getImpl()->value; } + +/// Parses an attribute with syntax: +/// attr ::= `#target.` `dl_entry` `<` (type | quoted-string) `,` attr `>` +DataLayoutEntryAttr DataLayoutEntryAttr::parse(DialectAsmParser &parser) { + if (failed(parser.parseLess())) + return {}; + + Type type = nullptr; + StringRef identifier; + llvm::SMLoc idLoc = parser.getCurrentLocation(); + OptionalParseResult parsedType = parser.parseOptionalType(type); + if (parsedType.hasValue() && failed(parsedType.getValue())) + return {}; + if (!parsedType.hasValue()) { + OptionalParseResult parsedString = parser.parseOptionalString(&identifier); + if (!parsedString.hasValue() || failed(parsedString.getValue())) { + parser.emitError(idLoc) << "expected a type or a quoted string"; + return {}; + } + } + + Attribute value; + if (failed(parser.parseComma()) || failed(parser.parseAttribute(value)) || + failed(parser.parseGreater())) + return {}; + + return type ? get(type, value) + : get(parser.getBuilder().getIdentifier(identifier), value); +} + +void DataLayoutEntryAttr::print(DialectAsmPrinter &os) const { + os << DataLayoutEntryAttr::kAttrKeyword << "<"; + if (auto type = getKey().dyn_cast()) + os << type; + else + os << "\"" << getKey().get().strref() << "\""; + os << ", " << getValue() << ">"; +} + +//===----------------------------------------------------------------------===// +// DataLayoutSpecAttr +//===----------------------------------------------------------------------===// + +namespace mlir { +namespace impl { +class DataLayoutSpecStorage : public AttributeStorage { +public: + using KeyTy = ArrayRef; + + DataLayoutSpecStorage(ArrayRef entries) + : entries(entries) {} + + bool operator==(const KeyTy &key) const { return key == entries; } + + static DataLayoutSpecStorage *construct(AttributeStorageAllocator &allocator, + const KeyTy &key) { + return new (allocator.allocate()) + DataLayoutSpecStorage(allocator.copyInto(key)); + } + + ArrayRef entries; +}; +} // namespace impl +} // namespace mlir + +DataLayoutSpecAttr +DataLayoutSpecAttr::get(MLIRContext *ctx, + ArrayRef entries) { + return Base::get(ctx, entries); +} + +DataLayoutSpecAttr +DataLayoutSpecAttr::getChecked(Location loc, + ArrayRef entries) { + return Base::getChecked(loc, entries); +} + +LogicalResult DataLayoutSpecAttr::verifyConstructionInvariants( + Location loc, ArrayRef entries) { + DenseSet types; + DenseSet ids; + for (DataLayoutEntryInterface entry : entries) { + if (auto type = entry.getKey().dyn_cast()) { + if (types.count(type)) + return emitError(loc) << "repeated layout entry key: " << type; + types.insert(type); + } else { + auto id = entry.getKey().get(); + if (ids.count(id)) + return emitError(loc) << "repeated layout entry key: " << id; + ids.insert(id); + } + } + return success(); +} + +/// Given a list of old and a list of new entries, overwrites old entries with +/// new ones if they have matching keys, appends new entries to the old entry +/// list otherwise. +static void +overwriteDuplicateEntries(SmallVectorImpl &oldEntries, + ArrayRef newEntries) { + for (DataLayoutEntryInterface entry : newEntries) { + // We expect a small (dozens) number of entries, so it is practically + // cheaper to iterate over the list linearly rather than to create an + // auxiliary hashmap to avoid duplication. Also note that we never need to + // check for duplicate keys the values that were added from `newEntries`. + bool replaced = false; + for (unsigned i = 0, e = oldEntries.size(); i < e; ++i) { + if (oldEntries[i].getKey() == entry.getKey()) { + oldEntries[i] = entry; + replaced = true; + break; + } + } + if (!replaced) + oldEntries.push_back(entry); + } +} + +/// Combines a data layout spec into the given lists of entries organized by +/// type class and identifier, overwriting them if necessary. Fails to combine +/// if the two entries with identical keys are not compatible. +static LogicalResult +combineOneSpec(DataLayoutSpecInterface spec, + DenseMap &entriesForType, + DenseMap &entriesForID) { + // A missing spec should be fine. + if (!spec) + return success(); + + DenseMap newEntriesForType; + DenseMap newEntriesForID; + spec.bucketEntriesByType(newEntriesForType, newEntriesForID); + + // Try overwriting the old entries with the new ones. + for (const auto &kvp : newEntriesForType) { + if (!entriesForType.count(kvp.first)) { + entriesForType[kvp.first] = std::move(kvp.second); + continue; + } + + Type typeSample = kvp.second.front().getKey().get(); + assert(&typeSample.getDialect() != + typeSample.getContext()->getLoadedDialect() && + "unexpected data layout entry for built-in type"); + + auto interface = typeSample.cast(); + if (!interface.areCompatible(entriesForType.lookup(kvp.first), kvp.second)) + return failure(); + + overwriteDuplicateEntries(entriesForType[kvp.first], kvp.second); + } + + for (const auto &kvp : newEntriesForID) { + Identifier id = kvp.second.getKey().get(); + Dialect *dialect = id.getDialect(); + if (!entriesForID.count(id)) { + entriesForID[id] = kvp.second; + continue; + } + + // Attempt to combine the enties using the dialect interface. If the + // dialect is not loaded for some reason, use the default combinator + // that conservatively accepts identical entries only. + entriesForID[id] = + dialect ? dialect->getRegisteredInterface() + ->combine(entriesForID[id], kvp.second) + : DataLayoutDialectInterface::defaultCombine(entriesForID[id], + kvp.second); + if (!entriesForID[id]) + return failure(); + } + + return success(); +} + +DataLayoutSpecAttr +DataLayoutSpecAttr::combineWith(ArrayRef specs) const { + // Only combine with attributes of the same kind. + // TODO: reconsider this when the need arises. + if (llvm::any_of(specs, [](DataLayoutSpecInterface spec) { + return spec && !spec.isa(); + })) + return {}; + + // Combine all specs in order, with `this` being the last one. + DenseMap entriesForType; + DenseMap entriesForID; + for (DataLayoutSpecInterface spec : specs) + if (failed(combineOneSpec(spec, entriesForType, entriesForID))) + return nullptr; + if (failed(combineOneSpec(*this, entriesForType, entriesForID))) + return nullptr; + + // Rebuild the linear list of entries. + SmallVector entries; + llvm::append_range( + entries, + llvm::map_range( + entriesForID, + [](const DenseMap::value_type + &v) { return v.second; })); + for (const auto &kvp : entriesForType) + llvm::append_range(entries, kvp.getSecond()); + + return DataLayoutSpecAttr::get(getContext(), entries); +} + +DataLayoutEntryListRef DataLayoutSpecAttr::getEntries() const { + return getImpl()->entries; +} + +/// Parses an attribute with syntax +/// attr ::= `#target.` `dl_spec` `<` attr-list? `>` +/// attr-list ::= attr +/// | attr `,` attr-list +DataLayoutSpecAttr DataLayoutSpecAttr::parse(DialectAsmParser &parser) { + if (failed(parser.parseLess())) + return {}; + + // Empty spec. + if (succeeded(parser.parseOptionalGreater())) + return get(parser.getBuilder().getContext(), {}); + + SmallVector entries; + do { + entries.emplace_back(); + if (failed(parser.parseAttribute(entries.back()))) + return {}; + } while (succeeded(parser.parseOptionalComma())); + + if (failed(parser.parseGreater())) + return {}; + return getChecked(parser.getEncodedSourceLoc(parser.getNameLoc()), entries); +} + +void DataLayoutSpecAttr::print(DialectAsmPrinter &os) const { + os << DataLayoutSpecAttr::kAttrKeyword << "<"; + llvm::interleaveComma(getEntries(), os); + os << ">"; +} + +//===----------------------------------------------------------------------===// +// TargetDialect +//===----------------------------------------------------------------------===// + +namespace { +class TargetDataLayoutInterface : public DataLayoutDialectInterface { +public: + using DataLayoutDialectInterface::DataLayoutDialectInterface; + + LogicalResult verifyEntry(DataLayoutEntryInterface entry, + Location loc) const final { + StringRef entryName = entry.getKey().get().strref(); + if (entryName == TargetDialect::kDataLayoutEndiannessKey) { + auto value = entry.getValue().dyn_cast(); + if (value && + (value.getValue() == TargetDialect::kDataLayoutEndiannessBig || + value.getValue() == TargetDialect::kDataLayoutEndiannessLittle)) + return success(); + return emitError(loc) + << "'" << entryName + << "' data layout entry is expected to be either '" + << TargetDialect::kDataLayoutEndiannessBig << "' or '" + << TargetDialect::kDataLayoutEndiannessLittle << "'"; + } + return emitError(loc) << "unknown data layout entry name: " << entryName; + } +}; +} // namespace + +void TargetDialect::initialize() { + addAttributes(); + addInterfaces(); +} + +Attribute TargetDialect::parseAttribute(DialectAsmParser &parser, + Type type) const { + StringRef attrKind; + if (parser.parseKeyword(&attrKind)) + return {}; + + if (attrKind == DataLayoutEntryAttr::kAttrKeyword) + return DataLayoutEntryAttr::parse(parser); + if (attrKind == DataLayoutSpecAttr::kAttrKeyword) + return DataLayoutSpecAttr::parse(parser); + + parser.emitError(parser.getNameLoc(), "unknown attrribute type: ") + << attrKind; + return {}; +} + +void TargetDialect::printAttribute(Attribute attr, + DialectAsmPrinter &os) const { + llvm::TypeSwitch(attr) + .Case( + [&](auto a) { a.print(os); }) + .Default([](Attribute) { llvm_unreachable("unknown attribute kind"); }); +} + +LogicalResult TargetDialect::verifyOperationAttribute(Operation *op, + NamedAttribute attr) { + if (attr.first == TargetDialect::kDataLayoutAttrName) { + if (!attr.second.isa()) { + return op->emitError() + << "'" << TargetDialect::kDataLayoutAttrName + << "' is expected to be a #target.dl_spec attribute"; + } + return success(); + } + + return op->emitError() << "attribute '" << attr.first + << "' not supported by dialect"; +} diff --git a/mlir/lib/Dialect/Target/Traits.cpp b/mlir/lib/Dialect/Target/Traits.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Dialect/Target/Traits.cpp @@ -0,0 +1,28 @@ +//===- Traits.cpp - Traits for MLIR Target dialect ------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Dialect/Target/Traits.h" +#include "mlir/Dialect/Target/Target.h" +#include "mlir/Interfaces/DataLayoutInterfaces.h" + +using namespace mlir; + +LogicalResult mlir::impl::verifyHasTargetDataLayoutTrait(Operation *op) { + // TODO: consider having trait inheritance so that HasTargetDataLayout trait + // can inherit DataLayoutOpInterface::Trait and enforece the validity of the + // assertion below. + assert(isa(op) && + "HasTargetDataLayout trait unexpectedly attached to an op that does " + "not implement DataLayoutOpInterface"); + return success(); +} + +DataLayoutSpecInterface mlir::impl::getDataLayoutSpec(Operation *op) { + return op->getAttrOfType( + TargetDialect::kDataLayoutAttrName); +} diff --git a/mlir/lib/Interfaces/CMakeLists.txt b/mlir/lib/Interfaces/CMakeLists.txt --- a/mlir/lib/Interfaces/CMakeLists.txt +++ b/mlir/lib/Interfaces/CMakeLists.txt @@ -3,6 +3,7 @@ CastInterfaces.cpp ControlFlowInterfaces.cpp CopyOpInterface.cpp + DataLayoutInterfaces.cpp DerivedAttributeOpInterface.cpp InferTypeOpInterface.cpp LoopLikeInterface.cpp @@ -31,6 +32,7 @@ add_mlir_interface_library(CastInterfaces) add_mlir_interface_library(ControlFlowInterfaces) add_mlir_interface_library(CopyOpInterface) +add_mlir_interface_library(DataLayoutInterfaces) add_mlir_interface_library(DerivedAttributeOpInterface) add_mlir_interface_library(InferTypeOpInterface) add_mlir_interface_library(LoopLikeInterface) diff --git a/mlir/lib/Interfaces/DataLayoutInterfaces.cpp b/mlir/lib/Interfaces/DataLayoutInterfaces.cpp new file mode 100644 --- /dev/null +++ b/mlir/lib/Interfaces/DataLayoutInterfaces.cpp @@ -0,0 +1,308 @@ +//===- DataLayoutInterfaces.cpp - Data Layout Interface Implementation ----===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Interfaces/DataLayoutInterfaces.h" +#include "mlir/IR/BuiltinDialect.h" +#include "mlir/IR/BuiltinTypes.h" +#include "mlir/IR/Operation.h" + +using namespace mlir; + +//===----------------------------------------------------------------------===// +// Default implementations +//===----------------------------------------------------------------------===// + +/// Reports that the given type is missing the data layout information and +/// exits. +static LLVM_ATTRIBUTE_NORETURN void reportMissingDataLayout(Type type) { + std::string message; + llvm::raw_string_ostream os(message); + os << "neither the scoping op nor the type class provide data layout " + "information for " + << type; + llvm::report_fatal_error(os.str()); +} + +unsigned +mlir::detail::getDefaultTypeSize(Type type, const DataLayout &dataLayout, + ArrayRef params) { + if (type.isa() || type.isa()) + return llvm::divideCeil(type.getIntOrFloatBitWidth(), 8); + + // Sizes of vector types are rounded up to those of types with closest + // power-of-two number of elements. + // TODO: make this extensible. + if (auto vecType = type.dyn_cast()) + return llvm::PowerOf2Ceil(vecType.getNumElements()) * + dataLayout.getTypeSize(vecType.getElementType()); + + if (auto typeInterface = type.dyn_cast()) + return typeInterface.getTypeSize(dataLayout, params); + + reportMissingDataLayout(type); +} + +unsigned mlir::detail::getDefaultABIAlignment( + Type type, const DataLayout &dataLayout, + ArrayRef params) { + // Natural alignment is the closest power-of-two number above. + if (type.isa()) + return llvm::PowerOf2Ceil(dataLayout.getTypeSize(type)); + + if (auto typeInterface = type.dyn_cast()) + return typeInterface.getABIAlignment(dataLayout, params); + + reportMissingDataLayout(type); +} + +unsigned mlir::detail::getDefaultPreferredAlignment( + Type type, const DataLayout &dataLayout, + ArrayRef params) { + // Preferred alignment is same as natural, but at least 4. + if (type.isa()) + return std::max(dataLayout.getTypeABIAlignment(type), 4u); + + if (auto typeInterface = type.dyn_cast()) + return typeInterface.getPreferredAlignment(dataLayout, params); + + reportMissingDataLayout(type); +} + +DataLayoutEntryList +mlir::detail::filterEntriesForType(DataLayoutEntryListRef entries, + TypeID typeID) { + return llvm::to_vector<4>(llvm::make_filter_range( + entries, [typeID](DataLayoutEntryInterface entry) { + auto type = entry.getKey().dyn_cast(); + return type && type.getTypeID() == typeID; + })); +} + +DataLayoutEntryInterface +mlir::detail::filterEntryForIdentifier(DataLayoutEntryListRef entries, + Identifier id) { + auto *it = llvm::find_if(entries, [id](DataLayoutEntryInterface entry) { + if (!entry.getKey().is()) + return false; + return entry.getKey().get() == id; + }); + if (it == entries.end()) + return {}; + return *it; +} + +/// Populates `opsWithLayout` with the list of proper ancestors of `leaf` that +/// implement the `DataLayoutOpInterface`. +static void findProperAscendantsWithLayout( + Operation *leaf, SmallVectorImpl &opsWithLayout) { + if (!leaf) + return; + + while (auto opLayout = leaf->getParentOfType()) { + opsWithLayout.push_back(opLayout); + leaf = opLayout; + } +} + +/// Returns a layout spec that is a combination of the layout specs attached +/// to the given operation and all its ancestors. +static DataLayoutSpecInterface +getCombinedDataLayout(DataLayoutOpInterface leaf) { + if (!leaf) + return {}; + + SmallVector opsWithLayout; + findProperAscendantsWithLayout(leaf, opsWithLayout); + + // Fast track if there are no ancestors. + if (opsWithLayout.empty()) + return leaf.getDataLayoutSpec(); + + // Create the list of non-null specs (null/missing specs can be safely + // ignored) from the outermost to the innermost. + SmallVector specs; + specs.reserve(opsWithLayout.size()); + for (DataLayoutOpInterface op : llvm::reverse(opsWithLayout)) + if (auto current = op.getDataLayoutSpec()) + specs.push_back(current); + + // Combine the specs using the innermost as anchor. + if (DataLayoutSpecInterface current = leaf.getDataLayoutSpec()) + return current.combineWith(specs); + if (specs.empty()) + return {}; + return specs.back().combineWith(llvm::makeArrayRef(specs).drop_back()); +} + +LogicalResult mlir::detail::verifyDataLayoutOp(DataLayoutOpInterface op) { + DataLayoutSpecInterface spec = op.getDataLayoutSpec(); + // The layout specification may be missing and it's fine. + if (!spec) + return success(); + + if (failed(spec.verify(op.getLoc()))) + return failure(); + if (!getCombinedDataLayout(op)) { + InFlightDiagnostic diag = + op.emitError() + << "data layout is not a refinement of the layouts in enclosing ops"; + SmallVector opsWithLayout; + findProperAscendantsWithLayout(op, opsWithLayout); + for (DataLayoutOpInterface parent : opsWithLayout) + diag.attachNote(parent.getLoc()) << "enclosing op with data layout"; + return diag; + } + return success(); +} + +//===----------------------------------------------------------------------===// +// DataLayout +//===----------------------------------------------------------------------===// + +mlir::DataLayout::DataLayout(DataLayoutOpInterface op) + : originalLayout(getCombinedDataLayout(op)), scope(op) { + if (!originalLayout) + assert((!op || !op.getDataLayoutSpec()) && + "could not compute layout information for an op (failed to " + "combine attributes?)"); +#ifndef NDEBUG + SmallVector opsWithLayout; + findProperAscendantsWithLayout(op, opsWithLayout); + layoutStack = llvm::to_vector<2>( + llvm::map_range(opsWithLayout, [](DataLayoutOpInterface iface) { + return iface.getDataLayoutSpec(); + })); +#endif +} + +void mlir::DataLayout::checkValid() const { +#ifndef NDEBUG + SmallVector opsWithLayout; + findProperAscendantsWithLayout(scope, opsWithLayout); + assert(opsWithLayout.size() == layoutStack.size() && + "data layout object used, but no longer valid due to the change in " + "number of nested layouts"); + for (auto pair : llvm::zip(opsWithLayout, layoutStack)) { + Attribute newLayout = std::get<0>(pair).getDataLayoutSpec(); + Attribute origLayout = std::get<1>(pair); + assert(newLayout == origLayout && + "data layout object used, but no longer valid " + "due to the change in layout attributes"); + } +#endif + assert(((!scope && !this->originalLayout) || + (scope && this->originalLayout == getCombinedDataLayout(scope))) && + "data layout object used, but no longer valid due to the change in " + "layout spec"); +} + +/// Looks up the value for the given type key in the given cache. If there is no +/// such value in the cache, compute it using the given callback and put it in +/// the cache before returning. +static unsigned cachedLookup(Type t, DenseMap &cache, + function_ref compute) { + auto it = cache.find(t); + if (it != cache.end()) + return it->second; + + auto result = cache.try_emplace(t, compute(t)); + return result.first->second; +} + +unsigned mlir::DataLayout::getTypeSize(Type t) const { + checkValid(); + return cachedLookup(t, sizes, [&](Type ty) { + return (scope && originalLayout) + ? scope.getTypeSize( + ty, *this, originalLayout.getSpecForType(ty.getTypeID())) + : detail::getDefaultTypeSize(ty, *this, {}); + }); +} + +unsigned mlir::DataLayout::getTypeABIAlignment(Type t) const { + checkValid(); + return cachedLookup(t, abiAlignments, [&](Type ty) { + return (scope && originalLayout) + ? scope.getTypeABIAlignment( + ty, *this, originalLayout.getSpecForType(ty.getTypeID())) + : detail::getDefaultABIAlignment(ty, *this, {}); + }); +} + +unsigned mlir::DataLayout::getTypePreferredAlignment(Type t) const { + checkValid(); + return cachedLookup(t, preferredAlignments, [&](Type ty) { + return (scope && originalLayout) + ? scope.getTypePreferredAlignment( + ty, *this, originalLayout.getSpecForType(ty.getTypeID())) + : detail::getDefaultPreferredAlignment(ty, *this, {}); + }); +} + +//===----------------------------------------------------------------------===// +// DataLayoutSpecInterface +//===----------------------------------------------------------------------===// + +void DataLayoutSpecInterface::bucketEntriesByType( + DenseMap &types, + DenseMap &ids) { + for (DataLayoutEntryInterface entry : getEntries()) { + if (auto type = entry.getKey().dyn_cast()) + types[type.getTypeID()].push_back(entry); + else + ids[entry.getKey().get()] = entry; + } +} + +LogicalResult mlir::detail::verifyDataLayoutSpec(DataLayoutSpecInterface spec, + Location loc) { + // First, verify individual entries. + for (DataLayoutEntryInterface entry : spec.getEntries()) + if (failed(entry.verify(loc))) + return failure(); + + // Second, dispatch verifications of entry groups to types or dialects they + // are are associated with. + DenseMap types; + DenseMap ids; + spec.bucketEntriesByType(types, ids); + + for (const auto &kvp : types) { + auto sampleType = kvp.second.front().getKey().get(); + if (&sampleType.getDialect() == + spec.getContext()->getLoadedDialect()) + return emitError(loc) << "unexpected data layout for a built-in type"; + + auto dlType = sampleType.dyn_cast(); + if (!dlType) + return emitError(loc) + << "data layout specified for a type that does not support it"; + if (failed(dlType.verifyEntries(kvp.second, loc))) + return failure(); + } + + for (const auto &kvp : ids) { + Identifier identifier = kvp.second.getKey().get(); + Dialect *dialect = identifier.getDialect(); + + // Ignore attributes that belong to an unknown dialect, the dialect may + // actually implement the relevant interface but we don't know about that. + if (!dialect) + continue; + + auto *iface = dialect->getRegisteredInterface(); + if (failed(iface->verifyEntry(kvp.second, loc))) + return failure(); + } + + return success(); +} + +#include "mlir/Interfaces/DataLayoutAttrInterface.cpp.inc" +#include "mlir/Interfaces/DataLayoutOpInterface.cpp.inc" +#include "mlir/Interfaces/DataLayoutTypeInterface.cpp.inc" diff --git a/mlir/test/Dialect/Target/invalid.mlir b/mlir/test/Dialect/Target/invalid.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Target/invalid.mlir @@ -0,0 +1,73 @@ +// RUN: mlir-opt -split-input-file -verify-diagnostics %s + +// expected-error@below {{attribute 'target.unknown' not supported by dialect}} +"test.unknown_op"() { target.unknown } : () -> () + +// ----- + +// expected-error@below {{'target.dl_spec' is expected to be a #target.dl_spec attribute}} +"test.unknown_op"() { target.dl_spec = 42 } : () -> () + +// ----- + +// expected-error@below {{invalid kind of attribute specified}} +"test.unknown_op"() { target.dl_spec = #target.dl_spec<[]> } : () -> () + +// ----- + +// expected-error@below {{expected a type or a quoted string}} +"test.unknown_op"() { test.unknown_attr = #target.dl_entry<42, 42> } : () -> () + +// ----- + +// expected-error@below {{repeated layout entry key: test.id}} +"test.unknown_op"() { test.unknown_attr = #target.dl_spec< + #target.dl_entry<"test.id", 42>, + #target.dl_entry<"test.id", 43> +>} : () -> () + +// ----- + +// expected-error@below {{repeated layout entry key: 'i32'}} +"test.unknown_op"() { test.unknown_attr = #target.dl_spec< + #target.dl_entry, + #target.dl_entry +>} : () -> () + +// ----- + +// expected-error@below {{unknown attrribute type: unknown}} +"test.unknown_op"() { test.unknown_attr = #target.unknown } : () -> () + +// ----- + +// expected-error@below {{unknown data layout entry name: target.unknown_id}} +"test.op_with_data_layout"() ({ +}) { target.dl_spec = #target.dl_spec<#target.dl_entry<"target.unknown_id", 42>> } : () -> () + +// ----- + +// expected-error@below {{'target.endianness' data layout entry is expected to be either 'big' or 'little'}} +"test.op_with_data_layout"() ({ +}) { target.dl_spec = #target.dl_spec<#target.dl_entry<"target.endianness", "some">> } : () -> () + +// ----- + +// Mismatching entries don't combine. +"test.op_with_data_layout"() ({ + // expected-error@below {{data layout is not a refinement of the layouts in enclosing ops}} + // expected-note@above {{enclosing op with data layout}} + "test.op_with_data_layout"() { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () + "test.maybe_terminator_op"() : () -> () +}) { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 33>> } : () -> () + +// ----- + +// Layout not supported for built-in types. +// expected-error@below {{unexpected data layout for a built-in type}} +"test.op_with_data_layout"() { target.dl_spec = #target.dl_spec<#target.dl_entry> } : () -> () + +// ----- + +// expected-error@below {{data layout specified for a type that does not support it}} +"test.op_with_data_layout"() { target.dl_spec = #target.dl_spec<#target.dl_entry> } : () -> () diff --git a/mlir/test/Dialect/Target/roundtrip.mlir b/mlir/test/Dialect/Target/roundtrip.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Target/roundtrip.mlir @@ -0,0 +1,53 @@ +// RUN: mlir-opt %s | mlir-opt | FileCheck %s + +// Round-tripping the syntax. + +"test.unknown_op"() { + // CHECK: #target.dl_entry<"test.identifier", 42 : i64> + test.unknown_attr_1 = #target.dl_entry<"test.identifier", 42 : i64>, + // CHECK: #target.dl_entry<"test.identifier", affine_map<(d0) -> (d0)>> + test.unknown_attr_2 = #target.dl_entry<"test.identifier", affine_map<(d0) -> (d0)>>, + // CHECK: #target.dl_entry + test.unknown_attr_3 = #target.dl_entry, + // CHECK: #target.dl_entry, ["string", 10]> + test.unknown_attr_4 = #target.dl_entry, ["string", 10]>, + // CHECK: #target.dl_spec<> + test.unknown_attr_5 = #target.dl_spec<>, + // CHECK: #target.dl_spec<#target.dl_entry<"test.id", 42 : i32>> + test.unknown_attr_6 = #target.dl_spec<#target.dl_entry<"test.id", 42 : i32>>, + // CHECK: #target.dl_spec< + // CHECK: #target.dl_entry<"test.id1", 43 : index> + // CHECK: #target.dl_entry<"test.id2", 44 : index> + // CHECK: #target.dl_entry<"test.id3", 45 : index>> + test.unknown_attr_7 = #target.dl_spec< + #target.dl_entry<"test.id1", 43 : index>, + #target.dl_entry<"test.id2", 44 : index>, + #target.dl_entry<"test.id3", 45 : index>> +} : () -> () + +// +// Supported cases where we shouldn't fail. No need to file-check these, not +// triggering an error or an assertion is enough. +// + +// Should not fail on missing spec. +"test.op_with_data_layout"() : () -> () + +// Should not fail on empty spec. +"test.op_with_data_layout"() { target.dl_spec = #target.dl_spec<> }: () -> () + +// Should not fail on nested compatible layouts. +"test.op_with_data_layout"() ({ + "test.op_with_data_layout"() { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () + "test.maybe_terminator_op"() : () -> () +}) { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () + +// Should not fail on deeper nested compatible layouts. +"test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() + { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () + "test.maybe_terminator_op"() : () -> () + }) { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () + "test.maybe_terminator_op"() : () -> () +}) { target.dl_spec = #target.dl_spec<#target.dl_entry<"unknown.unknown", 32>> } : () -> () diff --git a/mlir/test/Interfaces/DataLayoutInterfaces/query.mlir b/mlir/test/Interfaces/DataLayoutInterfaces/query.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Interfaces/DataLayoutInterfaces/query.mlir @@ -0,0 +1,172 @@ +// RUN: mlir-opt --test-data-layout-query %s | FileCheck %s + +// CHECK-LABEL: @no_layout_builtin +func @no_layout_builtin() { + // CHECK: alignment = 4 + // CHECK: preferred = 4 + // CHECK: size = 4 + "test.data_layout_query"() : () -> i32 + // CHECK: alignment = 8 + // CHECK: preferred = 8 + // CHECK: size = 8 + "test.data_layout_query"() : () -> f64 + return +} + +// CHECK-LABEL: @no_layout_custom +func @no_layout_custom() { + // CHECK: alignment = 1 + // CHECK: preferred = 1 + // CHECK: size = 1 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + return +} + +// CHECK-LABEL: @layout_op_no_layout +func @layout_op_no_layout() { + "test.op_with_data_layout"() ({ + // CHECK: alignment = 1 + // CHECK: preferred = 1 + // CHECK: size = 1 + "test.data_layout_query"() : () -> !test.test_layed_out<1000> + "test.maybe_terminator"() : () -> () + }) : () -> () + return +} + +// CHECK-LABEL: @layout_op +func @layout_op() { + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 1 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + return +} + +// Make sure the outer op with layout may be missing the spec. +// CHECK-LABEL: @nested_inner_only +func @nested_inner_only() { + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 1 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + "test.maybe_terminator"() : () -> () + }) : () -> () + return +} + +// Make sure the inner op with layout may be missing the spec. +// CHECK-LABEL: @nested_outer_only +func @nested_outer_only() { + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 1 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) : () -> () + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + return +} + +// CHECK-LABEL: @nested_middle_only +func @nested_middle_only() { + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 1 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) : () -> () + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + "test.maybe_terminator"() : () -> () + }) : () -> () + return +} + +// CHECK-LABEL: @nested_combine_with_missing +func @nested_combine_with_missing() { + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 30 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) : () -> () + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + // CHECK: alignment = 1 + // CHECK: preferred = 30 + // CHECK: size = 42 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 42]>, + #target.dl_entry, ["preferred", 30]> + >}: () -> () + return +} + +// CHECK-LABEL: @nested_combine_all +func @nested_combine_all() { + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + "test.op_with_data_layout"() ({ + // CHECK: alignment = 20 + // CHECK: preferred = 30 + // CHECK: size = 3 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 3]>, + #target.dl_entry, ["preferred", 30]> + >} : () -> () + // CHECK: alignment = 20 + // CHECK: preferred = 30 + // CHECK: size = 10 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 10]>, + #target.dl_entry, ["alignment", 20]> + >} : () -> () + // CHECK: alignment = 1 + // CHECK: preferred = 30 + // CHECK: size = 42 + "test.data_layout_query"() : () -> !test.test_layed_out<10> + "test.maybe_terminator"() : () -> () + }) { target.dl_spec = #target.dl_spec< + #target.dl_entry, ["size", 42]>, + #target.dl_entry, ["preferred", 30]> + >}: () -> () + return +} diff --git a/mlir/test/lib/Dialect/Test/CMakeLists.txt b/mlir/test/lib/Dialect/Test/CMakeLists.txt --- a/mlir/test/lib/Dialect/Test/CMakeLists.txt +++ b/mlir/test/lib/Dialect/Test/CMakeLists.txt @@ -20,7 +20,7 @@ set(LLVM_TARGET_DEFINITIONS TestOps.td) mlir_tablegen(TestOps.h.inc -gen-op-decls) mlir_tablegen(TestOps.cpp.inc -gen-op-defs) -mlir_tablegen(TestOpsDialect.h.inc -gen-dialect-decls) +mlir_tablegen(TestOpsDialect.h.inc -gen-dialect-decls -dialect=test) mlir_tablegen(TestOpEnums.h.inc -gen-enum-decls) mlir_tablegen(TestOpEnums.cpp.inc -gen-enum-defs) mlir_tablegen(TestOpStructs.h.inc -gen-struct-attr-decls) @@ -45,6 +45,7 @@ LINK_LIBS PUBLIC MLIRControlFlowInterfaces + MLIRDataLayoutInterfaces MLIRDerivedAttributeOpInterface MLIRDialect MLIRIR @@ -53,6 +54,7 @@ MLIRPass MLIRStandard MLIRStandardOpsTransforms + MLIRTarget MLIRTransformUtils MLIRTransforms ) diff --git a/mlir/test/lib/Dialect/Test/TestDialect.h b/mlir/test/lib/Dialect/Test/TestDialect.h --- a/mlir/test/lib/Dialect/Test/TestDialect.h +++ b/mlir/test/lib/Dialect/Test/TestDialect.h @@ -15,6 +15,8 @@ #define MLIR_TESTDIALECT_H #include "TestInterfaces.h" +#include "mlir/Dialect/Target/Target.h" +#include "mlir/Dialect/Target/Traits.h" #include "mlir/Dialect/Traits.h" #include "mlir/IR/BuiltinOps.h" #include "mlir/IR/BuiltinTypes.h" @@ -30,6 +32,10 @@ #include "mlir/Interfaces/InferTypeOpInterface.h" #include "mlir/Interfaces/SideEffectInterfaces.h" +namespace mlir { +class TargetDialect; +} // namespace mlir + #include "TestOpEnums.h.inc" #include "TestOpInterfaces.h.inc" #include "TestOpStructs.h.inc" diff --git a/mlir/test/lib/Dialect/Test/TestDialect.cpp b/mlir/test/lib/Dialect/Test/TestDialect.cpp --- a/mlir/test/lib/Dialect/Test/TestDialect.cpp +++ b/mlir/test/lib/Dialect/Test/TestDialect.cpp @@ -9,6 +9,7 @@ #include "TestDialect.h" #include "TestTypes.h" #include "mlir/Dialect/StandardOps/IR/Ops.h" +#include "mlir/Dialect/Target/Target.h" #include "mlir/IR/BuiltinOps.h" #include "mlir/IR/DialectImplementation.h" #include "mlir/IR/PatternMatch.h" @@ -170,7 +171,7 @@ >(); addInterfaces(); - addTypes(); diff --git a/mlir/test/lib/Dialect/Test/TestOps.td b/mlir/test/lib/Dialect/Test/TestOps.td --- a/mlir/test/lib/Dialect/Test/TestOps.td +++ b/mlir/test/lib/Dialect/Test/TestOps.td @@ -9,6 +9,7 @@ #ifndef TEST_OPS #define TEST_OPS +include "mlir/Dialect/Target/TargetBase.td" include "mlir/IR/OpBase.td" include "mlir/IR/OpAsmInterface.td" include "mlir/IR/RegionKindInterface.td" @@ -16,6 +17,7 @@ include "mlir/Interfaces/CallInterfaces.td" include "mlir/Interfaces/ControlFlowInterfaces.td" include "mlir/Interfaces/CopyOpInterface.td" +include "mlir/Interfaces/DataLayoutInterfaces.td" include "mlir/Interfaces/InferTypeOpInterface.td" include "mlir/Interfaces/SideEffectInterfaces.td" include "TestInterfaces.td" @@ -27,6 +29,7 @@ let hasOperationAttrVerify = 1; let hasRegionArgAttrVerify = 1; let hasRegionResultAttrVerify = 1; + let dependentDialects = ["::mlir::TargetDialect"]; } class TEST_Op traits = []> : @@ -2006,4 +2009,26 @@ let results = (outs TupleOf<[AnyType]>); } +//===----------------------------------------------------------------------===// +// Test Target DataLayout +//===----------------------------------------------------------------------===// + +def OpWithDataLayoutOp : TEST_Op<"op_with_data_layout", + [HasTargetDataLayout, DataLayoutOpInterface]> { + let summary = + "An op that uses DataLayout implementation from the Target dialect"; + let regions = (region VariadicRegion:$regions); +} + +def DataLayoutQueryOp : TEST_Op<"data_layout_query"> { + let summary = "A token op recognized by data layout query test pass"; + let description = [{ + The data layout query pass pattern-matches this op and attaches to it an + array attribute containing the result of data layout query of the result + type of this op. + }]; + + let results = (outs AnyType:$res); +} + #endif // TEST_OPS diff --git a/mlir/test/lib/Dialect/Test/TestTypes.h b/mlir/test/lib/Dialect/Test/TestTypes.h --- a/mlir/test/lib/Dialect/Test/TestTypes.h +++ b/mlir/test/lib/Dialect/Test/TestTypes.h @@ -21,6 +21,7 @@ #include "mlir/IR/DialectImplementation.h" #include "mlir/IR/Operation.h" #include "mlir/IR/Types.h" +#include "mlir/Interfaces/DataLayoutInterfaces.h" namespace mlir { namespace test { @@ -107,6 +108,62 @@ StringRef getName() { return getImpl()->name; } }; +struct TestLayedOutTypeStorage : public TypeStorage { + using KeyTy = unsigned; + + explicit TestLayedOutTypeStorage(unsigned key) : key(key) {} + bool operator==(const KeyTy &other) const { return other == key; } + + static TestLayedOutTypeStorage *construct(TypeStorageAllocator &allocator, + const KeyTy &key) { + return new (allocator.allocate()) + TestLayedOutTypeStorage(key); + } + + unsigned key; +}; + +class TestLayedOutType + : public Type::TypeBase { +public: + using Base::Base; + + static TestLayedOutType get(MLIRContext *ctx, unsigned key) { + return Base::get(ctx, key); + } + + unsigned getKey() { return getImpl()->key; } + + unsigned getTypeSize(const DataLayout &dataLayout, + DataLayoutEntryListRef params) const { + return extractKind(params, "size"); + } + + unsigned getABIAlignment(const DataLayout &dataLayout, + DataLayoutEntryListRef params) const { + return extractKind(params, "alignment"); + } + + unsigned getPreferredAlignment(const DataLayout &dataLayout, + DataLayoutEntryListRef params) const { + return extractKind(params, "preferred"); + } + + bool areCompatible(DataLayoutEntryListRef oldLayout, + DataLayoutEntryListRef newLayout) const { + unsigned old = extractKind(oldLayout, "alignment"); + return old == 1 || extractKind(newLayout, "alignment") <= old; + } + + LogicalResult verifyEntries(DataLayoutEntryListRef params, + Location loc) const; + +private: + unsigned extractKind(DataLayoutEntryListRef params, + StringRef expectedKind) const; +}; + } // namespace test } // namespace mlir diff --git a/mlir/test/lib/Dialect/Test/TestTypes.cpp b/mlir/test/lib/Dialect/Test/TestTypes.cpp --- a/mlir/test/lib/Dialect/Test/TestTypes.cpp +++ b/mlir/test/lib/Dialect/Test/TestTypes.cpp @@ -126,6 +126,40 @@ #define GET_TYPEDEF_CLASSES #include "TestTypeDefs.cpp.inc" +LogicalResult TestLayedOutType::verifyEntries(DataLayoutEntryListRef params, + Location loc) const { + for (DataLayoutEntryInterface entry : params) { + if (!entry.isTypeEntry()) + emitError(loc) << "unexpected identifier entry"; + + // This is for testing purposes only, so assert well-formedness. + assert(entry.getKey().get().isa() && + "wrong type passed in"); + auto array = entry.getValue().dyn_cast(); + assert(array && array.getValue().size() == 2 && + "expected array of two elements"); + auto kind = array.getValue().front().dyn_cast(); + (void)kind; + assert(kind && + (kind.getValue() == "size" || kind.getValue() == "alignment" || + kind.getValue() == "preferred") && + "unexpected kind"); + assert(array.getValue().back().isa()); + } + return success(); +} + +unsigned TestLayedOutType::extractKind(DataLayoutEntryListRef params, + StringRef expectedKind) const { + for (DataLayoutEntryInterface entry : params) { + ArrayRef pair = entry.getValue().cast().getValue(); + StringRef kind = pair.front().cast().getValue(); + if (kind == expectedKind) + return pair.back().cast().getValue().getZExtValue(); + } + return 1; +} + //===----------------------------------------------------------------------===// // TestDialect //===----------------------------------------------------------------------===// @@ -143,8 +177,20 @@ if (typeTag == "test_type") return TestType::get(parser.getBuilder().getContext()); - if (typeTag != "test_rec") + if (typeTag == "test_layed_out") { + unsigned val; + if (parser.parseLess() || parser.parseInteger(val) || + parser.parseGreater()) { + parser.emitError(parser.getNameLoc()) << "wrong syntax"; + return Type(); + } + return TestLayedOutType::get(parser.getBuilder().getContext(), val); + } + + if (typeTag != "test_rec") { + parser.emitError(parser.getNameLoc()) << "unknown type!"; return Type(); + } StringRef name; if (parser.parseLess() || parser.parseKeyword(&name)) @@ -185,6 +231,11 @@ return; } + if (auto t = type.dyn_cast()) { + printer << "test_layed_out<" << t.getKey() << ">"; + return; + } + auto rec = type.cast(); printer << "test_rec<" << rec.getName(); if (!stack.contains(rec)) { diff --git a/mlir/test/lib/Transforms/CMakeLists.txt b/mlir/test/lib/Transforms/CMakeLists.txt --- a/mlir/test/lib/Transforms/CMakeLists.txt +++ b/mlir/test/lib/Transforms/CMakeLists.txt @@ -9,6 +9,7 @@ TestConvertCallOp.cpp TestConvertGPUKernelToCubin.cpp TestConvertGPUKernelToHsaco.cpp + TestDataLayoutQuery.cpp TestDominance.cpp TestDynamicPipeline.cpp TestLoopFusion.cpp diff --git a/mlir/test/lib/Transforms/TestDataLayoutQuery.cpp b/mlir/test/lib/Transforms/TestDataLayoutQuery.cpp new file mode 100644 --- /dev/null +++ b/mlir/test/lib/Transforms/TestDataLayoutQuery.cpp @@ -0,0 +1,60 @@ +//===- TestDataLayoutQuery.cpp - Test Data Layout Queries -----------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "TestDialect.h" +#include "mlir/Dialect/Target/Target.h" +#include "mlir/IR/BuiltinAttributes.h" +#include "mlir/Pass/Pass.h" + +using namespace mlir; + +namespace { + +// A pass that finds "test.data_layout_query" operations and attaches to them +// attributes containing the results of data layout queries for operation +// result types. +struct TestDataLayoutQuery + : public PassWrapper { + void runOnFunction() override { + FuncOp func = getFunction(); + Builder builder(func.getContext()); + DenseMap layouts; + + func.walk([&](test::DataLayoutQueryOp op) { + // Skip the ops with already processed in a deeper call. + if (op->getAttr("size")) + return; + + auto scope = op->getParentOfType(); + if (!layouts.count(scope)) { + layouts.try_emplace( + scope, scope ? cast(scope.getOperation()) + : nullptr); + } + + const DataLayout &layout = layouts.find(scope)->getSecond(); + unsigned size = layout.getTypeSize(op.getType()); + unsigned alignment = layout.getTypeABIAlignment(op.getType()); + unsigned preferred = layout.getTypePreferredAlignment(op.getType()); + op->setAttrs( + {builder.getNamedAttr("size", builder.getIndexAttr(size)), + builder.getNamedAttr("alignment", builder.getIndexAttr(alignment)), + builder.getNamedAttr("preferred", builder.getIndexAttr(preferred))}); + }); + } +}; +} // namespace + +namespace mlir { +namespace test { +void registerTestDataLayoutQuery() { + PassRegistration("test-data-layout-query", + "Test data layout queries"); +} +} // namespace test +} // namespace mlir diff --git a/mlir/test/mlir-opt/commandline.mlir b/mlir/test/mlir-opt/commandline.mlir --- a/mlir/test/mlir-opt/commandline.mlir +++ b/mlir/test/mlir-opt/commandline.mlir @@ -25,6 +25,7 @@ // CHECK-NEXT: shape // CHECK-NEXT: spv // CHECK-NEXT: std +// CHECK-NEXT: target // CHECK-NEXT: tensor // CHECK-NEXT: test // CHECK-NEXT: tosa diff --git a/mlir/tools/mlir-opt/mlir-opt.cpp b/mlir/tools/mlir-opt/mlir-opt.cpp --- a/mlir/tools/mlir-opt/mlir-opt.cpp +++ b/mlir/tools/mlir-opt/mlir-opt.cpp @@ -66,6 +66,7 @@ void registerTestConvVectorization(); void registerTestConvertGPUKernelToCubinPass(); void registerTestConvertGPUKernelToHsacoPass(); +void registerTestDataLayoutQuery(); void registerTestDecomposeCallGraphTypes(); void registerTestDialect(DialectRegistry &); void registerTestDominancePass(); @@ -141,6 +142,7 @@ #endif test::registerTestConvVectorization(); test::registerTestDecomposeCallGraphTypes(); + test::registerTestDataLayoutQuery(); test::registerTestDominancePass(); test::registerTestDynamicPipelinePass(); test::registerTestExpandTanhPass(); diff --git a/mlir/unittests/CMakeLists.txt b/mlir/unittests/CMakeLists.txt --- a/mlir/unittests/CMakeLists.txt +++ b/mlir/unittests/CMakeLists.txt @@ -7,6 +7,7 @@ add_subdirectory(Analysis) add_subdirectory(Dialect) add_subdirectory(ExecutionEngine) +add_subdirectory(Interfaces) add_subdirectory(IR) add_subdirectory(Pass) add_subdirectory(SDBM) diff --git a/mlir/unittests/Interfaces/CMakeLists.txt b/mlir/unittests/Interfaces/CMakeLists.txt new file mode 100644 --- /dev/null +++ b/mlir/unittests/Interfaces/CMakeLists.txt @@ -0,0 +1,10 @@ +add_mlir_unittest(MLIRInterfacesTests + DataLayoutInterfacesTest.cpp +) + +target_link_libraries(MLIRInterfacesTests + PRIVATE + MLIRDataLayoutInterfaces + MLIRTarget + MLIRParser +) diff --git a/mlir/unittests/Interfaces/DataLayoutInterfacesTest.cpp b/mlir/unittests/Interfaces/DataLayoutInterfacesTest.cpp new file mode 100644 --- /dev/null +++ b/mlir/unittests/Interfaces/DataLayoutInterfacesTest.cpp @@ -0,0 +1,358 @@ +//===- DataLayoutInterfacesTest.cpp - Unit Tests for Data Layouts ---------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#include "mlir/Interfaces/DataLayoutInterfaces.h" +#include "mlir/Dialect/Target/Target.h" +#include "mlir/IR/Builders.h" +#include "mlir/IR/BuiltinOps.h" +#include "mlir/IR/Dialect.h" +#include "mlir/IR/DialectImplementation.h" +#include "mlir/IR/OpDefinition.h" +#include "mlir/IR/OpImplementation.h" +#include "mlir/Parser.h" + +#include + +using namespace mlir; + +namespace { +constexpr static const char *kDialectPrefix = "dltest"; +constexpr static const char *kAttrName = "dltest.layout"; + +/// Trivial array storage for the custom data layout spec attribute, just a list +/// of entries. +class DataLayoutSpecStorage : public AttributeStorage { +public: + using KeyTy = ArrayRef; + + DataLayoutSpecStorage(ArrayRef entries) + : entries(entries) {} + + bool operator==(const KeyTy &key) const { return key == entries; } + + static DataLayoutSpecStorage *construct(AttributeStorageAllocator &allocator, + const KeyTy &key) { + return new (allocator.allocate()) + DataLayoutSpecStorage(allocator.copyInto(key)); + } + + ArrayRef entries; +}; + +/// Simple data layout spec containing a list of entries that always verifies +/// as valid. +struct CustomDataLayoutSpec + : public Attribute::AttrBase { + using Base::Base; + static CustomDataLayoutSpec get(MLIRContext *ctx, + ArrayRef entries) { + return Base::get(ctx, entries); + } + CustomDataLayoutSpec + combineWith(ArrayRef specs) const { + return *this; + } + DataLayoutEntryListRef getEntries() const { return getImpl()->entries; } + LogicalResult verify(Location loc) { return success(); } +}; + +/// A type subject to data layout that exits the program if it is queried more +/// than once. Handy to check if the cache works. +struct SingleQueryType + : public Type::TypeBase { + using Base::Base; + + static SingleQueryType get(MLIRContext *ctx) { return Base::get(ctx); } + + unsigned getTypeSize(const DataLayout &layout, + DataLayoutEntryListRef params) { + static bool executed = false; + if (executed) + llvm::report_fatal_error("repeated call"); + + executed = true; + return 1; + } + + unsigned getABIAlignment(const DataLayout &layout, + DataLayoutEntryListRef params) { + static bool executed = false; + if (executed) + llvm::report_fatal_error("repeated call"); + + executed = true; + return 2; + } + + unsigned getPreferredAlignment(const DataLayout &layout, + DataLayoutEntryListRef params) { + static bool executed = false; + if (executed) + llvm::report_fatal_error("repeated call"); + + executed = true; + return 4; + } +}; + +/// A types that is not subject to data layout. +struct TypeNoLayout : public Type::TypeBase { + using Base::Base; + + static TypeNoLayout get(MLIRContext *ctx) { return Base::get(ctx); } +}; + +/// An op that serves as scope for data layout queries with the relevant +/// attribute attached. This can handle data layout requests for the built-in +/// types itself. +struct OpWithLayout : public Op { + using Op::Op; + + static StringRef getOperationName() { return "dltest.op_with_layout"; } + + DataLayoutSpecInterface getDataLayoutSpec() { + return getOperation()->getAttrOfType(kAttrName); + } + + static unsigned getTypeSize(Type type, const DataLayout &dataLayout, + DataLayoutEntryListRef params) { + // Make a recursive query. + if (type.isa()) + return dataLayout.getTypeSize( + IntegerType::get(type.getContext(), type.getIntOrFloatBitWidth())); + + // Handle built-in types that are not handled by the default process. + if (auto iType = type.dyn_cast()) { + for (DataLayoutEntryInterface entry : params) + if (entry.getKey().dyn_cast() == type) + return entry.getValue().cast().getValue().getZExtValue(); + return iType.getIntOrFloatBitWidth(); + } + + // Use the default process for everything else. + return detail::getDefaultTypeSize(type, dataLayout, params); + } + + static unsigned getTypeABIAlignment(Type type, const DataLayout &dataLayout, + DataLayoutEntryListRef params) { + return llvm::PowerOf2Ceil(getTypeSize(type, dataLayout, params)); + } + + static unsigned getTypePreferredAlignment(Type type, + const DataLayout &dataLayout, + DataLayoutEntryListRef params) { + return 2 * getTypeABIAlignment(type, dataLayout, params); + } +}; + +/// A dialect putting all the above together. +struct DLTestDialect : Dialect { + explicit DLTestDialect(MLIRContext *ctx) + : Dialect(getDialectNamespace(), ctx, TypeID::get()) { + ctx->getOrLoadDialect(); + addAttributes(); + addOperations(); + addTypes(); + } + static StringRef getDialectNamespace() { return "dltest"; } + + void printAttribute(Attribute attr, + DialectAsmPrinter &printer) const override { + printer << "spec<"; + llvm::interleaveComma(attr.cast().getEntries(), + printer); + printer << ">"; + } + + Attribute parseAttribute(DialectAsmParser &parser, Type type) const override { + bool ok = + succeeded(parser.parseKeyword("spec")) && succeeded(parser.parseLess()); + (void)ok; + assert(ok); + if (succeeded(parser.parseOptionalGreater())) + return CustomDataLayoutSpec::get(parser.getBuilder().getContext(), {}); + + SmallVector entries; + do { + entries.emplace_back(); + ok = succeeded(parser.parseAttribute(entries.back())); + assert(ok); + } while (succeeded(parser.parseOptionalComma())); + ok = succeeded(parser.parseGreater()); + assert(ok); + return CustomDataLayoutSpec::get(parser.getBuilder().getContext(), entries); + } + + void printType(Type type, DialectAsmPrinter &printer) const override { + if (type.isa()) + printer << "single_query"; + else + printer << "no_layout"; + } + + Type parseType(DialectAsmParser &parser) const override { + bool ok = succeeded(parser.parseKeyword("single_query")); + (void)ok; + assert(ok); + return SingleQueryType::get(parser.getBuilder().getContext()); + } +}; + +} // end namespace + +TEST(DataLayout, FallbackDefault) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + EXPECT_EQ(layout.getTypeSize(IntegerType::get(&ctx, 42)), 6); + EXPECT_EQ(layout.getTypeSize(Float16Type::get(&ctx)), 2); + EXPECT_EQ(layout.getTypeABIAlignment(IntegerType::get(&ctx, 42)), 8); + EXPECT_EQ(layout.getTypeABIAlignment(Float16Type::get(&ctx)), 2); + EXPECT_EQ(layout.getTypePreferredAlignment(IntegerType::get(&ctx, 42)), 8); + EXPECT_EQ(layout.getTypePreferredAlignment(Float16Type::get(&ctx)), 4); +} + +TEST(DataLayout, EmptySpec) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() { dltest.layout = #dltest.spec< > } : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + EXPECT_EQ(layout.getTypeSize(IntegerType::get(&ctx, 42)), 42); + EXPECT_EQ(layout.getTypeSize(Float16Type::get(&ctx)), 16); + EXPECT_EQ(layout.getTypeABIAlignment(IntegerType::get(&ctx, 42)), 64); + EXPECT_EQ(layout.getTypeABIAlignment(Float16Type::get(&ctx)), 16); + EXPECT_EQ(layout.getTypePreferredAlignment(IntegerType::get(&ctx, 42)), 128); + EXPECT_EQ(layout.getTypePreferredAlignment(Float16Type::get(&ctx)), 32); +} + +TEST(DataLayout, SpecWithEntries) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() { dltest.layout = #dltest.spec< + #target.dl_entry, + #target.dl_entry +> } : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + EXPECT_EQ(layout.getTypeSize(IntegerType::get(&ctx, 42)), 5); + EXPECT_EQ(layout.getTypeSize(Float16Type::get(&ctx)), 6); + EXPECT_EQ(layout.getTypeABIAlignment(IntegerType::get(&ctx, 42)), 8); + EXPECT_EQ(layout.getTypeABIAlignment(Float16Type::get(&ctx)), 8); + EXPECT_EQ(layout.getTypePreferredAlignment(IntegerType::get(&ctx, 42)), 16); + EXPECT_EQ(layout.getTypePreferredAlignment(Float16Type::get(&ctx)), 16); + + EXPECT_EQ(layout.getTypeSize(IntegerType::get(&ctx, 32)), 32); + EXPECT_EQ(layout.getTypeSize(Float32Type::get(&ctx)), 32); + EXPECT_EQ(layout.getTypeABIAlignment(IntegerType::get(&ctx, 32)), 32); + EXPECT_EQ(layout.getTypeABIAlignment(Float32Type::get(&ctx)), 32); + EXPECT_EQ(layout.getTypePreferredAlignment(IntegerType::get(&ctx, 32)), 64); + EXPECT_EQ(layout.getTypePreferredAlignment(Float32Type::get(&ctx)), 64); +} + +TEST(DataLayout, Caching) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() { dltest.layout = #dltest.spec<> } : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + + unsigned sum = 0; + sum += layout.getTypeSize(SingleQueryType::get(&ctx)); + // The second call should hit the cache. If it does not, the function in + // SingleQueryType will be called and will abort the process. + sum += layout.getTypeSize(SingleQueryType::get(&ctx)); + // Make sure the complier doesn't optimize away the query code. + EXPECT_EQ(sum, 2); + + // A fresh data layout has a new cache, so the call to it should be dispatched + // down to the type and abort the proces. + DataLayout second(op); + ASSERT_DEATH(second.getTypeSize(SingleQueryType::get(&ctx)), "repeated call"); +} + +TEST(DataLayout, CacheInvalidation) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() { dltest.layout = #dltest.spec< + #target.dl_entry, + #target.dl_entry +> } : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + + // Normal query is fine. + EXPECT_EQ(layout.getTypeSize(Float16Type::get(&ctx)), 6); + + // Replace the data layout spec with a new, empty spec. + op->setAttr(kAttrName, CustomDataLayoutSpec::get(&ctx, {})); + + // Data layout is no longer valid and should trigger assertion when queried. +#ifndef NDEBUG + ASSERT_DEATH(layout.getTypeSize(Float16Type::get(&ctx)), "no longer valid"); +#endif +} + +TEST(DataLayout, UnimplementedTypeInterface) { + const char *ir = R"MLIR( +"dltest.op_with_layout"() { dltest.layout = #dltest.spec<> } : () -> () + )MLIR"; + + DialectRegistry registry; + registry.insert(); + MLIRContext ctx(registry); + + OwningModuleRef module = parseSourceString(ir, &ctx); + auto op = + cast(module->getBody()->getOperations().front()); + DataLayout layout(op); + + ASSERT_DEATH(layout.getTypeSize(TypeNoLayout::get(&ctx)), + "neither the scoping op nor the type class provide data layout " + "information"); +}