1 //===-- Status.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 "lldb/Utility/Status.h"
11 #include "lldb/Utility/LLDBLog.h"
12 #include "lldb/Utility/Log.h"
13 #include "lldb/Utility/VASPrintf.h"
14 #include "lldb/lldb-defines.h"
15 #include "lldb/lldb-enumerations.h"
16 #include "llvm/ADT/SmallString.h"
17 #include "llvm/ADT/StringRef.h"
18 #include "llvm/Support/Errno.h"
19 #include "llvm/Support/FormatProviders.h"
24 #include <system_error>
27 #include <mach/mach.h>
40 using namespace lldb_private
;
42 char CloneableError::ID
;
43 char CloneableECError::ID
;
44 char MachKernelError::ID
;
48 /// A std::error_code category for eErrorTypeGeneric.
49 class LLDBGenericCategory
: public std::error_category
{
50 const char *name() const noexcept override
{ return "LLDBGenericCategory"; }
51 std::string
message(int __ev
) const override
{ return "generic LLDB error"; };
53 LLDBGenericCategory
&lldb_generic_category() {
54 static LLDBGenericCategory g_generic_category
;
55 return g_generic_category
;
59 Status::Status() : m_error(llvm::Error::success()) {}
61 static llvm::Error
ErrorFromEnums(Status::ValueType err
, ErrorType type
,
64 case eErrorTypeMachKernel
:
65 return llvm::make_error
<MachKernelError
>(
66 std::error_code(err
, std::system_category()));
70 return llvm::Error::success();
72 return llvm::make_error
<Win32Error
>(
73 std::error_code(err
, std::system_category()));
76 return llvm::errorCodeToError(
77 std::error_code(err
, std::generic_category()));
78 return llvm::createStringError(
79 std::move(msg
), std::error_code(err
, std::generic_category()));
81 return llvm::createStringError(
82 std::move(msg
), std::error_code(err
, lldb_generic_category()));
86 Status::Status(ValueType err
, ErrorType type
, std::string msg
)
87 : m_error(ErrorFromEnums(err
, type
, msg
)) {}
89 // This logic is confusing because C++ calls the traditional (posix) errno codes
90 // "generic errors", while we use the term "generic" to mean completely
91 // arbitrary (text-based) errors.
92 Status::Status(std::error_code EC
)
93 : m_error(!EC
? llvm::Error::success() : llvm::errorCodeToError(EC
)) {}
95 Status::Status(std::string err_str
)
97 llvm::createStringError(llvm::inconvertibleErrorCode(), err_str
)) {}
99 const Status
&Status::operator=(Status
&&other
) {
101 llvm::consumeError(std::move(m_error
));
102 m_error
= std::move(other
.m_error
);
106 Status
Status::FromErrorStringWithFormat(const char *format
, ...) {
109 va_start(args
, format
);
110 if (format
!= nullptr && format
[0]) {
111 llvm::SmallString
<1024> buf
;
112 VASprintf(buf
, format
, args
);
113 string
= std::string(buf
.str());
116 return Status(string
);
119 /// Creates a deep copy of all known errors and converts all other
120 /// errors to a new llvm::StringError.
121 static llvm::Error
CloneError(const llvm::Error
&error
) {
122 llvm::Error result
= llvm::Error::success();
123 auto clone
= [](const llvm::ErrorInfoBase
&e
) {
124 if (e
.isA
<CloneableError
>())
125 return llvm::Error(static_cast<const CloneableError
&>(e
).Clone());
126 if (e
.isA
<llvm::ECError
>())
127 return llvm::errorCodeToError(e
.convertToErrorCode());
128 return llvm::make_error
<llvm::StringError
>(e
.message(),
129 e
.convertToErrorCode(), true);
131 llvm::visitErrors(error
, [&](const llvm::ErrorInfoBase
&e
) {
132 result
= joinErrors(std::move(result
), clone(e
));
137 Status
Status::FromError(llvm::Error error
) { return Status(std::move(error
)); }
139 llvm::Error
Status::ToError() const { return CloneError(m_error
); }
141 Status::~Status() { llvm::consumeError(std::move(m_error
)); }
144 static std::string
RetrieveWin32ErrorString(uint32_t error_code
) {
145 char *buffer
= nullptr;
147 // Retrieve win32 system error.
148 // First, attempt to load a en-US message
149 if (::FormatMessageA(
150 FORMAT_MESSAGE_ALLOCATE_BUFFER
| FORMAT_MESSAGE_FROM_SYSTEM
|
151 FORMAT_MESSAGE_MAX_WIDTH_MASK
,
152 NULL
, error_code
, MAKELANGID(LANG_ENGLISH
, SUBLANG_ENGLISH_US
),
153 (LPSTR
)&buffer
, 0, NULL
)) {
154 message
.assign(buffer
);
157 // If the previous didn't work, use the default OS language
158 else if (::FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER
|
159 FORMAT_MESSAGE_FROM_SYSTEM
|
160 FORMAT_MESSAGE_MAX_WIDTH_MASK
,
161 NULL
, error_code
, 0, (LPSTR
)&buffer
, 0, NULL
)) {
162 message
.assign(buffer
);
169 std::string
MachKernelError::message() const {
170 #if defined(__APPLE__)
171 if (const char *s
= ::mach_error_string(convertToErrorCode().value()))
174 return "MachKernelError";
177 std::string
Win32Error::message() const {
179 return RetrieveWin32ErrorString(convertToErrorCode().value());
184 std::unique_ptr
<CloneableError
> MachKernelError::Clone() const {
185 return std::make_unique
<MachKernelError
>(convertToErrorCode());
188 std::unique_ptr
<CloneableError
> Win32Error::Clone() const {
189 return std::make_unique
<Win32Error
>(convertToErrorCode());
192 // Get the error value as a NULL C string. The error string will be fetched and
193 // cached on demand. The cached error string value will remain until the error
194 // value is changed or cleared.
195 const char *Status::AsCString(const char *default_error_str
) const {
199 m_string
= llvm::toStringWithoutConsuming(m_error
);
200 // Backwards compatibility with older implementations of Status.
201 if (m_error
.isA
<llvm::ECError
>())
202 if (!m_string
.empty() && m_string
[m_string
.size() - 1] == '\n')
205 if (m_string
.empty()) {
206 if (default_error_str
)
207 m_string
.assign(default_error_str
);
209 return nullptr; // User wanted a nullptr string back...
211 return m_string
.c_str();
214 // Clear the error and any cached error string that it might contain.
215 void Status::Clear() {
217 LLDB_LOG_ERRORV(GetLog(LLDBLog::API
), std::move(m_error
),
218 "dropping error {0}");
219 m_error
= llvm::Error::success();
222 Status::ValueType
Status::GetError() const {
223 Status::ValueType result
= 0;
224 llvm::visitErrors(m_error
, [&](const llvm::ErrorInfoBase
&error
) {
225 // Return the first only.
228 std::error_code ec
= error
.convertToErrorCode();
234 static ErrorType
ErrorCodeToErrorType(std::error_code ec
) {
235 if (ec
.category() == std::generic_category())
236 return eErrorTypePOSIX
;
237 if (ec
.category() == lldb_generic_category() ||
238 ec
== llvm::inconvertibleErrorCode())
239 return eErrorTypeGeneric
;
240 return eErrorTypeInvalid
;
243 ErrorType
CloneableECError::GetErrorType() const {
244 return ErrorCodeToErrorType(EC
);
247 lldb::ErrorType
MachKernelError::GetErrorType() const {
248 return lldb::eErrorTypeMachKernel
;
251 lldb::ErrorType
Win32Error::GetErrorType() const {
252 return lldb::eErrorTypeWin32
;
255 ErrorType
Status::GetType() const {
256 ErrorType result
= eErrorTypeInvalid
;
257 llvm::visitErrors(m_error
, [&](const llvm::ErrorInfoBase
&error
) {
258 // Return the first only.
259 if (result
!= eErrorTypeInvalid
)
261 result
= ErrorCodeToErrorType(error
.convertToErrorCode());
266 bool Status::Fail() const {
267 // Note that this does not clear the checked flag in
268 // m_error. Otherwise we'd need to make this thread-safe.
269 return m_error
.isA
<llvm::ErrorInfoBase
>();
272 Status
Status::FromErrno() { return Status(llvm::errnoAsErrorCode()); }
274 // Returns true if the error code in this object is considered a successful
276 bool Status::Success() const { return !Fail(); }
278 void llvm::format_provider
<lldb_private::Status
>::format(
279 const lldb_private::Status
&error
, llvm::raw_ostream
&OS
,
280 llvm::StringRef Options
) {
281 llvm::format_provider
<llvm::StringRef
>::format(error
.AsCString(), OS
,