2018-05-03 22:13:13 +03:00
/*
* Copyright ( C ) 2018 Red Hat , Inc . All rights reserved .
*
* This file is part of LVM2 .
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
* of the GNU General Public License v .2 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software Foundation ,
* Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
*/
# define _GNU_SOURCE
2018-05-12 00:02:10 +03:00
# include "bcache.h"
# include "framework.h"
# include "units.h"
2018-05-03 22:13:13 +03:00
# include <errno.h>
# include <stdio.h>
# include <stdlib.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <fcntl.h>
# include <unistd.h>
2018-05-12 00:02:10 +03:00
# include <sys/statvfs.h>
2018-05-03 22:13:13 +03:00
//----------------------------------------------------------------
# define T_BLOCK_SIZE 4096
# define NR_BLOCKS 64
# define INIT_PATTERN 123
struct fixture {
int fd ;
char fname [ 32 ] ;
struct bcache * cache ;
} ;
static inline uint8_t _pattern_at ( uint8_t pat , uint8_t byte )
{
return pat + byte ;
}
static uint64_t byte ( block_address b , uint64_t offset )
{
return b * T_BLOCK_SIZE + offset ;
}
2018-05-10 16:29:26 +03:00
static void * _fix_init ( struct io_engine * engine )
2018-05-03 22:13:13 +03:00
{
uint8_t buffer [ T_BLOCK_SIZE ] ;
struct fixture * f = malloc ( sizeof ( * f ) ) ;
unsigned b , i ;
2018-05-12 00:02:10 +03:00
struct statvfs fsdata ;
static int _runs_is_tmpfs = - 1 ;
if ( _runs_is_tmpfs = = - 1 ) {
// With testing in tmpfs directory O_DIRECT cannot be used
// tmpfs has f_fsid == 0 (unsure if this is best guess)
_runs_is_tmpfs = ( statvfs ( " . " , & fsdata ) = = 0 & & ! fsdata . f_fsid ) ? 1 : 0 ;
if ( _runs_is_tmpfs )
printf ( " Running test in tmpfs, *NOT* using O_DIRECT \n " ) ;
}
2018-05-03 22:13:13 +03:00
T_ASSERT ( f ) ;
snprintf ( f - > fname , sizeof ( f - > fname ) , " unit-test-XXXXXX " ) ;
2018-05-13 00:23:54 +03:00
f - > fd = mkstemp ( f - > fname ) ;
2018-05-03 22:13:13 +03:00
T_ASSERT ( f - > fd > = 0 ) ;
for ( b = 0 ; b < NR_BLOCKS ; b + + ) {
for ( i = 0 ; i < sizeof ( buffer ) ; i + + )
buffer [ i ] = _pattern_at ( INIT_PATTERN , byte ( b , i ) ) ;
T_ASSERT ( write ( f - > fd , buffer , T_BLOCK_SIZE ) > 0 ) ;
}
2018-05-12 00:02:10 +03:00
if ( ! _runs_is_tmpfs ) {
close ( f - > fd ) ;
// reopen with O_DIRECT
f - > fd = open ( f - > fname , O_RDWR | O_DIRECT ) ;
T_ASSERT ( f - > fd > = 0 ) ;
}
2018-05-03 22:13:13 +03:00
f - > cache = bcache_create ( T_BLOCK_SIZE / 512 , NR_BLOCKS , engine ) ;
T_ASSERT ( f - > cache ) ;
return f ;
}
2018-05-10 16:29:26 +03:00
static void * _async_init ( void )
{
struct io_engine * e = create_async_io_engine ( ) ;
T_ASSERT ( e ) ;
return _fix_init ( e ) ;
}
static void * _sync_init ( void )
{
struct io_engine * e = create_sync_io_engine ( ) ;
T_ASSERT ( e ) ;
return _fix_init ( e ) ;
}
2018-05-03 22:13:13 +03:00
static void _fix_exit ( void * fixture )
{
struct fixture * f = fixture ;
bcache_destroy ( f - > cache ) ;
close ( f - > fd ) ;
unlink ( f - > fname ) ;
free ( f ) ;
}
//----------------------------------------------------------------
static void _verify_bytes ( struct block * b , uint64_t base ,
uint64_t offset , uint64_t len , uint8_t pat )
{
unsigned i ;
for ( i = 0 ; i < len ; i + + )
T_ASSERT_EQUAL ( ( ( uint8_t * ) b - > data ) [ offset + i ] , _pattern_at ( pat , base + offset + i ) ) ;
}
static uint64_t _min ( uint64_t lhs , uint64_t rhs )
{
return rhs < lhs ? rhs : lhs ;
}
static void _verify ( struct fixture * f , uint64_t byte_b , uint64_t byte_e , uint8_t pat )
{
struct block * b ;
block_address bb = byte_b / T_BLOCK_SIZE ;
block_address be = ( byte_e + T_BLOCK_SIZE - 1 ) / T_BLOCK_SIZE ;
uint64_t offset = byte_b % T_BLOCK_SIZE ;
uint64_t blen , len = byte_e - byte_b ;
// Verify via bcache_read_bytes
{
unsigned i ;
size_t len2 = byte_e - byte_b ;
uint8_t * buffer = malloc ( len2 ) ;
T_ASSERT ( bcache_read_bytes ( f - > cache , f - > fd , byte_b , len2 , buffer ) ) ;
for ( i = 0 ; i < len ; i + + )
T_ASSERT_EQUAL ( buffer [ i ] , _pattern_at ( pat , byte_b + i ) ) ;
2018-05-10 16:29:26 +03:00
free ( buffer ) ;
2018-05-03 22:13:13 +03:00
}
// Verify again, driving bcache directly
for ( ; bb ! = be ; bb + + ) {
2018-05-10 15:26:08 +03:00
T_ASSERT ( bcache_get ( f - > cache , f - > fd , bb , 0 , & b ) ) ;
2018-05-03 22:13:13 +03:00
blen = _min ( T_BLOCK_SIZE - offset , len ) ;
_verify_bytes ( b , bb * T_BLOCK_SIZE , offset , blen , pat ) ;
offset = 0 ;
len - = blen ;
bcache_put ( b ) ;
}
}
2018-05-09 13:05:29 +03:00
static void _verify_set ( struct fixture * f , uint64_t byte_b , uint64_t byte_e , uint8_t val )
2018-05-03 22:13:13 +03:00
{
unsigned i ;
struct block * b ;
block_address bb = byte_b / T_BLOCK_SIZE ;
block_address be = ( byte_e + T_BLOCK_SIZE - 1 ) / T_BLOCK_SIZE ;
uint64_t offset = byte_b % T_BLOCK_SIZE ;
uint64_t blen , len = byte_e - byte_b ;
for ( ; bb ! = be ; bb + + ) {
2018-05-10 15:26:08 +03:00
T_ASSERT ( bcache_get ( f - > cache , f - > fd , bb , 0 , & b ) ) ;
2018-05-03 22:13:13 +03:00
blen = _min ( T_BLOCK_SIZE - offset , len ) ;
for ( i = 0 ; i < blen ; i + + )
2018-05-09 13:05:29 +03:00
T_ASSERT ( ( ( uint8_t * ) b - > data ) [ offset + i ] = = val ) ;
2018-05-03 22:13:13 +03:00
offset = 0 ;
len - = blen ;
bcache_put ( b ) ;
}
}
2018-05-09 13:05:29 +03:00
static void _verify_zeroes ( struct fixture * f , uint64_t byte_b , uint64_t byte_e )
{
_verify_set ( f , byte_b , byte_e , 0 ) ;
}
2018-05-03 22:13:13 +03:00
static void _do_write ( struct fixture * f , uint64_t byte_b , uint64_t byte_e , uint8_t pat )
{
unsigned i ;
size_t len = byte_e - byte_b ;
uint8_t * buffer = malloc ( len ) ;
T_ASSERT ( buffer ) ;
for ( i = 0 ; i < len ; i + + )
buffer [ i ] = _pattern_at ( pat , byte_b + i ) ;
2018-05-09 12:50:31 +03:00
T_ASSERT ( bcache_write_bytes ( f - > cache , f - > fd , byte_b , byte_e - byte_b , buffer ) ) ;
2018-05-03 22:13:13 +03:00
free ( buffer ) ;
}
static void _do_zero ( struct fixture * f , uint64_t byte_b , uint64_t byte_e )
{
2018-05-09 12:50:31 +03:00
T_ASSERT ( bcache_zero_bytes ( f - > cache , f - > fd , byte_b , byte_e - byte_b ) ) ;
2018-05-03 22:13:13 +03:00
}
2018-05-09 13:05:29 +03:00
static void _do_set ( struct fixture * f , uint64_t byte_b , uint64_t byte_e , uint8_t val )
{
T_ASSERT ( bcache_set_bytes ( f - > cache , f - > fd , byte_b , byte_e - byte_b , val ) ) ;
}
2018-05-03 22:13:13 +03:00
static void _reopen ( struct fixture * f )
{
struct io_engine * engine ;
bcache_destroy ( f - > cache ) ;
engine = create_async_io_engine ( ) ;
T_ASSERT ( engine ) ;
f - > cache = bcache_create ( T_BLOCK_SIZE / 512 , NR_BLOCKS , engine ) ;
T_ASSERT ( f - > cache ) ;
}
//----------------------------------------------------------------
static uint8_t _random_pattern ( void )
{
return random ( ) ;
}
static uint64_t _max_byte ( void )
{
return T_BLOCK_SIZE * NR_BLOCKS ;
}
static void _rwv_cycle ( struct fixture * f , uint64_t b , uint64_t e )
{
uint8_t pat = _random_pattern ( ) ;
_verify ( f , b , e , INIT_PATTERN ) ;
_do_write ( f , b , e , pat ) ;
_reopen ( f ) ;
_verify ( f , b < 128 ? 0 : b - 128 , b , INIT_PATTERN ) ;
_verify ( f , b , e , pat ) ;
_verify ( f , e , _min ( e + 128 , _max_byte ( ) ) , INIT_PATTERN ) ;
}
static void _test_rw_first_block ( void * fixture )
{
_rwv_cycle ( fixture , byte ( 0 , 0 ) , byte ( 0 , T_BLOCK_SIZE ) ) ;
}
static void _test_rw_last_block ( void * fixture )
{
uint64_t last_block = NR_BLOCKS - 1 ;
_rwv_cycle ( fixture , byte ( last_block , 0 ) ,
byte ( last_block , T_BLOCK_SIZE ) ) ;
}
static void _test_rw_several_whole_blocks ( void * fixture )
{
_rwv_cycle ( fixture , byte ( 5 , 0 ) , byte ( 10 , 0 ) ) ;
}
static void _test_rw_within_single_block ( void * fixture )
{
_rwv_cycle ( fixture , byte ( 7 , 3 ) , byte ( 7 , T_BLOCK_SIZE / 2 ) ) ;
}
static void _test_rw_cross_one_boundary ( void * fixture )
{
_rwv_cycle ( fixture , byte ( 13 , 43 ) , byte ( 14 , 43 ) ) ;
}
static void _test_rw_many_boundaries ( void * fixture )
{
_rwv_cycle ( fixture , byte ( 13 , 13 ) , byte ( 23 , 13 ) ) ;
}
//----------------------------------------------------------------
static void _zero_cycle ( struct fixture * f , uint64_t b , uint64_t e )
{
_verify ( f , b , e , INIT_PATTERN ) ;
_do_zero ( f , b , e ) ;
_reopen ( f ) ;
_verify ( f , b < 128 ? 0 : b - 128 , b , INIT_PATTERN ) ;
_verify_zeroes ( f , b , e ) ;
_verify ( f , e , _min ( e + 128 , _max_byte ( ) ) , INIT_PATTERN ) ;
}
static void _test_zero_first_block ( void * fixture )
{
_zero_cycle ( fixture , byte ( 0 , 0 ) , byte ( 0 , T_BLOCK_SIZE ) ) ;
}
static void _test_zero_last_block ( void * fixture )
{
uint64_t last_block = NR_BLOCKS - 1 ;
_zero_cycle ( fixture , byte ( last_block , 0 ) , byte ( last_block , T_BLOCK_SIZE ) ) ;
}
static void _test_zero_several_whole_blocks ( void * fixture )
{
_zero_cycle ( fixture , byte ( 5 , 0 ) , byte ( 10 , 0 ) ) ;
}
static void _test_zero_within_single_block ( void * fixture )
{
_zero_cycle ( fixture , byte ( 7 , 3 ) , byte ( 7 , T_BLOCK_SIZE / 2 ) ) ;
}
static void _test_zero_cross_one_boundary ( void * fixture )
{
_zero_cycle ( fixture , byte ( 13 , 43 ) , byte ( 14 , 43 ) ) ;
}
static void _test_zero_many_boundaries ( void * fixture )
{
_zero_cycle ( fixture , byte ( 13 , 13 ) , byte ( 23 , 13 ) ) ;
}
//----------------------------------------------------------------
2018-05-09 13:05:29 +03:00
static void _set_cycle ( struct fixture * f , uint64_t b , uint64_t e )
{
uint8_t val = random ( ) ;
_verify ( f , b , e , INIT_PATTERN ) ;
_do_set ( f , b , e , val ) ;
_reopen ( f ) ;
_verify ( f , b < 128 ? 0 : b - 128 , b , INIT_PATTERN ) ;
_verify_set ( f , b , e , val ) ;
_verify ( f , e , _min ( e + 128 , _max_byte ( ) ) , INIT_PATTERN ) ;
}
static void _test_set_first_block ( void * fixture )
{
_set_cycle ( fixture , byte ( 0 , 0 ) , byte ( 0 , T_BLOCK_SIZE ) ) ;
}
static void _test_set_last_block ( void * fixture )
{
uint64_t last_block = NR_BLOCKS - 1 ;
_set_cycle ( fixture , byte ( last_block , 0 ) , byte ( last_block , T_BLOCK_SIZE ) ) ;
}
static void _test_set_several_whole_blocks ( void * fixture )
{
_set_cycle ( fixture , byte ( 5 , 0 ) , byte ( 10 , 0 ) ) ;
}
static void _test_set_within_single_block ( void * fixture )
{
_set_cycle ( fixture , byte ( 7 , 3 ) , byte ( 7 , T_BLOCK_SIZE / 2 ) ) ;
}
static void _test_set_cross_one_boundary ( void * fixture )
{
_set_cycle ( fixture , byte ( 13 , 43 ) , byte ( 14 , 43 ) ) ;
}
static void _test_set_many_boundaries ( void * fixture )
{
_set_cycle ( fixture , byte ( 13 , 13 ) , byte ( 23 , 13 ) ) ;
}
//----------------------------------------------------------------
2018-05-10 16:29:26 +03:00
# define T(path, desc, fn) register_test(ts, " / base / device / bcache / utils / async / " path, desc, fn)
static struct test_suite * _async_tests ( void )
{
struct test_suite * ts = test_suite_create ( _async_init , _fix_exit ) ;
if ( ! ts ) {
fprintf ( stderr , " out of memory \n " ) ;
exit ( 1 ) ;
}
# define T(path, desc, fn) register_test(ts, " / base / device / bcache / utils / async / " path, desc, fn)
T ( " rw-first-block " , " read/write/verify the first block " , _test_rw_first_block ) ;
T ( " rw-last-block " , " read/write/verify the last block " , _test_rw_last_block ) ;
T ( " rw-several-blocks " , " read/write/verify several whole blocks " , _test_rw_several_whole_blocks ) ;
T ( " rw-within-single-block " , " read/write/verify within single block " , _test_rw_within_single_block ) ;
T ( " rw-cross-one-boundary " , " read/write/verify across one boundary " , _test_rw_cross_one_boundary ) ;
T ( " rw-many-boundaries " , " read/write/verify many boundaries " , _test_rw_many_boundaries ) ;
T ( " zero-first-block " , " zero the first block " , _test_zero_first_block ) ;
T ( " zero-last-block " , " zero the last block " , _test_zero_last_block ) ;
T ( " zero-several-blocks " , " zero several whole blocks " , _test_zero_several_whole_blocks ) ;
T ( " zero-within-single-block " , " zero within single block " , _test_zero_within_single_block ) ;
T ( " zero-cross-one-boundary " , " zero across one boundary " , _test_zero_cross_one_boundary ) ;
T ( " zero-many-boundaries " , " zero many boundaries " , _test_zero_many_boundaries ) ;
T ( " set-first-block " , " set the first block " , _test_set_first_block ) ;
T ( " set-last-block " , " set the last block " , _test_set_last_block ) ;
T ( " set-several-blocks " , " set several whole blocks " , _test_set_several_whole_blocks ) ;
T ( " set-within-single-block " , " set within single block " , _test_set_within_single_block ) ;
T ( " set-cross-one-boundary " , " set across one boundary " , _test_set_cross_one_boundary ) ;
T ( " set-many-boundaries " , " set many boundaries " , _test_set_many_boundaries ) ;
# undef T
return ts ;
}
2018-05-03 22:13:13 +03:00
2018-05-10 16:29:26 +03:00
static struct test_suite * _sync_tests ( void )
2018-05-03 22:13:13 +03:00
{
2018-05-10 16:29:26 +03:00
struct test_suite * ts = test_suite_create ( _sync_init , _fix_exit ) ;
2018-05-03 22:13:13 +03:00
if ( ! ts ) {
fprintf ( stderr , " out of memory \n " ) ;
exit ( 1 ) ;
}
2018-05-10 16:29:26 +03:00
# define T(path, desc, fn) register_test(ts, " / base / device / bcache / utils / sync / " path, desc, fn)
2018-05-03 22:13:13 +03:00
T ( " rw-first-block " , " read/write/verify the first block " , _test_rw_first_block ) ;
T ( " rw-last-block " , " read/write/verify the last block " , _test_rw_last_block ) ;
T ( " rw-several-blocks " , " read/write/verify several whole blocks " , _test_rw_several_whole_blocks ) ;
T ( " rw-within-single-block " , " read/write/verify within single block " , _test_rw_within_single_block ) ;
T ( " rw-cross-one-boundary " , " read/write/verify across one boundary " , _test_rw_cross_one_boundary ) ;
T ( " rw-many-boundaries " , " read/write/verify many boundaries " , _test_rw_many_boundaries ) ;
T ( " zero-first-block " , " zero the first block " , _test_zero_first_block ) ;
T ( " zero-last-block " , " zero the last block " , _test_zero_last_block ) ;
T ( " zero-several-blocks " , " zero several whole blocks " , _test_zero_several_whole_blocks ) ;
T ( " zero-within-single-block " , " zero within single block " , _test_zero_within_single_block ) ;
T ( " zero-cross-one-boundary " , " zero across one boundary " , _test_zero_cross_one_boundary ) ;
T ( " zero-many-boundaries " , " zero many boundaries " , _test_zero_many_boundaries ) ;
2018-05-09 13:05:29 +03:00
T ( " set-first-block " , " set the first block " , _test_set_first_block ) ;
T ( " set-last-block " , " set the last block " , _test_set_last_block ) ;
T ( " set-several-blocks " , " set several whole blocks " , _test_set_several_whole_blocks ) ;
T ( " set-within-single-block " , " set within single block " , _test_set_within_single_block ) ;
T ( " set-cross-one-boundary " , " set across one boundary " , _test_set_cross_one_boundary ) ;
T ( " set-many-boundaries " , " set many boundaries " , _test_set_many_boundaries ) ;
2018-05-10 16:29:26 +03:00
# undef T
2018-05-09 13:05:29 +03:00
2018-05-03 22:13:13 +03:00
return ts ;
}
void bcache_utils_tests ( struct dm_list * all_tests )
{
2018-05-10 16:29:26 +03:00
dm_list_add ( all_tests , & _async_tests ( ) - > list ) ;
dm_list_add ( all_tests , & _sync_tests ( ) - > list ) ;
2018-05-03 22:13:13 +03:00
}