
This reverts commit c861fe8a71e64f3d2108c58147e7375cd9314521. Unfortunately, this use of hidden visibility attributes causes user-defined specializations of standard-library types to also be marked hidden by default, which is incorrect. See discussion thread on #131156. ...and also reverts the follow-up commits: Revert "[libc++] Add explicit ABI annotations to functions from the block runtime declared in <__functional/function.h> (#140592)" This reverts commit 3e4c9dc299c35155934688184319d391b298fff7. Revert "[libc++] Make ABI annotations explicit for windows-specific code (#140507)" This reverts commit f73287e623a6c2e4a3485832bc3e10860cd26eb5. Revert "[libc++][NFC] Replace a few "namespace std" with the correct macro (#140510)" This reverts commit 1d411f27c769a32cb22ce50b9dc4421e34fd40dd.
136 lines
4.9 KiB
C++
136 lines
4.9 KiB
C++
// -*- C++ -*-
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// For information see https://libcxx.llvm.org/DesignDocs/TimeZone.html
|
|
|
|
#ifndef _LIBCPP___CHRONO_EXCEPTION_H
|
|
#define _LIBCPP___CHRONO_EXCEPTION_H
|
|
|
|
#include <version>
|
|
// Enable the contents of the header only when libc++ was built with experimental features enabled.
|
|
#if _LIBCPP_HAS_EXPERIMENTAL_TZDB
|
|
|
|
# include <__chrono/calendar.h>
|
|
# include <__chrono/local_info.h>
|
|
# include <__chrono/time_point.h>
|
|
# include <__config>
|
|
# include <__configuration/availability.h>
|
|
# include <__verbose_abort>
|
|
# include <format>
|
|
# include <stdexcept>
|
|
# include <string>
|
|
|
|
# if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
|
# pragma GCC system_header
|
|
# endif
|
|
|
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
|
|
|
# if _LIBCPP_STD_VER >= 20
|
|
|
|
namespace chrono {
|
|
|
|
class nonexistent_local_time : public runtime_error {
|
|
public:
|
|
template <class _Duration>
|
|
_LIBCPP_HIDE_FROM_ABI nonexistent_local_time(const local_time<_Duration>& __time, const local_info& __info)
|
|
: runtime_error{__create_message(__time, __info)} {
|
|
// [time.zone.exception.nonexist]/2
|
|
// Preconditions: i.result == local_info::nonexistent is true.
|
|
// The value of __info.result is not used.
|
|
_LIBCPP_ASSERT_PEDANTIC(__info.result == local_info::nonexistent,
|
|
"creating an nonexistent_local_time from a local_info that is not non-existent");
|
|
}
|
|
|
|
_LIBCPP_HIDE_FROM_ABI nonexistent_local_time(const nonexistent_local_time&) = default;
|
|
_LIBCPP_HIDE_FROM_ABI nonexistent_local_time& operator=(const nonexistent_local_time&) = default;
|
|
|
|
_LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI ~nonexistent_local_time() override; // exported as key function
|
|
|
|
private:
|
|
template <class _Duration>
|
|
_LIBCPP_HIDE_FROM_ABI string __create_message(const local_time<_Duration>& __time, const local_info& __info) {
|
|
return std::format(
|
|
R"({} is in a gap between
|
|
{} {} and
|
|
{} {} which are both equivalent to
|
|
{} UTC)",
|
|
__time,
|
|
local_seconds{__info.first.end.time_since_epoch()} + __info.first.offset,
|
|
__info.first.abbrev,
|
|
local_seconds{__info.second.begin.time_since_epoch()} + __info.second.offset,
|
|
__info.second.abbrev,
|
|
__info.first.end);
|
|
}
|
|
};
|
|
|
|
template <class _Duration>
|
|
[[noreturn]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_HIDE_FROM_ABI void __throw_nonexistent_local_time(
|
|
[[maybe_unused]] const local_time<_Duration>& __time, [[maybe_unused]] const local_info& __info) {
|
|
# if _LIBCPP_HAS_EXCEPTIONS
|
|
throw nonexistent_local_time(__time, __info);
|
|
# else
|
|
_LIBCPP_VERBOSE_ABORT("nonexistent_local_time was thrown in -fno-exceptions mode");
|
|
# endif
|
|
}
|
|
|
|
class ambiguous_local_time : public runtime_error {
|
|
public:
|
|
template <class _Duration>
|
|
_LIBCPP_HIDE_FROM_ABI ambiguous_local_time(const local_time<_Duration>& __time, const local_info& __info)
|
|
: runtime_error{__create_message(__time, __info)} {
|
|
// [time.zone.exception.ambig]/2
|
|
// Preconditions: i.result == local_info::ambiguous is true.
|
|
// The value of __info.result is not used.
|
|
_LIBCPP_ASSERT_PEDANTIC(__info.result == local_info::ambiguous,
|
|
"creating an ambiguous_local_time from a local_info that is not ambiguous");
|
|
}
|
|
|
|
_LIBCPP_HIDE_FROM_ABI ambiguous_local_time(const ambiguous_local_time&) = default;
|
|
_LIBCPP_HIDE_FROM_ABI ambiguous_local_time& operator=(const ambiguous_local_time&) = default;
|
|
|
|
_LIBCPP_AVAILABILITY_TZDB _LIBCPP_EXPORTED_FROM_ABI ~ambiguous_local_time() override; // exported as key function
|
|
|
|
private:
|
|
template <class _Duration>
|
|
_LIBCPP_HIDE_FROM_ABI string __create_message(const local_time<_Duration>& __time, const local_info& __info) {
|
|
return std::format(
|
|
// There are two spaces after the full-stop; this has been verified
|
|
// in the sources of the Standard.
|
|
R"({0} is ambiguous. It could be
|
|
{0} {1} == {2} UTC or
|
|
{0} {3} == {4} UTC)",
|
|
__time,
|
|
__info.first.abbrev,
|
|
__time - __info.first.offset,
|
|
__info.second.abbrev,
|
|
__time - __info.second.offset);
|
|
}
|
|
};
|
|
|
|
template <class _Duration>
|
|
[[noreturn]] _LIBCPP_AVAILABILITY_TZDB _LIBCPP_HIDE_FROM_ABI void __throw_ambiguous_local_time(
|
|
[[maybe_unused]] const local_time<_Duration>& __time, [[maybe_unused]] const local_info& __info) {
|
|
# if _LIBCPP_HAS_EXCEPTIONS
|
|
throw ambiguous_local_time(__time, __info);
|
|
# else
|
|
_LIBCPP_VERBOSE_ABORT("ambiguous_local_time was thrown in -fno-exceptions mode");
|
|
# endif
|
|
}
|
|
|
|
} // namespace chrono
|
|
|
|
# endif // _LIBCPP_STD_VER >= 20
|
|
|
|
_LIBCPP_END_NAMESPACE_STD
|
|
|
|
#endif // _LIBCPP_HAS_EXPERIMENTAL_TZDB
|
|
|
|
#endif // _LIBCPP___CHRONO_EXCEPTION_H
|