提交 ff6a4cf2 编写于 作者: C Christoph Hellwig 提交者: David S. Miller

net/ipv6: split up ipv6_flowlabel_opt

Split ipv6_flowlabel_opt into a subfunction for each action and a small
wrapper.
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 b43c6153
...@@ -533,27 +533,16 @@ int ipv6_flowlabel_opt_get(struct sock *sk, struct in6_flowlabel_req *freq, ...@@ -533,27 +533,16 @@ int ipv6_flowlabel_opt_get(struct sock *sk, struct in6_flowlabel_req *freq,
return -ENOENT; return -ENOENT;
} }
int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) #define socklist_dereference(__sflp) \
rcu_dereference_protected(__sflp, lockdep_is_held(&ip6_sk_fl_lock))
static int ipv6_flowlabel_put(struct sock *sk, struct in6_flowlabel_req *freq)
{ {
int uninitialized_var(err);
struct net *net = sock_net(sk);
struct ipv6_pinfo *np = inet6_sk(sk); struct ipv6_pinfo *np = inet6_sk(sk);
struct in6_flowlabel_req freq;
struct ipv6_fl_socklist *sfl1 = NULL;
struct ipv6_fl_socklist *sfl;
struct ipv6_fl_socklist __rcu **sflp; struct ipv6_fl_socklist __rcu **sflp;
struct ip6_flowlabel *fl, *fl1 = NULL; struct ipv6_fl_socklist *sfl;
if (optlen < sizeof(freq))
return -EINVAL;
if (copy_from_user(&freq, optval, sizeof(freq)))
return -EFAULT;
switch (freq.flr_action) { if (freq->flr_flags & IPV6_FL_F_REFLECT) {
case IPV6_FL_A_PUT:
if (freq.flr_flags & IPV6_FL_F_REFLECT) {
if (sk->sk_protocol != IPPROTO_TCP) if (sk->sk_protocol != IPPROTO_TCP)
return -ENOPROTOOPT; return -ENOPROTOOPT;
if (!np->repflow) if (!np->repflow)
...@@ -562,49 +551,68 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -562,49 +551,68 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
np->repflow = 0; np->repflow = 0;
return 0; return 0;
} }
spin_lock_bh(&ip6_sk_fl_lock); spin_lock_bh(&ip6_sk_fl_lock);
for (sflp = &np->ipv6_fl_list; for (sflp = &np->ipv6_fl_list;
(sfl = rcu_dereference_protected(*sflp, (sfl = socklist_dereference(*sflp)) != NULL;
lockdep_is_held(&ip6_sk_fl_lock))) != NULL;
sflp = &sfl->next) { sflp = &sfl->next) {
if (sfl->fl->label == freq.flr_label) { if (sfl->fl->label == freq->flr_label)
if (freq.flr_label == (np->flow_label&IPV6_FLOWLABEL_MASK)) goto found;
}
spin_unlock_bh(&ip6_sk_fl_lock);
return -ESRCH;
found:
if (freq->flr_label == (np->flow_label & IPV6_FLOWLABEL_MASK))
np->flow_label &= ~IPV6_FLOWLABEL_MASK; np->flow_label &= ~IPV6_FLOWLABEL_MASK;
*sflp = sfl->next; *sflp = sfl->next;
spin_unlock_bh(&ip6_sk_fl_lock); spin_unlock_bh(&ip6_sk_fl_lock);
fl_release(sfl->fl); fl_release(sfl->fl);
kfree_rcu(sfl, rcu); kfree_rcu(sfl, rcu);
return 0; return 0;
} }
}
spin_unlock_bh(&ip6_sk_fl_lock); static int ipv6_flowlabel_renew(struct sock *sk, struct in6_flowlabel_req *freq)
return -ESRCH; {
struct ipv6_pinfo *np = inet6_sk(sk);
struct net *net = sock_net(sk);
struct ipv6_fl_socklist *sfl;
int err;
case IPV6_FL_A_RENEW:
rcu_read_lock_bh(); rcu_read_lock_bh();
for_each_sk_fl_rcu(np, sfl) { for_each_sk_fl_rcu(np, sfl) {
if (sfl->fl->label == freq.flr_label) { if (sfl->fl->label == freq->flr_label) {
err = fl6_renew(sfl->fl, freq.flr_linger, freq.flr_expires); err = fl6_renew(sfl->fl, freq->flr_linger,
freq->flr_expires);
rcu_read_unlock_bh(); rcu_read_unlock_bh();
return err; return err;
} }
} }
rcu_read_unlock_bh(); rcu_read_unlock_bh();
if (freq.flr_share == IPV6_FL_S_NONE && if (freq->flr_share == IPV6_FL_S_NONE &&
ns_capable(net->user_ns, CAP_NET_ADMIN)) { ns_capable(net->user_ns, CAP_NET_ADMIN)) {
fl = fl_lookup(net, freq.flr_label); struct ip6_flowlabel *fl = fl_lookup(net, freq->flr_label);
if (fl) { if (fl) {
err = fl6_renew(fl, freq.flr_linger, freq.flr_expires); err = fl6_renew(fl, freq->flr_linger,
freq->flr_expires);
fl_release(fl); fl_release(fl);
return err; return err;
} }
} }
return -ESRCH; return -ESRCH;
}
case IPV6_FL_A_GET: static int ipv6_flowlabel_get(struct sock *sk, struct in6_flowlabel_req *freq,
if (freq.flr_flags & IPV6_FL_F_REFLECT) { void __user *optval, int optlen)
{
struct ipv6_fl_socklist *sfl, *sfl1 = NULL;
struct ip6_flowlabel *fl, *fl1 = NULL;
struct ipv6_pinfo *np = inet6_sk(sk);
struct net *net = sock_net(sk); struct net *net = sock_net(sk);
int uninitialized_var(err);
if (freq->flr_flags & IPV6_FL_F_REFLECT) {
if (net->ipv6.sysctl.flowlabel_consistency) { if (net->ipv6.sysctl.flowlabel_consistency) {
net_info_ratelimited("Can not set IPV6_FL_F_REFLECT if flowlabel_consistency sysctl is enable\n"); net_info_ratelimited("Can not set IPV6_FL_F_REFLECT if flowlabel_consistency sysctl is enable\n");
return -EPERM; return -EPERM;
...@@ -612,29 +620,28 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -612,29 +620,28 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
if (sk->sk_protocol != IPPROTO_TCP) if (sk->sk_protocol != IPPROTO_TCP)
return -ENOPROTOOPT; return -ENOPROTOOPT;
np->repflow = 1; np->repflow = 1;
return 0; return 0;
} }
if (freq.flr_label & ~IPV6_FLOWLABEL_MASK) if (freq->flr_label & ~IPV6_FLOWLABEL_MASK)
return -EINVAL; return -EINVAL;
if (net->ipv6.sysctl.flowlabel_state_ranges && if (net->ipv6.sysctl.flowlabel_state_ranges &&
(freq.flr_label & IPV6_FLOWLABEL_STATELESS_FLAG)) (freq->flr_label & IPV6_FLOWLABEL_STATELESS_FLAG))
return -ERANGE; return -ERANGE;
fl = fl_create(net, sk, &freq, optval, optlen, &err); fl = fl_create(net, sk, freq, optval, optlen, &err);
if (!fl) if (!fl)
return err; return err;
sfl1 = kmalloc(sizeof(*sfl1), GFP_KERNEL); sfl1 = kmalloc(sizeof(*sfl1), GFP_KERNEL);
if (freq.flr_label) { if (freq->flr_label) {
err = -EEXIST; err = -EEXIST;
rcu_read_lock_bh(); rcu_read_lock_bh();
for_each_sk_fl_rcu(np, sfl) { for_each_sk_fl_rcu(np, sfl) {
if (sfl->fl->label == freq.flr_label) { if (sfl->fl->label == freq->flr_label) {
if (freq.flr_flags&IPV6_FL_F_EXCL) { if (freq->flr_flags & IPV6_FL_F_EXCL) {
rcu_read_unlock_bh(); rcu_read_unlock_bh();
goto done; goto done;
} }
...@@ -647,11 +654,11 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -647,11 +654,11 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
rcu_read_unlock_bh(); rcu_read_unlock_bh();
if (!fl1) if (!fl1)
fl1 = fl_lookup(net, freq.flr_label); fl1 = fl_lookup(net, freq->flr_label);
if (fl1) { if (fl1) {
recheck: recheck:
err = -EEXIST; err = -EEXIST;
if (freq.flr_flags&IPV6_FL_F_EXCL) if (freq->flr_flags&IPV6_FL_F_EXCL)
goto release; goto release;
err = -EPERM; err = -EPERM;
if (fl1->share == IPV6_FL_S_EXCL || if (fl1->share == IPV6_FL_S_EXCL ||
...@@ -679,7 +686,7 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -679,7 +686,7 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
} }
} }
err = -ENOENT; err = -ENOENT;
if (!(freq.flr_flags&IPV6_FL_F_CREATE)) if (!(freq->flr_flags & IPV6_FL_F_CREATE))
goto done; goto done;
err = -ENOMEM; err = -ENOMEM;
...@@ -690,11 +697,11 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -690,11 +697,11 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
if (err != 0) if (err != 0)
goto done; goto done;
fl1 = fl_intern(net, fl, freq.flr_label); fl1 = fl_intern(net, fl, freq->flr_label);
if (fl1) if (fl1)
goto recheck; goto recheck;
if (!freq.flr_label) { if (!freq->flr_label) {
if (copy_to_user(&((struct in6_flowlabel_req __user *) optval)->flr_label, if (copy_to_user(&((struct in6_flowlabel_req __user *) optval)->flr_label,
&fl->label, sizeof(fl->label))) { &fl->label, sizeof(fl->label))) {
/* Intentionally ignore fault. */ /* Intentionally ignore fault. */
...@@ -703,17 +710,33 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen) ...@@ -703,17 +710,33 @@ int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
fl_link(np, sfl1, fl); fl_link(np, sfl1, fl);
return 0; return 0;
default:
return -EINVAL;
}
done: done:
fl_free(fl); fl_free(fl);
kfree(sfl1); kfree(sfl1);
return err; return err;
} }
int ipv6_flowlabel_opt(struct sock *sk, char __user *optval, int optlen)
{
struct in6_flowlabel_req freq;
if (optlen < sizeof(freq))
return -EINVAL;
if (copy_from_user(&freq, optval, sizeof(freq)))
return -EFAULT;
switch (freq.flr_action) {
case IPV6_FL_A_PUT:
return ipv6_flowlabel_put(sk, &freq);
case IPV6_FL_A_RENEW:
return ipv6_flowlabel_renew(sk, &freq);
case IPV6_FL_A_GET:
return ipv6_flowlabel_get(sk, &freq, optval, optlen);
default:
return -EINVAL;
}
}
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
struct ip6fl_iter_state { struct ip6fl_iter_state {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册