2005-04-16 15:20:36 -07:00
/* -*- linux-c -*- ------------------------------------------------------- *
*
* Copyright 2002 H . Peter Anvin - All Rights Reserved
*
* This program 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 , Inc . , 53 Temple Place Ste 330 ,
2009-03-31 14:57:37 +11:00
* Boston MA 02111 - 1307 , USA ; either version 2 of the License , or
2005-04-16 15:20:36 -07:00
* ( at your option ) any later version ; incorporated herein by reference .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
/*
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 [ ] = {
# if defined(__ia64__)
& raid6_intx16 ,
& raid6_intx32 ,
# endif
2007-10-29 04:31:16 +00:00
# if defined(__i386__) && !defined(__arch_um__)
2005-04-16 15:20:36 -07:00
& raid6_mmxx1 ,
& raid6_mmxx2 ,
& raid6_sse1x1 ,
& raid6_sse1x2 ,
& raid6_sse2x1 ,
& raid6_sse2x2 ,
2012-11-30 13:10:39 -08:00
# ifdef CONFIG_AS_AVX2
& raid6_avx2x1 ,
& raid6_avx2x2 ,
# endif
2005-04-16 15:20:36 -07:00
# endif
2007-10-29 04:31:16 +00:00
# if defined(__x86_64__) && !defined(__arch_um__)
2005-04-16 15:20:36 -07:00
& raid6_sse2x1 ,
& raid6_sse2x2 ,
& raid6_sse2x4 ,
2012-11-30 13:10:39 -08:00
# ifdef CONFIG_AS_AVX2
& raid6_avx2x1 ,
& raid6_avx2x2 ,
& raid6_avx2x4 ,
# endif
2005-04-16 15:20:36 -07:00
# endif
# ifdef CONFIG_ALTIVEC
& raid6_altivec1 ,
& raid6_altivec2 ,
& raid6_altivec4 ,
& raid6_altivec8 ,
# endif
2012-05-22 13:54:24 +10:00
& raid6_intx1 ,
& raid6_intx2 ,
& raid6_intx4 ,
& raid6_intx8 ,
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 [ ] = {
# if (defined(__i386__) || defined(__x86_64__)) && !defined(__arch_um__)
2012-11-08 13:47:44 -08:00
# ifdef CONFIG_AS_AVX2
& raid6_recov_avx2 ,
# endif
2012-05-22 13:54:18 +10:00
& raid6_recov_ssse3 ,
# 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 ;
printk ( " raid6: using %s recovery algorithm \n " , best - > name ) ;
} else
printk ( " raid6: Yikes! No recovery algorithm found! \n " ) ;
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
{
2012-05-22 13:54:24 +10:00
unsigned long perf , bestperf , j0 , j1 ;
const struct raid6_calls * const * algo ;
const struct raid6_calls * best ;
2005-04-16 15:20:36 -07:00
2012-05-22 13:54:24 +10:00
for ( bestperf = 0 , best = NULL , algo = raid6_algos ; * algo ; algo + + ) {
if ( ! best | | ( * algo ) - > prefer > = best - > prefer ) {
if ( ( * algo ) - > valid & & ! ( * algo ) - > valid ( ) )
continue ;
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 ( ) ;
2012-05-22 13:54:24 +10:00
if ( perf > bestperf ) {
2005-04-16 15:20:36 -07:00
bestperf = perf ;
2012-05-22 13:54:24 +10:00
best = * algo ;
2005-04-16 15:20:36 -07:00
}
printk ( " raid6: %-8s %5ld MB/s \n " , ( * algo ) - > name ,
( perf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 ) ) ;
}
}
2006-06-23 02:05:59 -07:00
if ( best ) {
2005-04-16 15:20:36 -07:00
printk ( " raid6: using algorithm %s (%ld MB/s) \n " ,
best - > name ,
( bestperf * HZ ) > > ( 20 - 16 + RAID6_TIME_JIFFIES_LG2 ) ) ;
2006-06-23 02:05:59 -07:00
raid6_call = * best ;
} else
2005-04-16 15:20:36 -07:00
printk ( " raid6: Yikes! No algorithm found! \n " ) ;
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 ) {
printk ( " raid6: Yikes! No memory available. \n " ) ;
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 " ) ;