2017-11-14 18:38:00 +01:00
// SPDX-License-Identifier: GPL-2.0
2005-04-16 15:20:36 -07:00
/*
* dcssblk . c - - the S / 390 block driver for dcss memory
*
* Authors : Carsten Otte , Stefan Weinhuber , Gerald Schaefer
*/
2008-12-25 13:39:45 +01:00
# define KMSG_COMPONENT "dcssblk"
# define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/ctype.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/blkdev.h>
# include <linux/completion.h>
# include <linux/interrupt.h>
2016-01-15 16:56:14 -08:00
# include <linux/pfn_t.h>
2017-06-27 13:06:22 -07:00
# include <linux/uio.h>
2017-01-26 13:54:35 -08:00
# include <linux/dax.h>
2009-06-16 10:30:51 +02:00
# include <asm/extmem.h>
# include <asm/io.h>
2005-04-16 15:20:36 -07:00
# define DCSSBLK_NAME "dcssblk"
# define DCSSBLK_MINORS_PER_DISK 1
# define DCSSBLK_PARM_LEN 400
2008-12-25 13:38:55 +01:00
# define DCSS_BUS_ID_SIZE 20
2005-04-16 15:20:36 -07:00
2008-03-02 10:37:41 -05:00
static int dcssblk_open ( struct block_device * bdev , fmode_t mode ) ;
2013-05-05 21:52:57 -04:00
static void dcssblk_release ( struct gendisk * disk , fmode_t mode ) ;
2021-10-12 13:12:24 +02:00
static void dcssblk_submit_bio ( struct bio * bio ) ;
2017-01-26 13:54:35 -08:00
static long dcssblk_dax_direct_access ( struct dax_device * dax_dev , pgoff_t pgoff ,
2022-05-13 15:10:58 -07:00
long nr_pages , enum dax_access_mode mode , void * * kaddr ,
pfn_t * pfn ) ;
2005-04-16 15:20:36 -07:00
static char dcssblk_segments [ DCSSBLK_PARM_LEN ] = " \0 " ;
static int dcssblk_major ;
2009-09-21 17:01:13 -07:00
static const struct block_device_operations dcssblk_devops = {
2005-06-23 22:05:23 -07:00
. owner = THIS_MODULE ,
2020-07-01 10:59:43 +02:00
. submit_bio = dcssblk_submit_bio ,
2008-03-02 10:37:41 -05:00
. open = dcssblk_open ,
. release = dcssblk_release ,
2017-01-26 13:54:35 -08:00
} ;
2020-02-28 11:34:53 -05:00
static int dcssblk_dax_zero_page_range ( struct dax_device * dax_dev ,
pgoff_t pgoff , size_t nr_pages )
{
long rc ;
void * kaddr ;
2022-05-13 15:10:58 -07:00
rc = dax_direct_access ( dax_dev , pgoff , nr_pages , DAX_ACCESS ,
& kaddr , NULL ) ;
2020-02-28 11:34:53 -05:00
if ( rc < 0 )
return rc ;
memset ( kaddr , 0 , nr_pages < < PAGE_SHIFT ) ;
dax_flush ( dax_dev , kaddr , nr_pages < < PAGE_SHIFT ) ;
return 0 ;
}
2017-01-26 13:54:35 -08:00
static const struct dax_operations dcssblk_dax_ops = {
. direct_access = dcssblk_dax_direct_access ,
2020-02-28 11:34:53 -05:00
. zero_page_range = dcssblk_dax_zero_page_range ,
2005-04-16 15:20:36 -07:00
} ;
2008-10-10 21:33:21 +02:00
struct dcssblk_dev_info {
struct list_head lh ;
struct device dev ;
2008-12-25 13:38:55 +01:00
char segment_name [ DCSS_BUS_ID_SIZE ] ;
2008-10-10 21:33:21 +02:00
atomic_t use_count ;
struct gendisk * gd ;
unsigned long start ;
unsigned long end ;
int segment_type ;
unsigned char save_pending ;
unsigned char is_shared ;
int num_of_segments ;
struct list_head seg_list ;
2017-01-26 13:54:35 -08:00
struct dax_device * dax_dev ;
2008-10-10 21:33:21 +02:00
} ;
struct segment_info {
struct list_head lh ;
2008-12-25 13:38:55 +01:00
char segment_name [ DCSS_BUS_ID_SIZE ] ;
2008-10-10 21:33:21 +02:00
unsigned long start ;
unsigned long end ;
int segment_type ;
} ;
2005-05-17 06:42:58 -04:00
static ssize_t dcssblk_add_store ( struct device * dev , struct device_attribute * attr , const char * buf ,
2005-04-16 15:20:36 -07:00
size_t count ) ;
2005-05-17 06:42:58 -04:00
static ssize_t dcssblk_remove_store ( struct device * dev , struct device_attribute * attr , const char * buf ,
2005-04-16 15:20:36 -07:00
size_t count ) ;
static DEVICE_ATTR ( add , S_IWUSR , NULL , dcssblk_add_store ) ;
static DEVICE_ATTR ( remove , S_IWUSR , NULL , dcssblk_remove_store ) ;
static struct device * dcssblk_root_dev ;
2008-01-26 14:11:13 +01:00
static LIST_HEAD ( dcssblk_devices ) ;
2005-04-16 15:20:36 -07:00
static struct rw_semaphore dcssblk_devices_sem ;
/*
* release function for segment device .
*/
static void
dcssblk_release_segment ( struct device * dev )
{
2008-10-10 21:33:21 +02:00
struct dcssblk_dev_info * dev_info ;
struct segment_info * entry , * temp ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
list_for_each_entry_safe ( entry , temp , & dev_info - > seg_list , lh ) {
list_del ( & entry - > lh ) ;
kfree ( entry ) ;
}
kfree ( dev_info ) ;
2005-04-16 15:20:36 -07:00
module_put ( THIS_MODULE ) ;
}
/*
* get a minor number . needs to be called with
* down_write ( & dcssblk_devices_sem ) and the
* device needs to be enqueued before the semaphore is
* freed .
*/
2007-02-05 21:18:53 +01:00
static int
2005-04-16 15:20:36 -07:00
dcssblk_assign_free_minor ( struct dcssblk_dev_info * dev_info )
{
int minor , found ;
struct dcssblk_dev_info * entry ;
if ( dev_info = = NULL )
return - EINVAL ;
for ( minor = 0 ; minor < ( 1 < < MINORBITS ) ; minor + + ) {
found = 0 ;
// test if minor available
list_for_each_entry ( entry , & dcssblk_devices , lh )
2009-06-12 10:26:35 +02:00
if ( minor = = entry - > gd - > first_minor )
2005-04-16 15:20:36 -07:00
found + + ;
if ( ! found ) break ; // got unused minor
}
if ( found )
return - EBUSY ;
dev_info - > gd - > first_minor = minor ;
return 0 ;
}
/*
* get the struct dcssblk_dev_info from dcssblk_devices
* for the given name .
* down_read ( & dcssblk_devices_sem ) must be held .
*/
static struct dcssblk_dev_info *
dcssblk_get_device_by_name ( char * name )
{
struct dcssblk_dev_info * entry ;
list_for_each_entry ( entry , & dcssblk_devices , lh ) {
if ( ! strcmp ( name , entry - > segment_name ) ) {
return entry ;
}
}
return NULL ;
}
2008-10-10 21:33:21 +02:00
/*
* get the struct segment_info from seg_list
* for the given name .
* down_read ( & dcssblk_devices_sem ) must be held .
*/
static struct segment_info *
dcssblk_get_segment_by_name ( char * name )
{
struct dcssblk_dev_info * dev_info ;
struct segment_info * entry ;
list_for_each_entry ( dev_info , & dcssblk_devices , lh ) {
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
if ( ! strcmp ( name , entry - > segment_name ) )
return entry ;
}
}
return NULL ;
}
/*
* get the highest address of the multi - segment block .
*/
static unsigned long
dcssblk_find_highest_addr ( struct dcssblk_dev_info * dev_info )
{
unsigned long highest_addr ;
struct segment_info * entry ;
highest_addr = 0 ;
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
if ( highest_addr < entry - > end )
highest_addr = entry - > end ;
}
return highest_addr ;
}
/*
* get the lowest address of the multi - segment block .
*/
static unsigned long
dcssblk_find_lowest_addr ( struct dcssblk_dev_info * dev_info )
{
int set_first ;
unsigned long lowest_addr ;
struct segment_info * entry ;
set_first = 0 ;
lowest_addr = 0 ;
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
if ( set_first = = 0 ) {
lowest_addr = entry - > start ;
set_first = 1 ;
} else {
if ( lowest_addr > entry - > start )
lowest_addr = entry - > start ;
}
}
return lowest_addr ;
}
/*
* Check continuity of segments .
*/
static int
dcssblk_is_continuous ( struct dcssblk_dev_info * dev_info )
{
int i , j , rc ;
struct segment_info * sort_list , * entry , temp ;
if ( dev_info - > num_of_segments < = 1 )
return 0 ;
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-12 14:03:40 -07:00
sort_list = kcalloc ( dev_info - > num_of_segments ,
sizeof ( struct segment_info ) ,
GFP_KERNEL ) ;
2008-10-10 21:33:21 +02:00
if ( sort_list = = NULL )
return - ENOMEM ;
i = 0 ;
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
memcpy ( & sort_list [ i ] , entry , sizeof ( struct segment_info ) ) ;
i + + ;
}
/* sort segments */
for ( i = 0 ; i < dev_info - > num_of_segments ; i + + )
for ( j = 0 ; j < dev_info - > num_of_segments ; j + + )
if ( sort_list [ j ] . start > sort_list [ i ] . start ) {
memcpy ( & temp , & sort_list [ i ] ,
sizeof ( struct segment_info ) ) ;
memcpy ( & sort_list [ i ] , & sort_list [ j ] ,
sizeof ( struct segment_info ) ) ;
memcpy ( & sort_list [ j ] , & temp ,
sizeof ( struct segment_info ) ) ;
}
/* check continuity */
for ( i = 0 ; i < dev_info - > num_of_segments - 1 ; i + + ) {
if ( ( sort_list [ i ] . end + 1 ) ! = sort_list [ i + 1 ] . start ) {
2008-12-25 13:39:45 +01:00
pr_err ( " Adjacent DCSSs %s and %s are not "
" contiguous \n " , sort_list [ i ] . segment_name ,
sort_list [ i + 1 ] . segment_name ) ;
2008-10-10 21:33:21 +02:00
rc = - EINVAL ;
goto out ;
}
/* EN and EW are allowed in a block device */
if ( sort_list [ i ] . segment_type ! = sort_list [ i + 1 ] . segment_type ) {
if ( ! ( sort_list [ i ] . segment_type & SEGMENT_EXCLUSIVE ) | |
( sort_list [ i ] . segment_type = = SEG_TYPE_ER ) | |
! ( sort_list [ i + 1 ] . segment_type &
SEGMENT_EXCLUSIVE ) | |
( sort_list [ i + 1 ] . segment_type = = SEG_TYPE_ER ) ) {
2008-12-25 13:39:45 +01:00
pr_err ( " DCSS %s and DCSS %s have "
" incompatible types \n " ,
sort_list [ i ] . segment_name ,
sort_list [ i + 1 ] . segment_name ) ;
2008-10-10 21:33:21 +02:00
rc = - EINVAL ;
goto out ;
}
}
}
rc = 0 ;
out :
kfree ( sort_list ) ;
return rc ;
}
/*
* Load a segment
*/
static int
dcssblk_load_segment ( char * name , struct segment_info * * seg_info )
{
int rc ;
/* already loaded? */
down_read ( & dcssblk_devices_sem ) ;
* seg_info = dcssblk_get_segment_by_name ( name ) ;
up_read ( & dcssblk_devices_sem ) ;
if ( * seg_info ! = NULL )
return - EEXIST ;
/* get a struct segment_info */
* seg_info = kzalloc ( sizeof ( struct segment_info ) , GFP_KERNEL ) ;
if ( * seg_info = = NULL )
return - ENOMEM ;
strcpy ( ( * seg_info ) - > segment_name , name ) ;
/* load the segment */
rc = segment_load ( name , SEGMENT_SHARED ,
& ( * seg_info ) - > start , & ( * seg_info ) - > end ) ;
if ( rc < 0 ) {
segment_warning ( rc , ( * seg_info ) - > segment_name ) ;
kfree ( * seg_info ) ;
} else {
INIT_LIST_HEAD ( & ( * seg_info ) - > lh ) ;
( * seg_info ) - > segment_type = rc ;
}
return rc ;
}
2005-04-16 15:20:36 -07:00
/*
* device attribute for switching shared / nonshared ( exclusive )
* operation ( show + store )
*/
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_shared_show ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
return sprintf ( buf , dev_info - > is_shared ? " 1 \n " : " 0 \n " ) ;
}
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_shared_store ( struct device * dev , struct device_attribute * attr , const char * inbuf , size_t count )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
2008-10-10 21:33:21 +02:00
struct segment_info * entry , * temp ;
2005-04-16 15:20:36 -07:00
int rc ;
2008-07-14 09:59:39 +02:00
if ( ( count > 1 ) & & ( inbuf [ 1 ] ! = ' \n ' ) & & ( inbuf [ 1 ] ! = ' \0 ' ) )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
down_write ( & dcssblk_devices_sem ) ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
if ( atomic_read ( & dev_info - > use_count ) ) {
rc = - EBUSY ;
goto out ;
}
if ( inbuf [ 0 ] = = ' 1 ' ) {
2008-10-10 21:33:21 +02:00
/* reload segments in shared mode */
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
rc = segment_modify_shared ( entry - > segment_name ,
SEGMENT_SHARED ) ;
if ( rc < 0 ) {
BUG_ON ( rc = = - EINVAL ) ;
if ( rc ! = - EAGAIN )
goto removeseg ;
2005-04-16 15:20:36 -07:00
}
}
2008-10-10 21:33:21 +02:00
dev_info - > is_shared = 1 ;
switch ( dev_info - > segment_type ) {
case SEG_TYPE_SR :
case SEG_TYPE_ER :
case SEG_TYPE_SC :
set_disk_ro ( dev_info - > gd , 1 ) ;
}
2005-04-16 15:20:36 -07:00
} else if ( inbuf [ 0 ] = = ' 0 ' ) {
2008-10-10 21:33:21 +02:00
/* reload segments in exclusive mode */
2005-04-16 15:20:36 -07:00
if ( dev_info - > segment_type = = SEG_TYPE_SC ) {
2008-12-25 13:39:45 +01:00
pr_err ( " DCSS %s is of type SC and cannot be "
" loaded as exclusive-writable \n " ,
dev_info - > segment_name ) ;
2005-04-16 15:20:36 -07:00
rc = - EINVAL ;
goto out ;
}
2008-10-10 21:33:21 +02:00
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
rc = segment_modify_shared ( entry - > segment_name ,
SEGMENT_EXCLUSIVE ) ;
if ( rc < 0 ) {
BUG_ON ( rc = = - EINVAL ) ;
if ( rc ! = - EAGAIN )
goto removeseg ;
}
2005-04-16 15:20:36 -07:00
}
2008-10-10 21:33:21 +02:00
dev_info - > is_shared = 0 ;
set_disk_ro ( dev_info - > gd , 0 ) ;
2005-04-16 15:20:36 -07:00
} else {
rc = - EINVAL ;
goto out ;
}
rc = count ;
goto out ;
removeseg :
2008-12-25 13:39:45 +01:00
pr_err ( " DCSS device %s is removed after a failed access mode "
" change \n " , dev_info - > segment_name ) ;
2008-10-10 21:33:21 +02:00
temp = entry ;
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
if ( entry ! = temp )
segment_unload ( entry - > segment_name ) ;
}
2005-04-16 15:20:36 -07:00
list_del ( & dev_info - > lh ) ;
2017-01-26 13:54:35 -08:00
kill_dax ( dev_info - > dax_dev ) ;
put_dax ( dev_info - > dax_dev ) ;
2005-04-16 15:20:36 -07:00
del_gendisk ( dev_info - > gd ) ;
2022-06-19 08:05:52 +02:00
put_disk ( dev_info - > gd ) ;
2014-02-03 14:03:04 -05:00
up_write ( & dcssblk_devices_sem ) ;
if ( device_remove_file_self ( dev , attr ) ) {
device_unregister ( dev ) ;
put_device ( dev ) ;
}
return rc ;
2005-04-16 15:20:36 -07:00
out :
up_write ( & dcssblk_devices_sem ) ;
return rc ;
}
2012-10-01 09:12:01 +02:00
static DEVICE_ATTR ( shared , S_IWUSR | S_IRUSR , dcssblk_shared_show ,
dcssblk_shared_store ) ;
2005-04-16 15:20:36 -07:00
/*
* device attribute for save operation on current copy
* of the segment . If the segment is busy , saving will
* become pending until it gets released , which can be
* undone by storing a non - true value to this entry .
* ( show + store )
*/
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_save_show ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
return sprintf ( buf , dev_info - > save_pending ? " 1 \n " : " 0 \n " ) ;
}
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_save_store ( struct device * dev , struct device_attribute * attr , const char * inbuf , size_t count )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
2008-10-10 21:33:21 +02:00
struct segment_info * entry ;
2005-04-16 15:20:36 -07:00
2008-07-14 09:59:39 +02:00
if ( ( count > 1 ) & & ( inbuf [ 1 ] ! = ' \n ' ) & & ( inbuf [ 1 ] ! = ' \0 ' ) )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
down_write ( & dcssblk_devices_sem ) ;
if ( inbuf [ 0 ] = = ' 1 ' ) {
if ( atomic_read ( & dev_info - > use_count ) = = 0 ) {
// device is idle => we save immediately
2008-12-25 13:39:45 +01:00
pr_info ( " All DCSSs that map to device %s are "
" saved \n " , dev_info - > segment_name ) ;
2008-10-10 21:33:21 +02:00
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
2015-01-16 18:29:04 +01:00
if ( entry - > segment_type = = SEG_TYPE_EN | |
entry - > segment_type = = SEG_TYPE_SN )
pr_warn ( " DCSS %s is of type SN or EN "
" and cannot be saved \n " ,
entry - > segment_name ) ;
else
segment_save ( entry - > segment_name ) ;
2008-10-10 21:33:21 +02:00
}
2005-04-16 15:20:36 -07:00
} else {
// device is busy => we save it when it becomes
// idle in dcssblk_release
2008-12-25 13:39:45 +01:00
pr_info ( " Device %s is in use, its DCSSs will be "
" saved when it becomes idle \n " ,
dev_info - > segment_name ) ;
2005-04-16 15:20:36 -07:00
dev_info - > save_pending = 1 ;
}
} else if ( inbuf [ 0 ] = = ' 0 ' ) {
if ( dev_info - > save_pending ) {
// device is busy & the user wants to undo his save
// request
dev_info - > save_pending = 0 ;
2008-12-25 13:39:45 +01:00
pr_info ( " A pending save request for device %s "
" has been canceled \n " ,
dev_info - > segment_name ) ;
2005-04-16 15:20:36 -07:00
}
} else {
up_write ( & dcssblk_devices_sem ) ;
return - EINVAL ;
}
up_write ( & dcssblk_devices_sem ) ;
return count ;
}
2012-10-01 09:12:01 +02:00
static DEVICE_ATTR ( save , S_IWUSR | S_IRUSR , dcssblk_save_show ,
dcssblk_save_store ) ;
2005-04-16 15:20:36 -07:00
2008-10-10 21:33:21 +02:00
/*
* device attribute for showing all segments in a device
*/
static ssize_t
dcssblk_seglist_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
int i ;
struct dcssblk_dev_info * dev_info ;
struct segment_info * entry ;
down_read ( & dcssblk_devices_sem ) ;
dev_info = container_of ( dev , struct dcssblk_dev_info , dev ) ;
i = 0 ;
buf [ 0 ] = ' \0 ' ;
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
strcpy ( & buf [ i ] , entry - > segment_name ) ;
i + = strlen ( entry - > segment_name ) ;
buf [ i ] = ' \n ' ;
i + + ;
}
up_read ( & dcssblk_devices_sem ) ;
return i ;
}
2012-10-01 09:12:01 +02:00
static DEVICE_ATTR ( seglist , S_IRUSR , dcssblk_seglist_show , NULL ) ;
static struct attribute * dcssblk_dev_attrs [ ] = {
& dev_attr_shared . attr ,
& dev_attr_save . attr ,
& dev_attr_seglist . attr ,
NULL ,
} ;
static struct attribute_group dcssblk_dev_attr_group = {
. attrs = dcssblk_dev_attrs ,
} ;
static const struct attribute_group * dcssblk_dev_attr_groups [ ] = {
& dcssblk_dev_attr_group ,
NULL ,
} ;
2008-10-10 21:33:21 +02:00
2005-04-16 15:20:36 -07:00
/*
* device attribute for adding devices
*/
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_add_store ( struct device * dev , struct device_attribute * attr , const char * buf , size_t count )
2005-04-16 15:20:36 -07:00
{
2008-10-10 21:33:21 +02:00
int rc , i , j , num_of_segments ;
2005-04-16 15:20:36 -07:00
struct dcssblk_dev_info * dev_info ;
2008-10-10 21:33:21 +02:00
struct segment_info * seg_info , * temp ;
2005-04-16 15:20:36 -07:00
char * local_buf ;
unsigned long seg_byte_size ;
dev_info = NULL ;
2008-10-10 21:33:21 +02:00
seg_info = NULL ;
2005-04-16 15:20:36 -07:00
if ( dev ! = dcssblk_root_dev ) {
rc = - EINVAL ;
goto out_nobuf ;
}
2008-10-10 21:33:21 +02:00
if ( ( count < 1 ) | | ( buf [ 0 ] = = ' \0 ' ) | | ( buf [ 0 ] = = ' \n ' ) ) {
rc = - ENAMETOOLONG ;
goto out_nobuf ;
}
2005-04-16 15:20:36 -07:00
local_buf = kmalloc ( count + 1 , GFP_KERNEL ) ;
if ( local_buf = = NULL ) {
rc = - ENOMEM ;
goto out_nobuf ;
}
2008-10-10 21:33:21 +02:00
2005-04-16 15:20:36 -07:00
/*
* parse input
*/
2008-10-10 21:33:21 +02:00
num_of_segments = 0 ;
2015-02-24 18:41:50 +02:00
for ( i = 0 ; ( i < count & & ( buf [ i ] ! = ' \0 ' ) & & ( buf [ i ] ! = ' \n ' ) ) ; i + + ) {
2015-08-19 10:42:23 +02:00
for ( j = i ; j < count & &
( buf [ j ] ! = ' : ' ) & &
2008-10-10 21:33:21 +02:00
( buf [ j ] ! = ' \0 ' ) & &
2015-08-19 10:42:23 +02:00
( buf [ j ] ! = ' \n ' ) ; j + + ) {
2008-10-10 21:33:21 +02:00
local_buf [ j - i ] = toupper ( buf [ j ] ) ;
}
local_buf [ j - i ] = ' \0 ' ;
if ( ( ( j - i ) = = 0 ) | | ( ( j - i ) > 8 ) ) {
rc = - ENAMETOOLONG ;
goto seg_list_del ;
}
rc = dcssblk_load_segment ( local_buf , & seg_info ) ;
if ( rc < 0 )
goto seg_list_del ;
/*
* get a struct dcssblk_dev_info
*/
if ( num_of_segments = = 0 ) {
dev_info = kzalloc ( sizeof ( struct dcssblk_dev_info ) ,
GFP_KERNEL ) ;
if ( dev_info = = NULL ) {
rc = - ENOMEM ;
goto out ;
}
strcpy ( dev_info - > segment_name , local_buf ) ;
dev_info - > segment_type = seg_info - > segment_type ;
INIT_LIST_HEAD ( & dev_info - > seg_list ) ;
}
list_add_tail ( & seg_info - > lh , & dev_info - > seg_list ) ;
num_of_segments + + ;
i = j ;
if ( ( buf [ j ] = = ' \0 ' ) | | ( buf [ j ] = = ' \n ' ) )
break ;
2005-04-16 15:20:36 -07:00
}
2008-10-10 21:33:21 +02:00
/* no trailing colon at the end of the input */
if ( ( i > 0 ) & & ( buf [ i - 1 ] = = ' : ' ) ) {
2005-04-16 15:20:36 -07:00
rc = - ENAMETOOLONG ;
2008-10-10 21:33:21 +02:00
goto seg_list_del ;
2005-04-16 15:20:36 -07:00
}
2022-08-18 22:59:47 +02:00
strscpy ( local_buf , buf , i + 1 ) ;
2008-10-10 21:33:21 +02:00
dev_info - > num_of_segments = num_of_segments ;
rc = dcssblk_is_continuous ( dev_info ) ;
if ( rc < 0 )
goto seg_list_del ;
dev_info - > start = dcssblk_find_lowest_addr ( dev_info ) ;
dev_info - > end = dcssblk_find_highest_addr ( dev_info ) ;
2005-04-16 15:20:36 -07:00
2014-06-10 10:46:20 -07:00
dev_set_name ( & dev_info - > dev , " %s " , dev_info - > segment_name ) ;
2005-04-16 15:20:36 -07:00
dev_info - > dev . release = dcssblk_release_segment ;
2012-10-01 09:12:01 +02:00
dev_info - > dev . groups = dcssblk_dev_attr_groups ;
2005-04-16 15:20:36 -07:00
INIT_LIST_HEAD ( & dev_info - > lh ) ;
2021-05-21 07:51:13 +02:00
dev_info - > gd = blk_alloc_disk ( NUMA_NO_NODE ) ;
2005-04-16 15:20:36 -07:00
if ( dev_info - > gd = = NULL ) {
rc = - ENOMEM ;
2008-10-10 21:33:21 +02:00
goto seg_list_del ;
2005-04-16 15:20:36 -07:00
}
dev_info - > gd - > major = dcssblk_major ;
2021-05-21 07:51:13 +02:00
dev_info - > gd - > minors = DCSSBLK_MINORS_PER_DISK ;
2005-04-16 15:20:36 -07:00
dev_info - > gd - > fops = & dcssblk_devops ;
dev_info - > gd - > private_data = dev_info ;
2021-05-21 07:51:13 +02:00
blk_queue_logical_block_size ( dev_info - > gd - > queue , 4096 ) ;
blk_queue_flag_set ( QUEUE_FLAG_DAX , dev_info - > gd - > queue ) ;
2008-10-10 21:33:21 +02:00
2005-04-16 15:20:36 -07:00
seg_byte_size = ( dev_info - > end - dev_info - > start + 1 ) ;
set_capacity ( dev_info - > gd , seg_byte_size > > 9 ) ; // size in sectors
2008-12-25 13:39:45 +01:00
pr_info ( " Loaded %s with total size %lu bytes and capacity %lu "
" sectors \n " , local_buf , seg_byte_size , seg_byte_size > > 9 ) ;
2005-04-16 15:20:36 -07:00
dev_info - > save_pending = 0 ;
dev_info - > is_shared = 1 ;
dev_info - > dev . parent = dcssblk_root_dev ;
/*
2008-10-10 21:33:21 +02:00
* get minor , add to list
2005-04-16 15:20:36 -07:00
*/
down_write ( & dcssblk_devices_sem ) ;
2008-10-10 21:33:21 +02:00
if ( dcssblk_get_segment_by_name ( local_buf ) ) {
2008-08-21 19:46:40 +02:00
rc = - EEXIST ;
2008-10-10 21:33:21 +02:00
goto release_gd ;
2008-08-21 19:46:40 +02:00
}
2005-04-16 15:20:36 -07:00
rc = dcssblk_assign_free_minor ( dev_info ) ;
2008-10-10 21:33:21 +02:00
if ( rc )
goto release_gd ;
2005-04-16 15:20:36 -07:00
sprintf ( dev_info - > gd - > disk_name , " dcssblk%d " ,
2009-06-12 10:26:35 +02:00
dev_info - > gd - > first_minor ) ;
2005-04-16 15:20:36 -07:00
list_add_tail ( & dev_info - > lh , & dcssblk_devices ) ;
if ( ! try_module_get ( THIS_MODULE ) ) {
rc = - ENODEV ;
2008-10-10 21:33:21 +02:00
goto dev_list_del ;
2005-04-16 15:20:36 -07:00
}
/*
* register the device
*/
rc = device_register ( & dev_info - > dev ) ;
2008-10-10 21:33:21 +02:00
if ( rc )
2012-10-01 09:12:01 +02:00
goto put_dev ;
2005-04-16 15:20:36 -07:00
2021-12-15 09:45:07 +01:00
dev_info - > dax_dev = alloc_dax ( dev_info , & dcssblk_dax_ops ) ;
2020-04-01 12:11:25 -04:00
if ( IS_ERR ( dev_info - > dax_dev ) ) {
rc = PTR_ERR ( dev_info - > dax_dev ) ;
dev_info - > dax_dev = NULL ;
2017-01-26 13:54:35 -08:00
goto put_dev ;
}
2021-12-15 09:45:07 +01:00
set_dax_synchronous ( dev_info - > dax_dev ) ;
2021-11-29 11:21:38 +01:00
rc = dax_add_host ( dev_info - > dax_dev , dev_info - > gd ) ;
if ( rc )
goto out_dax ;
2017-01-26 13:54:35 -08:00
2012-10-01 09:12:01 +02:00
get_device ( & dev_info - > dev ) ;
2021-09-27 15:02:31 -07:00
rc = device_add_disk ( & dev_info - > dev , dev_info - > gd , NULL ) ;
if ( rc )
2021-11-29 11:21:38 +01:00
goto out_dax_host ;
2007-12-04 16:09:03 +01:00
2005-04-16 15:20:36 -07:00
switch ( dev_info - > segment_type ) {
case SEG_TYPE_SR :
case SEG_TYPE_ER :
case SEG_TYPE_SC :
set_disk_ro ( dev_info - > gd , 1 ) ;
break ;
default :
set_disk_ro ( dev_info - > gd , 0 ) ;
break ;
}
up_write ( & dcssblk_devices_sem ) ;
rc = count ;
goto out ;
2021-11-29 11:21:38 +01:00
out_dax_host :
dax_remove_host ( dev_info - > gd ) ;
2021-09-27 15:02:31 -07:00
out_dax :
put_device ( & dev_info - > dev ) ;
kill_dax ( dev_info - > dax_dev ) ;
put_dax ( dev_info - > dax_dev ) ;
2012-10-01 09:12:01 +02:00
put_dev :
2005-04-16 15:20:36 -07:00
list_del ( & dev_info - > lh ) ;
2022-06-19 08:05:52 +02:00
put_disk ( dev_info - > gd ) ;
2008-10-10 21:33:21 +02:00
list_for_each_entry ( seg_info , & dev_info - > seg_list , lh ) {
segment_unload ( seg_info - > segment_name ) ;
}
2005-04-16 15:20:36 -07:00
put_device ( & dev_info - > dev ) ;
up_write ( & dcssblk_devices_sem ) ;
goto out ;
2008-10-10 21:33:21 +02:00
dev_list_del :
2005-04-16 15:20:36 -07:00
list_del ( & dev_info - > lh ) ;
2008-10-10 21:33:21 +02:00
release_gd :
2022-06-19 08:05:52 +02:00
put_disk ( dev_info - > gd ) ;
2008-10-10 21:33:21 +02:00
up_write ( & dcssblk_devices_sem ) ;
seg_list_del :
if ( dev_info = = NULL )
goto out ;
list_for_each_entry_safe ( seg_info , temp , & dev_info - > seg_list , lh ) {
list_del ( & seg_info - > lh ) ;
segment_unload ( seg_info - > segment_name ) ;
kfree ( seg_info ) ;
}
2005-04-16 15:20:36 -07:00
kfree ( dev_info ) ;
out :
kfree ( local_buf ) ;
out_nobuf :
return rc ;
}
/*
* device attribute for removing devices
*/
static ssize_t
2005-05-17 06:42:58 -04:00
dcssblk_remove_store ( struct device * dev , struct device_attribute * attr , const char * buf , size_t count )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
2008-10-10 21:33:21 +02:00
struct segment_info * entry ;
2005-04-16 15:20:36 -07:00
int rc , i ;
char * local_buf ;
if ( dev ! = dcssblk_root_dev ) {
return - EINVAL ;
}
local_buf = kmalloc ( count + 1 , GFP_KERNEL ) ;
if ( local_buf = = NULL ) {
return - ENOMEM ;
}
/*
* parse input
*/
2015-08-19 10:42:23 +02:00
for ( i = 0 ; ( i < count & & ( * ( buf + i ) ! = ' \0 ' ) & & ( * ( buf + i ) ! = ' \n ' ) ) ; i + + ) {
2005-04-16 15:20:36 -07:00
local_buf [ i ] = toupper ( buf [ i ] ) ;
}
local_buf [ i ] = ' \0 ' ;
if ( ( i = = 0 ) | | ( i > 8 ) ) {
rc = - ENAMETOOLONG ;
goto out_buf ;
}
down_write ( & dcssblk_devices_sem ) ;
dev_info = dcssblk_get_device_by_name ( local_buf ) ;
if ( dev_info = = NULL ) {
up_write ( & dcssblk_devices_sem ) ;
2016-03-03 20:49:57 -08:00
pr_warn ( " Device %s cannot be removed because it is not a known device \n " ,
local_buf ) ;
2005-04-16 15:20:36 -07:00
rc = - ENODEV ;
goto out_buf ;
}
if ( atomic_read ( & dev_info - > use_count ) ! = 0 ) {
up_write ( & dcssblk_devices_sem ) ;
2016-03-03 20:49:57 -08:00
pr_warn ( " Device %s cannot be removed while it is in use \n " ,
local_buf ) ;
2005-04-16 15:20:36 -07:00
rc = - EBUSY ;
goto out_buf ;
}
2008-10-10 21:33:21 +02:00
list_del ( & dev_info - > lh ) ;
2017-01-26 13:54:35 -08:00
kill_dax ( dev_info - > dax_dev ) ;
put_dax ( dev_info - > dax_dev ) ;
2005-04-16 15:20:36 -07:00
del_gendisk ( dev_info - > gd ) ;
2022-06-19 08:05:52 +02:00
put_disk ( dev_info - > gd ) ;
2008-10-10 21:33:21 +02:00
/* unload all related segments */
list_for_each_entry ( entry , & dev_info - > seg_list , lh )
segment_unload ( entry - > segment_name ) ;
2005-04-16 15:20:36 -07:00
up_write ( & dcssblk_devices_sem ) ;
2016-04-08 13:23:52 +02:00
device_unregister ( & dev_info - > dev ) ;
put_device ( & dev_info - > dev ) ;
2005-04-16 15:20:36 -07:00
rc = count ;
out_buf :
kfree ( local_buf ) ;
return rc ;
}
static int
2008-03-02 10:37:41 -05:00
dcssblk_open ( struct block_device * bdev , fmode_t mode )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
int rc ;
2008-03-02 10:37:41 -05:00
dev_info = bdev - > bd_disk - > private_data ;
2005-04-16 15:20:36 -07:00
if ( NULL = = dev_info ) {
rc = - ENODEV ;
goto out ;
}
atomic_inc ( & dev_info - > use_count ) ;
rc = 0 ;
out :
return rc ;
}
2013-05-05 21:52:57 -04:00
static void
2008-03-02 10:37:41 -05:00
dcssblk_release ( struct gendisk * disk , fmode_t mode )
2005-04-16 15:20:36 -07:00
{
2008-03-02 10:37:41 -05:00
struct dcssblk_dev_info * dev_info = disk - > private_data ;
2008-10-10 21:33:21 +02:00
struct segment_info * entry ;
2005-04-16 15:20:36 -07:00
2008-03-02 10:37:41 -05:00
if ( ! dev_info ) {
2013-05-05 21:52:57 -04:00
WARN_ON ( 1 ) ;
return ;
2005-04-16 15:20:36 -07:00
}
down_write ( & dcssblk_devices_sem ) ;
if ( atomic_dec_and_test ( & dev_info - > use_count )
& & ( dev_info - > save_pending ) ) {
2008-12-25 13:39:45 +01:00
pr_info ( " Device %s has become idle and is being saved "
" now \n " , dev_info - > segment_name ) ;
2008-10-10 21:33:21 +02:00
list_for_each_entry ( entry , & dev_info - > seg_list , lh ) {
2015-01-16 18:29:04 +01:00
if ( entry - > segment_type = = SEG_TYPE_EN | |
entry - > segment_type = = SEG_TYPE_SN )
pr_warn ( " DCSS %s is of type SN or EN and cannot "
" be saved \n " , entry - > segment_name ) ;
else
segment_save ( entry - > segment_name ) ;
2008-10-10 21:33:21 +02:00
}
2005-04-16 15:20:36 -07:00
dev_info - > save_pending = 0 ;
}
up_write ( & dcssblk_devices_sem ) ;
}
2021-10-12 13:12:24 +02:00
static void
2020-07-01 10:59:43 +02:00
dcssblk_submit_bio ( struct bio * bio )
2005-04-16 15:20:36 -07:00
{
struct dcssblk_dev_info * dev_info ;
2013-11-23 17:19:00 -08:00
struct bio_vec bvec ;
struct bvec_iter iter ;
2005-04-16 15:20:36 -07:00
unsigned long index ;
unsigned long page_addr ;
unsigned long source_addr ;
unsigned long bytes_done ;
2022-07-27 12:22:55 -04:00
bio = bio_split_to_limits ( bio ) ;
2015-04-23 22:37:18 -07:00
2005-04-16 15:20:36 -07:00
bytes_done = 0 ;
2021-01-24 11:02:34 +01:00
dev_info = bio - > bi_bdev - > bd_disk - > private_data ;
2005-04-16 15:20:36 -07:00
if ( dev_info = = NULL )
goto fail ;
2013-10-11 15:44:27 -07:00
if ( ( bio - > bi_iter . bi_sector & 7 ) ! = 0 | |
( bio - > bi_iter . bi_size & 4095 ) ! = 0 )
2005-04-16 15:20:36 -07:00
/* Request is not page-aligned. */
goto fail ;
2005-06-23 22:05:23 -07:00
/* verify data transfer direction */
if ( dev_info - > is_shared ) {
switch ( dev_info - > segment_type ) {
case SEG_TYPE_SR :
case SEG_TYPE_ER :
case SEG_TYPE_SC :
/* cannot write to these segments */
if ( bio_data_dir ( bio ) = = WRITE ) {
2016-03-03 20:49:57 -08:00
pr_warn ( " Writing to %s failed because it is a read-only device \n " ,
dev_name ( & dev_info - > dev ) ) ;
2005-06-23 22:05:23 -07:00
goto fail ;
}
}
}
2013-10-11 15:44:27 -07:00
index = ( bio - > bi_iter . bi_sector > > 3 ) ;
2013-11-23 17:19:00 -08:00
bio_for_each_segment ( bvec , bio , iter ) {
2021-08-04 11:56:33 +02:00
page_addr = ( unsigned long ) bvec_virt ( & bvec ) ;
2005-04-16 15:20:36 -07:00
source_addr = dev_info - > start + ( index < < 12 ) + bytes_done ;
2013-11-23 17:19:00 -08:00
if ( unlikely ( ( page_addr & 4095 ) ! = 0 ) | | ( bvec . bv_len & 4095 ) ! = 0 )
2005-04-16 15:20:36 -07:00
// More paranoia.
goto fail ;
if ( bio_data_dir ( bio ) = = READ ) {
memcpy ( ( void * ) page_addr , ( void * ) source_addr ,
2013-11-23 17:19:00 -08:00
bvec . bv_len ) ;
2005-04-16 15:20:36 -07:00
} else {
memcpy ( ( void * ) source_addr , ( void * ) page_addr ,
2013-11-23 17:19:00 -08:00
bvec . bv_len ) ;
2005-04-16 15:20:36 -07:00
}
2013-11-23 17:19:00 -08:00
bytes_done + = bvec . bv_len ;
2005-04-16 15:20:36 -07:00
}
2015-07-20 15:29:37 +02:00
bio_endio ( bio ) ;
2021-10-12 13:12:24 +02:00
return ;
2005-04-16 15:20:36 -07:00
fail :
2007-09-27 12:47:43 +02:00
bio_io_error ( bio ) ;
2005-06-23 22:05:23 -07:00
}
2015-01-07 18:05:34 +02:00
static long
2017-01-26 13:54:35 -08:00
__dcssblk_direct_access ( struct dcssblk_dev_info * dev_info , pgoff_t pgoff ,
long nr_pages , void * * kaddr , pfn_t * pfn )
{
resource_size_t offset = pgoff * PAGE_SIZE ;
unsigned long dev_sz ;
dev_sz = dev_info - > end - dev_info - > start + 1 ;
2018-07-30 15:15:44 +08:00
if ( kaddr )
* kaddr = ( void * ) dev_info - > start + offset ;
if ( pfn )
* pfn = __pfn_to_pfn_t ( PFN_DOWN ( dev_info - > start + offset ) ,
PFN_DEV | PFN_SPECIAL ) ;
2017-01-26 13:54:35 -08:00
return ( dev_sz - offset ) / PAGE_SIZE ;
}
static long
dcssblk_dax_direct_access ( struct dax_device * dax_dev , pgoff_t pgoff ,
2022-05-13 15:10:58 -07:00
long nr_pages , enum dax_access_mode mode , void * * kaddr ,
pfn_t * pfn )
2017-01-26 13:54:35 -08:00
{
struct dcssblk_dev_info * dev_info = dax_get_private ( dax_dev ) ;
2008-04-28 02:13:02 -07:00
2017-01-26 13:54:35 -08:00
return __dcssblk_direct_access ( dev_info , pgoff , nr_pages , kaddr , pfn ) ;
2005-04-16 15:20:36 -07:00
}
static void
dcssblk_check_params ( void )
{
int rc , i , j , k ;
2008-10-10 21:33:21 +02:00
char buf [ DCSSBLK_PARM_LEN + 1 ] ;
2005-04-16 15:20:36 -07:00
struct dcssblk_dev_info * dev_info ;
for ( i = 0 ; ( i < DCSSBLK_PARM_LEN ) & & ( dcssblk_segments [ i ] ! = ' \0 ' ) ;
i + + ) {
2015-08-19 10:42:23 +02:00
for ( j = i ; ( j < DCSSBLK_PARM_LEN ) & &
( dcssblk_segments [ j ] ! = ' , ' ) & &
2005-04-16 15:20:36 -07:00
( dcssblk_segments [ j ] ! = ' \0 ' ) & &
2015-08-19 10:42:23 +02:00
( dcssblk_segments [ j ] ! = ' ( ' ) ; j + + )
2005-04-16 15:20:36 -07:00
{
buf [ j - i ] = dcssblk_segments [ j ] ;
}
buf [ j - i ] = ' \0 ' ;
2005-06-25 14:55:29 -07:00
rc = dcssblk_add_store ( dcssblk_root_dev , NULL , buf , j - i ) ;
2005-04-16 15:20:36 -07:00
if ( ( rc > = 0 ) & & ( dcssblk_segments [ j ] = = ' ( ' ) ) {
2008-10-10 21:33:21 +02:00
for ( k = 0 ; ( buf [ k ] ! = ' : ' ) & & ( buf [ k ] ! = ' \0 ' ) ; k + + )
2005-04-16 15:20:36 -07:00
buf [ k ] = toupper ( buf [ k ] ) ;
2008-10-10 21:33:21 +02:00
buf [ k ] = ' \0 ' ;
2005-04-16 15:20:36 -07:00
if ( ! strncmp ( & dcssblk_segments [ j ] , " (local) " , 7 ) ) {
down_read ( & dcssblk_devices_sem ) ;
dev_info = dcssblk_get_device_by_name ( buf ) ;
up_read ( & dcssblk_devices_sem ) ;
if ( dev_info )
dcssblk_shared_store ( & dev_info - > dev ,
2005-06-25 14:55:29 -07:00
NULL , " 0 \n " , 2 ) ;
2005-04-16 15:20:36 -07:00
}
}
while ( ( dcssblk_segments [ j ] ! = ' , ' ) & &
( dcssblk_segments [ j ] ! = ' \0 ' ) )
{
j + + ;
}
if ( dcssblk_segments [ j ] = = ' \0 ' )
break ;
i = j ;
}
}
/*
* The init / exit functions .
*/
static void __exit
dcssblk_exit ( void )
{
2008-12-15 12:58:29 +00:00
root_device_unregister ( dcssblk_root_dev ) ;
2007-07-17 04:03:46 -07:00
unregister_blkdev ( dcssblk_major , DCSSBLK_NAME ) ;
2005-04-16 15:20:36 -07:00
}
static int __init
dcssblk_init ( void )
{
int rc ;
2009-06-16 10:30:51 +02:00
dcssblk_root_dev = root_device_register ( " dcssblk " ) ;
2021-06-09 15:46:17 +02:00
if ( IS_ERR ( dcssblk_root_dev ) )
return PTR_ERR ( dcssblk_root_dev ) ;
2009-06-16 10:30:51 +02:00
rc = device_create_file ( dcssblk_root_dev , & dev_attr_add ) ;
if ( rc )
goto out_root ;
rc = device_create_file ( dcssblk_root_dev , & dev_attr_remove ) ;
if ( rc )
goto out_root ;
2005-04-16 15:20:36 -07:00
rc = register_blkdev ( 0 , DCSSBLK_NAME ) ;
2009-06-16 10:30:51 +02:00
if ( rc < 0 )
goto out_root ;
2005-04-16 15:20:36 -07:00
dcssblk_major = rc ;
init_rwsem ( & dcssblk_devices_sem ) ;
dcssblk_check_params ( ) ;
return 0 ;
2009-06-16 10:30:51 +02:00
out_root :
root_device_unregister ( dcssblk_root_dev ) ;
2021-06-09 15:46:17 +02:00
2009-06-16 10:30:51 +02:00
return rc ;
2005-04-16 15:20:36 -07:00
}
module_init ( dcssblk_init ) ;
module_exit ( dcssblk_exit ) ;
module_param_string ( segments , dcssblk_segments , DCSSBLK_PARM_LEN , 0444 ) ;
MODULE_PARM_DESC ( segments , " Name of DCSS segment(s) to be loaded, "
2008-10-10 21:33:21 +02:00
" comma-separated list, names in each set separated "
" by commas are separated by colons, each set contains "
" names of contiguous segments and each name max. 8 chars. \n "
" Adding \" (local) \" to the end of each set equals echoing 0 "
" to /sys/devices/dcssblk/<device name>/shared after loading "
" the contiguous segments - \n "
" e.g. segments= \" mydcss1,mydcss2:mydcss3,mydcss4(local) \" " ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;