1 //===-- Implementation of a class for mapping errors to strings -----------===//
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 "error_to_string.h"
10 #include "platform_errors.h"
12 #include "src/__support/CPP/span.h"
13 #include "src/__support/CPP/string_view.h"
14 #include "src/__support/CPP/stringstream.h"
15 #include "src/__support/StringUtil/message_mapper.h"
16 #include "src/__support/integer_to_string.h"
17 #include "src/__support/macros/attributes.h"
18 #include "src/__support/macros/config.h"
22 namespace LIBC_NAMESPACE_DECL
{
25 constexpr size_t max_buff_size() {
26 constexpr size_t unknown_str_len
= sizeof("Unknown error");
27 // the buffer should be able to hold "Unknown error" + ' ' + num_str
28 return (unknown_str_len
+ 1 + IntegerToString
<int>::buffer_size()) *
32 // This is to hold error strings that have to be custom built. It may be
33 // rewritten on every call to strerror (or other error to string function).
34 constexpr size_t ERR_BUFFER_SIZE
= max_buff_size();
35 LIBC_THREAD_LOCAL
char error_buffer
[ERR_BUFFER_SIZE
];
37 constexpr size_t TOTAL_STR_LEN
= total_str_len(PLATFORM_ERRORS
);
39 // Since the StringMappings array is a map from error numbers to their
40 // corresponding strings, we have to have an array large enough we can use the
41 // error numbers as indexes. The current linux configuration has 132 values with
42 // the maximum value being 133 (41 and 58 are skipped). If other platforms use
43 // negative numbers or discontiguous ranges, then the array should be turned
44 // into a proper hashmap.
45 constexpr size_t ERR_ARRAY_SIZE
= max_key_val(PLATFORM_ERRORS
) + 1;
47 constexpr MessageMapper
<ERR_ARRAY_SIZE
, TOTAL_STR_LEN
>
48 ERROR_MAPPER(PLATFORM_ERRORS
);
50 constexpr MessageMapper
<ERR_ARRAY_SIZE
, TOTAL_STR_LEN
>
51 ERRNO_NAME_MAPPER(PLATFORM_ERRNO_NAMES
);
53 cpp::string_view
build_error_string(int err_num
, cpp::span
<char> buffer
) {
54 // if the buffer can't hold "Unknown error" + ' ' + num_str, then just
55 // return "Unknown error".
57 (sizeof("Unknown error") + 1 + IntegerToString
<int>::buffer_size()))
58 return const_cast<char *>("Unknown error");
60 cpp::StringStream
buffer_stream(
61 {const_cast<char *>(buffer
.data()), buffer
.size()});
62 buffer_stream
<< "Unknown error" << ' ' << err_num
<< '\0';
63 return buffer_stream
.str();
66 } // namespace internal
68 cpp::string_view
get_error_string(int err_num
) {
69 return get_error_string(err_num
,
70 {internal::error_buffer
, internal::ERR_BUFFER_SIZE
});
73 cpp::string_view
get_error_string(int err_num
, cpp::span
<char> buffer
) {
74 auto opt_str
= internal::ERROR_MAPPER
.get_str(err_num
);
78 return internal::build_error_string(err_num
, buffer
);
81 cpp::optional
<cpp::string_view
> try_get_errno_name(int err_num
) {
82 return internal::ERRNO_NAME_MAPPER
.get_str(err_num
);
85 } // namespace LIBC_NAMESPACE_DECL