| // 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_video_encoder_factory.h" |
| |
| #include <algorithm> |
| #include <vector> |
| |
| #include "base/callback.h" |
| #include "base/logging.h" |
| #include "base/memory/ptr_util.h" |
| #include "base/synchronization/lock.h" |
| |
| namespace remoting { |
| |
| WebRtcVideoEncoder::WebRtcVideoEncoder(webrtc::VideoCodecType codec) |
| : state_(kUninitialized), video_codec_type_(codec) { |
| VLOG(1) << "video codecType " << video_codec_type_; |
| } |
| |
| WebRtcVideoEncoder::~WebRtcVideoEncoder() {} |
| |
| int32_t WebRtcVideoEncoder::InitEncode(const webrtc::VideoCodec* codec_settings, |
| int32_t number_of_cores, |
| size_t max_payload_size) { |
| base::AutoLock lock(lock_); |
| DCHECK(codec_settings); |
| VLOG(1) << "video codecType " << codec_settings->codecType << " width " |
| << codec_settings->width << " height " << codec_settings->height |
| << " startBitrate " << codec_settings->startBitrate << " maxBitrate " |
| << codec_settings->maxBitrate << " minBitrate " |
| << codec_settings->minBitrate << " targetBitrate " |
| << codec_settings->targetBitrate << " maxFramerate " |
| << codec_settings->maxFramerate; |
| |
| int streamCount = codec_settings->numberOfSimulcastStreams; |
| // Validate request is to support a single stream. |
| if (streamCount > 1) { |
| for (int i = 0; i < streamCount; ++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 WebRtcVideoEncoder::RegisterEncodeCompleteCallback( |
| webrtc::EncodedImageCallback* callback) { |
| base::AutoLock lock(lock_); |
| DCHECK(callback); |
| encoded_callback_ = callback; |
| return WEBRTC_VIDEO_CODEC_OK; |
| } |
| |
| int32_t WebRtcVideoEncoder::Release() { |
| base::AutoLock lock(lock_); |
| encoded_callback_ = nullptr; |
| return WEBRTC_VIDEO_CODEC_OK; |
| } |
| |
| int32_t WebRtcVideoEncoder::Encode( |
| const webrtc::VideoFrame& frame, |
| const webrtc::CodecSpecificInfo* codec_specific_info, |
| const std::vector<webrtc::FrameType>* frame_types) { |
| base::AutoLock lock(lock_); |
| if (!key_frame_request_.is_null()) |
| key_frame_request_.Run(); |
| return WEBRTC_VIDEO_CODEC_OK; |
| } |
| |
| int32_t WebRtcVideoEncoder::SetChannelParameters(uint32_t packet_loss, |
| int64_t rtt) { |
| VLOG(1) << "WebRtcVideoEncoder::SetChannelParameters " |
| << "loss:RTT " << packet_loss << ":" << rtt; |
| // Unused right now. |
| return WEBRTC_VIDEO_CODEC_OK; |
| } |
| |
| int32_t WebRtcVideoEncoder::SetRates(uint32_t bitrate, uint32_t framerate) { |
| VLOG(1) << "WebRtcVideoEncoder::SetRates bitrate:framerate " << bitrate << ":" |
| << framerate; |
| if (!target_bitrate_cb_.is_null()) |
| target_bitrate_cb_.Run(bitrate); |
| // framerate is not expected to be valid given we never report captured |
| // frames |
| return WEBRTC_VIDEO_CODEC_OK; |
| } |
| |
| int WebRtcVideoEncoder::SendEncodedFrame(std::unique_ptr<VideoPacket> frame) { |
| 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 -1; |
| } |
| |
| webrtc::EncodedImage encoded_image(buffer, buffer_size, buffer_size); |
| encoded_image._encodedWidth = frame->format().screen_width(); |
| encoded_image._encodedHeight = frame->format().screen_height(); |
| encoded_image._completeFrame = true; |
| encoded_image._frameType = |
| frame->key_frame() ? webrtc::kVideoFrameKey : webrtc::kVideoFrameDelta; |
| encoded_image.capture_time_ms_ = frame->capture_time_ms(); |
| encoded_image._timeStamp = |
| static_cast<uint32_t>(frame->capture_time_ms() * 90); |
| |
| 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; |
| |
| int result = |
| encoded_callback_->Encoded(encoded_image, &codec_specific_info, &header); |
| if (result < 0) { |
| LOG(ERROR) << "Encoded callback failed: " << result; |
| } else if (result > 0) { |
| VLOG(1) << "Drop request from webrtc"; |
| } |
| return result; |
| } |
| |
| void WebRtcVideoEncoder::SetKeyFrameRequestCallback( |
| const base::Closure& key_frame_request) { |
| base::AutoLock lock(lock_); |
| key_frame_request_ = key_frame_request; |
| } |
| |
| void WebRtcVideoEncoder::SetTargetBitrateCallback( |
| const TargetBitrateCallback& target_bitrate_cb) { |
| base::AutoLock lock(lock_); |
| target_bitrate_cb_ = target_bitrate_cb; |
| } |
| |
| WebRtcVideoEncoderFactory::WebRtcVideoEncoderFactory() { |
| // TODO(isheriff): These do not really affect anything internally |
| // in webrtc. |
| codecs_.push_back(cricket::WebRtcVideoEncoderFactory::VideoCodec( |
| webrtc::kVideoCodecVP8, "VP8", 1280, 720, 30)); |
| } |
| |
| WebRtcVideoEncoderFactory::~WebRtcVideoEncoderFactory() { |
| DCHECK(encoders_.empty()); |
| } |
| |
| webrtc::VideoEncoder* WebRtcVideoEncoderFactory::CreateVideoEncoder( |
| webrtc::VideoCodecType type) { |
| VLOG(2) << "WebRtcVideoEncoderFactory::CreateVideoEncoder " << type; |
| DCHECK(type == webrtc::kVideoCodecVP8); |
| WebRtcVideoEncoder* encoder = new WebRtcVideoEncoder(type); |
| base::AutoLock lock(lock_); |
| encoder->SetKeyFrameRequestCallback(key_frame_request_); |
| encoder->SetTargetBitrateCallback(target_bitrate_cb_); |
| VLOG(1) << "Created " << encoder; |
| encoders_.push_back(base::WrapUnique(encoder)); |
| return encoder; |
| } |
| |
| const std::vector<cricket::WebRtcVideoEncoderFactory::VideoCodec>& |
| WebRtcVideoEncoderFactory::codecs() const { |
| VLOG(2) << "WebRtcVideoEncoderFactory::codecs"; |
| return codecs_; |
| } |
| |
| bool WebRtcVideoEncoderFactory::EncoderTypeHasInternalSource( |
| webrtc::VideoCodecType type) const { |
| VLOG(2) << "WebRtcVideoEncoderFactory::EncoderTypeHasInternalSource"; |
| return true; |
| } |
| |
| void WebRtcVideoEncoderFactory::DestroyVideoEncoder( |
| webrtc::VideoEncoder* encoder) { |
| VLOG(2) << "WebRtcVideoEncoderFactory::DestroyVideoEncoder"; |
| 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; |
| } |
| } |
| DCHECK(false) << "Asked to remove encoder not owned by factory"; |
| } |
| |
| int WebRtcVideoEncoderFactory::SendEncodedFrame( |
| std::unique_ptr<VideoPacket> frame) { |
| if (encoders_.size() != 1) { |
| LOG(ERROR) << "Unexpected number of encoders " << encoders_.size(); |
| return -1; |
| } |
| return encoders_.front()->SendEncodedFrame(std::move(frame)); |
| } |
| |
| void WebRtcVideoEncoderFactory::SetKeyFrameRequestCallback( |
| const base::Closure& key_frame_request) { |
| base::AutoLock lock(lock_); |
| key_frame_request_ = key_frame_request; |
| if (encoders_.size() == 1) { |
| encoders_.front()->SetKeyFrameRequestCallback(key_frame_request); |
| } else { |
| LOG(ERROR) << "Dropping key frame request callback with unexpected" |
| " number of encoders" |
| << encoders_.size(); |
| } |
| } |
| |
| void WebRtcVideoEncoderFactory::SetTargetBitrateCallback( |
| const TargetBitrateCallback& target_bitrate_cb) { |
| base::AutoLock lock(lock_); |
| target_bitrate_cb_ = target_bitrate_cb; |
| if (encoders_.size() == 1) { |
| encoders_.front()->SetTargetBitrateCallback(target_bitrate_cb); |
| } else { |
| LOG(ERROR) << "Dropping target bitrate request callback with unexpected" |
| " number of encoders" |
| << encoders_.size(); |
| } |
| } |
| } // namespace remoting |