[PATCH v2 03/15] vfs: make fstatat retry on ESTALE errors from getattr call

From: Jeff Layton
Date: Tue May 22 2012 - 10:15:59 EST


Signed-off-by: Jeff Layton <jlayton@xxxxxxxxxx>
---
fs/namei.c | 2 +-
fs/stat.c | 21 ++++++++++++++++-----
include/linux/fs.h | 3 ++-
3 files changed, 19 insertions(+), 7 deletions(-)

diff --git a/fs/namei.c b/fs/namei.c
index c63242f..516be34 100644
--- a/fs/namei.c
+++ b/fs/namei.c
@@ -116,7 +116,7 @@
* POSIX.1 2.4: an empty pathname is invalid (ENOENT).
* PATH_MAX includes the nul terminator --RR.
*/
-static char *getname_flags(const char __user *filename, int flags, int *empty)
+char *getname_flags(const char __user *filename, int flags, int *empty)
{
char *result = __getname(), *err;
int len;
diff --git a/fs/stat.c b/fs/stat.c
index 0cef336..60a8813 100644
--- a/fs/stat.c
+++ b/fs/stat.c
@@ -72,9 +72,11 @@ EXPORT_SYMBOL(vfs_fstat);
int vfs_fstatat(int dfd, const char __user *filename, struct kstat *stat,
int flag)
{
+ char *name;
struct path path;
int error = -EINVAL;
- int lookup_flags = 0;
+ unsigned int try = 0;
+ unsigned int lookup_flags = 0;

if ((flag & ~(AT_SYMLINK_NOFOLLOW | AT_NO_AUTOMOUNT |
AT_EMPTY_PATH)) != 0)
@@ -85,12 +87,21 @@ int vfs_fstatat(int dfd, const char __user *filename, struct kstat *stat,
if (flag & AT_EMPTY_PATH)
lookup_flags |= LOOKUP_EMPTY;

- error = user_path_at(dfd, filename, lookup_flags, &path);
- if (error)
+ name = getname_flags(filename, lookup_flags, NULL);
+ if (IS_ERR(name)) {
+ error = PTR_ERR(name);
goto out;
+ }
+ do {
+ error = kern_path_at(dfd, name, lookup_flags, &path);
+ if (error)
+ break;

- error = vfs_getattr(path.mnt, path.dentry, stat);
- path_put(&path);
+ error = vfs_getattr(path.mnt, path.dentry, stat);
+ path_put(&path);
+ lookup_flags |= LOOKUP_REVAL;
+ } while (retry_estale(error, try++));
+ putname(name);
out:
return error;
}
diff --git a/include/linux/fs.h b/include/linux/fs.h
index 0b8003b..2f92b93 100644
--- a/include/linux/fs.h
+++ b/include/linux/fs.h
@@ -2024,7 +2024,8 @@ extern struct file *file_open_root(struct dentry *, struct vfsmount *,
extern struct file * dentry_open(struct dentry *, struct vfsmount *, int,
const struct cred *);
extern int filp_close(struct file *, fl_owner_t id);
-extern char * getname(const char __user *);
+extern char *getname_flags(const char __user *, int, int *);
+extern char *getname(const char __user *);

/**
* retry_estale - determine whether the caller should retry an operation
--
1.7.7.6

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/