blob: e92d66cc1a8f1cbc7c5439fea0a44a21de662137 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MOJO_EDK_SYSTEM_DATA_PIPE_IMPL_H_
#define MOJO_EDK_SYSTEM_DATA_PIPE_IMPL_H_
#include <stdint.h>
#include <utility>
#include <vector>
#include "mojo/edk/platform/scoped_platform_handle.h"
#include "mojo/edk/system/data_pipe.h"
#include "mojo/edk/system/handle_signals_state.h"
#include "mojo/edk/system/memory.h"
#include "mojo/edk/util/thread_annotations.h"
#include "mojo/public/c/system/data_pipe.h"
#include "mojo/public/c/system/macros.h"
#include "mojo/public/c/system/result.h"
namespace mojo {
namespace system {
class Channel;
class MessageInTransit;
// Base class/interface for classes that "implement" |DataPipe| for various
// situations (local versus remote). The methods, other than the constructor,
// |set_owner()|, and the destructor, are always protected by |DataPipe|'s
// |mutex_|.
//
// Methods that access |owner_| are marked |MOJO_NO_THREAD_SAFETY_ANALYSIS| for
// the reason described below.
//
// Ideally, they'd be marked as requiring |owner_->mutex_|, but then we'd have
// trouble with the |DataPipe| methods that call |DataPipeImpl| methods (via
// |DataPipe|'s |impl_| member): we cannot tell the compiler that |impl_->owner_
// == this| in |DataPipe| methods, so that taking |mutex_| is equivalent to
// taking |impl_->owner_->mutex_|.
//
// Since |DataPipeImpl| methods are only called by |DataPipe| methods (which are
// required to hold |mutex_| since |impl_| marked as being guarded by it) and by
// |DataPipeImpl| and subclasses' methods (which are all transitively under
// |mutex_|, having originated from a call from |DataPipe|), we choose to turn
// off thread-safety analysis for our methods, which are trivial (they just
// thunk to |owner_|), rather than turn off thread-safety analysis for
// |DataPipe|'s methods, which aren't.
// Note that subclasses do not have access to the owning |DataPipe| (except as a
// |ChannelEndpointClient|), so that their methods should never need
// |MOJO_NO_THREAD_SAFETY_ANALYSIS| annotations.
class DataPipeImpl {
public:
virtual ~DataPipeImpl() {}
// This is only called by |DataPipe| during its construction.
void set_owner(DataPipe* owner) { owner_ = owner; }
virtual void ProducerClose() = 0;
// |num_bytes.Get()| will be a nonzero multiple of |element_num_bytes()|.
virtual MojoResult ProducerWriteData(UserPointer<const void> elements,
UserPointer<uint32_t> num_bytes,
uint32_t max_num_bytes_to_write,
uint32_t min_num_bytes_to_write) = 0;
virtual MojoResult ProducerBeginWriteData(
UserPointer<void*> buffer,
UserPointer<uint32_t> buffer_num_bytes) = 0;
virtual MojoResult ProducerEndWriteData(uint32_t num_bytes_written) = 0;
// Note: A producer should not be writable during a two-phase write.
virtual HandleSignalsState ProducerGetHandleSignalsState() const = 0;
virtual void ProducerStartSerialize(Channel* channel,
size_t* max_size,
size_t* max_platform_handles) = 0;
virtual bool ProducerEndSerialize(
Channel* channel,
void* destination,
size_t* actual_size,
std::vector<platform::ScopedPlatformHandle>* platform_handles) = 0;
virtual void ConsumerClose() = 0;
// |num_bytes.Get()| will be a nonzero multiple of |element_num_bytes()|.
virtual MojoResult ConsumerReadData(UserPointer<void> elements,
UserPointer<uint32_t> num_bytes,
uint32_t max_num_bytes_to_read,
uint32_t min_num_bytes_to_read,
bool peek) = 0;
virtual MojoResult ConsumerDiscardData(UserPointer<uint32_t> num_bytes,
uint32_t max_num_bytes_to_discard,
uint32_t min_num_bytes_to_discard) = 0;
// |num_bytes.Get()| will be a nonzero multiple of |element_num_bytes()|.
virtual MojoResult ConsumerQueryData(UserPointer<uint32_t> num_bytes) = 0;
virtual MojoResult ConsumerBeginReadData(
UserPointer<const void*> buffer,
UserPointer<uint32_t> buffer_num_bytes) = 0;
virtual MojoResult ConsumerEndReadData(uint32_t num_bytes_read) = 0;
// Note: A consumer should not be writable during a two-phase read.
virtual HandleSignalsState ConsumerGetHandleSignalsState() const = 0;
virtual void ConsumerStartSerialize(Channel* channel,
size_t* max_size,
size_t* max_platform_handles) = 0;
virtual bool ConsumerEndSerialize(
Channel* channel,
void* destination,
size_t* actual_size,
std::vector<platform::ScopedPlatformHandle>* platform_handles) = 0;
virtual bool OnReadMessage(unsigned port, MessageInTransit* message) = 0;
virtual void OnDetachFromChannel(unsigned port) = 0;
protected:
DataPipeImpl() : owner_() {}
// Helper to convert the given circular buffer into messages. The input is a
// circular buffer |buffer| (with appropriate element size and capacity), with
// current contents starting at |start_index| of length |current_num_bytes|.
// This will convert all of the contents.
void ConvertDataToMessages(const char* buffer,
size_t* start_index,
size_t* current_num_bytes,
MessageInTransitQueue* message_queue);
std::unique_ptr<DataPipeImpl> ReplaceImpl(
std::unique_ptr<DataPipeImpl> new_impl) MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->ReplaceImplNoLock(std::move(new_impl));
}
void SetProducerClosed() MOJO_NO_THREAD_SAFETY_ANALYSIS {
owner_->SetProducerClosedNoLock();
}
void SetConsumerClosed() MOJO_NO_THREAD_SAFETY_ANALYSIS {
owner_->SetConsumerClosedNoLock();
}
ChannelEndpointClient* channel_endpoint_client() const { return owner_; }
MojoCreateDataPipeOptions validated_options() const {
return owner_->validated_options();
}
size_t element_num_bytes() const { return owner_->element_num_bytes(); }
size_t capacity_num_bytes() const { return owner_->capacity_num_bytes(); }
bool producer_open() const MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->producer_open_no_lock();
}
bool consumer_open() const MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->consumer_open_no_lock();
}
size_t consumer_read_threshold_num_bytes() const
MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->consumer_read_threshold_num_bytes_no_lock();
}
uint32_t producer_two_phase_max_num_bytes_written() const
MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->producer_two_phase_max_num_bytes_written_no_lock();
}
uint32_t consumer_two_phase_max_num_bytes_read() const
MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->consumer_two_phase_max_num_bytes_read_no_lock();
}
void set_producer_two_phase_max_num_bytes_written(uint32_t num_bytes)
MOJO_NO_THREAD_SAFETY_ANALYSIS {
owner_->set_producer_two_phase_max_num_bytes_written_no_lock(num_bytes);
}
void set_consumer_two_phase_max_num_bytes_read(uint32_t num_bytes)
MOJO_NO_THREAD_SAFETY_ANALYSIS {
owner_->set_consumer_two_phase_max_num_bytes_read_no_lock(num_bytes);
}
bool producer_in_two_phase_write() const MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->producer_in_two_phase_write_no_lock();
}
bool consumer_in_two_phase_read() const MOJO_NO_THREAD_SAFETY_ANALYSIS {
return owner_->consumer_in_two_phase_read_no_lock();
}
private:
DataPipe* owner_;
MOJO_DISALLOW_COPY_AND_ASSIGN(DataPipeImpl);
};
// TODO(vtl): This is not the ideal place for the following structs; find
// somewhere better.
// Serialized form of a producer dispatcher. This will actually be followed by a
// serialized |ChannelEndpoint|; we want to preserve alignment guarantees.
struct MOJO_ALIGNAS(8) SerializedDataPipeProducerDispatcher {
// Only validated (and thus canonicalized) options should be serialized.
// However, the deserializer must revalidate (as with everything received).
MojoCreateDataPipeOptions validated_options;
// Number of bytes already enqueued to the consumer. Set to
// |static_cast<size_t>(-1)| if the consumer is already closed, in which case
// this will *not* be followed by a serialized |ChannelEndpoint|.
size_t consumer_num_bytes;
};
// Serialized form of a consumer dispatcher. This will actually be followed by a
// serialized |ChannelEndpoint|; we want to preserve alignment guarantees.
struct MOJO_ALIGNAS(8) SerializedDataPipeConsumerDispatcher {
// Only validated (and thus canonicalized) options should be serialized.
// However, the deserializer must revalidate (as with everything received).
MojoCreateDataPipeOptions validated_options;
};
} // namespace system
} // namespace mojo
#endif // MOJO_EDK_SYSTEM_DATA_PIPE_IMPL_H_