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

Merge branch 'net-dst_entry-shrink'

David Miller says:

====================
net: Significantly shrink the size of routes.

Through a combination of several things, our route structures are
larger than they need to be.

Mostly this stems from having members in dst_entry which are only used
by one class of routes.  So the majority of the work in this series is
about "un-commoning" these members and pushing them into the type
specific structures.

Unfortunately, IPSEC needed the most surgery.  The majority of the
changes here had to do with bundle creation and management.

The other issue is the refcount alignment in dst_entry.  Once we get
rid of the not-so-common members, it really opens the door to removing
that alignment entirely.

I think the new layout looks really nice, so I'll reproduce it here:

	struct net_device       *dev;
	struct  dst_ops	        *ops;
	unsigned long		_metrics;
	unsigned long           expires;
	struct xfrm_state	*xfrm;
	int			(*input)(struct sk_buff *);
	int			(*output)(struct net *net, struct sock *sk, struct sk_buff *skb);
	unsigned short		flags;
	short			obsolete;
	unsigned short		header_len;
	unsigned short		trailer_len;
	atomic_t		__refcnt;
	int			__use;
	unsigned long		lastuse;
	struct lwtunnel_state   *lwtstate;
	struct rcu_head		rcu_head;
	short			error;
	short			__pad;
	__u32			tclassid;

(This is for 64-bit, on 32-bit the __refcnt comes at the very end)

So, the good news:

1) struct dst_entry shrinks from 160 to 112 bytes.

2) struct rtable shrinks from 216 to 168 bytes.

3) struct rt6_info shrinks from 384 to 320 bytes.

Enjoy.

v2:
	Collapse some patches logically based upon feedback.
	Fix the strange patch #7.

v3:	xfrm_dst_path() needs inline keyword
	Properly align __refcnt on 32-bit.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents b78a6aa3 7149f813
...@@ -69,6 +69,7 @@ int dn_route_rcv(struct sk_buff *skb, struct net_device *dev, ...@@ -69,6 +69,7 @@ int dn_route_rcv(struct sk_buff *skb, struct net_device *dev,
*/ */
struct dn_route { struct dn_route {
struct dst_entry dst; struct dst_entry dst;
struct dn_route __rcu *dn_next;
struct neighbour *n; struct neighbour *n;
......
...@@ -34,13 +34,9 @@ struct sk_buff; ...@@ -34,13 +34,9 @@ struct sk_buff;
struct dst_entry { struct dst_entry {
struct net_device *dev; struct net_device *dev;
struct rcu_head rcu_head;
struct dst_entry *child;
struct dst_ops *ops; struct dst_ops *ops;
unsigned long _metrics; unsigned long _metrics;
unsigned long expires; unsigned long expires;
struct dst_entry *path;
struct dst_entry *from;
#ifdef CONFIG_XFRM #ifdef CONFIG_XFRM
struct xfrm_state *xfrm; struct xfrm_state *xfrm;
#else #else
...@@ -59,8 +55,6 @@ struct dst_entry { ...@@ -59,8 +55,6 @@ struct dst_entry {
#define DST_XFRM_QUEUE 0x0040 #define DST_XFRM_QUEUE 0x0040
#define DST_METADATA 0x0080 #define DST_METADATA 0x0080
short error;
/* A non-zero value of dst->obsolete forces by-hand validation /* A non-zero value of dst->obsolete forces by-hand validation
* of the route entry. Positive values are set by the generic * of the route entry. Positive values are set by the generic
* dst layer to indicate that the entry has been forcefully * dst layer to indicate that the entry has been forcefully
...@@ -76,35 +70,24 @@ struct dst_entry { ...@@ -76,35 +70,24 @@ struct dst_entry {
#define DST_OBSOLETE_KILL -2 #define DST_OBSOLETE_KILL -2
unsigned short header_len; /* more space at head required */ unsigned short header_len; /* more space at head required */
unsigned short trailer_len; /* space to reserve at tail */ unsigned short trailer_len; /* space to reserve at tail */
unsigned short __pad3;
#ifdef CONFIG_IP_ROUTE_CLASSID
__u32 tclassid;
#else
__u32 __pad2;
#endif
#ifdef CONFIG_64BIT
/*
* Align __refcnt to a 64 bytes alignment
* (L1_CACHE_SIZE would be too much)
*/
long __pad_to_align_refcnt[2];
#endif
/* /*
* __refcnt wants to be on a different cache line from * __refcnt wants to be on a different cache line from
* input/output/ops or performance tanks badly * input/output/ops or performance tanks badly
*/ */
atomic_t __refcnt; /* client references */ #ifdef CONFIG_64BIT
atomic_t __refcnt; /* 64-bit offset 64 */
#endif
int __use; int __use;
unsigned long lastuse; unsigned long lastuse;
struct lwtunnel_state *lwtstate; struct lwtunnel_state *lwtstate;
union { struct rcu_head rcu_head;
struct dst_entry *next; short error;
struct rtable __rcu *rt_next; short __pad;
struct rt6_info __rcu *rt6_next; __u32 tclassid;
struct dn_route __rcu *dn_next; #ifndef CONFIG_64BIT
}; atomic_t __refcnt; /* 32-bit offset 64 */
#endif
}; };
struct dst_metrics { struct dst_metrics {
...@@ -250,7 +233,7 @@ static inline void dst_hold(struct dst_entry *dst) ...@@ -250,7 +233,7 @@ static inline void dst_hold(struct dst_entry *dst)
{ {
/* /*
* If your kernel compilation stops here, please check * If your kernel compilation stops here, please check
* __pad_to_align_refcnt declaration in struct dst_entry * the placement of __refcnt in struct dst_entry
*/ */
BUILD_BUG_ON(offsetof(struct dst_entry, __refcnt) & 63); BUILD_BUG_ON(offsetof(struct dst_entry, __refcnt) & 63);
WARN_ON(atomic_inc_not_zero(&dst->__refcnt) == 0); WARN_ON(atomic_inc_not_zero(&dst->__refcnt) == 0);
......
...@@ -129,6 +129,8 @@ struct rt6_exception { ...@@ -129,6 +129,8 @@ struct rt6_exception {
struct rt6_info { struct rt6_info {
struct dst_entry dst; struct dst_entry dst;
struct rt6_info __rcu *rt6_next;
struct rt6_info *from;
/* /*
* Tail elements of dst_entry (__refcnt etc.) * Tail elements of dst_entry (__refcnt etc.)
...@@ -176,11 +178,11 @@ struct rt6_info { ...@@ -176,11 +178,11 @@ struct rt6_info {
#define for_each_fib6_node_rt_rcu(fn) \ #define for_each_fib6_node_rt_rcu(fn) \
for (rt = rcu_dereference((fn)->leaf); rt; \ for (rt = rcu_dereference((fn)->leaf); rt; \
rt = rcu_dereference(rt->dst.rt6_next)) rt = rcu_dereference(rt->rt6_next))
#define for_each_fib6_walker_rt(w) \ #define for_each_fib6_walker_rt(w) \
for (rt = (w)->leaf; rt; \ for (rt = (w)->leaf; rt; \
rt = rcu_dereference_protected(rt->dst.rt6_next, 1)) rt = rcu_dereference_protected(rt->rt6_next, 1))
static inline struct inet6_dev *ip6_dst_idev(struct dst_entry *dst) static inline struct inet6_dev *ip6_dst_idev(struct dst_entry *dst)
{ {
...@@ -203,11 +205,9 @@ static inline void rt6_update_expires(struct rt6_info *rt0, int timeout) ...@@ -203,11 +205,9 @@ static inline void rt6_update_expires(struct rt6_info *rt0, int timeout)
{ {
struct rt6_info *rt; struct rt6_info *rt;
for (rt = rt0; rt && !(rt->rt6i_flags & RTF_EXPIRES); for (rt = rt0; rt && !(rt->rt6i_flags & RTF_EXPIRES); rt = rt->from);
rt = (struct rt6_info *)rt->dst.from);
if (rt && rt != rt0) if (rt && rt != rt0)
rt0->dst.expires = rt->dst.expires; rt0->dst.expires = rt->dst.expires;
dst_set_expires(&rt0->dst, timeout); dst_set_expires(&rt0->dst, timeout);
rt0->rt6i_flags |= RTF_EXPIRES; rt0->rt6i_flags |= RTF_EXPIRES;
} }
...@@ -242,8 +242,8 @@ static inline u32 rt6_get_cookie(const struct rt6_info *rt) ...@@ -242,8 +242,8 @@ static inline u32 rt6_get_cookie(const struct rt6_info *rt)
u32 cookie = 0; u32 cookie = 0;
if (rt->rt6i_flags & RTF_PCPU || if (rt->rt6i_flags & RTF_PCPU ||
(unlikely(!list_empty(&rt->rt6i_uncached)) && rt->dst.from)) (unlikely(!list_empty(&rt->rt6i_uncached)) && rt->from))
rt = (struct rt6_info *)(rt->dst.from); rt = rt->from;
rt6_get_cookie_safe(rt, &cookie); rt6_get_cookie_safe(rt, &cookie);
......
...@@ -968,7 +968,7 @@ static inline bool xfrm_sec_ctx_match(struct xfrm_sec_ctx *s1, struct xfrm_sec_c ...@@ -968,7 +968,7 @@ static inline bool xfrm_sec_ctx_match(struct xfrm_sec_ctx *s1, struct xfrm_sec_c
/* A struct encoding bundle of transformations to apply to some set of flow. /* A struct encoding bundle of transformations to apply to some set of flow.
* *
* dst->child points to the next element of bundle. * xdst->child points to the next element of bundle.
* dst->xfrm points to an instanse of transformer. * dst->xfrm points to an instanse of transformer.
* *
* Due to unfortunate limitations of current routing cache, which we * Due to unfortunate limitations of current routing cache, which we
...@@ -984,6 +984,8 @@ struct xfrm_dst { ...@@ -984,6 +984,8 @@ struct xfrm_dst {
struct rt6_info rt6; struct rt6_info rt6;
} u; } u;
struct dst_entry *route; struct dst_entry *route;
struct dst_entry *child;
struct dst_entry *path;
struct xfrm_policy *pols[XFRM_POLICY_TYPE_MAX]; struct xfrm_policy *pols[XFRM_POLICY_TYPE_MAX];
int num_pols, num_xfrms; int num_pols, num_xfrms;
u32 xfrm_genid; u32 xfrm_genid;
...@@ -994,7 +996,35 @@ struct xfrm_dst { ...@@ -994,7 +996,35 @@ struct xfrm_dst {
u32 path_cookie; u32 path_cookie;
}; };
static inline struct dst_entry *xfrm_dst_path(const struct dst_entry *dst)
{
#ifdef CONFIG_XFRM #ifdef CONFIG_XFRM
if (dst->xfrm) {
const struct xfrm_dst *xdst = (const struct xfrm_dst *) dst;
return xdst->path;
}
#endif
return (struct dst_entry *) dst;
}
static inline struct dst_entry *xfrm_dst_child(const struct dst_entry *dst)
{
#ifdef CONFIG_XFRM
if (dst->xfrm) {
struct xfrm_dst *xdst = (struct xfrm_dst *) dst;
return xdst->child;
}
#endif
return NULL;
}
#ifdef CONFIG_XFRM
static inline void xfrm_dst_set_child(struct xfrm_dst *xdst, struct dst_entry *child)
{
xdst->child = child;
}
static inline void xfrm_dst_destroy(struct xfrm_dst *xdst) static inline void xfrm_dst_destroy(struct xfrm_dst *xdst)
{ {
xfrm_pols_put(xdst->pols, xdst->num_pols); xfrm_pols_put(xdst->pols, xdst->num_pols);
...@@ -1866,12 +1896,14 @@ bool xfrm_dev_offload_ok(struct sk_buff *skb, struct xfrm_state *x); ...@@ -1866,12 +1896,14 @@ bool xfrm_dev_offload_ok(struct sk_buff *skb, struct xfrm_state *x);
static inline bool xfrm_dst_offload_ok(struct dst_entry *dst) static inline bool xfrm_dst_offload_ok(struct dst_entry *dst)
{ {
struct xfrm_state *x = dst->xfrm; struct xfrm_state *x = dst->xfrm;
struct xfrm_dst *xdst;
if (!x || !x->type_offload) if (!x || !x->type_offload)
return false; return false;
if (x->xso.offload_handle && (x->xso.dev == dst->path->dev) && xdst = (struct xfrm_dst *) dst;
!dst->child->xfrm) if (x->xso.offload_handle && (x->xso.dev == xfrm_dst_path(dst)->dev) &&
!xdst->child->xfrm)
return true; return true;
return false; return false;
......
...@@ -78,7 +78,6 @@ void br_netfilter_rtable_init(struct net_bridge *br) ...@@ -78,7 +78,6 @@ void br_netfilter_rtable_init(struct net_bridge *br)
atomic_set(&rt->dst.__refcnt, 1); atomic_set(&rt->dst.__refcnt, 1);
rt->dst.dev = br->dev; rt->dst.dev = br->dev;
rt->dst.path = &rt->dst;
dst_init_metrics(&rt->dst, br_dst_default_metrics, true); dst_init_metrics(&rt->dst, br_dst_default_metrics, true);
rt->dst.flags = DST_NOXFRM | DST_FAKE_RTABLE; rt->dst.flags = DST_NOXFRM | DST_FAKE_RTABLE;
rt->dst.ops = &fake_dst_ops; rt->dst.ops = &fake_dst_ops;
......
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/prefetch.h> #include <linux/prefetch.h>
#include <net/lwtunnel.h> #include <net/lwtunnel.h>
#include <net/xfrm.h>
#include <net/dst.h> #include <net/dst.h>
#include <net/dst_metadata.h> #include <net/dst_metadata.h>
...@@ -62,15 +63,12 @@ void dst_init(struct dst_entry *dst, struct dst_ops *ops, ...@@ -62,15 +63,12 @@ void dst_init(struct dst_entry *dst, struct dst_ops *ops,
struct net_device *dev, int initial_ref, int initial_obsolete, struct net_device *dev, int initial_ref, int initial_obsolete,
unsigned short flags) unsigned short flags)
{ {
dst->child = NULL;
dst->dev = dev; dst->dev = dev;
if (dev) if (dev)
dev_hold(dev); dev_hold(dev);
dst->ops = ops; dst->ops = ops;
dst_init_metrics(dst, dst_default_metrics.metrics, true); dst_init_metrics(dst, dst_default_metrics.metrics, true);
dst->expires = 0UL; dst->expires = 0UL;
dst->path = dst;
dst->from = NULL;
#ifdef CONFIG_XFRM #ifdef CONFIG_XFRM
dst->xfrm = NULL; dst->xfrm = NULL;
#endif #endif
...@@ -88,7 +86,6 @@ void dst_init(struct dst_entry *dst, struct dst_ops *ops, ...@@ -88,7 +86,6 @@ void dst_init(struct dst_entry *dst, struct dst_ops *ops,
dst->__use = 0; dst->__use = 0;
dst->lastuse = jiffies; dst->lastuse = jiffies;
dst->flags = flags; dst->flags = flags;
dst->next = NULL;
if (!(flags & DST_NOCOUNT)) if (!(flags & DST_NOCOUNT))
dst_entries_add(ops, 1); dst_entries_add(ops, 1);
} }
...@@ -116,12 +113,17 @@ EXPORT_SYMBOL(dst_alloc); ...@@ -116,12 +113,17 @@ EXPORT_SYMBOL(dst_alloc);
struct dst_entry *dst_destroy(struct dst_entry * dst) struct dst_entry *dst_destroy(struct dst_entry * dst)
{ {
struct dst_entry *child; struct dst_entry *child = NULL;
smp_rmb(); smp_rmb();
child = dst->child; #ifdef CONFIG_XFRM
if (dst->xfrm) {
struct xfrm_dst *xdst = (struct xfrm_dst *) dst;
child = xdst->child;
}
#endif
if (!(dst->flags & DST_NOCOUNT)) if (!(dst->flags & DST_NOCOUNT))
dst_entries_add(dst->ops, -1); dst_entries_add(dst->ops, -1);
......
...@@ -399,7 +399,7 @@ struct pktgen_dev { ...@@ -399,7 +399,7 @@ struct pktgen_dev {
__u8 ipsmode; /* IPSEC mode (config) */ __u8 ipsmode; /* IPSEC mode (config) */
__u8 ipsproto; /* IPSEC type (config) */ __u8 ipsproto; /* IPSEC type (config) */
__u32 spi; __u32 spi;
struct dst_entry dst; struct xfrm_dst xdst;
struct dst_ops dstops; struct dst_ops dstops;
#endif #endif
char result[512]; char result[512];
...@@ -2609,7 +2609,7 @@ static int pktgen_output_ipsec(struct sk_buff *skb, struct pktgen_dev *pkt_dev) ...@@ -2609,7 +2609,7 @@ static int pktgen_output_ipsec(struct sk_buff *skb, struct pktgen_dev *pkt_dev)
* supports both transport/tunnel mode + ESP/AH type. * supports both transport/tunnel mode + ESP/AH type.
*/ */
if ((x->props.mode == XFRM_MODE_TUNNEL) && (pkt_dev->spi != 0)) if ((x->props.mode == XFRM_MODE_TUNNEL) && (pkt_dev->spi != 0))
skb->_skb_refdst = (unsigned long)&pkt_dev->dst | SKB_DST_NOREF; skb->_skb_refdst = (unsigned long)&pkt_dev->xdst.u.dst | SKB_DST_NOREF;
rcu_read_lock_bh(); rcu_read_lock_bh();
err = x->outer_mode->output(x, skb); err = x->outer_mode->output(x, skb);
...@@ -3742,10 +3742,10 @@ static int pktgen_add_device(struct pktgen_thread *t, const char *ifname) ...@@ -3742,10 +3742,10 @@ static int pktgen_add_device(struct pktgen_thread *t, const char *ifname)
* performance under such circumstance. * performance under such circumstance.
*/ */
pkt_dev->dstops.family = AF_INET; pkt_dev->dstops.family = AF_INET;
pkt_dev->dst.dev = pkt_dev->odev; pkt_dev->xdst.u.dst.dev = pkt_dev->odev;
dst_init_metrics(&pkt_dev->dst, pktgen_dst_metrics, false); dst_init_metrics(&pkt_dev->xdst.u.dst, pktgen_dst_metrics, false);
pkt_dev->dst.child = &pkt_dev->dst; pkt_dev->xdst.child = &pkt_dev->xdst.u.dst;
pkt_dev->dst.ops = &pkt_dev->dstops; pkt_dev->xdst.u.dst.ops = &pkt_dev->dstops;
#endif #endif
return add_dev_to_thread(t, pkt_dev); return add_dev_to_thread(t, pkt_dev);
......
...@@ -199,11 +199,11 @@ static void dn_dst_check_expire(struct timer_list *unused) ...@@ -199,11 +199,11 @@ static void dn_dst_check_expire(struct timer_list *unused)
lockdep_is_held(&dn_rt_hash_table[i].lock))) != NULL) { lockdep_is_held(&dn_rt_hash_table[i].lock))) != NULL) {
if (atomic_read(&rt->dst.__refcnt) > 1 || if (atomic_read(&rt->dst.__refcnt) > 1 ||
(now - rt->dst.lastuse) < expire) { (now - rt->dst.lastuse) < expire) {
rtp = &rt->dst.dn_next; rtp = &rt->dn_next;
continue; continue;
} }
*rtp = rt->dst.dn_next; *rtp = rt->dn_next;
rt->dst.dn_next = NULL; rt->dn_next = NULL;
dst_dev_put(&rt->dst); dst_dev_put(&rt->dst);
dst_release(&rt->dst); dst_release(&rt->dst);
} }
...@@ -233,11 +233,11 @@ static int dn_dst_gc(struct dst_ops *ops) ...@@ -233,11 +233,11 @@ static int dn_dst_gc(struct dst_ops *ops)
lockdep_is_held(&dn_rt_hash_table[i].lock))) != NULL) { lockdep_is_held(&dn_rt_hash_table[i].lock))) != NULL) {
if (atomic_read(&rt->dst.__refcnt) > 1 || if (atomic_read(&rt->dst.__refcnt) > 1 ||
(now - rt->dst.lastuse) < expire) { (now - rt->dst.lastuse) < expire) {
rtp = &rt->dst.dn_next; rtp = &rt->dn_next;
continue; continue;
} }
*rtp = rt->dst.dn_next; *rtp = rt->dn_next;
rt->dst.dn_next = NULL; rt->dn_next = NULL;
dst_dev_put(&rt->dst); dst_dev_put(&rt->dst);
dst_release(&rt->dst); dst_release(&rt->dst);
break; break;
...@@ -333,8 +333,8 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou ...@@ -333,8 +333,8 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou
lockdep_is_held(&dn_rt_hash_table[hash].lock))) != NULL) { lockdep_is_held(&dn_rt_hash_table[hash].lock))) != NULL) {
if (compare_keys(&rth->fld, &rt->fld)) { if (compare_keys(&rth->fld, &rt->fld)) {
/* Put it first */ /* Put it first */
*rthp = rth->dst.dn_next; *rthp = rth->dn_next;
rcu_assign_pointer(rth->dst.dn_next, rcu_assign_pointer(rth->dn_next,
dn_rt_hash_table[hash].chain); dn_rt_hash_table[hash].chain);
rcu_assign_pointer(dn_rt_hash_table[hash].chain, rth); rcu_assign_pointer(dn_rt_hash_table[hash].chain, rth);
...@@ -345,10 +345,10 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou ...@@ -345,10 +345,10 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou
*rp = rth; *rp = rth;
return 0; return 0;
} }
rthp = &rth->dst.dn_next; rthp = &rth->dn_next;
} }
rcu_assign_pointer(rt->dst.dn_next, dn_rt_hash_table[hash].chain); rcu_assign_pointer(rt->dn_next, dn_rt_hash_table[hash].chain);
rcu_assign_pointer(dn_rt_hash_table[hash].chain, rt); rcu_assign_pointer(dn_rt_hash_table[hash].chain, rt);
dst_hold_and_use(&rt->dst, now); dst_hold_and_use(&rt->dst, now);
...@@ -369,8 +369,8 @@ static void dn_run_flush(struct timer_list *unused) ...@@ -369,8 +369,8 @@ static void dn_run_flush(struct timer_list *unused)
goto nothing_to_declare; goto nothing_to_declare;
for(; rt; rt = next) { for(; rt; rt = next) {
next = rcu_dereference_raw(rt->dst.dn_next); next = rcu_dereference_raw(rt->dn_next);
RCU_INIT_POINTER(rt->dst.dn_next, NULL); RCU_INIT_POINTER(rt->dn_next, NULL);
dst_dev_put(&rt->dst); dst_dev_put(&rt->dst);
dst_release(&rt->dst); dst_release(&rt->dst);
} }
...@@ -1183,6 +1183,7 @@ static int dn_route_output_slow(struct dst_entry **pprt, const struct flowidn *o ...@@ -1183,6 +1183,7 @@ static int dn_route_output_slow(struct dst_entry **pprt, const struct flowidn *o
if (rt == NULL) if (rt == NULL)
goto e_nobufs; goto e_nobufs;
rt->dn_next = NULL;
memset(&rt->fld, 0, sizeof(rt->fld)); memset(&rt->fld, 0, sizeof(rt->fld));
rt->fld.saddr = oldflp->saddr; rt->fld.saddr = oldflp->saddr;
rt->fld.daddr = oldflp->daddr; rt->fld.daddr = oldflp->daddr;
...@@ -1252,7 +1253,7 @@ static int __dn_route_output_key(struct dst_entry **pprt, const struct flowidn * ...@@ -1252,7 +1253,7 @@ static int __dn_route_output_key(struct dst_entry **pprt, const struct flowidn *
if (!(flags & MSG_TRYHARD)) { if (!(flags & MSG_TRYHARD)) {
rcu_read_lock_bh(); rcu_read_lock_bh();
for (rt = rcu_dereference_bh(dn_rt_hash_table[hash].chain); rt; for (rt = rcu_dereference_bh(dn_rt_hash_table[hash].chain); rt;
rt = rcu_dereference_bh(rt->dst.dn_next)) { rt = rcu_dereference_bh(rt->dn_next)) {
if ((flp->daddr == rt->fld.daddr) && if ((flp->daddr == rt->fld.daddr) &&
(flp->saddr == rt->fld.saddr) && (flp->saddr == rt->fld.saddr) &&
(flp->flowidn_mark == rt->fld.flowidn_mark) && (flp->flowidn_mark == rt->fld.flowidn_mark) &&
...@@ -1448,6 +1449,7 @@ static int dn_route_input_slow(struct sk_buff *skb) ...@@ -1448,6 +1449,7 @@ static int dn_route_input_slow(struct sk_buff *skb)
if (rt == NULL) if (rt == NULL)
goto e_nobufs; goto e_nobufs;
rt->dn_next = NULL;
memset(&rt->fld, 0, sizeof(rt->fld)); memset(&rt->fld, 0, sizeof(rt->fld));
rt->rt_saddr = fld.saddr; rt->rt_saddr = fld.saddr;
rt->rt_daddr = fld.daddr; rt->rt_daddr = fld.daddr;
...@@ -1529,7 +1531,7 @@ static int dn_route_input(struct sk_buff *skb) ...@@ -1529,7 +1531,7 @@ static int dn_route_input(struct sk_buff *skb)
rcu_read_lock(); rcu_read_lock();
for(rt = rcu_dereference(dn_rt_hash_table[hash].chain); rt != NULL; for(rt = rcu_dereference(dn_rt_hash_table[hash].chain); rt != NULL;
rt = rcu_dereference(rt->dst.dn_next)) { rt = rcu_dereference(rt->dn_next)) {
if ((rt->fld.saddr == cb->src) && if ((rt->fld.saddr == cb->src) &&
(rt->fld.daddr == cb->dst) && (rt->fld.daddr == cb->dst) &&
(rt->fld.flowidn_oif == 0) && (rt->fld.flowidn_oif == 0) &&
...@@ -1749,7 +1751,7 @@ int dn_cache_dump(struct sk_buff *skb, struct netlink_callback *cb) ...@@ -1749,7 +1751,7 @@ int dn_cache_dump(struct sk_buff *skb, struct netlink_callback *cb)
rcu_read_lock_bh(); rcu_read_lock_bh();
for(rt = rcu_dereference_bh(dn_rt_hash_table[h].chain), idx = 0; for(rt = rcu_dereference_bh(dn_rt_hash_table[h].chain), idx = 0;
rt; rt;
rt = rcu_dereference_bh(rt->dst.dn_next), idx++) { rt = rcu_dereference_bh(rt->dn_next), idx++) {
if (idx < s_idx) if (idx < s_idx)
continue; continue;
skb_dst_set(skb, dst_clone(&rt->dst)); skb_dst_set(skb, dst_clone(&rt->dst));
...@@ -1795,7 +1797,7 @@ static struct dn_route *dn_rt_cache_get_next(struct seq_file *seq, struct dn_rou ...@@ -1795,7 +1797,7 @@ static struct dn_route *dn_rt_cache_get_next(struct seq_file *seq, struct dn_rou
{ {
struct dn_rt_cache_iter_state *s = seq->private; struct dn_rt_cache_iter_state *s = seq->private;
rt = rcu_dereference_bh(rt->dst.dn_next); rt = rcu_dereference_bh(rt->dn_next);
while (!rt) { while (!rt) {
rcu_read_unlock_bh(); rcu_read_unlock_bh();
if (--s->bucket < 0) if (--s->bucket < 0)
......
...@@ -1106,7 +1106,7 @@ void ipv4_sk_update_pmtu(struct sk_buff *skb, struct sock *sk, u32 mtu) ...@@ -1106,7 +1106,7 @@ void ipv4_sk_update_pmtu(struct sk_buff *skb, struct sock *sk, u32 mtu)
new = true; new = true;
} }
__ip_rt_update_pmtu((struct rtable *) rt->dst.path, &fl4, mtu); __ip_rt_update_pmtu((struct rtable *) xfrm_dst_path(&rt->dst), &fl4, mtu);
if (!dst_check(&rt->dst, 0)) { if (!dst_check(&rt->dst, 0)) {
if (new) if (new)
......
...@@ -62,7 +62,7 @@ static int xfrm4_mode_tunnel_output(struct xfrm_state *x, struct sk_buff *skb) ...@@ -62,7 +62,7 @@ static int xfrm4_mode_tunnel_output(struct xfrm_state *x, struct sk_buff *skb)
top_iph->frag_off = (flags & XFRM_STATE_NOPMTUDISC) ? top_iph->frag_off = (flags & XFRM_STATE_NOPMTUDISC) ?
0 : (XFRM_MODE_SKB_CB(skb)->frag_off & htons(IP_DF)); 0 : (XFRM_MODE_SKB_CB(skb)->frag_off & htons(IP_DF));
top_iph->ttl = ip4_dst_hoplimit(dst->child); top_iph->ttl = ip4_dst_hoplimit(xfrm_dst_child(dst));
top_iph->saddr = x->props.saddr.a4; top_iph->saddr = x->props.saddr.a4;
top_iph->daddr = x->id.daddr.a4; top_iph->daddr = x->id.daddr.a4;
......
...@@ -893,7 +893,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -893,7 +893,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
ins = &fn->leaf; ins = &fn->leaf;
for (iter = leaf; iter; for (iter = leaf; iter;
iter = rcu_dereference_protected(iter->dst.rt6_next, iter = rcu_dereference_protected(iter->rt6_next,
lockdep_is_held(&rt->rt6i_table->tb6_lock))) { lockdep_is_held(&rt->rt6i_table->tb6_lock))) {
/* /*
* Search for duplicates * Search for duplicates
...@@ -950,7 +950,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -950,7 +950,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
break; break;
next_iter: next_iter:
ins = &iter->dst.rt6_next; ins = &iter->rt6_next;
} }
if (fallback_ins && !found) { if (fallback_ins && !found) {
...@@ -979,7 +979,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -979,7 +979,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
&sibling->rt6i_siblings); &sibling->rt6i_siblings);
break; break;
} }
sibling = rcu_dereference_protected(sibling->dst.rt6_next, sibling = rcu_dereference_protected(sibling->rt6_next,
lockdep_is_held(&rt->rt6i_table->tb6_lock)); lockdep_is_held(&rt->rt6i_table->tb6_lock));
} }
/* For each sibling in the list, increment the counter of /* For each sibling in the list, increment the counter of
...@@ -1009,7 +1009,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -1009,7 +1009,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
if (err) if (err)
return err; return err;
rcu_assign_pointer(rt->dst.rt6_next, iter); rcu_assign_pointer(rt->rt6_next, iter);
atomic_inc(&rt->rt6i_ref); atomic_inc(&rt->rt6i_ref);
rcu_assign_pointer(rt->rt6i_node, fn); rcu_assign_pointer(rt->rt6i_node, fn);
rcu_assign_pointer(*ins, rt); rcu_assign_pointer(*ins, rt);
...@@ -1040,7 +1040,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -1040,7 +1040,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
atomic_inc(&rt->rt6i_ref); atomic_inc(&rt->rt6i_ref);
rcu_assign_pointer(rt->rt6i_node, fn); rcu_assign_pointer(rt->rt6i_node, fn);
rt->dst.rt6_next = iter->dst.rt6_next; rt->rt6_next = iter->rt6_next;
rcu_assign_pointer(*ins, rt); rcu_assign_pointer(*ins, rt);
call_fib6_entry_notifiers(info->nl_net, FIB_EVENT_ENTRY_REPLACE, call_fib6_entry_notifiers(info->nl_net, FIB_EVENT_ENTRY_REPLACE,
rt, extack); rt, extack);
...@@ -1059,14 +1059,14 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -1059,14 +1059,14 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
if (nsiblings) { if (nsiblings) {
/* Replacing an ECMP route, remove all siblings */ /* Replacing an ECMP route, remove all siblings */
ins = &rt->dst.rt6_next; ins = &rt->rt6_next;
iter = rcu_dereference_protected(*ins, iter = rcu_dereference_protected(*ins,
lockdep_is_held(&rt->rt6i_table->tb6_lock)); lockdep_is_held(&rt->rt6i_table->tb6_lock));
while (iter) { while (iter) {
if (iter->rt6i_metric > rt->rt6i_metric) if (iter->rt6i_metric > rt->rt6i_metric)
break; break;
if (rt6_qualify_for_ecmp(iter)) { if (rt6_qualify_for_ecmp(iter)) {
*ins = iter->dst.rt6_next; *ins = iter->rt6_next;
iter->rt6i_node = NULL; iter->rt6i_node = NULL;
fib6_purge_rt(iter, fn, info->nl_net); fib6_purge_rt(iter, fn, info->nl_net);
if (rcu_access_pointer(fn->rr_ptr) == iter) if (rcu_access_pointer(fn->rr_ptr) == iter)
...@@ -1075,7 +1075,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt, ...@@ -1075,7 +1075,7 @@ static int fib6_add_rt2node(struct fib6_node *fn, struct rt6_info *rt,
nsiblings--; nsiblings--;
info->nl_net->ipv6.rt6_stats->fib_rt_entries--; info->nl_net->ipv6.rt6_stats->fib_rt_entries--;
} else { } else {
ins = &iter->dst.rt6_next; ins = &iter->rt6_next;
} }
iter = rcu_dereference_protected(*ins, iter = rcu_dereference_protected(*ins,
lockdep_is_held(&rt->rt6i_table->tb6_lock)); lockdep_is_held(&rt->rt6i_table->tb6_lock));
...@@ -1644,7 +1644,7 @@ static void fib6_del_route(struct fib6_table *table, struct fib6_node *fn, ...@@ -1644,7 +1644,7 @@ static void fib6_del_route(struct fib6_table *table, struct fib6_node *fn,
WARN_ON_ONCE(rt->rt6i_flags & RTF_CACHE); WARN_ON_ONCE(rt->rt6i_flags & RTF_CACHE);
/* Unlink it */ /* Unlink it */
*rtp = rt->dst.rt6_next; *rtp = rt->rt6_next;
rt->rt6i_node = NULL; rt->rt6i_node = NULL;
net->ipv6.rt6_stats->fib_rt_entries--; net->ipv6.rt6_stats->fib_rt_entries--;
net->ipv6.rt6_stats->fib_discarded_routes++; net->ipv6.rt6_stats->fib_discarded_routes++;
...@@ -1672,7 +1672,7 @@ static void fib6_del_route(struct fib6_table *table, struct fib6_node *fn, ...@@ -1672,7 +1672,7 @@ static void fib6_del_route(struct fib6_table *table, struct fib6_node *fn,
FOR_WALKERS(net, w) { FOR_WALKERS(net, w) {
if (w->state == FWS_C && w->leaf == rt) { if (w->state == FWS_C && w->leaf == rt) {
RT6_TRACE("walker %p adjusted by delroute\n", w); RT6_TRACE("walker %p adjusted by delroute\n", w);
w->leaf = rcu_dereference_protected(rt->dst.rt6_next, w->leaf = rcu_dereference_protected(rt->rt6_next,
lockdep_is_held(&table->tb6_lock)); lockdep_is_held(&table->tb6_lock));
if (!w->leaf) if (!w->leaf)
w->state = FWS_U; w->state = FWS_U;
...@@ -1731,7 +1731,7 @@ int fib6_del(struct rt6_info *rt, struct nl_info *info) ...@@ -1731,7 +1731,7 @@ int fib6_del(struct rt6_info *rt, struct nl_info *info)
fib6_del_route(table, fn, rtp, info); fib6_del_route(table, fn, rtp, info);
return 0; return 0;
} }
rtp_next = &cur->dst.rt6_next; rtp_next = &cur->rt6_next;
} }
return -ENOENT; return -ENOENT;
} }
...@@ -2208,7 +2208,7 @@ static int ipv6_route_yield(struct fib6_walker *w) ...@@ -2208,7 +2208,7 @@ static int ipv6_route_yield(struct fib6_walker *w)
do { do {
iter->w.leaf = rcu_dereference_protected( iter->w.leaf = rcu_dereference_protected(
iter->w.leaf->dst.rt6_next, iter->w.leaf->rt6_next,
lockdep_is_held(&iter->tbl->tb6_lock)); lockdep_is_held(&iter->tbl->tb6_lock));
iter->skip--; iter->skip--;
if (!iter->skip && iter->w.leaf) if (!iter->skip && iter->w.leaf)
...@@ -2274,7 +2274,7 @@ static void *ipv6_route_seq_next(struct seq_file *seq, void *v, loff_t *pos) ...@@ -2274,7 +2274,7 @@ static void *ipv6_route_seq_next(struct seq_file *seq, void *v, loff_t *pos)
if (!v) if (!v)
goto iter_table; goto iter_table;
n = rcu_dereference_bh(((struct rt6_info *)v)->dst.rt6_next); n = rcu_dereference_bh(((struct rt6_info *)v)->rt6_next);
if (n) { if (n) {
++*pos; ++*pos;
return n; return n;
......
...@@ -1201,13 +1201,13 @@ static int ip6_setup_cork(struct sock *sk, struct inet_cork_full *cork, ...@@ -1201,13 +1201,13 @@ static int ip6_setup_cork(struct sock *sk, struct inet_cork_full *cork,
rt->dst.dev->mtu : dst_mtu(&rt->dst); rt->dst.dev->mtu : dst_mtu(&rt->dst);
else else
mtu = np->pmtudisc >= IPV6_PMTUDISC_PROBE ? mtu = np->pmtudisc >= IPV6_PMTUDISC_PROBE ?
rt->dst.dev->mtu : dst_mtu(rt->dst.path); rt->dst.dev->mtu : dst_mtu(xfrm_dst_path(&rt->dst));
if (np->frag_size < mtu) { if (np->frag_size < mtu) {
if (np->frag_size) if (np->frag_size)
mtu = np->frag_size; mtu = np->frag_size;
} }
cork->base.fragsize = mtu; cork->base.fragsize = mtu;
if (dst_allfrag(rt->dst.path)) if (dst_allfrag(xfrm_dst_path(&rt->dst)))
cork->base.flags |= IPCORK_ALLFRAG; cork->base.flags |= IPCORK_ALLFRAG;
cork->base.length = 0; cork->base.length = 0;
......
...@@ -186,7 +186,7 @@ static void rt6_uncached_list_flush_dev(struct net *net, struct net_device *dev) ...@@ -186,7 +186,7 @@ static void rt6_uncached_list_flush_dev(struct net *net, struct net_device *dev)
static u32 *rt6_pcpu_cow_metrics(struct rt6_info *rt) static u32 *rt6_pcpu_cow_metrics(struct rt6_info *rt)
{ {
return dst_metrics_write_ptr(rt->dst.from); return dst_metrics_write_ptr(&rt->from->dst);
} }
static u32 *ipv6_cow_metrics(struct dst_entry *dst, unsigned long old) static u32 *ipv6_cow_metrics(struct dst_entry *dst, unsigned long old)
...@@ -391,7 +391,7 @@ static void ip6_dst_destroy(struct dst_entry *dst) ...@@ -391,7 +391,7 @@ static void ip6_dst_destroy(struct dst_entry *dst)
{ {
struct rt6_info *rt = (struct rt6_info *)dst; struct rt6_info *rt = (struct rt6_info *)dst;
struct rt6_exception_bucket *bucket; struct rt6_exception_bucket *bucket;
struct dst_entry *from = dst->from; struct rt6_info *from = rt->from;
struct inet6_dev *idev; struct inet6_dev *idev;
dst_destroy_metrics_generic(dst); dst_destroy_metrics_generic(dst);
...@@ -409,8 +409,8 @@ static void ip6_dst_destroy(struct dst_entry *dst) ...@@ -409,8 +409,8 @@ static void ip6_dst_destroy(struct dst_entry *dst)
kfree(bucket); kfree(bucket);
} }
dst->from = NULL; rt->from = NULL;
dst_release(from); dst_release(&from->dst);
} }
static void ip6_dst_ifdown(struct dst_entry *dst, struct net_device *dev, static void ip6_dst_ifdown(struct dst_entry *dst, struct net_device *dev,
...@@ -443,9 +443,9 @@ static bool rt6_check_expired(const struct rt6_info *rt) ...@@ -443,9 +443,9 @@ static bool rt6_check_expired(const struct rt6_info *rt)
if (rt->rt6i_flags & RTF_EXPIRES) { if (rt->rt6i_flags & RTF_EXPIRES) {
if (time_after(jiffies, rt->dst.expires)) if (time_after(jiffies, rt->dst.expires))
return true; return true;
} else if (rt->dst.from) { } else if (rt->from) {
return rt->dst.obsolete != DST_OBSOLETE_FORCE_CHK || return rt->dst.obsolete != DST_OBSOLETE_FORCE_CHK ||
rt6_check_expired((struct rt6_info *)rt->dst.from); rt6_check_expired(rt->from);
} }
return false; return false;
} }
...@@ -502,7 +502,7 @@ static inline struct rt6_info *rt6_device_match(struct net *net, ...@@ -502,7 +502,7 @@ static inline struct rt6_info *rt6_device_match(struct net *net,
if (!oif && ipv6_addr_any(saddr)) if (!oif && ipv6_addr_any(saddr))
goto out; goto out;
for (sprt = rt; sprt; sprt = rcu_dereference(sprt->dst.rt6_next)) { for (sprt = rt; sprt; sprt = rcu_dereference(sprt->rt6_next)) {
struct net_device *dev = sprt->dst.dev; struct net_device *dev = sprt->dst.dev;
if (oif) { if (oif) {
...@@ -721,7 +721,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn, ...@@ -721,7 +721,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn,
match = NULL; match = NULL;
cont = NULL; cont = NULL;
for (rt = rr_head; rt; rt = rcu_dereference(rt->dst.rt6_next)) { for (rt = rr_head; rt; rt = rcu_dereference(rt->rt6_next)) {
if (rt->rt6i_metric != metric) { if (rt->rt6i_metric != metric) {
cont = rt; cont = rt;
break; break;
...@@ -731,7 +731,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn, ...@@ -731,7 +731,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn,
} }
for (rt = leaf; rt && rt != rr_head; for (rt = leaf; rt && rt != rr_head;
rt = rcu_dereference(rt->dst.rt6_next)) { rt = rcu_dereference(rt->rt6_next)) {
if (rt->rt6i_metric != metric) { if (rt->rt6i_metric != metric) {
cont = rt; cont = rt;
break; break;
...@@ -743,7 +743,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn, ...@@ -743,7 +743,7 @@ static struct rt6_info *find_rr_leaf(struct fib6_node *fn,
if (match || !cont) if (match || !cont)
return match; return match;
for (rt = cont; rt; rt = rcu_dereference(rt->dst.rt6_next)) for (rt = cont; rt; rt = rcu_dereference(rt->rt6_next))
match = find_match(rt, oif, strict, &mpri, match, do_rr); match = find_match(rt, oif, strict, &mpri, match, do_rr);
return match; return match;
...@@ -781,7 +781,7 @@ static struct rt6_info *rt6_select(struct net *net, struct fib6_node *fn, ...@@ -781,7 +781,7 @@ static struct rt6_info *rt6_select(struct net *net, struct fib6_node *fn,
&do_rr); &do_rr);
if (do_rr) { if (do_rr) {
struct rt6_info *next = rcu_dereference(rt0->dst.rt6_next); struct rt6_info *next = rcu_dereference(rt0->rt6_next);
/* no entries matched; do round-robin */ /* no entries matched; do round-robin */
if (!next || next->rt6i_metric != rt0->rt6i_metric) if (!next || next->rt6i_metric != rt0->rt6i_metric)
...@@ -1054,7 +1054,7 @@ static struct rt6_info *ip6_rt_cache_alloc(struct rt6_info *ort, ...@@ -1054,7 +1054,7 @@ static struct rt6_info *ip6_rt_cache_alloc(struct rt6_info *ort,
*/ */
if (ort->rt6i_flags & (RTF_CACHE | RTF_PCPU)) if (ort->rt6i_flags & (RTF_CACHE | RTF_PCPU))
ort = (struct rt6_info *)ort->dst.from; ort = ort->from;
rcu_read_lock(); rcu_read_lock();
dev = ip6_rt_get_dev_rcu(ort); dev = ip6_rt_get_dev_rcu(ort);
...@@ -1274,7 +1274,7 @@ static int rt6_insert_exception(struct rt6_info *nrt, ...@@ -1274,7 +1274,7 @@ static int rt6_insert_exception(struct rt6_info *nrt,
/* ort can't be a cache or pcpu route */ /* ort can't be a cache or pcpu route */
if (ort->rt6i_flags & (RTF_CACHE | RTF_PCPU)) if (ort->rt6i_flags & (RTF_CACHE | RTF_PCPU))
ort = (struct rt6_info *)ort->dst.from; ort = ort->from;
WARN_ON_ONCE(ort->rt6i_flags & (RTF_CACHE | RTF_PCPU)); WARN_ON_ONCE(ort->rt6i_flags & (RTF_CACHE | RTF_PCPU));
spin_lock_bh(&rt6_exception_lock); spin_lock_bh(&rt6_exception_lock);
...@@ -1415,8 +1415,8 @@ static struct rt6_info *rt6_find_cached_rt(struct rt6_info *rt, ...@@ -1415,8 +1415,8 @@ static struct rt6_info *rt6_find_cached_rt(struct rt6_info *rt,
/* Remove the passed in cached rt from the hash table that contains it */ /* Remove the passed in cached rt from the hash table that contains it */
int rt6_remove_exception_rt(struct rt6_info *rt) int rt6_remove_exception_rt(struct rt6_info *rt)
{ {
struct rt6_info *from = (struct rt6_info *)rt->dst.from;
struct rt6_exception_bucket *bucket; struct rt6_exception_bucket *bucket;
struct rt6_info *from = rt->from;
struct in6_addr *src_key = NULL; struct in6_addr *src_key = NULL;
struct rt6_exception *rt6_ex; struct rt6_exception *rt6_ex;
int err; int err;
...@@ -1460,8 +1460,8 @@ int rt6_remove_exception_rt(struct rt6_info *rt) ...@@ -1460,8 +1460,8 @@ int rt6_remove_exception_rt(struct rt6_info *rt)
*/ */
static void rt6_update_exception_stamp_rt(struct rt6_info *rt) static void rt6_update_exception_stamp_rt(struct rt6_info *rt)
{ {
struct rt6_info *from = (struct rt6_info *)rt->dst.from;
struct rt6_exception_bucket *bucket; struct rt6_exception_bucket *bucket;
struct rt6_info *from = rt->from;
struct in6_addr *src_key = NULL; struct in6_addr *src_key = NULL;
struct rt6_exception *rt6_ex; struct rt6_exception *rt6_ex;
...@@ -1929,9 +1929,9 @@ struct dst_entry *ip6_blackhole_route(struct net *net, struct dst_entry *dst_ori ...@@ -1929,9 +1929,9 @@ struct dst_entry *ip6_blackhole_route(struct net *net, struct dst_entry *dst_ori
static void rt6_dst_from_metrics_check(struct rt6_info *rt) static void rt6_dst_from_metrics_check(struct rt6_info *rt)
{ {
if (rt->dst.from && if (rt->from &&
dst_metrics_ptr(&rt->dst) != dst_metrics_ptr(rt->dst.from)) dst_metrics_ptr(&rt->dst) != dst_metrics_ptr(&rt->from->dst))
dst_init_metrics(&rt->dst, dst_metrics_ptr(rt->dst.from), true); dst_init_metrics(&rt->dst, dst_metrics_ptr(&rt->from->dst), true);
} }
static struct dst_entry *rt6_check(struct rt6_info *rt, u32 cookie) static struct dst_entry *rt6_check(struct rt6_info *rt, u32 cookie)
...@@ -1951,7 +1951,7 @@ static struct dst_entry *rt6_dst_from_check(struct rt6_info *rt, u32 cookie) ...@@ -1951,7 +1951,7 @@ static struct dst_entry *rt6_dst_from_check(struct rt6_info *rt, u32 cookie)
{ {
if (!__rt6_check_expired(rt) && if (!__rt6_check_expired(rt) &&
rt->dst.obsolete == DST_OBSOLETE_FORCE_CHK && rt->dst.obsolete == DST_OBSOLETE_FORCE_CHK &&
rt6_check((struct rt6_info *)(rt->dst.from), cookie)) rt6_check(rt->from, cookie))
return &rt->dst; return &rt->dst;
else else
return NULL; return NULL;
...@@ -1971,7 +1971,7 @@ static struct dst_entry *ip6_dst_check(struct dst_entry *dst, u32 cookie) ...@@ -1971,7 +1971,7 @@ static struct dst_entry *ip6_dst_check(struct dst_entry *dst, u32 cookie)
rt6_dst_from_metrics_check(rt); rt6_dst_from_metrics_check(rt);
if (rt->rt6i_flags & RTF_PCPU || if (rt->rt6i_flags & RTF_PCPU ||
(unlikely(!list_empty(&rt->rt6i_uncached)) && rt->dst.from)) (unlikely(!list_empty(&rt->rt6i_uncached)) && rt->from))
return rt6_dst_from_check(rt, cookie); return rt6_dst_from_check(rt, cookie);
else else
return rt6_check(rt, cookie); return rt6_check(rt, cookie);
...@@ -3055,11 +3055,11 @@ static void rt6_do_redirect(struct dst_entry *dst, struct sock *sk, struct sk_bu ...@@ -3055,11 +3055,11 @@ static void rt6_do_redirect(struct dst_entry *dst, struct sock *sk, struct sk_bu
static void rt6_set_from(struct rt6_info *rt, struct rt6_info *from) static void rt6_set_from(struct rt6_info *rt, struct rt6_info *from)
{ {
BUG_ON(from->dst.from); BUG_ON(from->from);
rt->rt6i_flags &= ~RTF_EXPIRES; rt->rt6i_flags &= ~RTF_EXPIRES;
dst_hold(&from->dst); dst_hold(&from->dst);
rt->dst.from = &from->dst; rt->from = from;
dst_init_metrics(&rt->dst, dst_metrics_ptr(&from->dst), true); dst_init_metrics(&rt->dst, dst_metrics_ptr(&from->dst), true);
} }
...@@ -4596,8 +4596,6 @@ static int __net_init ip6_route_net_init(struct net *net) ...@@ -4596,8 +4596,6 @@ static int __net_init ip6_route_net_init(struct net *net)
GFP_KERNEL); GFP_KERNEL);
if (!net->ipv6.ip6_null_entry) if (!net->ipv6.ip6_null_entry)
goto out_ip6_dst_entries; goto out_ip6_dst_entries;
net->ipv6.ip6_null_entry->dst.path =
(struct dst_entry *)net->ipv6.ip6_null_entry;
net->ipv6.ip6_null_entry->dst.ops = &net->ipv6.ip6_dst_ops; net->ipv6.ip6_null_entry->dst.ops = &net->ipv6.ip6_dst_ops;
dst_init_metrics(&net->ipv6.ip6_null_entry->dst, dst_init_metrics(&net->ipv6.ip6_null_entry->dst,
ip6_template_metrics, true); ip6_template_metrics, true);
...@@ -4609,8 +4607,6 @@ static int __net_init ip6_route_net_init(struct net *net) ...@@ -4609,8 +4607,6 @@ static int __net_init ip6_route_net_init(struct net *net)
GFP_KERNEL); GFP_KERNEL);
if (!net->ipv6.ip6_prohibit_entry) if (!net->ipv6.ip6_prohibit_entry)
goto out_ip6_null_entry; goto out_ip6_null_entry;
net->ipv6.ip6_prohibit_entry->dst.path =
(struct dst_entry *)net->ipv6.ip6_prohibit_entry;
net->ipv6.ip6_prohibit_entry->dst.ops = &net->ipv6.ip6_dst_ops; net->ipv6.ip6_prohibit_entry->dst.ops = &net->ipv6.ip6_dst_ops;
dst_init_metrics(&net->ipv6.ip6_prohibit_entry->dst, dst_init_metrics(&net->ipv6.ip6_prohibit_entry->dst,
ip6_template_metrics, true); ip6_template_metrics, true);
...@@ -4620,8 +4616,6 @@ static int __net_init ip6_route_net_init(struct net *net) ...@@ -4620,8 +4616,6 @@ static int __net_init ip6_route_net_init(struct net *net)
GFP_KERNEL); GFP_KERNEL);
if (!net->ipv6.ip6_blk_hole_entry) if (!net->ipv6.ip6_blk_hole_entry)
goto out_ip6_prohibit_entry; goto out_ip6_prohibit_entry;
net->ipv6.ip6_blk_hole_entry->dst.path =
(struct dst_entry *)net->ipv6.ip6_blk_hole_entry;
net->ipv6.ip6_blk_hole_entry->dst.ops = &net->ipv6.ip6_dst_ops; net->ipv6.ip6_blk_hole_entry->dst.ops = &net->ipv6.ip6_dst_ops;
dst_init_metrics(&net->ipv6.ip6_blk_hole_entry->dst, dst_init_metrics(&net->ipv6.ip6_blk_hole_entry->dst,
ip6_template_metrics, true); ip6_template_metrics, true);
......
...@@ -59,7 +59,7 @@ static int xfrm6_mode_tunnel_output(struct xfrm_state *x, struct sk_buff *skb) ...@@ -59,7 +59,7 @@ static int xfrm6_mode_tunnel_output(struct xfrm_state *x, struct sk_buff *skb)
if (x->props.flags & XFRM_STATE_NOECN) if (x->props.flags & XFRM_STATE_NOECN)
dsfield &= ~INET_ECN_MASK; dsfield &= ~INET_ECN_MASK;
ipv6_change_dsfield(top_iph, 0, dsfield); ipv6_change_dsfield(top_iph, 0, dsfield);
top_iph->hop_limit = ip6_dst_hoplimit(dst->child); top_iph->hop_limit = ip6_dst_hoplimit(xfrm_dst_child(dst));
top_iph->saddr = *(struct in6_addr *)&x->props.saddr; top_iph->saddr = *(struct in6_addr *)&x->props.saddr;
top_iph->daddr = *(struct in6_addr *)&x->id.daddr; top_iph->daddr = *(struct in6_addr *)&x->id.daddr;
return 0; return 0;
......
...@@ -265,7 +265,7 @@ static void xfrm6_dst_ifdown(struct dst_entry *dst, struct net_device *dev, ...@@ -265,7 +265,7 @@ static void xfrm6_dst_ifdown(struct dst_entry *dst, struct net_device *dev,
in6_dev_put(xdst->u.rt6.rt6i_idev); in6_dev_put(xdst->u.rt6.rt6i_idev);
xdst->u.rt6.rt6i_idev = loopback_idev; xdst->u.rt6.rt6i_idev = loopback_idev;
in6_dev_hold(loopback_idev); in6_dev_hold(loopback_idev);
xdst = (struct xfrm_dst *)xdst->u.dst.child; xdst = (struct xfrm_dst *)xfrm_dst_child(&xdst->u.dst);
} while (xdst->u.dst.xfrm); } while (xdst->u.dst.xfrm);
__in6_dev_put(loopback_idev); __in6_dev_put(loopback_idev);
......
...@@ -93,7 +93,8 @@ match_policy_out(const struct sk_buff *skb, const struct xt_policy_info *info, ...@@ -93,7 +93,8 @@ match_policy_out(const struct sk_buff *skb, const struct xt_policy_info *info,
if (dst->xfrm == NULL) if (dst->xfrm == NULL)
return -1; return -1;
for (i = 0; dst && dst->xfrm; dst = dst->child, i++) { for (i = 0; dst && dst->xfrm;
dst = ((struct xfrm_dst *)dst)->child, i++) {
pos = strict ? i : 0; pos = strict ? i : 0;
if (pos >= info->len) if (pos >= info->len)
return 0; return 0;
......
...@@ -120,8 +120,8 @@ bool xfrm_dev_offload_ok(struct sk_buff *skb, struct xfrm_state *x) ...@@ -120,8 +120,8 @@ bool xfrm_dev_offload_ok(struct sk_buff *skb, struct xfrm_state *x)
if (!x->type_offload || x->encap) if (!x->type_offload || x->encap)
return false; return false;
if ((x->xso.offload_handle && (dev == dst->path->dev)) && if ((x->xso.offload_handle && (dev == xfrm_dst_path(dst)->dev)) &&
!dst->child->xfrm && x->type->get_mtu) { !xdst->child->xfrm && x->type->get_mtu) {
mtu = x->type->get_mtu(x, xdst->child_mtu_cached); mtu = x->type->get_mtu(x, xdst->child_mtu_cached);
if (skb->len <= mtu) if (skb->len <= mtu)
......
...@@ -44,7 +44,7 @@ static int xfrm_skb_check_space(struct sk_buff *skb) ...@@ -44,7 +44,7 @@ static int xfrm_skb_check_space(struct sk_buff *skb)
static struct dst_entry *skb_dst_pop(struct sk_buff *skb) static struct dst_entry *skb_dst_pop(struct sk_buff *skb)
{ {
struct dst_entry *child = dst_clone(skb_dst(skb)->child); struct dst_entry *child = dst_clone(xfrm_dst_child(skb_dst(skb)));
skb_dst_drop(skb); skb_dst_drop(skb);
return child; return child;
......
...@@ -54,7 +54,7 @@ static struct xfrm_policy_afinfo const __rcu *xfrm_policy_afinfo[AF_INET6 + 1] ...@@ -54,7 +54,7 @@ static struct xfrm_policy_afinfo const __rcu *xfrm_policy_afinfo[AF_INET6 + 1]
static struct kmem_cache *xfrm_dst_cache __read_mostly; static struct kmem_cache *xfrm_dst_cache __read_mostly;
static __read_mostly seqcount_t xfrm_policy_hash_generation; static __read_mostly seqcount_t xfrm_policy_hash_generation;
static void xfrm_init_pmtu(struct dst_entry *dst); static void xfrm_init_pmtu(struct xfrm_dst **bundle, int nr);
static int stale_bundle(struct dst_entry *dst); static int stale_bundle(struct dst_entry *dst);
static int xfrm_bundle_ok(struct xfrm_dst *xdst); static int xfrm_bundle_ok(struct xfrm_dst *xdst);
static void xfrm_policy_queue_process(struct timer_list *t); static void xfrm_policy_queue_process(struct timer_list *t);
...@@ -1538,7 +1538,9 @@ static inline int xfrm_fill_dst(struct xfrm_dst *xdst, struct net_device *dev, ...@@ -1538,7 +1538,9 @@ static inline int xfrm_fill_dst(struct xfrm_dst *xdst, struct net_device *dev,
*/ */
static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy, static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy,
struct xfrm_state **xfrm, int nx, struct xfrm_state **xfrm,
struct xfrm_dst **bundle,
int nx,
const struct flowi *fl, const struct flowi *fl,
struct dst_entry *dst) struct dst_entry *dst)
{ {
...@@ -1546,8 +1548,8 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy, ...@@ -1546,8 +1548,8 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy,
unsigned long now = jiffies; unsigned long now = jiffies;
struct net_device *dev; struct net_device *dev;
struct xfrm_mode *inner_mode; struct xfrm_mode *inner_mode;
struct dst_entry *dst_prev = NULL; struct xfrm_dst *xdst_prev = NULL;
struct dst_entry *dst0 = NULL; struct xfrm_dst *xdst0 = NULL;
int i = 0; int i = 0;
int err; int err;
int header_len = 0; int header_len = 0;
...@@ -1573,13 +1575,14 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy, ...@@ -1573,13 +1575,14 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy,
goto put_states; goto put_states;
} }
if (!dst_prev) bundle[i] = xdst;
dst0 = dst1; if (!xdst_prev)
xdst0 = xdst;
else else
/* Ref count is taken during xfrm_alloc_dst() /* Ref count is taken during xfrm_alloc_dst()
* No need to do dst_clone() on dst1 * No need to do dst_clone() on dst1
*/ */
dst_prev->child = dst1; xfrm_dst_set_child(xdst_prev, &xdst->u.dst);
if (xfrm[i]->sel.family == AF_UNSPEC) { if (xfrm[i]->sel.family == AF_UNSPEC) {
inner_mode = xfrm_ip2inner_mode(xfrm[i], inner_mode = xfrm_ip2inner_mode(xfrm[i],
...@@ -1616,8 +1619,7 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy, ...@@ -1616,8 +1619,7 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy,
dst1->input = dst_discard; dst1->input = dst_discard;
dst1->output = inner_mode->afinfo->output; dst1->output = inner_mode->afinfo->output;
dst1->next = dst_prev; xdst_prev = xdst;
dst_prev = dst1;
header_len += xfrm[i]->props.header_len; header_len += xfrm[i]->props.header_len;
if (xfrm[i]->type->flags & XFRM_TYPE_NON_FRAGMENT) if (xfrm[i]->type->flags & XFRM_TYPE_NON_FRAGMENT)
...@@ -1625,40 +1627,39 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy, ...@@ -1625,40 +1627,39 @@ static struct dst_entry *xfrm_bundle_create(struct xfrm_policy *policy,
trailer_len += xfrm[i]->props.trailer_len; trailer_len += xfrm[i]->props.trailer_len;
} }
dst_prev->child = dst; xfrm_dst_set_child(xdst_prev, dst);
dst0->path = dst; xdst0->path = dst;
err = -ENODEV; err = -ENODEV;
dev = dst->dev; dev = dst->dev;
if (!dev) if (!dev)
goto free_dst; goto free_dst;
xfrm_init_path((struct xfrm_dst *)dst0, dst, nfheader_len); xfrm_init_path(xdst0, dst, nfheader_len);
xfrm_init_pmtu(dst_prev); xfrm_init_pmtu(bundle, nx);
for (dst_prev = dst0; dst_prev != dst; dst_prev = dst_prev->child) { for (xdst_prev = xdst0; xdst_prev != (struct xfrm_dst *)dst;
struct xfrm_dst *xdst = (struct xfrm_dst *)dst_prev; xdst_prev = (struct xfrm_dst *) xfrm_dst_child(&xdst_prev->u.dst)) {
err = xfrm_fill_dst(xdst_prev, dev, fl);
err = xfrm_fill_dst(xdst, dev, fl);
if (err) if (err)
goto free_dst; goto free_dst;
dst_prev->header_len = header_len; xdst_prev->u.dst.header_len = header_len;
dst_prev->trailer_len = trailer_len; xdst_prev->u.dst.trailer_len = trailer_len;
header_len -= xdst->u.dst.xfrm->props.header_len; header_len -= xdst_prev->u.dst.xfrm->props.header_len;
trailer_len -= xdst->u.dst.xfrm->props.trailer_len; trailer_len -= xdst_prev->u.dst.xfrm->props.trailer_len;
} }
out: out:
return dst0; return &xdst0->u.dst;
put_states: put_states:
for (; i < nx; i++) for (; i < nx; i++)
xfrm_state_put(xfrm[i]); xfrm_state_put(xfrm[i]);
free_dst: free_dst:
if (dst0) if (xdst0)
dst_release_immediate(dst0); dst_release_immediate(&xdst0->u.dst);
dst0 = ERR_PTR(err); xdst0 = ERR_PTR(err);
goto out; goto out;
} }
...@@ -1800,7 +1801,7 @@ static bool xfrm_xdst_can_reuse(struct xfrm_dst *xdst, ...@@ -1800,7 +1801,7 @@ static bool xfrm_xdst_can_reuse(struct xfrm_dst *xdst,
for (i = 0; i < num; i++) { for (i = 0; i < num; i++) {
if (!dst || dst->xfrm != xfrm[i]) if (!dst || dst->xfrm != xfrm[i])
return false; return false;
dst = dst->child; dst = xfrm_dst_child(dst);
} }
return xfrm_bundle_ok(xdst); return xfrm_bundle_ok(xdst);
...@@ -1813,6 +1814,7 @@ xfrm_resolve_and_create_bundle(struct xfrm_policy **pols, int num_pols, ...@@ -1813,6 +1814,7 @@ xfrm_resolve_and_create_bundle(struct xfrm_policy **pols, int num_pols,
{ {
struct net *net = xp_net(pols[0]); struct net *net = xp_net(pols[0]);
struct xfrm_state *xfrm[XFRM_MAX_DEPTH]; struct xfrm_state *xfrm[XFRM_MAX_DEPTH];
struct xfrm_dst *bundle[XFRM_MAX_DEPTH];
struct xfrm_dst *xdst, *old; struct xfrm_dst *xdst, *old;
struct dst_entry *dst; struct dst_entry *dst;
int err; int err;
...@@ -1840,7 +1842,7 @@ xfrm_resolve_and_create_bundle(struct xfrm_policy **pols, int num_pols, ...@@ -1840,7 +1842,7 @@ xfrm_resolve_and_create_bundle(struct xfrm_policy **pols, int num_pols,
old = xdst; old = xdst;
dst = xfrm_bundle_create(pols[0], xfrm, err, fl, dst_orig); dst = xfrm_bundle_create(pols[0], xfrm, bundle, err, fl, dst_orig);
if (IS_ERR(dst)) { if (IS_ERR(dst)) {
XFRM_INC_STATS(net, LINUX_MIB_XFRMOUTBUNDLEGENERROR); XFRM_INC_STATS(net, LINUX_MIB_XFRMOUTBUNDLEGENERROR);
return ERR_CAST(dst); return ERR_CAST(dst);
...@@ -1880,8 +1882,8 @@ static void xfrm_policy_queue_process(struct timer_list *t) ...@@ -1880,8 +1882,8 @@ static void xfrm_policy_queue_process(struct timer_list *t)
xfrm_decode_session(skb, &fl, dst->ops->family); xfrm_decode_session(skb, &fl, dst->ops->family);
spin_unlock(&pq->hold_queue.lock); spin_unlock(&pq->hold_queue.lock);
dst_hold(dst->path); dst_hold(xfrm_dst_path(dst));
dst = xfrm_lookup(net, dst->path, &fl, sk, 0); dst = xfrm_lookup(net, xfrm_dst_path(dst), &fl, sk, 0);
if (IS_ERR(dst)) if (IS_ERR(dst))
goto purge_queue; goto purge_queue;
...@@ -1910,8 +1912,8 @@ static void xfrm_policy_queue_process(struct timer_list *t) ...@@ -1910,8 +1912,8 @@ static void xfrm_policy_queue_process(struct timer_list *t)
skb = __skb_dequeue(&list); skb = __skb_dequeue(&list);
xfrm_decode_session(skb, &fl, skb_dst(skb)->ops->family); xfrm_decode_session(skb, &fl, skb_dst(skb)->ops->family);
dst_hold(skb_dst(skb)->path); dst_hold(xfrm_dst_path(skb_dst(skb)));
dst = xfrm_lookup(net, skb_dst(skb)->path, &fl, skb->sk, 0); dst = xfrm_lookup(net, xfrm_dst_path(skb_dst(skb)), &fl, skb->sk, 0);
if (IS_ERR(dst)) { if (IS_ERR(dst)) {
kfree_skb(skb); kfree_skb(skb);
continue; continue;
...@@ -2012,8 +2014,8 @@ static struct xfrm_dst *xfrm_create_dummy_bundle(struct net *net, ...@@ -2012,8 +2014,8 @@ static struct xfrm_dst *xfrm_create_dummy_bundle(struct net *net,
dst1->output = xdst_queue_output; dst1->output = xdst_queue_output;
dst_hold(dst); dst_hold(dst);
dst1->child = dst; xfrm_dst_set_child(xdst, dst);
dst1->path = dst; xdst->path = dst;
xfrm_init_path((struct xfrm_dst *)dst1, dst, 0); xfrm_init_path((struct xfrm_dst *)dst1, dst, 0);
...@@ -2576,7 +2578,7 @@ static int stale_bundle(struct dst_entry *dst) ...@@ -2576,7 +2578,7 @@ static int stale_bundle(struct dst_entry *dst)
void xfrm_dst_ifdown(struct dst_entry *dst, struct net_device *dev) void xfrm_dst_ifdown(struct dst_entry *dst, struct net_device *dev)
{ {
while ((dst = dst->child) && dst->xfrm && dst->dev == dev) { while ((dst = xfrm_dst_child(dst)) && dst->xfrm && dst->dev == dev) {
dst->dev = dev_net(dev)->loopback_dev; dst->dev = dev_net(dev)->loopback_dev;
dev_hold(dst->dev); dev_hold(dst->dev);
dev_put(dev); dev_put(dev);
...@@ -2600,13 +2602,15 @@ static struct dst_entry *xfrm_negative_advice(struct dst_entry *dst) ...@@ -2600,13 +2602,15 @@ static struct dst_entry *xfrm_negative_advice(struct dst_entry *dst)
return dst; return dst;
} }
static void xfrm_init_pmtu(struct dst_entry *dst) static void xfrm_init_pmtu(struct xfrm_dst **bundle, int nr)
{ {
do { while (nr--) {
struct xfrm_dst *xdst = (struct xfrm_dst *)dst; struct xfrm_dst *xdst = bundle[nr];
u32 pmtu, route_mtu_cached; u32 pmtu, route_mtu_cached;
struct dst_entry *dst;
pmtu = dst_mtu(dst->child); dst = &xdst->u.dst;
pmtu = dst_mtu(xfrm_dst_child(dst));
xdst->child_mtu_cached = pmtu; xdst->child_mtu_cached = pmtu;
pmtu = xfrm_state_mtu(dst->xfrm, pmtu); pmtu = xfrm_state_mtu(dst->xfrm, pmtu);
...@@ -2618,7 +2622,7 @@ static void xfrm_init_pmtu(struct dst_entry *dst) ...@@ -2618,7 +2622,7 @@ static void xfrm_init_pmtu(struct dst_entry *dst)
pmtu = route_mtu_cached; pmtu = route_mtu_cached;
dst_metric_set(dst, RTAX_MTU, pmtu); dst_metric_set(dst, RTAX_MTU, pmtu);
} while ((dst = dst->next)); }
} }
/* Check that the bundle accepts the flow and its components are /* Check that the bundle accepts the flow and its components are
...@@ -2627,19 +2631,20 @@ static void xfrm_init_pmtu(struct dst_entry *dst) ...@@ -2627,19 +2631,20 @@ static void xfrm_init_pmtu(struct dst_entry *dst)
static int xfrm_bundle_ok(struct xfrm_dst *first) static int xfrm_bundle_ok(struct xfrm_dst *first)
{ {
struct xfrm_dst *bundle[XFRM_MAX_DEPTH];
struct dst_entry *dst = &first->u.dst; struct dst_entry *dst = &first->u.dst;
struct xfrm_dst *last; struct xfrm_dst *xdst;
int start_from, nr;
u32 mtu; u32 mtu;
if (!dst_check(dst->path, ((struct xfrm_dst *)dst)->path_cookie) || if (!dst_check(xfrm_dst_path(dst), ((struct xfrm_dst *)dst)->path_cookie) ||
(dst->dev && !netif_running(dst->dev))) (dst->dev && !netif_running(dst->dev)))
return 0; return 0;
if (dst->flags & DST_XFRM_QUEUE) if (dst->flags & DST_XFRM_QUEUE)
return 1; return 1;
last = NULL; start_from = nr = 0;
do { do {
struct xfrm_dst *xdst = (struct xfrm_dst *)dst; struct xfrm_dst *xdst = (struct xfrm_dst *)dst;
...@@ -2651,9 +2656,11 @@ static int xfrm_bundle_ok(struct xfrm_dst *first) ...@@ -2651,9 +2656,11 @@ static int xfrm_bundle_ok(struct xfrm_dst *first)
xdst->policy_genid != atomic_read(&xdst->pols[0]->genid)) xdst->policy_genid != atomic_read(&xdst->pols[0]->genid))
return 0; return 0;
mtu = dst_mtu(dst->child); bundle[nr++] = xdst;
mtu = dst_mtu(xfrm_dst_child(dst));
if (xdst->child_mtu_cached != mtu) { if (xdst->child_mtu_cached != mtu) {
last = xdst; start_from = nr;
xdst->child_mtu_cached = mtu; xdst->child_mtu_cached = mtu;
} }
...@@ -2661,30 +2668,30 @@ static int xfrm_bundle_ok(struct xfrm_dst *first) ...@@ -2661,30 +2668,30 @@ static int xfrm_bundle_ok(struct xfrm_dst *first)
return 0; return 0;
mtu = dst_mtu(xdst->route); mtu = dst_mtu(xdst->route);
if (xdst->route_mtu_cached != mtu) { if (xdst->route_mtu_cached != mtu) {
last = xdst; start_from = nr;
xdst->route_mtu_cached = mtu; xdst->route_mtu_cached = mtu;
} }
dst = dst->child; dst = xfrm_dst_child(dst);
} while (dst->xfrm); } while (dst->xfrm);
if (likely(!last)) if (likely(!start_from))
return 1; return 1;
mtu = last->child_mtu_cached; xdst = bundle[start_from - 1];
for (;;) { mtu = xdst->child_mtu_cached;
dst = &last->u.dst; while (start_from--) {
dst = &xdst->u.dst;
mtu = xfrm_state_mtu(dst->xfrm, mtu); mtu = xfrm_state_mtu(dst->xfrm, mtu);
if (mtu > last->route_mtu_cached) if (mtu > xdst->route_mtu_cached)
mtu = last->route_mtu_cached; mtu = xdst->route_mtu_cached;
dst_metric_set(dst, RTAX_MTU, mtu); dst_metric_set(dst, RTAX_MTU, mtu);
if (!start_from)
if (last == first)
break; break;
last = (struct xfrm_dst *)last->u.dst.next; xdst = bundle[start_from - 1];
last->child_mtu_cached = mtu; xdst->child_mtu_cached = mtu;
} }
return 1; return 1;
...@@ -2692,22 +2699,20 @@ static int xfrm_bundle_ok(struct xfrm_dst *first) ...@@ -2692,22 +2699,20 @@ static int xfrm_bundle_ok(struct xfrm_dst *first)
static unsigned int xfrm_default_advmss(const struct dst_entry *dst) static unsigned int xfrm_default_advmss(const struct dst_entry *dst)
{ {
return dst_metric_advmss(dst->path); return dst_metric_advmss(xfrm_dst_path(dst));
} }
static unsigned int xfrm_mtu(const struct dst_entry *dst) static unsigned int xfrm_mtu(const struct dst_entry *dst)
{ {
unsigned int mtu = dst_metric_raw(dst, RTAX_MTU); unsigned int mtu = dst_metric_raw(dst, RTAX_MTU);
return mtu ? : dst_mtu(dst->path); return mtu ? : dst_mtu(xfrm_dst_path(dst));
} }
static const void *xfrm_get_dst_nexthop(const struct dst_entry *dst, static const void *xfrm_get_dst_nexthop(const struct dst_entry *dst,
const void *daddr) const void *daddr)
{ {
const struct dst_entry *path = dst->path; while (dst->xfrm) {
for (; dst != path; dst = dst->child) {
const struct xfrm_state *xfrm = dst->xfrm; const struct xfrm_state *xfrm = dst->xfrm;
if (xfrm->props.mode == XFRM_MODE_TRANSPORT) if (xfrm->props.mode == XFRM_MODE_TRANSPORT)
...@@ -2716,6 +2721,8 @@ static const void *xfrm_get_dst_nexthop(const struct dst_entry *dst, ...@@ -2716,6 +2721,8 @@ static const void *xfrm_get_dst_nexthop(const struct dst_entry *dst,
daddr = xfrm->coaddr; daddr = xfrm->coaddr;
else if (!(xfrm->type->flags & XFRM_TYPE_LOCAL_COADDR)) else if (!(xfrm->type->flags & XFRM_TYPE_LOCAL_COADDR))
daddr = &xfrm->id.daddr; daddr = &xfrm->id.daddr;
dst = xfrm_dst_child(dst);
} }
return daddr; return daddr;
} }
...@@ -2724,7 +2731,7 @@ static struct neighbour *xfrm_neigh_lookup(const struct dst_entry *dst, ...@@ -2724,7 +2731,7 @@ static struct neighbour *xfrm_neigh_lookup(const struct dst_entry *dst,
struct sk_buff *skb, struct sk_buff *skb,
const void *daddr) const void *daddr)
{ {
const struct dst_entry *path = dst->path; const struct dst_entry *path = xfrm_dst_path(dst);
if (!skb) if (!skb)
daddr = xfrm_get_dst_nexthop(dst, daddr); daddr = xfrm_get_dst_nexthop(dst, daddr);
...@@ -2733,7 +2740,7 @@ static struct neighbour *xfrm_neigh_lookup(const struct dst_entry *dst, ...@@ -2733,7 +2740,7 @@ static struct neighbour *xfrm_neigh_lookup(const struct dst_entry *dst,
static void xfrm_confirm_neigh(const struct dst_entry *dst, const void *daddr) static void xfrm_confirm_neigh(const struct dst_entry *dst, const void *daddr)
{ {
const struct dst_entry *path = dst->path; const struct dst_entry *path = xfrm_dst_path(dst);
daddr = xfrm_get_dst_nexthop(dst, daddr); daddr = xfrm_get_dst_nexthop(dst, daddr);
path->ops->confirm_neigh(path, daddr); path->ops->confirm_neigh(path, daddr);
......
...@@ -452,7 +452,7 @@ int selinux_xfrm_postroute_last(u32 sk_sid, struct sk_buff *skb, ...@@ -452,7 +452,7 @@ int selinux_xfrm_postroute_last(u32 sk_sid, struct sk_buff *skb,
if (dst) { if (dst) {
struct dst_entry *iter; struct dst_entry *iter;
for (iter = dst; iter != NULL; iter = iter->child) { for (iter = dst; iter != NULL; iter = xfrm_dst_child(iter)) {
struct xfrm_state *x = iter->xfrm; struct xfrm_state *x = iter->xfrm;
if (x && selinux_authorizable_xfrm(x)) if (x && selinux_authorizable_xfrm(x))
......
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