blob: 174c640992620a2053e15df6607da674fb3d12a8 [file] [log] [blame]
// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef IPCZ_SRC_IPCZ_PORTAL_H_
#define IPCZ_SRC_IPCZ_PORTAL_H_
#include <cstdint>
#include <utility>
#include "ipcz/api_object.h"
#include "ipcz/ipcz.h"
#include "ipcz/parcel.h"
#include "third_party/abseil-cpp/absl/container/flat_hash_map.h"
#include "third_party/abseil-cpp/absl/synchronization/mutex.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
#include "third_party/abseil-cpp/absl/types/span.h"
#include "third_party/abseil-cpp/absl/types/variant.h"
#include "util/ref_counted.h"
namespace ipcz {
class Node;
class Router;
// A Portal owns a terminal Router along a route. Portals are thread-safe and
// are manipulated directly by public ipcz API calls.
class Portal : public APIObjectImpl<Portal, APIObject::kPortal> {
public:
using Pair = std::pair<Ref<Portal>, Ref<Portal>>;
// Creates a new portal which assumes control over `router` and which lives on
// `node`.
Portal(Ref<Node> node, Ref<Router> router);
const Ref<Node>& node() const { return node_; }
const Ref<Router>& router() const { return router_; }
// Creates a new pair of portals which live on `node` and which are directly
// connected to each other by a LocalRouterLink.
static Pair CreatePair(Ref<Node> node);
// APIObject:
IpczResult Close() override;
bool CanSendFrom(Portal& sender) override;
// ipcz portal API implementation:
IpczResult QueryStatus(IpczPortalStatus& status);
IpczResult Merge(Portal& other);
IpczResult Put(absl::Span<const uint8_t> data,
absl::Span<const IpczHandle> handles,
const IpczPutLimits* limits);
IpczResult BeginPut(IpczBeginPutFlags flags,
const IpczPutLimits* limits,
size_t& num_data_bytes,
void*& data);
IpczResult CommitPut(const void* data,
size_t num_data_bytes_produced,
absl::Span<const IpczHandle> handles);
IpczResult AbortPut(const void* data);
IpczResult Get(IpczGetFlags flags,
void* data,
size_t* num_data_bytes,
IpczHandle* handles,
size_t* num_handles,
IpczHandle* parcel);
IpczResult BeginGet(const void** data,
size_t* num_data_bytes,
size_t* num_handles);
IpczResult CommitGet(size_t num_data_bytes_consumed,
absl::Span<IpczHandle> handles);
IpczResult AbortGet();
private:
~Portal() override;
const Ref<Node> node_;
const Ref<Router> router_;
absl::Mutex mutex_;
bool in_two_phase_get_ ABSL_GUARDED_BY(mutex_) = false;
// Tracks parcels being built for two-phase put operations. The most common
// case is a single concurrent put, so this case is optimized to store an
// inlined Parcel object with no hash table.
using PendingParcelMap = absl::flat_hash_map<const void*, Parcel>;
absl::variant<absl::monostate, Parcel, PendingParcelMap> pending_parcels_;
};
} // namespace ipcz
#endif // IPCZ_SRC_IPCZ_PORTAL_H_