2017-03-25 09:21:39 +00:00
|
|
|
/**
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
2019-12-30 02:10:35 +00:00
|
|
|
* Copyright (c) 2013-2020 Winlin
|
2017-03-25 09:21:39 +00:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
|
|
* this software and associated documentation files (the "Software"), to deal in
|
|
|
|
* the Software without restriction, including without limitation the rights to
|
|
|
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
|
|
|
* subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
|
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
2015-01-24 08:27:30 +00:00
|
|
|
|
|
|
|
#include <srs_app_mpegts_udp.hpp>
|
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
#include <stdlib.h>
|
2015-01-24 08:27:30 +00:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
2017-10-14 03:29:33 +00:00
|
|
|
#include <netdb.h>
|
2015-01-24 08:59:36 +00:00
|
|
|
using namespace std;
|
2015-01-24 08:27:30 +00:00
|
|
|
|
|
|
|
#include <srs_app_config.hpp>
|
|
|
|
#include <srs_kernel_error.hpp>
|
|
|
|
#include <srs_kernel_log.hpp>
|
2015-01-24 08:59:36 +00:00
|
|
|
#include <srs_app_config.hpp>
|
2015-01-25 05:19:22 +00:00
|
|
|
#include <srs_kernel_ts.hpp>
|
2015-09-22 00:48:55 +00:00
|
|
|
#include <srs_kernel_buffer.hpp>
|
2015-01-27 06:28:59 +00:00
|
|
|
#include <srs_kernel_ts.hpp>
|
2015-09-22 00:52:00 +00:00
|
|
|
#include <srs_kernel_stream.hpp>
|
2015-01-31 03:46:51 +00:00
|
|
|
#include <srs_kernel_file.hpp>
|
|
|
|
#include <srs_core_autofree.hpp>
|
2015-01-31 04:21:04 +00:00
|
|
|
#include <srs_kernel_utility.hpp>
|
2015-06-13 08:04:59 +00:00
|
|
|
#include <srs_rtmp_stack.hpp>
|
2015-06-14 00:43:38 +00:00
|
|
|
#include <srs_app_st.hpp>
|
2015-09-22 01:11:07 +00:00
|
|
|
#include <srs_protocol_utility.hpp>
|
2015-01-31 11:46:55 +00:00
|
|
|
#include <srs_app_utility.hpp>
|
2015-09-22 01:05:21 +00:00
|
|
|
#include <srs_protocol_amf0.hpp>
|
2015-01-31 11:46:55 +00:00
|
|
|
#include <srs_raw_avc.hpp>
|
2015-02-19 10:56:21 +00:00
|
|
|
#include <srs_app_pithy_print.hpp>
|
2015-10-14 07:51:01 +00:00
|
|
|
#include <srs_app_rtmp_conn.hpp>
|
2017-01-16 08:20:34 +00:00
|
|
|
#include <srs_protocol_utility.hpp>
|
2015-01-24 08:27:30 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
SrsMpegtsQueue::SrsMpegtsQueue()
|
|
|
|
{
|
|
|
|
nb_audios = nb_videos = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsMpegtsQueue::~SrsMpegtsQueue()
|
|
|
|
{
|
|
|
|
std::map<int64_t, SrsSharedPtrMessage*>::iterator it;
|
|
|
|
for (it = msgs.begin(); it != msgs.end(); ++it) {
|
|
|
|
SrsSharedPtrMessage* msg = it->second;
|
|
|
|
srs_freep(msg);
|
|
|
|
}
|
|
|
|
msgs.clear();
|
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsQueue::push(SrsSharedPtrMessage* msg)
|
2015-01-31 13:16:42 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// TODO: FIXME: use right way.
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
if (msgs.find(msg->timestamp) == msgs.end()) {
|
|
|
|
break;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// adjust the ts, add 1ms.
|
|
|
|
msg->timestamp += 1;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
if (i >= 5) {
|
2017-05-06 06:05:22 +00:00
|
|
|
srs_warn("mpegts: free the msg for dts exists, dts=%" PRId64, msg->timestamp);
|
2015-01-31 15:20:00 +00:00
|
|
|
srs_freep(msg);
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2015-01-31 13:16:42 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
if (msg->is_audio()) {
|
|
|
|
nb_audios++;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
if (msg->is_video()) {
|
|
|
|
nb_videos++;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
msgs[msg->timestamp] = msg;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 13:16:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsSharedPtrMessage* SrsMpegtsQueue::dequeue()
|
|
|
|
{
|
|
|
|
// got 2+ videos and audios, ok to dequeue.
|
|
|
|
bool av_ok = nb_videos >= 2 && nb_audios >= 2;
|
|
|
|
// 100 videos about 30s, while 300 audios about 30s
|
|
|
|
bool av_overflow = nb_videos > 100 || nb_audios > 300;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
if (av_ok || av_overflow) {
|
|
|
|
std::map<int64_t, SrsSharedPtrMessage*>::iterator it = msgs.begin();
|
|
|
|
SrsSharedPtrMessage* msg = it->second;
|
|
|
|
msgs.erase(it);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
if (msg->is_audio()) {
|
|
|
|
nb_audios--;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
if (msg->is_video()) {
|
|
|
|
nb_videos--;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
return msg;
|
2015-01-31 13:16:42 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-24 08:27:30 +00:00
|
|
|
SrsMpegtsOverUdp::SrsMpegtsOverUdp(SrsConfDirective* c)
|
|
|
|
{
|
2015-01-27 08:25:46 +00:00
|
|
|
context = new SrsTsContext();
|
2015-09-22 00:59:52 +00:00
|
|
|
buffer = new SrsSimpleStream();
|
2015-01-24 08:59:36 +00:00
|
|
|
output = _srs_config->get_stream_caster_output(c);
|
2015-03-10 10:07:43 +00:00
|
|
|
|
2017-01-17 02:44:13 +00:00
|
|
|
sdk = NULL;
|
2015-03-10 10:07:43 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
avc = new SrsRawH264Stream();
|
2015-01-31 15:20:00 +00:00
|
|
|
aac = new SrsRawAacStream();
|
2015-01-31 11:46:55 +00:00
|
|
|
h264_sps_changed = false;
|
|
|
|
h264_pps_changed = false;
|
|
|
|
h264_sps_pps_sent = false;
|
2015-01-31 13:16:42 +00:00
|
|
|
queue = new SrsMpegtsQueue();
|
2015-02-19 10:56:21 +00:00
|
|
|
pprint = SrsPithyPrint::create_caster();
|
2015-01-24 08:27:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsMpegtsOverUdp::~SrsMpegtsOverUdp()
|
|
|
|
{
|
2015-01-31 11:46:55 +00:00
|
|
|
close();
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 15:54:04 +00:00
|
|
|
srs_freep(buffer);
|
2015-01-27 08:25:46 +00:00
|
|
|
srs_freep(context);
|
2015-01-31 11:46:55 +00:00
|
|
|
srs_freep(avc);
|
2015-01-31 15:20:00 +00:00
|
|
|
srs_freep(aac);
|
2015-01-31 13:16:42 +00:00
|
|
|
srs_freep(queue);
|
2015-02-19 10:56:21 +00:00
|
|
|
srs_freep(pprint);
|
2015-01-24 08:27:30 +00:00
|
|
|
}
|
|
|
|
|
2017-10-14 03:29:33 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::on_udp_packet(const sockaddr* from, const int fromlen, char* buf, int nb_buf)
|
2015-01-24 08:27:30 +00:00
|
|
|
{
|
2017-10-14 03:29:33 +00:00
|
|
|
char address_string[64];
|
|
|
|
char port_string[16];
|
|
|
|
if(getnameinfo(from, fromlen,
|
|
|
|
(char*)&address_string, sizeof(address_string),
|
|
|
|
(char*)&port_string, sizeof(port_string),
|
2018-03-20 11:50:46 +00:00
|
|
|
NI_NUMERICHOST|NI_NUMERICSERV)) {
|
2017-10-14 03:29:33 +00:00
|
|
|
return srs_error_new(ERROR_SYSTEM_IP_INVALID, "bad address");
|
|
|
|
}
|
|
|
|
std::string peer_ip = std::string(address_string);
|
|
|
|
int peer_port = atoi(port_string);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 15:54:04 +00:00
|
|
|
// append to buffer.
|
|
|
|
buffer->append(buf, nb_buf);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = on_udp_bytes(peer_ip, peer_port, buf, nb_buf);
|
|
|
|
if (err != srs_success) {
|
|
|
|
return srs_error_wrap(err, "process udp");
|
2017-07-29 04:45:17 +00:00
|
|
|
}
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-03-10 10:07:43 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::on_udp_bytes(string host, int port, char* buf, int nb_buf)
|
2015-03-10 10:07:43 +00:00
|
|
|
{
|
2017-09-22 13:50:54 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 03:46:51 +00:00
|
|
|
// collect nMB data to parse in a time.
|
|
|
|
// TODO: FIXME: comment the following for release.
|
|
|
|
//if (buffer->length() < 3 * 1024 * 1024) return ret;
|
|
|
|
// TODO: FIXME: remove the debug to file.
|
|
|
|
#if 0
|
|
|
|
SrsFileWriter fw;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = fw.open("latest.ts")) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "open file");
|
2015-01-31 03:46:51 +00:00
|
|
|
}
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = fw.write(buffer->bytes(), buffer->length(), NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write data");
|
2015-01-31 03:46:51 +00:00
|
|
|
}
|
|
|
|
fw.close();
|
|
|
|
#endif
|
|
|
|
#if 0
|
|
|
|
SrsFileReader fr;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = fr.open("latest.ts")) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "open file");
|
2015-01-31 03:46:51 +00:00
|
|
|
}
|
|
|
|
buffer->erase(buffer->length());
|
|
|
|
int nb_fbuf = fr.filesize();
|
|
|
|
char* fbuf = new char[nb_fbuf];
|
2015-11-02 03:29:20 +00:00
|
|
|
SrsAutoFreeA(char, fbuf);
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = fr.read(fbuf, nb_fbuf, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read data");
|
2015-01-31 03:46:51 +00:00
|
|
|
}
|
|
|
|
fr.close();
|
|
|
|
buffer->append(fbuf, nb_fbuf);
|
|
|
|
#endif
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 15:54:04 +00:00
|
|
|
// find the sync byte of mpegts.
|
|
|
|
char* p = buffer->bytes();
|
|
|
|
for (int i = 0; i < buffer->length(); i++) {
|
|
|
|
if (p[i] != 0x47) {
|
|
|
|
continue;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 15:54:04 +00:00
|
|
|
if (i > 0) {
|
|
|
|
buffer->erase(i);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-25 02:54:25 +00:00
|
|
|
// drop ts packet when size not modulus by 188
|
2015-01-29 15:54:04 +00:00
|
|
|
if (buffer->length() < SRS_TS_PACKET_SIZE) {
|
2015-03-10 10:07:43 +00:00
|
|
|
srs_warn("udp: wait %s:%d packet %d/%d bytes", host.c_str(), port, nb_buf, buffer->length());
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-25 02:54:25 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-27 06:28:59 +00:00
|
|
|
// use stream to parse ts packet.
|
2017-01-16 07:47:26 +00:00
|
|
|
int nb_packet = buffer->length() / SRS_TS_PACKET_SIZE;
|
2015-01-29 15:54:04 +00:00
|
|
|
for (int i = 0; i < nb_packet; i++) {
|
|
|
|
char* p = buffer->bytes() + (i * SRS_TS_PACKET_SIZE);
|
2018-01-01 13:20:57 +00:00
|
|
|
|
|
|
|
SrsBuffer* stream = new SrsBuffer(p, SRS_TS_PACKET_SIZE);
|
|
|
|
SrsAutoFree(SrsBuffer, stream);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-27 06:28:59 +00:00
|
|
|
// process each ts packet
|
2017-09-22 13:50:54 +00:00
|
|
|
if ((err = context->decode(stream, this)) != srs_success) {
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_warn("parse ts packet err=%s", srs_error_desc(err).c_str());
|
|
|
|
srs_error_reset(err);
|
2015-01-29 14:58:02 +00:00
|
|
|
continue;
|
2015-01-25 02:54:25 +00:00
|
|
|
}
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 15:54:04 +00:00
|
|
|
// erase consumed bytes
|
|
|
|
if (nb_packet > 0) {
|
|
|
|
buffer->erase(nb_packet * SRS_TS_PACKET_SIZE);
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-25 02:54:25 +00:00
|
|
|
}
|
2015-01-24 08:27:30 +00:00
|
|
|
|
2017-09-22 13:50:54 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::on_ts_message(SrsTsMessage* msg)
|
2015-01-29 14:58:02 +00:00
|
|
|
{
|
2017-09-22 13:50:54 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-02-19 10:56:21 +00:00
|
|
|
pprint->elapse();
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 03:46:51 +00:00
|
|
|
// about the bytes of msg, specified by elementary stream which indicates by PES_packet_data_byte and stream_id
|
|
|
|
// for example, when SrsTsStream of SrsTsChannel indicates stream_type is SrsTsStreamVideoMpeg4 and SrsTsStreamAudioMpeg4,
|
|
|
|
// the elementary stream can be mux in "2.11 Carriage of ISO/IEC 14496 data" in hls-mpeg-ts-iso13818-1.pdf, page 103
|
2017-03-25 09:21:39 +00:00
|
|
|
// @remark, the most popular stream_id is 0xe0 for h.264 over mpegts, which indicates the stream_id is video and
|
2017-01-26 09:28:49 +00:00
|
|
|
// stream_number is 0, where I guess the elementary is specified in annexb format(ISO_IEC_14496-10-AVC-2003.pdf, page 211).
|
|
|
|
// because when audio stream_number is 0, the elementary is ADTS(ISO_IEC_14496-3-AAC-2001.pdf, page 75, 1.A.2.2 ADTS).
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 03:46:51 +00:00
|
|
|
// about the bytes of PES_packet_data_byte, defined in hls-mpeg-ts-iso13818-1.pdf, page 58
|
2015-05-09 15:12:12 +00:00
|
|
|
// PES_packet_data_byte "C PES_packet_data_bytes shall be contiguous bytes of data from the elementary stream
|
|
|
|
// indicated by the packets stream_id or PID. When the elementary stream data conforms to ITU-T
|
2015-01-31 03:46:51 +00:00
|
|
|
// Rec. H.262 | ISO/IEC 13818-2 or ISO/IEC 13818-3, the PES_packet_data_bytes shall be byte aligned to the bytes of this
|
|
|
|
// Recommendation | International Standard. The byte-order of the elementary stream shall be preserved. The number of
|
|
|
|
// PES_packet_data_bytes, N, is specified by the PES_packet_length field. N shall be equal to the value indicated in the
|
|
|
|
// PES_packet_length minus the number of bytes between the last byte of the PES_packet_length field and the first
|
|
|
|
// PES_packet_data_byte.
|
2017-03-25 09:21:39 +00:00
|
|
|
//
|
2015-01-31 03:46:51 +00:00
|
|
|
// In the case of a private_stream_1, private_stream_2, ECM_stream, or EMM_stream, the contents of the
|
|
|
|
// PES_packet_data_byte field are user definable and will not be specified by ITU-T | ISO/IEC in the future.
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 03:46:51 +00:00
|
|
|
// about the bytes of stream_id, define in hls-mpeg-ts-iso13818-1.pdf, page 49
|
2015-05-09 15:12:12 +00:00
|
|
|
// stream_id "C In Program Streams, the stream_id specifies the type and number of the elementary stream as defined by the
|
2015-01-31 03:46:51 +00:00
|
|
|
// stream_id Table 2-18. In Transport Streams, the stream_id may be set to any valid value which correctly describes the
|
|
|
|
// elementary stream type as defined in Table 2-18. In Transport Streams, the elementary stream type is specified in the
|
|
|
|
// Program Specific Information as specified in 2.4.4.
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-05-09 15:12:12 +00:00
|
|
|
// about the stream_id table, define in Table 2-18 "C Stream_id assignments, hls-mpeg-ts-iso13818-1.pdf, page 52.
|
2017-03-25 09:21:39 +00:00
|
|
|
//
|
2015-01-31 03:46:51 +00:00
|
|
|
// 110x xxxx
|
|
|
|
// ISO/IEC 13818-3 or ISO/IEC 11172-3 or ISO/IEC 13818-7 or ISO/IEC
|
|
|
|
// 14496-3 audio stream number x xxxx
|
|
|
|
// ((sid >> 5) & 0x07) == SrsTsPESStreamIdAudio
|
2017-03-25 09:21:39 +00:00
|
|
|
//
|
2015-01-31 03:46:51 +00:00
|
|
|
// 1110 xxxx
|
|
|
|
// ITU-T Rec. H.262 | ISO/IEC 13818-2 or ISO/IEC 11172-2 or ISO/IEC
|
|
|
|
// 14496-2 video stream number xxxx
|
|
|
|
// ((stream_id >> 4) & 0x0f) == SrsTsPESStreamIdVideo
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-02-19 10:56:21 +00:00
|
|
|
if (pprint->can_print()) {
|
2017-05-06 06:05:22 +00:00
|
|
|
srs_trace("<- " SRS_CONSTS_LOG_STREAM_CASTER " mpegts: got %s age=%d stream=%s, dts=%" PRId64 ", pts=%" PRId64 ", size=%d, us=%d, cc=%d, sid=%#x(%s-%d)",
|
2017-03-25 09:21:39 +00:00
|
|
|
(msg->channel->apply == SrsTsPidApplyVideo)? "Video":"Audio", pprint->age(), srs_ts_stream2string(msg->channel->stream).c_str(),
|
|
|
|
msg->dts, msg->pts, msg->payload->length(), msg->packet->payload_unit_start_indicator, msg->continuity_counter, msg->sid,
|
|
|
|
msg->is_audio()? "A":msg->is_video()? "V":"N", msg->stream_number());
|
2015-02-19 10:56:21 +00:00
|
|
|
}
|
2017-01-11 09:43:29 +00:00
|
|
|
|
|
|
|
// When the audio SID is private stream 1, we use common audio.
|
|
|
|
// @see https://github.com/ossrs/srs/issues/740
|
|
|
|
if (msg->channel->apply == SrsTsPidApplyAudio && msg->sid == SrsTsPESStreamIdPrivateStream1) {
|
|
|
|
msg->sid = SrsTsPESStreamIdAudioCommon;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 04:21:04 +00:00
|
|
|
// when not audio/video, or not adts/annexb format, donot support.
|
|
|
|
if (msg->stream_number() != 0) {
|
2017-09-22 13:50:54 +00:00
|
|
|
return srs_error_new(ERROR_STREAM_CASTER_TS_ES, "ts: unsupported stream format, sid=%#x(%s-%d)",
|
|
|
|
msg->sid, msg->is_audio()? "A":msg->is_video()? "V":"N", msg->stream_number());
|
2015-01-31 04:21:04 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// check supported codec
|
|
|
|
if (msg->channel->stream != SrsTsStreamVideoH264 && msg->channel->stream != SrsTsStreamAudioAAC) {
|
2017-09-22 13:50:54 +00:00
|
|
|
return srs_error_new(ERROR_STREAM_CASTER_TS_CODEC, "ts: unsupported stream codec=%d", msg->channel->stream);
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// parse the stream.
|
2018-01-01 13:20:57 +00:00
|
|
|
SrsBuffer avs(msg->payload->bytes(), msg->payload->length());
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// publish audio or video.
|
|
|
|
if (msg->channel->stream == SrsTsStreamVideoH264) {
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = on_ts_video(msg, &avs)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "ts: consume video");
|
2017-09-22 13:50:54 +00:00
|
|
|
}
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2015-01-31 15:20:00 +00:00
|
|
|
if (msg->channel->stream == SrsTsStreamAudioAAC) {
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = on_ts_audio(msg, &avs)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "ts: consume audio");
|
2017-09-22 13:50:54 +00:00
|
|
|
}
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-29 14:58:02 +00:00
|
|
|
// TODO: FIXME: implements it.
|
2017-09-22 13:50:54 +00:00
|
|
|
return err;
|
2015-01-29 14:58:02 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::on_ts_video(SrsTsMessage* msg, SrsBuffer* avs)
|
2015-01-31 11:46:55 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// ensure rtmp connected.
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = connect()) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "connect");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// ts tbn to flv tbn.
|
2017-01-30 09:32:18 +00:00
|
|
|
uint32_t dts = (uint32_t)(msg->dts / 90);
|
|
|
|
uint32_t pts = (uint32_t)(msg->dts / 90);
|
2015-01-31 11:46:55 +00:00
|
|
|
|
|
|
|
// send each frame.
|
|
|
|
while (!avs->empty()) {
|
|
|
|
char* frame = NULL;
|
|
|
|
int frame_size = 0;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->annexb_demux(avs, &frame, &frame_size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "demux annexb");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2015-04-04 07:53:36 +00:00
|
|
|
|
|
|
|
// 5bits, 7.3.1 NAL unit syntax,
|
2017-01-26 09:28:49 +00:00
|
|
|
// ISO_IEC_14496-10-AVC-2003.pdf, page 44.
|
2015-04-04 07:53:36 +00:00
|
|
|
// 7: SPS, 8: PPS, 5: I Frame, 1: P Frame
|
|
|
|
SrsAvcNaluType nal_unit_type = (SrsAvcNaluType)(frame[0] & 0x1f);
|
|
|
|
|
|
|
|
// ignore the nalu type sps(7), pps(8), aud(9)
|
2015-04-04 08:05:11 +00:00
|
|
|
if (nal_unit_type == SrsAvcNaluTypeAccessUnitDelimiter) {
|
|
|
|
continue;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// for sps
|
|
|
|
if (avc->is_sps(frame, frame_size)) {
|
|
|
|
std::string sps;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->sps_demux(frame, frame_size, sps)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "demux sps");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
if (h264_sps == sps) {
|
2015-01-31 11:46:55 +00:00
|
|
|
continue;
|
|
|
|
}
|
2015-01-31 15:20:00 +00:00
|
|
|
h264_sps_changed = true;
|
|
|
|
h264_sps = sps;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = write_h264_sps_pps(dts, pts)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write sps/pps");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
continue;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// for pps
|
|
|
|
if (avc->is_pps(frame, frame_size)) {
|
|
|
|
std::string pps;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->pps_demux(frame, frame_size, pps)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "demux pps");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
if (h264_pps == pps) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
h264_pps_changed = true;
|
|
|
|
h264_pps = pps;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = write_h264_sps_pps(dts, pts)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write sps/pps");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2015-03-26 11:52:21 +00:00
|
|
|
|
|
|
|
// ibp frame.
|
2015-04-04 07:53:36 +00:00
|
|
|
// TODO: FIXME: we should group all frames to a rtmp/flv message from one ts message.
|
2017-03-25 04:42:37 +00:00
|
|
|
srs_info("mpegts: demux avc ibp frame size=%d, dts=%d", frame_size, dts);
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = write_h264_ipb_frame(frame, frame_size, dts, pts)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write frame");
|
2015-03-26 11:52:21 +00:00
|
|
|
}
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2015-03-26 03:07:45 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::write_h264_sps_pps(uint32_t dts, uint32_t pts)
|
2015-01-31 11:46:55 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-01-31 11:46:55 +00:00
|
|
|
|
2015-04-13 08:40:31 +00:00
|
|
|
// TODO: FIMXE: there exists bug, see following comments.
|
|
|
|
// when sps or pps changed, update the sequence header,
|
|
|
|
// for the pps maybe not changed while sps changed.
|
|
|
|
// so, we must check when each video ts message frame parsed.
|
2015-01-31 11:46:55 +00:00
|
|
|
if (!h264_sps_changed || !h264_pps_changed) {
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// h264 raw to h264 packet.
|
|
|
|
std::string sh;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->mux_sequence_header(h264_sps, h264_pps, dts, pts, sh)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "mux sequence header");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// h264 packet to flv packet.
|
2017-02-12 12:38:39 +00:00
|
|
|
int8_t frame_type = SrsVideoAvcFrameTypeKeyFrame;
|
|
|
|
int8_t avc_packet_type = SrsVideoAvcFrameTraitSequenceHeader;
|
2015-01-31 11:46:55 +00:00
|
|
|
char* flv = NULL;
|
|
|
|
int nb_flv = 0;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->mux_avc2flv(sh, frame_type, avc_packet_type, dts, pts, &flv, &nb_flv)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "avc to flv");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// the timestamp in rtmp message header is dts.
|
2017-01-30 09:32:18 +00:00
|
|
|
uint32_t timestamp = dts;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = rtmp_write_packet(SrsFrameTypeVideo, timestamp, flv, nb_flv)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write packet");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
// reset sps and pps.
|
|
|
|
h264_sps_changed = false;
|
|
|
|
h264_pps_changed = false;
|
|
|
|
h264_sps_pps_sent = true;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::write_h264_ipb_frame(char* frame, int frame_size, uint32_t dts, uint32_t pts)
|
2015-01-31 11:46:55 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-01-31 11:46:55 +00:00
|
|
|
|
|
|
|
// when sps or pps not sent, ignore the packet.
|
2015-11-11 02:37:50 +00:00
|
|
|
// @see https://github.com/ossrs/srs/issues/203
|
2015-01-31 11:46:55 +00:00
|
|
|
if (!h264_sps_pps_sent) {
|
2018-01-01 13:20:57 +00:00
|
|
|
return srs_error_new(ERROR_H264_DROP_BEFORE_SPS_PPS, "drop sps/pps");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2015-04-04 07:53:36 +00:00
|
|
|
|
|
|
|
// 5bits, 7.3.1 NAL unit syntax,
|
2017-01-26 09:28:49 +00:00
|
|
|
// ISO_IEC_14496-10-AVC-2003.pdf, page 44.
|
2015-04-04 07:53:36 +00:00
|
|
|
// 7: SPS, 8: PPS, 5: I Frame, 1: P Frame
|
|
|
|
SrsAvcNaluType nal_unit_type = (SrsAvcNaluType)(frame[0] & 0x1f);
|
|
|
|
|
|
|
|
// for IDR frame, the frame is keyframe.
|
2017-02-12 12:38:39 +00:00
|
|
|
SrsVideoAvcFrameType frame_type = SrsVideoAvcFrameTypeInterFrame;
|
2015-04-04 07:53:36 +00:00
|
|
|
if (nal_unit_type == SrsAvcNaluTypeIDR) {
|
2017-02-12 12:38:39 +00:00
|
|
|
frame_type = SrsVideoAvcFrameTypeKeyFrame;
|
2015-04-04 07:53:36 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
std::string ibp;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->mux_ipb_frame(frame, frame_size, ibp)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "mux frame");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 12:38:39 +00:00
|
|
|
int8_t avc_packet_type = SrsVideoAvcFrameTraitNALU;
|
2015-01-31 11:46:55 +00:00
|
|
|
char* flv = NULL;
|
|
|
|
int nb_flv = 0;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = avc->mux_avc2flv(ibp, frame_type, avc_packet_type, dts, pts, &flv, &nb_flv)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "mux avc to flv");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// the timestamp in rtmp message header is dts.
|
2017-01-30 09:32:18 +00:00
|
|
|
uint32_t timestamp = dts;
|
2017-02-12 12:38:39 +00:00
|
|
|
return rtmp_write_packet(SrsFrameTypeVideo, timestamp, flv, nb_flv);
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::on_ts_audio(SrsTsMessage* msg, SrsBuffer* avs)
|
2015-01-31 15:20:00 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// ensure rtmp connected.
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = connect()) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "connect");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// ts tbn to flv tbn.
|
2017-01-30 09:32:18 +00:00
|
|
|
uint32_t dts = (uint32_t)(msg->dts / 90);
|
2015-01-31 15:20:00 +00:00
|
|
|
|
|
|
|
// send each frame.
|
|
|
|
while (!avs->empty()) {
|
|
|
|
char* frame = NULL;
|
|
|
|
int frame_size = 0;
|
|
|
|
SrsRawAacStreamCodec codec;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = aac->adts_demux(avs, &frame, &frame_size, codec)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "demux adts");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// ignore invalid frame,
|
|
|
|
// * atleast 1bytes for aac to decode the data.
|
|
|
|
if (frame_size <= 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
srs_info("mpegts: demux aac frame size=%d, dts=%d", frame_size, dts);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// generate sh.
|
|
|
|
if (aac_specific_config.empty()) {
|
|
|
|
std::string sh;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = aac->mux_sequence_header(&codec, sh)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "mux sequence header");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
aac_specific_config = sh;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
codec.aac_packet_type = 0;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = write_audio_raw_frame((char*)sh.data(), (int)sh.length(), &codec, dts)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write raw audio frame");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
// audio raw data.
|
|
|
|
codec.aac_packet_type = 1;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = write_audio_raw_frame(frame, frame_size, &codec, dts)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write audio raw frame");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::write_audio_raw_frame(char* frame, int frame_size, SrsRawAacStreamCodec* codec, uint32_t dts)
|
2015-01-31 15:20:00 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 15:20:00 +00:00
|
|
|
char* data = NULL;
|
|
|
|
int size = 0;
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = aac->mux_aac2flv(frame, frame_size, codec, dts, &data, &size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "mux aac to flv");
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 12:38:39 +00:00
|
|
|
return rtmp_write_packet(SrsFrameTypeAudio, dts, data, size);
|
2015-01-31 15:20:00 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::rtmp_write_packet(char type, uint32_t timestamp, char* data, int size)
|
2015-01-31 11:46:55 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-01-31 11:46:55 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = connect()) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "connect");
|
2017-01-17 02:44:13 +00:00
|
|
|
}
|
|
|
|
|
2015-01-31 11:46:55 +00:00
|
|
|
SrsSharedPtrMessage* msg = NULL;
|
2015-10-14 07:51:01 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = srs_rtmp_create_msg(type, timestamp, data, size, sdk->sid(), &msg)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "create message");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
srs_assert(msg);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
// push msg to queue.
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = queue->push(msg)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "push to queue");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
// for all ready msg, dequeue and send out.
|
|
|
|
for (;;) {
|
|
|
|
if ((msg = queue->dequeue()) == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-02-19 10:56:21 +00:00
|
|
|
if (pprint->can_print()) {
|
2017-05-06 06:05:22 +00:00
|
|
|
srs_trace("mpegts: send msg %s age=%d, dts=%" PRId64 ", size=%d",
|
2017-03-25 09:21:39 +00:00
|
|
|
msg->is_audio()? "A":msg->is_video()? "V":"N", pprint->age(), msg->timestamp, msg->size);
|
2015-02-19 10:56:21 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2015-01-31 13:16:42 +00:00
|
|
|
// send out encoded msg.
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = sdk->send_and_free_message(msg)) != srs_success) {
|
2017-01-17 02:44:13 +00:00
|
|
|
close();
|
2018-01-01 13:20:57 +00:00
|
|
|
return srs_error_wrap(err, "send messages");
|
2015-01-31 13:16:42 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-31 11:46:55 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t SrsMpegtsOverUdp::connect()
|
2015-01-31 11:46:55 +00:00
|
|
|
{
|
2018-01-01 13:20:57 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2017-01-17 02:44:13 +00:00
|
|
|
// Ignore when connected.
|
|
|
|
if (sdk) {
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2015-10-14 02:48:08 +00:00
|
|
|
|
2019-04-22 00:03:12 +00:00
|
|
|
srs_utime_t cto = SRS_CONSTS_RTMP_TIMEOUT;
|
|
|
|
srs_utime_t sto = SRS_CONSTS_RTMP_PULSE;
|
2017-01-17 04:25:30 +00:00
|
|
|
sdk = new SrsSimpleRtmpClient(output, cto, sto);
|
2017-01-17 02:44:13 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
if ((err = sdk->connect()) != srs_success) {
|
2017-01-17 02:44:13 +00:00
|
|
|
close();
|
2019-04-22 00:03:12 +00:00
|
|
|
return srs_error_wrap(err, "connect %s failed, cto=%dms, sto=%dms.", output.c_str(), srsu2msi(cto), srsu2msi(sto));
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
2018-08-02 01:17:49 +00:00
|
|
|
if ((err = sdk->publish(SRS_CONSTS_RTMP_PROTOCOL_CHUNK_SIZE)) != srs_success) {
|
2017-01-17 02:44:13 +00:00
|
|
|
close();
|
2018-01-01 13:20:57 +00:00
|
|
|
return srs_error_wrap(err, "publish");
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2018-01-01 13:20:57 +00:00
|
|
|
return err;
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SrsMpegtsOverUdp::close()
|
|
|
|
{
|
2017-01-17 02:44:13 +00:00
|
|
|
srs_freep(sdk);
|
2015-01-31 11:46:55 +00:00
|
|
|
}
|
|
|
|
|