2016-12-16 11:02:56 +01:00
/*
* Copyright ( C ) 2011 Novell Inc .
* Copyright ( C ) 2016 Red Hat , Inc .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 as published by
* the Free Software Foundation .
*/
# include <linux/fs.h>
# include <linux/namei.h>
# include <linux/xattr.h>
2016-12-16 11:02:56 +01:00
# include <linux/ratelimit.h>
2016-12-16 11:02:56 +01:00
# include "overlayfs.h"
# include "ovl_entry.h"
2016-12-16 11:02:56 +01:00
struct ovl_lookup_data {
struct qstr name ;
bool is_dir ;
bool opaque ;
bool stop ;
bool last ;
2016-12-16 11:02:56 +01:00
char * redirect ;
2016-12-16 11:02:56 +01:00
} ;
2016-12-16 11:02:56 +01:00
2016-12-16 11:02:56 +01:00
static int ovl_check_redirect ( struct dentry * dentry , struct ovl_lookup_data * d ,
size_t prelen , const char * post )
{
int res ;
char * s , * next , * buf = NULL ;
res = vfs_getxattr ( dentry , OVL_XATTR_REDIRECT , NULL , 0 ) ;
if ( res < 0 ) {
if ( res = = - ENODATA | | res = = - EOPNOTSUPP )
return 0 ;
goto fail ;
}
buf = kzalloc ( prelen + res + strlen ( post ) + 1 , GFP_TEMPORARY ) ;
if ( ! buf )
return - ENOMEM ;
if ( res = = 0 )
goto invalid ;
res = vfs_getxattr ( dentry , OVL_XATTR_REDIRECT , buf , res ) ;
if ( res < 0 )
goto fail ;
if ( res = = 0 )
goto invalid ;
if ( buf [ 0 ] = = ' / ' ) {
for ( s = buf ; * s + + = = ' / ' ; s = next ) {
next = strchrnul ( s , ' / ' ) ;
if ( s = = next )
goto invalid ;
}
} else {
if ( strchr ( buf , ' / ' ) ! = NULL )
goto invalid ;
memmove ( buf + prelen , buf , res ) ;
memcpy ( buf , d - > name . name , prelen ) ;
}
strcat ( buf , post ) ;
kfree ( d - > redirect ) ;
d - > redirect = buf ;
d - > name . name = d - > redirect ;
d - > name . len = strlen ( d - > redirect ) ;
return 0 ;
err_free :
kfree ( buf ) ;
return 0 ;
fail :
pr_warn_ratelimited ( " overlayfs: failed to get redirect (%i) \n " , res ) ;
goto err_free ;
invalid :
pr_warn_ratelimited ( " overlayfs: invalid redirect (%s) \n " , buf ) ;
goto err_free ;
}
2016-12-16 11:02:56 +01:00
static bool ovl_is_opaquedir ( struct dentry * dentry )
{
int res ;
char val ;
if ( ! d_is_dir ( dentry ) )
return false ;
res = vfs_getxattr ( dentry , OVL_XATTR_OPAQUE , & val , 1 ) ;
if ( res = = 1 & & val = = ' y ' )
return true ;
return false ;
}
2016-12-16 11:02:56 +01:00
static int ovl_lookup_single ( struct dentry * base , struct ovl_lookup_data * d ,
const char * name , unsigned int namelen ,
2016-12-16 11:02:56 +01:00
size_t prelen , const char * post ,
2016-12-16 11:02:56 +01:00
struct dentry * * ret )
{
struct dentry * this ;
int err ;
this = lookup_one_len_unlocked ( name , base , namelen ) ;
if ( IS_ERR ( this ) ) {
err = PTR_ERR ( this ) ;
this = NULL ;
if ( err = = - ENOENT | | err = = - ENAMETOOLONG )
goto out ;
goto out_err ;
}
if ( ! this - > d_inode )
goto put_and_out ;
if ( ovl_dentry_weird ( this ) ) {
/* Don't support traversing automounts and other weirdness */
err = - EREMOTE ;
goto out_err ;
}
if ( ovl_is_whiteout ( this ) ) {
d - > stop = d - > opaque = true ;
goto put_and_out ;
}
if ( ! d_can_lookup ( this ) ) {
d - > stop = true ;
if ( d - > is_dir )
goto put_and_out ;
goto out ;
}
d - > is_dir = true ;
if ( ! d - > last & & ovl_is_opaquedir ( this ) ) {
d - > stop = d - > opaque = true ;
goto out ;
}
2016-12-16 11:02:56 +01:00
err = ovl_check_redirect ( this , d , prelen , post ) ;
if ( err )
goto out_err ;
2016-12-16 11:02:56 +01:00
out :
* ret = this ;
return 0 ;
put_and_out :
dput ( this ) ;
this = NULL ;
goto out ;
out_err :
dput ( this ) ;
return err ;
}
static int ovl_lookup_layer ( struct dentry * base , struct ovl_lookup_data * d ,
struct dentry * * ret )
{
2017-01-18 15:19:54 +01:00
/* Counting down from the end, since the prefix can change */
size_t rem = d - > name . len - 1 ;
2016-12-16 11:02:56 +01:00
struct dentry * dentry = NULL ;
int err ;
2017-01-18 15:19:54 +01:00
if ( d - > name . name [ 0 ] ! = ' / ' )
2016-12-16 11:02:56 +01:00
return ovl_lookup_single ( base , d , d - > name . name , d - > name . len ,
0 , " " , ret ) ;
2017-01-18 15:19:54 +01:00
while ( ! IS_ERR_OR_NULL ( base ) & & d_can_lookup ( base ) ) {
const char * s = d - > name . name + d - > name . len - rem ;
2016-12-16 11:02:56 +01:00
const char * next = strchrnul ( s , ' / ' ) ;
2017-01-18 15:19:54 +01:00
size_t thislen = next - s ;
bool end = ! next [ 0 ] ;
2016-12-16 11:02:56 +01:00
2017-01-18 15:19:54 +01:00
/* Verify we did not go off the rails */
if ( WARN_ON ( s [ - 1 ] ! = ' / ' ) )
2016-12-16 11:02:56 +01:00
return - EIO ;
2017-01-18 15:19:54 +01:00
err = ovl_lookup_single ( base , d , s , thislen ,
d - > name . len - rem , next , & base ) ;
2016-12-16 11:02:56 +01:00
dput ( dentry ) ;
if ( err )
return err ;
dentry = base ;
2017-01-18 15:19:54 +01:00
if ( end )
break ;
rem - = thislen + 1 ;
if ( WARN_ON ( rem > = d - > name . len ) )
return - EIO ;
2016-12-16 11:02:56 +01:00
}
* ret = dentry ;
return 0 ;
2016-12-16 11:02:56 +01:00
}
2016-12-16 11:02:56 +01:00
/*
* Returns next layer in stack starting from top .
* Returns - 1 if this is the last layer .
*/
int ovl_path_next ( int idx , struct dentry * dentry , struct path * path )
{
struct ovl_entry * oe = dentry - > d_fsdata ;
BUG_ON ( idx < 0 ) ;
if ( idx = = 0 ) {
ovl_path_upper ( dentry , path ) ;
if ( path - > dentry )
return oe - > numlower ? 1 : - 1 ;
idx + + ;
}
BUG_ON ( idx > oe - > numlower ) ;
* path = oe - > lowerstack [ idx - 1 ] ;
return ( idx < oe - > numlower ) ? idx + 1 : - 1 ;
}
struct dentry * ovl_lookup ( struct inode * dir , struct dentry * dentry ,
unsigned int flags )
{
struct ovl_entry * oe ;
const struct cred * old_cred ;
2016-12-16 11:02:56 +01:00
struct ovl_fs * ofs = dentry - > d_sb - > s_fs_info ;
2016-12-16 11:02:56 +01:00
struct ovl_entry * poe = dentry - > d_parent - > d_fsdata ;
struct path * stack = NULL ;
struct dentry * upperdir , * upperdentry = NULL ;
unsigned int ctr = 0 ;
struct inode * inode = NULL ;
bool upperopaque = false ;
2016-12-16 11:02:56 +01:00
char * upperredirect = NULL ;
2016-12-16 11:02:56 +01:00
struct dentry * this ;
unsigned int i ;
int err ;
2016-12-16 11:02:56 +01:00
struct ovl_lookup_data d = {
. name = dentry - > d_name ,
. is_dir = false ,
. opaque = false ,
. stop = false ,
. last = ! poe - > numlower ,
2016-12-16 11:02:56 +01:00
. redirect = NULL ,
2016-12-16 11:02:56 +01:00
} ;
2016-12-16 11:02:56 +01:00
2016-12-16 11:02:56 +01:00
if ( dentry - > d_name . len > ofs - > namelen )
return ERR_PTR ( - ENAMETOOLONG ) ;
2016-12-16 11:02:56 +01:00
old_cred = ovl_override_creds ( dentry - > d_sb ) ;
upperdir = ovl_upperdentry_dereference ( poe ) ;
if ( upperdir ) {
2016-12-16 11:02:56 +01:00
err = ovl_lookup_layer ( upperdir , & d , & upperdentry ) ;
if ( err )
2016-12-16 11:02:56 +01:00
goto out ;
2016-12-16 11:02:56 +01:00
if ( upperdentry & & unlikely ( ovl_dentry_remote ( upperdentry ) ) ) {
dput ( upperdentry ) ;
err = - EREMOTE ;
goto out ;
2016-12-16 11:02:56 +01:00
}
2016-12-16 11:02:56 +01:00
if ( d . redirect ) {
upperredirect = kstrdup ( d . redirect , GFP_KERNEL ) ;
if ( ! upperredirect )
goto out_put_upper ;
if ( d . redirect [ 0 ] = = ' / ' )
poe = dentry - > d_sb - > s_root - > d_fsdata ;
}
2016-12-16 11:02:56 +01:00
upperopaque = d . opaque ;
2016-12-16 11:02:56 +01:00
}
2016-12-16 11:02:56 +01:00
if ( ! d . stop & & poe - > numlower ) {
2016-12-16 11:02:56 +01:00
err = - ENOMEM ;
2016-12-16 11:02:56 +01:00
stack = kcalloc ( ofs - > numlower , sizeof ( struct path ) ,
2016-12-16 11:02:56 +01:00
GFP_TEMPORARY ) ;
2016-12-16 11:02:56 +01:00
if ( ! stack )
goto out_put_upper ;
}
2016-12-16 11:02:56 +01:00
for ( i = 0 ; ! d . stop & & i < poe - > numlower ; i + + ) {
2016-12-16 11:02:56 +01:00
struct path lowerpath = poe - > lowerstack [ i ] ;
2016-12-16 11:02:56 +01:00
d . last = i = = poe - > numlower - 1 ;
err = ovl_lookup_layer ( lowerpath . dentry , & d , & this ) ;
if ( err )
2016-12-16 11:02:56 +01:00
goto out_put ;
2016-12-16 11:02:56 +01:00
2016-12-16 11:02:56 +01:00
if ( ! this )
continue ;
stack [ ctr ] . dentry = this ;
stack [ ctr ] . mnt = lowerpath . mnt ;
ctr + + ;
2016-12-16 11:02:56 +01:00
if ( d . stop )
break ;
if ( d . redirect & &
d . redirect [ 0 ] = = ' / ' & &
poe ! = dentry - > d_sb - > s_root - > d_fsdata ) {
poe = dentry - > d_sb - > s_root - > d_fsdata ;
/* Find the current layer on the root dentry */
for ( i = 0 ; i < poe - > numlower ; i + + )
if ( poe - > lowerstack [ i ] . mnt = = lowerpath . mnt )
break ;
if ( WARN_ON ( i = = poe - > numlower ) )
break ;
}
2016-12-16 11:02:56 +01:00
}
oe = ovl_alloc_entry ( ctr ) ;
err = - ENOMEM ;
if ( ! oe )
goto out_put ;
if ( upperdentry | | ctr ) {
struct dentry * realdentry ;
struct inode * realinode ;
realdentry = upperdentry ? upperdentry : stack [ 0 ] . dentry ;
realinode = d_inode ( realdentry ) ;
err = - ENOMEM ;
if ( upperdentry & & ! d_is_dir ( upperdentry ) ) {
inode = ovl_get_inode ( dentry - > d_sb , realinode ) ;
} else {
inode = ovl_new_inode ( dentry - > d_sb , realinode - > i_mode ,
realinode - > i_rdev ) ;
if ( inode )
ovl_inode_init ( inode , realinode , ! ! upperdentry ) ;
}
if ( ! inode )
goto out_free_oe ;
ovl_copyattr ( realdentry - > d_inode , inode ) ;
}
revert_creds ( old_cred ) ;
oe - > opaque = upperopaque ;
2016-12-16 11:02:56 +01:00
oe - > redirect = upperredirect ;
2016-12-16 11:02:56 +01:00
oe - > __upperdentry = upperdentry ;
memcpy ( oe - > lowerstack , stack , sizeof ( struct path ) * ctr ) ;
kfree ( stack ) ;
2016-12-16 11:02:56 +01:00
kfree ( d . redirect ) ;
2016-12-16 11:02:56 +01:00
dentry - > d_fsdata = oe ;
d_add ( dentry , inode ) ;
return NULL ;
out_free_oe :
kfree ( oe ) ;
out_put :
for ( i = 0 ; i < ctr ; i + + )
dput ( stack [ i ] . dentry ) ;
kfree ( stack ) ;
out_put_upper :
dput ( upperdentry ) ;
2016-12-16 11:02:56 +01:00
kfree ( upperredirect ) ;
2016-12-16 11:02:56 +01:00
out :
2016-12-16 11:02:56 +01:00
kfree ( d . redirect ) ;
2016-12-16 11:02:56 +01:00
revert_creds ( old_cred ) ;
return ERR_PTR ( err ) ;
}
bool ovl_lower_positive ( struct dentry * dentry )
{
struct ovl_entry * oe = dentry - > d_fsdata ;
struct ovl_entry * poe = dentry - > d_parent - > d_fsdata ;
const struct qstr * name = & dentry - > d_name ;
unsigned int i ;
bool positive = false ;
bool done = false ;
/*
* If dentry is negative , then lower is positive iff this is a
* whiteout .
*/
if ( ! dentry - > d_inode )
return oe - > opaque ;
/* Negative upper -> positive lower */
if ( ! oe - > __upperdentry )
return true ;
/* Positive upper -> have to look up lower to see whether it exists */
for ( i = 0 ; ! done & & ! positive & & i < poe - > numlower ; i + + ) {
struct dentry * this ;
struct dentry * lowerdir = poe - > lowerstack [ i ] . dentry ;
this = lookup_one_len_unlocked ( name - > name , lowerdir ,
name - > len ) ;
if ( IS_ERR ( this ) ) {
switch ( PTR_ERR ( this ) ) {
case - ENOENT :
case - ENAMETOOLONG :
break ;
default :
/*
* Assume something is there , we just couldn ' t
* access it .
*/
positive = true ;
break ;
}
} else {
if ( this - > d_inode ) {
positive = ! ovl_is_whiteout ( this ) ;
done = true ;
}
dput ( this ) ;
}
}
return positive ;
}