提交 0da4af00 编写于 作者: W Wei Wang 提交者: David S. Miller

ipv6: only update __use and lastusetime once per jiffy at most

In order to not dirty the cacheline too often, we try to only update
dst->__use and dst->lastusetime at most once per jiffy.
As dst->lastusetime is only used by ipv6 garbage collector, it should
be good enough time resolution.
And __use is only used in ipv6_route_seq_show() to show how many times a
dst has been used. And as __use is not atomic_t right now, it does not
show the precise number of usage times anyway. So we think it should be
OK to only update it at most once per jiffy.

According to my latest syn flood test on a machine with intel Xeon 6th
gen processor and 2 10G mlx nics bonded together, each with 8 rx queues
on 2 NUMA nodes:
With this patch, the packet process rate increases from ~3.49Mpps to
~3.75Mpps with a 7% increase rate.

Note: dst_use() is being renamed to dst_hold_and_use() to better specify
the purpose of the function.
Signed-off-by: NWei Wang <weiwan@google.com>
Acked-by: NEric Dumazet <edumazet@googl.com>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 0e80193b
...@@ -255,17 +255,18 @@ static inline void dst_hold(struct dst_entry *dst) ...@@ -255,17 +255,18 @@ static inline void dst_hold(struct dst_entry *dst)
WARN_ON(atomic_inc_not_zero(&dst->__refcnt) == 0); WARN_ON(atomic_inc_not_zero(&dst->__refcnt) == 0);
} }
static inline void dst_use(struct dst_entry *dst, unsigned long time) static inline void dst_use_noref(struct dst_entry *dst, unsigned long time)
{ {
dst_hold(dst); if (time != dst->lastuse) {
dst->__use++; dst->__use++;
dst->lastuse = time; dst->lastuse = time;
}
} }
static inline void dst_use_noref(struct dst_entry *dst, unsigned long time) static inline void dst_hold_and_use(struct dst_entry *dst, unsigned long time)
{ {
dst->__use++; dst_hold(dst);
dst->lastuse = time; dst_use_noref(dst, time);
} }
static inline struct dst_entry *dst_clone(struct dst_entry *dst) static inline struct dst_entry *dst_clone(struct dst_entry *dst)
......
...@@ -338,7 +338,7 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou ...@@ -338,7 +338,7 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou
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);
dst_use(&rth->dst, now); dst_hold_and_use(&rth->dst, now);
spin_unlock_bh(&dn_rt_hash_table[hash].lock); spin_unlock_bh(&dn_rt_hash_table[hash].lock);
dst_release_immediate(&rt->dst); dst_release_immediate(&rt->dst);
...@@ -351,7 +351,7 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou ...@@ -351,7 +351,7 @@ static int dn_insert_route(struct dn_route *rt, unsigned int hash, struct dn_rou
rcu_assign_pointer(rt->dst.dn_next, dn_rt_hash_table[hash].chain); rcu_assign_pointer(rt->dst.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_use(&rt->dst, now); dst_hold_and_use(&rt->dst, now);
spin_unlock_bh(&dn_rt_hash_table[hash].lock); spin_unlock_bh(&dn_rt_hash_table[hash].lock);
*rp = rt; *rp = rt;
return 0; return 0;
...@@ -1258,7 +1258,7 @@ static int __dn_route_output_key(struct dst_entry **pprt, const struct flowidn * ...@@ -1258,7 +1258,7 @@ static int __dn_route_output_key(struct dst_entry **pprt, const struct flowidn *
(flp->flowidn_mark == rt->fld.flowidn_mark) && (flp->flowidn_mark == rt->fld.flowidn_mark) &&
dn_is_output_route(rt) && dn_is_output_route(rt) &&
(rt->fld.flowidn_oif == flp->flowidn_oif)) { (rt->fld.flowidn_oif == flp->flowidn_oif)) {
dst_use(&rt->dst, jiffies); dst_hold_and_use(&rt->dst, jiffies);
rcu_read_unlock_bh(); rcu_read_unlock_bh();
*pprt = &rt->dst; *pprt = &rt->dst;
return 0; return 0;
...@@ -1535,7 +1535,7 @@ static int dn_route_input(struct sk_buff *skb) ...@@ -1535,7 +1535,7 @@ static int dn_route_input(struct sk_buff *skb)
(rt->fld.flowidn_oif == 0) && (rt->fld.flowidn_oif == 0) &&
(rt->fld.flowidn_mark == skb->mark) && (rt->fld.flowidn_mark == skb->mark) &&
(rt->fld.flowidn_iif == cb->iif)) { (rt->fld.flowidn_iif == cb->iif)) {
dst_use(&rt->dst, jiffies); dst_hold_and_use(&rt->dst, jiffies);
rcu_read_unlock(); rcu_read_unlock();
skb_dst_set(skb, (struct dst_entry *)rt); skb_dst_set(skb, (struct dst_entry *)rt);
return 0; return 0;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册