Re: [PATCH] [mdadm] add crc32c and use it for r5l checksum

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Thu, Oct 29 2015, Song Liu wrote:

> In kernel space, r5l checksum will use crc32c:
> http://marc.info/?l=linux-raid&m=144598970529191
> mdadm need to change too.
>
> This patch ports a simplified crc32c algorithm from kernel code,
> and used in super1.c:write_empty_r5l_meta_block();
>
> Signed-off-by: Song Liu <songliubraving@xxxxxx>
> Signed-off-by: Shaohua Li <shli@xxxxxx>


applied, thanks.

NeilBrown

> ---
>  Makefile |   2 +-
>  crc32c.c | 104 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
>  super1.c |  11 +++----
>  3 files changed, 109 insertions(+), 8 deletions(-)
>  create mode 100644 crc32c.c
>
> diff --git a/Makefile b/Makefile
> index 544e6cb..fde2e63 100644
> --- a/Makefile
> +++ b/Makefile
> @@ -133,7 +133,7 @@ OBJS =  mdadm.o config.o policy.o mdstat.o  ReadMe.o util.o maps.o lib.o \
>  	mdopen.o super0.o super1.o super-ddf.o super-intel.o bitmap.o \
>  	super-mbr.o super-gpt.o \
>  	restripe.o sysfs.o sha1.o mapfile.o crc32.o sg_io.o msg.o xmalloc.o \
> -	platform-intel.o probe_roms.o
> +	platform-intel.o probe_roms.o crc32c.o
>  
>  CHECK_OBJS = restripe.o sysfs.o maps.o lib.o xmalloc.o dlink.o
>  
> diff --git a/crc32c.c b/crc32c.c
> new file mode 100644
> index 0000000..156cba1
> --- /dev/null
> +++ b/crc32c.c
> @@ -0,0 +1,104 @@
> +/*
> + * Oct 28, 2015 Song Liu simplified the code and port it to mdadm
> + *
> + * Aug 8, 2011 Bob Pearson with help from Joakim Tjernlund and George Spelvin
> + * cleaned up code to current version of sparse and added the slicing-by-8
> + * algorithm to the closely similar existing slicing-by-4 algorithm.
> + *
> + * Oct 15, 2000 Matt Domsch <Matt_Domsch@xxxxxxxx>
> + * Nicer crc32 functions/docs submitted by linux@xxxxxxxxxxx.  Thanks!
> + * Code was from the public domain, copyright abandoned.  Code was
> + * subsequently included in the kernel, thus was re-licensed under the
> + * GNU GPL v2.
> + *
> + * Oct 12, 2000 Matt Domsch <Matt_Domsch@xxxxxxxx>
> + * Same crc32 function was used in 5 other places in the kernel.
> + * I made one version, and deleted the others.
> + * There are various incantations of crc32().  Some use a seed of 0 or ~0.
> + * Some xor at the end with ~0.  The generic crc32() function takes
> + * seed as an argument, and doesn't xor at the end.  Then individual
> + * users can do whatever they need.
> + *   drivers/net/smc9194.c uses seed ~0, doesn't xor with ~0.
> + *   fs/jffs2 uses seed 0, doesn't xor with ~0.
> + *   fs/partitions/efi.c uses seed ~0, xor's with ~0.
> + *
> + * This source code is licensed under the GNU General Public License,
> + * Version 2.  See the file COPYING for more details.
> + */
> +
> +#include <sys/types.h>
> +#include <asm/types.h>
> +#include <stdlib.h>
> +
> +/*
> + * There are multiple 16-bit CRC polynomials in common use, but this is
> + * *the* standard CRC-32 polynomial, first popularized by Ethernet.
> + * x^32+x^26+x^23+x^22+x^16+x^12+x^11+x^10+x^8+x^7+x^5+x^4+x^2+x^1+x^0
> + */
> +#define CRCPOLY_LE 0xedb88320
> +#define CRCPOLY_BE 0x04c11db7
> +
> +/*
> + * This is the CRC32c polynomial, as outlined by Castagnoli.
> + * x^32+x^28+x^27+x^26+x^25+x^23+x^22+x^20+x^19+x^18+x^14+x^13+x^11+x^10+x^9+
> + * x^8+x^6+x^0
> + */
> +#define CRC32C_POLY_LE 0x82F63B78
> +
> +/**
> + * crc32_le_generic() - Calculate bitwise little-endian Ethernet AUTODIN II
> + *			CRC32/CRC32C
> + * @crc: seed value for computation.  ~0 for Ethernet, sometimes 0 for other
> + *	 uses, or the previous crc32/crc32c value if computing incrementally.
> + * @p: pointer to buffer over which CRC32/CRC32C is run
> + * @len: length of buffer @p
> + * @polynomial: CRC32/CRC32c LE polynomial
> + */
> +static inline __u32 crc32_le_generic(__u32 crc, unsigned char const *p,
> +				     size_t len, __u32 polynomial)
> +{
> +	int i;
> +	while (len--) {
> +		crc ^= *p++;
> +		for (i = 0; i < 8; i++)
> +			crc = (crc >> 1) ^ ((crc & 1) ? polynomial : 0);
> +	}
> +	return crc;
> +}
> +
> +__u32 crc32_le(__u32 crc, unsigned char const *p, size_t len)
> +{
> +	return crc32_le_generic(crc, p, len, CRCPOLY_LE);
> +}
> +
> +__u32 crc32c_le(__u32 crc, unsigned char const *p, size_t len)
> +{
> +	return crc32_le_generic(crc, p, len, CRC32C_POLY_LE);
> +}
> +
> +/**
> + * crc32_be_generic() - Calculate bitwise big-endian Ethernet AUTODIN II CRC32
> + * @crc: seed value for computation.  ~0 for Ethernet, sometimes 0 for
> + *	other uses, or the previous crc32 value if computing incrementally.
> + * @p: pointer to buffer over which CRC32 is run
> + * @len: length of buffer @p
> + * @polynomial: CRC32 BE polynomial
> + */
> +static inline __u32 crc32_be_generic(__u32 crc, unsigned char const *p,
> +				     size_t len, __u32 polynomial)
> +{
> +	int i;
> +	while (len--) {
> +		crc ^= *p++ << 24;
> +		for (i = 0; i < 8; i++)
> +			crc =
> +			    (crc << 1) ^ ((crc & 0x80000000) ? polynomial :
> +					  0);
> +	}
> +	return crc;
> +}
> +
> +__u32 crc32_be(__u32 crc, unsigned char const *p, size_t len)
> +{
> +	return crc32_be_generic(crc, p, len, CRCPOLY_BE);
> +}
> diff --git a/super1.c b/super1.c
> index 58e6f9d..1735c2d 100644
> --- a/super1.c
> +++ b/super1.c
> @@ -1625,10 +1625,7 @@ static unsigned long choose_bm_space(unsigned long devsize)
>  static void free_super1(struct supertype *st);
>  
>  #define META_BLOCK_SIZE 4096
> -unsigned long crc32(
> -	unsigned long crc,
> -	const unsigned char *buf,
> -	unsigned len);
> +__u32 crc32c_le(__u32 crc, unsigned char const *p, size_t len);
>  
>  static int write_empty_r5l_meta_block(struct supertype *st, int fd)
>  {
> @@ -1652,9 +1649,9 @@ static int write_empty_r5l_meta_block(struct supertype *st, int fd)
>  	mb->seq = __cpu_to_le64(random32());
>  	mb->position = __cpu_to_le64(0);
>  
> -	crc = crc32(0xffffffff, sb->set_uuid, sizeof(sb->set_uuid));
> -	crc = crc32(crc, (void *)mb, META_BLOCK_SIZE);
> -	mb->checksum = __cpu_to_le32(crc);
> +	crc = crc32c_le(0xffffffff, sb->set_uuid, sizeof(sb->set_uuid));
> +	crc = crc32c_le(crc, (void *)mb, META_BLOCK_SIZE);
> +	mb->checksum = crc;
>  
>  	if (lseek64(fd, (sb->data_offset) * 512, 0) < 0LL) {
>  		pr_err("cannot seek to offset of the meta block\n");
> -- 
> 2.4.6

Attachment: signature.asc
Description: PGP signature


[Index of Archives]     [Linux RAID Wiki]     [ATA RAID]     [Linux SCSI Target Infrastructure]     [Linux Block]     [Linux IDE]     [Linux SCSI]     [Linux Hams]     [Device Mapper]     [Device Mapper Cryptographics]     [Kernel]     [Linux Admin]     [Linux Net]     [GFS]     [RPM]     [git]     [Yosemite Forum]


  Powered by Linux