blob: 4cc08c2daf72de702d6a5dca2ecf78caff6cad3e [file] [log] [blame]
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +02001/* OpenBSC interface to quagga VTY */
2/* (C) 2009 by Harald Welte <laforge@gnumonks.org>
Holger Hans Peter Freythere33966c2009-10-27 12:47:06 +01003 * (C) 2009 by Holger Hans Peter Freyther
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +02004 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 *
20 */
21
22#include <stdlib.h>
23#include <unistd.h>
24#include <sys/types.h>
25
26#include <vty/command.h>
27#include <vty/buffer.h>
28#include <vty/vty.h>
29
30#include <arpa/inet.h>
31
32#include <openbsc/linuxlist.h>
33#include <openbsc/gsm_data.h>
34#include <openbsc/gsm_subscriber.h>
Harald Welteb54d9502009-11-17 06:00:23 +010035#include <openbsc/silent_call.h>
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +020036#include <openbsc/gsm_04_11.h>
37#include <openbsc/e1_input.h>
38#include <openbsc/abis_nm.h>
39#include <openbsc/gsm_utils.h>
40#include <openbsc/db.h>
41#include <openbsc/talloc.h>
Harald Weltea1482332009-11-14 10:08:40 +010042#include <openbsc/signal.h>
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +020043
44/* forward declarations */
45void subscr_dump_vty(struct vty *vty, struct gsm_subscriber *subscr);
46
47static struct gsm_network *gsmnet;
48
49struct cmd_node subscr_node = {
50 SUBSCR_NODE,
51 "%s(subscriber)#",
52 1,
53};
54
55static int dummy_config_write(struct vty *v)
56{
57 return CMD_SUCCESS;
58}
59
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +020060static struct buffer *argv_to_buffer(int argc, const char *argv[], int base)
61{
62 struct buffer *b = buffer_new(1024);
63 int i;
64
65 if (!b)
66 return NULL;
67
68 for (i = base; i < argc; i++) {
69 buffer_putstr(b, argv[i]);
70 buffer_putc(b, ' ');
71 }
72 buffer_putc(b, '\0');
73
74 return b;
75}
76
77/* per-subscriber configuration */
78DEFUN(cfg_subscr,
79 cfg_subscr_cmd,
80 "subscriber IMSI",
81 "Select a Subscriber to configure\n")
82{
83 const char *imsi = argv[0];
84 struct gsm_subscriber *subscr;
85
86 subscr = subscr_get_by_imsi(gsmnet, imsi);
87 if (!subscr) {
88 vty_out(vty, "%% No subscriber for IMSI %s%s",
89 imsi, VTY_NEWLINE);
90 return CMD_WARNING;
91 }
92
Holger Hans Peter Freythere33966c2009-10-27 12:47:06 +010093 /* vty_go_parent should put this subscriber */
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +020094 vty->index = subscr;
95 vty->node = SUBSCR_NODE;
96
97 return CMD_SUCCESS;
98}
99
100/* Subscriber */
101DEFUN(show_subscr,
102 show_subscr_cmd,
103 "show subscriber [IMSI]",
104 SHOW_STR "Display information about a subscriber\n")
105{
106 const char *imsi;
107 struct gsm_subscriber *subscr;
108
109 if (argc >= 1) {
110 imsi = argv[0];
111 subscr = subscr_get_by_imsi(gsmnet, imsi);
112 if (!subscr) {
113 vty_out(vty, "%% unknown subscriber%s",
114 VTY_NEWLINE);
115 return CMD_WARNING;
116 }
117 subscr_dump_vty(vty, subscr);
Holger Hans Peter Freythere33966c2009-10-27 12:47:06 +0100118 subscr_put(subscr);
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200119
120 return CMD_SUCCESS;
121 }
122
123 /* FIXME: iterate over all subscribers ? */
124 return CMD_WARNING;
125
126 return CMD_SUCCESS;
127}
128
129DEFUN(show_subscr_cache,
130 show_subscr_cache_cmd,
131 "show subscriber cache",
132 SHOW_STR "Display contents of subscriber cache\n")
133{
134 struct gsm_subscriber *subscr;
135
136 llist_for_each_entry(subscr, &active_subscribers, entry) {
137 vty_out(vty, " Subscriber:%s", VTY_NEWLINE);
138 subscr_dump_vty(vty, subscr);
139 }
140
141 return CMD_SUCCESS;
142}
143
144DEFUN(sms_send_pend,
145 sms_send_pend_cmd,
146 "sms send pending MIN_ID",
147 "Send all pending SMS starting from MIN_ID")
148{
149 struct gsm_sms *sms;
150 int id = atoi(argv[0]);
151
152 while (1) {
153 sms = db_sms_get_unsent(gsmnet, id++);
154 if (!sms)
155 return CMD_WARNING;
156
157 if (!sms->receiver) {
158 sms_free(sms);
159 continue;
160 }
161
162 gsm411_send_sms_subscr(sms->receiver, sms);
163 }
164
165 return CMD_SUCCESS;
166}
167
168struct gsm_sms *sms_from_text(struct gsm_subscriber *receiver, const char *text)
169{
170 struct gsm_sms *sms = sms_alloc();
171
172 if (!sms)
173 return NULL;
174
175 if (!receiver->lac) {
176 /* subscriber currently not attached, store in database? */
177 return NULL;
178 }
179
180 sms->receiver = subscr_get(receiver);
181 strncpy(sms->text, text, sizeof(sms->text)-1);
182
183 /* FIXME: don't use ID 1 static */
184 sms->sender = subscr_get_by_id(gsmnet, 1);
185 sms->reply_path_req = 0;
186 sms->status_rep_req = 0;
187 sms->ud_hdr_ind = 0;
188 sms->protocol_id = 0; /* implicit */
189 sms->data_coding_scheme = 0; /* default 7bit */
190 strncpy(sms->dest_addr, receiver->extension, sizeof(sms->dest_addr)-1);
191 /* Generate user_data */
192 sms->user_data_len = gsm_7bit_encode(sms->user_data, sms->text);
193
194 return sms;
195}
196
197static int _send_sms_buffer(struct gsm_subscriber *receiver,
Harald Welte793a1352009-11-05 15:51:17 +0900198 struct buffer *b, u_int8_t tp_pid)
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200199{
200 struct gsm_sms *sms;
201
202 sms = sms_from_text(receiver, buffer_getstr(b));
Harald Welte793a1352009-11-05 15:51:17 +0900203 sms->protocol_id = tp_pid;
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200204 gsm411_send_sms_subscr(receiver, sms);
205
206 return CMD_SUCCESS;
207}
208
Harald Welte98f9c752009-11-14 08:00:53 +0100209static struct gsm_subscriber *get_subscr_by_argv(const char *type,
210 const char *id)
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200211{
Harald Welte98f9c752009-11-14 08:00:53 +0100212 if (!strcmp(type, "extension"))
213 return subscr_get_by_extension(gsmnet, id);
214 else if (!strcmp(type, "imsi"))
215 return subscr_get_by_imsi(gsmnet, id);
216 else if (!strcmp(type, "tmsi"))
217 return subscr_get_by_tmsi(gsmnet, atoi(id));
218 else if (!strcmp(type, "id"))
219 return subscr_get_by_id(gsmnet, atoi(id));
220
221 return NULL;
222}
223#define SUBSCR_TYPES "(extension|imsi|tmsi|id)"
224
225DEFUN(subscriber_send_sms,
226 subscriber_send_sms_cmd,
227 "subscriber " SUBSCR_TYPES " EXTEN sms send .LINE",
228 "Select subscriber based on extension")
229{
230 struct gsm_subscriber *subscr = get_subscr_by_argv(argv[0], argv[1]);
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200231 struct buffer *b;
232 int rc;
233
Harald Welte20f98312009-11-14 10:11:45 +0100234 if (!subscr) {
235 vty_out(vty, "%% No subscriber found for %s %s%s",
236 argv[0], argv[1], VTY_NEWLINE);
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200237 return CMD_WARNING;
Harald Welte20f98312009-11-14 10:11:45 +0100238 }
Harald Welte98f9c752009-11-14 08:00:53 +0100239 b = argv_to_buffer(argc, argv, 2);
240 rc = _send_sms_buffer(subscr, b, 0);
Harald Welte793a1352009-11-05 15:51:17 +0900241 buffer_free(b);
242
Harald Welteaf8c7b42009-11-14 10:10:54 +0100243 subscr_put(subscr);
244
Harald Welte793a1352009-11-05 15:51:17 +0900245 return rc;
246}
247
Harald Welte98f9c752009-11-14 08:00:53 +0100248DEFUN(subscriber_silent_sms,
249 subscriber_silent_sms_cmd,
250 "subscriber " SUBSCR_TYPES " EXTEN silent sms send .LINE",
251 "Select subscriber based on extension")
Harald Welte793a1352009-11-05 15:51:17 +0900252{
Harald Welte98f9c752009-11-14 08:00:53 +0100253 struct gsm_subscriber *subscr = get_subscr_by_argv(argv[0], argv[1]);
Harald Welte793a1352009-11-05 15:51:17 +0900254 struct buffer *b;
255 int rc;
256
Harald Welte20f98312009-11-14 10:11:45 +0100257 if (!subscr) {
258 vty_out(vty, "%% No subscriber found for %s %s%s",
259 argv[0], argv[1], VTY_NEWLINE);
Harald Welte793a1352009-11-05 15:51:17 +0900260 return CMD_WARNING;
Harald Welte20f98312009-11-14 10:11:45 +0100261 }
Harald Welte793a1352009-11-05 15:51:17 +0900262
Harald Welte98f9c752009-11-14 08:00:53 +0100263 b = argv_to_buffer(argc, argv, 2);
Harald Welteaf8c7b42009-11-14 10:10:54 +0100264 rc = _send_sms_buffer(subscr, b, 64);
Harald Welte793a1352009-11-05 15:51:17 +0900265 buffer_free(b);
266
Harald Welteaf8c7b42009-11-14 10:10:54 +0100267 subscr_put(subscr);
268
Harald Welte793a1352009-11-05 15:51:17 +0900269 return rc;
270}
271
Harald Weltea1482332009-11-14 10:08:40 +0100272DEFUN(subscriber_silent_call,
273 subscriber_silent_call_cmd,
274 "subscriber " SUBSCR_TYPES " EXTEN silent call (start|stop)",
275 "Send a silent call to a subscriber")
276{
277 struct gsm_subscriber *subscr = get_subscr_by_argv(argv[0], argv[1]);
278 int rc;
279
280 if (!subscr) {
281 vty_out(vty, "%% No subscriber found for %s %s%s",
Harald Welte20f98312009-11-14 10:11:45 +0100282 argv[0], argv[1], VTY_NEWLINE);
Harald Weltea1482332009-11-14 10:08:40 +0100283 return CMD_WARNING;
284 }
285
286 if (!strcmp(argv[2], "start")) {
287 rc = gsm_silent_call_start(subscr, vty);
288 if (rc <= 0) {
289 vty_out(vty, "%% Subscriber not attached%s",
290 VTY_NEWLINE);
291 return CMD_WARNING;
292 }
293 } else {
294 rc = gsm_silent_call_stop(subscr);
295 if (rc < 0)
296 return CMD_WARNING;
297 }
298
299 subscr_put(subscr);
300
301 return CMD_SUCCESS;
302}
303
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200304DEFUN(cfg_subscr_name,
305 cfg_subscr_name_cmd,
306 "name NAME",
307 "Set the name of the subscriber")
308{
309 const char *name = argv[0];
310 struct gsm_subscriber *subscr = vty->index;
311
312 strncpy(subscr->name, name, sizeof(subscr->name));
313
314 db_sync_subscriber(subscr);
315
316 return CMD_SUCCESS;
317}
318
319DEFUN(cfg_subscr_extension,
320 cfg_subscr_extension_cmd,
321 "extension EXTENSION",
322 "Set the extension of the subscriber")
323{
324 const char *name = argv[0];
325 struct gsm_subscriber *subscr = vty->index;
326
327 strncpy(subscr->extension, name, sizeof(subscr->extension));
328
329 db_sync_subscriber(subscr);
330
331 return CMD_SUCCESS;
332}
333
334DEFUN(cfg_subscr_authorized,
335 cfg_subscr_authorized_cmd,
336 "auth <0-1>",
337 "Set the authorization status of the subscriber")
338{
339 int auth = atoi(argv[0]);
340 struct gsm_subscriber *subscr = vty->index;
341
342 if (auth)
343 subscr->authorized = 1;
344 else
345 subscr->authorized = 0;
346
347 db_sync_subscriber(subscr);
348
349 return CMD_SUCCESS;
350}
351
Harald Weltea1482332009-11-14 10:08:40 +0100352static int scall_cbfn(unsigned int subsys, unsigned int signal,
353 void *handler_data, void *signal_data)
354{
355 struct scall_signal_data *sigdata = signal_data;
356 struct vty *vty = sigdata->data;
357
358 switch (signal) {
359 case S_SCALL_SUCCESS:
360 vty_out(vty, "%% silent call on ARFCN %u timeslot %u%s",
361 sigdata->lchan->ts->trx->arfcn, sigdata->lchan->ts->nr,
362 VTY_NEWLINE);
363 break;
364 case S_SCALL_EXPIRED:
365 vty_out(vty, "%% silent call expired paging%s", VTY_NEWLINE);
366 break;
367 }
368 return 0;
369}
370
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200371int bsc_vty_init_extra(struct gsm_network *net)
372{
373 gsmnet = net;
374
Harald Weltea1482332009-11-14 10:08:40 +0100375 register_signal_handler(SS_SCALL, scall_cbfn, NULL);
376
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200377 install_element(VIEW_NODE, &show_subscr_cmd);
378 install_element(VIEW_NODE, &show_subscr_cache_cmd);
379
380 install_element(VIEW_NODE, &sms_send_pend_cmd);
Harald Welte98f9c752009-11-14 08:00:53 +0100381
382 install_element(VIEW_NODE, &subscriber_send_sms_cmd);
383 install_element(VIEW_NODE, &subscriber_silent_sms_cmd);
Harald Weltea1482332009-11-14 10:08:40 +0100384 install_element(VIEW_NODE, &subscriber_silent_call_cmd);
Holger Hans Peter Freythercfa90d42009-08-10 10:17:50 +0200385
386 install_element(CONFIG_NODE, &cfg_subscr_cmd);
387 install_node(&subscr_node, dummy_config_write);
388
389 install_default(SUBSCR_NODE);
390 install_element(SUBSCR_NODE, &cfg_subscr_name_cmd);
391 install_element(SUBSCR_NODE, &cfg_subscr_extension_cmd);
392 install_element(SUBSCR_NODE, &cfg_subscr_authorized_cmd);
393
394 return 0;
395}