2005-04-17 02:20:36 +04:00
/*
* file . c
*
* Copyright ( C ) 1995 , 1996 by Volker Lendecke
* Modified 1997 Peter Waltenberg , Bill Hawes , David Woodhouse for 2.1 dcache
*
*/
2014-04-09 03:04:14 +04:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2016-12-24 22:46:01 +03:00
# include <linux/uaccess.h>
2005-04-17 02:20:36 +04:00
# include <linux/time.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/fcntl.h>
# include <linux/stat.h>
# include <linux/mm.h>
# include <linux/vmalloc.h>
Detach sched.h from mm.h
First thing mm.h does is including sched.h solely for can_do_mlock() inline
function which has "current" dereference inside. By dealing with can_do_mlock()
mm.h can be detached from sched.h which is good. See below, why.
This patch
a) removes unconditional inclusion of sched.h from mm.h
b) makes can_do_mlock() normal function in mm/mlock.c
c) exports can_do_mlock() to not break compilation
d) adds sched.h inclusions back to files that were getting it indirectly.
e) adds less bloated headers to some files (asm/signal.h, jiffies.h) that were
getting them indirectly
Net result is:
a) mm.h users would get less code to open, read, preprocess, parse, ... if
they don't need sched.h
b) sched.h stops being dependency for significant number of files:
on x86_64 allmodconfig touching sched.h results in recompile of 4083 files,
after patch it's only 3744 (-8.3%).
Cross-compile tested on
all arm defconfigs, all mips defconfigs, all powerpc defconfigs,
alpha alpha-up
arm
i386 i386-up i386-defconfig i386-allnoconfig
ia64 ia64-up
m68k
mips
parisc parisc-up
powerpc powerpc-up
s390 s390-up
sparc sparc-up
sparc64 sparc64-up
um-x86_64
x86_64 x86_64-up x86_64-defconfig x86_64-allnoconfig
as well as my two usual configs.
Signed-off-by: Alexey Dobriyan <adobriyan@gmail.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-21 01:22:52 +04:00
# include <linux/sched.h>
2005-04-17 02:20:36 +04:00
2011-01-13 01:37:47 +03:00
# include "ncp_fs.h"
2005-04-17 02:20:36 +04:00
2011-07-17 04:44:56 +04:00
static int ncp_fsync ( struct file * file , loff_t start , loff_t end , int datasync )
2005-04-17 02:20:36 +04:00
{
2011-07-17 04:44:56 +04:00
return filemap_write_and_wait_range ( file - > f_mapping , start , end ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Open a file with the specified read / write mode .
*/
int ncp_make_open ( struct inode * inode , int right )
{
int error ;
int access ;
error = - EINVAL ;
if ( ! inode ) {
2014-04-09 03:04:14 +04:00
pr_err ( " %s: got NULL inode \n " , __func__ ) ;
2005-04-17 02:20:36 +04:00
goto out ;
}
2014-04-09 03:04:15 +04:00
ncp_dbg ( 1 , " opened=%d, volume # %u, dir entry # %u \n " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & NCP_FINFO ( inode ) - > opened ) ,
NCP_FINFO ( inode ) - > volNumber ,
NCP_FINFO ( inode ) - > dirEntNum ) ;
error = - EACCES ;
2006-03-23 14:00:43 +03:00
mutex_lock ( & NCP_FINFO ( inode ) - > open_mutex ) ;
2005-04-17 02:20:36 +04:00
if ( ! atomic_read ( & NCP_FINFO ( inode ) - > opened ) ) {
struct ncp_entry_info finfo ;
int result ;
/* tries max. rights */
finfo . access = O_RDWR ;
result = ncp_open_create_file_or_subdir ( NCP_SERVER ( inode ) ,
inode , NULL , OC_MODE_OPEN ,
0 , AR_READ | AR_WRITE , & finfo ) ;
if ( ! result )
goto update ;
/* RDWR did not succeeded, try readonly or writeonly as requested */
switch ( right ) {
case O_RDONLY :
finfo . access = O_RDONLY ;
result = ncp_open_create_file_or_subdir ( NCP_SERVER ( inode ) ,
inode , NULL , OC_MODE_OPEN ,
0 , AR_READ , & finfo ) ;
break ;
case O_WRONLY :
finfo . access = O_WRONLY ;
result = ncp_open_create_file_or_subdir ( NCP_SERVER ( inode ) ,
inode , NULL , OC_MODE_OPEN ,
0 , AR_WRITE , & finfo ) ;
break ;
}
if ( result ) {
2014-04-09 03:04:16 +04:00
ncp_vdbg ( " failed, result=%d \n " , result ) ;
2005-04-17 02:20:36 +04:00
goto out_unlock ;
}
/*
* Update the inode information .
*/
update :
ncp_update_inode ( inode , & finfo ) ;
atomic_set ( & NCP_FINFO ( inode ) - > opened , 1 ) ;
}
access = NCP_FINFO ( inode ) - > access ;
2014-04-09 03:04:16 +04:00
ncp_vdbg ( " file open, access=%x \n " , access ) ;
2005-04-17 02:20:36 +04:00
if ( access = = right | | access = = O_RDWR ) {
atomic_inc ( & NCP_FINFO ( inode ) - > opened ) ;
error = 0 ;
}
out_unlock :
2006-03-23 14:00:43 +03:00
mutex_unlock ( & NCP_FINFO ( inode ) - > open_mutex ) ;
2005-04-17 02:20:36 +04:00
out :
return error ;
}
static ssize_t
2015-04-03 06:30:18 +03:00
ncp_file_read_iter ( struct kiocb * iocb , struct iov_iter * to )
2005-04-17 02:20:36 +04:00
{
2015-04-03 06:30:18 +03:00
struct file * file = iocb - > ki_filp ;
2014-10-31 09:41:28 +03:00
struct inode * inode = file_inode ( file ) ;
2005-04-17 02:20:36 +04:00
size_t already_read = 0 ;
2015-04-03 06:30:18 +03:00
off_t pos = iocb - > ki_pos ;
2005-04-17 02:20:36 +04:00
size_t bufsize ;
int error ;
2015-04-03 06:30:18 +03:00
void * freepage ;
2005-04-17 02:20:36 +04:00
size_t freelen ;
2014-10-31 09:41:28 +03:00
ncp_dbg ( 1 , " enter %pD2 \n " , file ) ;
2005-04-17 02:20:36 +04:00
2015-04-03 06:30:18 +03:00
if ( ! iov_iter_count ( to ) )
2005-04-17 02:20:36 +04:00
return 0 ;
if ( pos > inode - > i_sb - > s_maxbytes )
return 0 ;
2015-04-03 06:30:18 +03:00
iov_iter_truncate ( to , inode - > i_sb - > s_maxbytes - pos ) ;
2005-04-17 02:20:36 +04:00
error = ncp_make_open ( inode , O_RDONLY ) ;
if ( error ) {
2014-04-09 03:04:15 +04:00
ncp_dbg ( 1 , " open failed, error=%d \n " , error ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
bufsize = NCP_SERVER ( inode ) - > buffer_size ;
error = - EIO ;
freelen = ncp_read_bounce_size ( bufsize ) ;
freepage = vmalloc ( freelen ) ;
if ( ! freepage )
goto outrel ;
error = 0 ;
/* First read in as much as possible for each bufsize. */
2015-04-03 06:30:18 +03:00
while ( iov_iter_count ( to ) ) {
2005-04-17 02:20:36 +04:00
int read_this_time ;
2015-04-03 06:30:18 +03:00
size_t to_read = min_t ( size_t ,
2005-04-17 02:20:36 +04:00
bufsize - ( pos % bufsize ) ,
2015-04-03 06:30:18 +03:00
iov_iter_count ( to ) ) ;
2005-04-17 02:20:36 +04:00
error = ncp_read_bounce ( NCP_SERVER ( inode ) ,
NCP_FINFO ( inode ) - > file_handle ,
2015-04-03 06:30:18 +03:00
pos , to_read , to , & read_this_time ,
2005-04-17 02:20:36 +04:00
freepage , freelen ) ;
if ( error ) {
error = - EIO ; /* NW errno -> Linux errno */
break ;
}
pos + = read_this_time ;
already_read + = read_this_time ;
2015-04-03 06:30:18 +03:00
if ( read_this_time ! = to_read )
2005-04-17 02:20:36 +04:00
break ;
}
vfree ( freepage ) ;
2015-04-03 06:30:18 +03:00
iocb - > ki_pos = pos ;
2005-04-17 02:20:36 +04:00
file_accessed ( file ) ;
2014-10-31 09:41:28 +03:00
ncp_dbg ( 1 , " exit %pD2 \n " , file ) ;
2005-04-17 02:20:36 +04:00
outrel :
ncp_inode_close ( inode ) ;
return already_read ? already_read : error ;
}
static ssize_t
2015-04-03 06:30:18 +03:00
ncp_file_write_iter ( struct kiocb * iocb , struct iov_iter * from )
2005-04-17 02:20:36 +04:00
{
2015-04-03 06:30:18 +03:00
struct file * file = iocb - > ki_filp ;
2014-10-31 09:41:28 +03:00
struct inode * inode = file_inode ( file ) ;
2005-04-17 02:20:36 +04:00
size_t already_written = 0 ;
size_t bufsize ;
int errno ;
2015-04-03 06:30:18 +03:00
void * bouncebuffer ;
2015-04-09 19:55:47 +03:00
off_t pos ;
2005-04-17 02:20:36 +04:00
2014-10-31 09:41:28 +03:00
ncp_dbg ( 1 , " enter %pD2 \n " , file ) ;
2015-04-09 19:55:47 +03:00
errno = generic_write_checks ( iocb , from ) ;
if ( errno < = 0 )
2015-04-03 06:30:18 +03:00
return errno ;
2005-04-17 02:20:36 +04:00
errno = ncp_make_open ( inode , O_WRONLY ) ;
if ( errno ) {
2014-04-09 03:04:15 +04:00
ncp_dbg ( 1 , " open failed, error=%d \n " , errno ) ;
2005-04-17 02:20:36 +04:00
return errno ;
}
bufsize = NCP_SERVER ( inode ) - > buffer_size ;
2012-03-26 17:59:21 +04:00
errno = file_update_time ( file ) ;
if ( errno )
goto outrel ;
2005-04-17 02:20:36 +04:00
bouncebuffer = vmalloc ( bufsize ) ;
if ( ! bouncebuffer ) {
errno = - EIO ; /* -ENOMEM */
goto outrel ;
}
2015-04-09 19:55:47 +03:00
pos = iocb - > ki_pos ;
2015-04-03 06:30:18 +03:00
while ( iov_iter_count ( from ) ) {
2005-04-17 02:20:36 +04:00
int written_this_time ;
2015-04-03 06:30:18 +03:00
size_t to_write = min_t ( size_t ,
2015-04-09 19:55:47 +03:00
bufsize - ( pos % bufsize ) ,
2015-04-03 06:30:18 +03:00
iov_iter_count ( from ) ) ;
2005-04-17 02:20:36 +04:00
2016-11-02 05:09:04 +03:00
if ( ! copy_from_iter_full ( bouncebuffer , to_write , from ) ) {
2005-04-17 02:20:36 +04:00
errno = - EFAULT ;
break ;
}
if ( ncp_write_kernel ( NCP_SERVER ( inode ) ,
NCP_FINFO ( inode ) - > file_handle ,
pos , to_write , bouncebuffer , & written_this_time ) ! = 0 ) {
errno = - EIO ;
break ;
}
pos + = written_this_time ;
already_written + = written_this_time ;
2015-04-03 06:30:18 +03:00
if ( written_this_time ! = to_write )
2005-04-17 02:20:36 +04:00
break ;
}
vfree ( bouncebuffer ) ;
2015-04-03 06:30:18 +03:00
iocb - > ki_pos = pos ;
2005-04-17 02:20:36 +04:00
2010-09-27 03:47:33 +04:00
if ( pos > i_size_read ( inode ) ) {
2016-01-22 23:40:57 +03:00
inode_lock ( inode ) ;
2010-09-27 03:47:33 +04:00
if ( pos > i_size_read ( inode ) )
i_size_write ( inode , pos ) ;
2016-01-22 23:40:57 +03:00
inode_unlock ( inode ) ;
2005-04-17 02:20:36 +04:00
}
2014-10-31 09:41:28 +03:00
ncp_dbg ( 1 , " exit %pD2 \n " , file ) ;
2005-04-17 02:20:36 +04:00
outrel :
ncp_inode_close ( inode ) ;
return already_written ? already_written : errno ;
}
static int ncp_release ( struct inode * inode , struct file * file ) {
if ( ncp_make_closed ( inode ) ) {
2014-04-09 03:04:15 +04:00
ncp_dbg ( 1 , " failed to close \n " ) ;
2005-04-17 02:20:36 +04:00
}
return 0 ;
}
2006-03-28 13:56:42 +04:00
const struct file_operations ncp_file_operations =
2005-04-17 02:20:36 +04:00
{
2010-09-27 03:47:33 +04:00
. llseek = generic_file_llseek ,
2015-04-03 06:30:18 +03:00
. read_iter = ncp_file_read_iter ,
. write_iter = ncp_file_write_iter ,
2010-05-05 17:15:37 +04:00
. unlocked_ioctl = ncp_ioctl ,
2006-10-01 10:27:55 +04:00
# ifdef CONFIG_COMPAT
. compat_ioctl = ncp_compat_ioctl ,
# endif
2005-04-17 02:20:36 +04:00
. mmap = ncp_mmap ,
. release = ncp_release ,
. fsync = ncp_fsync ,
} ;
2007-02-12 11:55:39 +03:00
const struct inode_operations ncp_file_inode_operations =
2005-04-17 02:20:36 +04:00
{
. setattr = ncp_notify_change ,
} ;