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.
|
|
|
|
*/
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2014-05-28 07:37:06 +00:00
|
|
|
#include <srs_kernel_flv.hpp>
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2015-11-11 02:37:50 +00:00
|
|
|
// for srs-librtmp, @see https://github.com/ossrs/srs/issues/213
|
2014-11-19 08:16:04 +00:00
|
|
|
#ifndef _WIN32
|
2014-06-14 08:24:15 +00:00
|
|
|
#include <unistd.h>
|
2014-11-19 08:16:04 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
2014-05-23 05:56:40 +00:00
|
|
|
#include <sstream>
|
|
|
|
using namespace std;
|
|
|
|
|
2014-05-28 07:37:06 +00:00
|
|
|
#include <srs_kernel_log.hpp>
|
2014-05-23 05:56:40 +00:00
|
|
|
#include <srs_kernel_error.hpp>
|
2015-09-22 00:48:55 +00:00
|
|
|
#include <srs_kernel_buffer.hpp>
|
2014-07-04 23:33:18 +00:00
|
|
|
#include <srs_kernel_file.hpp>
|
2015-03-10 09:04:02 +00:00
|
|
|
#include <srs_kernel_codec.hpp>
|
2015-05-24 17:02:06 +00:00
|
|
|
#include <srs_kernel_utility.hpp>
|
2015-06-07 07:13:41 +00:00
|
|
|
#include <srs_core_mem_watch.hpp>
|
2017-12-31 04:11:48 +00:00
|
|
|
#include <srs_core_autofree.hpp>
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
SrsMessageHeader::SrsMessageHeader()
|
|
|
|
{
|
|
|
|
message_type = 0;
|
|
|
|
payload_length = 0;
|
|
|
|
timestamp_delta = 0;
|
|
|
|
stream_id = 0;
|
|
|
|
|
|
|
|
timestamp = 0;
|
|
|
|
// we always use the connection chunk-id
|
|
|
|
perfer_cid = RTMP_CID_OverConnection;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsMessageHeader::~SrsMessageHeader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_audio()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AudioMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_video()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_VideoMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_amf0_command()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AMF0CommandMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_amf0_data()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AMF0DataMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_amf3_command()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AMF3CommandMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_amf3_data()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AMF3DataMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_window_ackledgement_size()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_WindowAcknowledgementSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_ackledgement()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_Acknowledgement;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_set_chunk_size()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_SetChunkSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_user_control_message()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_UserControlMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_set_peer_bandwidth()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_SetPeerBandwidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsMessageHeader::is_aggregate()
|
|
|
|
{
|
|
|
|
return message_type == RTMP_MSG_AggregateMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SrsMessageHeader::initialize_amf0_script(int size, int stream)
|
|
|
|
{
|
|
|
|
message_type = RTMP_MSG_AMF0DataMessage;
|
|
|
|
payload_length = (int32_t)size;
|
|
|
|
timestamp_delta = (int32_t)0;
|
|
|
|
timestamp = (int64_t)0;
|
|
|
|
stream_id = (int32_t)stream;
|
|
|
|
|
|
|
|
// amf0 script use connection2 chunk-id
|
|
|
|
perfer_cid = RTMP_CID_OverConnection2;
|
|
|
|
}
|
|
|
|
|
2017-01-30 09:32:18 +00:00
|
|
|
void SrsMessageHeader::initialize_audio(int size, uint32_t time, int stream)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
message_type = RTMP_MSG_AudioMessage;
|
|
|
|
payload_length = (int32_t)size;
|
|
|
|
timestamp_delta = (int32_t)time;
|
|
|
|
timestamp = (int64_t)time;
|
|
|
|
stream_id = (int32_t)stream;
|
|
|
|
|
|
|
|
// audio chunk-id
|
|
|
|
perfer_cid = RTMP_CID_Audio;
|
|
|
|
}
|
|
|
|
|
2017-01-30 09:32:18 +00:00
|
|
|
void SrsMessageHeader::initialize_video(int size, uint32_t time, int stream)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
message_type = RTMP_MSG_VideoMessage;
|
|
|
|
payload_length = (int32_t)size;
|
|
|
|
timestamp_delta = (int32_t)time;
|
|
|
|
timestamp = (int64_t)time;
|
|
|
|
stream_id = (int32_t)stream;
|
|
|
|
|
|
|
|
// video chunk-id
|
|
|
|
perfer_cid = RTMP_CID_Video;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsCommonMessage::SrsCommonMessage()
|
|
|
|
{
|
|
|
|
payload = NULL;
|
|
|
|
size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsCommonMessage::~SrsCommonMessage()
|
|
|
|
{
|
2015-06-13 07:45:25 +00:00
|
|
|
#ifdef SRS_AUTO_MEM_WATCH
|
2015-06-07 07:13:41 +00:00
|
|
|
srs_memory_unwatch(payload);
|
|
|
|
#endif
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(payload);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2015-06-09 02:13:25 +00:00
|
|
|
void SrsCommonMessage::create_payload(int size)
|
|
|
|
{
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(payload);
|
2015-06-09 02:13:25 +00:00
|
|
|
|
|
|
|
payload = new char[size];
|
|
|
|
srs_verbose("create payload for RTMP message. size=%d", size);
|
|
|
|
|
2015-06-13 07:45:25 +00:00
|
|
|
#ifdef SRS_AUTO_MEM_WATCH
|
2015-06-09 02:13:25 +00:00
|
|
|
srs_memory_watch(payload, "RTMP.msg.payload", size);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsCommonMessage::create(SrsMessageHeader* pheader, char* body, int size)
|
2015-12-29 10:33:02 +00:00
|
|
|
{
|
|
|
|
// drop previous payload.
|
|
|
|
srs_freepa(payload);
|
|
|
|
|
|
|
|
this->header = *pheader;
|
|
|
|
this->payload = body;
|
|
|
|
this->size = size;
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_success;
|
2015-12-29 10:33:02 +00:00
|
|
|
}
|
|
|
|
|
2017-06-13 08:10:46 +00:00
|
|
|
SrsSharedMessageHeader::SrsSharedMessageHeader() : payload_length(0), message_type(0), perfer_cid(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsSharedMessageHeader::~SrsSharedMessageHeader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-05-24 17:02:06 +00:00
|
|
|
SrsSharedPtrMessage::SrsSharedPtrPayload::SrsSharedPtrPayload()
|
|
|
|
{
|
|
|
|
payload = NULL;
|
|
|
|
size = 0;
|
2020-03-08 16:40:30 +00:00
|
|
|
rtp_fragments = NULL;
|
|
|
|
nb_rtp_fragments = 0;
|
2015-05-24 17:02:06 +00:00
|
|
|
shared_count = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsSharedPtrMessage::SrsSharedPtrPayload::~SrsSharedPtrPayload()
|
|
|
|
{
|
2015-06-13 07:45:25 +00:00
|
|
|
#ifdef SRS_AUTO_MEM_WATCH
|
2015-06-07 07:13:41 +00:00
|
|
|
srs_memory_unwatch(payload);
|
|
|
|
#endif
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(payload);
|
2020-03-09 11:46:27 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < nb_rtp_fragments; ++i) {
|
2020-03-11 11:21:44 +00:00
|
|
|
srs_freepa(rtp_fragments[i].bytes);
|
2020-03-09 11:46:27 +00:00
|
|
|
}
|
|
|
|
|
2020-03-11 11:21:44 +00:00
|
|
|
if (rtp_fragments != NULL && nb_rtp_fragments > 0) {
|
2020-03-09 11:46:27 +00:00
|
|
|
srs_freepa(rtp_fragments);
|
|
|
|
}
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2020-03-08 16:40:30 +00:00
|
|
|
SrsSharedPtrMessage::SrsSharedPtrMessage() : timestamp(0), stream_id(0), size(0), payload(NULL), rtp_fragments(NULL), nb_rtp_fragments(0)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
ptr = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsSharedPtrMessage::~SrsSharedPtrMessage()
|
|
|
|
{
|
|
|
|
if (ptr) {
|
|
|
|
if (ptr->shared_count == 0) {
|
|
|
|
srs_freep(ptr);
|
|
|
|
} else {
|
|
|
|
ptr->shared_count--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsSharedPtrMessage::create(SrsCommonMessage* msg)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-05-24 17:02:06 +00:00
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = create(&msg->header, msg->payload, msg->size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "create message");
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// to prevent double free of payload:
|
|
|
|
// initialize already attach the payload of msg,
|
|
|
|
// detach the payload to transfer the owner to shared ptr.
|
|
|
|
msg->payload = NULL;
|
|
|
|
msg->size = 0;
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsSharedPtrMessage::create(SrsMessageHeader* pheader, char* payload, int size)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2019-05-20 00:45:42 +00:00
|
|
|
|
|
|
|
if (size < 0) {
|
|
|
|
return srs_error_new(ERROR_RTMP_MESSAGE_CREATE, "create message size=%d", size);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
2019-05-20 00:45:42 +00:00
|
|
|
|
|
|
|
srs_assert(!ptr);
|
2015-05-24 17:02:06 +00:00
|
|
|
ptr = new SrsSharedPtrPayload();
|
|
|
|
|
|
|
|
// direct attach the data.
|
|
|
|
if (pheader) {
|
|
|
|
ptr->header.message_type = pheader->message_type;
|
|
|
|
ptr->header.payload_length = size;
|
|
|
|
ptr->header.perfer_cid = pheader->perfer_cid;
|
|
|
|
this->timestamp = pheader->timestamp;
|
|
|
|
this->stream_id = pheader->stream_id;
|
|
|
|
}
|
|
|
|
ptr->payload = payload;
|
|
|
|
ptr->size = size;
|
|
|
|
|
|
|
|
// message can access it.
|
|
|
|
this->payload = ptr->payload;
|
|
|
|
this->size = ptr->size;
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int SrsSharedPtrMessage::count()
|
|
|
|
{
|
|
|
|
srs_assert(ptr);
|
|
|
|
return ptr->shared_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsSharedPtrMessage::check(int stream_id)
|
|
|
|
{
|
2019-11-04 01:31:30 +00:00
|
|
|
// Ignore error when message has no payload.
|
|
|
|
if (!ptr) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-24 17:02:06 +00:00
|
|
|
// we donot use the complex basic header,
|
|
|
|
// ensure the basic header is 1bytes.
|
2020-01-19 06:34:42 +00:00
|
|
|
if (ptr->header.perfer_cid < 2 || ptr->header.perfer_cid > 63) {
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_info("change the chunk_id=%d to default=%d", ptr->header.perfer_cid, RTMP_CID_ProtocolControl);
|
2015-05-24 17:02:06 +00:00
|
|
|
ptr->header.perfer_cid = RTMP_CID_ProtocolControl;
|
|
|
|
}
|
|
|
|
|
|
|
|
// we assume that the stream_id in a group must be the same.
|
|
|
|
if (this->stream_id == stream_id) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
this->stream_id = stream_id;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-03-09 11:46:27 +00:00
|
|
|
void SrsSharedPtrMessage::set_rtp_fragments(SrsSample* samples, int nb_samples)
|
|
|
|
{
|
|
|
|
ptr->rtp_fragments = samples;
|
|
|
|
ptr->nb_rtp_fragments = nb_samples;
|
|
|
|
|
|
|
|
rtp_fragments = samples;
|
|
|
|
nb_rtp_fragments = nb_samples;
|
|
|
|
}
|
|
|
|
|
2015-05-24 17:02:06 +00:00
|
|
|
bool SrsSharedPtrMessage::is_av()
|
|
|
|
{
|
|
|
|
return ptr->header.message_type == RTMP_MSG_AudioMessage
|
2017-03-25 09:21:39 +00:00
|
|
|
|| ptr->header.message_type == RTMP_MSG_VideoMessage;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsSharedPtrMessage::is_audio()
|
|
|
|
{
|
|
|
|
return ptr->header.message_type == RTMP_MSG_AudioMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SrsSharedPtrMessage::is_video()
|
|
|
|
{
|
|
|
|
return ptr->header.message_type == RTMP_MSG_VideoMessage;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SrsSharedPtrMessage::chunk_header(char* cache, int nb_cache, bool c0)
|
|
|
|
{
|
|
|
|
if (c0) {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_chunk_header_c0(ptr->header.perfer_cid, (uint32_t)timestamp,
|
|
|
|
ptr->header.payload_length, ptr->header.message_type, stream_id, cache, nb_cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
} else {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_chunk_header_c3(ptr->header.perfer_cid, (uint32_t)timestamp,
|
|
|
|
cache, nb_cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SrsSharedPtrMessage* SrsSharedPtrMessage::copy()
|
|
|
|
{
|
|
|
|
srs_assert(ptr);
|
|
|
|
|
|
|
|
SrsSharedPtrMessage* copy = new SrsSharedPtrMessage();
|
|
|
|
|
|
|
|
copy->ptr = ptr;
|
|
|
|
ptr->shared_count++;
|
|
|
|
|
|
|
|
copy->timestamp = timestamp;
|
|
|
|
copy->stream_id = stream_id;
|
|
|
|
copy->payload = ptr->payload;
|
|
|
|
copy->size = ptr->size;
|
2020-03-08 16:40:30 +00:00
|
|
|
copy->rtp_fragments = ptr->rtp_fragments;
|
|
|
|
copy->nb_rtp_fragments = ptr->nb_rtp_fragments;
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
return copy;
|
|
|
|
}
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2017-02-12 13:50:02 +00:00
|
|
|
SrsFlvTransmuxer::SrsFlvTransmuxer()
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-01-30 12:12:36 +00:00
|
|
|
writer = NULL;
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
nb_tag_headers = 0;
|
|
|
|
tag_headers = NULL;
|
|
|
|
nb_iovss_cache = 0;
|
|
|
|
iovss_cache = NULL;
|
|
|
|
nb_ppts = 0;
|
|
|
|
ppts = NULL;
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 13:50:02 +00:00
|
|
|
SrsFlvTransmuxer::~SrsFlvTransmuxer()
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(tag_headers);
|
|
|
|
srs_freepa(iovss_cache);
|
|
|
|
srs_freepa(ppts);
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::initialize(ISrsWriter* fw)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-01-30 12:12:36 +00:00
|
|
|
srs_assert(fw);
|
|
|
|
writer = fw;
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_success;
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2020-02-04 08:13:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_header(bool has_video, bool has_audio)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2020-02-04 08:13:48 +00:00
|
|
|
|
|
|
|
uint8_t av_flag = 0;
|
|
|
|
av_flag += (has_audio? 4:0);
|
|
|
|
av_flag += (has_video? 1:0);
|
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
// 9bytes header and 4bytes first previous-tag-size
|
2015-03-23 13:19:04 +00:00
|
|
|
char flv_header[] = {
|
2014-05-23 05:56:40 +00:00
|
|
|
'F', 'L', 'V', // Signatures "FLV"
|
|
|
|
(char)0x01, // File version (for example, 0x01 for FLV version 1)
|
2020-02-04 08:13:48 +00:00
|
|
|
(char)av_flag, // 4, audio; 1, video; 5 audio+video.
|
2014-05-28 11:01:47 +00:00
|
|
|
(char)0x00, (char)0x00, (char)0x00, (char)0x09 // DataOffset UI32 The length of this header in bytes
|
2014-05-23 05:56:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// flv specification should set the audio and video flag,
|
|
|
|
// actually in practise, application generally ignore this flag,
|
|
|
|
// so we generally set the audio/video to 0.
|
|
|
|
|
2014-05-28 11:01:47 +00:00
|
|
|
// write 9bytes header.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = write_header(flv_header)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write header");
|
2014-05-28 11:01:47 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-28 11:01:47 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_header(char flv_header[9])
|
2014-05-28 11:01:47 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-28 11:01:47 +00:00
|
|
|
|
2014-05-23 05:56:40 +00:00
|
|
|
// write data.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = writer->write(flv_header, 9, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write flv header failed");
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2015-03-26 03:07:45 +00:00
|
|
|
// previous tag size.
|
2014-07-05 03:10:42 +00:00
|
|
|
char pts[] = { (char)0x00, (char)0x00, (char)0x00, (char)0x00 };
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = writer->write(pts, 4, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write pts");
|
2014-05-28 11:01:47 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_metadata(char type, char* data, int size)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
if (size > 0) {
|
|
|
|
cache_metadata(type, data, size, tag_header);
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = write_tag(tag_header, sizeof(tag_header), data, size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write tag");
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_audio(int64_t timestamp, char* data, int size)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
if (size > 0) {
|
|
|
|
cache_audio(timestamp, data, size, tag_header);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = write_tag(tag_header, sizeof(tag_header), data, size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write tag");
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_video(int64_t timestamp, char* data, int size)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-05-24 17:02:06 +00:00
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
if (size > 0) {
|
|
|
|
cache_video(timestamp, data, size, tag_header);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = write_tag(tag_header, sizeof(tag_header), data, size)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write flv video tag failed");
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-02-12 13:50:02 +00:00
|
|
|
int SrsFlvTransmuxer::size_tag(int data_size)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
srs_assert(data_size >= 0);
|
|
|
|
return SRS_FLV_TAG_HEADER_SIZE + data_size + SRS_FLV_PREVIOUS_TAG_SIZE;
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_tags(SrsSharedPtrMessage** msgs, int count)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
// realloc the iovss.
|
|
|
|
int nb_iovss = 3 * count;
|
|
|
|
iovec* iovss = iovss_cache;
|
|
|
|
if (nb_iovss_cache < nb_iovss) {
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(iovss_cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
nb_iovss_cache = nb_iovss;
|
|
|
|
iovss = iovss_cache = new iovec[nb_iovss];
|
|
|
|
}
|
|
|
|
|
|
|
|
// realloc the tag headers.
|
|
|
|
char* cache = tag_headers;
|
|
|
|
if (nb_tag_headers < count) {
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(tag_headers);
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
nb_tag_headers = count;
|
|
|
|
cache = tag_headers = new char[SRS_FLV_TAG_HEADER_SIZE * count];
|
|
|
|
}
|
|
|
|
|
|
|
|
// realloc the pts.
|
|
|
|
char* pts = ppts;
|
|
|
|
if (nb_ppts < count) {
|
2015-11-02 03:05:39 +00:00
|
|
|
srs_freepa(ppts);
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
nb_ppts = count;
|
|
|
|
pts = ppts = new char[SRS_FLV_PREVIOUS_TAG_SIZE * count];
|
|
|
|
}
|
|
|
|
|
|
|
|
// the cache is ok, write each messages.
|
|
|
|
iovec* iovs = iovss;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
SrsSharedPtrMessage* msg = msgs[i];
|
|
|
|
|
|
|
|
// cache all flv header.
|
|
|
|
if (msg->is_audio()) {
|
2019-05-21 00:20:04 +00:00
|
|
|
cache_audio(msg->timestamp, msg->payload, msg->size, cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
} else if (msg->is_video()) {
|
2019-05-21 00:20:04 +00:00
|
|
|
cache_video(msg->timestamp, msg->payload, msg->size, cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
} else {
|
2019-05-21 00:20:04 +00:00
|
|
|
cache_metadata(SrsFrameTypeScript, msg->payload, msg->size, cache);
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// cache all pts.
|
2019-05-21 00:20:04 +00:00
|
|
|
cache_pts(SRS_FLV_TAG_HEADER_SIZE + msg->size, pts);
|
2015-05-24 17:02:06 +00:00
|
|
|
|
|
|
|
// all ioves.
|
|
|
|
iovs[0].iov_base = cache;
|
|
|
|
iovs[0].iov_len = SRS_FLV_TAG_HEADER_SIZE;
|
|
|
|
iovs[1].iov_base = msg->payload;
|
|
|
|
iovs[1].iov_len = msg->size;
|
|
|
|
iovs[2].iov_base = pts;
|
|
|
|
iovs[2].iov_len = SRS_FLV_PREVIOUS_TAG_SIZE;
|
|
|
|
|
|
|
|
// move next.
|
|
|
|
cache += SRS_FLV_TAG_HEADER_SIZE;
|
|
|
|
pts += SRS_FLV_PREVIOUS_TAG_SIZE;
|
|
|
|
iovs += 3;
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = writer->writev(iovss, nb_iovss, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write flv tags failed");
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2015-05-24 17:02:06 +00:00
|
|
|
}
|
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
void SrsFlvTransmuxer::cache_metadata(char type, char* data, int size, char* cache)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
srs_assert(data);
|
|
|
|
|
|
|
|
// 11 bytes tag header
|
|
|
|
/*char tag_header[] = {
|
|
|
|
(char)type, // TagType UB [5], 18 = script data
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // DataSize UI24 Length of the message.
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // Timestamp UI24 Time in milliseconds at which the data in this tag applies.
|
|
|
|
(char)0x00, // TimestampExtended UI8
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // StreamID UI24 Always 0.
|
|
|
|
};*/
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
SrsBuffer* tag_stream = new SrsBuffer(cache, 11);
|
|
|
|
SrsAutoFree(SrsBuffer, tag_stream);
|
|
|
|
|
2015-05-24 17:02:06 +00:00
|
|
|
// write data size.
|
|
|
|
tag_stream->write_1bytes(type);
|
|
|
|
tag_stream->write_3bytes(size);
|
|
|
|
tag_stream->write_3bytes(0x00);
|
|
|
|
tag_stream->write_1bytes(0x00);
|
|
|
|
tag_stream->write_3bytes(0x00);
|
|
|
|
}
|
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
void SrsFlvTransmuxer::cache_audio(int64_t timestamp, char* data, int size, char* cache)
|
2015-05-24 17:02:06 +00:00
|
|
|
{
|
|
|
|
srs_assert(data);
|
|
|
|
|
2014-05-23 05:56:40 +00:00
|
|
|
timestamp &= 0x7fffffff;
|
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
// 11bytes tag header
|
2015-05-24 15:16:56 +00:00
|
|
|
/*char tag_header[] = {
|
2017-02-12 12:38:39 +00:00
|
|
|
(char)SrsFrameTypeAudio, // TagType UB [5], 8 = audio
|
2015-05-24 17:02:06 +00:00
|
|
|
(char)0x00, (char)0x00, (char)0x00, // DataSize UI24 Length of the message.
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // Timestamp UI24 Time in milliseconds at which the data in this tag applies.
|
|
|
|
(char)0x00, // TimestampExtended UI8
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // StreamID UI24 Always 0.
|
|
|
|
};*/
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
SrsBuffer* tag_stream = new SrsBuffer(cache, 11);
|
|
|
|
SrsAutoFree(SrsBuffer, tag_stream);
|
|
|
|
|
2014-05-23 05:56:40 +00:00
|
|
|
// write data size.
|
2017-02-12 12:38:39 +00:00
|
|
|
tag_stream->write_1bytes(SrsFrameTypeAudio);
|
2014-05-23 05:56:40 +00:00
|
|
|
tag_stream->write_3bytes(size);
|
2014-11-19 08:16:04 +00:00
|
|
|
tag_stream->write_3bytes((int32_t)timestamp);
|
2014-05-23 05:56:40 +00:00
|
|
|
// default to little-endian
|
|
|
|
tag_stream->write_1bytes((timestamp >> 24) & 0xFF);
|
2015-05-24 17:02:06 +00:00
|
|
|
tag_stream->write_3bytes(0x00);
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
void SrsFlvTransmuxer::cache_video(int64_t timestamp, char* data, int size, char* cache)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(data);
|
|
|
|
|
2014-05-23 05:56:40 +00:00
|
|
|
timestamp &= 0x7fffffff;
|
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
// 11bytes tag header
|
2015-05-24 15:16:56 +00:00
|
|
|
/*char tag_header[] = {
|
2017-02-12 12:38:39 +00:00
|
|
|
(char)SrsFrameTypeVideo, // TagType UB [5], 9 = video
|
2015-05-24 17:02:06 +00:00
|
|
|
(char)0x00, (char)0x00, (char)0x00, // DataSize UI24 Length of the message.
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // Timestamp UI24 Time in milliseconds at which the data in this tag applies.
|
|
|
|
(char)0x00, // TimestampExtended UI8
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // StreamID UI24 Always 0.
|
|
|
|
};*/
|
2014-05-23 05:56:40 +00:00
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
SrsBuffer* tag_stream = new SrsBuffer(cache, 11);
|
|
|
|
SrsAutoFree(SrsBuffer, tag_stream);
|
|
|
|
|
2014-05-23 05:56:40 +00:00
|
|
|
// write data size.
|
2017-02-12 12:38:39 +00:00
|
|
|
tag_stream->write_1bytes(SrsFrameTypeVideo);
|
2014-05-23 05:56:40 +00:00
|
|
|
tag_stream->write_3bytes(size);
|
2014-11-19 08:16:04 +00:00
|
|
|
tag_stream->write_3bytes((int32_t)timestamp);
|
2014-05-23 05:56:40 +00:00
|
|
|
// default to little-endian
|
|
|
|
tag_stream->write_1bytes((timestamp >> 24) & 0xFF);
|
2015-05-24 17:02:06 +00:00
|
|
|
tag_stream->write_3bytes(0x00);
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2019-05-21 00:20:04 +00:00
|
|
|
void SrsFlvTransmuxer::cache_pts(int size, char* cache)
|
2014-05-29 04:09:26 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
SrsBuffer* tag_stream = new SrsBuffer(cache, 11);
|
|
|
|
SrsAutoFree(SrsBuffer, tag_stream);
|
2015-05-24 17:02:06 +00:00
|
|
|
tag_stream->write_4bytes(size);
|
2014-05-29 04:09:26 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvTransmuxer::write_tag(char* header, int header_size, char* tag, int tag_size)
|
2014-05-23 05:56:40 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-23 05:56:40 +00:00
|
|
|
|
|
|
|
// PreviousTagSizeN UI32 Size of last tag, including its header, in bytes.
|
2015-03-23 13:19:04 +00:00
|
|
|
char pre_size[SRS_FLV_PREVIOUS_TAG_SIZE];
|
2019-05-21 00:20:04 +00:00
|
|
|
cache_pts(tag_size + header_size, pre_size);
|
2015-05-24 14:43:02 +00:00
|
|
|
|
|
|
|
iovec iovs[3];
|
|
|
|
iovs[0].iov_base = header;
|
|
|
|
iovs[0].iov_len = header_size;
|
|
|
|
iovs[1].iov_base = tag;
|
|
|
|
iovs[1].iov_len = tag_size;
|
|
|
|
iovs[2].iov_base = pre_size;
|
2015-05-24 17:02:06 +00:00
|
|
|
iovs[2].iov_len = SRS_FLV_PREVIOUS_TAG_SIZE;
|
2015-05-24 14:43:02 +00:00
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = writer->writev(iovs, 3, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "write flv tag failed");
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-23 05:56:40 +00:00
|
|
|
}
|
|
|
|
|
2014-07-04 23:40:55 +00:00
|
|
|
SrsFlvDecoder::SrsFlvDecoder()
|
|
|
|
{
|
2015-07-28 09:56:50 +00:00
|
|
|
reader = NULL;
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsFlvDecoder::~SrsFlvDecoder()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvDecoder::initialize(ISrsReader* fr)
|
2014-07-04 23:40:55 +00:00
|
|
|
{
|
2015-07-28 09:56:50 +00:00
|
|
|
srs_assert(fr);
|
|
|
|
reader = fr;
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_success;
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvDecoder::read_header(char header[9])
|
2014-07-04 23:40:55 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(header);
|
|
|
|
|
2017-02-01 13:57:32 +00:00
|
|
|
// TODO: FIXME: Should use readfully.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(header, 9, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read header");
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
char* h = header;
|
|
|
|
if (h[0] != 'F' || h[1] != 'L' || h[2] != 'V') {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_error_new(ERROR_KERNEL_FLV_HEADER, "flv header must start with FLV");
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvDecoder::read_tag_header(char* ptype, int32_t* pdata_size, uint32_t* ptime)
|
2014-07-04 23:40:55 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(ptype);
|
|
|
|
srs_assert(pdata_size);
|
|
|
|
srs_assert(ptime);
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-04 23:40:55 +00:00
|
|
|
char th[11]; // tag header
|
|
|
|
|
|
|
|
// read tag header
|
2017-02-01 13:57:32 +00:00
|
|
|
// TODO: FIXME: Should use readfully.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(th, 11, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read flv tag header failed");
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Reserved UB [2]
|
|
|
|
// Filter UB [1]
|
|
|
|
// TagType UB [5]
|
2014-09-26 08:34:13 +00:00
|
|
|
*ptype = (th[0] & 0x1F);
|
2014-07-04 23:40:55 +00:00
|
|
|
|
|
|
|
// DataSize UI24
|
|
|
|
char* pp = (char*)pdata_size;
|
2015-01-02 01:05:16 +00:00
|
|
|
pp[3] = 0;
|
2014-07-04 23:40:55 +00:00
|
|
|
pp[2] = th[1];
|
|
|
|
pp[1] = th[2];
|
|
|
|
pp[0] = th[3];
|
|
|
|
|
|
|
|
// Timestamp UI24
|
|
|
|
pp = (char*)ptime;
|
|
|
|
pp[2] = th[4];
|
|
|
|
pp[1] = th[5];
|
|
|
|
pp[0] = th[6];
|
|
|
|
|
|
|
|
// TimestampExtended UI8
|
|
|
|
pp[3] = th[7];
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvDecoder::read_tag_data(char* data, int32_t size)
|
2014-07-04 23:40:55 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(data);
|
2014-07-04 23:40:55 +00:00
|
|
|
|
2017-02-01 13:57:32 +00:00
|
|
|
// TODO: FIXME: Should use readfully.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(data, size, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read flv tag header failed");
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvDecoder::read_previous_tag_size(char previous_tag_size[4])
|
2014-07-04 23:40:55 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(previous_tag_size);
|
2014-07-04 23:40:55 +00:00
|
|
|
|
|
|
|
// ignore 4bytes tag size.
|
2017-02-01 13:57:32 +00:00
|
|
|
// TODO: FIXME: Should use readfully.
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(previous_tag_size, 4, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read flv previous tag size failed");
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-07-04 23:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SrsFlvVodStreamDecoder::SrsFlvVodStreamDecoder()
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
2015-07-28 09:56:50 +00:00
|
|
|
reader = NULL;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2014-07-04 23:40:55 +00:00
|
|
|
SrsFlvVodStreamDecoder::~SrsFlvVodStreamDecoder()
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvVodStreamDecoder::initialize(ISrsReader* fr)
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-26 05:57:08 +00:00
|
|
|
|
2015-07-28 09:56:50 +00:00
|
|
|
srs_assert(fr);
|
2017-01-30 12:12:36 +00:00
|
|
|
reader = dynamic_cast<SrsFileReader*>(fr);
|
|
|
|
if (!reader) {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_error_new(ERROR_EXPECT_FILE_IO, "stream is not file io");
|
2017-01-30 12:12:36 +00:00
|
|
|
}
|
2014-07-05 09:30:13 +00:00
|
|
|
|
2017-01-30 12:12:36 +00:00
|
|
|
if (!reader->is_open()) {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_error_new(ERROR_KERNEL_FLV_STREAM_CLOSED, "stream is not open for decoder");
|
2014-07-05 03:10:42 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvVodStreamDecoder::read_header_ext(char header[13])
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(header);
|
2014-05-26 05:57:08 +00:00
|
|
|
|
2014-07-05 03:10:42 +00:00
|
|
|
// @remark, always false, for sizeof(char[13]) equals to sizeof(char*)
|
|
|
|
//srs_assert(13 == sizeof(header));
|
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
// 9bytes header and 4bytes first previous-tag-size
|
|
|
|
int size = 13;
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(header, size, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read header");
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvVodStreamDecoder::read_sequence_header_summary(int64_t* pstart, int* psize)
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2017-03-25 09:21:39 +00:00
|
|
|
|
2014-07-05 09:30:13 +00:00
|
|
|
srs_assert(pstart);
|
|
|
|
srs_assert(psize);
|
2014-05-26 05:57:08 +00:00
|
|
|
|
|
|
|
// simply, the first video/audio must be the sequence header.
|
|
|
|
// and must be a sequence video and audio.
|
|
|
|
|
|
|
|
// 11bytes tag header
|
2015-03-23 13:19:04 +00:00
|
|
|
char tag_header[] = {
|
2014-05-26 05:57:08 +00:00
|
|
|
(char)0x00, // TagType UB [5], 9 = video, 8 = audio, 18 = script data
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // DataSize UI24 Length of the message.
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // Timestamp UI24 Time in milliseconds at which the data in this tag applies.
|
|
|
|
(char)0x00, // TimestampExtended UI8
|
|
|
|
(char)0x00, (char)0x00, (char)0x00, // StreamID UI24 Always 0.
|
|
|
|
};
|
|
|
|
|
|
|
|
// discovery the sequence header video and audio.
|
|
|
|
// @remark, maybe no video or no audio.
|
|
|
|
bool got_video = false;
|
|
|
|
bool got_audio = false;
|
|
|
|
// audio/video sequence and data offset.
|
|
|
|
int64_t av_sequence_offset_start = -1;
|
|
|
|
int64_t av_sequence_offset_end = -1;
|
|
|
|
for (;;) {
|
2017-12-31 04:11:48 +00:00
|
|
|
if ((err = reader->read(tag_header, SRS_FLV_TAG_HEADER_SIZE, NULL)) != srs_success) {
|
|
|
|
return srs_error_wrap(err, "read tag header");
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
SrsBuffer* tag_stream = new SrsBuffer(tag_header, SRS_FLV_TAG_HEADER_SIZE);
|
|
|
|
SrsAutoFree(SrsBuffer, tag_stream);
|
2014-05-26 05:57:08 +00:00
|
|
|
|
|
|
|
int8_t tag_type = tag_stream->read_1bytes();
|
|
|
|
int32_t data_size = tag_stream->read_3bytes();
|
|
|
|
|
|
|
|
bool is_video = tag_type == 0x09;
|
|
|
|
bool is_audio = tag_type == 0x08;
|
|
|
|
bool is_not_av = !is_video && !is_audio;
|
|
|
|
if (is_not_av) {
|
|
|
|
// skip body and tag size.
|
2015-07-28 09:56:50 +00:00
|
|
|
reader->skip(data_size + SRS_FLV_PREVIOUS_TAG_SIZE);
|
2014-05-26 05:57:08 +00:00
|
|
|
continue;
|
|
|
|
}
|
2019-05-21 00:43:20 +00:00
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
// if video duplicated, no audio
|
|
|
|
if (is_video && got_video) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// if audio duplicated, no video
|
|
|
|
if (is_audio && got_audio) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// video
|
|
|
|
if (is_video) {
|
|
|
|
srs_assert(!got_video);
|
|
|
|
got_video = true;
|
|
|
|
|
|
|
|
if (av_sequence_offset_start < 0) {
|
2015-07-28 09:56:50 +00:00
|
|
|
av_sequence_offset_start = reader->tellg() - SRS_FLV_TAG_HEADER_SIZE;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
2015-07-28 09:56:50 +00:00
|
|
|
av_sequence_offset_end = reader->tellg() + data_size + SRS_FLV_PREVIOUS_TAG_SIZE;
|
|
|
|
reader->skip(data_size + SRS_FLV_PREVIOUS_TAG_SIZE);
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// audio
|
|
|
|
if (is_audio) {
|
|
|
|
srs_assert(!got_audio);
|
|
|
|
got_audio = true;
|
2019-05-21 00:43:20 +00:00
|
|
|
|
2014-05-26 05:57:08 +00:00
|
|
|
if (av_sequence_offset_start < 0) {
|
2015-07-28 09:56:50 +00:00
|
|
|
av_sequence_offset_start = reader->tellg() - SRS_FLV_TAG_HEADER_SIZE;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
2015-07-28 09:56:50 +00:00
|
|
|
av_sequence_offset_end = reader->tellg() + data_size + SRS_FLV_PREVIOUS_TAG_SIZE;
|
|
|
|
reader->skip(data_size + SRS_FLV_PREVIOUS_TAG_SIZE);
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
2019-02-07 14:49:41 +00:00
|
|
|
|
|
|
|
if (got_audio && got_video) {
|
|
|
|
break;
|
|
|
|
}
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// seek to the sequence header start offset.
|
|
|
|
if (av_sequence_offset_start > 0) {
|
2017-02-03 14:49:19 +00:00
|
|
|
reader->seek2(av_sequence_offset_start);
|
2014-05-26 05:57:08 +00:00
|
|
|
*pstart = av_sequence_offset_start;
|
|
|
|
*psize = (int)(av_sequence_offset_end - av_sequence_offset_start);
|
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t SrsFlvVodStreamDecoder::seek2(int64_t offset)
|
2014-05-26 05:57:08 +00:00
|
|
|
{
|
2017-12-31 04:11:48 +00:00
|
|
|
srs_error_t err = srs_success;
|
2014-05-26 05:57:08 +00:00
|
|
|
|
2015-07-28 09:56:50 +00:00
|
|
|
if (offset >= reader->filesize()) {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_error_new(ERROR_SYSTEM_FILE_EOF, "flv fast decoder seek overflow file, size=%d, offset=%d", (int)reader->filesize(), (int)offset);
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-02-03 14:49:19 +00:00
|
|
|
if (reader->seek2(offset) < 0) {
|
2017-12-31 04:11:48 +00:00
|
|
|
return srs_error_new(ERROR_SYSTEM_FILE_SEEK, "flv fast decoder seek error, size=%d, offset=%d", (int)reader->filesize(), (int)offset);
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-31 04:11:48 +00:00
|
|
|
return err;
|
2014-05-26 05:57:08 +00:00
|
|
|
}
|
2014-05-28 07:37:06 +00:00
|
|
|
|
2014-08-02 14:18:39 +00:00
|
|
|
|