summaryrefslogtreecommitdiff
path: root/gr-digital/lib/packet_header_default.cc
blob: 0b3692bc9d3d43f4f6e19b81a64d78fdf60db3ab (plain)
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
/* -*- c++ -*- */
/* Copyright 2012 Free Software Foundation, Inc.
 *
 * This file is part of GNU Radio
 *
 * SPDX-License-Identifier: GPL-3.0-or-later
 *
 */

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <gnuradio/digital/packet_header_default.h>
#include <cstring>

namespace gr {
namespace digital {

packet_header_default::sptr packet_header_default::make(long header_len,
                                                        const std::string& len_tag_key,
                                                        const std::string& num_tag_key,
                                                        int bits_per_byte)
{
    return packet_header_default::sptr(
        new packet_header_default(header_len, len_tag_key, num_tag_key, bits_per_byte));
}

const unsigned MASK_LUT[9] = { 0x00, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x2F, 0x7F, 0xFF };
packet_header_default::packet_header_default(long header_len,
                                             const std::string& len_tag_key,
                                             const std::string& num_tag_key,
                                             int bits_per_byte)
    : d_header_len(header_len),
      d_len_tag_key(pmt::string_to_symbol(len_tag_key)),
      d_num_tag_key(num_tag_key.empty() ? pmt::PMT_NIL
                                        : pmt::string_to_symbol(num_tag_key)),
      d_bits_per_byte(bits_per_byte),
      d_header_number(0)
{
    if (d_bits_per_byte < 1 || d_bits_per_byte > 8) {
        throw std::invalid_argument("bits_per_byte must be in [1, 8]");
    }
    d_mask = MASK_LUT[d_bits_per_byte];
}

packet_header_default::~packet_header_default() {}

bool packet_header_default::header_formatter(long packet_len,
                                             unsigned char* out,
                                             const std::vector<tag_t>& tags)
{
    packet_len &= 0x0FFF;
    d_crc_impl.reset();
    d_crc_impl.process_bytes((void const*)&packet_len, 2);
    d_crc_impl.process_bytes((void const*)&d_header_number, 2);
    unsigned char crc = d_crc_impl();

    memset(out, 0x00, d_header_len);
    int k = 0; // Position in out
    for (int i = 0; i < 12 && k < d_header_len; i += d_bits_per_byte, k++) {
        out[k] = (unsigned char)((packet_len >> i) & d_mask);
    }
    for (int i = 0; i < 12 && k < d_header_len; i += d_bits_per_byte, k++) {
        out[k] = (unsigned char)((d_header_number >> i) & d_mask);
    }
    for (int i = 0; i < 8 && k < d_header_len; i += d_bits_per_byte, k++) {
        out[k] = (unsigned char)((crc >> i) & d_mask);
    }
    d_header_number++;
    d_header_number &= 0x0FFF;

    return true;
}

bool packet_header_default::header_parser(const unsigned char* in,
                                          std::vector<tag_t>& tags)
{
    unsigned header_len = 0;
    unsigned header_num = 0;
    tag_t tag;

    int k = 0; // Position in "in"
    for (int i = 0; i < 12 && k < d_header_len; i += d_bits_per_byte, k++) {
        header_len |= (((int)in[k]) & d_mask) << i;
    }
    tag.key = d_len_tag_key;
    tag.value = pmt::from_long(header_len);
    tags.push_back(tag);
    if (k >= d_header_len) {
        return true;
    }
    if (d_num_tag_key == pmt::PMT_NIL) {
        k += 12;
    } else {
        for (int i = 0; i < 12 && k < d_header_len; i += d_bits_per_byte, k++) {
            header_num |= (((int)in[k]) & d_mask) << i;
        }
        tag.key = d_num_tag_key;
        tag.value = pmt::from_long(header_num);
        tags.push_back(tag);
    }
    if (k >= d_header_len) {
        return true;
    }

    d_crc_impl.reset();
    d_crc_impl.process_bytes((void const*)&header_len, 2);
    d_crc_impl.process_bytes((void const*)&header_num, 2);
    unsigned char crc_calcd = d_crc_impl();
    for (int i = 0; i < 8 && k < d_header_len; i += d_bits_per_byte, k++) {
        if ((((int)in[k]) & d_mask) != (((int)crc_calcd >> i) & d_mask)) {
            return false;
        }
    }

    return true;
}

} /* namespace digital */
} /* namespace gr */