WebRtcTransport.cpp 58 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545
  1. /*
  2. * Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
  3. *
  4. * This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
  5. *
  6. * Use of this source code is governed by MIT-like license that can be found in the
  7. * LICENSE file in the root of the source tree. All contributing project authors
  8. * may be found in the AUTHORS file in the root of the source tree.
  9. */
  10. #include <iostream>
  11. #include <srtp2/srtp.h>
  12. #include "Util/base64.h"
  13. #include "Network/sockutil.h"
  14. #include "Common/config.h"
  15. #include "Nack.h"
  16. #include "RtpExt.h"
  17. #include "Rtcp/Rtcp.h"
  18. #include "Rtcp/RtcpFCI.h"
  19. #include "Rtcp/RtcpContext.h"
  20. #include "Rtsp/Rtsp.h"
  21. #include "Rtsp/RtpReceiver.h"
  22. #include "WebRtcTransport.h"
  23. #include "WebRtcEchoTest.h"
  24. #include "WebRtcPlayer.h"
  25. #include "WebRtcPusher.h"
  26. #include "Rtsp/RtspMediaSourceImp.h"
  27. #define RTP_SSRC_OFFSET 1
  28. #define RTX_SSRC_OFFSET 2
  29. #define RTP_CNAME "zlmediakit-rtp"
  30. #define RTP_LABEL "zlmediakit-label"
  31. #define RTP_MSLABEL "zlmediakit-mslabel"
  32. using namespace std;
  33. namespace mediakit {
  34. // RTC配置项目 [AUTO-TRANSLATED:19940011]
  35. // RTC configuration project
  36. namespace Rtc {
  37. #define RTC_FIELD "rtc."
  38. // rtp和rtcp接受超时时间 [AUTO-TRANSLATED:0f318cc0]
  39. // rtp and rtcp receive timeout
  40. const string kTimeOutSec = RTC_FIELD "timeoutSec";
  41. // 服务器外网ip [AUTO-TRANSLATED:23283ba6]
  42. // Server external network ip
  43. const string kExternIP = RTC_FIELD "externIP";
  44. // 设置remb比特率,非0时关闭twcc并开启remb。该设置在rtc推流时有效,可以控制推流画质 [AUTO-TRANSLATED:412801db]
  45. // Set remb bitrate, when it is not 0, turn off twcc and turn on remb. This setting is valid when rtc pushes the stream, and can control the pushing stream quality
  46. const string kRembBitRate = RTC_FIELD "rembBitRate";
  47. // webrtc单端口udp服务器 [AUTO-TRANSLATED:d17271ea]
  48. // webrtc single-port udp server
  49. const string kPort = RTC_FIELD "port";
  50. const string kTcpPort = RTC_FIELD "tcpPort";
  51. // 比特率设置 [AUTO-TRANSLATED:2c75f5bc]
  52. // Bitrate setting
  53. const string kStartBitrate = RTC_FIELD "start_bitrate";
  54. const string kMaxBitrate = RTC_FIELD "max_bitrate";
  55. const string kMinBitrate = RTC_FIELD "min_bitrate";
  56. // 数据通道设置 [AUTO-TRANSLATED:2dc48bc3]
  57. // Data channel setting
  58. const string kDataChannelEcho = RTC_FIELD "datachannel_echo";
  59. static onceToken token([]() {
  60. mINI::Instance()[kTimeOutSec] = 15;
  61. mINI::Instance()[kExternIP] = "";
  62. mINI::Instance()[kRembBitRate] = 0;
  63. mINI::Instance()[kPort] = 8000;
  64. mINI::Instance()[kTcpPort] = 8000;
  65. mINI::Instance()[kStartBitrate] = 0;
  66. mINI::Instance()[kMaxBitrate] = 0;
  67. mINI::Instance()[kMinBitrate] = 0;
  68. mINI::Instance()[kDataChannelEcho] = true;
  69. });
  70. } // namespace RTC
  71. static atomic<uint64_t> s_key { 0 };
  72. static void translateIPFromEnv(std::vector<std::string> &v) {
  73. for (auto iter = v.begin(); iter != v.end();) {
  74. if (start_with(*iter, "$")) {
  75. auto ip = toolkit::getEnv(*iter);
  76. if (ip.empty()) {
  77. iter = v.erase(iter);
  78. } else {
  79. *iter++ = ip;
  80. }
  81. } else {
  82. ++iter;
  83. }
  84. }
  85. }
  86. static std::string getServerPrefix() {
  87. // stun_user_name格式: base64(ip+udp_port+tcp_port) + _ + number [AUTO-TRANSLATED:cc3c5902]
  88. // stun_user_name format: base64(ip+udp_port+tcp_port) + _ + number
  89. // 其中ip为二进制char[4], udp_port/tcp_port为大端 uint16. [AUTO-TRANSLATED:92ea5521]
  90. // Where ip is binary char[4], udp_port/tcp_port is big-endian uint16.
  91. // number为自增长数,确保短时间内唯一 [AUTO-TRANSLATED:d31aada9]
  92. // number is an auto-incrementing number, ensuring uniqueness in a short period of time
  93. GET_CONFIG(uint16_t, udp_port, Rtc::kPort);
  94. GET_CONFIG(uint16_t, tcp_port, Rtc::kTcpPort);
  95. char buf[8];
  96. auto host = SockUtil::get_local_ip();
  97. auto addr = SockUtil::make_sockaddr(host.data(), udp_port);
  98. // 拷贝ipv4地址 [AUTO-TRANSLATED:49c16eed]
  99. // Copy ipv4 address
  100. memcpy(buf, &(reinterpret_cast<sockaddr_in *>(&addr)->sin_addr), 4);
  101. // 拷贝udp端口 [AUTO-TRANSLATED:ebb750d3]
  102. // Copy udp port
  103. memcpy(buf + 4, &(reinterpret_cast<sockaddr_in *>(&addr)->sin_port), 2);
  104. // tcp端口转大端模式 [AUTO-TRANSLATED:4f2293de]
  105. // Convert tcp port to big-endian mode
  106. addr = SockUtil::make_sockaddr(host.data(), tcp_port);
  107. // 拷贝tcp端口 [AUTO-TRANSLATED:23191878]
  108. // Copy tcp port
  109. memcpy(buf + 6, &(reinterpret_cast<sockaddr_in *>(&addr)->sin_port), 2);
  110. auto ret = encodeBase64(string(buf, 8)) + '_';
  111. InfoL << "MediaServer(" << host << ":" << udp_port << ":" << tcp_port << ") prefix: " << ret;
  112. return ret;
  113. }
  114. const char* sockTypeStr(Session* session) {
  115. if (session) {
  116. switch (session->getSock()->sockType()) {
  117. case SockNum::Sock_TCP: return "tcp";
  118. case SockNum::Sock_UDP: return "udp";
  119. default: break;
  120. }
  121. }
  122. return "unknown";
  123. }
  124. WebRtcTransport::WebRtcTransport(const EventPoller::Ptr &poller) {
  125. _poller = poller;
  126. static auto prefix = getServerPrefix();
  127. _identifier = prefix + to_string(++s_key);
  128. _packet_pool.setSize(64);
  129. }
  130. void WebRtcTransport::onCreate() {
  131. _dtls_transport = std::make_shared<RTC::DtlsTransport>(_poller, this);
  132. _ice_server = std::make_shared<RTC::IceServer>(this, _identifier, makeRandStr(24));
  133. }
  134. void WebRtcTransport::onDestory() {
  135. #ifdef ENABLE_SCTP
  136. _sctp = nullptr;
  137. #endif
  138. _dtls_transport = nullptr;
  139. _ice_server = nullptr;
  140. }
  141. const EventPoller::Ptr &WebRtcTransport::getPoller() const {
  142. return _poller;
  143. }
  144. const string &WebRtcTransport::getIdentifier() const {
  145. return _identifier;
  146. }
  147. const std::string& WebRtcTransport::deleteRandStr() const {
  148. if (_delete_rand_str.empty()) {
  149. _delete_rand_str = makeRandStr(32);
  150. }
  151. return _delete_rand_str;
  152. }
  153. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  154. void WebRtcTransport::OnIceServerSendStunPacket(
  155. const RTC::IceServer *iceServer, const RTC::StunPacket *packet, RTC::TransportTuple *tuple) {
  156. sendSockData((char *)packet->GetData(), packet->GetSize(), tuple);
  157. }
  158. void WebRtcTransportImp::OnIceServerSelectedTuple(const RTC::IceServer *iceServer, RTC::TransportTuple *tuple) {
  159. InfoL << getIdentifier() << " select tuple " << sockTypeStr(tuple) << " " << tuple->get_peer_ip() << ":" << tuple->get_peer_port();
  160. tuple->setSendFlushFlag(false);
  161. unrefSelf();
  162. }
  163. void WebRtcTransport::OnIceServerConnected(const RTC::IceServer *iceServer) {
  164. InfoL << getIdentifier();
  165. }
  166. void WebRtcTransport::OnIceServerCompleted(const RTC::IceServer *iceServer) {
  167. InfoL << getIdentifier();
  168. if (_answer_sdp->media[0].role == DtlsRole::passive) {
  169. _dtls_transport->Run(RTC::DtlsTransport::Role::SERVER);
  170. } else {
  171. _dtls_transport->Run(RTC::DtlsTransport::Role::CLIENT);
  172. }
  173. }
  174. void WebRtcTransport::OnIceServerDisconnected(const RTC::IceServer *iceServer) {
  175. InfoL << getIdentifier();
  176. }
  177. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  178. void WebRtcTransport::OnDtlsTransportConnected(
  179. const RTC::DtlsTransport *dtlsTransport, RTC::SrtpSession::CryptoSuite srtpCryptoSuite, uint8_t *srtpLocalKey,
  180. size_t srtpLocalKeyLen, uint8_t *srtpRemoteKey, size_t srtpRemoteKeyLen, std::string &remoteCert) {
  181. InfoL << getIdentifier();
  182. _srtp_session_send = std::make_shared<RTC::SrtpSession>(
  183. RTC::SrtpSession::Type::OUTBOUND, srtpCryptoSuite, srtpLocalKey, srtpLocalKeyLen);
  184. _srtp_session_recv = std::make_shared<RTC::SrtpSession>(
  185. RTC::SrtpSession::Type::INBOUND, srtpCryptoSuite, srtpRemoteKey, srtpRemoteKeyLen);
  186. #ifdef ENABLE_SCTP
  187. _sctp = std::make_shared<RTC::SctpAssociationImp>(getPoller(), this, 128, 128, 262144, true);
  188. _sctp->TransportConnected();
  189. #endif
  190. onStartWebRTC();
  191. }
  192. #pragma pack(push, 1)
  193. struct DtlsHeader {
  194. uint8_t content_type;
  195. uint16_t dtls_version;
  196. uint16_t epoch;
  197. uint8_t seq[6];
  198. uint16_t length;
  199. uint8_t payload[1];
  200. };
  201. #pragma pack(pop)
  202. void WebRtcTransport::OnDtlsTransportSendData(
  203. const RTC::DtlsTransport *dtlsTransport, const uint8_t *data, size_t len) {
  204. size_t offset = 0;
  205. while(offset < len) {
  206. auto *header = reinterpret_cast<const DtlsHeader *>(data + offset);
  207. auto length = ntohs(header->length) + offsetof(DtlsHeader, payload);
  208. sendSockData((char *)data + offset, length, nullptr);
  209. offset += length;
  210. }
  211. }
  212. void WebRtcTransport::OnDtlsTransportConnecting(const RTC::DtlsTransport *dtlsTransport) {
  213. InfoL << getIdentifier();
  214. }
  215. void WebRtcTransport::OnDtlsTransportFailed(const RTC::DtlsTransport *dtlsTransport) {
  216. InfoL << getIdentifier();
  217. onShutdown(SockException(Err_shutdown, "dtls transport failed"));
  218. }
  219. void WebRtcTransport::OnDtlsTransportClosed(const RTC::DtlsTransport *dtlsTransport) {
  220. InfoL << getIdentifier();
  221. onShutdown(SockException(Err_shutdown, "dtls close notify received"));
  222. }
  223. void WebRtcTransport::OnDtlsTransportApplicationDataReceived(
  224. const RTC::DtlsTransport *dtlsTransport, const uint8_t *data, size_t len) {
  225. #ifdef ENABLE_SCTP
  226. _sctp->ProcessSctpData(data, len);
  227. #else
  228. InfoL << hexdump(data, len);
  229. #endif
  230. }
  231. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  232. #ifdef ENABLE_SCTP
  233. void WebRtcTransport::OnSctpAssociationConnecting(RTC::SctpAssociation *sctpAssociation) {
  234. TraceL << getIdentifier();
  235. try {
  236. NOTICE_EMIT(BroadcastRtcSctpConnectArgs, Broadcast::kBroadcastRtcSctpConnecting, *this);
  237. } catch (std::exception &ex) {
  238. WarnL << "Exception occurred: " << ex.what();
  239. }
  240. }
  241. void WebRtcTransport::OnSctpAssociationConnected(RTC::SctpAssociation *sctpAssociation) {
  242. InfoL << getIdentifier();
  243. try {
  244. NOTICE_EMIT(BroadcastRtcSctpConnectArgs, Broadcast::kBroadcastRtcSctpConnected, *this);
  245. } catch (std::exception &ex) {
  246. WarnL << "Exception occurred: " << ex.what();
  247. }
  248. }
  249. void WebRtcTransport::OnSctpAssociationFailed(RTC::SctpAssociation *sctpAssociation) {
  250. WarnL << getIdentifier();
  251. try {
  252. NOTICE_EMIT(BroadcastRtcSctpConnectArgs, Broadcast::kBroadcastRtcSctpFailed, *this);
  253. } catch (std::exception &ex) {
  254. WarnL << "Exception occurred: " << ex.what();
  255. }
  256. }
  257. void WebRtcTransport::OnSctpAssociationClosed(RTC::SctpAssociation *sctpAssociation) {
  258. InfoL << getIdentifier();
  259. try {
  260. NOTICE_EMIT(BroadcastRtcSctpConnectArgs, Broadcast::kBroadcastRtcSctpClosed, *this);
  261. } catch (std::exception &ex) {
  262. WarnL << "Exception occurred: " << ex.what();
  263. }
  264. }
  265. void WebRtcTransport::OnSctpAssociationSendData(
  266. RTC::SctpAssociation *sctpAssociation, const uint8_t *data, size_t len) {
  267. try {
  268. NOTICE_EMIT(BroadcastRtcSctpSendArgs, Broadcast::kBroadcastRtcSctpSend, *this, data, len);
  269. } catch (std::exception &ex) {
  270. WarnL << "Exception occurred: " << ex.what();
  271. }
  272. _dtls_transport->SendApplicationData(data, len);
  273. }
  274. void WebRtcTransport::OnSctpAssociationMessageReceived(
  275. RTC::SctpAssociation *sctpAssociation, uint16_t streamId, uint32_t ppid, const uint8_t *msg, size_t len) {
  276. InfoL << getIdentifier() << " " << streamId << " " << ppid << " " << len << " " << string((char *)msg, len);
  277. RTC::SctpStreamParameters params;
  278. params.streamId = streamId;
  279. GET_CONFIG(bool, datachannel_echo, Rtc::kDataChannelEcho);
  280. if (datachannel_echo) {
  281. // 回显数据 [AUTO-TRANSLATED:7868d3a4]
  282. // Echo data
  283. _sctp->SendSctpMessage(params, ppid, msg, len);
  284. }
  285. try {
  286. NOTICE_EMIT(BroadcastRtcSctpReceivedArgs, Broadcast::kBroadcastRtcSctpReceived, *this, streamId, ppid, msg, len);
  287. } catch (std::exception &ex) {
  288. WarnL << "Exception occurred: " << ex.what();
  289. }
  290. }
  291. #endif
  292. void WebRtcTransport::sendDatachannel(uint16_t streamId, uint32_t ppid, const char *msg, size_t len) {
  293. #ifdef ENABLE_SCTP
  294. if (_sctp) {
  295. RTC::SctpStreamParameters params;
  296. params.streamId = streamId;
  297. _sctp->SendSctpMessage(params, ppid, (uint8_t *)msg, len);
  298. }
  299. #else
  300. WarnL << "WebRTC datachannel disabled!";
  301. #endif
  302. }
  303. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  304. void WebRtcTransport::sendSockData(const char *buf, size_t len, RTC::TransportTuple *tuple) {
  305. auto pkt = _packet_pool.obtain2();
  306. pkt->assign(buf, len);
  307. onSendSockData(std::move(pkt), true, tuple ? tuple : _ice_server->GetSelectedTuple());
  308. }
  309. Session::Ptr WebRtcTransport::getSession() const {
  310. auto tuple = _ice_server ? _ice_server->GetSelectedTuple(true) : nullptr;
  311. return tuple ? static_pointer_cast<Session>(tuple->shared_from_this()) : nullptr;
  312. }
  313. void WebRtcTransport::sendRtcpRemb(uint32_t ssrc, size_t bit_rate) {
  314. auto remb = FCI_REMB::create({ ssrc }, (uint32_t)bit_rate);
  315. auto fb = RtcpFB::create(PSFBType::RTCP_PSFB_REMB, remb.data(), remb.size());
  316. fb->ssrc = htonl(0);
  317. fb->ssrc_media = htonl(ssrc);
  318. sendRtcpPacket((char *)fb.get(), fb->getSize(), true);
  319. }
  320. void WebRtcTransport::sendRtcpPli(uint32_t ssrc) {
  321. auto pli = RtcpFB::create(PSFBType::RTCP_PSFB_PLI);
  322. pli->ssrc = htonl(0);
  323. pli->ssrc_media = htonl(ssrc);
  324. sendRtcpPacket((char *)pli.get(), pli->getSize(), true);
  325. }
  326. string getFingerprint(const string &algorithm_str, const std::shared_ptr<RTC::DtlsTransport> &transport) {
  327. auto algorithm = RTC::DtlsTransport::GetFingerprintAlgorithm(algorithm_str);
  328. for (auto &finger_prints : transport->GetLocalFingerprints()) {
  329. if (finger_prints.algorithm == algorithm) {
  330. return finger_prints.value;
  331. }
  332. }
  333. throw std::invalid_argument(StrPrinter << "不支持的加密算法:" << algorithm_str);
  334. }
  335. void WebRtcTransport::setRemoteDtlsFingerprint(const RtcSession &remote) {
  336. // 设置远端dtls签名 [AUTO-TRANSLATED:746d5f9c]
  337. // Set remote dtls signature
  338. RTC::DtlsTransport::Fingerprint remote_fingerprint;
  339. remote_fingerprint.algorithm
  340. = RTC::DtlsTransport::GetFingerprintAlgorithm(_offer_sdp->media[0].fingerprint.algorithm);
  341. remote_fingerprint.value = _offer_sdp->media[0].fingerprint.hash;
  342. _dtls_transport->SetRemoteFingerprint(remote_fingerprint);
  343. }
  344. void WebRtcTransport::onRtcConfigure(RtcConfigure &configure) const {
  345. SdpAttrFingerprint fingerprint;
  346. fingerprint.algorithm = _offer_sdp->media[0].fingerprint.algorithm;
  347. fingerprint.hash = getFingerprint(fingerprint.algorithm, _dtls_transport);
  348. configure.setDefaultSetting(
  349. _ice_server->GetUsernameFragment(), _ice_server->GetPassword(), RtpDirection::sendrecv, fingerprint);
  350. // 开启remb后关闭twcc,因为开启twcc后remb无效 [AUTO-TRANSLATED:8a8feca2]
  351. // Turn off twcc after turning on remb, because remb is invalid after turning on twcc
  352. GET_CONFIG(size_t, remb_bit_rate, Rtc::kRembBitRate);
  353. configure.enableTWCC(!remb_bit_rate);
  354. }
  355. static void setSdpBitrate(RtcSession &sdp) {
  356. GET_CONFIG(size_t, max_bitrate, Rtc::kMaxBitrate);
  357. GET_CONFIG(size_t, min_bitrate, Rtc::kMinBitrate);
  358. GET_CONFIG(size_t, start_bitrate, Rtc::kStartBitrate);
  359. auto m = (RtcMedia *)(sdp.getMedia(TrackType::TrackVideo));
  360. if (m) {
  361. auto &plan = m->plan[0];
  362. if (max_bitrate) plan.fmtp["x-google-max-bitrate"] = std::to_string(max_bitrate);
  363. if (min_bitrate) plan.fmtp["x-google-min-bitrate"] = std::to_string(min_bitrate);
  364. if (start_bitrate) plan.fmtp["x-google-start-bitrate"] = std::to_string(start_bitrate);
  365. }
  366. }
  367. std::string WebRtcTransport::getAnswerSdp(const string &offer) {
  368. try {
  369. // // 解析offer sdp //// [AUTO-TRANSLATED:87c1f337]
  370. // // Parse offer sdp ////
  371. _offer_sdp = std::make_shared<RtcSession>();
  372. _offer_sdp->loadFrom(offer);
  373. onCheckSdp(SdpType::offer, *_offer_sdp);
  374. _offer_sdp->checkValid();
  375. setRemoteDtlsFingerprint(*_offer_sdp);
  376. // // sdp 配置 //// [AUTO-TRANSLATED:718a72e2]
  377. // // sdp configuration ////
  378. RtcConfigure configure;
  379. onRtcConfigure(configure);
  380. // // 生成answer sdp //// [AUTO-TRANSLATED:a139475e]
  381. // // Generate answer sdp ////
  382. _answer_sdp = configure.createAnswer(*_offer_sdp);
  383. onCheckSdp(SdpType::answer, *_answer_sdp);
  384. setSdpBitrate(*_answer_sdp);
  385. _answer_sdp->checkValid();
  386. return _answer_sdp->toString();
  387. } catch (exception &ex) {
  388. onShutdown(SockException(Err_shutdown, ex.what()));
  389. throw;
  390. }
  391. }
  392. static bool isDtls(char *buf) {
  393. return ((*buf > 19) && (*buf < 64));
  394. }
  395. void WebRtcTransport::inputSockData(char *buf, int len, RTC::TransportTuple *tuple) {
  396. if (RTC::StunPacket::IsStun((const uint8_t *)buf, len)) {
  397. std::unique_ptr<RTC::StunPacket> packet(RTC::StunPacket::Parse((const uint8_t *)buf, len));
  398. if (!packet) {
  399. WarnL << "parse stun error";
  400. return;
  401. }
  402. _ice_server->ProcessStunPacket(packet.get(), tuple);
  403. return;
  404. }
  405. if (isDtls(buf)) {
  406. _dtls_transport->ProcessDtlsData((uint8_t *)buf, len);
  407. return;
  408. }
  409. if (isRtp(buf, len)) {
  410. if (!_srtp_session_recv) {
  411. WarnL << "received rtp packet when dtls not completed from:" << tuple->get_peer_ip();
  412. return;
  413. }
  414. if (_srtp_session_recv->DecryptSrtp((uint8_t *)buf, &len)) {
  415. onRtp(buf, len, _ticker.createdTime());
  416. }
  417. return;
  418. }
  419. if (isRtcp(buf, len)) {
  420. if (!_srtp_session_recv) {
  421. WarnL << "received rtcp packet when dtls not completed from:" << tuple->get_peer_ip();
  422. return;
  423. }
  424. if (_srtp_session_recv->DecryptSrtcp((uint8_t *)buf, &len)) {
  425. onRtcp(buf, len);
  426. }
  427. return;
  428. }
  429. }
  430. void WebRtcTransport::sendRtpPacket(const char *buf, int len, bool flush, void *ctx) {
  431. if (_srtp_session_send) {
  432. auto pkt = _packet_pool.obtain2();
  433. // 预留rtx加入的两个字节 [AUTO-TRANSLATED:d1eb5cd7]
  434. // Reserve two bytes for rtx joining
  435. pkt->setCapacity((size_t)len + SRTP_MAX_TRAILER_LEN + 2);
  436. memcpy(pkt->data(), buf, len);
  437. onBeforeEncryptRtp(pkt->data(), len, ctx);
  438. if (_srtp_session_send->EncryptRtp(reinterpret_cast<uint8_t *>(pkt->data()), &len)) {
  439. pkt->setSize(len);
  440. onSendSockData(std::move(pkt), flush);
  441. }
  442. }
  443. }
  444. void WebRtcTransport::sendRtcpPacket(const char *buf, int len, bool flush, void *ctx) {
  445. if (_srtp_session_send) {
  446. auto pkt = _packet_pool.obtain2();
  447. // 预留rtx加入的两个字节 [AUTO-TRANSLATED:d1eb5cd7]
  448. // Reserve two bytes for rtx joining
  449. pkt->setCapacity((size_t)len + SRTP_MAX_TRAILER_LEN + 2);
  450. memcpy(pkt->data(), buf, len);
  451. onBeforeEncryptRtcp(pkt->data(), len, ctx);
  452. if (_srtp_session_send->EncryptRtcp(reinterpret_cast<uint8_t *>(pkt->data()), &len)) {
  453. pkt->setSize(len);
  454. onSendSockData(std::move(pkt), flush);
  455. }
  456. }
  457. }
  458. ///////////////////////////////////////////////////////////////////////////////////
  459. void WebRtcTransportImp::onCreate() {
  460. WebRtcTransport::onCreate();
  461. registerSelf();
  462. weak_ptr<WebRtcTransportImp> weak_self = static_pointer_cast<WebRtcTransportImp>(shared_from_this());
  463. GET_CONFIG(float, timeoutSec, Rtc::kTimeOutSec);
  464. _timer = std::make_shared<Timer>(
  465. timeoutSec / 2,
  466. [weak_self]() {
  467. auto strong_self = weak_self.lock();
  468. if (!strong_self) {
  469. return false;
  470. }
  471. if (strong_self->_alive_ticker.elapsedTime() > timeoutSec * 1000) {
  472. strong_self->onShutdown(SockException(Err_timeout, "接受rtp/rtcp/datachannel超时"));
  473. }
  474. return true;
  475. },
  476. getPoller());
  477. _twcc_ctx.setOnSendTwccCB([this](uint32_t ssrc, string fci) { onSendTwcc(ssrc, fci); });
  478. }
  479. void WebRtcTransportImp::OnDtlsTransportApplicationDataReceived(const RTC::DtlsTransport *dtlsTransport, const uint8_t *data, size_t len) {
  480. WebRtcTransport::OnDtlsTransportApplicationDataReceived(dtlsTransport, data, len);
  481. #ifdef ENABLE_SCTP
  482. if (_answer_sdp->isOnlyDatachannel()) {
  483. _alive_ticker.resetTime();
  484. }
  485. #endif
  486. }
  487. WebRtcTransportImp::WebRtcTransportImp(const EventPoller::Ptr &poller) : WebRtcTransport(poller) {
  488. InfoL << getIdentifier();
  489. }
  490. WebRtcTransportImp::~WebRtcTransportImp() {
  491. InfoL << getIdentifier();
  492. }
  493. void WebRtcTransportImp::onDestory() {
  494. WebRtcTransport::onDestory();
  495. unregisterSelf();
  496. }
  497. void WebRtcTransportImp::onSendSockData(Buffer::Ptr buf, bool flush, RTC::TransportTuple *tuple) {
  498. if (tuple == nullptr) {
  499. tuple = _ice_server->GetSelectedTuple();
  500. if (!tuple) {
  501. WarnL << "send data failed:" << buf->size();
  502. return;
  503. }
  504. }
  505. // 一次性发送一帧的rtp数据,提高网络io性能 [AUTO-TRANSLATED:fbab421e]
  506. // Send one frame of rtp data at a time to improve network io performance
  507. if (tuple->getSock()->sockType() == SockNum::Sock_TCP) {
  508. // 增加tcp两字节头 [AUTO-TRANSLATED:62159f79]
  509. // Add two-byte header to tcp
  510. auto len = buf->size();
  511. char tcp_len[2] = { 0 };
  512. tcp_len[0] = (len >> 8) & 0xff;
  513. tcp_len[1] = len & 0xff;
  514. tuple->SockSender::send(tcp_len, 2);
  515. }
  516. tuple->send(std::move(buf));
  517. if (flush) {
  518. tuple->flushAll();
  519. }
  520. }
  521. ///////////////////////////////////////////////////////////////////
  522. bool WebRtcTransportImp::canSendRtp() const {
  523. for (auto &m : _answer_sdp->media) {
  524. if (m.direction == RtpDirection::sendrecv || m.direction == RtpDirection::sendonly) {
  525. return true;
  526. }
  527. }
  528. return false;
  529. }
  530. bool WebRtcTransportImp::canRecvRtp() const {
  531. for (auto &m : _answer_sdp->media) {
  532. if (m.direction == RtpDirection::sendrecv || m.direction == RtpDirection::recvonly) {
  533. return true;
  534. }
  535. }
  536. return false;
  537. }
  538. void WebRtcTransportImp::onStartWebRTC() {
  539. // 获取ssrc和pt相关信息,届时收到rtp和rtcp时分别可以根据pt和ssrc找到相关的信息 [AUTO-TRANSLATED:39828247]
  540. // Get ssrc and pt related information, so that when receiving rtp and rtcp, you can find the relevant information according to pt and ssrc respectively
  541. for (auto &m_answer : _answer_sdp->media) {
  542. if (m_answer.type == TrackApplication) {
  543. continue;
  544. }
  545. auto m_offer = _offer_sdp->getMedia(m_answer.type);
  546. auto track = std::make_shared<MediaTrack>();
  547. track->media = &m_answer;
  548. track->answer_ssrc_rtp = m_answer.getRtpSSRC();
  549. track->answer_ssrc_rtx = m_answer.getRtxSSRC();
  550. track->offer_ssrc_rtp = m_offer->getRtpSSRC();
  551. track->offer_ssrc_rtx = m_offer->getRtxSSRC();
  552. track->plan_rtp = &m_answer.plan[0];
  553. track->plan_rtx = m_answer.getRelatedRtxPlan(track->plan_rtp->pt);
  554. track->rtcp_context_send = std::make_shared<RtcpContextForSend>();
  555. // rtp track type --> MediaTrack
  556. if (m_answer.direction == RtpDirection::sendonly || m_answer.direction == RtpDirection::sendrecv) {
  557. // 该类型的track 才支持发送 [AUTO-TRANSLATED:b7c1e631]
  558. // This type of track supports sending
  559. _type_to_track[m_answer.type] = track;
  560. }
  561. // send ssrc --> MediaTrack
  562. _ssrc_to_track[track->answer_ssrc_rtp] = track;
  563. _ssrc_to_track[track->answer_ssrc_rtx] = track;
  564. // recv ssrc --> MediaTrack
  565. _ssrc_to_track[track->offer_ssrc_rtp] = track;
  566. _ssrc_to_track[track->offer_ssrc_rtx] = track;
  567. // rtp pt --> MediaTrack
  568. _pt_to_track.emplace(
  569. track->plan_rtp->pt, std::unique_ptr<WrappedMediaTrack>(new WrappedRtpTrack(track, _twcc_ctx, *this)));
  570. if (track->plan_rtx) {
  571. // rtx pt --> MediaTrack
  572. _pt_to_track.emplace(track->plan_rtx->pt, std::unique_ptr<WrappedMediaTrack>(new WrappedRtxTrack(track)));
  573. }
  574. // 记录rtp ext类型与id的关系,方便接收或发送rtp时修改rtp ext id [AUTO-TRANSLATED:5736bd34]
  575. // Record the relationship between rtp ext type and id, which is convenient for modifying rtp ext id when receiving or sending rtp
  576. track->rtp_ext_ctx = std::make_shared<RtpExtContext>(m_answer);
  577. weak_ptr<MediaTrack> weak_track = track;
  578. track->rtp_ext_ctx->setOnGetRtp([this, weak_track](uint8_t pt, uint32_t ssrc, const string &rid) {
  579. // ssrc --> MediaTrack
  580. auto track = weak_track.lock();
  581. assert(track);
  582. _ssrc_to_track[ssrc] = std::move(track);
  583. InfoL << "get rtp, pt:" << (int)pt << ", ssrc:" << ssrc << ", rid:" << rid;
  584. });
  585. size_t index = 0;
  586. for (auto &ssrc : m_offer->rtp_ssrc_sim) {
  587. // 记录ssrc对应的MediaTrack [AUTO-TRANSLATED:8e344bc1]
  588. // Record the MediaTrack corresponding to ssrc
  589. _ssrc_to_track[ssrc.ssrc] = track;
  590. if (m_offer->rtp_rids.size() > index) {
  591. // 支持firefox的simulcast, 提前映射好ssrc和rid的关系 [AUTO-TRANSLATED:86f3e5bf]
  592. // Support firefox's simulcast, map the relationship between ssrc and rid in advance
  593. track->rtp_ext_ctx->setRid(ssrc.ssrc, m_offer->rtp_rids[index]);
  594. } else {
  595. // SDP munging没有rid, 它通过group-ssrc:SIM给出ssrc列表; [AUTO-TRANSLATED:d6cd0b5b]
  596. // SDP munging does not have rid, it gives the ssrc list through group-ssrc:SIM;
  597. // 系统又要有rid,这里手工生成rid,并为其绑定ssrc [AUTO-TRANSLATED:f4988139]
  598. // The system also needs a rid, so we manually generate a rid and bind it to the ssrc
  599. std::string rid = "r" + std::to_string(index);
  600. track->rtp_ext_ctx->setRid(ssrc.ssrc, rid);
  601. if (ssrc.rtx_ssrc) {
  602. track->rtp_ext_ctx->setRid(ssrc.rtx_ssrc, rid);
  603. }
  604. }
  605. ++index;
  606. }
  607. }
  608. }
  609. void WebRtcTransportImp::onCheckAnswer(RtcSession &sdp) {
  610. // 修改answer sdp的ip、端口信息 [AUTO-TRANSLATED:ff72ec1e]
  611. // Modify the ip and port information of the answer sdp
  612. GET_CONFIG_FUNC(std::vector<std::string>, extern_ips, Rtc::kExternIP, [](string str) {
  613. std::vector<std::string> ret;
  614. if (str.length()) {
  615. ret = split(str, ",");
  616. }
  617. translateIPFromEnv(ret);
  618. return ret;
  619. });
  620. for (auto &m : sdp.media) {
  621. m.addr.reset();
  622. m.addr.address = extern_ips.empty() ? _local_ip.empty() ? SockUtil::get_local_ip() : _local_ip : extern_ips[0];
  623. m.rtcp_addr.reset();
  624. m.rtcp_addr.address = m.addr.address;
  625. GET_CONFIG(uint16_t, udp_port, Rtc::kPort);
  626. GET_CONFIG(uint16_t, tcp_port, Rtc::kTcpPort);
  627. m.port = m.port ? (udp_port ? udp_port : tcp_port) : 0;
  628. if (m.type != TrackApplication) {
  629. m.rtcp_addr.port = m.port;
  630. }
  631. sdp.origin.address = m.addr.address;
  632. }
  633. if (!canSendRtp()) {
  634. // 设置我们发送的rtp的ssrc [AUTO-TRANSLATED:3704484a]
  635. // Set the ssrc of the rtp we send
  636. return;
  637. }
  638. for (auto &m : sdp.media) {
  639. if (m.type == TrackApplication) {
  640. continue;
  641. }
  642. if (!m.rtp_rtx_ssrc.empty()) {
  643. // 已经生成了ssrc [AUTO-TRANSLATED:5883cab8]
  644. // The ssrc has been generated
  645. continue;
  646. }
  647. // 添加answer sdp的ssrc信息 [AUTO-TRANSLATED:ab4c3fad]
  648. // Add the ssrc information to the answer sdp
  649. m.rtp_rtx_ssrc.emplace_back();
  650. auto &ssrc = m.rtp_rtx_ssrc.back();
  651. // 发送的ssrc我们随便定义,因为在发送rtp时会修改为此值 [AUTO-TRANSLATED:ee8d77f0]
  652. // We can define the ssrc we send at will, because it will be modified to this value when sending rtp
  653. ssrc.ssrc = m.type + RTP_SSRC_OFFSET;
  654. ssrc.cname = RTP_CNAME;
  655. ssrc.label = std::string(RTP_LABEL) + '-' + m.mid;
  656. ssrc.mslabel = RTP_MSLABEL;
  657. ssrc.msid = ssrc.mslabel + ' ' + ssrc.label;
  658. if (m.getRelatedRtxPlan(m.plan[0].pt)) {
  659. // rtx ssrc
  660. ssrc.rtx_ssrc = ssrc.ssrc + RTX_SSRC_OFFSET;
  661. }
  662. }
  663. }
  664. void WebRtcTransportImp::onCheckSdp(SdpType type, RtcSession &sdp) {
  665. switch (type) {
  666. case SdpType::answer:
  667. onCheckAnswer(sdp);
  668. break;
  669. case SdpType::offer:
  670. break;
  671. default: /*不可达*/
  672. assert(0);
  673. break;
  674. }
  675. }
  676. SdpAttrCandidate::Ptr
  677. makeIceCandidate(std::string ip, uint16_t port, uint32_t priority = 100, std::string proto = "udp") {
  678. auto candidate = std::make_shared<SdpAttrCandidate>();
  679. // rtp端口 [AUTO-TRANSLATED:b0addb27]
  680. // rtp port
  681. candidate->component = 1;
  682. candidate->transport = proto;
  683. candidate->foundation = proto + "candidate";
  684. // 优先级,单candidate时随便 [AUTO-TRANSLATED:7c85d820]
  685. // Priority, random when there is only one candidate
  686. candidate->priority = priority;
  687. candidate->address = std::move(ip);
  688. candidate->port = port;
  689. candidate->type = "host";
  690. if (proto == "tcp") {
  691. candidate->type += " tcptype passive";
  692. }
  693. return candidate;
  694. }
  695. void WebRtcTransportImp::onRtcConfigure(RtcConfigure &configure) const {
  696. WebRtcTransport::onRtcConfigure(configure);
  697. if (!_cands.empty()) {
  698. for (auto &cand : _cands) {
  699. configure.addCandidate(cand);
  700. }
  701. return;
  702. }
  703. GET_CONFIG(uint16_t, local_udp_port, Rtc::kPort);
  704. GET_CONFIG(uint16_t, local_tcp_port, Rtc::kTcpPort);
  705. // 添加接收端口candidate信息 [AUTO-TRANSLATED:cc9a6a90]
  706. // Add the receiving port candidate information
  707. GET_CONFIG_FUNC(std::vector<std::string>, extern_ips, Rtc::kExternIP, [](string str) {
  708. std::vector<std::string> ret;
  709. if (str.length()) {
  710. ret = split(str, ",");
  711. }
  712. translateIPFromEnv(ret);
  713. return ret;
  714. });
  715. if (extern_ips.empty()) {
  716. std::string local_ip = _local_ip.empty() ? SockUtil::get_local_ip() : _local_ip;
  717. if (local_udp_port) { configure.addCandidate(*makeIceCandidate(local_ip, local_udp_port, 120, "udp")); }
  718. if (local_tcp_port) { configure.addCandidate(*makeIceCandidate(local_ip, local_tcp_port, _preferred_tcp ? 125 : 115, "tcp")); }
  719. } else {
  720. const uint32_t delta = 10;
  721. uint32_t priority = 100 + delta * extern_ips.size();
  722. for (auto ip : extern_ips) {
  723. if (local_udp_port) { configure.addCandidate(*makeIceCandidate(ip, local_udp_port, priority, "udp")); }
  724. if (local_tcp_port) { configure.addCandidate(*makeIceCandidate(ip, local_tcp_port, priority - (_preferred_tcp ? -5 : 5), "tcp")); }
  725. priority -= delta;
  726. }
  727. }
  728. }
  729. void WebRtcTransportImp::setPreferredTcp(bool flag) {
  730. _preferred_tcp = flag;
  731. }
  732. void WebRtcTransportImp::setLocalIp(std::string local_ip) {
  733. _local_ip = std::move(local_ip);
  734. }
  735. void WebRtcTransportImp::setIceCandidate(vector<SdpAttrCandidate> cands) {
  736. _cands = std::move(cands);
  737. }
  738. ///////////////////////////////////////////////////////////////////
  739. class RtpChannel : public RtpTrackImp, public std::enable_shared_from_this<RtpChannel> {
  740. public:
  741. RtpChannel(EventPoller::Ptr poller, RtpTrackImp::OnSorted cb, function<void(const FCI_NACK &nack)> on_nack) {
  742. _poller = std::move(poller);
  743. _on_nack = std::move(on_nack);
  744. setOnSorted(std::move(cb));
  745. // 设置jitter buffer参数 [AUTO-TRANSLATED:eede98b6]
  746. // Set jitter buffer parameters
  747. GET_CONFIG(uint32_t, nack_maxms, Rtc::kNackMaxMS);
  748. GET_CONFIG(uint32_t, nack_max_rtp, Rtc::kNackMaxSize);
  749. RtpTrackImp::setParams(nack_max_rtp, nack_maxms, nack_max_rtp / 2);
  750. _nack_ctx.setOnNack([this](const FCI_NACK &nack) { onNack(nack); });
  751. }
  752. RtpPacket::Ptr inputRtp(TrackType type, int sample_rate, uint8_t *ptr, size_t len, bool is_rtx) {
  753. auto rtp = RtpTrack::inputRtp(type, sample_rate, ptr, len);
  754. if (!rtp) {
  755. return rtp;
  756. }
  757. auto seq = rtp->getSeq();
  758. _nack_ctx.received(seq, is_rtx);
  759. if (!is_rtx) {
  760. // 统计rtp接受情况,便于生成nack rtcp包 [AUTO-TRANSLATED:57e0f80d]
  761. // Statistics of rtp reception, which is convenient for generating nack rtcp packets
  762. _rtcp_context.onRtp(seq, rtp->getStamp(), rtp->ntp_stamp, sample_rate, len);
  763. }
  764. return rtp;
  765. }
  766. Buffer::Ptr createRtcpRR(RtcpHeader *sr, uint32_t ssrc) {
  767. _rtcp_context.onRtcp(sr);
  768. return _rtcp_context.createRtcpRR(ssrc, getSSRC());
  769. }
  770. float getLossRate() {
  771. auto expected = _rtcp_context.getExpectedPacketsInterval();
  772. if (!expected) {
  773. return -1;
  774. }
  775. return _rtcp_context.getLostInterval() * 100 / expected;
  776. }
  777. private:
  778. void starNackTimer() {
  779. if (_delay_task) {
  780. return;
  781. }
  782. weak_ptr<RtpChannel> weak_self = shared_from_this();
  783. _delay_task = _poller->doDelayTask(10, [weak_self]() -> uint64_t {
  784. auto strong_self = weak_self.lock();
  785. if (!strong_self) {
  786. return 0;
  787. }
  788. auto ret = strong_self->_nack_ctx.reSendNack();
  789. if (!ret) {
  790. strong_self->_delay_task = nullptr;
  791. }
  792. return ret;
  793. });
  794. }
  795. void onNack(const FCI_NACK &nack) {
  796. _on_nack(nack);
  797. starNackTimer();
  798. }
  799. private:
  800. NackContext _nack_ctx;
  801. RtcpContextForRecv _rtcp_context;
  802. EventPoller::Ptr _poller;
  803. EventPoller::DelayTask::Ptr _delay_task;
  804. function<void(const FCI_NACK &nack)> _on_nack;
  805. };
  806. std::shared_ptr<RtpChannel> MediaTrack::getRtpChannel(uint32_t ssrc) const {
  807. auto it_chn = rtp_channel.find(rtp_ext_ctx->getRid(ssrc));
  808. if (it_chn == rtp_channel.end()) {
  809. return nullptr;
  810. }
  811. return it_chn->second;
  812. }
  813. float WebRtcTransportImp::getLossRate(TrackType type) {
  814. for (auto &pr : _ssrc_to_track) {
  815. auto ssrc = pr.first;
  816. auto &track = pr.second;
  817. auto rtp_chn = track->getRtpChannel(ssrc);
  818. if (rtp_chn) {
  819. if (track->media && type == track->media->type) {
  820. return rtp_chn->getLossRate();
  821. }
  822. }
  823. }
  824. return -1;
  825. }
  826. void WebRtcTransportImp::onRtcp(const char *buf, size_t len) {
  827. _bytes_usage += len;
  828. auto rtcps = RtcpHeader::loadFromBytes((char *)buf, len);
  829. for (auto rtcp : rtcps) {
  830. switch ((RtcpType)rtcp->pt) {
  831. case RtcpType::RTCP_SR: {
  832. _alive_ticker.resetTime();
  833. // 对方汇报rtp发送情况 [AUTO-TRANSLATED:1389b0c8]
  834. // The other party reports the rtp sending situation
  835. RtcpSR *sr = (RtcpSR *)rtcp;
  836. auto it = _ssrc_to_track.find(sr->ssrc);
  837. if (it != _ssrc_to_track.end()) {
  838. auto &track = it->second;
  839. auto rtp_chn = track->getRtpChannel(sr->ssrc);
  840. if (!rtp_chn) {
  841. WarnL << "未识别的sr rtcp包:" << rtcp->dumpString();
  842. } else {
  843. // 设置rtp时间戳与ntp时间戳的对应关系 [AUTO-TRANSLATED:e92f4749]
  844. // Set the correspondence between rtp timestamp and ntp timestamp
  845. rtp_chn->setNtpStamp(sr->rtpts, sr->getNtpUnixStampMS());
  846. auto rr = rtp_chn->createRtcpRR(sr, track->answer_ssrc_rtp);
  847. sendRtcpPacket(rr->data(), rr->size(), true);
  848. }
  849. } else {
  850. WarnL << "未识别的sr rtcp包:" << rtcp->dumpString();
  851. }
  852. break;
  853. }
  854. case RtcpType::RTCP_RR: {
  855. _alive_ticker.resetTime();
  856. // 对方汇报rtp接收情况 [AUTO-TRANSLATED:77f50a28]
  857. // The other party reports the rtp receiving situation
  858. RtcpRR *rr = (RtcpRR *)rtcp;
  859. for (auto item : rr->getItemList()) {
  860. auto it = _ssrc_to_track.find(item->ssrc);
  861. if (it != _ssrc_to_track.end()) {
  862. auto &track = it->second;
  863. track->rtcp_context_send->onRtcp(rtcp);
  864. auto sr = track->rtcp_context_send->createRtcpSR(track->answer_ssrc_rtp);
  865. sendRtcpPacket(sr->data(), sr->size(), true);
  866. } else {
  867. WarnL << "未识别的rr rtcp包:" << rtcp->dumpString();
  868. }
  869. }
  870. break;
  871. }
  872. case RtcpType::RTCP_BYE: {
  873. // 对方汇报停止发送rtp [AUTO-TRANSLATED:96ad0cf3]
  874. // The other party reports the stop sending rtp
  875. RtcpBye *bye = (RtcpBye *)rtcp;
  876. for (auto ssrc : bye->getSSRC()) {
  877. auto it = _ssrc_to_track.find(*ssrc);
  878. if (it == _ssrc_to_track.end()) {
  879. WarnL << "未识别的bye rtcp包:" << rtcp->dumpString();
  880. continue;
  881. }
  882. _ssrc_to_track.erase(it);
  883. }
  884. onRtcpBye();
  885. // bye 会在 sender audio track mute 时出现, 因此不能作为 shutdown 的依据 [AUTO-TRANSLATED:d9fdfaac]
  886. // Bye will appear when the sender audio track is muted, so it cannot be used as the basis for shutdown
  887. break;
  888. }
  889. case RtcpType::RTCP_PSFB:
  890. case RtcpType::RTCP_RTPFB: {
  891. if ((RtcpType)rtcp->pt == RtcpType::RTCP_PSFB) {
  892. break;
  893. }
  894. // RTPFB
  895. switch ((RTPFBType)rtcp->report_count) {
  896. case RTPFBType::RTCP_RTPFB_NACK: {
  897. RtcpFB *fb = (RtcpFB *)rtcp;
  898. auto it = _ssrc_to_track.find(fb->ssrc_media);
  899. if (it == _ssrc_to_track.end()) {
  900. WarnL << "未识别的 rtcp包:" << rtcp->dumpString();
  901. return;
  902. }
  903. auto &track = it->second;
  904. auto &fci = fb->getFci<FCI_NACK>();
  905. track->nack_list.forEach(fci, [&](const RtpPacket::Ptr &rtp) {
  906. // rtp重传 [AUTO-TRANSLATED:62a37e46]
  907. // rtp retransmission
  908. onSendRtp(rtp, true, true);
  909. });
  910. break;
  911. }
  912. default:
  913. break;
  914. }
  915. break;
  916. }
  917. case RtcpType::RTCP_XR: {
  918. RtcpXRRRTR *xr = (RtcpXRRRTR *)rtcp;
  919. if (xr->bt != 4) {
  920. break;
  921. }
  922. auto it = _ssrc_to_track.find(xr->ssrc);
  923. if (it == _ssrc_to_track.end()) {
  924. WarnL << "未识别的 rtcp包:" << rtcp->dumpString();
  925. return;
  926. }
  927. auto &track = it->second;
  928. track->rtcp_context_send->onRtcp(rtcp);
  929. auto xrdlrr = track->rtcp_context_send->createRtcpXRDLRR(track->answer_ssrc_rtp, track->answer_ssrc_rtp);
  930. sendRtcpPacket(xrdlrr->data(), xrdlrr->size(), true);
  931. break;
  932. }
  933. default:
  934. break;
  935. }
  936. }
  937. }
  938. ///////////////////////////////////////////////////////////////////
  939. void WebRtcTransportImp::createRtpChannel(const string &rid, uint32_t ssrc, MediaTrack &track) {
  940. // rid --> RtpReceiverImp
  941. auto &ref = track.rtp_channel[rid];
  942. weak_ptr<WebRtcTransportImp> weak_self = static_pointer_cast<WebRtcTransportImp>(shared_from_this());
  943. ref = std::make_shared<RtpChannel>(
  944. getPoller(), [&track, this, rid](RtpPacket::Ptr rtp) mutable { onSortedRtp(track, rid, std::move(rtp)); },
  945. [&track, weak_self, ssrc](const FCI_NACK &nack) mutable {
  946. // nack发送可能由定时器异步触发 [AUTO-TRANSLATED:186d6723]
  947. // Nack sending may be triggered asynchronously by a timer
  948. auto strong_self = weak_self.lock();
  949. if (strong_self) {
  950. strong_self->onSendNack(track, nack, ssrc);
  951. }
  952. });
  953. InfoL << "create rtp receiver of ssrc:" << ssrc << ", rid:" << rid << ", codec:" << track.plan_rtp->codec;
  954. }
  955. void WebRtcTransportImp::updateTicker() {
  956. _alive_ticker.resetTime();
  957. }
  958. void WebRtcTransportImp::onRtp(const char *buf, size_t len, uint64_t stamp_ms) {
  959. _bytes_usage += len;
  960. _alive_ticker.resetTime();
  961. RtpHeader *rtp = (RtpHeader *)buf;
  962. // 根据接收到的rtp的pt信息,找到该流的信息 [AUTO-TRANSLATED:9a97682c]
  963. // Find the information of the stream according to the pt information of the received rtp
  964. auto it = _pt_to_track.find(rtp->pt);
  965. if (it == _pt_to_track.end()) {
  966. WarnL << "unknown rtp pt:" << (int)rtp->pt;
  967. return;
  968. }
  969. it->second->inputRtp(buf, len, stamp_ms, rtp);
  970. }
  971. void WrappedRtpTrack::inputRtp(const char *buf, size_t len, uint64_t stamp_ms, RtpHeader *rtp) {
  972. #if 0
  973. auto seq = ntohs(rtp->seq);
  974. if (track->media->type == TrackVideo && seq % 100 == 0) {
  975. // 此处模拟接受丢包 [AUTO-TRANSLATED:7e8c2e5c]
  976. // Simulate packet loss here
  977. return;
  978. }
  979. #endif
  980. auto ssrc = ntohl(rtp->ssrc);
  981. // 修改ext id至统一 [AUTO-TRANSLATED:0769b0ec]
  982. // Modify the ext id to be unified
  983. string rid;
  984. auto twcc_ext = track->rtp_ext_ctx->changeRtpExtId(rtp, true, &rid, RtpExtType::transport_cc);
  985. if (twcc_ext) {
  986. _twcc_ctx.onRtp(ssrc, twcc_ext.getTransportCCSeq(), stamp_ms);
  987. }
  988. auto &ref = track->rtp_channel[rid];
  989. if (!ref) {
  990. _transport.createRtpChannel(rid, ssrc, *track);
  991. }
  992. // 解析并排序rtp [AUTO-TRANSLATED:d382b65d]
  993. // Parse and sort rtp
  994. ref->inputRtp(track->media->type, track->plan_rtp->sample_rate, (uint8_t *)buf, len, false);
  995. }
  996. void WrappedRtxTrack::inputRtp(const char *buf, size_t len, uint64_t stamp_ms, RtpHeader *rtp) {
  997. // 修改ext id至统一 [AUTO-TRANSLATED:0769b0ec]
  998. // Modify the ext id to be unified
  999. string rid;
  1000. track->rtp_ext_ctx->changeRtpExtId(rtp, true, &rid, RtpExtType::transport_cc);
  1001. auto &ref = track->rtp_channel[rid];
  1002. if (!ref) {
  1003. // 再接收到对应的rtp前,丢弃rtx包 [AUTO-TRANSLATED:d4ca6d69]
  1004. // Discard rtx packets before receiving the corresponding rtp
  1005. WarnL << "unknown rtx rtp, rid:" << rid << ", ssrc:" << ntohl(rtp->ssrc) << ", codec:" << track->plan_rtp->codec
  1006. << ", seq:" << ntohs(rtp->seq);
  1007. return;
  1008. }
  1009. // 这里是rtx重传包 [AUTO-TRANSLATED:6efd3766]
  1010. // This is the rtx retransmission packet
  1011. // https://datatracker.ietf.org/doc/html/rfc4588#section-4
  1012. auto payload = rtp->getPayloadData();
  1013. auto size = rtp->getPayloadSize(len);
  1014. if (size < 2) {
  1015. return;
  1016. }
  1017. // 前两个字节是原始的rtp的seq [AUTO-TRANSLATED:c57ff92d]
  1018. // The first two bytes are the original rtp seq
  1019. auto origin_seq = payload[0] << 8 | payload[1];
  1020. // rtx 转换为 rtp [AUTO-TRANSLATED:be27f61b]
  1021. // rtx converted to rtp
  1022. rtp->pt = track->plan_rtp->pt;
  1023. rtp->seq = htons(origin_seq);
  1024. rtp->ssrc = htonl(ref->getSSRC());
  1025. memmove((uint8_t *)buf + 2, buf, payload - (uint8_t *)buf);
  1026. buf += 2;
  1027. len -= 2;
  1028. ref->inputRtp(track->media->type, track->plan_rtp->sample_rate, (uint8_t *)buf, len, true);
  1029. }
  1030. void WebRtcTransportImp::onSendNack(MediaTrack &track, const FCI_NACK &nack, uint32_t ssrc) {
  1031. auto rtcp = RtcpFB::create(RTPFBType::RTCP_RTPFB_NACK, &nack, FCI_NACK::kSize);
  1032. rtcp->ssrc = htonl(track.answer_ssrc_rtp);
  1033. rtcp->ssrc_media = htonl(ssrc);
  1034. sendRtcpPacket((char *)rtcp.get(), rtcp->getSize(), true);
  1035. }
  1036. void WebRtcTransportImp::onSendTwcc(uint32_t ssrc, const string &twcc_fci) {
  1037. auto rtcp = RtcpFB::create(RTPFBType::RTCP_RTPFB_TWCC, twcc_fci.data(), twcc_fci.size());
  1038. rtcp->ssrc = htonl(0);
  1039. rtcp->ssrc_media = htonl(ssrc);
  1040. sendRtcpPacket((char *)rtcp.get(), rtcp->getSize(), true);
  1041. }
  1042. ///////////////////////////////////////////////////////////////////
  1043. void WebRtcTransportImp::onSortedRtp(MediaTrack &track, const string &rid, RtpPacket::Ptr rtp) {
  1044. if (track.media->type == TrackVideo && _pli_ticker.elapsedTime() > 2000) {
  1045. // 定期发送pli请求关键帧,方便非rtc等协议 [AUTO-TRANSLATED:b992f020]
  1046. // Regularly send pli requests for key frames, which is convenient for non-rtc protocols
  1047. _pli_ticker.resetTime();
  1048. sendRtcpPli(rtp->getSSRC());
  1049. // 开启remb,则发送remb包调节比特率 [AUTO-TRANSLATED:20e98cea]
  1050. // If remb is enabled, send remb packets to adjust the bitrate
  1051. GET_CONFIG(size_t, remb_bit_rate, Rtc::kRembBitRate);
  1052. if (remb_bit_rate && _answer_sdp->supportRtcpFb(SdpConst::kRembRtcpFb)) {
  1053. sendRtcpRemb(rtp->getSSRC(), remb_bit_rate);
  1054. }
  1055. }
  1056. onRecvRtp(track, rid, std::move(rtp));
  1057. }
  1058. ///////////////////////////////////////////////////////////////////
  1059. void WebRtcTransportImp::onSendRtp(const RtpPacket::Ptr &rtp, bool flush, bool rtx) {
  1060. auto &track = _type_to_track[rtp->type];
  1061. if (!track) {
  1062. // 忽略,对方不支持该编码类型 [AUTO-TRANSLATED:498ee936]
  1063. // Ignore, the other party does not support this encoding type
  1064. return;
  1065. }
  1066. if (!rtx) {
  1067. // 统计rtp发送情况,好做sr汇报 [AUTO-TRANSLATED:142028b2]
  1068. // Statistics of RTP sending, for SR reporting
  1069. track->rtcp_context_send->onRtp(
  1070. rtp->getSeq(), rtp->getStamp(), rtp->ntp_stamp, rtp->sample_rate,
  1071. rtp->size() - RtpPacket::kRtpTcpHeaderSize);
  1072. track->nack_list.pushBack(rtp);
  1073. #if 0
  1074. // 此处模拟发送丢包 [AUTO-TRANSLATED:9612f08e]
  1075. // Simulate packet loss here
  1076. if (rtp->type == TrackVideo && rtp->getSeq() % 100 == 0) {
  1077. return;
  1078. }
  1079. #endif
  1080. } else {
  1081. // 发送rtx重传包 [AUTO-TRANSLATED:ae60e1fd]
  1082. // Send RTX retransmission packets
  1083. // TraceL << "send rtx rtp:" << rtp->getSeq();
  1084. }
  1085. pair<bool /*rtx*/, MediaTrack *> ctx { rtx, track.get() };
  1086. sendRtpPacket(rtp->data() + RtpPacket::kRtpTcpHeaderSize, rtp->size() - RtpPacket::kRtpTcpHeaderSize, flush, &ctx);
  1087. _bytes_usage += rtp->size() - RtpPacket::kRtpTcpHeaderSize;
  1088. }
  1089. void WebRtcTransportImp::onBeforeEncryptRtp(const char *buf, int &len, void *ctx) {
  1090. auto pr = (pair<bool /*rtx*/, MediaTrack *> *)ctx;
  1091. auto header = (RtpHeader *)buf;
  1092. if (!pr->first || !pr->second->plan_rtx) {
  1093. // 普通的rtp,或者不支持rtx, 修改目标pt和ssrc [AUTO-TRANSLATED:e1264971]
  1094. // Ordinary RTP, or does not support RTX, modify the target PT and SSRC
  1095. pr->second->rtp_ext_ctx->changeRtpExtId(header, false);
  1096. header->pt = pr->second->plan_rtp->pt;
  1097. header->ssrc = htonl(pr->second->answer_ssrc_rtp);
  1098. } else {
  1099. // 重传的rtp, rtx [AUTO-TRANSLATED:e863a518]
  1100. // Retransmitted RTP, RTX
  1101. pr->second->rtp_ext_ctx->changeRtpExtId(header, false);
  1102. header->pt = pr->second->plan_rtx->pt;
  1103. if (pr->second->answer_ssrc_rtx) {
  1104. // 有rtx单独的ssrc,有些情况下,浏览器支持rtx,但是未指定rtx单独的ssrc [AUTO-TRANSLATED:181cee9a]
  1105. // RTX has a separate SSRC, in some cases, the browser supports RTX, but does not specify a separate SSRC for RTX
  1106. header->ssrc = htonl(pr->second->answer_ssrc_rtx);
  1107. } else {
  1108. // 未单独指定rtx的ssrc,那么使用rtp的ssrc [AUTO-TRANSLATED:dcafdd75]
  1109. // If RTX SSRC is not specified separately, use the RTP SSRC
  1110. header->ssrc = htonl(pr->second->answer_ssrc_rtp);
  1111. }
  1112. auto origin_seq = ntohs(header->seq);
  1113. // seq跟原来的不一样 [AUTO-TRANSLATED:803f9a5e]
  1114. // The sequence is different from the original
  1115. header->seq = htons(_rtx_seq[pr->second->media->type]);
  1116. ++_rtx_seq[pr->second->media->type];
  1117. auto payload = header->getPayloadData();
  1118. auto payload_size = header->getPayloadSize(len);
  1119. if (payload_size) {
  1120. // rtp负载后移两个字节,这两个字节用于存放osn [AUTO-TRANSLATED:b7eed70e]
  1121. // The RTP payload is shifted two bytes, these two bytes are used to store OSN
  1122. // https://datatracker.ietf.org/doc/html/rfc4588#section-4
  1123. memmove(payload + 2, payload, payload_size);
  1124. }
  1125. payload[0] = origin_seq >> 8;
  1126. payload[1] = origin_seq & 0xFF;
  1127. len += 2;
  1128. }
  1129. }
  1130. void WebRtcTransportImp::safeShutdown(const SockException &ex) {
  1131. std::weak_ptr<WebRtcTransportImp> weak_self = static_pointer_cast<WebRtcTransportImp>(shared_from_this());
  1132. getPoller()->async([ex, weak_self]() {
  1133. if (auto strong_self = weak_self.lock()) {
  1134. strong_self->onShutdown(ex);
  1135. }
  1136. });
  1137. }
  1138. void WebRtcTransportImp::onShutdown(const SockException &ex) {
  1139. WarnL << ex;
  1140. unrefSelf();
  1141. for (auto &tuple : _ice_server->GetTuples()) {
  1142. tuple->shutdown(ex);
  1143. }
  1144. }
  1145. void WebRtcTransportImp::removeTuple(RTC::TransportTuple *tuple) {
  1146. InfoL << getIdentifier() << " remove tuple " << tuple->get_peer_ip() << ":" << tuple->get_peer_port();
  1147. this->_ice_server->RemoveTuple(tuple);
  1148. }
  1149. uint64_t WebRtcTransportImp::getBytesUsage() const {
  1150. return _bytes_usage;
  1151. }
  1152. uint64_t WebRtcTransportImp::getDuration() const {
  1153. return _alive_ticker.createdTime() / 1000;
  1154. }
  1155. void WebRtcTransportImp::onRtcpBye(){}
  1156. /////////////////////////////////////////////////////////////////////////////////////////////
  1157. void WebRtcTransportImp::registerSelf() {
  1158. _self = static_pointer_cast<WebRtcTransportImp>(shared_from_this());
  1159. WebRtcTransportManager::Instance().addItem(getIdentifier(), _self);
  1160. }
  1161. void WebRtcTransportImp::unrefSelf() {
  1162. _self = nullptr;
  1163. }
  1164. void WebRtcTransportImp::unregisterSelf() {
  1165. unrefSelf();
  1166. WebRtcTransportManager::Instance().removeItem(getIdentifier());
  1167. }
  1168. WebRtcTransportManager &WebRtcTransportManager::Instance() {
  1169. static WebRtcTransportManager s_instance;
  1170. return s_instance;
  1171. }
  1172. void WebRtcTransportManager::addItem(const string &key, const WebRtcTransportImp::Ptr &ptr) {
  1173. lock_guard<mutex> lck(_mtx);
  1174. _map[key] = ptr;
  1175. }
  1176. WebRtcTransportImp::Ptr WebRtcTransportManager::getItem(const string &key) {
  1177. if (key.empty()) {
  1178. return nullptr;
  1179. }
  1180. lock_guard<mutex> lck(_mtx);
  1181. auto it = _map.find(key);
  1182. if (it == _map.end()) {
  1183. return nullptr;
  1184. }
  1185. return it->second.lock();
  1186. }
  1187. void WebRtcTransportManager::removeItem(const string &key) {
  1188. lock_guard<mutex> lck(_mtx);
  1189. _map.erase(key);
  1190. }
  1191. //////////////////////////////////////////////////////////////////////////////////////////////
  1192. WebRtcPluginManager &WebRtcPluginManager::Instance() {
  1193. static WebRtcPluginManager s_instance;
  1194. return s_instance;
  1195. }
  1196. void WebRtcPluginManager::registerPlugin(const string &type, Plugin cb) {
  1197. lock_guard<mutex> lck(_mtx_creator);
  1198. _map_creator[type] = std::move(cb);
  1199. }
  1200. void WebRtcPluginManager::setListener(Listener cb) {
  1201. lock_guard<mutex> lck(_mtx_creator);
  1202. _listener = std::move(cb);
  1203. }
  1204. void WebRtcPluginManager::negotiateSdp(Session &sender, const string &type, const WebRtcArgs &args, const onCreateWebRtc &cb_in) {
  1205. onCreateWebRtc cb;
  1206. lock_guard<mutex> lck(_mtx_creator);
  1207. if (_listener) {
  1208. auto listener = _listener;
  1209. auto args_ptr = args.shared_from_this();
  1210. auto sender_ptr = static_pointer_cast<Session>(sender.shared_from_this());
  1211. cb = [listener, sender_ptr, type, args_ptr, cb_in](const WebRtcInterface &rtc) {
  1212. listener(*sender_ptr, type, *args_ptr, rtc);
  1213. cb_in(rtc);
  1214. };
  1215. } else {
  1216. cb = cb_in;
  1217. }
  1218. auto it = _map_creator.find(type);
  1219. if (it == _map_creator.end()) {
  1220. cb_in(WebRtcException(SockException(Err_other, "the type can not supported")));
  1221. return;
  1222. }
  1223. it->second(sender, args, cb);
  1224. }
  1225. void echo_plugin(Session &sender, const WebRtcArgs &args, const onCreateWebRtc &cb) {
  1226. cb(*WebRtcEchoTest::create(EventPollerPool::Instance().getPoller()));
  1227. }
  1228. void push_plugin(Session &sender, const WebRtcArgs &args, const onCreateWebRtc &cb) {
  1229. MediaInfo info(args["url"]);
  1230. Broadcast::PublishAuthInvoker invoker = [cb, info](const string &err, const ProtocolOption &option) mutable {
  1231. if (!err.empty()) {
  1232. cb(WebRtcException(SockException(Err_other, err)));
  1233. return;
  1234. }
  1235. RtspMediaSourceImp::Ptr push_src;
  1236. std::shared_ptr<void> push_src_ownership;
  1237. auto src = MediaSource::find(RTSP_SCHEMA, info.vhost, info.app, info.stream);
  1238. auto push_failed = (bool)src;
  1239. while (src) {
  1240. // 尝试断连后继续推流 [AUTO-TRANSLATED:9eaaa6dd]
  1241. // Try to continue streaming after disconnecting
  1242. auto rtsp_src = dynamic_pointer_cast<RtspMediaSourceImp>(src);
  1243. if (!rtsp_src) {
  1244. // 源不是rtsp推流产生的 [AUTO-TRANSLATED:47b87993]
  1245. // The source is not generated by RTSP streaming
  1246. break;
  1247. }
  1248. auto ownership = rtsp_src->getOwnership();
  1249. if (!ownership) {
  1250. // 获取推流源所有权失败 [AUTO-TRANSLATED:256190b2]
  1251. // Failed to get the ownership of the streaming source
  1252. break;
  1253. }
  1254. push_src = std::move(rtsp_src);
  1255. push_src_ownership = std::move(ownership);
  1256. push_failed = false;
  1257. break;
  1258. }
  1259. if (push_failed) {
  1260. cb(WebRtcException(SockException(Err_other, "already publishing")));
  1261. return;
  1262. }
  1263. if (!push_src) {
  1264. push_src = std::make_shared<RtspMediaSourceImp>(info);
  1265. push_src_ownership = push_src->getOwnership();
  1266. push_src->setProtocolOption(option);
  1267. }
  1268. auto rtc = WebRtcPusher::create(EventPollerPool::Instance().getPoller(), push_src, push_src_ownership, info, option);
  1269. push_src->setListener(rtc);
  1270. cb(*rtc);
  1271. };
  1272. // rtsp推流需要鉴权 [AUTO-TRANSLATED:c2cbb7ed]
  1273. // RTSP streaming requires authentication
  1274. auto flag = NOTICE_EMIT(BroadcastMediaPublishArgs, Broadcast::kBroadcastMediaPublish, MediaOriginType::rtc_push, info, invoker, sender);
  1275. if (!flag) {
  1276. // 该事件无人监听,默认不鉴权 [AUTO-TRANSLATED:e1fbc6ae]
  1277. // No one is listening to this event, authentication is not enabled by default
  1278. invoker("", ProtocolOption());
  1279. }
  1280. }
  1281. void play_plugin(Session &sender, const WebRtcArgs &args, const onCreateWebRtc &cb) {
  1282. MediaInfo info(args["url"]);
  1283. auto session_ptr = static_pointer_cast<Session>(sender.shared_from_this());
  1284. Broadcast::AuthInvoker invoker = [cb, info, session_ptr](const string &err) mutable {
  1285. if (!err.empty()) {
  1286. cb(WebRtcException(SockException(Err_other, err)));
  1287. return;
  1288. }
  1289. // webrtc播放的是rtsp的源 [AUTO-TRANSLATED:649ae489]
  1290. // WebRTC plays the RTSP source
  1291. info.schema = RTSP_SCHEMA;
  1292. MediaSource::findAsync(info, session_ptr, [=](const MediaSource::Ptr &src_in) mutable {
  1293. auto src = dynamic_pointer_cast<RtspMediaSource>(src_in);
  1294. if (!src) {
  1295. cb(WebRtcException(SockException(Err_other, "stream not found")));
  1296. return;
  1297. }
  1298. // 还原成rtc,目的是为了hook时识别哪种播放协议 [AUTO-TRANSLATED:fe8dd2dc]
  1299. // Restore to RTC, the purpose is to identify which playback protocol during hooking
  1300. info.schema = "rtc";
  1301. auto rtc = WebRtcPlayer::create(EventPollerPool::Instance().getPoller(), src, info);
  1302. cb(*rtc);
  1303. });
  1304. };
  1305. // 广播通用播放url鉴权事件 [AUTO-TRANSLATED:81e24be4]
  1306. // Broadcast a universal playback URL authentication event
  1307. auto flag = NOTICE_EMIT(BroadcastMediaPlayedArgs, Broadcast::kBroadcastMediaPlayed, info, invoker, sender);
  1308. if (!flag) {
  1309. // 该事件无人监听,默认不鉴权 [AUTO-TRANSLATED:e1fbc6ae]
  1310. // No one is listening to this event, authentication is not enabled by default
  1311. invoker("");
  1312. }
  1313. }
  1314. static void setWebRtcArgs(const WebRtcArgs &args, WebRtcInterface &rtc) {
  1315. {
  1316. static auto is_vaild_ip = [](const std::string &ip) -> bool {
  1317. int a, b, c, d;
  1318. return sscanf(ip.c_str(), "%d.%d.%d.%d", &a, &b, &c, &d) == 4;
  1319. };
  1320. std::string host = args["Host"];
  1321. if (!host.empty()) {
  1322. auto local_ip = host.substr(0, host.find(':'));
  1323. if (!is_vaild_ip(local_ip) || local_ip == "127.0.0.1") {
  1324. local_ip = "";
  1325. }
  1326. rtc.setLocalIp(std::move(local_ip));
  1327. }
  1328. }
  1329. bool preferred_tcp = args["preferred_tcp"];
  1330. {
  1331. rtc.setPreferredTcp(preferred_tcp);
  1332. }
  1333. {
  1334. vector<SdpAttrCandidate> cands;
  1335. {
  1336. auto cand_str = trim(args["cand_udp"]);
  1337. auto ip_port = toolkit::split(cand_str, ":");
  1338. if (ip_port.size() == 2) {
  1339. // udp优先 [AUTO-TRANSLATED:b428f63d]
  1340. // UDP priority
  1341. auto ice_cand = makeIceCandidate(ip_port[0], atoi(ip_port[1].data()), preferred_tcp ? 100 : 120, "udp");
  1342. cands.emplace_back(std::move(*ice_cand));
  1343. }
  1344. }
  1345. {
  1346. auto cand_str = trim(args["cand_tcp"]);
  1347. auto ip_port = toolkit::split(cand_str, ":");
  1348. if (ip_port.size() == 2) {
  1349. // tcp模式 [AUTO-TRANSLATED:62fddf91]
  1350. // TCP mode
  1351. auto ice_cand = makeIceCandidate(ip_port[0], atoi(ip_port[1].data()), preferred_tcp ? 120 : 100, "tcp");
  1352. cands.emplace_back(std::move(*ice_cand));
  1353. }
  1354. }
  1355. if (!cands.empty()) {
  1356. // udp优先 [AUTO-TRANSLATED:b428f63d]
  1357. // UDP priority
  1358. rtc.setIceCandidate(std::move(cands));
  1359. }
  1360. }
  1361. }
  1362. static onceToken s_rtc_auto_register([]() {
  1363. #if !defined (NDEBUG)
  1364. // debug模式才开启echo插件 [AUTO-TRANSLATED:48fcb116]
  1365. // Enable echo plugin only in debug mode
  1366. WebRtcPluginManager::Instance().registerPlugin("echo", echo_plugin);
  1367. #endif
  1368. WebRtcPluginManager::Instance().registerPlugin("push", push_plugin);
  1369. WebRtcPluginManager::Instance().registerPlugin("play", play_plugin);
  1370. WebRtcPluginManager::Instance().setListener([](Session &sender, const std::string &type, const WebRtcArgs &args, const WebRtcInterface &rtc) {
  1371. setWebRtcArgs(args, const_cast<WebRtcInterface&>(rtc));
  1372. });
  1373. });
  1374. }// namespace mediakit