[mlir][spirv] Use separate attribute for (version, capabilities, extensions)

We also need the (version, capabilities, extensions) triple on the
spv.module op. Thus far we have been using separate 'extensions'
and 'capabilities' attributes there and 'version' is missing. Creating
a separate attribute for the trip allows us to reuse the assembly
form and verification.

Differential Revision: https://reviews.llvm.org/D75868
This commit is contained in:
Lei Zhang 2020-03-11 16:02:46 -04:00
parent 69ce2fd2df
commit e115a40f50
9 changed files with 368 additions and 223 deletions

View File

@ -742,11 +742,11 @@ instructions.
SPIR-V compilation should also take into consideration of the execution SPIR-V compilation should also take into consideration of the execution
environment, so we generate SPIR-V modules valid for the target environment. environment, so we generate SPIR-V modules valid for the target environment.
This is conveyed by the `spv.target_env` attribute. It should be of This is conveyed by the `spv.target_env` (`spirv::TargetEnvAttr`) attribute. It
`#spv.target_env` attribute kind, which is defined as: should be of `#spv.target_env` attribute kind, which is defined as:
``` ```
spirv-version ::= `V_1_0` | `V_1_1` | ... spirv-version ::= `v1.0` | `v1.1` | ...
spirv-extension ::= `SPV_KHR_16bit_storage` | `SPV_EXT_physical_storage_buffer` | ... spirv-extension ::= `SPV_KHR_16bit_storage` | `SPV_EXT_physical_storage_buffer` | ...
spirv-capability ::= `Shader` | `Kernel` | `GroupNonUniform` | ... spirv-capability ::= `Shader` | `Kernel` | `GroupNonUniform` | ...
@ -758,18 +758,22 @@ spirv-capability-elements ::= spirv-capability (`,` spirv-capability)*
spirv-resource-limits ::= dictionary-attribute spirv-resource-limits ::= dictionary-attribute
spirv-vce-attribute ::= `#` `spv.vce` `<`
spirv-version `,`
spirv-capability-list `,`
spirv-extensions-list `>`
spirv-target-env-attribute ::= `#` `spv.target_env` `<` spirv-target-env-attribute ::= `#` `spv.target_env` `<`
spirv-version `,` spirv-vce-attribute,
spirv-extensions-list `,`
spirv-capability-list `,`
spirv-resource-limits `>` spirv-resource-limits `>`
``` ```
The attribute has a few fields: The attribute has a few fields:
* The target SPIR-V version. * A `#spv.vce` (`spirv::VerCapExtAttr`) attribute:
* A list of SPIR-V extensions for the target. * The target SPIR-V version.
* A list of SPIR-V capabilities for the target. * A list of SPIR-V extensions for the target.
* A list of SPIR-V capabilities for the target.
* A dictionary of target resource limits (see the * A dictionary of target resource limits (see the
[Vulkan spec][VulkanResourceLimits] for explanation): [Vulkan spec][VulkanResourceLimits] for explanation):
* `max_compute_workgroup_invocations` * `max_compute_workgroup_invocations`
@ -780,7 +784,7 @@ For example,
``` ```
module attributes { module attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_3, [SPV_KHR_8bit_storage], [Shader, GroupNonUniform] #spv.vce<v1.3, [Shader, GroupNonUniform], [SPV_KHR_8bit_storage]>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>

View File

@ -96,12 +96,12 @@ class SPV_StrEnumAttr<string name, string description,
// SPIR-V availability definitions // SPIR-V availability definitions
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
def SPV_V_1_0 : I32EnumAttrCase<"V_1_0", 0>; def SPV_V_1_0 : I32EnumAttrCase<"V_1_0", 0, "v1.0">;
def SPV_V_1_1 : I32EnumAttrCase<"V_1_1", 1>; def SPV_V_1_1 : I32EnumAttrCase<"V_1_1", 1, "v1.1">;
def SPV_V_1_2 : I32EnumAttrCase<"V_1_2", 2>; def SPV_V_1_2 : I32EnumAttrCase<"V_1_2", 2, "v1.2">;
def SPV_V_1_3 : I32EnumAttrCase<"V_1_3", 3>; def SPV_V_1_3 : I32EnumAttrCase<"V_1_3", 3, "v1.3">;
def SPV_V_1_4 : I32EnumAttrCase<"V_1_4", 4>; def SPV_V_1_4 : I32EnumAttrCase<"V_1_4", 4, "v1.4">;
def SPV_V_1_5 : I32EnumAttrCase<"V_1_5", 5>; def SPV_V_1_5 : I32EnumAttrCase<"V_1_5", 5, "v1.5">;
def SPV_VersionAttr : SPV_I32EnumAttr<"Version", "valid SPIR-V version", [ def SPV_VersionAttr : SPV_I32EnumAttr<"Version", "valid SPIR-V version", [
SPV_V_1_0, SPV_V_1_1, SPV_V_1_2, SPV_V_1_3, SPV_V_1_4, SPV_V_1_5]>; SPV_V_1_0, SPV_V_1_1, SPV_V_1_2, SPV_V_1_3, SPV_V_1_4, SPV_V_1_5]>;

View File

@ -32,16 +32,71 @@ enum class Version : uint32_t;
namespace detail { namespace detail {
struct TargetEnvAttributeStorage; struct TargetEnvAttributeStorage;
struct VerCapExtAttributeStorage;
} // namespace detail } // namespace detail
/// SPIR-V dialect-specific attribute kinds. /// SPIR-V dialect-specific attribute kinds.
// TODO(antiagainst): move to a more suitable place if we have more attributes. // TODO(antiagainst): move to a more suitable place if we have more attributes.
namespace AttrKind { namespace AttrKind {
enum Kind { enum Kind {
TargetEnv = Attribute::FIRST_SPIRV_ATTR, TargetEnv = Attribute::FIRST_SPIRV_ATTR, /// Target environment
VerCapExt, /// (version, extension, capability) triple
}; };
} // namespace AttrKind } // namespace AttrKind
/// An attribute that specifies the SPIR-V (version, capabilities, extensions)
/// triple.
class VerCapExtAttr
: public Attribute::AttrBase<VerCapExtAttr, Attribute,
detail::VerCapExtAttributeStorage> {
public:
using Base::Base;
/// Gets a VerCapExtAttr instance.
static VerCapExtAttr get(Version version, ArrayRef<Capability> capabilities,
ArrayRef<Extension> extensions,
MLIRContext *context);
static VerCapExtAttr get(IntegerAttr version, ArrayAttr capabilities,
ArrayAttr extensions);
/// Returns the attribute kind's name (without the 'spv.' prefix).
static StringRef getKindName();
/// Returns the version.
Version getVersion();
struct ext_iterator final
: public llvm::mapped_iterator<ArrayAttr::iterator,
Extension (*)(Attribute)> {
explicit ext_iterator(ArrayAttr::iterator it);
};
using ext_range = llvm::iterator_range<ext_iterator>;
/// Returns the extensions.
ext_range getExtensions();
/// Returns the extensions as a string array attribute.
ArrayAttr getExtensionsAttr();
struct cap_iterator final
: public llvm::mapped_iterator<ArrayAttr::iterator,
Capability (*)(Attribute)> {
explicit cap_iterator(ArrayAttr::iterator it);
};
using cap_range = llvm::iterator_range<cap_iterator>;
/// Returns the capabilities.
cap_range getCapabilities();
/// Returns the capabilities as an integer array attribute.
ArrayAttr getCapabilitiesAttr();
static bool kindof(unsigned kind) { return kind == AttrKind::VerCapExt; }
static LogicalResult verifyConstructionInvariants(Location loc,
IntegerAttr version,
ArrayAttr capabilities,
ArrayAttr extensions);
};
/// An attribute that specifies the target version, allowed extensions and /// An attribute that specifies the target version, allowed extensions and
/// capabilities, and resource limits. These information describles a SPIR-V /// capabilities, and resource limits. These information describles a SPIR-V
/// target environment. /// target environment.
@ -52,39 +107,24 @@ public:
using Base::Base; using Base::Base;
/// Gets a TargetEnvAttr instance. /// Gets a TargetEnvAttr instance.
static TargetEnvAttr get(Version version, ArrayRef<Extension> extensions, static TargetEnvAttr get(VerCapExtAttr triple, DictionaryAttr limits);
ArrayRef<Capability> capabilities,
DictionaryAttr limits);
static TargetEnvAttr get(IntegerAttr version, ArrayAttr extensions,
ArrayAttr capabilities, DictionaryAttr limits);
/// Returns the attribute kind's name (without the 'spv.' prefix). /// Returns the attribute kind's name (without the 'spv.' prefix).
static StringRef getKindName(); static StringRef getKindName();
/// Returns the (version, capabilities, extensions) triple attribute.
VerCapExtAttr getTripleAttr();
/// Returns the target version. /// Returns the target version.
Version getVersion(); Version getVersion();
struct ext_iterator final
: public llvm::mapped_iterator<ArrayAttr::iterator,
Extension (*)(Attribute)> {
explicit ext_iterator(ArrayAttr::iterator it);
};
using ext_range = llvm::iterator_range<ext_iterator>;
/// Returns the target extensions. /// Returns the target extensions.
ext_range getExtensions(); VerCapExtAttr::ext_range getExtensions();
/// Returns the target extensions as a string array attribute. /// Returns the target extensions as a string array attribute.
ArrayAttr getExtensionsAttr(); ArrayAttr getExtensionsAttr();
struct cap_iterator final
: public llvm::mapped_iterator<ArrayAttr::iterator,
Capability (*)(Attribute)> {
explicit cap_iterator(ArrayAttr::iterator it);
};
using cap_range = llvm::iterator_range<cap_iterator>;
/// Returns the target capabilities. /// Returns the target capabilities.
cap_range getCapabilities(); VerCapExtAttr::cap_range getCapabilities();
/// Returns the target capabilities as an integer array attribute. /// Returns the target capabilities as an integer array attribute.
ArrayAttr getCapabilitiesAttr(); ArrayAttr getCapabilitiesAttr();
@ -94,9 +134,7 @@ public:
static bool kindof(unsigned kind) { return kind == AttrKind::TargetEnv; } static bool kindof(unsigned kind) { return kind == AttrKind::TargetEnv; }
static LogicalResult verifyConstructionInvariants(Location loc, static LogicalResult verifyConstructionInvariants(Location loc,
IntegerAttr version, VerCapExtAttr triple,
ArrayAttr extensions,
ArrayAttr capabilities,
DictionaryAttr limits); DictionaryAttr limits);
}; };

View File

@ -118,7 +118,7 @@ SPIRVDialect::SPIRVDialect(MLIRContext *context)
: Dialect(getDialectNamespace(), context) { : Dialect(getDialectNamespace(), context) {
addTypes<ArrayType, ImageType, PointerType, RuntimeArrayType, StructType>(); addTypes<ArrayType, ImageType, PointerType, RuntimeArrayType, StructType>();
addAttributes<TargetEnvAttr>(); addAttributes<TargetEnvAttr, VerCapExtAttr>();
// Add SPIR-V ops. // Add SPIR-V ops.
addOperations< addOperations<
@ -662,8 +662,7 @@ static ParseResult parseKeywordList(
return success(); return success();
} }
/// Parses a spirv::TargetEnvAttr. static Attribute parseVerCapExtAttr(DialectAsmParser &parser) {
static Attribute parseTargetAttr(DialectAsmParser &parser) {
if (parser.parseLess()) if (parser.parseLess())
return {}; return {};
@ -685,28 +684,6 @@ static Attribute parseTargetAttr(DialectAsmParser &parser) {
} }
} }
ArrayAttr extensionsAttr;
{
SmallVector<Attribute, 1> extensions;
llvm::SMLoc errorloc;
StringRef errorKeyword;
auto processExtension = [&](llvm::SMLoc loc, StringRef extension) {
if (spirv::symbolizeExtension(extension)) {
extensions.push_back(builder.getStringAttr(extension));
return success();
}
return errorloc = loc, errorKeyword = extension, failure();
};
if (parseKeywordList(parser, processExtension) || parser.parseComma()) {
if (!errorKeyword.empty())
parser.emitError(errorloc, "unknown extension: ") << errorKeyword;
return {};
}
extensionsAttr = builder.getArrayAttr(extensions);
}
ArrayAttr capabilitiesAttr; ArrayAttr capabilitiesAttr;
{ {
SmallVector<Attribute, 4> capabilities; SmallVector<Attribute, 4> capabilities;
@ -730,6 +707,44 @@ static Attribute parseTargetAttr(DialectAsmParser &parser) {
capabilitiesAttr = builder.getArrayAttr(capabilities); capabilitiesAttr = builder.getArrayAttr(capabilities);
} }
ArrayAttr extensionsAttr;
{
SmallVector<Attribute, 1> extensions;
llvm::SMLoc errorloc;
StringRef errorKeyword;
auto processExtension = [&](llvm::SMLoc loc, StringRef extension) {
if (spirv::symbolizeExtension(extension)) {
extensions.push_back(builder.getStringAttr(extension));
return success();
}
return errorloc = loc, errorKeyword = extension, failure();
};
if (parseKeywordList(parser, processExtension)) {
if (!errorKeyword.empty())
parser.emitError(errorloc, "unknown extension: ") << errorKeyword;
return {};
}
extensionsAttr = builder.getArrayAttr(extensions);
}
if (parser.parseGreater())
return {};
return spirv::VerCapExtAttr::get(versionAttr, capabilitiesAttr,
extensionsAttr);
}
/// Parses a spirv::TargetEnvAttr.
static Attribute parseTargetEnvAttr(DialectAsmParser &parser) {
if (parser.parseLess())
return {};
spirv::VerCapExtAttr tripleAttr;
if (parser.parseAttribute(tripleAttr) || parser.parseComma())
return {};
DictionaryAttr limitsAttr; DictionaryAttr limitsAttr;
{ {
auto loc = parser.getCurrentLocation(); auto loc = parser.getCurrentLocation();
@ -749,8 +764,7 @@ static Attribute parseTargetAttr(DialectAsmParser &parser) {
if (parser.parseGreater()) if (parser.parseGreater())
return {}; return {};
return spirv::TargetEnvAttr::get(versionAttr, extensionsAttr, return spirv::TargetEnvAttr::get(tripleAttr, limitsAttr);
capabilitiesAttr, limitsAttr);
} }
Attribute SPIRVDialect::parseAttribute(DialectAsmParser &parser, Attribute SPIRVDialect::parseAttribute(DialectAsmParser &parser,
@ -767,7 +781,9 @@ Attribute SPIRVDialect::parseAttribute(DialectAsmParser &parser,
return {}; return {};
if (attrKind == spirv::TargetEnvAttr::getKindName()) if (attrKind == spirv::TargetEnvAttr::getKindName())
return parseTargetAttr(parser); return parseTargetEnvAttr(parser);
if (attrKind == spirv::VerCapExtAttr::getKindName())
return parseVerCapExtAttr(parser);
parser.emitError(parser.getNameLoc(), "unknown SPIR-V attriubte kind: ") parser.emitError(parser.getNameLoc(), "unknown SPIR-V attriubte kind: ")
<< attrKind; << attrKind;
@ -778,24 +794,32 @@ Attribute SPIRVDialect::parseAttribute(DialectAsmParser &parser,
// Attribute Printing // Attribute Printing
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
static void print(spirv::TargetEnvAttr targetEnv, DialectAsmPrinter &printer) { static void print(spirv::VerCapExtAttr triple, DialectAsmPrinter &printer) {
auto &os = printer.getStream(); auto &os = printer.getStream();
printer << spirv::TargetEnvAttr::getKindName() << "<" printer << spirv::VerCapExtAttr::getKindName() << "<"
<< spirv::stringifyVersion(targetEnv.getVersion()) << ", ["; << spirv::stringifyVersion(triple.getVersion()) << ", [";
interleaveComma(targetEnv.getExtensionsAttr(), os, [&](Attribute attr) { interleaveComma(triple.getCapabilities(), os, [&](spirv::Capability cap) {
os << attr.cast<StringAttr>().getValue();
});
printer << "], [";
interleaveComma(targetEnv.getCapabilities(), os, [&](spirv::Capability cap) {
os << spirv::stringifyCapability(cap); os << spirv::stringifyCapability(cap);
}); });
printer << "], " << targetEnv.getResourceLimits() << ">"; printer << "], [";
interleaveComma(triple.getExtensionsAttr(), os, [&](Attribute attr) {
os << attr.cast<StringAttr>().getValue();
});
printer << "]>";
}
static void print(spirv::TargetEnvAttr targetEnv, DialectAsmPrinter &printer) {
printer << spirv::TargetEnvAttr::getKindName() << "<#spv.";
print(targetEnv.getTripleAttr(), printer);
printer << ", " << targetEnv.getResourceLimits() << ">";
} }
void SPIRVDialect::printAttribute(Attribute attr, void SPIRVDialect::printAttribute(Attribute attr,
DialectAsmPrinter &printer) const { DialectAsmPrinter &printer) const {
if (auto targetEnv = attr.dyn_cast<TargetEnvAttr>()) if (auto targetEnv = attr.dyn_cast<TargetEnvAttr>())
print(targetEnv, printer); print(targetEnv, printer);
else if (auto vceAttr = attr.dyn_cast<VerCapExtAttr>())
print(vceAttr, printer);
else else
llvm_unreachable("unhandled SPIR-V attribute kind"); llvm_unreachable("unhandled SPIR-V attribute kind");
} }
@ -807,7 +831,7 @@ void SPIRVDialect::printAttribute(Attribute attr,
Operation *SPIRVDialect::materializeConstant(OpBuilder &builder, Operation *SPIRVDialect::materializeConstant(OpBuilder &builder,
Attribute value, Type type, Attribute value, Type type,
Location loc) { Location loc) {
if (!ConstantOp::isBuildableWith(type)) if (!spirv::ConstantOp::isBuildableWith(type))
return nullptr; return nullptr;
return builder.create<spirv::ConstantOp>(loc, type, value); return builder.create<spirv::ConstantOp>(loc, type, value);
@ -832,12 +856,7 @@ LogicalResult SPIRVDialect::verifyOperationAttribute(Operation *op,
"32-bit integer elements attribute: 'local_size'"; "32-bit integer elements attribute: 'local_size'";
} else if (symbol == spirv::getTargetEnvAttrName()) { } else if (symbol == spirv::getTargetEnvAttrName()) {
if (!attr.isa<spirv::TargetEnvAttr>()) if (!attr.isa<spirv::TargetEnvAttr>())
return op->emitError("'") return op->emitError("'") << symbol << "' must be a spirv::TargetEnvAttr";
<< symbol
<< "' must be a dictionary attribute containing one 32-bit "
"integer attribute 'version', one string array attribute "
"'extensions', one 32-bit integer array attribute "
"'capabilities', and one dictionary attribute 'limits'";
} else { } else {
return op->emitError("found unsupported '") return op->emitError("found unsupported '")
<< symbol << "' attribute on operation"; << symbol << "' attribute on operation";

View File

@ -14,95 +14,123 @@
using namespace mlir; using namespace mlir;
//===----------------------------------------------------------------------===//
// DictionaryDict derived attributes
//===----------------------------------------------------------------------===//
namespace mlir { namespace mlir {
#include "mlir/Dialect/SPIRV/TargetAndABI.cpp.inc" #include "mlir/Dialect/SPIRV/TargetAndABI.cpp.inc"
//===----------------------------------------------------------------------===//
// Attribute storage classes
//===----------------------------------------------------------------------===//
namespace spirv { namespace spirv {
namespace detail { namespace detail {
struct TargetEnvAttributeStorage : public AttributeStorage { struct VerCapExtAttributeStorage : public AttributeStorage {
using KeyTy = std::tuple<Attribute, Attribute, Attribute, Attribute>; using KeyTy = std::tuple<Attribute, Attribute, Attribute>;
TargetEnvAttributeStorage(Attribute version, Attribute extensions, VerCapExtAttributeStorage(Attribute version, Attribute capabilities,
Attribute capabilities, Attribute limits) Attribute extensions)
: version(version), extensions(extensions), capabilities(capabilities), : version(version), capabilities(capabilities), extensions(extensions) {}
limits(limits) {}
bool operator==(const KeyTy &key) const { bool operator==(const KeyTy &key) const {
return std::get<0>(key) == version && std::get<1>(key) == extensions && return std::get<0>(key) == version && std::get<1>(key) == capabilities &&
std::get<2>(key) == capabilities && std::get<3>(key) == limits; std::get<2>(key) == extensions;
}
static VerCapExtAttributeStorage *
construct(AttributeStorageAllocator &allocator, const KeyTy &key) {
return new (allocator.allocate<VerCapExtAttributeStorage>())
VerCapExtAttributeStorage(std::get<0>(key), std::get<1>(key),
std::get<2>(key));
}
Attribute version;
Attribute capabilities;
Attribute extensions;
};
struct TargetEnvAttributeStorage : public AttributeStorage {
using KeyTy = std::pair<Attribute, Attribute>;
TargetEnvAttributeStorage(Attribute triple, Attribute limits)
: triple(triple), limits(limits) {}
bool operator==(const KeyTy &key) const {
return key.first == triple && key.second == limits;
} }
static TargetEnvAttributeStorage * static TargetEnvAttributeStorage *
construct(AttributeStorageAllocator &allocator, const KeyTy &key) { construct(AttributeStorageAllocator &allocator, const KeyTy &key) {
return new (allocator.allocate<TargetEnvAttributeStorage>()) return new (allocator.allocate<TargetEnvAttributeStorage>())
TargetEnvAttributeStorage(std::get<0>(key), std::get<1>(key), TargetEnvAttributeStorage(key.first, key.second);
std::get<2>(key), std::get<3>(key));
} }
Attribute version; Attribute triple;
Attribute extensions;
Attribute capabilities;
Attribute limits; Attribute limits;
}; };
} // namespace detail } // namespace detail
} // namespace spirv } // namespace spirv
} // namespace mlir } // namespace mlir
spirv::TargetEnvAttr spirv::TargetEnvAttr::get( //===----------------------------------------------------------------------===//
spirv::Version version, ArrayRef<spirv::Extension> extensions, // VerCapExtAttr
ArrayRef<spirv::Capability> capabilities, DictionaryAttr limits) { //===----------------------------------------------------------------------===//
Builder b(limits.getContext());
spirv::VerCapExtAttr spirv::VerCapExtAttr::get(
spirv::Version version, ArrayRef<spirv::Capability> capabilities,
ArrayRef<spirv::Extension> extensions, MLIRContext *context) {
Builder b(context);
auto versionAttr = b.getI32IntegerAttr(static_cast<uint32_t>(version)); auto versionAttr = b.getI32IntegerAttr(static_cast<uint32_t>(version));
SmallVector<Attribute, 4> extAttrs;
extAttrs.reserve(extensions.size());
for (spirv::Extension ext : extensions)
extAttrs.push_back(b.getStringAttr(spirv::stringifyExtension(ext)));
SmallVector<Attribute, 4> capAttrs; SmallVector<Attribute, 4> capAttrs;
capAttrs.reserve(capabilities.size()); capAttrs.reserve(capabilities.size());
for (spirv::Capability cap : capabilities) for (spirv::Capability cap : capabilities)
capAttrs.push_back(b.getI32IntegerAttr(static_cast<uint32_t>(cap))); capAttrs.push_back(b.getI32IntegerAttr(static_cast<uint32_t>(cap)));
return get(versionAttr, b.getArrayAttr(extAttrs), b.getArrayAttr(capAttrs), SmallVector<Attribute, 4> extAttrs;
limits); extAttrs.reserve(extensions.size());
for (spirv::Extension ext : extensions)
extAttrs.push_back(b.getStringAttr(spirv::stringifyExtension(ext)));
return get(versionAttr, b.getArrayAttr(capAttrs), b.getArrayAttr(extAttrs));
} }
spirv::TargetEnvAttr spirv::TargetEnvAttr::get(IntegerAttr version, spirv::VerCapExtAttr spirv::VerCapExtAttr::get(IntegerAttr version,
ArrayAttr extensions,
ArrayAttr capabilities, ArrayAttr capabilities,
DictionaryAttr limits) { ArrayAttr extensions) {
assert(version && extensions && capabilities && limits); assert(version && capabilities && extensions);
MLIRContext *context = version.getContext(); MLIRContext *context = version.getContext();
return Base::get(context, spirv::AttrKind::TargetEnv, version, extensions, return Base::get(context, spirv::AttrKind::VerCapExt, version, capabilities,
capabilities, limits); extensions);
} }
StringRef spirv::TargetEnvAttr::getKindName() { return "target_env"; } StringRef spirv::VerCapExtAttr::getKindName() { return "vce"; }
spirv::Version spirv::TargetEnvAttr::getVersion() { spirv::Version spirv::VerCapExtAttr::getVersion() {
return static_cast<spirv::Version>( return static_cast<spirv::Version>(
getImpl()->version.cast<IntegerAttr>().getValue().getZExtValue()); getImpl()->version.cast<IntegerAttr>().getValue().getZExtValue());
} }
spirv::TargetEnvAttr::ext_iterator::ext_iterator(ArrayAttr::iterator it) spirv::VerCapExtAttr::ext_iterator::ext_iterator(ArrayAttr::iterator it)
: llvm::mapped_iterator<ArrayAttr::iterator, : llvm::mapped_iterator<ArrayAttr::iterator,
spirv::Extension (*)(Attribute)>( spirv::Extension (*)(Attribute)>(
it, [](Attribute attr) { it, [](Attribute attr) {
return *symbolizeExtension(attr.cast<StringAttr>().getValue()); return *symbolizeExtension(attr.cast<StringAttr>().getValue());
}) {} }) {}
spirv::TargetEnvAttr::ext_range spirv::TargetEnvAttr::getExtensions() { spirv::VerCapExtAttr::ext_range spirv::VerCapExtAttr::getExtensions() {
auto range = getExtensionsAttr().getValue(); auto range = getExtensionsAttr().getValue();
return {ext_iterator(range.begin()), ext_iterator(range.end())}; return {ext_iterator(range.begin()), ext_iterator(range.end())};
} }
ArrayAttr spirv::TargetEnvAttr::getExtensionsAttr() { ArrayAttr spirv::VerCapExtAttr::getExtensionsAttr() {
return getImpl()->extensions.cast<ArrayAttr>(); return getImpl()->extensions.cast<ArrayAttr>();
} }
spirv::TargetEnvAttr::cap_iterator::cap_iterator(ArrayAttr::iterator it) spirv::VerCapExtAttr::cap_iterator::cap_iterator(ArrayAttr::iterator it)
: llvm::mapped_iterator<ArrayAttr::iterator, : llvm::mapped_iterator<ArrayAttr::iterator,
spirv::Capability (*)(Attribute)>( spirv::Capability (*)(Attribute)>(
it, [](Attribute attr) { it, [](Attribute attr) {
@ -110,33 +138,21 @@ spirv::TargetEnvAttr::cap_iterator::cap_iterator(ArrayAttr::iterator it)
attr.cast<IntegerAttr>().getValue().getZExtValue()); attr.cast<IntegerAttr>().getValue().getZExtValue());
}) {} }) {}
spirv::TargetEnvAttr::cap_range spirv::TargetEnvAttr::getCapabilities() { spirv::VerCapExtAttr::cap_range spirv::VerCapExtAttr::getCapabilities() {
auto range = getCapabilitiesAttr().getValue(); auto range = getCapabilitiesAttr().getValue();
return {cap_iterator(range.begin()), cap_iterator(range.end())}; return {cap_iterator(range.begin()), cap_iterator(range.end())};
} }
ArrayAttr spirv::TargetEnvAttr::getCapabilitiesAttr() { ArrayAttr spirv::VerCapExtAttr::getCapabilitiesAttr() {
return getImpl()->capabilities.cast<ArrayAttr>(); return getImpl()->capabilities.cast<ArrayAttr>();
} }
spirv::ResourceLimitsAttr spirv::TargetEnvAttr::getResourceLimits() { LogicalResult spirv::VerCapExtAttr::verifyConstructionInvariants(
return getImpl()->limits.cast<spirv::ResourceLimitsAttr>(); Location loc, IntegerAttr version, ArrayAttr capabilities,
} ArrayAttr extensions) {
LogicalResult spirv::TargetEnvAttr::verifyConstructionInvariants(
Location loc, IntegerAttr version, ArrayAttr extensions,
ArrayAttr capabilities, DictionaryAttr limits) {
if (!version.getType().isSignlessInteger(32)) if (!version.getType().isSignlessInteger(32))
return emitError(loc, "expected 32-bit integer for version"); return emitError(loc, "expected 32-bit integer for version");
if (!llvm::all_of(extensions.getValue(), [](Attribute attr) {
if (auto strAttr = attr.dyn_cast<StringAttr>())
if (spirv::symbolizeExtension(strAttr.getValue()))
return true;
return false;
}))
return emitError(loc, "unknown extension in extension list");
if (!llvm::all_of(capabilities.getValue(), [](Attribute attr) { if (!llvm::all_of(capabilities.getValue(), [](Attribute attr) {
if (auto intAttr = attr.dyn_cast<IntegerAttr>()) if (auto intAttr = attr.dyn_cast<IntegerAttr>())
if (spirv::symbolizeCapability(intAttr.getValue().getZExtValue())) if (spirv::symbolizeCapability(intAttr.getValue().getZExtValue()))
@ -145,12 +161,70 @@ LogicalResult spirv::TargetEnvAttr::verifyConstructionInvariants(
})) }))
return emitError(loc, "unknown capability in capability list"); return emitError(loc, "unknown capability in capability list");
if (!llvm::all_of(extensions.getValue(), [](Attribute attr) {
if (auto strAttr = attr.dyn_cast<StringAttr>())
if (spirv::symbolizeExtension(strAttr.getValue()))
return true;
return false;
}))
return emitError(loc, "unknown extension in extension list");
return success();
}
//===----------------------------------------------------------------------===//
// TargetEnvAttr
//===----------------------------------------------------------------------===//
spirv::TargetEnvAttr spirv::TargetEnvAttr::get(spirv::VerCapExtAttr triple,
DictionaryAttr limits) {
assert(triple && limits && "expected valid triple and limits");
MLIRContext *context = triple.getContext();
return Base::get(context, spirv::AttrKind::TargetEnv, triple, limits);
}
StringRef spirv::TargetEnvAttr::getKindName() { return "target_env"; }
spirv::VerCapExtAttr spirv::TargetEnvAttr::getTripleAttr() {
return getImpl()->triple.cast<spirv::VerCapExtAttr>();
}
spirv::Version spirv::TargetEnvAttr::getVersion() {
return getTripleAttr().getVersion();
}
spirv::VerCapExtAttr::ext_range spirv::TargetEnvAttr::getExtensions() {
return getTripleAttr().getExtensions();
}
ArrayAttr spirv::TargetEnvAttr::getExtensionsAttr() {
return getTripleAttr().getExtensionsAttr();
}
spirv::VerCapExtAttr::cap_range spirv::TargetEnvAttr::getCapabilities() {
return getTripleAttr().getCapabilities();
}
ArrayAttr spirv::TargetEnvAttr::getCapabilitiesAttr() {
return getTripleAttr().getCapabilitiesAttr();
}
spirv::ResourceLimitsAttr spirv::TargetEnvAttr::getResourceLimits() {
return getImpl()->limits.cast<spirv::ResourceLimitsAttr>();
}
LogicalResult spirv::TargetEnvAttr::verifyConstructionInvariants(
Location loc, spirv::VerCapExtAttr triple, DictionaryAttr limits) {
if (!limits.isa<spirv::ResourceLimitsAttr>()) if (!limits.isa<spirv::ResourceLimitsAttr>())
return emitError(loc, "expected spirv::ResourceLimitsAttr for limits"); return emitError(loc, "expected spirv::ResourceLimitsAttr for limits");
return success(); return success();
} }
//===----------------------------------------------------------------------===//
// Utility functions
//===----------------------------------------------------------------------===//
StringRef spirv::getInterfaceVarABIAttrName() { StringRef spirv::getInterfaceVarABIAttrName() {
return "spv.interface_var_abi"; return "spv.interface_var_abi";
} }
@ -212,13 +286,11 @@ spirv::getDefaultResourceLimits(MLIRContext *context) {
StringRef spirv::getTargetEnvAttrName() { return "spv.target_env"; } StringRef spirv::getTargetEnvAttrName() { return "spv.target_env"; }
spirv::TargetEnvAttr spirv::getDefaultTargetEnv(MLIRContext *context) { spirv::TargetEnvAttr spirv::getDefaultTargetEnv(MLIRContext *context) {
Builder builder(context); auto triple = spirv::VerCapExtAttr::get(spirv::Version::V_1_0,
return spirv::TargetEnvAttr::get( {spirv::Capability::Shader},
builder.getI32IntegerAttr(static_cast<uint32_t>(spirv::Version::V_1_0)), ArrayRef<Extension>(), context);
builder.getI32ArrayAttr({}), return spirv::TargetEnvAttr::get(triple,
builder.getI32ArrayAttr( spirv::getDefaultResourceLimits(context));
{static_cast<uint32_t>(spirv::Capability::Shader)}),
spirv::getDefaultResourceLimits(context));
} }
spirv::TargetEnvAttr spirv::lookupTargetEnvOrDefault(Operation *op) { spirv::TargetEnvAttr spirv::lookupTargetEnvOrDefault(Operation *op) {

View File

@ -16,7 +16,7 @@
module attributes { module attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_3, [], [Shader, GroupNonUniformArithmetic], #spv.vce<v1.3, [Shader, GroupNonUniformArithmetic], []>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>
@ -78,7 +78,7 @@ func @single_workgroup_reduction(%input: memref<16xi32>, %output: memref<1xi32>)
module attributes { module attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_3, [], [Shader, GroupNonUniformArithmetic], #spv.vce<v1.3, [Shader, GroupNonUniformArithmetic], []>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>
@ -111,7 +111,7 @@ func @single_workgroup_reduction(%input: memref<16xi32>, %output: memref<1xi32>)
module attributes { module attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_3, [], [Shader, GroupNonUniformArithmetic], #spv.vce<v1.3, [Shader, GroupNonUniformArithmetic], []>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>
@ -146,7 +146,7 @@ func @single_workgroup_reduction(%input: memref<16xi32>, %output: memref<1xi32>)
module attributes { module attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_3, [], [Shader, GroupNonUniformArithmetic], #spv.vce<v1.3, [Shader, GroupNonUniformArithmetic], []>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>

View File

@ -2,8 +2,8 @@
// CHECK-LABEL: iadd // CHECK-LABEL: iadd
func @iadd(%arg: i32) -> i32 { func @iadd(%arg: i32) -> i32 {
// CHECK: min version: V_1_0 // CHECK: min version: v1.0
// CHECK: max version: V_1_5 // CHECK: max version: v1.5
// CHECK: extensions: [ ] // CHECK: extensions: [ ]
// CHECK: capabilities: [ ] // CHECK: capabilities: [ ]
%0 = spv.IAdd %arg, %arg: i32 %0 = spv.IAdd %arg, %arg: i32
@ -12,8 +12,8 @@ func @iadd(%arg: i32) -> i32 {
// CHECK: atomic_compare_exchange_weak // CHECK: atomic_compare_exchange_weak
func @atomic_compare_exchange_weak(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 { func @atomic_compare_exchange_weak(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 {
// CHECK: min version: V_1_0 // CHECK: min version: v1.0
// CHECK: max version: V_1_3 // CHECK: max version: v1.3
// CHECK: extensions: [ ] // CHECK: extensions: [ ]
// CHECK: capabilities: [ [Kernel] ] // CHECK: capabilities: [ [Kernel] ]
%0 = spv.AtomicCompareExchangeWeak "Workgroup" "Release" "Acquire" %ptr, %value, %comparator: !spv.ptr<i32, Workgroup> %0 = spv.AtomicCompareExchangeWeak "Workgroup" "Release" "Acquire" %ptr, %value, %comparator: !spv.ptr<i32, Workgroup>
@ -22,8 +22,8 @@ func @atomic_compare_exchange_weak(%ptr: !spv.ptr<i32, Workgroup>, %value: i32,
// CHECK-LABEL: subgroup_ballot // CHECK-LABEL: subgroup_ballot
func @subgroup_ballot(%predicate: i1) -> vector<4xi32> { func @subgroup_ballot(%predicate: i1) -> vector<4xi32> {
// CHECK: min version: V_1_3 // CHECK: min version: v1.3
// CHECK: max version: V_1_5 // CHECK: max version: v1.5
// CHECK: extensions: [ ] // CHECK: extensions: [ ]
// CHECK: capabilities: [ [GroupNonUniformBallot] ] // CHECK: capabilities: [ [GroupNonUniformBallot] ]
%0 = spv.GroupNonUniformBallot "Workgroup" %predicate : vector<4xi32> %0 = spv.GroupNonUniformBallot "Workgroup" %predicate : vector<4xi32>
@ -32,8 +32,8 @@ func @subgroup_ballot(%predicate: i1) -> vector<4xi32> {
// CHECK-LABEL: module_logical_glsl450 // CHECK-LABEL: module_logical_glsl450
func @module_logical_glsl450() { func @module_logical_glsl450() {
// CHECK: spv.module min version: V_1_0 // CHECK: spv.module min version: v1.0
// CHECK: spv.module max version: V_1_5 // CHECK: spv.module max version: v1.5
// CHECK: spv.module extensions: [ ] // CHECK: spv.module extensions: [ ]
// CHECK: spv.module capabilities: [ [Shader] ] // CHECK: spv.module capabilities: [ [Shader] ]
spv.module "Logical" "GLSL450" { } spv.module "Logical" "GLSL450" { }
@ -42,8 +42,8 @@ func @module_logical_glsl450() {
// CHECK-LABEL: module_physical_storage_buffer64_vulkan // CHECK-LABEL: module_physical_storage_buffer64_vulkan
func @module_physical_storage_buffer64_vulkan() { func @module_physical_storage_buffer64_vulkan() {
// CHECK: spv.module min version: V_1_0 // CHECK: spv.module min version: v1.0
// CHECK: spv.module max version: V_1_5 // CHECK: spv.module max version: v1.5
// CHECK: spv.module extensions: [ [SPV_EXT_physical_storage_buffer, SPV_KHR_physical_storage_buffer] [SPV_KHR_vulkan_memory_model] ] // CHECK: spv.module extensions: [ [SPV_EXT_physical_storage_buffer, SPV_KHR_physical_storage_buffer] [SPV_KHR_vulkan_memory_model] ]
// CHECK: spv.module capabilities: [ [PhysicalStorageBufferAddresses] [VulkanMemoryModel] ] // CHECK: spv.module capabilities: [ [PhysicalStorageBufferAddresses] [VulkanMemoryModel] ]
spv.module "PhysicalStorageBuffer64" "Vulkan" { } spv.module "PhysicalStorageBuffer64" "Vulkan" { }

View File

@ -106,51 +106,9 @@ func @interface_var() -> (f32 {spv.interface_var_abi = {
// spv.target_env // spv.target_env
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
func @target_env_wrong_type() attributes {
// expected-error @+1 {{expected valid keyword}}
spv.target_env = #spv.target_env<64>
} { return }
// -----
func @target_env_missing_fields() attributes {
// expected-error @+1 {{expected ','}}
spv.target_env = #spv.target_env<V_1_0>
} { return }
// -----
func @target_env_wrong_version() attributes {
// expected-error @+1 {{unknown version: V_x_y}}
spv.target_env = #spv.target_env<V_x_y, []>
} { return }
// -----
func @target_env_wrong_extension_type() attributes {
// expected-error @+1 {{expected valid keyword}}
spv.target_env = #spv.target_env<V_1_0, [32: i32], [Shader]>
} { return }
// -----
func @target_env_wrong_extension() attributes {
// expected-error @+1 {{unknown extension: SPV_Something}}
spv.target_env = #spv.target_env<V_1_0, [SPV_Something], [Shader]>
} { return }
// -----
func @target_env_wrong_capability() attributes {
// expected-error @+1 {{unknown capability: Something}}
spv.target_env = #spv.target_env<V_1_0, [], [Something]>
} { return }
// -----
func @target_env_missing_limits() attributes { func @target_env_missing_limits() attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_0, [SPV_KHR_storage_buffer_storage_class], [Shader], #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>,
// expected-error @+1 {{limits must be a dictionary attribute containing two 32-bit integer attributes 'max_compute_workgroup_invocations' and 'max_compute_workgroup_size'}} // expected-error @+1 {{limits must be a dictionary attribute containing two 32-bit integer attributes 'max_compute_workgroup_invocations' and 'max_compute_workgroup_size'}}
{max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}> {max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}>
} { return } } { return }
@ -159,7 +117,7 @@ func @target_env_missing_limits() attributes {
func @target_env_wrong_limits() attributes { func @target_env_wrong_limits() attributes {
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_0, [SPV_KHR_storage_buffer_storage_class], [Shader], #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>,
// expected-error @+1 {{limits must be a dictionary attribute containing two 32-bit integer attributes 'max_compute_workgroup_invocations' and 'max_compute_workgroup_size'}} // expected-error @+1 {{limits must be a dictionary attribute containing two 32-bit integer attributes 'max_compute_workgroup_invocations' and 'max_compute_workgroup_size'}}
{max_compute_workgroup_invocations = 128 : i64, max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}> {max_compute_workgroup_invocations = 128 : i64, max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}>
} { return } } { return }
@ -167,10 +125,11 @@ func @target_env_wrong_limits() attributes {
// ----- // -----
func @target_env() attributes { func @target_env() attributes {
// CHECK: spv.target_env = #spv.target_env<
// CHECK: spv.target_env = #spv.target_env<V_1_0, [SPV_KHR_storage_buffer_storage_class], [Shader], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}> // CHECK-SAME: #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>,
// CHECK-SAME: {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>}>
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_0, [SPV_KHR_storage_buffer_storage_class], [Shader], #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>
@ -182,7 +141,7 @@ func @target_env() attributes {
func @target_env_extra_fields() attributes { func @target_env_extra_fields() attributes {
// expected-error @+6 {{expected '>'}} // expected-error @+6 {{expected '>'}}
spv.target_env = #spv.target_env< spv.target_env = #spv.target_env<
V_1_0, [SPV_KHR_storage_buffer_storage_class], [Shader], #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>,
{ {
max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_invocations = 128 : i32,
max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32> max_compute_workgroup_size = dense<[128, 64, 64]> : vector<3xi32>
@ -190,3 +149,56 @@ func @target_env_extra_fields() attributes {
more_stuff more_stuff
> >
} { return } } { return }
// -----
//===----------------------------------------------------------------------===//
// spv.vce
//===----------------------------------------------------------------------===//
func @vce_wrong_type() attributes {
// expected-error @+1 {{expected valid keyword}}
vce = #spv.vce<64>
} { return }
// -----
func @vce_missing_fields() attributes {
// expected-error @+1 {{expected ','}}
vce = #spv.vce<v1.0>
} { return }
// -----
func @vce_wrong_version() attributes {
// expected-error @+1 {{unknown version: V_x_y}}
vce = #spv.vce<V_x_y, []>
} { return }
// -----
func @vce_wrong_extension_type() attributes {
// expected-error @+1 {{expected valid keyword}}
vce = #spv.vce<v1.0, [32: i32], [Shader]>
} { return }
// -----
func @vce_wrong_extension() attributes {
// expected-error @+1 {{unknown extension: SPV_Something}}
vce = #spv.vce<v1.0, [Shader], [SPV_Something]>
} { return }
// -----
func @vce_wrong_capability() attributes {
// expected-error @+1 {{unknown capability: Something}}
vce = #spv.vce<v1.0, [Something], []>
} { return }
// -----
func @vce() attributes {
// CHECK: #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>
vce = #spv.vce<v1.0, [Shader], [SPV_KHR_storage_buffer_storage_class]>
} { return }

View File

@ -35,7 +35,7 @@
// CHECK-LABEL: @cmp_exchange_weak_suitable_version_capabilities // CHECK-LABEL: @cmp_exchange_weak_suitable_version_capabilities
func @cmp_exchange_weak_suitable_version_capabilities(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes { func @cmp_exchange_weak_suitable_version_capabilities(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_1, [], [Kernel, AtomicStorage], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.1, [Kernel, AtomicStorage], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.AtomicCompareExchangeWeak "Workgroup" "AcquireRelease|AtomicCounterMemory" "Acquire" // CHECK: spv.AtomicCompareExchangeWeak "Workgroup" "AcquireRelease|AtomicCounterMemory" "Acquire"
%0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32) %0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32)
@ -44,7 +44,7 @@ func @cmp_exchange_weak_suitable_version_capabilities(%ptr: !spv.ptr<i32, Workgr
// CHECK-LABEL: @cmp_exchange_weak_unsupported_version // CHECK-LABEL: @cmp_exchange_weak_unsupported_version
func @cmp_exchange_weak_unsupported_version(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes { func @cmp_exchange_weak_unsupported_version(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_4, [], [Kernel, AtomicStorage], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.4, [Kernel, AtomicStorage], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_atomic_compare_exchange_weak_op // CHECK: test.convert_to_atomic_compare_exchange_weak_op
%0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32) %0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32)
@ -57,7 +57,7 @@ func @cmp_exchange_weak_unsupported_version(%ptr: !spv.ptr<i32, Workgroup>, %val
// CHECK-LABEL: @group_non_uniform_ballot_suitable_version // CHECK-LABEL: @group_non_uniform_ballot_suitable_version
func @group_non_uniform_ballot_suitable_version(%predicate: i1) -> vector<4xi32> attributes { func @group_non_uniform_ballot_suitable_version(%predicate: i1) -> vector<4xi32> attributes {
spv.target_env = #spv.target_env<V_1_4, [], [GroupNonUniformBallot], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.4, [GroupNonUniformBallot], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.GroupNonUniformBallot "Workgroup" // CHECK: spv.GroupNonUniformBallot "Workgroup"
%0 = "test.convert_to_group_non_uniform_ballot_op"(%predicate): (i1) -> (vector<4xi32>) %0 = "test.convert_to_group_non_uniform_ballot_op"(%predicate): (i1) -> (vector<4xi32>)
@ -66,7 +66,7 @@ func @group_non_uniform_ballot_suitable_version(%predicate: i1) -> vector<4xi32>
// CHECK-LABEL: @group_non_uniform_ballot_unsupported_version // CHECK-LABEL: @group_non_uniform_ballot_unsupported_version
func @group_non_uniform_ballot_unsupported_version(%predicate: i1) -> vector<4xi32> attributes { func @group_non_uniform_ballot_unsupported_version(%predicate: i1) -> vector<4xi32> attributes {
spv.target_env = #spv.target_env<V_1_1, [], [GroupNonUniformBallot], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.1, [GroupNonUniformBallot], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_group_non_uniform_ballot_op // CHECK: test.convert_to_group_non_uniform_ballot_op
%0 = "test.convert_to_group_non_uniform_ballot_op"(%predicate): (i1) -> (vector<4xi32>) %0 = "test.convert_to_group_non_uniform_ballot_op"(%predicate): (i1) -> (vector<4xi32>)
@ -79,7 +79,7 @@ func @group_non_uniform_ballot_unsupported_version(%predicate: i1) -> vector<4xi
// CHECK-LABEL: @cmp_exchange_weak_missing_capability_kernel // CHECK-LABEL: @cmp_exchange_weak_missing_capability_kernel
func @cmp_exchange_weak_missing_capability_kernel(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes { func @cmp_exchange_weak_missing_capability_kernel(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_3, [], [AtomicStorage], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.3, [AtomicStorage], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_atomic_compare_exchange_weak_op // CHECK: test.convert_to_atomic_compare_exchange_weak_op
%0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32) %0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32)
@ -88,7 +88,7 @@ func @cmp_exchange_weak_missing_capability_kernel(%ptr: !spv.ptr<i32, Workgroup>
// CHECK-LABEL: @cmp_exchange_weak_missing_capability_atomic_storage // CHECK-LABEL: @cmp_exchange_weak_missing_capability_atomic_storage
func @cmp_exchange_weak_missing_capability_atomic_storage(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes { func @cmp_exchange_weak_missing_capability_atomic_storage(%ptr: !spv.ptr<i32, Workgroup>, %value: i32, %comparator: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_3, [], [Kernel], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.3, [Kernel], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_atomic_compare_exchange_weak_op // CHECK: test.convert_to_atomic_compare_exchange_weak_op
%0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32) %0 = "test.convert_to_atomic_compare_exchange_weak_op"(%ptr, %value, %comparator): (!spv.ptr<i32, Workgroup>, i32, i32) -> (i32)
@ -97,7 +97,7 @@ func @cmp_exchange_weak_missing_capability_atomic_storage(%ptr: !spv.ptr<i32, Wo
// CHECK-LABEL: @subgroup_ballot_missing_capability // CHECK-LABEL: @subgroup_ballot_missing_capability
func @subgroup_ballot_missing_capability(%predicate: i1) -> vector<4xi32> attributes { func @subgroup_ballot_missing_capability(%predicate: i1) -> vector<4xi32> attributes {
spv.target_env = #spv.target_env<V_1_4, [SPV_KHR_shader_ballot], [], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.4, [], [SPV_KHR_shader_ballot]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_subgroup_ballot_op // CHECK: test.convert_to_subgroup_ballot_op
%0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>) %0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>)
@ -106,7 +106,7 @@ func @subgroup_ballot_missing_capability(%predicate: i1) -> vector<4xi32> attrib
// CHECK-LABEL: @bit_reverse_directly_implied_capability // CHECK-LABEL: @bit_reverse_directly_implied_capability
func @bit_reverse_directly_implied_capability(%operand: i32) -> i32 attributes { func @bit_reverse_directly_implied_capability(%operand: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_0, [], [Geometry], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [Geometry], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.BitReverse // CHECK: spv.BitReverse
%0 = "test.convert_to_bit_reverse_op"(%operand): (i32) -> (i32) %0 = "test.convert_to_bit_reverse_op"(%operand): (i32) -> (i32)
@ -115,7 +115,7 @@ func @bit_reverse_directly_implied_capability(%operand: i32) -> i32 attributes {
// CHECK-LABEL: @bit_reverse_recursively_implied_capability // CHECK-LABEL: @bit_reverse_recursively_implied_capability
func @bit_reverse_recursively_implied_capability(%operand: i32) -> i32 attributes { func @bit_reverse_recursively_implied_capability(%operand: i32) -> i32 attributes {
spv.target_env = #spv.target_env<V_1_0, [], [GeometryPointSize], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [GeometryPointSize], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.BitReverse // CHECK: spv.BitReverse
%0 = "test.convert_to_bit_reverse_op"(%operand): (i32) -> (i32) %0 = "test.convert_to_bit_reverse_op"(%operand): (i32) -> (i32)
@ -128,7 +128,7 @@ func @bit_reverse_recursively_implied_capability(%operand: i32) -> i32 attribute
// CHECK-LABEL: @subgroup_ballot_suitable_extension // CHECK-LABEL: @subgroup_ballot_suitable_extension
func @subgroup_ballot_suitable_extension(%predicate: i1) -> vector<4xi32> attributes { func @subgroup_ballot_suitable_extension(%predicate: i1) -> vector<4xi32> attributes {
spv.target_env = #spv.target_env<V_1_4, [SPV_KHR_shader_ballot], [SubgroupBallotKHR], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.4, [SubgroupBallotKHR], [SPV_KHR_shader_ballot]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.SubgroupBallotKHR // CHECK: spv.SubgroupBallotKHR
%0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>) %0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>)
@ -137,7 +137,7 @@ func @subgroup_ballot_suitable_extension(%predicate: i1) -> vector<4xi32> attrib
// CHECK-LABEL: @subgroup_ballot_missing_extension // CHECK-LABEL: @subgroup_ballot_missing_extension
func @subgroup_ballot_missing_extension(%predicate: i1) -> vector<4xi32> attributes { func @subgroup_ballot_missing_extension(%predicate: i1) -> vector<4xi32> attributes {
spv.target_env = #spv.target_env<V_1_4, [], [SubgroupBallotKHR], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.4, [SubgroupBallotKHR], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_subgroup_ballot_op // CHECK: test.convert_to_subgroup_ballot_op
%0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>) %0 = "test.convert_to_subgroup_ballot_op"(%predicate): (i1) -> (vector<4xi32>)
@ -146,7 +146,7 @@ func @subgroup_ballot_missing_extension(%predicate: i1) -> vector<4xi32> attribu
// CHECK-LABEL: @module_suitable_extension1 // CHECK-LABEL: @module_suitable_extension1
func @module_suitable_extension1() attributes { func @module_suitable_extension1() attributes {
spv.target_env = #spv.target_env<V_1_0, [SPV_KHR_vulkan_memory_model, SPV_EXT_physical_storage_buffer], [VulkanMemoryModel, PhysicalStorageBufferAddresses], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [VulkanMemoryModel, PhysicalStorageBufferAddresses], [SPV_KHR_vulkan_memory_model, SPV_EXT_physical_storage_buffer]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan" // CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan"
"test.convert_to_module_op"() : () ->() "test.convert_to_module_op"() : () ->()
@ -155,7 +155,7 @@ func @module_suitable_extension1() attributes {
// CHECK-LABEL: @module_suitable_extension2 // CHECK-LABEL: @module_suitable_extension2
func @module_suitable_extension2() attributes { func @module_suitable_extension2() attributes {
spv.target_env = #spv.target_env<V_1_0, [SPV_KHR_vulkan_memory_model, SPV_KHR_physical_storage_buffer], [VulkanMemoryModel, PhysicalStorageBufferAddresses], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [VulkanMemoryModel, PhysicalStorageBufferAddresses], [SPV_KHR_vulkan_memory_model, SPV_KHR_physical_storage_buffer]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan" // CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan"
"test.convert_to_module_op"() : () -> () "test.convert_to_module_op"() : () -> ()
@ -164,7 +164,7 @@ func @module_suitable_extension2() attributes {
// CHECK-LABEL: @module_missing_extension_mm // CHECK-LABEL: @module_missing_extension_mm
func @module_missing_extension_mm() attributes { func @module_missing_extension_mm() attributes {
spv.target_env = #spv.target_env<V_1_0, [SPV_KHR_physical_storage_buffer], [VulkanMemoryModel, PhysicalStorageBufferAddresses], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [VulkanMemoryModel, PhysicalStorageBufferAddresses], [SPV_KHR_physical_storage_buffer]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_module_op // CHECK: test.convert_to_module_op
"test.convert_to_module_op"() : () -> () "test.convert_to_module_op"() : () -> ()
@ -173,7 +173,7 @@ func @module_missing_extension_mm() attributes {
// CHECK-LABEL: @module_missing_extension_am // CHECK-LABEL: @module_missing_extension_am
func @module_missing_extension_am() attributes { func @module_missing_extension_am() attributes {
spv.target_env = #spv.target_env<V_1_0, [SPV_KHR_vulkan_memory_model], [VulkanMemoryModel, PhysicalStorageBufferAddresses], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.0, [VulkanMemoryModel, PhysicalStorageBufferAddresses], [SPV_KHR_vulkan_memory_model]>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: test.convert_to_module_op // CHECK: test.convert_to_module_op
"test.convert_to_module_op"() : () -> () "test.convert_to_module_op"() : () -> ()
@ -183,7 +183,7 @@ func @module_missing_extension_am() attributes {
// CHECK-LABEL: @module_implied_extension // CHECK-LABEL: @module_implied_extension
func @module_implied_extension() attributes { func @module_implied_extension() attributes {
// Version 1.5 implies SPV_KHR_vulkan_memory_model and SPV_KHR_physical_storage_buffer. // Version 1.5 implies SPV_KHR_vulkan_memory_model and SPV_KHR_physical_storage_buffer.
spv.target_env = #spv.target_env<V_1_5, [], [VulkanMemoryModel, PhysicalStorageBufferAddresses], {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}> spv.target_env = #spv.target_env<#spv.vce<v1.5, [VulkanMemoryModel, PhysicalStorageBufferAddresses], []>, {max_compute_workgroup_invocations = 128 : i32, max_compute_workgroup_size = dense<[128, 128, 64]> : vector<3xi32>}>
} { } {
// CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan" // CHECK: spv.module "PhysicalStorageBuffer64" "Vulkan"
"test.convert_to_module_op"() : () -> () "test.convert_to_module_op"() : () -> ()