aboutsummaryrefslogtreecommitdiffstats
path: root/net/ipv4
diff options
context:
space:
mode:
authorEric Dumazet <eric.dumazet@gmail.com>2010-10-20 18:03:38 -0400
committerDavid S. Miller <davem@davemloft.net>2010-10-21 06:09:41 -0400
commit9b0c290e78d667e6a483bde8c7cef7dd15f49017 (patch)
treeed68f6cbcc9a512106217694b16a846147cdf54a /net/ipv4
parent7b5edbc4cfe2297b0915adea5aa1eafcafadbf06 (diff)
fib: introduce fib_alias_accessed() helper
Perf tools session at NFWS 2010 pointed out a false sharing on struct fib_alias that can be avoided pretty easily, if we set FA_S_ACCESSED bit only if needed (ie : not already set) Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4')
-rw-r--r--net/ipv4/fib_hash.c3
-rw-r--r--net/ipv4/fib_lookup.h7
-rw-r--r--net/ipv4/fib_semantics.c2
-rw-r--r--net/ipv4/fib_trie.c3
4 files changed, 12 insertions, 3 deletions
diff --git a/net/ipv4/fib_hash.c b/net/ipv4/fib_hash.c
index 4f1aafd3ba89..43e1c594ce8f 100644
--- a/net/ipv4/fib_hash.c
+++ b/net/ipv4/fib_hash.c
@@ -335,7 +335,8 @@ void fib_table_select_default(struct fib_table *tb,
335 if (!next_fi->fib_nh[0].nh_gw || 335 if (!next_fi->fib_nh[0].nh_gw ||
336 next_fi->fib_nh[0].nh_scope != RT_SCOPE_LINK) 336 next_fi->fib_nh[0].nh_scope != RT_SCOPE_LINK)
337 continue; 337 continue;
338 fa->fa_state |= FA_S_ACCESSED; 338
339 fib_alias_accessed(fa);
339 340
340 if (fi == NULL) { 341 if (fi == NULL) {
341 if (next_fi != res->fi) 342 if (next_fi != res->fi)
diff --git a/net/ipv4/fib_lookup.h b/net/ipv4/fib_lookup.h
index 5072d8effd5d..a29edf2219c8 100644
--- a/net/ipv4/fib_lookup.h
+++ b/net/ipv4/fib_lookup.h
@@ -17,6 +17,13 @@ struct fib_alias {
17 17
18#define FA_S_ACCESSED 0x01 18#define FA_S_ACCESSED 0x01
19 19
20/* Dont write on fa_state unless needed, to keep it shared on all cpus */
21static inline void fib_alias_accessed(struct fib_alias *fa)
22{
23 if (!(fa->fa_state & FA_S_ACCESSED))
24 fa->fa_state |= FA_S_ACCESSED;
25}
26
20/* Exported by fib_semantics.c */ 27/* Exported by fib_semantics.c */
21extern int fib_semantic_match(struct list_head *head, 28extern int fib_semantic_match(struct list_head *head,
22 const struct flowi *flp, 29 const struct flowi *flp,
diff --git a/net/ipv4/fib_semantics.c b/net/ipv4/fib_semantics.c
index 6734c9cab248..3e0da3ef6116 100644
--- a/net/ipv4/fib_semantics.c
+++ b/net/ipv4/fib_semantics.c
@@ -901,7 +901,7 @@ int fib_semantic_match(struct list_head *head, const struct flowi *flp,
901 if (fa->fa_scope < flp->fl4_scope) 901 if (fa->fa_scope < flp->fl4_scope)
902 continue; 902 continue;
903 903
904 fa->fa_state |= FA_S_ACCESSED; 904 fib_alias_accessed(fa);
905 905
906 err = fib_props[fa->fa_type].error; 906 err = fib_props[fa->fa_type].error;
907 if (err == 0) { 907 if (err == 0) {
diff --git a/net/ipv4/fib_trie.c b/net/ipv4/fib_trie.c
index 31494f335686..cd5e13aee7d5 100644
--- a/net/ipv4/fib_trie.c
+++ b/net/ipv4/fib_trie.c
@@ -1838,7 +1838,8 @@ void fib_table_select_default(struct fib_table *tb,
1838 if (!next_fi->fib_nh[0].nh_gw || 1838 if (!next_fi->fib_nh[0].nh_gw ||
1839 next_fi->fib_nh[0].nh_scope != RT_SCOPE_LINK) 1839 next_fi->fib_nh[0].nh_scope != RT_SCOPE_LINK)
1840 continue; 1840 continue;
1841 fa->fa_state |= FA_S_ACCESSED; 1841
1842 fib_alias_accessed(fa);
1842 1843
1843 if (fi == NULL) { 1844 if (fi == NULL) {
1844 if (next_fi != res->fi) 1845 if (next_fi != res->fi)