forked from OSchip/llvm-project
[mlir:PDL] Rework errors for pdl.operations with non-inferrable results
We currently emit an error during verification if a pdl.operation with non-inferrable results is used within a rewrite. This allows for catching some errors during compile time, but is slightly broken. For one, the verification at the PDL level assumes that all dialects have been loaded, which is true at run time, but may not be true when the PDL is generated (such as via PDLL). This commit fixes this by not emitting the error if the operation isn't registered, i.e. it uses the `mightHave` variant of trait/interface methods. Secondly, we currently don't verify when a pdl.operation has no explicit results, but the operation being created is known to expect at least one. This commit adds a heuristic error to detect these cases when possible and fail. We can't always capture when the user made an error, but we can capture the most common case where the user expected an operation to infer its result types (when it actually isn't possible). Differential Revision: https://reviews.llvm.org/D124583
This commit is contained in:
parent
d4381b3f93
commit
9613a850b6
|
@ -369,6 +369,12 @@ def PDL_OperationOp : PDL_Op<"operation", [AttrSizedOperandSegments]> {
|
|||
/// Returns true if the operation type referenced supports result type
|
||||
/// inference.
|
||||
bool hasTypeInference();
|
||||
|
||||
/// Returns true if the operation type referenced might support result type
|
||||
/// inference, i.e. it supports type reference or is currently not
|
||||
/// registered in the context. Returns false if the root operation name
|
||||
/// has not been set.
|
||||
bool mightHaveTypeInference();
|
||||
}];
|
||||
let hasVerifier = 1;
|
||||
}
|
||||
|
|
|
@ -167,6 +167,17 @@ public:
|
|||
return impl->interfaceMap.contains(interfaceID);
|
||||
}
|
||||
|
||||
/// Returns true if the operation *might* have the provided interface. This
|
||||
/// means that either the operation is unregistered, or it was registered with
|
||||
/// the provide interface.
|
||||
template <typename T>
|
||||
bool mightHaveInterface() const {
|
||||
return mightHaveInterface(TypeID::get<T>());
|
||||
}
|
||||
bool mightHaveInterface(TypeID interfaceID) const {
|
||||
return !isRegistered() || hasInterface(interfaceID);
|
||||
}
|
||||
|
||||
/// Return the dialect this operation is registered to if the dialect is
|
||||
/// loaded in the context, or nullptr if the dialect isn't loaded.
|
||||
Dialect *getDialect() const {
|
||||
|
|
|
@ -198,6 +198,36 @@ static LogicalResult verifyResultTypesAreInferrable(OperationOp op,
|
|||
if (llvm::any_of(op.op().getUses(), canInferTypeFromUse))
|
||||
return success();
|
||||
|
||||
// Handle the case where the operation has no explicit result types.
|
||||
if (resultTypes.empty()) {
|
||||
// If we don't know the concrete operation, don't attempt any verification.
|
||||
// We can't make assumptions if we don't know the concrete operation.
|
||||
Optional<StringRef> rawOpName = op.name();
|
||||
if (!rawOpName)
|
||||
return success();
|
||||
Optional<RegisteredOperationName> opName =
|
||||
RegisteredOperationName::lookup(*rawOpName, op.getContext());
|
||||
if (!opName)
|
||||
return success();
|
||||
|
||||
// If no explicit result types were provided, check to see if the operation
|
||||
// expected at least one result. This doesn't cover all cases, but this
|
||||
// should cover many cases in which the user intended to infer the results
|
||||
// of an operation, but it isn't actually possible.
|
||||
bool expectedAtLeastOneResult =
|
||||
!opName->hasTrait<OpTrait::ZeroResult>() &&
|
||||
!opName->hasTrait<OpTrait::VariadicResults>();
|
||||
if (expectedAtLeastOneResult) {
|
||||
return op
|
||||
.emitOpError("must have inferable or constrained result types when "
|
||||
"nested within `pdl.rewrite`")
|
||||
.attachNote()
|
||||
.append("operation is created in a non-inferrable context, but '",
|
||||
*opName, "' does not implement InferTypeOpInterface");
|
||||
}
|
||||
return success();
|
||||
}
|
||||
|
||||
// Otherwise, make sure each of the types can be inferred.
|
||||
for (const auto &it : llvm::enumerate(resultTypes)) {
|
||||
Operation *resultTypeOp = it.value().getDefiningOp();
|
||||
|
@ -248,7 +278,7 @@ LogicalResult OperationOp::verify() {
|
|||
|
||||
// If the operation is within a rewrite body and doesn't have type inference,
|
||||
// ensure that the result types can be resolved.
|
||||
if (isWithinRewrite && !hasTypeInference()) {
|
||||
if (isWithinRewrite && !mightHaveTypeInference()) {
|
||||
if (failed(verifyResultTypesAreInferrable(*this, types())))
|
||||
return failure();
|
||||
}
|
||||
|
@ -257,12 +287,18 @@ LogicalResult OperationOp::verify() {
|
|||
}
|
||||
|
||||
bool OperationOp::hasTypeInference() {
|
||||
Optional<StringRef> opName = name();
|
||||
if (!opName)
|
||||
return false;
|
||||
if (Optional<StringRef> rawOpName = name()) {
|
||||
OperationName opName(*rawOpName, getContext());
|
||||
return opName.hasInterface<InferTypeOpInterface>();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
if (auto rInfo = RegisteredOperationName::lookup(*opName, getContext()))
|
||||
return rInfo->hasInterface<InferTypeOpInterface>();
|
||||
bool OperationOp::mightHaveTypeInference() {
|
||||
if (Optional<StringRef> rawOpName = name()) {
|
||||
OperationName opName(*rawOpName, getContext());
|
||||
return opName.mightHaveInterface<InferTypeOpInterface>();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
|
|
@ -136,7 +136,21 @@ pdl.pattern : benefit(1) {
|
|||
|
||||
// expected-error@below {{op must have inferable or constrained result types when nested within `pdl.rewrite`}}
|
||||
// expected-note@below {{result type #0 was not constrained}}
|
||||
%newOp = operation "foo.op" -> (%type : !pdl.type)
|
||||
%newOp = operation "builtin.unrealized_conversion_cast" -> (%type : !pdl.type)
|
||||
}
|
||||
}
|
||||
|
||||
// -----
|
||||
|
||||
// Unused operation only necessary to ensure the func dialect is loaded.
|
||||
func.func private @unusedOpToLoadFuncDialect()
|
||||
|
||||
pdl.pattern : benefit(1) {
|
||||
%op = operation "foo.op"
|
||||
rewrite %op {
|
||||
// expected-error@below {{op must have inferable or constrained result types when nested within `pdl.rewrite`}}
|
||||
// expected-note@below {{operation is created in a non-inferrable context, but 'func.constant' does not implement InferTypeOpInterface}}
|
||||
%newOp = operation "func.constant"
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue