-- 23500704dd7c2642fad49f88c07ce41ebaab12e4 by Abseil Team <absl-team@google.com>: Change fetch_add() to store(1 + load()) As there is only one concurrent writer to the Info struct, we can avoid using the more expensive fetch_add() function. PiperOrigin-RevId: 329523785 -- 79e5018dba2e117ad89b76367165a604b3f24045 by Abseil Team <absl-team@google.com>: Record rehash count in hashtablez This will help identify which containers could benefit from a reserve call. PiperOrigin-RevId: 329510552 -- e327e54b805d67556f934fa7f7dc2d4e72fa066a by Abseil Team <absl-team@google.com>: Fix -Wsign-compare issues. These lines could theoretically have overflowed in cases of very large stack traces etc. Very unlikely, but this was causing warnings when built with -Wsign-compare, which we intend to enable with Chromium. In none of these three cases is it trivial to switch to a range-based for loop. PiperOrigin-RevId: 329415195 -- 08aca2fc75e8b3ad1201849987b64148fe48f283 by Xiaoyi Zhang <zhangxy@google.com>: Release absl::StatusOr. PiperOrigin-RevId: 329353348 -- bf4d2a7f8b089e2adf14d32b0e39de0a981005c3 by Xiaoyi Zhang <zhangxy@google.com>: Internal change PiperOrigin-RevId: 329337031 -- 42fa7d2fb993bbfc344954227cf1eeb801eca065 by Abseil Team <absl-team@google.com>: Internal change PiperOrigin-RevId: 329099807 GitOrigin-RevId: 23500704dd7c2642fad49f88c07ce41ebaab12e4 Change-Id: I6713e4ca3bb0ab2ced5e487827ae036ab8ac61f1pull/780/head
parent
a4cbb5f698
commit
0e9921b75a
11 changed files with 3031 additions and 4 deletions
@ -0,0 +1,399 @@ |
||||
// Copyright 2020 The Abseil Authors.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// https://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
#ifndef ABSL_STATUS_INTERNAL_STATUSOR_INTERNAL_H_ |
||||
#define ABSL_STATUS_INTERNAL_STATUSOR_INTERNAL_H_ |
||||
|
||||
#include <type_traits> |
||||
#include <utility> |
||||
|
||||
#include "absl/meta/type_traits.h" |
||||
#include "absl/status/status.h" |
||||
#include "absl/utility/utility.h" |
||||
|
||||
namespace absl { |
||||
ABSL_NAMESPACE_BEGIN |
||||
|
||||
template <typename T> |
||||
class ABSL_MUST_USE_RESULT StatusOr; |
||||
|
||||
namespace internal_statusor { |
||||
|
||||
// Detects whether `U` has conversion operator to `StatusOr<T>`, i.e. `operator
|
||||
// StatusOr<T>()`.
|
||||
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 absl::StatusOr<T>())]); |
||||
|
||||
template <typename T, typename U> |
||||
struct HasConversionOperatorToStatusOr<T, U, decltype(test<T, U>(0))> |
||||
: std::true_type {}; |
||||
|
||||
// Detects whether `T` is constructible or convertible from `StatusOr<U>`.
|
||||
template <typename T, typename U> |
||||
using IsConstructibleOrConvertibleFromStatusOr = |
||||
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>>; |
||||
|
||||
// Detects whether `T` is constructible or convertible or assignable from
|
||||
// `StatusOr<U>`.
|
||||
template <typename T, typename U> |
||||
using IsConstructibleOrConvertibleOrAssignableFromStatusOr = |
||||
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>&&>>; |
||||
|
||||
// Detects whether direct initializing `StatusOr<T>` from `U` is ambiguous, i.e.
|
||||
// when `U` is `StatusOr<V>` and `T` is constructible or convertible from `V`.
|
||||
template <typename T, typename U> |
||||
struct IsDirectInitializationAmbiguous |
||||
: public absl::conditional_t< |
||||
std::is_same<absl::remove_cv_t<absl::remove_reference_t<U>>, |
||||
U>::value, |
||||
std::false_type, |
||||
IsDirectInitializationAmbiguous< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>> {}; |
||||
|
||||
template <typename T, typename V> |
||||
struct IsDirectInitializationAmbiguous<T, absl::StatusOr<V>> |
||||
: public IsConstructibleOrConvertibleFromStatusOr<T, V> {}; |
||||
|
||||
// Checks against the constraints of the direction initialization, i.e. when
|
||||
// `StatusOr<T>::StatusOr(U&&)` should participate in overload resolution.
|
||||
template <typename T, typename U> |
||||
using IsDirectInitializationValid = absl::disjunction< |
||||
// Short circuits if T is basically U.
|
||||
std::is_same<T, absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
absl::negation<absl::disjunction< |
||||
std::is_same<absl::StatusOr<T>, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
std::is_same<absl::Status, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
std::is_same<absl::in_place_t, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
IsDirectInitializationAmbiguous<T, U>>>>; |
||||
|
||||
// This trait detects whether `StatusOr<T>::operator=(U&&)` is ambiguous, which
|
||||
// is equivalent to whether all the following conditions are met:
|
||||
// 1. `U` is `StatusOr<V>`.
|
||||
// 2. `T` is constructible and assignable from `V`.
|
||||
// 3. `T` is constructible and assignable from `U` (i.e. `StatusOr<V>`).
|
||||
// For example, the following code is considered ambiguous:
|
||||
// (`T` is `bool`, `U` is `StatusOr<bool>`, `V` is `bool`)
|
||||
// StatusOr<bool> s1 = true; // s1.ok() && s1.ValueOrDie() == true
|
||||
// StatusOr<bool> s2 = false; // s2.ok() && s2.ValueOrDie() == false
|
||||
// s1 = s2; // ambiguous, `s1 = s2.ValueOrDie()` or `s1 = bool(s2)`?
|
||||
template <typename T, typename U> |
||||
struct IsForwardingAssignmentAmbiguous |
||||
: public absl::conditional_t< |
||||
std::is_same<absl::remove_cv_t<absl::remove_reference_t<U>>, |
||||
U>::value, |
||||
std::false_type, |
||||
IsForwardingAssignmentAmbiguous< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>> {}; |
||||
|
||||
template <typename T, typename U> |
||||
struct IsForwardingAssignmentAmbiguous<T, absl::StatusOr<U>> |
||||
: public IsConstructibleOrConvertibleOrAssignableFromStatusOr<T, U> {}; |
||||
|
||||
// Checks against the constraints of the forwarding assignment, i.e. whether
|
||||
// `StatusOr<T>::operator(U&&)` should participate in overload resolution.
|
||||
template <typename T, typename U> |
||||
using IsForwardingAssignmentValid = absl::disjunction< |
||||
// Short circuits if T is basically U.
|
||||
std::is_same<T, absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
absl::negation<absl::disjunction< |
||||
std::is_same<absl::StatusOr<T>, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
std::is_same<absl::Status, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
std::is_same<absl::in_place_t, |
||||
absl::remove_cv_t<absl::remove_reference_t<U>>>, |
||||
IsForwardingAssignmentAmbiguous<T, U>>>>; |
||||
|
||||
class Helper { |
||||
public: |
||||
// Move type-agnostic error handling to the .cc.
|
||||
static void HandleInvalidStatusCtorArg(Status*); |
||||
static void Crash(const absl::Status& status); |
||||
}; |
||||
|
||||
// Construct an instance of T in `p` through placement new, passing Args... to
|
||||
// the constructor.
|
||||
// This abstraction is here mostly for the gcc performance fix.
|
||||
template <typename T, typename... Args> |
||||
void PlacementNew(void* p, Args&&... args) { |
||||
#if defined(__GNUC__) && !defined(__clang__) |
||||
// Teach gcc that 'p' cannot be null, fixing code size issues.
|
||||
if (p == nullptr) __builtin_unreachable(); |
||||
#endif |
||||
new (p) T(std::forward<Args>(args)...); |
||||
} |
||||
|
||||
// Helper base class to hold the data and all operations.
|
||||
// We move all this to a base class to allow mixing with the appropriate
|
||||
// TraitsBase specialization.
|
||||
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(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, |
||||
absl::enable_if_t<std::is_constructible<absl::Status, U&&>::value, |
||||
int> = 0> |
||||
explicit StatusOrData(U&& v) : status_(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<absl::Status>(std::forward<U>(v)); |
||||
EnsureNotOk(); |
||||
} |
||||
|
||||
bool ok() const { return status_.ok(); } |
||||
|
||||
protected: |
||||
// status_ will always be active after the constructor.
|
||||
// We make it a union to be able to initialize exactly how we need without
|
||||
// waste.
|
||||
// Eg. in the copy constructor we use the default constructor of Status in
|
||||
// the ok() path to avoid an extra Ref call.
|
||||
union { |
||||
Status status_; |
||||
}; |
||||
|
||||
// data_ is active iff status_.ok()==true
|
||||
struct Dummy {}; |
||||
union { |
||||
// When T is const, we need some non-const object we can cast to void* for
|
||||
// the placement new. dummy_ is that object.
|
||||
Dummy dummy_; |
||||
T data_; |
||||
}; |
||||
|
||||
void Clear() { |
||||
if (ok()) data_.~T(); |
||||
} |
||||
|
||||
void EnsureOk() const { |
||||
if (ABSL_PREDICT_FALSE(!ok())) Helper::Crash(status_); |
||||
} |
||||
|
||||
void EnsureNotOk() { |
||||
if (ABSL_PREDICT_FALSE(ok())) Helper::HandleInvalidStatusCtorArg(&status_); |
||||
} |
||||
|
||||
// Construct the value (ie. data_) through placement new with the passed
|
||||
// argument.
|
||||
template <typename... Arg> |
||||
void MakeValue(Arg&&... arg) { |
||||
internal_statusor::PlacementNew<T>(&dummy_, std::forward<Arg>(arg)...); |
||||
} |
||||
|
||||
// Construct the status (ie. status_) through placement new with the passed
|
||||
// argument.
|
||||
template <typename... Args> |
||||
void MakeStatus(Args&&... args) { |
||||
internal_statusor::PlacementNew<Status>(&status_, |
||||
std::forward<Args>(args)...); |
||||
} |
||||
}; |
||||
|
||||
// Helper base classes to allow implicitly deleted constructors and assignment
|
||||
// operators in `StatusOr`. For example, `CopyCtorBase` will explicitly delete
|
||||
// the copy constructor when T is not copy constructible and `StatusOr` will
|
||||
// inherit that behavior implicitly.
|
||||
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; |
||||
}; |
||||
|
||||
ABSL_ATTRIBUTE_NORETURN void ThrowBadStatusOrAccess(absl::Status status); |
||||
|
||||
} // namespace internal_statusor
|
||||
ABSL_NAMESPACE_END |
||||
} // namespace absl
|
||||
|
||||
#endif // ABSL_STATUS_INTERNAL_STATUSOR_INTERNAL_H_
|
@ -0,0 +1,71 @@ |
||||
// Copyright 2020 The Abseil Authors.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// https://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
#include "absl/status/statusor.h" |
||||
|
||||
#include <cstdlib> |
||||
#include <utility> |
||||
|
||||
#include "absl/base/internal/raw_logging.h" |
||||
#include "absl/status/status.h" |
||||
#include "absl/strings/str_cat.h" |
||||
|
||||
namespace absl { |
||||
ABSL_NAMESPACE_BEGIN |
||||
|
||||
BadStatusOrAccess::BadStatusOrAccess(absl::Status status) |
||||
: status_(std::move(status)) {} |
||||
|
||||
BadStatusOrAccess::~BadStatusOrAccess() = default; |
||||
const char* BadStatusOrAccess::what() const noexcept { |
||||
return "Bad StatusOr access"; |
||||
} |
||||
|
||||
const absl::Status& BadStatusOrAccess::status() const { return status_; } |
||||
|
||||
namespace internal_statusor { |
||||
|
||||
void Helper::HandleInvalidStatusCtorArg(absl::Status* status) { |
||||
const char* kMessage = |
||||
"An OK status is not a valid constructor argument to StatusOr<T>"; |
||||
#ifdef NDEBUG |
||||
ABSL_INTERNAL_LOG(ERROR, kMessage); |
||||
#else |
||||
ABSL_INTERNAL_LOG(FATAL, kMessage); |
||||
#endif |
||||
// In optimized builds, we will fall back to InternalError.
|
||||
*status = absl::InternalError(kMessage); |
||||
} |
||||
|
||||
void Helper::Crash(const absl::Status& status) { |
||||
ABSL_INTERNAL_LOG( |
||||
FATAL, |
||||
absl::StrCat("Attempting to fetch value instead of handling error ", |
||||
status.ToString())); |
||||
} |
||||
|
||||
void ThrowBadStatusOrAccess(absl::Status status) { |
||||
#ifdef ABSL_HAVE_EXCEPTIONS |
||||
throw absl::BadStatusOrAccess(std::move(status)); |
||||
#else |
||||
ABSL_INTERNAL_LOG( |
||||
FATAL, |
||||
absl::StrCat("Attempting to fetch value instead of handling error ", |
||||
status.ToString())); |
||||
std::abort(); |
||||
#endif |
||||
} |
||||
|
||||
} // namespace internal_statusor
|
||||
ABSL_NAMESPACE_END |
||||
} // namespace absl
|
@ -0,0 +1,670 @@ |
||||
// Copyright 2020 The Abseil Authors.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||
// you may not use this file except in compliance with the License.
|
||||
// You may obtain a copy of the License at
|
||||
//
|
||||
// https://www.apache.org/licenses/LICENSE-2.0
|
||||
//
|
||||
// Unless required by applicable law or agreed to in writing, software
|
||||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
//
|
||||
// StatusOr<T> is the union of a Status object and a T
|
||||
// object. StatusOr models the concept of an object that is either a
|
||||
// usable value, or an error Status explaining why such a value is
|
||||
// not present. To this end, StatusOr<T> does not allow its Status
|
||||
// value to be absl::OkStatus().
|
||||
//
|
||||
// The primary use-case for StatusOr<T> is as the return value of a
|
||||
// function which may fail.
|
||||
//
|
||||
// Example usage of a StatusOr<T>:
|
||||
//
|
||||
// StatusOr<Foo> result = DoBigCalculationThatCouldFail();
|
||||
// if (result.ok()) {
|
||||
// result->DoSomethingCool();
|
||||
// } else {
|
||||
// LOG(ERROR) << result.status();
|
||||
// }
|
||||
//
|
||||
// Example that is guaranteed to crash if the result holds no value:
|
||||
//
|
||||
// StatusOr<Foo> result = DoBigCalculationThatCouldFail();
|
||||
// const Foo& foo = result.value();
|
||||
// foo.DoSomethingCool();
|
||||
//
|
||||
// Example usage of a StatusOr<std::unique_ptr<T>>:
|
||||
//
|
||||
// StatusOr<std::unique_ptr<Foo>> result = FooFactory::MakeNewFoo(arg);
|
||||
// if (!result.ok()) { // Don't omit .ok()
|
||||
// LOG(ERROR) << result.status();
|
||||
// } else if (*result == nullptr) {
|
||||
// LOG(ERROR) << "Unexpected null pointer";
|
||||
// } else {
|
||||
// (*result)->DoSomethingCool();
|
||||
// }
|
||||
//
|
||||
// Example factory implementation returning StatusOr<T>:
|
||||
//
|
||||
// StatusOr<Foo> FooFactory::MakeFoo(int arg) {
|
||||
// if (arg <= 0) {
|
||||
// return absl::Status(absl::StatusCode::kInvalidArgument,
|
||||
// "Arg must be positive");
|
||||
// }
|
||||
// return Foo(arg);
|
||||
// }
|
||||
//
|
||||
// NULL POINTERS
|
||||
//
|
||||
// Historically StatusOr<T*> treated null pointers specially. This is no longer
|
||||
// true -- a StatusOr<T*> can be constructed from a null pointer like any other
|
||||
// pointer value, and the result will be that ok() returns true and value()
|
||||
// returns null.
|
||||
|
||||
#ifndef ABSL_STATUS_STATUSOR_H_ |
||||
#define ABSL_STATUS_STATUSOR_H_ |
||||
|
||||
#include <exception> |
||||
#include <initializer_list> |
||||
#include <new> |
||||
#include <string> |
||||
#include <type_traits> |
||||
#include <utility> |
||||
|
||||
#include "absl/base/attributes.h" |
||||
#include "absl/meta/type_traits.h" |
||||
#include "absl/status/internal/statusor_internal.h" |
||||
#include "absl/status/status.h" |
||||
#include "absl/types/variant.h" |
||||
#include "absl/utility/utility.h" |
||||
|
||||
namespace absl { |
||||
ABSL_NAMESPACE_BEGIN |
||||
class BadStatusOrAccess : public std::exception { |
||||
public: |
||||
explicit BadStatusOrAccess(absl::Status status); |
||||
~BadStatusOrAccess() override; |
||||
const char* what() const noexcept override; |
||||
const absl::Status& status() const; |
||||
|
||||
private: |
||||
absl::Status status_; |
||||
}; |
||||
|
||||
// Returned StatusOr objects may not be ignored.
|
||||
template <typename T> |
||||
class ABSL_MUST_USE_RESULT StatusOr; |
||||
|
||||
template <typename T> |
||||
class StatusOr : private internal_statusor::StatusOrData<T>, |
||||
private internal_statusor::CopyCtorBase<T>, |
||||
private internal_statusor::MoveCtorBase<T>, |
||||
private internal_statusor::CopyAssignBase<T>, |
||||
private internal_statusor::MoveAssignBase<T> { |
||||
template <typename U> |
||||
friend class StatusOr; |
||||
|
||||
typedef internal_statusor::StatusOrData<T> Base; |
||||
|
||||
public: |
||||
typedef T value_type; |
||||
|
||||
// Constructs a new StatusOr with Status::UNKNOWN status. This is marked
|
||||
// 'explicit' to try to catch cases like 'return {};', where people think
|
||||
// absl::StatusOr<std::vector<int>> will be initialized with an empty vector,
|
||||
// instead of a Status::UNKNOWN status.
|
||||
explicit StatusOr(); |
||||
|
||||
// StatusOr<T> is copy constructible if T is copy constructible.
|
||||
StatusOr(const StatusOr&) = default; |
||||
// StatusOr<T> is copy assignable if T is copy constructible and copy
|
||||
// assignable.
|
||||
StatusOr& operator=(const StatusOr&) = default; |
||||
|
||||
// StatusOr<T> is move constructible if T is move constructible.
|
||||
StatusOr(StatusOr&&) = default; |
||||
// StatusOr<T> is moveAssignable if T is move constructible and move
|
||||
// assignable.
|
||||
StatusOr& operator=(StatusOr&&) = default; |
||||
|
||||
// Converting constructors from StatusOr<U>, when T is constructible from U.
|
||||
// To avoid ambiguity, they are disabled if T is also constructible from
|
||||
// StatusOr<U>. Explicit iff the corresponding construction of T from U is
|
||||
// explicit.
|
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, |
||||
std::is_constructible<T, const U&>, |
||||
std::is_convertible<const U&, T>, |
||||
absl::negation< |
||||
internal_statusor::IsConstructibleOrConvertibleFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
StatusOr(const StatusOr<U>& other) // NOLINT
|
||||
: Base(static_cast<const typename StatusOr<U>::Base&>(other)) {} |
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, |
||||
std::is_constructible<T, const U&>, |
||||
absl::negation<std::is_convertible<const U&, T>>, |
||||
absl::negation< |
||||
internal_statusor::IsConstructibleOrConvertibleFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
explicit StatusOr(const StatusOr<U>& other) |
||||
: Base(static_cast<const typename StatusOr<U>::Base&>(other)) {} |
||||
|
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, std::is_constructible<T, U&&>, |
||||
std::is_convertible<U&&, T>, |
||||
absl::negation< |
||||
internal_statusor::IsConstructibleOrConvertibleFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
StatusOr(StatusOr<U>&& other) // NOLINT
|
||||
: Base(static_cast<typename StatusOr<U>::Base&&>(other)) {} |
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, std::is_constructible<T, U&&>, |
||||
absl::negation<std::is_convertible<U&&, T>>, |
||||
absl::negation< |
||||
internal_statusor::IsConstructibleOrConvertibleFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
explicit StatusOr(StatusOr<U>&& other) |
||||
: Base(static_cast<typename StatusOr<U>::Base&&>(other)) {} |
||||
|
||||
// Conversion copy/move assignment operator, T must be constructible and
|
||||
// assignable from U. Only enable if T cannot be directly assigned from
|
||||
// StatusOr<U>.
|
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, |
||||
std::is_constructible<T, const U&>, |
||||
std::is_assignable<T, const U&>, |
||||
absl::negation< |
||||
internal_statusor:: |
||||
IsConstructibleOrConvertibleOrAssignableFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
StatusOr& operator=(const StatusOr<U>& other) { |
||||
this->Assign(other); |
||||
return *this; |
||||
} |
||||
template < |
||||
typename U, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_same<T, U>>, std::is_constructible<T, U&&>, |
||||
std::is_assignable<T, U&&>, |
||||
absl::negation< |
||||
internal_statusor:: |
||||
IsConstructibleOrConvertibleOrAssignableFromStatusOr< |
||||
T, U>>>::value, |
||||
int> = 0> |
||||
StatusOr& operator=(StatusOr<U>&& other) { |
||||
this->Assign(std::move(other)); |
||||
return *this; |
||||
} |
||||
|
||||
// Constructs a new StatusOr with a non-ok status. After calling this
|
||||
// constructor, this->ok() will be false and calls to value() will CHECK-fail.
|
||||
// The constructor also takes any type `U` that is convertible to `Status`.
|
||||
//
|
||||
// NOTE: Not explicit - we want to use StatusOr<T> as a return
|
||||
// value, so it is convenient and sensible to be able to do
|
||||
// `return Status()` or `return ConvertibleToStatus()` when the return type
|
||||
// is `StatusOr<T>`.
|
||||
//
|
||||
// REQUIRES: !Status(std::forward<U>(v)).ok(). This requirement is DCHECKed.
|
||||
// In optimized builds, passing absl::OkStatus() here will have the effect
|
||||
// of passing absl::StatusCode::kInternal as a fallback.
|
||||
template < |
||||
typename U = absl::Status, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
std::is_convertible<U&&, absl::Status>, |
||||
std::is_constructible<absl::Status, U&&>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::StatusOr<T>>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, T>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::in_place_t>>, |
||||
absl::negation<internal_statusor::HasConversionOperatorToStatusOr< |
||||
T, U&&>>>::value, |
||||
int> = 0> |
||||
StatusOr(U&& v) : Base(std::forward<U>(v)) {} |
||||
|
||||
template < |
||||
typename U = absl::Status, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
absl::negation<std::is_convertible<U&&, absl::Status>>, |
||||
std::is_constructible<absl::Status, U&&>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::StatusOr<T>>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, T>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::in_place_t>>, |
||||
absl::negation<internal_statusor::HasConversionOperatorToStatusOr< |
||||
T, U&&>>>::value, |
||||
int> = 0> |
||||
explicit StatusOr(U&& v) : Base(std::forward<U>(v)) {} |
||||
|
||||
template < |
||||
typename U = absl::Status, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
std::is_convertible<U&&, absl::Status>, |
||||
std::is_constructible<absl::Status, U&&>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::StatusOr<T>>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, T>>, |
||||
absl::negation<std::is_same<absl::decay_t<U>, absl::in_place_t>>, |
||||
absl::negation<internal_statusor::HasConversionOperatorToStatusOr< |
||||
T, U&&>>>::value, |
||||
int> = 0> |
||||
StatusOr& operator=(U&& v) { |
||||
this->AssignStatus(std::forward<U>(v)); |
||||
return *this; |
||||
} |
||||
|
||||
// Perfect-forwarding value assignment operator.
|
||||
// If `*this` contains a `T` value before the call, the contained value is
|
||||
// assigned from `std::forward<U>(v)`; Otherwise, it is directly-initialized
|
||||
// from `std::forward<U>(v)`.
|
||||
// This function does not participate in overload unless:
|
||||
// 1. `std::is_constructible_v<T, U>` is true,
|
||||
// 2. `std::is_assignable_v<T&, U>` is true.
|
||||
// 3. `std::is_same_v<StatusOr<T>, std::remove_cvref_t<U>>` is false.
|
||||
// 4. Assigning `U` to `T` is not ambiguous:
|
||||
// If `U` is `StatusOr<V>` and `T` is constructible and assignable from
|
||||
// both `StatusOr<V>` and `V`, the assignment is considered bug-prone and
|
||||
// ambiguous thus will fail to compile. For example:
|
||||
// StatusOr<bool> s1 = true; // s1.ok() && *s1 == true
|
||||
// StatusOr<bool> s2 = false; // s2.ok() && *s2 == false
|
||||
// s1 = s2; // ambiguous, `s1 = *s2` or `s1 = bool(s2)`?
|
||||
template < |
||||
typename U = T, |
||||
typename = typename std::enable_if<absl::conjunction< |
||||
std::is_constructible<T, U&&>, std::is_assignable<T&, U&&>, |
||||
absl::disjunction< |
||||
std::is_same<absl::remove_cv_t<absl::remove_reference_t<U>>, T>, |
||||
absl::conjunction< |
||||
absl::negation<std::is_convertible<U&&, absl::Status>>, |
||||
absl::negation<internal_statusor:: |
||||
HasConversionOperatorToStatusOr<T, U&&>>>>, |
||||
internal_statusor::IsForwardingAssignmentValid<T, U&&>>::value>::type> |
||||
StatusOr& operator=(U&& v) { |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_constructible<T, U&&>, std::is_assignable<T&, U&&>, |
||||
std::is_constructible<absl::Status, U&&>, |
||||
std::is_assignable<absl::Status&, U&&>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U can assign to both T and Status, will result in semantic change"); |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_constructible<T, U&&>, std::is_assignable<T&, U&&>, |
||||
internal_statusor::HasConversionOperatorToStatusOr<T, U&&>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U can assign to T and convert to StatusOr<T>, will result in semantic " |
||||
"change"); |
||||
this->Assign(std::forward<U>(v)); |
||||
return *this; |
||||
} |
||||
|
||||
// Constructs the inner value T in-place using the provided args, using the
|
||||
// T(args...) constructor.
|
||||
template <typename... Args> |
||||
explicit StatusOr(absl::in_place_t, Args&&... args); |
||||
template <typename U, typename... Args> |
||||
explicit StatusOr(absl::in_place_t, std::initializer_list<U> ilist, |
||||
Args&&... args); |
||||
|
||||
// Constructs the inner value T in-place using the provided args, using the
|
||||
// T(U) (direct-initialization) constructor. Only valid if T can be
|
||||
// constructed from a U. Can accept move or copy constructors. Explicit if
|
||||
// U is not convertible to T. To avoid ambiguity, this is disabled if U is
|
||||
// a StatusOr<J>, where J is convertible to T.
|
||||
template < |
||||
typename U = T, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
internal_statusor::IsDirectInitializationValid<T, U&&>, |
||||
std::is_constructible<T, U&&>, std::is_convertible<U&&, T>, |
||||
absl::disjunction< |
||||
std::is_same<absl::remove_cv_t<absl::remove_reference_t<U>>, |
||||
T>, |
||||
absl::conjunction< |
||||
absl::negation<std::is_convertible<U&&, absl::Status>>, |
||||
absl::negation< |
||||
internal_statusor::HasConversionOperatorToStatusOr< |
||||
T, U&&>>>>>::value, |
||||
int> = 0> |
||||
StatusOr(U&& u) // NOLINT
|
||||
: StatusOr(absl::in_place, std::forward<U>(u)) { |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_convertible<U&&, T>, std::is_convertible<U&&, absl::Status>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U is convertible to both T and Status, will result in semantic " |
||||
"change"); |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_convertible<U&&, T>, |
||||
internal_statusor::HasConversionOperatorToStatusOr<T, U&&>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U can construct T and convert to StatusOr<T>, will result in semantic " |
||||
"change"); |
||||
} |
||||
|
||||
template < |
||||
typename U = T, |
||||
absl::enable_if_t< |
||||
absl::conjunction< |
||||
internal_statusor::IsDirectInitializationValid<T, U&&>, |
||||
absl::disjunction< |
||||
std::is_same<absl::remove_cv_t<absl::remove_reference_t<U>>, |
||||
T>, |
||||
absl::conjunction< |
||||
absl::negation<std::is_constructible<absl::Status, U&&>>, |
||||
absl::negation< |
||||
internal_statusor::HasConversionOperatorToStatusOr< |
||||
T, U&&>>>>, |
||||
std::is_constructible<T, U&&>, |
||||
absl::negation<std::is_convertible<U&&, T>>>::value, |
||||
int> = 0> |
||||
explicit StatusOr(U&& u) // NOLINT
|
||||
: StatusOr(absl::in_place, std::forward<U>(u)) { |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_constructible<T, U&&>, |
||||
std::is_constructible<absl::Status, U&&>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U can construct both T and Status, will result in semantic " |
||||
"change"); |
||||
static_assert( |
||||
!absl::conjunction< |
||||
std::is_constructible<T, U&&>, |
||||
internal_statusor::HasConversionOperatorToStatusOr<T, U&&>, |
||||
absl::negation<std::is_same< |
||||
T, absl::remove_cv_t<absl::remove_reference_t<U>>>>>::value, |
||||
"U can construct T and convert to StatusOr<T>, will result in semantic " |
||||
"change"); |
||||
} |
||||
|
||||
// Returns this->status().ok()
|
||||
ABSL_MUST_USE_RESULT bool ok() const { return this->status_.ok(); } |
||||
|
||||
// Returns a reference to our status. If this contains a T, then
|
||||
// returns absl::OkStatus().
|
||||
const Status& status() const &; |
||||
Status status() &&; |
||||
|
||||
// Returns a reference to the held value if `this->ok()`. Otherwise, throws
|
||||
// `absl::BadStatusOrAccess` if exception is enabled, or `LOG(FATAL)` if
|
||||
// exception is disabled.
|
||||
// If you have already checked the status using `this->ok()`, you probably
|
||||
// want to use `operator*()` or `operator->()` to access the value instead of
|
||||
// `value`.
|
||||
// Note: for value types that are cheap to copy, prefer simple code:
|
||||
//
|
||||
// T value = statusor.value();
|
||||
//
|
||||
// Otherwise, if the value type is expensive to copy, but can be left
|
||||
// in the StatusOr, simply assign to a reference:
|
||||
//
|
||||
// T& value = statusor.value(); // or `const T&`
|
||||
//
|
||||
// Otherwise, if the value type supports an efficient move, it can be
|
||||
// used as follows:
|
||||
//
|
||||
// T value = std::move(statusor).value();
|
||||
//
|
||||
// The `std::move` on statusor instead of on the whole expression enables
|
||||
// warnings about possible uses of the statusor object after the move.
|
||||
const T& value() const&; |
||||
T& value() &; |
||||
const T&& value() const&&; |
||||
T&& value() &&; |
||||
|
||||
// Returns a reference to the current value.
|
||||
//
|
||||
// REQUIRES: this->ok() == true, otherwise the behavior is undefined.
|
||||
//
|
||||
// Use this->ok() to verify that there is a current value.
|
||||
// Alternatively, see value() for a similar API that guarantees
|
||||
// CHECK-failing if there is no current value.
|
||||
const T& operator*() const&; |
||||
T& operator*() &; |
||||
const T&& operator*() const&&; |
||||
T&& operator*() &&; |
||||
|
||||
// Returns a pointer to the current value.
|
||||
//
|
||||
// REQUIRES: this->ok() == true, otherwise the behavior is undefined.
|
||||
//
|
||||
// Use this->ok() to verify that there is a current value.
|
||||
const T* operator->() const; |
||||
T* operator->(); |
||||
|
||||
// Returns the current value this->ok() == true. Otherwise constructs a value
|
||||
// using `default_value`.
|
||||
//
|
||||
// Unlike `value`, this function returns by value, copying the current value
|
||||
// if necessary. If the value type supports an efficient move, it can be used
|
||||
// as follows:
|
||||
//
|
||||
// T value = std::move(statusor).value_or(def);
|
||||
//
|
||||
// Unlike with `value`, calling `std::move` on the result of `value_or` will
|
||||
// still trigger a copy.
|
||||
template <typename U> |
||||
T value_or(U&& default_value) const&; |
||||
template <typename U> |
||||
T value_or(U&& default_value) &&; |
||||
|
||||
// Ignores any errors. This method does nothing except potentially suppress
|
||||
// complaints from any tools that are checking that errors are not dropped on
|
||||
// the floor.
|
||||
void IgnoreError() const; |
||||
|
||||
// Reconstructs the inner value T in-place using the provided args, using the
|
||||
// T(args...) constructor. Returns reference to the reconstructed `T`.
|
||||
template <typename... Args> |
||||
T& emplace(Args&&... args) { |
||||
if (ok()) { |
||||
this->Clear(); |
||||
this->MakeValue(std::forward<Args>(args)...); |
||||
} else { |
||||
this->MakeValue(std::forward<Args>(args)...); |
||||
this->status_ = absl::OkStatus(); |
||||
} |
||||
return this->data_; |
||||
} |
||||
|
||||
template < |
||||
typename U, typename... Args, |
||||
absl::enable_if_t< |
||||
std::is_constructible<T, std::initializer_list<U>&, Args&&...>::value, |
||||
int> = 0> |
||||
T& emplace(std::initializer_list<U> ilist, Args&&... args) { |
||||
if (ok()) { |
||||
this->Clear(); |
||||
this->MakeValue(ilist, std::forward<Args>(args)...); |
||||
} else { |
||||
this->MakeValue(ilist, std::forward<Args>(args)...); |
||||
this->status_ = absl::OkStatus(); |
||||
} |
||||
return this->data_; |
||||
} |
||||
|
||||
private: |
||||
using internal_statusor::StatusOrData<T>::Assign; |
||||
template <typename U> |
||||
void Assign(const absl::StatusOr<U>& other); |
||||
template <typename U> |
||||
void Assign(absl::StatusOr<U>&& other); |
||||
}; |
||||
|
||||
template <typename T> |
||||
bool operator==(const StatusOr<T>& lhs, const StatusOr<T>& rhs) { |
||||
if (lhs.ok() && rhs.ok()) return *lhs == *rhs; |
||||
return lhs.status() == rhs.status(); |
||||
} |
||||
|
||||
template <typename T> |
||||
bool operator!=(const StatusOr<T>& lhs, const StatusOr<T>& rhs) { |
||||
return !(lhs == rhs); |
||||
} |
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// Implementation details for StatusOr<T>
|
||||
|
||||
// TODO(sbenza): avoid the string here completely.
|
||||
template <typename T> |
||||
StatusOr<T>::StatusOr() : Base(Status(absl::StatusCode::kUnknown, "")) {} |
||||
|
||||
template <typename T> |
||||
template <typename U> |
||||
inline void StatusOr<T>::Assign(const StatusOr<U>& other) { |
||||
if (other.ok()) { |
||||
this->Assign(*other); |
||||
} else { |
||||
this->AssignStatus(other.status()); |
||||
} |
||||
} |
||||
|
||||
template <typename T> |
||||
template <typename U> |
||||
inline void StatusOr<T>::Assign(StatusOr<U>&& other) { |
||||
if (other.ok()) { |
||||
this->Assign(*std::move(other)); |
||||
} else { |
||||
this->AssignStatus(std::move(other).status()); |
||||
} |
||||
} |
||||
template <typename T> |
||||
template <typename... Args> |
||||
StatusOr<T>::StatusOr(absl::in_place_t, Args&&... args) |
||||
: Base(absl::in_place, std::forward<Args>(args)...) {} |
||||
|
||||
template <typename T> |
||||
template <typename U, typename... Args> |
||||
StatusOr<T>::StatusOr(absl::in_place_t, std::initializer_list<U> ilist, |
||||
Args&&... args) |
||||
: Base(absl::in_place, ilist, std::forward<Args>(args)...) {} |
||||
|
||||
template <typename T> |
||||
const Status& StatusOr<T>::status() const & { return this->status_; } |
||||
template <typename T> |
||||
Status StatusOr<T>::status() && { |
||||
return ok() ? OkStatus() : std::move(this->status_); |
||||
} |
||||
|
||||
template <typename T> |
||||
const T& StatusOr<T>::value() const& { |
||||
if (!this->ok()) internal_statusor::ThrowBadStatusOrAccess(this->status_); |
||||
return this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
T& StatusOr<T>::value() & { |
||||
if (!this->ok()) internal_statusor::ThrowBadStatusOrAccess(this->status_); |
||||
return this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
const T&& StatusOr<T>::value() const&& { |
||||
if (!this->ok()) { |
||||
internal_statusor::ThrowBadStatusOrAccess(std::move(this->status_)); |
||||
} |
||||
return std::move(this->data_); |
||||
} |
||||
|
||||
template <typename T> |
||||
T&& StatusOr<T>::value() && { |
||||
if (!this->ok()) { |
||||
internal_statusor::ThrowBadStatusOrAccess(std::move(this->status_)); |
||||
} |
||||
return std::move(this->data_); |
||||
} |
||||
|
||||
template <typename T> |
||||
const T& StatusOr<T>::operator*() const& { |
||||
this->EnsureOk(); |
||||
return this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
T& StatusOr<T>::operator*() & { |
||||
this->EnsureOk(); |
||||
return this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
const T&& StatusOr<T>::operator*() const&& { |
||||
this->EnsureOk(); |
||||
return std::move(this->data_); |
||||
} |
||||
|
||||
template <typename T> |
||||
T&& StatusOr<T>::operator*() && { |
||||
this->EnsureOk(); |
||||
return std::move(this->data_); |
||||
} |
||||
|
||||
template <typename T> |
||||
const T* StatusOr<T>::operator->() const { |
||||
this->EnsureOk(); |
||||
return &this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
T* StatusOr<T>::operator->() { |
||||
this->EnsureOk(); |
||||
return &this->data_; |
||||
} |
||||
|
||||
template <typename T> |
||||
template <typename U> |
||||
T StatusOr<T>::value_or(U&& default_value) const& { |
||||
if (ok()) { |
||||
return this->data_; |
||||
} |
||||
return std::forward<U>(default_value); |
||||
} |
||||
|
||||
template <typename T> |
||||
template <typename U> |
||||
T StatusOr<T>::value_or(U&& default_value) && { |
||||
if (ok()) { |
||||
return std::move(this->data_); |
||||
} |
||||
return std::forward<U>(default_value); |
||||
} |
||||
|
||||
template <typename T> |
||||
void StatusOr<T>::IgnoreError() const { |
||||
// no-op
|
||||
} |
||||
|
||||
ABSL_NAMESPACE_END |
||||
} // namespace absl
|
||||
|
||||
#endif // ABSL_STATUS_STATUSOR_H_
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in new issue