299 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			299 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
//===- ErrorHandler.cpp ---------------------------------------------------===//
 | 
						|
//
 | 
						|
// 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 "lld/Common/ErrorHandler.h"
 | 
						|
 | 
						|
#include "llvm/Support/Parallel.h"
 | 
						|
 | 
						|
#include "llvm/ADT/Twine.h"
 | 
						|
#include "llvm/IR/DiagnosticInfo.h"
 | 
						|
#include "llvm/IR/DiagnosticPrinter.h"
 | 
						|
#include "llvm/Support/CrashRecoveryContext.h"
 | 
						|
#include "llvm/Support/ManagedStatic.h"
 | 
						|
#include "llvm/Support/Process.h"
 | 
						|
#include "llvm/Support/Program.h"
 | 
						|
#include "llvm/Support/raw_ostream.h"
 | 
						|
#include <mutex>
 | 
						|
#include <regex>
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
using namespace lld;
 | 
						|
 | 
						|
// The functions defined in this file can be called from multiple threads,
 | 
						|
// but lld::outs() or lld::errs() are not thread-safe. We protect them using a
 | 
						|
// mutex.
 | 
						|
static std::mutex mu;
 | 
						|
 | 
						|
// We want to separate multi-line messages with a newline. `sep` is "\n"
 | 
						|
// if the last messages was multi-line. Otherwise "".
 | 
						|
static StringRef sep;
 | 
						|
 | 
						|
static StringRef getSeparator(const Twine &msg) {
 | 
						|
  if (StringRef(msg.str()).contains('\n'))
 | 
						|
    return "\n";
 | 
						|
  return "";
 | 
						|
}
 | 
						|
 | 
						|
raw_ostream *lld::stdoutOS;
 | 
						|
raw_ostream *lld::stderrOS;
 | 
						|
 | 
						|
ErrorHandler &lld::errorHandler() {
 | 
						|
  static ErrorHandler handler;
 | 
						|
  return handler;
 | 
						|
}
 | 
						|
 | 
						|
raw_ostream &lld::outs() {
 | 
						|
  if (errorHandler().disableOutput)
 | 
						|
    return llvm::nulls();
 | 
						|
  return stdoutOS ? *stdoutOS : llvm::outs();
 | 
						|
}
 | 
						|
 | 
						|
raw_ostream &lld::errs() {
 | 
						|
  if (errorHandler().disableOutput)
 | 
						|
    return llvm::nulls();
 | 
						|
  return stderrOS ? *stderrOS : llvm::errs();
 | 
						|
}
 | 
						|
 | 
						|
void lld::exitLld(int val) {
 | 
						|
  // Delete any temporary file, while keeping the memory mapping open.
 | 
						|
  if (errorHandler().outputBuffer)
 | 
						|
    errorHandler().outputBuffer->discard();
 | 
						|
 | 
						|
  // Re-throw a possible signal or exception once/if it was catched by
 | 
						|
  // safeLldMain().
 | 
						|
  CrashRecoveryContext::throwIfCrash(val);
 | 
						|
 | 
						|
  // Dealloc/destroy ManagedStatic variables before calling _exit().
 | 
						|
  // In an LTO build, allows us to get the output of -time-passes.
 | 
						|
  // Ensures that the thread pool for the parallel algorithms is stopped to
 | 
						|
  // avoid intermittent crashes on Windows when exiting.
 | 
						|
  if (!CrashRecoveryContext::GetCurrent())
 | 
						|
    llvm_shutdown();
 | 
						|
 | 
						|
  {
 | 
						|
    std::lock_guard<std::mutex> lock(mu);
 | 
						|
    lld::outs().flush();
 | 
						|
    lld::errs().flush();
 | 
						|
  }
 | 
						|
  // When running inside safeLldMain(), restore the control flow back to the
 | 
						|
  // CrashRecoveryContext. Otherwise simply use _exit(), meanning no cleanup,
 | 
						|
  // since we want to avoid further crashes on shutdown.
 | 
						|
  llvm::sys::Process::Exit(val, /*NoCleanup=*/true);
 | 
						|
}
 | 
						|
 | 
						|
void lld::diagnosticHandler(const DiagnosticInfo &di) {
 | 
						|
  SmallString<128> s;
 | 
						|
  raw_svector_ostream os(s);
 | 
						|
  DiagnosticPrinterRawOStream dp(os);
 | 
						|
  di.print(dp);
 | 
						|
  switch (di.getSeverity()) {
 | 
						|
  case DS_Error:
 | 
						|
    error(s);
 | 
						|
    break;
 | 
						|
  case DS_Warning:
 | 
						|
    warn(s);
 | 
						|
    break;
 | 
						|
  case DS_Remark:
 | 
						|
  case DS_Note:
 | 
						|
    message(s);
 | 
						|
    break;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void lld::checkError(Error e) {
 | 
						|
  handleAllErrors(std::move(e),
 | 
						|
                  [&](ErrorInfoBase &eib) { error(eib.message()); });
 | 
						|
}
 | 
						|
 | 
						|
// This is for --vs-diagnostics.
 | 
						|
//
 | 
						|
// Normally, lld's error message starts with argv[0]. Therefore, it usually
 | 
						|
// looks like this:
 | 
						|
//
 | 
						|
//   ld.lld: error: ...
 | 
						|
//
 | 
						|
// This error message style is unfortunately unfriendly to Visual Studio
 | 
						|
// IDE. VS interprets the first word of the first line as an error location
 | 
						|
// and make it clickable, thus "ld.lld" in the above message would become a
 | 
						|
// clickable text. When you click it, VS opens "ld.lld" executable file with
 | 
						|
// a binary editor.
 | 
						|
//
 | 
						|
// As a workaround, we print out an error location instead of "ld.lld" if
 | 
						|
// lld is running in VS diagnostics mode. As a result, error message will
 | 
						|
// look like this:
 | 
						|
//
 | 
						|
//   src/foo.c(35): error: ...
 | 
						|
//
 | 
						|
// This function returns an error location string. An error location is
 | 
						|
// extracted from an error message using regexps.
 | 
						|
std::string ErrorHandler::getLocation(const Twine &msg) {
 | 
						|
  if (!vsDiagnostics)
 | 
						|
    return std::string(logName);
 | 
						|
 | 
						|
  static std::regex regexes[] = {
 | 
						|
      std::regex(
 | 
						|
          R"(^undefined (?:\S+ )?symbol:.*\n)"
 | 
						|
          R"(>>> referenced by .+\((\S+):(\d+)\))"),
 | 
						|
      std::regex(
 | 
						|
          R"(^undefined (?:\S+ )?symbol:.*\n>>> referenced by (\S+):(\d+))"),
 | 
						|
      std::regex(R"(^undefined symbol:.*\n>>> referenced by (.*):)"),
 | 
						|
      std::regex(
 | 
						|
          R"(^duplicate symbol: .*\n>>> defined in (\S+)\n>>> defined in.*)"),
 | 
						|
      std::regex(
 | 
						|
          R"(^duplicate symbol: .*\n>>> defined at .+\((\S+):(\d+)\))"),
 | 
						|
      std::regex(R"(^duplicate symbol: .*\n>>> defined at (\S+):(\d+))"),
 | 
						|
      std::regex(
 | 
						|
          R"(.*\n>>> defined in .*\n>>> referenced by .+\((\S+):(\d+)\))"),
 | 
						|
      std::regex(R"(.*\n>>> defined in .*\n>>> referenced by (\S+):(\d+))"),
 | 
						|
      std::regex(R"((\S+):(\d+): unclosed quote)"),
 | 
						|
  };
 | 
						|
 | 
						|
  std::string str = msg.str();
 | 
						|
  for (std::regex &re : regexes) {
 | 
						|
    std::smatch m;
 | 
						|
    if (!std::regex_search(str, m, re))
 | 
						|
      continue;
 | 
						|
 | 
						|
    assert(m.size() == 2 || m.size() == 3);
 | 
						|
    if (m.size() == 2)
 | 
						|
      return m.str(1);
 | 
						|
    return m.str(1) + "(" + m.str(2) + ")";
 | 
						|
  }
 | 
						|
 | 
						|
  return std::string(logName);
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::reportDiagnostic(StringRef location, Colors c,
 | 
						|
                                    StringRef diagKind, const Twine &msg) {
 | 
						|
  SmallString<256> buf;
 | 
						|
  raw_svector_ostream os(buf);
 | 
						|
  os << sep << location << ": ";
 | 
						|
  if (!diagKind.empty()) {
 | 
						|
    if (lld::errs().colors_enabled()) {
 | 
						|
      os.enable_colors(true);
 | 
						|
      os << c << diagKind << ": " << Colors::RESET;
 | 
						|
    } else {
 | 
						|
      os << diagKind << ": ";
 | 
						|
    }
 | 
						|
  }
 | 
						|
  os << msg << '\n';
 | 
						|
  lld::errs() << buf;
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::log(const Twine &msg) {
 | 
						|
  if (!verbose || disableOutput)
 | 
						|
    return;
 | 
						|
  std::lock_guard<std::mutex> lock(mu);
 | 
						|
  reportDiagnostic(logName, Colors::RESET, "", msg);
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::message(const Twine &msg, llvm::raw_ostream &s) {
 | 
						|
  if (disableOutput)
 | 
						|
    return;
 | 
						|
  std::lock_guard<std::mutex> lock(mu);
 | 
						|
  s << msg << "\n";
 | 
						|
  s.flush();
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::warn(const Twine &msg) {
 | 
						|
  if (fatalWarnings) {
 | 
						|
    error(msg);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  std::lock_guard<std::mutex> lock(mu);
 | 
						|
  reportDiagnostic(getLocation(msg), Colors::MAGENTA, "warning", msg);
 | 
						|
  sep = getSeparator(msg);
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::error(const Twine &msg) {
 | 
						|
  // If Visual Studio-style error message mode is enabled,
 | 
						|
  // this particular error is printed out as two errors.
 | 
						|
  if (vsDiagnostics) {
 | 
						|
    static std::regex re(R"(^(duplicate symbol: .*))"
 | 
						|
                         R"((\n>>> defined at \S+:\d+.*\n>>>.*))"
 | 
						|
                         R"((\n>>> defined at \S+:\d+.*\n>>>.*))");
 | 
						|
    std::string str = msg.str();
 | 
						|
    std::smatch m;
 | 
						|
 | 
						|
    if (std::regex_match(str, m, re)) {
 | 
						|
      error(m.str(1) + m.str(2));
 | 
						|
      error(m.str(1) + m.str(3));
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  bool exit = false;
 | 
						|
  {
 | 
						|
    std::lock_guard<std::mutex> lock(mu);
 | 
						|
 | 
						|
    if (errorLimit == 0 || errorCount < errorLimit) {
 | 
						|
      reportDiagnostic(getLocation(msg), Colors::RED, "error", msg);
 | 
						|
    } else if (errorCount == errorLimit) {
 | 
						|
      reportDiagnostic(logName, Colors::RED, "error", errorLimitExceededMsg);
 | 
						|
      exit = exitEarly;
 | 
						|
    }
 | 
						|
 | 
						|
    sep = getSeparator(msg);
 | 
						|
    ++errorCount;
 | 
						|
  }
 | 
						|
 | 
						|
  if (exit)
 | 
						|
    exitLld(1);
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::error(const Twine &msg, ErrorTag tag,
 | 
						|
                         ArrayRef<StringRef> args) {
 | 
						|
  if (errorHandlingScript.empty()) {
 | 
						|
    error(msg);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  SmallVector<StringRef, 4> scriptArgs;
 | 
						|
  scriptArgs.push_back(errorHandlingScript);
 | 
						|
  switch (tag) {
 | 
						|
  case ErrorTag::LibNotFound:
 | 
						|
    scriptArgs.push_back("missing-lib");
 | 
						|
    break;
 | 
						|
  case ErrorTag::SymbolNotFound:
 | 
						|
    scriptArgs.push_back("undefined-symbol");
 | 
						|
    break;
 | 
						|
  }
 | 
						|
  scriptArgs.insert(scriptArgs.end(), args.begin(), args.end());
 | 
						|
  int res = llvm::sys::ExecuteAndWait(errorHandlingScript, scriptArgs);
 | 
						|
  if (res == 0) {
 | 
						|
    return error(msg);
 | 
						|
  } else {
 | 
						|
    // Temporarily disable error limit to make sure the two calls to error(...)
 | 
						|
    // only count as one.
 | 
						|
    uint64_t currentErrorLimit = errorLimit;
 | 
						|
    errorLimit = 0;
 | 
						|
    error(msg);
 | 
						|
    errorLimit = currentErrorLimit;
 | 
						|
    --errorCount;
 | 
						|
 | 
						|
    switch (res) {
 | 
						|
    case -1:
 | 
						|
      error("error handling script '" + errorHandlingScript +
 | 
						|
            "' failed to execute");
 | 
						|
      break;
 | 
						|
    case -2:
 | 
						|
      error("error handling script '" + errorHandlingScript +
 | 
						|
            "' crashed or timeout");
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      error("error handling script '" + errorHandlingScript +
 | 
						|
            "' exited with code " + Twine(res));
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void ErrorHandler::fatal(const Twine &msg) {
 | 
						|
  error(msg);
 | 
						|
  exitLld(1);
 | 
						|
}
 |