/** * 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 #include #include #include #include #include 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::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::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::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::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); }