2007-12-17 06:06:41 +08:00
|
|
|
/* SCTP kernel reference Implementation
|
|
|
|
* Copyright (c) 1999-2001 Motorola, Inc.
|
|
|
|
* Copyright (c) 2001-2003 International Business Machines, Corp.
|
|
|
|
*
|
|
|
|
* This file is part of the SCTP kernel reference Implementation
|
|
|
|
*
|
|
|
|
* SCTP Checksum functions
|
|
|
|
*
|
|
|
|
* The SCTP reference implementation 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, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* The SCTP reference implementation is distributed in the hope that it
|
|
|
|
* will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
|
|
* ************************
|
|
|
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
* See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with GNU CC; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* Please send any bug reports or fixes you make to the
|
|
|
|
* email address(es):
|
2013-07-23 20:51:47 +08:00
|
|
|
* lksctp developers <linux-sctp@vger.kernel.org>
|
2007-12-17 06:06:41 +08:00
|
|
|
*
|
|
|
|
* Written or modified by:
|
|
|
|
* Dinakaran Joseph
|
|
|
|
* Jon Grimm <jgrimm@us.ibm.com>
|
|
|
|
* Sridhar Samudrala <sri@us.ibm.com>
|
|
|
|
*
|
|
|
|
* Rewritten to use libcrc32c by:
|
|
|
|
* Vlad Yasevich <vladislav.yasevich@hp.com>
|
|
|
|
*/
|
|
|
|
|
2013-07-02 00:10:36 +08:00
|
|
|
#ifndef __sctp_checksum_h__
|
|
|
|
#define __sctp_checksum_h__
|
|
|
|
|
2007-12-17 06:06:41 +08:00
|
|
|
#include <linux/types.h>
|
|
|
|
#include <net/sctp/sctp.h>
|
|
|
|
#include <linux/crc32c.h>
|
|
|
|
|
2009-02-13 16:33:42 +08:00
|
|
|
static inline __u32 sctp_crc32c(__u32 crc, u8 *buffer, u16 length)
|
2007-12-17 06:06:41 +08:00
|
|
|
{
|
2009-02-13 16:33:42 +08:00
|
|
|
return crc32c(crc, buffer, length);
|
2008-07-19 14:07:09 +08:00
|
|
|
}
|
|
|
|
|
2009-02-13 16:33:42 +08:00
|
|
|
static inline __u32 sctp_start_cksum(__u8 *buffer, __u16 length)
|
2008-07-19 14:07:09 +08:00
|
|
|
{
|
2009-02-13 16:33:42 +08:00
|
|
|
__u32 crc = ~(__u32)0;
|
2007-12-17 06:06:41 +08:00
|
|
|
__u8 zero[sizeof(__u32)] = {0};
|
|
|
|
|
|
|
|
/* Optimize this routine to be SCTP specific, knowing how
|
|
|
|
* to skip the checksum field of the SCTP header.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Calculate CRC up to the checksum. */
|
2008-07-19 14:07:09 +08:00
|
|
|
crc = sctp_crc32c(crc, buffer, sizeof(struct sctphdr) - sizeof(__u32));
|
2007-12-17 06:06:41 +08:00
|
|
|
|
|
|
|
/* Skip checksum field of the header. */
|
2008-07-19 14:07:09 +08:00
|
|
|
crc = sctp_crc32c(crc, zero, sizeof(__u32));
|
2007-12-17 06:06:41 +08:00
|
|
|
|
|
|
|
/* Calculate the rest of the CRC. */
|
2008-07-19 14:07:09 +08:00
|
|
|
crc = sctp_crc32c(crc, &buffer[sizeof(struct sctphdr)],
|
2007-12-17 06:06:41 +08:00
|
|
|
length - sizeof(struct sctphdr));
|
|
|
|
return crc;
|
|
|
|
}
|
|
|
|
|
2009-02-13 16:33:42 +08:00
|
|
|
static inline __u32 sctp_update_cksum(__u8 *buffer, __u16 length, __u32 crc32)
|
2007-12-17 06:06:41 +08:00
|
|
|
{
|
2008-07-19 14:07:09 +08:00
|
|
|
return sctp_crc32c(crc32, buffer, length);
|
2007-12-17 06:06:41 +08:00
|
|
|
}
|
|
|
|
|
2013-04-19 09:54:58 +08:00
|
|
|
static inline __le32 sctp_end_cksum(__u32 crc32)
|
2007-12-17 06:06:41 +08:00
|
|
|
{
|
2009-02-13 16:33:42 +08:00
|
|
|
return cpu_to_le32(~crc32);
|
2007-12-17 06:06:41 +08:00
|
|
|
}
|
2013-07-02 00:10:36 +08:00
|
|
|
|
2013-07-25 09:52:05 +08:00
|
|
|
/* Calculate the CRC32C checksum of an SCTP packet. */
|
|
|
|
static inline __le32 sctp_compute_cksum(const struct sk_buff *skb,
|
|
|
|
unsigned int offset)
|
|
|
|
{
|
|
|
|
const struct sk_buff *iter;
|
|
|
|
|
|
|
|
__u32 crc32 = sctp_start_cksum(skb->data + offset,
|
|
|
|
skb_headlen(skb) - offset);
|
|
|
|
skb_walk_frags(skb, iter)
|
|
|
|
crc32 = sctp_update_cksum((__u8 *) iter->data,
|
|
|
|
skb_headlen(iter), crc32);
|
|
|
|
|
|
|
|
return sctp_end_cksum(crc32);
|
|
|
|
}
|
|
|
|
|
2013-07-02 00:10:36 +08:00
|
|
|
#endif /* __sctp_checksum_h__ */
|