2018-03-06 03:15:50 +03:00
/* SPDX-License-Identifier: GPL-2.0 */
# include <linux/syscalls.h>
# include <linux/export.h>
# include <linux/uaccess.h>
# include <linux/fs_struct.h>
# include <linux/fs.h>
# include <linux/slab.h>
# include <linux/prefetch.h>
# include "mount.h"
2021-05-12 21:51:03 +03:00
struct prepend_buffer {
char * buf ;
int len ;
} ;
# define DECLARE_BUFFER(__name, __buf, __len) \
struct prepend_buffer __name = { . buf = __buf + __len , . len = __len }
static char * extract_string ( struct prepend_buffer * p )
2018-03-06 03:15:50 +03:00
{
2021-05-12 21:51:03 +03:00
if ( likely ( p - > len > = 0 ) )
return p - > buf ;
return ERR_PTR ( - ENAMETOOLONG ) ;
}
static void prepend ( struct prepend_buffer * p , const char * str , int namelen )
{
p - > len - = namelen ;
if ( likely ( p - > len > = 0 ) ) {
p - > buf - = namelen ;
memcpy ( p - > buf , str , namelen ) ;
2021-05-18 05:05:23 +03:00
}
2018-03-06 03:15:50 +03:00
}
/**
* prepend_name - prepend a pathname in front of current buffer pointer
* @ buffer : buffer pointer
* @ buflen : allocated length of the buffer
* @ name : name string and length qstr structure
*
* With RCU path tracing , it may race with d_move ( ) . Use READ_ONCE ( ) to
* make sure that either the old or the new name pointer and length are
* fetched . However , there may be mismatch between length and pointer .
* The length cannot be trusted , we need to copy it byte - by - byte until
* the length is reached or a null byte is found . It also prepends " / " at
* the beginning of the name . The sequence number check at the caller will
* retry it again when a d_move ( ) does happen . So any garbage in the buffer
* due to mismatched pointer and length will be discarded .
*
* Load acquire is needed to make sure that we see that terminating NUL .
*/
2021-05-12 21:51:03 +03:00
static bool prepend_name ( struct prepend_buffer * p , const struct qstr * name )
2018-03-06 03:15:50 +03:00
{
const char * dname = smp_load_acquire ( & name - > name ) ; /* ^^^ */
u32 dlen = READ_ONCE ( name - > len ) ;
2021-05-12 21:51:03 +03:00
char * s ;
2018-03-06 03:15:50 +03:00
2021-05-12 21:51:03 +03:00
p - > len - = dlen + 1 ;
if ( unlikely ( p - > len < 0 ) )
2021-05-18 05:41:11 +03:00
return false ;
2021-05-12 21:51:03 +03:00
s = p - > buf - = dlen + 1 ;
* s + + = ' / ' ;
2018-03-06 03:15:50 +03:00
while ( dlen - - ) {
char c = * dname + + ;
if ( ! c )
break ;
2021-05-12 21:51:03 +03:00
* s + + = c ;
2018-03-06 03:15:50 +03:00
}
2021-05-18 05:41:11 +03:00
return true ;
2018-03-06 03:15:50 +03:00
}
/**
* prepend_path - Prepend path string to a buffer
* @ path : the dentry / vfsmount to report
* @ root : root vfsmnt / dentry
* @ buffer : pointer to the end of the buffer
* @ buflen : pointer to buffer length
*
* The function will first try to write out the pathname without taking any
* lock other than the RCU read lock to make sure that dentries won ' t go away .
* It only checks the sequence number of the global rename_lock as any change
* in the dentry ' s d_seq will be preceded by changes in the rename_lock
* sequence number . If the sequence number had been changed , it will restart
* the whole pathname back - tracing sequence again by taking the rename_lock .
* In this case , there is no need to take the RCU read lock as the recursive
* parent pointer references will keep the dentry chain alive as long as no
* rename operation is performed .
*/
static int prepend_path ( const struct path * path ,
const struct path * root ,
2021-05-12 21:51:03 +03:00
struct prepend_buffer * p )
2018-03-06 03:15:50 +03:00
{
struct dentry * dentry ;
struct mount * mnt ;
int error = 0 ;
unsigned seq , m_seq = 0 ;
2021-05-12 21:51:03 +03:00
struct prepend_buffer b ;
2018-03-06 03:15:50 +03:00
rcu_read_lock ( ) ;
restart_mnt :
read_seqbegin_or_lock ( & mount_lock , & m_seq ) ;
seq = 0 ;
rcu_read_lock ( ) ;
restart :
2021-05-12 21:51:03 +03:00
b = * p ;
2018-03-06 03:15:50 +03:00
error = 0 ;
dentry = path - > dentry ;
2021-05-12 23:24:12 +03:00
mnt = real_mount ( path - > mnt ) ;
2018-03-06 03:15:50 +03:00
read_seqbegin_or_lock ( & rename_lock , & seq ) ;
2021-05-12 23:24:12 +03:00
while ( dentry ! = root - > dentry | | & mnt - > mnt ! = root - > mnt ) {
2018-03-06 03:15:50 +03:00
struct dentry * parent ;
2021-05-12 23:24:12 +03:00
if ( dentry = = mnt - > mnt . mnt_root | | IS_ROOT ( dentry ) ) {
2018-03-06 03:15:50 +03:00
struct mount * parent = READ_ONCE ( mnt - > mnt_parent ) ;
2020-10-14 23:45:28 +03:00
struct mnt_namespace * mnt_ns ;
2018-03-06 03:15:50 +03:00
/* Escaped? */
2021-05-12 23:24:12 +03:00
if ( dentry ! = mnt - > mnt . mnt_root ) {
2021-05-12 21:51:03 +03:00
b = * p ;
2018-03-06 03:15:50 +03:00
error = 3 ;
break ;
}
/* Global root? */
if ( mnt ! = parent ) {
dentry = READ_ONCE ( mnt - > mnt_mountpoint ) ;
mnt = parent ;
continue ;
}
2020-10-14 23:45:28 +03:00
mnt_ns = READ_ONCE ( mnt - > mnt_ns ) ;
/* open-coded is_mounted() to use local mnt_ns */
if ( ! IS_ERR_OR_NULL ( mnt_ns ) & & ! is_anon_ns ( mnt_ns ) )
2019-08-31 02:31:09 +03:00
error = 1 ; // absolute root
else
error = 2 ; // detached or not attached yet
2018-03-06 03:15:50 +03:00
break ;
}
parent = dentry - > d_parent ;
prefetch ( parent ) ;
2021-05-12 21:51:03 +03:00
if ( ! prepend_name ( & b , & dentry - > d_name ) )
2018-03-06 03:15:50 +03:00
break ;
dentry = parent ;
}
if ( ! ( seq & 1 ) )
rcu_read_unlock ( ) ;
if ( need_seqretry ( & rename_lock , seq ) ) {
seq = 1 ;
goto restart ;
}
done_seqretry ( & rename_lock , seq ) ;
if ( ! ( m_seq & 1 ) )
rcu_read_unlock ( ) ;
if ( need_seqretry ( & mount_lock , m_seq ) ) {
m_seq = 1 ;
goto restart_mnt ;
}
done_seqretry ( & mount_lock , m_seq ) ;
2021-05-12 21:51:03 +03:00
if ( b . len = = p - > len )
prepend ( & b , " / " , 1 ) ;
2021-05-18 05:29:03 +03:00
2021-05-12 21:51:03 +03:00
* p = b ;
2018-03-06 03:15:50 +03:00
return error ;
}
/**
* __d_path - return the path of a dentry
* @ path : the dentry / vfsmount to report
* @ root : root vfsmnt / dentry
* @ buf : buffer to return value in
* @ buflen : buffer length
*
* Convert a dentry into an ASCII path name .
*
* Returns a pointer into the buffer or an error code if the
* path was too long .
*
* " buflen " should be positive .
*
* If the path is not reachable from the supplied root , return % NULL .
*/
char * __d_path ( const struct path * path ,
const struct path * root ,
char * buf , int buflen )
{
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buf , buflen ) ;
2018-03-06 03:15:50 +03:00
2021-05-12 21:51:03 +03:00
prepend ( & b , " " , 1 ) ;
if ( prepend_path ( path , root , & b ) > 0 )
2018-03-06 03:15:50 +03:00
return NULL ;
2021-05-12 21:51:03 +03:00
return extract_string ( & b ) ;
2018-03-06 03:15:50 +03:00
}
char * d_absolute_path ( const struct path * path ,
char * buf , int buflen )
{
struct path root = { } ;
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buf , buflen ) ;
2018-03-06 03:15:50 +03:00
2021-05-12 21:51:03 +03:00
prepend ( & b , " " , 1 ) ;
if ( prepend_path ( path , & root , & b ) > 1 )
2021-05-18 05:29:03 +03:00
return ERR_PTR ( - EINVAL ) ;
2021-05-12 21:51:03 +03:00
return extract_string ( & b ) ;
2018-03-06 03:15:50 +03:00
}
static void get_fs_root_rcu ( struct fs_struct * fs , struct path * root )
{
unsigned seq ;
do {
seq = read_seqcount_begin ( & fs - > seq ) ;
* root = fs - > root ;
} while ( read_seqcount_retry ( & fs - > seq , seq ) ) ;
}
/**
* d_path - return the path of a dentry
* @ path : path to report
* @ buf : buffer to return value in
* @ buflen : buffer length
*
* Convert a dentry into an ASCII path name . If the entry has been deleted
* the string " (deleted) " is appended . Note that this is ambiguous .
*
* Returns a pointer into the buffer or an error code if the path was
* too long . Note : Callers should use the returned pointer , not the passed
* in buffer , to use the name ! The implementation often starts at an offset
* into the buffer , and may leave 0 bytes at the start .
*
* " buflen " should be positive .
*/
char * d_path ( const struct path * path , char * buf , int buflen )
{
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buf , buflen ) ;
2018-03-06 03:15:50 +03:00
struct path root ;
/*
* We have various synthetic filesystems that never get mounted . On
* these filesystems dentries are never used for lookup purposes , and
* thus don ' t need to be hashed . They also don ' t need a name until a
* user wants to identify the object in / proc / pid / fd / . The little hack
* below allows us to generate a name for these objects on demand :
*
* Some pseudo inodes are mountable . When they are mounted
* path - > dentry = = path - > mnt - > mnt_root . In that case don ' t call d_dname
* and instead have d_path return the mounted path .
*/
if ( path - > dentry - > d_op & & path - > dentry - > d_op - > d_dname & &
( ! IS_ROOT ( path - > dentry ) | | path - > dentry ! = path - > mnt - > mnt_root ) )
return path - > dentry - > d_op - > d_dname ( path - > dentry , buf , buflen ) ;
rcu_read_lock ( ) ;
get_fs_root_rcu ( current - > fs , & root ) ;
2021-05-18 04:43:01 +03:00
if ( unlikely ( d_unlinked ( path - > dentry ) ) )
2021-05-12 21:51:03 +03:00
prepend ( & b , " (deleted) " , 11 ) ;
2021-05-18 04:43:01 +03:00
else
2021-05-12 21:51:03 +03:00
prepend ( & b , " " , 1 ) ;
prepend_path ( path , & root , & b ) ;
2018-03-06 03:15:50 +03:00
rcu_read_unlock ( ) ;
2021-05-12 21:51:03 +03:00
return extract_string ( & b ) ;
2018-03-06 03:15:50 +03:00
}
EXPORT_SYMBOL ( d_path ) ;
/*
* Helper function for dentry_operations . d_dname ( ) members
*/
char * dynamic_dname ( struct dentry * dentry , char * buffer , int buflen ,
const char * fmt , . . . )
{
va_list args ;
char temp [ 64 ] ;
int sz ;
va_start ( args , fmt ) ;
sz = vsnprintf ( temp , sizeof ( temp ) , fmt , args ) + 1 ;
va_end ( args ) ;
if ( sz > sizeof ( temp ) | | sz > buflen )
return ERR_PTR ( - ENAMETOOLONG ) ;
buffer + = buflen - sz ;
return memcpy ( buffer , temp , sz ) ;
}
char * simple_dname ( struct dentry * dentry , char * buffer , int buflen )
{
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buffer , buflen ) ;
2018-03-06 03:15:50 +03:00
/* these dentries are never renamed, so d_lock is not needed */
2021-05-12 21:51:03 +03:00
prepend ( & b , " (deleted) " , 11 ) ;
prepend ( & b , dentry - > d_name . name , dentry - > d_name . len ) ;
prepend ( & b , " / " , 1 ) ;
return extract_string ( & b ) ;
2018-03-06 03:15:50 +03:00
}
/*
* Write full pathname from the root of the filesystem into the buffer .
*/
2021-05-12 21:51:03 +03:00
static char * __dentry_path ( const struct dentry * d , struct prepend_buffer * p )
2018-03-06 03:15:50 +03:00
{
2019-07-07 16:57:53 +03:00
const struct dentry * dentry ;
2021-05-12 21:51:03 +03:00
struct prepend_buffer b ;
int seq = 0 ;
2018-03-06 03:15:50 +03:00
rcu_read_lock ( ) ;
restart :
dentry = d ;
2021-05-12 21:51:03 +03:00
b = * p ;
2018-03-06 03:15:50 +03:00
read_seqbegin_or_lock ( & rename_lock , & seq ) ;
while ( ! IS_ROOT ( dentry ) ) {
2019-07-07 16:57:53 +03:00
const struct dentry * parent = dentry - > d_parent ;
2018-03-06 03:15:50 +03:00
prefetch ( parent ) ;
2021-05-12 21:51:03 +03:00
if ( ! prepend_name ( & b , & dentry - > d_name ) )
2018-03-06 03:15:50 +03:00
break ;
dentry = parent ;
}
if ( ! ( seq & 1 ) )
rcu_read_unlock ( ) ;
if ( need_seqretry ( & rename_lock , seq ) ) {
seq = 1 ;
goto restart ;
}
done_seqretry ( & rename_lock , seq ) ;
2021-05-12 21:51:03 +03:00
if ( b . len = = p - > len )
prepend ( & b , " / " , 1 ) ;
return extract_string ( & b ) ;
2018-03-06 03:15:50 +03:00
}
2019-07-07 16:57:53 +03:00
char * dentry_path_raw ( const struct dentry * dentry , char * buf , int buflen )
2018-03-06 03:15:50 +03:00
{
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buf , buflen ) ;
prepend ( & b , " " , 1 ) ;
return __dentry_path ( dentry , & b ) ;
2018-03-06 03:15:50 +03:00
}
EXPORT_SYMBOL ( dentry_path_raw ) ;
2019-07-07 16:57:53 +03:00
char * dentry_path ( const struct dentry * dentry , char * buf , int buflen )
2018-03-06 03:15:50 +03:00
{
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , buf , buflen ) ;
2021-05-18 03:16:51 +03:00
if ( unlikely ( d_unlinked ( dentry ) ) )
2021-05-12 21:51:03 +03:00
prepend ( & b , " //deleted " , 10 ) ;
2021-05-18 03:16:51 +03:00
else
2021-05-12 21:51:03 +03:00
prepend ( & b , " " , 1 ) ;
return __dentry_path ( dentry , & b ) ;
2018-03-06 03:15:50 +03:00
}
static void get_fs_root_and_pwd_rcu ( struct fs_struct * fs , struct path * root ,
struct path * pwd )
{
unsigned seq ;
do {
seq = read_seqcount_begin ( & fs - > seq ) ;
* root = fs - > root ;
* pwd = fs - > pwd ;
} while ( read_seqcount_retry ( & fs - > seq , seq ) ) ;
}
/*
* NOTE ! The user - level library version returns a
* character pointer . The kernel system call just
* returns the length of the buffer filled ( which
* includes the ending ' \0 ' character ) , or a negative
* error value . So libc would do something like
*
* char * getcwd ( char * buf , size_t size )
* {
* int retval ;
*
* retval = sys_getcwd ( buf , size ) ;
* if ( retval > = 0 )
* return buf ;
* errno = - retval ;
* return NULL ;
* }
*/
SYSCALL_DEFINE2 ( getcwd , char __user * , buf , unsigned long , size )
{
int error ;
struct path pwd , root ;
char * page = __getname ( ) ;
if ( ! page )
return - ENOMEM ;
rcu_read_lock ( ) ;
get_fs_root_and_pwd_rcu ( current - > fs , & root , & pwd ) ;
error = - ENOENT ;
if ( ! d_unlinked ( pwd . dentry ) ) {
unsigned long len ;
2021-05-12 21:51:03 +03:00
DECLARE_BUFFER ( b , page , PATH_MAX ) ;
2018-03-06 03:15:50 +03:00
2021-05-12 21:51:03 +03:00
prepend ( & b , " " , 1 ) ;
if ( prepend_path ( & pwd , & root , & b ) > 0 )
prepend ( & b , " (unreachable) " , 13 ) ;
2018-03-06 03:15:50 +03:00
rcu_read_unlock ( ) ;
2021-05-12 21:51:03 +03:00
if ( b . len < 0 ) {
2021-05-18 04:56:38 +03:00
error = - ENAMETOOLONG ;
2018-03-06 03:15:50 +03:00
goto out ;
}
error = - ERANGE ;
2021-05-12 21:51:03 +03:00
len = PATH_MAX - b . len ;
2018-03-06 03:15:50 +03:00
if ( len < = size ) {
error = len ;
2021-05-12 21:51:03 +03:00
if ( copy_to_user ( buf , b . buf , len ) )
2018-03-06 03:15:50 +03:00
error = - EFAULT ;
}
} else {
rcu_read_unlock ( ) ;
}
out :
__putname ( page ) ;
return error ;
}