mirror of
https://github.com/ossrs/srs.git
synced 2025-02-15 04:42:04 +00:00
283 lines
6.8 KiB
C++
283 lines
6.8 KiB
C++
/**
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2013-2020 Winlin
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <srs_app_fragment.hpp>
|
|
|
|
#include <srs_kernel_utility.hpp>
|
|
#include <srs_kernel_log.hpp>
|
|
#include <srs_kernel_error.hpp>
|
|
|
|
#include <unistd.h>
|
|
#include <sstream>
|
|
using namespace std;
|
|
|
|
SrsFragment::SrsFragment()
|
|
{
|
|
dur = 0;
|
|
start_dts = -1;
|
|
sequence_header = false;
|
|
}
|
|
|
|
SrsFragment::~SrsFragment()
|
|
{
|
|
}
|
|
|
|
void SrsFragment::append(int64_t dts)
|
|
{
|
|
// The max positive ms is 0x7fffffffffffffff/1000.
|
|
static const int64_t maxMS = 0x20c49ba5e353f7LL;
|
|
|
|
// We reset negative or overflow dts to zero.
|
|
if (dts > maxMS || dts < 0) {
|
|
dts = 0;
|
|
}
|
|
|
|
srs_utime_t dts_in_tbn = dts * SRS_UTIME_MILLISECONDS;
|
|
|
|
if (start_dts == -1) {
|
|
start_dts = dts_in_tbn;
|
|
}
|
|
|
|
// TODO: FIXME: Use cumulus dts.
|
|
start_dts = srs_min(start_dts, dts_in_tbn);
|
|
dur = dts_in_tbn - start_dts;
|
|
}
|
|
|
|
srs_utime_t SrsFragment::duration()
|
|
{
|
|
return dur;
|
|
}
|
|
|
|
bool SrsFragment::is_sequence_header()
|
|
{
|
|
return sequence_header;
|
|
}
|
|
|
|
void SrsFragment::set_sequence_header(bool v)
|
|
{
|
|
sequence_header = v;
|
|
}
|
|
|
|
string SrsFragment::fullpath()
|
|
{
|
|
return filepath;
|
|
}
|
|
|
|
void SrsFragment::set_path(string v)
|
|
{
|
|
filepath = v;
|
|
}
|
|
|
|
srs_error_t SrsFragment::unlink_file()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
if (::unlink(filepath.c_str()) < 0) {
|
|
return srs_error_new(ERROR_SYSTEM_FRAGMENT_UNLINK, "unlink %s", filepath.c_str());
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
srs_error_t SrsFragment::create_dir()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
std::string segment_dir = srs_path_dirname(filepath);
|
|
|
|
if ((err = srs_create_dir_recursively(segment_dir)) != srs_success) {
|
|
return srs_error_wrap(err, "create %s", segment_dir.c_str());
|
|
}
|
|
|
|
srs_info("Create dir %s ok", segment_dir.c_str());
|
|
|
|
return err;
|
|
}
|
|
|
|
string SrsFragment::tmppath()
|
|
{
|
|
return filepath + ".tmp";
|
|
}
|
|
|
|
srs_error_t SrsFragment::unlink_tmpfile()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
string filepath = tmppath();
|
|
if (::unlink(filepath.c_str()) < 0) {
|
|
return srs_error_new(ERROR_SYSTEM_FRAGMENT_UNLINK, "unlink tmp file %s", filepath.c_str());
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
srs_error_t SrsFragment::rename()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
string full_path = fullpath();
|
|
string tmp_file = tmppath();
|
|
int tempdur = srsu2msi(duration());
|
|
if (true) {
|
|
std::stringstream ss;
|
|
ss << tempdur;
|
|
full_path = srs_string_replace(full_path, "[duration]", ss.str());
|
|
}
|
|
|
|
int r0 = ::rename(tmp_file.c_str(), full_path.c_str());
|
|
if (r0 < 0) {
|
|
return srs_error_new(ERROR_SYSTEM_FRAGMENT_RENAME, "rename %s to %s", tmp_file.c_str(), full_path.c_str());
|
|
}
|
|
|
|
filepath = full_path;
|
|
return err;
|
|
}
|
|
|
|
SrsFragmentWindow::SrsFragmentWindow()
|
|
{
|
|
}
|
|
|
|
SrsFragmentWindow::~SrsFragmentWindow()
|
|
{
|
|
vector<SrsFragment*>::iterator it;
|
|
|
|
for (it = fragments.begin(); it != fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
srs_freep(fragment);
|
|
}
|
|
fragments.clear();
|
|
|
|
for (it = expired_fragments.begin(); it != expired_fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
srs_freep(fragment);
|
|
}
|
|
expired_fragments.clear();
|
|
}
|
|
|
|
void SrsFragmentWindow::dispose()
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
std::vector<SrsFragment*>::iterator it;
|
|
|
|
for (it = fragments.begin(); it != fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
if ((err = fragment->unlink_file()) != srs_success) {
|
|
srs_warn("Unlink ts failed %s", srs_error_desc(err).c_str());
|
|
srs_freep(err);
|
|
}
|
|
srs_freep(fragment);
|
|
}
|
|
fragments.clear();
|
|
|
|
for (it = expired_fragments.begin(); it != expired_fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
if ((err = fragment->unlink_file()) != srs_success) {
|
|
srs_warn("Unlink ts failed %s", srs_error_desc(err).c_str());
|
|
srs_freep(err);
|
|
}
|
|
srs_freep(fragment);
|
|
}
|
|
expired_fragments.clear();
|
|
}
|
|
|
|
void SrsFragmentWindow::append(SrsFragment* fragment)
|
|
{
|
|
fragments.push_back(fragment);
|
|
}
|
|
|
|
void SrsFragmentWindow::shrink(srs_utime_t window)
|
|
{
|
|
srs_utime_t duration = 0;
|
|
|
|
int remove_index = -1;
|
|
|
|
for (int i = (int)fragments.size() - 1; i >= 0; i--) {
|
|
SrsFragment* fragment = fragments[i];
|
|
duration += fragment->duration();
|
|
|
|
if (duration > window) {
|
|
remove_index = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < remove_index && !fragments.empty(); i++) {
|
|
SrsFragment* fragment = *fragments.begin();
|
|
fragments.erase(fragments.begin());
|
|
expired_fragments.push_back(fragment);
|
|
}
|
|
}
|
|
|
|
void SrsFragmentWindow::clear_expired(bool delete_files)
|
|
{
|
|
srs_error_t err = srs_success;
|
|
|
|
std::vector<SrsFragment*>::iterator it;
|
|
|
|
for (it = expired_fragments.begin(); it != expired_fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
if (delete_files && (err = fragment->unlink_file()) != srs_success) {
|
|
srs_warn("Unlink ts failed, %s", srs_error_desc(err).c_str());
|
|
srs_freep(err);
|
|
}
|
|
srs_freep(fragment);
|
|
}
|
|
|
|
expired_fragments.clear();
|
|
}
|
|
|
|
srs_utime_t SrsFragmentWindow::max_duration()
|
|
{
|
|
srs_utime_t v = 0;
|
|
|
|
std::vector<SrsFragment*>::iterator it;
|
|
|
|
for (it = fragments.begin(); it != fragments.end(); ++it) {
|
|
SrsFragment* fragment = *it;
|
|
v = srs_max(v, fragment->duration());
|
|
}
|
|
|
|
return v;
|
|
}
|
|
|
|
bool SrsFragmentWindow::empty()
|
|
{
|
|
return fragments.empty();
|
|
}
|
|
|
|
SrsFragment* SrsFragmentWindow::first()
|
|
{
|
|
return fragments.at(0);
|
|
}
|
|
|
|
int SrsFragmentWindow::size()
|
|
{
|
|
return (int)fragments.size();
|
|
}
|
|
|
|
SrsFragment* SrsFragmentWindow::at(int index)
|
|
{
|
|
return fragments.at(index);
|
|
}
|
|
|