15 #include "llvm/ADT/MapVector.h"
16 #include "llvm/ADT/SmallString.h"
17 #include "llvm/ADT/StringMap.h"
18 #include "llvm/ADT/TypeSwitch.h"
19 #include "llvm/Support/Mutex.h"
20 #include "llvm/Support/PrettyStackTrace.h"
21 #include "llvm/Support/Regex.h"
22 #include "llvm/Support/Signals.h"
23 #include "llvm/Support/SourceMgr.h"
24 #include "llvm/Support/raw_ostream.h"
37 opaqueVal(reinterpret_cast<intptr_t>(attr.getAsOpaquePointer())) {}
42 opaqueVal(reinterpret_cast<intptr_t>(val.getAsOpaquePointer())) {}
48 reinterpret_cast<const void *
>(
opaqueVal));
88 std::vector<std::unique_ptr<
char[]>> &strings) {
91 auto strRef = val.toStringRef(data);
95 strings.push_back(std::unique_ptr<
char[]>(
new char[strRef.size()]));
96 memcpy(&strings.back()[0], strRef.data(), strRef.size());
98 return StringRef(&strings.back()[0], strRef.size());
143 llvm::raw_string_ostream os(
str);
147 if (
str.find(
'\n') != std::string::npos)
155 llvm::raw_string_ostream os(
str);
169 llvm::raw_string_ostream os(
str);
180 "cannot attach a note to a note");
189 return *notes.back();
193 Diagnostic::operator LogicalResult()
const {
return failure(); }
201 InFlightDiagnostic::operator LogicalResult()
const {
202 return failure(isActive());
248 llvm::sys::SmartScopedLock<true> lock(mutex);
253 for (
auto &handlerIt : llvm::reverse(handlers))
254 if (succeeded(handlerIt.second(
diag)))
261 auto &os = llvm::errs();
262 if (!llvm::isa<UnknownLoc>(
diag.getLocation()))
263 os <<
diag.getLocation() <<
": ";
282 llvm::sys::SmartScopedLock<true> lock(
impl->mutex);
283 auto uniqueID =
impl->uniqueHandlerId++;
284 impl->handlers.insert({uniqueID, std::move(handler)});
290 llvm::sys::SmartScopedLock<true> lock(
impl->mutex);
291 impl->handlers.erase(handlerID);
298 "notes should not be emitted directly");
309 auto diag = diagEngine.emit(location, severity);
310 if (!message.isTriviallyEmpty())
317 llvm::raw_string_ostream stream(bt);
318 llvm::sys::PrintStackTrace(stream);
321 diag.attachNote() <<
"diagnostic emitted with trace:\n" << bt;
355 ctx->getDiagEngine().eraseHandler(handlerID);
367 StringRef filename) {
369 auto bufferIt = filenameToBufId.find(filename);
370 if (bufferIt != filenameToBufId.end())
371 return bufferIt->second;
374 for (
unsigned i = 1, e = mgr.getNumBuffers() + 1; i != e; ++i) {
375 auto *buf = mgr.getMemoryBuffer(i);
376 if (buf->getBufferIdentifier() == filename)
377 return filenameToBufId[filename] = i;
381 auto bufferOrErr = llvm::MemoryBuffer::getFile(filename);
384 unsigned id = mgr.AddNewSourceBuffer(std::move(*bufferOrErr), SMLoc());
385 filenameToBufId[filename] = id;
397 if (isa<NameLoc>(loc))
399 if (
auto callLoc = dyn_cast<CallSiteLoc>(loc))
401 if (isa<FusedLoc>(loc)) {
402 for (
auto subLoc : cast<FusedLoc>(loc).getLocations()) {
416 return llvm::SourceMgr::DK_Note;
418 return llvm::SourceMgr::DK_Warning;
420 return llvm::SourceMgr::DK_Error;
422 return llvm::SourceMgr::DK_Remark;
424 llvm_unreachable(
"Unknown DiagnosticSeverity");
428 llvm::SourceMgr &mgr,
MLIRContext *ctx, raw_ostream &os,
431 shouldShowLocFn(std::move(shouldShowLocFn)),
439 std::move(shouldShowLocFn)) {}
445 bool displaySourceLine) {
452 llvm::raw_string_ostream strOS(str);
453 if (!llvm::isa<UnknownLoc>(loc))
454 strOS << loc <<
": ";
461 if (displaySourceLine) {
462 auto smloc = convertLocToSMLoc(fileLoc);
471 llvm::raw_string_ostream locOS(locStr);
472 locOS << fileLoc.getFilename().getValue() <<
":" << fileLoc.getLine() <<
":"
473 << fileLoc.getColumn();
481 auto addLocToStack = [&](
Location loc, StringRef locContext) {
482 if (std::optional<Location> showableLoc = findLocToShow(loc))
483 locationStack.emplace_back(*showableLoc, locContext);
488 addLocToStack(loc, {});
494 loc = callLoc->getCaller();
495 for (
unsigned curDepth = 0; curDepth < callStackLimit; ++curDepth) {
496 addLocToStack(loc,
"called from");
498 loc = callLoc->getCaller();
505 if (locationStack.empty()) {
511 for (
auto &it : llvm::drop_begin(locationStack))
517 for (
auto ¬e :
diag.getNotes()) {
518 emitDiagnostic(note.getLocation(), note.str(), note.getSeverity(),
519 loc != note.getLocation());
520 loc = note.getLocation();
525 callStackLimit = limit;
529 const llvm::MemoryBuffer *
531 if (
unsigned id =
impl->getSourceMgrBufferIDForFile(
mgr, filename))
532 return mgr.getMemoryBuffer(
id);
536 std::optional<Location>
537 SourceMgrDiagnosticHandler::findLocToShow(
Location loc) {
545 .Case([&](CallSiteLoc callLoc) -> std::optional<Location> {
548 return findLocToShow(callLoc.getCallee());
550 .Case([&](
FileLineColLoc) -> std::optional<Location> {
return loc; })
551 .Case([&](
FusedLoc fusedLoc) -> std::optional<Location> {
554 for (
Location childLoc : fusedLoc.getLocations())
555 if (std::optional<Location> showableLoc = findLocToShow(childLoc))
559 .Case([&](NameLoc nameLoc) -> std::optional<Location> {
560 return findLocToShow(nameLoc.getChildLoc());
562 .Case([&](OpaqueLoc opaqueLoc) -> std::optional<Location> {
564 return findLocToShow(opaqueLoc.getFallbackLocation());
566 .Case([](UnknownLoc) -> std::optional<Location> {
574 SMLoc SourceMgrDiagnosticHandler::convertLocToSMLoc(
FileLineColLoc loc) {
599 LogicalResult
emitError(raw_ostream &os, llvm::SourceMgr &mgr,
601 SMRange range(
fileLoc, SMLoc::getFromPointer(
fileLoc.getPointer() +
603 mgr.PrintMessage(os,
fileLoc, llvm::SourceMgr::DK_Error, msg, range);
619 std::string regexStr;
620 llvm::raw_string_ostream regexOS(regexStr);
622 while (!strToProcess.empty()) {
624 size_t regexIt = strToProcess.find(
"{{");
625 if (regexIt == StringRef::npos) {
626 regexOS << llvm::Regex::escape(strToProcess);
629 regexOS << llvm::Regex::escape(strToProcess.take_front(regexIt));
630 strToProcess = strToProcess.drop_front(regexIt + 2);
633 size_t regexEndIt = strToProcess.find(
"}}");
634 if (regexEndIt == StringRef::npos)
635 return emitError(os, mgr,
"found start of regex with no end '}}'");
636 StringRef regexStr = strToProcess.take_front(regexEndIt);
639 std::string regexError;
640 if (!llvm::Regex(regexStr).isValid(regexError))
641 return emitError(os, mgr,
"invalid regex: " + regexError);
643 regexOS <<
'(' << regexStr <<
')';
644 strToProcess = strToProcess.drop_front(regexEndIt + 2);
671 std::optional<MutableArrayRef<ExpectedDiag>>
677 const llvm::MemoryBuffer *buf);
694 llvm::Regex(
"expected-(error|note|remark|warning)(-re)? "
695 "*(@([+-][0-9]+|above|below|unknown))? *{{(.*)}}$");
716 llvm_unreachable(
"Unknown DiagnosticSeverity");
719 std::optional<MutableArrayRef<ExpectedDiag>>
720 SourceMgrDiagnosticVerifierHandlerImpl::getExpectedDiags(StringRef bufName) {
721 auto expectedDiags = expectedDiagsPerFile.find(bufName);
722 if (expectedDiags != expectedDiagsPerFile.end())
728 SourceMgrDiagnosticVerifierHandlerImpl::computeExpectedDiags(
729 raw_ostream &os, llvm::SourceMgr &mgr,
const llvm::MemoryBuffer *buf) {
733 auto &expectedDiags = expectedDiagsPerFile[buf->getBufferIdentifier()];
736 unsigned lastNonDesignatorLine = 0;
743 buf->getBuffer().split(lines,
'\n');
744 for (
unsigned lineNo = 0, e = lines.size(); lineNo < e; ++lineNo) {
746 if (!expected.match(lines[lineNo].rtrim(), &matches)) {
748 if (!designatorsForNextLine.empty()) {
749 for (
unsigned diagIndex : designatorsForNextLine)
750 expectedDiags[diagIndex].lineNo = lineNo + 1;
751 designatorsForNextLine.clear();
753 lastNonDesignatorLine = lineNo;
758 SMLoc expectedStart = SMLoc::getFromPointer(matches[0].data());
761 if (matches[1] ==
"error")
763 else if (matches[1] ==
"warning")
765 else if (matches[1] ==
"remark")
768 assert(matches[1] ==
"note");
779 StringRef offsetMatch = matches[3];
780 if (!offsetMatch.empty()) {
781 offsetMatch = offsetMatch.drop_front(1);
784 if (offsetMatch[0] ==
'+' || offsetMatch[0] ==
'-') {
786 offsetMatch.drop_front().getAsInteger(0, offset);
788 if (offsetMatch.front() ==
'+')
792 }
else if (offsetMatch.consume_front(
"unknown")) {
795 expectedUnknownLocDiags.emplace_back(std::move(record));
797 }
else if (offsetMatch.consume_front(
"above")) {
800 record.
lineNo = lastNonDesignatorLine + 1;
804 assert(offsetMatch.consume_front(
"below"));
805 designatorsForNextLine.push_back(expectedDiags.size());
812 expectedDiags.emplace_back(std::move(record));
814 return expectedDiags;
823 for (
unsigned i = 0, e =
mgr.getNumBuffers(); i != e; ++i)
824 (void)
impl->computeExpectedDiags(out,
mgr,
mgr.getMemoryBuffer(i + 1));
846 err.emitError(
os,
mgr,
848 err.substring +
"\" was not produced");
850 for (
auto &expectedDiagsPair :
impl->expectedDiagsPerFile)
851 for (
auto &err : expectedDiagsPair.second)
852 checkExpectedDiags(err);
853 for (
auto &err :
impl->expectedUnknownLocDiags)
854 checkExpectedDiags(err);
855 impl->expectedDiagsPerFile.clear();
865 for (
auto ¬e :
diag.getNotes())
871 void SourceMgrDiagnosticVerifierHandler::process(
Diagnostic &
diag) {
872 return process(
diag.getLocation(),
diag.str(),
diag.getSeverity());
876 void SourceMgrDiagnosticVerifierHandler::process(
LocationAttr loc,
887 diags =
impl->computeExpectedDiags(
892 diags =
impl->expectedUnknownLocDiags;
900 for (
auto &e : diags) {
902 if (fileLoc && fileLoc.
getLine() != e.lineNo)
905 if (e.kind ==
kind) {
921 mgr.PrintMessage(
os, nearMiss->
fileLoc, llvm::SourceMgr::DK_Error,
923 "' diagnostic emitted when expecting a '" +
928 impl->status = failure();
954 uint64_t tid = llvm::get_threadid();
955 llvm::sys::SmartScopedLock<true> lock(
mutex);
996 uint64_t tid = llvm::get_threadid();
997 llvm::sys::SmartScopedLock<true> lock(
mutex);
1003 uint64_t tid = llvm::get_threadid();
1004 llvm::sys::SmartScopedLock<true> lock(
mutex);
1009 void print(raw_ostream &os)
const override {
1014 os <<
"In-Flight Diagnostics:\n";
1020 if (!llvm::isa<UnknownLoc>(
diag.getLocation()))
1021 os <<
diag.getLocation() <<
": ";
1022 switch (
diag.getSeverity()) {
1023 case DiagnosticSeverity::Error:
1026 case DiagnosticSeverity::Warning:
1029 case DiagnosticSeverity::Note:
1032 case DiagnosticSeverity::Remark:
1064 impl->setOrderIDForThread(orderID);
1070 impl->eraseOrderIDForThread();
static OpPrintingFlags adjustPrintingFlags(OpPrintingFlags flags, DiagnosticSeverity severity)
Adjusts operation printing flags used in diagnostics for the given severity level.
static InFlightDiagnostic emitDiag(Location location, DiagnosticSeverity severity, const Twine &message)
Helper function used to emit a diagnostic with an optionally empty twine message.
static StringRef getDiagKindStr(DiagnosticSeverity kind)
Given a diagnostic kind, return a human readable string for it.
static StringRef twineToStrRef(const Twine &val, std::vector< std::unique_ptr< char[]>> &strings)
Convert a Twine to a StringRef.
static std::optional< CallSiteLoc > getCallSiteLoc(Location loc)
Return a processable CallSiteLoc from the given location.
static llvm::SourceMgr::DiagKind getDiagKind(DiagnosticSeverity kind)
Given a diagnostic kind, returns the LLVM DiagKind.
union mlir::linalg::@1254::ArityGroupAndKind::Kind kind
static std::string diag(const llvm::Value &value)
static LogicalResult emit(SolverOp solver, const SMTEmissionOptions &options, mlir::raw_indented_ostream &stream)
Emit the SMT operations in the given 'solver' to the 'stream'.
Attributes are known-constant values of operations.
MLIRContext * getContext() const
Return the context this attribute belongs to.
static Attribute getFromOpaquePointer(const void *ptr)
Construct an attribute from the opaque pointer representation.
A variant type that holds a single argument for a diagnostic.
DiagnosticArgumentKind
Enum that represents the different kinds of diagnostic arguments supported.
StringRef getAsString() const
Returns this argument as a string.
double getAsDouble() const
Returns this argument as a double.
DiagnosticArgument(Attribute attr)
Note: The constructors below are only exposed due to problems accessing constructors from type traits...
Type getAsType() const
Returns this argument as a Type.
int64_t getAsInteger() const
Returns this argument as a signed integer.
DiagnosticArgumentKind getKind() const
Returns the kind of this argument.
Attribute getAsAttribute() const
Returns this argument as an Attribute.
void print(raw_ostream &os) const
Outputs this argument to a stream.
uint64_t getAsUnsigned() const
Returns this argument as an unsigned integer.
uint64_t HandlerID
A handle to a specific registered handler object.
InFlightDiagnostic emit(Location loc, DiagnosticSeverity severity)
Create a new inflight diagnostic with the given location and severity.
void eraseHandler(HandlerID id)
Erase the registered diagnostic handler with the given identifier.
llvm::unique_function< LogicalResult(Diagnostic &)> HandlerTy
The handler type for MLIR diagnostics.
HandlerID registerHandler(HandlerTy handler)
Register a new handler for diagnostics to the engine.
This class contains all of the information necessary to report a diagnostic to the DiagnosticEngine.
std::string str() const
Converts the diagnostic to a string.
Diagnostic & attachNote(std::optional< Location > noteLoc=std::nullopt)
Attaches a note to this diagnostic.
MutableArrayRef< DiagnosticArgument > getArguments()
Returns the current list of diagnostic arguments.
std::enable_if_t<!std::is_convertible< Arg, StringRef >::value &&std::is_constructible< DiagnosticArgument, Arg >::value, Diagnostic & > operator<<(Arg &&val)
Stream operator for inserting new diagnostic arguments.
void print(raw_ostream &os) const
Outputs this diagnostic to a stream.
Diagnostic & appendOp(Operation &op, const OpPrintingFlags &flags)
Append an operation with the given printing flags.
An instance of this location represents a tuple of file, line number, and column number.
StringAttr getFilename() const
unsigned getColumn() const
This class represents a diagnostic that is inflight and set to be reported.
void report()
Reports the diagnostic to the engine.
void abandon()
Abandons this diagnostic so that it will no longer be reported.
Location objects represent source locations information in MLIR.
T findInstanceOf()
Return an instance of the given location type if one is nested under the current location.
This class defines the main interface for locations in MLIR and acts as a non-nullable wrapper around...
MLIRContext is the top-level object for a collection of MLIR operations.
bool shouldPrintStackTraceOnDiagnostic()
Return true if we should attach the current stacktrace to diagnostics when emitted.
DiagnosticEngine & getDiagEngine()
Returns the diagnostic engine for this context.
Set of flags used to control the behavior of the various IR print methods (e.g.
OpPrintingFlags & elideLargeElementsAttrs(int64_t largeElementLimit=16)
Enables the elision of large elements attributes by printing a lexically valid but otherwise meaningl...
OpPrintingFlags & printGenericOpForm(bool enable=true)
Always print operations in the generic form.
OpPrintingFlags & useLocalScope(bool enable=true)
Use local scope when printing the operation.
StringRef getStringRef() const
Return the name of this operation. This always succeeds.
Operation is the basic unit of execution within MLIR.
void print(raw_ostream &os, const OpPrintingFlags &flags={})
void eraseOrderIDForThread()
Remove the order id for the current thread.
~ParallelDiagnosticHandler()
ParallelDiagnosticHandler(MLIRContext *ctx)
void setOrderIDForThread(size_t orderID)
Set the order id for the current thread.
This diagnostic handler is a simple RAII class that registers and erases a diagnostic handler on a gi...
~ScopedDiagnosticHandler()
void setHandler(FuncTy &&handler)
Set the handler to manage via RAII.
This class is a utility diagnostic handler for use with llvm::SourceMgr.
void setCallStackLimit(unsigned limit)
Set the maximum depth that a call stack will be printed. Defaults to 10.
void emitDiagnostic(Location loc, Twine message, DiagnosticSeverity kind, bool displaySourceLine=true)
Emit the given diagnostic information with the held source manager.
raw_ostream & os
The output stream to use when printing diagnostics.
SourceMgrDiagnosticHandler(llvm::SourceMgr &mgr, MLIRContext *ctx, raw_ostream &os, ShouldShowLocFn &&shouldShowLocFn={})
~SourceMgrDiagnosticHandler()
ShouldShowLocFn shouldShowLocFn
A functor used when determining if a location for a diagnostic should be shown.
const llvm::MemoryBuffer * getBufferForFile(StringRef filename)
Get a memory buffer for the given file, or nullptr if no file is available.
llvm::SourceMgr & mgr
The source manager that we are wrapping.
llvm::unique_function< bool(Location)> ShouldShowLocFn
This type represents a functor used to filter out locations when printing a diagnostic.
This class is a utility diagnostic handler for use with llvm::SourceMgr that verifies that emitted di...
~SourceMgrDiagnosticVerifierHandler()
SourceMgrDiagnosticVerifierHandler(llvm::SourceMgr &srcMgr, MLIRContext *ctx, raw_ostream &out, Level level=Level::All)
void registerInContext(MLIRContext *ctx)
Register this handler with the given context.
LogicalResult verify()
Returns the status of the handler and verifies that all expected diagnostics were emitted.
Instances of the Type class are uniqued, have an immutable identifier and an optional mutable compone...
static Type getFromOpaquePointer(const void *pointer)
This class represents an instance of an SSA value in the MLIR system, representing a computable value...
void print(raw_ostream &os) const
The OpAsmOpInterface, see OpAsmInterface.td for more details.
Include the generated interface declarations.
InFlightDiagnostic emitWarning(Location loc)
Utility method to emit a warning message using this location.
DiagnosticSeverity
Defines the different supported severity of a diagnostic.
InFlightDiagnostic emitError(Location loc)
Utility method to emit an error message using this location.
InFlightDiagnostic emitRemark(Location loc)
Utility method to emit a remark message using this location.
DiagnosticEngine::HandlerID uniqueHandlerId
This is a unique identifier counter for diagnostic handlers in the context.
llvm::sys::SmartMutex< true > mutex
A mutex to ensure that diagnostics emission is thread-safe.
void emit(Diagnostic &&diag)
Emit a diagnostic using the registered issue handle if present, or with the default behavior if not.
llvm::SmallMapVector< DiagnosticEngine::HandlerID, DiagnosticEngine::HandlerTy, 2 > handlers
These are the handlers used to report diagnostics.
This class represents an expected output diagnostic.
LogicalResult emitError(raw_ostream &os, llvm::SourceMgr &mgr, const Twine &msg)
Emit an error at the location referenced by this diagnostic.
unsigned lineNo
The line number the expected diagnostic should be on.
LogicalResult computeRegex(raw_ostream &os, llvm::SourceMgr &mgr)
Compute the regex matcher for this diagnostic, using the provided stream and manager to emit diagnost...
bool match(StringRef str) const
Returns true if this diagnostic matches the given string.
bool matched
A flag indicating if the expected diagnostic has been matched yet.
DiagnosticSeverity kind
The severity of the diagnosic expected.
StringRef substring
The substring that is expected to be within the diagnostic.
std::optional< llvm::Regex > substringRegex
An optional regex matcher, if the expected diagnostic sub-string was a regex string.
SMLoc fileLoc
The location of the expected diagnostic within the input file.
ExpectedDiag(DiagnosticSeverity kind, unsigned lineNo, SMLoc fileLoc, StringRef substring)
size_t id
The id for this diagnostic, this is used for ordering.
bool operator<(const ThreadDiagnostic &rhs) const
ThreadDiagnostic(size_t id, Diagnostic diag)
Diagnostic diag
The diagnostic.
void print(raw_ostream &os) const override
Dump the current diagnostics that were inflight.
MLIRContext * context
The context to emit the diagnostics to.
ParallelDiagnosticHandlerImpl(MLIRContext *ctx)
~ParallelDiagnosticHandlerImpl() override
void emitDiagnostics(llvm::function_ref< void(Diagnostic &)> emitFn) const
Utility method to emit any held diagnostics.
DiagnosticEngine::HandlerID handlerID
The unique id for the parallel handler.
DenseMap< uint64_t, size_t > threadToOrderID
A mapping between the thread id and the current order id.
void setOrderIDForThread(size_t orderID)
Set the order id for the current thread.
std::vector< ThreadDiagnostic > diagnostics
An unordered list of diagnostics that were emitted.
void eraseOrderIDForThread()
Remove the order id for the current thread.
llvm::sys::SmartMutex< true > mutex
A smart mutex to lock access to the internal state.
llvm::StringMap< unsigned > filenameToBufId
Mapping between file name and buffer ID's.
unsigned getSourceMgrBufferIDForFile(llvm::SourceMgr &mgr, StringRef filename)
Return the SrcManager buffer id for the specified file, or zero if none can be found.
llvm::StringMap< SmallVector< ExpectedDiag, 2 > > expectedDiagsPerFile
A list of expected diagnostics for each buffer of the source manager.
LogicalResult status
The current status of the verifier.
MutableArrayRef< ExpectedDiag > computeExpectedDiags(raw_ostream &os, llvm::SourceMgr &mgr, const llvm::MemoryBuffer *buf)
Computes the expected diagnostics for the given source buffer.
SourceMgrDiagnosticVerifierHandler::Level getVerifyLevel() const
llvm::Regex expected
Regex to match the expected diagnostics format.
SourceMgrDiagnosticVerifierHandler::Level level
Verification level.
SmallVector< ExpectedDiag, 2 > expectedUnknownLocDiags
A list of expected diagnostics with unknown locations.
SourceMgrDiagnosticVerifierHandlerImpl(SourceMgrDiagnosticVerifierHandler::Level level)
std::optional< MutableArrayRef< ExpectedDiag > > getExpectedDiags(StringRef bufName)
Returns the expected diagnostics for the given source file.