diff options
-rw-r--r-- | fs/nfs/dir.c | 15 | ||||
-rw-r--r-- | fs/nfs/internal.h | 9 | ||||
-rw-r--r-- | fs/nfs/nfs2xdr.c | 18 | ||||
-rw-r--r-- | fs/nfs/nfs3xdr.c | 28 | ||||
-rw-r--r-- | fs/nfs/nfs4_fs.h | 1 | ||||
-rw-r--r-- | fs/nfs/nfs4xdr.c | 29 | ||||
-rw-r--r-- | include/linux/nfs_xdr.h | 3 |
7 files changed, 61 insertions, 42 deletions
diff --git a/fs/nfs/dir.c b/fs/nfs/dir.c index 996dd8989a91..3e2123fe79f5 100644 --- a/fs/nfs/dir.c +++ b/fs/nfs/dir.c | |||
@@ -172,7 +172,7 @@ struct nfs_cache_array { | |||
172 | struct nfs_cache_array_entry array[0]; | 172 | struct nfs_cache_array_entry array[0]; |
173 | }; | 173 | }; |
174 | 174 | ||
175 | typedef __be32 * (*decode_dirent_t)(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int); | 175 | typedef int (*decode_dirent_t)(struct xdr_stream *, struct nfs_entry *, int); |
176 | typedef struct { | 176 | typedef struct { |
177 | struct file *file; | 177 | struct file *file; |
178 | struct page *page; | 178 | struct page *page; |
@@ -378,14 +378,14 @@ error: | |||
378 | return error; | 378 | return error; |
379 | } | 379 | } |
380 | 380 | ||
381 | /* Fill in an entry based on the xdr code stored in desc->page */ | 381 | static int xdr_decode(nfs_readdir_descriptor_t *desc, |
382 | static | 382 | struct nfs_entry *entry, struct xdr_stream *xdr) |
383 | int xdr_decode(nfs_readdir_descriptor_t *desc, struct nfs_entry *entry, struct xdr_stream *stream) | ||
384 | { | 383 | { |
385 | __be32 *p = desc->decode(stream, entry, NFS_SERVER(desc->file->f_path.dentry->d_inode), desc->plus); | 384 | int error; |
386 | if (IS_ERR(p)) | ||
387 | return PTR_ERR(p); | ||
388 | 385 | ||
386 | error = desc->decode(xdr, entry, desc->plus); | ||
387 | if (error) | ||
388 | return error; | ||
389 | entry->fattr->time_start = desc->timestamp; | 389 | entry->fattr->time_start = desc->timestamp; |
390 | entry->fattr->gencount = desc->gencount; | 390 | entry->fattr->gencount = desc->gencount; |
391 | return 0; | 391 | return 0; |
@@ -566,6 +566,7 @@ int nfs_readdir_xdr_to_array(nfs_readdir_descriptor_t *desc, struct page *page, | |||
566 | entry.eof = 0; | 566 | entry.eof = 0; |
567 | entry.fh = nfs_alloc_fhandle(); | 567 | entry.fh = nfs_alloc_fhandle(); |
568 | entry.fattr = nfs_alloc_fattr(); | 568 | entry.fattr = nfs_alloc_fattr(); |
569 | entry.server = NFS_SERVER(inode); | ||
569 | if (entry.fh == NULL || entry.fattr == NULL) | 570 | if (entry.fh == NULL || entry.fattr == NULL) |
570 | goto out; | 571 | goto out; |
571 | 572 | ||
diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h index 6c6a9955bae9..435eae3666bd 100644 --- a/fs/nfs/internal.h +++ b/fs/nfs/internal.h | |||
@@ -187,15 +187,18 @@ extern void nfs_destroy_directcache(void); | |||
187 | /* nfs2xdr.c */ | 187 | /* nfs2xdr.c */ |
188 | extern int nfs_stat_to_errno(enum nfs_stat); | 188 | extern int nfs_stat_to_errno(enum nfs_stat); |
189 | extern struct rpc_procinfo nfs_procedures[]; | 189 | extern struct rpc_procinfo nfs_procedures[]; |
190 | extern __be32 *nfs2_decode_dirent(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int); | 190 | extern int nfs2_decode_dirent(struct xdr_stream *, |
191 | struct nfs_entry *, int); | ||
191 | 192 | ||
192 | /* nfs3xdr.c */ | 193 | /* nfs3xdr.c */ |
193 | extern struct rpc_procinfo nfs3_procedures[]; | 194 | extern struct rpc_procinfo nfs3_procedures[]; |
194 | extern __be32 *nfs3_decode_dirent(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int); | 195 | extern int nfs3_decode_dirent(struct xdr_stream *, |
196 | struct nfs_entry *, int); | ||
195 | 197 | ||
196 | /* nfs4xdr.c */ | 198 | /* nfs4xdr.c */ |
197 | #ifdef CONFIG_NFS_V4 | 199 | #ifdef CONFIG_NFS_V4 |
198 | extern __be32 *nfs4_decode_dirent(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int); | 200 | extern int nfs4_decode_dirent(struct xdr_stream *, |
201 | struct nfs_entry *, int); | ||
199 | #endif | 202 | #endif |
200 | #ifdef CONFIG_NFS_V4_1 | 203 | #ifdef CONFIG_NFS_V4_1 |
201 | extern const u32 nfs41_maxread_overhead; | 204 | extern const u32 nfs41_maxread_overhead; |
diff --git a/fs/nfs/nfs2xdr.c b/fs/nfs/nfs2xdr.c index 0343175fe6c0..a9b848edbd2e 100644 --- a/fs/nfs/nfs2xdr.c +++ b/fs/nfs/nfs2xdr.c | |||
@@ -936,10 +936,10 @@ static int nfs2_xdr_dec_writeres(struct rpc_rqst *req, __be32 *p, | |||
936 | * the local page cache. | 936 | * the local page cache. |
937 | * @xdr: XDR stream where entry resides | 937 | * @xdr: XDR stream where entry resides |
938 | * @entry: buffer to fill in with entry data | 938 | * @entry: buffer to fill in with entry data |
939 | * @server: nfs_server data for this directory | ||
940 | * @plus: boolean indicating whether this should be a readdirplus entry | 939 | * @plus: boolean indicating whether this should be a readdirplus entry |
941 | * | 940 | * |
942 | * Returns the position of the next item in the buffer, or an ERR_PTR. | 941 | * Returns zero if successful, otherwise a negative errno value is |
942 | * returned. | ||
943 | * | 943 | * |
944 | * This function is not invoked during READDIR reply decoding, but | 944 | * This function is not invoked during READDIR reply decoding, but |
945 | * rather whenever an application invokes the getdents(2) system call | 945 | * rather whenever an application invokes the getdents(2) system call |
@@ -954,8 +954,8 @@ static int nfs2_xdr_dec_writeres(struct rpc_rqst *req, __be32 *p, | |||
954 | * entry *nextentry; | 954 | * entry *nextentry; |
955 | * }; | 955 | * }; |
956 | */ | 956 | */ |
957 | __be32 *nfs2_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | 957 | int nfs2_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, |
958 | struct nfs_server *server, int plus) | 958 | int plus) |
959 | { | 959 | { |
960 | __be32 *p; | 960 | __be32 *p; |
961 | int error; | 961 | int error; |
@@ -968,9 +968,9 @@ __be32 *nfs2_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
968 | if (unlikely(p == NULL)) | 968 | if (unlikely(p == NULL)) |
969 | goto out_overflow; | 969 | goto out_overflow; |
970 | if (*p++ == xdr_zero) | 970 | if (*p++ == xdr_zero) |
971 | return ERR_PTR(-EAGAIN); | 971 | return -EAGAIN; |
972 | entry->eof = 1; | 972 | entry->eof = 1; |
973 | return ERR_PTR(-EBADCOOKIE); | 973 | return -EBADCOOKIE; |
974 | } | 974 | } |
975 | 975 | ||
976 | p = xdr_inline_decode(xdr, 4); | 976 | p = xdr_inline_decode(xdr, 4); |
@@ -980,7 +980,7 @@ __be32 *nfs2_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
980 | 980 | ||
981 | error = decode_filename_inline(xdr, &entry->name, &entry->len); | 981 | error = decode_filename_inline(xdr, &entry->name, &entry->len); |
982 | if (unlikely(error)) | 982 | if (unlikely(error)) |
983 | return ERR_PTR(error); | 983 | return error; |
984 | 984 | ||
985 | /* | 985 | /* |
986 | * The type (size and byte order) of nfscookie isn't defined in | 986 | * The type (size and byte order) of nfscookie isn't defined in |
@@ -999,11 +999,11 @@ __be32 *nfs2_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
999 | entry->eof = 0; | 999 | entry->eof = 0; |
1000 | if (p != NULL) | 1000 | if (p != NULL) |
1001 | entry->eof = (p[0] == xdr_zero) && (p[1] != xdr_zero); | 1001 | entry->eof = (p[0] == xdr_zero) && (p[1] != xdr_zero); |
1002 | return p; | 1002 | return 0; |
1003 | 1003 | ||
1004 | out_overflow: | 1004 | out_overflow: |
1005 | print_overflow_msg(__func__, xdr); | 1005 | print_overflow_msg(__func__, xdr); |
1006 | return ERR_PTR(-EAGAIN); | 1006 | return -EAGAIN; |
1007 | } | 1007 | } |
1008 | 1008 | ||
1009 | /* | 1009 | /* |
diff --git a/fs/nfs/nfs3xdr.c b/fs/nfs/nfs3xdr.c index c97d00fe849a..15c93ccd90c5 100644 --- a/fs/nfs/nfs3xdr.c +++ b/fs/nfs/nfs3xdr.c | |||
@@ -1970,10 +1970,10 @@ out_status: | |||
1970 | * the local page cache | 1970 | * the local page cache |
1971 | * @xdr: XDR stream where entry resides | 1971 | * @xdr: XDR stream where entry resides |
1972 | * @entry: buffer to fill in with entry data | 1972 | * @entry: buffer to fill in with entry data |
1973 | * @server: nfs_server data for this directory | ||
1974 | * @plus: boolean indicating whether this should be a readdirplus entry | 1973 | * @plus: boolean indicating whether this should be a readdirplus entry |
1975 | * | 1974 | * |
1976 | * Returns the position of the next item in the buffer, or an ERR_PTR. | 1975 | * Returns zero if successful, otherwise a negative errno value is |
1976 | * returned. | ||
1977 | * | 1977 | * |
1978 | * This function is not invoked during READDIR reply decoding, but | 1978 | * This function is not invoked during READDIR reply decoding, but |
1979 | * rather whenever an application invokes the getdents(2) system call | 1979 | * rather whenever an application invokes the getdents(2) system call |
@@ -2000,8 +2000,8 @@ out_status: | |||
2000 | * entryplus3 *nextentry; | 2000 | * entryplus3 *nextentry; |
2001 | * }; | 2001 | * }; |
2002 | */ | 2002 | */ |
2003 | __be32 *nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | 2003 | int nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, |
2004 | struct nfs_server *server, int plus) | 2004 | int plus) |
2005 | { | 2005 | { |
2006 | struct nfs_entry old = *entry; | 2006 | struct nfs_entry old = *entry; |
2007 | __be32 *p; | 2007 | __be32 *p; |
@@ -2015,23 +2015,23 @@ __be32 *nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
2015 | if (unlikely(p == NULL)) | 2015 | if (unlikely(p == NULL)) |
2016 | goto out_overflow; | 2016 | goto out_overflow; |
2017 | if (*p == xdr_zero) | 2017 | if (*p == xdr_zero) |
2018 | return ERR_PTR(-EAGAIN); | 2018 | return -EAGAIN; |
2019 | entry->eof = 1; | 2019 | entry->eof = 1; |
2020 | return ERR_PTR(-EBADCOOKIE); | 2020 | return -EBADCOOKIE; |
2021 | } | 2021 | } |
2022 | 2022 | ||
2023 | error = decode_fileid3(xdr, &entry->ino); | 2023 | error = decode_fileid3(xdr, &entry->ino); |
2024 | if (unlikely(error)) | 2024 | if (unlikely(error)) |
2025 | return ERR_PTR(error); | 2025 | return error; |
2026 | 2026 | ||
2027 | error = decode_inline_filename3(xdr, &entry->name, &entry->len); | 2027 | error = decode_inline_filename3(xdr, &entry->name, &entry->len); |
2028 | if (unlikely(error)) | 2028 | if (unlikely(error)) |
2029 | return ERR_PTR(error); | 2029 | return error; |
2030 | 2030 | ||
2031 | entry->prev_cookie = entry->cookie; | 2031 | entry->prev_cookie = entry->cookie; |
2032 | error = decode_cookie3(xdr, &entry->cookie); | 2032 | error = decode_cookie3(xdr, &entry->cookie); |
2033 | if (unlikely(error)) | 2033 | if (unlikely(error)) |
2034 | return ERR_PTR(error); | 2034 | return error; |
2035 | 2035 | ||
2036 | entry->d_type = DT_UNKNOWN; | 2036 | entry->d_type = DT_UNKNOWN; |
2037 | 2037 | ||
@@ -2039,7 +2039,7 @@ __be32 *nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
2039 | entry->fattr->valid = 0; | 2039 | entry->fattr->valid = 0; |
2040 | error = decode_post_op_attr(xdr, entry->fattr); | 2040 | error = decode_post_op_attr(xdr, entry->fattr); |
2041 | if (unlikely(error)) | 2041 | if (unlikely(error)) |
2042 | return ERR_PTR(error); | 2042 | return error; |
2043 | if (entry->fattr->valid & NFS_ATTR_FATTR_V3) | 2043 | if (entry->fattr->valid & NFS_ATTR_FATTR_V3) |
2044 | entry->d_type = nfs_umode_to_dtype(entry->fattr->mode); | 2044 | entry->d_type = nfs_umode_to_dtype(entry->fattr->mode); |
2045 | 2045 | ||
@@ -2052,7 +2052,7 @@ __be32 *nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
2052 | if (unlikely(error)) { | 2052 | if (unlikely(error)) { |
2053 | if (error == -E2BIG) | 2053 | if (error == -E2BIG) |
2054 | goto out_truncated; | 2054 | goto out_truncated; |
2055 | return ERR_PTR(error); | 2055 | return error; |
2056 | } | 2056 | } |
2057 | } else | 2057 | } else |
2058 | zero_nfs_fh3(entry->fh); | 2058 | zero_nfs_fh3(entry->fh); |
@@ -2063,15 +2063,15 @@ __be32 *nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
2063 | entry->eof = 0; | 2063 | entry->eof = 0; |
2064 | if (p != NULL) | 2064 | if (p != NULL) |
2065 | entry->eof = (p[0] == xdr_zero) && (p[1] != xdr_zero); | 2065 | entry->eof = (p[0] == xdr_zero) && (p[1] != xdr_zero); |
2066 | return p; | 2066 | return 0; |
2067 | 2067 | ||
2068 | out_overflow: | 2068 | out_overflow: |
2069 | print_overflow_msg(__func__, xdr); | 2069 | print_overflow_msg(__func__, xdr); |
2070 | return ERR_PTR(-EAGAIN); | 2070 | return -EAGAIN; |
2071 | out_truncated: | 2071 | out_truncated: |
2072 | dprintk("NFS: directory entry contains invalid file handle\n"); | 2072 | dprintk("NFS: directory entry contains invalid file handle\n"); |
2073 | *entry = old; | 2073 | *entry = old; |
2074 | return ERR_PTR(-EAGAIN); | 2074 | return -EAGAIN; |
2075 | } | 2075 | } |
2076 | 2076 | ||
2077 | /* | 2077 | /* |
diff --git a/fs/nfs/nfs4_fs.h b/fs/nfs/nfs4_fs.h index 9fa496387fdf..7a6eecffcaeb 100644 --- a/fs/nfs/nfs4_fs.h +++ b/fs/nfs/nfs4_fs.h | |||
@@ -331,7 +331,6 @@ extern void nfs_free_seqid(struct nfs_seqid *seqid); | |||
331 | extern const nfs4_stateid zero_stateid; | 331 | extern const nfs4_stateid zero_stateid; |
332 | 332 | ||
333 | /* nfs4xdr.c */ | 333 | /* nfs4xdr.c */ |
334 | extern __be32 *nfs4_decode_dirent(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int); | ||
335 | extern struct rpc_procinfo nfs4_procedures[]; | 334 | extern struct rpc_procinfo nfs4_procedures[]; |
336 | 335 | ||
337 | struct nfs4_mount_data; | 336 | struct nfs4_mount_data; |
diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c index 868815c55450..be9f00ab0d18 100644 --- a/fs/nfs/nfs4xdr.c +++ b/fs/nfs/nfs4xdr.c | |||
@@ -6159,8 +6159,22 @@ out: | |||
6159 | } | 6159 | } |
6160 | #endif /* CONFIG_NFS_V4_1 */ | 6160 | #endif /* CONFIG_NFS_V4_1 */ |
6161 | 6161 | ||
6162 | __be32 *nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | 6162 | /** |
6163 | struct nfs_server *server, int plus) | 6163 | * nfs4_decode_dirent - Decode a single NFSv4 directory entry stored in |
6164 | * the local page cache. | ||
6165 | * @xdr: XDR stream where entry resides | ||
6166 | * @entry: buffer to fill in with entry data | ||
6167 | * @plus: boolean indicating whether this should be a readdirplus entry | ||
6168 | * | ||
6169 | * Returns zero if successful, otherwise a negative errno value is | ||
6170 | * returned. | ||
6171 | * | ||
6172 | * This function is not invoked during READDIR reply decoding, but | ||
6173 | * rather whenever an application invokes the getdents(2) system call | ||
6174 | * on a directory already in our cache. | ||
6175 | */ | ||
6176 | int nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | ||
6177 | int plus) | ||
6164 | { | 6178 | { |
6165 | uint32_t bitmap[2] = {0}; | 6179 | uint32_t bitmap[2] = {0}; |
6166 | uint32_t len; | 6180 | uint32_t len; |
@@ -6172,9 +6186,9 @@ __be32 *nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
6172 | if (unlikely(!p)) | 6186 | if (unlikely(!p)) |
6173 | goto out_overflow; | 6187 | goto out_overflow; |
6174 | if (!ntohl(*p++)) | 6188 | if (!ntohl(*p++)) |
6175 | return ERR_PTR(-EAGAIN); | 6189 | return -EAGAIN; |
6176 | entry->eof = 1; | 6190 | entry->eof = 1; |
6177 | return ERR_PTR(-EBADCOOKIE); | 6191 | return -EBADCOOKIE; |
6178 | } | 6192 | } |
6179 | 6193 | ||
6180 | p = xdr_inline_decode(xdr, 12); | 6194 | p = xdr_inline_decode(xdr, 12); |
@@ -6203,7 +6217,8 @@ __be32 *nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
6203 | if (decode_attr_length(xdr, &len, &p) < 0) | 6217 | if (decode_attr_length(xdr, &len, &p) < 0) |
6204 | goto out_overflow; | 6218 | goto out_overflow; |
6205 | 6219 | ||
6206 | if (decode_getfattr_attrs(xdr, bitmap, entry->fattr, entry->fh, server, 1) < 0) | 6220 | if (decode_getfattr_attrs(xdr, bitmap, entry->fattr, entry->fh, |
6221 | entry->server, 1) < 0) | ||
6207 | goto out_overflow; | 6222 | goto out_overflow; |
6208 | if (entry->fattr->valid & NFS_ATTR_FATTR_FILEID) | 6223 | if (entry->fattr->valid & NFS_ATTR_FATTR_FILEID) |
6209 | entry->ino = entry->fattr->fileid; | 6224 | entry->ino = entry->fattr->fileid; |
@@ -6221,11 +6236,11 @@ __be32 *nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
6221 | else | 6236 | else |
6222 | entry->eof = 0; | 6237 | entry->eof = 0; |
6223 | 6238 | ||
6224 | return p; | 6239 | return 0; |
6225 | 6240 | ||
6226 | out_overflow: | 6241 | out_overflow: |
6227 | print_overflow_msg(__func__, xdr); | 6242 | print_overflow_msg(__func__, xdr); |
6228 | return ERR_PTR(-EAGAIN); | 6243 | return -EAGAIN; |
6229 | } | 6244 | } |
6230 | 6245 | ||
6231 | /* | 6246 | /* |
diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h index 80f07198a31a..236e7e4b99a0 100644 --- a/include/linux/nfs_xdr.h +++ b/include/linux/nfs_xdr.h | |||
@@ -484,6 +484,7 @@ struct nfs_entry { | |||
484 | struct nfs_fh * fh; | 484 | struct nfs_fh * fh; |
485 | struct nfs_fattr * fattr; | 485 | struct nfs_fattr * fattr; |
486 | unsigned char d_type; | 486 | unsigned char d_type; |
487 | struct nfs_server * server; | ||
487 | }; | 488 | }; |
488 | 489 | ||
489 | /* | 490 | /* |
@@ -1089,7 +1090,7 @@ struct nfs_rpc_ops { | |||
1089 | int (*pathconf) (struct nfs_server *, struct nfs_fh *, | 1090 | int (*pathconf) (struct nfs_server *, struct nfs_fh *, |
1090 | struct nfs_pathconf *); | 1091 | struct nfs_pathconf *); |
1091 | int (*set_capabilities)(struct nfs_server *, struct nfs_fh *); | 1092 | int (*set_capabilities)(struct nfs_server *, struct nfs_fh *); |
1092 | __be32 *(*decode_dirent)(struct xdr_stream *, struct nfs_entry *, struct nfs_server *, int plus); | 1093 | int (*decode_dirent)(struct xdr_stream *, struct nfs_entry *, int); |
1093 | void (*read_setup) (struct nfs_read_data *, struct rpc_message *); | 1094 | void (*read_setup) (struct nfs_read_data *, struct rpc_message *); |
1094 | int (*read_done) (struct rpc_task *, struct nfs_read_data *); | 1095 | int (*read_done) (struct rpc_task *, struct nfs_read_data *); |
1095 | void (*write_setup) (struct nfs_write_data *, struct rpc_message *); | 1096 | void (*write_setup) (struct nfs_write_data *, struct rpc_message *); |