mirror of
				https://github.com/ossrs/srs.git
				synced 2025-03-09 15:49:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			283 lines
		
	
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			283 lines
		
	
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * SRT - Secure, Reliable, Transport
 | |
|  * Copyright (c) 2019 Haivision Systems Inc.
 | |
|  * 
 | |
|  * This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
 | |
|  * 
 | |
|  */
 | |
| 
 | |
| 
 | |
| #ifndef INC_SRT_FEC_H
 | |
| #define INC_SRT_FEC_H
 | |
| 
 | |
| #include <string>
 | |
| #include <map>
 | |
| #include <vector>
 | |
| #include <deque>
 | |
| 
 | |
| #include "packetfilter_api.h"
 | |
| 
 | |
| namespace srt {
 | |
| 
 | |
| class FECFilterBuiltin: public SrtPacketFilterBase
 | |
| {
 | |
|     SrtFilterConfig cfg;
 | |
|     size_t m_number_cols;
 | |
|     size_t m_number_rows;
 | |
| 
 | |
|     // Configuration
 | |
|     SRT_ARQLevel m_fallback_level;
 | |
|     bool m_cols_only;
 | |
|     bool m_arrangement_staircase;
 | |
| 
 | |
| public:
 | |
| 
 | |
|     size_t numberCols() const { return m_number_cols; }
 | |
|     size_t numberRows() const { return m_number_rows; }
 | |
| 
 | |
|     size_t sizeCol() const { return m_number_rows; }
 | |
|     size_t sizeRow() const { return m_number_cols; }
 | |
| 
 | |
|     struct Group
 | |
|     {
 | |
|         int32_t base;     //< Sequence of the first packet in the group
 | |
|         size_t step;      //< by how many packets the sequence should increase to get the next packet
 | |
|         size_t drop;      //< by how much the sequence should increase to get to the next series
 | |
|         size_t collected; //< how many packets were taken to collect the clip
 | |
| 
 | |
|         Group(): base(SRT_SEQNO_NONE), step(0), drop(0), collected(0)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         uint16_t length_clip;
 | |
|         uint8_t flag_clip;
 | |
|         uint32_t timestamp_clip;
 | |
|         std::vector<char> payload_clip;
 | |
| 
 | |
|         // This is mutable because it's an intermediate buffer for
 | |
|         // the purpose of output.
 | |
|         //mutable vector<char> output_buffer;
 | |
| 
 | |
|         enum Type
 | |
|         {
 | |
|             HORIZ,  // Horizontal, recursive
 | |
|             VERT,    // Vertical, recursive
 | |
| 
 | |
|             // NOTE: HORIZ/VERT are defined as 0/1 so that not-inversion
 | |
|             // can flip between them.
 | |
|             SINGLE  // Horizontal-only with no recursion
 | |
|         };
 | |
| 
 | |
|         static Type FlipType(Type t)
 | |
|         {
 | |
|             SRT_ASSERT(t != SINGLE);
 | |
|             return (t == HORIZ) ? VERT : HORIZ;
 | |
|         }
 | |
| 
 | |
|     };
 | |
| 
 | |
|     struct RcvGroup: Group
 | |
|     {
 | |
|         bool fec;
 | |
|         bool dismissed;
 | |
|         RcvGroup(): fec(false), dismissed(false) {}
 | |
| 
 | |
| #if ENABLE_HEAVY_LOGGING
 | |
|         std::string DisplayStats()
 | |
|         {
 | |
|             if (base == SRT_SEQNO_NONE)
 | |
|                 return "UNINITIALIZED!!!";
 | |
| 
 | |
|             std::ostringstream os;
 | |
|             os << "base=" << base << " step=" << step << " drop=" << drop << " collected=" << collected
 | |
|                 << " " << (fec ? "+" : "-") << "FEC " << (dismissed ? "DISMISSED" : "active");
 | |
|             return os.str();
 | |
|         }
 | |
| #endif
 | |
|     };
 | |
| 
 | |
| private:
 | |
| 
 | |
|     // Row Groups: every item represents a single row group and collects clips for one row.
 | |
|     // Col Groups: every item represents a signel column group and collect clips for packets represented in one column
 | |
| 
 | |
|     struct Send
 | |
|     {
 | |
|         // We need only ONE horizontal group. Simply after the group
 | |
|         // is closed (last packet supplied), and the FEC packet extracted,
 | |
|         // the group is no longer in use.
 | |
|         Group row;
 | |
|         std::vector<Group> cols;
 | |
|     } snd;
 | |
| 
 | |
|     struct Receive
 | |
|     {
 | |
|         SRTSOCKET id;
 | |
|         bool order_required;
 | |
| 
 | |
|         Receive(std::vector<SrtPacket>& provided): id(SRT_INVALID_SOCK), order_required(false), rebuilt(provided)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         // In reception we need to keep as many horizontal groups as required
 | |
|         // for possible later tracking. A horizontal group should be dismissed
 | |
|         // when the size of this container exceeds the `m_number_rows` (size of the column).
 | |
|         //
 | |
|         // The 'std::deque' type is used here for a trial implementation. A desired solution
 | |
|         // would be a kind of a ring buffer where new groups are added and old (exceeding
 | |
|         // the size) automatically dismissed.
 | |
|         std::deque<RcvGroup> rowq;
 | |
| 
 | |
|         // Base index at the oldest column platform determines
 | |
|         // the base index of the queue. Meaning, first you need
 | |
|         // to determnine the column index, where the index 0 is
 | |
|         // the fistmost element of this queue. After determining
 | |
|         // the column index, there must be also a second factor
 | |
|         // deteremined - which column series it is. So, this can
 | |
|         // start by extracting the base sequence of the element
 | |
|         // at the index column. This is the series 0. Now, the
 | |
|         // distance between these two sequences, divided by
 | |
|         // rowsize*colsize should return %index-in-column,
 | |
|         // /number-series. The latter multiplied by the row size
 | |
|         // is the offset between the firstmost column and the
 | |
|         // searched column.
 | |
|         std::deque<RcvGroup> colq;
 | |
| 
 | |
|         // This keeps the value of "packet received or not".
 | |
|         // The sequence number of the first cell is rowq[0].base.
 | |
|         // When dropping a row,
 | |
|         // - the firstmost element of rowq is removed
 | |
|         // - the length of one row is removed from this std::vector
 | |
|         int32_t cell_base;
 | |
|         std::deque<bool> cells;
 | |
| 
 | |
|         // Note this function will automatically extend the container
 | |
|         // with empty cells if the index exceeds the size, HOWEVER
 | |
|         // the caller must make sure that this index isn't any "crazy",
 | |
|         // that is, it fits somehow in reasonable ranges.
 | |
|         bool CellAt(size_t index)
 | |
|         {
 | |
|             if (index >= cells.size())
 | |
|             {
 | |
|                 // Cells not prepared for this sequence yet,
 | |
|                 // so extend in advance.
 | |
|                 cells.resize(index+1, false);
 | |
|                 return false; // It wasn't marked, anyway.
 | |
|             }
 | |
| 
 | |
|             return cells[index];
 | |
|         }
 | |
| 
 | |
|         typedef SrtPacket PrivPacket;
 | |
|         std::vector<PrivPacket>& rebuilt;
 | |
|     } rcv;
 | |
| 
 | |
|     void ConfigureGroup(Group& g, int32_t seqno, size_t gstep, size_t drop);
 | |
|     template <class Container>
 | |
|     void ConfigureColumns(Container& which, int32_t isn);
 | |
| 
 | |
|     void ResetGroup(Group& g);
 | |
| 
 | |
|     // Universal
 | |
|     void ClipData(Group& g, uint16_t length_net, uint8_t kflg,
 | |
|             uint32_t timestamp_hw, const char* payload, size_t payload_size);
 | |
|     void ClipPacket(Group& g, const CPacket& pkt);
 | |
| 
 | |
|     // Sending
 | |
|     bool CheckGroupClose(Group& g, size_t pos, size_t size);
 | |
|     void PackControl(const Group& g, signed char groupix, SrtPacket& pkt, int32_t seqno);
 | |
| 
 | |
|     // Receiving
 | |
|     void CheckLargeDrop(int32_t seqno);
 | |
|     size_t ExtendRows(size_t rowx);
 | |
|     size_t ExtendColumns(size_t colgx);
 | |
| 
 | |
|     enum ECellReceived
 | |
|     {
 | |
|         CELL_RECEIVED, //< mark cell for a received packet (no matter current value)
 | |
|         CELL_EXTEND,  //< just make sure there's a place for a packet, set false if not
 | |
|         CELL_REMOVE  //< even if a packet was marked true, remove the cell existence confirmation
 | |
|     };
 | |
|     void MarkCellReceived(int32_t seq, ECellReceived recv = CELL_RECEIVED);
 | |
| 
 | |
|     enum EHangStatus
 | |
|     {
 | |
|         HANG_NOTDONE,
 | |
|         HANG_SUCCESS,
 | |
|         HANG_PAST,
 | |
|         HANG_CRAZY
 | |
|     };
 | |
| 
 | |
|     friend bool operator <(FECFilterBuiltin::EHangStatus a, FECFilterBuiltin::EHangStatus b)
 | |
|     {
 | |
|         return int(a) < int(b);
 | |
|     }
 | |
| 
 | |
|     EHangStatus HangHorizontal(const CPacket& pkt, bool fec_ctl, loss_seqs_t& irrecover);
 | |
|     EHangStatus HangVertical(const CPacket& pkt, signed char fec_colx, loss_seqs_t& irrecover);
 | |
|     void ClipControlPacket(Group& g, const CPacket& pkt);
 | |
|     void ClipRebuiltPacket(Group& g, Receive::PrivPacket& pkt);
 | |
|     void RcvRebuild(Group& g, int32_t seqno, Group::Type tp);
 | |
|     int32_t RcvGetLossSeqHoriz(Group& g);
 | |
|     int32_t RcvGetLossSeqVert(Group& g);
 | |
|     bool CheckEmergencyShrink(size_t n_series, size_t size_in_packets);
 | |
| 
 | |
|     static void TranslateLossRecords(const std::set<int32_t>& loss, loss_seqs_t& irrecover);
 | |
|     void RcvCheckDismissColumn(int32_t seqno, int colgx, loss_seqs_t& irrecover);
 | |
|     int RcvGetRowGroupIndex(int32_t seq, EHangStatus& w_status);
 | |
|     int RcvGetColumnGroupIndex(int32_t seqno, EHangStatus& w_status);
 | |
|     void CollectIrrecoverRow(RcvGroup& g, loss_seqs_t& irrecover) const;
 | |
|     bool IsLost(int32_t seq) const;
 | |
| 
 | |
| public:
 | |
| 
 | |
|     FECFilterBuiltin(const SrtFilterInitializer& init, std::vector<SrtPacket>& provided, const std::string& confstr);
 | |
| 
 | |
|     // Sender side
 | |
| 
 | |
|     // This function creates and stores the FEC control packet with
 | |
|     // a prediction to be immediately sent. This is called in the function
 | |
|     // that normally is prepared for extracting a data packet from the sender
 | |
|     // buffer and send it over the channel.
 | |
|     virtual bool packControlPacket(SrtPacket& r_packet, int32_t seq) ATR_OVERRIDE;
 | |
| 
 | |
|     // This is called at the moment when the sender queue decided to pick up
 | |
|     // a new packet from the scheduled packets. This should be then used to
 | |
|     // continue filling the group, possibly followed by final calculating the
 | |
|     // FEC control packet ready to send.
 | |
|     virtual void feedSource(CPacket& r_packet) ATR_OVERRIDE;
 | |
| 
 | |
|     // Receiver side
 | |
| 
 | |
|     // This function is called at the moment when a new data packet has
 | |
|     // arrived (no matter if subsequent or recovered). The 'state' value
 | |
|     // defines the configured level of loss state required to send the
 | |
|     // loss report.
 | |
|     virtual bool receive(const CPacket& pkt, loss_seqs_t& loss_seqs) ATR_OVERRIDE;
 | |
| 
 | |
|     // Configuration
 | |
| 
 | |
|     // This is the size that is needed extra by packets operated by this corrector.
 | |
|     // It should be subtracted from a current maximum value for SRTO_PAYLOADSIZE
 | |
| 
 | |
|     // The default FEC uses extra space only for FEC/CTL packet.
 | |
|     // The timestamp clip is placed in the timestamp field in the header.
 | |
|     // The payload contains:
 | |
|     // - the length clip
 | |
|     // - the flag spec
 | |
|     // - the payload clip
 | |
|     // The payload clip takes simply the current length of SRTO_PAYLOADSIZE.
 | |
|     // So extra 4 bytes are needed, 2 for flags, 2 for length clip.
 | |
|     static const size_t EXTRA_SIZE = 4;
 | |
| 
 | |
|     virtual SRT_ARQLevel arqLevel() ATR_OVERRIDE { return m_fallback_level; }
 | |
| 
 | |
|     static const char defaultConfig [];
 | |
|     static bool verifyConfig(const SrtFilterConfig& config, std::string& w_errormsg);
 | |
| };
 | |
| 
 | |
| } // namespace srt
 | |
| 
 | |
| #endif
 |