The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#) https://grpc.io/
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

267 lines
7.1 KiB

// Copyright 2021 gRPC 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
//
// http://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 GRPC_SRC_CORE_LIB_PROMISE_POLL_H
#define GRPC_SRC_CORE_LIB_PROMISE_POLL_H
#include <string>
#include <utility>
#include "absl/log/check.h"
#include "absl/strings/str_format.h"
#include <grpc/support/log.h>
#include <grpc/support/port_platform.h>
#include "src/core/lib/gprpp/construct_destruct.h"
namespace grpc_core {
// A type that signals a Promise is still pending and not yet completed.
// Allows writing 'return Pending{}' and with automatic conversions gets
// upgraded to a Poll<> object.
struct Pending {};
inline bool operator==(const Pending&, const Pending&) { return true; }
// A type that contains no value. Useful for simulating 'void' in promises that
// always need to return some kind of value.
struct Empty {};
inline bool operator==(const Empty&, const Empty&) { return true; }
// The result of polling a Promise once.
//
// Can be either pending - the Promise has not yet completed, or ready -
// indicating that the Promise has completed AND should not be polled again.
template <typename T>
class Poll {
public:
// NOLINTNEXTLINE(google-explicit-constructor)
Poll(Pending) : ready_(false) {}
Poll() : ready_(false) {}
Poll(const Poll& other) : ready_(other.ready_) {
if (ready_) Construct(&value_, other.value_);
}
Poll(Poll&& other) noexcept : ready_(other.ready_) {
if (ready_) Construct(&value_, std::move(other.value_));
}
Poll& operator=(const Poll& other) {
if (ready_) {
if (other.ready_) {
value_ = other.value_;
} else {
Destruct(&value_);
ready_ = false;
}
} else if (other.ready_) {
Construct(&value_, other.value_);
ready_ = true;
}
return *this;
}
Poll& operator=(Poll&& other) noexcept {
if (ready_) {
if (other.ready_) {
value_ = std::move(other.value_);
} else {
Destruct(&value_);
ready_ = false;
}
} else if (other.ready_) {
Construct(&value_, std::move(other.value_));
ready_ = true;
}
return *this;
}
template <typename U>
// NOLINTNEXTLINE(google-explicit-constructor)
Poll(U value) : ready_(true) {
Construct(&value_, std::move(value));
}
// NOLINTNEXTLINE(google-explicit-constructor)
Poll(T&& value) : ready_(true) { Construct(&value_, std::forward<T>(value)); }
~Poll() {
if (ready_) Destruct(&value_);
}
bool pending() const { return !ready_; }
bool ready() const { return ready_; }
T& value() {
DCHECK(ready());
return value_;
}
const T& value() const {
DCHECK(ready());
return value_;
}
T* value_if_ready() {
if (ready()) return &value_;
return nullptr;
}
const T* value_if_ready() const {
if (ready()) return &value_;
return nullptr;
}
private:
// Flag indicating readiness, followed by an optional value.
//
// Why not optional<T>?
//
// We have cases where we want to return absl::nullopt{} from a promise, and
// have that upgraded to a Poll<absl::nullopt_t> prior to a cast to some
// Poll<optional<T>>.
//
// Since optional<nullopt_t> is not allowed, we'd not be allowed to make
// Poll<nullopt_t> and so we'd need to pollute all poll handling code with
// some edge case handling template magic - the complexity would explode and
// grow over time - versus hand coding the pieces we need here and containing
// that quirk to one place.
bool ready_;
// We do a single element union so we can choose when to construct/destruct
// this value.
union {
T value_;
};
};
template <>
class Poll<Empty> {
public:
// NOLINTNEXTLINE(google-explicit-constructor)
Poll(Pending) : ready_(false) {}
Poll() : ready_(false) {}
Poll(const Poll& other) = default;
Poll(Poll&& other) noexcept = default;
Poll& operator=(const Poll& other) = default;
Poll& operator=(Poll&& other) = default;
// NOLINTNEXTLINE(google-explicit-constructor)
Poll(Empty) : ready_(true) {}
~Poll() = default;
bool pending() const { return !ready_; }
bool ready() const { return ready_; }
Empty value() const {
DCHECK(ready());
return Empty{};
}
Empty* value_if_ready() {
static Empty value;
if (ready()) return &value;
return nullptr;
}
const Empty* value_if_ready() const {
static Empty value;
if (ready()) return &value;
return nullptr;
}
private:
// Flag indicating readiness.
bool ready_;
};
// Ensure degenerate cases are not defined:
// Can't poll for a Pending
template <>
class Poll<Pending>;
// Can't poll for a poll
template <class T>
class Poll<Poll<T>>;
// PollTraits tells us whether a type is Poll<> or some other type, and is
// leveraged in the PromiseLike/PromiseFactory machinery to select the
// appropriate implementation of those concepts based upon the return type of a
// lambda, for example (via enable_if).
template <typename T>
struct PollTraits {
using Type = T;
static constexpr bool is_poll() { return false; }
};
template <typename T>
struct PollTraits<Poll<T>> {
using Type = T;
static constexpr bool is_poll() { return true; }
};
template <typename T>
bool operator==(const Poll<T>& a, const Poll<T>& b) {
if (a.pending() && b.pending()) return true;
if (a.ready() && b.ready()) return a.value() == b.value();
return false;
}
template <typename T, typename U, typename SfinaeVoid = void>
struct PollCastImpl;
template <typename T, typename U>
struct PollCastImpl<T, Poll<U>> {
static Poll<T> Cast(Poll<U>&& poll) {
if (poll.pending()) return Pending{};
return static_cast<T>(std::move(poll.value()));
}
};
template <typename T, typename U>
struct PollCastImpl<T, U, std::enable_if<!PollTraits<U>::is_poll()>> {
static Poll<T> Cast(U&& poll) { return Poll<T>(T(std::move(poll))); }
};
template <typename T>
struct PollCastImpl<T, T> {
static Poll<T> Cast(T&& poll) { return Poll<T>(std::move(poll)); }
};
template <typename T>
struct PollCastImpl<T, Poll<T>> {
static Poll<T> Cast(Poll<T>&& poll) { return std::move(poll); }
};
template <typename T, typename U>
Poll<T> poll_cast(U poll) {
return PollCastImpl<T, U>::Cast(std::move(poll));
}
// Convert a poll to a string
template <typename T, typename F>
std::string PollToString(
const Poll<T>& poll,
F t_to_string = [](const T& t) { return t.ToString(); }) {
if (poll.pending()) {
return "<<pending>>";
}
return t_to_string(poll.value());
}
template <typename Sink, typename T>
void AbslStringify(Sink& sink, const Poll<T>& poll) {
if (poll.pending()) {
absl::Format(&sink, "<<pending>>");
return;
}
absl::Format(&sink, "%v", poll.value());
}
} // namespace grpc_core
#endif // GRPC_SRC_CORE_LIB_PROMISE_POLL_H