/* * Copyright 2017 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "sdk/objc/native/src/objc_video_encoder_factory.h" #include #import "base/RTCMacros.h" #import "base/RTCVideoEncoder.h" #import "base/RTCVideoEncoderFactory.h" #import "components/video_codec/RTCCodecSpecificInfoH264+Private.h" #import "sdk/objc/api/peerconnection/RTCEncodedImage+Private.h" #import "sdk/objc/api/peerconnection/RTCRtpFragmentationHeader+Private.h" #import "sdk/objc/api/peerconnection/RTCVideoCodecInfo+Private.h" #import "sdk/objc/api/peerconnection/RTCVideoEncoderSettings+Private.h" #import "sdk/objc/api/video_codec/RTCVideoCodecConstants.h" #import "sdk/objc/api/video_codec/RTCWrappedNativeVideoEncoder.h" #import "sdk/objc/helpers/NSString+StdString.h" #include "api/video/video_frame.h" #include "api/video_codecs/sdp_video_format.h" #include "api/video_codecs/video_encoder.h" #include "modules/include/module_common_types.h" #include "modules/video_coding/include/video_codec_interface.h" #include "modules/video_coding/include/video_error_codes.h" #include "rtc_base/logging.h" #include "sdk/objc/native/src/objc_video_frame.h" namespace webrtc { namespace { class ObjCVideoEncoder : public VideoEncoder { public: ObjCVideoEncoder(id encoder) : encoder_(encoder), implementation_name_([encoder implementationName].stdString) {} int32_t InitEncode(const VideoCodec *codec_settings, const Settings &encoder_settings) override { RTC_OBJC_TYPE(RTCVideoEncoderSettings) *settings = [[RTC_OBJC_TYPE(RTCVideoEncoderSettings) alloc] initWithNativeVideoCodec:codec_settings]; return [encoder_ startEncodeWithSettings:settings numberOfCores:encoder_settings.number_of_cores]; } int32_t RegisterEncodeCompleteCallback(EncodedImageCallback *callback) override { [encoder_ setCallback:^BOOL(RTC_OBJC_TYPE(RTCEncodedImage) * _Nonnull frame, id _Nonnull info, RTC_OBJC_TYPE(RTCRtpFragmentationHeader) * _Nonnull header) { EncodedImage encodedImage = [frame nativeEncodedImage]; // Handle types that can be converted into one of CodecSpecificInfo's hard coded cases. CodecSpecificInfo codecSpecificInfo; if ([info isKindOfClass:[RTC_OBJC_TYPE(RTCCodecSpecificInfoH264) class]]) { codecSpecificInfo = [(RTC_OBJC_TYPE(RTCCodecSpecificInfoH264) *)info nativeCodecSpecificInfo]; } std::unique_ptr fragmentationHeader = [header createNativeFragmentationHeader]; EncodedImageCallback::Result res = callback->OnEncodedImage(encodedImage, &codecSpecificInfo, fragmentationHeader.get()); return res.error == EncodedImageCallback::Result::OK; }]; return WEBRTC_VIDEO_CODEC_OK; } int32_t Release() override { return [encoder_ releaseEncoder]; } int32_t Encode(const VideoFrame &frame, const std::vector *frame_types) override { NSMutableArray *rtcFrameTypes = [NSMutableArray array]; for (size_t i = 0; i < frame_types->size(); ++i) { [rtcFrameTypes addObject:@(RTCFrameType(frame_types->at(i)))]; } return [encoder_ encode:ToObjCVideoFrame(frame) codecSpecificInfo:nil frameTypes:rtcFrameTypes]; } void SetRates(const RateControlParameters ¶meters) override { const uint32_t bitrate = parameters.bitrate.get_sum_kbps(); const uint32_t framerate = static_cast(parameters.framerate_fps + 0.5); [encoder_ setBitrate:bitrate framerate:framerate]; } VideoEncoder::EncoderInfo GetEncoderInfo() const override { EncoderInfo info; info.supports_native_handle = true; info.implementation_name = implementation_name_; RTC_OBJC_TYPE(RTCVideoEncoderQpThresholds) *qp_thresholds = [encoder_ scalingSettings]; info.scaling_settings = qp_thresholds ? ScalingSettings(qp_thresholds.low, qp_thresholds.high) : ScalingSettings::kOff; info.is_hardware_accelerated = true; info.has_internal_source = false; return info; } private: id encoder_; const std::string implementation_name_; }; class ObjcVideoEncoderSelector : public VideoEncoderFactory::EncoderSelectorInterface { public: ObjcVideoEncoderSelector(id selector) { selector_ = selector; } void OnCurrentEncoder(const SdpVideoFormat &format) override { RTC_OBJC_TYPE(RTCVideoCodecInfo) *info = [[RTC_OBJC_TYPE(RTCVideoCodecInfo) alloc] initWithNativeSdpVideoFormat:format]; [selector_ registerCurrentEncoderInfo:info]; } absl::optional OnEncoderBroken() override { RTC_OBJC_TYPE(RTCVideoCodecInfo) *info = [selector_ encoderForBrokenEncoder]; if (info) { return [info nativeSdpVideoFormat]; } return absl::nullopt; } absl::optional OnAvailableBitrate(const DataRate &rate) override { RTC_OBJC_TYPE(RTCVideoCodecInfo) *info = [selector_ encoderForBitrate:rate.kbps()]; if (info) { return [info nativeSdpVideoFormat]; } return absl::nullopt; } private: id selector_; }; } // namespace ObjCVideoEncoderFactory::ObjCVideoEncoderFactory( id encoder_factory) : encoder_factory_(encoder_factory) {} ObjCVideoEncoderFactory::~ObjCVideoEncoderFactory() {} id ObjCVideoEncoderFactory::wrapped_encoder_factory() const { return encoder_factory_; } std::vector ObjCVideoEncoderFactory::GetSupportedFormats() const { std::vector supported_formats; for (RTC_OBJC_TYPE(RTCVideoCodecInfo) * supportedCodec in [encoder_factory_ supportedCodecs]) { SdpVideoFormat format = [supportedCodec nativeSdpVideoFormat]; supported_formats.push_back(format); } return supported_formats; } std::vector ObjCVideoEncoderFactory::GetImplementations() const { if ([encoder_factory_ respondsToSelector:@selector(implementations)]) { std::vector supported_formats; for (RTC_OBJC_TYPE(RTCVideoCodecInfo) * supportedCodec in [encoder_factory_ implementations]) { SdpVideoFormat format = [supportedCodec nativeSdpVideoFormat]; supported_formats.push_back(format); } return supported_formats; } return GetSupportedFormats(); } VideoEncoderFactory::CodecInfo ObjCVideoEncoderFactory::QueryVideoEncoder( const SdpVideoFormat &format) const { // TODO(andersc): This is a hack until we figure out how this should be done properly. NSString *formatName = [NSString stringForStdString:format.name]; NSSet *wrappedSoftwareFormats = [NSSet setWithObjects:kRTCVideoCodecVp8Name, kRTCVideoCodecVp9Name, nil]; CodecInfo codec_info; codec_info.is_hardware_accelerated = ![wrappedSoftwareFormats containsObject:formatName]; codec_info.has_internal_source = false; return codec_info; } std::unique_ptr ObjCVideoEncoderFactory::CreateVideoEncoder( const SdpVideoFormat &format) { RTC_OBJC_TYPE(RTCVideoCodecInfo) *info = [[RTC_OBJC_TYPE(RTCVideoCodecInfo) alloc] initWithNativeSdpVideoFormat:format]; id encoder = [encoder_factory_ createEncoder:info]; if ([encoder isKindOfClass:[RTCWrappedNativeVideoEncoder class]]) { return [(RTCWrappedNativeVideoEncoder *)encoder releaseWrappedEncoder]; } else { return std::unique_ptr(new ObjCVideoEncoder(encoder)); } } std::unique_ptr ObjCVideoEncoderFactory::GetEncoderSelector() const { if ([encoder_factory_ respondsToSelector:@selector(encoderSelector)]) { id selector = [encoder_factory_ encoderSelector]; if (selector) { return absl::make_unique(selector); } } return nullptr; } } // namespace webrtc