2020-02-09 07:01:34 +08:00
|
|
|
//===- LocationSnapshot.cpp - Location Snapshot Utilities -----------------===//
|
|
|
|
//
|
|
|
|
// 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/Transforms/LocationSnapshot.h"
|
2020-04-08 04:58:12 +08:00
|
|
|
#include "PassDetail.h"
|
2020-02-09 07:01:34 +08:00
|
|
|
#include "mlir/IR/AsmState.h"
|
|
|
|
#include "mlir/IR/Builders.h"
|
|
|
|
#include "mlir/Support/FileUtilities.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/ToolOutputFile.h"
|
|
|
|
|
|
|
|
using namespace mlir;
|
|
|
|
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given stream, and using the printed locations within that stream.
|
|
|
|
/// If a 'tag' is non-empty, the generated locations are represented as a
|
|
|
|
/// NameLoc with the given tag as the name, and then fused with the existing
|
|
|
|
/// locations. Otherwise, the existing locations are replaced.
|
|
|
|
static void generateLocationsFromIR(raw_ostream &os, StringRef fileName,
|
|
|
|
Operation *op, OpPrintingFlags flags,
|
|
|
|
StringRef tag) {
|
|
|
|
// Print the IR to the stream, and collect the raw line+column information.
|
|
|
|
AsmState::LocationMap opToLineCol;
|
|
|
|
AsmState state(op, &opToLineCol);
|
|
|
|
op->print(os, state, flags);
|
|
|
|
|
|
|
|
Builder builder(op->getContext());
|
|
|
|
Optional<Identifier> tagIdentifier;
|
|
|
|
if (!tag.empty())
|
|
|
|
tagIdentifier = builder.getIdentifier(tag);
|
|
|
|
|
|
|
|
// Walk and generate new locations for each of the operations.
|
|
|
|
Identifier file = builder.getIdentifier(fileName);
|
|
|
|
op->walk([&](Operation *opIt) {
|
|
|
|
// Check to see if this operation has a mapped location. Some operations may
|
|
|
|
// be elided from the printed form, e.g. the body terminators of some region
|
|
|
|
// operations.
|
|
|
|
auto it = opToLineCol.find(opIt);
|
|
|
|
if (it == opToLineCol.end())
|
|
|
|
return;
|
|
|
|
const std::pair<unsigned, unsigned> &lineCol = it->second;
|
|
|
|
auto newLoc =
|
|
|
|
builder.getFileLineColLoc(file, lineCol.first, lineCol.second);
|
|
|
|
|
|
|
|
// If we don't have a tag, set the location directly
|
|
|
|
if (!tagIdentifier) {
|
|
|
|
opIt->setLoc(newLoc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, build a fused location with the existing op loc.
|
|
|
|
opIt->setLoc(builder.getFusedLoc(
|
|
|
|
{opIt->getLoc(), NameLoc::get(*tagIdentifier, newLoc)}));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given file, and using the printed locations within that file. If
|
|
|
|
/// `filename` is empty, a temporary file is generated instead.
|
|
|
|
static LogicalResult generateLocationsFromIR(StringRef fileName, Operation *op,
|
|
|
|
OpPrintingFlags flags,
|
|
|
|
StringRef tag) {
|
|
|
|
// If a filename wasn't provided, then generate one.
|
|
|
|
SmallString<32> filepath(fileName);
|
|
|
|
if (filepath.empty()) {
|
|
|
|
if (std::error_code error = llvm::sys::fs::createTemporaryFile(
|
|
|
|
"mlir_snapshot", "tmp.mlir", filepath)) {
|
|
|
|
return op->emitError()
|
|
|
|
<< "failed to generate temporary file for location snapshot: "
|
|
|
|
<< error.message();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open the output file for emission.
|
|
|
|
std::string error;
|
|
|
|
std::unique_ptr<llvm::ToolOutputFile> outputFile =
|
|
|
|
openOutputFile(filepath, &error);
|
|
|
|
if (!outputFile)
|
|
|
|
return op->emitError() << error;
|
|
|
|
|
|
|
|
// Generate the intermediate locations.
|
|
|
|
generateLocationsFromIR(outputFile->os(), filepath, op, flags, tag);
|
|
|
|
outputFile->keep();
|
|
|
|
return success();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given stream, and using the printed locations within that stream.
|
|
|
|
/// The generated locations replace the current operation locations.
|
|
|
|
void mlir::generateLocationsFromIR(raw_ostream &os, StringRef fileName,
|
|
|
|
Operation *op, OpPrintingFlags flags) {
|
|
|
|
::generateLocationsFromIR(os, fileName, op, flags, /*tag=*/StringRef());
|
|
|
|
}
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given file, and using the printed locations within that file. If
|
|
|
|
/// `filename` is empty, a temporary file is generated instead.
|
|
|
|
LogicalResult mlir::generateLocationsFromIR(StringRef fileName, Operation *op,
|
|
|
|
OpPrintingFlags flags) {
|
|
|
|
return ::generateLocationsFromIR(fileName, op, flags, /*tag=*/StringRef());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given stream, and using the printed locations within that stream.
|
|
|
|
/// The generated locations are represented as a NameLoc with the given tag as
|
|
|
|
/// the name, and then fused with the existing locations.
|
|
|
|
void mlir::generateLocationsFromIR(raw_ostream &os, StringRef fileName,
|
|
|
|
StringRef tag, Operation *op,
|
|
|
|
OpPrintingFlags flags) {
|
|
|
|
::generateLocationsFromIR(os, fileName, op, flags, tag);
|
|
|
|
}
|
|
|
|
/// This function generates new locations from the given IR by snapshotting the
|
|
|
|
/// IR to the given file, and using the printed locations within that file. If
|
|
|
|
/// `filename` is empty, a temporary file is generated instead.
|
|
|
|
LogicalResult mlir::generateLocationsFromIR(StringRef fileName, StringRef tag,
|
|
|
|
Operation *op,
|
|
|
|
OpPrintingFlags flags) {
|
|
|
|
return ::generateLocationsFromIR(fileName, op, flags, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
2020-04-08 04:56:16 +08:00
|
|
|
struct LocationSnapshotPass
|
2020-04-08 04:58:12 +08:00
|
|
|
: public LocationSnapshotBase<LocationSnapshotPass> {
|
2020-02-09 07:01:34 +08:00
|
|
|
LocationSnapshotPass() = default;
|
|
|
|
LocationSnapshotPass(OpPrintingFlags flags, StringRef fileName, StringRef tag)
|
|
|
|
: flags(flags) {
|
|
|
|
this->fileName = fileName.str();
|
|
|
|
this->tag = tag.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
void runOnOperation() override {
|
|
|
|
Operation *op = getOperation();
|
|
|
|
if (failed(generateLocationsFromIR(fileName, op, OpPrintingFlags(), tag)))
|
|
|
|
return signalPassFailure();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The printing flags to use when creating the snapshot.
|
|
|
|
OpPrintingFlags flags;
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
std::unique_ptr<Pass> mlir::createLocationSnapshotPass(OpPrintingFlags flags,
|
|
|
|
StringRef fileName,
|
|
|
|
StringRef tag) {
|
|
|
|
return std::make_unique<LocationSnapshotPass>(flags, fileName, tag);
|
|
|
|
}
|
2020-04-01 16:48:34 +08:00
|
|
|
std::unique_ptr<Pass> mlir::createLocationSnapshotPass() {
|
|
|
|
return std::make_unique<LocationSnapshotPass>();
|
|
|
|
}
|