Commit 3e3eec09 authored by David S. Miller's avatar David S. Miller

Merge branch 'gtp-misc-improvements'

Andreas Schultz says:

====================
gtp: misc improvements

This is a resent of last series that missed the merge window. There
are no changes compared to v4.

v4: Compared to v3 it contains mostly smallish naming and spelling fixes.
It also drops the documentation patch, Harald did a better job with the
documentation and the some things I described do not yet match the implementation.
I'll readd the relevant parts with a follow up series.

This series lays the groundwork for removing the socket references from
the GTP netdevice by removing duplicate code and simplifying the logic on
some code paths.

It slighly changes the GTP genl API by making the socket parameters optional
(though one of them is still required).

The removal of the socket references will break the 1:1 releation between
GTP netdevice and GTP socket that prevents us to support multiple VRFs with
overlapping IP addresse spaces attached to the same GTP-U entity (needed for
multi APN support, coming a follow up series).

Pablo found a socket hold problem in v2. In order to solve that I had to
switch the socket references from the struct socket to the internal
struct sock. This should have no functionl impact, but we can now hang
on to the reference without blocking user space from closing the GTP socket.

v4->v5:
 * resent for new merge window
v3->v4:
 * drop the documentation patch
 * spelling fixes
 * pass nlattr instead of genl_info into gtp_find_dev,
   makes the code slightly more compact and readable
v2->v3:
 * add documentation to explain the goal of all these changes
 * incorporate review comments
 * switch from struct socket to struct sock
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 68e5cfaf 101cfbc1
...@@ -58,6 +58,9 @@ struct pdp_ctx { ...@@ -58,6 +58,9 @@ struct pdp_ctx {
struct in_addr ms_addr_ip4; struct in_addr ms_addr_ip4;
struct in_addr sgsn_addr_ip4; struct in_addr sgsn_addr_ip4;
struct sock *sk;
struct net_device *dev;
atomic_t tx_seq; atomic_t tx_seq;
struct rcu_head rcu_head; struct rcu_head rcu_head;
}; };
...@@ -66,8 +69,8 @@ struct pdp_ctx { ...@@ -66,8 +69,8 @@ struct pdp_ctx {
struct gtp_dev { struct gtp_dev {
struct list_head list; struct list_head list;
struct socket *sock0; struct sock *sk0;
struct socket *sock1u; struct sock *sk1u;
struct net_device *dev; struct net_device *dev;
...@@ -84,6 +87,8 @@ struct gtp_net { ...@@ -84,6 +87,8 @@ struct gtp_net {
static u32 gtp_h_initval; static u32 gtp_h_initval;
static void pdp_context_delete(struct pdp_ctx *pctx);
static inline u32 gtp0_hashfn(u64 tid) static inline u32 gtp0_hashfn(u64 tid)
{ {
u32 *tid32 = (u32 *) &tid; u32 *tid32 = (u32 *) &tid;
...@@ -175,9 +180,42 @@ static bool gtp_check_src_ms(struct sk_buff *skb, struct pdp_ctx *pctx, ...@@ -175,9 +180,42 @@ static bool gtp_check_src_ms(struct sk_buff *skb, struct pdp_ctx *pctx,
return false; return false;
} }
static int gtp_rx(struct pdp_ctx *pctx, struct sk_buff *skb, unsigned int hdrlen)
{
struct pcpu_sw_netstats *stats;
if (!gtp_check_src_ms(skb, pctx, hdrlen)) {
netdev_dbg(pctx->dev, "No PDP ctx for this MS\n");
return 1;
}
/* Get rid of the GTP + UDP headers. */
if (iptunnel_pull_header(skb, hdrlen, skb->protocol,
!net_eq(sock_net(pctx->sk), dev_net(pctx->dev))))
return -1;
netdev_dbg(pctx->dev, "forwarding packet from GGSN to uplink\n");
/* Now that the UDP and the GTP header have been removed, set up the
* new network header. This is required by the upper layer to
* calculate the transport header.
*/
skb_reset_network_header(skb);
skb->dev = pctx->dev;
stats = this_cpu_ptr(pctx->dev->tstats);
u64_stats_update_begin(&stats->syncp);
stats->rx_packets++;
stats->rx_bytes += skb->len;
u64_stats_update_end(&stats->syncp);
netif_rx(skb);
return 0;
}
/* 1 means pass up to the stack, -1 means drop and 0 means decapsulated. */ /* 1 means pass up to the stack, -1 means drop and 0 means decapsulated. */
static int gtp0_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb, static int gtp0_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb)
bool xnet)
{ {
unsigned int hdrlen = sizeof(struct udphdr) + unsigned int hdrlen = sizeof(struct udphdr) +
sizeof(struct gtp0_header); sizeof(struct gtp0_header);
...@@ -201,17 +239,10 @@ static int gtp0_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb, ...@@ -201,17 +239,10 @@ static int gtp0_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb,
return 1; return 1;
} }
if (!gtp_check_src_ms(skb, pctx, hdrlen)) { return gtp_rx(pctx, skb, hdrlen);
netdev_dbg(gtp->dev, "No PDP ctx for this MS\n");
return 1;
}
/* Get rid of the GTP + UDP headers. */
return iptunnel_pull_header(skb, hdrlen, skb->protocol, xnet);
} }
static int gtp1u_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb, static int gtp1u_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb)
bool xnet)
{ {
unsigned int hdrlen = sizeof(struct udphdr) + unsigned int hdrlen = sizeof(struct udphdr) +
sizeof(struct gtp1_header); sizeof(struct gtp1_header);
...@@ -250,37 +281,33 @@ static int gtp1u_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb, ...@@ -250,37 +281,33 @@ static int gtp1u_udp_encap_recv(struct gtp_dev *gtp, struct sk_buff *skb,
return 1; return 1;
} }
if (!gtp_check_src_ms(skb, pctx, hdrlen)) { return gtp_rx(pctx, skb, hdrlen);
netdev_dbg(gtp->dev, "No PDP ctx for this MS\n");
return 1;
}
/* Get rid of the GTP + UDP headers. */
return iptunnel_pull_header(skb, hdrlen, skb->protocol, xnet);
} }
static void gtp_encap_disable(struct gtp_dev *gtp) static void gtp_encap_destroy(struct sock *sk)
{ {
if (gtp->sock0 && gtp->sock0->sk) { struct gtp_dev *gtp;
udp_sk(gtp->sock0->sk)->encap_type = 0;
rcu_assign_sk_user_data(gtp->sock0->sk, NULL);
}
if (gtp->sock1u && gtp->sock1u->sk) {
udp_sk(gtp->sock1u->sk)->encap_type = 0;
rcu_assign_sk_user_data(gtp->sock1u->sk, NULL);
}
gtp->sock0 = NULL; gtp = rcu_dereference_sk_user_data(sk);
gtp->sock1u = NULL; if (gtp) {
udp_sk(sk)->encap_type = 0;
rcu_assign_sk_user_data(sk, NULL);
sock_put(sk);
}
} }
static void gtp_encap_destroy(struct sock *sk) static void gtp_encap_disable_sock(struct sock *sk)
{ {
struct gtp_dev *gtp; if (!sk)
return;
gtp = rcu_dereference_sk_user_data(sk); gtp_encap_destroy(sk);
if (gtp) }
gtp_encap_disable(gtp);
static void gtp_encap_disable(struct gtp_dev *gtp)
{
gtp_encap_disable_sock(gtp->sk0);
gtp_encap_disable_sock(gtp->sk1u);
} }
/* UDP encapsulation receive handler. See net/ipv4/udp.c. /* UDP encapsulation receive handler. See net/ipv4/udp.c.
...@@ -288,10 +315,8 @@ static void gtp_encap_destroy(struct sock *sk) ...@@ -288,10 +315,8 @@ static void gtp_encap_destroy(struct sock *sk)
*/ */
static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb) static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb)
{ {
struct pcpu_sw_netstats *stats;
struct gtp_dev *gtp; struct gtp_dev *gtp;
bool xnet; int ret = 0;
int ret;
gtp = rcu_dereference_sk_user_data(sk); gtp = rcu_dereference_sk_user_data(sk);
if (!gtp) if (!gtp)
...@@ -299,16 +324,14 @@ static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb) ...@@ -299,16 +324,14 @@ static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb)
netdev_dbg(gtp->dev, "encap_recv sk=%p\n", sk); netdev_dbg(gtp->dev, "encap_recv sk=%p\n", sk);
xnet = !net_eq(sock_net(sk), dev_net(gtp->dev));
switch (udp_sk(sk)->encap_type) { switch (udp_sk(sk)->encap_type) {
case UDP_ENCAP_GTP0: case UDP_ENCAP_GTP0:
netdev_dbg(gtp->dev, "received GTP0 packet\n"); netdev_dbg(gtp->dev, "received GTP0 packet\n");
ret = gtp0_udp_encap_recv(gtp, skb, xnet); ret = gtp0_udp_encap_recv(gtp, skb);
break; break;
case UDP_ENCAP_GTP1U: case UDP_ENCAP_GTP1U:
netdev_dbg(gtp->dev, "received GTP1U packet\n"); netdev_dbg(gtp->dev, "received GTP1U packet\n");
ret = gtp1u_udp_encap_recv(gtp, skb, xnet); ret = gtp1u_udp_encap_recv(gtp, skb);
break; break;
default: default:
ret = -1; /* Shouldn't happen. */ ret = -1; /* Shouldn't happen. */
...@@ -317,33 +340,17 @@ static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb) ...@@ -317,33 +340,17 @@ static int gtp_encap_recv(struct sock *sk, struct sk_buff *skb)
switch (ret) { switch (ret) {
case 1: case 1:
netdev_dbg(gtp->dev, "pass up to the process\n"); netdev_dbg(gtp->dev, "pass up to the process\n");
return 1; break;
case 0: case 0:
netdev_dbg(gtp->dev, "forwarding packet from GGSN to uplink\n");
break; break;
case -1: case -1:
netdev_dbg(gtp->dev, "GTP packet has been dropped\n"); netdev_dbg(gtp->dev, "GTP packet has been dropped\n");
kfree_skb(skb); kfree_skb(skb);
return 0; ret = 0;
break;
} }
/* Now that the UDP and the GTP header have been removed, set up the return ret;
* new network header. This is required by the upper layer to
* calculate the transport header.
*/
skb_reset_network_header(skb);
skb->dev = gtp->dev;
stats = this_cpu_ptr(gtp->dev->tstats);
u64_stats_update_begin(&stats->syncp);
stats->rx_packets++;
stats->rx_bytes += skb->len;
u64_stats_update_end(&stats->syncp);
netif_rx(skb);
return 0;
} }
static int gtp_dev_init(struct net_device *dev) static int gtp_dev_init(struct net_device *dev)
...@@ -367,8 +374,9 @@ static void gtp_dev_uninit(struct net_device *dev) ...@@ -367,8 +374,9 @@ static void gtp_dev_uninit(struct net_device *dev)
free_percpu(dev->tstats); free_percpu(dev->tstats);
} }
static struct rtable *ip4_route_output_gtp(struct net *net, struct flowi4 *fl4, static struct rtable *ip4_route_output_gtp(struct flowi4 *fl4,
const struct sock *sk, __be32 daddr) const struct sock *sk,
__be32 daddr)
{ {
memset(fl4, 0, sizeof(*fl4)); memset(fl4, 0, sizeof(*fl4));
fl4->flowi4_oif = sk->sk_bound_dev_if; fl4->flowi4_oif = sk->sk_bound_dev_if;
...@@ -377,7 +385,7 @@ static struct rtable *ip4_route_output_gtp(struct net *net, struct flowi4 *fl4, ...@@ -377,7 +385,7 @@ static struct rtable *ip4_route_output_gtp(struct net *net, struct flowi4 *fl4,
fl4->flowi4_tos = RT_CONN_FLAGS(sk); fl4->flowi4_tos = RT_CONN_FLAGS(sk);
fl4->flowi4_proto = sk->sk_protocol; fl4->flowi4_proto = sk->sk_protocol;
return ip_route_output_key(net, fl4); return ip_route_output_key(sock_net(sk), fl4);
} }
static inline void gtp0_push_header(struct sk_buff *skb, struct pdp_ctx *pctx) static inline void gtp0_push_header(struct sk_buff *skb, struct pdp_ctx *pctx)
...@@ -466,7 +474,6 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev, ...@@ -466,7 +474,6 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev,
struct rtable *rt; struct rtable *rt;
struct flowi4 fl4; struct flowi4 fl4;
struct iphdr *iph; struct iphdr *iph;
struct sock *sk;
__be16 df; __be16 df;
int mtu; int mtu;
...@@ -482,30 +489,7 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev, ...@@ -482,30 +489,7 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev,
} }
netdev_dbg(dev, "found PDP context %p\n", pctx); netdev_dbg(dev, "found PDP context %p\n", pctx);
switch (pctx->gtp_version) { rt = ip4_route_output_gtp(&fl4, pctx->sk, pctx->sgsn_addr_ip4.s_addr);
case GTP_V0:
if (gtp->sock0)
sk = gtp->sock0->sk;
else
sk = NULL;
break;
case GTP_V1:
if (gtp->sock1u)
sk = gtp->sock1u->sk;
else
sk = NULL;
break;
default:
return -ENOENT;
}
if (!sk) {
netdev_dbg(dev, "no userspace socket is available, skip\n");
return -ENOENT;
}
rt = ip4_route_output_gtp(sock_net(sk), &fl4, gtp->sock0->sk,
pctx->sgsn_addr_ip4.s_addr);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
netdev_dbg(dev, "no route to SSGN %pI4\n", netdev_dbg(dev, "no route to SSGN %pI4\n",
&pctx->sgsn_addr_ip4.s_addr); &pctx->sgsn_addr_ip4.s_addr);
...@@ -550,7 +534,7 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev, ...@@ -550,7 +534,7 @@ static int gtp_build_skb_ip4(struct sk_buff *skb, struct net_device *dev,
goto err_rt; goto err_rt;
} }
gtp_set_pktinfo_ipv4(pktinfo, sk, iph, pctx, rt, &fl4, dev); gtp_set_pktinfo_ipv4(pktinfo, pctx->sk, iph, pctx, rt, &fl4, dev);
gtp_push_header(skb, pktinfo); gtp_push_header(skb, pktinfo);
return 0; return 0;
...@@ -640,27 +624,23 @@ static void gtp_link_setup(struct net_device *dev) ...@@ -640,27 +624,23 @@ static void gtp_link_setup(struct net_device *dev)
static int gtp_hashtable_new(struct gtp_dev *gtp, int hsize); static int gtp_hashtable_new(struct gtp_dev *gtp, int hsize);
static void gtp_hashtable_free(struct gtp_dev *gtp); static void gtp_hashtable_free(struct gtp_dev *gtp);
static int gtp_encap_enable(struct net_device *dev, struct gtp_dev *gtp, static int gtp_encap_enable(struct gtp_dev *gtp, struct nlattr *data[]);
int fd_gtp0, int fd_gtp1);
static int gtp_newlink(struct net *src_net, struct net_device *dev, static int gtp_newlink(struct net *src_net, struct net_device *dev,
struct nlattr *tb[], struct nlattr *data[]) struct nlattr *tb[], struct nlattr *data[])
{ {
int hashsize, err, fd0, fd1;
struct gtp_dev *gtp; struct gtp_dev *gtp;
struct gtp_net *gn; struct gtp_net *gn;
int hashsize, err;
if (!data[IFLA_GTP_FD0] || !data[IFLA_GTP_FD1]) if (!data[IFLA_GTP_FD0] && !data[IFLA_GTP_FD1])
return -EINVAL; return -EINVAL;
gtp = netdev_priv(dev); gtp = netdev_priv(dev);
fd0 = nla_get_u32(data[IFLA_GTP_FD0]); err = gtp_encap_enable(gtp, data);
fd1 = nla_get_u32(data[IFLA_GTP_FD1]);
err = gtp_encap_enable(dev, gtp, fd0, fd1);
if (err < 0) if (err < 0)
goto out_err; return err;
if (!data[IFLA_GTP_PDP_HASHSIZE]) if (!data[IFLA_GTP_PDP_HASHSIZE])
hashsize = 1024; hashsize = 1024;
...@@ -688,7 +668,6 @@ static int gtp_newlink(struct net *src_net, struct net_device *dev, ...@@ -688,7 +668,6 @@ static int gtp_newlink(struct net *src_net, struct net_device *dev,
gtp_hashtable_free(gtp); gtp_hashtable_free(gtp);
out_encap: out_encap:
gtp_encap_disable(gtp); gtp_encap_disable(gtp);
out_err:
return err; return err;
} }
...@@ -747,21 +726,6 @@ static struct rtnl_link_ops gtp_link_ops __read_mostly = { ...@@ -747,21 +726,6 @@ static struct rtnl_link_ops gtp_link_ops __read_mostly = {
.fill_info = gtp_fill_info, .fill_info = gtp_fill_info,
}; };
static struct net *gtp_genl_get_net(struct net *src_net, struct nlattr *tb[])
{
struct net *net;
/* Examine the link attributes and figure out which network namespace
* we are talking about.
*/
if (tb[GTPA_NET_NS_FD])
net = get_net_ns_by_fd(nla_get_u32(tb[GTPA_NET_NS_FD]));
else
net = get_net(src_net);
return net;
}
static int gtp_hashtable_new(struct gtp_dev *gtp, int hsize) static int gtp_hashtable_new(struct gtp_dev *gtp, int hsize)
{ {
int i; int i;
...@@ -791,85 +755,111 @@ static void gtp_hashtable_free(struct gtp_dev *gtp) ...@@ -791,85 +755,111 @@ static void gtp_hashtable_free(struct gtp_dev *gtp)
struct pdp_ctx *pctx; struct pdp_ctx *pctx;
int i; int i;
for (i = 0; i < gtp->hash_size; i++) { for (i = 0; i < gtp->hash_size; i++)
hlist_for_each_entry_rcu(pctx, &gtp->tid_hash[i], hlist_tid) { hlist_for_each_entry_rcu(pctx, &gtp->tid_hash[i], hlist_tid)
hlist_del_rcu(&pctx->hlist_tid); pdp_context_delete(pctx);
hlist_del_rcu(&pctx->hlist_addr);
kfree_rcu(pctx, rcu_head);
}
}
synchronize_rcu(); synchronize_rcu();
kfree(gtp->addr_hash); kfree(gtp->addr_hash);
kfree(gtp->tid_hash); kfree(gtp->tid_hash);
} }
static int gtp_encap_enable(struct net_device *dev, struct gtp_dev *gtp, static struct sock *gtp_encap_enable_socket(int fd, int type,
int fd_gtp0, int fd_gtp1) struct gtp_dev *gtp)
{ {
struct udp_tunnel_sock_cfg tuncfg = {NULL}; struct udp_tunnel_sock_cfg tuncfg = {NULL};
struct socket *sock0, *sock1u; struct socket *sock;
struct sock *sk;
int err; int err;
netdev_dbg(dev, "enable gtp on %d, %d\n", fd_gtp0, fd_gtp1); pr_debug("enable gtp on %d, %d\n", fd, type);
sock0 = sockfd_lookup(fd_gtp0, &err); sock = sockfd_lookup(fd, &err);
if (sock0 == NULL) { if (!sock) {
netdev_dbg(dev, "socket fd=%d not found (gtp0)\n", fd_gtp0); pr_debug("gtp socket fd=%d not found\n", fd);
return -ENOENT; return NULL;
}
if (sock0->sk->sk_protocol != IPPROTO_UDP) {
netdev_dbg(dev, "socket fd=%d not UDP\n", fd_gtp0);
err = -EINVAL;
goto err1;
} }
sock1u = sockfd_lookup(fd_gtp1, &err); if (sock->sk->sk_protocol != IPPROTO_UDP) {
if (sock1u == NULL) { pr_debug("socket fd=%d not UDP\n", fd);
netdev_dbg(dev, "socket fd=%d not found (gtp1u)\n", fd_gtp1); sk = ERR_PTR(-EINVAL);
err = -ENOENT; goto out_sock;
goto err1;
} }
if (sock1u->sk->sk_protocol != IPPROTO_UDP) { if (rcu_dereference_sk_user_data(sock->sk)) {
netdev_dbg(dev, "socket fd=%d not UDP\n", fd_gtp1); sk = ERR_PTR(-EBUSY);
err = -EINVAL; goto out_sock;
goto err2;
} }
netdev_dbg(dev, "enable gtp on %p, %p\n", sock0, sock1u); sk = sock->sk;
sock_hold(sk);
gtp->sock0 = sock0;
gtp->sock1u = sock1u;
tuncfg.sk_user_data = gtp; tuncfg.sk_user_data = gtp;
tuncfg.encap_type = type;
tuncfg.encap_rcv = gtp_encap_recv; tuncfg.encap_rcv = gtp_encap_recv;
tuncfg.encap_destroy = gtp_encap_destroy; tuncfg.encap_destroy = gtp_encap_destroy;
tuncfg.encap_type = UDP_ENCAP_GTP0; setup_udp_tunnel_sock(sock_net(sock->sk), sock, &tuncfg);
setup_udp_tunnel_sock(sock_net(gtp->sock0->sk), gtp->sock0, &tuncfg);
tuncfg.encap_type = UDP_ENCAP_GTP1U;
setup_udp_tunnel_sock(sock_net(gtp->sock1u->sk), gtp->sock1u, &tuncfg);
err = 0; out_sock:
err2: sockfd_put(sock);
sockfd_put(sock1u); return sk;
err1:
sockfd_put(sock0);
return err;
} }
static struct net_device *gtp_find_dev(struct net *net, int ifindex) static int gtp_encap_enable(struct gtp_dev *gtp, struct nlattr *data[])
{ {
struct gtp_net *gn = net_generic(net, gtp_net_id); struct sock *sk1u = NULL;
struct gtp_dev *gtp; struct sock *sk0 = NULL;
list_for_each_entry_rcu(gtp, &gn->gtp_dev_list, list) { if (data[IFLA_GTP_FD0]) {
if (ifindex == gtp->dev->ifindex) u32 fd0 = nla_get_u32(data[IFLA_GTP_FD0]);
return gtp->dev;
sk0 = gtp_encap_enable_socket(fd0, UDP_ENCAP_GTP0, gtp);
if (IS_ERR(sk0))
return PTR_ERR(sk0);
}
if (data[IFLA_GTP_FD1]) {
u32 fd1 = nla_get_u32(data[IFLA_GTP_FD1]);
sk1u = gtp_encap_enable_socket(fd1, UDP_ENCAP_GTP1U, gtp);
if (IS_ERR(sk1u)) {
if (sk0)
gtp_encap_disable_sock(sk0);
return PTR_ERR(sk1u);
} }
}
gtp->sk0 = sk0;
gtp->sk1u = sk1u;
return 0;
}
static struct gtp_dev *gtp_find_dev(struct net *src_net, struct nlattr *nla[])
{
struct gtp_dev *gtp = NULL;
struct net_device *dev;
struct net *net;
/* Examine the link attributes and figure out which network namespace
* we are talking about.
*/
if (nla[GTPA_NET_NS_FD])
net = get_net_ns_by_fd(nla_get_u32(nla[GTPA_NET_NS_FD]));
else
net = get_net(src_net);
if (IS_ERR(net))
return NULL; return NULL;
/* Check if there's an existing gtpX device to configure */
dev = dev_get_by_index_rcu(net, nla_get_u32(nla[GTPA_LINK]));
if (dev->netdev_ops == &gtp_netdev_ops)
gtp = netdev_priv(dev);
put_net(net);
return gtp;
} }
static void ipv4_pdp_fill(struct pdp_ctx *pctx, struct genl_info *info) static void ipv4_pdp_fill(struct pdp_ctx *pctx, struct genl_info *info)
...@@ -899,9 +889,10 @@ static void ipv4_pdp_fill(struct pdp_ctx *pctx, struct genl_info *info) ...@@ -899,9 +889,10 @@ static void ipv4_pdp_fill(struct pdp_ctx *pctx, struct genl_info *info)
} }
} }
static int ipv4_pdp_add(struct net_device *dev, struct genl_info *info) static int ipv4_pdp_add(struct gtp_dev *gtp, struct sock *sk,
struct genl_info *info)
{ {
struct gtp_dev *gtp = netdev_priv(dev); struct net_device *dev = gtp->dev;
u32 hash_ms, hash_tid = 0; u32 hash_ms, hash_tid = 0;
struct pdp_ctx *pctx; struct pdp_ctx *pctx;
bool found = false; bool found = false;
...@@ -940,6 +931,9 @@ static int ipv4_pdp_add(struct net_device *dev, struct genl_info *info) ...@@ -940,6 +931,9 @@ static int ipv4_pdp_add(struct net_device *dev, struct genl_info *info)
if (pctx == NULL) if (pctx == NULL)
return -ENOMEM; return -ENOMEM;
sock_hold(sk);
pctx->sk = sk;
pctx->dev = gtp->dev;
ipv4_pdp_fill(pctx, info); ipv4_pdp_fill(pctx, info);
atomic_set(&pctx->tx_seq, 0); atomic_set(&pctx->tx_seq, 0);
...@@ -976,10 +970,27 @@ static int ipv4_pdp_add(struct net_device *dev, struct genl_info *info) ...@@ -976,10 +970,27 @@ static int ipv4_pdp_add(struct net_device *dev, struct genl_info *info)
return 0; return 0;
} }
static void pdp_context_free(struct rcu_head *head)
{
struct pdp_ctx *pctx = container_of(head, struct pdp_ctx, rcu_head);
sock_put(pctx->sk);
kfree(pctx);
}
static void pdp_context_delete(struct pdp_ctx *pctx)
{
hlist_del_rcu(&pctx->hlist_tid);
hlist_del_rcu(&pctx->hlist_addr);
call_rcu(&pctx->rcu_head, pdp_context_free);
}
static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info) static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info)
{ {
struct net_device *dev; unsigned int version;
struct net *net; struct gtp_dev *gtp;
struct sock *sk;
int err;
if (!info->attrs[GTPA_VERSION] || if (!info->attrs[GTPA_VERSION] ||
!info->attrs[GTPA_LINK] || !info->attrs[GTPA_LINK] ||
...@@ -987,7 +998,9 @@ static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info) ...@@ -987,7 +998,9 @@ static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info)
!info->attrs[GTPA_MS_ADDRESS]) !info->attrs[GTPA_MS_ADDRESS])
return -EINVAL; return -EINVAL;
switch (nla_get_u32(info->attrs[GTPA_VERSION])) { version = nla_get_u32(info->attrs[GTPA_VERSION]);
switch (version) {
case GTP_V0: case GTP_V0:
if (!info->attrs[GTPA_TID] || if (!info->attrs[GTPA_TID] ||
!info->attrs[GTPA_FLOW]) !info->attrs[GTPA_FLOW])
...@@ -1003,77 +1016,101 @@ static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info) ...@@ -1003,77 +1016,101 @@ static int gtp_genl_new_pdp(struct sk_buff *skb, struct genl_info *info)
return -EINVAL; return -EINVAL;
} }
net = gtp_genl_get_net(sock_net(skb->sk), info->attrs); rcu_read_lock();
if (IS_ERR(net))
return PTR_ERR(net);
/* Check if there's an existing gtpX device to configure */ gtp = gtp_find_dev(sock_net(skb->sk), info->attrs);
dev = gtp_find_dev(net, nla_get_u32(info->attrs[GTPA_LINK])); if (!gtp) {
if (dev == NULL) { err = -ENODEV;
put_net(net); goto out_unlock;
return -ENODEV;
} }
put_net(net);
return ipv4_pdp_add(dev, info); if (version == GTP_V0)
sk = gtp->sk0;
else if (version == GTP_V1)
sk = gtp->sk1u;
else
sk = NULL;
if (!sk) {
err = -ENODEV;
goto out_unlock;
}
err = ipv4_pdp_add(gtp, sk, info);
out_unlock:
rcu_read_unlock();
return err;
} }
static int gtp_genl_del_pdp(struct sk_buff *skb, struct genl_info *info) static struct pdp_ctx *gtp_find_pdp_by_link(struct net *net,
struct nlattr *nla[])
{ {
struct net_device *dev;
struct pdp_ctx *pctx;
struct gtp_dev *gtp; struct gtp_dev *gtp;
struct net *net;
if (!info->attrs[GTPA_VERSION] || gtp = gtp_find_dev(net, nla);
!info->attrs[GTPA_LINK]) if (!gtp)
return -EINVAL; return ERR_PTR(-ENODEV);
net = gtp_genl_get_net(sock_net(skb->sk), info->attrs); if (nla[GTPA_MS_ADDRESS]) {
if (IS_ERR(net)) __be32 ip = nla_get_be32(nla[GTPA_MS_ADDRESS]);
return PTR_ERR(net);
/* Check if there's an existing gtpX device to configure */ return ipv4_pdp_find(gtp, ip);
dev = gtp_find_dev(net, nla_get_u32(info->attrs[GTPA_LINK])); } else if (nla[GTPA_VERSION]) {
if (dev == NULL) { u32 gtp_version = nla_get_u32(nla[GTPA_VERSION]);
put_net(net);
return -ENODEV; if (gtp_version == GTP_V0 && nla[GTPA_TID])
return gtp0_pdp_find(gtp, nla_get_u64(nla[GTPA_TID]));
else if (gtp_version == GTP_V1 && nla[GTPA_I_TEI])
return gtp1_pdp_find(gtp, nla_get_u32(nla[GTPA_I_TEI]));
} }
put_net(net);
gtp = netdev_priv(dev); return ERR_PTR(-EINVAL);
}
switch (nla_get_u32(info->attrs[GTPA_VERSION])) { static struct pdp_ctx *gtp_find_pdp(struct net *net, struct nlattr *nla[])
case GTP_V0: {
if (!info->attrs[GTPA_TID]) struct pdp_ctx *pctx;
return -EINVAL;
pctx = gtp0_pdp_find(gtp, nla_get_u64(info->attrs[GTPA_TID]));
break;
case GTP_V1:
if (!info->attrs[GTPA_I_TEI])
return -EINVAL;
pctx = gtp1_pdp_find(gtp, nla_get_u64(info->attrs[GTPA_I_TEI]));
break;
default: if (nla[GTPA_LINK])
pctx = gtp_find_pdp_by_link(net, nla);
else
pctx = ERR_PTR(-EINVAL);
if (!pctx)
pctx = ERR_PTR(-ENOENT);
return pctx;
}
static int gtp_genl_del_pdp(struct sk_buff *skb, struct genl_info *info)
{
struct pdp_ctx *pctx;
int err = 0;
if (!info->attrs[GTPA_VERSION])
return -EINVAL; return -EINVAL;
}
if (pctx == NULL) rcu_read_lock();
return -ENOENT;
pctx = gtp_find_pdp(sock_net(skb->sk), info->attrs);
if (IS_ERR(pctx)) {
err = PTR_ERR(pctx);
goto out_unlock;
}
if (pctx->gtp_version == GTP_V0) if (pctx->gtp_version == GTP_V0)
netdev_dbg(dev, "GTPv0-U: deleting tunnel id = %llx (pdp %p)\n", netdev_dbg(pctx->dev, "GTPv0-U: deleting tunnel id = %llx (pdp %p)\n",
pctx->u.v0.tid, pctx); pctx->u.v0.tid, pctx);
else if (pctx->gtp_version == GTP_V1) else if (pctx->gtp_version == GTP_V1)
netdev_dbg(dev, "GTPv1-U: deleting tunnel id = %x/%x (pdp %p)\n", netdev_dbg(pctx->dev, "GTPv1-U: deleting tunnel id = %x/%x (pdp %p)\n",
pctx->u.v1.i_tei, pctx->u.v1.o_tei, pctx); pctx->u.v1.i_tei, pctx->u.v1.o_tei, pctx);
hlist_del_rcu(&pctx->hlist_tid); pdp_context_delete(pctx);
hlist_del_rcu(&pctx->hlist_addr);
kfree_rcu(pctx, rcu_head);
return 0; out_unlock:
rcu_read_unlock();
return err;
} }
static struct genl_family gtp_genl_family; static struct genl_family gtp_genl_family;
...@@ -1117,59 +1154,17 @@ static int gtp_genl_fill_info(struct sk_buff *skb, u32 snd_portid, u32 snd_seq, ...@@ -1117,59 +1154,17 @@ static int gtp_genl_fill_info(struct sk_buff *skb, u32 snd_portid, u32 snd_seq,
static int gtp_genl_get_pdp(struct sk_buff *skb, struct genl_info *info) static int gtp_genl_get_pdp(struct sk_buff *skb, struct genl_info *info)
{ {
struct pdp_ctx *pctx = NULL; struct pdp_ctx *pctx = NULL;
struct net_device *dev;
struct sk_buff *skb2; struct sk_buff *skb2;
struct gtp_dev *gtp;
u32 gtp_version;
struct net *net;
int err; int err;
if (!info->attrs[GTPA_VERSION] || if (!info->attrs[GTPA_VERSION])
!info->attrs[GTPA_LINK])
return -EINVAL; return -EINVAL;
gtp_version = nla_get_u32(info->attrs[GTPA_VERSION]);
switch (gtp_version) {
case GTP_V0:
case GTP_V1:
break;
default:
return -EINVAL;
}
net = gtp_genl_get_net(sock_net(skb->sk), info->attrs);
if (IS_ERR(net))
return PTR_ERR(net);
/* Check if there's an existing gtpX device to configure */
dev = gtp_find_dev(net, nla_get_u32(info->attrs[GTPA_LINK]));
if (dev == NULL) {
put_net(net);
return -ENODEV;
}
put_net(net);
gtp = netdev_priv(dev);
rcu_read_lock(); rcu_read_lock();
if (gtp_version == GTP_V0 &&
info->attrs[GTPA_TID]) {
u64 tid = nla_get_u64(info->attrs[GTPA_TID]);
pctx = gtp0_pdp_find(gtp, tid);
} else if (gtp_version == GTP_V1 &&
info->attrs[GTPA_I_TEI]) {
u32 tid = nla_get_u32(info->attrs[GTPA_I_TEI]);
pctx = gtp1_pdp_find(gtp, tid);
} else if (info->attrs[GTPA_MS_ADDRESS]) {
__be32 ip = nla_get_be32(info->attrs[GTPA_MS_ADDRESS]);
pctx = ipv4_pdp_find(gtp, ip);
}
if (pctx == NULL) { pctx = gtp_find_pdp(sock_net(skb->sk), info->attrs);
err = -ENOENT; if (IS_ERR(pctx)) {
err = PTR_ERR(pctx);
goto err_unlock; goto err_unlock;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment