2019-05-20 19:08:13 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/* -*- linux-c -*- ------------------------------------------------------- *
*
* Copyright 2002 H . Peter Anvin - All Rights Reserved
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
/*
2010-08-12 06:44:54 +10:00
* raid6 / algos . c
2005-04-16 15:20:36 -07:00
*
* Algorithm list and algorithm selection for RAID - 6
*/
2009-03-31 15:09:39 +11:00
# include <linux/raid/pq.h>
2005-04-16 15:20:36 -07:00
# ifndef __KERNEL__
# include <sys/mman.h>
2005-09-16 19:27:29 -07:00
# include <stdio.h>
2009-03-31 15:09:39 +11:00
# else
2012-05-22 13:54:16 +10:00
# include <linux/module.h>
2010-08-12 06:38:24 +10:00
# include <linux/gfp.h>
2009-03-31 15:09:39 +11:00
# if !RAID6_USE_EMPTY_ZERO_PAGE
/* In .bss so it's zeroed */
const char raid6_empty_zero_page [ PAGE_SIZE ] __attribute__ ( ( aligned ( 256 ) ) ) ;
EXPORT_SYMBOL ( raid6_empty_zero_page ) ;
# endif
2005-04-16 15:20:36 -07:00
# endif
struct raid6_calls raid6_call ;
2009-03-31 15:09:39 +11:00
EXPORT_SYMBOL_GPL ( raid6_call ) ;
2005-04-16 15:20:36 -07:00
const struct raid6_calls * const raid6_algos [ ] = {
2007-10-29 04:31:16 +00:00
# if defined(__i386__) && !defined(__arch_um__)
2016-08-12 18:03:19 -07:00
# ifdef CONFIG_AS_AVX512
& raid6_avx512x2 ,
2018-11-12 15:26:51 -08:00
& raid6_avx512x1 ,
2016-08-12 18:03:19 -07:00
# endif
2012-11-30 13:10:39 -08:00
# ifdef CONFIG_AS_AVX2
& raid6_avx2x2 ,
2018-11-12 15:26:51 -08:00
& raid6_avx2x1 ,
2012-11-30 13:10:39 -08:00
# endif
2018-11-12 15:26:51 -08:00
& raid6_sse2x2 ,
& raid6_sse2x1 ,
& raid6_sse1x2 ,
& raid6_sse1x1 ,
& raid6_mmxx2 ,
& raid6_mmxx1 ,
# endif
# if defined(__x86_64__) && !defined(__arch_um__)
2016-08-12 18:03:19 -07:00
# ifdef CONFIG_AS_AVX512
& raid6_avx512x4 ,
2018-11-12 15:26:51 -08:00
& raid6_avx512x2 ,
& raid6_avx512x1 ,
# endif
# ifdef CONFIG_AS_AVX2
& raid6_avx2x4 ,
& raid6_avx2x2 ,
& raid6_avx2x1 ,
2016-08-12 18:03:19 -07:00
# endif
2018-11-12 15:26:51 -08:00
& raid6_sse2x4 ,
& raid6_sse2x2 ,
& raid6_sse2x1 ,
2005-04-16 15:20:36 -07:00
# endif
# ifdef CONFIG_ALTIVEC
2017-08-04 13:42:32 +10:00
& raid6_vpermxor8 ,
2018-11-12 15:26:51 -08:00
& raid6_vpermxor4 ,
& raid6_vpermxor2 ,
& raid6_vpermxor1 ,
& raid6_altivec8 ,
& raid6_altivec4 ,
& raid6_altivec2 ,
& raid6_altivec1 ,
2013-08-07 12:39:56 -04:00
# endif
2016-08-23 13:30:24 +02:00
# if defined(CONFIG_S390)
& raid6_s390vx8 ,
2005-04-16 15:20:36 -07:00
# endif
2013-05-16 17:20:32 +02:00
# ifdef CONFIG_KERNEL_MODE_NEON
& raid6_neonx8 ,
2018-11-12 15:26:51 -08:00
& raid6_neonx4 ,
& raid6_neonx2 ,
& raid6_neonx1 ,
# endif
# if defined(__ia64__)
& raid6_intx32 ,
& raid6_intx16 ,
2013-05-16 17:20:32 +02:00
# endif
2018-11-12 15:26:51 -08:00
& raid6_intx8 ,
& raid6_intx4 ,
& raid6_intx2 ,
& raid6_intx1 ,
2005-04-16 15:20:36 -07:00
NULL
} ;
2012-05-22 13:54:18 +10:00
void ( * raid6_2data_recov ) ( int , size_t , int , int , void * * ) ;
EXPORT_SYMBOL_GPL ( raid6_2data_recov ) ;
void ( * raid6_datap_recov ) ( int , size_t , int , void * * ) ;
EXPORT_SYMBOL_GPL ( raid6_datap_recov ) ;
const struct raid6_recov_calls * const raid6_recov_algos [ ] = {
2016-08-12 18:03:20 -07:00
# ifdef CONFIG_AS_AVX512
& raid6_recov_avx512 ,
# endif
2012-11-08 13:47:44 -08:00
# ifdef CONFIG_AS_AVX2
& raid6_recov_avx2 ,
# endif
2015-01-23 08:29:50 +00:00
# ifdef CONFIG_AS_SSSE3
2012-05-22 13:54:18 +10:00
& raid6_recov_ssse3 ,
2016-08-31 09:27:35 +02:00
# endif
# ifdef CONFIG_S390
& raid6_recov_s390xc ,
2017-07-13 18:16:01 +01:00
# endif
# if defined(CONFIG_KERNEL_MODE_NEON)
& raid6_recov_neon ,
2012-05-22 13:54:18 +10:00
# endif
& raid6_recov_intx1 ,
NULL
} ;
2005-04-16 15:20:36 -07:00
# ifdef __KERNEL__
# define RAID6_TIME_JIFFIES_LG2 4
# else
/* Need more time to be stable in userspace */
# define RAID6_TIME_JIFFIES_LG2 9
2009-03-31 15:09:39 +11:00
# define time_before(x, y) ((x) < (y))
2005-04-16 15:20:36 -07:00
# endif
2012-05-22 13:54:24 +10:00
static inline const struct raid6_recov_calls * raid6_choose_recov ( void )
2012-05-22 13:54:18 +10:00
{
const struct raid6_recov_calls * const * algo ;
const struct raid6_recov_calls * best ;
for ( best = NULL , algo = raid6_recov_algos ; * algo ; algo + + )
if ( ! best | | ( * algo ) - > priority > best - > priority )
if ( ! ( * algo ) - > valid | | ( * algo ) - > valid ( ) )
best = * algo ;
if ( best ) {
raid6_2data_recov = best - > data2 ;
raid6_datap_recov = best - > datap ;
2014-10-13 23:03:16 +11:00
pr_info ( " raid6: using %s recovery algorithm \n " , best - > name ) ;
2012-05-22 13:54:18 +10:00
} else
2014-10-13 23:03:16 +11:00
pr_err ( " raid6: Yikes! No recovery algorithm found! \n " ) ;
2012-05-22 13:54:18 +10:00
2012-05-22 13:54:24 +10:00
return best ;
}
2005-04-16 15:20:36 -07:00
2012-05-22 13:54:24 +10:00
static inline const struct raid6_calls * raid6_choose_gen (
void * ( * const dptrs ) [ ( 65536 / PAGE_SIZE ) + 2 ] , const int disks )
2005-04-16 15:20:36 -07:00
{
2014-12-15 12:57:04 +11:00
unsigned long perf , bestgenperf , bestxorperf , j0 , j1 ;
int start = ( disks > > 1 ) - 1 , stop = disks - 3 ; /* work on the second half of the disks */
2012-05-22 13:54:24 +10:00
const struct raid6_calls * const * algo ;
const struct raid6_calls * best ;
2005-04-16 15:20:36 -07:00
2014-12-15 12:57:04 +11:00
for ( bestgenperf = 0 , bestxorperf = 0 , best = NULL , algo = raid6_algos ; * algo ; algo + + ) {
2012-05-22 13:54:24 +10:00
if ( ! best | | ( * algo ) - > prefer > = best - > prefer ) {
if ( ( * algo ) - > valid & & ! ( * algo ) - > valid ( ) )
continue ;
2005-04-16 15:20:36 -07:00
2018-11-12 15:26:52 -08:00
if ( ! IS_ENABLED ( CONFIG_RAID6_PQ_BENCHMARK ) ) {
best = * algo ;
break ;
}
2005-04-16 15:20:36 -07:00
perf = 0 ;
preempt_disable ( ) ;
j0 = jiffies ;
2012-05-22 13:54:24 +10:00
while ( ( j1 = jiffies ) = = j0 )
2005-04-16 15:20:36 -07:00
cpu_relax ( ) ;
2008-04-28 02:15:56 -07:00
while ( time_before ( jiffies ,
j1 + ( 1 < < RAID6_TIME_JIFFIES_LG2 ) ) ) {
2012-05-22 13:54:24 +10:00
( * algo ) - > gen_syndrome ( disks , PAGE_SIZE , * dptrs ) ;
2005-04-16 15:20:36 -07:00
perf + + ;
}
preempt_enable ( ) ;
2014-12-15 12:57:04 +11:00
if ( perf > bestgenperf ) {
bestgenperf = perf ;
2012-05-22 13:54:24 +10:00
best = * algo ;
2005-04-16 15:20:36 -07:00
}
2014-12-15 12:57:04 +11:00
pr_info ( " raid6: %-8s gen() %5ld MB/s \n " , ( * algo ) - > name ,
2005-04-16 15:20:36 -07:00
( perf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 ) ) ;
2014-12-15 12:57:04 +11:00
if ( ! ( * algo ) - > xor_syndrome )
continue ;
perf = 0 ;
preempt_disable ( ) ;
j0 = jiffies ;
while ( ( j1 = jiffies ) = = j0 )
cpu_relax ( ) ;
while ( time_before ( jiffies ,
j1 + ( 1 < < RAID6_TIME_JIFFIES_LG2 ) ) ) {
( * algo ) - > xor_syndrome ( disks , start , stop ,
PAGE_SIZE , * dptrs ) ;
perf + + ;
}
preempt_enable ( ) ;
if ( best = = * algo )
bestxorperf = perf ;
pr_info ( " raid6: %-8s xor() %5ld MB/s \n " , ( * algo ) - > name ,
( perf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 + 1 ) ) ;
2005-04-16 15:20:36 -07:00
}
}
2006-06-23 02:05:59 -07:00
if ( best ) {
2014-12-15 12:57:04 +11:00
pr_info ( " raid6: using algorithm %s gen() %ld MB/s \n " ,
2005-04-16 15:20:36 -07:00
best - > name ,
2014-12-15 12:57:04 +11:00
( bestgenperf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 ) ) ;
if ( best - > xor_syndrome )
pr_info ( " raid6: .... xor() %ld MB/s, rmw enabled \n " ,
( bestxorperf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 + 1 ) ) ;
2006-06-23 02:05:59 -07:00
raid6_call = * best ;
} else
2014-10-13 23:03:16 +11:00
pr_err ( " raid6: Yikes! No algorithm found! \n " ) ;
2005-04-16 15:20:36 -07:00
2012-05-22 13:54:24 +10:00
return best ;
}
/* Try to pick the best algorithm */
/* This code uses the gfmul table as convenient data set to abuse */
int __init raid6_select_algo ( void )
{
const int disks = ( 65536 / PAGE_SIZE ) + 2 ;
const struct raid6_calls * gen_best ;
const struct raid6_recov_calls * rec_best ;
char * syndromes ;
void * dptrs [ ( 65536 / PAGE_SIZE ) + 2 ] ;
int i ;
for ( i = 0 ; i < disks - 2 ; i + + )
dptrs [ i ] = ( ( char * ) raid6_gfmul ) + PAGE_SIZE * i ;
/* Normal code - use a 2-page allocation to avoid D$ conflict */
syndromes = ( void * ) __get_free_pages ( GFP_KERNEL , 1 ) ;
if ( ! syndromes ) {
2014-10-13 23:03:16 +11:00
pr_err ( " raid6: Yikes! No memory available. \n " ) ;
2012-05-22 13:54:24 +10:00
return - ENOMEM ;
}
dptrs [ disks - 2 ] = syndromes ;
dptrs [ disks - 1 ] = syndromes + PAGE_SIZE ;
/* select raid gen_syndrome function */
gen_best = raid6_choose_gen ( & dptrs , disks ) ;
2005-04-16 15:20:36 -07:00
2012-05-22 13:54:18 +10:00
/* select raid recover functions */
2012-05-22 13:54:24 +10:00
rec_best = raid6_choose_recov ( ) ;
free_pages ( ( unsigned long ) syndromes , 1 ) ;
2012-05-22 13:54:18 +10:00
2012-05-22 13:54:24 +10:00
return gen_best & & rec_best ? 0 : - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2009-03-31 15:09:39 +11:00
static void raid6_exit ( void )
{
do { } while ( 0 ) ;
}
subsys_initcall ( raid6_select_algo ) ;
module_exit ( raid6_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
2009-12-14 12:49:58 +11:00
MODULE_DESCRIPTION ( " RAID6 Q-syndrome calculations " ) ;