blob: 0881f150f8c53e43c45c2cc1c60a6f607cbf2152 [file] [log] [blame]
Harald Welte51b00a62014-04-03 09:37:38 -04001/* Command line utility to create GTP link */
2
3/* (C) 2014 by sysmocom - s.f.m.c. GmbH
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +02004 * (C) 2016 by Pablo Neira Ayuso <pablo@netfilter.org>
5 *
Harald Welte51b00a62014-04-03 09:37:38 -04006 * Author: Pablo Neira Ayuso <pablo@gnumonks.org>
7 *
8 * All Rights Reserved
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU Affero General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 *
23 */
24
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010025#include <stdio.h>
26#include <stdlib.h>
27#include <unistd.h>
28#include <string.h>
29#include <time.h>
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +020030#include <netinet/in.h>
31#include <arpa/inet.h>
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010032
33#include <libmnl/libmnl.h>
34#include <linux/if.h>
35#include <linux/if_link.h>
36#include <linux/rtnetlink.h>
37
Pablo Neira Ayusob9f6ffe2016-05-08 18:27:55 +020038#include <linux/gtp.h>
39#include <linux/if_link.h>
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010040
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +020041#include <libgtpnl/gtpnl.h>
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010042#include <errno.h>
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +020043
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010044struct gtp_server_sock {
45 int family;
46 int fd1;
47 int fd2;
48 socklen_t len;
49 struct {
50 union {
51 struct sockaddr_in in;
52 struct sockaddr_in6 in6;
53 } fd1;
54 union {
55 struct sockaddr_in in;
56 struct sockaddr_in6 in6;
57 } fd2;
58 } sockaddr;
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010059 union {
60 struct in_addr v4;
61 struct in6_addr v6;
62 } addr;
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010063};
64
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010065static void setup_sockaddr_in(struct sockaddr_in *sockaddr, struct in_addr *in,
66 uint16_t port)
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010067{
68 sockaddr->sin_family = AF_INET;
69 sockaddr->sin_port = htons(port);
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010070 sockaddr->sin_addr = *in;
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010071}
72
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010073static void setup_sockaddr_in6(struct sockaddr_in6 *sockaddr, struct in6_addr *in6,
74 uint16_t port)
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010075{
76 sockaddr->sin6_family = AF_INET6;
77 sockaddr->sin6_port = htons(port);
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010078 sockaddr->sin6_addr = *in6;
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010079}
80
81static int setup_socket(struct gtp_server_sock *gtp_sock, int family)
82{
83 int fd1 = socket(family, SOCK_DGRAM, 0);
84 int fd2 = socket(family, SOCK_DGRAM, 0);
Pablo Neira Ayuso9911f6b2024-01-31 18:17:51 +010085 int one = 1;
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010086
87 if (fd1 < 0 || fd2 < 0)
88 return -1;
89
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +010090 gtp_sock->family = family;
91 gtp_sock->fd1 = fd1;
92 gtp_sock->fd2 = fd2;
93
94 switch (family) {
95 case AF_INET:
96 gtp_sock->len = sizeof(struct sockaddr_in);
Pablo Neira Ayusod4894882024-02-15 21:08:26 +010097 setup_sockaddr_in(&gtp_sock->sockaddr.fd1.in,
98 &gtp_sock->addr.v4, 3386);
99 setup_sockaddr_in(&gtp_sock->sockaddr.fd2.in,
100 &gtp_sock->addr.v4, 2152);
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100101 break;
102 case AF_INET6:
103 gtp_sock->len = sizeof(struct sockaddr_in6);
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100104 setup_sockaddr_in6(&gtp_sock->sockaddr.fd1.in6,
105 &gtp_sock->addr.v6, 3386);
106 setup_sockaddr_in6(&gtp_sock->sockaddr.fd2.in6,
107 &gtp_sock->addr.v6, 2152);
Pablo Neira Ayuso9911f6b2024-01-31 18:17:51 +0100108 if (setsockopt(fd1, IPPROTO_IPV6, IPV6_V6ONLY, &one, sizeof(one)) < 0)
109 perror("setsockopt IPV6_V6ONLY: ");
110 if (setsockopt(fd2, IPPROTO_IPV6, IPV6_V6ONLY, &one, sizeof(one)) < 0)
111 perror("setsockopt IPV6_V6ONLY: ");
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100112 break;
113 }
114
115 return 0;
116}
117
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100118int main(int argc, char *argv[])
119{
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100120 char buf[MNL_SOCKET_BUFFER_SIZE];
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100121 struct gtp_server_sock gtp_sock;
122 int ret, sgsn_mode = 0, family;
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100123 const char *addr = NULL;
124
125 memset(&gtp_sock, 0, sizeof(gtp_sock));
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100126
Harald Welte3bf55c32017-03-15 18:03:42 +0100127 if (argc < 3) {
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100128 printf("Usage: %s add <device> <family> [--sgsn] [<address>]\n", argv[0]);
Oliver Smith6954da72022-09-07 15:48:16 +0200129 printf(" %s del <device>\n", argv[0]);
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100130 exit(EXIT_FAILURE);
131 }
132
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +0200133 if (!strcmp(argv[1], "del")) {
134 printf("destroying gtp interface...\n");
135 if (gtp_dev_destroy(argv[2]) < 0)
136 perror("gtp_dev_destroy");
137
138 return 0;
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100139 } else if (!strcmp(argv[1], "add")) {
140 if (argc < 4) {
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100141 printf("Usage: %s add <device> <family> [--sgsn] [<address>]\n", argv[0]);
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100142 exit(EXIT_FAILURE);
143 }
144
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100145 if (argc == 5) {
146 if (!strcmp(argv[4], "--sgsn"))
147 sgsn_mode = 1;
148 else
149 addr = argv[4];
150 }
151
152 if (argc == 6) {
153 if (!strcmp(argv[4], "--sgsn"))
154 sgsn_mode = 1;
155
156 addr = argv[5];
157 }
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +0200158 }
159
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100160 if (!strcmp(argv[3], "ip"))
161 family = AF_INET;
162 else if (!strcmp(argv[3], "ip6"))
163 family = AF_INET6;
164 else {
165 fprintf(stderr, "unsupport family `%s', expecting `ip' or `ip6'\n",
166 argv[3]);
167 exit(EXIT_FAILURE);
168 }
Harald Welte3bf55c32017-03-15 18:03:42 +0100169
Pablo Neira Ayusod4894882024-02-15 21:08:26 +0100170 if (addr) {
171 if (!inet_pton(family, addr, &gtp_sock.addr)) {
172 fprintf(stderr, "invalid listener address: %s\n",
173 strerror(errno));
174 exit(EXIT_FAILURE);
175 }
176 } else {
177 switch (family) {
178 case AF_INET:
179 gtp_sock.addr.v4.s_addr = INADDR_ANY;
180 break;
181 case AF_INET6:
182 gtp_sock.addr.v6 = in6addr_any;
183 break;
184 }
185 }
186
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100187 if (setup_socket(&gtp_sock, family) < 0) {
188 perror("socket");
189 exit(EXIT_FAILURE);
190 }
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200191
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100192 if (bind(gtp_sock.fd1, (struct sockaddr *) &gtp_sock.sockaddr.fd1, gtp_sock.len) < 0) {
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200193 perror("bind");
194 exit(EXIT_FAILURE);
195 }
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100196 if (bind(gtp_sock.fd2, (struct sockaddr *) &gtp_sock.sockaddr.fd2, gtp_sock.len) < 0) {
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200197 perror("bind");
198 exit(EXIT_FAILURE);
199 }
Pablo Neira Ayuso7aa20872014-02-24 11:38:52 +0100200
Harald Welte3bf55c32017-03-15 18:03:42 +0100201 if (sgsn_mode)
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100202 ret = gtp_dev_create_sgsn(-1, argv[2], gtp_sock.fd1, gtp_sock.fd2);
Harald Welte3bf55c32017-03-15 18:03:42 +0100203 else
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100204 ret = gtp_dev_create(-1, argv[2], gtp_sock.fd1, gtp_sock.fd2);
Harald Weltea7a4df32017-03-15 20:53:08 +0100205 if (ret < 0) {
206 perror("cannot create GTP device\n");
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100207 exit(EXIT_FAILURE);
208 }
209
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +0200210 fprintf(stderr, "WARNING: attaching dummy socket descriptors. Keep "
211 "this process running for testing purposes.\n");
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200212
213 while (1) {
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100214 union {
215 struct sockaddr_in addr;
216 struct sockaddr_in6 addr6;
217 } sock;
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200218
Pablo Neira Ayuso38b5e232023-11-03 17:58:43 +0100219 ret = recvfrom(gtp_sock.fd1, buf, sizeof(buf), 0,
220 (struct sockaddr *)&sock, &gtp_sock.len);
Pablo Neira Ayuso50826a52016-05-10 18:00:51 +0200221 printf("received %d bytes via UDP socket\n", ret);
222 }
Pablo Neira Ayuso448bce42016-05-08 21:43:40 +0200223
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100224 return 0;
225}