blob: e2922996bdcf00478c8ab58bafe5b0af1ed8ebcf [file] [log] [blame]
// Copyright 2014 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 CONTENT_RENDERER_MEDIA_STREAM_VIDEO_TRACK_ADAPTER_H_
#define CONTENT_RENDERER_MEDIA_STREAM_VIDEO_TRACK_ADAPTER_H_
#include <stdint.h>
#include <vector>
#include "base/macros.h"
#include "base/memory/ref_counted.h"
#include "base/single_thread_task_runner.h"
#include "base/time/time.h"
#include "content/common/content_export.h"
#include "content/renderer/media/stream/media_stream_types.h"
#include "content/renderer/media/stream/media_stream_video_track.h"
#include "media/base/video_frame.h"
#include "ui/gfx/geometry/size.h"
namespace content {
class CONTENT_EXPORT VideoTrackAdapterSettings {
public:
// Creates a VideoTrackAdapterSettings with no target resolution or frame rate
// and without any constraints on the resolution.
VideoTrackAdapterSettings();
// Creates a VideoTrackAdapterSettings with a given target resolution and
// and frame rate, and without any constraints on the resolution.
VideoTrackAdapterSettings(const gfx::Size& target_size,
double max_frame_rate);
// Creates a VideoTrackAdapterSettings with the specified resolution, frame
// rate and resolution constraints. If |target_size| is null, it means that
// no video processing is desired.
VideoTrackAdapterSettings(base::Optional<gfx::Size> target_size,
double min_aspect_ratio,
double max_aspect_ratio,
double max_frame_rate);
VideoTrackAdapterSettings(const VideoTrackAdapterSettings& other);
VideoTrackAdapterSettings& operator=(const VideoTrackAdapterSettings& other);
bool operator==(const VideoTrackAdapterSettings& other) const;
const base::Optional<gfx::Size>& target_size() const { return target_size_; }
int target_width() const {
DCHECK(target_size_);
return target_size_->width();
}
int target_height() const {
DCHECK(target_size_);
return target_size_->height();
}
double min_aspect_ratio() const { return min_aspect_ratio_; }
double max_aspect_ratio() const { return max_aspect_ratio_; }
double max_frame_rate() const { return max_frame_rate_; }
void set_max_frame_rate(double max_frame_rate) {
max_frame_rate_ = max_frame_rate;
}
private:
base::Optional<gfx::Size> target_size_;
double min_aspect_ratio_;
double max_aspect_ratio_;
// A |max_frame_rate| of zero is used to signal that no frame-rate
// adjustment is necessary.
// TODO(guidou): Change this to base::Optional. https://crbug.com/734528
double max_frame_rate_;
};
// VideoTrackAdapter is a helper class used by MediaStreamVideoSource used for
// adapting the video resolution from a source implementation to the resolution
// a track requires. Different tracks can have different resolution constraints.
// The constraints can be set as max width and height as well as max and min
// aspect ratio.
// Video frames are delivered to a track using a VideoCaptureDeliverFrameCB on
// the IO-thread.
// Adaptations is done by wrapping the original media::VideoFrame in a new
// media::VideoFrame with a new visible_rect and natural_size.
class VideoTrackAdapter
: public base::RefCountedThreadSafe<VideoTrackAdapter> {
public:
using OnMutedCallback = base::Callback<void(bool mute_state)>;
explicit VideoTrackAdapter(
scoped_refptr<base::SingleThreadTaskRunner> io_task_runner);
// Register |track| to receive video frames in |frame_callback| with
// a resolution within the boundaries of the arguments, and settings
// updates in |settings_callback|.
// Must be called on the main render thread.
// |source_frame_rate| is used to calculate a prudent interval to check for
// passing frames and inform of the result via |on_muted_state_callback|.
void AddTrack(const MediaStreamVideoTrack* track,
VideoCaptureDeliverFrameCB frame_callback,
VideoTrackSettingsCallback settings_callback,
VideoTrackFormatCallback track_callback,
const VideoTrackAdapterSettings& settings);
void RemoveTrack(const MediaStreamVideoTrack* track);
void ReconfigureTrack(const MediaStreamVideoTrack* track,
const VideoTrackAdapterSettings& settings);
// Delivers |frame| to all tracks that have registered a callback.
// Must be called on the IO-thread.
void DeliverFrameOnIO(const scoped_refptr<media::VideoFrame>& frame,
base::TimeTicks estimated_capture_time);
base::SingleThreadTaskRunner* io_task_runner() const {
DCHECK_CALLED_ON_VALID_THREAD(thread_checker_);
return io_task_runner_.get();
}
// Start monitor that frames are delivered to this object. I.E, that
// |DeliverFrameOnIO| is called with a frame rate of |source_frame_rate|.
// |on_muted_callback| is triggered on the main render thread.
void StartFrameMonitoring(double source_frame_rate,
const OnMutedCallback& on_muted_callback);
void StopFrameMonitoring();
void SetSourceFrameSize(const gfx::Size& source_frame_size);
// Exported for testing.
// Returns true if |desired_size| is updated successfully, false otherwise.
// |desired_size| is not updated |settings| has rescaling disabled and
// |input_size| is invalid.
CONTENT_EXPORT static bool CalculateDesiredSize(
bool is_rotated,
const gfx::Size& input_size,
const VideoTrackAdapterSettings& settings,
gfx::Size* desired_size);
private:
virtual ~VideoTrackAdapter();
friend class base::RefCountedThreadSafe<VideoTrackAdapter>;
void AddTrackOnIO(const MediaStreamVideoTrack* track,
VideoCaptureDeliverFrameCB frame_callback,
VideoTrackSettingsCallback settings_callback,
VideoTrackFormatCallback track_callback,
const VideoTrackAdapterSettings& settings);
void RemoveTrackOnIO(const MediaStreamVideoTrack* track);
void ReconfigureTrackOnIO(const MediaStreamVideoTrack* track,
const VideoTrackAdapterSettings& settings);
void StartFrameMonitoringOnIO(
const OnMutedCallback& on_muted_state_callback,
double source_frame_rate);
void StopFrameMonitoringOnIO();
void SetSourceFrameSizeOnIO(const gfx::Size& frame_size);
// Compare |frame_counter_snapshot| with the current |frame_counter_|, and
// inform of the situation (muted, not muted) via |set_muted_state_callback|.
void CheckFramesReceivedOnIO(const OnMutedCallback& set_muted_state_callback,
uint64_t old_frame_counter_snapshot);
// |thread_checker_| is bound to the main render thread.
THREAD_CHECKER(thread_checker_);
const scoped_refptr<base::SingleThreadTaskRunner> io_task_runner_;
// |renderer_task_runner_| is used to ensure that
// VideoCaptureDeliverFrameCB is released on the main render thread.
const scoped_refptr<base::SingleThreadTaskRunner> renderer_task_runner_;
// VideoFrameResolutionAdapter is an inner class that lives on the IO-thread.
// It does the resolution adaptation and delivers frames to all registered
// tracks.
class VideoFrameResolutionAdapter;
using FrameAdapters = std::vector<scoped_refptr<VideoFrameResolutionAdapter>>;
FrameAdapters adapters_;
// Set to true if frame monitoring has been started. It is only accessed on
// the IO-thread.
bool monitoring_frame_rate_;
// Keeps track of it frames have been received. It is only accessed on the
// IO-thread.
bool muted_state_;
// Running frame counter, accessed on the IO-thread.
uint64_t frame_counter_;
// Frame rate configured on the video source, accessed on the IO-thread.
float source_frame_rate_;
// Resolution configured on the video source, accessed on the IO-thread.
base::Optional<gfx::Size> source_frame_size_;
DISALLOW_COPY_AND_ASSIGN(VideoTrackAdapter);
};
} // namespace content
#endif // CONTENT_RENDERER_MEDIA_STREAM_VIDEO_TRACK_ADAPTER_H_