blob: 22592a16249499db54f078de246ebf3b67a47408 [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
Jacob Erlbeck7430da62014-09-12 15:09:56 +020053static const struct value_string keep_modes[] = {
54 {GBPROX_KEEP_NEVER, "never"},
55 {GBPROX_KEEP_REATTACH, "re-attach"},
56 {GBPROX_KEEP_IDENTIFIED, "identified"},
57 {GBPROX_KEEP_ALWAYS, "always"},
58 {0, NULL}
59};
60
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +020061static void gbprox_vty_print_peer(struct vty *vty, struct gbproxy_peer *peer)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020062{
63 struct gprs_ra_id raid;
64 gsm48_parse_ra(&raid, peer->ra);
65
66 vty_out(vty, "NSEI %5u, PTP-BVCI %5u, "
67 "RAI %u-%u-%u-%u",
68 peer->nsei, peer->bvci,
69 raid.mcc, raid.mnc, raid.lac, raid.rac);
70 if (peer->blocked)
71 vty_out(vty, " [BVC-BLOCKED]");
72
73 vty_out(vty, "%s", VTY_NEWLINE);
74}
75
Harald Welte799e0c92010-04-30 21:49:24 +020076static int config_write_gbproxy(struct vty *vty)
77{
Harald Welte799e0c92010-04-30 21:49:24 +020078 vty_out(vty, "gbproxy%s", VTY_NEWLINE);
79
Harald Welteff3bde82010-05-19 15:09:09 +020080 vty_out(vty, " sgsn nsei %u%s", g_cfg->nsip_sgsn_nsei,
Harald Welte799e0c92010-04-30 21:49:24 +020081 VTY_NEWLINE);
Harald Welte799e0c92010-04-30 21:49:24 +020082
Jacob Erlbeck67a44452014-05-19 10:14:58 +020083 if (g_cfg->core_mcc > 0)
84 vty_out(vty, " core-mobile-country-code %d%s",
85 g_cfg->core_mcc, VTY_NEWLINE);
86 if (g_cfg->core_mnc > 0)
87 vty_out(vty, " core-mobile-network-code %d%s",
88 g_cfg->core_mnc, VTY_NEWLINE);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020089
90 if (g_cfg->match_re)
91 vty_out(vty, " match-imsi %s%s",
92 g_cfg->match_re, VTY_NEWLINE);
93
Jacob Erlbeck73685282014-05-23 20:48:07 +020094 if (g_cfg->core_apn != NULL) {
95 if (g_cfg->core_apn_size > 0) {
96 char str[500] = {0};
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020097 vty_out(vty, " core-access-point-name %s%s",
Holger Hans Peter Freyther7127b022014-08-04 11:52:52 +020098 gprs_apn_to_str(str, g_cfg->core_apn,
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020099 g_cfg->core_apn_size),
100 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200101 } else {
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200102 vty_out(vty, " core-access-point-name none%s",
103 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200104 }
105 }
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200106
Jacob Erlbeck25f98e62014-08-28 13:47:53 +0200107 if (g_cfg->route_to_sgsn2)
108 vty_out(vty, " secondary-sgsn nsei %u%s", g_cfg->nsip_sgsn2_nsei,
109 VTY_NEWLINE);
110
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200111 if (g_cfg->tlli_max_age > 0)
112 vty_out(vty, " tlli-list max-age %d%s",
113 g_cfg->tlli_max_age, VTY_NEWLINE);
114 if (g_cfg->tlli_max_len > 0)
115 vty_out(vty, " tlli-list max-length %d%s",
116 g_cfg->tlli_max_len, VTY_NEWLINE);
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200117 vty_out(vty, " tlli-list keep-mode %s%s",
118 get_value_string(keep_modes, g_cfg->keep_tlli_infos),
119 VTY_NEWLINE);
120
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200121
Harald Welte799e0c92010-04-30 21:49:24 +0200122 return CMD_SUCCESS;
123}
124
Harald Welte799e0c92010-04-30 21:49:24 +0200125DEFUN(cfg_gbproxy,
126 cfg_gbproxy_cmd,
127 "gbproxy",
128 "Configure the Gb proxy")
129{
130 vty->node = GBPROXY_NODE;
131 return CMD_SUCCESS;
132}
133
Harald Welte799e0c92010-04-30 21:49:24 +0200134DEFUN(cfg_nsip_sgsn_nsei,
135 cfg_nsip_sgsn_nsei_cmd,
Harald Welteff3bde82010-05-19 15:09:09 +0200136 "sgsn nsei <0-65534>",
Holger Hans Peter Freyther2eb6e2c2011-11-05 15:14:59 +0100137 "SGSN information\n"
138 "NSEI to be used in the connection with the SGSN\n"
139 "The NSEI\n")
Harald Welte799e0c92010-04-30 21:49:24 +0200140{
141 unsigned int port = atoi(argv[0]);
142
143 g_cfg->nsip_sgsn_nsei = port;
144 return CMD_SUCCESS;
145}
146
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200147#define GBPROXY_CORE_MNC_STR "Use this network code for the core network\n"
148
149DEFUN(cfg_gbproxy_core_mnc,
150 cfg_gbproxy_core_mnc_cmd,
151 "core-mobile-network-code <1-999>",
152 GBPROXY_CORE_MNC_STR "NCC value\n")
153{
154 g_cfg->core_mnc = atoi(argv[0]);
155 return CMD_SUCCESS;
156}
157
158DEFUN(cfg_gbproxy_no_core_mnc,
159 cfg_gbproxy_no_core_mnc_cmd,
160 "no core-mobile-network-code",
161 NO_STR GBPROXY_CORE_MNC_STR)
162{
163 g_cfg->core_mnc = 0;
164 return CMD_SUCCESS;
165}
166
167#define GBPROXY_CORE_MCC_STR "Use this country code for the core network\n"
168
169DEFUN(cfg_gbproxy_core_mcc,
170 cfg_gbproxy_core_mcc_cmd,
171 "core-mobile-country-code <1-999>",
172 GBPROXY_CORE_MCC_STR "MCC value\n")
173{
174 g_cfg->core_mcc = atoi(argv[0]);
175 return CMD_SUCCESS;
176}
177
178DEFUN(cfg_gbproxy_no_core_mcc,
179 cfg_gbproxy_no_core_mcc_cmd,
180 "no core-mobile-country-code",
181 NO_STR GBPROXY_CORE_MCC_STR)
182{
183 g_cfg->core_mcc = 0;
184 return CMD_SUCCESS;
185}
186
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200187#define GBPROXY_MATCH_IMSI_STR "Restrict DTAP patching to certain IMSIs\n"
188
189DEFUN(cfg_gbproxy_match_imsi,
190 cfg_gbproxy_match_imsi_cmd,
191 "match-imsi .REGEXP",
192 GBPROXY_MATCH_IMSI_STR
193 "Regular expression for the match\n")
194{
195 const char *filter = argv[0];
196 const char *err_msg = NULL;
197
198 if (gbproxy_set_patch_filter(g_cfg, filter, &err_msg) != 0) {
199 vty_out(vty, "Match expression invalid: %s%s",
200 err_msg, VTY_NEWLINE);
201 return CMD_WARNING;
202 }
203 talloc_free(g_cfg->match_re);
204 /* TODO: replace NULL */
205 g_cfg->match_re = talloc_strdup(NULL, filter);
206
207 g_cfg->acquire_imsi = 1;
208
209 return CMD_SUCCESS;
210}
211
212DEFUN(cfg_gbproxy_no_match_imsi,
213 cfg_gbproxy_no_match_imsi_cmd,
214 "no match-imsi",
215 NO_STR GBPROXY_MATCH_IMSI_STR)
216{
217 gbproxy_clear_patch_filter(g_cfg);
218
219 talloc_free(g_cfg->match_re);
220 g_cfg->match_re = NULL;
221
222 g_cfg->acquire_imsi = 0;
223
224 return CMD_SUCCESS;
225}
226
Jacob Erlbeck73685282014-05-23 20:48:07 +0200227#define GBPROXY_CORE_APN_STR "Use this access point name (APN) for the backbone\n"
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200228#define GBPROXY_CORE_APN_ARG_STR "Replace APN by this string\n" "Remove APN\n"
Jacob Erlbeck73685282014-05-23 20:48:07 +0200229
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200230static int set_core_apn(struct vty *vty, const char *apn)
Jacob Erlbeck73685282014-05-23 20:48:07 +0200231{
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200232 int apn_len;
Jacob Erlbeck73685282014-05-23 20:48:07 +0200233
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200234 if (!apn) {
235 talloc_free(g_cfg->core_apn);
236 g_cfg->core_apn = NULL;
237 g_cfg->core_apn_size = 0;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200238 return CMD_SUCCESS;
239 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200240
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200241 apn_len = strlen(apn);
242
243 if (apn_len >= 100) {
Jacob Erlbeck73685282014-05-23 20:48:07 +0200244 vty_out(vty, "APN string too long (max 99 chars)%s",
245 VTY_NEWLINE);
246 return CMD_WARNING;
247 }
248
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200249 if (apn_len == 0) {
250 talloc_free(g_cfg->core_apn);
251 /* TODO: replace NULL */
252 g_cfg->core_apn = talloc_zero_size(NULL, 2);
253 g_cfg->core_apn_size = 0;
254 } else {
255 /* TODO: replace NULL */
256 g_cfg->core_apn =
257 talloc_realloc_size(NULL, g_cfg->core_apn, apn_len + 1);
258 g_cfg->core_apn_size =
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +0200259 gprs_str_to_apn(g_cfg->core_apn, apn_len + 1, apn);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200260 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200261
262 return CMD_SUCCESS;
263}
264
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200265DEFUN(cfg_gbproxy_core_apn,
266 cfg_gbproxy_core_apn_cmd,
267 "core-access-point-name (APN|none)",
268 GBPROXY_CORE_APN_STR GBPROXY_CORE_APN_ARG_STR)
269{
270 if (strcmp(argv[0], "none") == 0)
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200271 return set_core_apn(vty, "");
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200272 else
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200273 return set_core_apn(vty, argv[0]);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200274}
275
Jacob Erlbeck73685282014-05-23 20:48:07 +0200276DEFUN(cfg_gbproxy_no_core_apn,
277 cfg_gbproxy_no_core_apn_cmd,
278 "no core-access-point-name",
279 NO_STR GBPROXY_CORE_APN_STR)
280{
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200281 return set_core_apn(vty, NULL);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200282}
283
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200284/* TODO: Remove the patch-ptmsi command, since P-TMSI patching is enabled
285 * automatically when needed. This command is only left for manual testing
286 * (e.g. doing P-TMSI patching without using a secondary SGSN)
287 */
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200288#define GBPROXY_PATCH_PTMSI_STR "Patch P-TMSI/TLLI\n"
289
290DEFUN(cfg_gbproxy_patch_ptmsi,
291 cfg_gbproxy_patch_ptmsi_cmd,
292 "patch-ptmsi",
293 GBPROXY_PATCH_PTMSI_STR)
294{
295 g_cfg->patch_ptmsi = 1;
296
297 return CMD_SUCCESS;
298}
299
300DEFUN(cfg_gbproxy_no_patch_ptmsi,
301 cfg_gbproxy_no_patch_ptmsi_cmd,
302 "no patch-ptmsi",
303 NO_STR GBPROXY_PATCH_PTMSI_STR)
304{
305 g_cfg->patch_ptmsi = 0;
306
307 return CMD_SUCCESS;
308}
309
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200310/* TODO: Remove the acquire-imsi command, since that feature is enabled
311 * automatically when IMSI matching is enabled. This command is only left for
312 * manual testing (e.g. doing IMSI acquisition without IMSI based patching)
313 */
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200314#define GBPROXY_ACQUIRE_IMSI_STR "Acquire the IMSI before establishing a LLC connection (Experimental)\n"
315
316DEFUN(cfg_gbproxy_acquire_imsi,
317 cfg_gbproxy_acquire_imsi_cmd,
318 "acquire-imsi",
319 GBPROXY_ACQUIRE_IMSI_STR)
320{
321 g_cfg->acquire_imsi = 1;
322
323 return CMD_SUCCESS;
324}
325
326DEFUN(cfg_gbproxy_no_acquire_imsi,
327 cfg_gbproxy_no_acquire_imsi_cmd,
328 "no acquire-imsi",
329 NO_STR GBPROXY_ACQUIRE_IMSI_STR)
330{
331 g_cfg->acquire_imsi = 0;
332
333 return CMD_SUCCESS;
334}
335
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200336#define GBPROXY_SECOND_SGSN_STR "Route matching LLC connections to a second SGSN (Experimental)\n"
337
338DEFUN(cfg_gbproxy_secondary_sgsn,
339 cfg_gbproxy_secondary_sgsn_cmd,
340 "secondary-sgsn nsei <0-65534>",
341 GBPROXY_SECOND_SGSN_STR
342 "NSEI to be used in the connection with the SGSN\n"
343 "The NSEI\n")
344{
345 g_cfg->route_to_sgsn2 = 1;
346 g_cfg->nsip_sgsn2_nsei = atoi(argv[0]);
347
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200348 g_cfg->patch_ptmsi = 1;
349
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200350 return CMD_SUCCESS;
351}
352
353DEFUN(cfg_gbproxy_no_secondary_sgsn,
354 cfg_gbproxy_no_secondary_sgsn_cmd,
355 "no secondary-sgsn",
356 NO_STR GBPROXY_SECOND_SGSN_STR)
357{
358 g_cfg->route_to_sgsn2 = 0;
359 g_cfg->nsip_sgsn2_nsei = 0xFFFF;
360
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200361 g_cfg->patch_ptmsi = 0;
362
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200363 return CMD_SUCCESS;
364}
365
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200366#define GBPROXY_TLLI_LIST_STR "Set TLLI list parameters\n"
367#define GBPROXY_MAX_AGE_STR "Limit maximum age\n"
368
369DEFUN(cfg_gbproxy_tlli_list_max_age,
370 cfg_gbproxy_tlli_list_max_age_cmd,
371 "tlli-list max-age <1-999999>",
372 GBPROXY_TLLI_LIST_STR GBPROXY_MAX_AGE_STR
373 "Maximum age in seconds\n")
374{
375 g_cfg->tlli_max_age = atoi(argv[0]);
376
Jacob Erlbeck73685282014-05-23 20:48:07 +0200377 return CMD_SUCCESS;
378}
379
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200380DEFUN(cfg_gbproxy_tlli_list_no_max_age,
381 cfg_gbproxy_tlli_list_no_max_age_cmd,
382 "no tlli-list max-age",
383 NO_STR GBPROXY_TLLI_LIST_STR GBPROXY_MAX_AGE_STR)
384{
385 g_cfg->tlli_max_age = 0;
386
387 return CMD_SUCCESS;
388}
389
390#define GBPROXY_MAX_LEN_STR "Limit list length\n"
391
392DEFUN(cfg_gbproxy_tlli_list_max_len,
393 cfg_gbproxy_tlli_list_max_len_cmd,
394 "tlli-list max-length <1-99999>",
395 GBPROXY_TLLI_LIST_STR GBPROXY_MAX_LEN_STR
396 "Maximum number of TLLIs in the list\n")
397{
398 g_cfg->tlli_max_len = atoi(argv[0]);
399
400 return CMD_SUCCESS;
401}
402
403DEFUN(cfg_gbproxy_tlli_list_no_max_len,
404 cfg_gbproxy_tlli_list_no_max_len_cmd,
405 "no tlli-list max-length",
406 NO_STR GBPROXY_TLLI_LIST_STR GBPROXY_MAX_LEN_STR)
407{
408 g_cfg->tlli_max_len = 0;
409
410 return CMD_SUCCESS;
411}
412
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200413DEFUN(cfg_gbproxy_tlli_list_keep_mode,
414 cfg_gbproxy_tlli_list_keep_mode_cmd,
415 "tlli-list keep-mode (never|re-attach|identified|always)",
416 GBPROXY_TLLI_LIST_STR "How to keep entries for detached TLLIs\n"
417 "Discard entry immediately after detachment\n"
418 "Keep entry if a re-attachment has be requested\n"
419 "Keep entry if it associated with an IMSI\n"
420 "Don't discard entries after detachment\n")
421{
422 int val = get_string_value(keep_modes, argv[0]);
423 OSMO_ASSERT(val >= GBPROX_KEEP_NEVER && val <= GBPROX_KEEP_ALWAYS);
424 g_cfg->keep_tlli_infos = val;
425
426 return CMD_SUCCESS;
427}
428
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200429
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200430DEFUN(show_gbproxy, show_gbproxy_cmd, "show gbproxy [stats]",
431 SHOW_STR "Display information about the Gb proxy\n" "Show statistics\n")
432{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200433 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200434 int show_stats = argc >= 1;
Jacob Erlbeck91fb6802014-05-28 10:59:10 +0200435
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200436 if (show_stats)
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200437 vty_out_rate_ctr_group(vty, "", g_cfg->ctrg);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200438
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200439 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200440 gbprox_vty_print_peer(vty, peer);
441
442 if (show_stats)
443 vty_out_rate_ctr_group(vty, " ", peer->ctrg);
444 }
445 return CMD_SUCCESS;
446}
447
448DEFUN(show_gbproxy_tllis, show_gbproxy_tllis_cmd, "show gbproxy tllis",
449 SHOW_STR "Display information about the Gb proxy\n" "Show TLLIs\n")
450{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200451 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200452 char mi_buf[200];
453 time_t now = time(NULL);
454
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200455 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200456 struct gbproxy_tlli_info *tlli_info;
457 struct gbproxy_patch_state *state = &peer->patch_state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200458
459 gbprox_vty_print_peer(vty, peer);
460
461 llist_for_each_entry(tlli_info, &state->enabled_tllis, list) {
462 time_t age = now - tlli_info->timestamp;
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200463 int stored_msgs = 0;
464 struct llist_head *iter;
465 llist_for_each(iter, &tlli_info->stored_msgs)
466 stored_msgs++;
467
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200468 if (tlli_info->imsi > 0) {
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200469 snprintf(mi_buf, sizeof(mi_buf), "(invalid)");
470 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200471 tlli_info->imsi,
472 tlli_info->imsi_len);
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200473 } else {
474 snprintf(mi_buf, sizeof(mi_buf), "(none)");
475 }
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200476 vty_out(vty, " TLLI %08x, IMSI %s, AGE %d",
477 tlli_info->tlli.current, mi_buf, (int)age);
478
479 if (stored_msgs)
480 vty_out(vty, ", STORED %d", stored_msgs);
481
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200482 if (tlli_info->is_deregistered)
483 vty_out(vty, ", DE-REGISTERED");
484
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200485 vty_out(vty, "%s", VTY_NEWLINE);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200486 }
487 }
488 return CMD_SUCCESS;
489}
490
491DEFUN(delete_gb_bvci, delete_gb_bvci_cmd,
492 "delete-gbproxy-peer <0-65534> bvci <2-65534>",
493 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
494 "NSEI number\n"
495 "Only delete peer with a matching BVCI\n"
496 "BVCI number\n")
497{
498 const uint16_t nsei = atoi(argv[0]);
499 const uint16_t bvci = atoi(argv[1]);
500 int counter;
501
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200502 counter = gbproxy_cleanup_peers(g_cfg, nsei, bvci);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200503
504 if (counter == 0) {
505 vty_out(vty, "BVC not found%s", VTY_NEWLINE);
506 return CMD_WARNING;
507 }
508
509 return CMD_SUCCESS;
510}
511
512DEFUN(delete_gb_nsei, delete_gb_nsei_cmd,
513 "delete-gbproxy-peer <0-65534> (only-bvc|only-nsvc|all) [dry-run]",
514 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
515 "NSEI number\n"
516 "Only delete BSSGP connections (BVC)\n"
517 "Only delete dynamic NS connections (NS-VC)\n"
518 "Delete BVC and dynamic NS connections\n"
519 "Show what would be deleted instead of actually deleting\n"
520 )
521{
522 const uint16_t nsei = atoi(argv[0]);
523 const char *mode = argv[1];
524 int dry_run = argc > 2;
525 int delete_bvc = 0;
526 int delete_nsvc = 0;
527 int counter;
528
529 if (strcmp(mode, "only-bvc") == 0)
530 delete_bvc = 1;
531 else if (strcmp(mode, "only-nsvc") == 0)
532 delete_nsvc = 1;
533 else
534 delete_bvc = delete_nsvc = 1;
535
536 if (delete_bvc) {
537 if (!dry_run)
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200538 counter = gbproxy_cleanup_peers(g_cfg, nsei, 0);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200539 else {
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200540 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200541 counter = 0;
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200542 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200543 if (peer->nsei != nsei)
544 continue;
545
546 vty_out(vty, "BVC: ");
547 gbprox_vty_print_peer(vty, peer);
548 counter += 1;
549 }
550 }
551 vty_out(vty, "%sDeleted %d BVC%s",
552 dry_run ? "Not " : "", counter, VTY_NEWLINE);
553 }
554
555 if (delete_nsvc) {
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200556 struct gprs_ns_inst *nsi = g_cfg->nsi;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200557 struct gprs_nsvc *nsvc, *nsvc2;
558
559 counter = 0;
560 llist_for_each_entry_safe(nsvc, nsvc2, &nsi->gprs_nsvcs, list) {
561 if (nsvc->nsei != nsei)
562 continue;
563 if (nsvc->persistent)
564 continue;
565
566 if (!dry_run)
567 gprs_nsvc_delete(nsvc);
568 else
569 vty_out(vty, "NS-VC: NSEI %5u, NS-VCI %5u, "
570 "remote %s%s",
571 nsvc->nsei, nsvc->nsvci,
572 gprs_ns_ll_str(nsvc), VTY_NEWLINE);
573 counter += 1;
574 }
575 vty_out(vty, "%sDeleted %d NS-VC%s",
576 dry_run ? "Not " : "", counter, VTY_NEWLINE);
577 }
578
579 return CMD_SUCCESS;
580}
581
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200582#define GBPROXY_DELETE_TLLI_STR \
583 "Delete a GBProxy TLLI entry by NSEI and identification\nNSEI number\n"
584
585DEFUN(delete_gb_tlli_by_id, delete_gb_tlli_by_id_cmd,
586 "delete-gbproxy-tlli <0-65534> (tlli|imsi) IDENT",
587 GBPROXY_DELETE_TLLI_STR
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200588 "Delete entries with a matching TLLI (hex)\n"
589 "Delete entries with a matching IMSI\n"
590 "Identification to match\n")
591{
592 const uint16_t nsei = atoi(argv[0]);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200593 enum {MATCH_TLLI = 't', MATCH_IMSI = 'i'} match;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200594 uint32_t tlli = 0;
595 const char *imsi = NULL;
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200596 struct gbproxy_peer *peer = 0;
597 struct gbproxy_tlli_info *tlli_info, *nxt;
598 struct gbproxy_patch_state *state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200599 char mi_buf[200];
600 int found = 0;
601
602 match = argv[1][0];
603
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200604 if (match == MATCH_TLLI)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200605 tlli = strtoll(argv[2], NULL, 16);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200606 else
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200607 imsi = argv[2];
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200608
609 peer = gbproxy_peer_by_nsei(g_cfg, nsei);
610 if (!peer) {
611 vty_out(vty, "Didn't find peer with NSEI %d%s",
612 nsei, VTY_NEWLINE);
613 return CMD_WARNING;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200614 }
615
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200616 state = &peer->patch_state;
617
618 llist_for_each_entry_safe(tlli_info, nxt, &state->enabled_tllis, list) {
619 if (match == MATCH_TLLI) {
620 if (tlli_info->tlli.current != tlli)
621 continue;
622 } else {
623 mi_buf[0] = '\0';
624 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
625 tlli_info->imsi,
626 tlli_info->imsi_len);
627
628 if (strcmp(mi_buf, imsi) != 0)
629 continue;
630 }
631
632 vty_out(vty, "Deleting TLLI %08x%s", tlli_info->tlli.current,
633 VTY_NEWLINE);
634 gbproxy_delete_tlli(peer, tlli_info);
635 found += 1;
636 }
637
638 if (!found && argc >= 2) {
639 vty_out(vty, "Didn't find TLLI entry with %s %s%s",
640 argv[1], argv[2], VTY_NEWLINE);
641 }
642
643 return CMD_SUCCESS;
644}
645
646DEFUN(delete_gb_tlli, delete_gb_tlli_cmd,
647 "delete-gbproxy-tlli <0-65534> (stale|de-registered)",
648 GBPROXY_DELETE_TLLI_STR
649 "Delete stale entries\n"
650 "Delete de-registered entries\n")
651{
652 const uint16_t nsei = atoi(argv[0]);
653 enum {MATCH_STALE = 's', MATCH_DEREGISTERED = 'd'} match;
654 struct gbproxy_peer *peer = 0;
655 struct gbproxy_tlli_info *tlli_info, *nxt;
656 struct gbproxy_patch_state *state;
657 int found = 0;
658
659 match = argv[1][0];
660
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200661 peer = gbproxy_peer_by_nsei(g_cfg, nsei);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200662 if (!peer) {
663 vty_out(vty, "Didn't find peer with NSEI %d%s",
664 nsei, VTY_NEWLINE);
665 return CMD_WARNING;
666 }
667
668 state = &peer->patch_state;
669
670 if (match == MATCH_STALE) {
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200671 found = gbproxy_remove_stale_tllis(peer, time(NULL));
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200672 if (found)
673 vty_out(vty, "Deleted %d stale TLLI%s%s",
674 found, found == 1 ? "" : "s", VTY_NEWLINE);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200675 } else {
676 llist_for_each_entry_safe(tlli_info, nxt,
677 &state->enabled_tllis, list) {
678 if (!tlli_info->is_deregistered)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200679 continue;
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200680
681 gbproxy_delete_tlli(peer, tlli_info);
682 found += 1;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200683 }
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200684 }
685
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200686 if (found)
687 vty_out(vty, "Deleted %d %s TLLI%s%s",
688 found, argv[1], found == 1 ? "" : "s", VTY_NEWLINE);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200689
690 return CMD_SUCCESS;
691}
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200692
Harald Welte799e0c92010-04-30 21:49:24 +0200693int gbproxy_vty_init(void)
694{
Harald Welte995a2d32010-05-12 16:50:52 +0000695 install_element_ve(&show_gbproxy_cmd);
Jacob Erlbeck7dd498d2014-06-19 10:15:27 +0200696 install_element_ve(&show_gbproxy_tllis_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200697
Jacob Erlbeck4211d792013-10-24 12:48:23 +0200698 install_element(ENABLE_NODE, &delete_gb_bvci_cmd);
699 install_element(ENABLE_NODE, &delete_gb_nsei_cmd);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200700 install_element(ENABLE_NODE, &delete_gb_tlli_by_id_cmd);
Jacob Erlbeck7dd498d2014-06-19 10:15:27 +0200701 install_element(ENABLE_NODE, &delete_gb_tlli_cmd);
Holger Hans Peter Freyther90267a92013-10-23 11:24:17 +0200702
Harald Welte799e0c92010-04-30 21:49:24 +0200703 install_element(CONFIG_NODE, &cfg_gbproxy_cmd);
704 install_node(&gbproxy_node, config_write_gbproxy);
Jacob Erlbeck36722e12013-10-29 09:30:30 +0100705 vty_install_default(GBPROXY_NODE);
Harald Welte799e0c92010-04-30 21:49:24 +0200706 install_element(GBPROXY_NODE, &cfg_nsip_sgsn_nsei_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200707 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mcc_cmd);
708 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200709 install_element(GBPROXY_NODE, &cfg_gbproxy_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200710 install_element(GBPROXY_NODE, &cfg_gbproxy_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200711 install_element(GBPROXY_NODE, &cfg_gbproxy_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200712 install_element(GBPROXY_NODE, &cfg_gbproxy_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200713 install_element(GBPROXY_NODE, &cfg_gbproxy_acquire_imsi_cmd);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200714 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_max_age_cmd);
715 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_max_len_cmd);
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200716 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_keep_mode_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200717 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mcc_cmd);
718 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200719 install_element(GBPROXY_NODE, &cfg_gbproxy_no_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200720 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200721 install_element(GBPROXY_NODE, &cfg_gbproxy_no_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200722 install_element(GBPROXY_NODE, &cfg_gbproxy_no_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200723 install_element(GBPROXY_NODE, &cfg_gbproxy_no_acquire_imsi_cmd);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200724 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_no_max_age_cmd);
725 install_element(GBPROXY_NODE, &cfg_gbproxy_tlli_list_no_max_len_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200726
727 return 0;
728}
729
730int gbproxy_parse_config(const char *config_file, struct gbproxy_config *cfg)
731{
732 int rc;
733
734 g_cfg = cfg;
Harald Weltedcccb182010-05-16 20:52:23 +0200735 rc = vty_read_config_file(config_file, NULL);
Harald Welte799e0c92010-04-30 21:49:24 +0200736 if (rc < 0) {
737 fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file);
738 return rc;
739 }
740
741 return 0;
742}
743