blob: 5fb230f4c4390d5072dedcf470bc57879e274282 [file] [log] [blame]
Harald Welte288be162010-05-01 16:48:27 +02001/*
Harald Welte7f6da482013-03-19 11:00:13 +01002 * (C) 2010-2013 by Harald Welte <laforge@gnumonks.org>
Harald Welte288be162010-05-01 16:48:27 +02003 * (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 Welte288be162010-05-01 16:48:27 +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 Welte288be162010-05-01 16:48:27 +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 Welte288be162010-05-01 16:48:27 +020018 *
19 */
20
Harald Welte288be162010-05-01 16:48:27 +020021#include <sys/socket.h>
22#include <netinet/in.h>
23#include <arpa/inet.h>
24
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010025#include <osmocom/core/talloc.h>
26#include <osmocom/core/utils.h>
27#include <osmocom/core/rate_ctr.h>
Harald Welte288be162010-05-01 16:48:27 +020028
29#include <openbsc/debug.h>
30#include <openbsc/sgsn.h>
Harald Welteea34a4e2012-06-16 14:59:56 +080031#include <osmocom/gprs/gprs_ns.h>
Harald Welted193cb32010-05-17 22:58:03 +020032#include <openbsc/gprs_sgsn.h>
Harald Welte62ab20c2010-05-14 18:59:17 +020033#include <openbsc/vty.h>
Harald Weltec5d4a0c2010-07-02 22:47:59 +020034#include <openbsc/gsm_04_08_gprs.h>
Harald Welte288be162010-05-01 16:48:27 +020035
Harald Welte4b037e42010-05-19 19:45:32 +020036#include <osmocom/vty/command.h>
37#include <osmocom/vty/vty.h>
Pablo Neira Ayuso6110a3f2011-03-28 19:35:00 +020038#include <osmocom/vty/misc.h>
Harald Welte288be162010-05-01 16:48:27 +020039
Harald Welted193cb32010-05-17 22:58:03 +020040#include <pdp.h>
41
Harald Welte288be162010-05-01 16:48:27 +020042static struct sgsn_config *g_cfg = NULL;
43
Jacob Erlbeck106f5472014-11-04 10:08:37 +010044const struct value_string sgsn_auth_pol_strs[] = {
45 { SGSN_AUTH_POLICY_OPEN, "accept-all" },
46 { SGSN_AUTH_POLICY_CLOSED, "closed" },
47 { SGSN_AUTH_POLICY_ACL_ONLY, "acl-only" },
Jacob Erlbeckbe2c8d92014-11-12 10:18:09 +010048 { SGSN_AUTH_POLICY_REMOTE, "remote" },
Jacob Erlbeck106f5472014-11-04 10:08:37 +010049 { 0, NULL }
50};
51
52
Harald Weltec5d4a0c2010-07-02 22:47:59 +020053#define GSM48_MAX_APN_LEN 102 /* 10.5.6.1 */
54static char *gprs_apn2str(uint8_t *apn, unsigned int len)
55{
56 static char apnbuf[GSM48_MAX_APN_LEN+1];
Holger Hans Peter Freyther80e03652013-07-04 18:44:16 +020057 unsigned int i = 0;
Harald Weltec5d4a0c2010-07-02 22:47:59 +020058
59 if (!apn)
60 return "";
61
62 if (len > sizeof(apnbuf)-1)
63 len = sizeof(apnbuf)-1;
64
65 memcpy(apnbuf, apn, len);
66 apnbuf[len] = '\0';
67
68 /* replace the domain name step sizes with dots */
69 while (i < len) {
70 unsigned int step = apnbuf[i];
71 apnbuf[i] = '.';
72 i += step+1;
73 }
74
75 return apnbuf+1;
76}
77
Holger Hans Peter Freythera2730302014-03-23 18:08:26 +010078char *gprs_pdpaddr2str(uint8_t *pdpa, uint8_t len)
Harald Weltec5d4a0c2010-07-02 22:47:59 +020079{
80 static char str[INET6_ADDRSTRLEN + 10];
81
82 if (!pdpa || len < 2)
83 return "none";
84
85 switch (pdpa[0] & 0x0f) {
86 case PDP_TYPE_ORG_IETF:
87 switch (pdpa[1]) {
88 case PDP_TYPE_N_IETF_IPv4:
89 if (len < 2 + 4)
90 break;
91 strcpy(str, "IPv4 ");
92 inet_ntop(AF_INET, pdpa+2, str+5, sizeof(str)-5);
93 return str;
94 case PDP_TYPE_N_IETF_IPv6:
95 if (len < 2 + 8)
96 break;
97 strcpy(str, "IPv6 ");
98 inet_ntop(AF_INET6, pdpa+2, str+5, sizeof(str)-5);
99 return str;
100 default:
101 break;
102 }
103 break;
104 case PDP_TYPE_ORG_ETSI:
105 if (pdpa[1] == PDP_TYPE_N_ETSI_PPP)
106 return "PPP";
107 break;
108 default:
109 break;
110 }
111
112 return "invalid";
113}
114
Harald Welte288be162010-05-01 16:48:27 +0200115static struct cmd_node sgsn_node = {
116 SGSN_NODE,
Harald Welte570ce242012-08-17 13:16:10 +0200117 "%s(config-sgsn)# ",
Harald Welte288be162010-05-01 16:48:27 +0200118 1,
119};
120
121static int config_write_sgsn(struct vty *vty)
122{
Harald Welte77289c22010-05-18 14:32:29 +0200123 struct sgsn_ggsn_ctx *gctx;
Harald Welte7f6da482013-03-19 11:00:13 +0100124 struct imsi_acl_entry *acl;
Harald Welte288be162010-05-01 16:48:27 +0200125
126 vty_out(vty, "sgsn%s", VTY_NEWLINE);
127
Harald Weltee300d002010-06-02 12:41:34 +0200128 vty_out(vty, " gtp local-ip %s%s",
129 inet_ntoa(g_cfg->gtp_listenaddr.sin_addr), VTY_NEWLINE);
130
Harald Welted193cb32010-05-17 22:58:03 +0200131 llist_for_each_entry(gctx, &sgsn_ggsn_ctxts, list) {
Harald Welteff3bde82010-05-19 15:09:09 +0200132 vty_out(vty, " ggsn %u remote-ip %s%s", gctx->id,
Harald Welted193cb32010-05-17 22:58:03 +0200133 inet_ntoa(gctx->remote_addr), VTY_NEWLINE);
Harald Welteff3bde82010-05-19 15:09:09 +0200134 vty_out(vty, " ggsn %u gtp-version %u%s", gctx->id,
Harald Welted193cb32010-05-17 22:58:03 +0200135 gctx->gtp_version, VTY_NEWLINE);
Harald Welte288be162010-05-01 16:48:27 +0200136 }
137
Harald Welte3dfb5492013-03-19 11:48:54 +0100138 vty_out(vty, " auth-policy %s%s",
Jacob Erlbeck106f5472014-11-04 10:08:37 +0100139 get_value_string(sgsn_auth_pol_strs, g_cfg->auth_policy),
140 VTY_NEWLINE);
Harald Welte7f6da482013-03-19 11:00:13 +0100141 llist_for_each_entry(acl, &g_cfg->imsi_acl, list)
142 vty_out(vty, " imsi-acl add %s%s", acl->imsi, VTY_NEWLINE);
143
Harald Welte288be162010-05-01 16:48:27 +0200144 return CMD_SUCCESS;
145}
146
Holger Hans Peter Freyther1491f2e2011-11-05 15:21:16 +0100147#define SGSN_STR "Configure the SGSN\n"
148#define GGSN_STR "Configure the GGSN information\n"
Harald Weltee300d002010-06-02 12:41:34 +0200149
150DEFUN(cfg_sgsn, cfg_sgsn_cmd,
151 "sgsn",
152 SGSN_STR)
Harald Welte288be162010-05-01 16:48:27 +0200153{
154 vty->node = SGSN_NODE;
155 return CMD_SUCCESS;
156}
157
Harald Weltee300d002010-06-02 12:41:34 +0200158DEFUN(cfg_sgsn_bind_addr, cfg_sgsn_bind_addr_cmd,
159 "gtp local-ip A.B.C.D",
160 "GTP Parameters\n"
Holger Hans Peter Freyther1491f2e2011-11-05 15:21:16 +0100161 "Set the IP address for the local GTP bind\n"
162 "IPv4 Address\n")
Harald Weltee300d002010-06-02 12:41:34 +0200163{
164 inet_aton(argv[0], &g_cfg->gtp_listenaddr.sin_addr);
165
166 return CMD_SUCCESS;
167}
168
Harald Welted193cb32010-05-17 22:58:03 +0200169DEFUN(cfg_ggsn_remote_ip, cfg_ggsn_remote_ip_cmd,
170 "ggsn <0-255> remote-ip A.B.C.D",
Holger Hans Peter Freyther1491f2e2011-11-05 15:21:16 +0100171 GGSN_STR "GGSN Number\n" IP_STR "IPv4 Address\n")
Harald Welted193cb32010-05-17 22:58:03 +0200172{
173 uint32_t id = atoi(argv[0]);
Harald Welte77289c22010-05-18 14:32:29 +0200174 struct sgsn_ggsn_ctx *ggc = sgsn_ggsn_ctx_find_alloc(id);
Harald Welte288be162010-05-01 16:48:27 +0200175
Harald Welted193cb32010-05-17 22:58:03 +0200176 inet_aton(argv[1], &ggc->remote_addr);
Harald Welte288be162010-05-01 16:48:27 +0200177
Harald Welted193cb32010-05-17 22:58:03 +0200178 return CMD_SUCCESS;
179}
180
181#if 0
182DEFUN(cfg_ggsn_remote_port, cfg_ggsn_remote_port_cmd,
183 "ggsn <0-255> remote-port <0-65535>",
184 "")
185{
186 uint32_t id = atoi(argv[0]);
Harald Welte77289c22010-05-18 14:32:29 +0200187 struct sgsn_ggsn_ctx *ggc = sgsn_ggsn_ctx_find_alloc(id);
Harald Welted193cb32010-05-17 22:58:03 +0200188 uint16_t port = atoi(argv[1]);
189
190}
191#endif
192
193DEFUN(cfg_ggsn_gtp_version, cfg_ggsn_gtp_version_cmd,
194 "ggsn <0-255> gtp-version (0|1)",
Holger Hans Peter Freyther1491f2e2011-11-05 15:21:16 +0100195 GGSN_STR "GGSN Number\n" "GTP Version\n"
196 "Version 0\n" "Version 1\n")
Harald Welted193cb32010-05-17 22:58:03 +0200197{
198 uint32_t id = atoi(argv[0]);
Harald Welte77289c22010-05-18 14:32:29 +0200199 struct sgsn_ggsn_ctx *ggc = sgsn_ggsn_ctx_find_alloc(id);
Harald Welted193cb32010-05-17 22:58:03 +0200200
201 if (atoi(argv[1]))
202 ggc->gtp_version = 1;
203 else
204 ggc->gtp_version = 0;
205
206 return CMD_SUCCESS;
207}
208
209#if 0
210DEFUN(cfg_apn_ggsn, cfg_apn_ggsn_cmd,
211 "apn APNAME ggsn <0-255>",
212 "")
213{
214 struct apn_ctx **
215}
216#endif
217
218const struct value_string gprs_mm_st_strs[] = {
219 { GMM_DEREGISTERED, "DEREGISTERED" },
220 { GMM_COMMON_PROC_INIT, "COMMON PROCEDURE (INIT)" },
221 { GMM_REGISTERED_NORMAL, "REGISTERED (NORMAL)" },
Harald Weltebffeff82010-06-09 15:50:45 +0200222 { GMM_REGISTERED_SUSPENDED, "REGISTERED (SUSPENDED)" },
Harald Welted193cb32010-05-17 22:58:03 +0200223 { GMM_DEREGISTERED_INIT, "DEREGISTERED (INIT)" },
224 { 0, NULL }
225};
226
227static void vty_dump_pdp(struct vty *vty, const char *pfx,
228 struct sgsn_pdp_ctx *pdp)
229{
Jacob Erlbeck99985b52014-10-13 10:32:00 +0200230 const char *imsi = pdp->mm ? pdp->mm->imsi : "(detaching)";
Harald Welted193cb32010-05-17 22:58:03 +0200231 vty_out(vty, "%sPDP Context IMSI: %s, SAPI: %u, NSAPI: %u%s",
Jacob Erlbeck99985b52014-10-13 10:32:00 +0200232 pfx, imsi, pdp->sapi, pdp->nsapi, VTY_NEWLINE);
Harald Weltec5d4a0c2010-07-02 22:47:59 +0200233 vty_out(vty, "%s APN: %s%s", pfx,
234 gprs_apn2str(pdp->lib->apn_use.v, pdp->lib->apn_use.l),
235 VTY_NEWLINE);
236 vty_out(vty, "%s PDP Address: %s%s", pfx,
237 gprs_pdpaddr2str(pdp->lib->eua.v, pdp->lib->eua.l),
238 VTY_NEWLINE);
Harald Welteefbdee92010-06-10 00:20:12 +0200239 vty_out_rate_ctr_group(vty, " ", pdp->ctrg);
Harald Welted193cb32010-05-17 22:58:03 +0200240}
241
242static void vty_dump_mmctx(struct vty *vty, const char *pfx,
243 struct sgsn_mm_ctx *mm, int pdp)
244{
245 vty_out(vty, "%sMM Context for IMSI %s, IMEI %s, P-TMSI %08x%s",
246 pfx, mm->imsi, mm->imei, mm->p_tmsi, VTY_NEWLINE);
247 vty_out(vty, "%s MSISDN: %s, TLLI: %08x%s", pfx, mm->msisdn,
248 mm->tlli, VTY_NEWLINE);
249 vty_out(vty, "%s MM State: %s, Routeing Area: %u-%u-%u-%u, "
250 "Cell ID: %u%s", pfx,
251 get_value_string(gprs_mm_st_strs, mm->mm_state),
252 mm->ra.mcc, mm->ra.mnc, mm->ra.lac, mm->ra.rac,
253 mm->cell_id, VTY_NEWLINE);
254
Harald Welte8acd88f2010-05-18 10:57:45 +0200255 vty_out_rate_ctr_group(vty, " ", mm->ctrg);
256
Harald Welted193cb32010-05-17 22:58:03 +0200257 if (pdp) {
258 struct sgsn_pdp_ctx *pdp;
259
260 llist_for_each_entry(pdp, &mm->pdp_list, list)
261 vty_dump_pdp(vty, " ", pdp);
262 }
263}
264
265DEFUN(show_sgsn, show_sgsn_cmd, "show sgsn",
266 SHOW_STR "Display information about the SGSN")
267{
268 /* FIXME: statistics */
269 return CMD_SUCCESS;
270}
271
272#define MMCTX_STR "MM Context\n"
273#define INCLUDE_PDP_STR "Include PDP Context Information\n"
274
275#if 0
276DEFUN(show_mmctx_tlli, show_mmctx_tlli_cmd,
277 "show mm-context tlli HEX [pdp]",
278 SHOW_STR MMCTX_STR "Identify by TLLI\n" "TLLI\n" INCLUDE_PDP_STR)
279{
280 uint32_t tlli;
281 struct sgsn_mm_ctx *mm;
282
283 tlli = strtoul(argv[0], NULL, 16);
284 mm = sgsn_mm_ctx_by_tlli(tlli);
285 if (!mm) {
286 vty_out(vty, "No MM context for TLLI %08x%s",
287 tlli, VTY_NEWLINE);
288 return CMD_WARNING;
289 }
290 vty_dump_mmctx(vty, "", mm, argv[1] ? 1 : 0);
291 return CMD_SUCCESS;
292}
293#endif
294
295DEFUN(swow_mmctx_imsi, show_mmctx_imsi_cmd,
296 "show mm-context imsi IMSI [pdp]",
297 SHOW_STR MMCTX_STR "Identify by IMSI\n" "IMSI of the MM Context\n"
298 INCLUDE_PDP_STR)
299{
300 struct sgsn_mm_ctx *mm;
301
302 mm = sgsn_mm_ctx_by_imsi(argv[0]);
303 if (!mm) {
304 vty_out(vty, "No MM context for IMSI %s%s",
305 argv[0], VTY_NEWLINE);
306 return CMD_WARNING;
307 }
308 vty_dump_mmctx(vty, "", mm, argv[1] ? 1 : 0);
309 return CMD_SUCCESS;
310}
311
312DEFUN(swow_mmctx_all, show_mmctx_all_cmd,
313 "show mm-context all [pdp]",
314 SHOW_STR MMCTX_STR "All MM Contexts\n" INCLUDE_PDP_STR)
315{
316 struct sgsn_mm_ctx *mm;
317
318 llist_for_each_entry(mm, &sgsn_mm_ctxts, list)
319 vty_dump_mmctx(vty, "", mm, argv[0] ? 1 : 0);
320
321 return CMD_SUCCESS;
322}
323
Harald Welted193cb32010-05-17 22:58:03 +0200324DEFUN(show_pdpctx_all, show_pdpctx_all_cmd,
325 "show pdp-context all",
Holger Hans Peter Freyther1491f2e2011-11-05 15:21:16 +0100326 SHOW_STR "Display information on PDP Context\n" "Show everything\n")
Harald Welted193cb32010-05-17 22:58:03 +0200327{
328 struct sgsn_pdp_ctx *pdp;
329
330 llist_for_each_entry(pdp, &sgsn_pdp_ctxts, g_list)
331 vty_dump_pdp(vty, "", pdp);
332
333 return CMD_SUCCESS;
334}
Harald Welte288be162010-05-01 16:48:27 +0200335
Harald Welte7f6da482013-03-19 11:00:13 +0100336
337DEFUN(imsi_acl, cfg_imsi_acl_cmd,
338 "imsi-acl (add|del) IMSI",
339 "Access Control List of foreign IMSIs\n"
340 "Add IMSI to ACL\n"
341 "Remove IMSI from ACL\n"
342 "IMSI of subscriber\n")
343{
344 const char *op = argv[0];
345 const char *imsi = argv[1];
346 int rc;
347
348 if (!strcmp(op, "add"))
Jacob Erlbeck3b5d4072014-10-24 15:11:03 +0200349 rc = sgsn_acl_add(imsi, g_cfg);
Harald Welte7f6da482013-03-19 11:00:13 +0100350 else
Jacob Erlbeck3b5d4072014-10-24 15:11:03 +0200351 rc = sgsn_acl_del(imsi, g_cfg);
Harald Welte7f6da482013-03-19 11:00:13 +0100352
353 if (rc < 0) {
354 vty_out(vty, "%% unable to %s ACL\n", op);
355 return CMD_WARNING;
356 }
357
358 return CMD_SUCCESS;
359}
360
Harald Welte3dfb5492013-03-19 11:48:54 +0100361DEFUN(cfg_auth_policy, cfg_auth_policy_cmd,
Jacob Erlbeckbe2c8d92014-11-12 10:18:09 +0100362 "auth-policy (accept-all|closed|acl-only|remote)",
Harald Welte3dfb5492013-03-19 11:48:54 +0100363 "Autorization Policy of SGSN\n"
Jacob Erlbeck106f5472014-11-04 10:08:37 +0100364 "Accept all IMSIs (DANGEROUS)\n"
365 "Accept only home network subscribers or those in the ACL\n"
Jacob Erlbeckbe2c8d92014-11-12 10:18:09 +0100366 "Accept only subscribers in the ACL\n"
367 "Use remote subscription data only (HLR)\n")
Harald Welte3dfb5492013-03-19 11:48:54 +0100368{
Jacob Erlbeck106f5472014-11-04 10:08:37 +0100369 int val = get_string_value(sgsn_auth_pol_strs, argv[0]);
Jacob Erlbeckbe2c8d92014-11-12 10:18:09 +0100370 OSMO_ASSERT(val >= SGSN_AUTH_POLICY_OPEN && val <= SGSN_AUTH_POLICY_REMOTE);
Jacob Erlbeck106f5472014-11-04 10:08:37 +0100371 g_cfg->auth_policy = val;
Harald Welte3dfb5492013-03-19 11:48:54 +0100372
373 return CMD_SUCCESS;
374}
375
Harald Welte288be162010-05-01 16:48:27 +0200376int sgsn_vty_init(void)
377{
Harald Welted193cb32010-05-17 22:58:03 +0200378 install_element_ve(&show_sgsn_cmd);
379 //install_element_ve(&show_mmctx_tlli_cmd);
380 install_element_ve(&show_mmctx_imsi_cmd);
381 install_element_ve(&show_mmctx_all_cmd);
382 install_element_ve(&show_pdpctx_all_cmd);
Harald Welte288be162010-05-01 16:48:27 +0200383
384 install_element(CONFIG_NODE, &cfg_sgsn_cmd);
385 install_node(&sgsn_node, config_write_sgsn);
Jacob Erlbeck36722e12013-10-29 09:30:30 +0100386 vty_install_default(SGSN_NODE);
Harald Weltee300d002010-06-02 12:41:34 +0200387 install_element(SGSN_NODE, &cfg_sgsn_bind_addr_cmd);
Harald Welted193cb32010-05-17 22:58:03 +0200388 install_element(SGSN_NODE, &cfg_ggsn_remote_ip_cmd);
389 //install_element(SGSN_NODE, &cfg_ggsn_remote_port_cmd);
390 install_element(SGSN_NODE, &cfg_ggsn_gtp_version_cmd);
Harald Welte7f6da482013-03-19 11:00:13 +0100391 install_element(SGSN_NODE, &cfg_imsi_acl_cmd);
Harald Welte3dfb5492013-03-19 11:48:54 +0100392 install_element(SGSN_NODE, &cfg_auth_policy_cmd);
Harald Welte288be162010-05-01 16:48:27 +0200393
394 return 0;
395}
396
397int sgsn_parse_config(const char *config_file, struct sgsn_config *cfg)
398{
399 int rc;
400
401 g_cfg = cfg;
Harald Welte7f6da482013-03-19 11:00:13 +0100402
Harald Weltedcccb182010-05-16 20:52:23 +0200403 rc = vty_read_config_file(config_file, NULL);
Harald Welte288be162010-05-01 16:48:27 +0200404 if (rc < 0) {
405 fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file);
406 return rc;
407 }
408
409 return 0;
410}