Commit 366e41d9 authored by Vlad Yasevich's avatar Vlad Yasevich Committed by David S. Miller

ipv6: pull cork initialization into its own function.

Pull IPv6 cork initialization into its own function that
can be re-used.  IPv6 specific cork data did not have an
explicit data structure.  This patch creats eone so that
just ipv6 cork data can be as arguemts.  Also, since
IPv6 tries to save the flow label into inet_cork_full
tructure, pass the full cork.

Adjust ip6_cork_release() to take cork data structures.
Signed-off-by: default avatarVladislav Yasevich <vyasevic@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ba0c39cb
...@@ -125,6 +125,12 @@ struct ipv6_mc_socklist; ...@@ -125,6 +125,12 @@ struct ipv6_mc_socklist;
struct ipv6_ac_socklist; struct ipv6_ac_socklist;
struct ipv6_fl_socklist; struct ipv6_fl_socklist;
struct inet6_cork {
struct ipv6_txoptions *opt;
u8 hop_limit;
u8 tclass;
};
/** /**
* struct ipv6_pinfo - ipv6 private area * struct ipv6_pinfo - ipv6 private area
* *
...@@ -217,11 +223,7 @@ struct ipv6_pinfo { ...@@ -217,11 +223,7 @@ struct ipv6_pinfo {
struct ipv6_txoptions *opt; struct ipv6_txoptions *opt;
struct sk_buff *pktoptions; struct sk_buff *pktoptions;
struct sk_buff *rxpmtu; struct sk_buff *rxpmtu;
struct { struct inet6_cork cork;
struct ipv6_txoptions *opt;
u8 hop_limit;
u8 tclass;
} cork;
}; };
/* WARNING: don't change the layout of the members in {raw,udp,tcp}6_sock! */ /* WARNING: don't change the layout of the members in {raw,udp,tcp}6_sock! */
......
...@@ -1135,72 +1135,56 @@ static void ip6_append_data_mtu(unsigned int *mtu, ...@@ -1135,72 +1135,56 @@ static void ip6_append_data_mtu(unsigned int *mtu,
} }
} }
int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, static int ip6_setup_cork(struct sock *sk, struct inet_cork_full *cork,
int offset, int len, int odd, struct sk_buff *skb), struct inet6_cork *v6_cork,
void *from, int length, int transhdrlen, int hlimit, int tclass, struct ipv6_txoptions *opt,
int hlimit, int tclass, struct ipv6_txoptions *opt, struct flowi6 *fl6, struct rt6_info *rt, struct flowi6 *fl6)
struct rt6_info *rt, unsigned int flags, int dontfrag)
{ {
struct inet_sock *inet = inet_sk(sk);
struct ipv6_pinfo *np = inet6_sk(sk); struct ipv6_pinfo *np = inet6_sk(sk);
struct inet_cork *cork; unsigned int mtu;
struct sk_buff *skb, *skb_prev = NULL;
unsigned int maxfraglen, fragheaderlen, mtu, orig_mtu;
int exthdrlen;
int dst_exthdrlen;
int hh_len;
int copy;
int err;
int offset = 0;
__u8 tx_flags = 0;
u32 tskey = 0;
if (flags&MSG_PROBE)
return 0;
cork = &inet->cork.base;
if (skb_queue_empty(&sk->sk_write_queue)) {
/* /*
* setup for corking * setup for corking
*/ */
if (opt) { if (opt) {
if (WARN_ON(np->cork.opt)) if (WARN_ON(v6_cork->opt))
return -EINVAL; return -EINVAL;
np->cork.opt = kzalloc(opt->tot_len, sk->sk_allocation); v6_cork->opt = kzalloc(opt->tot_len, sk->sk_allocation);
if (unlikely(np->cork.opt == NULL)) if (unlikely(v6_cork->opt == NULL))
return -ENOBUFS; return -ENOBUFS;
np->cork.opt->tot_len = opt->tot_len; v6_cork->opt->tot_len = opt->tot_len;
np->cork.opt->opt_flen = opt->opt_flen; v6_cork->opt->opt_flen = opt->opt_flen;
np->cork.opt->opt_nflen = opt->opt_nflen; v6_cork->opt->opt_nflen = opt->opt_nflen;
np->cork.opt->dst0opt = ip6_opt_dup(opt->dst0opt, v6_cork->opt->dst0opt = ip6_opt_dup(opt->dst0opt,
sk->sk_allocation); sk->sk_allocation);
if (opt->dst0opt && !np->cork.opt->dst0opt) if (opt->dst0opt && !v6_cork->opt->dst0opt)
return -ENOBUFS; return -ENOBUFS;
np->cork.opt->dst1opt = ip6_opt_dup(opt->dst1opt, v6_cork->opt->dst1opt = ip6_opt_dup(opt->dst1opt,
sk->sk_allocation); sk->sk_allocation);
if (opt->dst1opt && !np->cork.opt->dst1opt) if (opt->dst1opt && !v6_cork->opt->dst1opt)
return -ENOBUFS; return -ENOBUFS;
np->cork.opt->hopopt = ip6_opt_dup(opt->hopopt, v6_cork->opt->hopopt = ip6_opt_dup(opt->hopopt,
sk->sk_allocation); sk->sk_allocation);
if (opt->hopopt && !np->cork.opt->hopopt) if (opt->hopopt && !v6_cork->opt->hopopt)
return -ENOBUFS; return -ENOBUFS;
np->cork.opt->srcrt = ip6_rthdr_dup(opt->srcrt, v6_cork->opt->srcrt = ip6_rthdr_dup(opt->srcrt,
sk->sk_allocation); sk->sk_allocation);
if (opt->srcrt && !np->cork.opt->srcrt) if (opt->srcrt && !v6_cork->opt->srcrt)
return -ENOBUFS; return -ENOBUFS;
/* need source address above miyazawa*/ /* need source address above miyazawa*/
} }
dst_hold(&rt->dst); dst_hold(&rt->dst);
cork->dst = &rt->dst; cork->base.dst = &rt->dst;
inet->cork.fl.u.ip6 = *fl6; cork->fl.u.ip6 = *fl6;
np->cork.hop_limit = hlimit; v6_cork->hop_limit = hlimit;
np->cork.tclass = tclass; v6_cork->tclass = tclass;
if (rt->dst.flags & DST_XFRM_TUNNEL) if (rt->dst.flags & DST_XFRM_TUNNEL)
mtu = np->pmtudisc >= IPV6_PMTUDISC_PROBE ? mtu = np->pmtudisc >= IPV6_PMTUDISC_PROBE ?
rt->dst.dev->mtu : dst_mtu(&rt->dst); rt->dst.dev->mtu : dst_mtu(&rt->dst);
...@@ -1211,10 +1195,45 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, ...@@ -1211,10 +1195,45 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to,
if (np->frag_size) if (np->frag_size)
mtu = np->frag_size; mtu = np->frag_size;
} }
cork->fragsize = mtu; cork->base.fragsize = mtu;
if (dst_allfrag(rt->dst.path)) if (dst_allfrag(rt->dst.path))
cork->flags |= IPCORK_ALLFRAG; cork->base.flags |= IPCORK_ALLFRAG;
cork->length = 0; cork->base.length = 0;
return 0;
}
int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to,
int offset, int len, int odd, struct sk_buff *skb),
void *from, int length, int transhdrlen,
int hlimit, int tclass, struct ipv6_txoptions *opt, struct flowi6 *fl6,
struct rt6_info *rt, unsigned int flags, int dontfrag)
{
struct inet_sock *inet = inet_sk(sk);
struct ipv6_pinfo *np = inet6_sk(sk);
struct inet_cork *cork;
struct sk_buff *skb, *skb_prev = NULL;
unsigned int maxfraglen, fragheaderlen, mtu, orig_mtu;
int exthdrlen;
int dst_exthdrlen;
int hh_len;
int copy;
int err;
int offset = 0;
__u8 tx_flags = 0;
u32 tskey = 0;
if (flags&MSG_PROBE)
return 0;
cork = &inet->cork.base;
if (skb_queue_empty(&sk->sk_write_queue)) {
/*
* setup for corking
*/
err = ip6_setup_cork(sk, &inet->cork, &np->cork, hlimit,
tclass, opt, rt, fl6);
if (err)
return err;
exthdrlen = (opt ? opt->opt_flen : 0); exthdrlen = (opt ? opt->opt_flen : 0);
length += exthdrlen; length += exthdrlen;
transhdrlen += exthdrlen; transhdrlen += exthdrlen;
...@@ -1226,8 +1245,8 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, ...@@ -1226,8 +1245,8 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to,
transhdrlen = 0; transhdrlen = 0;
exthdrlen = 0; exthdrlen = 0;
dst_exthdrlen = 0; dst_exthdrlen = 0;
mtu = cork->fragsize;
} }
mtu = cork->fragsize;
orig_mtu = mtu; orig_mtu = mtu;
hh_len = LL_RESERVED_SPACE(rt->dst.dev); hh_len = LL_RESERVED_SPACE(rt->dst.dev);
...@@ -1503,23 +1522,24 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to, ...@@ -1503,23 +1522,24 @@ int ip6_append_data(struct sock *sk, int getfrag(void *from, char *to,
} }
EXPORT_SYMBOL_GPL(ip6_append_data); EXPORT_SYMBOL_GPL(ip6_append_data);
static void ip6_cork_release(struct inet_sock *inet, struct ipv6_pinfo *np) static void ip6_cork_release(struct inet_cork_full *cork,
struct inet6_cork *v6_cork)
{ {
if (np->cork.opt) { if (v6_cork->opt) {
kfree(np->cork.opt->dst0opt); kfree(v6_cork->opt->dst0opt);
kfree(np->cork.opt->dst1opt); kfree(v6_cork->opt->dst1opt);
kfree(np->cork.opt->hopopt); kfree(v6_cork->opt->hopopt);
kfree(np->cork.opt->srcrt); kfree(v6_cork->opt->srcrt);
kfree(np->cork.opt); kfree(v6_cork->opt);
np->cork.opt = NULL; v6_cork->opt = NULL;
} }
if (inet->cork.base.dst) { if (cork->base.dst) {
dst_release(inet->cork.base.dst); dst_release(cork->base.dst);
inet->cork.base.dst = NULL; cork->base.dst = NULL;
inet->cork.base.flags &= ~IPCORK_ALLFRAG; cork->base.flags &= ~IPCORK_ALLFRAG;
} }
memset(&inet->cork.fl, 0, sizeof(inet->cork.fl)); memset(&cork->fl, 0, sizeof(cork->fl));
} }
int ip6_push_pending_frames(struct sock *sk) int ip6_push_pending_frames(struct sock *sk)
...@@ -1599,7 +1619,7 @@ int ip6_push_pending_frames(struct sock *sk) ...@@ -1599,7 +1619,7 @@ int ip6_push_pending_frames(struct sock *sk)
} }
out: out:
ip6_cork_release(inet, np); ip6_cork_release(&inet->cork, &np->cork);
return err; return err;
error: error:
IP6_INC_STATS(net, rt->rt6i_idev, IPSTATS_MIB_OUTDISCARDS); IP6_INC_STATS(net, rt->rt6i_idev, IPSTATS_MIB_OUTDISCARDS);
...@@ -1618,6 +1638,6 @@ void ip6_flush_pending_frames(struct sock *sk) ...@@ -1618,6 +1638,6 @@ void ip6_flush_pending_frames(struct sock *sk)
kfree_skb(skb); kfree_skb(skb);
} }
ip6_cork_release(inet_sk(sk), inet6_sk(sk)); ip6_cork_release(&inet_sk(sk)->cork, &inet6_sk(sk)->cork);
} }
EXPORT_SYMBOL_GPL(ip6_flush_pending_frames); EXPORT_SYMBOL_GPL(ip6_flush_pending_frames);
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