blob: 5e5c1ddf719795576eb3e3ff995f0c90947d4275 [file] [log] [blame]
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +02001/* (C) 2015 by sysmocom s.f.m.c. GmbH
2 * All Rights Reserved
3 *
4 * Author: Neels Hofmeyr
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * 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
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Affero General Public License for more details.
15 *
16 * 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/>.
18 *
19 */
20
21#include <string.h>
22
Neels Hofmeyrb6c2db52015-11-18 18:11:32 +010023#include <ares.h>
24#include <sys/socket.h>
25#include <netinet/in.h>
26#include <arpa/inet.h>
27
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +020028#include <osmocom/core/talloc.h>
29#include <osmocom/vty/command.h>
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +010030#include <osmocom/vty/misc.h>
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +020031
32#include <openbsc/vty.h>
33#include <openbsc/gtphub.h>
34
Neels Hofmeyrb6c2db52015-11-18 18:11:32 +010035/* TODO split GRX ares from sgsn into a separate struct and allow use without
36 * globals. */
37#include <openbsc/sgsn.h>
38extern struct sgsn_instance *sgsn;
39
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +010040static struct gtphub *g_hub = 0;
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +020041static struct gtphub_cfg *g_cfg = 0;
42
43static struct cmd_node gtphub_node = {
44 GTPHUB_NODE,
45 "%s(config-gtphub)# ",
46 1,
47};
48
49#define GTPH_DEFAULT_CONTROL_PORT 2123
50#define GTPH_DEFAULT_USER_PORT 2152
51
52static void write_addrs(struct vty *vty, const char *name,
53 struct gtphub_cfg_addr *c, struct gtphub_cfg_addr *u)
54{
55 if ((c->port == GTPH_DEFAULT_CONTROL_PORT)
56 && (u->port == GTPH_DEFAULT_USER_PORT)
57 && (strcmp(c->addr_str, u->addr_str) == 0)) {
58 /* Default port numbers and same IP address: write "short"
59 * variant. */
60 vty_out(vty, " %s %s%s",
61 name,
62 c->addr_str,
63 VTY_NEWLINE);
64 return;
65 }
66
67 vty_out(vty, " %s ctrl %s %d user %s %d%s",
68 name,
69 c->addr_str, (int)c->port,
70 u->addr_str, (int)u->port,
71 VTY_NEWLINE);
Neels Hofmeyrb6c2db52015-11-18 18:11:32 +010072
73 struct ares_addr_node *server;
74 for (server = sgsn->ares_servers; server; server = server->next)
75 vty_out(vty, " grx-dns-add %s%s", inet_ntoa(server->addr.addr4), VTY_NEWLINE);
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +020076}
77
78static int config_write_gtphub(struct vty *vty)
79{
80 vty_out(vty, "gtphub%s", VTY_NEWLINE);
81
82 write_addrs(vty, "bind-to-sgsns",
83 &g_cfg->to_sgsns[GTPH_PLANE_CTRL].bind,
84 &g_cfg->to_sgsns[GTPH_PLANE_USER].bind);
85
86 write_addrs(vty, "bind-to-ggsns",
87 &g_cfg->to_ggsns[GTPH_PLANE_CTRL].bind,
88 &g_cfg->to_ggsns[GTPH_PLANE_USER].bind);
89
90 if (g_cfg->ggsn_proxy[GTPH_PLANE_CTRL].addr_str) {
91 write_addrs(vty, "ggsn-proxy",
92 &g_cfg->ggsn_proxy[GTPH_PLANE_CTRL],
93 &g_cfg->ggsn_proxy[GTPH_PLANE_USER]);
94 }
95
96 return CMD_SUCCESS;
97}
98
99DEFUN(cfg_gtphub, cfg_gtphub_cmd,
100 "gtphub",
101 "Configure the GTP hub")
102{
103 vty->node = GTPHUB_NODE;
104 return CMD_SUCCESS;
105}
106
107#define BIND_ARGS "ctrl ADDR <0-65535> user ADDR <0-65535>"
108#define BIND_DOCS \
109 "Set GTP-C bind\n" \
110 "GTP-C local IP address (v4 or v6)\n" \
111 "GTP-C local port\n" \
112 "Set GTP-U bind\n" \
113 "GTP-U local IP address (v4 or v6)\n" \
114 "GTP-U local port\n"
115
116
117DEFUN(cfg_gtphub_bind_to_sgsns_short, cfg_gtphub_bind_to_sgsns_short_cmd,
118 "bind-to-sgsns ADDR",
119 "GTP Hub Parameters\n"
120 "Set the local bind address to listen for SGSNs, for both GTP-C and GTP-U\n"
121 "Local IP address (v4 or v6)\n"
122 )
123{
124 int i;
Neels Hofmeyrf9773202015-11-27 00:05:56 +0100125 for_each_plane(i)
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200126 g_cfg->to_sgsns[i].bind.addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
127 g_cfg->to_sgsns[GTPH_PLANE_CTRL].bind.port = GTPH_DEFAULT_CONTROL_PORT;
128 g_cfg->to_sgsns[GTPH_PLANE_USER].bind.port = GTPH_DEFAULT_USER_PORT;
129 return CMD_SUCCESS;
130}
131
132DEFUN(cfg_gtphub_bind_to_ggsns_short, cfg_gtphub_bind_to_ggsns_short_cmd,
133 "bind-to-ggsns ADDR",
134 "GTP Hub Parameters\n"
135 "Set the local bind address to listen for GGSNs, for both GTP-C and GTP-U\n"
136 "Local IP address (v4 or v6)\n"
137 )
138{
139 int i;
Neels Hofmeyrf9773202015-11-27 00:05:56 +0100140 for_each_plane(i)
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200141 g_cfg->to_ggsns[i].bind.addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
142 g_cfg->to_ggsns[GTPH_PLANE_CTRL].bind.port = GTPH_DEFAULT_CONTROL_PORT;
143 g_cfg->to_ggsns[GTPH_PLANE_USER].bind.port = GTPH_DEFAULT_USER_PORT;
144 return CMD_SUCCESS;
145}
146
147
148static int handle_binds(struct gtphub_cfg_bind *b, const char **argv)
149{
150 b[GTPH_PLANE_CTRL].bind.addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
151 b[GTPH_PLANE_CTRL].bind.port = atoi(argv[1]);
152 b[GTPH_PLANE_USER].bind.addr_str = talloc_strdup(tall_vty_ctx, argv[2]);
153 b[GTPH_PLANE_USER].bind.port = atoi(argv[3]);
154 return CMD_SUCCESS;
155}
156
157DEFUN(cfg_gtphub_bind_to_sgsns, cfg_gtphub_bind_to_sgsns_cmd,
158 "bind-to-sgsns " BIND_ARGS,
159 "GTP Hub Parameters\n"
160 "Set the local bind addresses and ports to listen for SGSNs\n"
161 BIND_DOCS
162 )
163{
164 return handle_binds(g_cfg->to_sgsns, argv);
165}
166
167DEFUN(cfg_gtphub_bind_to_ggsns, cfg_gtphub_bind_to_ggsns_cmd,
168 "bind-to-ggsns " BIND_ARGS,
169 "GTP Hub Parameters\n"
170 "Set the local bind addresses and ports to listen for GGSNs\n"
171 BIND_DOCS
172 )
173{
174 return handle_binds(g_cfg->to_ggsns, argv);
175}
176
177DEFUN(cfg_gtphub_ggsn_proxy_short, cfg_gtphub_ggsn_proxy_short_cmd,
178 "ggsn-proxy ADDR",
179 "GTP Hub Parameters\n"
180 "Redirect all GGSN bound traffic to default ports on this address (another gtphub)\n"
181 "Remote IP address (v4 or v6)\n"
182 )
183{
184 g_cfg->ggsn_proxy[GTPH_PLANE_CTRL].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
185 g_cfg->ggsn_proxy[GTPH_PLANE_CTRL].port = GTPH_DEFAULT_CONTROL_PORT;
186 g_cfg->ggsn_proxy[GTPH_PLANE_USER].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
187 g_cfg->ggsn_proxy[GTPH_PLANE_USER].port = GTPH_DEFAULT_USER_PORT;
188 return CMD_SUCCESS;
189}
190
191DEFUN(cfg_gtphub_ggsn_proxy, cfg_gtphub_ggsn_proxy_cmd,
192 "ggsn-proxy " BIND_ARGS,
193 "GTP Hub Parameters\n"
194 "Redirect all GGSN bound traffic to these addresses and ports (another gtphub)\n"
195 BIND_DOCS
196 )
197{
198 g_cfg->ggsn_proxy[GTPH_PLANE_CTRL].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
199 g_cfg->ggsn_proxy[GTPH_PLANE_CTRL].port = atoi(argv[1]);
200 g_cfg->ggsn_proxy[GTPH_PLANE_USER].addr_str = talloc_strdup(tall_vty_ctx, argv[2]);
201 g_cfg->ggsn_proxy[GTPH_PLANE_USER].port = atoi(argv[3]);
202 return CMD_SUCCESS;
203}
204
205DEFUN(cfg_gtphub_sgsn_proxy_short, cfg_gtphub_sgsn_proxy_short_cmd,
206 "sgsn-proxy ADDR",
207 "GTP Hub Parameters\n"
208 "Redirect all SGSN bound traffic to default ports on this address (another gtphub)\n"
209 "Remote IP address (v4 or v6)\n"
210 )
211{
212 g_cfg->sgsn_proxy[GTPH_PLANE_CTRL].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
213 g_cfg->sgsn_proxy[GTPH_PLANE_CTRL].port = GTPH_DEFAULT_CONTROL_PORT;
214 g_cfg->sgsn_proxy[GTPH_PLANE_USER].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
215 g_cfg->sgsn_proxy[GTPH_PLANE_USER].port = GTPH_DEFAULT_USER_PORT;
216 return CMD_SUCCESS;
217}
218
219DEFUN(cfg_gtphub_sgsn_proxy, cfg_gtphub_sgsn_proxy_cmd,
220 "sgsn-proxy " BIND_ARGS,
221 "GTP Hub Parameters\n"
222 "Redirect all SGSN bound traffic to these addresses and ports (another gtphub)\n"
223 BIND_DOCS
224 )
225{
226 g_cfg->sgsn_proxy[GTPH_PLANE_CTRL].addr_str = talloc_strdup(tall_vty_ctx, argv[0]);
227 g_cfg->sgsn_proxy[GTPH_PLANE_CTRL].port = atoi(argv[1]);
228 g_cfg->sgsn_proxy[GTPH_PLANE_USER].addr_str = talloc_strdup(tall_vty_ctx, argv[2]);
229 g_cfg->sgsn_proxy[GTPH_PLANE_USER].port = atoi(argv[3]);
230 return CMD_SUCCESS;
231}
232
Neels Hofmeyrb6c2db52015-11-18 18:11:32 +0100233
234/* Copied from sgsn_vty.h */
235DEFUN(cfg_grx_ggsn, cfg_grx_ggsn_cmd,
236 "grx-dns-add A.B.C.D",
237 "Add DNS server\nIPv4 address\n")
238{
239 struct ares_addr_node *node = talloc_zero(tall_bsc_ctx, struct ares_addr_node);
240 node->family = AF_INET;
241 inet_aton(argv[0], &node->addr.addr4);
242
243 node->next = sgsn->ares_servers;
244 sgsn->ares_servers = node;
245 return CMD_SUCCESS;
246}
247
248
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100249/*
250(show gtphub all, show gtphub stats, show gtphub teidmap,
251 show gtphub peers, ...)
252*/
253
254static void show_bind_stats_all(struct vty *vty)
255{
256 int plane_idx;
Neels Hofmeyrf9773202015-11-27 00:05:56 +0100257 for_each_plane(plane_idx) {
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100258 vty_out(vty, "- %s Plane:%s",
259 gtphub_plane_idx_names[plane_idx], VTY_NEWLINE);
260
261 struct gtphub_bind *b = &g_hub->to_ggsns[plane_idx];
262 vty_out(vty, " - to/from GGSNs: %s port %d%s",
263 gsn_addr_to_str(&b->local_addr), (int)b->local_port,
264 VTY_NEWLINE);
265 vty_out_rate_ctr_group(vty, " ", b->counters_io);
266
267 b = &g_hub->to_sgsns[plane_idx];
268 vty_out(vty, " - to/from SGSNs: %s port %d%s",
269 gsn_addr_to_str(&b->local_addr), (int)b->local_port,
270 VTY_NEWLINE);
271 vty_out_rate_ctr_group(vty, " ", b->counters_io);
272 }
273}
274
275/*
276static void show_peers_summary(struct vty *vty)
277{
278 int c
279 int plane_idx;
280}
281*/
282
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100283static void show_tunnels_summary(struct vty *vty)
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100284{
285 time_t now = gtphub_now();
286
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100287 const int w = 36;
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100288 int max_expiry = g_hub->expire_slowly.expiry_in_seconds;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100289 float seconds_per_step = ((float)max_expiry) / w;
290
291 /* Print TEI mapping expiry in an ASCII histogram, like:
292 TEI map summary
293 Legend: '_'=0 '.'<=1% ':'<=2% '|'<=10% '#'>10% (10.0 m/step)
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100294 CTRL: 30 mappings, valid for 360m[# :. | . : . ]1m
295 USER: 30 mappings, valid for 360m[# :. | . : . ]1m
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100296 4 TEI mappings in total, last expiry in 359.4 min
297 */
298 vty_out(vty,
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100299 "Tunnels summary%s"
300 " Legend: ' '=0 '.'<=1%% ':'<=2%% '|'<=10%% '#'>10%% (%.1f m/step)%s",
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100301 VTY_NEWLINE,
302 seconds_per_step / 60.,
303 VTY_NEWLINE);
304
305 int last_expiry = 0;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100306
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100307 unsigned int count = 0;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100308
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100309 int histogram[w];
310 memset(histogram, 0, sizeof(histogram));
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100311
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100312 struct gtphub_tunnel *t;
313 llist_for_each_entry(t, &g_hub->tunnels, entry) {
314 count ++;
315 int expiry = t->expiry_entry.expiry - now;
316 last_expiry = (last_expiry > expiry) ? last_expiry : expiry;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100317
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100318 int hi = ((float)expiry) / seconds_per_step;
319 if (hi < 0)
320 hi = 0;
321 if (hi > (w - 1))
322 hi = w - 1;
323 histogram[hi] ++;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100324 }
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100325
326 vty_out(vty,
327 " %u tunnels, valid for %dm[",
328 count, max_expiry / 60);
329
330 int i;
331 for (i = w - 1; i >= 0; i--) {
332 char c;
333 int val = histogram[i];
334 int percent = 100. * val / count;
335 if (!val)
336 c = ' ';
337 else if (percent <= 1)
338 c = '.';
339 else if (percent <= 2)
340 c = ':';
341 else if (percent <= 10)
342 c = '|';
343 else c = '#';
344 vty_out(vty, "%c", c);
345 }
346 vty_out(vty, "]1m%s", VTY_NEWLINE);
347
348 vty_out(vty, " last expiry in %.1f min%s",
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100349 ((float)last_expiry) / 60.,
350 VTY_NEWLINE);
351}
352
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100353static void show_tunnels_all(struct vty *vty)
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100354{
355 time_t now = gtphub_now();
356
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100357 vty_out(vty, "All tunnels:%s"
358 "Legend: SGSN <-> GGSN, with each:%s"
359 " <IP-Ctrl>[/<IP-User>] (<TEI-Ctrl>=<mapped>/<TEI-User>=<mapped>)%s",
360 VTY_NEWLINE, VTY_NEWLINE, VTY_NEWLINE);
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100361
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100362 unsigned int count = 0;
363 unsigned int incomplete = 0;
364 struct gtphub_tunnel *t;
365 llist_for_each_entry(t, &g_hub->tunnels, entry) {
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100366 vty_out(vty,
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100367 "(%4dm) %s%s",
368 -(int)((t->expiry_entry.expiry - now) / 60),
369 gtphub_tunnel_str(t),
370 VTY_NEWLINE);
371 count ++;
372 if (!gtphub_tunnel_complete(t))
373 incomplete ++;
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100374 }
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100375 vty_out(vty, "Total: %u tunnels%s", count, VTY_NEWLINE);
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100376}
377
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100378DEFUN(show_gtphub_tunnels_summary, show_gtphub_tunnels_summary_cmd, "show gtphub tunnels summary",
379 SHOW_STR "Summary of all tunnels")
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100380{
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100381 show_tunnels_summary(vty);
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100382 return CMD_SUCCESS;
383}
384
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100385DEFUN(show_gtphub_tunnels_list, show_gtphub_tunnels_list_cmd, "show gtphub tunnels list",
386 SHOW_STR "List all tunnels")
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100387{
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100388 show_tunnels_all(vty);
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100389 return CMD_SUCCESS;
390}
391
392DEFUN(show_gtphub, show_gtphub_cmd, "show gtphub all",
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200393 SHOW_STR "Display information about the GTP hub")
394{
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100395 show_bind_stats_all(vty);
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100396 show_tunnels_summary(vty);
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200397 return CMD_SUCCESS;
398}
399
400
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100401int gtphub_vty_init(struct gtphub *global_hub, struct gtphub_cfg *global_cfg)
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200402{
Neels Hofmeyr4b2cbda2015-11-20 03:16:19 +0100403 g_hub = global_hub;
404 g_cfg = global_cfg;
405
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200406 install_element_ve(&show_gtphub_cmd);
Neels Hofmeyre54cd152015-11-24 13:31:06 +0100407 install_element_ve(&show_gtphub_tunnels_summary_cmd);
408 install_element_ve(&show_gtphub_tunnels_list_cmd);
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200409
410 install_element(CONFIG_NODE, &cfg_gtphub_cmd);
411 install_node(&gtphub_node, config_write_gtphub);
412 vty_install_default(GTPHUB_NODE);
413
414 install_element(GTPHUB_NODE, &cfg_gtphub_bind_to_sgsns_short_cmd);
415 install_element(GTPHUB_NODE, &cfg_gtphub_bind_to_sgsns_cmd);
416 install_element(GTPHUB_NODE, &cfg_gtphub_bind_to_ggsns_short_cmd);
417 install_element(GTPHUB_NODE, &cfg_gtphub_bind_to_ggsns_cmd);
418 install_element(GTPHUB_NODE, &cfg_gtphub_ggsn_proxy_short_cmd);
419 install_element(GTPHUB_NODE, &cfg_gtphub_ggsn_proxy_cmd);
420 install_element(GTPHUB_NODE, &cfg_gtphub_sgsn_proxy_short_cmd);
421 install_element(GTPHUB_NODE, &cfg_gtphub_sgsn_proxy_cmd);
Neels Hofmeyrb6c2db52015-11-18 18:11:32 +0100422 install_element(GTPHUB_NODE, &cfg_grx_ggsn_cmd);
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200423
424 return 0;
425}
426
427int gtphub_cfg_read(struct gtphub_cfg *cfg, const char *config_file)
428{
429 int rc;
430
Neels Hofmeyrc8a614d2015-09-24 17:32:30 +0200431 rc = vty_read_config_file(config_file, NULL);
432 if (rc < 0) {
433 fprintf(stderr, "Failed to parse the config file: '%s'\n", config_file);
434 return rc;
435 }
436
437 return 0;
438}