|
@@ -143,8 +143,11 @@ static bool frag_expire_skip_icmp(u32 user)
|
|
|
static void ip_expire(struct timer_list *t)
|
|
|
{
|
|
|
struct inet_frag_queue *frag = from_timer(frag, t, timer);
|
|
|
- struct ipq *qp;
|
|
|
+ struct sk_buff *clone, *head;
|
|
|
+ const struct iphdr *iph;
|
|
|
struct net *net;
|
|
|
+ struct ipq *qp;
|
|
|
+ int err;
|
|
|
|
|
|
qp = container_of(frag, struct ipq, q);
|
|
|
net = container_of(qp->q.net, struct net, ipv4.frags);
|
|
@@ -158,45 +161,41 @@ static void ip_expire(struct timer_list *t)
|
|
|
ipq_kill(qp);
|
|
|
__IP_INC_STATS(net, IPSTATS_MIB_REASMFAILS);
|
|
|
|
|
|
- if (!inet_frag_evicting(&qp->q)) {
|
|
|
- struct sk_buff *clone, *head = qp->q.fragments;
|
|
|
- const struct iphdr *iph;
|
|
|
- int err;
|
|
|
+ head = qp->q.fragments;
|
|
|
|
|
|
- __IP_INC_STATS(net, IPSTATS_MIB_REASMTIMEOUT);
|
|
|
+ __IP_INC_STATS(net, IPSTATS_MIB_REASMTIMEOUT);
|
|
|
|
|
|
- if (!(qp->q.flags & INET_FRAG_FIRST_IN) || !qp->q.fragments)
|
|
|
- goto out;
|
|
|
+ if (!(qp->q.flags & INET_FRAG_FIRST_IN) || !head)
|
|
|
+ goto out;
|
|
|
|
|
|
- head->dev = dev_get_by_index_rcu(net, qp->iif);
|
|
|
- if (!head->dev)
|
|
|
- goto out;
|
|
|
+ head->dev = dev_get_by_index_rcu(net, qp->iif);
|
|
|
+ if (!head->dev)
|
|
|
+ goto out;
|
|
|
|
|
|
|
|
|
- /* skb has no dst, perform route lookup again */
|
|
|
- iph = ip_hdr(head);
|
|
|
- err = ip_route_input_noref(head, iph->daddr, iph->saddr,
|
|
|
+ /* skb has no dst, perform route lookup again */
|
|
|
+ iph = ip_hdr(head);
|
|
|
+ err = ip_route_input_noref(head, iph->daddr, iph->saddr,
|
|
|
iph->tos, head->dev);
|
|
|
- if (err)
|
|
|
- goto out;
|
|
|
+ if (err)
|
|
|
+ goto out;
|
|
|
|
|
|
- /* Only an end host needs to send an ICMP
|
|
|
- * "Fragment Reassembly Timeout" message, per RFC792.
|
|
|
- */
|
|
|
- if (frag_expire_skip_icmp(qp->q.key.v4.user) &&
|
|
|
- (skb_rtable(head)->rt_type != RTN_LOCAL))
|
|
|
- goto out;
|
|
|
-
|
|
|
- clone = skb_clone(head, GFP_ATOMIC);
|
|
|
-
|
|
|
- /* Send an ICMP "Fragment Reassembly Timeout" message. */
|
|
|
- if (clone) {
|
|
|
- spin_unlock(&qp->q.lock);
|
|
|
- icmp_send(clone, ICMP_TIME_EXCEEDED,
|
|
|
- ICMP_EXC_FRAGTIME, 0);
|
|
|
- consume_skb(clone);
|
|
|
- goto out_rcu_unlock;
|
|
|
- }
|
|
|
+ /* Only an end host needs to send an ICMP
|
|
|
+ * "Fragment Reassembly Timeout" message, per RFC792.
|
|
|
+ */
|
|
|
+ if (frag_expire_skip_icmp(qp->q.key.v4.user) &&
|
|
|
+ (skb_rtable(head)->rt_type != RTN_LOCAL))
|
|
|
+ goto out;
|
|
|
+
|
|
|
+ clone = skb_clone(head, GFP_ATOMIC);
|
|
|
+
|
|
|
+ /* Send an ICMP "Fragment Reassembly Timeout" message. */
|
|
|
+ if (clone) {
|
|
|
+ spin_unlock(&qp->q.lock);
|
|
|
+ icmp_send(clone, ICMP_TIME_EXCEEDED,
|
|
|
+ ICMP_EXC_FRAGTIME, 0);
|
|
|
+ consume_skb(clone);
|
|
|
+ goto out_rcu_unlock;
|
|
|
}
|
|
|
out:
|
|
|
spin_unlock(&qp->q.lock);
|