2010-10-19 00:59:08 +08:00
|
|
|
/*
|
|
|
|
# Copyright 2010 Nick Foster
|
|
|
|
#
|
|
|
|
# This file is part of gr-air-modes
|
|
|
|
#
|
|
|
|
# gr-air-modes 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; either version 3, or (at your option)
|
|
|
|
# any later version.
|
|
|
|
#
|
|
|
|
# gr-air-modes 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 for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with gr-air-modes; see the file COPYING. If not, write to
|
|
|
|
# the Free Software Foundation, Inc., 51 Franklin Street,
|
|
|
|
# Boston, MA 02110-1301, USA.
|
|
|
|
#
|
|
|
|
*/
|
|
|
|
|
2010-09-15 13:01:56 +08:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <air_modes_slicer.h>
|
|
|
|
#include <gr_io_signature.h>
|
|
|
|
#include <air_modes_types.h>
|
|
|
|
#include <sstream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <modes_parity.h>
|
2010-11-19 06:25:07 +08:00
|
|
|
#include <gr_tag_info.h>
|
2011-06-12 12:32:28 +08:00
|
|
|
#include <iostream>
|
2010-09-15 13:01:56 +08:00
|
|
|
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
}
|
|
|
|
|
|
|
|
air_modes_slicer_sptr air_make_modes_slicer(int channel_rate, gr_msg_queue_sptr queue)
|
|
|
|
{
|
|
|
|
return air_modes_slicer_sptr (new air_modes_slicer(channel_rate, queue));
|
|
|
|
}
|
|
|
|
|
|
|
|
air_modes_slicer::air_modes_slicer(int channel_rate, gr_msg_queue_sptr queue) :
|
|
|
|
gr_sync_block ("modes_slicer",
|
2010-11-21 09:06:48 +08:00
|
|
|
gr_make_io_signature (1, 1, sizeof(float)), //stream 0 is received data, stream 1 is binary preamble detector output
|
2010-09-15 13:01:56 +08:00
|
|
|
gr_make_io_signature (0, 0, 0) )
|
|
|
|
{
|
|
|
|
//initialize private data here
|
|
|
|
d_chip_rate = 2000000; //2Mchips per second
|
2011-06-12 12:32:28 +08:00
|
|
|
d_samples_per_chip = 2;//FIXME this is constant now channel_rate / d_chip_rate;
|
2010-09-15 13:01:56 +08:00
|
|
|
d_samples_per_symbol = d_samples_per_chip * 2;
|
|
|
|
d_check_width = 120 * d_samples_per_symbol; //how far you will have to look ahead
|
|
|
|
d_queue = queue;
|
2011-06-12 13:07:20 +08:00
|
|
|
d_secs_per_sample = 1.0 / d_chip_rate;
|
2010-09-15 13:01:56 +08:00
|
|
|
|
|
|
|
set_output_multiple(1+d_check_width * 2); //how do you specify buffer size for sinks?
|
|
|
|
}
|
|
|
|
|
2011-06-12 13:07:20 +08:00
|
|
|
//FIXME i'm sure this exists in gr
|
2010-11-19 06:25:07 +08:00
|
|
|
static bool pmtcompare(pmt::pmt_t x, pmt::pmt_t y)
|
|
|
|
{
|
|
|
|
uint64_t t_x, t_y;
|
|
|
|
t_x = pmt::pmt_to_uint64(pmt::pmt_tuple_ref(x, 0));
|
|
|
|
t_y = pmt::pmt_to_uint64(pmt::pmt_tuple_ref(y, 0));
|
|
|
|
return t_x < t_y;
|
|
|
|
}
|
2011-06-12 12:32:28 +08:00
|
|
|
|
|
|
|
//this slicer is courtesy of Lincoln Labs. supposedly it is more resistant to mode A/C FRUIT.
|
|
|
|
//see http://adsb.tc.faa.gov/WG3_Meetings/Meeting8/Squitter-Lon.pdf
|
2011-06-12 12:40:51 +08:00
|
|
|
static slice_result_t slicer(const float bit0, const float bit1, const float ref) {
|
|
|
|
slice_result_t result;
|
2011-06-12 12:32:28 +08:00
|
|
|
|
|
|
|
//3dB limits for bit slicing and confidence measurement
|
|
|
|
float highlimit=ref*2;
|
|
|
|
float lowlimit=ref*0.5;
|
|
|
|
|
|
|
|
bool firstchip_inref = ((bit0 > lowlimit) && (bit0 < highlimit));
|
|
|
|
bool secondchip_inref = ((bit1 > lowlimit) && (bit1 < highlimit));
|
|
|
|
|
|
|
|
if(firstchip_inref && !secondchip_inref) {
|
2011-06-12 12:40:51 +08:00
|
|
|
result.decision = 1;
|
|
|
|
result.confidence = 1;
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
|
|
|
else if(secondchip_inref && !firstchip_inref) {
|
2011-06-12 12:40:51 +08:00
|
|
|
result.decision = 0;
|
|
|
|
result.confidence = 1;
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
|
|
|
else if(firstchip_inref && secondchip_inref) {
|
2011-06-12 12:40:51 +08:00
|
|
|
result.decision = bit0 > bit1;
|
|
|
|
result.confidence = 0;
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
2011-06-12 15:56:22 +08:00
|
|
|
else {//if(!firstchip_inref && !secondchip_inref) {
|
2011-06-12 12:40:51 +08:00
|
|
|
result.decision = bit0 > bit1;
|
|
|
|
if(result.decision) {
|
|
|
|
if(bit1 < lowlimit * 0.5) result.confidence = 1;
|
|
|
|
else result.confidence = 0;
|
2011-06-12 12:32:28 +08:00
|
|
|
} else {
|
2011-06-12 12:40:51 +08:00
|
|
|
if(bit0 < lowlimit * 0.5) result.confidence = 1;
|
|
|
|
else result.confidence = 0;
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
|
|
|
}
|
2011-06-12 12:40:51 +08:00
|
|
|
return result;
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 11:26:58 +08:00
|
|
|
/*
|
|
|
|
static double pmt_to_timestamp(pmt::pmt_t tstamp, uint64_t sample_cnt, double secs_per_sample) {
|
2011-05-31 10:57:47 +08:00
|
|
|
double frac;
|
|
|
|
uint64_t secs, sample, sample_age;
|
|
|
|
|
|
|
|
if(gr_tags::get_name(tstamp) != "time") return 0;
|
|
|
|
|
|
|
|
secs = pmt_to_uint64(pmt_tuple_ref(gr_tags::get_value(tstamp), 0));
|
|
|
|
frac = pmt_to_double(pmt_tuple_ref(gr_tags::get_value(tstamp), 1));
|
|
|
|
sample = gr_tags::get_nitems(d_timestamp);
|
|
|
|
//now we have to offset the timestamp based on the current sample number
|
|
|
|
sample_age = (sample_cnt + i) - sample;
|
|
|
|
return sample_age * secs_per_sample + frac + secs;
|
|
|
|
}
|
2011-05-31 11:26:58 +08:00
|
|
|
*/
|
2010-09-15 13:01:56 +08:00
|
|
|
int air_modes_slicer::work(int noutput_items,
|
|
|
|
gr_vector_const_void_star &input_items,
|
|
|
|
gr_vector_void_star &output_items)
|
|
|
|
{
|
2011-06-12 12:32:28 +08:00
|
|
|
const float *in = (const float *) input_items[0];
|
2010-09-15 13:01:56 +08:00
|
|
|
int size = noutput_items - d_check_width; //since it's a sync block, i assume that it runs with ninput_items = noutput_items
|
|
|
|
|
|
|
|
int i;
|
2011-06-12 15:56:22 +08:00
|
|
|
static int n_ok=0, n_badcrc=0, n_loconf=0, n_zeroes=0;
|
2010-11-21 09:06:48 +08:00
|
|
|
|
|
|
|
std::vector<pmt::pmt_t> tags;
|
|
|
|
uint64_t abs_sample_cnt = nitems_read(0);
|
2011-06-12 12:32:28 +08:00
|
|
|
get_tags_in_range(tags, 0, abs_sample_cnt, abs_sample_cnt + size, pmt::pmt_string_to_symbol("preamble_found"));
|
2010-11-21 09:06:48 +08:00
|
|
|
std::vector<pmt::pmt_t>::iterator tag_iter;
|
|
|
|
|
|
|
|
for(tag_iter = tags.begin(); tag_iter != tags.end(); tag_iter++) {
|
|
|
|
uint64_t i = gr_tags::get_nitems(*tag_iter) - abs_sample_cnt;
|
|
|
|
modes_packet rx_packet;
|
|
|
|
|
|
|
|
memset(&rx_packet.data, 0x00, 14 * sizeof(unsigned char));
|
|
|
|
memset(&rx_packet.lowconfbits, 0x00, 24 * sizeof(unsigned char));
|
|
|
|
rx_packet.numlowconf = 0;
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2011-06-12 12:32:28 +08:00
|
|
|
//let's use the preamble to get a reference level for the packet
|
|
|
|
//fixme: a better thing to do is create a bi-level avg 1 and avg 0
|
|
|
|
//through simple statistics, then take the median for your slice level
|
|
|
|
//this won't improve decoding but will improve confidence
|
|
|
|
rx_packet.reference_level = (in[i]
|
|
|
|
+ in[i+2]
|
|
|
|
+ in[i+7]
|
|
|
|
+ in[i+9]) / 4.0;
|
|
|
|
|
|
|
|
i += 16; //move on up to the first bit of the packet data
|
|
|
|
//now let's slice the header so we can determine if it's a short pkt or a long pkt
|
|
|
|
unsigned char pkt_hdr = 0;
|
|
|
|
for(int j=0; j < 5; j++) {
|
2011-06-12 12:40:51 +08:00
|
|
|
slice_result_t slice_result = slicer(in[i+j*2], in[i+j*2+1], rx_packet.reference_level);
|
|
|
|
if(slice_result.decision) pkt_hdr += 1 << (4-j);
|
2011-06-12 12:32:28 +08:00
|
|
|
}
|
|
|
|
if(pkt_hdr == 17) rx_packet.type = Long_Packet;
|
|
|
|
else rx_packet.type = Short_Packet;
|
2011-06-12 13:07:20 +08:00
|
|
|
int packet_length = (rx_packet.type == framer_packet_type(Short_Packet)) ? 56 : 112;
|
2010-11-21 09:06:48 +08:00
|
|
|
|
2011-06-12 12:32:28 +08:00
|
|
|
//it's slice time!
|
|
|
|
//TODO: don't repeat your work here, you already have the first 5 bits
|
2010-11-21 09:06:48 +08:00
|
|
|
for(int j = 0; j < packet_length; j++) {
|
2011-06-12 12:40:51 +08:00
|
|
|
slice_result_t slice_result = slicer(in[i+j*2], in[i+j*2+1], rx_packet.reference_level);
|
2011-06-12 12:32:28 +08:00
|
|
|
|
|
|
|
//put the data into the packet
|
2011-06-12 12:40:51 +08:00
|
|
|
if(slice_result.decision) {
|
2010-11-21 09:06:48 +08:00
|
|
|
rx_packet.data[j/8] += 1 << (7-(j%8));
|
|
|
|
}
|
|
|
|
//put the confidence decision into the packet
|
2011-06-12 12:40:51 +08:00
|
|
|
if(slice_result.confidence) {
|
2010-11-21 09:06:48 +08:00
|
|
|
//rx_packet.confidence[j/8] += 1 << (7-(j%8));
|
2011-06-12 12:40:51 +08:00
|
|
|
} else {
|
|
|
|
if(rx_packet.numlowconf < 24) rx_packet.lowconfbits[rx_packet.numlowconf++] = j;
|
|
|
|
}
|
2010-11-21 09:06:48 +08:00
|
|
|
}
|
2010-11-19 06:25:07 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
/******************** BEGIN TIMESTAMP BS ******************/
|
2011-05-17 09:06:54 +08:00
|
|
|
rx_packet.timestamp = 0;
|
2011-05-31 11:26:58 +08:00
|
|
|
/*
|
2010-11-21 09:06:48 +08:00
|
|
|
uint64_t abs_sample_cnt = nitems_read(0);
|
|
|
|
std::vector<pmt::pmt_t> tags;
|
2011-05-31 10:57:47 +08:00
|
|
|
uint64_t timestamp_secs, timestamp_sample, timestamp_delta;
|
|
|
|
double timestamp_frac;
|
2010-11-21 09:06:48 +08:00
|
|
|
|
2011-05-17 09:06:54 +08:00
|
|
|
get_tags_in_range(tags, 0, abs_sample_cnt, abs_sample_cnt + i, pmt::pmt_string_to_symbol("time"));
|
2010-11-21 09:06:48 +08:00
|
|
|
//tags.back() is the most recent timestamp, then.
|
|
|
|
if(tags.size() > 0) {
|
2011-05-31 10:57:47 +08:00
|
|
|
d_timestamp = tags.back();
|
2011-05-17 09:06:54 +08:00
|
|
|
}
|
|
|
|
|
2011-05-31 10:57:47 +08:00
|
|
|
if(d_timestamp) {
|
|
|
|
rx_packet.timestamp = pmt_to_timestamp(d_timestamp, abs_sample_cnt + i, d_secs_per_sample);
|
2010-11-21 09:06:48 +08:00
|
|
|
}
|
2011-05-31 11:26:58 +08:00
|
|
|
*/
|
2010-11-21 09:06:48 +08:00
|
|
|
/******************* END TIMESTAMP BS *********************/
|
2010-11-19 06:25:07 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
//increment for the next round
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
//here you might want to traverse the whole packet and if you find all 0's, just toss it. don't know why these packets turn up, but they pass ECC.
|
|
|
|
bool zeroes = 1;
|
|
|
|
for(int m = 0; m < 14; m++) {
|
|
|
|
if(rx_packet.data[m]) zeroes = 0;
|
|
|
|
}
|
2011-06-12 15:56:22 +08:00
|
|
|
if(zeroes) {n_zeroes++; continue;} //toss it
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
rx_packet.message_type = (rx_packet.data[0] >> 3) & 0x1F; //get the message type for the parser to conveniently use, and to make decisions on ECC methods
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2011-06-12 15:59:41 +08:00
|
|
|
if(rx_packet.type == Short_Packet && rx_packet.message_type != 11 && rx_packet.numlowconf > 0) {n_loconf++; continue;}
|
2011-06-12 15:56:22 +08:00
|
|
|
if(rx_packet.message_type == 11 && rx_packet.numlowconf >= 10) {n_loconf++; continue;}
|
2010-10-18 10:45:19 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
rx_packet.parity = modes_check_parity(rx_packet.data, packet_length);
|
|
|
|
|
2011-06-12 15:56:22 +08:00
|
|
|
//parity for packets that aren't type 11 or type 17 is encoded with the transponder ID, which we don't know
|
|
|
|
//therefore we toss 'em if there's syndrome
|
|
|
|
//parity for the other short packets is usually nonzero, so they can't really be trusted that far
|
|
|
|
if(rx_packet.parity && (rx_packet.message_type == 11 || rx_packet.message_type == 17)) {n_badcrc++; continue;}
|
2010-11-21 09:06:48 +08:00
|
|
|
|
2011-06-12 15:56:22 +08:00
|
|
|
//we no longer attempt to brute force error correct via syndrome. it really only gets you 1% additional returns,
|
|
|
|
//at the expense of a lot of CPU time and complexity
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
//we'll replicate some data by sending the message type as the first field, followed by the first 8+24=32 bits of the packet, followed by
|
|
|
|
//56 long packet data bits if applicable (zero-padded if not), followed by parity
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
d_payload.str("");
|
|
|
|
d_payload << std::dec << std::setw(2) << std::setfill('0') << rx_packet.message_type << std::hex << " ";
|
|
|
|
for(int m = 0; m < 4; m++) {
|
|
|
|
d_payload << std::setw(2) << std::setfill('0') << unsigned(rx_packet.data[m]);
|
|
|
|
}
|
|
|
|
d_payload << " ";
|
|
|
|
if(packet_length == 112) {
|
|
|
|
for(int m = 4; m < 11; m++) {
|
2010-09-15 13:01:56 +08:00
|
|
|
d_payload << std::setw(2) << std::setfill('0') << unsigned(rx_packet.data[m]);
|
|
|
|
}
|
|
|
|
d_payload << " ";
|
2010-11-21 09:06:48 +08:00
|
|
|
for(int m = 11; m < 14; m++) {
|
|
|
|
d_payload << std::setw(2) << std::setfill('0') << unsigned(rx_packet.data[m]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for(int m = 4; m < 11; m++) {
|
|
|
|
d_payload << std::setw(2) << std::setfill('0') << unsigned(0);
|
2010-09-15 13:01:56 +08:00
|
|
|
}
|
2010-11-21 09:06:48 +08:00
|
|
|
d_payload << " ";
|
|
|
|
for(int m = 4; m < 7; m++) {
|
|
|
|
d_payload << std::setw(2) << std::setfill('0') << unsigned(rx_packet.data[m]);
|
|
|
|
}
|
|
|
|
}
|
2010-09-15 13:01:56 +08:00
|
|
|
|
2010-11-21 09:06:48 +08:00
|
|
|
d_payload << " " << std::setw(6) << rx_packet.parity << " " << std::dec << rx_packet.reference_level
|
2011-05-17 09:06:54 +08:00
|
|
|
<< " " << std::setprecision(10) << std::setw(10) << rx_packet.timestamp;
|
2010-09-15 13:01:56 +08:00
|
|
|
gr_message_sptr msg = gr_make_message_from_string(std::string(d_payload.str()));
|
2010-11-21 09:06:48 +08:00
|
|
|
d_queue->handle(msg);
|
2011-06-12 15:56:22 +08:00
|
|
|
n_ok++;
|
|
|
|
std::cout << "n_ok: " << n_ok << " n_loconf: " << n_loconf << " n_badcrc: " << n_badcrc << " n_zeroes: " << n_zeroes << std::endl;
|
2010-09-15 13:01:56 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|