mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-24 17:40:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			410 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			410 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2021 Mobilinkd LLC.
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <QDebug>
 | |
| #include <QString>
 | |
| 
 | |
| #include "M17Randomizer.h"
 | |
| #include "PolynomialInterleaver.h"
 | |
| #include "Trellis.h"
 | |
| #include "Viterbi.h"
 | |
| #include "CRC16.h"
 | |
| #include "LinkSetupFrame.h"
 | |
| #include "Golay24.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <array>
 | |
| #include <cstddef>
 | |
| #include <functional>
 | |
| #include <iostream>
 | |
| #include <iomanip>
 | |
| 
 | |
| namespace modemm17
 | |
| {
 | |
| 
 | |
| 
 | |
| template <typename C, size_t N>
 | |
| QString dump(const std::array<C,N>& data, char header = 'D')
 | |
| {
 | |
|     QString s(header);
 | |
|     s += "=";
 | |
| 
 | |
|     for (auto c : data) {
 | |
|         s += QString("%1 ").arg((int) c, 2, 16, QChar('0'));
 | |
|     }
 | |
| 
 | |
|     return s;
 | |
| }
 | |
| 
 | |
| struct M17FrameDecoder
 | |
| {
 | |
|     static const size_t MAX_LICH_FRAGMENT = 5;
 | |
| 
 | |
|     M17Randomizer derandomize_;
 | |
|     PolynomialInterleaver<45, 92, 368> interleaver_;
 | |
|     Trellis<4,2> trellis_{makeTrellis<4, 2>({031,027})};
 | |
|     Viterbi<decltype(trellis_), 4> viterbi_{trellis_};
 | |
|     CRC16 crc_;
 | |
| 
 | |
|     enum class State { LSF, STREAM, BASIC_PACKET, FULL_PACKET, BERT };
 | |
|     enum class SyncWordType { LSF, STREAM, PACKET, BERT };
 | |
|     enum class DecodeResult { FAIL, OK, EOS, INCOMPLETE, PACKET_INCOMPLETE };
 | |
|     enum class FrameType { LSF, LICH, STREAM, BASIC_PACKET, FULL_PACKET, BERT };
 | |
| 
 | |
|     State state_ = State::LSF;
 | |
| 
 | |
|     using input_buffer_t = std::array<int8_t, 368>;
 | |
| 
 | |
|     using lsf_conv_buffer_t = std::array<uint8_t, 46>;
 | |
|     using audio_conv_buffer_t = std::array<uint8_t, 34>;
 | |
| 
 | |
|     using lsf_buffer_t = std::array<uint8_t, 30>;
 | |
|     using lich_buffer_t = std::array<uint8_t, 6>;
 | |
|     using audio_buffer_t = std::array<uint8_t, 18>;
 | |
|     using packet_buffer_t = std::array<uint8_t, 26>;
 | |
|     using bert_buffer_t = std::array<uint8_t, 25>;
 | |
| 
 | |
|     using output_buffer_t = struct {
 | |
|         FrameType type;
 | |
|         union {
 | |
|             lich_buffer_t lich;
 | |
|             audio_buffer_t stream;
 | |
|             packet_buffer_t packet;
 | |
|             bert_buffer_t bert;
 | |
|         };
 | |
|         lsf_buffer_t lsf;
 | |
|     };
 | |
| 
 | |
|     using depunctured_buffer_t = union {
 | |
|         std::array<int8_t, 488> lsf;
 | |
|         std::array<int8_t, 296> stream;
 | |
|         std::array<int8_t, 420> packet;
 | |
|         std::array<int8_t, 402> bert;
 | |
|     };
 | |
| 
 | |
|     using decode_buffer_t = union {
 | |
|         std::array<uint8_t, 240> lsf;
 | |
|         std::array<uint8_t, 144> stream;
 | |
|         std::array<uint8_t, 206> packet;
 | |
|         std::array<uint8_t, 197> bert;
 | |
|     };
 | |
| 
 | |
|     /**
 | |
|      * Callback function for frame types.  The caller is expected to return
 | |
|      * true if the data was good or unknown and false if the data is known
 | |
|      * to be bad.
 | |
|      */
 | |
|     using callback_t = std::function<bool(const output_buffer_t&, int)>;
 | |
| 
 | |
|     callback_t callback_;
 | |
| 
 | |
|     output_buffer_t output_buffer;
 | |
|     decode_buffer_t decode_buffer;
 | |
|     uint16_t frame_number = 0;
 | |
| 
 | |
|     uint8_t lich_segments{0};       ///< one bit per received LICH fragment.
 | |
| 
 | |
|     M17FrameDecoder(callback_t callback) :
 | |
|         crc_(0x5935, 0xFFFF),
 | |
|         callback_(callback)
 | |
|     {}
 | |
| 
 | |
|     void update_state(std::array<uint8_t, 240>& lsf_output)
 | |
|     {
 | |
|         if (lsf_output[111]) // LSF type bit 0
 | |
|         {
 | |
|             if (lsf_output[109] != 0) {
 | |
|                 state_ = State::STREAM;
 | |
|             }
 | |
|         }
 | |
|         else    // packet frame comes next.
 | |
|         {
 | |
|             uint8_t packet_type = (lsf_output[109] << 1) | lsf_output[110];
 | |
|             switch (packet_type)
 | |
|             {
 | |
|             case 1: // RAW -- ignore LSF.
 | |
|                 state_ = State::BASIC_PACKET;
 | |
|                 break;
 | |
|             case 2: // ENCAPSULATED
 | |
|                 state_ = State::FULL_PACKET;
 | |
|                 break;
 | |
|             default:
 | |
|                 state_ = State::FULL_PACKET;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void reset()
 | |
|     {
 | |
|         state_ = State::LSF;
 | |
|         frame_number = 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Decode the LSF and, if it is valid, transition to the next state.
 | |
|      *
 | |
|      * The LSF is returned for STREAM mode, dropped for BASIC_PACKET mode,
 | |
|      * and captured for FULL_PACKET mode.
 | |
|      *
 | |
|      * @param buffer
 | |
|      * @param viterbi_cost
 | |
|      * @return
 | |
|      */
 | |
|     DecodeResult decode_lsf(input_buffer_t& buffer, int& viterbi_cost)
 | |
|     {
 | |
|         depunctured_buffer_t depuncture_buffer;
 | |
|         depuncture<368, 488, 61>(buffer, depuncture_buffer.lsf, P1);
 | |
|         viterbi_cost = viterbi_.decode(depuncture_buffer.lsf, decode_buffer.lsf);
 | |
|         to_byte_array(decode_buffer.lsf, output_buffer.lsf);
 | |
| 
 | |
|         // qDebug() << "modemm17::M17FrameDecoder::decode_lsf: vierbi:" << viterbi_cost <<dump(output_buffer.lsf);
 | |
| 
 | |
|         crc_.reset();
 | |
|         for (auto c : output_buffer.lsf) crc_(c);
 | |
|         auto checksum = crc_.get();
 | |
| 
 | |
|         if (checksum == 0)
 | |
|         {
 | |
|             update_state(decode_buffer.lsf);
 | |
|             output_buffer.type = FrameType::LSF;
 | |
|             callback_(output_buffer, viterbi_cost);
 | |
|             return DecodeResult::OK;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             qDebug() << "modemm17::M17FrameDecoder::decode_lsf: bad CRC:" << dump(output_buffer.lsf);
 | |
|         }
 | |
| 
 | |
|         lich_segments = 0;
 | |
|         output_buffer.lsf.fill(0);
 | |
|         return DecodeResult::FAIL;
 | |
|     }
 | |
| 
 | |
|     // Unpack  & decode LICH fragments into tmp_buffer.
 | |
|     bool unpack_lich(input_buffer_t& buffer)
 | |
|     {
 | |
|         size_t index = 0;
 | |
|         // Read the 4 24-bit codewords from LICH
 | |
|         for (size_t i = 0; i != 4; ++i) // for each codeword
 | |
|         {
 | |
|             uint32_t codeword = 0;
 | |
|             for (size_t j = 0; j != 24; ++j) // for each bit in codeword
 | |
|             {
 | |
|                 codeword <<= 1;
 | |
|                 codeword |= (buffer[i * 24 + j] > 0);
 | |
|             }
 | |
|             uint32_t decoded = 0;
 | |
|             if (!Golay24::decode(codeword, decoded))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
|             decoded >>= 12; // Remove check bits and parity.
 | |
|             // append codeword.
 | |
|             if (i & 1)
 | |
|             {
 | |
|                 output_buffer.lich[index++] |= (decoded >> 8);     // upper 4 bits
 | |
|                 output_buffer.lich[index++] = (decoded & 0xFF);    // lower 8 bits
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 output_buffer.lich[index++] |= (decoded >> 4);     // upper 8 bits
 | |
|                 output_buffer.lich[index] = (decoded & 0x0F) << 4; // lower 4 bits
 | |
|             }
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     DecodeResult decode_lich(input_buffer_t& buffer, int& viterbi_cost)
 | |
|     {
 | |
|         output_buffer.lich.fill(0);
 | |
|         // Read the 4 12-bit codewords from LICH into buffers.lich.
 | |
|         if (!unpack_lich(buffer)) return DecodeResult::FAIL;
 | |
| 
 | |
|         output_buffer.type = FrameType::LICH;
 | |
|         callback_(output_buffer, 0);
 | |
| 
 | |
|         uint8_t fragment_number = output_buffer.lich[5];   // Get fragment number.
 | |
|         fragment_number = (fragment_number >> 5) & 7;
 | |
| 
 | |
|         if (fragment_number > MAX_LICH_FRAGMENT)
 | |
|         {
 | |
|             viterbi_cost = -1;
 | |
|             return DecodeResult::INCOMPLETE;    // More to go...
 | |
|         }
 | |
| 
 | |
|         // Copy decoded LICH to superframe buffer.
 | |
|         std::copy(output_buffer.lich.begin(), output_buffer.lich.begin() + 5,
 | |
|             output_buffer.lsf.begin() + (fragment_number * 5));
 | |
| 
 | |
|         lich_segments |= (1 << fragment_number);        // Indicate segment received.
 | |
|         if ((lich_segments & 0x3F) != 0x3F)
 | |
|         {
 | |
|             viterbi_cost = -1;
 | |
|             return DecodeResult::INCOMPLETE;        // More to go...
 | |
|         }
 | |
| 
 | |
|         crc_.reset();
 | |
|         for (auto c : output_buffer.lsf) crc_(c);
 | |
|         auto checksum = crc_.get();
 | |
| 
 | |
|         if (checksum == 0)
 | |
|         {
 | |
|         	lich_segments = 0;
 | |
|             state_ = State::STREAM;
 | |
|             viterbi_cost = 0;
 | |
|             output_buffer.type = FrameType::LSF;
 | |
|             callback_(output_buffer, viterbi_cost);
 | |
|             return DecodeResult::OK;
 | |
|         }
 | |
| 
 | |
|         // Failed CRC... try again.
 | |
|         // lich_segments = 0;
 | |
|         // output_buffer.lsf.fill(0);
 | |
|         viterbi_cost = 128;
 | |
|         return DecodeResult::INCOMPLETE;
 | |
|     }
 | |
| 
 | |
|     DecodeResult decode_bert(input_buffer_t& buffer, int& viterbi_cost)
 | |
|     {
 | |
|         depunctured_buffer_t depuncture_buffer;
 | |
|         depuncture<368, 402, 12>(buffer, depuncture_buffer.bert, P2);
 | |
|         viterbi_cost = viterbi_.decode(depuncture_buffer.bert, decode_buffer.bert);
 | |
|         to_byte_array(decode_buffer.bert, output_buffer.bert);
 | |
| 
 | |
|         output_buffer.type = FrameType::BERT;
 | |
|         callback_(output_buffer, viterbi_cost);
 | |
| 
 | |
|         return DecodeResult::OK;
 | |
|     }
 | |
| 
 | |
|     DecodeResult decode_stream(input_buffer_t& buffer, int& viterbi_cost)
 | |
|     {
 | |
|         std::array<int8_t, 272> tmp;
 | |
|         std::copy(buffer.begin() + 96, buffer.end(), tmp.begin());
 | |
|         depunctured_buffer_t depuncture_buffer;
 | |
| 
 | |
|         depuncture<272, 296, 12>(tmp, depuncture_buffer.stream, P2);
 | |
|         viterbi_cost = viterbi_.decode(depuncture_buffer.stream, decode_buffer.stream);
 | |
|         to_byte_array(decode_buffer.stream, output_buffer.stream);
 | |
| 
 | |
|         if ((viterbi_cost < 60) && (output_buffer.stream[0] & 0x80))
 | |
|         {
 | |
|             // fputs("\nEOS\n", stderr);
 | |
|             state_ = State::LSF;
 | |
|         }
 | |
| 
 | |
|         output_buffer.type = FrameType::STREAM;
 | |
|         callback_(output_buffer, viterbi_cost);
 | |
| 
 | |
|         return state_ == State::LSF ? DecodeResult::EOS : DecodeResult::OK;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Capture packet frames until an EOF bit is found.
 | |
| 
 | |
|      * @param buffer the demodulated M17 symbols in LLR format.
 | |
|      * @param viterbi_cost the cost of traversing the trellis.
 | |
|      * @param frame_type is either BASIC_PACKET or FULL_PACKET.
 | |
|      * @return the result of decoding the packet frame.
 | |
|      */
 | |
|     DecodeResult decode_packet(input_buffer_t& buffer, int& viterbi_cost, FrameType type)
 | |
|     {
 | |
|         depunctured_buffer_t depuncture_buffer;
 | |
|         depuncture<368, 420, 8>(buffer, depuncture_buffer.packet, P3);
 | |
|         viterbi_cost = viterbi_.decode(depuncture_buffer.packet, decode_buffer.packet);
 | |
|         to_byte_array(decode_buffer.packet, output_buffer.packet);
 | |
| 
 | |
|         output_buffer.type = type;
 | |
|         auto result = callback_(output_buffer, viterbi_cost);
 | |
| 
 | |
|         if (output_buffer.packet[25] & 0x80) // last packet;
 | |
|         {
 | |
|             state_ = State::LSF;
 | |
|             return result ? DecodeResult::OK : DecodeResult::FAIL;
 | |
|         }
 | |
| 
 | |
|         return DecodeResult::PACKET_INCOMPLETE;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Decode M17 frames.  The decoder uses the sync word to determine frame
 | |
|      * type and to update its state machine.
 | |
|      *
 | |
|      * The decoder receives M17 frame type indicator (based on sync word) and
 | |
|      * frames from the M17 demodulator.
 | |
|      *
 | |
|      * If the frame is an LSF, the state immediately changes to LSF. When
 | |
|      * in LSF mode, the state machine can transition to:
 | |
|      *
 | |
|      *  - LSF if the CRC is bad.
 | |
|      *  - STREAM if the LSF type field indicates Stream.
 | |
|      *  - BASIC_PACKET if the LSF type field indicates Packet and the packet
 | |
|      *    type is RAW.
 | |
|      *  - FULL_PACKET if the LSF type field indicates Packet and the packet
 | |
|      *    type is ENCAPSULATED or RESERVED.
 | |
|      *
 | |
|      * When in LSF mode, if an LSF frame is received it is parsed as an LSF.
 | |
|      * When a STREAM frame is received, it attempts to recover an LSF from
 | |
|      * the LICH.  PACKET frame types are ignored when state is LSF.
 | |
|      *
 | |
|      * When in STREAM mode, the state machine can transition to either:
 | |
|      *
 | |
|      *  - STREAM when a any stream frame is received.
 | |
|      *  - LSF when the EOS indicator is set, or when a packet frame is received.
 | |
|      *
 | |
|      * When in BASIC_PACKET mode, the state machine can transition to either:
 | |
|      *
 | |
|      *  - BASIC_PACKET when any packet frame is received.
 | |
|      *  - LSF when the EOS indicator is set, or when a stream frame is received.
 | |
|      *
 | |
|      * When in FULL_PACKET mode, the state machine can transition to either:
 | |
|      *
 | |
|      *  - FULL_PACKET when any packet frame is received.
 | |
|      *  - LSF when the EOS indicator is set, or when a stream frame is received.
 | |
|      */
 | |
|     DecodeResult operator()(SyncWordType frame_type, input_buffer_t& buffer, int& viterbi_cost)
 | |
|     {
 | |
|         derandomize_(buffer);
 | |
|         interleaver_.deinterleave(buffer);
 | |
| 
 | |
|         // This is out state machined.
 | |
|         switch(frame_type)
 | |
|         {
 | |
|         case SyncWordType::LSF:
 | |
|             state_ = State::LSF;
 | |
|             return decode_lsf(buffer, viterbi_cost);
 | |
|         case SyncWordType::STREAM:
 | |
|             switch (state_)
 | |
|             {
 | |
|             case State::LSF:
 | |
|                 return decode_lich(buffer, viterbi_cost);
 | |
|             case State::STREAM:
 | |
|                 return decode_stream(buffer, viterbi_cost);
 | |
|             default:
 | |
|                 state_ = State::LSF;
 | |
|             }
 | |
|             break;
 | |
|         case SyncWordType::PACKET:
 | |
|             switch (state_)
 | |
|             {
 | |
|             case State::BASIC_PACKET:
 | |
|                 return decode_packet(buffer, viterbi_cost, FrameType::BASIC_PACKET);
 | |
|             case State::FULL_PACKET:
 | |
|                 return decode_packet(buffer, viterbi_cost, FrameType::FULL_PACKET);
 | |
|             default:
 | |
|                 state_ = State::LSF;
 | |
|             }
 | |
|             break;
 | |
|         case SyncWordType::BERT:
 | |
|             state_ = State::BERT;
 | |
|             return decode_bert(buffer, viterbi_cost);
 | |
|         }
 | |
| 
 | |
|         return DecodeResult::FAIL;
 | |
|     }
 | |
| 
 | |
|     State state() const { return state_; }
 | |
| };
 | |
| 
 | |
| } // modemm17
 |