blob: efdf751d9bd6ca7030f28c11c74ec0771d5a3211 [file] [log] [blame]
Harald Welte799e0c92010-04-30 21:49:24 +02001/*
2 * (C) 2010 by Harald Welte <laforge@gnumonks.org>
3 * (C) 2010 by On-Waves
4 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01007 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
Harald Welte799e0c92010-04-30 21:49:24 +02009 * (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
Harald Welte9af6ddf2011-01-01 15:25:50 +010014 * GNU Affero General Public License for more details.
Harald Welte799e0c92010-04-30 21:49:24 +020015 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010016 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Harald Welte799e0c92010-04-30 21:49:24 +020018 *
19 */
20
Harald Welte799e0c92010-04-30 21:49:24 +020021#include <sys/socket.h>
22#include <netinet/in.h>
23#include <arpa/inet.h>
Jacob Erlbeck73685282014-05-23 20:48:07 +020024#include <string.h>
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020025#include <time.h>
Harald Welte799e0c92010-04-30 21:49:24 +020026
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010027#include <osmocom/core/talloc.h>
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020028#include <osmocom/core/rate_ctr.h>
Harald Welte799e0c92010-04-30 21:49:24 +020029
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020030#include <openbsc/gsm_04_08.h>
Harald Welteea34a4e2012-06-16 14:59:56 +080031#include <osmocom/gprs/gprs_ns.h>
32
Harald Welte799e0c92010-04-30 21:49:24 +020033#include <openbsc/debug.h>
Harald Welteb77c6972010-05-01 11:28:43 +020034#include <openbsc/gb_proxy.h>
Holger Hans Peter Freyther7127b022014-08-04 11:52:52 +020035#include <openbsc/gprs_utils.h>
Harald Welte62ab20c2010-05-14 18:59:17 +020036#include <openbsc/vty.h>
Harald Welte799e0c92010-04-30 21:49:24 +020037
Harald Welte4b037e42010-05-19 19:45:32 +020038#include <osmocom/vty/command.h>
39#include <osmocom/vty/vty.h>
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020040#include <osmocom/vty/misc.h>
Harald Welte799e0c92010-04-30 21:49:24 +020041
Harald Welte799e0c92010-04-30 21:49:24 +020042static struct gbproxy_config *g_cfg = NULL;
43
44/*
45 * vty code for mgcp below
46 */
Harald Welteb77c6972010-05-01 11:28:43 +020047static struct cmd_node gbproxy_node = {
Harald Welte799e0c92010-04-30 21:49:24 +020048 GBPROXY_NODE,
Harald Welte570ce242012-08-17 13:16:10 +020049 "%s(config-gbproxy)# ",
Harald Welte799e0c92010-04-30 21:49:24 +020050 1,
51};
52
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +020053static void gbprox_vty_print_peer(struct vty *vty, struct gbproxy_peer *peer)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020054{
55 struct gprs_ra_id raid;
56 gsm48_parse_ra(&raid, peer->ra);
57
58 vty_out(vty, "NSEI %5u, PTP-BVCI %5u, "
59 "RAI %u-%u-%u-%u",
60 peer->nsei, peer->bvci,
61 raid.mcc, raid.mnc, raid.lac, raid.rac);
62 if (peer->blocked)
63 vty_out(vty, " [BVC-BLOCKED]");
64
65 vty_out(vty, "%s", VTY_NEWLINE);
66}
67
Harald Welte799e0c92010-04-30 21:49:24 +020068static int config_write_gbproxy(struct vty *vty)
69{
Harald Welte799e0c92010-04-30 21:49:24 +020070 vty_out(vty, "gbproxy%s", VTY_NEWLINE);
71
Harald Welteff3bde82010-05-19 15:09:09 +020072 vty_out(vty, " sgsn nsei %u%s", g_cfg->nsip_sgsn_nsei,
Harald Welte799e0c92010-04-30 21:49:24 +020073 VTY_NEWLINE);
Harald Welte799e0c92010-04-30 21:49:24 +020074
Jacob Erlbeck67a44452014-05-19 10:14:58 +020075 if (g_cfg->core_mcc > 0)
76 vty_out(vty, " core-mobile-country-code %d%s",
77 g_cfg->core_mcc, VTY_NEWLINE);
78 if (g_cfg->core_mnc > 0)
79 vty_out(vty, " core-mobile-network-code %d%s",
80 g_cfg->core_mnc, VTY_NEWLINE);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020081
82 if (g_cfg->match_re)
83 vty_out(vty, " match-imsi %s%s",
84 g_cfg->match_re, VTY_NEWLINE);
85
Jacob Erlbeck73685282014-05-23 20:48:07 +020086 if (g_cfg->core_apn != NULL) {
87 if (g_cfg->core_apn_size > 0) {
88 char str[500] = {0};
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020089 vty_out(vty, " core-access-point-name %s%s",
Holger Hans Peter Freyther7127b022014-08-04 11:52:52 +020090 gprs_apn_to_str(str, g_cfg->core_apn,
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020091 g_cfg->core_apn_size),
92 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +020093 } else {
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020094 vty_out(vty, " core-access-point-name none%s",
95 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +020096 }
97 }
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +020098
Jacob Erlbeck25f98e62014-08-28 13:47:53 +020099 if (g_cfg->route_to_sgsn2)
100 vty_out(vty, " secondary-sgsn nsei %u%s", g_cfg->nsip_sgsn2_nsei,
101 VTY_NEWLINE);
102
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200103 if (g_cfg->tlli_max_age > 0)
104 vty_out(vty, " tlli-list max-age %d%s",
105 g_cfg->tlli_max_age, VTY_NEWLINE);
106 if (g_cfg->tlli_max_len > 0)
107 vty_out(vty, " tlli-list max-length %d%s",
108 g_cfg->tlli_max_len, VTY_NEWLINE);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200109
Harald Welte799e0c92010-04-30 21:49:24 +0200110 return CMD_SUCCESS;
111}
112
Harald Welte799e0c92010-04-30 21:49:24 +0200113DEFUN(cfg_gbproxy,
114 cfg_gbproxy_cmd,
115 "gbproxy",
116 "Configure the Gb proxy")
117{
118 vty->node = GBPROXY_NODE;
119 return CMD_SUCCESS;
120}
121
Harald Welte799e0c92010-04-30 21:49:24 +0200122DEFUN(cfg_nsip_sgsn_nsei,
123 cfg_nsip_sgsn_nsei_cmd,
Harald Welteff3bde82010-05-19 15:09:09 +0200124 "sgsn nsei <0-65534>",
Holger Hans Peter Freyther2eb6e2c2011-11-05 15:14:59 +0100125 "SGSN information\n"
126 "NSEI to be used in the connection with the SGSN\n"
127 "The NSEI\n")
Harald Welte799e0c92010-04-30 21:49:24 +0200128{
129 unsigned int port = atoi(argv[0]);
130
131 g_cfg->nsip_sgsn_nsei = port;
132 return CMD_SUCCESS;
133}
134
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200135#define GBPROXY_CORE_MNC_STR "Use this network code for the core network\n"
136
137DEFUN(cfg_gbproxy_core_mnc,
138 cfg_gbproxy_core_mnc_cmd,
139 "core-mobile-network-code <1-999>",
140 GBPROXY_CORE_MNC_STR "NCC value\n")
141{
142 g_cfg->core_mnc = atoi(argv[0]);
143 return CMD_SUCCESS;
144}
145
146DEFUN(cfg_gbproxy_no_core_mnc,
147 cfg_gbproxy_no_core_mnc_cmd,
148 "no core-mobile-network-code",
149 NO_STR GBPROXY_CORE_MNC_STR)
150{
151 g_cfg->core_mnc = 0;
152 return CMD_SUCCESS;
153}
154
155#define GBPROXY_CORE_MCC_STR "Use this country code for the core network\n"
156
157DEFUN(cfg_gbproxy_core_mcc,
158 cfg_gbproxy_core_mcc_cmd,
159 "core-mobile-country-code <1-999>",
160 GBPROXY_CORE_MCC_STR "MCC value\n")
161{
162 g_cfg->core_mcc = atoi(argv[0]);
163 return CMD_SUCCESS;
164}
165
166DEFUN(cfg_gbproxy_no_core_mcc,
167 cfg_gbproxy_no_core_mcc_cmd,
168 "no core-mobile-country-code",
169 NO_STR GBPROXY_CORE_MCC_STR)
170{
171 g_cfg->core_mcc = 0;
172 return CMD_SUCCESS;
173}
174
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200175#define GBPROXY_MATCH_IMSI_STR "Restrict DTAP patching to certain IMSIs\n"
176
177DEFUN(cfg_gbproxy_match_imsi,
178 cfg_gbproxy_match_imsi_cmd,
179 "match-imsi .REGEXP",
180 GBPROXY_MATCH_IMSI_STR
181 "Regular expression for the match\n")
182{
183 const char *filter = argv[0];
184 const char *err_msg = NULL;
185
186 if (gbproxy_set_patch_filter(g_cfg, filter, &err_msg) != 0) {
187 vty_out(vty, "Match expression invalid: %s%s",
188 err_msg, VTY_NEWLINE);
189 return CMD_WARNING;
190 }
191 talloc_free(g_cfg->match_re);
192 /* TODO: replace NULL */
193 g_cfg->match_re = talloc_strdup(NULL, filter);
194
195 g_cfg->acquire_imsi = 1;
196
197 return CMD_SUCCESS;
198}
199
200DEFUN(cfg_gbproxy_no_match_imsi,
201 cfg_gbproxy_no_match_imsi_cmd,
202 "no match-imsi",
203 NO_STR GBPROXY_MATCH_IMSI_STR)
204{
205 gbproxy_clear_patch_filter(g_cfg);
206
207 talloc_free(g_cfg->match_re);
208 g_cfg->match_re = NULL;
209
210 g_cfg->acquire_imsi = 0;
211
212 return CMD_SUCCESS;
213}
214
Jacob Erlbeck73685282014-05-23 20:48:07 +0200215#define GBPROXY_CORE_APN_STR "Use this access point name (APN) for the backbone\n"
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200216#define GBPROXY_CORE_APN_ARG_STR "Replace APN by this string\n" "Remove APN\n"
Jacob Erlbeck73685282014-05-23 20:48:07 +0200217
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200218static int set_core_apn(struct vty *vty, const char *apn)
Jacob Erlbeck73685282014-05-23 20:48:07 +0200219{
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200220 int apn_len;
Jacob Erlbeck73685282014-05-23 20:48:07 +0200221
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200222 if (!apn) {
223 talloc_free(g_cfg->core_apn);
224 g_cfg->core_apn = NULL;
225 g_cfg->core_apn_size = 0;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200226 return CMD_SUCCESS;
227 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200228
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200229 apn_len = strlen(apn);
230
231 if (apn_len >= 100) {
Jacob Erlbeck73685282014-05-23 20:48:07 +0200232 vty_out(vty, "APN string too long (max 99 chars)%s",
233 VTY_NEWLINE);
234 return CMD_WARNING;
235 }
236
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200237 if (apn_len == 0) {
238 talloc_free(g_cfg->core_apn);
239 /* TODO: replace NULL */
240 g_cfg->core_apn = talloc_zero_size(NULL, 2);
241 g_cfg->core_apn_size = 0;
242 } else {
243 /* TODO: replace NULL */
244 g_cfg->core_apn =
245 talloc_realloc_size(NULL, g_cfg->core_apn, apn_len + 1);
246 g_cfg->core_apn_size =
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +0200247 gprs_str_to_apn(g_cfg->core_apn, apn_len + 1, apn);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200248 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200249
250 return CMD_SUCCESS;
251}
252
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200253DEFUN(cfg_gbproxy_core_apn,
254 cfg_gbproxy_core_apn_cmd,
255 "core-access-point-name (APN|none)",
256 GBPROXY_CORE_APN_STR GBPROXY_CORE_APN_ARG_STR)
257{
258 if (strcmp(argv[0], "none") == 0)
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200259 return set_core_apn(vty, "");
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200260 else
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200261 return set_core_apn(vty, argv[0]);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200262}
263
Jacob Erlbeck73685282014-05-23 20:48:07 +0200264DEFUN(cfg_gbproxy_no_core_apn,
265 cfg_gbproxy_no_core_apn_cmd,
266 "no core-access-point-name",
267 NO_STR GBPROXY_CORE_APN_STR)
268{
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200269 return set_core_apn(vty, NULL);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200270}
271
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200272/* TODO: Remove the patch-ptmsi command, since P-TMSI patching is enabled
273 * automatically when needed. This command is only left for manual testing
274 * (e.g. doing P-TMSI patching without using a secondary SGSN)
275 */
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200276#define GBPROXY_PATCH_PTMSI_STR "Patch P-TMSI/TLLI\n"
277
278DEFUN(cfg_gbproxy_patch_ptmsi,
279 cfg_gbproxy_patch_ptmsi_cmd,
280 "patch-ptmsi",
281 GBPROXY_PATCH_PTMSI_STR)
282{
283 g_cfg->patch_ptmsi = 1;
284
285 return CMD_SUCCESS;
286}
287
288DEFUN(cfg_gbproxy_no_patch_ptmsi,
289 cfg_gbproxy_no_patch_ptmsi_cmd,
290 "no patch-ptmsi",
291 NO_STR GBPROXY_PATCH_PTMSI_STR)
292{
293 g_cfg->patch_ptmsi = 0;
294
295 return CMD_SUCCESS;
296}
297
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200298/* TODO: Remove the acquire-imsi command, since that feature is enabled
299 * automatically when IMSI matching is enabled. This command is only left for
300 * manual testing (e.g. doing IMSI acquisition without IMSI based patching)
301 */
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200302#define GBPROXY_ACQUIRE_IMSI_STR "Acquire the IMSI before establishing a LLC connection (Experimental)\n"
303
304DEFUN(cfg_gbproxy_acquire_imsi,
305 cfg_gbproxy_acquire_imsi_cmd,
306 "acquire-imsi",
307 GBPROXY_ACQUIRE_IMSI_STR)
308{
309 g_cfg->acquire_imsi = 1;
310
311 return CMD_SUCCESS;
312}
313
314DEFUN(cfg_gbproxy_no_acquire_imsi,
315 cfg_gbproxy_no_acquire_imsi_cmd,
316 "no acquire-imsi",
317 NO_STR GBPROXY_ACQUIRE_IMSI_STR)
318{
319 g_cfg->acquire_imsi = 0;
320
321 return CMD_SUCCESS;
322}
323
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200324#define GBPROXY_SECOND_SGSN_STR "Route matching LLC connections to a second SGSN (Experimental)\n"
325
326DEFUN(cfg_gbproxy_secondary_sgsn,
327 cfg_gbproxy_secondary_sgsn_cmd,
328 "secondary-sgsn nsei <0-65534>",
329 GBPROXY_SECOND_SGSN_STR
330 "NSEI to be used in the connection with the SGSN\n"
331 "The NSEI\n")
332{
333 g_cfg->route_to_sgsn2 = 1;
334 g_cfg->nsip_sgsn2_nsei = atoi(argv[0]);
335
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200336 g_cfg->patch_ptmsi = 1;
337
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200338 return CMD_SUCCESS;
339}
340
341DEFUN(cfg_gbproxy_no_secondary_sgsn,
342 cfg_gbproxy_no_secondary_sgsn_cmd,
343 "no secondary-sgsn",
344 NO_STR GBPROXY_SECOND_SGSN_STR)
345{
346 g_cfg->route_to_sgsn2 = 0;
347 g_cfg->nsip_sgsn2_nsei = 0xFFFF;
348
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200349 g_cfg->patch_ptmsi = 0;
350
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200351 return CMD_SUCCESS;
352}
353
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200354#define GBPROXY_TLLI_LIST_STR "Set TLLI list parameters\n"
355#define GBPROXY_MAX_AGE_STR "Limit maximum age\n"
356
357DEFUN(cfg_gbproxy_tlli_list_max_age,
358 cfg_gbproxy_tlli_list_max_age_cmd,
359 "tlli-list max-age <1-999999>",
360 GBPROXY_TLLI_LIST_STR GBPROXY_MAX_AGE_STR
361 "Maximum age in seconds\n")
362{
363 g_cfg->tlli_max_age = atoi(argv[0]);
364
Jacob Erlbeck73685282014-05-23 20:48:07 +0200365 return CMD_SUCCESS;
366}
367
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200368DEFUN(cfg_gbproxy_tlli_list_no_max_age,
369 cfg_gbproxy_tlli_list_no_max_age_cmd,
370 "no tlli-list max-age",
371 NO_STR GBPROXY_TLLI_LIST_STR GBPROXY_MAX_AGE_STR)
372{
373 g_cfg->tlli_max_age = 0;
374
375 return CMD_SUCCESS;
376}
377
378#define GBPROXY_MAX_LEN_STR "Limit list length\n"
379
380DEFUN(cfg_gbproxy_tlli_list_max_len,
381 cfg_gbproxy_tlli_list_max_len_cmd,
382 "tlli-list max-length <1-99999>",
383 GBPROXY_TLLI_LIST_STR GBPROXY_MAX_LEN_STR
384 "Maximum number of TLLIs in the list\n")
385{
386 g_cfg->tlli_max_len = atoi(argv[0]);
387
388 return CMD_SUCCESS;
389}
390
391DEFUN(cfg_gbproxy_tlli_list_no_max_len,
392 cfg_gbproxy_tlli_list_no_max_len_cmd,
393 "no tlli-list max-length",
394 NO_STR GBPROXY_TLLI_LIST_STR GBPROXY_MAX_LEN_STR)
395{
396 g_cfg->tlli_max_len = 0;
397
398 return CMD_SUCCESS;
399}
400
401
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200402DEFUN(show_gbproxy, show_gbproxy_cmd, "show gbproxy [stats]",
403 SHOW_STR "Display information about the Gb proxy\n" "Show statistics\n")
404{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200405 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200406 int show_stats = argc >= 1;
Jacob Erlbeck91fb6802014-05-28 10:59:10 +0200407
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200408 if (show_stats)
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200409 vty_out_rate_ctr_group(vty, "", g_cfg->ctrg);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200410
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200411 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200412 gbprox_vty_print_peer(vty, peer);
413
414 if (show_stats)
415 vty_out_rate_ctr_group(vty, " ", peer->ctrg);
416 }
417 return CMD_SUCCESS;
418}
419
420DEFUN(show_gbproxy_tllis, show_gbproxy_tllis_cmd, "show gbproxy tllis",
421 SHOW_STR "Display information about the Gb proxy\n" "Show TLLIs\n")
422{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200423 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200424 char mi_buf[200];
425 time_t now = time(NULL);
426
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200427 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200428 struct gbproxy_tlli_info *tlli_info;
429 struct gbproxy_patch_state *state = &peer->patch_state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200430
431 gbprox_vty_print_peer(vty, peer);
432
433 llist_for_each_entry(tlli_info, &state->enabled_tllis, list) {
434 time_t age = now - tlli_info->timestamp;
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200435 int stored_msgs = 0;
436 struct llist_head *iter;
437 llist_for_each(iter, &tlli_info->stored_msgs)
438 stored_msgs++;
439
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200440 if (tlli_info->imsi > 0) {
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200441 snprintf(mi_buf, sizeof(mi_buf), "(invalid)");
442 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200443 tlli_info->imsi,
444 tlli_info->imsi_len);
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200445 } else {
446 snprintf(mi_buf, sizeof(mi_buf), "(none)");
447 }
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200448 vty_out(vty, " TLLI %08x, IMSI %s, AGE %d",
449 tlli_info->tlli.current, mi_buf, (int)age);
450
451 if (stored_msgs)
452 vty_out(vty, ", STORED %d", stored_msgs);
453
454 vty_out(vty, "%s", VTY_NEWLINE);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200455 }
456 }
457 return CMD_SUCCESS;
458}
459
460DEFUN(delete_gb_bvci, delete_gb_bvci_cmd,
461 "delete-gbproxy-peer <0-65534> bvci <2-65534>",
462 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
463 "NSEI number\n"
464 "Only delete peer with a matching BVCI\n"
465 "BVCI number\n")
466{
467 const uint16_t nsei = atoi(argv[0]);
468 const uint16_t bvci = atoi(argv[1]);
469 int counter;
470
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200471 counter = gbproxy_cleanup_peers(g_cfg, nsei, bvci);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200472
473 if (counter == 0) {
474 vty_out(vty, "BVC not found%s", VTY_NEWLINE);
475 return CMD_WARNING;
476 }
477
478 return CMD_SUCCESS;
479}
480
481DEFUN(delete_gb_nsei, delete_gb_nsei_cmd,
482 "delete-gbproxy-peer <0-65534> (only-bvc|only-nsvc|all) [dry-run]",
483 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
484 "NSEI number\n"
485 "Only delete BSSGP connections (BVC)\n"
486 "Only delete dynamic NS connections (NS-VC)\n"
487 "Delete BVC and dynamic NS connections\n"
488 "Show what would be deleted instead of actually deleting\n"
489 )
490{
491 const uint16_t nsei = atoi(argv[0]);
492 const char *mode = argv[1];
493 int dry_run = argc > 2;
494 int delete_bvc = 0;
495 int delete_nsvc = 0;
496 int counter;
497
498 if (strcmp(mode, "only-bvc") == 0)
499 delete_bvc = 1;
500 else if (strcmp(mode, "only-nsvc") == 0)
501 delete_nsvc = 1;
502 else
503 delete_bvc = delete_nsvc = 1;
504
505 if (delete_bvc) {
506 if (!dry_run)
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200507 counter = gbproxy_cleanup_peers(g_cfg, nsei, 0);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200508 else {
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200509 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200510 counter = 0;
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200511 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200512 if (peer->nsei != nsei)
513 continue;
514
515 vty_out(vty, "BVC: ");
516 gbprox_vty_print_peer(vty, peer);
517 counter += 1;
518 }
519 }
520 vty_out(vty, "%sDeleted %d BVC%s",
521 dry_run ? "Not " : "", counter, VTY_NEWLINE);
522 }
523
524 if (delete_nsvc) {
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200525 struct gprs_ns_inst *nsi = g_cfg->nsi;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200526 struct gprs_nsvc *nsvc, *nsvc2;
527
528 counter = 0;
529 llist_for_each_entry_safe(nsvc, nsvc2, &nsi->gprs_nsvcs, list) {
530 if (nsvc->nsei != nsei)
531 continue;
532 if (nsvc->persistent)
533 continue;
534
535 if (!dry_run)
536 gprs_nsvc_delete(nsvc);
537 else
538 vty_out(vty, "NS-VC: NSEI %5u, NS-VCI %5u, "
539 "remote %s%s",
540 nsvc->nsei, nsvc->nsvci,
541 gprs_ns_ll_str(nsvc), VTY_NEWLINE);
542 counter += 1;
543 }
544 vty_out(vty, "%sDeleted %d NS-VC%s",
545 dry_run ? "Not " : "", counter, VTY_NEWLINE);
546 }
547
548 return CMD_SUCCESS;
549}
550
551DEFUN(delete_gb_tlli, delete_gb_tlli_cmd,
552 "delete-gbproxy-tlli <0-65534> (tlli|imsi|stale) [IDENT]",
553 "Delete a GBProxy TLLI entry by NSEI and identification\n"
554 "NSEI number\n"
555 "Delete entries with a matching TLLI (hex)\n"
556 "Delete entries with a matching IMSI\n"
Holger Hans Peter Freytherfcc30a32014-08-29 10:50:50 +0200557 "Delete stale entries\n"
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200558 "Identification to match\n")
559{
560 const uint16_t nsei = atoi(argv[0]);
561 enum {MATCH_TLLI = 't', MATCH_IMSI = 'i', MATCH_STALE = 's'} match;
562 uint32_t tlli = 0;
563 const char *imsi = NULL;
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200564 struct gbproxy_peer *peer = 0;
565 struct gbproxy_tlli_info *tlli_info, *nxt;
566 struct gbproxy_patch_state *state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200567 char mi_buf[200];
568 int found = 0;
569
570 match = argv[1][0];
571
572 switch (match) {
573 case MATCH_TLLI:
574 if (argc < 2 || !argv[2][0]) {
575 vty_out(vty, "%% Missing TLLI%s", VTY_NEWLINE);
576 return CMD_WARNING;
577 }
578 tlli = strtoll(argv[2], NULL, 16);
579 break;
580 case MATCH_IMSI:
581 if (argc < 2 || !argv[2][0]) {
582 vty_out(vty, "%% Missing IMSI%s", VTY_NEWLINE);
583 return CMD_WARNING;
584 }
585 imsi = argv[2];
586 break;
587 default:
588 break;
589 }
590
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200591 peer = gbproxy_peer_by_nsei(g_cfg, nsei);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200592 if (!peer) {
593 vty_out(vty, "Didn't find peer with NSEI %d%s",
594 nsei, VTY_NEWLINE);
595 return CMD_WARNING;
596 }
597
598 state = &peer->patch_state;
599
600 if (match == MATCH_STALE) {
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200601 found = gbproxy_remove_stale_tllis(peer, time(NULL));
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200602 if (found)
603 vty_out(vty, "Deleted %d stale TLLI%s%s",
604 found, found == 1 ? "" : "s", VTY_NEWLINE);
605 return CMD_SUCCESS;
606 }
607
608 llist_for_each_entry_safe(tlli_info, nxt, &state->enabled_tllis, list) {
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200609 if (match == MATCH_TLLI && tlli_info->tlli.current != tlli)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200610 continue;
611
612 if (match == MATCH_IMSI) {
613 mi_buf[0] = '\0';
614 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200615 tlli_info->imsi,
616 tlli_info->imsi_len);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200617
618 if (strcmp(mi_buf, imsi) != 0)
619 continue;
620 }
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200621 vty_out(vty, "Deleting TLLI %08x%s", tlli_info->tlli.current,
622 VTY_NEWLINE);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200623 gbproxy_delete_tlli(peer, tlli_info);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200624 found += 1;
625 }
626
627 if (!found && argc >= 2) {
628 vty_out(vty, "Didn't find TLLI entry with %s %s%s",
629 argv[1], argv[2], VTY_NEWLINE);
630 }
631
632 return CMD_SUCCESS;
633}
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200634
Harald Welte799e0c92010-04-30 21:49:24 +0200635int gbproxy_vty_init(void)
636{
Harald Welte995a2d32010-05-12 16:50:52 +0000637 install_element_ve(&show_gbproxy_cmd);
Jacob Erlbeck7dd498d2014-06-19 10:15:27 +0200638 install_element_ve(&show_gbproxy_tllis_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200639
Jacob Erlbeck4211d792013-10-24 12:48:23 +0200640 install_element(ENABLE_NODE, &delete_gb_bvci_cmd);
641 install_element(ENABLE_NODE, &delete_gb_nsei_cmd);
Jacob Erlbeck7dd498d2014-06-19 10:15:27 +0200642 install_element(ENABLE_NODE, &delete_gb_tlli_cmd);
Holger Hans Peter Freyther90267a92013-10-23 11:24:17 +0200643
Harald Welte799e0c92010-04-30 21:49:24 +0200644 install_element(CONFIG_NODE, &cfg_gbproxy_cmd);
645 install_node(&gbproxy_node, config_write_gbproxy);
Jacob Erlbeck36722e12013-10-29 09:30:30 +0100646 vty_install_default(GBPROXY_NODE);
Harald Welte799e0c92010-04-30 21:49:24 +0200647 install_element(GBPROXY_NODE, &cfg_nsip_sgsn_nsei_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200648 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mcc_cmd);
649 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200650 install_element(GBPROXY_NODE, &cfg_gbproxy_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200651 install_element(GBPROXY_NODE, &cfg_gbproxy_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200652 install_element(GBPROXY_NODE, &cfg_gbproxy_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200653 install_element(GBPROXY_NODE, &cfg_gbproxy_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200654 install_element(GBPROXY_NODE, &cfg_gbproxy_acquire_imsi_cmd);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200655 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_max_age_cmd);
656 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_max_len_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200657 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mcc_cmd);
658 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200659 install_element(GBPROXY_NODE, &cfg_gbproxy_no_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200660 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200661 install_element(GBPROXY_NODE, &cfg_gbproxy_no_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200662 install_element(GBPROXY_NODE, &cfg_gbproxy_no_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200663 install_element(GBPROXY_NODE, &cfg_gbproxy_no_acquire_imsi_cmd);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200664 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_no_max_age_cmd);
665 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_no_max_len_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200666
667 return 0;
668}
669
670int gbproxy_parse_config(const char *config_file, struct gbproxy_config *cfg)
671{
672 int rc;
673
674 g_cfg = cfg;
Harald Weltedcccb182010-05-16 20:52:23 +0200675 rc = vty_read_config_file(config_file, NULL);
Harald Welte799e0c92010-04-30 21:49:24 +0200676 if (rc < 0) {
677 fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file);
678 return rc;
679 }
680
681 return 0;
682}
683