Several fixes:
- Fixed bug in stream health function causing loop if track not active - Fixed DTSC pulls ignoring data before the live point - Improved async buffers (deque mode) to spread the tracks more fairly - DTSC pull now implements "ping" and "error" commands - DTSC pulls report suspicious keyframe intervals to the origin and ask for confirmation - DTSC output now accepts these reports and disconnects if there is no match in keyframe intervals - Outputs in async mode now keep the seek point in all tracks when reselecting - Outputs in async mode now default to a starting position in each track that is at a keyframe roughly halfway in the buffer - Outputs in async mode now ignore playback rate (always fastest possible) - Removed code duplication in prepareNext function - Reordered the prepareNext function somewhat to be easier to follow for humans - DTSC output no longer overrides initialSeek function, now uses default implementation - Sanitycheck output now supports both sync and async modes, supports printing multiple timestamps for multiple tracks
This commit is contained in:
		
							parent
							
								
									b89875ea37
								
							
						
					
					
						commit
						f560b88bfe
					
				
					 9 changed files with 257 additions and 222 deletions
				
			
		|  | @ -1384,8 +1384,13 @@ namespace DTSC{ | ||||||
|       setType(newIdx, M.getType(*it)); |       setType(newIdx, M.getType(*it)); | ||||||
|       setCodec(newIdx, M.getCodec(*it)); |       setCodec(newIdx, M.getCodec(*it)); | ||||||
|       setLang(newIdx, M.getLang(*it)); |       setLang(newIdx, M.getLang(*it)); | ||||||
|  |       if (copyData){ | ||||||
|         setFirstms(newIdx, M.getFirstms(*it)); |         setFirstms(newIdx, M.getFirstms(*it)); | ||||||
|         setLastms(newIdx, M.getLastms(*it)); |         setLastms(newIdx, M.getLastms(*it)); | ||||||
|  |       }else{ | ||||||
|  |         setFirstms(newIdx, 0); | ||||||
|  |         setLastms(newIdx, 0); | ||||||
|  |       } | ||||||
|       setBps(newIdx, M.getBps(*it)); |       setBps(newIdx, M.getBps(*it)); | ||||||
|       setMaxBps(newIdx, M.getMaxBps(*it)); |       setMaxBps(newIdx, M.getMaxBps(*it)); | ||||||
|       setFpks(newIdx, M.getFpks(*it)); |       setFpks(newIdx, M.getFpks(*it)); | ||||||
|  | @ -3241,8 +3246,8 @@ namespace DTSC{ | ||||||
|       uint32_t longest_cnt = 0; |       uint32_t longest_cnt = 0; | ||||||
|       DTSC::Keys Mkeys(keys(i)); |       DTSC::Keys Mkeys(keys(i)); | ||||||
|       uint32_t firstKey = Mkeys.getFirstValid(); |       uint32_t firstKey = Mkeys.getFirstValid(); | ||||||
|       uint32_t endKey = Mkeys.getEndValid() - 1; |       uint32_t endKey = Mkeys.getEndValid(); | ||||||
|       for (int k = firstKey; k < endKey; k++){ |       for (uint32_t k = firstKey; k+1 < endKey; k++){ | ||||||
|         uint64_t kDur = Mkeys.getDuration(k); |         uint64_t kDur = Mkeys.getDuration(k); | ||||||
|         uint64_t kParts = Mkeys.getParts(k); |         uint64_t kParts = Mkeys.getParts(k); | ||||||
|         if (!kDur){continue;} |         if (!kDur){continue;} | ||||||
|  |  | ||||||
|  | @ -280,12 +280,10 @@ void Util::packetSorter::dropTrack(size_t tid){ | ||||||
| /// Removes the first packet from the sorter and inserts the given packet.
 | /// Removes the first packet from the sorter and inserts the given packet.
 | ||||||
| void Util::packetSorter::replaceFirst(const sortedPageInfo &pInfo){ | void Util::packetSorter::replaceFirst(const sortedPageInfo &pInfo){ | ||||||
|   if (dequeMode){ |   if (dequeMode){ | ||||||
|  |     //in deque mode, insertion of the new packet is at the back
 | ||||||
|  |     //this works, as a failure to retrieve a packet will swap the front entry to the back as well
 | ||||||
|     dequeBuffer.pop_front(); |     dequeBuffer.pop_front(); | ||||||
|     if (dequeBuffer.size() && dequeBuffer.front().time > pInfo.time){ |  | ||||||
|       dequeBuffer.push_front(pInfo); |  | ||||||
|     }else{ |  | ||||||
|     dequeBuffer.push_back(pInfo); |     dequeBuffer.push_back(pInfo); | ||||||
|     } |  | ||||||
|   }else{ |   }else{ | ||||||
|     setBuffer.erase(setBuffer.begin()); |     setBuffer.erase(setBuffer.begin()); | ||||||
|     setBuffer.insert(pInfo); |     setBuffer.insert(pInfo); | ||||||
|  | @ -328,6 +326,20 @@ void Util::packetSorter::getTrackList(std::set<size_t> &toFill) const{ | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Fills toFill with track IDs and current playback position of tracks that are in the sorter.
 | ||||||
|  | void Util::packetSorter::getTrackList(std::map<size_t, uint64_t> &toFill) const{ | ||||||
|  |   toFill.clear(); | ||||||
|  |   if (dequeMode){ | ||||||
|  |     for (std::deque<Util::sortedPageInfo>::const_iterator it = dequeBuffer.begin(); it != dequeBuffer.end(); ++it){ | ||||||
|  |       toFill[it->tid] = it->time; | ||||||
|  |     } | ||||||
|  |   }else{ | ||||||
|  |     for (std::set<Util::sortedPageInfo>::const_iterator it = setBuffer.begin(); it != setBuffer.end(); ++it){ | ||||||
|  |       toFill[it->tid] = it->time; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| JSON::Value Util::getStreamConfig(const std::string &streamname){ | JSON::Value Util::getStreamConfig(const std::string &streamname){ | ||||||
|   JSON::Value result; |   JSON::Value result; | ||||||
|   if (streamname.size() > 100){ |   if (streamname.size() > 100){ | ||||||
|  |  | ||||||
|  | @ -75,6 +75,7 @@ namespace Util{ | ||||||
|       void moveFirstToEnd(); |       void moveFirstToEnd(); | ||||||
|       bool hasEntry(size_t tid) const; |       bool hasEntry(size_t tid) const; | ||||||
|       void getTrackList(std::set<size_t> &toFill) const; |       void getTrackList(std::set<size_t> &toFill) const; | ||||||
|  |       void getTrackList(std::map<size_t, uint64_t> &toFill) const; | ||||||
|       void setSyncMode(bool synced); |       void setSyncMode(bool synced); | ||||||
|       bool getSyncMode() const; |       bool getSyncMode() const; | ||||||
|     private: |     private: | ||||||
|  |  | ||||||
|  | @ -165,7 +165,7 @@ namespace Mist{ | ||||||
|       DTSC::Packet metaPack(dataPacket.data(), dataPacket.size()); |       DTSC::Packet metaPack(dataPacket.data(), dataPacket.size()); | ||||||
|       DTSC::Meta nM("", metaPack.getScan()); |       DTSC::Meta nM("", metaPack.getScan()); | ||||||
|       meta.reInit(streamName, false); |       meta.reInit(streamName, false); | ||||||
|       meta.merge(nM); |       meta.merge(nM, true, false); | ||||||
|       std::set<size_t> validTracks = M.getMySourceTracks(getpid()); |       std::set<size_t> validTracks = M.getMySourceTracks(getpid()); | ||||||
|       userSelect.clear(); |       userSelect.clear(); | ||||||
|       for (std::set<size_t>::iterator it = validTracks.begin(); it != validTracks.end(); ++it){ |       for (std::set<size_t>::iterator it = validTracks.begin(); it != validTracks.end(); ++it){ | ||||||
|  | @ -342,10 +342,7 @@ namespace Mist{ | ||||||
|         // userClient.keepAlive();
 |         // userClient.keepAlive();
 | ||||||
|         std::string cmd; |         std::string cmd; | ||||||
|         thisPacket.getString("cmd", cmd); |         thisPacket.getString("cmd", cmd); | ||||||
|         if (cmd != "reset"){ |         if (cmd == "reset"){ | ||||||
|           thisPacket.reInit(srcConn); |  | ||||||
|           continue; |  | ||||||
|         } |  | ||||||
|           // Read next packet
 |           // Read next packet
 | ||||||
|           thisPacket.reInit(srcConn); |           thisPacket.reInit(srcConn); | ||||||
|           if (thisPacket.getVersion() != DTSC::DTSC_HEAD){ |           if (thisPacket.getVersion() != DTSC::DTSC_HEAD){ | ||||||
|  | @ -357,6 +354,28 @@ namespace Mist{ | ||||||
|           thisPacket.reInit(srcConn); // read the next packet before continuing
 |           thisPacket.reInit(srcConn); // read the next packet before continuing
 | ||||||
|           continue;                   // parse the next packet before returning
 |           continue;                   // parse the next packet before returning
 | ||||||
|         } |         } | ||||||
|  |         if (cmd == "error"){ | ||||||
|  |           thisPacket.getString("msg", cmd); | ||||||
|  |           Util::logExitReason("%s", cmd.c_str()); | ||||||
|  |           thisPacket.null(); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         if (cmd == "ping"){ | ||||||
|  |           thisPacket.reInit(srcConn); | ||||||
|  |           JSON::Value prep; | ||||||
|  |           prep["cmd"] = "ok"; | ||||||
|  |           prep["msg"] = "Pong!"; | ||||||
|  |           srcConn.SendNow("DTCM"); | ||||||
|  |           char sSize[4] ={0, 0, 0, 0}; | ||||||
|  |           Bit::htobl(sSize, prep.packedSize()); | ||||||
|  |           srcConn.SendNow(sSize, 4); | ||||||
|  |           prep.sendTo(srcConn); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         INFO_MSG("Unhandled command: %s", cmd.c_str()); | ||||||
|  |         thisPacket.reInit(srcConn); | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|       if (thisPacket.getVersion() == DTSC::DTSC_HEAD){ |       if (thisPacket.getVersion() == DTSC::DTSC_HEAD){ | ||||||
|         DTSC::Meta nM("", thisPacket.getScan()); |         DTSC::Meta nM("", thisPacket.getScan()); | ||||||
|         meta.merge(nM, false, false); |         meta.merge(nM, false, false); | ||||||
|  | @ -364,7 +383,33 @@ namespace Mist{ | ||||||
|         continue;                   // parse the next packet before returning
 |         continue;                   // parse the next packet before returning
 | ||||||
|       } |       } | ||||||
|       thisTime = thisPacket.getTime(); |       thisTime = thisPacket.getTime(); | ||||||
|       thisIdx = thisPacket.getTrackId(); |       thisIdx = M.trackIDToIndex(thisPacket.getTrackId()); | ||||||
|  |       if (thisPacket.getFlag("keyframe") && M.trackValid(thisIdx)){ | ||||||
|  |         uint32_t shrtest_key = 0xFFFFFFFFul; | ||||||
|  |         uint32_t longest_key = 0; | ||||||
|  |         DTSC::Keys Mkeys(M.keys(thisIdx)); | ||||||
|  |         uint32_t firstKey = Mkeys.getFirstValid(); | ||||||
|  |         uint32_t endKey = Mkeys.getEndValid(); | ||||||
|  |         uint32_t checkKey = (endKey-firstKey <= 3)?firstKey:endKey-3; | ||||||
|  |         for (uint32_t k = firstKey; k+1 < endKey; k++){ | ||||||
|  |           uint64_t kDur = Mkeys.getDuration(k); | ||||||
|  |           if (!kDur){continue;} | ||||||
|  |           if (kDur > longest_key && k >= checkKey){longest_key = kDur;} | ||||||
|  |           if (kDur < shrtest_key){shrtest_key = kDur;} | ||||||
|  |         } | ||||||
|  |         if (longest_key > shrtest_key*2){ | ||||||
|  |           JSON::Value prep; | ||||||
|  |           prep["cmd"] = "check_key_duration"; | ||||||
|  |           prep["id"] = thisPacket.getTrackId(); | ||||||
|  |           prep["duration"] = longest_key; | ||||||
|  |           srcConn.SendNow("DTCM"); | ||||||
|  |           char sSize[4] ={0, 0, 0, 0}; | ||||||
|  |           Bit::htobl(sSize, prep.packedSize()); | ||||||
|  |           srcConn.SendNow(sSize, 4); | ||||||
|  |           prep.sendTo(srcConn); | ||||||
|  |           INFO_MSG("Key duration %" PRIu32 " is quite long - confirming with upstream source", longest_key); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|       return; // we have a packet
 |       return; // we have a packet
 | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -463,7 +463,7 @@ namespace Mist{ | ||||||
|     meta.reloadReplacedPagesIfNeeded(); |     meta.reloadReplacedPagesIfNeeded(); | ||||||
| 
 | 
 | ||||||
|     bool autoSeek = buffer.size(); |     bool autoSeek = buffer.size(); | ||||||
|     uint64_t seekTarget = currentTime(); |     uint64_t seekTarget = buffer.getSyncMode()?currentTime():0; | ||||||
|     std::set<size_t> newSelects = |     std::set<size_t> newSelects = | ||||||
|         Util::wouldSelect(M, targetParams, capa, UA, autoSeek ? seekTarget : 0); |         Util::wouldSelect(M, targetParams, capa, UA, autoSeek ? seekTarget : 0); | ||||||
| 
 | 
 | ||||||
|  | @ -482,9 +482,10 @@ namespace Mist{ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::set<size_t> oldSelects; |     std::set<size_t> oldSelects; | ||||||
|     for (std::map<size_t, Comms::Users>::iterator it = userSelect.begin(); it != userSelect.end(); ++it){ |     buffer.getTrackList(oldSelects); | ||||||
|       oldSelects.insert(it->first); |     std::map<size_t, uint64_t> seekTargets; | ||||||
|     } |     buffer.getTrackList(seekTargets); | ||||||
|  | 
 | ||||||
|     //No changes? Abort and return false;
 |     //No changes? Abort and return false;
 | ||||||
|     if (oldSelects == newSelects){return false;} |     if (oldSelects == newSelects){return false;} | ||||||
| 
 | 
 | ||||||
|  | @ -510,6 +511,7 @@ namespace Mist{ | ||||||
|         WARN_MSG("Could not select track %zu, dropping track", *it); |         WARN_MSG("Could not select track %zu, dropping track", *it); | ||||||
|         newSelects.erase(*it); |         newSelects.erase(*it); | ||||||
|         userSelect.erase(*it); |         userSelect.erase(*it); | ||||||
|  |         continue; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -521,10 +523,16 @@ namespace Mist{ | ||||||
|     //After attempting to add/remove tracks, now no changes? Abort and return false;
 |     //After attempting to add/remove tracks, now no changes? Abort and return false;
 | ||||||
|     if (oldSelects == newSelects){return false;} |     if (oldSelects == newSelects){return false;} | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     if (autoSeek){ |     if (autoSeek){ | ||||||
|       INFO_MSG("Automatically seeking to position %" PRIu64 " to resume playback", seekTarget); |       buffer.clear(); | ||||||
|       seek(seekTarget); |       INFO_MSG("Automatically seeking to resume playback"); | ||||||
|  |       for (std::set<size_t>::iterator it = newSelects.begin(); it != newSelects.end(); it++){ | ||||||
|  |         if (seekTargets.count(*it)){ | ||||||
|  |           seek(*it, seekTargets[*it], false); | ||||||
|  |         }else{ | ||||||
|  |           seek(*it, 0, false); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|  | @ -793,6 +801,11 @@ namespace Mist{ | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|     DTSC::Keys keys(M.keys(tid)); |     DTSC::Keys keys(M.keys(tid)); | ||||||
|  |     if (M.getLive() && !pos && !buffer.getSyncMode()){ | ||||||
|  |       uint64_t tmpTime = (M.getFirstms(tid) + M.getLastms(tid))/2; | ||||||
|  |       uint32_t tmpKey = M.getKeyNumForTime(tid, tmpTime); | ||||||
|  |       pos = keys.getTime(tmpKey); | ||||||
|  |     } | ||||||
|     uint32_t keyNum = M.getKeyNumForTime(tid, pos); |     uint32_t keyNum = M.getKeyNumForTime(tid, pos); | ||||||
|     if (keyNum == INVALID_KEY_NUM){ |     if (keyNum == INVALID_KEY_NUM){ | ||||||
|       FAIL_MSG("Attempted seek on empty track %zu", tid); |       FAIL_MSG("Attempted seek on empty track %zu", tid); | ||||||
|  | @ -827,8 +840,7 @@ namespace Mist{ | ||||||
|       tmp.time = tmpPack.getTime(); |       tmp.time = tmpPack.getTime(); | ||||||
|     } |     } | ||||||
|     if (tmpPack){ |     if (tmpPack){ | ||||||
|       HIGH_MSG("Sought to time %" PRIu64 " (yields a packet at %" PRIu64 "ms) in %s@%zu", tmp.time, |       HIGH_MSG("Sought to time %" PRIu64 " in %s", tmp.time, curPage[tid].name.c_str()); | ||||||
|                tmpPack.getTime(), streamName.c_str(), tid); |  | ||||||
|       tmp.partIndex = M.getPartIndex(tmpPack.getTime(), tmp.tid); |       tmp.partIndex = M.getPartIndex(tmpPack.getTime(), tmp.tid); | ||||||
|       buffer.insert(tmp); |       buffer.insert(tmp); | ||||||
|       return true; |       return true; | ||||||
|  | @ -862,7 +874,7 @@ namespace Mist{ | ||||||
|   void Output::initialSeek(){ |   void Output::initialSeek(){ | ||||||
|     if (!meta){return;} |     if (!meta){return;} | ||||||
|     uint64_t seekPos = 0; |     uint64_t seekPos = 0; | ||||||
|     if (meta.getLive()){ |     if (meta.getLive() && buffer.getSyncMode()){ | ||||||
|       size_t mainTrack = getMainSelectedTrack(); |       size_t mainTrack = getMainSelectedTrack(); | ||||||
|       if (mainTrack == INVALID_TRACK_ID){return;} |       if (mainTrack == INVALID_TRACK_ID){return;} | ||||||
|       DTSC::Keys keys(M.keys(mainTrack)); |       DTSC::Keys keys(M.keys(mainTrack)); | ||||||
|  | @ -1208,7 +1220,7 @@ namespace Mist{ | ||||||
|   /// Waits for the given amount of millis, increasing the realtime playback
 |   /// Waits for the given amount of millis, increasing the realtime playback
 | ||||||
|   /// related times as needed to keep smooth playback intact.
 |   /// related times as needed to keep smooth playback intact.
 | ||||||
|   void Output::playbackSleep(uint64_t millis){ |   void Output::playbackSleep(uint64_t millis){ | ||||||
|     if (realTime && M.getLive()){ |     if (realTime && M.getLive() && buffer.getSyncMode()){ | ||||||
|       firstTime += millis; |       firstTime += millis; | ||||||
|       extraKeepAway += millis; |       extraKeepAway += millis; | ||||||
|     } |     } | ||||||
|  | @ -1321,7 +1333,7 @@ namespace Mist{ | ||||||
|             if (firstPacketTime == 0xFFFFFFFFFFFFFFFFull){firstPacketTime = lastPacketTime;} |             if (firstPacketTime == 0xFFFFFFFFFFFFFFFFull){firstPacketTime = lastPacketTime;} | ||||||
| 
 | 
 | ||||||
|             // slow down processing, if real time speed is wanted
 |             // slow down processing, if real time speed is wanted
 | ||||||
|             if (realTime){ |             if (realTime && buffer.getSyncMode()){ | ||||||
|               uint8_t i = 6; |               uint8_t i = 6; | ||||||
|               while (--i && thisPacket.getTime() > (((Util::bootMS() - firstTime) * 1000) / realTime + maxSkipAhead) && |               while (--i && thisPacket.getTime() > (((Util::bootMS() - firstTime) * 1000) / realTime + maxSkipAhead) && | ||||||
|                      keepGoing()){ |                      keepGoing()){ | ||||||
|  | @ -1476,9 +1488,14 @@ namespace Mist{ | ||||||
|     // depending on whether this is probably bad and the current debug level, print a message
 |     // depending on whether this is probably bad and the current debug level, print a message
 | ||||||
|     size_t printLevel = (probablyBad ? DLVL_WARN : DLVL_INFO); |     size_t printLevel = (probablyBad ? DLVL_WARN : DLVL_INFO); | ||||||
|     const Comms::Users &usr = userSelect.at(trackId); |     const Comms::Users &usr = userSelect.at(trackId); | ||||||
|  |     if (!usr){ | ||||||
|  |       DEBUG_MSG(printLevel, "Dropping %s track %zu (lastP=%" PRIu64 "): %s", | ||||||
|  |                 meta.getCodec(trackId).c_str(), trackId, pageNumMax(trackId), reason.c_str()); | ||||||
|  |     }else{ | ||||||
|       DEBUG_MSG(printLevel, "Dropping %s track %zu@k%zu (nextP=%" PRIu64 ", lastP=%" PRIu64 "): %s", |       DEBUG_MSG(printLevel, "Dropping %s track %zu@k%zu (nextP=%" PRIu64 ", lastP=%" PRIu64 "): %s", | ||||||
|                 meta.getCodec(trackId).c_str(), trackId, usr.getKeyNum() + 1, |                 meta.getCodec(trackId).c_str(), trackId, usr.getKeyNum() + 1, | ||||||
|                 pageNumForKey(trackId, usr.getKeyNum() + 1), pageNumMax(trackId), reason.c_str()); |                 pageNumForKey(trackId, usr.getKeyNum() + 1), pageNumMax(trackId), reason.c_str()); | ||||||
|  |     } | ||||||
|     // now actually drop the track from the buffer
 |     // now actually drop the track from the buffer
 | ||||||
|     buffer.dropTrack(trackId); |     buffer.dropTrack(trackId); | ||||||
|     userSelect.erase(trackId); |     userSelect.erase(trackId); | ||||||
|  | @ -1576,42 +1593,12 @@ namespace Mist{ | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Util::sortedPageInfo nxt = *(buffer.begin()); |     Util::sortedPageInfo nxt; | ||||||
| 
 | 
 | ||||||
|     if (meta.reloadReplacedPagesIfNeeded()){return false;} |     uint64_t nextTime; | ||||||
|     if (!M.getValidTracks().count(nxt.tid)){ |     size_t trackTries = 0; | ||||||
|       dropTrack(nxt.tid, "disappeared from metadata"); |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // if we're going to read past the end of the data page, load the next page
 |  | ||||||
|     // this only happens for VoD
 |  | ||||||
|     if (nxt.offset >= curPage[nxt.tid].len || |  | ||||||
|         (!memcmp(curPage[nxt.tid].mapped + nxt.offset, "\000\000\000\000", 4))){ |  | ||||||
|       if (!M.getLive() && nxt.time >= M.getLastms(nxt.tid)){ |  | ||||||
|         dropTrack(nxt.tid, "end of non-live track reached", false); |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       if (M.getPageNumberForTime(nxt.tid, nxt.time) != currentPage[nxt.tid]){ |  | ||||||
|         loadPageForKey(nxt.tid, M.getPageNumberForTime(nxt.tid, nxt.time)); |  | ||||||
|         nxt.offset = 0; |  | ||||||
|         //Only read the next time if the page load succeeded and there is a packet to read from
 |  | ||||||
|         if (curPage[nxt.tid].mapped && curPage[nxt.tid].mapped[0] == 'D'){ |  | ||||||
|           nxt.time = getDTSCTime(curPage[nxt.tid].mapped, 0); |  | ||||||
|         } |  | ||||||
|         buffer.replaceFirst(nxt); |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       dropTrack(nxt.tid, "VoD page load failure"); |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // We know this packet will be valid, pre-load it so we know its length
 |  | ||||||
|     DTSC::Packet preLoad(curPage[nxt.tid].mapped + nxt.offset, 0, true); |  | ||||||
| 
 |  | ||||||
|     uint64_t nextTime = 0; |  | ||||||
|     //In case we're not in sync mode, we might have to retry a few times
 |     //In case we're not in sync mode, we might have to retry a few times
 | ||||||
|     for (size_t trackTries = 0; trackTries < buffer.size(); ++trackTries){ |     for (; trackTries < buffer.size(); ++trackTries){ | ||||||
| 
 | 
 | ||||||
|       nxt = *(buffer.begin()); |       nxt = *(buffer.begin()); | ||||||
| 
 | 
 | ||||||
|  | @ -1639,8 +1626,13 @@ namespace Mist{ | ||||||
|           buffer.replaceFirst(nxt); |           buffer.replaceFirst(nxt); | ||||||
|           return false; |           return false; | ||||||
|         } |         } | ||||||
|         INFO_MSG("Invalid packet: no data @%" PRIu64 " for time %" PRIu64 " on track %zu", nxt.offset, nxt.time, nxt.tid); |         if (nxt.offset >= curPage[nxt.tid].len){ | ||||||
|         dropTrack(nxt.tid, "VoD page load failure"); |           INFO_MSG("Reading past end of page %s: %" PRIu64 " > %" PRIu64 " for time %" PRIu64 " on track %zu", curPage[nxt.tid].name.c_str(), nxt.offset, curPage[nxt.tid].len, nxt.time, nxt.tid); | ||||||
|  |           dropTrack(nxt.tid, "reading past end of page"); | ||||||
|  |         }else{ | ||||||
|  |           INFO_MSG("Invalid packet: no data @%" PRIu64 " in %s for time %" PRIu64 " on track %zu", nxt.offset, curPage[nxt.tid].name.c_str(), nxt.time, nxt.tid); | ||||||
|  |           dropTrack(nxt.tid, "zero packet"); | ||||||
|  |         } | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|       // We know this packet will be valid, pre-load it so we know its length
 |       // We know this packet will be valid, pre-load it so we know its length
 | ||||||
|  | @ -1662,8 +1654,11 @@ namespace Mist{ | ||||||
|           dropTrack(nxt.tid, errMsg.str().c_str()); |           dropTrack(nxt.tid, errMsg.str().c_str()); | ||||||
|           return false; |           return false; | ||||||
|         } |         } | ||||||
|       }else{ |         break;//Packet valid!
 | ||||||
|         //no next packet yet!
 |       } | ||||||
|  | 
 | ||||||
|  |       //no next packet on the current page
 | ||||||
|  | 
 | ||||||
|       //Check if this is the last packet of a VoD stream. Return success and drop the track.
 |       //Check if this is the last packet of a VoD stream. Return success and drop the track.
 | ||||||
|       if (!M.getLive() && nxt.time >= M.getLastms(nxt.tid)){ |       if (!M.getLive() && nxt.time >= M.getLastms(nxt.tid)){ | ||||||
|         thisPacket.reInit(curPage[nxt.tid].mapped + nxt.offset, 0, true); |         thisPacket.reInit(curPage[nxt.tid].mapped + nxt.offset, 0, true); | ||||||
|  | @ -1671,8 +1666,9 @@ namespace Mist{ | ||||||
|         dropTrack(nxt.tid, "end of non-live track reached", false); |         dropTrack(nxt.tid, "end of non-live track reached", false); | ||||||
|         return true; |         return true; | ||||||
|       } |       } | ||||||
|         uint32_t thisKey = M.getKeyNumForTime(nxt.tid, nxt.time); | 
 | ||||||
|       //Check if there exists a different page for the next key
 |       //Check if there exists a different page for the next key
 | ||||||
|  |       uint32_t thisKey = M.getKeyNumForTime(nxt.tid, nxt.time); | ||||||
|       uint32_t nextKeyPage = INVALID_KEY_NUM; |       uint32_t nextKeyPage = INVALID_KEY_NUM; | ||||||
|       //Make sure we only try to read the page for the next key if it actually should be available
 |       //Make sure we only try to read the page for the next key if it actually should be available
 | ||||||
|       DTSC::Keys keys(M.keys(nxt.tid)); |       DTSC::Keys keys(M.keys(nxt.tid)); | ||||||
|  | @ -1690,15 +1686,19 @@ namespace Mist{ | ||||||
|           dropTrack(nxt.tid, errMsg.str().c_str()); |           dropTrack(nxt.tid, errMsg.str().c_str()); | ||||||
|           return false; |           return false; | ||||||
|         } |         } | ||||||
|         }else{ |         break;//Valid packet!
 | ||||||
|           if (!buffer.getSyncMode() && trackTries < buffer.size()-1){ |       } | ||||||
|             //We shuffle the just-tried packet back to the end of the queue, then retry up to buffer.size() times
 | 
 | ||||||
|  |       //Okay, there's no next page yet, and no next packet on this page either.
 | ||||||
|  |       //That means we're waiting for data to show up, somewhere.
 | ||||||
|  |        | ||||||
|  |       //In non-sync mode, shuffle the just-tried packet to the end of queue and retry
 | ||||||
|  |       if (!buffer.getSyncMode()){ | ||||||
|         buffer.moveFirstToEnd(); |         buffer.moveFirstToEnd(); | ||||||
|         continue; |         continue; | ||||||
|       } |       } | ||||||
|           //Okay, there's no next page yet, and no next packet on this page either.
 | 
 | ||||||
|           //That means we're waiting for data to show up, somewhere.
 |       // in sync mode, after ~25 seconds, give up and drop the track.
 | ||||||
|           // after ~25 seconds, give up and drop the track.
 |  | ||||||
|       if (++emptyCount >= dataWaitTimeout){ |       if (++emptyCount >= dataWaitTimeout){ | ||||||
|         dropTrack(nxt.tid, "EOP: data wait timeout"); |         dropTrack(nxt.tid, "EOP: data wait timeout"); | ||||||
|         return false; |         return false; | ||||||
|  | @ -1726,24 +1726,29 @@ namespace Mist{ | ||||||
|         } |         } | ||||||
|         return false;//no sleep after reconnect
 |         return false;//no sleep after reconnect
 | ||||||
|       } |       } | ||||||
|  |        | ||||||
|       //Fine! We didn't want a packet, anyway. Let's try again later.
 |       //Fine! We didn't want a packet, anyway. Let's try again later.
 | ||||||
|       playbackSleep(10); |       playbackSleep(10); | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|       } | 
 | ||||||
|  |     if (trackTries == buffer.size()){ | ||||||
|  |       //Fine! We didn't want a packet, anyway. Let's try again later.
 | ||||||
|  |       playbackSleep(10); | ||||||
|  |       return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // we've handled all special cases - at this point the packet should exist
 |     // we've handled all special cases - at this point the packet should exist
 | ||||||
|     // let's load it
 |     // let's load it
 | ||||||
|     thisPacket.reInit(curPage[nxt.tid].mapped + nxt.offset, 0, true); |     thisPacket.reInit(curPage[nxt.tid].mapped + nxt.offset, 0, true); | ||||||
|     thisIdx = nxt.tid; |  | ||||||
|     thisTime = thisPacket.getTime(); |  | ||||||
|     // if it failed, drop the track and continue
 |     // if it failed, drop the track and continue
 | ||||||
|     if (!thisPacket){ |     if (!thisPacket){ | ||||||
|       dropTrack(nxt.tid, "packet load failure"); |       dropTrack(nxt.tid, "packet load failure"); | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|     emptyCount = 0; // valid packet - reset empty counter
 |     emptyCount = 0; // valid packet - reset empty counter
 | ||||||
|  |     thisIdx = nxt.tid; | ||||||
|  |     thisTime = thisPacket.getTime(); | ||||||
| 
 | 
 | ||||||
|     if (!userSelect[nxt.tid]){ |     if (!userSelect[nxt.tid]){ | ||||||
|       dropTrack(nxt.tid, "track is not alive!"); |       dropTrack(nxt.tid, "track is not alive!"); | ||||||
|  |  | ||||||
|  | @ -115,47 +115,6 @@ namespace Mist{ | ||||||
| 
 | 
 | ||||||
|   std::string OutDTSC::getStatsName(){return (pushing ? "INPUT:DTSC" : "OUTPUT:DTSC");} |   std::string OutDTSC::getStatsName(){return (pushing ? "INPUT:DTSC" : "OUTPUT:DTSC");} | ||||||
| 
 | 
 | ||||||
|   /// Seeks to the first sync'ed keyframe of the main track.
 |  | ||||||
|   /// Aborts if there is no main track or it has no keyframes.
 |  | ||||||
|   void OutDTSC::initialSeek(){ |  | ||||||
|     uint64_t seekPos = 0; |  | ||||||
|     if (M.getLive()){ |  | ||||||
|       size_t mainTrack = getMainSelectedTrack(); |  | ||||||
|       // cancel if there are no keys in the main track
 |  | ||||||
|       if (mainTrack == INVALID_TRACK_ID){return;} |  | ||||||
| 
 |  | ||||||
|       DTSC::Keys keys(M.keys(mainTrack)); |  | ||||||
|       if (!keys.getValidCount()){return;} |  | ||||||
|       // seek to the oldest keyframe
 |  | ||||||
|       std::set<size_t> validTracks = M.getValidTracks(); |  | ||||||
|       for (size_t i = keys.getFirstValid(); i < keys.getEndValid(); ++i){ |  | ||||||
|         seekPos = keys.getTime(i); |  | ||||||
|         bool good = true; |  | ||||||
|         // check if all tracks have data for this point in time
 |  | ||||||
|         for (std::map<size_t, Comms::Users>::iterator ti = userSelect.begin(); ti != userSelect.end(); ++ti){ |  | ||||||
|           if (mainTrack == ti->first){continue;}// skip self
 |  | ||||||
|           if (!validTracks.count(ti->first)){ |  | ||||||
|             HIGH_MSG("Skipping track %zu, not in tracks", ti->first); |  | ||||||
|             continue; |  | ||||||
|           }// ignore missing tracks
 |  | ||||||
|           if (M.getLastms(ti->first) == M.getFirstms(ti->first)){ |  | ||||||
|             HIGH_MSG("Skipping track %zu, last equals first", ti->first); |  | ||||||
|             continue; |  | ||||||
|           }// ignore point-tracks
 |  | ||||||
|           if (M.getFirstms(ti->first) > seekPos){ |  | ||||||
|             good = false; |  | ||||||
|             break; |  | ||||||
|           } |  | ||||||
|           HIGH_MSG("Track %zu is good", ti->first); |  | ||||||
|         } |  | ||||||
|         // if yes, seek here
 |  | ||||||
|         if (good){break;} |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     MEDIUM_MSG("Initial seek to %" PRIu64 "ms", seekPos); |  | ||||||
|     seek(seekPos); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   void OutDTSC::sendNext(){ |   void OutDTSC::sendNext(){ | ||||||
|     // If selectable tracks changed, set sentHeader to false to force it to send init data
 |     // If selectable tracks changed, set sentHeader to false to force it to send init data
 | ||||||
|     static uint64_t lastMeta = 0; |     static uint64_t lastMeta = 0; | ||||||
|  | @ -201,6 +160,10 @@ namespace Mist{ | ||||||
|         std::string dataPacket = myConn.Received().remove(rSize); |         std::string dataPacket = myConn.Received().remove(rSize); | ||||||
|         DTSC::Scan dScan((char *)dataPacket.data(), rSize); |         DTSC::Scan dScan((char *)dataPacket.data(), rSize); | ||||||
|         HIGH_MSG("Received DTCM: %s", dScan.asJSON().toString().c_str()); |         HIGH_MSG("Received DTCM: %s", dScan.asJSON().toString().c_str()); | ||||||
|  |         if (dScan.getMember("cmd").asString() == "ok"){ | ||||||
|  |           INFO_MSG("Remote OK: %s", dScan.getMember("msg").asString().c_str()); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|         if (dScan.getMember("cmd").asString() == "push"){ |         if (dScan.getMember("cmd").asString() == "push"){ | ||||||
|           handlePush(dScan); |           handlePush(dScan); | ||||||
|           continue; |           continue; | ||||||
|  | @ -230,6 +193,29 @@ namespace Mist{ | ||||||
|           sendOk("Internal state reset"); |           sendOk("Internal state reset"); | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|  |         if (dScan.getMember("cmd").asString() == "check_key_duration"){ | ||||||
|  |           size_t idx = dScan.getMember("id").asInt() - 1; | ||||||
|  |           size_t dur = dScan.getMember("duration").asInt(); | ||||||
|  |           if (!M.trackValid(idx)){ | ||||||
|  |             ERROR_MSG("Cannot check key duration %zu for track %zu: not valid", dur, idx); | ||||||
|  |             return; | ||||||
|  |           } | ||||||
|  |           uint32_t longest_key = 0; | ||||||
|  |           DTSC::Keys Mkeys(M.keys(idx)); | ||||||
|  |           uint32_t firstKey = Mkeys.getFirstValid(); | ||||||
|  |           uint32_t endKey = Mkeys.getEndValid(); | ||||||
|  |           for (uint32_t k = firstKey; k+1 < endKey; k++){ | ||||||
|  |             uint64_t kDur = Mkeys.getDuration(k); | ||||||
|  |             if (kDur > longest_key){longest_key = kDur;} | ||||||
|  |           } | ||||||
|  |           if (dur > longest_key*1.2){ | ||||||
|  |             onFail("Key duration mismatch; disconnecting "+myConn.getHost()+" to recover ("+JSON::Value(longest_key).asString()+" -> "+JSON::Value(dur).asString()+")", true); | ||||||
|  |             return; | ||||||
|  |           }else{ | ||||||
|  |             sendOk("Key duration matches upstream"); | ||||||
|  |           } | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|         WARN_MSG("Unhandled DTCM command: '%s'", dScan.getMember("cmd").asString().c_str()); |         WARN_MSG("Unhandled DTCM command: '%s'", dScan.getMember("cmd").asString().c_str()); | ||||||
|       }else if (myConn.Received().copy(4) == "DTSC"){ |       }else if (myConn.Received().copy(4) == "DTSC"){ | ||||||
|         // Header packet
 |         // Header packet
 | ||||||
|  |  | ||||||
|  | @ -11,7 +11,6 @@ namespace Mist{ | ||||||
|     void onRequest(); |     void onRequest(); | ||||||
|     void sendNext(); |     void sendNext(); | ||||||
|     void sendHeader(); |     void sendHeader(); | ||||||
|     void initialSeek(); |  | ||||||
|     static bool listenMode(){return !(config->getString("target").size());} |     static bool listenMode(){return !(config->getString("target").size());} | ||||||
|     void onFail(const std::string &msg, bool critical = false); |     void onFail(const std::string &msg, bool critical = false); | ||||||
|     void stats(bool force = false); |     void stats(bool force = false); | ||||||
|  |  | ||||||
|  | @ -14,6 +14,11 @@ namespace Mist{ | ||||||
|     //}
 |     //}
 | ||||||
|     parseData = true; |     parseData = true; | ||||||
|     wantRequest = false; |     wantRequest = false; | ||||||
|  |     if (config->getBool("sync")){ | ||||||
|  |       setSyncMode(true); | ||||||
|  |     }else{ | ||||||
|  |       setSyncMode(false); | ||||||
|  |     } | ||||||
|     initialize(); |     initialize(); | ||||||
|     initialSeek(); |     initialSeek(); | ||||||
|     sortSet.clear(); |     sortSet.clear(); | ||||||
|  | @ -50,19 +55,7 @@ namespace Mist{ | ||||||
|         seek(seekPoint); |         seek(seekPoint); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   void OutSanityCheck::initialSeek(){ |  | ||||||
|     if (M.getLive()){ |  | ||||||
|       liveSeek(); |  | ||||||
|       if (getKeyFrame() && thisPacket){ |  | ||||||
|         sendNext(); |  | ||||||
|         INFO_MSG("Initial sent!"); |  | ||||||
|       } |  | ||||||
|       firstTime = Util::getMS() - currentTime(); |  | ||||||
|     }else{ |  | ||||||
|       Output::initialSeek(); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   void OutSanityCheck::init(Util::Config *cfg){ |   void OutSanityCheck::init(Util::Config *cfg){ | ||||||
|  | @ -74,8 +67,11 @@ namespace Mist{ | ||||||
|                                                   "\"stream\",\"help\":\"The name of the stream " |                                                   "\"stream\",\"help\":\"The name of the stream " | ||||||
|                                                   "that this connector will transmit.\"}")); |                                                   "that this connector will transmit.\"}")); | ||||||
|     cfg->addOption( |     cfg->addOption( | ||||||
|         "seek", JSON::fromString("{\"arg\":\"string\",\"short\":\"S\",\"long\":\"seek\",\"help\":" |         "seek", JSON::fromString("{\"arg\":\"string\",\"short\":\"k\",\"long\":\"seek\",\"help\":" | ||||||
|                                  "\"Time in ms to check from - by default start of stream\"}")); |                                  "\"Time in ms to check from - by default start of stream\"}")); | ||||||
|  |     cfg->addOption( | ||||||
|  |         "sync", JSON::fromString("{\"short\":\"y\",\"long\":\"sync\",\"help\":" | ||||||
|  |                                  "\"Retrieve tracks in sync (default async)\"}")); | ||||||
|     cfg->addBasicConnectorOptions(capa); |     cfg->addBasicConnectorOptions(capa); | ||||||
|     config = cfg; |     config = cfg; | ||||||
|   } |   } | ||||||
|  | @ -89,36 +85,23 @@ namespace Mist{ | ||||||
|   } |   } | ||||||
|   */ |   */ | ||||||
| 
 | 
 | ||||||
|  | #define printTime(t) std::setfill('0') << std::setw(2) << (t / 3600000) << ":" << std::setw(2) << ((t % 3600000) / 60000) << ":" << std::setw(2) << ((t % 60000) / 1000) << "." << std::setw(3) << (t % 1000) | ||||||
|  | 
 | ||||||
|   void OutSanityCheck::sendNext(){ |   void OutSanityCheck::sendNext(){ | ||||||
|  |     static std::map<size_t, uint64_t> trkTime; | ||||||
|     if (M.getLive()){ |     if (M.getLive()){ | ||||||
|       static uint64_t prevTime = 0; |       if (thisTime < trkTime[thisIdx]){ | ||||||
|       static size_t prevTrack = 0; |         std::cout << "Time error in track " << thisIdx << ": "; | ||||||
|       uint64_t t = thisPacket.getTime(); |         std::cout << printTime(thisTime) << " < " << printTime(trkTime[thisIdx]) << std::endl << std::endl; | ||||||
|       if (t < prevTime){ |  | ||||||
|         std::cout << "Time error: "; |  | ||||||
|         std::cout << std::setfill('0') << std::setw(2) << (t / 3600000) << ":" << std::setw(2) |  | ||||||
|                   << ((t % 3600000) / 60000) << ":" << std::setw(2) << ((t % 60000) / 1000) << "." |  | ||||||
|                   << std::setw(3) << (t % 1000); |  | ||||||
|         std::cout << " (" << thisIdx << ")"; |  | ||||||
|         std::cout << " < "; |  | ||||||
|         std::cout << std::setfill('0') << std::setw(2) << (prevTime / 3600000) << ":" |  | ||||||
|                   << std::setw(2) << ((prevTime % 3600000) / 60000) << ":" << std::setw(2) |  | ||||||
|                   << ((prevTime % 60000) / 1000) << "." << std::setw(3) << (prevTime % 1000); |  | ||||||
|         std::cout << " (" << prevTrack << ")"; |  | ||||||
|         std::cout << std::endl << std::endl; |  | ||||||
|       }else{ |       }else{ | ||||||
|         prevTime = t; |         trkTime[thisIdx] = thisTime; | ||||||
|         prevTrack = thisIdx; |  | ||||||
|       } |       } | ||||||
|       std::cout << "\033[A" << std::setfill('0') << std::setw(2) << (t / 3600000) << ":" |       std::cout << "\033[A"; | ||||||
|                 << std::setw(2) << ((t % 3600000) / 60000) << ":" << std::setw(2) |       for (std::map<size_t, uint64_t>::iterator it = trkTime.begin(); it != trkTime.end(); ++it){ | ||||||
|                 << ((t % 60000) / 1000) << "." << std::setw(3) << (t % 1000) << "   "; |         uint64_t t = M.getLastms(it->first); | ||||||
|       uint32_t mainTrack = M.mainTrack(); |         std::cout << it->first << ":" << printTime(it->second) << "/" << printTime(t) << ", "; | ||||||
|       if (mainTrack == INVALID_TRACK_ID){return;} |       } | ||||||
|       t = M.getLastms(mainTrack); |       std::cout << std::endl; | ||||||
|       std::cout << std::setfill('0') << std::setw(2) << (t / 3600000) << ":" << std::setw(2) |  | ||||||
|                 << ((t % 3600000) / 60000) << ":" << std::setw(2) << ((t % 60000) / 1000) << "." |  | ||||||
|                 << std::setw(3) << (t % 1000) << "   " << std::endl; |  | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -27,7 +27,6 @@ namespace Mist{ | ||||||
|     OutSanityCheck(Socket::Connection &conn); |     OutSanityCheck(Socket::Connection &conn); | ||||||
|     static void init(Util::Config *cfg); |     static void init(Util::Config *cfg); | ||||||
|     void sendNext(); |     void sendNext(); | ||||||
|     void initialSeek(); |  | ||||||
|     static bool listenMode(){return false;} |     static bool listenMode(){return false;} | ||||||
| 
 | 
 | ||||||
|   protected: |   protected: | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Thulinma
						Thulinma