blob: eb8f4d56f4d8c2a7295c0cee78c505779ff05487 [file] [log] [blame]
Harald Weltefc61a3f2011-02-12 14:44:16 +01001/* VTY interface for A-bis OM2000 */
2
3/* (C) 2010-2011 by Harald Welte <laforge@gnumonks.org>
4 *
5 * All Rights Reserved
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU Affero General Public License as published by
9 * the Free Software Foundation; either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU Affero General Public License for more details.
16 *
17 * You should have received a copy of the GNU Affero General Public License
18 * along with this program. If not, see <http://www.gnu.org/licenses/>.
19 *
20 */
21
22#include <stdlib.h>
23#include <unistd.h>
24#include <errno.h>
25#include <stdint.h>
26
27#include <arpa/inet.h>
28
29#include <openbsc/gsm_data.h>
Pablo Neira Ayusodd5fff42011-03-22 16:47:59 +010030#include <osmocom/core/msgb.h>
31#include <osmocom/gsm/tlv.h>
32#include <osmocom/core/talloc.h>
Harald Weltefc61a3f2011-02-12 14:44:16 +010033#include <openbsc/debug.h>
34#include <openbsc/signal.h>
35#include <openbsc/abis_om2000.h>
36#include <openbsc/vty.h>
37
38#include <osmocom/vty/vty.h>
39#include <osmocom/vty/command.h>
40#include <osmocom/vty/logging.h>
41#include <osmocom/vty/telnet_interface.h>
42
43extern struct gsm_network *bsc_gsmnet;
44
45static struct cmd_node om2k_node = {
46 OM2K_NODE,
Harald Welte60acee22011-02-12 20:50:58 +010047 "%s(om2k)# ",
Harald Weltefc61a3f2011-02-12 14:44:16 +010048 1,
49};
50
51struct oml_node_state {
52 struct gsm_bts *bts;
53 struct abis_om2k_mo mo;
54};
55
56static int dummy_config_write(struct vty *v)
57{
58 return CMD_SUCCESS;
59}
60
61/* FIXME: auto-generate those strings from the value_string lists */
62#define OM2K_OBJCLASS_VTY "(trxc|ts|tf|is|con|dp|cf|tx|rx)"
Harald Welte0f916202011-02-13 22:51:03 +010063#define OM2K_OBJCLASS_VTY_HELP "TRX Controller\n" \
64 "Timeslot\n" \
65 "Timing Function\n" \
66 "Interface Switch\n" \
67 "Abis Concentrator\n" \
68 "Digital Path\n" \
69 "Central Function\n" \
70 "Transmitter\n" \
71 "Receiver\n"
Harald Weltefc61a3f2011-02-12 14:44:16 +010072
73DEFUN(om2k_class_inst, om2k_class_inst_cmd,
74 "bts <0-255> om2000 class " OM2K_OBJCLASS_VTY
75 " <0-255> <0-255> <0-255>",
76 "BTS related commands\n" "BTS Number\n"
77 "Manipulate the OM2000 managed objects\n"
78 "Object Class\n" OM2K_OBJCLASS_VTY_HELP
79 "BTS Number\n" "Associated SO Instance\n" "Instance Number\n")
80{
81 struct gsm_bts *bts;
82 struct oml_node_state *oms;
83 int bts_nr = atoi(argv[0]);
84
85 bts = gsm_bts_num(bsc_gsmnet, bts_nr);
86 if (!bts) {
87 vty_out(vty, "%% No such BTS (%d)%s", bts_nr, VTY_NEWLINE);
88 return CMD_WARNING;
89 }
90
91 if (bts->type != GSM_BTS_TYPE_RBS2000) {
92 vty_out(vty, "%% BTS %d not an Ericsson RBS%s",
93 bts_nr, VTY_NEWLINE);
94 return CMD_WARNING;
95 }
96
97 oms = talloc_zero(tall_bsc_ctx, struct oml_node_state);
98 if (!oms)
99 return CMD_WARNING;
100
101 oms->bts = bts;
102 oms->mo.class = get_string_value(om2k_mo_class_short_vals, argv[1]);
103 oms->mo.bts = atoi(argv[2]);
104 oms->mo.assoc_so = atoi(argv[3]);
105 oms->mo.inst = atoi(argv[4]);
106
107 vty->index = oms;
108 vty->node = OM2K_NODE;
109
110 return CMD_SUCCESS;
111
112}
113
114DEFUN(om2k_classnum_inst, om2k_classnum_inst_cmd,
115 "bts <0-255> om2000 class <0-255> <0-255> <0-255> <0-255>",
116 "BTS related commands\n" "BTS Number\n"
117 "Manipulate the OML managed objects\n"
118 "Object Class\n" "Object Class\n"
119 "BTS Number\n" "Associated SO Instance\n" "Instance Number\n")
120{
121 struct gsm_bts *bts;
122 struct oml_node_state *oms;
123 int bts_nr = atoi(argv[0]);
124
125 bts = gsm_bts_num(bsc_gsmnet, bts_nr);
126 if (!bts) {
127 vty_out(vty, "%% No such BTS (%d)%s", bts_nr, VTY_NEWLINE);
128 return CMD_WARNING;
129 }
130
131 oms = talloc_zero(tall_bsc_ctx, struct oml_node_state);
132 if (!oms)
133 return CMD_WARNING;
134
135 oms->bts = bts;
136 oms->mo.class = atoi(argv[1]);
137 oms->mo.bts = atoi(argv[2]);
138 oms->mo.assoc_so = atoi(argv[3]);
139 oms->mo.inst = atoi(argv[4]);
140
141 vty->index = oms;
142 vty->node = OM2K_NODE;
143
144 return CMD_SUCCESS;
145}
146
147DEFUN(om2k_reset, om2k_reset_cmd,
148 "reset-command",
149 "Reset the MO\n")
150{
151 struct oml_node_state *oms = vty->index;
152
153 abis_om2k_tx_reset_cmd(oms->bts, &oms->mo);
154 return CMD_SUCCESS;
155}
156
157DEFUN(om2k_start, om2k_start_cmd,
158 "start-request",
159 "Start the MO\n")
160{
161 struct oml_node_state *oms = vty->index;
162
163 abis_om2k_tx_start_req(oms->bts, &oms->mo);
164 return CMD_SUCCESS;
165}
166
167DEFUN(om2k_status, om2k_status_cmd,
168 "status-request",
169 "Get the MO Status\n")
170{
171 struct oml_node_state *oms = vty->index;
172
173 abis_om2k_tx_status_req(oms->bts, &oms->mo);
174 return CMD_SUCCESS;
175}
176
Harald Welte9bf8e9e2011-02-12 14:57:17 +0100177DEFUN(om2k_connect, om2k_connect_cmd,
178 "connect-command",
179 "Connect the MO\n")
180{
181 struct oml_node_state *oms = vty->index;
182
183 abis_om2k_tx_connect_cmd(oms->bts, &oms->mo);
184 return CMD_SUCCESS;
185}
186
187DEFUN(om2k_disconnect, om2k_disconnect_cmd,
188 "disconnect-command",
189 "Disconnect the MO\n")
190{
191 struct oml_node_state *oms = vty->index;
192
193 abis_om2k_tx_disconnect_cmd(oms->bts, &oms->mo);
194 return CMD_SUCCESS;
195}
196
Harald Welte6d5a0392011-02-12 18:48:53 +0100197DEFUN(om2k_enable, om2k_enable_cmd,
198 "enable-request",
199 "Enable the MO\n")
200{
201 struct oml_node_state *oms = vty->index;
202
203 abis_om2k_tx_enable_req(oms->bts, &oms->mo);
204 return CMD_SUCCESS;
205}
206
207DEFUN(om2k_disable, om2k_disable_cmd,
208 "disable-request",
209 "Disable the MO\n")
210{
211 struct oml_node_state *oms = vty->index;
212
213 abis_om2k_tx_disable_req(oms->bts, &oms->mo);
214 return CMD_SUCCESS;
215}
216
Harald Welte9bf8e9e2011-02-12 14:57:17 +0100217DEFUN(om2k_op_info, om2k_op_info_cmd,
218 "operational-info <0-1>",
Harald Welte557f3992012-08-16 23:23:50 +0200219 "Set operational information\n"
Harald Weltefa2015f2012-08-17 09:52:03 +0200220 "Set operational info to 0 or 1\n")
Harald Welte9bf8e9e2011-02-12 14:57:17 +0100221{
222 struct oml_node_state *oms = vty->index;
223 int oper = atoi(argv[0]);
224
225 abis_om2k_tx_op_info(oms->bts, &oms->mo, oper);
226 return CMD_SUCCESS;
227}
228
Harald Welte40ad3b32011-02-12 15:07:30 +0100229DEFUN(om2k_test, om2k_test_cmd,
230 "test-request",
231 "Test the MO\n")
232{
233 struct oml_node_state *oms = vty->index;
234
235 abis_om2k_tx_test_req(oms->bts, &oms->mo);
236 return CMD_SUCCESS;
237}
238
Harald Weltea5526552011-02-13 22:45:02 +0100239struct con_conn_group {
240 struct llist_head list;
241
242 uint8_t cg;
243 uint16_t ccp;
244 uint8_t tag;
245 uint8_t tei;
246};
247
248static void add_con_list(struct gsm_bts *bts, uint8_t cg, uint16_t ccp,
249 uint8_t tag, uint8_t tei)
250{
251 struct con_conn_group *ent = talloc_zero(bts, struct con_conn_group);
252
253 ent->cg = cg;
254 ent->ccp = ccp;
255 ent->tag = tag;
256 ent->tei = tei;
257
Harald Weltec72ba482011-02-14 16:32:44 +0100258 llist_add_tail(&ent->list, &bts->rbs2000.con.conn_groups);
Harald Weltea5526552011-02-13 22:45:02 +0100259}
260
261static int del_con_list(struct gsm_bts *bts, uint8_t cg, uint16_t ccp,
262 uint8_t tag, uint8_t tei)
263{
264 struct con_conn_group *grp, *grp2;
265
266 llist_for_each_entry_safe(grp, grp2, &bts->rbs2000.con.conn_groups, list) {
267 if (grp->cg == cg && grp->ccp == ccp && grp->tag == tag
268 && grp->tei == tei) {
269 llist_del(&grp->list);
270 talloc_free(grp);
271 return 0;
272 }
273 }
274 return -ENOENT;
275}
276
277#define CON_LIST_HELP "CON connetiton list\n" \
278 "Add entry to CON list\n" \
279 "Delete entry from CON list\n" \
280 "Connection Group Number\n" \
281 "CON Connection Point\n" \
282
283DEFUN(om2k_con_list_dec, om2k_con_list_dec_cmd,
284 "con-connection-list (add|del) <1-255> <0-1023> deconcentrated",
285 CON_LIST_HELP "De-concentrated in/outlet\n")
286{
287 struct oml_node_state *oms = vty->index;
288 struct gsm_bts *bts = oms->bts;
289 uint8_t cg = atoi(argv[1]);
290 uint16_t ccp = atoi(argv[2]);
291
292 if (!strcmp(argv[0], "add"))
293 add_con_list(bts, cg, ccp, 0, 0xff);
294 else {
295 if (del_con_list(bts, cg, ccp, 0, 0xff) < 0) {
296 vty_out(vty, "%% No matching CON list entry%s",
297 VTY_NEWLINE);
298 return CMD_WARNING;
299 }
300 }
301
302 return CMD_SUCCESS;
303}
304
305DEFUN(om2k_con_list_tei, om2k_con_list_tei_cmd,
306 "con-connection-list (add|del) <1-255> <0-1023> tei <0-63>",
307 CON_LIST_HELP "Concentrated in/outlet with TEI\n" "TEI Number\n")
308{
309 struct oml_node_state *oms = vty->index;
310 struct gsm_bts *bts = oms->bts;
311 uint8_t cg = atoi(argv[1]);
312 uint16_t ccp = atoi(argv[2]);
313 uint8_t tei = atoi(argv[3]);
314
315 if (!strcmp(argv[0], "add"))
316 add_con_list(bts, cg, ccp, cg, tei);
317 else {
318 if (del_con_list(bts, cg, ccp, cg, tei) < 0) {
319 vty_out(vty, "%% No matching CON list entry%s",
320 VTY_NEWLINE);
321 return CMD_WARNING;
322 }
323 }
324
325 return CMD_SUCCESS;
326}
327
Harald Weltebf7a5342011-02-14 16:17:49 +0100328DEFUN(cfg_bts_is_conn_list, cfg_bts_is_conn_list_cmd,
Harald Welte738310b2011-02-13 22:13:28 +0100329 "is-connection-list (add|del) <0-2047> <0-2047> <0-255>",
330 "Interface Switch Connnection List\n"
331 "Add to IS list\n" "Delete from IS list\n"
332 "ICP1\n" "ICP2\n" "Contiguity Index\n")
Harald Welte4a8f22e2011-02-12 20:23:40 +0100333{
Harald Weltebf7a5342011-02-14 16:17:49 +0100334 struct gsm_bts *bts = vty->index;
Harald Welte738310b2011-02-13 22:13:28 +0100335 uint16_t icp1 = atoi(argv[1]);
336 uint16_t icp2 = atoi(argv[2]);
337 uint8_t ci = atoi(argv[3]);
338 struct is_conn_group *grp, *grp2;
Harald Welte4a8f22e2011-02-12 20:23:40 +0100339
Harald Welte738310b2011-02-13 22:13:28 +0100340 if (!strcmp(argv[0], "add")) {
341 grp = talloc_zero(bts, struct is_conn_group);
342 grp->icp1 = icp1;
343 grp->icp2 = icp2;
344 grp->ci = ci;
Harald Weltec72ba482011-02-14 16:32:44 +0100345 llist_add_tail(&grp->list, &bts->rbs2000.is.conn_groups);
Harald Welte738310b2011-02-13 22:13:28 +0100346 } else {
347 llist_for_each_entry_safe(grp, grp2, &bts->rbs2000.is.conn_groups, list) {
348 if (grp->icp1 == icp1 && grp->icp2 == icp2
349 && grp->ci == ci) {
350 llist_del(&grp->list);
351 talloc_free(grp);
352 return CMD_SUCCESS;
353 }
354 }
355 vty_out(vty, "%% No matching IS Conn Group found!%s",
356 VTY_NEWLINE);
357 return CMD_WARNING;
358 }
Harald Welte4a8f22e2011-02-12 20:23:40 +0100359
Harald Welte738310b2011-02-13 22:13:28 +0100360 return CMD_SUCCESS;
361}
362
363
Harald Welte00585d92011-03-05 14:13:14 +0100364DEFUN(om2k_conf_req, om2k_conf_req_cmd,
365 "configuration-request",
366 "Send the configuration request for current MO\n")
367{
368 struct oml_node_state *oms = vty->index;
369 struct gsm_bts *bts = oms->bts;
370 struct gsm_bts_trx *trx = NULL;
371 struct gsm_bts_trx_ts *ts = NULL;
372
373 switch (oms->mo.class) {
Harald Welte2f53e152011-03-06 19:46:15 +0100374 case OM2K_MO_CLS_IS:
Harald Welte3ee11892011-03-06 21:49:21 +0100375 abis_om2k_tx_is_conf_req(bts);
Harald Welte2f53e152011-03-06 19:46:15 +0100376 break;
Harald Welte00585d92011-03-05 14:13:14 +0100377 case OM2K_MO_CLS_TS:
378 trx = gsm_bts_trx_by_nr(bts, oms->mo.assoc_so);
379 if (!trx) {
380 vty_out(vty, "%% BTS %u has no TRX %u%s", bts->nr,
381 oms->mo.assoc_so, VTY_NEWLINE);
382 return CMD_WARNING;
383 }
384 if (oms->mo.inst >= ARRAY_SIZE(trx->ts)) {
385 vty_out(vty, "%% Timeslot %u out of range%s",
386 oms->mo.inst, VTY_NEWLINE);
387 return CMD_WARNING;
388 }
389 ts = &trx->ts[oms->mo.inst];
390 abis_om2k_tx_ts_conf_req(ts);
391 break;
392 case OM2K_MO_CLS_RX:
393 case OM2K_MO_CLS_TX:
394 case OM2K_MO_CLS_TRXC:
395 trx = gsm_bts_trx_by_nr(bts, oms->mo.inst);
396 if (!trx) {
397 vty_out(vty, "%% BTS %u has no TRX %u%s", bts->nr,
398 oms->mo.inst, VTY_NEWLINE);
399 return CMD_WARNING;
400 }
401 switch (oms->mo.class) {
402 case OM2K_MO_CLS_RX:
403 abis_om2k_tx_rx_conf_req(trx);
404 break;
405 case OM2K_MO_CLS_TX:
Harald Weltef69c0582011-03-05 16:14:34 +0100406 abis_om2k_tx_tx_conf_req(trx);
Harald Welte00585d92011-03-05 14:13:14 +0100407 break;
408 default:
409 break;
410 }
411 break;
Harald Welte429bed02011-03-05 14:36:47 +0100412 case OM2K_MO_CLS_TF:
413 abis_om2k_tx_tf_conf_req(bts);
414 break;
Harald Welte00585d92011-03-05 14:13:14 +0100415 default:
416 vty_out(vty, "%% Don't know how to configure MO%s",
417 VTY_NEWLINE);
418 }
419
420 return CMD_SUCCESS;
421}
422
Harald Weltebf7a5342011-02-14 16:17:49 +0100423void abis_om2k_config_write_bts(struct vty *vty, struct gsm_bts *bts)
424{
425 struct is_conn_group *igrp;
426 struct con_conn_group *cgrp;
427
428 llist_for_each_entry(igrp, &bts->rbs2000.is.conn_groups, list)
429 vty_out(vty, " is-connection-list add %u %u %u%s",
430 igrp->icp1, igrp->icp2, igrp->ci, VTY_NEWLINE);
431
432 llist_for_each_entry(cgrp, &bts->rbs2000.con.conn_groups, list) {
433 vty_out(vty, " con-connection-list add %u %u ",
434 cgrp->cg, cgrp->ccp);
435 if (cgrp->tei == 0xff)
436 vty_out(vty, "deconcentrated%s", VTY_NEWLINE);
437 else
438 vty_out(vty, "tei %u%s", cgrp->tei, VTY_NEWLINE);
439 }
440}
441
Harald Weltefc61a3f2011-02-12 14:44:16 +0100442int abis_om2k_vty_init(void)
443{
444 install_element(ENABLE_NODE, &om2k_class_inst_cmd);
445 install_element(ENABLE_NODE, &om2k_classnum_inst_cmd);
446 install_node(&om2k_node, dummy_config_write);
447
Jacob Erlbeckbf8eec72013-09-02 13:17:16 +0200448 bsc_install_default(OM2K_NODE);
Harald Weltefc61a3f2011-02-12 14:44:16 +0100449 install_element(OM2K_NODE, &om2k_reset_cmd);
450 install_element(OM2K_NODE, &om2k_start_cmd);
451 install_element(OM2K_NODE, &om2k_status_cmd);
Harald Welte9bf8e9e2011-02-12 14:57:17 +0100452 install_element(OM2K_NODE, &om2k_connect_cmd);
453 install_element(OM2K_NODE, &om2k_disconnect_cmd);
Harald Welte6d5a0392011-02-12 18:48:53 +0100454 install_element(OM2K_NODE, &om2k_enable_cmd);
455 install_element(OM2K_NODE, &om2k_disable_cmd);
Harald Welte9bf8e9e2011-02-12 14:57:17 +0100456 install_element(OM2K_NODE, &om2k_op_info_cmd);
Harald Welte40ad3b32011-02-12 15:07:30 +0100457 install_element(OM2K_NODE, &om2k_test_cmd);
Harald Welte00585d92011-03-05 14:13:14 +0100458 install_element(OM2K_NODE, &om2k_conf_req_cmd);
Harald Weltea5526552011-02-13 22:45:02 +0100459 install_element(OM2K_NODE, &om2k_con_list_dec_cmd);
460 install_element(OM2K_NODE, &om2k_con_list_tei_cmd);
Harald Weltefc61a3f2011-02-12 14:44:16 +0100461
Harald Weltebf7a5342011-02-14 16:17:49 +0100462 install_element(BTS_NODE, &cfg_bts_is_conn_list_cmd);
463
Harald Weltefc61a3f2011-02-12 14:44:16 +0100464 return 0;
465}