2007-10-15 17:31:52 +08:00
|
|
|
/*
|
|
|
|
* 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 <net/inet_frag.h>
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(inet_frags_init);
|
|
|
|
|
|
|
|
void inet_frags_fini(struct inet_frags *f)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(inet_frags_fini);
|
2007-10-15 17:37:18 +08:00
|
|
|
|
|
|
|
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);
|