| //===-- Status.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 "lldb/Utility/Status.h" |
| |
| #include "lldb/Utility/LLDBLog.h" |
| #include "lldb/Utility/Log.h" |
| #include "lldb/Utility/VASPrintf.h" |
| #include "lldb/lldb-defines.h" |
| #include "lldb/lldb-enumerations.h" |
| #include "llvm/ADT/SmallString.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/Support/Errno.h" |
| #include "llvm/Support/FormatProviders.h" |
| |
| #include <cerrno> |
| #include <cstdarg> |
| #include <string> |
| #include <system_error> |
| |
| #ifdef __APPLE__ |
| #include <mach/mach.h> |
| #endif |
| |
| #ifdef _WIN32 |
| #include <windows.h> |
| #endif |
| #include <cstdint> |
| |
| namespace llvm { |
| class raw_ostream; |
| } |
| |
| using namespace lldb; |
| using namespace lldb_private; |
| |
| char CloneableError::ID; |
| char CloneableECError::ID; |
| char MachKernelError::ID; |
| char Win32Error::ID; |
| |
| namespace { |
| /// A std::error_code category for eErrorTypeGeneric. |
| class LLDBGenericCategory : public std::error_category { |
| const char *name() const noexcept override { return "LLDBGenericCategory"; } |
| std::string message(int __ev) const override { return "generic LLDB error"; }; |
| }; |
| LLDBGenericCategory &lldb_generic_category() { |
| static LLDBGenericCategory g_generic_category; |
| return g_generic_category; |
| } |
| } // namespace |
| |
| Status::Status() : m_error(llvm::Error::success()) {} |
| |
| static llvm::Error ErrorFromEnums(Status::ValueType err, ErrorType type, |
| std::string msg) { |
| switch (type) { |
| case eErrorTypeMachKernel: |
| return llvm::make_error<MachKernelError>( |
| std::error_code(err, std::system_category())); |
| case eErrorTypeWin32: |
| #ifdef _WIN32 |
| if (err == NO_ERROR) |
| return llvm::Error::success(); |
| #endif |
| return llvm::make_error<Win32Error>( |
| std::error_code(err, std::system_category())); |
| case eErrorTypePOSIX: |
| if (msg.empty()) |
| return llvm::errorCodeToError( |
| std::error_code(err, std::generic_category())); |
| return llvm::createStringError( |
| std::move(msg), std::error_code(err, std::generic_category())); |
| default: |
| return llvm::createStringError( |
| std::move(msg), std::error_code(err, lldb_generic_category())); |
| } |
| } |
| |
| Status::Status(ValueType err, ErrorType type, std::string msg) |
| : m_error(ErrorFromEnums(err, type, msg)) {} |
| |
| // This logic is confusing because C++ calls the traditional (posix) errno codes |
| // "generic errors", while we use the term "generic" to mean completely |
| // arbitrary (text-based) errors. |
| Status::Status(std::error_code EC) |
| : m_error(!EC ? llvm::Error::success() : llvm::errorCodeToError(EC)) {} |
| |
| Status::Status(std::string err_str) |
| : m_error( |
| llvm::createStringError(llvm::inconvertibleErrorCode(), err_str)) {} |
| |
| const Status &Status::operator=(Status &&other) { |
| Clear(); |
| llvm::consumeError(std::move(m_error)); |
| m_error = std::move(other.m_error); |
| return *this; |
| } |
| |
| Status Status::FromErrorStringWithFormat(const char *format, ...) { |
| std::string string; |
| va_list args; |
| va_start(args, format); |
| if (format != nullptr && format[0]) { |
| llvm::SmallString<1024> buf; |
| VASprintf(buf, format, args); |
| string = std::string(buf.str()); |
| } |
| va_end(args); |
| return Status(string); |
| } |
| |
| /// Creates a deep copy of all known errors and converts all other |
| /// errors to a new llvm::StringError. |
| static llvm::Error CloneError(const llvm::Error &error) { |
| llvm::Error result = llvm::Error::success(); |
| auto clone = [](const llvm::ErrorInfoBase &e) { |
| if (e.isA<CloneableError>()) |
| return llvm::Error(static_cast<const CloneableError &>(e).Clone()); |
| if (e.isA<llvm::ECError>()) |
| return llvm::errorCodeToError(e.convertToErrorCode()); |
| return llvm::make_error<llvm::StringError>(e.message(), |
| e.convertToErrorCode(), true); |
| }; |
| llvm::visitErrors(error, [&](const llvm::ErrorInfoBase &e) { |
| result = joinErrors(std::move(result), clone(e)); |
| }); |
| return result; |
| } |
| |
| Status Status::FromError(llvm::Error error) { return Status(std::move(error)); } |
| |
| llvm::Error Status::ToError() const { return CloneError(m_error); } |
| |
| Status::~Status() { llvm::consumeError(std::move(m_error)); } |
| |
| #ifdef _WIN32 |
| static std::string RetrieveWin32ErrorString(uint32_t error_code) { |
| char *buffer = nullptr; |
| std::string message; |
| // Retrieve win32 system error. |
| // First, attempt to load a en-US message |
| if (::FormatMessageA( |
| FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | |
| FORMAT_MESSAGE_MAX_WIDTH_MASK, |
| NULL, error_code, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), |
| (LPSTR)&buffer, 0, NULL)) { |
| message.assign(buffer); |
| ::LocalFree(buffer); |
| } |
| // If the previous didn't work, use the default OS language |
| else if (::FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | |
| FORMAT_MESSAGE_FROM_SYSTEM | |
| FORMAT_MESSAGE_MAX_WIDTH_MASK, |
| NULL, error_code, 0, (LPSTR)&buffer, 0, NULL)) { |
| message.assign(buffer); |
| ::LocalFree(buffer); |
| } |
| return message; |
| } |
| #endif |
| |
| std::string MachKernelError::message() const { |
| #if defined(__APPLE__) |
| if (const char *s = ::mach_error_string(convertToErrorCode().value())) |
| return s; |
| #endif |
| return "MachKernelError"; |
| } |
| |
| std::string Win32Error::message() const { |
| #if defined(_WIN32) |
| return RetrieveWin32ErrorString(convertToErrorCode().value()); |
| #endif |
| return "Win32Error"; |
| } |
| |
| std::unique_ptr<CloneableError> MachKernelError::Clone() const { |
| return std::make_unique<MachKernelError>(convertToErrorCode()); |
| } |
| |
| std::unique_ptr<CloneableError> Win32Error::Clone() const { |
| return std::make_unique<Win32Error>(convertToErrorCode()); |
| } |
| |
| // Get the error value as a NULL C string. The error string will be fetched and |
| // cached on demand. The cached error string value will remain until the error |
| // value is changed or cleared. |
| const char *Status::AsCString(const char *default_error_str) const { |
| if (Success()) |
| return nullptr; |
| |
| m_string = llvm::toStringWithoutConsuming(m_error); |
| // Backwards compatibility with older implementations of Status. |
| if (m_error.isA<llvm::ECError>()) |
| if (!m_string.empty() && m_string[m_string.size() - 1] == '\n') |
| m_string.pop_back(); |
| |
| if (m_string.empty()) { |
| if (default_error_str) |
| m_string.assign(default_error_str); |
| else |
| return nullptr; // User wanted a nullptr string back... |
| } |
| return m_string.c_str(); |
| } |
| |
| // Clear the error and any cached error string that it might contain. |
| void Status::Clear() { |
| if (m_error) |
| LLDB_LOG_ERRORV(GetLog(LLDBLog::API), std::move(m_error), |
| "dropping error {0}"); |
| m_error = llvm::Error::success(); |
| } |
| |
| Status::ValueType Status::GetError() const { |
| Status::ValueType result = 0; |
| llvm::visitErrors(m_error, [&](const llvm::ErrorInfoBase &error) { |
| // Return the first only. |
| if (result) |
| return; |
| std::error_code ec = error.convertToErrorCode(); |
| result = ec.value(); |
| }); |
| return result; |
| } |
| |
| static ErrorType ErrorCodeToErrorType(std::error_code ec) { |
| if (ec.category() == std::generic_category()) |
| return eErrorTypePOSIX; |
| if (ec.category() == lldb_generic_category() || |
| ec == llvm::inconvertibleErrorCode()) |
| return eErrorTypeGeneric; |
| return eErrorTypeInvalid; |
| } |
| |
| ErrorType CloneableECError::GetErrorType() const { |
| return ErrorCodeToErrorType(EC); |
| } |
| |
| lldb::ErrorType MachKernelError::GetErrorType() const { |
| return lldb::eErrorTypeMachKernel; |
| } |
| |
| lldb::ErrorType Win32Error::GetErrorType() const { |
| return lldb::eErrorTypeWin32; |
| } |
| |
| StructuredData::ObjectSP Status::GetAsStructuredData() const { |
| auto dict_up = std::make_unique<StructuredData::Dictionary>(); |
| auto array_up = std::make_unique<StructuredData::Array>(); |
| llvm::visitErrors(m_error, [&](const llvm::ErrorInfoBase &error) { |
| if (error.isA<CloneableError>()) |
| array_up->AddItem( |
| static_cast<const CloneableError &>(error).GetAsStructuredData()); |
| else |
| array_up->AddStringItem(error.message()); |
| }); |
| dict_up->AddIntegerItem("version", 1u); |
| dict_up->AddIntegerItem("type", (unsigned)GetType()); |
| dict_up->AddItem("errors", std::move(array_up)); |
| return dict_up; |
| } |
| |
| StructuredData::ObjectSP CloneableECError::GetAsStructuredData() const { |
| auto dict_up = std::make_unique<StructuredData::Dictionary>(); |
| dict_up->AddIntegerItem("version", 1u); |
| dict_up->AddIntegerItem("error_code", EC.value()); |
| dict_up->AddStringItem("message", message()); |
| return dict_up; |
| } |
| |
| ErrorType Status::GetType() const { |
| ErrorType result = eErrorTypeInvalid; |
| llvm::visitErrors(m_error, [&](const llvm::ErrorInfoBase &error) { |
| // Return the first only. |
| if (result != eErrorTypeInvalid) |
| return; |
| if (error.isA<CloneableError>()) |
| result = static_cast<const CloneableError &>(error).GetErrorType(); |
| else |
| result = ErrorCodeToErrorType(error.convertToErrorCode()); |
| |
| }); |
| return result; |
| } |
| |
| bool Status::Fail() const { |
| // Note that this does not clear the checked flag in |
| // m_error. Otherwise we'd need to make this thread-safe. |
| return m_error.isA<llvm::ErrorInfoBase>(); |
| } |
| |
| Status Status::FromErrno() { return Status(llvm::errnoAsErrorCode()); } |
| |
| // Returns true if the error code in this object is considered a successful |
| // return value. |
| bool Status::Success() const { return !Fail(); } |
| |
| void llvm::format_provider<lldb_private::Status>::format( |
| const lldb_private::Status &error, llvm::raw_ostream &OS, |
| llvm::StringRef Options) { |
| llvm::format_provider<llvm::StringRef>::format(error.AsCString(), OS, |
| Options); |
| } |