diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2010-10-20 18:03:38 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2010-10-21 06:09:41 -0400 |
commit | 9b0c290e78d667e6a483bde8c7cef7dd15f49017 (patch) | |
tree | ed68f6cbcc9a512106217694b16a846147cdf54a /net | |
parent | 7b5edbc4cfe2297b0915adea5aa1eafcafadbf06 (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')
-rw-r--r-- | net/ipv4/fib_hash.c | 3 | ||||
-rw-r--r-- | net/ipv4/fib_lookup.h | 7 | ||||
-rw-r--r-- | net/ipv4/fib_semantics.c | 2 | ||||
-rw-r--r-- | net/ipv4/fib_trie.c | 3 |
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 */ | ||
21 | static 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 */ |
21 | extern int fib_semantic_match(struct list_head *head, | 28 | extern 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) |