2015-12-15 22:53:43 +01:00
|
|
|
/*
|
2019-08-19 18:53:08 +02:00
|
|
|
* Copyright 2013-2015 Max Kellermann <max.kellermann@gmail.com>
|
2015-12-15 22:53:43 +01:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* - Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
*
|
|
|
|
* - Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
|
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SYSTEM_ERROR_HXX
|
|
|
|
#define SYSTEM_ERROR_HXX
|
|
|
|
|
2020-07-23 16:14:23 +02:00
|
|
|
#include <system_error> // IWYU pragma: export
|
2015-12-15 22:53:43 +01:00
|
|
|
#include <utility>
|
|
|
|
|
2016-07-02 13:59:47 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2015-12-15 22:53:43 +01:00
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
FormatSystemError(std::error_code code, const char *fmt,
|
|
|
|
Args&&... args) noexcept
|
2015-12-15 22:53:43 +01:00
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
snprintf(buffer, sizeof(buffer), fmt, std::forward<Args>(args)...);
|
|
|
|
return std::system_error(code, buffer);
|
|
|
|
}
|
|
|
|
|
2017-12-12 10:22:20 +01:00
|
|
|
#ifdef _WIN32
|
2015-12-15 22:53:43 +01:00
|
|
|
|
2021-05-19 17:24:02 +02:00
|
|
|
#include <errhandlingapi.h> // for GetLastError()
|
|
|
|
#include <windef.h> // for HWND (needed by winbase.h)
|
|
|
|
#include <winbase.h> // for FormatMessageA()
|
2015-12-15 22:53:43 +01:00
|
|
|
|
2015-12-18 01:06:12 +01:00
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
MakeLastError(DWORD code, const char *msg) noexcept
|
2015-12-18 01:06:12 +01:00
|
|
|
{
|
|
|
|
return std::system_error(std::error_code(code, std::system_category()),
|
|
|
|
msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
MakeLastError(const char *msg) noexcept
|
2015-12-18 01:06:12 +01:00
|
|
|
{
|
|
|
|
return MakeLastError(GetLastError(), msg);
|
|
|
|
}
|
|
|
|
|
2015-12-15 22:53:43 +01:00
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
FormatLastError(DWORD code, const char *fmt, Args&&... args) noexcept
|
2015-12-15 22:53:43 +01:00
|
|
|
{
|
|
|
|
char buffer[512];
|
|
|
|
const auto end = buffer + sizeof(buffer);
|
2022-11-28 09:27:53 +01:00
|
|
|
constexpr std::size_t max_prefix = sizeof(buffer) - 128;
|
|
|
|
size_t length = snprintf(buffer, max_prefix,
|
2015-12-15 22:53:43 +01:00
|
|
|
fmt, std::forward<Args>(args)...);
|
2022-11-28 09:27:53 +01:00
|
|
|
if (length >= max_prefix)
|
|
|
|
length = max_prefix - 1;
|
2015-12-15 22:53:43 +01:00
|
|
|
char *p = buffer + length;
|
|
|
|
*p++ = ':';
|
|
|
|
*p++ = ' ';
|
|
|
|
|
|
|
|
FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM |
|
|
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
|
|
nullptr, code, 0, p, end - p, nullptr);
|
2015-12-18 01:06:12 +01:00
|
|
|
return MakeLastError(code, buffer);
|
2015-12-15 22:53:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
FormatLastError(const char *fmt, Args&&... args) noexcept
|
2015-12-15 22:53:43 +01:00
|
|
|
{
|
|
|
|
return FormatLastError(GetLastError(), fmt,
|
|
|
|
std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
|
2017-12-12 10:22:20 +01:00
|
|
|
#endif /* _WIN32 */
|
2015-12-15 22:53:43 +01:00
|
|
|
|
2020-07-23 16:14:23 +02:00
|
|
|
#include <cerrno> // IWYU pragma: export
|
2020-03-12 23:51:16 +01:00
|
|
|
|
2015-12-15 22:53:43 +01:00
|
|
|
#include <string.h>
|
|
|
|
|
2016-12-04 19:50:21 +01:00
|
|
|
/**
|
|
|
|
* Returns the error_category to be used to wrap errno values. The
|
|
|
|
* C++ standard does not define this well, so this code is based on
|
|
|
|
* observations what C++ standard library implementations actually
|
|
|
|
* use.
|
|
|
|
*
|
|
|
|
* @see https://stackoverflow.com/questions/28746372/system-error-categories-and-standard-system-error-codes
|
|
|
|
*/
|
|
|
|
static inline const std::error_category &
|
2017-09-13 17:48:13 +02:00
|
|
|
ErrnoCategory() noexcept
|
2016-12-04 19:50:21 +01:00
|
|
|
{
|
2017-12-12 10:22:20 +01:00
|
|
|
#ifdef _WIN32
|
2016-12-04 19:50:21 +01:00
|
|
|
/* on Windows, the generic_category() is used for errno
|
|
|
|
values */
|
|
|
|
return std::generic_category();
|
|
|
|
#else
|
|
|
|
/* on POSIX, system_category() appears to be the best
|
|
|
|
choice */
|
|
|
|
return std::system_category();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-18 01:06:12 +01:00
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
MakeErrno(int code, const char *msg) noexcept
|
2015-12-18 01:06:12 +01:00
|
|
|
{
|
2016-12-04 19:50:21 +01:00
|
|
|
return std::system_error(std::error_code(code, ErrnoCategory()),
|
2015-12-18 01:06:12 +01:00
|
|
|
msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
MakeErrno(const char *msg) noexcept
|
2015-12-18 01:06:12 +01:00
|
|
|
{
|
|
|
|
return MakeErrno(errno, msg);
|
|
|
|
}
|
|
|
|
|
2015-12-15 22:53:43 +01:00
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
FormatErrno(int code, const char *fmt, Args&&... args) noexcept
|
2015-12-15 22:53:43 +01:00
|
|
|
{
|
|
|
|
char buffer[512];
|
2015-12-27 06:04:57 +01:00
|
|
|
snprintf(buffer, sizeof(buffer),
|
|
|
|
fmt, std::forward<Args>(args)...);
|
2015-12-18 01:06:12 +01:00
|
|
|
return MakeErrno(code, buffer);
|
2015-12-15 22:53:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
2017-09-13 17:48:13 +02:00
|
|
|
FormatErrno(const char *fmt, Args&&... args) noexcept
|
2015-12-15 22:53:43 +01:00
|
|
|
{
|
|
|
|
return FormatErrno(errno, fmt, std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
|
2019-12-04 12:31:38 +01:00
|
|
|
template<typename... Args>
|
|
|
|
static inline std::system_error
|
|
|
|
FormatFileNotFound(const char *fmt, Args&&... args) noexcept
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
return FormatLastError(ERROR_FILE_NOT_FOUND, fmt,
|
|
|
|
std::forward<Args>(args)...);
|
|
|
|
#else
|
|
|
|
return FormatErrno(ENOENT, fmt, std::forward<Args>(args)...);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2019-07-09 18:17:41 +02:00
|
|
|
inline bool
|
|
|
|
IsErrno(const std::system_error &e, int code) noexcept
|
|
|
|
{
|
|
|
|
return e.code().category() == ErrnoCategory() &&
|
|
|
|
e.code().value() == code;
|
|
|
|
}
|
|
|
|
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2015-12-18 00:54:08 +01:00
|
|
|
static inline bool
|
2017-05-08 14:44:49 +02:00
|
|
|
IsFileNotFound(const std::system_error &e) noexcept
|
2015-12-18 00:54:08 +01:00
|
|
|
{
|
2017-12-12 10:22:20 +01:00
|
|
|
#ifdef _WIN32
|
2015-12-18 00:54:08 +01:00
|
|
|
return e.code().category() == std::system_category() &&
|
|
|
|
e.code().value() == ERROR_FILE_NOT_FOUND;
|
|
|
|
#else
|
2019-07-09 18:17:41 +02:00
|
|
|
return IsErrno(e, ENOENT);
|
2015-12-18 00:54:08 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2016-09-09 15:37:06 +02:00
|
|
|
static inline bool
|
2017-05-08 14:44:49 +02:00
|
|
|
IsPathNotFound(const std::system_error &e) noexcept
|
2016-09-09 15:37:06 +02:00
|
|
|
{
|
2017-12-12 10:22:20 +01:00
|
|
|
#ifdef _WIN32
|
2016-09-09 15:37:06 +02:00
|
|
|
return e.code().category() == std::system_category() &&
|
|
|
|
e.code().value() == ERROR_PATH_NOT_FOUND;
|
|
|
|
#else
|
2019-07-09 18:17:41 +02:00
|
|
|
return IsErrno(e, ENOTDIR);
|
2016-09-09 15:37:06 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-02-08 14:59:40 +01:00
|
|
|
[[gnu::pure]]
|
2015-12-29 12:56:23 +01:00
|
|
|
static inline bool
|
2017-05-08 14:44:49 +02:00
|
|
|
IsAccessDenied(const std::system_error &e) noexcept
|
2015-12-29 12:56:23 +01:00
|
|
|
{
|
2017-12-12 10:22:20 +01:00
|
|
|
#ifdef _WIN32
|
2015-12-29 12:56:23 +01:00
|
|
|
return e.code().category() == std::system_category() &&
|
|
|
|
e.code().value() == ERROR_ACCESS_DENIED;
|
|
|
|
#else
|
2019-07-09 18:17:41 +02:00
|
|
|
return IsErrno(e, EACCES);
|
2015-12-29 12:56:23 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-15 22:53:43 +01:00
|
|
|
#endif
|