diff --git a/mlir/include/mlir/IR/BuiltinLocationAttributes.td b/mlir/include/mlir/IR/BuiltinLocationAttributes.td --- a/mlir/include/mlir/IR/BuiltinLocationAttributes.td +++ b/mlir/include/mlir/IR/BuiltinLocationAttributes.td @@ -202,12 +202,12 @@ /// Returns an instance of opaque location which contains a given pointer to /// an object. The corresponding MLIR location is set to UnknownLoc. template - static Location get(T underlyingLocation, MLIRContext *context); + static OpaqueLoc get(T underlyingLocation, MLIRContext *context); /// Returns an instance of opaque location which contains a given pointer to /// an object and an additional MLIR location. template - static Location get(T underlyingLocation, Location fallbackLocation) { + static OpaqueLoc get(T underlyingLocation, Location fallbackLocation) { return get(reinterpret_cast(underlyingLocation), TypeID::get(), fallbackLocation); } diff --git a/mlir/lib/Parser/Parser.cpp b/mlir/lib/Parser/Parser.cpp --- a/mlir/lib/Parser/Parser.cpp +++ b/mlir/lib/Parser/Parser.cpp @@ -476,10 +476,14 @@ /// their first reference, to allow checking for use of undefined values. DenseMap forwardRefPlaceholders; - /// A set of operations whose locations reference aliases that have yet to - /// be resolved. - SmallVector, 8> - opsAndArgumentsWithDeferredLocs; + /// Deffered locations: when parsing `loc(#loc42)` we add an entry to this + /// map. After parsing the definition `#loc42 = ...` we'll patch back users + /// of this location. + struct DeferredLocInfo { + llvm::SMLoc loc; + std::string identifier; + }; + std::vector> allocatedDeferredLocs; /// The builder used when creating parsed operation instances. OpBuilder opBuilder; @@ -537,23 +541,40 @@ // Resolve the locations of any deferred operations. auto &attributeAliases = state.symbols.attributeAliasDefinitions; - for (std::pair &it : opsAndArgumentsWithDeferredLocs) { - llvm::SMLoc tokLoc = it.second.getLoc(); - StringRef identifier = it.second.getSpelling().drop_front(); - Attribute attr = attributeAliases.lookup(identifier); - if (!attr) - return emitError(tokLoc) << "operation location alias was never defined"; - - LocationAttr locAttr = attr.dyn_cast(); - if (!locAttr) - return emitError(tokLoc) - << "expected location, but found '" << attr << "'"; - auto opOrArgument = it.first; - if (auto *op = opOrArgument.dyn_cast()) - op->setLoc(locAttr); - else - opOrArgument.get().setLoc(locAttr); - } + auto locID = TypeID::get(); + auto resolveLocation = [&](auto &opOrArgument) -> LogicalResult { + auto fwdLoc = opOrArgument.getLoc().template dyn_cast(); + if (!fwdLoc || fwdLoc.getUnderlyingTypeID() != locID) + return success(); + auto *locInfo = reinterpret_cast( + fwdLoc.getUnderlyingLocation()); + Attribute attr = attributeAliases.lookup(locInfo->identifier); + if (!attr) { + emitError(locInfo->loc) << "operation location alias was never defined"; + return failure(); + } + auto locAttr = attr.dyn_cast(); + if (!locAttr) { + emitError(locInfo->loc) + << "expected location, but found '" << attr << "'"; + return failure(); + } + opOrArgument.setLoc(locAttr); + return success(); + }; + + auto walkRes = topLevelOp->walk([&](Operation *op) { + if (failed(resolveLocation(*op))) + return WalkResult::interrupt(); + for (Region ®ion : op->getRegions()) + for (Block &block : region.getBlocks()) + for (BlockArgument arg : block.getArguments()) + if (failed(resolveLocation(arg))) + return WalkResult::interrupt(); + return WalkResult::advance(); + }); + if (walkRes.wasInterrupted()) + return failure(); // Pop the top level name scope. if (failed(popSSANameScope())) @@ -1731,7 +1752,11 @@ << "expected location, but found '" << attr << "'"; } else { // Otherwise, remember this operation and resolve its location later. - opsAndArgumentsWithDeferredLocs.emplace_back(opOrArgument, tok); + // In the meantime, use a special OpaqueLoc as a marker. + allocatedDeferredLocs.emplace_back( + new DeferredLocInfo{tok.getLoc(), identifier.str()}); + directLoc = + OpaqueLoc::get(allocatedDeferredLocs.back().get(), getContext()); } // Otherwise, we parse the location directly.