mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-16 13:16:49 +00:00

Adding support for the extra SUMMARY line that is output by most compilers. This also adds the ability for end-users to specify their own handlers for reporting these errors (see the test).
96 lines
3.0 KiB
C++
96 lines
3.0 KiB
C++
//===--- rtsan_diagnostics.cpp - Realtime Sanitizer -------------*- C++ -*-===//
|
|
//
|
|
// 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 "rtsan/rtsan_diagnostics.h"
|
|
|
|
#include "sanitizer_common/sanitizer_flags.h"
|
|
#include "sanitizer_common/sanitizer_report_decorator.h"
|
|
#include "sanitizer_common/sanitizer_stacktrace.h"
|
|
|
|
using namespace __sanitizer;
|
|
using namespace __rtsan;
|
|
|
|
// We must define our own implementation of this method for our runtime.
|
|
// This one is just copied from UBSan.
|
|
namespace __sanitizer {
|
|
void BufferedStackTrace::UnwindImpl(uptr pc, uptr bp, void *context,
|
|
bool request_fast, u32 max_depth) {
|
|
uptr top = 0;
|
|
uptr bottom = 0;
|
|
GetThreadStackTopAndBottom(false, &top, &bottom);
|
|
bool fast = StackTrace::WillUseFastUnwind(request_fast);
|
|
Unwind(max_depth, pc, bp, context, top, bottom, fast);
|
|
}
|
|
} // namespace __sanitizer
|
|
|
|
namespace {
|
|
class Decorator : public SanitizerCommonDecorator {
|
|
public:
|
|
Decorator() : SanitizerCommonDecorator() {}
|
|
const char *FunctionName() const { return Green(); }
|
|
const char *Reason() const { return Blue(); }
|
|
};
|
|
} // namespace
|
|
|
|
static const char *GetErrorTypeStr(const DiagnosticsInfo &info) {
|
|
switch (info.type) {
|
|
case DiagnosticsInfoType::InterceptedCall:
|
|
return "unsafe-library-call";
|
|
case DiagnosticsInfoType::BlockingCall:
|
|
return "blocking-call";
|
|
}
|
|
CHECK(false);
|
|
return "(unknown error)";
|
|
}
|
|
|
|
static void PrintError(const Decorator &decorator,
|
|
const DiagnosticsInfo &info) {
|
|
|
|
Printf("%s", decorator.Error());
|
|
Report("ERROR: RealtimeSanitizer: %s\n", GetErrorTypeStr(info));
|
|
}
|
|
|
|
static void PrintReason(const Decorator &decorator,
|
|
const DiagnosticsInfo &info) {
|
|
Printf("%s", decorator.Reason());
|
|
|
|
switch (info.type) {
|
|
case DiagnosticsInfoType::InterceptedCall: {
|
|
Printf("Intercepted call to real-time unsafe function "
|
|
"`%s%s%s` in real-time context!",
|
|
decorator.FunctionName(), info.func_name, decorator.Reason());
|
|
break;
|
|
}
|
|
case DiagnosticsInfoType::BlockingCall: {
|
|
Printf("Call to blocking function "
|
|
"`%s%s%s` in real-time context!",
|
|
decorator.FunctionName(), info.func_name, decorator.Reason());
|
|
break;
|
|
}
|
|
}
|
|
|
|
Printf("\n");
|
|
}
|
|
|
|
void __rtsan::PrintDiagnostics(const DiagnosticsInfo &info) {
|
|
ScopedErrorReportLock::CheckLocked();
|
|
|
|
Decorator d;
|
|
PrintError(d, info);
|
|
PrintReason(d, info);
|
|
Printf("%s", d.Default());
|
|
}
|
|
|
|
void __rtsan::PrintErrorSummary(const DiagnosticsInfo &info,
|
|
const BufferedStackTrace &stack) {
|
|
ScopedErrorReportLock::CheckLocked();
|
|
ReportErrorSummary(GetErrorTypeStr(info), &stack);
|
|
}
|