1 //===- ErrorHandler.cpp ---------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "lld/Common/ErrorHandler.h"
11 #include "llvm/Support/Parallel.h"
13 #include "lld/Common/CommonLinkerContext.h"
14 #include "llvm/ADT/Twine.h"
15 #include "llvm/IR/DiagnosticInfo.h"
16 #include "llvm/IR/DiagnosticPrinter.h"
17 #include "llvm/Support/CrashRecoveryContext.h"
18 #include "llvm/Support/ManagedStatic.h"
19 #include "llvm/Support/Process.h"
20 #include "llvm/Support/Program.h"
21 #include "llvm/Support/raw_ostream.h"
27 static StringRef
getSeparator(const Twine
&msg
) {
28 if (StringRef(msg
.str()).contains('\n'))
33 ErrorHandler::~ErrorHandler() {
38 void ErrorHandler::initialize(llvm::raw_ostream
&stdoutOS
,
39 llvm::raw_ostream
&stderrOS
, bool exitEarly
,
41 this->stdoutOS
= &stdoutOS
;
42 this->stderrOS
= &stderrOS
;
43 stderrOS
.enable_colors(stderrOS
.has_colors());
44 this->exitEarly
= exitEarly
;
45 this->disableOutput
= disableOutput
;
48 void ErrorHandler::flushStreams() {
49 std::lock_guard
<std::mutex
> lock(mu
);
54 ErrorHandler
&lld::errorHandler() { return context().e
; }
56 void lld::error(const Twine
&msg
) { errorHandler().error(msg
); }
57 void lld::error(const Twine
&msg
, ErrorTag tag
, ArrayRef
<StringRef
> args
) {
58 errorHandler().error(msg
, tag
, args
);
60 void lld::fatal(const Twine
&msg
) { errorHandler().fatal(msg
); }
61 void lld::log(const Twine
&msg
) { errorHandler().log(msg
); }
62 void lld::message(const Twine
&msg
, llvm::raw_ostream
&s
) {
63 errorHandler().message(msg
, s
);
65 void lld::warn(const Twine
&msg
) { errorHandler().warn(msg
); }
66 uint64_t lld::errorCount() { return errorHandler().errorCount
; }
68 raw_ostream
&lld::outs() {
69 ErrorHandler
&e
= errorHandler();
73 raw_ostream
&lld::errs() {
74 ErrorHandler
&e
= errorHandler();
78 raw_ostream
&ErrorHandler::outs() {
81 return stdoutOS
? *stdoutOS
: llvm::outs();
84 raw_ostream
&ErrorHandler::errs() {
87 return stderrOS
? *stderrOS
: llvm::errs();
90 void lld::exitLld(int val
) {
92 ErrorHandler
&e
= errorHandler();
93 // Delete any temporary file, while keeping the memory mapping open.
95 e
.outputBuffer
->discard();
98 // Re-throw a possible signal or exception once/if it was caught by
100 CrashRecoveryContext::throwIfCrash(val
);
102 // Dealloc/destroy ManagedStatic variables before calling _exit().
103 // In an LTO build, allows us to get the output of -time-passes.
104 // Ensures that the thread pool for the parallel algorithms is stopped to
105 // avoid intermittent crashes on Windows when exiting.
106 if (!CrashRecoveryContext::GetCurrent())
110 lld::errorHandler().flushStreams();
112 // When running inside safeLldMain(), restore the control flow back to the
113 // CrashRecoveryContext. Otherwise simply use _exit(), meanning no cleanup,
114 // since we want to avoid further crashes on shutdown.
115 llvm::sys::Process::Exit(val
, /*NoCleanup=*/true);
118 void lld::diagnosticHandler(const DiagnosticInfo
&di
) {
120 raw_svector_ostream
os(s
);
121 DiagnosticPrinterRawOStream
dp(os
);
123 // For an inline asm diagnostic, prepend the module name to get something like
124 // "$module <inline asm>:1:5: ".
125 if (auto *dism
= dyn_cast
<DiagnosticInfoSrcMgr
>(&di
))
126 if (dism
->isInlineAsmDiag())
127 os
<< dism
->getModuleName() << ' ';
130 switch (di
.getSeverity()) {
144 void lld::checkError(Error e
) {
145 handleAllErrors(std::move(e
),
146 [&](ErrorInfoBase
&eib
) { error(eib
.message()); });
149 // This is for --vs-diagnostics.
151 // Normally, lld's error message starts with argv[0]. Therefore, it usually
154 // ld.lld: error: ...
156 // This error message style is unfortunately unfriendly to Visual Studio
157 // IDE. VS interprets the first word of the first line as an error location
158 // and make it clickable, thus "ld.lld" in the above message would become a
159 // clickable text. When you click it, VS opens "ld.lld" executable file with
162 // As a workaround, we print out an error location instead of "ld.lld" if
163 // lld is running in VS diagnostics mode. As a result, error message will
166 // src/foo.c(35): error: ...
168 // This function returns an error location string. An error location is
169 // extracted from an error message using regexps.
170 std::string
ErrorHandler::getLocation(const Twine
&msg
) {
172 return std::string(logName
);
174 static std::regex regexes
[] = {
176 R
"(^undefined (?:\S+ )?symbol:.*\n)"
177 R
"(>>> referenced by .+\((\S+):(\d+)\))"),
179 R
"(^undefined (?:\S+ )?symbol:.*\n>>> referenced by (\S+):(\d+))"),
180 std::regex(R
"(^undefined symbol:.*\n>>> referenced by (.*):)"),
182 R
"(^duplicate symbol: .*\n>>> defined in (\S+)\n>>> defined in.*)"),
184 R
"(^duplicate symbol: .*\n>>> defined at .+\((\S+):(\d+)\))"),
185 std::regex(R
"(^duplicate symbol: .*\n>>> defined at (\S+):(\d+))"),
187 R
"(.*\n>>> defined in .*\n>>> referenced by .+\((\S+):(\d+)\))"),
188 std::regex(R
"(.*\n>>> defined in .*\n>>> referenced by (\S+):(\d+))"),
189 std::regex(R
"((\S+):(\d+): unclosed quote)"),
192 std::string str
= msg
.str();
193 for (std::regex
&re
: regexes
) {
195 if (!std::regex_search(str
, m
, re
))
198 assert(m
.size() == 2 || m
.size() == 3);
201 return m
.str(1) + "(" + m
.str(2) + ")";
204 return std::string(logName
);
207 void ErrorHandler::reportDiagnostic(StringRef location
, Colors c
,
208 StringRef diagKind
, const Twine
&msg
) {
209 SmallString
<256> buf
;
210 raw_svector_ostream
os(buf
);
211 os
<< sep
<< location
<< ": ";
212 if (!diagKind
.empty()) {
213 if (lld::errs().colors_enabled()) {
214 os
.enable_colors(true);
215 os
<< c
<< diagKind
<< ": " << Colors::RESET
;
217 os
<< diagKind
<< ": ";
224 void ErrorHandler::log(const Twine
&msg
) {
225 if (!verbose
|| disableOutput
)
227 std::lock_guard
<std::mutex
> lock(mu
);
228 reportDiagnostic(logName
, Colors::RESET
, "", msg
);
231 void ErrorHandler::message(const Twine
&msg
, llvm::raw_ostream
&s
) {
234 std::lock_guard
<std::mutex
> lock(mu
);
239 void ErrorHandler::warn(const Twine
&msg
) {
245 if (suppressWarnings
)
248 std::lock_guard
<std::mutex
> lock(mu
);
249 reportDiagnostic(getLocation(msg
), Colors::MAGENTA
, "warning", msg
);
250 sep
= getSeparator(msg
);
253 void ErrorHandler::error(const Twine
&msg
) {
254 // If Visual Studio-style error message mode is enabled,
255 // this particular error is printed out as two errors.
257 static std::regex
re(R
"(^(duplicate symbol: .*))"
258 R
"((\n>>> defined at \S+:\d+.*\n>>>.*))"
259 R
"((\n>>> defined at \S+:\d+.*\n>>>.*))");
260 std::string str
= msg
.str();
263 if (std::regex_match(str
, m
, re
)) {
264 error(m
.str(1) + m
.str(2));
265 error(m
.str(1) + m
.str(3));
272 std::lock_guard
<std::mutex
> lock(mu
);
274 if (errorLimit
== 0 || errorCount
< errorLimit
) {
275 reportDiagnostic(getLocation(msg
), Colors::RED
, "error", msg
);
276 } else if (errorCount
== errorLimit
) {
277 reportDiagnostic(logName
, Colors::RED
, "error", errorLimitExceededMsg
);
281 sep
= getSeparator(msg
);
289 void ErrorHandler::error(const Twine
&msg
, ErrorTag tag
,
290 ArrayRef
<StringRef
> args
) {
291 if (errorHandlingScript
.empty()) {
295 SmallVector
<StringRef
, 4> scriptArgs
;
296 scriptArgs
.push_back(errorHandlingScript
);
298 case ErrorTag::LibNotFound
:
299 scriptArgs
.push_back("missing-lib");
301 case ErrorTag::SymbolNotFound
:
302 scriptArgs
.push_back("undefined-symbol");
305 scriptArgs
.insert(scriptArgs
.end(), args
.begin(), args
.end());
306 int res
= llvm::sys::ExecuteAndWait(errorHandlingScript
, scriptArgs
);
310 // Temporarily disable error limit to make sure the two calls to error(...)
311 // only count as one.
312 uint64_t currentErrorLimit
= errorLimit
;
315 errorLimit
= currentErrorLimit
;
320 error("error handling script '" + errorHandlingScript
+
321 "' failed to execute");
324 error("error handling script '" + errorHandlingScript
+
325 "' crashed or timeout");
328 error("error handling script '" + errorHandlingScript
+
329 "' exited with code " + Twine(res
));
334 void ErrorHandler::fatal(const Twine
&msg
) {