2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2012-10-10 23:07:19 +05:30
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-10-22 16:53:33 +09:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/list.h>
# include <linux/random.h>
# include <linux/string.h>
# include <linux/bitops.h>
2012-09-03 22:00:01 +09:00
# include <linux/slab.h>
2009-10-22 16:53:33 +09:00
# include <linux/mtd/nand_ecc.h>
2015-03-29 21:52:06 +02:00
# include "mtd_test.h"
2012-09-08 01:48:07 +09:00
/*
* Test the implementation for software ECC
*
* No actual MTD device is needed , So we don ' t need to warry about losing
* important data by human error .
*
* This covers possible patterns of corruption which can be reliably corrected
* or detected .
*/
2019-02-06 16:47:44 +01:00
# if IS_ENABLED(CONFIG_MTD_RAW_NAND)
2009-10-22 16:53:33 +09:00
2012-09-08 01:48:07 +09:00
struct nand_ecc_test {
const char * name ;
void ( * prepare ) ( void * , void * , void * , void * , const size_t ) ;
int ( * verify ) ( void * , void * , void * , const size_t ) ;
} ;
2012-09-08 01:48:06 +09:00
/*
* The reason for this __change_bit_le ( ) instead of __change_bit ( ) is to inject
* bit error properly within the region which is not a multiple of
* sizeof ( unsigned long ) on big - endian systems
*/
# ifdef __LITTLE_ENDIAN
# define __change_bit_le(nr, addr) __change_bit(nr, addr)
# elif defined(__BIG_ENDIAN)
# define __change_bit_le(nr, addr) \
__change_bit ( ( nr ) ^ ( ( BITS_PER_LONG - 1 ) & ~ 0x7 ) , addr )
# else
# error "Unknown byte order"
# endif
2012-09-08 01:48:07 +09:00
static void single_bit_error_data ( void * error_data , void * correct_data ,
size_t size )
2009-10-22 16:53:33 +09:00
{
2013-01-03 21:19:13 +09:00
unsigned int offset = prandom_u32 ( ) % ( size * BITS_PER_BYTE ) ;
2009-10-22 16:53:33 +09:00
2012-09-08 01:48:07 +09:00
memcpy ( error_data , correct_data , size ) ;
__change_bit_le ( offset , error_data ) ;
}
2012-09-08 01:48:10 +09:00
static void double_bit_error_data ( void * error_data , void * correct_data ,
size_t size )
{
unsigned int offset [ 2 ] ;
2013-01-03 21:19:13 +09:00
offset [ 0 ] = prandom_u32 ( ) % ( size * BITS_PER_BYTE ) ;
2012-09-08 01:48:10 +09:00
do {
2013-01-03 21:19:13 +09:00
offset [ 1 ] = prandom_u32 ( ) % ( size * BITS_PER_BYTE ) ;
2012-09-08 01:48:10 +09:00
} while ( offset [ 0 ] = = offset [ 1 ] ) ;
memcpy ( error_data , correct_data , size ) ;
__change_bit_le ( offset [ 0 ] , error_data ) ;
__change_bit_le ( offset [ 1 ] , error_data ) ;
}
2012-09-08 01:48:09 +09:00
static unsigned int random_ecc_bit ( size_t size )
{
2013-01-03 21:19:13 +09:00
unsigned int offset = prandom_u32 ( ) % ( 3 * BITS_PER_BYTE ) ;
2012-09-08 01:48:09 +09:00
if ( size = = 256 ) {
/*
* Don ' t inject a bit error into the insignificant bits ( 16 th
* and 17 th bit ) in ECC code for 256 byte data block
*/
while ( offset = = 16 | | offset = = 17 )
2013-01-03 21:19:13 +09:00
offset = prandom_u32 ( ) % ( 3 * BITS_PER_BYTE ) ;
2012-09-08 01:48:09 +09:00
}
return offset ;
}
static void single_bit_error_ecc ( void * error_ecc , void * correct_ecc ,
size_t size )
{
unsigned int offset = random_ecc_bit ( size ) ;
memcpy ( error_ecc , correct_ecc , 3 ) ;
__change_bit_le ( offset , error_ecc ) ;
}
2012-09-08 01:48:10 +09:00
static void double_bit_error_ecc ( void * error_ecc , void * correct_ecc ,
size_t size )
{
unsigned int offset [ 2 ] ;
offset [ 0 ] = random_ecc_bit ( size ) ;
do {
offset [ 1 ] = random_ecc_bit ( size ) ;
} while ( offset [ 0 ] = = offset [ 1 ] ) ;
memcpy ( error_ecc , correct_ecc , 3 ) ;
__change_bit_le ( offset [ 0 ] , error_ecc ) ;
__change_bit_le ( offset [ 1 ] , error_ecc ) ;
}
2012-09-08 01:48:08 +09:00
static void no_bit_error ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
memcpy ( error_data , correct_data , size ) ;
memcpy ( error_ecc , correct_ecc , 3 ) ;
}
static int no_bit_error_verify ( void * error_data , void * error_ecc ,
void * correct_data , const size_t size )
{
unsigned char calc_ecc [ 3 ] ;
int ret ;
2018-09-04 16:23:28 +02:00
__nand_calculate_ecc ( error_data , size , calc_ecc ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2018-09-04 16:23:28 +02:00
ret = __nand_correct_data ( error_data , error_ecc , calc_ecc , size ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2012-09-08 01:48:08 +09:00
if ( ret = = 0 & & ! memcmp ( correct_data , error_data , size ) )
return 0 ;
return - EINVAL ;
}
2012-09-08 01:48:07 +09:00
static void single_bit_error_in_data ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
single_bit_error_data ( error_data , correct_data , size ) ;
memcpy ( error_ecc , correct_ecc , 3 ) ;
}
2012-09-08 01:48:09 +09:00
static void single_bit_error_in_ecc ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
memcpy ( error_data , correct_data , size ) ;
single_bit_error_ecc ( error_ecc , correct_ecc , size ) ;
}
2012-09-08 01:48:07 +09:00
static int single_bit_error_correct ( void * error_data , void * error_ecc ,
void * correct_data , const size_t size )
{
unsigned char calc_ecc [ 3 ] ;
int ret ;
2018-09-04 16:23:28 +02:00
__nand_calculate_ecc ( error_data , size , calc_ecc ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2018-09-04 16:23:28 +02:00
ret = __nand_correct_data ( error_data , error_ecc , calc_ecc , size ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2012-09-08 01:48:07 +09:00
if ( ret = = 1 & & ! memcmp ( correct_data , error_data , size ) )
return 0 ;
return - EINVAL ;
2009-10-22 16:53:33 +09:00
}
2012-09-08 01:48:10 +09:00
static void double_bit_error_in_data ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
double_bit_error_data ( error_data , correct_data , size ) ;
memcpy ( error_ecc , correct_ecc , 3 ) ;
}
static void single_bit_error_in_data_and_ecc ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
single_bit_error_data ( error_data , correct_data , size ) ;
single_bit_error_ecc ( error_ecc , correct_ecc , size ) ;
}
static void double_bit_error_in_ecc ( void * error_data , void * error_ecc ,
void * correct_data , void * correct_ecc , const size_t size )
{
memcpy ( error_data , correct_data , size ) ;
double_bit_error_ecc ( error_ecc , correct_ecc , size ) ;
}
static int double_bit_error_detect ( void * error_data , void * error_ecc ,
void * correct_data , const size_t size )
{
unsigned char calc_ecc [ 3 ] ;
int ret ;
2018-09-04 16:23:28 +02:00
__nand_calculate_ecc ( error_data , size , calc_ecc ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2018-09-04 16:23:28 +02:00
ret = __nand_correct_data ( error_data , error_ecc , calc_ecc , size ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2012-09-08 01:48:10 +09:00
2016-03-01 16:04:00 -05:00
return ( ret = = - EBADMSG ) ? 0 : - EINVAL ;
2012-09-08 01:48:10 +09:00
}
2012-09-08 01:48:07 +09:00
static const struct nand_ecc_test nand_ecc_test [ ] = {
2012-09-08 01:48:08 +09:00
{
. name = " no-bit-error " ,
. prepare = no_bit_error ,
. verify = no_bit_error_verify ,
} ,
2012-09-08 01:48:07 +09:00
{
. name = " single-bit-error-in-data-correct " ,
. prepare = single_bit_error_in_data ,
. verify = single_bit_error_correct ,
} ,
2012-09-08 01:48:09 +09:00
{
. name = " single-bit-error-in-ecc-correct " ,
. prepare = single_bit_error_in_ecc ,
. verify = single_bit_error_correct ,
} ,
2012-09-08 01:48:10 +09:00
{
. name = " double-bit-error-in-data-detect " ,
. prepare = double_bit_error_in_data ,
. verify = double_bit_error_detect ,
} ,
{
. name = " single-bit-error-in-data-and-ecc-detect " ,
. prepare = single_bit_error_in_data_and_ecc ,
. verify = double_bit_error_detect ,
} ,
{
. name = " double-bit-error-in-ecc-detect " ,
. prepare = double_bit_error_in_ecc ,
. verify = double_bit_error_detect ,
} ,
2012-09-08 01:48:07 +09:00
} ;
2012-09-03 22:00:00 +09:00
static void dump_data_ecc ( void * error_data , void * error_ecc , void * correct_data ,
void * correct_ecc , const size_t size )
{
pr_info ( " hexdump of error data: \n " ) ;
print_hex_dump ( KERN_INFO , " " , DUMP_PREFIX_OFFSET , 16 , 4 ,
error_data , size , false ) ;
print_hex_dump ( KERN_INFO , " hexdump of error ecc: " ,
DUMP_PREFIX_NONE , 16 , 1 , error_ecc , 3 , false ) ;
pr_info ( " hexdump of correct data: \n " ) ;
print_hex_dump ( KERN_INFO , " " , DUMP_PREFIX_OFFSET , 16 , 4 ,
correct_data , size , false ) ;
print_hex_dump ( KERN_INFO , " hexdump of correct ecc: " ,
DUMP_PREFIX_NONE , 16 , 1 , correct_ecc , 3 , false ) ;
}
2012-09-08 01:48:07 +09:00
static int nand_ecc_test_run ( const size_t size )
2009-10-22 16:53:33 +09:00
{
2012-09-08 01:48:07 +09:00
int i ;
2012-09-03 22:00:01 +09:00
int err = 0 ;
void * error_data ;
void * error_ecc ;
void * correct_data ;
void * correct_ecc ;
2009-10-22 16:53:33 +09:00
2012-09-03 22:00:01 +09:00
error_data = kmalloc ( size , GFP_KERNEL ) ;
error_ecc = kmalloc ( 3 , GFP_KERNEL ) ;
correct_data = kmalloc ( size , GFP_KERNEL ) ;
correct_ecc = kmalloc ( 3 , GFP_KERNEL ) ;
if ( ! error_data | | ! error_ecc | | ! correct_data | | ! correct_ecc ) {
err = - ENOMEM ;
goto error ;
}
2009-10-22 16:53:33 +09:00
2013-02-27 17:05:31 -08:00
prandom_bytes ( correct_data , size ) ;
2018-09-04 16:23:28 +02:00
__nand_calculate_ecc ( correct_data , size , correct_ecc ,
2019-02-08 08:48:37 +01:00
IS_ENABLED ( CONFIG_MTD_NAND_ECC_SW_HAMMING_SMC ) ) ;
2012-09-08 01:48:07 +09:00
for ( i = 0 ; i < ARRAY_SIZE ( nand_ecc_test ) ; i + + ) {
nand_ecc_test [ i ] . prepare ( error_data , error_ecc ,
correct_data , correct_ecc , size ) ;
err = nand_ecc_test [ i ] . verify ( error_data , error_ecc ,
correct_data , size ) ;
if ( err ) {
2012-10-10 23:07:19 +05:30
pr_err ( " not ok - %s-%zd \n " ,
2012-09-08 01:48:07 +09:00
nand_ecc_test [ i ] . name , size ) ;
dump_data_ecc ( error_data , error_ecc ,
correct_data , correct_ecc , size ) ;
break ;
}
2012-10-10 23:07:19 +05:30
pr_info ( " ok - %s-%zd \n " ,
2012-09-08 01:48:07 +09:00
nand_ecc_test [ i ] . name , size ) ;
2015-03-29 21:52:06 +02:00
err = mtdtest_relax ( ) ;
if ( err )
break ;
2009-10-22 16:53:33 +09:00
}
2012-09-03 22:00:01 +09:00
error :
kfree ( error_data ) ;
kfree ( error_ecc ) ;
kfree ( correct_data ) ;
kfree ( correct_ecc ) ;
return err ;
2009-10-22 16:53:33 +09:00
}
# else
2012-09-08 01:48:07 +09:00
static int nand_ecc_test_run ( const size_t size )
2009-10-22 16:53:33 +09:00
{
return 0 ;
}
# endif
static int __init ecc_test_init ( void )
{
2012-08-26 21:06:44 +09:00
int err ;
2009-10-22 16:53:33 +09:00
2012-09-08 01:48:07 +09:00
err = nand_ecc_test_run ( 256 ) ;
2012-08-26 21:06:44 +09:00
if ( err )
return err ;
2012-09-08 01:48:07 +09:00
return nand_ecc_test_run ( 512 ) ;
2009-10-22 16:53:33 +09:00
}
static void __exit ecc_test_exit ( void )
{
}
module_init ( ecc_test_init ) ;
module_exit ( ecc_test_exit ) ;
MODULE_DESCRIPTION ( " NAND ECC function test module " ) ;
MODULE_AUTHOR ( " Akinobu Mita " ) ;
MODULE_LICENSE ( " GPL " ) ;