Commit 1ee89bd0 authored by Nicolas Dichtel's avatar Nicolas Dichtel Committed by David S. Miller

netfilter: discard overlapping IPv6 fragment

RFC5722 prohibits reassembling IPv6 fragments when some data overlaps.

Bug spotted by Zhang Zuotao <zuotao.zhang@6wind.com>.
Signed-off-by: default avatarNicolas Dichtel <nicolas.dichtel@6wind.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 70789d70
...@@ -113,14 +113,6 @@ static void nf_skb_free(struct sk_buff *skb) ...@@ -113,14 +113,6 @@ static void nf_skb_free(struct sk_buff *skb)
kfree_skb(NFCT_FRAG6_CB(skb)->orig); kfree_skb(NFCT_FRAG6_CB(skb)->orig);
} }
/* Memory Tracking Functions. */
static void frag_kfree_skb(struct sk_buff *skb)
{
atomic_sub(skb->truesize, &nf_init_frags.mem);
nf_skb_free(skb);
kfree_skb(skb);
}
/* Destruction primitives. */ /* Destruction primitives. */
static __inline__ void fq_put(struct nf_ct_frag6_queue *fq) static __inline__ void fq_put(struct nf_ct_frag6_queue *fq)
...@@ -282,66 +274,22 @@ static int nf_ct_frag6_queue(struct nf_ct_frag6_queue *fq, struct sk_buff *skb, ...@@ -282,66 +274,22 @@ static int nf_ct_frag6_queue(struct nf_ct_frag6_queue *fq, struct sk_buff *skb,
} }
found: found:
/* We found where to put this one. Check for overlap with /* RFC5722, Section 4:
* preceding fragment, and, if needed, align things so that * When reassembling an IPv6 datagram, if
* any overlaps are eliminated. * one or more its constituent fragments is determined to be an
*/ * overlapping fragment, the entire datagram (and any constituent
if (prev) { * fragments, including those not yet received) MUST be silently
int i = (NFCT_FRAG6_CB(prev)->offset + prev->len) - offset; * discarded.
if (i > 0) {
offset += i;
if (end <= offset) {
pr_debug("overlap\n");
goto err;
}
if (!pskb_pull(skb, i)) {
pr_debug("Can't pull\n");
goto err;
}
if (skb->ip_summed != CHECKSUM_UNNECESSARY)
skb->ip_summed = CHECKSUM_NONE;
}
}
/* Look for overlap with succeeding segments.
* If we can merge fragments, do it.
*/ */
while (next && NFCT_FRAG6_CB(next)->offset < end) {
/* overlap is 'i' bytes */
int i = end - NFCT_FRAG6_CB(next)->offset;
if (i < next->len) { /* Check for overlap with preceding fragment. */
/* Eat head of the next overlapped fragment if (prev &&
* and leave the loop. The next ones cannot overlap. (NFCT_FRAG6_CB(prev)->offset + prev->len) - offset > 0)
*/ goto discard_fq;
pr_debug("Eat head of the overlapped parts.: %d", i);
if (!pskb_pull(next, i))
goto err;
/* next fragment */
NFCT_FRAG6_CB(next)->offset += i;
fq->q.meat -= i;
if (next->ip_summed != CHECKSUM_UNNECESSARY)
next->ip_summed = CHECKSUM_NONE;
break;
} else {
struct sk_buff *free_it = next;
/* Old fragmnet is completely overridden with
* new one drop it.
*/
next = next->next;
if (prev) /* Look for overlap with succeeding segment. */
prev->next = next; if (next && NFCT_FRAG6_CB(next)->offset < end)
else goto discard_fq;
fq->q.fragments = next;
fq->q.meat -= free_it->len;
frag_kfree_skb(free_it);
}
}
NFCT_FRAG6_CB(skb)->offset = offset; NFCT_FRAG6_CB(skb)->offset = offset;
...@@ -371,6 +319,8 @@ static int nf_ct_frag6_queue(struct nf_ct_frag6_queue *fq, struct sk_buff *skb, ...@@ -371,6 +319,8 @@ static int nf_ct_frag6_queue(struct nf_ct_frag6_queue *fq, struct sk_buff *skb,
write_unlock(&nf_frags.lock); write_unlock(&nf_frags.lock);
return 0; return 0;
discard_fq:
fq_kill(fq);
err: err:
return -1; return -1;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment