2009-10-28 16:57:14 +01:00
/*
* storage_common . c - - Common definitions for mass storage functionality
*
* Copyright ( C ) 2003 - 2008 Alan Stern
* Copyeight ( C ) 2009 Samsung Electronics
2012-01-13 15:05:16 +01:00
* Author : Michal Nazarewicz ( mina86 @ mina86 . com )
2009-10-28 16:57:15 +01:00
*
* 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 ; either version 2 of the License , or
* ( at your option ) any later version .
*/
/*
* This file requires the following identifiers used in USB strings to
* be defined ( each of type pointer to char ) :
* - fsg_string_interface - - name of the interface
2009-10-28 16:57:19 +01:00
*/
usb: gadget: storage: make FSG_NUM_BUFFERS variable size
FSG_NUM_BUFFERS is set to 2 as default.
Usually 2 buffers are enough to establish a good buffering pipeline.
The number may be increased in order to compensate a for bursty VFS
behaviour.
Here follows a description of system that may require more than
2 buffers.
* CPU ondemand governor active
* latency cost for wake up and/or frequency change
* DMA for IO
Use case description.
* Data transfer from MMC via VFS to USB.
* DMA shuffles data from MMC and to USB.
* The CPU wakes up every now and then to pass data in and out from VFS,
which cause the bursty VFS behaviour.
Test set up
* Running dd on the host reading from the mass storage device
* cmdline: dd if=/dev/sdb of=/dev/null bs=4k count=$((256*100))
* Caches are dropped on the host and on the device before each run
Measurements on a Snowball board with ondemand_governor active.
FSG_NUM_BUFFERS 2
104857600 bytes (105 MB) copied, 5.62173 s, 18.7 MB/s
104857600 bytes (105 MB) copied, 5.61811 s, 18.7 MB/s
104857600 bytes (105 MB) copied, 5.57817 s, 18.8 MB/s
FSG_NUM_BUFFERS 4
104857600 bytes (105 MB) copied, 5.26839 s, 19.9 MB/s
104857600 bytes (105 MB) copied, 5.2691 s, 19.9 MB/s
104857600 bytes (105 MB) copied, 5.2711 s, 19.9 MB/s
There may not be one optimal number for all boards. This is why
the number is added to Kconfig. If selecting USB_GADGET_DEBUG_FILES
this value may be set by a module parameter as well.
Signed-off-by: Per Forlin <per.forlin@linaro.org>
Acked-by: Michal Nazarewicz <mina86@mina86.com>
Acked-by: Alan Stern <stern@rowland.harvard.edu>
Signed-off-by: Felipe Balbi <balbi@ti.com>
2011-08-19 21:21:27 +02:00
/*
* When USB_GADGET_DEBUG_FILES is defined the module param num_buffers
* sets the number of pipeline buffers ( length of the fsg_buffhd array ) .
* The valid range of num_buffers is : num > = 2 & & num < = 4.
*/
2013-09-26 14:38:16 +02:00
# include <linux/module.h>
# include <linux/blkdev.h>
# include <linux/file.h>
# include <linux/fs.h>
# include <linux/usb/composite.h>
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
# include "storage_common.h"
2009-10-28 16:57:14 +01:00
/* There is only one interface. */
2013-09-26 14:38:16 +02:00
struct usb_interface_descriptor fsg_intf_desc = {
2009-10-28 16:57:15 +01:00
. bLength = sizeof fsg_intf_desc ,
2009-10-28 16:57:14 +01:00
. bDescriptorType = USB_DT_INTERFACE ,
2009-11-09 14:15:23 +01:00
. bNumEndpoints = 2 , /* Adjusted during fsg_bind() */
2009-10-28 16:57:14 +01:00
. bInterfaceClass = USB_CLASS_MASS_STORAGE ,
2009-11-09 14:15:23 +01:00
. bInterfaceSubClass = USB_SC_SCSI , /* Adjusted during fsg_bind() */
. bInterfaceProtocol = USB_PR_BULK , /* Adjusted during fsg_bind() */
2009-10-28 16:57:15 +01:00
. iInterface = FSG_STRING_INTERFACE ,
2009-10-28 16:57:14 +01:00
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_intf_desc ) ;
2009-10-28 16:57:14 +01:00
2010-07-05 16:38:04 +02:00
/*
* Three full - speed endpoint descriptors : bulk - in , bulk - out , and
* interrupt - in .
*/
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_fs_bulk_in_desc = {
2009-10-28 16:57:14 +01:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
. bEndpointAddress = USB_DIR_IN ,
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
/* wMaxPacketSize set by autoconfiguration */
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_fs_bulk_in_desc ) ;
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_fs_bulk_out_desc = {
2009-10-28 16:57:14 +01:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
. bEndpointAddress = USB_DIR_OUT ,
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
/* wMaxPacketSize set by autoconfiguration */
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_fs_bulk_out_desc ) ;
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
struct usb_descriptor_header * fsg_fs_function [ ] = {
2009-10-28 16:57:15 +01:00
( struct usb_descriptor_header * ) & fsg_intf_desc ,
( struct usb_descriptor_header * ) & fsg_fs_bulk_in_desc ,
( struct usb_descriptor_header * ) & fsg_fs_bulk_out_desc ,
2009-10-28 16:57:14 +01:00
NULL ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_fs_function ) ;
2009-10-28 16:57:14 +01:00
/*
* USB 2.0 devices need to expose both high speed and full speed
* descriptors , unless they only run at full speed .
*
* That means alternate endpoint descriptors ( bigger packets )
* and a " device qualifier " . . . plus more construction options
2010-07-05 16:38:04 +02:00
* for the configuration descriptor .
2009-10-28 16:57:14 +01:00
*/
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_hs_bulk_in_desc = {
2009-10-28 16:57:14 +01:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
/* bEndpointAddress copied from fs_bulk_in_desc during fsg_bind() */
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
. wMaxPacketSize = cpu_to_le16 ( 512 ) ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_hs_bulk_in_desc ) ;
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_hs_bulk_out_desc = {
2009-10-28 16:57:14 +01:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
/* bEndpointAddress copied from fs_bulk_out_desc during fsg_bind() */
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
. wMaxPacketSize = cpu_to_le16 ( 512 ) ,
2009-11-09 14:15:23 +01:00
. bInterval = 1 , /* NAK every 1 uframe */
2009-10-28 16:57:14 +01:00
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_hs_bulk_out_desc ) ;
2009-10-28 16:57:14 +01:00
2009-10-28 16:57:19 +01:00
2013-09-26 14:38:16 +02:00
struct usb_descriptor_header * fsg_hs_function [ ] = {
2009-10-28 16:57:15 +01:00
( struct usb_descriptor_header * ) & fsg_intf_desc ,
( struct usb_descriptor_header * ) & fsg_hs_bulk_in_desc ,
( struct usb_descriptor_header * ) & fsg_hs_bulk_out_desc ,
2009-10-28 16:57:14 +01:00
NULL ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_hs_function ) ;
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_ss_bulk_in_desc = {
2011-08-03 14:33:27 +03:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
/* bEndpointAddress copied from fs_bulk_in_desc during fsg_bind() */
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
. wMaxPacketSize = cpu_to_le16 ( 1024 ) ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_ss_bulk_in_desc ) ;
2011-08-03 14:33:27 +03:00
2013-09-26 14:38:16 +02:00
struct usb_ss_ep_comp_descriptor fsg_ss_bulk_in_comp_desc = {
2011-08-03 14:33:27 +03:00
. bLength = sizeof ( fsg_ss_bulk_in_comp_desc ) ,
. bDescriptorType = USB_DT_SS_ENDPOINT_COMP ,
/*.bMaxBurst = DYNAMIC, */
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_ss_bulk_in_comp_desc ) ;
2011-08-03 14:33:27 +03:00
2013-09-26 14:38:16 +02:00
struct usb_endpoint_descriptor fsg_ss_bulk_out_desc = {
2011-08-03 14:33:27 +03:00
. bLength = USB_DT_ENDPOINT_SIZE ,
. bDescriptorType = USB_DT_ENDPOINT ,
/* bEndpointAddress copied from fs_bulk_out_desc during fsg_bind() */
. bmAttributes = USB_ENDPOINT_XFER_BULK ,
. wMaxPacketSize = cpu_to_le16 ( 1024 ) ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_ss_bulk_out_desc ) ;
2011-08-03 14:33:27 +03:00
2013-09-26 14:38:16 +02:00
struct usb_ss_ep_comp_descriptor fsg_ss_bulk_out_comp_desc = {
2011-08-03 14:33:27 +03:00
. bLength = sizeof ( fsg_ss_bulk_in_comp_desc ) ,
. bDescriptorType = USB_DT_SS_ENDPOINT_COMP ,
/*.bMaxBurst = DYNAMIC, */
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_ss_bulk_out_comp_desc ) ;
2011-08-03 14:33:27 +03:00
2013-09-26 14:38:16 +02:00
struct usb_descriptor_header * fsg_ss_function [ ] = {
2011-08-03 14:33:27 +03:00
( struct usb_descriptor_header * ) & fsg_intf_desc ,
( struct usb_descriptor_header * ) & fsg_ss_bulk_in_desc ,
( struct usb_descriptor_header * ) & fsg_ss_bulk_in_comp_desc ,
( struct usb_descriptor_header * ) & fsg_ss_bulk_out_desc ,
( struct usb_descriptor_header * ) & fsg_ss_bulk_out_comp_desc ,
NULL ,
} ;
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_ss_function ) ;
2009-10-28 16:57:14 +01:00
/*-------------------------------------------------------------------------*/
2010-07-05 16:38:04 +02:00
/*
* If the next two routines are called while the gadget is registered ,
* the caller must own fsg - > filesem for writing .
*/
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
void fsg_lun_close ( struct fsg_lun * curlun )
2012-06-18 14:37:20 +02:00
{
if ( curlun - > filp ) {
LDBG ( curlun , " close backing file \n " ) ;
fput ( curlun - > filp ) ;
curlun - > filp = NULL ;
}
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_lun_close ) ;
2012-06-18 14:37:20 +02:00
2013-09-26 14:38:16 +02:00
int fsg_lun_open ( struct fsg_lun * curlun , const char * filename )
2009-10-28 16:57:14 +01:00
{
int ro ;
struct file * filp = NULL ;
int rc = - EINVAL ;
struct inode * inode = NULL ;
loff_t size ;
loff_t num_sectors ;
loff_t min_sectors ;
2012-06-18 14:37:20 +02:00
unsigned int blkbits ;
unsigned int blksize ;
2009-10-28 16:57:14 +01:00
/* R/W if we can, R/O if we must */
2009-10-28 16:57:16 +01:00
ro = curlun - > initially_ro ;
2009-10-28 16:57:14 +01:00
if ( ! ro ) {
filp = filp_open ( filename , O_RDWR | O_LARGEFILE , 0 ) ;
2010-11-13 11:55:17 +01:00
if ( PTR_ERR ( filp ) = = - EROFS | | PTR_ERR ( filp ) = = - EACCES )
2009-10-28 16:57:14 +01:00
ro = 1 ;
}
if ( ro )
filp = filp_open ( filename , O_RDONLY | O_LARGEFILE , 0 ) ;
if ( IS_ERR ( filp ) ) {
LINFO ( curlun , " unable to open backing file: %s \n " , filename ) ;
return PTR_ERR ( filp ) ;
}
if ( ! ( filp - > f_mode & FMODE_WRITE ) )
ro = 1 ;
2013-01-23 17:07:38 -05:00
inode = file_inode ( filp ) ;
2012-07-22 21:23:33 +04:00
if ( ( ! S_ISREG ( inode - > i_mode ) & & ! S_ISBLK ( inode - > i_mode ) ) ) {
2009-10-28 16:57:14 +01:00
LINFO ( curlun , " invalid file type: %s \n " , filename ) ;
goto out ;
}
2010-07-05 16:38:04 +02:00
/*
* If we can ' t read the file , it ' s no good .
* If we can ' t write the file , use it read - only .
*/
2012-07-22 21:23:33 +04:00
if ( ! ( filp - > f_op - > read | | filp - > f_op - > aio_read ) ) {
2009-10-28 16:57:14 +01:00
LINFO ( curlun , " file not readable: %s \n " , filename ) ;
goto out ;
}
if ( ! ( filp - > f_op - > write | | filp - > f_op - > aio_write ) )
ro = 1 ;
size = i_size_read ( inode - > i_mapping - > host ) ;
if ( size < 0 ) {
LINFO ( curlun , " unable to find file size: %s \n " , filename ) ;
rc = ( int ) size ;
goto out ;
}
2011-08-17 22:52:59 -07:00
if ( curlun - > cdrom ) {
2012-06-18 14:37:20 +02:00
blksize = 2048 ;
blkbits = 11 ;
2011-08-17 22:52:59 -07:00
} else if ( inode - > i_bdev ) {
2012-06-18 14:37:20 +02:00
blksize = bdev_logical_block_size ( inode - > i_bdev ) ;
blkbits = blksize_bits ( blksize ) ;
2011-08-17 22:52:59 -07:00
} else {
2012-06-18 14:37:20 +02:00
blksize = 512 ;
blkbits = 9 ;
2011-08-17 22:52:59 -07:00
}
2012-06-18 14:37:20 +02:00
num_sectors = size > > blkbits ; /* File size in logic-block-size blocks */
2009-10-28 16:57:14 +01:00
min_sectors = 1 ;
2009-10-28 16:57:16 +01:00
if ( curlun - > cdrom ) {
2011-08-17 22:52:59 -07:00
min_sectors = 300 ; /* Smallest track is 300 frames */
if ( num_sectors > = 256 * 60 * 75 ) {
num_sectors = 256 * 60 * 75 - 1 ;
2009-10-28 16:57:14 +01:00
LINFO ( curlun , " file too big: %s \n " , filename ) ;
LINFO ( curlun , " using only first %d blocks \n " ,
( int ) num_sectors ) ;
}
}
if ( num_sectors < min_sectors ) {
LINFO ( curlun , " file too small: %s \n " , filename ) ;
rc = - ETOOSMALL ;
goto out ;
}
2012-06-18 14:37:20 +02:00
if ( fsg_lun_is_open ( curlun ) )
fsg_lun_close ( curlun ) ;
curlun - > blksize = blksize ;
curlun - > blkbits = blkbits ;
2009-10-28 16:57:14 +01:00
curlun - > ro = ro ;
curlun - > filp = filp ;
curlun - > file_length = size ;
curlun - > num_sectors = num_sectors ;
LDBG ( curlun , " open backing file: %s \n " , filename ) ;
2012-07-22 21:23:33 +04:00
return 0 ;
2009-10-28 16:57:14 +01:00
out :
2012-07-22 21:23:33 +04:00
fput ( filp ) ;
2009-10-28 16:57:14 +01:00
return rc ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_lun_open ) ;
2009-10-28 16:57:14 +01:00
/*-------------------------------------------------------------------------*/
2010-07-05 16:38:04 +02:00
/*
* Sync the file data , don ' t bother with the metadata .
* This code was copied from fs / buffer . c : sys_fdatasync ( ) .
*/
2013-09-26 14:38:16 +02:00
int fsg_lun_fsync_sub ( struct fsg_lun * curlun )
2009-10-28 16:57:14 +01:00
{
struct file * filp = curlun - > filp ;
if ( curlun - > ro | | ! filp )
return 0 ;
2010-03-22 17:32:25 +01:00
return vfs_fsync ( filp , 1 ) ;
2009-10-28 16:57:14 +01:00
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_lun_fsync_sub ) ;
2009-10-28 16:57:14 +01:00
2013-09-26 14:38:16 +02:00
void store_cdrom_address ( u8 * dest , int msf , u32 addr )
2009-10-28 16:57:14 +01:00
{
if ( msf ) {
/* Convert to Minutes-Seconds-Frames */
addr > > = 2 ; /* Convert to 2048-byte frames */
addr + = 2 * 75 ; /* Lead-in occupies 2 seconds */
dest [ 3 ] = addr % 75 ; /* Frames */
addr / = 75 ;
dest [ 2 ] = addr % 60 ; /* Seconds */
addr / = 60 ;
dest [ 1 ] = addr ; /* Minutes */
dest [ 0 ] = 0 ; /* Reserved */
} else {
/* Absolute sector */
put_unaligned_be32 ( addr , dest ) ;
}
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( store_cdrom_address ) ;
2009-10-28 16:57:17 +01:00
/*-------------------------------------------------------------------------*/
2013-10-09 10:06:03 +02:00
ssize_t fsg_show_ro ( struct fsg_lun * curlun , char * buf )
2009-10-28 16:57:17 +01:00
{
return sprintf ( buf , " %d \n " , fsg_lun_is_open ( curlun )
? curlun - > ro
: curlun - > initially_ro ) ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_show_ro ) ;
2009-10-28 16:57:17 +01:00
2013-10-09 10:06:03 +02:00
ssize_t fsg_show_nofua ( struct fsg_lun * curlun , char * buf )
2010-07-22 17:53:56 +03:00
{
return sprintf ( buf , " %u \n " , curlun - > nofua ) ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_show_nofua ) ;
2010-07-22 17:53:56 +03:00
2013-10-09 10:06:03 +02:00
ssize_t fsg_show_file ( struct fsg_lun * curlun , struct rw_semaphore * filesem ,
2013-09-26 14:38:16 +02:00
char * buf )
2009-10-28 16:57:17 +01:00
{
char * p ;
ssize_t rc ;
down_read ( filesem ) ;
2009-11-09 14:15:23 +01:00
if ( fsg_lun_is_open ( curlun ) ) { /* Get the complete pathname */
2009-10-28 16:57:17 +01:00
p = d_path ( & curlun - > filp - > f_path , buf , PAGE_SIZE - 1 ) ;
if ( IS_ERR ( p ) )
rc = PTR_ERR ( p ) ;
else {
rc = strlen ( p ) ;
memmove ( buf , p , rc ) ;
2009-11-09 14:15:23 +01:00
buf [ rc ] = ' \n ' ; /* Add a newline */
2009-10-28 16:57:17 +01:00
buf [ + + rc ] = 0 ;
}
2009-11-09 14:15:23 +01:00
} else { /* No file, return 0 bytes */
2009-10-28 16:57:17 +01:00
* buf = 0 ;
rc = 0 ;
}
up_read ( filesem ) ;
return rc ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_show_file ) ;
2009-10-28 16:57:17 +01:00
2013-10-09 10:06:04 +02:00
ssize_t fsg_show_cdrom ( struct fsg_lun * curlun , char * buf )
{
return sprintf ( buf , " %u \n " , curlun - > cdrom ) ;
}
EXPORT_SYMBOL ( fsg_show_cdrom ) ;
ssize_t fsg_show_removable ( struct fsg_lun * curlun , char * buf )
{
return sprintf ( buf , " %u \n " , curlun - > removable ) ;
}
EXPORT_SYMBOL ( fsg_show_removable ) ;
2009-10-28 16:57:17 +01:00
2013-10-15 08:33:13 +02:00
/*
* The caller must hold fsg - > filesem for reading when calling this function .
*/
static ssize_t _fsg_store_ro ( struct fsg_lun * curlun , bool ro )
{
if ( fsg_lun_is_open ( curlun ) ) {
LDBG ( curlun , " read-only status change prevented \n " ) ;
return - EBUSY ;
}
curlun - > ro = ro ;
curlun - > initially_ro = ro ;
LDBG ( curlun , " read-only status set to %d \n " , curlun - > ro ) ;
return 0 ;
}
2013-10-09 10:06:03 +02:00
ssize_t fsg_store_ro ( struct fsg_lun * curlun , struct rw_semaphore * filesem ,
2013-09-26 14:38:16 +02:00
const char * buf , size_t count )
2009-10-28 16:57:17 +01:00
{
2011-04-14 11:55:43 +02:00
ssize_t rc ;
2013-10-15 08:33:12 +02:00
bool ro ;
2009-10-28 16:57:17 +01:00
2013-10-15 08:33:12 +02:00
rc = strtobool ( buf , & ro ) ;
2011-04-14 00:37:00 +02:00
if ( rc )
return rc ;
2009-10-28 16:57:17 +01:00
2010-07-05 16:38:04 +02:00
/*
* Allow the write - enable status to change only while the
* backing file is closed .
*/
2009-10-28 16:57:17 +01:00
down_read ( filesem ) ;
2013-10-15 08:33:13 +02:00
rc = _fsg_store_ro ( curlun , ro ) ;
if ( ! rc )
2011-04-14 11:55:43 +02:00
rc = count ;
2009-10-28 16:57:17 +01:00
up_read ( filesem ) ;
2013-10-15 08:33:13 +02:00
2009-10-28 16:57:17 +01:00
return rc ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_store_ro ) ;
2009-10-28 16:57:17 +01:00
2013-10-09 10:06:03 +02:00
ssize_t fsg_store_nofua ( struct fsg_lun * curlun , const char * buf , size_t count )
2010-07-22 17:53:56 +03:00
{
2013-10-15 08:33:12 +02:00
bool nofua ;
2011-04-14 00:37:00 +02:00
int ret ;
2010-07-22 17:53:56 +03:00
2013-10-15 08:33:12 +02:00
ret = strtobool ( buf , & nofua ) ;
2011-04-14 00:37:00 +02:00
if ( ret )
return ret ;
2010-07-22 17:53:56 +03:00
/* Sync data when switching from async mode to sync */
if ( ! nofua & & curlun - > nofua )
fsg_lun_fsync_sub ( curlun ) ;
curlun - > nofua = nofua ;
return count ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_store_nofua ) ;
2010-07-22 17:53:56 +03:00
2013-10-09 10:06:03 +02:00
ssize_t fsg_store_file ( struct fsg_lun * curlun , struct rw_semaphore * filesem ,
2013-09-26 14:38:16 +02:00
const char * buf , size_t count )
2009-10-28 16:57:17 +01:00
{
int rc = 0 ;
if ( curlun - > prevent_medium_removal & & fsg_lun_is_open ( curlun ) ) {
LDBG ( curlun , " eject attempt prevented \n " ) ;
2009-11-09 14:15:23 +01:00
return - EBUSY ; /* "Door is locked" */
2009-10-28 16:57:17 +01:00
}
/* Remove a trailing newline */
if ( count > 0 & & buf [ count - 1 ] = = ' \n ' )
2009-11-09 14:15:23 +01:00
( ( char * ) buf ) [ count - 1 ] = 0 ; /* Ugh! */
2009-10-28 16:57:17 +01:00
/* Load new medium */
2012-06-18 14:37:20 +02:00
down_write ( filesem ) ;
2009-10-28 16:57:17 +01:00
if ( count > 0 & & buf [ 0 ] ) {
2012-06-18 14:37:20 +02:00
/* fsg_lun_open() will close existing file if any. */
2009-10-28 16:57:17 +01:00
rc = fsg_lun_open ( curlun , buf ) ;
if ( rc = = 0 )
curlun - > unit_attention_data =
SS_NOT_READY_TO_READY_TRANSITION ;
2012-06-18 14:37:20 +02:00
} else if ( fsg_lun_is_open ( curlun ) ) {
fsg_lun_close ( curlun ) ;
curlun - > unit_attention_data = SS_MEDIUM_NOT_PRESENT ;
2009-10-28 16:57:17 +01:00
}
up_write ( filesem ) ;
return ( rc < 0 ? rc : count ) ;
}
2013-09-26 14:38:16 +02:00
EXPORT_SYMBOL ( fsg_store_file ) ;
2013-10-15 08:33:13 +02:00
ssize_t fsg_store_cdrom ( struct fsg_lun * curlun , struct rw_semaphore * filesem ,
const char * buf , size_t count )
2013-10-09 10:06:04 +02:00
{
2013-10-15 08:33:12 +02:00
bool cdrom ;
2013-10-09 10:06:04 +02:00
int ret ;
2013-10-15 08:33:12 +02:00
ret = strtobool ( buf , & cdrom ) ;
2013-10-09 10:06:04 +02:00
if ( ret )
return ret ;
2013-10-15 08:33:13 +02:00
down_read ( filesem ) ;
ret = cdrom ? _fsg_store_ro ( curlun , true ) : 0 ;
2013-10-09 10:06:04 +02:00
2013-10-15 08:33:13 +02:00
if ( ! ret ) {
curlun - > cdrom = cdrom ;
ret = count ;
}
up_read ( filesem ) ;
return ret ;
2013-10-09 10:06:04 +02:00
}
EXPORT_SYMBOL ( fsg_store_cdrom ) ;
ssize_t fsg_store_removable ( struct fsg_lun * curlun , const char * buf ,
size_t count )
{
2013-10-15 08:33:12 +02:00
bool removable ;
2013-10-09 10:06:04 +02:00
int ret ;
2013-10-15 08:33:12 +02:00
ret = strtobool ( buf , & removable ) ;
2013-10-09 10:06:04 +02:00
if ( ret )
return ret ;
curlun - > removable = removable ;
return count ;
}
EXPORT_SYMBOL ( fsg_store_removable ) ;
2013-09-26 14:38:16 +02:00
MODULE_LICENSE ( " GPL " ) ;