| /* |
| * inet fragments management |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License |
| * as published by the Free Software Foundation; either version |
| * 2 of the License, or (at your option) any later version. |
| * |
| * Authors: Pavel Emelyanov <xemul@openvz.org> |
| * Started as consolidation of ipv4/ip_fragment.c, |
| * ipv6/reassembly. and ipv6 nf conntrack reassembly |
| */ |
| |
| #include <linux/list.h> |
| #include <linux/spinlock.h> |
| #include <linux/module.h> |
| #include <linux/timer.h> |
| #include <linux/mm.h> |
| #include <linux/random.h> |
| |
| #include <net/inet_frag.h> |
| |
| static void inet_frag_secret_rebuild(unsigned long dummy) |
| { |
| struct inet_frags *f = (struct inet_frags *)dummy; |
| unsigned long now = jiffies; |
| int i; |
| |
| write_lock(&f->lock); |
| get_random_bytes(&f->rnd, sizeof(u32)); |
| for (i = 0; i < INETFRAGS_HASHSZ; i++) { |
| struct inet_frag_queue *q; |
| struct hlist_node *p, *n; |
| |
| hlist_for_each_entry_safe(q, p, n, &f->hash[i], list) { |
| unsigned int hval = f->hashfn(q); |
| |
| if (hval != i) { |
| hlist_del(&q->list); |
| |
| /* Relink to new hash chain. */ |
| hlist_add_head(&q->list, &f->hash[hval]); |
| } |
| } |
| } |
| write_unlock(&f->lock); |
| |
| mod_timer(&f->secret_timer, now + f->ctl->secret_interval); |
| } |
| |
| void inet_frags_init(struct inet_frags *f) |
| { |
| int i; |
| |
| for (i = 0; i < INETFRAGS_HASHSZ; i++) |
| INIT_HLIST_HEAD(&f->hash[i]); |
| |
| INIT_LIST_HEAD(&f->lru_list); |
| rwlock_init(&f->lock); |
| |
| f->rnd = (u32) ((num_physpages ^ (num_physpages>>7)) ^ |
| (jiffies ^ (jiffies >> 6))); |
| |
| f->nqueues = 0; |
| atomic_set(&f->mem, 0); |
| |
| init_timer(&f->secret_timer); |
| f->secret_timer.function = inet_frag_secret_rebuild; |
| f->secret_timer.data = (unsigned long)f; |
| f->secret_timer.expires = jiffies + f->ctl->secret_interval; |
| add_timer(&f->secret_timer); |
| } |
| EXPORT_SYMBOL(inet_frags_init); |
| |
| void inet_frags_fini(struct inet_frags *f) |
| { |
| del_timer(&f->secret_timer); |
| } |
| EXPORT_SYMBOL(inet_frags_fini); |
| |
| static inline void fq_unlink(struct inet_frag_queue *fq, struct inet_frags *f) |
| { |
| write_lock(&f->lock); |
| hlist_del(&fq->list); |
| list_del(&fq->lru_list); |
| f->nqueues--; |
| write_unlock(&f->lock); |
| } |
| |
| void inet_frag_kill(struct inet_frag_queue *fq, struct inet_frags *f) |
| { |
| if (del_timer(&fq->timer)) |
| atomic_dec(&fq->refcnt); |
| |
| if (!(fq->last_in & COMPLETE)) { |
| fq_unlink(fq, f); |
| atomic_dec(&fq->refcnt); |
| fq->last_in |= COMPLETE; |
| } |
| } |
| |
| EXPORT_SYMBOL(inet_frag_kill); |