| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  | #include <QDebug>
 | 
					
						
							|  |  |  | #include <QString>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | #include "FirFilter.h"
 | 
					
						
							|  |  |  | #include "LinkSetupFrame.h"
 | 
					
						
							|  |  |  | #include "CRC16.h"
 | 
					
						
							|  |  |  | #include "Convolution.h"
 | 
					
						
							|  |  |  | #include "PolynomialInterleaver.h"
 | 
					
						
							|  |  |  | #include "M17Randomizer.h"
 | 
					
						
							|  |  |  | #include "Util.h"
 | 
					
						
							|  |  |  | #include "Golay24.h"
 | 
					
						
							|  |  |  | #include "Trellis.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							|  |  |  | #include <atomic>
 | 
					
						
							|  |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <cstdint>
 | 
					
						
							|  |  |  | #include <future>
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-31 05:15:58 +02:00
										 |  |  | #include "export.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  | namespace modemm17 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Common routines extracted from the original asynchronous M17 modulator. | 
					
						
							|  |  |  |  * It is used to produce the various symbol sequences but modulation is handled at | 
					
						
							|  |  |  |  * upper level. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2022-07-31 05:15:58 +02:00
										 |  |  | struct MODEMM17_API M17Modulator | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using symbols_t = std::array<int8_t, 192>;      // One frame of symbols.
 | 
					
						
							|  |  |  |     using baseband_t = std::array<int16_t, 1920>;   // One frame of baseband data @ 48ksps
 | 
					
						
							|  |  |  |     using bitstream_t = std::array<uint8_t, 48>;    // M17 frame of bits (in bytes).
 | 
					
						
							|  |  |  |     using lsf_t = std::array<uint8_t, 30>;          // Link setup frame bytes.
 | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |     using lich_segment_t = std::array<uint8_t, 96>; // Golay-encoded LICH bits.
 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |     using lich_t = std::array<lich_segment_t, 6>;   // All LICH segments.
 | 
					
						
							|  |  |  |     using audio_frame_t = std::array<int16_t, 320>; | 
					
						
							|  |  |  |     using codec_frame_t = std::array<uint8_t, 16>; | 
					
						
							|  |  |  |     using payload_t = std::array<uint8_t, 34>;      // Bytes in the payload of a data frame.
 | 
					
						
							|  |  |  |     using frame_t = std::array<uint8_t, 46>;        // M17 frame (without sync word).
 | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |     using packet_t = std::array<uint8_t, 25>;       // Packet payload
 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     static const std::array<uint8_t, 2> SYNC_WORD; | 
					
						
							|  |  |  |     static const std::array<uint8_t, 2> LSF_SYNC_WORD; | 
					
						
							|  |  |  |     static const std::array<uint8_t, 2> STREAM_SYNC_WORD; | 
					
						
							|  |  |  |     static const std::array<uint8_t, 2> PACKET_SYNC_WORD; | 
					
						
							|  |  |  |     static const std::array<uint8_t, 2> BERT_SYNC_WORD; | 
					
						
							|  |  |  |     static const std::array<uint8_t, 2> EOT_SYNC; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-27 18:15:47 +02:00
										 |  |  |     static int8_t bits_to_symbol(uint8_t bits) | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         switch (bits) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         case 0: return 1; | 
					
						
							|  |  |  |         case 1: return 3; | 
					
						
							|  |  |  |         case 2: return -1; | 
					
						
							|  |  |  |         case 3: return -3; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     template <typename T, size_t N> | 
					
						
							|  |  |  |     static std::array<int8_t, N / 2> bits_to_symbols(const std::array<T, N>& bits) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::array<int8_t, N / 2> result; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         for (size_t i = 0; i != N; i += 2) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             result[index++] = bits_to_symbol((bits[i] << 1) | bits[i + 1]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |     template <typename T, size_t N> | 
					
						
							|  |  |  |     static std::array<int8_t, N * 4> bytes_to_symbols(const std::array<T, N>& bytes) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::array<int8_t, N * 4> result; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         for (auto b : bytes) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 result[index++] = bits_to_symbol(b >> 6); | 
					
						
							|  |  |  |                 b <<= 2; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     /*
 | 
					
						
							|  |  |  |     * Converts a suite of 192 symbols (from the 384 bits of a frame) into 1920 16 bit integer samples to be used | 
					
						
							|  |  |  |     * in the final FM modulator (baseband). Sample rate is expected to be 48 kS/s. This is the original 48 kS/s | 
					
						
							|  |  |  |     * 16 bit audio output of the modulator. | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     template <size_t N> | 
					
						
							|  |  |  |     std::array<int16_t, N*10> symbols_to_baseband(std::array<int8_t, N> symbols, bool invert = false) | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<int16_t, N*10> baseband; | 
					
						
							|  |  |  |         baseband.fill(0); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         for (size_t i = 0; i != symbols.size(); ++i) { | 
					
						
							|  |  |  |             baseband[i * 10] = symbols[i]; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         for (auto& b : baseband) { | 
					
						
							|  |  |  |             b = rrc(b) * 7168.0 * (invert ? -1.0 : 1.0); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         return baseband; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |     std::array<int8_t, 368> make_lsf(lsf_t& lsf, bool streamElsePacket = false) | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         lsf.fill(0); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-31 04:54:39 +02:00
										 |  |  |         M17Randomizer randomizer; | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         PolynomialInterleaver<45, 92, 368> interleaver; | 
					
						
							| 
									
										
										
										
											2022-07-27 18:15:47 +02:00
										 |  |  |         CRC16 crc(0x5935, 0xFFFF); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |         auto rit = std::copy(dest_.begin(), dest_.end(), lsf.begin()); | 
					
						
							|  |  |  |         std::copy(source_.begin(), source_.end(), rit); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |         lsf[12] = can_ >> 1; | 
					
						
							|  |  |  |         lsf[13] = (streamElsePacket ? 5 : 4) | ((can_ & 1) << 7); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |         if (gnss_on_) | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |             lsf[13] |= (1<<5); | 
					
						
							|  |  |  |             std::copy(gnss_.begin(), gnss_.end(), &lsf[14]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             lsf[13] |= (3<<5); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         crc.reset(); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 28; ++i) { | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |             crc(lsf[i]); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<uint8_t, 2> checksum = crc.get_bytes(); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         lsf[28] = checksum[0]; | 
					
						
							|  |  |  |         lsf[29] = checksum[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<uint8_t, 488> encoded; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         uint32_t memory = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto b : lsf) | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |             for (size_t i = 0; i != 8; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |                 b <<= 1; | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |                 memory = modemm17::update_memory<4>(memory, x); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         // Flush the encoder.
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |             memory = modemm17::update_memory<4>(memory, 0); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<int8_t, 368> punctured; | 
					
						
							| 
									
										
										
										
											2022-08-02 23:42:50 +02:00
										 |  |  |         auto size = puncture<488, 368, 61>(encoded, punctured, P1); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         if (size != 368) { | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |             qWarning() << "modemm17::M17Modulator::make_lsf: incorrect size (not 368)" << size; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         interleaver.interleave(punctured); | 
					
						
							|  |  |  |         randomizer.randomize(punctured); | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return punctured; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |     static lich_segment_t make_lich_segment(std::array<uint8_t, 5> segment, uint8_t segment_number) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         lich_segment_t result; | 
					
						
							|  |  |  |         uint16_t tmp; | 
					
						
							|  |  |  |         uint32_t encoded; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tmp = segment[0]; | 
					
						
							|  |  |  |         tmp <<= 4; | 
					
						
							|  |  |  |         tmp |= ((segment[1] >> 4) & 0x0F); | 
					
						
							|  |  |  |         // tmp = segment[0] << 4 | ((segment[1] >> 4) & 0x0F);
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |         encoded = modemm17::Golay24::encode24(tmp); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 24; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             result[i] = (encoded & (1 << 23)) != 0 ? 1 : 0; | 
					
						
							|  |  |  |             encoded <<= 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tmp = segment[1] & 0x0F; | 
					
						
							|  |  |  |         tmp <<= 8; | 
					
						
							|  |  |  |         tmp |= segment[2]; | 
					
						
							|  |  |  |         // tmp = ((segment[1] & 0x0F) << 8) | segment[2];
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |         encoded = modemm17::Golay24::encode24(tmp); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 24; i != 48; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             result[i] = (encoded & (1 << 23)) != 0 ? 1 : 0; | 
					
						
							|  |  |  |             encoded <<= 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tmp = segment[3]; | 
					
						
							|  |  |  |         tmp <<= 4; | 
					
						
							|  |  |  |         tmp |= ((segment[4] >> 4) & 0x0F); | 
					
						
							|  |  |  |         // tmp = segment[3] << 4 | ((segment[4] >> 4) & 0x0F);
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |         encoded = modemm17::Golay24::encode24(tmp); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 48; i != 72; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             result[i] = (encoded & (1 << 23)) != 0 ? 1 : 0; | 
					
						
							|  |  |  |             encoded <<= 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tmp = segment[4] & 0x0F; | 
					
						
							|  |  |  |         tmp <<= 8; | 
					
						
							|  |  |  |         tmp |= (segment_number << 5); | 
					
						
							|  |  |  |         // tmp = ((segment[4] & 0x0F) << 8) | (segment_number << 5);
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |         encoded = modemm17::Golay24::encode24(tmp); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 72; i != 96; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             result[i] = (encoded & (1 << 23)) != 0 ? 1 : 0; | 
					
						
							|  |  |  |             encoded <<= 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static std::array<int8_t, 272> make_stream_data_frame(uint16_t frame_number, const codec_frame_t& payload) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::array<uint8_t, 18> data;   // FN, payload = 2 + 16;
 | 
					
						
							|  |  |  |         data[0] = uint8_t((frame_number >> 8) & 0xFF); | 
					
						
							|  |  |  |         data[1] = uint8_t(frame_number & 0xFF); | 
					
						
							|  |  |  |         std::copy(payload.begin(), payload.end(), data.begin() + 2); | 
					
						
							|  |  |  |         std::array<uint8_t, 296> encoded; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         uint32_t memory = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto b : data) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             for (size_t i = 0; i != 8; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |                 b <<= 1; | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |                 memory = modemm17::update_memory<4>(memory, x); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Flush the encoder.
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |             memory = modemm17::update_memory<4>(memory, 0); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::array<int8_t, 272> punctured; | 
					
						
							| 
									
										
										
										
											2022-08-02 23:42:50 +02:00
										 |  |  |         auto size = modemm17::puncture<296, 272, 12>(encoded, punctured, modemm17::P2); | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (size != 272) { | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |             qWarning() << "modemm17::M17Modulator::make_stream_data_frame: incorrect size (not 272)" << size; | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return punctured; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     std::array<int8_t, 368> make_packet_frame( | 
					
						
							|  |  |  |         uint8_t packet_number, | 
					
						
							|  |  |  |         int packet_size, | 
					
						
							|  |  |  |         bool last_packet, | 
					
						
							|  |  |  |         const std::array<uint8_t, 25> packet | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-07-31 04:54:39 +02:00
										 |  |  |         M17Randomizer randomizer; | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         PolynomialInterleaver<45, 92, 368> interleaver; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |         std::array<uint8_t, 26> packet_assembly; | 
					
						
							|  |  |  |         packet_assembly.fill(0); | 
					
						
							|  |  |  |         std::copy(packet.begin(), packet.begin() + packet_size, packet_assembly.begin()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (packet_number == 0) { | 
					
						
							|  |  |  |             crc_.reset(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int i = 0; i < packet_size; i++) { | 
					
						
							|  |  |  |             crc_(packet[i]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (last_packet) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |             packet_assembly[25] = 0x80 | ((packet_size+2)<<2); // sent packet size includes CRC
 | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |             packet_assembly[packet_size]   = crc_.get_bytes()[1]; | 
					
						
							|  |  |  |             packet_assembly[packet_size+1] = crc_.get_bytes()[0]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             packet_assembly[25] = (packet_number<<2); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<uint8_t, 420> encoded; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         uint32_t memory = 0; | 
					
						
							|  |  |  |         uint8_t b; | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         for (int bi = 0; bi < 25; bi++) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             b = packet_assembly[bi]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (size_t i = 0; i != 8; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |                 b <<= 1; | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |                 memory = modemm17::update_memory<4>(memory, x); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |                 encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-06-12 22:51:18 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         b = packet_assembly[25]; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         for (size_t i = 0; i != 6; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |             b <<= 1; | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |             memory = modemm17::update_memory<4>(memory, x); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         // Flush the encoder.
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |             memory = modemm17::update_memory<4>(memory, 0); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = modemm17::convolve_bit(027, memory); | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         std::array<int8_t, 368> punctured; | 
					
						
							| 
									
										
										
										
											2022-08-02 23:42:50 +02:00
										 |  |  |         auto size = puncture<420, 368, 8>(encoded, punctured, P3); | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (size != 368) { | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |             qWarning() << "modemm17::M17Modulator::make_packet_frame: incorrect size (not 368)" << size; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         interleaver.interleave(punctured); | 
					
						
							|  |  |  |         randomizer.randomize(punctured); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return punctured; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |     static std::array<int8_t, 368> make_bert_frame(PRBS9& prbs) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::array<uint8_t, 25> data;   // 24.6125 bytes, 197 bits
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |         // Generate the data (24*8 = 192 bits).
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         for (size_t i = 0; i != data.size() - 1; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint8_t byte = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (int i = 0; i != 8; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 byte <<= 1; | 
					
						
							|  |  |  |                 byte |= prbs.generate(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             data[i] = byte; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |         // Generate the data (last 5 bits).
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         uint8_t byte = 0; | 
					
						
							|  |  |  |         for (int i = 0; i != 5; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             byte <<= 1; | 
					
						
							|  |  |  |             byte |= prbs.generate(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         byte <<= 3; | 
					
						
							|  |  |  |         data[24] = byte; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |         // Convolutional encode
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         std::array<uint8_t, 402> encoded; | 
					
						
							|  |  |  |         size_t index = 0; | 
					
						
							|  |  |  |         uint32_t memory = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |         // 24*8 = 192 first bits
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         for (size_t i = 0; i != data.size() - 1; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             auto b = data[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (size_t j = 0; j != 8; ++j) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |                 b <<= 1; | 
					
						
							|  |  |  |                 memory = update_memory<4>(memory, x); | 
					
						
							|  |  |  |                 encoded[index++] = convolve_bit(031, memory); | 
					
						
							|  |  |  |                 encoded[index++] = convolve_bit(027, memory); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |         // last 5 bits
 | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         auto b = data[24]; | 
					
						
							|  |  |  |         for (size_t j = 0; j != 5; ++j) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |             b <<= 1; | 
					
						
							|  |  |  |             memory = update_memory<4>(memory, x); | 
					
						
							|  |  |  |             encoded[index++] = convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = convolve_bit(027, memory); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Flush the encoder.
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             memory = update_memory<4>(memory, 0); | 
					
						
							|  |  |  |             encoded[index++] = convolve_bit(031, memory); | 
					
						
							|  |  |  |             encoded[index++] = convolve_bit(027, memory); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::array<int8_t, 368> punctured; | 
					
						
							| 
									
										
										
										
											2022-08-02 23:42:50 +02:00
										 |  |  |         auto size = puncture<402, 368, 12>(encoded, punctured, P2); | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (size != 368) { | 
					
						
							| 
									
										
										
										
											2022-07-06 07:01:42 +02:00
										 |  |  |             qWarning() << "modemm17::M17Modulator::make_bert_frame: incorrect size (not 368)" << size; | 
					
						
							| 
									
										
										
										
											2022-07-04 22:45:16 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return punctured; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |     static void interleave_and_randomize(std::array<int8_t, 368>& punctured) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-07-31 04:54:39 +02:00
										 |  |  |         M17Randomizer randomizer; | 
					
						
							| 
									
										
										
										
											2022-06-29 08:45:44 +02:00
										 |  |  |         PolynomialInterleaver<45, 92, 368> interleaver; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         interleaver.interleave(punctured); | 
					
						
							|  |  |  |         randomizer.randomize(punctured); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |     M17Modulator(const std::string& source, const std::string& dest = "") : | 
					
						
							|  |  |  |         source_(encode_callsign(source)), | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |         dest_(encode_callsign(dest)), | 
					
						
							| 
									
										
										
										
											2022-06-30 00:17:31 +02:00
										 |  |  |         can_(10), | 
					
						
							| 
									
										
										
										
											2022-07-27 18:15:47 +02:00
										 |  |  |         rrc(makeFirFilter(rrc_taps)), | 
					
						
							|  |  |  |         crc_(0x5935, 0xFFFF) | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         gnss_.fill(0); | 
					
						
							|  |  |  |         gnss_on_ = false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Set the source identifier (callsign) for the transmitter. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void source(const std::string& callsign) { | 
					
						
							|  |  |  |         source_ = encode_callsign(callsign); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Set the destination identifier for the transmitter.  A blank value is | 
					
						
							|  |  |  |      * interpreted as the broadcast address.  This is the default. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void dest(const std::string& callsign) { | 
					
						
							|  |  |  |         dest_ = encode_callsign(callsign); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-30 00:17:31 +02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Set the Channel Access Number (0..15) | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void can(uint8_t can) { | 
					
						
							|  |  |  |         can_ = can & 0xF; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |     * Set GNSS data | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     void set_gnss(float lat, float lon, float alt) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         gnss_ = LinkSetupFrame::encode_gnss(lat, lon, alt); | 
					
						
							|  |  |  |         gnss_on_ = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |     * Reset GNSS data | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     void reset_gnss() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         gnss_.fill(0); | 
					
						
							|  |  |  |         gnss_on_ = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | private: | 
					
						
							|  |  |  |     LinkSetupFrame::encoded_call_t source_; | 
					
						
							|  |  |  |     LinkSetupFrame::encoded_call_t dest_; | 
					
						
							| 
									
										
										
										
											2022-07-16 03:48:33 +02:00
										 |  |  |     LinkSetupFrame::gnss_t gnss_; | 
					
						
							|  |  |  |     bool gnss_on_; | 
					
						
							| 
									
										
										
										
											2022-06-30 00:17:31 +02:00
										 |  |  |     uint8_t can_; | 
					
						
							| 
									
										
										
										
											2022-06-17 02:25:34 +02:00
										 |  |  |     BaseFirFilter<150> rrc; | 
					
						
							|  |  |  |     static const std::array<float, 150> rrc_taps; | 
					
						
							| 
									
										
										
										
											2022-07-27 18:15:47 +02:00
										 |  |  |     CRC16 crc_; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static LinkSetupFrame::encoded_call_t encode_callsign(std::string callsign) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         LinkSetupFrame::encoded_call_t encoded_call = {0xff,0xff,0xff,0xff,0xff,0xff}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (callsign.empty() || callsign.size() > 9) { | 
					
						
							|  |  |  |             return encoded_call; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  |         modemm17::LinkSetupFrame::call_t call; | 
					
						
							| 
									
										
										
										
											2022-06-10 19:17:56 +02:00
										 |  |  |         call.fill(0); | 
					
						
							|  |  |  |         std::copy(callsign.begin(), callsign.end(), call.begin()); | 
					
						
							|  |  |  |         encoded_call = LinkSetupFrame::encode_callsign(call); | 
					
						
							|  |  |  |         return encoded_call; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <typename T, size_t N> | 
					
						
							|  |  |  |     static std::array<T, N * 2 + 1> conv_encode(std::array<T, N> data) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::array<T, N * 2 + 1> result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         uint8_t bit_index = 0; | 
					
						
							|  |  |  |         uint8_t byte_index = 0; | 
					
						
							|  |  |  |         uint8_t tmp = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         uint32_t memory = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto b : data) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             for (size_t i = 0; i != 8; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 uint32_t x = (b & 0x80) >> 7; | 
					
						
							|  |  |  |                 b <<= 1; | 
					
						
							|  |  |  |                 memory = update_memory<4>(memory, x); | 
					
						
							|  |  |  |                 tmp = (tmp << 1) | convolve_bit(031, memory); | 
					
						
							|  |  |  |                 tmp = (tmp << 1) | convolve_bit(027, memory); | 
					
						
							|  |  |  |                 bit_index += 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (bit_index == 8) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     bit_index = 0; | 
					
						
							|  |  |  |                     result[byte_index++] = tmp; | 
					
						
							|  |  |  |                     tmp = 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Flush the encoder.
 | 
					
						
							|  |  |  |         for (size_t i = 0; i != 4; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             memory = update_memory<4>(memory, 0); | 
					
						
							|  |  |  |             tmp = (tmp << 1) | convolve_bit(031, memory); | 
					
						
							|  |  |  |             tmp = (tmp << 1) | convolve_bit(027, memory); | 
					
						
							|  |  |  |             bit_index += 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (bit_index == 8) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 bit_index = 0; | 
					
						
							|  |  |  |                 result[byte_index++] = tmp; | 
					
						
							|  |  |  |                 tmp = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Frame may not end on a byte boundary.
 | 
					
						
							|  |  |  |         if (bit_index != 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             while (bit_index++ != 8) { | 
					
						
							|  |  |  |                 tmp <<= 1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             result[byte_index] = tmp; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-04 23:03:07 +02:00
										 |  |  | } // modemm17
 |