| 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
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
 | //
// Copyright 2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0-or-later
//
#include "rhodium_cpld_ctrl.hpp"
#include "rhodium_constants.hpp"
#include <uhd/utils/log.hpp>
#include <uhd/utils/math.hpp>
#include <boost/format.hpp>
#include <chrono>
using namespace uhd;
using namespace uhd::math::fp_compare;
namespace {
//! Address of the CPLD scratch register
constexpr uint8_t CPLD_REGS_SCRATCH = 0x05;
//! Address of the CPLD gain table selection register
// constexpr uint8_t CPLD_REGS_GAIN_TBL_SEL = 0x06;
constexpr uint32_t MAX_GAIN_INDEX    = 60;
constexpr uint32_t MAX_LO_GAIN_INDEX = 30;
//! RX Demodulator Adjustment thresholds
constexpr double RX_DEMOD_ADJ_1500OHM_THRESHOLD = 3e9;
constexpr double RX_DEMOD_ADJ_200OHM_THRESHOLD  = 4.5e9;
/*
Unlike other CPLD fields, gain control doesn't use a register and instead
commands are directly sent over SPI. The format of these 24-bit commands is
as follows.
    23:22 Table select (1 = RX, 2 = TX)
    21:16 Gain index
    15:14 Reserved
    13    Write enable for DSA1
    12:7  Reserved
    6     Write enable for DSA2
    5:0   Reserved
The CPLD replies with the current attenuation settings as follows, but we
ignore this reply in our code.
    23:13 Reserved
    12:7  Current attenuator setting for DSA1
    6     Reserved
    5:0   Current attenuator setting for DSA2
*/
//! Gain loader constants
constexpr size_t GAIN_CTRL_TABLE_FIELD = 22;
constexpr size_t GAIN_CTRL_INDEX_FIELD = 16;
constexpr size_t GAIN_CTRL_DSA1_FIELD  = 13;
constexpr size_t GAIN_CTRL_DSA2_FIELD  = 6;
constexpr uint32_t GAIN_CTRL_TABLE_RX          = 1;
constexpr uint32_t GAIN_CTRL_TABLE_TX          = 2;
constexpr uint32_t GAIN_CTRL_DSA1_WRITE_ENABLE = 1;
constexpr uint32_t GAIN_CTRL_DSA2_WRITE_ENABLE = 1;
/*
Similar to gain control, LO control doesn't use a register and instead
commands are directly sent over SPI. The format of these 24-bit commands is
as follows:
    23:22 Table select (Always 3 = LO)
    21:16 Attenuator setting
    15:14 Reserved
    13    Write enable for RX LO DSA
    12:7  Reserved
    6     Write enable for TX LO DSA
    5:0   Reserved
The CPLD replies with the current attenuator settings as follows, but we
ignore this reply in our code.
    23:13 Reserved
    12:7  Current attenuator setting for RX LO DSA
    6     Reserved
    5:0   Current attenuator setting for TX LO DSA
*/
//! LO Gain loader constants
constexpr size_t LO_GAIN_CTRL_TABLE_FIELD = 22;
constexpr size_t LO_GAIN_CTRL_INDEX_FIELD = 16;
constexpr size_t LO_GAIN_CTRL_RX_LO_FIELD = 13;
constexpr size_t LO_GAIN_CTRL_TX_LO_FIELD = 6;
constexpr uint32_t LO_GAIN_CTRL_TABLE_LO            = 3;
constexpr uint32_t LO_GAIN_CTRL_RX_LO_WRITE_ENABLE  = 1;
constexpr uint32_t LO_GAIN_CTRL_RX_LO_WRITE_DISABLE = 0;
constexpr uint32_t LO_GAIN_CTRL_TX_LO_WRITE_ENABLE  = 1;
constexpr uint32_t LO_GAIN_CTRL_TX_LO_WRITE_DISABLE = 0;
} // namespace
rhodium_cpld_ctrl::rhodium_cpld_ctrl(write_spi_t write_fn, read_spi_t read_fn)
{
    _write_reg_fn = [write_fn](const uint8_t addr, const uint32_t data) {
        UHD_LOG_TRACE("RH_CPLD",
            str(boost::format("Writing to CPLD: 0x%02X -> 0x%04X") % uint32_t(addr)
                % data));
        const uint32_t spi_transaction = 0 | ((addr & 0x7F) << 17) | data;
        UHD_LOG_TRACE(
            "RH_CPLD", str(boost::format("SPI Transaction: 0x%04X") % spi_transaction));
        write_fn(spi_transaction);
    };
    _write_raw_fn = [write_fn](const uint32_t data) {
        UHD_LOG_TRACE("RH_CPLD", str(boost::format("Writing to CPLD: 0x%06X") % data));
        UHD_LOG_TRACE("RH_CPLD", str(boost::format("SPI Transaction: 0x%06X") % data));
        write_fn(data);
    };
    _read_reg_fn = [read_fn](const uint8_t addr) {
        UHD_LOG_TRACE("RH_CPLD",
            str(boost::format("Reading from CPLD address 0x%02X") % uint32_t(addr)));
        const uint32_t spi_transaction = (1 << 16) | ((addr & 0x7F) << 17);
        UHD_LOG_TRACE(
            "RH_CPLD", str(boost::format("SPI Transaction: 0x%04X") % spi_transaction));
        return read_fn(spi_transaction);
    };
    reset();
    _loopback_test();
}
/******************************************************************************
 * API
 *****************************************************************************/
void rhodium_cpld_ctrl::reset()
{
    std::lock_guard<std::mutex> l(_set_mutex);
    UHD_LOG_TRACE("RH_CPLD", "Resetting CPLD to default state");
    // Set local register cache to default values and commit
    _regs = rhodium_cpld_regs_t();
    _gain_queue.clear();
    commit(true);
}
uint16_t rhodium_cpld_ctrl::get_reg(const uint8_t addr)
{
    std::lock_guard<std::mutex> l(_set_mutex);
    return _read_reg_fn(addr);
}
void rhodium_cpld_ctrl::set_scratch(const uint16_t val)
{
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.scratch = val;
    commit();
}
uint16_t rhodium_cpld_ctrl::get_scratch()
{
    return get_reg(CPLD_REGS_SCRATCH);
}
void rhodium_cpld_ctrl::set_tx_switches(const tx_sw2_t tx_sw2,
    const tx_sw3_sw4_t tx_sw3_sw4,
    const tx_sw5_t tx_sw5,
    const tx_hb_lb_sel_t tx_hb_lb_sel,
    const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD",
        "Configuring TX band selection switches."
        " sw2="
            << tx_sw2 << " sw3_sw4=" << tx_sw3_sw4 << " sw5=" << tx_sw5
            << " hb_lb_sel=" << tx_hb_lb_sel);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.tx_sw2       = rhodium_cpld_regs_t::tx_sw2_t(tx_sw2);
    _regs.tx_sw3_sw4   = rhodium_cpld_regs_t::tx_sw3_sw4_t(tx_sw3_sw4);
    _regs.tx_sw5       = rhodium_cpld_regs_t::tx_sw5_t(tx_sw5);
    _regs.tx_hb_lb_sel = rhodium_cpld_regs_t::tx_hb_lb_sel_t(tx_hb_lb_sel);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_rx_switches(const rx_sw2_sw7_t rx_sw2_sw7,
    const rx_sw3_t rx_sw3,
    const rx_sw4_sw5_t rx_sw4_sw5,
    const rx_sw6_t rx_sw6,
    const rx_hb_lb_sel_t rx_hb_lb_sel,
    const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD",
        "Configuring RX band selection switches."
        " sw2_sw7="
            << rx_sw2_sw7 << " sw3=" << rx_sw3 << " sw4_sw5=" << rx_sw4_sw5
            << " sw6=" << rx_sw6 << " hb_lb_sel=" << rx_hb_lb_sel);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.rx_sw2_sw7   = rhodium_cpld_regs_t::rx_sw2_sw7_t(rx_sw2_sw7);
    _regs.rx_sw3       = rhodium_cpld_regs_t::rx_sw3_t(rx_sw3);
    _regs.rx_sw4_sw5   = rhodium_cpld_regs_t::rx_sw4_sw5_t(rx_sw4_sw5);
    _regs.rx_sw6       = rhodium_cpld_regs_t::rx_sw6_t(rx_sw6);
    _regs.rx_hb_lb_sel = rhodium_cpld_regs_t::rx_hb_lb_sel_t(rx_hb_lb_sel);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_rx_input_switches(
    const rx_sw1_t rx_sw1, const cal_iso_sw_t cal_iso_sw, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD",
        "Configuring RX input selection switches."
        " sw1="
            << rx_sw1 << " cal_iso=" << cal_iso_sw);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.rx_sw1     = rhodium_cpld_regs_t::rx_sw1_t(rx_sw1);
    _regs.cal_iso_sw = rhodium_cpld_regs_t::cal_iso_sw_t(cal_iso_sw);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_tx_output_switches(
    const tx_sw1_t tx_sw1, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD",
        "Configuring TX output selection switches."
        " sw1="
            << tx_sw1);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.tx_sw1 = rhodium_cpld_regs_t::tx_sw1_t(tx_sw1);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_rx_lo_source(
    const rx_lo_input_sel_t rx_lo_input_sel, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD", "Setting RX LO source to " << rx_lo_input_sel);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.rx_lo_input_sel = rhodium_cpld_regs_t::rx_lo_input_sel_t(rx_lo_input_sel);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_tx_lo_source(
    const tx_lo_input_sel_t tx_lo_input_sel, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD", "Setting TX LO source to " << tx_lo_input_sel);
    std::lock_guard<std::mutex> l(_set_mutex);
    _regs.tx_lo_input_sel = rhodium_cpld_regs_t::tx_lo_input_sel_t(tx_lo_input_sel);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_rx_lo_path(const double freq, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD", "Configuring RX LO filter and settings. freq=" << freq);
    std::lock_guard<std::mutex> l(_set_mutex);
    auto freq_compare = fp_compare_epsilon<double>(freq, RHODIUM_FREQ_COMPARE_EPSILON);
    if (freq_compare < RX_DEMOD_ADJ_1500OHM_THRESHOLD) {
        _regs.rx_demod_adj =
            rhodium_cpld_regs_t::rx_demod_adj_t::RX_DEMOD_ADJ_RES_1500_OHM;
    } else if (freq_compare < RX_DEMOD_ADJ_200OHM_THRESHOLD) {
        _regs.rx_demod_adj =
            rhodium_cpld_regs_t::rx_demod_adj_t::RX_DEMOD_ADJ_RES_200_OHM;
    } else {
        _regs.rx_demod_adj = rhodium_cpld_regs_t::rx_demod_adj_t::RX_DEMOD_ADJ_RES_OPEN;
    }
    if (freq_compare < RHODIUM_LO_0_9_GHZ_LPF_THRESHOLD_FREQ) {
        _regs.rx_lo_filter_sel =
            rhodium_cpld_regs_t::rx_lo_filter_sel_t::RX_LO_FILTER_SEL_0_9GHZ_LPF;
    } else if (freq_compare < RHODIUM_LO_2_25_GHZ_LPF_THRESHOLD_FREQ) {
        _regs.rx_lo_filter_sel =
            rhodium_cpld_regs_t::rx_lo_filter_sel_t::RX_LO_FILTER_SEL_2_25GHZ_LPF;
    } else {
        _regs.rx_lo_filter_sel =
            rhodium_cpld_regs_t::rx_lo_filter_sel_t::RX_LO_FILTER_SEL_5_85GHZ_LPF;
    }
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_tx_lo_path(const double freq, const bool defer_commit)
{
    UHD_LOG_TRACE("RH_CPLD", "Configuring TX LO filter and settings. freq=" << freq);
    std::lock_guard<std::mutex> l(_set_mutex);
    auto freq_compare = fp_compare_epsilon<double>(freq, RHODIUM_FREQ_COMPARE_EPSILON);
    if (freq_compare < RHODIUM_LO_0_9_GHZ_LPF_THRESHOLD_FREQ) {
        _regs.tx_lo_filter_sel =
            rhodium_cpld_regs_t::tx_lo_filter_sel_t::TX_LO_FILTER_SEL_0_9GHZ_LPF;
    } else if (freq_compare < RHODIUM_LO_2_25_GHZ_LPF_THRESHOLD_FREQ) {
        _regs.tx_lo_filter_sel =
            rhodium_cpld_regs_t::tx_lo_filter_sel_t::TX_LO_FILTER_SEL_2_25GHZ_LPF;
    } else {
        _regs.tx_lo_filter_sel =
            rhodium_cpld_regs_t::tx_lo_filter_sel_t::TX_LO_FILTER_SEL_5_85GHZ_LPF;
    }
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_gain_index(const uint32_t index,
    const gain_band_t band,
    const uhd::direction_t dir,
    const bool defer_commit)
{
    UHD_ASSERT_THROW(index <= MAX_GAIN_INDEX);
    UHD_ASSERT_THROW(dir == RX_DIRECTION or dir == TX_DIRECTION);
    if (band == HIGH) {
        if (dir == RX_DIRECTION) {
            _regs.rx_gain_tbl_sel = rhodium_cpld_regs_t::RX_GAIN_TBL_SEL_HIGHBAND;
        } else {
            _regs.tx_gain_tbl_sel = rhodium_cpld_regs_t::TX_GAIN_TBL_SEL_HIGHBAND;
        }
    } else {
        if (dir == RX_DIRECTION) {
            _regs.rx_gain_tbl_sel = rhodium_cpld_regs_t::RX_GAIN_TBL_SEL_LOWBAND;
        } else {
            _regs.tx_gain_tbl_sel = rhodium_cpld_regs_t::TX_GAIN_TBL_SEL_LOWBAND;
        }
    }
    const uint8_t table_id = (dir == RX_DIRECTION) ? GAIN_CTRL_TABLE_RX
                                                   : GAIN_CTRL_TABLE_TX;
    const uint32_t cmd = (table_id << GAIN_CTRL_TABLE_FIELD)
                         | (index << GAIN_CTRL_INDEX_FIELD)
                         | (GAIN_CTRL_DSA1_WRITE_ENABLE << GAIN_CTRL_DSA1_FIELD)
                         | (GAIN_CTRL_DSA2_WRITE_ENABLE << GAIN_CTRL_DSA2_FIELD);
    std::lock_guard<std::mutex> l(_set_mutex);
    _gain_queue.emplace_back(cmd);
    if (not defer_commit) {
        commit();
    }
}
void rhodium_cpld_ctrl::set_lo_gain(
    const uint32_t index, const uhd::direction_t dir, const bool defer_commit)
{
    UHD_ASSERT_THROW(index <= MAX_LO_GAIN_INDEX);
    // The DSA has 0-30 dB of attenuation in 1 dB steps
    // This index directly controls the attenuation value of the LO DSA,
    // so reverse the gain value to write the value
    const uint32_t attenuation = MAX_LO_GAIN_INDEX - index;
    const uint8_t set_rx       = (dir == RX_DIRECTION or dir == DX_DIRECTION)
                               ? LO_GAIN_CTRL_RX_LO_WRITE_ENABLE
                               : LO_GAIN_CTRL_RX_LO_WRITE_DISABLE;
    const uint8_t set_tx = (dir == TX_DIRECTION or dir == DX_DIRECTION)
                               ? LO_GAIN_CTRL_TX_LO_WRITE_ENABLE
                               : LO_GAIN_CTRL_TX_LO_WRITE_DISABLE;
    const uint32_t cmd = (LO_GAIN_CTRL_TABLE_LO << LO_GAIN_CTRL_TABLE_FIELD)
                         | (attenuation << LO_GAIN_CTRL_INDEX_FIELD)
                         | (set_rx << LO_GAIN_CTRL_RX_LO_FIELD)
                         | (set_tx << LO_GAIN_CTRL_TX_LO_FIELD);
    std::lock_guard<std::mutex> l(_set_mutex);
    _gain_queue.emplace_back(cmd);
    if (not defer_commit) {
        commit();
    }
}
/******************************************************************************
 * Private methods
 *****************************************************************************/
void rhodium_cpld_ctrl::_loopback_test()
{
    UHD_LOG_TRACE("RH_CPLD", "Performing CPLD scratch loopback test...");
    using namespace std::chrono;
    const uint16_t random_number = // Derived from current time
        uint16_t(system_clock::to_time_t(system_clock::now()) & 0xFFFF);
    set_scratch(random_number);
    const uint16_t actual = get_scratch();
    if (actual != random_number) {
        UHD_LOGGER_ERROR("RH_CPLD")
            << "CPLD scratch loopback failed! "
            << boost::format("Expected: 0x%04X Got: 0x%04X") % random_number % actual;
        throw uhd::runtime_error("CPLD scratch loopback failed!");
    }
    UHD_LOG_TRACE("RH_CPLD", "CPLD scratch loopback test passed!");
}
void rhodium_cpld_ctrl::commit(const bool save_all)
{
    UHD_LOGGER_TRACE("RH_CPLD")
        << "Storing register cache " << (save_all ? "completely" : "selectively")
        << " to CPLD via SPI...";
    auto changed_addrs = save_all ? _regs.get_all_addrs()
                                  : _regs.get_changed_addrs<size_t>();
    for (const auto addr : changed_addrs) {
        _write_reg_fn(addr, _regs.get_reg(addr));
    }
    _regs.save_state();
    UHD_LOG_TRACE("RH_CPLD",
        "Storing cache complete: "
        "Updated "
            << changed_addrs.size() << " registers.");
    UHD_LOGGER_TRACE("RH_CPLD") << "Writing queued gain commands "
                                << "to CPLD via SPI...";
    for (const auto cmd : _gain_queue) {
        _write_raw_fn(cmd);
    }
    UHD_LOG_TRACE("RH_CPLD",
        "Writing queued gain commands complete: "
        "Wrote "
            << _gain_queue.size() << " commands.");
    _gain_queue.clear();
}
 |