blob: 6dc0e343741496f87abe08ff302613309eefcdf3 [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{
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +020078 enum gbproxy_match_id match_id;
79
Harald Welte799e0c92010-04-30 21:49:24 +020080 vty_out(vty, "gbproxy%s", VTY_NEWLINE);
81
Harald Welteff3bde82010-05-19 15:09:09 +020082 vty_out(vty, " sgsn nsei %u%s", g_cfg->nsip_sgsn_nsei,
Harald Welte799e0c92010-04-30 21:49:24 +020083 VTY_NEWLINE);
Harald Welte799e0c92010-04-30 21:49:24 +020084
Jacob Erlbeck67a44452014-05-19 10:14:58 +020085 if (g_cfg->core_mcc > 0)
86 vty_out(vty, " core-mobile-country-code %d%s",
87 g_cfg->core_mcc, VTY_NEWLINE);
88 if (g_cfg->core_mnc > 0)
89 vty_out(vty, " core-mobile-network-code %d%s",
90 g_cfg->core_mnc, VTY_NEWLINE);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020091
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +020092 for (match_id = 0; match_id < ARRAY_SIZE(g_cfg->matches); ++match_id) {
93 struct gbproxy_match *match = &g_cfg->matches[match_id];
94 if (match->re_str)
95 vty_out(vty, " match-imsi %s%s",
96 match->re_str, VTY_NEWLINE);
97 }
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +020098
Jacob Erlbeck73685282014-05-23 20:48:07 +020099 if (g_cfg->core_apn != NULL) {
100 if (g_cfg->core_apn_size > 0) {
101 char str[500] = {0};
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200102 vty_out(vty, " core-access-point-name %s%s",
Holger Hans Peter Freyther7127b022014-08-04 11:52:52 +0200103 gprs_apn_to_str(str, g_cfg->core_apn,
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200104 g_cfg->core_apn_size),
105 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200106 } else {
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200107 vty_out(vty, " core-access-point-name none%s",
108 VTY_NEWLINE);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200109 }
110 }
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200111
Jacob Erlbeck25f98e62014-08-28 13:47:53 +0200112 if (g_cfg->route_to_sgsn2)
113 vty_out(vty, " secondary-sgsn nsei %u%s", g_cfg->nsip_sgsn2_nsei,
114 VTY_NEWLINE);
115
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200116 if (g_cfg->tlli_max_age > 0)
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200117 vty_out(vty, " link-list max-age %d%s",
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200118 g_cfg->tlli_max_age, VTY_NEWLINE);
119 if (g_cfg->tlli_max_len > 0)
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200120 vty_out(vty, " link-list max-length %d%s",
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200121 g_cfg->tlli_max_len, VTY_NEWLINE);
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200122 vty_out(vty, " link-list keep-mode %s%s",
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200123 get_value_string(keep_modes, g_cfg->keep_link_infos),
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200124 VTY_NEWLINE);
125
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200126
Harald Welte799e0c92010-04-30 21:49:24 +0200127 return CMD_SUCCESS;
128}
129
Harald Welte799e0c92010-04-30 21:49:24 +0200130DEFUN(cfg_gbproxy,
131 cfg_gbproxy_cmd,
132 "gbproxy",
133 "Configure the Gb proxy")
134{
135 vty->node = GBPROXY_NODE;
136 return CMD_SUCCESS;
137}
138
Harald Welte799e0c92010-04-30 21:49:24 +0200139DEFUN(cfg_nsip_sgsn_nsei,
140 cfg_nsip_sgsn_nsei_cmd,
Harald Welteff3bde82010-05-19 15:09:09 +0200141 "sgsn nsei <0-65534>",
Holger Hans Peter Freyther2eb6e2c2011-11-05 15:14:59 +0100142 "SGSN information\n"
143 "NSEI to be used in the connection with the SGSN\n"
144 "The NSEI\n")
Harald Welte799e0c92010-04-30 21:49:24 +0200145{
146 unsigned int port = atoi(argv[0]);
147
148 g_cfg->nsip_sgsn_nsei = port;
149 return CMD_SUCCESS;
150}
151
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200152#define GBPROXY_CORE_MNC_STR "Use this network code for the core network\n"
153
154DEFUN(cfg_gbproxy_core_mnc,
155 cfg_gbproxy_core_mnc_cmd,
156 "core-mobile-network-code <1-999>",
157 GBPROXY_CORE_MNC_STR "NCC value\n")
158{
159 g_cfg->core_mnc = atoi(argv[0]);
160 return CMD_SUCCESS;
161}
162
163DEFUN(cfg_gbproxy_no_core_mnc,
164 cfg_gbproxy_no_core_mnc_cmd,
165 "no core-mobile-network-code",
166 NO_STR GBPROXY_CORE_MNC_STR)
167{
168 g_cfg->core_mnc = 0;
169 return CMD_SUCCESS;
170}
171
172#define GBPROXY_CORE_MCC_STR "Use this country code for the core network\n"
173
174DEFUN(cfg_gbproxy_core_mcc,
175 cfg_gbproxy_core_mcc_cmd,
176 "core-mobile-country-code <1-999>",
177 GBPROXY_CORE_MCC_STR "MCC value\n")
178{
179 g_cfg->core_mcc = atoi(argv[0]);
180 return CMD_SUCCESS;
181}
182
183DEFUN(cfg_gbproxy_no_core_mcc,
184 cfg_gbproxy_no_core_mcc_cmd,
185 "no core-mobile-country-code",
186 NO_STR GBPROXY_CORE_MCC_STR)
187{
188 g_cfg->core_mcc = 0;
189 return CMD_SUCCESS;
190}
191
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200192#define GBPROXY_MATCH_IMSI_STR "Restrict DTAP patching to certain IMSIs\n"
193
194DEFUN(cfg_gbproxy_match_imsi,
195 cfg_gbproxy_match_imsi_cmd,
196 "match-imsi .REGEXP",
197 GBPROXY_MATCH_IMSI_STR
198 "Regular expression for the match\n")
199{
200 const char *filter = argv[0];
201 const char *err_msg = NULL;
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +0200202 struct gbproxy_match *match = &g_cfg->matches[GBPROX_MATCH_PATCHING];
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200203
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +0200204 if (gbproxy_set_patch_filter(match, filter, &err_msg) != 0) {
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200205 vty_out(vty, "Match expression invalid: %s%s",
206 err_msg, VTY_NEWLINE);
207 return CMD_WARNING;
208 }
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200209
210 g_cfg->acquire_imsi = 1;
211
212 return CMD_SUCCESS;
213}
214
215DEFUN(cfg_gbproxy_no_match_imsi,
216 cfg_gbproxy_no_match_imsi_cmd,
217 "no match-imsi",
218 NO_STR GBPROXY_MATCH_IMSI_STR)
219{
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +0200220 struct gbproxy_match *match = &g_cfg->matches[GBPROX_MATCH_PATCHING];
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200221
Jacob Erlbeck9a83d7a2014-09-25 11:17:31 +0200222 gbproxy_clear_patch_filter(match);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200223
224 g_cfg->acquire_imsi = 0;
225
226 return CMD_SUCCESS;
227}
228
Jacob Erlbeck73685282014-05-23 20:48:07 +0200229#define GBPROXY_CORE_APN_STR "Use this access point name (APN) for the backbone\n"
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200230#define GBPROXY_CORE_APN_ARG_STR "Replace APN by this string\n" "Remove APN\n"
Jacob Erlbeck73685282014-05-23 20:48:07 +0200231
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200232static int set_core_apn(struct vty *vty, const char *apn)
Jacob Erlbeck73685282014-05-23 20:48:07 +0200233{
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200234 int apn_len;
Jacob Erlbeck73685282014-05-23 20:48:07 +0200235
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200236 if (!apn) {
237 talloc_free(g_cfg->core_apn);
238 g_cfg->core_apn = NULL;
239 g_cfg->core_apn_size = 0;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200240 return CMD_SUCCESS;
241 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200242
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200243 apn_len = strlen(apn);
244
245 if (apn_len >= 100) {
Jacob Erlbeck73685282014-05-23 20:48:07 +0200246 vty_out(vty, "APN string too long (max 99 chars)%s",
247 VTY_NEWLINE);
248 return CMD_WARNING;
249 }
250
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200251 if (apn_len == 0) {
252 talloc_free(g_cfg->core_apn);
253 /* TODO: replace NULL */
254 g_cfg->core_apn = talloc_zero_size(NULL, 2);
255 g_cfg->core_apn_size = 0;
256 } else {
257 /* TODO: replace NULL */
258 g_cfg->core_apn =
259 talloc_realloc_size(NULL, g_cfg->core_apn, apn_len + 1);
260 g_cfg->core_apn_size =
Holger Hans Peter Freytherce1b22e2014-08-04 14:22:13 +0200261 gprs_str_to_apn(g_cfg->core_apn, apn_len + 1, apn);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200262 }
Jacob Erlbeck73685282014-05-23 20:48:07 +0200263
264 return CMD_SUCCESS;
265}
266
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200267DEFUN(cfg_gbproxy_core_apn,
268 cfg_gbproxy_core_apn_cmd,
269 "core-access-point-name (APN|none)",
270 GBPROXY_CORE_APN_STR GBPROXY_CORE_APN_ARG_STR)
271{
272 if (strcmp(argv[0], "none") == 0)
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200273 return set_core_apn(vty, "");
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200274 else
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200275 return set_core_apn(vty, argv[0]);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200276}
277
Jacob Erlbeck73685282014-05-23 20:48:07 +0200278DEFUN(cfg_gbproxy_no_core_apn,
279 cfg_gbproxy_no_core_apn_cmd,
280 "no core-access-point-name",
281 NO_STR GBPROXY_CORE_APN_STR)
282{
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200283 return set_core_apn(vty, NULL);
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200284}
285
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200286/* TODO: Remove the patch-ptmsi command, since P-TMSI patching is enabled
287 * automatically when needed. This command is only left for manual testing
288 * (e.g. doing P-TMSI patching without using a secondary SGSN)
289 */
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200290#define GBPROXY_PATCH_PTMSI_STR "Patch P-TMSI/TLLI\n"
291
292DEFUN(cfg_gbproxy_patch_ptmsi,
293 cfg_gbproxy_patch_ptmsi_cmd,
294 "patch-ptmsi",
295 GBPROXY_PATCH_PTMSI_STR)
296{
297 g_cfg->patch_ptmsi = 1;
298
299 return CMD_SUCCESS;
300}
301
302DEFUN(cfg_gbproxy_no_patch_ptmsi,
303 cfg_gbproxy_no_patch_ptmsi_cmd,
304 "no patch-ptmsi",
305 NO_STR GBPROXY_PATCH_PTMSI_STR)
306{
307 g_cfg->patch_ptmsi = 0;
308
309 return CMD_SUCCESS;
310}
311
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200312/* TODO: Remove the acquire-imsi command, since that feature is enabled
313 * automatically when IMSI matching is enabled. This command is only left for
314 * manual testing (e.g. doing IMSI acquisition without IMSI based patching)
315 */
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200316#define GBPROXY_ACQUIRE_IMSI_STR "Acquire the IMSI before establishing a LLC connection (Experimental)\n"
317
318DEFUN(cfg_gbproxy_acquire_imsi,
319 cfg_gbproxy_acquire_imsi_cmd,
320 "acquire-imsi",
321 GBPROXY_ACQUIRE_IMSI_STR)
322{
323 g_cfg->acquire_imsi = 1;
324
325 return CMD_SUCCESS;
326}
327
328DEFUN(cfg_gbproxy_no_acquire_imsi,
329 cfg_gbproxy_no_acquire_imsi_cmd,
330 "no acquire-imsi",
331 NO_STR GBPROXY_ACQUIRE_IMSI_STR)
332{
333 g_cfg->acquire_imsi = 0;
334
335 return CMD_SUCCESS;
336}
337
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200338#define GBPROXY_SECOND_SGSN_STR "Route matching LLC connections to a second SGSN (Experimental)\n"
339
340DEFUN(cfg_gbproxy_secondary_sgsn,
341 cfg_gbproxy_secondary_sgsn_cmd,
342 "secondary-sgsn nsei <0-65534>",
343 GBPROXY_SECOND_SGSN_STR
344 "NSEI to be used in the connection with the SGSN\n"
345 "The NSEI\n")
346{
347 g_cfg->route_to_sgsn2 = 1;
348 g_cfg->nsip_sgsn2_nsei = atoi(argv[0]);
349
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200350 g_cfg->patch_ptmsi = 1;
351
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200352 return CMD_SUCCESS;
353}
354
355DEFUN(cfg_gbproxy_no_secondary_sgsn,
356 cfg_gbproxy_no_secondary_sgsn_cmd,
357 "no secondary-sgsn",
358 NO_STR GBPROXY_SECOND_SGSN_STR)
359{
360 g_cfg->route_to_sgsn2 = 0;
361 g_cfg->nsip_sgsn2_nsei = 0xFFFF;
362
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200363 g_cfg->patch_ptmsi = 0;
364
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +0200365 return CMD_SUCCESS;
366}
367
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200368#define GBPROXY_LINK_LIST_STR "Set TLLI list parameters\n"
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200369#define GBPROXY_MAX_AGE_STR "Limit maximum age\n"
370
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200371DEFUN(cfg_gbproxy_link_list_max_age,
372 cfg_gbproxy_link_list_max_age_cmd,
373 "link-list max-age <1-999999>",
374 GBPROXY_LINK_LIST_STR GBPROXY_MAX_AGE_STR
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200375 "Maximum age in seconds\n")
376{
377 g_cfg->tlli_max_age = atoi(argv[0]);
378
Jacob Erlbeck73685282014-05-23 20:48:07 +0200379 return CMD_SUCCESS;
380}
381
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200382DEFUN(cfg_gbproxy_link_list_no_max_age,
383 cfg_gbproxy_link_list_no_max_age_cmd,
384 "no link-list max-age",
385 NO_STR GBPROXY_LINK_LIST_STR GBPROXY_MAX_AGE_STR)
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200386{
387 g_cfg->tlli_max_age = 0;
388
389 return CMD_SUCCESS;
390}
391
392#define GBPROXY_MAX_LEN_STR "Limit list length\n"
393
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200394DEFUN(cfg_gbproxy_link_list_max_len,
395 cfg_gbproxy_link_list_max_len_cmd,
396 "link-list max-length <1-99999>",
397 GBPROXY_LINK_LIST_STR GBPROXY_MAX_LEN_STR
398 "Maximum number of logical links in the list\n")
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200399{
400 g_cfg->tlli_max_len = atoi(argv[0]);
401
402 return CMD_SUCCESS;
403}
404
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200405DEFUN(cfg_gbproxy_link_list_no_max_len,
406 cfg_gbproxy_link_list_no_max_len_cmd,
407 "no link-list max-length",
408 NO_STR GBPROXY_LINK_LIST_STR GBPROXY_MAX_LEN_STR)
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200409{
410 g_cfg->tlli_max_len = 0;
411
412 return CMD_SUCCESS;
413}
414
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200415DEFUN(cfg_gbproxy_link_list_keep_mode,
416 cfg_gbproxy_link_list_keep_mode_cmd,
417 "link-list keep-mode (never|re-attach|identified|always)",
418 GBPROXY_LINK_LIST_STR "How to keep entries for detached logical links\n"
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200419 "Discard entry immediately after detachment\n"
420 "Keep entry if a re-attachment has be requested\n"
421 "Keep entry if it associated with an IMSI\n"
422 "Don't discard entries after detachment\n")
423{
424 int val = get_string_value(keep_modes, argv[0]);
425 OSMO_ASSERT(val >= GBPROX_KEEP_NEVER && val <= GBPROX_KEEP_ALWAYS);
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200426 g_cfg->keep_link_infos = val;
Jacob Erlbeck7430da62014-09-12 15:09:56 +0200427
428 return CMD_SUCCESS;
429}
430
Jacob Erlbeck7c101d92014-06-06 18:49:23 +0200431
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200432DEFUN(show_gbproxy, show_gbproxy_cmd, "show gbproxy [stats]",
433 SHOW_STR "Display information about the Gb proxy\n" "Show statistics\n")
434{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200435 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200436 int show_stats = argc >= 1;
Jacob Erlbeck91fb6802014-05-28 10:59:10 +0200437
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200438 if (show_stats)
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200439 vty_out_rate_ctr_group(vty, "", g_cfg->ctrg);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200440
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200441 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200442 gbprox_vty_print_peer(vty, peer);
443
444 if (show_stats)
445 vty_out_rate_ctr_group(vty, " ", peer->ctrg);
446 }
447 return CMD_SUCCESS;
448}
449
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200450DEFUN(show_gbproxy_links, show_gbproxy_links_cmd, "show gbproxy links",
451 SHOW_STR "Display information about the Gb proxy\n" "Show logical links\n")
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200452{
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200453 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200454 char mi_buf[200];
Jacob Erlbeckc6807c42014-09-19 16:34:01 +0200455 time_t now;
456 struct timespec ts = {0,};
457
458 clock_gettime(CLOCK_MONOTONIC, &ts);
459 now = ts.tv_sec;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200460
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200461 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200462 struct gbproxy_link_info *link_info;
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200463 struct gbproxy_patch_state *state = &peer->patch_state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200464
465 gbprox_vty_print_peer(vty, peer);
466
Jacob Erlbeckf8562e32014-09-19 16:03:07 +0200467 llist_for_each_entry(link_info, &state->logical_links, list) {
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200468 time_t age = now - link_info->timestamp;
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200469 int stored_msgs = 0;
470 struct llist_head *iter;
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200471 llist_for_each(iter, &link_info->stored_msgs)
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200472 stored_msgs++;
473
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200474 if (link_info->imsi > 0) {
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200475 snprintf(mi_buf, sizeof(mi_buf), "(invalid)");
476 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200477 link_info->imsi,
478 link_info->imsi_len);
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200479 } else {
480 snprintf(mi_buf, sizeof(mi_buf), "(none)");
481 }
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200482 vty_out(vty, " TLLI %08x, IMSI %s, AGE %d",
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200483 link_info->tlli.current, mi_buf, (int)age);
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200484
485 if (stored_msgs)
486 vty_out(vty, ", STORED %d", stored_msgs);
487
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200488 if (g_cfg->route_to_sgsn2)
489 vty_out(vty, ", SGSN NSEI %d",
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200490 link_info->sgsn_nsei);
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200491
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200492 if (link_info->is_deregistered)
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200493 vty_out(vty, ", DE-REGISTERED");
494
Jacob Erlbeck146e3072014-09-02 14:09:01 +0200495 vty_out(vty, "%s", VTY_NEWLINE);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200496 }
497 }
498 return CMD_SUCCESS;
499}
500
501DEFUN(delete_gb_bvci, delete_gb_bvci_cmd,
502 "delete-gbproxy-peer <0-65534> bvci <2-65534>",
503 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
504 "NSEI number\n"
505 "Only delete peer with a matching BVCI\n"
506 "BVCI number\n")
507{
508 const uint16_t nsei = atoi(argv[0]);
509 const uint16_t bvci = atoi(argv[1]);
510 int counter;
511
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200512 counter = gbproxy_cleanup_peers(g_cfg, nsei, bvci);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200513
514 if (counter == 0) {
515 vty_out(vty, "BVC not found%s", VTY_NEWLINE);
516 return CMD_WARNING;
517 }
518
519 return CMD_SUCCESS;
520}
521
522DEFUN(delete_gb_nsei, delete_gb_nsei_cmd,
523 "delete-gbproxy-peer <0-65534> (only-bvc|only-nsvc|all) [dry-run]",
524 "Delete a GBProxy peer by NSEI and optionally BVCI\n"
525 "NSEI number\n"
526 "Only delete BSSGP connections (BVC)\n"
527 "Only delete dynamic NS connections (NS-VC)\n"
528 "Delete BVC and dynamic NS connections\n"
529 "Show what would be deleted instead of actually deleting\n"
530 )
531{
532 const uint16_t nsei = atoi(argv[0]);
533 const char *mode = argv[1];
534 int dry_run = argc > 2;
535 int delete_bvc = 0;
536 int delete_nsvc = 0;
537 int counter;
538
539 if (strcmp(mode, "only-bvc") == 0)
540 delete_bvc = 1;
541 else if (strcmp(mode, "only-nsvc") == 0)
542 delete_nsvc = 1;
543 else
544 delete_bvc = delete_nsvc = 1;
545
546 if (delete_bvc) {
547 if (!dry_run)
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200548 counter = gbproxy_cleanup_peers(g_cfg, nsei, 0);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200549 else {
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200550 struct gbproxy_peer *peer;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200551 counter = 0;
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200552 llist_for_each_entry(peer, &g_cfg->bts_peers, list) {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200553 if (peer->nsei != nsei)
554 continue;
555
556 vty_out(vty, "BVC: ");
557 gbprox_vty_print_peer(vty, peer);
558 counter += 1;
559 }
560 }
561 vty_out(vty, "%sDeleted %d BVC%s",
562 dry_run ? "Not " : "", counter, VTY_NEWLINE);
563 }
564
565 if (delete_nsvc) {
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200566 struct gprs_ns_inst *nsi = g_cfg->nsi;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200567 struct gprs_nsvc *nsvc, *nsvc2;
568
569 counter = 0;
570 llist_for_each_entry_safe(nsvc, nsvc2, &nsi->gprs_nsvcs, list) {
571 if (nsvc->nsei != nsei)
572 continue;
573 if (nsvc->persistent)
574 continue;
575
576 if (!dry_run)
577 gprs_nsvc_delete(nsvc);
578 else
579 vty_out(vty, "NS-VC: NSEI %5u, NS-VCI %5u, "
580 "remote %s%s",
581 nsvc->nsei, nsvc->nsvci,
582 gprs_ns_ll_str(nsvc), VTY_NEWLINE);
583 counter += 1;
584 }
585 vty_out(vty, "%sDeleted %d NS-VC%s",
586 dry_run ? "Not " : "", counter, VTY_NEWLINE);
587 }
588
589 return CMD_SUCCESS;
590}
591
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200592#define GBPROXY_DELETE_LINK_STR \
593 "Delete a GBProxy logical link entry by NSEI and identification\nNSEI number\n"
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200594
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200595DEFUN(delete_gb_link_by_id, delete_gb_link_by_id_cmd,
596 "delete-gbproxy-link <0-65534> (tlli|imsi|sgsn-nsei) IDENT",
597 GBPROXY_DELETE_LINK_STR
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200598 "Delete entries with a matching TLLI (hex)\n"
599 "Delete entries with a matching IMSI\n"
600 "Identification to match\n")
601{
602 const uint16_t nsei = atoi(argv[0]);
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200603 enum {MATCH_TLLI = 't', MATCH_IMSI = 'i', MATCH_SGSN = 's'} match;
604 uint32_t ident = 0;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200605 const char *imsi = NULL;
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200606 struct gbproxy_peer *peer = 0;
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200607 struct gbproxy_link_info *link_info, *nxt;
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200608 struct gbproxy_patch_state *state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200609 char mi_buf[200];
610 int found = 0;
611
612 match = argv[1][0];
613
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200614 switch (match) {
615 case MATCH_TLLI: ident = strtoll(argv[2], NULL, 16); break;
616 case MATCH_IMSI: imsi = argv[2]; break;
617 case MATCH_SGSN: ident = strtoll(argv[2], NULL, 0); break;
618 };
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200619
620 peer = gbproxy_peer_by_nsei(g_cfg, nsei);
621 if (!peer) {
622 vty_out(vty, "Didn't find peer with NSEI %d%s",
623 nsei, VTY_NEWLINE);
624 return CMD_WARNING;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200625 }
626
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200627 state = &peer->patch_state;
628
Jacob Erlbeckf8562e32014-09-19 16:03:07 +0200629 llist_for_each_entry_safe(link_info, nxt, &state->logical_links, list) {
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200630 switch (match) {
631 case MATCH_TLLI:
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200632 if (link_info->tlli.current != ident)
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200633 continue;
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200634 break;
635 case MATCH_SGSN:
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200636 if (link_info->sgsn_nsei != ident)
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200637 continue;
638 break;
639 case MATCH_IMSI:
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200640 mi_buf[0] = '\0';
641 gsm48_mi_to_string(mi_buf, sizeof(mi_buf),
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200642 link_info->imsi,
643 link_info->imsi_len);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200644
645 if (strcmp(mi_buf, imsi) != 0)
646 continue;
Jacob Erlbeck91a0e862014-09-17 10:56:38 +0200647 break;
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200648 }
649
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200650 vty_out(vty, "Deleting link with TLLI %08x%s", link_info->tlli.current,
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200651 VTY_NEWLINE);
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200652 gbproxy_delete_link_info(peer, link_info);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200653 found += 1;
654 }
655
656 if (!found && argc >= 2) {
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200657 vty_out(vty, "Didn't find link entry with %s %s%s",
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200658 argv[1], argv[2], VTY_NEWLINE);
659 }
660
661 return CMD_SUCCESS;
662}
663
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200664DEFUN(delete_gb_link, delete_gb_link_cmd,
665 "delete-gbproxy-link <0-65534> (stale|de-registered)",
666 GBPROXY_DELETE_LINK_STR
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200667 "Delete stale entries\n"
668 "Delete de-registered entries\n")
669{
670 const uint16_t nsei = atoi(argv[0]);
671 enum {MATCH_STALE = 's', MATCH_DEREGISTERED = 'd'} match;
672 struct gbproxy_peer *peer = 0;
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200673 struct gbproxy_link_info *link_info, *nxt;
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200674 struct gbproxy_patch_state *state;
Jacob Erlbeckc6807c42014-09-19 16:34:01 +0200675 time_t now;
676 struct timespec ts = {0,};
677
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200678 int found = 0;
679
680 match = argv[1][0];
681
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200682 peer = gbproxy_peer_by_nsei(g_cfg, nsei);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200683 if (!peer) {
684 vty_out(vty, "Didn't find peer with NSEI %d%s",
685 nsei, VTY_NEWLINE);
686 return CMD_WARNING;
687 }
688
689 state = &peer->patch_state;
690
Jacob Erlbeckc6807c42014-09-19 16:34:01 +0200691 clock_gettime(CLOCK_MONOTONIC, &ts);
692 now = ts.tv_sec;
693
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200694 if (match == MATCH_STALE) {
Jacob Erlbeckc6807c42014-09-19 16:34:01 +0200695 found = gbproxy_remove_stale_link_infos(peer, now);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200696 if (found)
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200697 vty_out(vty, "Deleted %d stale logical link%s%s",
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200698 found, found == 1 ? "" : "s", VTY_NEWLINE);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200699 } else {
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200700 llist_for_each_entry_safe(link_info, nxt,
Jacob Erlbeckf8562e32014-09-19 16:03:07 +0200701 &state->logical_links, list) {
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200702 if (!link_info->is_deregistered)
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200703 continue;
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200704
Jacob Erlbeck91d2f8a2014-09-19 15:07:27 +0200705 gbproxy_delete_link_info(peer, link_info);
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200706 found += 1;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200707 }
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200708 }
709
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200710 if (found)
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200711 vty_out(vty, "Deleted %d %s logical link%s%s",
Jacob Erlbeckcdd37832014-09-16 08:56:26 +0200712 found, argv[1], found == 1 ? "" : "s", VTY_NEWLINE);
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200713
714 return CMD_SUCCESS;
715}
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200716
Harald Welte799e0c92010-04-30 21:49:24 +0200717int gbproxy_vty_init(void)
718{
Harald Welte995a2d32010-05-12 16:50:52 +0000719 install_element_ve(&show_gbproxy_cmd);
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200720 install_element_ve(&show_gbproxy_links_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200721
Jacob Erlbeck4211d792013-10-24 12:48:23 +0200722 install_element(ENABLE_NODE, &delete_gb_bvci_cmd);
723 install_element(ENABLE_NODE, &delete_gb_nsei_cmd);
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200724 install_element(ENABLE_NODE, &delete_gb_link_by_id_cmd);
725 install_element(ENABLE_NODE, &delete_gb_link_cmd);
Holger Hans Peter Freyther90267a92013-10-23 11:24:17 +0200726
Harald Welte799e0c92010-04-30 21:49:24 +0200727 install_element(CONFIG_NODE, &cfg_gbproxy_cmd);
728 install_node(&gbproxy_node, config_write_gbproxy);
Jacob Erlbeck36722e12013-10-29 09:30:30 +0100729 vty_install_default(GBPROXY_NODE);
Harald Welte799e0c92010-04-30 21:49:24 +0200730 install_element(GBPROXY_NODE, &cfg_nsip_sgsn_nsei_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200731 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mcc_cmd);
732 install_element(GBPROXY_NODE, &cfg_gbproxy_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200733 install_element(GBPROXY_NODE, &cfg_gbproxy_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200734 install_element(GBPROXY_NODE, &cfg_gbproxy_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200735 install_element(GBPROXY_NODE, &cfg_gbproxy_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200736 install_element(GBPROXY_NODE, &cfg_gbproxy_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200737 install_element(GBPROXY_NODE, &cfg_gbproxy_acquire_imsi_cmd);
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200738 install_element(GBPROXY_NODE, &cfg_gbproxy_link_list_max_age_cmd);
739 install_element(GBPROXY_NODE, &cfg_gbproxy_link_list_max_len_cmd);
740 install_element(GBPROXY_NODE, &cfg_gbproxy_link_list_keep_mode_cmd);
Jacob Erlbeck67a44452014-05-19 10:14:58 +0200741 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mcc_cmd);
742 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_mnc_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200743 install_element(GBPROXY_NODE, &cfg_gbproxy_no_match_imsi_cmd);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200744 install_element(GBPROXY_NODE, &cfg_gbproxy_no_core_apn_cmd);
Jacob Erlbeckcd9e1c92014-09-02 13:57:19 +0200745 install_element(GBPROXY_NODE, &cfg_gbproxy_no_secondary_sgsn_cmd);
Jacob Erlbeck0d4236b2014-08-18 14:54:37 +0200746 install_element(GBPROXY_NODE, &cfg_gbproxy_no_patch_ptmsi_cmd);
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200747 install_element(GBPROXY_NODE, &cfg_gbproxy_no_acquire_imsi_cmd);
Jacob Erlbeckd4c79a42014-09-19 15:36:44 +0200748 install_element(GBPROXY_NODE, &cfg_gbproxy_link_list_no_max_age_cmd);
749 install_element(GBPROXY_NODE, &cfg_gbproxy_link_list_no_max_len_cmd);
Harald Welte799e0c92010-04-30 21:49:24 +0200750
751 return 0;
752}
753
754int gbproxy_parse_config(const char *config_file, struct gbproxy_config *cfg)
755{
756 int rc;
757
758 g_cfg = cfg;
Harald Weltedcccb182010-05-16 20:52:23 +0200759 rc = vty_read_config_file(config_file, NULL);
Harald Welte799e0c92010-04-30 21:49:24 +0200760 if (rc < 0) {
761 fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file);
762 return rc;
763 }
764
765 return 0;
766}
767