blob: 4105216f5a2eca0d37b87a73aa71b0a98502642e [file] [log] [blame]
// Copyright 2018 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.
#include <map>
#include "base/gtest_prod_util.h"
#include "base/memory/weak_ptr.h"
#include "base/sequence_checker.h"
#include "build/build_config.h"
#include "media/capture/video/video_frame_receiver.h"
#include "media/capture/video_capture_types.h"
#include "services/video_capture/public/mojom/receiver.mojom.h"
#include "services/video_capture/public/mojom/scoped_access_permission.mojom.h"
namespace video_capture {
// Implementation of mojom::VideoFrameReceiver that distributes frames to
// potentially multiple clients.
class BroadcastingReceiver : public mojom::Receiver {
class BufferContext {
BufferContext(int32_t buffer_id,
media::mojom::VideoBufferHandlePtr buffer_handle);
BufferContext(BufferContext&& other);
BufferContext& operator=(BufferContext&& other);
int32_t buffer_context_id() const { return buffer_context_id_; }
int32_t buffer_id() const { return buffer_id_; }
void set_access_permission(
mojom::ScopedAccessPermissionPtr access_permission) {
access_permission_ = std::move(access_permission);
void IncreaseConsumerCount();
void DecreaseConsumerCount();
bool IsStillBeingConsumed() const;
bool is_retired() const { return is_retired_; }
void set_retired() { is_retired_ = true; }
media::mojom::VideoBufferHandlePtr CloneBufferHandle(
media::VideoCaptureBufferType target_buffer_type);
// If the source handle is shared_memory_via_raw_file_descriptor, we first
// have to unwrap it before we can clone it. Instead of unwrapping, cloning,
// and than wrapping back each time we need to clone it, we convert it to
// a regular shared memory and keep it in this form.
void ConvertRawFileDescriptorToSharedBuffer();
int32_t buffer_context_id_;
int32_t buffer_id_;
media::mojom::VideoBufferHandlePtr buffer_handle_;
// Indicates how many consumers are currently relying on
// |access_permission_|.
int32_t consumer_hold_count_;
bool is_retired_;
mojom::ScopedAccessPermissionPtr access_permission_;
~BroadcastingReceiver() override;
// Indicates to the BroadcastingReceiver that we want to restart the source
// without letting connected clients know about the restart. The
// BroadcastingReceiver will hide the OnStopped() event sent by the source
// from the connected clients and instead invoke the given
// |on_stopped_handler|. It will also not forward the subsequent
// OnStarted() and possibly OnStartedUsingGpuDecode() events to clients who
// have already received these events.
void HideSourceRestartFromClients(base::OnceClosure on_stopped_handler);
void SetOnStoppedHandler(base::OnceClosure on_stopped_handler);
// Returns a client_id that can be used for a call to Suspend/Resume/Remove.
int32_t AddClient(mojom::ReceiverPtr client,
media::VideoCaptureBufferType target_buffer_type);
void SuspendClient(int32_t client_id);
void ResumeClient(int32_t client_id);
// Returns ownership of the client back to the caller.
mojom::ReceiverPtr RemoveClient(int32_t client_id);
// video_capture::mojom::Receiver:
void OnNewBuffer(int32_t buffer_id,
media::mojom::VideoBufferHandlePtr buffer_handle) override;
void OnFrameReadyInBuffer(
int32_t buffer_id,
int32_t frame_feedback_id,
mojom::ScopedAccessPermissionPtr access_permission,
media::mojom::VideoFrameInfoPtr frame_info) override;
void OnBufferRetired(int32_t buffer_id) override;
void OnError(media::VideoCaptureError error) override;
void OnFrameDropped(media::VideoCaptureFrameDropReason reason) override;
void OnLog(const std::string& message) override;
void OnStarted() override;
void OnStartedUsingGpuDecode() override;
void OnStopped() override;
enum class Status {
// Wrapper that suppresses calls to OnStarted() and OnStartedUsingGpuDecode()
// after they have already been called once. Keeps track of whether or not
// a client is suspended.
class ClientContext {
ClientContext(mojom::ReceiverPtr client,
media::VideoCaptureBufferType target_buffer_type);
ClientContext(ClientContext&& other);
ClientContext& operator=(ClientContext&& other);
void OnStarted();
void OnStartedUsingGpuDecode();
mojom::ReceiverPtr& client() { return client_; }
media::VideoCaptureBufferType target_buffer_type() {
return target_buffer_type_;
void set_is_suspended(bool suspended) { is_suspended_ = suspended; }
bool is_suspended() const { return is_suspended_; }
mojom::ReceiverPtr client_;
media::VideoCaptureBufferType target_buffer_type_;
bool is_suspended_;
bool on_started_has_been_called_;
bool on_started_using_gpu_decode_has_been_called_;
void OnClientFinishedConsumingFrame(int32_t buffer_context_id);
void OnClientDisconnected(int32_t client_id);
std::vector<BufferContext>::iterator FindUnretiredBufferContextFromBufferId(
int32_t buffer_id);
std::map<int32_t /*client_id*/, ClientContext> clients_;
std::vector<BufferContext> buffer_contexts_;
Status status_;
base::OnceClosure on_stopped_handler_;
// Keeps track of the last VideoCaptureError that arrived via OnError().
// This is used for relaying the error event to clients that connect after the
// OnError() call has been received.
media::VideoCaptureError error_;
// Keeps track of the id value for the next client to be added. This member is
// incremented each time a client is added and represents a unique identifier
// for each client.
int32_t next_client_id_;
base::WeakPtrFactory<BroadcastingReceiver> weak_factory_;
} // namespace video_capture