/////////////////////////////////////////////////////////////////////////////////// // Copyright (C) 2017 Edouard Griffiths, F4EXB // // // // This program is free software; you can redistribute it and/or modify // // it under the terms of the GNU General Public License as published by // // the Free Software Foundation as version 3 of the License, or // // // // This program is distributed in the hope that it will be useful, // // but WITHOUT ANY WARRANTY; without even the implied warranty of // // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // // GNU General Public License V3 for more details. // // // // You should have received a copy of the GNU General Public License // // along with this program. If not, see . // /////////////////////////////////////////////////////////////////////////////////// #ifndef PLUGINS_SAMPLESINK_SDRDAEMONSINK_UDPSINKFEC_H_ #define PLUGINS_SAMPLESINK_SDRDAEMONSINK_UDPSINKFEC_H_ #include #include #include #include #include #include #include "cm256.h" #include "dsp/dsptypes.h" #include "util/CRC64.h" #include "util/messagequeue.h" #include "util/message.h" #include "channel/sdrdaemondatablock.h" #include "UDPSocket.h" class UDPSinkFECWorker; class UDPSinkFEC : public QObject { Q_OBJECT public: static const uint32_t m_udpSize = 512; //!< Size of UDP block in number of bytes static const uint32_t m_nbOriginalBlocks = 128; //!< Number of original blocks in a protected block sequence /** * Construct UDP sink */ UDPSinkFEC(); /** Destroy UDP sink */ ~UDPSinkFEC(); /** * Write IQ samples */ void write(const SampleVector::iterator& begin, uint32_t sampleChunkSize); /** Return the last error, or return an empty string if there is no error. */ std::string error() { std::string ret(m_error); m_error.clear(); return ret; } /** Set sample rate given in S/s */ void setSampleRate(uint32_t sampleRate); void setNbBlocksFEC(uint32_t nbBlocksFEC); void setTxDelay(float txDelayRatio); void setRemoteAddress(const QString& address, uint16_t port); /** Return true if the stream is OK, return false if there is an error. */ operator bool() const { return m_error.empty(); } private: std::string m_error; uint32_t m_sampleRate; //!< sample rate in Hz uint32_t m_nbSamples; //!< total number of samples sent int the last frame QHostAddress m_ownAddress; CRC64 m_crc64; uint8_t* m_bufMeta; uint8_t* m_buf; SDRDaemonMetaDataFEC m_currentMetaFEC; //!< Meta data for current frame uint32_t m_nbBlocksFEC; //!< Variable number of FEC blocks float m_txDelayRatio; //!< Delay in ratio of nominal frame period uint32_t m_txDelay; //!< Delay in microseconds (usleep) between each sending of an UDP datagram SDRDaemonSuperBlock m_txBlocks[4][256]; //!< UDP blocks to send with original data + FEC SDRDaemonSuperBlock m_superBlock; //!< current super block being built int m_txBlockIndex; //!< Current index in blocks to transmit in the Tx row int m_txBlocksIndex; //!< Current index of Tx blocks row uint16_t m_frameCount; //!< transmission frame count int m_sampleIndex; //!< Current sample index in protected block data QThread *m_udpThread; UDPSinkFECWorker *m_udpWorker; }; class UDPSinkFECWorker : public QObject { Q_OBJECT public: class MsgUDPFECEncodeAndSend : public Message { MESSAGE_CLASS_DECLARATION public: SDRDaemonSuperBlock *getTxBlocks() const { return m_txBlockx; } uint32_t getNbBlocsFEC() const { return m_nbBlocksFEC; } uint32_t getTxDelay() const { return m_txDelay; } uint16_t getFrameIndex() const { return m_frameIndex; } static MsgUDPFECEncodeAndSend* create( SDRDaemonSuperBlock *txBlocks, uint32_t nbBlocksFEC, uint32_t txDelay, uint16_t frameIndex) { return new MsgUDPFECEncodeAndSend(txBlocks, nbBlocksFEC, txDelay, frameIndex); } private: SDRDaemonSuperBlock *m_txBlockx; uint32_t m_nbBlocksFEC; uint32_t m_txDelay; uint16_t m_frameIndex; MsgUDPFECEncodeAndSend( SDRDaemonSuperBlock *txBlocks, uint32_t nbBlocksFEC, uint32_t txDelay, uint16_t frameIndex) : m_txBlockx(txBlocks), m_nbBlocksFEC(nbBlocksFEC), m_txDelay(txDelay), m_frameIndex(frameIndex) {} }; class MsgConfigureRemoteAddress : public Message { MESSAGE_CLASS_DECLARATION public: const QString& getAddress() const { return m_address; } uint16_t getPort() const { return m_port; } static MsgConfigureRemoteAddress* create(const QString& address, uint16_t port) { return new MsgConfigureRemoteAddress(address, port); } private: QString m_address; uint16_t m_port; MsgConfigureRemoteAddress(const QString& address, uint16_t port) : m_address(address), m_port(port) {} }; UDPSinkFECWorker(); ~UDPSinkFECWorker(); void pushTxFrame(SDRDaemonSuperBlock *txBlocks, uint32_t nbBlocksFEC, uint32_t txDelay, uint16_t frameIndex); void setRemoteAddress(const QString& address, uint16_t port); void stop(); MessageQueue m_inputMessageQueue; //!< Queue for asynchronous inbound communication signals: void finished(); public slots: void process(); private slots: void handleInputMessages(); private: void encodeAndTransmit(SDRDaemonSuperBlock *txBlockx, uint16_t frameIndex, uint32_t nbBlocksFEC, uint32_t txDelay); volatile bool m_running; CM256 m_cm256; //!< CM256 library object bool m_cm256Valid; //!< true if CM256 library is initialized correctly UDPSocket m_socket; QString m_remoteAddress; uint16_t m_remotePort; }; #endif /* PLUGINS_SAMPLESINK_SDRDAEMONSINK_UDPSINKFEC_H_ */