netfilter: nf_conntrack: use atomic64 for accounting counters
authorEric Dumazet <eric.dumazet@gmail.com>
Wed, 14 Dec 2011 13:45:20 +0000 (14:45 +0100)
committerPablo Neira Ayuso <pablo@netfilter.org>
Sun, 18 Dec 2011 00:19:19 +0000 (01:19 +0100)
We can use atomic64_t infrastructure to avoid taking a spinlock in fast
path, and remove inaccuracies while reading values in
ctnetlink_dump_counters() and connbytes_mt() on 32bit arches.

Suggested by Pablo.

Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
include/net/netfilter/nf_conntrack_acct.h
net/netfilter/nf_conntrack_acct.c
net/netfilter/nf_conntrack_core.c
net/netfilter/nf_conntrack_netlink.c
net/netfilter/xt_connbytes.c

index 4e9c63a20db22dd498933745ed9cf2627e74fd47..463ae8e166965908d2fc0a60e92296123b846f2e 100644 (file)
@@ -15,8 +15,8 @@
 #include <net/netfilter/nf_conntrack_extend.h>
 
 struct nf_conn_counter {
-       u_int64_t packets;
-       u_int64_t bytes;
+       atomic64_t packets;
+       atomic64_t bytes;
 };
 
 static inline
index 369df3f08d42635da9245558eca0cd950d1417f5..93329067a6a2e1726268ccfd6917d33cdecc989f 100644 (file)
@@ -46,8 +46,8 @@ seq_print_acct(struct seq_file *s, const struct nf_conn *ct, int dir)
                return 0;
 
        return seq_printf(s, "packets=%llu bytes=%llu ",
-                         (unsigned long long)acct[dir].packets,
-                         (unsigned long long)acct[dir].bytes);
+                         (unsigned long long)atomic64_read(&acct[dir].packets),
+                         (unsigned long long)atomic64_read(&acct[dir].bytes));
 };
 EXPORT_SYMBOL_GPL(seq_print_acct);
 
index 7202b0631cd6eb725debc17062a0d1f4a1087f41..8b2842e321fb9e77ce51f385cb3d543a9b4554c3 100644 (file)
@@ -1044,10 +1044,8 @@ acct:
 
                acct = nf_conn_acct_find(ct);
                if (acct) {
-                       spin_lock_bh(&ct->lock);
-                       acct[CTINFO2DIR(ctinfo)].packets++;
-                       acct[CTINFO2DIR(ctinfo)].bytes += skb->len;
-                       spin_unlock_bh(&ct->lock);
+                       atomic64_inc(&acct[CTINFO2DIR(ctinfo)].packets);
+                       atomic64_add(skb->len, &acct[CTINFO2DIR(ctinfo)].bytes);
                }
        }
 }
@@ -1063,11 +1061,9 @@ bool __nf_ct_kill_acct(struct nf_conn *ct,
 
                acct = nf_conn_acct_find(ct);
                if (acct) {
-                       spin_lock_bh(&ct->lock);
-                       acct[CTINFO2DIR(ctinfo)].packets++;
-                       acct[CTINFO2DIR(ctinfo)].bytes +=
-                               skb->len - skb_network_offset(skb);
-                       spin_unlock_bh(&ct->lock);
+                       atomic64_inc(&acct[CTINFO2DIR(ctinfo)].packets);
+                       atomic64_add(skb->len - skb_network_offset(skb),
+                                    &acct[CTINFO2DIR(ctinfo)].bytes);
                }
        }
 
index ef21b221f0363a8700fbbc3aadd316423c4f9d66..a36e6553ddb323149ebcf7a5e61d6e6a840f61e8 100644 (file)
@@ -219,9 +219,9 @@ ctnetlink_dump_counters(struct sk_buff *skb, const struct nf_conn *ct,
                goto nla_put_failure;
 
        NLA_PUT_BE64(skb, CTA_COUNTERS_PACKETS,
-                    cpu_to_be64(acct[dir].packets));
+                    cpu_to_be64(atomic64_read(&acct[dir].packets)));
        NLA_PUT_BE64(skb, CTA_COUNTERS_BYTES,
-                    cpu_to_be64(acct[dir].bytes));
+                    cpu_to_be64(atomic64_read(&acct[dir].bytes)));
 
        nla_nest_end(skb, nest_count);
 
@@ -720,8 +720,12 @@ restart:
                                struct nf_conn_counter *acct;
 
                                acct = nf_conn_acct_find(ct);
-                               if (acct)
-                                       memset(acct, 0, sizeof(struct nf_conn_counter[IP_CT_DIR_MAX]));
+                               if (acct) {
+                                       atomic64_set(&acct[IP_CT_DIR_ORIGINAL].bytes, 0);
+                                       atomic64_set(&acct[IP_CT_DIR_ORIGINAL].packets, 0);
+                                       atomic64_set(&acct[IP_CT_DIR_REPLY].bytes, 0);
+                                       atomic64_set(&acct[IP_CT_DIR_REPLY].packets, 0);
+                                       }
                        }
                }
                if (cb->args[1]) {
index 5b138506690ec578105911ffb64309b5297d6458..2b8418c257c9c556bf3ed38135c2c8cefb2a642a 100644 (file)
@@ -40,46 +40,46 @@ connbytes_mt(const struct sk_buff *skb, struct xt_action_param *par)
        case XT_CONNBYTES_PKTS:
                switch (sinfo->direction) {
                case XT_CONNBYTES_DIR_ORIGINAL:
-                       what = counters[IP_CT_DIR_ORIGINAL].packets;
+                       what = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].packets);
                        break;
                case XT_CONNBYTES_DIR_REPLY:
-                       what = counters[IP_CT_DIR_REPLY].packets;
+                       what = atomic64_read(&counters[IP_CT_DIR_REPLY].packets);
                        break;
                case XT_CONNBYTES_DIR_BOTH:
-                       what = counters[IP_CT_DIR_ORIGINAL].packets;
-                       what += counters[IP_CT_DIR_REPLY].packets;
+                       what = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].packets);
+                       what += atomic64_read(&counters[IP_CT_DIR_REPLY].packets);
                        break;
                }
                break;
        case XT_CONNBYTES_BYTES:
                switch (sinfo->direction) {
                case XT_CONNBYTES_DIR_ORIGINAL:
-                       what = counters[IP_CT_DIR_ORIGINAL].bytes;
+                       what = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].bytes);
                        break;
                case XT_CONNBYTES_DIR_REPLY:
-                       what = counters[IP_CT_DIR_REPLY].bytes;
+                       what = atomic64_read(&counters[IP_CT_DIR_REPLY].bytes);
                        break;
                case XT_CONNBYTES_DIR_BOTH:
-                       what = counters[IP_CT_DIR_ORIGINAL].bytes;
-                       what += counters[IP_CT_DIR_REPLY].bytes;
+                       what = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].bytes);
+                       what += atomic64_read(&counters[IP_CT_DIR_REPLY].bytes);
                        break;
                }
                break;
        case XT_CONNBYTES_AVGPKT:
                switch (sinfo->direction) {
                case XT_CONNBYTES_DIR_ORIGINAL:
-                       bytes = counters[IP_CT_DIR_ORIGINAL].bytes;
-                       pkts  = counters[IP_CT_DIR_ORIGINAL].packets;
+                       bytes = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].bytes);
+                       pkts  = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].packets);
                        break;
                case XT_CONNBYTES_DIR_REPLY:
-                       bytes = counters[IP_CT_DIR_REPLY].bytes;
-                       pkts  = counters[IP_CT_DIR_REPLY].packets;
+                       bytes = atomic64_read(&counters[IP_CT_DIR_REPLY].bytes);
+                       pkts  = atomic64_read(&counters[IP_CT_DIR_REPLY].packets);
                        break;
                case XT_CONNBYTES_DIR_BOTH:
-                       bytes = counters[IP_CT_DIR_ORIGINAL].bytes +
-                               counters[IP_CT_DIR_REPLY].bytes;
-                       pkts  = counters[IP_CT_DIR_ORIGINAL].packets +
-                               counters[IP_CT_DIR_REPLY].packets;
+                       bytes = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].bytes) +
+                               atomic64_read(&counters[IP_CT_DIR_REPLY].bytes);
+                       pkts  = atomic64_read(&counters[IP_CT_DIR_ORIGINAL].packets) +
+                               atomic64_read(&counters[IP_CT_DIR_REPLY].packets);
                        break;
                }
                if (pkts != 0)