RtspPlayer.cpp 28.2 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
xiongziliang committed
3 4 5
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
xiongziliang committed
6 7 8
 * Use of this source code is governed by MIT license that can be found in the
 * LICENSE file in the root of the source tree. All contributing project authors
 * may be found in the AUTHORS file in the root of the source tree.
xiongziliang committed
9
 */
xiongziliang committed
10

xzl committed
11 12 13
#include <set>
#include <cmath>
#include <algorithm>
xiongziliang committed
14
#include <iomanip>
xiongziliang committed
15
#include "Common/config.h"
xiongzilaing committed
16
#include "RtspPlayer.h"
17
#include "Util/MD5.h"
18
#include "Util/util.h"
xiongziliang committed
19
#include "Util/base64.h"
xzl committed
20
#include "Network/sockutil.h"
xiongziliang committed
21
using namespace toolkit;
xiongziliang committed
22
using namespace mediakit::Client;
xzl committed
23

xiongziliang committed
24
namespace mediakit {
xzl committed
25

26
enum PlayType {
27 28 29
    type_play = 0,
    type_pause,
    type_seek
30 31
};

32
RtspPlayer::RtspPlayer(const EventPoller::Ptr &poller) : TcpClient(poller){
33
    RtpReceiver::setPoolSize(64);
xzl committed
34 35
}
RtspPlayer::~RtspPlayer(void) {
36
    DebugL << endl;
xzl committed
37 38
}
void RtspPlayer::teardown(){
39
    if (alive()) {
40 41 42 43
        if (!_content_base.empty()) {
            sendRtspRequest("TEARDOWN", _content_base);
        }
        shutdown(SockException(Err_shutdown, "teardown"));
44 45
    }

46 47 48 49 50
    _md5_nonce.clear();
    _realm.clear();
    _sdp_track.clear();
    _session_id.clear();
    _content_base.clear();
51
    RtpReceiver::clear();
xiongziliang committed
52

53 54 55 56 57 58 59 60 61 62 63
    CLEAR_ARR(_rtp_sock);
    CLEAR_ARR(_rtcp_sock);
    CLEAR_ARR(_rtp_seq_start)
    CLEAR_ARR(_rtp_recv_count)
    CLEAR_ARR(_rtp_recv_count)
    CLEAR_ARR(_rtp_seq_now)

    _play_check_timer.reset();
    _rtp_check_timer.reset();
    _cseq_send = 1;
    _on_response = nullptr;
xzl committed
64
}
xiongziliang committed
65

xiongziliang committed
66
void RtspPlayer::play(const string &strUrl){
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
    RtspUrl url;
    if(!url.parse(strUrl)){
        onPlayResult_l(SockException(Err_other,StrPrinter << "illegal rtsp url:" << strUrl),false);
        return;
    }

    teardown();

    if (url._user.size()) {
        (*this)[kRtspUser] = url._user;
    }
    if (url._passwd.size()) {
        (*this)[kRtspPwd] = url._passwd;
        (*this)[kRtspPwdIsMD5] = false;
    }

83 84 85
    _play_url = url._url;
    _rtp_type = (Rtsp::eRtpType)(int)(*this)[kRtpType];
    DebugL << url._url << " " << (url._user.size() ? url._user : "null") << " " << (url._passwd.size() ? url._passwd : "null") << " " << _rtp_type;
86 87 88

    weak_ptr<RtspPlayer> weakSelf = dynamic_pointer_cast<RtspPlayer>(shared_from_this());
    float playTimeOutSec = (*this)[kTimeoutMS].as<int>() / 1000.0;
89
    _play_check_timer.reset(new Timer(playTimeOutSec, [weakSelf]() {
90 91 92 93 94 95
        auto strongSelf=weakSelf.lock();
        if(!strongSelf) {
            return false;
        }
        strongSelf->onPlayResult_l(SockException(Err_timeout,"play rtsp timeout"),false);
        return false;
96
    }, getPoller()));
97 98 99 100 101

    if(!(*this)[kNetAdapter].empty()){
        setNetAdapter((*this)[kNetAdapter]);
    }
    startConnect(url._host, url._port, playTimeOutSec);
102
}
103

104
void RtspPlayer::onConnect(const SockException &err){
105 106 107 108
    if(err.getErrCode() != Err_success) {
        onPlayResult_l(err,false);
        return;
    }
109
    sendOptions();
xzl committed
110 111
}

112
void RtspPlayer::onRecv(const Buffer::Ptr& buf) {
113
    if(_benchmark_mode && !_play_check_timer){
114
        //在性能测试模式下,如果rtsp握手完毕后,不再解析rtp包
115
        _rtp_recv_ticker.resetTime();
116 117
        return;
    }
118
    input(buf->data(), buf->size());
xzl committed
119
}
xiongziliang committed
120

xzl committed
121
void RtspPlayer::onErr(const SockException &ex) {
122
    //定时器_pPlayTimer为空后表明握手结束了
123
    onPlayResult_l(ex,!_play_check_timer);
xzl committed
124
}
xiongziliang committed
125

126 127
// from live555
bool RtspPlayer::handleAuthenticationFailure(const string &paramsStr) {
128
    if(!_realm.empty()){
129 130 131 132
        //已经认证过了
        return false;
    }

xiongziliang committed
133 134 135 136 137 138 139 140
    char *realm = new char[paramsStr.size()];
    char *nonce = new char[paramsStr.size()];
    char *stale = new char[paramsStr.size()];
    onceToken token(nullptr,[&](){
        delete[] realm;
        delete[] nonce;
        delete[] stale;
    });
141 142

    if (sscanf(paramsStr.data(), "Digest realm=\"%[^\"]\", nonce=\"%[^\"]\", stale=%[a-zA-Z]", realm, nonce, stale) == 3) {
143 144
        _realm = (const char *)realm;
        _md5_nonce = (const char *)nonce;
145 146 147
        return true;
    }
    if (sscanf(paramsStr.data(), "Digest realm=\"%[^\"]\", nonce=\"%[^\"]\"", realm, nonce) == 2) {
148 149
        _realm = (const char *)realm;
        _md5_nonce = (const char *)nonce;
150 151 152
        return true;
    }
    if (sscanf(paramsStr.data(), "Basic realm=\"%[^\"]\"", realm) == 1) {
153
        _realm = (const char *)realm;
154 155 156 157
        return true;
    }
    return false;
}
xiongziliang committed
158

159
bool RtspPlayer::handleResponse(const string &cmd, const Parser &parser){
160 161 162
    string authInfo = parser["WWW-Authenticate"];
    //发送DESCRIBE命令后的回复
    if ((parser.Url() == "401") && handleAuthenticationFailure(authInfo)) {
163
        sendOptions();
164
        return false;
165 166 167 168 169 170 171
    }
    if(parser.Url() == "302" || parser.Url() == "301"){
        auto newUrl = parser["Location"];
        if(newUrl.empty()){
            throw std::runtime_error("未找到Location字段(跳转url)");
        }
        play(newUrl);
172
        return false;
173 174
    }
    if (parser.Url() != "200") {
175
        throw std::runtime_error(StrPrinter << cmd << ":" << parser.Url() << " " << parser.Tail() << endl);
176
    }
177 178
    return true;
}
179

180 181 182 183 184
void RtspPlayer::handleResDESCRIBE(const Parser& parser) {
    if (!handleResponse("DESCRIBE", parser)) {
        return;
    }
    _content_base = parser["Content-Base"];
185 186
    if(_content_base.empty()){
        _content_base = _play_url;
187
    }
188 189
    if (_content_base.back() == '/') {
        _content_base.pop_back();
190 191
    }

192 193
    SdpParser sdpParser(parser.Content());
    //解析sdp
194
    _sdp_track = sdpParser.getAvailableTrack();
195
    auto title = sdpParser.getTrack(TrackTitle);
196
    bool is_play_back = false;
197
    if(title && title->_duration ){
198
        is_play_back = true;
xiongziliang committed
199
    }
xiongziliang committed
200

201
    if (_sdp_track.empty()) {
202 203 204 205 206
        throw std::runtime_error("无有效的Sdp Track");
    }
    if (!onCheckSDP(sdpParser.toString())) {
        throw std::runtime_error("onCheckSDP faied");
    }
xiongziliang committed
207

208
    sendSetup(0);
xzl committed
209
}
210

xiongziliang committed
211
//有必要的情况下创建udp端口
212
void RtspPlayer::createUdpSockIfNecessary(int track_idx){
213 214
    auto &rtpSockRef = _rtp_sock[track_idx];
    auto &rtcpSockRef = _rtcp_sock[track_idx];
215
    if (!rtpSockRef || !rtcpSockRef) {
216 217
        std::pair<Socket::Ptr, Socket::Ptr> pr = std::make_pair(createSocket(), createSocket());
        makeSockPair(pr, get_local_ip());
218 219
        rtpSockRef = pr.first;
        rtcpSockRef = pr.second;
220
    }
221 222
}

xzl committed
223
//发送SETUP命令
224 225 226
void RtspPlayer::sendSetup(unsigned int track_idx) {
    _on_response = std::bind(&RtspPlayer::handleResSETUP, this, placeholders::_1, track_idx);
    auto &track = _sdp_track[track_idx];
227
    auto baseUrl = _content_base + "/" + track->_control_surffix;
228 229 230
    if (track->_control.find("://") != string::npos) {
        baseUrl = track->_control;
    }
231
    switch (_rtp_type) {
232 233 234 235 236
        case Rtsp::RTP_TCP: {
            sendRtspRequest("SETUP",baseUrl,{"Transport",StrPrinter << "RTP/AVP/TCP;unicast;interleaved=" << track->_type * 2 << "-" << track->_type * 2 + 1});
        }
            break;
        case Rtsp::RTP_MULTICAST: {
xiongziliang committed
237
            sendRtspRequest("SETUP",baseUrl,{"Transport","RTP/AVP;multicast"});
238 239 240
        }
            break;
        case Rtsp::RTP_UDP: {
241
            createUdpSockIfNecessary(track_idx);
242 243
            sendRtspRequest("SETUP", baseUrl, {"Transport",
                                               StrPrinter << "RTP/AVP;unicast;client_port="
244 245
                                                          << _rtp_sock[track_idx]->get_local_port() << "-"
                                                          << _rtcp_sock[track_idx]->get_local_port()});
246 247 248 249 250
        }
            break;
        default:
            break;
    }
xzl committed
251 252
}

253
void RtspPlayer::handleResSETUP(const Parser &parser, unsigned int track_idx) {
254 255 256 257
    if (parser.Url() != "200") {
        throw std::runtime_error(
        StrPrinter << "SETUP:" << parser.Url() << " " << parser.Tail() << endl);
    }
258
    if (track_idx == 0) {
259 260 261
        _session_id = parser["Session"];
        _session_id.append(";");
        _session_id = FindField(_session_id.data(), nullptr, ";");
262 263 264 265
    }

    auto strTransport = parser["Transport"];
    if(strTransport.find("TCP") != string::npos || strTransport.find("interleaved") != string::npos){
266
        _rtp_type = Rtsp::RTP_TCP;
267
    }else if(strTransport.find("multicast") != string::npos){
268
        _rtp_type = Rtsp::RTP_MULTICAST;
269
    }else{
270
        _rtp_type = Rtsp::RTP_UDP;
271 272
    }

273
    RtspSplitter::enableRecvRtp(_rtp_type == Rtsp::RTP_TCP);
274

275
    if(_rtp_type == Rtsp::RTP_TCP)  {
276
        string interleaved = FindField( FindField((strTransport + ";").data(), "interleaved=", ";").data(), NULL, "-");
277
        _sdp_track[track_idx]->_interleaved = atoi(interleaved.data());
278
    }else{
279
        const char *strPos = (_rtp_type == Rtsp::RTP_MULTICAST ? "port=" : "server_port=") ;
280 281
        auto port_str = FindField((strTransport + ";").data(), strPos, ";");
        uint16_t rtp_port = atoi(FindField(port_str.data(), NULL, "-").data());
282
        uint16_t rtcp_port = atoi(FindField(port_str.data(), "-",NULL).data());
283 284
        auto &pRtpSockRef = _rtp_sock[track_idx];
        auto &pRtcpSockRef = _rtcp_sock[track_idx];
xiongziliang committed
285

286
        if (_rtp_type == Rtsp::RTP_MULTICAST) {
287 288
            //udp组播
            auto multiAddr = FindField((strTransport + ";").data(), "destination=", ";");
289
            pRtpSockRef = createSocket();
290 291 292 293 294 295 296 297 298
            if (!pRtpSockRef->bindUdpSock(rtp_port, multiAddr.data())) {
                pRtpSockRef.reset();
                throw std::runtime_error("open udp sock err");
            }
            auto fd = pRtpSockRef->rawFD();
            if (-1 == SockUtil::joinMultiAddrFilter(fd, multiAddr.data(), get_peer_ip().data(),get_local_ip().data())) {
                SockUtil::joinMultiAddr(fd, multiAddr.data(),get_local_ip().data());
            }
        } else {
299
            createUdpSockIfNecessary(track_idx);
300 301 302 303 304 305 306 307 308 309
            //udp单播
            struct sockaddr_in rtpto;
            rtpto.sin_port = ntohs(rtp_port);
            rtpto.sin_family = AF_INET;
            rtpto.sin_addr.s_addr = inet_addr(get_peer_ip().data());
            pRtpSockRef->setSendPeerAddr((struct sockaddr *)&(rtpto));
            //发送rtp打洞包
            pRtpSockRef->send("\xce\xfa\xed\xfe", 4);

            //设置rtcp发送目标,为后续发送rtcp做准备
310 311 312 313
            rtpto.sin_port = ntohs(rtcp_port);
            rtpto.sin_family = AF_INET;
            rtpto.sin_addr.s_addr = inet_addr(get_peer_ip().data());
            pRtcpSockRef->setSendPeerAddr((struct sockaddr *)&(rtpto));
314
        }
315 316 317 318

        auto srcIP = inet_addr(get_peer_ip().data());
        weak_ptr<RtspPlayer> weakSelf = dynamic_pointer_cast<RtspPlayer>(shared_from_this());
        //设置rtp over udp接收回调处理函数
319
        pRtpSockRef->setOnRead([srcIP, track_idx, weakSelf](const Buffer::Ptr &buf, struct sockaddr *addr , int addr_len) {
320 321 322 323 324
            auto strongSelf = weakSelf.lock();
            if (!strongSelf) {
                return;
            }
            if (((struct sockaddr_in *) addr)->sin_addr.s_addr != srcIP) {
325
                WarnL << "收到其他地址的rtp数据:" << SockUtil::inet_ntoa(((struct sockaddr_in *) addr)->sin_addr);
326 327
                return;
            }
328 329
            strongSelf->handleOneRtp(track_idx, strongSelf->_sdp_track[track_idx]->_type,
                                     strongSelf->_sdp_track[track_idx]->_samplerate, (unsigned char *) buf->data(), buf->size());
330 331 332 333
        });

        if(pRtcpSockRef) {
            //设置rtcp over udp接收回调处理函数
334
            pRtcpSockRef->setOnRead([srcIP, track_idx, weakSelf](const Buffer::Ptr &buf, struct sockaddr *addr , int addr_len) {
335 336 337 338 339
                auto strongSelf = weakSelf.lock();
                if (!strongSelf) {
                    return;
                }
                if (((struct sockaddr_in *) addr)->sin_addr.s_addr != srcIP) {
340
                    WarnL << "收到其他地址的rtcp数据:" << SockUtil::inet_ntoa(((struct sockaddr_in *) addr)->sin_addr);
341 342
                    return;
                }
343
                strongSelf->onRtcpPacket(track_idx, strongSelf->_sdp_track[track_idx], (unsigned char *) buf->data(), buf->size());
344 345
            });
        }
346 347
    }

348
    if (track_idx < _sdp_track.size() - 1) {
349
        //需要继续发送SETUP命令
350
        sendSetup(track_idx + 1);
351 352 353 354
        return;
    }
    //所有setup命令发送完毕
    //发送play命令
355
    sendPause(type_play, 0);
xzl committed
356
}
357

358
void RtspPlayer::sendDescribe() {
359
    //发送DESCRIBE命令后处理函数:handleResDESCRIBE
360 361
    _on_response = std::bind(&RtspPlayer::handleResDESCRIBE, this, placeholders::_1);
    sendRtspRequest("DESCRIBE", _play_url, {"Accept", "application/sdp"});
362 363
}

364
void RtspPlayer::sendOptions(){
365
    _on_response = [this](const Parser& parser){
366 367
        if (!handleResponse("OPTIONS", parser)) {
            return;
368 369 370 371 372 373 374 375 376 377 378
        }
        //获取服务器支持的命令
        _supported_cmd.clear();
        auto public_val = split(parser["Public"],",");
        for(auto &cmd : public_val){
            trim(cmd);
            _supported_cmd.emplace(cmd);
        }
        //发送Describe请求,获取sdp
        sendDescribe();
    };
379
    sendRtspRequest("OPTIONS", _play_url);
380 381 382
}

void RtspPlayer::sendKeepAlive(){
383
    _on_response = [this](const Parser& parser){};
384 385
    if(_supported_cmd.find("GET_PARAMETER") != _supported_cmd.end()){
        //支持GET_PARAMETER,用此命令保活
386
        sendRtspRequest("GET_PARAMETER", _play_url);
387 388
    }else{
        //不支持GET_PARAMETER,用OPTIONS命令保活
389
        sendRtspRequest("OPTIONS", _play_url);
390
    }
391 392
}

393
void RtspPlayer::sendPause(int type , uint32_t seekMS){
394
    _on_response = std::bind(&RtspPlayer::handleResPAUSE, this, placeholders::_1, type);
395 396 397
    //开启或暂停rtsp
    switch (type){
        case type_pause:
398
            sendRtspRequest("PAUSE", _content_base);
399 400
            break;
        case type_play:
401
            sendRtspRequest("PLAY", _content_base);
402 403
            break;
        case type_seek:
404
            sendRtspRequest("PLAY", _content_base, {"Range",StrPrinter << "npt=" << setiosflags(ios::fixed) << setprecision(2) << seekMS / 1000.0 << "-"});
405 406 407
            break;
        default:
            WarnL << "unknown type : " << type;
408
            _on_response = nullptr;
409 410
            break;
    }
xzl committed
411
}
xiongziliang committed
412

413 414
void RtspPlayer::pause(bool pause_flag) {
    sendPause(pause_flag ? type_pause : type_seek, getProgressMilliSecond());
xzl committed
415 416
}

417
void RtspPlayer::handleResPAUSE(const Parser& parser,int type) {
418 419 420 421 422 423 424
    if (parser.Url() != "200") {
        switch (type) {
            case type_pause:
                WarnL << "Pause failed:" << parser.Url() << " " << parser.Tail() << endl;
                break;
            case type_play:
                WarnL << "Play failed:" << parser.Url() << " " << parser.Tail() << endl;
425
                onPlayResult_l(SockException(Err_shutdown, StrPrinter << "rtsp play failed:" << parser.Url() << " " << parser.Tail() ), !_play_check_timer);
426 427 428 429 430 431 432 433 434 435
                break;
            case type_seek:
                WarnL << "Seek failed:" << parser.Url() << " " << parser.Tail() << endl;
                break;
        }
        return;
    }

    if (type == type_pause) {
        //暂停成功!
436
        _rtp_check_timer.reset();
437 438 439 440 441 442 443 444 445 446 447 448 449 450 451
        return;
    }

    //play或seek成功
    uint32_t iSeekTo = 0;
    //修正时间轴
    auto strRange = parser["Range"];
    if (strRange.size()) {
        auto strStart = FindField(strRange.data(), "npt=", "-");
        if (strStart == "now") {
            strStart = "0";
        }
        iSeekTo = 1000 * atof(strStart.data());
        DebugL << "seekTo(ms):" << iSeekTo;
    }
452 453

    onPlayResult_l(SockException(Err_success, type == type_seek ? "resum rtsp success" : "rtsp play success"), !_play_check_timer);
xzl committed
454 455
}

456 457
void RtspPlayer::onWholeRtspPacket(Parser &parser) {
    try {
458 459 460 461
        decltype(_on_response) func;
        _on_response.swap(func);
        if(func){
            func(parser);
462 463 464
        }
        parser.Clear();
    } catch (std::exception &err) {
465
        //定时器_pPlayTimer为空后表明握手结束了
466
        onPlayResult_l(SockException(Err_other, err.what()),!_play_check_timer);
xzl committed
467 468 469
    }
}

470 471 472
void RtspPlayer::onRtpPacket(const char *data, uint64_t len) {
    int trackIdx = -1;
    uint8_t interleaved = data[1];
473
    if(interleaved %2 == 0){
474
        trackIdx = getTrackIndexByInterleaved(interleaved);
xiongziliang committed
475
        handleOneRtp(trackIdx, _sdp_track[trackIdx]->_type, _sdp_track[trackIdx]->_samplerate, (unsigned char *)data + 4, len - 4);
476 477
    }else{
        trackIdx = getTrackIndexByInterleaved(interleaved - 1);
478
        onRtcpPacket(trackIdx, _sdp_track[trackIdx], (unsigned char *) data + 4, len - 4);
479
    }
xzl committed
480 481
}

xiongziliang committed
482
//此处预留rtcp处理函数
483
void RtspPlayer::onRtcpPacket(int track_idx, SdpTrack::Ptr &track, unsigned char *data, unsigned int len){}
xiongziliang committed
484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542

#if 0
//改代码提取自FFmpeg,参考之
// Receiver Report
    avio_w8(pb, (RTP_VERSION << 6) + 1); /* 1 report block */
    avio_w8(pb, RTCP_RR);
    avio_wb16(pb, 7); /* length in words - 1 */
    // our own SSRC: we use the server's SSRC + 1 to avoid conflicts
    avio_wb32(pb, s->ssrc + 1);
    avio_wb32(pb, s->ssrc); // server SSRC
    // some placeholders we should really fill...
    // RFC 1889/p64
    extended_max          = stats->cycles + stats->max_seq;
    expected              = extended_max - stats->base_seq;
    lost                  = expected - stats->received;
    lost                  = FFMIN(lost, 0xffffff); // clamp it since it's only 24 bits...
    expected_interval     = expected - stats->expected_prior;
    stats->expected_prior = expected;
    received_interval     = stats->received - stats->received_prior;
    stats->received_prior = stats->received;
    lost_interval         = expected_interval - received_interval;
    if (expected_interval == 0 || lost_interval <= 0)
        fraction = 0;
    else
        fraction = (lost_interval << 8) / expected_interval;

    fraction = (fraction << 24) | lost;

    avio_wb32(pb, fraction); /* 8 bits of fraction, 24 bits of total packets lost */
    avio_wb32(pb, extended_max); /* max sequence received */
    avio_wb32(pb, stats->jitter >> 4); /* jitter */

    if (s->last_rtcp_ntp_time == AV_NOPTS_VALUE) {
        avio_wb32(pb, 0); /* last SR timestamp */
        avio_wb32(pb, 0); /* delay since last SR */
    } else {
        uint32_t middle_32_bits   = s->last_rtcp_ntp_time >> 16; // this is valid, right? do we need to handle 64 bit values special?
        uint32_t delay_since_last = av_rescale(av_gettime_relative() - s->last_rtcp_reception_time,
                                               65536, AV_TIME_BASE);

        avio_wb32(pb, middle_32_bits); /* last SR timestamp */
        avio_wb32(pb, delay_since_last); /* delay since last SR */
    }

    // CNAME
    avio_w8(pb, (RTP_VERSION << 6) + 1); /* 1 report block */
    avio_w8(pb, RTCP_SDES);
    len = strlen(s->hostname);
    avio_wb16(pb, (7 + len + 3) / 4); /* length in words - 1 */
    avio_wb32(pb, s->ssrc + 1);
    avio_w8(pb, 0x01);
    avio_w8(pb, len);
    avio_write(pb, s->hostname, len);
    avio_w8(pb, 0); /* END */
    // padding
    for (len = (7 + len) % 4; len % 4; len++)
        avio_w8(pb, 0);
#endif

543
void RtspPlayer::sendReceiverReport(bool over_tcp, int track_idx){
xiongziliang committed
544 545 546
    static const char s_cname[] = "ZLMediaKitRtsp";
    uint8_t aui8Rtcp[4 + 32 + 10 + sizeof(s_cname) + 1] = {0};
    uint8_t *pui8Rtcp_RR = aui8Rtcp + 4, *pui8Rtcp_SDES = pui8Rtcp_RR + 32;
547 548
    auto &track = _sdp_track[track_idx];
    auto &counter = _rtcp_counter[track_idx];
xiongziliang committed
549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573

    aui8Rtcp[0] = '$';
    aui8Rtcp[1] = track->_interleaved + 1;
    aui8Rtcp[2] = (sizeof(aui8Rtcp) - 4) >>  8;
    aui8Rtcp[3] = (sizeof(aui8Rtcp) - 4) & 0xFF;

    pui8Rtcp_RR[0] = 0x81;/* 1 report block */
    pui8Rtcp_RR[1] = 0xC9;//RTCP_RR
    pui8Rtcp_RR[2] = 0x00;
    pui8Rtcp_RR[3] = 0x07;/* length in words - 1 */

    uint32_t ssrc=htonl(track->_ssrc + 1);
    // our own SSRC: we use the server's SSRC + 1 to avoid conflicts
    memcpy(&pui8Rtcp_RR[4], &ssrc, 4);
    ssrc=htonl(track->_ssrc);
    // server SSRC
    memcpy(&pui8Rtcp_RR[8], &ssrc, 4);

    //FIXME: 8 bits of fraction, 24 bits of total packets lost
    pui8Rtcp_RR[12] = 0x00;
    pui8Rtcp_RR[13] = 0x00;
    pui8Rtcp_RR[14] = 0x00;
    pui8Rtcp_RR[15] = 0x00;

    //FIXME: max sequence received
574
    int cycleCount = getCycleCount(track_idx);
xiongziliang committed
575 576 577 578 579
    pui8Rtcp_RR[16] = cycleCount >> 8;
    pui8Rtcp_RR[17] = cycleCount & 0xFF;
    pui8Rtcp_RR[18] = counter.pktCnt >> 8;
    pui8Rtcp_RR[19] = counter.pktCnt & 0xFF;

580
    uint32_t  jitter = htonl(getJitterSize(track_idx));
xiongziliang committed
581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601
    //FIXME: jitter
    memcpy(pui8Rtcp_RR + 20, &jitter , 4);
    /* last SR timestamp */
    memcpy(pui8Rtcp_RR + 24, &counter.lastTimeStamp, 4);
    uint32_t msInc = htonl(ntohl(counter.timeStamp) - ntohl(counter.lastTimeStamp));
    /* delay since last SR */
    memcpy(pui8Rtcp_RR + 28, &msInc, 4);

    // CNAME
    pui8Rtcp_SDES[0] = 0x81;
    pui8Rtcp_SDES[1] = 0xCA;
    pui8Rtcp_SDES[2] = 0x00;
    pui8Rtcp_SDES[3] = 0x06;

    memcpy(&pui8Rtcp_SDES[4], &ssrc, 4);

    pui8Rtcp_SDES[8] = 0x01;
    pui8Rtcp_SDES[9] = 0x0f;
    memcpy(&pui8Rtcp_SDES[10], s_cname, sizeof(s_cname));
    pui8Rtcp_SDES[10 + sizeof(s_cname)] = 0x00;

602
    if (over_tcp) {
xiongziliang committed
603
        send(obtainBuffer((char *) aui8Rtcp, sizeof(aui8Rtcp)));
604 605
    } else if (_rtcp_sock[track_idx]) {
        _rtcp_sock[track_idx]->send((char *) aui8Rtcp + 4, sizeof(aui8Rtcp) - 4);
xiongziliang committed
606
    }
607
}
xzl committed
608

xiongziliang committed
609
void RtspPlayer::onRtpSorted(const RtpPacket::Ptr &rtppt, int trackidx){
610
    //统计丢包率
611 612 613
    if (_rtp_seq_start[trackidx] == 0 || rtppt->sequence < _rtp_seq_start[trackidx]) {
        _rtp_seq_start[trackidx] = rtppt->sequence;
        _rtp_recv_count[trackidx] = 0;
614
    }
615 616 617
    _rtp_recv_count[trackidx] ++;
    _rtp_seq_now[trackidx] = rtppt->sequence;
    _stamp[trackidx] = rtppt->timeStamp;
618
    //计算相对时间戳
619
    onRecvRTP_l(rtppt, _sdp_track[trackidx]);
xzl committed
620
}
xiongziliang committed
621

622
float RtspPlayer::getPacketLossRate(TrackType type) const{
623 624 625 626 627 628 629
    try {
        auto track_idx = getTrackIndexByTrackType(type);
        if (_rtp_seq_now[track_idx] - _rtp_seq_start[track_idx] + 1 == 0) {
            return 0;
        }
        return 1.0 - (double) _rtp_recv_count[track_idx] / (_rtp_seq_now[track_idx] - _rtp_seq_start[track_idx] + 1);
    } catch (...) {
630 631
        uint64_t totalRecv = 0;
        uint64_t totalSend = 0;
632 633 634
        for (unsigned int i = 0; i < _sdp_track.size(); i++) {
            totalRecv += _rtp_recv_count[i];
            totalSend += (_rtp_seq_now[i] - _rtp_seq_start[i] + 1);
635
        }
636
        if (totalSend == 0) {
637 638
            return 0;
        }
639
        return 1.0 - (double) totalRecv / totalSend;
640
    }
xzl committed
641 642
}

643
uint32_t RtspPlayer::getProgressMilliSecond() const{
644
    return MAX(_stamp[0],_stamp[1]);
xzl committed
645
}
xiongziliang committed
646

647
void RtspPlayer::seekToMilliSecond(uint32_t ms) {
648
    sendPause(type_seek,ms);
xzl committed
649
}
650

651
void RtspPlayer::sendRtspRequest(const string &cmd, const string &url, const std::initializer_list<string> &header) {
652 653 654 655 656 657 658 659 660 661 662
    string key;
    StrCaseMap header_map;
    int i = 0;
    for(auto &val : header){
        if(++i % 2 == 0){
            header_map.emplace(key,val);
        }else{
            key = val;
        }
    }
    sendRtspRequest(cmd,url,header_map);
xiongziliang committed
663
}
xiongziliang committed
664

665
void RtspPlayer::sendRtspRequest(const string &cmd, const string &url,const StrCaseMap &header_const) {
666
    auto header = header_const;
667
    header.emplace("CSeq",StrPrinter << _cseq_send++);
xiongziliang committed
668
    header.emplace("User-Agent",SERVER_NAME);
669

670 671
    if(!_session_id.empty()){
        header.emplace("Session", _session_id);
672 673
    }

674 675
    if(!_realm.empty() && !(*this)[kRtspUser].empty()){
        if(!_md5_nonce.empty()){
676 677 678 679 680 681 682 683 684 685 686
            //MD5认证
            /*
            response计算方法如下:
            RTSP客户端应该使用username + password并计算response如下:
            (1)当password为MD5编码,则
                response = md5( password:nonce:md5(public_method:url)  );
            (2)当password为ANSI字符串,则
                response= md5( md5(username:realm:password):nonce:md5(public_method:url) );
             */
            string encrypted_pwd = (*this)[kRtspPwd];
            if(!(*this)[kRtspPwdIsMD5].as<bool>()){
687
                encrypted_pwd = MD5((*this)[kRtspUser] + ":" + _realm + ":" + encrypted_pwd).hexdigest();
688
            }
689
            auto response = MD5(encrypted_pwd + ":" + _md5_nonce + ":" + MD5(cmd + ":" + url).hexdigest()).hexdigest();
690 691 692
            _StrPrinter printer;
            printer << "Digest ";
            printer << "username=\"" << (*this)[kRtspUser] << "\", ";
693 694
            printer << "realm=\"" << _realm << "\", ";
            printer << "nonce=\"" << _md5_nonce << "\", ";
695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711
            printer << "uri=\"" << url << "\", ";
            printer << "response=\"" << response << "\"";
            header.emplace("Authorization",printer);
        }else if(!(*this)[kRtspPwdIsMD5].as<bool>()){
            //base64认证
            string authStr = StrPrinter << (*this)[kRtspUser] << ":" << (*this)[kRtspPwd];
            char authStrBase64[1024] = {0};
            av_base64_encode(authStrBase64,sizeof(authStrBase64),(uint8_t *)authStr.data(),authStr.size());
            header.emplace("Authorization",StrPrinter << "Basic " << authStrBase64 );
        }
    }

    _StrPrinter printer;
    printer << cmd << " " << url << " RTSP/1.0\r\n";
    for (auto &pr : header){
        printer << pr.first << ": " << pr.second << "\r\n";
    }
712 713
    printer << "\r\n";
    SockSender::send(std::move(printer));
714 715
}

716
void RtspPlayer::onRecvRTP_l(const RtpPacket::Ptr &rtp, const SdpTrack::Ptr &track) {
717
    _rtp_recv_ticker.resetTime();
718
    onRecvRTP(rtp, track);
xiongziliang committed
719

720 721 722 723
    int track_idx = getTrackIndexByTrackType(rtp->type);
    RtcpCounter &counter = _rtcp_counter[track_idx];
    counter.pktCnt = rtp->sequence;
    auto &ticker = _rtcp_send_ticker[track_idx];
xiongziliang committed
724 725 726 727
    if (ticker.elapsedTime() > 5 * 1000) {
        //send rtcp every 5 second
        counter.lastTimeStamp = counter.timeStamp;
        //直接保存网络字节序
728
        memcpy(&counter.timeStamp, rtp->data() + 8, 4);
729
        if (counter.lastTimeStamp != 0) {
730
            sendReceiverReport(_rtp_type == Rtsp::RTP_TCP, track_idx);
xiongziliang committed
731 732
            ticker.resetTime();
        }
733 734

        //有些rtsp服务器需要rtcp保活,有些需要发送信令保活
735
        if (track_idx == 0) {
736 737 738
            //只需要发送一次心跳信令包
            sendKeepAlive();
        }
xiongziliang committed
739
    }
740
}
xiongziliang committed
741

xiongziliang committed
742
void RtspPlayer::onPlayResult_l(const SockException &ex , bool handshake_done) {
xiongziliang committed
743 744 745
    if (ex.getErrCode() == Err_shutdown) {
        //主动shutdown的,不触发回调
        return;
746 747
    }

xiongziliang committed
748
    WarnL << ex.getErrCode() << " " << ex.what();
xiongziliang committed
749
    if (!handshake_done) {
750
        //开始播放阶段
751
        _play_check_timer.reset();
752
        onPlayResult(ex);
753 754
        //是否为性能测试模式
        _benchmark_mode = (*this)[Client::kBenchmarkMode].as<int>();
755 756 757 758 759 760
    } else if (ex) {
        //播放成功后异常断开回调
        onShutdown(ex);
    } else {
        //恢复播放
        onResume();
761 762
    }

xiongziliang committed
763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782
    if (!ex) {
        //播放成功,恢复rtp接收超时定时器
        _rtp_recv_ticker.resetTime();
        int timeoutMS = (*this)[kMediaTimeoutMS].as<int>();
        weak_ptr<RtspPlayer> weakSelf = dynamic_pointer_cast<RtspPlayer>(shared_from_this());
        auto lam = [weakSelf, timeoutMS]() {
            auto strongSelf = weakSelf.lock();
            if (!strongSelf) {
                return false;
            }
            if (strongSelf->_rtp_recv_ticker.elapsedTime() > timeoutMS) {
                //接收rtp媒体数据包超时
                strongSelf->onPlayResult_l(SockException(Err_timeout, "receive rtp timeout"), true);
                return false;
            }
            return true;
        };
        //创建rtp数据接收超时检测定时器
        _rtp_check_timer = std::make_shared<Timer>(timeoutMS / 2000.0, lam, getPoller());
    } else {
783 784
        teardown();
    }
785 786
}

787 788 789
int RtspPlayer::getTrackIndexByInterleaved(int interleaved) const {
    for (unsigned int i = 0; i < _sdp_track.size(); i++) {
        if (_sdp_track[i]->_interleaved == interleaved) {
790 791 792
            return i;
        }
    }
793
    if (_sdp_track.size() == 1) {
794 795
        return 0;
    }
796
    throw SockException(Err_shutdown, StrPrinter << "no such track with interleaved:" << interleaved);
xiongziliang committed
797 798
}

799
int RtspPlayer::getTrackIndexByTrackType(TrackType track_type) const {
800
    for (unsigned int i = 0; i < _sdp_track.size(); i++) {
801
        if (_sdp_track[i]->_type == track_type) {
802 803 804
            return i;
        }
    }
805
    if (_sdp_track.size() == 1) {
806 807
        return 0;
    }
808
    throw SockException(Err_shutdown, StrPrinter << "no such track with type:" << (int) track_type);
809 810
}

xiongziliang committed
811
} /* namespace mediakit */