123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396 |
- #ifndef Y_ABSL_STATUS_INTERNAL_STATUSOR_INTERNAL_H_
- #define Y_ABSL_STATUS_INTERNAL_STATUSOR_INTERNAL_H_
- #include <type_traits>
- #include <utility>
- #include "y_absl/base/attributes.h"
- #include "y_absl/meta/type_traits.h"
- #include "y_absl/status/status.h"
- #include "y_absl/utility/utility.h"
- namespace y_absl {
- Y_ABSL_NAMESPACE_BEGIN
- template <typename T>
- class Y_ABSL_MUST_USE_RESULT StatusOr;
- namespace internal_statusor {
- template <typename T, typename U, typename = void>
- struct HasConversionOperatorToStatusOr : std::false_type {};
- template <typename T, typename U>
- void test(char (*)[sizeof(std::declval<U>().operator y_absl::StatusOr<T>())]);
- template <typename T, typename U>
- struct HasConversionOperatorToStatusOr<T, U, decltype(test<T, U>(0))>
- : std::true_type {};
- template <typename T, typename U>
- using IsConstructibleOrConvertibleFromStatusOr =
- y_absl::disjunction<std::is_constructible<T, StatusOr<U>&>,
- std::is_constructible<T, const StatusOr<U>&>,
- std::is_constructible<T, StatusOr<U>&&>,
- std::is_constructible<T, const StatusOr<U>&&>,
- std::is_convertible<StatusOr<U>&, T>,
- std::is_convertible<const StatusOr<U>&, T>,
- std::is_convertible<StatusOr<U>&&, T>,
- std::is_convertible<const StatusOr<U>&&, T>>;
- template <typename T, typename U>
- using IsConstructibleOrConvertibleOrAssignableFromStatusOr =
- y_absl::disjunction<IsConstructibleOrConvertibleFromStatusOr<T, U>,
- std::is_assignable<T&, StatusOr<U>&>,
- std::is_assignable<T&, const StatusOr<U>&>,
- std::is_assignable<T&, StatusOr<U>&&>,
- std::is_assignable<T&, const StatusOr<U>&&>>;
- template <typename T, typename U>
- struct IsDirectInitializationAmbiguous
- : public y_absl::conditional_t<
- std::is_same<y_absl::remove_cv_t<y_absl::remove_reference_t<U>>,
- U>::value,
- std::false_type,
- IsDirectInitializationAmbiguous<
- T, y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>> {};
- template <typename T, typename V>
- struct IsDirectInitializationAmbiguous<T, y_absl::StatusOr<V>>
- : public IsConstructibleOrConvertibleFromStatusOr<T, V> {};
- template <typename T, typename U>
- using IsDirectInitializationValid = y_absl::disjunction<
-
- std::is_same<T, y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- y_absl::negation<y_absl::disjunction<
- std::is_same<y_absl::StatusOr<T>,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- std::is_same<y_absl::Status,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- std::is_same<y_absl::in_place_t,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- IsDirectInitializationAmbiguous<T, U>>>>;
- template <typename T, typename U>
- struct IsForwardingAssignmentAmbiguous
- : public y_absl::conditional_t<
- std::is_same<y_absl::remove_cv_t<y_absl::remove_reference_t<U>>,
- U>::value,
- std::false_type,
- IsForwardingAssignmentAmbiguous<
- T, y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>> {};
- template <typename T, typename U>
- struct IsForwardingAssignmentAmbiguous<T, y_absl::StatusOr<U>>
- : public IsConstructibleOrConvertibleOrAssignableFromStatusOr<T, U> {};
- template <typename T, typename U>
- using IsForwardingAssignmentValid = y_absl::disjunction<
-
- std::is_same<T, y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- y_absl::negation<y_absl::disjunction<
- std::is_same<y_absl::StatusOr<T>,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- std::is_same<y_absl::Status,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- std::is_same<y_absl::in_place_t,
- y_absl::remove_cv_t<y_absl::remove_reference_t<U>>>,
- IsForwardingAssignmentAmbiguous<T, U>>>>;
- class Helper {
- public:
-
- static void HandleInvalidStatusCtorArg(Status*);
- Y_ABSL_ATTRIBUTE_NORETURN static void Crash(const y_absl::Status& status);
- };
- template <typename T, typename... Args>
- Y_ABSL_ATTRIBUTE_NONNULL(1) void PlacementNew(void* p, Args&&... args) {
- new (p) T(std::forward<Args>(args)...);
- }
- template <typename T>
- class StatusOrData {
- template <typename U>
- friend class StatusOrData;
- public:
- StatusOrData() = delete;
- StatusOrData(const StatusOrData& other) {
- if (other.ok()) {
- MakeValue(other.data_);
- MakeStatus();
- } else {
- MakeStatus(other.status_);
- }
- }
- StatusOrData(StatusOrData&& other) noexcept {
- if (other.ok()) {
- MakeValue(std::move(other.data_));
- MakeStatus();
- } else {
- MakeStatus(std::move(other.status_));
- }
- }
- template <typename U>
- explicit StatusOrData(const StatusOrData<U>& other) {
- if (other.ok()) {
- MakeValue(other.data_);
- MakeStatus();
- } else {
- MakeStatus(other.status_);
- }
- }
- template <typename U>
- explicit StatusOrData(StatusOrData<U>&& other) {
- if (other.ok()) {
- MakeValue(std::move(other.data_));
- MakeStatus();
- } else {
- MakeStatus(std::move(other.status_));
- }
- }
- template <typename... Args>
- explicit StatusOrData(y_absl::in_place_t, Args&&... args)
- : data_(std::forward<Args>(args)...) {
- MakeStatus();
- }
- explicit StatusOrData(const T& value) : data_(value) {
- MakeStatus();
- }
- explicit StatusOrData(T&& value) : data_(std::move(value)) {
- MakeStatus();
- }
- template <typename U,
- y_absl::enable_if_t<std::is_constructible<y_absl::Status, U&&>::value,
- int> = 0>
- explicit StatusOrData(U&& v) : status_(std::forward<U>(v)) {
- EnsureNotOk();
- }
- StatusOrData& operator=(const StatusOrData& other) {
- if (this == &other) return *this;
- if (other.ok())
- Assign(other.data_);
- else
- AssignStatus(other.status_);
- return *this;
- }
- StatusOrData& operator=(StatusOrData&& other) {
- if (this == &other) return *this;
- if (other.ok())
- Assign(std::move(other.data_));
- else
- AssignStatus(std::move(other.status_));
- return *this;
- }
- ~StatusOrData() {
- if (ok()) {
- status_.~Status();
- data_.~T();
- } else {
- status_.~Status();
- }
- }
- template <typename U>
- void Assign(U&& value) {
- if (ok()) {
- data_ = std::forward<U>(value);
- } else {
- MakeValue(std::forward<U>(value));
- status_ = OkStatus();
- }
- }
- template <typename U>
- void AssignStatus(U&& v) {
- Clear();
- status_ = static_cast<y_absl::Status>(std::forward<U>(v));
- EnsureNotOk();
- }
- bool ok() const { return status_.ok(); }
- protected:
-
-
-
-
-
- union {
- Status status_;
- };
-
- struct Dummy {};
- union {
-
-
- Dummy dummy_;
- T data_;
- };
- void Clear() {
- if (ok()) data_.~T();
- }
- void EnsureOk() const {
- if (Y_ABSL_PREDICT_FALSE(!ok())) Helper::Crash(status_);
- }
- void EnsureNotOk() {
- if (Y_ABSL_PREDICT_FALSE(ok())) Helper::HandleInvalidStatusCtorArg(&status_);
- }
-
-
- template <typename... Arg>
- void MakeValue(Arg&&... arg) {
- internal_statusor::PlacementNew<T>(&dummy_, std::forward<Arg>(arg)...);
- }
-
-
- template <typename... Args>
- void MakeStatus(Args&&... args) {
- internal_statusor::PlacementNew<Status>(&status_,
- std::forward<Args>(args)...);
- }
- };
- template <typename T, bool = std::is_copy_constructible<T>::value>
- struct CopyCtorBase {
- CopyCtorBase() = default;
- CopyCtorBase(const CopyCtorBase&) = default;
- CopyCtorBase(CopyCtorBase&&) = default;
- CopyCtorBase& operator=(const CopyCtorBase&) = default;
- CopyCtorBase& operator=(CopyCtorBase&&) = default;
- };
- template <typename T>
- struct CopyCtorBase<T, false> {
- CopyCtorBase() = default;
- CopyCtorBase(const CopyCtorBase&) = delete;
- CopyCtorBase(CopyCtorBase&&) = default;
- CopyCtorBase& operator=(const CopyCtorBase&) = default;
- CopyCtorBase& operator=(CopyCtorBase&&) = default;
- };
- template <typename T, bool = std::is_move_constructible<T>::value>
- struct MoveCtorBase {
- MoveCtorBase() = default;
- MoveCtorBase(const MoveCtorBase&) = default;
- MoveCtorBase(MoveCtorBase&&) = default;
- MoveCtorBase& operator=(const MoveCtorBase&) = default;
- MoveCtorBase& operator=(MoveCtorBase&&) = default;
- };
- template <typename T>
- struct MoveCtorBase<T, false> {
- MoveCtorBase() = default;
- MoveCtorBase(const MoveCtorBase&) = default;
- MoveCtorBase(MoveCtorBase&&) = delete;
- MoveCtorBase& operator=(const MoveCtorBase&) = default;
- MoveCtorBase& operator=(MoveCtorBase&&) = default;
- };
- template <typename T, bool = std::is_copy_constructible<T>::value&&
- std::is_copy_assignable<T>::value>
- struct CopyAssignBase {
- CopyAssignBase() = default;
- CopyAssignBase(const CopyAssignBase&) = default;
- CopyAssignBase(CopyAssignBase&&) = default;
- CopyAssignBase& operator=(const CopyAssignBase&) = default;
- CopyAssignBase& operator=(CopyAssignBase&&) = default;
- };
- template <typename T>
- struct CopyAssignBase<T, false> {
- CopyAssignBase() = default;
- CopyAssignBase(const CopyAssignBase&) = default;
- CopyAssignBase(CopyAssignBase&&) = default;
- CopyAssignBase& operator=(const CopyAssignBase&) = delete;
- CopyAssignBase& operator=(CopyAssignBase&&) = default;
- };
- template <typename T, bool = std::is_move_constructible<T>::value&&
- std::is_move_assignable<T>::value>
- struct MoveAssignBase {
- MoveAssignBase() = default;
- MoveAssignBase(const MoveAssignBase&) = default;
- MoveAssignBase(MoveAssignBase&&) = default;
- MoveAssignBase& operator=(const MoveAssignBase&) = default;
- MoveAssignBase& operator=(MoveAssignBase&&) = default;
- };
- template <typename T>
- struct MoveAssignBase<T, false> {
- MoveAssignBase() = default;
- MoveAssignBase(const MoveAssignBase&) = default;
- MoveAssignBase(MoveAssignBase&&) = default;
- MoveAssignBase& operator=(const MoveAssignBase&) = default;
- MoveAssignBase& operator=(MoveAssignBase&&) = delete;
- };
- Y_ABSL_ATTRIBUTE_NORETURN void ThrowBadStatusOrAccess(y_absl::Status status);
- }
- Y_ABSL_NAMESPACE_END
- }
- #endif
|