netfilter: conntrack: restart iteration on resize
authorFlorian Westphal <fw@strlen.de>
Sun, 21 May 2017 10:52:58 +0000 (12:52 +0200)
committerPablo Neira Ayuso <pablo@netfilter.org>
Mon, 29 May 2017 10:46:11 +0000 (12:46 +0200)
We could some conntracks when a resize occurs in parallel.

Avoid this by sampling generation seqcnt and doing a restart if needed.

Signed-off-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
net/netfilter/nf_conntrack_core.c

index 7ecee79c78b84892caa213aa92ea12988c11ac55..c3bd9b086dcc9d60e62e232c078bd915b133bb8c 100644 (file)
@@ -1623,17 +1623,25 @@ found:
 static void nf_ct_iterate_cleanup(int (*iter)(struct nf_conn *i, void *data),
                                  void *data, u32 portid, int report)
 {
+       unsigned int bucket = 0, sequence;
        struct nf_conn *ct;
-       unsigned int bucket = 0;
 
        might_sleep();
 
-       while ((ct = get_next_corpse(iter, data, &bucket)) != NULL) {
-               /* Time to push up daises... */
+       for (;;) {
+               sequence = read_seqcount_begin(&nf_conntrack_generation);
 
-               nf_ct_delete(ct, portid, report);
-               nf_ct_put(ct);
-               cond_resched();
+               while ((ct = get_next_corpse(iter, data, &bucket)) != NULL) {
+                       /* Time to push up daises... */
+
+                       nf_ct_delete(ct, portid, report);
+                       nf_ct_put(ct);
+                       cond_resched();
+               }
+
+               if (!read_seqcount_retry(&nf_conntrack_generation, sequence))
+                       break;
+               bucket = 0;
        }
 }