csum_tcpudp_magic should not be called to compute checksum for non-TCP/UDP packets, like ICMP with wrong checksum Fixes: 5d1549847c76 ("netfilter: Fix remainder of pseudo-header protocol 0") Suggested-by: Pablo Neira Ayuso <pablo@xxxxxxxxxxxxx> Signed-off-by: Li RongQing <lirongqing@xxxxxxxxx> --- diff with v1: rewrite the code as suggested add fixes tag net/netfilter/utils.c | 21 +++++++++++++++------ 1 file changed, 15 insertions(+), 6 deletions(-) diff --git a/net/netfilter/utils.c b/net/netfilter/utils.c index 51b454d8fa9c..0f78416566fa 100644 --- a/net/netfilter/utils.c +++ b/net/netfilter/utils.c @@ -17,12 +17,21 @@ __sum16 nf_ip_checksum(struct sk_buff *skb, unsigned int hook, case CHECKSUM_COMPLETE: if (hook != NF_INET_PRE_ROUTING && hook != NF_INET_LOCAL_IN) break; - if ((protocol != IPPROTO_TCP && protocol != IPPROTO_UDP && - !csum_fold(skb->csum)) || - !csum_tcpudp_magic(iph->saddr, iph->daddr, - skb->len - dataoff, protocol, - skb->csum)) { - skb->ip_summed = CHECKSUM_UNNECESSARY; + switch (protocol) { + case IPPROTO_TCP: + case IPPROTO_UDP: + if (!csum_tcpudp_magic(iph->saddr, iph->daddr, + skb->len - dataoff, + protocol, skb->csum)) { + skb->ip_summed = CHECKSUM_UNNECESSARY; + return 0; + } + break; + default: + if (!csum_fold(skb->csum)) { + skb->ip_summed = CHECKSUM_UNNECESSARY; + return 0; + } break; } /* fall through */ -- 2.16.2