2005-04-17 02:20:36 +04:00
|
|
|
/*
|
|
|
|
* INET An implementation of the TCP/IP protocol suite for the LINUX
|
|
|
|
* operating system. INET is implemented using the BSD Socket
|
|
|
|
* interface as the means of communication with the user level.
|
|
|
|
*
|
|
|
|
* The IP fragmentation functionality.
|
2007-02-09 17:24:47 +03:00
|
|
|
*
|
2005-04-17 02:20:36 +04:00
|
|
|
* Authors: Fred N. van Kempen <waltje@uWalt.NL.Mugnet.ORG>
|
2008-10-14 06:01:08 +04:00
|
|
|
* Alan Cox <alan@lxorguk.ukuu.org.uk>
|
2005-04-17 02:20:36 +04:00
|
|
|
*
|
|
|
|
* Fixes:
|
|
|
|
* Alan Cox : Split from ip.c , see ip_input.c for history.
|
|
|
|
* David S. Miller : Begin massive cleanup...
|
|
|
|
* Andi Kleen : Add sysctls.
|
|
|
|
* xxxx : Overlapfrag bug.
|
|
|
|
* Ultima : ip_expire() kernel panic.
|
|
|
|
* Bill Hawes : Frag accounting and evictor fixes.
|
|
|
|
* John McDonald : 0 length frag bug.
|
|
|
|
* Alexey Kuznetsov: SMP races, threading, cleanup.
|
|
|
|
* Patrick McHardy : LRU queue of frag heads for evictor.
|
|
|
|
*/
|
|
|
|
|
2012-03-12 11:03:32 +04:00
|
|
|
#define pr_fmt(fmt) "IPv4: " fmt
|
|
|
|
|
2005-12-14 10:14:27 +03:00
|
|
|
#include <linux/compiler.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/jiffies.h>
|
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/ip.h>
|
|
|
|
#include <linux/icmp.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/jhash.h>
|
|
|
|
#include <linux/random.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 11:04:11 +03:00
|
|
|
#include <linux/slab.h>
|
2010-01-23 12:57:42 +03:00
|
|
|
#include <net/route.h>
|
|
|
|
#include <net/dst.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include <net/sock.h>
|
|
|
|
#include <net/ip.h>
|
|
|
|
#include <net/icmp.h>
|
|
|
|
#include <net/checksum.h>
|
2005-12-14 10:14:27 +03:00
|
|
|
#include <net/inetpeer.h>
|
2007-10-15 13:24:19 +04:00
|
|
|
#include <net/inet_frag.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include <linux/tcp.h>
|
|
|
|
#include <linux/udp.h>
|
|
|
|
#include <linux/inet.h>
|
|
|
|
#include <linux/netfilter_ipv4.h>
|
2011-01-05 10:52:55 +03:00
|
|
|
#include <net/inet_ecn.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* NOTE. Logic of IP defragmentation is parallel to corresponding IPv6
|
|
|
|
* code now. If you change something here, _PLEASE_ update ipv6/reassembly.c
|
|
|
|
* as well. Or notify me, at least. --ANK
|
|
|
|
*/
|
|
|
|
|
2008-01-22 16:58:31 +03:00
|
|
|
static int sysctl_ipfrag_max_dist __read_mostly = 64;
|
2005-12-14 10:14:27 +03:00
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
struct ipfrag_skb_cb
|
|
|
|
{
|
|
|
|
struct inet_skb_parm h;
|
|
|
|
int offset;
|
|
|
|
};
|
|
|
|
|
2008-11-03 13:47:38 +03:00
|
|
|
#define FRAG_CB(skb) ((struct ipfrag_skb_cb *)((skb)->cb))
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* Describe an entry in the "incomplete datagrams" queue. */
|
|
|
|
struct ipq {
|
2007-10-15 13:24:19 +04:00
|
|
|
struct inet_frag_queue q;
|
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
u32 user;
|
2006-09-27 09:19:02 +04:00
|
|
|
__be32 saddr;
|
|
|
|
__be32 daddr;
|
|
|
|
__be16 id;
|
2005-04-17 02:20:36 +04:00
|
|
|
u8 protocol;
|
2011-01-05 10:52:55 +03:00
|
|
|
u8 ecn; /* RFC3168 support */
|
2005-12-14 10:14:27 +03:00
|
|
|
int iif;
|
|
|
|
unsigned int rid;
|
|
|
|
struct inet_peer *peer;
|
2005-04-17 02:20:36 +04:00
|
|
|
};
|
|
|
|
|
2011-01-05 10:52:55 +03:00
|
|
|
static inline u8 ip4_frag_ecn(u8 tos)
|
|
|
|
{
|
2011-05-16 12:37:37 +04:00
|
|
|
return 1 << (tos & INET_ECN_MASK);
|
2011-01-05 10:52:55 +03:00
|
|
|
}
|
|
|
|
|
2007-10-15 13:31:52 +04:00
|
|
|
static struct inet_frags ip4_frags;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2008-01-22 17:07:25 +03:00
|
|
|
int ip_frag_mem(struct net *net)
|
2007-10-15 13:31:52 +04:00
|
|
|
{
|
2013-01-29 03:45:12 +04:00
|
|
|
return sum_frag_mem_limit(&net->ipv4.frags);
|
2007-10-15 13:31:52 +04:00
|
|
|
}
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev,
|
|
|
|
struct net_device *dev);
|
|
|
|
|
2007-10-18 06:46:47 +04:00
|
|
|
struct ip4_create_arg {
|
|
|
|
struct iphdr *iph;
|
|
|
|
u32 user;
|
|
|
|
};
|
|
|
|
|
2006-09-27 09:19:02 +04:00
|
|
|
static unsigned int ipqhashfn(__be16 id, __be32 saddr, __be32 daddr, u8 prot)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2013-10-23 13:06:55 +04:00
|
|
|
net_get_random_once(&ip4_frags.rnd, sizeof(ip4_frags.rnd));
|
2006-09-27 09:19:02 +04:00
|
|
|
return jhash_3words((__force u32)id << 16 | prot,
|
|
|
|
(__force u32)saddr, (__force u32)daddr,
|
2014-07-24 18:50:30 +04:00
|
|
|
ip4_frags.rnd);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2014-07-24 18:50:29 +04:00
|
|
|
static unsigned int ip4_hashfn(const struct inet_frag_queue *q)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2014-07-24 18:50:29 +04:00
|
|
|
const struct ipq *ipq;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-15 13:38:08 +04:00
|
|
|
ipq = container_of(q, struct ipq, q);
|
|
|
|
return ipqhashfn(ipq->id, ipq->saddr, ipq->daddr, ipq->protocol);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2014-07-24 18:50:29 +04:00
|
|
|
static bool ip4_frag_match(const struct inet_frag_queue *q, const void *a)
|
2007-10-18 06:47:21 +04:00
|
|
|
{
|
2014-07-24 18:50:29 +04:00
|
|
|
const struct ipq *qp;
|
|
|
|
const struct ip4_create_arg *arg = a;
|
2007-10-18 06:47:21 +04:00
|
|
|
|
|
|
|
qp = container_of(q, struct ipq, q);
|
2010-09-23 00:43:57 +04:00
|
|
|
return qp->id == arg->iph->id &&
|
2012-05-18 07:57:13 +04:00
|
|
|
qp->saddr == arg->iph->saddr &&
|
|
|
|
qp->daddr == arg->iph->daddr &&
|
|
|
|
qp->protocol == arg->iph->protocol &&
|
|
|
|
qp->user == arg->user;
|
2007-10-18 06:47:21 +04:00
|
|
|
}
|
|
|
|
|
2014-07-24 18:50:29 +04:00
|
|
|
static void ip4_frag_init(struct inet_frag_queue *q, const void *a)
|
2007-10-18 06:46:47 +04:00
|
|
|
{
|
|
|
|
struct ipq *qp = container_of(q, struct ipq, q);
|
2012-06-08 05:21:40 +04:00
|
|
|
struct netns_ipv4 *ipv4 = container_of(q->net, struct netns_ipv4,
|
|
|
|
frags);
|
|
|
|
struct net *net = container_of(ipv4, struct net, ipv4);
|
|
|
|
|
2014-07-24 18:50:29 +04:00
|
|
|
const struct ip4_create_arg *arg = a;
|
2007-10-18 06:46:47 +04:00
|
|
|
|
|
|
|
qp->protocol = arg->iph->protocol;
|
|
|
|
qp->id = arg->iph->id;
|
2011-01-05 10:52:55 +03:00
|
|
|
qp->ecn = ip4_frag_ecn(arg->iph->tos);
|
2007-10-18 06:46:47 +04:00
|
|
|
qp->saddr = arg->iph->saddr;
|
|
|
|
qp->daddr = arg->iph->daddr;
|
|
|
|
qp->user = arg->user;
|
|
|
|
qp->peer = sysctl_ipfrag_max_dist ?
|
2012-06-10 06:12:36 +04:00
|
|
|
inet_getpeer_v4(net->ipv4.peers, arg->iph->saddr, 1) : NULL;
|
2007-10-18 06:46:47 +04:00
|
|
|
}
|
|
|
|
|
2007-10-15 13:39:14 +04:00
|
|
|
static __inline__ void ip4_frag_free(struct inet_frag_queue *q)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2007-10-15 13:39:14 +04:00
|
|
|
struct ipq *qp;
|
|
|
|
|
|
|
|
qp = container_of(q, struct ipq, q);
|
|
|
|
if (qp->peer)
|
|
|
|
inet_putpeer(qp->peer);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Destruction primitives. */
|
|
|
|
|
2007-10-15 13:41:09 +04:00
|
|
|
static __inline__ void ipq_put(struct ipq *ipq)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2007-10-15 13:41:56 +04:00
|
|
|
inet_frag_put(&ipq->q, &ip4_frags);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Kill ipq entry. It is not destroyed immediately,
|
|
|
|
* because caller (and someone more) holds reference count.
|
|
|
|
*/
|
|
|
|
static void ipq_kill(struct ipq *ipq)
|
|
|
|
{
|
2007-10-15 13:37:18 +04:00
|
|
|
inet_frag_kill(&ipq->q, &ip4_frags);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Oops, a fragment queue timed out. Kill it and send an ICMP reply.
|
|
|
|
*/
|
|
|
|
static void ip_expire(unsigned long arg)
|
|
|
|
{
|
2007-10-18 06:45:23 +04:00
|
|
|
struct ipq *qp;
|
2008-07-17 07:19:08 +04:00
|
|
|
struct net *net;
|
2007-10-18 06:45:23 +04:00
|
|
|
|
|
|
|
qp = container_of((struct inet_frag_queue *) arg, struct ipq, q);
|
2008-07-17 07:19:08 +04:00
|
|
|
net = container_of(qp->q.net, struct net, ipv4.frags);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
spin_lock(&qp->q.lock);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-08-01 14:29:44 +04:00
|
|
|
if (qp->q.flags & INET_FRAG_COMPLETE)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto out;
|
|
|
|
|
|
|
|
ipq_kill(qp);
|
2008-07-17 07:20:11 +04:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMFAILS);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-08-01 14:29:47 +04:00
|
|
|
if (!(qp->q.flags & INET_FRAG_EVICTED)) {
|
2007-10-15 13:24:19 +04:00
|
|
|
struct sk_buff *head = qp->q.fragments;
|
2011-05-04 14:02:26 +04:00
|
|
|
const struct iphdr *iph;
|
|
|
|
int err;
|
2008-03-25 01:31:00 +03:00
|
|
|
|
2014-08-01 14:29:47 +04:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMTIMEOUT);
|
|
|
|
|
|
|
|
if (!(qp->q.flags & INET_FRAG_FIRST_IN) || !qp->q.fragments)
|
|
|
|
goto out;
|
|
|
|
|
2009-11-06 07:59:47 +03:00
|
|
|
rcu_read_lock();
|
|
|
|
head->dev = dev_get_by_index_rcu(net, qp->iif);
|
2010-01-23 12:57:42 +03:00
|
|
|
if (!head->dev)
|
|
|
|
goto out_rcu_unlock;
|
|
|
|
|
2013-04-16 16:55:41 +04:00
|
|
|
/* skb has no dst, perform route lookup again */
|
2011-05-04 14:02:26 +04:00
|
|
|
iph = ip_hdr(head);
|
2012-07-26 15:14:38 +04:00
|
|
|
err = ip_route_input_noref(head, iph->daddr, iph->saddr,
|
|
|
|
iph->tos, head->dev);
|
2011-05-04 14:02:26 +04:00
|
|
|
if (err)
|
|
|
|
goto out_rcu_unlock;
|
|
|
|
|
2014-08-01 14:29:47 +04:00
|
|
|
/* Only an end host needs to send an ICMP
|
2011-05-04 14:02:26 +04:00
|
|
|
* "Fragment Reassembly Timeout" message, per RFC792.
|
2010-01-23 12:57:42 +03:00
|
|
|
*/
|
2011-07-05 12:05:48 +04:00
|
|
|
if (qp->user == IP_DEFRAG_AF_PACKET ||
|
2014-05-03 03:14:04 +04:00
|
|
|
((qp->user >= IP_DEFRAG_CONNTRACK_IN) &&
|
|
|
|
(qp->user <= __IP_DEFRAG_CONNTRACK_IN_END) &&
|
|
|
|
(skb_rtable(head)->rt_type != RTN_LOCAL)))
|
2011-05-04 14:02:26 +04:00
|
|
|
goto out_rcu_unlock;
|
|
|
|
|
2010-01-23 12:57:42 +03:00
|
|
|
/* Send an ICMP "Fragment Reassembly Timeout" message. */
|
|
|
|
icmp_send(head, ICMP_TIME_EXCEEDED, ICMP_EXC_FRAGTIME, 0);
|
|
|
|
out_rcu_unlock:
|
2010-02-02 22:46:50 +03:00
|
|
|
rcu_read_unlock();
|
|
|
|
}
|
2005-04-17 02:20:36 +04:00
|
|
|
out:
|
2007-10-15 13:24:19 +04:00
|
|
|
spin_unlock(&qp->q.lock);
|
2007-10-15 13:41:09 +04:00
|
|
|
ipq_put(qp);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2007-10-18 06:47:21 +04:00
|
|
|
/* Find the correct entry in the "incomplete datagrams" queue for
|
|
|
|
* this IP datagram, and create new one, if nothing is found.
|
|
|
|
*/
|
2008-01-22 17:02:14 +03:00
|
|
|
static inline struct ipq *ip_find(struct net *net, struct iphdr *iph, u32 user)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2007-10-18 06:46:47 +04:00
|
|
|
struct inet_frag_queue *q;
|
|
|
|
struct ip4_create_arg arg;
|
2007-10-18 06:47:21 +04:00
|
|
|
unsigned int hash;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-18 06:46:47 +04:00
|
|
|
arg.iph = iph;
|
|
|
|
arg.user = user;
|
2008-06-28 07:06:08 +04:00
|
|
|
|
2007-10-18 06:47:21 +04:00
|
|
|
hash = ipqhashfn(iph->id, iph->saddr, iph->daddr, iph->protocol);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2008-01-22 17:02:14 +03:00
|
|
|
q = inet_frag_find(&net->ipv4.frags, &ip4_frags, &arg, hash);
|
2013-03-15 15:32:30 +04:00
|
|
|
if (IS_ERR_OR_NULL(q)) {
|
|
|
|
inet_frag_maybe_warn_overflow(q, pr_fmt());
|
|
|
|
return NULL;
|
|
|
|
}
|
2007-10-18 06:46:47 +04:00
|
|
|
return container_of(q, struct ipq, q);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2005-12-14 10:14:27 +03:00
|
|
|
/* Is the fragment too far ahead to be part of ipq? */
|
|
|
|
static inline int ip_frag_too_far(struct ipq *qp)
|
|
|
|
{
|
|
|
|
struct inet_peer *peer = qp->peer;
|
|
|
|
unsigned int max = sysctl_ipfrag_max_dist;
|
|
|
|
unsigned int start, end;
|
|
|
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (!peer || !max)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
start = qp->rid;
|
|
|
|
end = atomic_inc_return(&peer->rid);
|
|
|
|
qp->rid = end;
|
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
rc = qp->q.fragments && (end - start) > max;
|
2005-12-14 10:14:27 +03:00
|
|
|
|
|
|
|
if (rc) {
|
2008-07-17 07:20:11 +04:00
|
|
|
struct net *net;
|
|
|
|
|
|
|
|
net = container_of(qp->q.net, struct net, ipv4.frags);
|
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMFAILS);
|
2005-12-14 10:14:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ip_frag_reinit(struct ipq *qp)
|
|
|
|
{
|
|
|
|
struct sk_buff *fp;
|
2013-01-29 03:45:12 +04:00
|
|
|
unsigned int sum_truesize = 0;
|
2005-12-14 10:14:27 +03:00
|
|
|
|
2008-01-22 17:09:37 +03:00
|
|
|
if (!mod_timer(&qp->q.timer, jiffies + qp->q.net->timeout)) {
|
2007-10-15 13:24:19 +04:00
|
|
|
atomic_inc(&qp->q.refcnt);
|
2005-12-14 10:14:27 +03:00
|
|
|
return -ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
fp = qp->q.fragments;
|
2005-12-14 10:14:27 +03:00
|
|
|
do {
|
|
|
|
struct sk_buff *xp = fp->next;
|
2013-01-29 03:45:12 +04:00
|
|
|
|
|
|
|
sum_truesize += fp->truesize;
|
|
|
|
kfree_skb(fp);
|
2005-12-14 10:14:27 +03:00
|
|
|
fp = xp;
|
|
|
|
} while (fp);
|
2013-01-29 03:45:12 +04:00
|
|
|
sub_frag_mem_limit(&qp->q, sum_truesize);
|
2005-12-14 10:14:27 +03:00
|
|
|
|
2014-08-01 14:29:44 +04:00
|
|
|
qp->q.flags = 0;
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.len = 0;
|
|
|
|
qp->q.meat = 0;
|
|
|
|
qp->q.fragments = NULL;
|
2010-06-29 08:39:37 +04:00
|
|
|
qp->q.fragments_tail = NULL;
|
2005-12-14 10:14:27 +03:00
|
|
|
qp->iif = 0;
|
2011-01-05 10:52:55 +03:00
|
|
|
qp->ecn = 0;
|
2005-12-14 10:14:27 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
/* Add new segment to existing queue. */
|
2007-10-14 11:38:15 +04:00
|
|
|
static int ip_frag_queue(struct ipq *qp, struct sk_buff *skb)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
|
|
|
struct sk_buff *prev, *next;
|
2007-10-14 11:38:15 +04:00
|
|
|
struct net_device *dev;
|
2005-04-17 02:20:36 +04:00
|
|
|
int flags, offset;
|
|
|
|
int ihl, end;
|
2007-10-14 11:38:15 +04:00
|
|
|
int err = -ENOENT;
|
2011-01-05 10:52:55 +03:00
|
|
|
u8 ecn;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-08-01 14:29:44 +04:00
|
|
|
if (qp->q.flags & INET_FRAG_COMPLETE)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto err;
|
|
|
|
|
2005-12-14 10:14:27 +03:00
|
|
|
if (!(IPCB(skb)->flags & IPSKB_FRAG_COMPLETE) &&
|
2007-10-14 11:38:15 +04:00
|
|
|
unlikely(ip_frag_too_far(qp)) &&
|
|
|
|
unlikely(err = ip_frag_reinit(qp))) {
|
2005-12-14 10:14:27 +03:00
|
|
|
ipq_kill(qp);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2011-01-05 10:52:55 +03:00
|
|
|
ecn = ip4_frag_ecn(ip_hdr(skb)->tos);
|
2007-04-21 09:47:35 +04:00
|
|
|
offset = ntohs(ip_hdr(skb)->frag_off);
|
2005-04-17 02:20:36 +04:00
|
|
|
flags = offset & ~IP_OFFSET;
|
|
|
|
offset &= IP_OFFSET;
|
|
|
|
offset <<= 3; /* offset is in 8-byte chunks */
|
2007-03-13 02:09:15 +03:00
|
|
|
ihl = ip_hdrlen(skb);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* Determine the position of this fragment. */
|
2007-02-09 17:24:47 +03:00
|
|
|
end = offset + skb->len - ihl;
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -EINVAL;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* Is this the final fragment? */
|
|
|
|
if ((flags & IP_MF) == 0) {
|
|
|
|
/* If we already have some bits beyond end
|
2011-11-29 08:31:00 +04:00
|
|
|
* or have different end, the segment is corrupted.
|
2005-04-17 02:20:36 +04:00
|
|
|
*/
|
2007-10-15 13:24:19 +04:00
|
|
|
if (end < qp->q.len ||
|
2014-08-01 14:29:44 +04:00
|
|
|
((qp->q.flags & INET_FRAG_LAST_IN) && end != qp->q.len))
|
2005-04-17 02:20:36 +04:00
|
|
|
goto err;
|
2014-08-01 14:29:44 +04:00
|
|
|
qp->q.flags |= INET_FRAG_LAST_IN;
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.len = end;
|
2005-04-17 02:20:36 +04:00
|
|
|
} else {
|
|
|
|
if (end&7) {
|
|
|
|
end &= ~7;
|
|
|
|
if (skb->ip_summed != CHECKSUM_UNNECESSARY)
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
|
|
|
}
|
2007-10-15 13:24:19 +04:00
|
|
|
if (end > qp->q.len) {
|
2005-04-17 02:20:36 +04:00
|
|
|
/* Some bits beyond end -> corruption. */
|
2014-08-01 14:29:44 +04:00
|
|
|
if (qp->q.flags & INET_FRAG_LAST_IN)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto err;
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.len = end;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (end == offset)
|
|
|
|
goto err;
|
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -ENOMEM;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (pskb_pull(skb, ihl) == NULL)
|
|
|
|
goto err;
|
2007-10-14 11:38:15 +04:00
|
|
|
|
|
|
|
err = pskb_trim_rcsum(skb, end - offset);
|
|
|
|
if (err)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto err;
|
|
|
|
|
|
|
|
/* Find out which fragments are in front and at the back of us
|
|
|
|
* in the chain of fragments so far. We must know where to put
|
|
|
|
* this fragment, right?
|
|
|
|
*/
|
2010-06-29 08:39:37 +04:00
|
|
|
prev = qp->q.fragments_tail;
|
|
|
|
if (!prev || FRAG_CB(prev)->offset < offset) {
|
|
|
|
next = NULL;
|
|
|
|
goto found;
|
|
|
|
}
|
2005-04-17 02:20:36 +04:00
|
|
|
prev = NULL;
|
2007-10-15 13:24:19 +04:00
|
|
|
for (next = qp->q.fragments; next != NULL; next = next->next) {
|
2005-04-17 02:20:36 +04:00
|
|
|
if (FRAG_CB(next)->offset >= offset)
|
|
|
|
break; /* bingo! */
|
|
|
|
prev = next;
|
|
|
|
}
|
|
|
|
|
2010-06-29 08:39:37 +04:00
|
|
|
found:
|
2005-04-17 02:20:36 +04:00
|
|
|
/* We found where to put this one. Check for overlap with
|
|
|
|
* preceding fragment, and, if needed, align things so that
|
|
|
|
* any overlaps are eliminated.
|
|
|
|
*/
|
|
|
|
if (prev) {
|
|
|
|
int i = (FRAG_CB(prev)->offset + prev->len) - offset;
|
|
|
|
|
|
|
|
if (i > 0) {
|
|
|
|
offset += i;
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -EINVAL;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (end <= offset)
|
|
|
|
goto err;
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -ENOMEM;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (!pskb_pull(skb, i))
|
|
|
|
goto err;
|
|
|
|
if (skb->ip_summed != CHECKSUM_UNNECESSARY)
|
|
|
|
skb->ip_summed = CHECKSUM_NONE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -ENOMEM;
|
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
while (next && FRAG_CB(next)->offset < end) {
|
|
|
|
int i = end - FRAG_CB(next)->offset; /* overlap is 'i' bytes */
|
|
|
|
|
|
|
|
if (i < next->len) {
|
|
|
|
/* Eat head of the next overlapped fragment
|
|
|
|
* and leave the loop. The next ones cannot overlap.
|
|
|
|
*/
|
|
|
|
if (!pskb_pull(next, i))
|
|
|
|
goto err;
|
|
|
|
FRAG_CB(next)->offset += i;
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.meat -= i;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (next->ip_summed != CHECKSUM_UNNECESSARY)
|
|
|
|
next->ip_summed = CHECKSUM_NONE;
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
struct sk_buff *free_it = next;
|
|
|
|
|
2006-12-12 21:48:59 +03:00
|
|
|
/* Old fragment is completely overridden with
|
2005-04-17 02:20:36 +04:00
|
|
|
* new one drop it.
|
|
|
|
*/
|
|
|
|
next = next->next;
|
|
|
|
|
|
|
|
if (prev)
|
|
|
|
prev->next = next;
|
|
|
|
else
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.fragments = next;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.meat -= free_it->len;
|
2013-01-29 03:45:12 +04:00
|
|
|
sub_frag_mem_limit(&qp->q, free_it->truesize);
|
|
|
|
kfree_skb(free_it);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FRAG_CB(skb)->offset = offset;
|
|
|
|
|
|
|
|
/* Insert this fragment in the chain of fragments. */
|
|
|
|
skb->next = next;
|
2010-06-29 08:39:37 +04:00
|
|
|
if (!next)
|
|
|
|
qp->q.fragments_tail = skb;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (prev)
|
|
|
|
prev->next = skb;
|
|
|
|
else
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.fragments = skb;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
dev = skb->dev;
|
|
|
|
if (dev) {
|
|
|
|
qp->iif = dev->ifindex;
|
|
|
|
skb->dev = NULL;
|
|
|
|
}
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.stamp = skb->tstamp;
|
|
|
|
qp->q.meat += skb->len;
|
2011-01-05 10:52:55 +03:00
|
|
|
qp->ecn |= ecn;
|
2013-01-29 03:45:12 +04:00
|
|
|
add_frag_mem_limit(&qp->q, skb->truesize);
|
2005-04-17 02:20:36 +04:00
|
|
|
if (offset == 0)
|
2014-08-01 14:29:44 +04:00
|
|
|
qp->q.flags |= INET_FRAG_FIRST_IN;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2012-08-26 21:13:55 +04:00
|
|
|
if (ip_hdr(skb)->frag_off & htons(IP_DF) &&
|
|
|
|
skb->len + ihl > qp->q.max_size)
|
|
|
|
qp->q.max_size = skb->len + ihl;
|
|
|
|
|
2014-08-01 14:29:44 +04:00
|
|
|
if (qp->q.flags == (INET_FRAG_FIRST_IN | INET_FRAG_LAST_IN) &&
|
2013-04-16 16:55:41 +04:00
|
|
|
qp->q.meat == qp->q.len) {
|
|
|
|
unsigned long orefdst = skb->_skb_refdst;
|
2007-10-14 11:38:15 +04:00
|
|
|
|
2013-04-16 16:55:41 +04:00
|
|
|
skb->_skb_refdst = 0UL;
|
|
|
|
err = ip_frag_reasm(qp, prev, dev);
|
|
|
|
skb->_skb_refdst = orefdst;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
skb_dst_drop(skb);
|
2007-10-14 11:38:15 +04:00
|
|
|
return -EINPROGRESS;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
err:
|
|
|
|
kfree_skb(skb);
|
2007-10-14 11:38:15 +04:00
|
|
|
return err;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Build a new IP datagram from all its fragments. */
|
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
static int ip_frag_reasm(struct ipq *qp, struct sk_buff *prev,
|
|
|
|
struct net_device *dev)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2009-03-19 09:26:11 +03:00
|
|
|
struct net *net = container_of(qp->q.net, struct net, ipv4.frags);
|
2005-04-17 02:20:36 +04:00
|
|
|
struct iphdr *iph;
|
2007-10-15 13:24:19 +04:00
|
|
|
struct sk_buff *fp, *head = qp->q.fragments;
|
2005-04-17 02:20:36 +04:00
|
|
|
int len;
|
|
|
|
int ihlen;
|
2007-10-14 11:38:15 +04:00
|
|
|
int err;
|
2012-05-19 07:02:20 +04:00
|
|
|
int sum_truesize;
|
2011-05-16 12:37:37 +04:00
|
|
|
u8 ecn;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
ipq_kill(qp);
|
|
|
|
|
2013-03-22 12:24:37 +04:00
|
|
|
ecn = ip_frag_ecn_table[qp->ecn];
|
2011-05-16 12:37:37 +04:00
|
|
|
if (unlikely(ecn == 0xff)) {
|
|
|
|
err = -EINVAL;
|
|
|
|
goto out_fail;
|
|
|
|
}
|
2007-10-14 11:38:15 +04:00
|
|
|
/* Make the one we just received the head. */
|
|
|
|
if (prev) {
|
|
|
|
head = prev->next;
|
|
|
|
fp = skb_clone(head, GFP_ATOMIC);
|
|
|
|
if (!fp)
|
|
|
|
goto out_nomem;
|
|
|
|
|
|
|
|
fp->next = head->next;
|
2010-06-29 08:39:37 +04:00
|
|
|
if (!fp->next)
|
|
|
|
qp->q.fragments_tail = fp;
|
2007-10-14 11:38:15 +04:00
|
|
|
prev->next = fp;
|
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
skb_morph(head, qp->q.fragments);
|
|
|
|
head->next = qp->q.fragments->next;
|
2007-10-14 11:38:15 +04:00
|
|
|
|
2012-04-19 10:10:26 +04:00
|
|
|
consume_skb(qp->q.fragments);
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.fragments = head;
|
2007-10-14 11:38:15 +04:00
|
|
|
}
|
|
|
|
|
2008-07-26 08:43:18 +04:00
|
|
|
WARN_ON(head == NULL);
|
|
|
|
WARN_ON(FRAG_CB(head)->offset != 0);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* Allocate a new buffer for the datagram. */
|
2007-03-13 02:09:15 +03:00
|
|
|
ihlen = ip_hdrlen(head);
|
2007-10-15 13:24:19 +04:00
|
|
|
len = ihlen + qp->q.len;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
err = -E2BIG;
|
2007-03-09 07:44:43 +03:00
|
|
|
if (len > 65535)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto out_oversize;
|
|
|
|
|
|
|
|
/* Head of list must not be cloned. */
|
2013-02-14 13:44:49 +04:00
|
|
|
if (skb_unclone(head, GFP_ATOMIC))
|
2005-04-17 02:20:36 +04:00
|
|
|
goto out_nomem;
|
|
|
|
|
|
|
|
/* If the first fragment is fragmented itself, we split
|
|
|
|
* it to two chunks: the first with data and paged part
|
|
|
|
* and the second, holding only fragments. */
|
2010-08-23 11:13:46 +04:00
|
|
|
if (skb_has_frag_list(head)) {
|
2005-04-17 02:20:36 +04:00
|
|
|
struct sk_buff *clone;
|
|
|
|
int i, plen = 0;
|
|
|
|
|
|
|
|
if ((clone = alloc_skb(0, GFP_ATOMIC)) == NULL)
|
|
|
|
goto out_nomem;
|
|
|
|
clone->next = head->next;
|
|
|
|
head->next = clone;
|
|
|
|
skb_shinfo(clone)->frag_list = skb_shinfo(head)->frag_list;
|
2009-06-09 11:19:37 +04:00
|
|
|
skb_frag_list_init(head);
|
2011-10-19 01:00:24 +04:00
|
|
|
for (i = 0; i < skb_shinfo(head)->nr_frags; i++)
|
|
|
|
plen += skb_frag_size(&skb_shinfo(head)->frags[i]);
|
2005-04-17 02:20:36 +04:00
|
|
|
clone->len = clone->data_len = head->data_len - plen;
|
|
|
|
head->data_len -= clone->len;
|
|
|
|
head->len -= clone->len;
|
|
|
|
clone->csum = 0;
|
|
|
|
clone->ip_summed = head->ip_summed;
|
2013-01-29 03:45:12 +04:00
|
|
|
add_frag_mem_limit(&qp->q, clone->truesize);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2007-04-11 07:50:43 +04:00
|
|
|
skb_push(head, head->data - skb_network_header(head));
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2012-05-19 07:02:20 +04:00
|
|
|
sum_truesize = head->truesize;
|
|
|
|
for (fp = head->next; fp;) {
|
|
|
|
bool headstolen;
|
|
|
|
int delta;
|
|
|
|
struct sk_buff *next = fp->next;
|
|
|
|
|
|
|
|
sum_truesize += fp->truesize;
|
2005-04-17 02:20:36 +04:00
|
|
|
if (head->ip_summed != fp->ip_summed)
|
|
|
|
head->ip_summed = CHECKSUM_NONE;
|
2006-08-30 03:44:56 +04:00
|
|
|
else if (head->ip_summed == CHECKSUM_COMPLETE)
|
2005-04-17 02:20:36 +04:00
|
|
|
head->csum = csum_add(head->csum, fp->csum);
|
2012-05-19 07:02:20 +04:00
|
|
|
|
|
|
|
if (skb_try_coalesce(head, fp, &headstolen, &delta)) {
|
|
|
|
kfree_skb_partial(fp, headstolen);
|
|
|
|
} else {
|
|
|
|
if (!skb_shinfo(head)->frag_list)
|
|
|
|
skb_shinfo(head)->frag_list = fp;
|
|
|
|
head->data_len += fp->len;
|
|
|
|
head->len += fp->len;
|
|
|
|
head->truesize += fp->truesize;
|
|
|
|
}
|
|
|
|
fp = next;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
2013-01-29 03:45:12 +04:00
|
|
|
sub_frag_mem_limit(&qp->q, sum_truesize);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
head->next = NULL;
|
|
|
|
head->dev = dev;
|
2007-10-15 13:24:19 +04:00
|
|
|
head->tstamp = qp->q.stamp;
|
2012-08-26 21:13:55 +04:00
|
|
|
IPCB(head)->frag_max_size = qp->q.max_size;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-04-21 09:47:35 +04:00
|
|
|
iph = ip_hdr(head);
|
2012-08-26 21:13:55 +04:00
|
|
|
/* max_size != 0 implies at least one fragment had IP_DF set */
|
|
|
|
iph->frag_off = qp->q.max_size ? htons(IP_DF) : 0;
|
2005-04-17 02:20:36 +04:00
|
|
|
iph->tot_len = htons(len);
|
2011-05-16 12:37:37 +04:00
|
|
|
iph->tos |= ecn;
|
2009-03-19 09:26:11 +03:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMOKS);
|
2007-10-15 13:24:19 +04:00
|
|
|
qp->q.fragments = NULL;
|
2010-06-29 08:39:37 +04:00
|
|
|
qp->q.fragments_tail = NULL;
|
2007-10-14 11:38:15 +04:00
|
|
|
return 0;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
out_nomem:
|
2012-03-12 11:03:32 +04:00
|
|
|
LIMIT_NETDEBUG(KERN_ERR pr_fmt("queue_glue: no memory for gluing queue %p\n"),
|
|
|
|
qp);
|
2007-10-18 08:37:22 +04:00
|
|
|
err = -ENOMEM;
|
2005-04-17 02:20:36 +04:00
|
|
|
goto out_fail;
|
|
|
|
out_oversize:
|
2012-05-14 01:56:26 +04:00
|
|
|
net_info_ratelimited("Oversized IP packet from %pI4\n", &qp->saddr);
|
2005-04-17 02:20:36 +04:00
|
|
|
out_fail:
|
2009-11-30 10:02:22 +03:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMFAILS);
|
2007-10-14 11:38:15 +04:00
|
|
|
return err;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Process an incoming IP datagram fragment. */
|
2007-10-14 11:38:32 +04:00
|
|
|
int ip_defrag(struct sk_buff *skb, u32 user)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
|
|
|
struct ipq *qp;
|
2008-01-22 17:02:14 +03:00
|
|
|
struct net *net;
|
2007-02-09 17:24:47 +03:00
|
|
|
|
2009-06-02 09:19:30 +04:00
|
|
|
net = skb->dev ? dev_net(skb->dev) : dev_net(skb_dst(skb)->dev);
|
2008-07-17 07:20:11 +04:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMREQDS);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* Lookup (or create) queue header */
|
2008-01-22 17:02:14 +03:00
|
|
|
if ((qp = ip_find(net, ip_hdr(skb), user)) != NULL) {
|
2007-10-14 11:38:15 +04:00
|
|
|
int ret;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
spin_lock(&qp->q.lock);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-14 11:38:15 +04:00
|
|
|
ret = ip_frag_queue(qp, skb);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2007-10-15 13:24:19 +04:00
|
|
|
spin_unlock(&qp->q.lock);
|
2007-10-15 13:41:09 +04:00
|
|
|
ipq_put(qp);
|
2007-10-14 11:38:32 +04:00
|
|
|
return ret;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2008-07-17 07:20:11 +04:00
|
|
|
IP_INC_STATS_BH(net, IPSTATS_MIB_REASMFAILS);
|
2005-04-17 02:20:36 +04:00
|
|
|
kfree_skb(skb);
|
2007-10-14 11:38:32 +04:00
|
|
|
return -ENOMEM;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
2010-07-10 01:22:10 +04:00
|
|
|
EXPORT_SYMBOL(ip_defrag);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2011-10-06 14:28:31 +04:00
|
|
|
struct sk_buff *ip_check_defrag(struct sk_buff *skb, u32 user)
|
|
|
|
{
|
2012-12-10 03:41:06 +04:00
|
|
|
struct iphdr iph;
|
2011-10-06 14:28:31 +04:00
|
|
|
u32 len;
|
|
|
|
|
|
|
|
if (skb->protocol != htons(ETH_P_IP))
|
|
|
|
return skb;
|
|
|
|
|
2012-12-10 03:41:06 +04:00
|
|
|
if (!skb_copy_bits(skb, 0, &iph, sizeof(iph)))
|
2011-10-06 14:28:31 +04:00
|
|
|
return skb;
|
|
|
|
|
2012-12-10 03:41:06 +04:00
|
|
|
if (iph.ihl < 5 || iph.version != 4)
|
2011-10-06 14:28:31 +04:00
|
|
|
return skb;
|
2012-12-10 03:41:06 +04:00
|
|
|
|
|
|
|
len = ntohs(iph.tot_len);
|
|
|
|
if (skb->len < len || len < (iph.ihl * 4))
|
2011-10-06 14:28:31 +04:00
|
|
|
return skb;
|
|
|
|
|
2012-12-10 03:41:06 +04:00
|
|
|
if (ip_is_fragment(&iph)) {
|
2011-10-06 14:28:31 +04:00
|
|
|
skb = skb_share_check(skb, GFP_ATOMIC);
|
|
|
|
if (skb) {
|
2012-12-10 03:41:06 +04:00
|
|
|
if (!pskb_may_pull(skb, iph.ihl*4))
|
|
|
|
return skb;
|
2011-10-06 14:28:31 +04:00
|
|
|
if (pskb_trim_rcsum(skb, len))
|
|
|
|
return skb;
|
|
|
|
memset(IPCB(skb), 0, sizeof(struct inet_skb_parm));
|
|
|
|
if (ip_defrag(skb, user))
|
|
|
|
return NULL;
|
2013-12-16 10:12:18 +04:00
|
|
|
skb_clear_hash(skb);
|
2011-10-06 14:28:31 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return skb;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(ip_check_defrag);
|
|
|
|
|
2008-01-22 16:58:31 +03:00
|
|
|
#ifdef CONFIG_SYSCTL
|
|
|
|
static int zero;
|
|
|
|
|
2008-05-20 00:51:29 +04:00
|
|
|
static struct ctl_table ip4_frags_ns_ctl_table[] = {
|
2008-01-22 16:58:31 +03:00
|
|
|
{
|
|
|
|
.procname = "ipfrag_high_thresh",
|
2008-01-22 17:10:13 +03:00
|
|
|
.data = &init_net.ipv4.frags.high_thresh,
|
2008-01-22 16:58:31 +03:00
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
2014-07-24 18:50:37 +04:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
|
|
|
.extra1 = &init_net.ipv4.frags.low_thresh
|
2008-01-22 16:58:31 +03:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.procname = "ipfrag_low_thresh",
|
2008-01-22 17:10:13 +03:00
|
|
|
.data = &init_net.ipv4.frags.low_thresh,
|
2008-01-22 16:58:31 +03:00
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
2014-07-24 18:50:37 +04:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
|
|
|
.extra1 = &zero,
|
|
|
|
.extra2 = &init_net.ipv4.frags.high_thresh
|
2008-01-22 16:58:31 +03:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.procname = "ipfrag_time",
|
2008-01-22 17:09:37 +03:00
|
|
|
.data = &init_net.ipv4.frags.timeout,
|
2008-01-22 16:58:31 +03:00
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
2008-11-04 05:21:05 +03:00
|
|
|
.proc_handler = proc_dointvec_jiffies,
|
2008-01-22 16:58:31 +03:00
|
|
|
},
|
2008-05-20 00:53:02 +04:00
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2014-07-24 18:50:35 +04:00
|
|
|
/* secret interval has been deprecated */
|
|
|
|
static int ip4_frags_secret_interval_unused;
|
2008-05-20 00:53:02 +04:00
|
|
|
static struct ctl_table ip4_frags_ctl_table[] = {
|
2008-01-22 16:58:31 +03:00
|
|
|
{
|
|
|
|
.procname = "ipfrag_secret_interval",
|
2014-07-24 18:50:35 +04:00
|
|
|
.data = &ip4_frags_secret_interval_unused,
|
2008-01-22 16:58:31 +03:00
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
2008-11-04 05:21:05 +03:00
|
|
|
.proc_handler = proc_dointvec_jiffies,
|
2008-01-22 16:58:31 +03:00
|
|
|
},
|
|
|
|
{
|
|
|
|
.procname = "ipfrag_max_dist",
|
|
|
|
.data = &sysctl_ipfrag_max_dist,
|
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
2008-11-04 05:21:05 +03:00
|
|
|
.proc_handler = proc_dointvec_minmax,
|
2008-01-22 16:58:31 +03:00
|
|
|
.extra1 = &zero
|
|
|
|
},
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2010-01-17 06:35:32 +03:00
|
|
|
static int __net_init ip4_frags_ns_ctl_register(struct net *net)
|
2008-01-22 16:58:31 +03:00
|
|
|
{
|
2008-01-22 17:08:36 +03:00
|
|
|
struct ctl_table *table;
|
2008-01-22 16:58:31 +03:00
|
|
|
struct ctl_table_header *hdr;
|
|
|
|
|
2008-05-20 00:51:29 +04:00
|
|
|
table = ip4_frags_ns_ctl_table;
|
2009-11-26 02:14:13 +03:00
|
|
|
if (!net_eq(net, &init_net)) {
|
2008-05-20 00:51:29 +04:00
|
|
|
table = kmemdup(table, sizeof(ip4_frags_ns_ctl_table), GFP_KERNEL);
|
2008-01-22 17:08:36 +03:00
|
|
|
if (table == NULL)
|
|
|
|
goto err_alloc;
|
|
|
|
|
2008-01-22 17:10:13 +03:00
|
|
|
table[0].data = &net->ipv4.frags.high_thresh;
|
2014-07-24 18:50:37 +04:00
|
|
|
table[0].extra1 = &net->ipv4.frags.low_thresh;
|
|
|
|
table[0].extra2 = &init_net.ipv4.frags.high_thresh;
|
2008-01-22 17:10:13 +03:00
|
|
|
table[1].data = &net->ipv4.frags.low_thresh;
|
2014-07-24 18:50:37 +04:00
|
|
|
table[1].extra2 = &net->ipv4.frags.high_thresh;
|
2008-01-22 17:09:37 +03:00
|
|
|
table[2].data = &net->ipv4.frags.timeout;
|
2012-11-16 07:02:59 +04:00
|
|
|
|
|
|
|
/* Don't export sysctls to unprivileged users */
|
|
|
|
if (net->user_ns != &init_user_ns)
|
|
|
|
table[0].procname = NULL;
|
2008-01-22 17:08:36 +03:00
|
|
|
}
|
|
|
|
|
2012-04-19 17:44:49 +04:00
|
|
|
hdr = register_net_sysctl(net, "net/ipv4", table);
|
2008-01-22 17:08:36 +03:00
|
|
|
if (hdr == NULL)
|
|
|
|
goto err_reg;
|
|
|
|
|
|
|
|
net->ipv4.frags_hdr = hdr;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_reg:
|
2009-11-26 02:14:13 +03:00
|
|
|
if (!net_eq(net, &init_net))
|
2008-01-22 17:08:36 +03:00
|
|
|
kfree(table);
|
|
|
|
err_alloc:
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2010-01-17 06:35:32 +03:00
|
|
|
static void __net_exit ip4_frags_ns_ctl_unregister(struct net *net)
|
2008-01-22 17:08:36 +03:00
|
|
|
{
|
|
|
|
struct ctl_table *table;
|
|
|
|
|
|
|
|
table = net->ipv4.frags_hdr->ctl_table_arg;
|
|
|
|
unregister_net_sysctl_table(net->ipv4.frags_hdr);
|
|
|
|
kfree(table);
|
2008-01-22 16:58:31 +03:00
|
|
|
}
|
2008-05-20 00:53:02 +04:00
|
|
|
|
|
|
|
static void ip4_frags_ctl_register(void)
|
|
|
|
{
|
2012-04-19 17:22:55 +04:00
|
|
|
register_net_sysctl(&init_net, "net/ipv4", ip4_frags_ctl_table);
|
2008-05-20 00:53:02 +04:00
|
|
|
}
|
2008-01-22 16:58:31 +03:00
|
|
|
#else
|
2008-05-20 00:51:29 +04:00
|
|
|
static inline int ip4_frags_ns_ctl_register(struct net *net)
|
2008-01-22 16:58:31 +03:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2008-01-22 17:08:36 +03:00
|
|
|
|
2008-05-20 00:51:29 +04:00
|
|
|
static inline void ip4_frags_ns_ctl_unregister(struct net *net)
|
2008-01-22 17:08:36 +03:00
|
|
|
{
|
|
|
|
}
|
2008-05-20 00:53:02 +04:00
|
|
|
|
|
|
|
static inline void ip4_frags_ctl_register(void)
|
|
|
|
{
|
|
|
|
}
|
2008-01-22 16:58:31 +03:00
|
|
|
#endif
|
|
|
|
|
2010-01-17 06:35:32 +03:00
|
|
|
static int __net_init ipv4_frags_init_net(struct net *net)
|
2008-01-22 16:58:31 +03:00
|
|
|
{
|
net: increase fragment memory usage limits
Increase the amount of memory usage limits for incomplete
IP fragments.
Arguing for new thresh high/low values:
High threshold = 4 MBytes
Low threshold = 3 MBytes
The fragmentation memory accounting code, tries to account for the
real memory usage, by measuring both the size of frag queue struct
(inet_frag_queue (ipv4:ipq/ipv6:frag_queue)) and the SKB's truesize.
We want to be able to handle/hold-on-to enough fragments, to ensure
good performance, without causing incomplete fragments to hurt
scalability, by causing the number of inet_frag_queue to grow too much
(resulting longer searches for frag queues).
For IPv4, how much memory does the largest frag consume.
Maximum size fragment is 64K, which is approx 44 fragments with
MTU(1500) sized packets. Sizeof(struct ipq) is 200. A 1500 byte
packet results in a truesize of 2944 (not 2048 as I first assumed)
(44*2944)+200 = 129736 bytes
The current default high thresh of 262144 bytes, is obviously
problematic, as only two 64K fragments can fit in the queue at the
same time.
How many 64K fragment can we fit into 4 MBytes:
4*2^20/((44*2944)+200) = 32.34 fragment in queues
An attacker could send a separate/distinct fake fragment packets per
queue, causing us to allocate one inet_frag_queue per packet, and thus
attacking the hash table and its lists.
How many frag queue do we need to store, and given a current hash size
of 64, what is the average list length.
Using one MTU sized fragment per inet_frag_queue, each consuming
(2944+200) 3144 bytes.
4*2^20/(2944+200) = 1334 frag queues -> 21 avg list length
An attack could send small fragments, the smallest packet I could send
resulted in a truesize of 896 bytes (I'm a little surprised by this).
4*2^20/(896+200) = 3827 frag queues -> 59 avg list length
When increasing these number, we also need to followup with
improvements, that is going to help scalability. Simply increasing
the hash size, is not enough as the current implementation does not
have a per hash bucket locking.
Signed-off-by: Jesper Dangaard Brouer <brouer@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2013-01-15 11:16:35 +04:00
|
|
|
/* Fragment cache limits.
|
|
|
|
*
|
|
|
|
* The fragment memory accounting code, (tries to) account for
|
|
|
|
* the real memory usage, by measuring both the size of frag
|
|
|
|
* queue struct (inet_frag_queue (ipv4:ipq/ipv6:frag_queue))
|
|
|
|
* and the SKB's truesize.
|
|
|
|
*
|
|
|
|
* A 64K fragment consumes 129736 bytes (44*2944)+200
|
|
|
|
* (1500 truesize == 2944, sizeof(struct ipq) == 200)
|
|
|
|
*
|
|
|
|
* We will commit 4MB at one time. Should we cross that limit
|
|
|
|
* we will prune down to 3MB, making room for approx 8 big 64K
|
|
|
|
* fragments 8x128k.
|
2008-01-22 17:10:13 +03:00
|
|
|
*/
|
net: increase fragment memory usage limits
Increase the amount of memory usage limits for incomplete
IP fragments.
Arguing for new thresh high/low values:
High threshold = 4 MBytes
Low threshold = 3 MBytes
The fragmentation memory accounting code, tries to account for the
real memory usage, by measuring both the size of frag queue struct
(inet_frag_queue (ipv4:ipq/ipv6:frag_queue)) and the SKB's truesize.
We want to be able to handle/hold-on-to enough fragments, to ensure
good performance, without causing incomplete fragments to hurt
scalability, by causing the number of inet_frag_queue to grow too much
(resulting longer searches for frag queues).
For IPv4, how much memory does the largest frag consume.
Maximum size fragment is 64K, which is approx 44 fragments with
MTU(1500) sized packets. Sizeof(struct ipq) is 200. A 1500 byte
packet results in a truesize of 2944 (not 2048 as I first assumed)
(44*2944)+200 = 129736 bytes
The current default high thresh of 262144 bytes, is obviously
problematic, as only two 64K fragments can fit in the queue at the
same time.
How many 64K fragment can we fit into 4 MBytes:
4*2^20/((44*2944)+200) = 32.34 fragment in queues
An attacker could send a separate/distinct fake fragment packets per
queue, causing us to allocate one inet_frag_queue per packet, and thus
attacking the hash table and its lists.
How many frag queue do we need to store, and given a current hash size
of 64, what is the average list length.
Using one MTU sized fragment per inet_frag_queue, each consuming
(2944+200) 3144 bytes.
4*2^20/(2944+200) = 1334 frag queues -> 21 avg list length
An attack could send small fragments, the smallest packet I could send
resulted in a truesize of 896 bytes (I'm a little surprised by this).
4*2^20/(896+200) = 3827 frag queues -> 59 avg list length
When increasing these number, we also need to followup with
improvements, that is going to help scalability. Simply increasing
the hash size, is not enough as the current implementation does not
have a per hash bucket locking.
Signed-off-by: Jesper Dangaard Brouer <brouer@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2013-01-15 11:16:35 +04:00
|
|
|
net->ipv4.frags.high_thresh = 4 * 1024 * 1024;
|
|
|
|
net->ipv4.frags.low_thresh = 3 * 1024 * 1024;
|
2008-01-22 17:09:37 +03:00
|
|
|
/*
|
|
|
|
* Important NOTE! Fragment queue must be destroyed before MSL expires.
|
|
|
|
* RFC791 is wrong proposing to prolongate timer each fragment arrival
|
|
|
|
* by TTL.
|
|
|
|
*/
|
|
|
|
net->ipv4.frags.timeout = IP_FRAG_TIME;
|
|
|
|
|
2008-01-22 17:06:23 +03:00
|
|
|
inet_frags_init_net(&net->ipv4.frags);
|
|
|
|
|
2008-05-20 00:51:29 +04:00
|
|
|
return ip4_frags_ns_ctl_register(net);
|
2008-01-22 16:58:31 +03:00
|
|
|
}
|
|
|
|
|
2010-01-17 06:35:32 +03:00
|
|
|
static void __net_exit ipv4_frags_exit_net(struct net *net)
|
2008-01-22 17:12:39 +03:00
|
|
|
{
|
2008-05-20 00:51:29 +04:00
|
|
|
ip4_frags_ns_ctl_unregister(net);
|
2008-01-22 17:12:39 +03:00
|
|
|
inet_frags_exit_net(&net->ipv4.frags, &ip4_frags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct pernet_operations ip4_frags_ops = {
|
|
|
|
.init = ipv4_frags_init_net,
|
|
|
|
.exit = ipv4_frags_exit_net,
|
|
|
|
};
|
|
|
|
|
2007-04-20 03:16:32 +04:00
|
|
|
void __init ipfrag_init(void)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2008-05-20 00:53:02 +04:00
|
|
|
ip4_frags_ctl_register();
|
2008-01-22 17:12:39 +03:00
|
|
|
register_pernet_subsys(&ip4_frags_ops);
|
2007-10-15 13:38:08 +04:00
|
|
|
ip4_frags.hashfn = ip4_hashfn;
|
2007-10-18 06:46:47 +04:00
|
|
|
ip4_frags.constructor = ip4_frag_init;
|
2007-10-15 13:39:14 +04:00
|
|
|
ip4_frags.destructor = ip4_frag_free;
|
|
|
|
ip4_frags.skb_free = NULL;
|
|
|
|
ip4_frags.qsize = sizeof(struct ipq);
|
2007-10-18 06:47:21 +04:00
|
|
|
ip4_frags.match = ip4_frag_match;
|
2007-10-18 06:45:23 +04:00
|
|
|
ip4_frags.frag_expire = ip_expire;
|
2007-10-15 13:31:52 +04:00
|
|
|
inet_frags_init(&ip4_frags);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|