f2fs: fix to return error number of read_all_xattrs correctly
authorChao Yu <yuchao0@huawei.com>
Sun, 18 Sep 2016 15:30:04 +0000 (23:30 +0800)
committerJaegeuk Kim <jaegeuk@kernel.org>
Thu, 22 Sep 2016 18:43:05 +0000 (11:43 -0700)
We treat all error in read_all_xattrs as a no memory error, which covers
the real reason of failure in it. Fix it by return correct errno in order
to reflect the real cause.

Signed-off-by: Chao Yu <yuchao0@huawei.com>
Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
fs/f2fs/xattr.c

index d39a7923de423bdf5f599851566f24e54302e011..1f74876233b66f2e1b4c6aea17b740eaae29ff3f 100644 (file)
@@ -217,18 +217,20 @@ static struct f2fs_xattr_entry *__find_xattr(void *base_addr, int index,
        return entry;
 }
 
-static void *read_all_xattrs(struct inode *inode, struct page *ipage)
+static int read_all_xattrs(struct inode *inode, struct page *ipage,
+                                                       void **base_addr)
 {
        struct f2fs_sb_info *sbi = F2FS_I_SB(inode);
        struct f2fs_xattr_header *header;
        size_t size = PAGE_SIZE, inline_size = 0;
        void *txattr_addr;
+       int err;
 
        inline_size = inline_xattr_size(inode);
 
        txattr_addr = kzalloc(inline_size + size, GFP_F2FS_ZERO);
        if (!txattr_addr)
-               return NULL;
+               return -ENOMEM;
 
        /* read from inline xattr */
        if (inline_size) {
@@ -239,8 +241,10 @@ static void *read_all_xattrs(struct inode *inode, struct page *ipage)
                        inline_addr = inline_xattr_addr(ipage);
                } else {
                        page = get_node_page(sbi, inode->i_ino);
-                       if (IS_ERR(page))
+                       if (IS_ERR(page)) {
+                               err = PTR_ERR(page);
                                goto fail;
+                       }
                        inline_addr = inline_xattr_addr(page);
                }
                memcpy(txattr_addr, inline_addr, inline_size);
@@ -254,8 +258,10 @@ static void *read_all_xattrs(struct inode *inode, struct page *ipage)
 
                /* The inode already has an extended attribute block. */
                xpage = get_node_page(sbi, F2FS_I(inode)->i_xattr_nid);
-               if (IS_ERR(xpage))
+               if (IS_ERR(xpage)) {
+                       err = PTR_ERR(xpage);
                        goto fail;
+               }
 
                xattr_addr = page_address(xpage);
                memcpy(txattr_addr + inline_size, xattr_addr, PAGE_SIZE);
@@ -269,10 +275,11 @@ static void *read_all_xattrs(struct inode *inode, struct page *ipage)
                header->h_magic = cpu_to_le32(F2FS_XATTR_MAGIC);
                header->h_refcount = cpu_to_le32(1);
        }
-       return txattr_addr;
+       *base_addr = txattr_addr;
+       return 0;
 fail:
        kzfree(txattr_addr);
-       return NULL;
+       return err;
 }
 
 static inline int write_all_xattrs(struct inode *inode, __u32 hsize,
@@ -366,9 +373,9 @@ int f2fs_getxattr(struct inode *inode, int index, const char *name,
        if (len > F2FS_NAME_LEN)
                return -ERANGE;
 
-       base_addr = read_all_xattrs(inode, ipage);
-       if (!base_addr)
-               return -ENOMEM;
+       error = read_all_xattrs(inode, ipage, &base_addr);
+       if (error)
+               return error;
 
        entry = __find_xattr(base_addr, index, len, name);
        if (IS_XATTR_LAST_ENTRY(entry)) {
@@ -402,9 +409,9 @@ ssize_t f2fs_listxattr(struct dentry *dentry, char *buffer, size_t buffer_size)
        int error = 0;
        size_t rest = buffer_size;
 
-       base_addr = read_all_xattrs(inode, NULL);
-       if (!base_addr)
-               return -ENOMEM;
+       error = read_all_xattrs(inode, NULL, &base_addr);
+       if (error)
+               return error;
 
        list_for_each_xattr(entry, base_addr) {
                const struct xattr_handler *handler =
@@ -463,9 +470,9 @@ static int __f2fs_setxattr(struct inode *inode, int index,
        if (size > MAX_VALUE_LEN(inode))
                return -E2BIG;
 
-       base_addr = read_all_xattrs(inode, ipage);
-       if (!base_addr)
-               return -ENOMEM;
+       error = read_all_xattrs(inode, ipage, &base_addr);
+       if (error)
+               return error;
 
        /* find entry with wanted name. */
        here = __find_xattr(base_addr, index, len, name);