mirror of https://github.com/ossrs/srs.git
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
8.9 KiB
C++
249 lines
8.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"
|
|
|
|
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(CSeqNo::m_iMaxSeqNo), 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
|
|
};
|
|
|
|
};
|
|
|
|
struct RcvGroup: Group
|
|
{
|
|
bool fec;
|
|
bool dismissed;
|
|
RcvGroup(): fec(false), dismissed(false) {}
|
|
|
|
#if ENABLE_HEAVY_LOGGING
|
|
std::string DisplayStats()
|
|
{
|
|
if (base == CSeqNo::m_iMaxSeqNo)
|
|
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);
|
|
int ExtendRows(int rowx);
|
|
int ExtendColumns(int colgx);
|
|
void MarkCellReceived(int32_t seq);
|
|
bool HangHorizontal(const CPacket& pkt, bool fec_ctl, loss_seqs_t& irrecover);
|
|
bool 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);
|
|
|
|
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);
|
|
int RcvGetColumnGroupIndex(int32_t seq);
|
|
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; }
|
|
};
|
|
|
|
#endif
|