forked from flightaware/dump978
-
Notifications
You must be signed in to change notification settings - Fork 0
/
fec.cc
148 lines (122 loc) · 4.98 KB
/
fec.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
// Copyright (c) 2019, FlightAware LLC.
// All rights reserved.
// Licensed under the 2-clause BSD license; see the LICENSE file
#include "fec.h"
#include "uat_protocol.h"
extern "C" {
#include "fec/rs.h"
}
using namespace flightaware::uat;
using namespace flightaware::uat::fec;
FEC::FEC(void) {
rs_downlink_short_ = ::init_rs_char(
/* symsize */ 8, /* gfpoly */ DOWNLINK_SHORT_POLY, /* fcr */ 120,
/* prim */ 1, /* nroots */ DOWNLINK_SHORT_ROOTS,
/* pad */ DOWNLINK_SHORT_PAD);
rs_downlink_long_ = ::init_rs_char(
/* symsize */ 8, /* gfpoly */ DOWNLINK_LONG_POLY, /* fcr */ 120,
/* prim */ 1, /* nroots */ DOWNLINK_LONG_ROOTS,
/* pad */ DOWNLINK_LONG_PAD);
rs_uplink_ = ::init_rs_char(/* symsize */ 8, /* gfpoly */ UPLINK_BLOCK_POLY,
/* fcr */ 120, /* prim */ 1,
/* nroots */ UPLINK_BLOCK_ROOTS,
/* pad */ UPLINK_BLOCK_PAD);
}
FEC::~FEC(void) {
::free_rs_char(rs_downlink_short_);
::free_rs_char(rs_downlink_long_);
::free_rs_char(rs_uplink_);
}
std::tuple<bool, Bytes, unsigned> FEC::CorrectDownlink(const Bytes &raw, const std::vector<std::size_t> &erasures) {
using R = std::tuple<bool, Bytes, unsigned>;
if (raw.size() != DOWNLINK_LONG_BYTES) {
return R{false, {}, 0};
}
if (erasures.size() > DOWNLINK_LONG_ROOTS) {
// too many
return R{false, {}, 0};
}
// Try decoding as a Long UAT.
Bytes corrected;
std::copy(raw.begin(), raw.end(), std::back_inserter(corrected));
int erasures_array[DOWNLINK_LONG_ROOTS];
for (std::size_t i = 0; i < erasures.size(); ++i) {
erasures_array[i] = erasures[i];
corrected[erasures[i]] = 0;
}
int n_corrected = ::decode_rs_char(rs_downlink_long_, corrected.data(), erasures_array, erasures.size());
if (n_corrected >= 0 && n_corrected <= DOWNLINK_LONG_ROOTS && (corrected[0] >> 3) != 0) {
// Valid long frame.
corrected.resize(DOWNLINK_LONG_DATA_BYTES);
return R{true, std::move(corrected), n_corrected};
}
// Retry as Basic UAT
// We rely on decode_rs_char not modifying the data if there were
// uncorrectable errors in the previous step.
// Only pass in erasures that lie within the short message length
int short_erasures = 0;
for (auto e : erasures) {
if (e < DOWNLINK_SHORT_BYTES) {
if (short_erasures < DOWNLINK_SHORT_ROOTS) {
erasures_array[short_erasures] = e;
}
++short_erasures;
}
}
if (short_erasures > DOWNLINK_SHORT_ROOTS) {
// too many
return R{false, {}, 0};
}
n_corrected = ::decode_rs_char(rs_downlink_short_, corrected.data(), erasures_array, short_erasures);
if (n_corrected >= 0 && n_corrected <= DOWNLINK_SHORT_ROOTS && (corrected[0] >> 3) == 0) {
// Valid short frame
corrected.resize(DOWNLINK_SHORT_DATA_BYTES);
return R{true, std::move(corrected), n_corrected};
}
// Failed.
return R{false, {}, 0};
}
std::tuple<bool, Bytes, unsigned> FEC::CorrectUplink(const Bytes &raw, const std::vector<std::size_t> &erasures) {
using R = std::tuple<bool, Bytes, unsigned>;
if (raw.size() != UPLINK_BYTES) {
return R{false, {}, 0};
}
// uplink messages consist of 6 blocks, interleaved; each block consists of a
// data section then an ECC section; we need to deinterleave, check/correct
// the data, then join the blocks removing the ECC sections.
unsigned total_errors = 0;
Bytes corrected;
Bytes blockdata;
corrected.reserve(UPLINK_DATA_BYTES);
blockdata.resize(UPLINK_BLOCK_BYTES);
for (unsigned block = 0; block < UPLINK_BLOCKS_PER_FRAME; ++block) {
// deinterleave
for (unsigned i = 0; i < UPLINK_BLOCK_BYTES; ++i) {
blockdata[i] = raw[i * UPLINK_BLOCKS_PER_FRAME + block];
}
// build erasures for this block
int block_erasures[UPLINK_BLOCK_ROOTS];
int num_erasures = 0;
for (auto index : erasures) {
if (index % UPLINK_BLOCKS_PER_FRAME == block) {
if (num_erasures < UPLINK_BLOCK_ROOTS)
block_erasures[num_erasures] = index / UPLINK_BLOCKS_PER_FRAME + UPLINK_BLOCK_PAD;
num_erasures++;
}
}
// too many erasures in this block?
if (num_erasures > UPLINK_BLOCK_ROOTS) {
return R{false, {}, 0};
}
// error-correct
int n_corrected = ::decode_rs_char(rs_uplink_, blockdata.data(), block_erasures, num_erasures);
if (n_corrected < 0 || n_corrected > UPLINK_BLOCK_ROOTS) {
// Failed
return R{false, {}, 0};
}
total_errors += n_corrected;
// copy the data into the right place
std::copy(blockdata.begin(), blockdata.begin() + UPLINK_BLOCK_DATA_BYTES, std::back_inserter(corrected));
}
return R{true, std::move(corrected), total_errors};
}