2005-04-17 02:20:36 +04:00
/*
* linux / fs / readdir . c
*
* Copyright ( C ) 1995 Linus Torvalds
*/
2010-08-10 04:20:22 +04:00
# include <linux/stddef.h>
2007-05-08 11:29:02 +04:00
# include <linux/kernel.h>
2011-11-17 08:57:37 +04:00
# include <linux/export.h>
2005-04-17 02:20:36 +04:00
# include <linux/time.h>
# include <linux/mm.h>
# include <linux/errno.h>
# include <linux/stat.h>
# include <linux/file.h>
# include <linux/fs.h>
2014-06-05 03:05:41 +04:00
# include <linux/fsnotify.h>
2005-04-17 02:20:36 +04:00
# include <linux/dirent.h>
# include <linux/security.h>
# include <linux/syscalls.h>
# include <linux/unistd.h>
# include <asm/uaccess.h>
2013-05-15 21:52:59 +04:00
int iterate_dir ( struct file * file , struct dir_context * ctx )
2005-04-17 02:20:36 +04:00
{
2013-01-24 02:07:38 +04:00
struct inode * inode = file_inode ( file ) ;
2005-04-17 02:20:36 +04:00
int res = - ENOTDIR ;
2013-09-23 00:27:52 +04:00
if ( ! file - > f_op - > iterate )
2005-04-17 02:20:36 +04:00
goto out ;
res = security_file_permission ( file , MAY_READ ) ;
if ( res )
goto out ;
2007-12-07 01:39:54 +03:00
res = mutex_lock_killable ( & inode - > i_mutex ) ;
if ( res )
goto out ;
2005-04-17 02:20:36 +04:00
res = - ENOENT ;
if ( ! IS_DEADDIR ( inode ) ) {
2013-05-23 05:44:23 +04:00
ctx - > pos = file - > f_pos ;
res = file - > f_op - > iterate ( file , ctx ) ;
file - > f_pos = ctx - > pos ;
2014-06-05 03:05:41 +04:00
fsnotify_access ( file ) ;
2005-04-17 02:20:36 +04:00
file_accessed ( file ) ;
}
2006-01-10 02:59:24 +03:00
mutex_unlock ( & inode - > i_mutex ) ;
2005-04-17 02:20:36 +04:00
out :
return res ;
}
2013-05-15 21:52:59 +04:00
EXPORT_SYMBOL ( iterate_dir ) ;
2005-04-17 02:20:36 +04:00
/*
* Traditional linux readdir ( ) handling . .
*
* " count=1 " is a special case , meaning that the buffer is one
* dirent - structure in size and that the code can ' t handle more
* anyway . Thus the special " fillonedir() " function for that
* case ( the low - level handlers don ' t need to care about this ) .
*/
# ifdef __ARCH_WANT_OLD_READDIR
struct old_linux_dirent {
unsigned long d_ino ;
unsigned long d_offset ;
unsigned short d_namlen ;
char d_name [ 1 ] ;
} ;
struct readdir_callback {
2013-05-15 21:52:59 +04:00
struct dir_context ctx ;
2005-04-17 02:20:36 +04:00
struct old_linux_dirent __user * dirent ;
int result ;
} ;
static int fillonedir ( void * __buf , const char * name , int namlen , loff_t offset ,
2006-10-03 12:13:46 +04:00
u64 ino , unsigned int d_type )
2005-04-17 02:20:36 +04:00
{
2013-05-15 21:52:59 +04:00
struct readdir_callback * buf = ( struct readdir_callback * ) __buf ;
2005-04-17 02:20:36 +04:00
struct old_linux_dirent __user * dirent ;
2006-10-03 12:13:46 +04:00
unsigned long d_ino ;
2005-04-17 02:20:36 +04:00
if ( buf - > result )
return - EINVAL ;
2006-10-03 12:13:46 +04:00
d_ino = ino ;
2008-08-12 08:28:24 +04:00
if ( sizeof ( d_ino ) < sizeof ( ino ) & & d_ino ! = ino ) {
buf - > result = - EOVERFLOW ;
2006-10-03 12:13:46 +04:00
return - EOVERFLOW ;
2008-08-12 08:28:24 +04:00
}
2005-04-17 02:20:36 +04:00
buf - > result + + ;
dirent = buf - > dirent ;
if ( ! access_ok ( VERIFY_WRITE , dirent ,
( unsigned long ) ( dirent - > d_name + namlen + 1 ) -
( unsigned long ) dirent ) )
goto efault ;
2006-10-03 12:13:46 +04:00
if ( __put_user ( d_ino , & dirent - > d_ino ) | |
2005-04-17 02:20:36 +04:00
__put_user ( offset , & dirent - > d_offset ) | |
__put_user ( namlen , & dirent - > d_namlen ) | |
__copy_to_user ( dirent - > d_name , name , namlen ) | |
__put_user ( 0 , dirent - > d_name + namlen ) )
goto efault ;
return 0 ;
efault :
buf - > result = - EFAULT ;
return - EFAULT ;
}
2009-01-14 16:14:34 +03:00
SYSCALL_DEFINE3 ( old_readdir , unsigned int , fd ,
struct old_linux_dirent __user * , dirent , unsigned int , count )
2005-04-17 02:20:36 +04:00
{
int error ;
2012-08-28 20:52:22 +04:00
struct fd f = fdget ( fd ) ;
2013-05-23 06:22:04 +04:00
struct readdir_callback buf = {
. ctx . actor = fillonedir ,
. dirent = dirent
} ;
2005-04-17 02:20:36 +04:00
2012-08-28 20:52:22 +04:00
if ( ! f . file )
2012-04-22 02:40:32 +04:00
return - EBADF ;
2005-04-17 02:20:36 +04:00
2013-05-15 21:52:59 +04:00
error = iterate_dir ( f . file , & buf . ctx ) ;
2008-08-24 15:29:52 +04:00
if ( buf . result )
2005-04-17 02:20:36 +04:00
error = buf . result ;
2012-08-28 20:52:22 +04:00
fdput ( f ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
# endif /* __ARCH_WANT_OLD_READDIR */
/*
* New , all - improved , singing , dancing , iBCS2 - compliant getdents ( )
* interface .
*/
struct linux_dirent {
unsigned long d_ino ;
unsigned long d_off ;
unsigned short d_reclen ;
char d_name [ 1 ] ;
} ;
struct getdents_callback {
2013-05-15 21:52:59 +04:00
struct dir_context ctx ;
2005-04-17 02:20:36 +04:00
struct linux_dirent __user * current_dir ;
struct linux_dirent __user * previous ;
int count ;
int error ;
} ;
static int filldir ( void * __buf , const char * name , int namlen , loff_t offset ,
2006-10-03 12:13:46 +04:00
u64 ino , unsigned int d_type )
2005-04-17 02:20:36 +04:00
{
struct linux_dirent __user * dirent ;
struct getdents_callback * buf = ( struct getdents_callback * ) __buf ;
2006-10-03 12:13:46 +04:00
unsigned long d_ino ;
2010-08-10 04:20:22 +04:00
int reclen = ALIGN ( offsetof ( struct linux_dirent , d_name ) + namlen + 2 ,
sizeof ( long ) ) ;
2005-04-17 02:20:36 +04:00
buf - > error = - EINVAL ; /* only used if we fail.. */
if ( reclen > buf - > count )
return - EINVAL ;
2006-10-03 12:13:46 +04:00
d_ino = ino ;
2008-08-12 08:28:24 +04:00
if ( sizeof ( d_ino ) < sizeof ( ino ) & & d_ino ! = ino ) {
buf - > error = - EOVERFLOW ;
2006-10-03 12:13:46 +04:00
return - EOVERFLOW ;
2008-08-12 08:28:24 +04:00
}
2005-04-17 02:20:36 +04:00
dirent = buf - > previous ;
if ( dirent ) {
if ( __put_user ( offset , & dirent - > d_off ) )
goto efault ;
}
dirent = buf - > current_dir ;
2006-10-03 12:13:46 +04:00
if ( __put_user ( d_ino , & dirent - > d_ino ) )
2005-04-17 02:20:36 +04:00
goto efault ;
if ( __put_user ( reclen , & dirent - > d_reclen ) )
goto efault ;
if ( copy_to_user ( dirent - > d_name , name , namlen ) )
goto efault ;
if ( __put_user ( 0 , dirent - > d_name + namlen ) )
goto efault ;
if ( __put_user ( d_type , ( char __user * ) dirent + reclen - 1 ) )
goto efault ;
buf - > previous = dirent ;
dirent = ( void __user * ) dirent + reclen ;
buf - > current_dir = dirent ;
buf - > count - = reclen ;
return 0 ;
efault :
buf - > error = - EFAULT ;
return - EFAULT ;
}
2009-01-14 16:14:23 +03:00
SYSCALL_DEFINE3 ( getdents , unsigned int , fd ,
struct linux_dirent __user * , dirent , unsigned int , count )
2005-04-17 02:20:36 +04:00
{
2012-08-28 20:52:22 +04:00
struct fd f ;
2005-04-17 02:20:36 +04:00
struct linux_dirent __user * lastdirent ;
2013-05-23 06:22:04 +04:00
struct getdents_callback buf = {
. ctx . actor = filldir ,
. count = count ,
. current_dir = dirent
} ;
2005-04-17 02:20:36 +04:00
int error ;
if ( ! access_ok ( VERIFY_WRITE , dirent , count ) )
2012-04-22 02:40:32 +04:00
return - EFAULT ;
2005-04-17 02:20:36 +04:00
2012-08-28 20:52:22 +04:00
f = fdget ( fd ) ;
if ( ! f . file )
2012-04-22 02:40:32 +04:00
return - EBADF ;
2005-04-17 02:20:36 +04:00
2013-05-15 21:52:59 +04:00
error = iterate_dir ( f . file , & buf . ctx ) ;
2008-08-24 15:29:52 +04:00
if ( error > = 0 )
error = buf . error ;
2005-04-17 02:20:36 +04:00
lastdirent = buf . previous ;
if ( lastdirent ) {
2013-05-16 02:49:12 +04:00
if ( put_user ( buf . ctx . pos , & lastdirent - > d_off ) )
2005-04-17 02:20:36 +04:00
error = - EFAULT ;
else
error = count - buf . count ;
}
2012-08-28 20:52:22 +04:00
fdput ( f ) ;
2005-04-17 02:20:36 +04:00
return error ;
}
struct getdents_callback64 {
2013-05-15 21:52:59 +04:00
struct dir_context ctx ;
2005-04-17 02:20:36 +04:00
struct linux_dirent64 __user * current_dir ;
struct linux_dirent64 __user * previous ;
int count ;
int error ;
} ;
static int filldir64 ( void * __buf , const char * name , int namlen , loff_t offset ,
2006-10-03 12:13:46 +04:00
u64 ino , unsigned int d_type )
2005-04-17 02:20:36 +04:00
{
struct linux_dirent64 __user * dirent ;
struct getdents_callback64 * buf = ( struct getdents_callback64 * ) __buf ;
2010-08-10 04:20:22 +04:00
int reclen = ALIGN ( offsetof ( struct linux_dirent64 , d_name ) + namlen + 1 ,
sizeof ( u64 ) ) ;
2005-04-17 02:20:36 +04:00
buf - > error = - EINVAL ; /* only used if we fail.. */
if ( reclen > buf - > count )
return - EINVAL ;
dirent = buf - > previous ;
if ( dirent ) {
if ( __put_user ( offset , & dirent - > d_off ) )
goto efault ;
}
dirent = buf - > current_dir ;
if ( __put_user ( ino , & dirent - > d_ino ) )
goto efault ;
if ( __put_user ( 0 , & dirent - > d_off ) )
goto efault ;
if ( __put_user ( reclen , & dirent - > d_reclen ) )
goto efault ;
if ( __put_user ( d_type , & dirent - > d_type ) )
goto efault ;
if ( copy_to_user ( dirent - > d_name , name , namlen ) )
goto efault ;
if ( __put_user ( 0 , dirent - > d_name + namlen ) )
goto efault ;
buf - > previous = dirent ;
dirent = ( void __user * ) dirent + reclen ;
buf - > current_dir = dirent ;
buf - > count - = reclen ;
return 0 ;
efault :
buf - > error = - EFAULT ;
return - EFAULT ;
}
2009-01-14 16:14:23 +03:00
SYSCALL_DEFINE3 ( getdents64 , unsigned int , fd ,
struct linux_dirent64 __user * , dirent , unsigned int , count )
2005-04-17 02:20:36 +04:00
{
2012-08-28 20:52:22 +04:00
struct fd f ;
2005-04-17 02:20:36 +04:00
struct linux_dirent64 __user * lastdirent ;
2013-05-23 06:22:04 +04:00
struct getdents_callback64 buf = {
. ctx . actor = filldir64 ,
. count = count ,
. current_dir = dirent
} ;
2005-04-17 02:20:36 +04:00
int error ;
if ( ! access_ok ( VERIFY_WRITE , dirent , count ) )
2012-04-22 02:40:32 +04:00
return - EFAULT ;
2005-04-17 02:20:36 +04:00
2012-08-28 20:52:22 +04:00
f = fdget ( fd ) ;
if ( ! f . file )
2012-04-22 02:40:32 +04:00
return - EBADF ;
2005-04-17 02:20:36 +04:00
2013-05-15 21:52:59 +04:00
error = iterate_dir ( f . file , & buf . ctx ) ;
2008-08-24 15:29:52 +04:00
if ( error > = 0 )
error = buf . error ;
2005-04-17 02:20:36 +04:00
lastdirent = buf . previous ;
if ( lastdirent ) {
2013-05-16 02:49:12 +04:00
typeof ( lastdirent - > d_off ) d_off = buf . ctx . pos ;
2005-04-17 02:20:36 +04:00
if ( __put_user ( d_off , & lastdirent - > d_off ) )
2008-08-24 15:29:52 +04:00
error = - EFAULT ;
else
error = count - buf . count ;
2005-04-17 02:20:36 +04:00
}
2012-08-28 20:52:22 +04:00
fdput ( f ) ;
2005-04-17 02:20:36 +04:00
return error ;
}