webrtc Android源码分析一

nativeCreateVideoSource

初始化

PeerConnectionFactory(pc/peerconnectionfactory) 创建PeerConnection方法中:

rtc::scoped_refptr<PeerConnectionInterface>
PeerConnectionFactory::CreatePeerConnection(const PeerConnectionInterface::RTCConfiguration& configuration,std::unique_ptr<cricket::PortAllocator> allocator,std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,PeerConnectionObserver* observer) {RTC_DCHECK(signaling_thread_->IsCurrent());if (!cert_generator.get()) {// No certificate generator specified, use the default one.cert_generator.reset(new rtc::RTCCertificateGenerator(signaling_thread_, network_thread_));}if (!allocator) {allocator.reset(new cricket::BasicPortAllocator(default_network_manager_.get(), default_socket_factory_.get()));}network_thread_->Invoke<void>(RTC_FROM_HERE, rtc::Bind(&cricket::PortAllocator::SetNetworkIgnoreMask,allocator.get(), options_.network_ignore_mask));rtc::scoped_refptr<PeerConnection> pc(new rtc::RefCountedObject<PeerConnection>(this));if (!pc->Initialize(configuration, std::move(allocator),std::move(cert_generator), observer)) {return nullptr;}return PeerConnectionProxy::Create(signaling_thread(), pc);
}

构造PeerConnection对象pc,并调用初始化方法Initialize,Initialize中:

ool PeerConnection::Initialize(const PeerConnectionInterface::RTCConfiguration& configuration,std::unique_ptr<cricket::PortAllocator> allocator,std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,PeerConnectionObserver* observer) {TRACE_EVENT0("webrtc", "PeerConnection::Initialize");if (!allocator) {LOG(LS_ERROR) << "PeerConnection initialized without a PortAllocator? "<< "This shouldn't happen if using PeerConnectionFactory.";return false;}if (!observer) {// TODO(deadbeef): Why do we do this?LOG(LS_ERROR) << "PeerConnection initialized without a "<< "PeerConnectionObserver";return false;}observer_ = observer;port_allocator_ = std::move(allocator);// The port allocator lives on the network thread and should be initialized// there.if (!network_thread()->Invoke<bool>(RTC_FROM_HERE, rtc::Bind(&PeerConnection::InitializePortAllocator_n,this, configuration))) {return false;}// Call must be constructed on the worker thread.factory_->worker_thread()->Invoke<void>(RTC_FROM_HERE, rtc::Bind(&PeerConnection::CreateCall_w,this));session_.reset(new WebRtcSession(call_.get(), factory_->channel_manager(), configuration.media_config,event_log_.get(),factory_->network_thread(),factory_->worker_thread(), factory_->signaling_thread(),port_allocator_.get(),std::unique_ptr<cricket::TransportController>(factory_->CreateTransportController(port_allocator_.get(),configuration.redetermine_role_on_ice_restart)),
#ifdef HAVE_SCTPstd::unique_ptr<cricket::SctpTransportInternalFactory>(new cricket::SctpTransportFactory(factory_->network_thread()))
#elsenullptr
#endif));stats_.reset(new StatsCollector(this));stats_collector_ = RTCStatsCollector::Create(this);// Initialize the WebRtcSession. It creates transport channels etc.if (!session_->Initialize(factory_->options(), std::move(cert_generator),configuration)) {return false;}// Register PeerConnection as receiver of local ice candidates.// All the callbacks will be posted to the application from PeerConnection.session_->RegisterIceObserver(this);session_->SignalState.connect(this, &PeerConnection::OnSessionStateChange);session_->SignalVoiceChannelCreated.connect(this, &PeerConnection::OnVoiceChannelCreated);session_->SignalVoiceChannelDestroyed.connect(this, &PeerConnection::OnVoiceChannelDestroyed);session_->SignalVideoChannelCreated.connect(this, &PeerConnection::OnVideoChannelCreated);session_->SignalVideoChannelDestroyed.connect(this, &PeerConnection::OnVideoChannelDestroyed);session_->SignalDataChannelCreated.connect(this, &PeerConnection::OnDataChannelCreated);session_->SignalDataChannelDestroyed.connect(this, &PeerConnection::OnDataChannelDestroyed);session_->SignalDataChannelOpenMessage.connect(this, &PeerConnection::OnDataChannelOpenMessage);configuration_ = configuration;return true;
}

调用CreateCall_w创建call对象:

void PeerConnection::CreateCall_w() {RTC_DCHECK(!call_);const int kMinBandwidthBps = 30000;const int kStartBandwidthBps = 300000;const int kMaxBandwidthBps = 2000000;webrtc::Call::Config call_config(event_log_.get());call_config.audio_state =factory_->channel_manager() ->media_engine()->GetAudioState();call_config.bitrate_config.min_bitrate_bps = kMinBandwidthBps;call_config.bitrate_config.start_bitrate_bps = kStartBandwidthBps;call_config.bitrate_config.max_bitrate_bps = kMaxBandwidthBps;call_.reset(webrtc::Call::Create(call_config));
}

使用call对象以及PeerConnectionFactory中channel_manager(PeerConnectionFactory中Initialize中创建)构造WebRtcSession对象session_,调用Initialize方法初始化session_,初始化session_槽函数等.session_初始化方法中创建WebRtcSessionDescriptionFactory对象webrtc_session_desc_factory_.

创建Channel

WebRtcSession::SetLocalDescription:

bool WebRtcSession::SetLocalDescription(SessionDescriptionInterface* desc,std::string* err_desc) {RTC_DCHECK(signaling_thread()->IsCurrent());// Takes the ownership of |desc| regardless of the result.std::unique_ptr<SessionDescriptionInterface> desc_temp(desc);// Validate SDP.if (!ValidateSessionDescription(desc, cricket::CS_LOCAL, err_desc)) {return false;}// Update the initial_offerer flag if this session is the initial_offerer.Action action = GetAction(desc->type());if (state() == STATE_INIT && action == kOffer) {initial_offerer_ = true;transport_controller_->SetIceRole(cricket::ICEROLE_CONTROLLING);}if (action == kAnswer) {current_local_description_.reset(desc_temp.release());pending_local_description_.reset(nullptr);current_remote_description_.reset(pending_remote_description_.release());} else {pending_local_description_.reset(desc_temp.release());}// Transport and Media channels will be created only when offer is set.if (action == kOffer && !CreateChannels(local_description()->description())) {// TODO(mallinath) - Handle CreateChannel failure, as new local description// is applied. Restore back to old description.return BadLocalSdp(desc->type(), kCreateChannelFailed, err_desc);}// Remove unused channels if MediaContentDescription is rejected.RemoveUnusedChannels(local_description()->description());if (!UpdateSessionState(action, cricket::CS_LOCAL, err_desc)) {return false;}if (remote_description()) {// Now that we have a local description, we can push down remote candidates.UseCandidatesInSessionDescription(remote_description());}pending_ice_restarts_.clear();if (error() != ERROR_NONE) {return BadLocalSdp(desc->type(), GetSessionErrorMsg(), err_desc);}return true;
}

action为offer时CreateChannel创建channels:

bool WebRtcSession::CreateChannels(const SessionDescription* desc) {const cricket::ContentGroup* bundle_group = nullptr;if (bundle_policy_ == PeerConnectionInterface::kBundlePolicyMaxBundle) {bundle_group = desc->GetGroupByName(cricket::GROUP_TYPE_BUNDLE);if (!bundle_group) {LOG(LS_WARNING) << "max-bundle specified without BUNDLE specified";return false;}}// Creating the media channels and transport proxies.const cricket::ContentInfo* voice = cricket::GetFirstAudioContent(desc);if (voice && !voice->rejected && !voice_channel_) {if (!CreateVoiceChannel(voice,GetBundleTransportName(voice, bundle_group))) {LOG(LS_ERROR) << "Failed to create voice channel.";return false;}}const cricket::ContentInfo* video = cricket::GetFirstVideoContent(desc);if (video && !video->rejected && !video_channel_) {if (!CreateVideoChannel(video,GetBundleTransportName(video, bundle_group))) {LOG(LS_ERROR) << "Failed to create video channel.";return false;}}const cricket::ContentInfo* data = cricket::GetFirstDataContent(desc);if (data_channel_type_ != cricket::DCT_NONE && data && !data->rejected &&!rtp_data_channel_ && !sctp_transport_) {if (!CreateDataChannel(data, GetBundleTransportName(data, bundle_group))) {LOG(LS_ERROR) << "Failed to create data channel.";return false;}}return true;
}

CreateChannels中创建三个Channel,其中CreateVideoChannel创建视频Channel:

bool WebRtcSession::CreateVideoChannel(const cricket::ContentInfo* content,const std::string* bundle_transport) {bool require_rtcp_mux =rtcp_mux_policy_ == PeerConnectionInterface::kRtcpMuxPolicyRequire;std::string transport_name =bundle_transport ? *bundle_transport : content->name;cricket::DtlsTransportInternal* rtp_dtls_transport =transport_controller_->CreateDtlsTransport(transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);cricket::DtlsTransportInternal* rtcp_dtls_transport = nullptr;if (!require_rtcp_mux) {rtcp_dtls_transport = transport_controller_->CreateDtlsTransport(transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTCP);}video_channel_.reset(channel_manager_->CreateVideoChannel(call_, media_config_, rtp_dtls_transport, rtcp_dtls_transport,transport_controller_->signaling_thread(), content->name, SrtpRequired(),video_options_));if (!video_channel_) {transport_controller_->DestroyDtlsTransport(transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);if (rtcp_dtls_transport) {transport_controller_->DestroyDtlsTransport(transport_name, cricket::ICE_CANDIDATE_COMPONENT_RTP);}return false;}video_channel_->SignalRtcpMuxFullyActive.connect(this, &WebRtcSession::DestroyRtcpTransport_n);video_channel_->SignalDtlsSrtpSetupFailure.connect(this, &WebRtcSession::OnDtlsSrtpSetupFailure);SignalVideoChannelCreated();video_channel_->SignalSentPacket.connect(this,&WebRtcSession::OnSentPacket_w);return true;
}

调用channel_manager的CreateVideoChannel创建BaseChannel基类的cricket::VideoChannel.
VideoChannel需要传入VideoMediaChannel作为构造参数:

//pc/channelmanager.h/cc
VideoChannel* ChannelManager::CreateVideoChannel_w(webrtc::Call* call,const cricket::MediaConfig& media_config,DtlsTransportInternal* rtp_dtls_transport,DtlsTransportInternal* rtcp_dtls_transport,rtc::PacketTransportInternal* rtp_packet_transport,rtc::PacketTransportInternal* rtcp_packet_transport,rtc::Thread* signaling_thread,const std::string& content_name,bool srtp_required,const VideoOptions& options) {RTC_DCHECK(initialized_);RTC_DCHECK(worker_thread_ == rtc::Thread::Current());RTC_DCHECK(nullptr != call);VideoMediaChannel* media_channel = media_engine_->CreateVideoChannel(call, media_config, options);if (media_channel == NULL) {return NULL;}VideoChannel* video_channel = new VideoChannel(worker_thread_, network_thread_, signaling_thread, media_channel,content_name, rtcp_packet_transport == nullptr, srtp_required);if (!video_channel->Init_w(rtp_dtls_transport, rtcp_dtls_transport,rtp_packet_transport, rtcp_packet_transport)) {delete video_channel;return NULL;}video_channels_.push_back(video_channel);return video_channel;
}

VideoMediaChannel实例media_channel由MediaEngineInterface对象media_engine创建,media_engine由ChannelManager构造方法传入并初始化,ChannelManager由PeerConnectionFactory创建,在PeerConnection初始化方法中,media_engine被创建:

//pc/peerconnectionfactory.cc
bool PeerConnectionFactory::Initialize() {RTC_DCHECK(signaling_thread_->IsCurrent());rtc::InitRandom(rtc::Time32());default_network_manager_.reset(new rtc::BasicNetworkManager());if (!default_network_manager_) {return false;}default_socket_factory_.reset(new rtc::BasicPacketSocketFactory(network_thread_));if (!default_socket_factory_) {return false;}std::unique_ptr<cricket::MediaEngineInterface> media_engine =worker_thread_->Invoke<std::unique_ptr<cricket::MediaEngineInterface>>(RTC_FROM_HERE,rtc::Bind(&PeerConnectionFactory::CreateMediaEngine_w, this));channel_manager_.reset(new cricket::ChannelManager(std::move(media_engine), worker_thread_, network_thread_));channel_manager_->SetVideoRtxEnabled(true);if (!channel_manager_->Init()) {return false;}return true;
}std::unique_ptr<cricket::MediaEngineInterface>
PeerConnectionFactory::CreateMediaEngine_w() {RTC_DCHECK(worker_thread_ == rtc::Thread::Current());return std::unique_ptr<cricket::MediaEngineInterface>(cricket::WebRtcMediaEngineFactory::Create(default_adm_.get(), audio_encoder_factory_,audio_decoder_factory_,video_encoder_factory_.get(), video_decoder_factory_.get(),external_audio_mixer_));
}

WebRtcMediaEngine2继承自CompositeMediaEngine,CompositeMediaEngine父类MediaEngineInterface有WebRtcVoiceEngine voiceWebRtcVideoEngine2 video两个对象

WebRtcVideoEngine2

WebRtcVideoEngine2定义在media/engine/webrtcvideoengine2.h下,用于创建WebRtcVideoChannel2(定义在同一头文件),WebRtcVideoChannel2定义了WebRtcVideoSendStream与WebRtcVideoReceiveStream两个内部类.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/50795.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器学习第四十九周周报 GT

文章目录 week49 GY摘要Abstract1. 题目2. Abstract3. 网络结构3.1 graphon3.2 框架概览 4. 文献解读4.1 Introduction4.2 创新点4.3 实验过程4.3.1 有效性4.3.2 可转移性4.3.3 消融研究4.3.4 运行时间 5. 结论6.代码复现小结参考文献 week49 GY 摘要 本周阅读了题为Fine-tun…

46、PHP实现矩阵中的路径

题目&#xff1a; PHP实现矩阵中的路径 描述&#xff1a; 请设计一个函数&#xff0c;用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。 路径可以从矩阵中的任意一个格子开始&#xff0c;每一步可以在矩阵中向左&#xff0c;向右&#xff0c;向上&#xff0c;向…

几个小创新模型,Transformer与SVM、LSTM、BiLSTM、Adaboost的结合,MATLAB分类全家桶再更新!...

截止到本期MATLAB机器学习分类全家桶&#xff0c;一共发了5篇&#xff0c;参考文章如下&#xff1a; 1.机器学习分类全家桶&#xff0c;模式识别&#xff0c;故障诊断的看这一篇绝对够了&#xff01;MATLAB代码 2. 再更新&#xff0c;机器学习分类全家桶&#xff0c;模式识别&a…

【四】jdk8基于m2芯片arm架构Ubuntu24虚拟机下载与安装

文章目录 1. 安装版本2. 开始安装3. 集群安装 1. 安装版本 如无特别说明&#xff0c;本文均在root权限下安装。进入oracle官网&#xff1a;https://www.oracle.com/java/technologies/downloads/找到最下面Java SE 看到java 8&#xff0c;下载使用 ARM64 Compressed Archive版…

vue3+vite纯前端实现自动触发浏览器刷新更新版本内容,并在打包时生成版本号文件

前言 在前端项目中&#xff0c;有时候为了实现自动触发浏览器刷新并更新版本内容&#xff0c;可以采取一系列巧妙的措施。我的项目中是需要在打包时候生成一个version.js文件&#xff0c;用当前打包时间作为版本的唯一标识&#xff0c;然后打包发版 &#xff0c;从实现对版本更…

五大设备制造商的 200 多种机型的安全启动功能完全失效

2012 年&#xff0c;一个由硬件和软件制造商组成的行业联盟采用了安全启动技术&#xff0c;以防范长期存在的安全威胁。这种威胁是恶意软件的幽灵&#xff0c;它可以感染 BIOS&#xff0c;即每次计算机启动时加载操作系统的固件。从那里&#xff0c;它可以保持不受检测和删除&a…

从零开始学Java(超详细韩顺平老师笔记梳理)08——面向对象编程中级(上)IDEA常用快捷键、包、封装、继承

文章目录 前言一、IDEA使用常用快捷键模板/自定义模板 二、包package1. 基本介绍2. 包的命名规范3. 常用的包和如何引入4. 注意事项和细节 三、访问修饰符&#xff08;四类&#xff09;四、封装Encapsulation&#xff08;重点&#xff09;1. 封装介绍2. 封装步骤3. 快速入门4. …

单链表的建立

一.前言 单链表的建立一共有两种方法&#xff0c;一种是头插法&#xff0c;将元素插入在链表的头部&#xff0c;也叫前插法。另外一种则就是尾插法&#xff0c;将元素插入在链表尾部&#xff0c;也叫后插法。 二. 头插法 首先从一个空表开始&#xff0c;重复读入数据&#xff1…

金蝶插件调用HTTP请求 并解析JSON

返回数据如下&#xff1a; { "mainTable": { "create_time": "2023-01-03", "RECORD_DATE": "2023-01-03", "id": "6", "ASSEMBLY_ITEM": ""…

SpringCloud Nacos的配置与使用

Spring Cloud Nacos的配置与使用 文章目录 Spring Cloud Nacos的配置与使用1. 简单介绍2. 环境搭建3. 服务注册/服务发现4. Nacos 负载均衡4.1 服务下线4.2 权重配置4.3 同集群优先访问 5. Nacos 健康检查5.1 两种健康检查机制5.2 服务实例类型 6.Nacos 环境隔离6.1 创建namesp…

【MySQL进阶之路 | 高级篇】表级锁之S锁,X锁,意向锁

1. 从数据操作的粒度划分&#xff1a;表级锁&#xff0c;页级锁&#xff0c;行锁 为了尽可能提高数据库的并发度&#xff0c;每次锁定的数据范围越小越好&#xff0c;理论上每次只锁定当前操作的数据的方案会得到最大的并发度&#xff0c;但是管理锁是很耗资源的事情&#xff…

windows使用ssh-agent管理私钥

主要有以下几个方面: 开启openssh 的 ssh-agent 服务 打开powershell 输入 Get-Service -Name ssh-agent 查看服务是否起来Start-Service ssh-agent 启动服务Stop-Service ssh-agent 关闭服务将私钥添加到ssh-agent 添加私钥 ssh-add ~/.ssh/id_rsa查询添加哪些私钥 ssh-add -…

驾驭代码的无形疆界:动态内存管理揭秘

目录 1.:为什么要有动态内存分配 2.malloc和free 2.1:malloc 2.2:free 3.calloc和realloc 3.1:calloc 3.1.1:代码1(malloc) 3.1.2:代码2(calloc) 3.2:realloc 3.2.1:原地扩容 3.2.2:异地扩容 3.2.3:代码1(原地扩容) 3.2.3:代码2(异地扩容) 4:常见的动态内存的错误…

【windows Docker desktop】在git bash中报错 docker: command not found 解决办法

【windows Docker desktop】在git bash中报错 docker: command not found 解决办法 1. 首先检查在windows中环境变量是否设置成功2. 检查docker在git bash中环境变量是否配置3. 重新加载终端配置4. 最后在校验一下是否配置成功 1. 首先检查在windows中环境变量是否设置成功 启…

vite + xlsx + xlsx-style 导出 Excel

如下 npm i 依赖 npm i xlsxnpm i xlsx-style-vite1、简单的使用&#xff1a;.vue文件中使用 const dataSource ref([]) // 数据源const columns [{title: 用户名,key: userName,width: 120,},{title: 用户组,key: userGroup,width: 120,},{title: 状态,key: enable,width: …

Vue插槽 (Slots)详解

目录 前言基础插槽具名插槽作用域插槽默认插槽动态插槽名总结相关阅读 前言 Vue的插槽&#xff08;Slots&#xff09;是一个非常强大的特性&#xff0c;它允许你在组件的模板中嵌入父组件的内容。插槽使得组件之间的内容分发变得灵活&#xff0c;尤其在构建可复用组件时非常…

鸿蒙(HarmonyOS)下拉选择控件

一、操作环境 操作系统: Windows 11 专业版、IDE:DevEco Studio 3.1.1 Release、SDK:HarmonyOS 3.1.0&#xff08;API 9&#xff09; 二、效果图 三、代码 SelectPVComponent.ets Component export default struct SelectPVComponent {Link selection: SelectOption[]priva…

浅谈我对RESTful架构的理解

总结说在前面&#xff1a; RESTful API是目前比较成熟的一套互联网应用程序的 API 设计理论&#xff0c;他是一种理论规范&#xff0c;方便不同的前端设备与后端进行通信&#xff0c;在 RESTful 风格的 API 设计架构中&#xff0c;每个网址代表一种资源&#xff08;resource&am…

maven介绍 搭建Nexus3(maven私服搭建)

Maven是一个强大的项目管理工具&#xff0c;它基于项目对象模型&#xff08;POM&#xff1a;Project Object Model&#xff09;的概念&#xff0c;通过XML格式的配置文件&#xff08;pom.xml&#xff09;来管理项目的构建 Maven确实可以被视为一种工程管理工具或项目自动化构…

飞凌嵌入式技术创新日深圳站,8月26日见!

飞凌嵌入式技术创新日&#xff08;深圳站&#xff09;将于8月26日举行&#xff0c;一场嵌入式前沿科技的高端局就在眼前。届时&#xff0c;将有多位重量级技术大咖出席&#xff0c;为大家分享最新的研究成果、独到的行业见解和典型的应用案例&#xff0c;紧密结合当前行业热点和…