blob: 4aa648bcb05d2f2559f9e81e7ec02f4b8f760975 [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.
#ifndef SERVICES_VIDEO_CAPTURE_BROADCASTING_RECEIVER_H_
#define SERVICES_VIDEO_CAPTURE_BROADCASTING_RECEIVER_H_
#include <map>
#include "base/memory/weak_ptr.h"
#include "base/sequence_checker.h"
#include "media/capture/video/video_frame_receiver.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 {
public:
BroadcastingReceiver();
~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);
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;
private:
enum class Status {
kOnStartedHasNotYetBeenCalled,
kOnStartedHasBeenCalled,
kOnStartedUsingGpuDecodeHasBeenCalled,
kDeviceIsRestarting,
kOnErrorHasBeenCalled,
kOnStoppedHasBeenCalled
};
// 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 {
public:
explicit ClientContext(mojom::ReceiverPtr client);
~ClientContext();
ClientContext(ClientContext&& other);
ClientContext& operator=(ClientContext&& other);
void OnStarted();
void OnStartedUsingGpuDecode();
mojom::ReceiverPtr& client() { return client_; }
void set_is_suspended(bool suspended) { is_suspended_ = suspended; }
bool is_suspended() const { return is_suspended_; }
private:
mojom::ReceiverPtr client_;
bool is_suspended_;
bool on_started_has_been_called_;
bool on_started_using_gpu_decode_has_been_called_;
};
class BufferContext {
public:
BufferContext(int32_t buffer_id,
media::mojom::VideoBufferHandlePtr buffer_handle);
~BufferContext();
BufferContext(BufferContext&& other);
BufferContext& operator=(BufferContext&& other);
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;
media::mojom::VideoBufferHandlePtr CloneBufferHandle();
private:
int32_t buffer_id_;
media::mojom::VideoBufferHandlePtr buffer_handle_;
// Indicates how many consumers are currently relying on
// |access_permission_|.
int32_t consumer_hold_count_;
mojom::ScopedAccessPermissionPtr access_permission_;
};
void OnClientFinishedConsumingFrame(int32_t buffer_id);
void OnClientDisconnected(int32_t client_id);
BufferContext& LookupBufferContextFromBufferId(int32_t buffer_id);
SEQUENCE_CHECKER(sequence_checker_);
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
#endif // SERVICES_VIDEO_CAPTURE_BROADCASTING_RECEIVER_H_