blob: c7a052285f1577004f04784a6538a7f8cebb7069 [file] [log] [blame]
Pablo Neira Ayuso14506662014-02-20 18:43:15 +01001#include <stdio.h>
2#include <stdlib.h>
3#include <string.h>
4#include <unistd.h>
5#include <time.h>
6#include <arpa/inet.h>
7#include <sys/socket.h>
8#include <netinet/in.h>
9#include <inttypes.h>
10
11#include <libmnl/libmnl.h>
12#include <linux/genetlink.h>
13
Pablo Neira Ayuso18532952014-02-22 22:09:59 +010014#include <libgtpnl/gtp.h>
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010015#include <libgtpnl/gtpnl.h>
16
17#include <net/if.h>
18#include <linux/gtp_nl.h>
19
20#include "internal.h"
21
Pablo Neira Ayusob976ffa2014-03-20 13:56:55 +010022static void gtp_build_payload(struct nlmsghdr *nlh, struct gtp_tunnel *t)
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010023{
Pablo Neira Ayusob976ffa2014-03-20 13:56:55 +010024 mnl_attr_put_u32(nlh, GTPA_VERSION, t->gtp_version);
25 mnl_attr_put_u32(nlh, GTPA_LINK, t->ifidx);
26 mnl_attr_put_u32(nlh, GTPA_SGSN_ADDRESS, t->sgsn_addr.s_addr);
27 mnl_attr_put_u32(nlh, GTPA_MS_ADDRESS, t->ms_addr.s_addr);
28 mnl_attr_put_u64(nlh, GTPA_TID, t->tid);
29 mnl_attr_put_u16(nlh, GTPA_FLOWID, t->flowid);
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010030}
31
Pablo Neira Ayuso18532952014-02-22 22:09:59 +010032int gtp_add_tunnel(int genl_id, struct mnl_socket *nl, struct gtp_tunnel *t)
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010033{
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010034 struct nlmsghdr *nlh;
35 char buf[MNL_SOCKET_BUFFER_SIZE];
36 uint32_t seq = time(NULL);
37
Pablo Neira Ayuso18532952014-02-22 22:09:59 +010038 if (t->gtp_version > GTP_V1) {
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010039 fprintf(stderr, "wrong GTP version %u, use v0 or v1\n",
Pablo Neira Ayuso18532952014-02-22 22:09:59 +010040 t->gtp_version);
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010041 return -1;
42 }
43
44 nlh = genl_nlmsg_build_hdr(buf, genl_id, NLM_F_EXCL | NLM_F_ACK, ++seq,
45 GTP_CMD_TUNNEL_NEW);
Pablo Neira Ayusob976ffa2014-03-20 13:56:55 +010046 gtp_build_payload(nlh, t);
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010047
48 if (genl_socket_talk(nl, nlh, seq, NULL, NULL) < 0)
49 perror("genl_socket_talk");
50
51 return 0;
52}
53EXPORT_SYMBOL(gtp_add_tunnel);
54
Pablo Neira Ayuso18532952014-02-22 22:09:59 +010055int gtp_del_tunnel(int genl_id, struct mnl_socket *nl, struct gtp_tunnel *t)
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010056{
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010057 char buf[MNL_SOCKET_BUFFER_SIZE];
58 struct nlmsghdr *nlh;
59 uint32_t seq = time(NULL);
60
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010061 nlh = genl_nlmsg_build_hdr(buf, genl_id, NLM_F_ACK, ++seq,
62 GTP_CMD_TUNNEL_DELETE);
Pablo Neira Ayusob976ffa2014-03-20 13:56:55 +010063 gtp_build_payload(nlh, t);
Pablo Neira Ayuso14506662014-02-20 18:43:15 +010064
65 if (genl_socket_talk(nl, nlh, seq, NULL, NULL) < 0)
66 perror("genl_socket_talk");
67
68 return 0;
69}
70EXPORT_SYMBOL(gtp_del_tunnel);
71
72struct gtp_pdp {
73 uint32_t version;
74 uint64_t tid;
75 struct in_addr sgsn_addr;
76 struct in_addr ms_addr;
77};
78
79static int genl_gtp_validate_cb(const struct nlattr *attr, void *data)
80{
81 const struct nlattr **tb = data;
82 int type = mnl_attr_get_type(attr);
83
84 if (mnl_attr_type_valid(attr, CTRL_ATTR_MAX) < 0)
85 return MNL_CB_OK;
86
87 switch(type) {
88 case GTPA_TID:
89 if (mnl_attr_validate(attr, MNL_TYPE_U64) < 0) {
90 perror("mnl_attr_validate");
91 return MNL_CB_ERROR;
92 }
93 break;
94 case GTPA_SGSN_ADDRESS:
95 case GTPA_MS_ADDRESS:
96 case GTPA_VERSION:
97 if (mnl_attr_validate(attr, MNL_TYPE_U32) < 0) {
98 perror("mnl_attr_validate");
99 return MNL_CB_ERROR;
100 }
101 break;
102 default:
103 break;
104 }
105 tb[type] = attr;
106 return MNL_CB_OK;
107}
108
109static int genl_gtp_attr_cb(const struct nlmsghdr *nlh, void *data)
110{
111 struct nlattr *tb[GTPA_MAX + 1] = {};
Pablo Neira Ayuso0829e0e2014-02-22 22:19:35 +0100112 struct gtp_pdp pdp = {};
Pablo Neira Ayuso14506662014-02-20 18:43:15 +0100113 struct genlmsghdr *genl;
114
115 mnl_attr_parse(nlh, sizeof(*genl), genl_gtp_validate_cb, tb);
116 if (tb[GTPA_TID])
117 pdp.tid = mnl_attr_get_u64(tb[GTPA_TID]);
118 if (tb[GTPA_SGSN_ADDRESS]) {
119 pdp.sgsn_addr.s_addr =
120 mnl_attr_get_u32(tb[GTPA_SGSN_ADDRESS]);
121 }
122 if (tb[GTPA_MS_ADDRESS]) {
123 pdp.ms_addr.s_addr = mnl_attr_get_u32(tb[GTPA_MS_ADDRESS]);
124 }
125 if (tb[GTPA_VERSION]) {
126 pdp.version = mnl_attr_get_u32(tb[GTPA_VERSION]);
127 }
128
129 printf("version %u ", pdp.version);
130 printf("tid %"PRIu64" ms_addr %s ", pdp.tid, inet_ntoa(pdp.sgsn_addr));
131 printf("sgsn_addr %s\n", inet_ntoa(pdp.ms_addr));
132
133 return MNL_CB_OK;
134}
135
136int gtp_list_tunnel(int genl_id, struct mnl_socket *nl)
137{
138 char buf[MNL_SOCKET_BUFFER_SIZE];
139 struct nlmsghdr *nlh;
140 uint32_t seq = time(NULL);
141
142 nlh = genl_nlmsg_build_hdr(buf, genl_id, NLM_F_DUMP, 0,
143 GTP_CMD_TUNNEL_GET);
144
145 if (genl_socket_talk(nl, nlh, seq, genl_gtp_attr_cb, NULL) < 0) {
146 perror("genl_socket_talk");
147 return 0;
148 }
149
150 return 0;
151}
152EXPORT_SYMBOL(gtp_list_tunnel);