blob: bd1b0ce9315ec9d63cb1137133b20bd348e604b3 [file] [log] [blame]
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +01001/*
Pau Espin Pedrole9ce77b2019-06-25 12:29:01 +02002 * Copyright (C) 2018-2019 sysmocom - s.f.m.c. GmbH
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +01003 * All Rights Reserved
4 *
Pau Espin Pedrole9ce77b2019-06-25 12:29:01 +02005 * SPDX-License-Identifier: AGPL-3.0+
6 *
7 * Author: Pau Espin Pedrol <pespin@sysmocom.de>
8 *
9 * This program is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU Affero General Public License as published by
11 * the Free Software Foundation, either version 3 of the License, or
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010012 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Pau Espin Pedrole9ce77b2019-06-25 12:29:01 +020017 * GNU Affero General Public License for more details.
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010018 *
Pau Espin Pedrole9ce77b2019-06-25 12:29:01 +020019 * You should have received a copy of the GNU Affero General Public License
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010020 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Pau Espin Pedrole9ce77b2019-06-25 12:29:01 +020021 * See the COPYING file in the main directory for details.
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010022 */
23
24#include <string.h>
25#include <stdint.h>
26#include <inttypes.h>
27#include <netinet/in.h>
28#include <arpa/inet.h>
29
30#include <osmocom/core/talloc.h>
31#include <osmocom/core/utils.h>
32#include <osmocom/core/rate_ctr.h>
33
34#include <osmocom/vty/command.h>
Pau Espin Pedrolaebbfe02020-01-02 16:39:11 +010035#include <osmocom/vty/logging.h>
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010036#include <osmocom/vty/vty.h>
37#include <osmocom/vty/misc.h>
38
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +020039#include "trx_rate_ctr.h"
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010040#include "trx_vty.h"
41#include "../config.h"
42
43static struct trx_ctx* g_trx_ctx;
44
Pau Espin Pedrola7bf6cd2020-01-14 17:52:15 +010045const struct value_string clock_ref_names[] = {
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +010046 { REF_INTERNAL, "internal" },
47 { REF_EXTERNAL, "external" },
Pau Espin Pedrolaae403f2018-08-27 16:50:19 +020048 { REF_GPS, "gpsdo" },
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +010049 { 0, NULL }
50};
51
Alexander Chemeris9a87d902019-10-15 00:33:07 +030052const struct value_string filler_names[] = {
53 { FILLER_DUMMY, "Dummy bursts (C0 only)" },
54 { FILLER_ZERO, "Empty bursts" },
55 { FILLER_NORM_RAND, "GMSK Normal Bursts with random payload" },
56 { FILLER_EDGE_RAND, "8-PSK Normal Bursts with random payload" },
57 { FILLER_ACCESS_RAND, "Access Bursts with random payload" },
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +010058 { 0, NULL }
59};
60
Alexander Chemeris9a87d902019-10-15 00:33:07 +030061static const struct value_string filler_types[] = {
62 { FILLER_DUMMY, "dummy" },
63 { FILLER_ZERO, "zero" },
64 { FILLER_NORM_RAND, "random-nb-gmsk" },
65 { FILLER_EDGE_RAND, "random-nb-8psk" },
66 { FILLER_ACCESS_RAND, "random-ab" },
67 { 0, NULL }
68};
69
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +070070static const struct value_string filler_docs[] = {
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +070071 { FILLER_DUMMY, "Send a Dummy Burst on C0 (TRX0) and empty burst on other channels" },
Vadim Yanitskiy15873072020-12-08 15:35:52 +010072 { FILLER_ZERO, "Send an empty burst (default)" },
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +070073 { FILLER_NORM_RAND, "Send a GMSK modulated Normal Burst with random bits (spectrum mask testing)" },
74 { FILLER_EDGE_RAND, "Send an 8-PSK modulated Normal Burst with random bits (spectrum mask testing)" },
75 { FILLER_ACCESS_RAND, "Send an Access Burst with random bits (Rx/Tx alignment testing)" },
76 { 0, NULL }
77};
78
Alexander Chemeris9a87d902019-10-15 00:33:07 +030079
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010080struct trx_ctx *trx_from_vty(struct vty *v)
81{
82 /* It can't hurt to force callers to continue to pass the vty instance
83 * to this function, in case we'd like to retrieve the global
84 * trx instance from the vty at some point in the future. But
85 * until then, just return the global pointer, which should have been
86 * initialized by trx_vty_init().
87 */
88 OSMO_ASSERT(g_trx_ctx);
89 return g_trx_ctx;
90}
91
92enum trx_vty_node {
93 TRX_NODE = _LAST_OSMOVTY_NODE + 1,
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +010094 CHAN_NODE,
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +010095};
96
97static struct cmd_node trx_node = {
98 TRX_NODE,
99 "%s(config-trx)# ",
100 1,
101};
102
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100103static struct cmd_node chan_node = {
104 CHAN_NODE,
105 "%s(config-trx-chan)# ",
106 1,
107};
108
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100109DEFUN(cfg_trx, cfg_trx_cmd,
110 "trx",
111 "Configure the TRX\n")
112{
113 struct trx_ctx *trx = trx_from_vty(vty);
114
115 if (!trx)
116 return CMD_WARNING;
117
118 vty->node = TRX_NODE;
119
120 return CMD_SUCCESS;
121}
122
123DEFUN(cfg_bind_ip, cfg_bind_ip_cmd,
Vadim Yanitskiyc620ced2020-10-30 01:47:35 +0700124 "bind-ip " VTY_IPV4_CMD,
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100125 "Set the IP address for the local bind\n"
126 "IPv4 Address\n")
127{
128 struct trx_ctx *trx = trx_from_vty(vty);
129
130 osmo_talloc_replace_string(trx, &trx->cfg.bind_addr, argv[0]);
131
132 return CMD_SUCCESS;
133}
134
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100135DEFUN(cfg_remote_ip, cfg_remote_ip_cmd,
Vadim Yanitskiyc620ced2020-10-30 01:47:35 +0700136 "remote-ip " VTY_IPV4_CMD,
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100137 "Set the IP address for the remote BTS\n"
138 "IPv4 Address\n")
139{
140 struct trx_ctx *trx = trx_from_vty(vty);
141
142 osmo_talloc_replace_string(trx, &trx->cfg.remote_addr, argv[0]);
143
144 return CMD_SUCCESS;
145}
146
147DEFUN(cfg_base_port, cfg_base_port_cmd,
148 "base-port <1-65535>",
149 "Set the TRX Base Port\n"
150 "TRX Base Port\n")
151{
152 struct trx_ctx *trx = trx_from_vty(vty);
153
154 trx->cfg.base_port = atoi(argv[0]);
155
156 return CMD_SUCCESS;
157}
158
159DEFUN(cfg_dev_args, cfg_dev_args_cmd,
160 "dev-args DESC",
161 "Set the device-specific arguments to pass to the device\n"
162 "Device-specific arguments\n")
163{
164 struct trx_ctx *trx = trx_from_vty(vty);
165
166 osmo_talloc_replace_string(trx, &trx->cfg.dev_args, argv[0]);
167
168 return CMD_SUCCESS;
169}
170
171DEFUN(cfg_tx_sps, cfg_tx_sps_cmd,
172 "tx-sps (1|4)",
173 "Set the Tx Samples-per-Symbol\n"
Vadim Yanitskiyef79fd92020-10-30 01:44:29 +0700174 "Tx Samples-per-Symbol\n"
175 "1 Sample-per-Symbol\n"
176 "4 Samples-per-Symbol\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100177{
178 struct trx_ctx *trx = trx_from_vty(vty);
179
180 trx->cfg.tx_sps = atoi(argv[0]);
181
182 return CMD_SUCCESS;
183}
184
185DEFUN(cfg_rx_sps, cfg_rx_sps_cmd,
186 "rx-sps (1|4)",
187 "Set the Rx Samples-per-Symbol\n"
Vadim Yanitskiyef79fd92020-10-30 01:44:29 +0700188 "Rx Samples-per-Symbol\n"
189 "1 Sample-per-Symbol\n"
190 "4 Samples-per-Symbol\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100191{
192 struct trx_ctx *trx = trx_from_vty(vty);
193
194 trx->cfg.rx_sps = atoi(argv[0]);
195
196 return CMD_SUCCESS;
197}
198
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100199DEFUN(cfg_clock_ref, cfg_clock_ref_cmd,
200 "clock-ref (internal|external|gpsdo)",
201 "Set the Reference Clock\n"
Martin Hauke066fd042019-10-13 19:08:00 +0200202 "Enable internal reference (default)\n"
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100203 "Enable external 10 MHz reference\n"
204 "Enable GPSDO reference\n")
205{
206 struct trx_ctx *trx = trx_from_vty(vty);
207
208 trx->cfg.clock_ref = get_string_value(clock_ref_names, argv[0]);
209
210 return CMD_SUCCESS;
211}
212
213DEFUN(cfg_multi_arfcn, cfg_multi_arfcn_cmd,
214 "multi-arfcn (disable|enable)",
Vadim Yanitskiy5d6504c2020-10-30 01:50:05 +0700215 "Multi-ARFCN transceiver mode (default=disable)\n"
216 "Enable multi-ARFCN mode\n" "Disable multi-ARFCN mode\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100217{
218 struct trx_ctx *trx = trx_from_vty(vty);
219
220 if (strcmp("disable", argv[0]) == 0) {
221 trx->cfg.multi_arfcn = false;
Tom Tsoud2800452019-04-01 07:55:48 +0700222 return CMD_SUCCESS;
223 }
224
225 if (trx->cfg.num_chans > TRX_MCHAN_MAX) {
226 vty_out(vty, "Up to %i channels are supported for multi-TRX mode%s",
227 TRX_MCHAN_MAX, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100228 return CMD_WARNING;
229 }
230
Tom Tsoud2800452019-04-01 07:55:48 +0700231 trx->cfg.multi_arfcn = true;
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100232 return CMD_SUCCESS;
233}
234
235DEFUN(cfg_offset, cfg_offset_cmd,
236 "offset FLOAT",
237 "Set the baseband frequency offset (default=0, auto)\n"
238 "Baseband Frequency Offset\n")
239{
240 struct trx_ctx *trx = trx_from_vty(vty);
241
242 trx->cfg.offset = atof(argv[0]);
243
244 return CMD_SUCCESS;
245}
246
Vadim Yanitskiy3b8f7c42018-08-23 13:41:06 +0700247DEFUN_ATTR(cfg_freq_offset, cfg_freq_offset_cmd,
248 "freq-offset FLOAT",
249 "Apply an artificial offset to Rx/Tx carrier frequency\n"
250 "Frequency offset in kHz (e.g. -145300)\n",
251 CMD_ATTR_HIDDEN)
252{
253 struct trx_ctx *trx = trx_from_vty(vty);
254
255 trx->cfg.freq_offset_khz = atof(argv[0]);
256
257 return CMD_SUCCESS;
258}
259
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100260DEFUN(cfg_rssi_offset, cfg_rssi_offset_cmd,
Pau Espin Pedrole91544d2020-10-13 17:03:37 +0200261 "rssi-offset FLOAT [relative]",
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100262 "Set the RSSI to dBm offset in dB (default=0)\n"
Pau Espin Pedrole91544d2020-10-13 17:03:37 +0200263 "RSSI to dBm offset in dB\n"
264 "Add to the default rssi-offset value instead of completely replacing it\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100265{
266 struct trx_ctx *trx = trx_from_vty(vty);
267
268 trx->cfg.rssi_offset = atof(argv[0]);
Pau Espin Pedrole91544d2020-10-13 17:03:37 +0200269 trx->cfg.force_rssi_offset = (argc == 1);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100270
271 return CMD_SUCCESS;
272}
273
Ericc27fe602021-05-05 17:33:18 +0200274
275DEFUN_ATTR(cfg_ul_fn_offset, cfg_ul_fn_offset_cmd,
276 "ul-fn-offset <-10-10>",
277 "Adjusts the uplink frame FN by the specified amount\n"
278 "Frame Number offset\n",
279 CMD_ATTR_HIDDEN)
280{
281 struct trx_ctx *trx = trx_from_vty(vty);
282
283 trx->cfg.ul_fn_offset = atoi(argv[0]);
284
285 return CMD_SUCCESS;
286}
287
Eric Wildf8c7a522023-05-17 17:44:08 +0200288DEFUN_ATTR(cfg_ul_freq_override, cfg_ul_freq_override_cmd,
289 "ul-freq-override FLOAT",
290 "Overrides Rx carrier frequency\n"
291 "Frequency in Hz (e.g. 145300000)\n",
292 CMD_ATTR_HIDDEN)
293{
294 struct trx_ctx *trx = trx_from_vty(vty);
295
296 trx->cfg.overrides.ul_freq_override = true;
297 trx->cfg.overrides.ul_freq = atof(argv[0]);
298
299 return CMD_SUCCESS;
300}
301DEFUN_ATTR(cfg_dl_freq_override, cfg_dl_freq_override_cmd,
302 "dl-freq-override FLOAT",
303 "Overrides Tx carrier frequency\n"
304 "Frequency in Hz (e.g. 145300000)\n",
305 CMD_ATTR_HIDDEN)
306{
307 struct trx_ctx *trx = trx_from_vty(vty);
308
309 trx->cfg.overrides.dl_freq_override = true;
310 trx->cfg.overrides.dl_freq = atof(argv[0]);
311
312 return CMD_SUCCESS;
313}
314
315DEFUN_ATTR(cfg_ul_gain_override, cfg_ul_gain_override_cmd,
316 "ul-gain-override FLOAT",
317 "Overrides Rx gain\n"
318 "gain in dB\n",
319 CMD_ATTR_HIDDEN)
320{
321 struct trx_ctx *trx = trx_from_vty(vty);
322
323 trx->cfg.overrides.ul_gain_override = true;
324 trx->cfg.overrides.ul_gain = atof(argv[0]);
325
326 return CMD_SUCCESS;
327}
328DEFUN_ATTR(cfg_dl_gain_override, cfg_dl_gain_override_cmd,
329 "dl-gain-override FLOAT",
330 "Overrides Tx gain\n"
331 "gain in dB\n",
332 CMD_ATTR_HIDDEN)
333{
334 struct trx_ctx *trx = trx_from_vty(vty);
335
336 trx->cfg.overrides.dl_gain_override = true;
337 trx->cfg.overrides.dl_gain = atof(argv[0]);
338
339 return CMD_SUCCESS;
340}
341
Ericcca5d932023-07-25 18:44:03 +0200342DEFUN_ATTR(cfg_use_viterbi, cfg_use_viterbi_cmd,
343 "viterbi-eq (disable|enable)",
344 "Use viterbi equalizer for gmsk (default=disable)\n"
345 "Disable VA\n"
346 "Enable VA\n",
347 CMD_ATTR_HIDDEN)
348{
349 struct trx_ctx *trx = trx_from_vty(vty);
350
351 if (strcmp("disable", argv[0]) == 0)
352 trx->cfg.use_va = false;
353 else if (strcmp("enable", argv[0]) == 0)
354 trx->cfg.use_va = true;
355 else
356 return CMD_WARNING;
357
358 return CMD_SUCCESS;
359}
360
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100361DEFUN(cfg_swap_channels, cfg_swap_channels_cmd,
362 "swap-channels (disable|enable)",
Vadim Yanitskiy948b4e42020-10-30 01:27:21 +0700363 "Swap primary and secondary channels of the PHY (if any)\n"
364 "Do not swap primary and secondary channels (default)\n"
365 "Swap primary and secondary channels\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100366{
367 struct trx_ctx *trx = trx_from_vty(vty);
368
369 if (strcmp("disable", argv[0]) == 0) {
370 trx->cfg.swap_channels = false;
371 } else if (strcmp("enable", argv[0]) == 0) {
372 trx->cfg.swap_channels = true;
373 } else {
374 return CMD_WARNING;
375 }
376
377 return CMD_SUCCESS;
378}
379
380DEFUN(cfg_egprs, cfg_egprs_cmd,
381 "egprs (disable|enable)",
Vadim Yanitskiye0bdb6b2020-10-30 01:35:19 +0700382 "EGPRS (8-PSK demodulation) support (default=disable)\n"
383 "Disable EGPRS (8-PSK demodulation) support\n"
384 "Enable EGPRS (8-PSK demodulation) support\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100385{
386 struct trx_ctx *trx = trx_from_vty(vty);
387
388 if (strcmp("disable", argv[0]) == 0) {
389 trx->cfg.egprs = false;
390 } else if (strcmp("enable", argv[0]) == 0) {
391 trx->cfg.egprs = true;
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100392 } else {
393 return CMD_WARNING;
394 }
395
396 return CMD_SUCCESS;
397}
398
Vadim Yanitskiya8b35652018-10-22 02:52:18 +0200399DEFUN(cfg_ext_rach, cfg_ext_rach_cmd,
400 "ext-rach (disable|enable)",
Vadim Yanitskiyfaacb192020-10-30 01:35:49 +0700401 "11-bit Access Burst correlation support (default=disable)\n"
402 "Disable 11-bit Access Burst (TS1 & TS2) correlation\n"
403 "Enable 11-bit Access Burst (TS1 & TS2) correlation\n")
Vadim Yanitskiya8b35652018-10-22 02:52:18 +0200404{
405 struct trx_ctx *trx = trx_from_vty(vty);
406
407 if (strcmp("disable", argv[0]) == 0)
408 trx->cfg.ext_rach = false;
409
410 if (strcmp("enable", argv[0]) == 0)
411 trx->cfg.ext_rach = true;
412
413 return CMD_SUCCESS;
414}
415
Pau Espin Pedrol553a2502020-07-29 18:05:25 +0200416DEFUN_DEPRECATED(cfg_rt_prio, cfg_rt_prio_cmd,
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100417 "rt-prio <1-32>",
418 "Set the SCHED_RR real-time priority\n"
419 "Real time priority\n")
420{
421 struct trx_ctx *trx = trx_from_vty(vty);
422
423 trx->cfg.sched_rr = atoi(argv[0]);
Pau Espin Pedrol553a2502020-07-29 18:05:25 +0200424 vty_out (vty, "%% 'rt-prio %u' is deprecated, use 'policy rr %u' under 'sched' node instead%s",
425 trx->cfg.sched_rr, trx->cfg.sched_rr, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100426
427 return CMD_SUCCESS;
428}
429
Eric Wildac0487e2019-06-17 13:02:44 +0200430DEFUN(cfg_stack_size, cfg_stack_size_cmd,
431 "stack-size <0-2147483647>",
432 "Set the stack size per thread in BYTE, 0 = OS default\n"
433 "Stack size per thread in BYTE\n")
434{
435 struct trx_ctx *trx = trx_from_vty(vty);
436
437 trx->cfg.stack_size = atoi(argv[0]);
438
439 return CMD_SUCCESS;
440}
441
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +0700442#define CFG_FILLER_DOC_STR \
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300443 "Filler burst settings\n"
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +0700444
445DEFUN(cfg_filler, cfg_filler_type_cmd,
446 "AUTO-GENERATED", "AUTO-GENERATED")
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300447{
448 struct trx_ctx *trx = trx_from_vty(vty);
449 // trx->cfg.filler is unsigned, so we need an interim int var to detect errors
450 int type = get_string_value(filler_types, argv[0]);
451
452 if (type < 0) {
453 trx->cfg.filler = FILLER_ZERO;
454 return CMD_WARNING;
455 }
456 trx->cfg.filler = type;
457
458 return CMD_SUCCESS;
459}
460
461DEFUN(cfg_test_rtsc, cfg_filler_tsc_cmd,
462 "filler tsc <0-7>",
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +0700463 CFG_FILLER_DOC_STR
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300464 "Set the TSC for GMSK/8-PSK Normal Burst random fillers. Used only with 'random-nb-gmsk' and"
465 " 'random-nb-8psk' filler types. (default=0)\n"
466 "TSC\n")
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100467{
468 struct trx_ctx *trx = trx_from_vty(vty);
469
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300470 trx->cfg.rtsc = atoi(argv[0]);
471
472 return CMD_SUCCESS;
473}
474
475DEFUN(cfg_test_rach_delay, cfg_filler_rach_delay_cmd,
476 "filler access-burst-delay <0-68>",
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +0700477 CFG_FILLER_DOC_STR
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300478 "Set the delay for Access Burst random fillers. Used only with 'random-ab' filler type. (default=0)\n"
479 "RACH delay in symbols\n")
480{
481 struct trx_ctx *trx = trx_from_vty(vty);
482
483 trx->cfg.rach_delay = atoi(argv[0]);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100484
485 return CMD_SUCCESS;
486}
487
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200488static int vty_ctr_name_2_id(const char* str) {
489 size_t i;
490 for (i = 0; trx_chan_ctr_names[i].str; i++) {
491 if (strstr(trx_chan_ctr_names[i].str, str)) {
492 return i;
493 }
494 }
495 return -1;
496}
497
498static int vty_intv_name_2_id(const char* str) {
499 size_t i;
500 for (i = 0; rate_ctr_intv[i].str; i++) {
501 if (strcmp(rate_ctr_intv[i].str, str) == 0) {
502 return i;
503 }
504 }
505 return -1;
506}
507
Pau Espin Pedrol1d0c6fe2020-07-09 18:09:10 +0200508#define THRESHOLD_ARGS "(rx_overruns|tx_underruns|rx_drop_events|rx_drop_samples|tx_drop_events|tx_drop_samples|tx_stale_bursts|tx_unavailable_bursts|tx_trxd_fn_repeated|tx_trxd_fn_outoforder|tx_trxd_fn_skipped)"
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200509#define THRESHOLD_STR_VAL(s) "Set threshold value for rate_ctr device:" OSMO_STRINGIFY_VAL(s) "\n"
510#define THRESHOLD_STRS \
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200511 THRESHOLD_STR_VAL(rx_overruns) \
512 THRESHOLD_STR_VAL(tx_underruns) \
513 THRESHOLD_STR_VAL(rx_drop_events) \
Pau Espin Pedrol68a78092019-07-29 20:11:25 +0200514 THRESHOLD_STR_VAL(rx_drop_samples) \
515 THRESHOLD_STR_VAL(tx_drop_events) \
Pau Espin Pedrol92ba59d2020-06-29 14:34:59 +0200516 THRESHOLD_STR_VAL(tx_drop_samples) \
Pau Espin Pedrolc0d6fd22020-07-09 16:51:47 +0200517 THRESHOLD_STR_VAL(tx_stale_bursts) \
Pau Espin Pedrol1d0c6fe2020-07-09 18:09:10 +0200518 THRESHOLD_STR_VAL(tx_unavailable_bursts) \
Pau Espin Pedrolc0d6fd22020-07-09 16:51:47 +0200519 THRESHOLD_STR_VAL(tx_trxd_fn_repeated) \
520 THRESHOLD_STR_VAL(tx_trxd_fn_outoforder) \
521 THRESHOLD_STR_VAL(tx_trxd_fn_skipped) \
522 ""
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200523#define INTV_ARGS "(per-second|per-minute|per-hour|per-day)"
524#define INTV_STR_VAL(s) "Threshold value sampled " OSMO_STRINGIFY_VAL(s) "\n"
525#define INTV_STRS \
526 INTV_STR_VAL(per-second) \
527 INTV_STR_VAL(per-minute) \
528 INTV_STR_VAL(per-hour) \
529 INTV_STR_VAL(per-day)
530
Philipp Maier30863e82020-09-29 22:03:51 +0200531DEFUN_ATTR(cfg_ctr_error_threshold, cfg_ctr_error_threshold_cmd,
532 "ctr-error-threshold " THRESHOLD_ARGS " <0-65535> " INTV_ARGS,
533 "Threshold rate for error counter\n"
534 THRESHOLD_STRS
535 "Value to set for threshold\n"
536 INTV_STRS,
537 CMD_ATTR_IMMEDIATE)
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200538{
539 int rc;
540 struct ctr_threshold ctr;
541
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200542 rc = vty_ctr_name_2_id(argv[0]);
543 if (rc < 0) {
544 vty_out(vty, "No valid ctr_name found for ctr-error-threshold %s%s",
545 argv[0], VTY_NEWLINE);
546 return CMD_WARNING;
547 }
548 ctr.ctr_id = (enum TrxCtr)rc;
549 ctr.val = atoi(argv[1]);
550 rc = vty_intv_name_2_id(argv[2]);
551 if (rc < 0) {
552 vty_out(vty, "No valid time frame found for ctr-error-threshold %s %d %s%s",
553 argv[0], ctr.val, argv[2], VTY_NEWLINE);
554 return CMD_WARNING;
555 }
556 ctr.intv = (enum rate_ctr_intv) rc;
557 trx_rate_ctr_threshold_add(&ctr);
558
559 return CMD_SUCCESS;
560}
561
Philipp Maier30863e82020-09-29 22:03:51 +0200562DEFUN_ATTR(cfg_no_ctr_error_threshold, cfg_no_ctr_error_threshold_cmd,
563 "no ctr-error-threshold " THRESHOLD_ARGS " <0-65535> " INTV_ARGS,
564 NO_STR "Threshold rate for error counter\n"
565 THRESHOLD_STRS
566 "Value to set for threshold\n"
567 INTV_STRS,
568 CMD_ATTR_IMMEDIATE)
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200569{
570 int rc;
571 struct ctr_threshold ctr;
572
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200573 rc = vty_ctr_name_2_id(argv[0]);
574 if (rc < 0) {
575 vty_out(vty, "No valid ctr_name found for ctr-error-threshold %s%s",
576 argv[0], VTY_NEWLINE);
577 return CMD_WARNING;
578 }
579 ctr.ctr_id = (enum TrxCtr)rc;
580 ctr.val = atoi(argv[1]);
581 rc = vty_intv_name_2_id(argv[2]);
582 if (rc < 0) {
583 vty_out(vty, "No valid time frame found for ctr-error-threshold %s %d %s%s",
584 argv[0], ctr.val, argv[2], VTY_NEWLINE);
585 return CMD_WARNING;
586 }
587 ctr.intv = (enum rate_ctr_intv) rc;
588 if (trx_rate_ctr_threshold_del(&ctr) < 0) {
589 vty_out(vty, "no ctr-error-threshold: Entry to delete not found%s", VTY_NEWLINE);
590 return CMD_WARNING;
591 }
592
593 return CMD_SUCCESS;
594}
595
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100596DEFUN(cfg_chan, cfg_chan_cmd,
597 "chan <0-100>",
598 "Select a channel to configure\n"
599 "Channel index\n")
600{
601 struct trx_ctx *trx = trx_from_vty(vty);
602 int idx = atoi(argv[0]);
603
604 if (idx >= TRX_CHAN_MAX) {
605 vty_out(vty, "Chan list full.%s", VTY_NEWLINE);
606 return CMD_WARNING;
Tom Tsoud2800452019-04-01 07:55:48 +0700607 } else if (trx->cfg.multi_arfcn && trx->cfg.num_chans >= TRX_MCHAN_MAX) {
608 vty_out(vty, "Up to %i channels are supported for multi-TRX mode%s",
609 TRX_MCHAN_MAX, VTY_NEWLINE);
610 return CMD_WARNING;
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100611 }
Tom Tsoud2800452019-04-01 07:55:48 +0700612
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100613 if (trx->cfg.num_chans < idx) { /* Unexisting or creating non-consecutive */
614 vty_out(vty, "Non-existent or non-consecutive chan %d.%s",
615 idx, VTY_NEWLINE);
616 return CMD_WARNING;
617 } else if (trx->cfg.num_chans == idx) { /* creating it */
618 trx->cfg.num_chans++;
619 trx->cfg.chans[idx].trx = trx;
620 trx->cfg.chans[idx].idx = idx;
621 }
622
623 vty->node = CHAN_NODE;
624 vty->index = &trx->cfg.chans[idx];
625
626 return CMD_SUCCESS;
627}
628
629DEFUN(cfg_chan_rx_path, cfg_chan_rx_path_cmd,
630 "rx-path NAME",
631 "Set the Rx Path\n"
632 "Rx Path name\n")
633{
634 struct trx_chan *chan = vty->index;
635
Vadim Yanitskiy4d436842020-09-11 18:31:08 +0700636 if (chan->trx->cfg.multi_arfcn && chan->idx > 0) {
637 vty_out(vty, "%% Setting 'rx-path' for chan %u in multi-ARFCN mode "
638 "does not make sense, because only chan 0 is used%s",
639 chan->idx, VTY_NEWLINE);
640 }
641
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100642 osmo_talloc_replace_string(chan->trx, &chan->rx_path, argv[0]);
643
644 return CMD_SUCCESS;
645}
646
647DEFUN(cfg_chan_tx_path, cfg_chan_tx_path_cmd,
648 "tx-path NAME",
649 "Set the Tx Path\n"
650 "Tx Path name\n")
651{
652 struct trx_chan *chan = vty->index;
653
Vadim Yanitskiy4d436842020-09-11 18:31:08 +0700654 if (chan->trx->cfg.multi_arfcn && chan->idx > 0) {
655 vty_out(vty, "%% Setting 'tx-path' for chan %u in multi-ARFCN mode "
656 "does not make sense, because only chan 0 is used%s",
657 chan->idx, VTY_NEWLINE);
658 }
659
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100660 osmo_talloc_replace_string(chan->trx, &chan->tx_path, argv[0]);
661
662 return CMD_SUCCESS;
663}
664
665static int dummy_config_write(struct vty *v)
666{
667 return CMD_SUCCESS;
668}
669
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100670static int config_write_trx(struct vty *vty)
671{
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100672 struct trx_chan *chan;
673 int i;
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100674 struct trx_ctx *trx = trx_from_vty(vty);
675
676 vty_out(vty, "trx%s", VTY_NEWLINE);
677 if (trx->cfg.bind_addr)
678 vty_out(vty, " bind-ip %s%s", trx->cfg.bind_addr, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100679 if (trx->cfg.remote_addr)
680 vty_out(vty, " remote-ip %s%s", trx->cfg.remote_addr, VTY_NEWLINE);
681 if (trx->cfg.base_port != DEFAULT_TRX_PORT)
682 vty_out(vty, " base-port %u%s", trx->cfg.base_port, VTY_NEWLINE);
Eric19e134a2023-05-10 23:50:38 +0200683 if (strlen(trx->cfg.dev_args))
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100684 vty_out(vty, " dev-args %s%s", trx->cfg.dev_args, VTY_NEWLINE);
685 if (trx->cfg.tx_sps != DEFAULT_TX_SPS)
686 vty_out(vty, " tx-sps %u%s", trx->cfg.tx_sps, VTY_NEWLINE);
687 if (trx->cfg.rx_sps != DEFAULT_RX_SPS)
688 vty_out(vty, " rx-sps %u%s", trx->cfg.rx_sps, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100689 if (trx->cfg.clock_ref != REF_INTERNAL)
690 vty_out(vty, " clock-ref %s%s", get_value_string(clock_ref_names, trx->cfg.clock_ref), VTY_NEWLINE);
691 vty_out(vty, " multi-arfcn %s%s", trx->cfg.multi_arfcn ? "enable" : "disable", VTY_NEWLINE);
692 if (trx->cfg.offset != 0)
693 vty_out(vty, " offset %f%s", trx->cfg.offset, VTY_NEWLINE);
Vadim Yanitskiy3b8f7c42018-08-23 13:41:06 +0700694 if (trx->cfg.freq_offset_khz != 0)
695 vty_out(vty, " freq-offset %f%s", trx->cfg.freq_offset_khz, VTY_NEWLINE);
Pau Espin Pedrole91544d2020-10-13 17:03:37 +0200696 if (!(trx->cfg.rssi_offset == 0 && !trx->cfg.force_rssi_offset))
697 vty_out(vty, " rssi-offset %f%s%s", trx->cfg.rssi_offset,
698 trx->cfg.force_rssi_offset ? " relative": "", VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100699 vty_out(vty, " swap-channels %s%s", trx->cfg.swap_channels ? "enable" : "disable", VTY_NEWLINE);
700 vty_out(vty, " egprs %s%s", trx->cfg.egprs ? "enable" : "disable", VTY_NEWLINE);
Vadim Yanitskiya8b35652018-10-22 02:52:18 +0200701 vty_out(vty, " ext-rach %s%s", trx->cfg.ext_rach ? "enable" : "disable", VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100702 if (trx->cfg.sched_rr != 0)
703 vty_out(vty, " rt-prio %u%s", trx->cfg.sched_rr, VTY_NEWLINE);
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300704 if (trx->cfg.filler != FILLER_ZERO)
705 vty_out(vty, " filler type %s%s", get_value_string(filler_types, trx->cfg.filler), VTY_NEWLINE);
706 if (trx->cfg.rtsc > 0)
707 vty_out(vty, " filler tsc %u%s", trx->cfg.rtsc, VTY_NEWLINE);
708 if (trx->cfg.rach_delay > 0)
709 vty_out(vty, " filler access-burst-delay %u%s", trx->cfg.rach_delay, VTY_NEWLINE);
Eric Wildac0487e2019-06-17 13:02:44 +0200710 if (trx->cfg.stack_size != 0)
711 vty_out(vty, " stack-size %u%s", trx->cfg.stack_size, VTY_NEWLINE);
Ericc27fe602021-05-05 17:33:18 +0200712 if (trx->cfg.ul_fn_offset != 0)
Eric0c34c642021-06-16 15:20:51 +0200713 vty_out(vty, " ul-fn-offset %d%s", trx->cfg.ul_fn_offset, VTY_NEWLINE);
Eric Wildf8c7a522023-05-17 17:44:08 +0200714 if (trx->cfg.overrides.dl_freq_override)
715 vty_out(vty, " dl-freq-override %f%s", trx->cfg.overrides.dl_freq, VTY_NEWLINE);
716 if (trx->cfg.overrides.ul_freq_override)
717 vty_out(vty, " ul-freq-override %f%s", trx->cfg.overrides.ul_freq, VTY_NEWLINE);
718 if (trx->cfg.overrides.dl_gain_override)
719 vty_out(vty, " dl-gain-override %f%s", trx->cfg.overrides.dl_gain, VTY_NEWLINE);
720 if (trx->cfg.overrides.ul_gain_override)
721 vty_out(vty, " ul-gain-override %f%s", trx->cfg.overrides.ul_gain, VTY_NEWLINE);
Ericcca5d932023-07-25 18:44:03 +0200722 if (trx->cfg.use_va)
723 vty_out(vty, " viterbi-eq %s%s", trx->cfg.use_va ? "enable" : "disable", VTY_NEWLINE);
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200724 trx_rate_ctr_threshold_write_config(vty, " ");
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100725
726 for (i = 0; i < trx->cfg.num_chans; i++) {
727 chan = &trx->cfg.chans[i];
728 vty_out(vty, " chan %u%s", chan->idx, VTY_NEWLINE);
729 if (chan->rx_path)
Harald Welte03b3c302018-07-31 15:48:18 +0200730 vty_out(vty, " rx-path %s%s", chan->rx_path, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100731 if (chan->tx_path)
Harald Welte03b3c302018-07-31 15:48:18 +0200732 vty_out(vty, " tx-path %s%s", chan->tx_path, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100733 }
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100734
735 return CMD_SUCCESS;
736}
737
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100738static void trx_dump_vty(struct vty *vty, struct trx_ctx *trx)
739{
740 struct trx_chan *chan;
741 int i;
742 vty_out(vty, "TRX Config:%s", VTY_NEWLINE);
743 vty_out(vty, " Local IP: %s%s", trx->cfg.bind_addr, VTY_NEWLINE);
744 vty_out(vty, " Remote IP: %s%s", trx->cfg.remote_addr, VTY_NEWLINE);
745 vty_out(vty, " TRX Base Port: %u%s", trx->cfg.base_port, VTY_NEWLINE);
746 vty_out(vty, " Device args: %s%s", trx->cfg.dev_args, VTY_NEWLINE);
747 vty_out(vty, " Tx Samples-per-Symbol: %u%s", trx->cfg.tx_sps, VTY_NEWLINE);
748 vty_out(vty, " Rx Samples-per-Symbol: %u%s", trx->cfg.rx_sps, VTY_NEWLINE);
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300749 vty_out(vty, " Filler Burst Type: %s%s", get_value_string(filler_names, trx->cfg.filler), VTY_NEWLINE);
750 vty_out(vty, " Filler Burst TSC: %u%s", trx->cfg.rtsc, VTY_NEWLINE);
751 vty_out(vty, " Filler Burst RACH Delay: %u%s", trx->cfg.rach_delay, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100752 vty_out(vty, " Clock Reference: %s%s", get_value_string(clock_ref_names, trx->cfg.clock_ref), VTY_NEWLINE);
753 vty_out(vty, " Multi-Carrier: %s%s", trx->cfg.multi_arfcn ? "Enabled" : "Disabled", VTY_NEWLINE);
754 vty_out(vty, " Tuning offset: %f%s", trx->cfg.offset, VTY_NEWLINE);
755 vty_out(vty, " RSSI to dBm offset: %f%s", trx->cfg.rssi_offset, VTY_NEWLINE);
756 vty_out(vty, " Swap channels: %s%s", trx->cfg.swap_channels ? "Enabled" : "Disabled", VTY_NEWLINE);
757 vty_out(vty, " EDGE support: %s%s", trx->cfg.egprs ? "Enabled" : "Disabled", VTY_NEWLINE);
Vadim Yanitskiya8b35652018-10-22 02:52:18 +0200758 vty_out(vty, " Extended RACH support: %s%s", trx->cfg.ext_rach ? "Enabled" : "Disabled", VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100759 vty_out(vty, " Real Time Priority: %u (%s)%s", trx->cfg.sched_rr,
760 trx->cfg.sched_rr ? "Enabled" : "Disabled", VTY_NEWLINE);
Eric Wildac0487e2019-06-17 13:02:44 +0200761 vty_out(vty, " Stack size per Thread in BYTE (0 = OS default): %u%s", trx->cfg.stack_size, VTY_NEWLINE);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100762 vty_out(vty, " Channels: %u%s", trx->cfg.num_chans, VTY_NEWLINE);
763 for (i = 0; i < trx->cfg.num_chans; i++) {
764 chan = &trx->cfg.chans[i];
765 vty_out(vty, " Channel %u:%s", chan->idx, VTY_NEWLINE);
766 if (chan->rx_path)
767 vty_out(vty, " Rx Path: %s%s", chan->rx_path, VTY_NEWLINE);
768 if (chan->tx_path)
769 vty_out(vty, " Tx Path: %s%s", chan->tx_path, VTY_NEWLINE);
770 }
771}
772
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100773DEFUN(show_trx, show_trx_cmd,
774 "show trx",
775 SHOW_STR "Display information on the TRX\n")
776{
777 struct trx_ctx *trx = trx_from_vty(vty);
778
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100779 trx_dump_vty(vty, trx);
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100780
781 return CMD_SUCCESS;
782}
783
784static int trx_vty_is_config_node(struct vty *vty, int node)
785{
786 switch (node) {
787 case TRX_NODE:
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100788 case CHAN_NODE:
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100789 return 1;
790 default:
791 return 0;
792 }
793}
794
795static int trx_vty_go_parent(struct vty *vty)
796{
797 switch (vty->node) {
798 case TRX_NODE:
799 vty->node = CONFIG_NODE;
800 vty->index = NULL;
801 vty->index_sub = NULL;
802 break;
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100803 case CHAN_NODE:
804 vty->node = TRX_NODE;
805 vty->index = NULL;
806 vty->index_sub = NULL;
807 break;
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100808 default:
Vadim Yanitskiy01eea0a2018-05-09 15:19:48 +0700809 vty->node = CONFIG_NODE;
810 vty->index = NULL;
811 vty->index_sub = NULL;
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100812 }
813
814 return vty->node;
815}
816
817static const char trx_copyright[] =
818 "Copyright (C) 2007-2014 Free Software Foundation, Inc.\r\n"
819 "Copyright (C) 2013 Thomas Tsou <tom@tsou.cc>\r\n"
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300820 "Copyright (C) 2013-2019 Fairwaves, Inc.\r\n"
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100821 "Copyright (C) 2015 Ettus Research LLC\r\n"
822 "Copyright (C) 2017-2018 by sysmocom s.f.m.c. GmbH <info@sysmocom.de>\r\n"
823 "License AGPLv3+: GNU AGPL version 3 or later <http://gnu.org/licenses/agpl-3.0.html>\r\n"
824 "This is free software: you are free to change and redistribute it.\r\n"
825 "There is NO WARRANTY, to the extent permitted by law.\r\n";
826
827struct vty_app_info g_vty_info = {
828 .name = "OsmoTRX",
829 .version = PACKAGE_VERSION,
830 .copyright = trx_copyright,
831 .go_parent_cb = trx_vty_go_parent,
832 .is_config_node = trx_vty_is_config_node,
833};
834
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100835struct trx_ctx *vty_trx_ctx_alloc(void *talloc_ctx)
836{
837 struct trx_ctx * trx = talloc_zero(talloc_ctx, struct trx_ctx);
838
839 trx->cfg.bind_addr = talloc_strdup(trx, DEFAULT_TRX_IP);
840 trx->cfg.remote_addr = talloc_strdup(trx, DEFAULT_TRX_IP);
841 trx->cfg.base_port = DEFAULT_TRX_PORT;
842 trx->cfg.tx_sps = DEFAULT_TX_SPS;
843 trx->cfg.rx_sps = DEFAULT_RX_SPS;
844 trx->cfg.filler = FILLER_ZERO;
Pau Espin Pedrole91544d2020-10-13 17:03:37 +0200845 trx->cfg.rssi_offset = 0.0f;
Eric8c1d5902023-08-24 11:42:57 +0200846 trx->cfg.dev_args = talloc_strdup(trx, "");
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100847
848 return trx;
849}
850
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100851int trx_vty_init(struct trx_ctx* trx)
852{
Vadim Yanitskiyc0e7ce92020-10-30 02:19:55 +0700853 cfg_filler_type_cmd.string = vty_cmd_string_from_valstr(trx, filler_types,
854 "filler type (", "|", ")", 0);
855 cfg_filler_type_cmd.doc = vty_cmd_string_from_valstr(trx, filler_docs,
856 CFG_FILLER_DOC_STR "What to do when there is nothing to send "
857 "(filler type, default=zero)\n", "\n", "", 0);
858
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100859 g_trx_ctx = trx;
860 install_element_ve(&show_trx_cmd);
861
862 install_element(CONFIG_NODE, &cfg_trx_cmd);
863
864 install_node(&trx_node, config_write_trx);
865 install_element(TRX_NODE, &cfg_bind_ip_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100866 install_element(TRX_NODE, &cfg_remote_ip_cmd);
867 install_element(TRX_NODE, &cfg_base_port_cmd);
868 install_element(TRX_NODE, &cfg_dev_args_cmd);
869 install_element(TRX_NODE, &cfg_tx_sps_cmd);
870 install_element(TRX_NODE, &cfg_rx_sps_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100871 install_element(TRX_NODE, &cfg_clock_ref_cmd);
872 install_element(TRX_NODE, &cfg_multi_arfcn_cmd);
873 install_element(TRX_NODE, &cfg_offset_cmd);
Vadim Yanitskiy3b8f7c42018-08-23 13:41:06 +0700874 install_element(TRX_NODE, &cfg_freq_offset_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100875 install_element(TRX_NODE, &cfg_rssi_offset_cmd);
876 install_element(TRX_NODE, &cfg_swap_channels_cmd);
877 install_element(TRX_NODE, &cfg_egprs_cmd);
Vadim Yanitskiya8b35652018-10-22 02:52:18 +0200878 install_element(TRX_NODE, &cfg_ext_rach_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100879 install_element(TRX_NODE, &cfg_rt_prio_cmd);
Alexander Chemeris9a87d902019-10-15 00:33:07 +0300880 install_element(TRX_NODE, &cfg_filler_type_cmd);
881 install_element(TRX_NODE, &cfg_filler_tsc_cmd);
882 install_element(TRX_NODE, &cfg_filler_rach_delay_cmd);
Pau Espin Pedrol6a305fe2019-05-24 19:58:20 +0200883 install_element(TRX_NODE, &cfg_ctr_error_threshold_cmd);
884 install_element(TRX_NODE, &cfg_no_ctr_error_threshold_cmd);
Eric Wildac0487e2019-06-17 13:02:44 +0200885 install_element(TRX_NODE, &cfg_stack_size_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100886
887 install_element(TRX_NODE, &cfg_chan_cmd);
Ericc27fe602021-05-05 17:33:18 +0200888 install_element(TRX_NODE, &cfg_ul_fn_offset_cmd);
Eric Wildf8c7a522023-05-17 17:44:08 +0200889 install_element(TRX_NODE, &cfg_ul_freq_override_cmd);
890 install_element(TRX_NODE, &cfg_dl_freq_override_cmd);
891 install_element(TRX_NODE, &cfg_ul_gain_override_cmd);
892 install_element(TRX_NODE, &cfg_dl_gain_override_cmd);
Ericcca5d932023-07-25 18:44:03 +0200893 install_element(TRX_NODE, &cfg_use_viterbi_cmd);
Pau Espin Pedrola3ab8c22018-02-21 15:41:03 +0100894 install_node(&chan_node, dummy_config_write);
895 install_element(CHAN_NODE, &cfg_chan_rx_path_cmd);
896 install_element(CHAN_NODE, &cfg_chan_tx_path_cmd);
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100897
Pau Espin Pedrolaebbfe02020-01-02 16:39:11 +0100898 logging_vty_add_deprecated_subsys(g_trx_ctx, "lms");
899
Pau Espin Pedrol5ea18172018-02-20 16:48:15 +0100900 return 0;
901}