blob: 4d28176f40208c3ecf686d262ffc032f8190bd5d [file] [log] [blame]
// Copyright 2016 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 "remoting/protocol/webrtc_dummy_video_encoder.h"
#include <algorithm>
#include <vector>
#include "base/bind.h"
#include "base/callback.h"
#include "base/logging.h"
#include "base/memory/ptr_util.h"
#include "base/synchronization/lock.h"
#include "base/threading/thread_task_runner_handle.h"
#include "remoting/protocol/video_channel_state_observer.h"
namespace remoting {
namespace protocol {
WebrtcDummyVideoEncoder::WebrtcDummyVideoEncoder(
scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
base::WeakPtr<VideoChannelStateObserver> video_channel_state_observer)
: main_task_runner_(main_task_runner),
state_(kUninitialized),
video_channel_state_observer_(video_channel_state_observer) {}
WebrtcDummyVideoEncoder::~WebrtcDummyVideoEncoder() {}
int32_t WebrtcDummyVideoEncoder::InitEncode(
const webrtc::VideoCodec* codec_settings,
int32_t number_of_cores,
size_t max_payload_size) {
DCHECK(codec_settings);
base::AutoLock lock(lock_);
int stream_count = codec_settings->numberOfSimulcastStreams;
// Validate request is to support a single stream.
if (stream_count > 1) {
for (int i = 0; i < stream_count; ++i) {
if (codec_settings->simulcastStream[i].maxBitrate != 0) {
LOG(ERROR) << "Simulcast unsupported";
return WEBRTC_VIDEO_CODEC_ERR_PARAMETER;
}
}
}
state_ = kInitialized;
return WEBRTC_VIDEO_CODEC_OK;
}
int32_t WebrtcDummyVideoEncoder::RegisterEncodeCompleteCallback(
webrtc::EncodedImageCallback* callback) {
DCHECK(callback);
base::AutoLock lock(lock_);
encoded_callback_ = callback;
return WEBRTC_VIDEO_CODEC_OK;
}
int32_t WebrtcDummyVideoEncoder::Release() {
base::AutoLock lock(lock_);
encoded_callback_ = nullptr;
return WEBRTC_VIDEO_CODEC_OK;
}
int32_t WebrtcDummyVideoEncoder::Encode(
const webrtc::VideoFrame& frame,
const webrtc::CodecSpecificInfo* codec_specific_info,
const std::vector<webrtc::FrameType>* frame_types) {
// WebrtcDummyVideoCapturer doesn't generate any video frames, so Encode() can
// be called only from VCMGenericEncoder::RequestFrame() to request a key
// frame.
main_task_runner_->PostTask(
FROM_HERE, base::Bind(&VideoChannelStateObserver::OnKeyFrameRequested,
video_channel_state_observer_));
return WEBRTC_VIDEO_CODEC_OK;
}
int32_t WebrtcDummyVideoEncoder::SetChannelParameters(uint32_t packet_loss,
int64_t rtt) {
main_task_runner_->PostTask(
FROM_HERE, base::Bind(&VideoChannelStateObserver::OnChannelParameters,
video_channel_state_observer_, packet_loss,
base::TimeDelta::FromMilliseconds(rtt)));
return WEBRTC_VIDEO_CODEC_OK;
}
int32_t WebrtcDummyVideoEncoder::SetRates(uint32_t bitrate,
uint32_t framerate) {
main_task_runner_->PostTask(
FROM_HERE, base::Bind(&VideoChannelStateObserver::OnTargetBitrateChanged,
video_channel_state_observer_, bitrate));
// framerate is not expected to be valid given we never report captured
// frames.
return WEBRTC_VIDEO_CODEC_OK;
}
webrtc::EncodedImageCallback::Result WebrtcDummyVideoEncoder::SendEncodedFrame(
const WebrtcVideoEncoder::EncodedFrame& frame,
base::TimeTicks capture_time) {
DCHECK(main_task_runner_->BelongsToCurrentThread());
uint8_t* buffer =
reinterpret_cast<uint8_t*>(const_cast<char*>(frame.data.data()));
size_t buffer_size = frame.data.size();
base::AutoLock lock(lock_);
if (state_ == kUninitialized) {
LOG(ERROR) << "encoder interface uninitialized";
return webrtc::EncodedImageCallback::Result(
webrtc::EncodedImageCallback::Result::ERROR_SEND_FAILED);
}
webrtc::EncodedImage encoded_image(buffer, buffer_size, buffer_size);
encoded_image._encodedWidth = frame.size.width();
encoded_image._encodedHeight = frame.size.height();
encoded_image._completeFrame = true;
encoded_image._frameType =
frame.key_frame ? webrtc::kVideoFrameKey : webrtc::kVideoFrameDelta;
int64_t capture_time_ms = (capture_time - base::TimeTicks()).InMilliseconds();
encoded_image.capture_time_ms_ = capture_time_ms;
encoded_image._timeStamp = static_cast<uint32_t>(capture_time_ms * 90);
encoded_image.playout_delay_.min_ms = 0;
encoded_image.playout_delay_.max_ms = 0;
webrtc::CodecSpecificInfo codec_specific_info;
memset(&codec_specific_info, 0, sizeof(codec_specific_info));
codec_specific_info.codecType = webrtc::kVideoCodecVP8;
webrtc::RTPFragmentationHeader header;
memset(&header, 0, sizeof(header));
codec_specific_info.codecSpecific.VP8.simulcastIdx = 0;
codec_specific_info.codecSpecific.VP8.temporalIdx = webrtc::kNoTemporalIdx;
codec_specific_info.codecSpecific.VP8.tl0PicIdx = webrtc::kNoTl0PicIdx;
codec_specific_info.codecSpecific.VP8.pictureId = webrtc::kNoPictureId;
header.VerifyAndAllocateFragmentationHeader(1);
header.fragmentationOffset[0] = 0;
header.fragmentationLength[0] = buffer_size;
header.fragmentationPlType[0] = 0;
header.fragmentationTimeDiff[0] = 0;
return encoded_callback_->OnEncodedImage(encoded_image, &codec_specific_info,
&header);
}
WebrtcDummyVideoEncoderFactory::WebrtcDummyVideoEncoderFactory()
: main_task_runner_(base::ThreadTaskRunnerHandle::Get()) {
// TODO(isheriff): These do not really affect anything internally
// in webrtc.
codecs_.push_back(cricket::WebRtcVideoEncoderFactory::VideoCodec(
webrtc::kVideoCodecVP8, "VP8", 1280, 720, 30));
}
WebrtcDummyVideoEncoderFactory::~WebrtcDummyVideoEncoderFactory() {
DCHECK(encoders_.empty());
}
webrtc::VideoEncoder* WebrtcDummyVideoEncoderFactory::CreateVideoEncoder(
webrtc::VideoCodecType type) {
DCHECK_EQ(type, webrtc::kVideoCodecVP8);
WebrtcDummyVideoEncoder* encoder = new WebrtcDummyVideoEncoder(
main_task_runner_, video_channel_state_observer_);
base::AutoLock lock(lock_);
encoders_.push_back(base::WrapUnique(encoder));
return encoder;
}
const std::vector<cricket::WebRtcVideoEncoderFactory::VideoCodec>&
WebrtcDummyVideoEncoderFactory::codecs() const {
return codecs_;
}
bool WebrtcDummyVideoEncoderFactory::EncoderTypeHasInternalSource(
webrtc::VideoCodecType type) const {
// Returns true to directly provide encoded frames to webrtc.
return true;
}
void WebrtcDummyVideoEncoderFactory::DestroyVideoEncoder(
webrtc::VideoEncoder* encoder) {
base::AutoLock lock(lock_);
if (encoder == nullptr) {
LOG(ERROR) << "Attempting to destroy null encoder";
return;
}
for (auto pos = encoders_.begin(); pos != encoders_.end(); ++pos) {
if ((*pos).get() == encoder) {
encoders_.erase(pos);
return;
}
}
NOTREACHED() << "Asked to remove encoder not owned by factory.";
}
webrtc::EncodedImageCallback::Result
WebrtcDummyVideoEncoderFactory::SendEncodedFrame(
const WebrtcVideoEncoder::EncodedFrame& frame,
base::TimeTicks capture_time) {
DCHECK(main_task_runner_->BelongsToCurrentThread());
base::AutoLock lock(lock_);
if (encoders_.size() != 1) {
LOG(ERROR) << "Unexpected number of encoders " << encoders_.size();
return webrtc::EncodedImageCallback::Result(
webrtc::EncodedImageCallback::Result::ERROR_SEND_FAILED);
}
return encoders_.front()->SendEncodedFrame(frame, capture_time);
}
void WebrtcDummyVideoEncoderFactory::SetVideoChannelStateObserver(
base::WeakPtr<VideoChannelStateObserver> video_channel_state_observer) {
DCHECK(main_task_runner_->BelongsToCurrentThread());
DCHECK(encoders_.empty());
base::AutoLock lock(lock_);
video_channel_state_observer_ = video_channel_state_observer;
}
} // namespace protocol
} // namespace remoting