diff options
Diffstat (limited to 'fs/nfs')
-rw-r--r-- | fs/nfs/Makefile | 3 | ||||
-rw-r--r-- | fs/nfs/cache_lib.c | 140 | ||||
-rw-r--r-- | fs/nfs/cache_lib.h | 27 | ||||
-rw-r--r-- | fs/nfs/callback.c | 26 | ||||
-rw-r--r-- | fs/nfs/callback_xdr.c | 2 | ||||
-rw-r--r-- | fs/nfs/client.c | 43 | ||||
-rw-r--r-- | fs/nfs/direct.c | 3 | ||||
-rw-r--r-- | fs/nfs/dns_resolve.c | 335 | ||||
-rw-r--r-- | fs/nfs/dns_resolve.h | 14 | ||||
-rw-r--r-- | fs/nfs/file.c | 54 | ||||
-rw-r--r-- | fs/nfs/fscache.c | 25 | ||||
-rw-r--r-- | fs/nfs/fscache.h | 6 | ||||
-rw-r--r-- | fs/nfs/idmap.c | 6 | ||||
-rw-r--r-- | fs/nfs/inode.c | 154 | ||||
-rw-r--r-- | fs/nfs/internal.h | 39 | ||||
-rw-r--r-- | fs/nfs/mount_clnt.c | 83 | ||||
-rw-r--r-- | fs/nfs/nfs2xdr.c | 1 | ||||
-rw-r--r-- | fs/nfs/nfs3proc.c | 2 | ||||
-rw-r--r-- | fs/nfs/nfs3xdr.c | 1 | ||||
-rw-r--r-- | fs/nfs/nfs4namespace.c | 24 | ||||
-rw-r--r-- | fs/nfs/nfs4proc.c | 41 | ||||
-rw-r--r-- | fs/nfs/nfs4state.c | 2 | ||||
-rw-r--r-- | fs/nfs/nfs4xdr.c | 1461 | ||||
-rw-r--r-- | fs/nfs/proc.c | 1 | ||||
-rw-r--r-- | fs/nfs/super.c | 521 | ||||
-rw-r--r-- | fs/nfs/write.c | 92 |
26 files changed, 2097 insertions, 1009 deletions
diff --git a/fs/nfs/Makefile b/fs/nfs/Makefile index 845159814de2..da7fda639eac 100644 --- a/fs/nfs/Makefile +++ b/fs/nfs/Makefile | |||
@@ -6,7 +6,8 @@ obj-$(CONFIG_NFS_FS) += nfs.o | |||
6 | 6 | ||
7 | nfs-y := client.o dir.o file.o getroot.o inode.o super.o nfs2xdr.o \ | 7 | nfs-y := client.o dir.o file.o getroot.o inode.o super.o nfs2xdr.o \ |
8 | direct.o pagelist.o proc.o read.o symlink.o unlink.o \ | 8 | direct.o pagelist.o proc.o read.o symlink.o unlink.o \ |
9 | write.o namespace.o mount_clnt.o | 9 | write.o namespace.o mount_clnt.o \ |
10 | dns_resolve.o cache_lib.o | ||
10 | nfs-$(CONFIG_ROOT_NFS) += nfsroot.o | 11 | nfs-$(CONFIG_ROOT_NFS) += nfsroot.o |
11 | nfs-$(CONFIG_NFS_V3) += nfs3proc.o nfs3xdr.o | 12 | nfs-$(CONFIG_NFS_V3) += nfs3proc.o nfs3xdr.o |
12 | nfs-$(CONFIG_NFS_V3_ACL) += nfs3acl.o | 13 | nfs-$(CONFIG_NFS_V3_ACL) += nfs3acl.o |
diff --git a/fs/nfs/cache_lib.c b/fs/nfs/cache_lib.c new file mode 100644 index 000000000000..b4ffd0146ea6 --- /dev/null +++ b/fs/nfs/cache_lib.c | |||
@@ -0,0 +1,140 @@ | |||
1 | /* | ||
2 | * linux/fs/nfs/cache_lib.c | ||
3 | * | ||
4 | * Helper routines for the NFS client caches | ||
5 | * | ||
6 | * Copyright (c) 2009 Trond Myklebust <Trond.Myklebust@netapp.com> | ||
7 | */ | ||
8 | #include <linux/kmod.h> | ||
9 | #include <linux/module.h> | ||
10 | #include <linux/moduleparam.h> | ||
11 | #include <linux/mount.h> | ||
12 | #include <linux/namei.h> | ||
13 | #include <linux/sunrpc/cache.h> | ||
14 | #include <linux/sunrpc/rpc_pipe_fs.h> | ||
15 | |||
16 | #include "cache_lib.h" | ||
17 | |||
18 | #define NFS_CACHE_UPCALL_PATHLEN 256 | ||
19 | #define NFS_CACHE_UPCALL_TIMEOUT 15 | ||
20 | |||
21 | static char nfs_cache_getent_prog[NFS_CACHE_UPCALL_PATHLEN] = | ||
22 | "/sbin/nfs_cache_getent"; | ||
23 | static unsigned long nfs_cache_getent_timeout = NFS_CACHE_UPCALL_TIMEOUT; | ||
24 | |||
25 | module_param_string(cache_getent, nfs_cache_getent_prog, | ||
26 | sizeof(nfs_cache_getent_prog), 0600); | ||
27 | MODULE_PARM_DESC(cache_getent, "Path to the client cache upcall program"); | ||
28 | module_param_named(cache_getent_timeout, nfs_cache_getent_timeout, ulong, 0600); | ||
29 | MODULE_PARM_DESC(cache_getent_timeout, "Timeout (in seconds) after which " | ||
30 | "the cache upcall is assumed to have failed"); | ||
31 | |||
32 | int nfs_cache_upcall(struct cache_detail *cd, char *entry_name) | ||
33 | { | ||
34 | static char *envp[] = { "HOME=/", | ||
35 | "TERM=linux", | ||
36 | "PATH=/sbin:/usr/sbin:/bin:/usr/bin", | ||
37 | NULL | ||
38 | }; | ||
39 | char *argv[] = { | ||
40 | nfs_cache_getent_prog, | ||
41 | cd->name, | ||
42 | entry_name, | ||
43 | NULL | ||
44 | }; | ||
45 | int ret = -EACCES; | ||
46 | |||
47 | if (nfs_cache_getent_prog[0] == '\0') | ||
48 | goto out; | ||
49 | ret = call_usermodehelper(argv[0], argv, envp, UMH_WAIT_EXEC); | ||
50 | /* | ||
51 | * Disable the upcall mechanism if we're getting an ENOENT or | ||
52 | * EACCES error. The admin can re-enable it on the fly by using | ||
53 | * sysfs to set the 'cache_getent' parameter once the problem | ||
54 | * has been fixed. | ||
55 | */ | ||
56 | if (ret == -ENOENT || ret == -EACCES) | ||
57 | nfs_cache_getent_prog[0] = '\0'; | ||
58 | out: | ||
59 | return ret > 0 ? 0 : ret; | ||
60 | } | ||
61 | |||
62 | /* | ||
63 | * Deferred request handling | ||
64 | */ | ||
65 | void nfs_cache_defer_req_put(struct nfs_cache_defer_req *dreq) | ||
66 | { | ||
67 | if (atomic_dec_and_test(&dreq->count)) | ||
68 | kfree(dreq); | ||
69 | } | ||
70 | |||
71 | static void nfs_dns_cache_revisit(struct cache_deferred_req *d, int toomany) | ||
72 | { | ||
73 | struct nfs_cache_defer_req *dreq; | ||
74 | |||
75 | dreq = container_of(d, struct nfs_cache_defer_req, deferred_req); | ||
76 | |||
77 | complete_all(&dreq->completion); | ||
78 | nfs_cache_defer_req_put(dreq); | ||
79 | } | ||
80 | |||
81 | static struct cache_deferred_req *nfs_dns_cache_defer(struct cache_req *req) | ||
82 | { | ||
83 | struct nfs_cache_defer_req *dreq; | ||
84 | |||
85 | dreq = container_of(req, struct nfs_cache_defer_req, req); | ||
86 | dreq->deferred_req.revisit = nfs_dns_cache_revisit; | ||
87 | atomic_inc(&dreq->count); | ||
88 | |||
89 | return &dreq->deferred_req; | ||
90 | } | ||
91 | |||
92 | struct nfs_cache_defer_req *nfs_cache_defer_req_alloc(void) | ||
93 | { | ||
94 | struct nfs_cache_defer_req *dreq; | ||
95 | |||
96 | dreq = kzalloc(sizeof(*dreq), GFP_KERNEL); | ||
97 | if (dreq) { | ||
98 | init_completion(&dreq->completion); | ||
99 | atomic_set(&dreq->count, 1); | ||
100 | dreq->req.defer = nfs_dns_cache_defer; | ||
101 | } | ||
102 | return dreq; | ||
103 | } | ||
104 | |||
105 | int nfs_cache_wait_for_upcall(struct nfs_cache_defer_req *dreq) | ||
106 | { | ||
107 | if (wait_for_completion_timeout(&dreq->completion, | ||
108 | nfs_cache_getent_timeout * HZ) == 0) | ||
109 | return -ETIMEDOUT; | ||
110 | return 0; | ||
111 | } | ||
112 | |||
113 | int nfs_cache_register(struct cache_detail *cd) | ||
114 | { | ||
115 | struct nameidata nd; | ||
116 | struct vfsmount *mnt; | ||
117 | int ret; | ||
118 | |||
119 | mnt = rpc_get_mount(); | ||
120 | if (IS_ERR(mnt)) | ||
121 | return PTR_ERR(mnt); | ||
122 | ret = vfs_path_lookup(mnt->mnt_root, mnt, "/cache", 0, &nd); | ||
123 | if (ret) | ||
124 | goto err; | ||
125 | ret = sunrpc_cache_register_pipefs(nd.path.dentry, | ||
126 | cd->name, 0600, cd); | ||
127 | path_put(&nd.path); | ||
128 | if (!ret) | ||
129 | return ret; | ||
130 | err: | ||
131 | rpc_put_mount(); | ||
132 | return ret; | ||
133 | } | ||
134 | |||
135 | void nfs_cache_unregister(struct cache_detail *cd) | ||
136 | { | ||
137 | sunrpc_cache_unregister_pipefs(cd); | ||
138 | rpc_put_mount(); | ||
139 | } | ||
140 | |||
diff --git a/fs/nfs/cache_lib.h b/fs/nfs/cache_lib.h new file mode 100644 index 000000000000..76f856e284e4 --- /dev/null +++ b/fs/nfs/cache_lib.h | |||
@@ -0,0 +1,27 @@ | |||
1 | /* | ||
2 | * Helper routines for the NFS client caches | ||
3 | * | ||
4 | * Copyright (c) 2009 Trond Myklebust <Trond.Myklebust@netapp.com> | ||
5 | */ | ||
6 | |||
7 | #include <linux/completion.h> | ||
8 | #include <linux/sunrpc/cache.h> | ||
9 | #include <asm/atomic.h> | ||
10 | |||
11 | /* | ||
12 | * Deferred request handling | ||
13 | */ | ||
14 | struct nfs_cache_defer_req { | ||
15 | struct cache_req req; | ||
16 | struct cache_deferred_req deferred_req; | ||
17 | struct completion completion; | ||
18 | atomic_t count; | ||
19 | }; | ||
20 | |||
21 | extern int nfs_cache_upcall(struct cache_detail *cd, char *entry_name); | ||
22 | extern struct nfs_cache_defer_req *nfs_cache_defer_req_alloc(void); | ||
23 | extern void nfs_cache_defer_req_put(struct nfs_cache_defer_req *dreq); | ||
24 | extern int nfs_cache_wait_for_upcall(struct nfs_cache_defer_req *dreq); | ||
25 | |||
26 | extern int nfs_cache_register(struct cache_detail *cd); | ||
27 | extern void nfs_cache_unregister(struct cache_detail *cd); | ||
diff --git a/fs/nfs/callback.c b/fs/nfs/callback.c index 7f604c7941fb..293fa0528a6e 100644 --- a/fs/nfs/callback.c +++ b/fs/nfs/callback.c | |||
@@ -43,21 +43,29 @@ static struct svc_program nfs4_callback_program; | |||
43 | unsigned int nfs_callback_set_tcpport; | 43 | unsigned int nfs_callback_set_tcpport; |
44 | unsigned short nfs_callback_tcpport; | 44 | unsigned short nfs_callback_tcpport; |
45 | unsigned short nfs_callback_tcpport6; | 45 | unsigned short nfs_callback_tcpport6; |
46 | static const int nfs_set_port_min = 0; | 46 | #define NFS_CALLBACK_MAXPORTNR (65535U) |
47 | static const int nfs_set_port_max = 65535; | ||
48 | 47 | ||
49 | static int param_set_port(const char *val, struct kernel_param *kp) | 48 | static int param_set_portnr(const char *val, struct kernel_param *kp) |
50 | { | 49 | { |
51 | char *endp; | 50 | unsigned long num; |
52 | int num = simple_strtol(val, &endp, 0); | 51 | int ret; |
53 | if (endp == val || *endp || num < nfs_set_port_min || num > nfs_set_port_max) | 52 | |
53 | if (!val) | ||
54 | return -EINVAL; | ||
55 | ret = strict_strtoul(val, 0, &num); | ||
56 | if (ret == -EINVAL || num > NFS_CALLBACK_MAXPORTNR) | ||
54 | return -EINVAL; | 57 | return -EINVAL; |
55 | *((int *)kp->arg) = num; | 58 | *((unsigned int *)kp->arg) = num; |
56 | return 0; | 59 | return 0; |
57 | } | 60 | } |
58 | 61 | ||
59 | module_param_call(callback_tcpport, param_set_port, param_get_int, | 62 | static int param_get_portnr(char *buffer, struct kernel_param *kp) |
60 | &nfs_callback_set_tcpport, 0644); | 63 | { |
64 | return param_get_uint(buffer, kp); | ||
65 | } | ||
66 | #define param_check_portnr(name, p) __param_check(name, p, unsigned int); | ||
67 | |||
68 | module_param_named(callback_tcpport, nfs_callback_set_tcpport, portnr, 0644); | ||
61 | 69 | ||
62 | /* | 70 | /* |
63 | * This is the NFSv4 callback kernel thread. | 71 | * This is the NFSv4 callback kernel thread. |
diff --git a/fs/nfs/callback_xdr.c b/fs/nfs/callback_xdr.c index e5a2dac5f715..76b0aa0f73bf 100644 --- a/fs/nfs/callback_xdr.c +++ b/fs/nfs/callback_xdr.c | |||
@@ -222,7 +222,7 @@ static unsigned decode_sessionid(struct xdr_stream *xdr, | |||
222 | 222 | ||
223 | p = read_buf(xdr, len); | 223 | p = read_buf(xdr, len); |
224 | if (unlikely(p == NULL)) | 224 | if (unlikely(p == NULL)) |
225 | return htonl(NFS4ERR_RESOURCE);; | 225 | return htonl(NFS4ERR_RESOURCE); |
226 | 226 | ||
227 | memcpy(sid->data, p, len); | 227 | memcpy(sid->data, p, len); |
228 | return 0; | 228 | return 0; |
diff --git a/fs/nfs/client.c b/fs/nfs/client.c index 8d25ccb2d51d..63976c0ccc25 100644 --- a/fs/nfs/client.c +++ b/fs/nfs/client.c | |||
@@ -648,8 +648,6 @@ static int nfs_start_lockd(struct nfs_server *server) | |||
648 | .hostname = clp->cl_hostname, | 648 | .hostname = clp->cl_hostname, |
649 | .address = (struct sockaddr *)&clp->cl_addr, | 649 | .address = (struct sockaddr *)&clp->cl_addr, |
650 | .addrlen = clp->cl_addrlen, | 650 | .addrlen = clp->cl_addrlen, |
651 | .protocol = server->flags & NFS_MOUNT_TCP ? | ||
652 | IPPROTO_TCP : IPPROTO_UDP, | ||
653 | .nfs_version = clp->rpc_ops->version, | 651 | .nfs_version = clp->rpc_ops->version, |
654 | .noresvport = server->flags & NFS_MOUNT_NORESVPORT ? | 652 | .noresvport = server->flags & NFS_MOUNT_NORESVPORT ? |
655 | 1 : 0, | 653 | 1 : 0, |
@@ -660,6 +658,14 @@ static int nfs_start_lockd(struct nfs_server *server) | |||
660 | if (server->flags & NFS_MOUNT_NONLM) | 658 | if (server->flags & NFS_MOUNT_NONLM) |
661 | return 0; | 659 | return 0; |
662 | 660 | ||
661 | switch (clp->cl_proto) { | ||
662 | default: | ||
663 | nlm_init.protocol = IPPROTO_TCP; | ||
664 | break; | ||
665 | case XPRT_TRANSPORT_UDP: | ||
666 | nlm_init.protocol = IPPROTO_UDP; | ||
667 | } | ||
668 | |||
663 | host = nlmclnt_init(&nlm_init); | 669 | host = nlmclnt_init(&nlm_init); |
664 | if (IS_ERR(host)) | 670 | if (IS_ERR(host)) |
665 | return PTR_ERR(host); | 671 | return PTR_ERR(host); |
@@ -787,7 +793,7 @@ static int nfs_init_server(struct nfs_server *server, | |||
787 | dprintk("--> nfs_init_server()\n"); | 793 | dprintk("--> nfs_init_server()\n"); |
788 | 794 | ||
789 | #ifdef CONFIG_NFS_V3 | 795 | #ifdef CONFIG_NFS_V3 |
790 | if (data->flags & NFS_MOUNT_VER3) | 796 | if (data->version == 3) |
791 | cl_init.rpc_ops = &nfs_v3_clientops; | 797 | cl_init.rpc_ops = &nfs_v3_clientops; |
792 | #endif | 798 | #endif |
793 | 799 | ||
@@ -809,6 +815,9 @@ static int nfs_init_server(struct nfs_server *server, | |||
809 | /* Initialise the client representation from the mount data */ | 815 | /* Initialise the client representation from the mount data */ |
810 | server->flags = data->flags; | 816 | server->flags = data->flags; |
811 | server->options = data->options; | 817 | server->options = data->options; |
818 | server->caps |= NFS_CAP_HARDLINKS|NFS_CAP_SYMLINKS|NFS_CAP_FILEID| | ||
819 | NFS_CAP_MODE|NFS_CAP_NLINK|NFS_CAP_OWNER|NFS_CAP_OWNER_GROUP| | ||
820 | NFS_CAP_ATIME|NFS_CAP_CTIME|NFS_CAP_MTIME; | ||
812 | 821 | ||
813 | if (data->rsize) | 822 | if (data->rsize) |
814 | server->rsize = nfs_block_size(data->rsize, NULL); | 823 | server->rsize = nfs_block_size(data->rsize, NULL); |
@@ -879,6 +888,7 @@ static void nfs_server_set_fsinfo(struct nfs_server *server, struct nfs_fsinfo * | |||
879 | server->rsize = NFS_MAX_FILE_IO_SIZE; | 888 | server->rsize = NFS_MAX_FILE_IO_SIZE; |
880 | server->rpages = (server->rsize + PAGE_CACHE_SIZE - 1) >> PAGE_CACHE_SHIFT; | 889 | server->rpages = (server->rsize + PAGE_CACHE_SIZE - 1) >> PAGE_CACHE_SHIFT; |
881 | 890 | ||
891 | server->backing_dev_info.name = "nfs"; | ||
882 | server->backing_dev_info.ra_pages = server->rpages * NFS_MAX_READAHEAD; | 892 | server->backing_dev_info.ra_pages = server->rpages * NFS_MAX_READAHEAD; |
883 | 893 | ||
884 | if (server->wsize > max_rpc_payload) | 894 | if (server->wsize > max_rpc_payload) |
@@ -929,10 +939,6 @@ static int nfs_probe_fsinfo(struct nfs_server *server, struct nfs_fh *mntfh, str | |||
929 | goto out_error; | 939 | goto out_error; |
930 | 940 | ||
931 | nfs_server_set_fsinfo(server, &fsinfo); | 941 | nfs_server_set_fsinfo(server, &fsinfo); |
932 | error = bdi_init(&server->backing_dev_info); | ||
933 | if (error) | ||
934 | goto out_error; | ||
935 | |||
936 | 942 | ||
937 | /* Get some general file system info */ | 943 | /* Get some general file system info */ |
938 | if (server->namelen == 0) { | 944 | if (server->namelen == 0) { |
@@ -964,6 +970,7 @@ static void nfs_server_copy_userdata(struct nfs_server *target, struct nfs_serve | |||
964 | target->acdirmin = source->acdirmin; | 970 | target->acdirmin = source->acdirmin; |
965 | target->acdirmax = source->acdirmax; | 971 | target->acdirmax = source->acdirmax; |
966 | target->caps = source->caps; | 972 | target->caps = source->caps; |
973 | target->options = source->options; | ||
967 | } | 974 | } |
968 | 975 | ||
969 | /* | 976 | /* |
@@ -991,6 +998,12 @@ static struct nfs_server *nfs_alloc_server(void) | |||
991 | return NULL; | 998 | return NULL; |
992 | } | 999 | } |
993 | 1000 | ||
1001 | if (bdi_init(&server->backing_dev_info)) { | ||
1002 | nfs_free_iostats(server->io_stats); | ||
1003 | kfree(server); | ||
1004 | return NULL; | ||
1005 | } | ||
1006 | |||
994 | return server; | 1007 | return server; |
995 | } | 1008 | } |
996 | 1009 | ||
@@ -1074,10 +1087,6 @@ struct nfs_server *nfs_create_server(const struct nfs_parsed_mount_data *data, | |||
1074 | (unsigned long long) server->fsid.major, | 1087 | (unsigned long long) server->fsid.major, |
1075 | (unsigned long long) server->fsid.minor); | 1088 | (unsigned long long) server->fsid.minor); |
1076 | 1089 | ||
1077 | BUG_ON(!server->nfs_client); | ||
1078 | BUG_ON(!server->nfs_client->rpc_ops); | ||
1079 | BUG_ON(!server->nfs_client->rpc_ops->file_inode_ops); | ||
1080 | |||
1081 | spin_lock(&nfs_client_lock); | 1090 | spin_lock(&nfs_client_lock); |
1082 | list_add_tail(&server->client_link, &server->nfs_client->cl_superblocks); | 1091 | list_add_tail(&server->client_link, &server->nfs_client->cl_superblocks); |
1083 | list_add_tail(&server->master_link, &nfs_volume_list); | 1092 | list_add_tail(&server->master_link, &nfs_volume_list); |
@@ -1274,7 +1283,7 @@ static int nfs4_init_server(struct nfs_server *server, | |||
1274 | 1283 | ||
1275 | /* Initialise the client representation from the mount data */ | 1284 | /* Initialise the client representation from the mount data */ |
1276 | server->flags = data->flags; | 1285 | server->flags = data->flags; |
1277 | server->caps |= NFS_CAP_ATOMIC_OPEN; | 1286 | server->caps |= NFS_CAP_ATOMIC_OPEN|NFS_CAP_CHANGE_ATTR; |
1278 | server->options = data->options; | 1287 | server->options = data->options; |
1279 | 1288 | ||
1280 | /* Get a client record */ | 1289 | /* Get a client record */ |
@@ -1359,10 +1368,6 @@ struct nfs_server *nfs4_create_server(const struct nfs_parsed_mount_data *data, | |||
1359 | if (server->namelen == 0 || server->namelen > NFS4_MAXNAMLEN) | 1368 | if (server->namelen == 0 || server->namelen > NFS4_MAXNAMLEN) |
1360 | server->namelen = NFS4_MAXNAMLEN; | 1369 | server->namelen = NFS4_MAXNAMLEN; |
1361 | 1370 | ||
1362 | BUG_ON(!server->nfs_client); | ||
1363 | BUG_ON(!server->nfs_client->rpc_ops); | ||
1364 | BUG_ON(!server->nfs_client->rpc_ops->file_inode_ops); | ||
1365 | |||
1366 | spin_lock(&nfs_client_lock); | 1371 | spin_lock(&nfs_client_lock); |
1367 | list_add_tail(&server->client_link, &server->nfs_client->cl_superblocks); | 1372 | list_add_tail(&server->client_link, &server->nfs_client->cl_superblocks); |
1368 | list_add_tail(&server->master_link, &nfs_volume_list); | 1373 | list_add_tail(&server->master_link, &nfs_volume_list); |
@@ -1400,7 +1405,7 @@ struct nfs_server *nfs4_create_referral_server(struct nfs_clone_mount *data, | |||
1400 | 1405 | ||
1401 | /* Initialise the client representation from the parent server */ | 1406 | /* Initialise the client representation from the parent server */ |
1402 | nfs_server_copy_userdata(server, parent_server); | 1407 | nfs_server_copy_userdata(server, parent_server); |
1403 | server->caps |= NFS_CAP_ATOMIC_OPEN; | 1408 | server->caps |= NFS_CAP_ATOMIC_OPEN|NFS_CAP_CHANGE_ATTR; |
1404 | 1409 | ||
1405 | /* Get a client representation. | 1410 | /* Get a client representation. |
1406 | * Note: NFSv4 always uses TCP, */ | 1411 | * Note: NFSv4 always uses TCP, */ |
@@ -1533,7 +1538,7 @@ static void *nfs_server_list_next(struct seq_file *p, void *v, loff_t *pos); | |||
1533 | static void nfs_server_list_stop(struct seq_file *p, void *v); | 1538 | static void nfs_server_list_stop(struct seq_file *p, void *v); |
1534 | static int nfs_server_list_show(struct seq_file *m, void *v); | 1539 | static int nfs_server_list_show(struct seq_file *m, void *v); |
1535 | 1540 | ||
1536 | static struct seq_operations nfs_server_list_ops = { | 1541 | static const struct seq_operations nfs_server_list_ops = { |
1537 | .start = nfs_server_list_start, | 1542 | .start = nfs_server_list_start, |
1538 | .next = nfs_server_list_next, | 1543 | .next = nfs_server_list_next, |
1539 | .stop = nfs_server_list_stop, | 1544 | .stop = nfs_server_list_stop, |
@@ -1554,7 +1559,7 @@ static void *nfs_volume_list_next(struct seq_file *p, void *v, loff_t *pos); | |||
1554 | static void nfs_volume_list_stop(struct seq_file *p, void *v); | 1559 | static void nfs_volume_list_stop(struct seq_file *p, void *v); |
1555 | static int nfs_volume_list_show(struct seq_file *m, void *v); | 1560 | static int nfs_volume_list_show(struct seq_file *m, void *v); |
1556 | 1561 | ||
1557 | static struct seq_operations nfs_volume_list_ops = { | 1562 | static const struct seq_operations nfs_volume_list_ops = { |
1558 | .start = nfs_volume_list_start, | 1563 | .start = nfs_volume_list_start, |
1559 | .next = nfs_volume_list_next, | 1564 | .next = nfs_volume_list_next, |
1560 | .stop = nfs_volume_list_stop, | 1565 | .stop = nfs_volume_list_stop, |
diff --git a/fs/nfs/direct.c b/fs/nfs/direct.c index e4e089a8f294..6c3210099d51 100644 --- a/fs/nfs/direct.c +++ b/fs/nfs/direct.c | |||
@@ -934,9 +934,6 @@ out: | |||
934 | * back into its cache. We let the server do generic write | 934 | * back into its cache. We let the server do generic write |
935 | * parameter checking and report problems. | 935 | * parameter checking and report problems. |
936 | * | 936 | * |
937 | * We also avoid an unnecessary invocation of generic_osync_inode(), | ||
938 | * as it is fairly meaningless to sync the metadata of an NFS file. | ||
939 | * | ||
940 | * We eliminate local atime updates, see direct read above. | 937 | * We eliminate local atime updates, see direct read above. |
941 | * | 938 | * |
942 | * We avoid unnecessary page cache invalidations for normal cached | 939 | * We avoid unnecessary page cache invalidations for normal cached |
diff --git a/fs/nfs/dns_resolve.c b/fs/nfs/dns_resolve.c new file mode 100644 index 000000000000..f4d54ba97cc6 --- /dev/null +++ b/fs/nfs/dns_resolve.c | |||
@@ -0,0 +1,335 @@ | |||
1 | /* | ||
2 | * linux/fs/nfs/dns_resolve.c | ||
3 | * | ||
4 | * Copyright (c) 2009 Trond Myklebust <Trond.Myklebust@netapp.com> | ||
5 | * | ||
6 | * Resolves DNS hostnames into valid ip addresses | ||
7 | */ | ||
8 | |||
9 | #include <linux/hash.h> | ||
10 | #include <linux/string.h> | ||
11 | #include <linux/kmod.h> | ||
12 | #include <linux/module.h> | ||
13 | #include <linux/socket.h> | ||
14 | #include <linux/seq_file.h> | ||
15 | #include <linux/inet.h> | ||
16 | #include <linux/sunrpc/clnt.h> | ||
17 | #include <linux/sunrpc/cache.h> | ||
18 | #include <linux/sunrpc/svcauth.h> | ||
19 | |||
20 | #include "dns_resolve.h" | ||
21 | #include "cache_lib.h" | ||
22 | |||
23 | #define NFS_DNS_HASHBITS 4 | ||
24 | #define NFS_DNS_HASHTBL_SIZE (1 << NFS_DNS_HASHBITS) | ||
25 | |||
26 | static struct cache_head *nfs_dns_table[NFS_DNS_HASHTBL_SIZE]; | ||
27 | |||
28 | struct nfs_dns_ent { | ||
29 | struct cache_head h; | ||
30 | |||
31 | char *hostname; | ||
32 | size_t namelen; | ||
33 | |||
34 | struct sockaddr_storage addr; | ||
35 | size_t addrlen; | ||
36 | }; | ||
37 | |||
38 | |||
39 | static void nfs_dns_ent_init(struct cache_head *cnew, | ||
40 | struct cache_head *ckey) | ||
41 | { | ||
42 | struct nfs_dns_ent *new; | ||
43 | struct nfs_dns_ent *key; | ||
44 | |||
45 | new = container_of(cnew, struct nfs_dns_ent, h); | ||
46 | key = container_of(ckey, struct nfs_dns_ent, h); | ||
47 | |||
48 | kfree(new->hostname); | ||
49 | new->hostname = kstrndup(key->hostname, key->namelen, GFP_KERNEL); | ||
50 | if (new->hostname) { | ||
51 | new->namelen = key->namelen; | ||
52 | memcpy(&new->addr, &key->addr, key->addrlen); | ||
53 | new->addrlen = key->addrlen; | ||
54 | } else { | ||
55 | new->namelen = 0; | ||
56 | new->addrlen = 0; | ||
57 | } | ||
58 | } | ||
59 | |||
60 | static void nfs_dns_ent_put(struct kref *ref) | ||
61 | { | ||
62 | struct nfs_dns_ent *item; | ||
63 | |||
64 | item = container_of(ref, struct nfs_dns_ent, h.ref); | ||
65 | kfree(item->hostname); | ||
66 | kfree(item); | ||
67 | } | ||
68 | |||
69 | static struct cache_head *nfs_dns_ent_alloc(void) | ||
70 | { | ||
71 | struct nfs_dns_ent *item = kmalloc(sizeof(*item), GFP_KERNEL); | ||
72 | |||
73 | if (item != NULL) { | ||
74 | item->hostname = NULL; | ||
75 | item->namelen = 0; | ||
76 | item->addrlen = 0; | ||
77 | return &item->h; | ||
78 | } | ||
79 | return NULL; | ||
80 | }; | ||
81 | |||
82 | static unsigned int nfs_dns_hash(const struct nfs_dns_ent *key) | ||
83 | { | ||
84 | return hash_str(key->hostname, NFS_DNS_HASHBITS); | ||
85 | } | ||
86 | |||
87 | static void nfs_dns_request(struct cache_detail *cd, | ||
88 | struct cache_head *ch, | ||
89 | char **bpp, int *blen) | ||
90 | { | ||
91 | struct nfs_dns_ent *key = container_of(ch, struct nfs_dns_ent, h); | ||
92 | |||
93 | qword_add(bpp, blen, key->hostname); | ||
94 | (*bpp)[-1] = '\n'; | ||
95 | } | ||
96 | |||
97 | static int nfs_dns_upcall(struct cache_detail *cd, | ||
98 | struct cache_head *ch) | ||
99 | { | ||
100 | struct nfs_dns_ent *key = container_of(ch, struct nfs_dns_ent, h); | ||
101 | int ret; | ||
102 | |||
103 | ret = nfs_cache_upcall(cd, key->hostname); | ||
104 | if (ret) | ||
105 | ret = sunrpc_cache_pipe_upcall(cd, ch, nfs_dns_request); | ||
106 | return ret; | ||
107 | } | ||
108 | |||
109 | static int nfs_dns_match(struct cache_head *ca, | ||
110 | struct cache_head *cb) | ||
111 | { | ||
112 | struct nfs_dns_ent *a; | ||
113 | struct nfs_dns_ent *b; | ||
114 | |||
115 | a = container_of(ca, struct nfs_dns_ent, h); | ||
116 | b = container_of(cb, struct nfs_dns_ent, h); | ||
117 | |||
118 | if (a->namelen == 0 || a->namelen != b->namelen) | ||
119 | return 0; | ||
120 | return memcmp(a->hostname, b->hostname, a->namelen) == 0; | ||
121 | } | ||
122 | |||
123 | static int nfs_dns_show(struct seq_file *m, struct cache_detail *cd, | ||
124 | struct cache_head *h) | ||
125 | { | ||
126 | struct nfs_dns_ent *item; | ||
127 | long ttl; | ||
128 | |||
129 | if (h == NULL) { | ||
130 | seq_puts(m, "# ip address hostname ttl\n"); | ||
131 | return 0; | ||
132 | } | ||
133 | item = container_of(h, struct nfs_dns_ent, h); | ||
134 | ttl = (long)item->h.expiry_time - (long)get_seconds(); | ||
135 | if (ttl < 0) | ||
136 | ttl = 0; | ||
137 | |||
138 | if (!test_bit(CACHE_NEGATIVE, &h->flags)) { | ||
139 | char buf[INET6_ADDRSTRLEN+IPV6_SCOPE_ID_LEN+1]; | ||
140 | |||
141 | rpc_ntop((struct sockaddr *)&item->addr, buf, sizeof(buf)); | ||
142 | seq_printf(m, "%15s ", buf); | ||
143 | } else | ||
144 | seq_puts(m, "<none> "); | ||
145 | seq_printf(m, "%15s %ld\n", item->hostname, ttl); | ||
146 | return 0; | ||
147 | } | ||
148 | |||
149 | struct nfs_dns_ent *nfs_dns_lookup(struct cache_detail *cd, | ||
150 | struct nfs_dns_ent *key) | ||
151 | { | ||
152 | struct cache_head *ch; | ||
153 | |||
154 | ch = sunrpc_cache_lookup(cd, | ||
155 | &key->h, | ||
156 | nfs_dns_hash(key)); | ||
157 | if (!ch) | ||
158 | return NULL; | ||
159 | return container_of(ch, struct nfs_dns_ent, h); | ||
160 | } | ||
161 | |||
162 | struct nfs_dns_ent *nfs_dns_update(struct cache_detail *cd, | ||
163 | struct nfs_dns_ent *new, | ||
164 | struct nfs_dns_ent *key) | ||
165 | { | ||
166 | struct cache_head *ch; | ||
167 | |||
168 | ch = sunrpc_cache_update(cd, | ||
169 | &new->h, &key->h, | ||
170 | nfs_dns_hash(key)); | ||
171 | if (!ch) | ||
172 | return NULL; | ||
173 | return container_of(ch, struct nfs_dns_ent, h); | ||
174 | } | ||
175 | |||
176 | static int nfs_dns_parse(struct cache_detail *cd, char *buf, int buflen) | ||
177 | { | ||
178 | char buf1[NFS_DNS_HOSTNAME_MAXLEN+1]; | ||
179 | struct nfs_dns_ent key, *item; | ||
180 | unsigned long ttl; | ||
181 | ssize_t len; | ||
182 | int ret = -EINVAL; | ||
183 | |||
184 | if (buf[buflen-1] != '\n') | ||
185 | goto out; | ||
186 | buf[buflen-1] = '\0'; | ||
187 | |||
188 | len = qword_get(&buf, buf1, sizeof(buf1)); | ||
189 | if (len <= 0) | ||
190 | goto out; | ||
191 | key.addrlen = rpc_pton(buf1, len, | ||
192 | (struct sockaddr *)&key.addr, | ||
193 | sizeof(key.addr)); | ||
194 | |||
195 | len = qword_get(&buf, buf1, sizeof(buf1)); | ||
196 | if (len <= 0) | ||
197 | goto out; | ||
198 | |||
199 | key.hostname = buf1; | ||
200 | key.namelen = len; | ||
201 | memset(&key.h, 0, sizeof(key.h)); | ||
202 | |||
203 | ttl = get_expiry(&buf); | ||
204 | if (ttl == 0) | ||
205 | goto out; | ||
206 | key.h.expiry_time = ttl + get_seconds(); | ||
207 | |||
208 | ret = -ENOMEM; | ||
209 | item = nfs_dns_lookup(cd, &key); | ||
210 | if (item == NULL) | ||
211 | goto out; | ||
212 | |||
213 | if (key.addrlen == 0) | ||
214 | set_bit(CACHE_NEGATIVE, &key.h.flags); | ||
215 | |||
216 | item = nfs_dns_update(cd, &key, item); | ||
217 | if (item == NULL) | ||
218 | goto out; | ||
219 | |||
220 | ret = 0; | ||
221 | cache_put(&item->h, cd); | ||
222 | out: | ||
223 | return ret; | ||
224 | } | ||
225 | |||
226 | static struct cache_detail nfs_dns_resolve = { | ||
227 | .owner = THIS_MODULE, | ||
228 | .hash_size = NFS_DNS_HASHTBL_SIZE, | ||
229 | .hash_table = nfs_dns_table, | ||
230 | .name = "dns_resolve", | ||
231 | .cache_put = nfs_dns_ent_put, | ||
232 | .cache_upcall = nfs_dns_upcall, | ||
233 | .cache_parse = nfs_dns_parse, | ||
234 | .cache_show = nfs_dns_show, | ||
235 | .match = nfs_dns_match, | ||
236 | .init = nfs_dns_ent_init, | ||
237 | .update = nfs_dns_ent_init, | ||
238 | .alloc = nfs_dns_ent_alloc, | ||
239 | }; | ||
240 | |||
241 | static int do_cache_lookup(struct cache_detail *cd, | ||
242 | struct nfs_dns_ent *key, | ||
243 | struct nfs_dns_ent **item, | ||
244 | struct nfs_cache_defer_req *dreq) | ||
245 | { | ||
246 | int ret = -ENOMEM; | ||
247 | |||
248 | *item = nfs_dns_lookup(cd, key); | ||
249 | if (*item) { | ||
250 | ret = cache_check(cd, &(*item)->h, &dreq->req); | ||
251 | if (ret) | ||
252 | *item = NULL; | ||
253 | } | ||
254 | return ret; | ||
255 | } | ||
256 | |||
257 | static int do_cache_lookup_nowait(struct cache_detail *cd, | ||
258 | struct nfs_dns_ent *key, | ||
259 | struct nfs_dns_ent **item) | ||
260 | { | ||
261 | int ret = -ENOMEM; | ||
262 | |||
263 | *item = nfs_dns_lookup(cd, key); | ||
264 | if (!*item) | ||
265 | goto out_err; | ||
266 | ret = -ETIMEDOUT; | ||
267 | if (!test_bit(CACHE_VALID, &(*item)->h.flags) | ||
268 | || (*item)->h.expiry_time < get_seconds() | ||
269 | || cd->flush_time > (*item)->h.last_refresh) | ||
270 | goto out_put; | ||
271 | ret = -ENOENT; | ||
272 | if (test_bit(CACHE_NEGATIVE, &(*item)->h.flags)) | ||
273 | goto out_put; | ||
274 | return 0; | ||
275 | out_put: | ||
276 | cache_put(&(*item)->h, cd); | ||
277 | out_err: | ||
278 | *item = NULL; | ||
279 | return ret; | ||
280 | } | ||
281 | |||
282 | static int do_cache_lookup_wait(struct cache_detail *cd, | ||
283 | struct nfs_dns_ent *key, | ||
284 | struct nfs_dns_ent **item) | ||
285 | { | ||
286 | struct nfs_cache_defer_req *dreq; | ||
287 | int ret = -ENOMEM; | ||
288 | |||
289 | dreq = nfs_cache_defer_req_alloc(); | ||
290 | if (!dreq) | ||
291 | goto out; | ||
292 | ret = do_cache_lookup(cd, key, item, dreq); | ||
293 | if (ret == -EAGAIN) { | ||
294 | ret = nfs_cache_wait_for_upcall(dreq); | ||
295 | if (!ret) | ||
296 | ret = do_cache_lookup_nowait(cd, key, item); | ||
297 | } | ||
298 | nfs_cache_defer_req_put(dreq); | ||
299 | out: | ||
300 | return ret; | ||
301 | } | ||
302 | |||
303 | ssize_t nfs_dns_resolve_name(char *name, size_t namelen, | ||
304 | struct sockaddr *sa, size_t salen) | ||
305 | { | ||
306 | struct nfs_dns_ent key = { | ||
307 | .hostname = name, | ||
308 | .namelen = namelen, | ||
309 | }; | ||
310 | struct nfs_dns_ent *item = NULL; | ||
311 | ssize_t ret; | ||
312 | |||
313 | ret = do_cache_lookup_wait(&nfs_dns_resolve, &key, &item); | ||
314 | if (ret == 0) { | ||
315 | if (salen >= item->addrlen) { | ||
316 | memcpy(sa, &item->addr, item->addrlen); | ||
317 | ret = item->addrlen; | ||
318 | } else | ||
319 | ret = -EOVERFLOW; | ||
320 | cache_put(&item->h, &nfs_dns_resolve); | ||
321 | } else if (ret == -ENOENT) | ||
322 | ret = -ESRCH; | ||
323 | return ret; | ||
324 | } | ||
325 | |||
326 | int nfs_dns_resolver_init(void) | ||
327 | { | ||
328 | return nfs_cache_register(&nfs_dns_resolve); | ||
329 | } | ||
330 | |||
331 | void nfs_dns_resolver_destroy(void) | ||
332 | { | ||
333 | nfs_cache_unregister(&nfs_dns_resolve); | ||
334 | } | ||
335 | |||
diff --git a/fs/nfs/dns_resolve.h b/fs/nfs/dns_resolve.h new file mode 100644 index 000000000000..a3f0938babf7 --- /dev/null +++ b/fs/nfs/dns_resolve.h | |||
@@ -0,0 +1,14 @@ | |||
1 | /* | ||
2 | * Resolve DNS hostnames into valid ip addresses | ||
3 | */ | ||
4 | #ifndef __LINUX_FS_NFS_DNS_RESOLVE_H | ||
5 | #define __LINUX_FS_NFS_DNS_RESOLVE_H | ||
6 | |||
7 | #define NFS_DNS_HOSTNAME_MAXLEN (128) | ||
8 | |||
9 | extern int nfs_dns_resolver_init(void); | ||
10 | extern void nfs_dns_resolver_destroy(void); | ||
11 | extern ssize_t nfs_dns_resolve_name(char *name, size_t namelen, | ||
12 | struct sockaddr *sa, size_t salen); | ||
13 | |||
14 | #endif | ||
diff --git a/fs/nfs/file.c b/fs/nfs/file.c index 05062329b678..f5fdd39e037a 100644 --- a/fs/nfs/file.c +++ b/fs/nfs/file.c | |||
@@ -59,7 +59,7 @@ static int nfs_lock(struct file *filp, int cmd, struct file_lock *fl); | |||
59 | static int nfs_flock(struct file *filp, int cmd, struct file_lock *fl); | 59 | static int nfs_flock(struct file *filp, int cmd, struct file_lock *fl); |
60 | static int nfs_setlease(struct file *file, long arg, struct file_lock **fl); | 60 | static int nfs_setlease(struct file *file, long arg, struct file_lock **fl); |
61 | 61 | ||
62 | static struct vm_operations_struct nfs_file_vm_ops; | 62 | static const struct vm_operations_struct nfs_file_vm_ops; |
63 | 63 | ||
64 | const struct file_operations nfs_file_operations = { | 64 | const struct file_operations nfs_file_operations = { |
65 | .llseek = nfs_file_llseek, | 65 | .llseek = nfs_file_llseek, |
@@ -328,6 +328,42 @@ nfs_file_fsync(struct file *file, struct dentry *dentry, int datasync) | |||
328 | } | 328 | } |
329 | 329 | ||
330 | /* | 330 | /* |
331 | * Decide whether a read/modify/write cycle may be more efficient | ||
332 | * then a modify/write/read cycle when writing to a page in the | ||
333 | * page cache. | ||
334 | * | ||
335 | * The modify/write/read cycle may occur if a page is read before | ||
336 | * being completely filled by the writer. In this situation, the | ||
337 | * page must be completely written to stable storage on the server | ||
338 | * before it can be refilled by reading in the page from the server. | ||
339 | * This can lead to expensive, small, FILE_SYNC mode writes being | ||
340 | * done. | ||
341 | * | ||
342 | * It may be more efficient to read the page first if the file is | ||
343 | * open for reading in addition to writing, the page is not marked | ||
344 | * as Uptodate, it is not dirty or waiting to be committed, | ||
345 | * indicating that it was previously allocated and then modified, | ||
346 | * that there were valid bytes of data in that range of the file, | ||
347 | * and that the new data won't completely replace the old data in | ||
348 | * that range of the file. | ||
349 | */ | ||
350 | static int nfs_want_read_modify_write(struct file *file, struct page *page, | ||
351 | loff_t pos, unsigned len) | ||
352 | { | ||
353 | unsigned int pglen = nfs_page_length(page); | ||
354 | unsigned int offset = pos & (PAGE_CACHE_SIZE - 1); | ||
355 | unsigned int end = offset + len; | ||
356 | |||
357 | if ((file->f_mode & FMODE_READ) && /* open for read? */ | ||
358 | !PageUptodate(page) && /* Uptodate? */ | ||
359 | !PagePrivate(page) && /* i/o request already? */ | ||
360 | pglen && /* valid bytes of file? */ | ||
361 | (end < pglen || offset)) /* replace all valid bytes? */ | ||
362 | return 1; | ||
363 | return 0; | ||
364 | } | ||
365 | |||
366 | /* | ||
331 | * This does the "real" work of the write. We must allocate and lock the | 367 | * This does the "real" work of the write. We must allocate and lock the |
332 | * page to be sent back to the generic routine, which then copies the | 368 | * page to be sent back to the generic routine, which then copies the |
333 | * data from user space. | 369 | * data from user space. |
@@ -340,15 +376,16 @@ static int nfs_write_begin(struct file *file, struct address_space *mapping, | |||
340 | struct page **pagep, void **fsdata) | 376 | struct page **pagep, void **fsdata) |
341 | { | 377 | { |
342 | int ret; | 378 | int ret; |
343 | pgoff_t index; | 379 | pgoff_t index = pos >> PAGE_CACHE_SHIFT; |
344 | struct page *page; | 380 | struct page *page; |
345 | index = pos >> PAGE_CACHE_SHIFT; | 381 | int once_thru = 0; |
346 | 382 | ||
347 | dfprintk(PAGECACHE, "NFS: write_begin(%s/%s(%ld), %u@%lld)\n", | 383 | dfprintk(PAGECACHE, "NFS: write_begin(%s/%s(%ld), %u@%lld)\n", |
348 | file->f_path.dentry->d_parent->d_name.name, | 384 | file->f_path.dentry->d_parent->d_name.name, |
349 | file->f_path.dentry->d_name.name, | 385 | file->f_path.dentry->d_name.name, |
350 | mapping->host->i_ino, len, (long long) pos); | 386 | mapping->host->i_ino, len, (long long) pos); |
351 | 387 | ||
388 | start: | ||
352 | /* | 389 | /* |
353 | * Prevent starvation issues if someone is doing a consistency | 390 | * Prevent starvation issues if someone is doing a consistency |
354 | * sync-to-disk | 391 | * sync-to-disk |
@@ -367,6 +404,13 @@ static int nfs_write_begin(struct file *file, struct address_space *mapping, | |||
367 | if (ret) { | 404 | if (ret) { |
368 | unlock_page(page); | 405 | unlock_page(page); |
369 | page_cache_release(page); | 406 | page_cache_release(page); |
407 | } else if (!once_thru && | ||
408 | nfs_want_read_modify_write(file, page, pos, len)) { | ||
409 | once_thru = 1; | ||
410 | ret = nfs_readpage(file, page); | ||
411 | page_cache_release(page); | ||
412 | if (!ret) | ||
413 | goto start; | ||
370 | } | 414 | } |
371 | return ret; | 415 | return ret; |
372 | } | 416 | } |
@@ -479,7 +523,9 @@ const struct address_space_operations nfs_file_aops = { | |||
479 | .invalidatepage = nfs_invalidate_page, | 523 | .invalidatepage = nfs_invalidate_page, |
480 | .releasepage = nfs_release_page, | 524 | .releasepage = nfs_release_page, |
481 | .direct_IO = nfs_direct_IO, | 525 | .direct_IO = nfs_direct_IO, |
526 | .migratepage = nfs_migrate_page, | ||
482 | .launder_page = nfs_launder_page, | 527 | .launder_page = nfs_launder_page, |
528 | .error_remove_page = generic_error_remove_page, | ||
483 | }; | 529 | }; |
484 | 530 | ||
485 | /* | 531 | /* |
@@ -526,7 +572,7 @@ out_unlock: | |||
526 | return VM_FAULT_SIGBUS; | 572 | return VM_FAULT_SIGBUS; |
527 | } | 573 | } |
528 | 574 | ||
529 | static struct vm_operations_struct nfs_file_vm_ops = { | 575 | static const struct vm_operations_struct nfs_file_vm_ops = { |
530 | .fault = filemap_fault, | 576 | .fault = filemap_fault, |
531 | .page_mkwrite = nfs_vm_page_mkwrite, | 577 | .page_mkwrite = nfs_vm_page_mkwrite, |
532 | }; | 578 | }; |
diff --git a/fs/nfs/fscache.c b/fs/nfs/fscache.c index 379be678cb7e..70fad69eb959 100644 --- a/fs/nfs/fscache.c +++ b/fs/nfs/fscache.c | |||
@@ -58,17 +58,34 @@ void nfs_fscache_release_client_cookie(struct nfs_client *clp) | |||
58 | /* | 58 | /* |
59 | * Get the cache cookie for an NFS superblock. We have to handle | 59 | * Get the cache cookie for an NFS superblock. We have to handle |
60 | * uniquification here because the cache doesn't do it for us. | 60 | * uniquification here because the cache doesn't do it for us. |
61 | * | ||
62 | * The default uniquifier is just an empty string, but it may be overridden | ||
63 | * either by the 'fsc=xxx' option to mount, or by inheriting it from the parent | ||
64 | * superblock across an automount point of some nature. | ||
61 | */ | 65 | */ |
62 | void nfs_fscache_get_super_cookie(struct super_block *sb, | 66 | void nfs_fscache_get_super_cookie(struct super_block *sb, const char *uniq, |
63 | struct nfs_parsed_mount_data *data) | 67 | struct nfs_clone_mount *mntdata) |
64 | { | 68 | { |
65 | struct nfs_fscache_key *key, *xkey; | 69 | struct nfs_fscache_key *key, *xkey; |
66 | struct nfs_server *nfss = NFS_SB(sb); | 70 | struct nfs_server *nfss = NFS_SB(sb); |
67 | struct rb_node **p, *parent; | 71 | struct rb_node **p, *parent; |
68 | const char *uniq = data->fscache_uniq ?: ""; | ||
69 | int diff, ulen; | 72 | int diff, ulen; |
70 | 73 | ||
71 | ulen = strlen(uniq); | 74 | if (uniq) { |
75 | ulen = strlen(uniq); | ||
76 | } else if (mntdata) { | ||
77 | struct nfs_server *mnt_s = NFS_SB(mntdata->sb); | ||
78 | if (mnt_s->fscache_key) { | ||
79 | uniq = mnt_s->fscache_key->key.uniquifier; | ||
80 | ulen = mnt_s->fscache_key->key.uniq_len; | ||
81 | } | ||
82 | } | ||
83 | |||
84 | if (!uniq) { | ||
85 | uniq = ""; | ||
86 | ulen = 1; | ||
87 | } | ||
88 | |||
72 | key = kzalloc(sizeof(*key) + ulen, GFP_KERNEL); | 89 | key = kzalloc(sizeof(*key) + ulen, GFP_KERNEL); |
73 | if (!key) | 90 | if (!key) |
74 | return; | 91 | return; |
diff --git a/fs/nfs/fscache.h b/fs/nfs/fscache.h index 6e809bb0ff08..b9c572d0679f 100644 --- a/fs/nfs/fscache.h +++ b/fs/nfs/fscache.h | |||
@@ -74,7 +74,8 @@ extern void nfs_fscache_get_client_cookie(struct nfs_client *); | |||
74 | extern void nfs_fscache_release_client_cookie(struct nfs_client *); | 74 | extern void nfs_fscache_release_client_cookie(struct nfs_client *); |
75 | 75 | ||
76 | extern void nfs_fscache_get_super_cookie(struct super_block *, | 76 | extern void nfs_fscache_get_super_cookie(struct super_block *, |
77 | struct nfs_parsed_mount_data *); | 77 | const char *, |
78 | struct nfs_clone_mount *); | ||
78 | extern void nfs_fscache_release_super_cookie(struct super_block *); | 79 | extern void nfs_fscache_release_super_cookie(struct super_block *); |
79 | 80 | ||
80 | extern void nfs_fscache_init_inode_cookie(struct inode *); | 81 | extern void nfs_fscache_init_inode_cookie(struct inode *); |
@@ -173,7 +174,8 @@ static inline void nfs_fscache_release_client_cookie(struct nfs_client *clp) {} | |||
173 | 174 | ||
174 | static inline void nfs_fscache_get_super_cookie( | 175 | static inline void nfs_fscache_get_super_cookie( |
175 | struct super_block *sb, | 176 | struct super_block *sb, |
176 | struct nfs_parsed_mount_data *data) | 177 | const char *uniq, |
178 | struct nfs_clone_mount *mntdata) | ||
177 | { | 179 | { |
178 | } | 180 | } |
179 | static inline void nfs_fscache_release_super_cookie(struct super_block *sb) {} | 181 | static inline void nfs_fscache_release_super_cookie(struct super_block *sb) {} |
diff --git a/fs/nfs/idmap.c b/fs/nfs/idmap.c index 86147b0ab2cf..21a84d45916f 100644 --- a/fs/nfs/idmap.c +++ b/fs/nfs/idmap.c | |||
@@ -101,7 +101,7 @@ static void idmap_pipe_destroy_msg(struct rpc_pipe_msg *); | |||
101 | 101 | ||
102 | static unsigned int fnvhash32(const void *, size_t); | 102 | static unsigned int fnvhash32(const void *, size_t); |
103 | 103 | ||
104 | static struct rpc_pipe_ops idmap_upcall_ops = { | 104 | static const struct rpc_pipe_ops idmap_upcall_ops = { |
105 | .upcall = idmap_pipe_upcall, | 105 | .upcall = idmap_pipe_upcall, |
106 | .downcall = idmap_pipe_downcall, | 106 | .downcall = idmap_pipe_downcall, |
107 | .destroy_msg = idmap_pipe_destroy_msg, | 107 | .destroy_msg = idmap_pipe_destroy_msg, |
@@ -119,8 +119,8 @@ nfs_idmap_new(struct nfs_client *clp) | |||
119 | if (idmap == NULL) | 119 | if (idmap == NULL) |
120 | return -ENOMEM; | 120 | return -ENOMEM; |
121 | 121 | ||
122 | idmap->idmap_dentry = rpc_mkpipe(clp->cl_rpcclient->cl_dentry, "idmap", | 122 | idmap->idmap_dentry = rpc_mkpipe(clp->cl_rpcclient->cl_path.dentry, |
123 | idmap, &idmap_upcall_ops, 0); | 123 | "idmap", idmap, &idmap_upcall_ops, 0); |
124 | if (IS_ERR(idmap->idmap_dentry)) { | 124 | if (IS_ERR(idmap->idmap_dentry)) { |
125 | error = PTR_ERR(idmap->idmap_dentry); | 125 | error = PTR_ERR(idmap->idmap_dentry); |
126 | kfree(idmap); | 126 | kfree(idmap); |
diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c index bd7938eda6a8..faa091865ad0 100644 --- a/fs/nfs/inode.c +++ b/fs/nfs/inode.c | |||
@@ -46,6 +46,7 @@ | |||
46 | #include "iostat.h" | 46 | #include "iostat.h" |
47 | #include "internal.h" | 47 | #include "internal.h" |
48 | #include "fscache.h" | 48 | #include "fscache.h" |
49 | #include "dns_resolve.h" | ||
49 | 50 | ||
50 | #define NFSDBG_FACILITY NFSDBG_VFS | 51 | #define NFSDBG_FACILITY NFSDBG_VFS |
51 | 52 | ||
@@ -286,6 +287,11 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr) | |||
286 | /* We can't support update_atime(), since the server will reset it */ | 287 | /* We can't support update_atime(), since the server will reset it */ |
287 | inode->i_flags |= S_NOATIME|S_NOCMTIME; | 288 | inode->i_flags |= S_NOATIME|S_NOCMTIME; |
288 | inode->i_mode = fattr->mode; | 289 | inode->i_mode = fattr->mode; |
290 | if ((fattr->valid & NFS_ATTR_FATTR_MODE) == 0 | ||
291 | && nfs_server_capable(inode, NFS_CAP_MODE)) | ||
292 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
293 | | NFS_INO_INVALID_ACCESS | ||
294 | | NFS_INO_INVALID_ACL; | ||
289 | /* Why so? Because we want revalidate for devices/FIFOs, and | 295 | /* Why so? Because we want revalidate for devices/FIFOs, and |
290 | * that's precisely what we have in nfs_file_inode_operations. | 296 | * that's precisely what we have in nfs_file_inode_operations. |
291 | */ | 297 | */ |
@@ -330,20 +336,46 @@ nfs_fhget(struct super_block *sb, struct nfs_fh *fh, struct nfs_fattr *fattr) | |||
330 | nfsi->attr_gencount = fattr->gencount; | 336 | nfsi->attr_gencount = fattr->gencount; |
331 | if (fattr->valid & NFS_ATTR_FATTR_ATIME) | 337 | if (fattr->valid & NFS_ATTR_FATTR_ATIME) |
332 | inode->i_atime = fattr->atime; | 338 | inode->i_atime = fattr->atime; |
339 | else if (nfs_server_capable(inode, NFS_CAP_ATIME)) | ||
340 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR; | ||
333 | if (fattr->valid & NFS_ATTR_FATTR_MTIME) | 341 | if (fattr->valid & NFS_ATTR_FATTR_MTIME) |
334 | inode->i_mtime = fattr->mtime; | 342 | inode->i_mtime = fattr->mtime; |
343 | else if (nfs_server_capable(inode, NFS_CAP_MTIME)) | ||
344 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
345 | | NFS_INO_INVALID_DATA; | ||
335 | if (fattr->valid & NFS_ATTR_FATTR_CTIME) | 346 | if (fattr->valid & NFS_ATTR_FATTR_CTIME) |
336 | inode->i_ctime = fattr->ctime; | 347 | inode->i_ctime = fattr->ctime; |
348 | else if (nfs_server_capable(inode, NFS_CAP_CTIME)) | ||
349 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
350 | | NFS_INO_INVALID_ACCESS | ||
351 | | NFS_INO_INVALID_ACL; | ||
337 | if (fattr->valid & NFS_ATTR_FATTR_CHANGE) | 352 | if (fattr->valid & NFS_ATTR_FATTR_CHANGE) |
338 | nfsi->change_attr = fattr->change_attr; | 353 | nfsi->change_attr = fattr->change_attr; |
354 | else if (nfs_server_capable(inode, NFS_CAP_CHANGE_ATTR)) | ||
355 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
356 | | NFS_INO_INVALID_DATA; | ||
339 | if (fattr->valid & NFS_ATTR_FATTR_SIZE) | 357 | if (fattr->valid & NFS_ATTR_FATTR_SIZE) |
340 | inode->i_size = nfs_size_to_loff_t(fattr->size); | 358 | inode->i_size = nfs_size_to_loff_t(fattr->size); |
359 | else | ||
360 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
361 | | NFS_INO_INVALID_DATA | ||
362 | | NFS_INO_REVAL_PAGECACHE; | ||
341 | if (fattr->valid & NFS_ATTR_FATTR_NLINK) | 363 | if (fattr->valid & NFS_ATTR_FATTR_NLINK) |
342 | inode->i_nlink = fattr->nlink; | 364 | inode->i_nlink = fattr->nlink; |
365 | else if (nfs_server_capable(inode, NFS_CAP_NLINK)) | ||
366 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR; | ||
343 | if (fattr->valid & NFS_ATTR_FATTR_OWNER) | 367 | if (fattr->valid & NFS_ATTR_FATTR_OWNER) |
344 | inode->i_uid = fattr->uid; | 368 | inode->i_uid = fattr->uid; |
369 | else if (nfs_server_capable(inode, NFS_CAP_OWNER)) | ||
370 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
371 | | NFS_INO_INVALID_ACCESS | ||
372 | | NFS_INO_INVALID_ACL; | ||
345 | if (fattr->valid & NFS_ATTR_FATTR_GROUP) | 373 | if (fattr->valid & NFS_ATTR_FATTR_GROUP) |
346 | inode->i_gid = fattr->gid; | 374 | inode->i_gid = fattr->gid; |
375 | else if (nfs_server_capable(inode, NFS_CAP_OWNER_GROUP)) | ||
376 | nfsi->cache_validity |= NFS_INO_INVALID_ATTR | ||
377 | | NFS_INO_INVALID_ACCESS | ||
378 | | NFS_INO_INVALID_ACL; | ||
347 | if (fattr->valid & NFS_ATTR_FATTR_BLOCKS_USED) | 379 | if (fattr->valid & NFS_ATTR_FATTR_BLOCKS_USED) |
348 | inode->i_blocks = fattr->du.nfs2.blocks; | 380 | inode->i_blocks = fattr->du.nfs2.blocks; |
349 | if (fattr->valid & NFS_ATTR_FATTR_SPACE_USED) { | 381 | if (fattr->valid & NFS_ATTR_FATTR_SPACE_USED) { |
@@ -426,49 +458,21 @@ nfs_setattr(struct dentry *dentry, struct iattr *attr) | |||
426 | */ | 458 | */ |
427 | static int nfs_vmtruncate(struct inode * inode, loff_t offset) | 459 | static int nfs_vmtruncate(struct inode * inode, loff_t offset) |
428 | { | 460 | { |
429 | if (i_size_read(inode) < offset) { | 461 | loff_t oldsize; |
430 | unsigned long limit; | 462 | int err; |
431 | |||
432 | limit = current->signal->rlim[RLIMIT_FSIZE].rlim_cur; | ||
433 | if (limit != RLIM_INFINITY && offset > limit) | ||
434 | goto out_sig; | ||
435 | if (offset > inode->i_sb->s_maxbytes) | ||
436 | goto out_big; | ||
437 | spin_lock(&inode->i_lock); | ||
438 | i_size_write(inode, offset); | ||
439 | spin_unlock(&inode->i_lock); | ||
440 | } else { | ||
441 | struct address_space *mapping = inode->i_mapping; | ||
442 | 463 | ||
443 | /* | 464 | err = inode_newsize_ok(inode, offset); |
444 | * truncation of in-use swapfiles is disallowed - it would | 465 | if (err) |
445 | * cause subsequent swapout to scribble on the now-freed | 466 | goto out; |
446 | * blocks. | ||
447 | */ | ||
448 | if (IS_SWAPFILE(inode)) | ||
449 | return -ETXTBSY; | ||
450 | spin_lock(&inode->i_lock); | ||
451 | i_size_write(inode, offset); | ||
452 | spin_unlock(&inode->i_lock); | ||
453 | 467 | ||
454 | /* | 468 | spin_lock(&inode->i_lock); |
455 | * unmap_mapping_range is called twice, first simply for | 469 | oldsize = inode->i_size; |
456 | * efficiency so that truncate_inode_pages does fewer | 470 | i_size_write(inode, offset); |
457 | * single-page unmaps. However after this first call, and | 471 | spin_unlock(&inode->i_lock); |
458 | * before truncate_inode_pages finishes, it is possible for | 472 | |
459 | * private pages to be COWed, which remain after | 473 | truncate_pagecache(inode, oldsize, offset); |
460 | * truncate_inode_pages finishes, hence the second | 474 | out: |
461 | * unmap_mapping_range call must be made for correctness. | 475 | return err; |
462 | */ | ||
463 | unmap_mapping_range(mapping, offset + PAGE_SIZE - 1, 0, 1); | ||
464 | truncate_inode_pages(mapping, offset); | ||
465 | unmap_mapping_range(mapping, offset + PAGE_SIZE - 1, 0, 1); | ||
466 | } | ||
467 | return 0; | ||
468 | out_sig: | ||
469 | send_sig(SIGXFSZ, current, 0); | ||
470 | out_big: | ||
471 | return -EFBIG; | ||
472 | } | 476 | } |
473 | 477 | ||
474 | /** | 478 | /** |
@@ -1145,6 +1149,7 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1145 | loff_t cur_isize, new_isize; | 1149 | loff_t cur_isize, new_isize; |
1146 | unsigned long invalid = 0; | 1150 | unsigned long invalid = 0; |
1147 | unsigned long now = jiffies; | 1151 | unsigned long now = jiffies; |
1152 | unsigned long save_cache_validity; | ||
1148 | 1153 | ||
1149 | dfprintk(VFS, "NFS: %s(%s/%ld ct=%d info=0x%x)\n", | 1154 | dfprintk(VFS, "NFS: %s(%s/%ld ct=%d info=0x%x)\n", |
1150 | __func__, inode->i_sb->s_id, inode->i_ino, | 1155 | __func__, inode->i_sb->s_id, inode->i_ino, |
@@ -1171,10 +1176,11 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1171 | */ | 1176 | */ |
1172 | nfsi->read_cache_jiffies = fattr->time_start; | 1177 | nfsi->read_cache_jiffies = fattr->time_start; |
1173 | 1178 | ||
1174 | if ((fattr->valid & NFS_ATTR_FATTR_CHANGE) || (fattr->valid & (NFS_ATTR_FATTR_MTIME|NFS_ATTR_FATTR_CTIME))) | 1179 | save_cache_validity = nfsi->cache_validity; |
1175 | nfsi->cache_validity &= ~(NFS_INO_INVALID_ATTR | 1180 | nfsi->cache_validity &= ~(NFS_INO_INVALID_ATTR |
1176 | | NFS_INO_INVALID_ATIME | 1181 | | NFS_INO_INVALID_ATIME |
1177 | | NFS_INO_REVAL_PAGECACHE); | 1182 | | NFS_INO_REVAL_FORCED |
1183 | | NFS_INO_REVAL_PAGECACHE); | ||
1178 | 1184 | ||
1179 | /* Do atomic weak cache consistency updates */ | 1185 | /* Do atomic weak cache consistency updates */ |
1180 | nfs_wcc_update_inode(inode, fattr); | 1186 | nfs_wcc_update_inode(inode, fattr); |
@@ -1189,7 +1195,8 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1189 | nfs_force_lookup_revalidate(inode); | 1195 | nfs_force_lookup_revalidate(inode); |
1190 | nfsi->change_attr = fattr->change_attr; | 1196 | nfsi->change_attr = fattr->change_attr; |
1191 | } | 1197 | } |
1192 | } | 1198 | } else if (server->caps & NFS_CAP_CHANGE_ATTR) |
1199 | invalid |= save_cache_validity; | ||
1193 | 1200 | ||
1194 | if (fattr->valid & NFS_ATTR_FATTR_MTIME) { | 1201 | if (fattr->valid & NFS_ATTR_FATTR_MTIME) { |
1195 | /* NFSv2/v3: Check if the mtime agrees */ | 1202 | /* NFSv2/v3: Check if the mtime agrees */ |
@@ -1201,7 +1208,12 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1201 | nfs_force_lookup_revalidate(inode); | 1208 | nfs_force_lookup_revalidate(inode); |
1202 | memcpy(&inode->i_mtime, &fattr->mtime, sizeof(inode->i_mtime)); | 1209 | memcpy(&inode->i_mtime, &fattr->mtime, sizeof(inode->i_mtime)); |
1203 | } | 1210 | } |
1204 | } | 1211 | } else if (server->caps & NFS_CAP_MTIME) |
1212 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1213 | | NFS_INO_INVALID_DATA | ||
1214 | | NFS_INO_REVAL_PAGECACHE | ||
1215 | | NFS_INO_REVAL_FORCED); | ||
1216 | |||
1205 | if (fattr->valid & NFS_ATTR_FATTR_CTIME) { | 1217 | if (fattr->valid & NFS_ATTR_FATTR_CTIME) { |
1206 | /* If ctime has changed we should definitely clear access+acl caches */ | 1218 | /* If ctime has changed we should definitely clear access+acl caches */ |
1207 | if (!timespec_equal(&inode->i_ctime, &fattr->ctime)) { | 1219 | if (!timespec_equal(&inode->i_ctime, &fattr->ctime)) { |
@@ -1215,7 +1227,11 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1215 | } | 1227 | } |
1216 | memcpy(&inode->i_ctime, &fattr->ctime, sizeof(inode->i_ctime)); | 1228 | memcpy(&inode->i_ctime, &fattr->ctime, sizeof(inode->i_ctime)); |
1217 | } | 1229 | } |
1218 | } | 1230 | } else if (server->caps & NFS_CAP_CTIME) |
1231 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1232 | | NFS_INO_INVALID_ACCESS | ||
1233 | | NFS_INO_INVALID_ACL | ||
1234 | | NFS_INO_REVAL_FORCED); | ||
1219 | 1235 | ||
1220 | /* Check if our cached file size is stale */ | 1236 | /* Check if our cached file size is stale */ |
1221 | if (fattr->valid & NFS_ATTR_FATTR_SIZE) { | 1237 | if (fattr->valid & NFS_ATTR_FATTR_SIZE) { |
@@ -1231,30 +1247,50 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1231 | dprintk("NFS: isize change on server for file %s/%ld\n", | 1247 | dprintk("NFS: isize change on server for file %s/%ld\n", |
1232 | inode->i_sb->s_id, inode->i_ino); | 1248 | inode->i_sb->s_id, inode->i_ino); |
1233 | } | 1249 | } |
1234 | } | 1250 | } else |
1251 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1252 | | NFS_INO_REVAL_PAGECACHE | ||
1253 | | NFS_INO_REVAL_FORCED); | ||
1235 | 1254 | ||
1236 | 1255 | ||
1237 | if (fattr->valid & NFS_ATTR_FATTR_ATIME) | 1256 | if (fattr->valid & NFS_ATTR_FATTR_ATIME) |
1238 | memcpy(&inode->i_atime, &fattr->atime, sizeof(inode->i_atime)); | 1257 | memcpy(&inode->i_atime, &fattr->atime, sizeof(inode->i_atime)); |
1258 | else if (server->caps & NFS_CAP_ATIME) | ||
1259 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATIME | ||
1260 | | NFS_INO_REVAL_FORCED); | ||
1239 | 1261 | ||
1240 | if (fattr->valid & NFS_ATTR_FATTR_MODE) { | 1262 | if (fattr->valid & NFS_ATTR_FATTR_MODE) { |
1241 | if ((inode->i_mode & S_IALLUGO) != (fattr->mode & S_IALLUGO)) { | 1263 | if ((inode->i_mode & S_IALLUGO) != (fattr->mode & S_IALLUGO)) { |
1242 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; | 1264 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; |
1243 | inode->i_mode = fattr->mode; | 1265 | inode->i_mode = fattr->mode; |
1244 | } | 1266 | } |
1245 | } | 1267 | } else if (server->caps & NFS_CAP_MODE) |
1268 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1269 | | NFS_INO_INVALID_ACCESS | ||
1270 | | NFS_INO_INVALID_ACL | ||
1271 | | NFS_INO_REVAL_FORCED); | ||
1272 | |||
1246 | if (fattr->valid & NFS_ATTR_FATTR_OWNER) { | 1273 | if (fattr->valid & NFS_ATTR_FATTR_OWNER) { |
1247 | if (inode->i_uid != fattr->uid) { | 1274 | if (inode->i_uid != fattr->uid) { |
1248 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; | 1275 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; |
1249 | inode->i_uid = fattr->uid; | 1276 | inode->i_uid = fattr->uid; |
1250 | } | 1277 | } |
1251 | } | 1278 | } else if (server->caps & NFS_CAP_OWNER) |
1279 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1280 | | NFS_INO_INVALID_ACCESS | ||
1281 | | NFS_INO_INVALID_ACL | ||
1282 | | NFS_INO_REVAL_FORCED); | ||
1283 | |||
1252 | if (fattr->valid & NFS_ATTR_FATTR_GROUP) { | 1284 | if (fattr->valid & NFS_ATTR_FATTR_GROUP) { |
1253 | if (inode->i_gid != fattr->gid) { | 1285 | if (inode->i_gid != fattr->gid) { |
1254 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; | 1286 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_ACCESS|NFS_INO_INVALID_ACL; |
1255 | inode->i_gid = fattr->gid; | 1287 | inode->i_gid = fattr->gid; |
1256 | } | 1288 | } |
1257 | } | 1289 | } else if (server->caps & NFS_CAP_OWNER_GROUP) |
1290 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1291 | | NFS_INO_INVALID_ACCESS | ||
1292 | | NFS_INO_INVALID_ACL | ||
1293 | | NFS_INO_REVAL_FORCED); | ||
1258 | 1294 | ||
1259 | if (fattr->valid & NFS_ATTR_FATTR_NLINK) { | 1295 | if (fattr->valid & NFS_ATTR_FATTR_NLINK) { |
1260 | if (inode->i_nlink != fattr->nlink) { | 1296 | if (inode->i_nlink != fattr->nlink) { |
@@ -1263,7 +1299,9 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1263 | invalid |= NFS_INO_INVALID_DATA; | 1299 | invalid |= NFS_INO_INVALID_DATA; |
1264 | inode->i_nlink = fattr->nlink; | 1300 | inode->i_nlink = fattr->nlink; |
1265 | } | 1301 | } |
1266 | } | 1302 | } else if (server->caps & NFS_CAP_NLINK) |
1303 | invalid |= save_cache_validity & (NFS_INO_INVALID_ATTR | ||
1304 | | NFS_INO_REVAL_FORCED); | ||
1267 | 1305 | ||
1268 | if (fattr->valid & NFS_ATTR_FATTR_SPACE_USED) { | 1306 | if (fattr->valid & NFS_ATTR_FATTR_SPACE_USED) { |
1269 | /* | 1307 | /* |
@@ -1293,9 +1331,8 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
1293 | || S_ISLNK(inode->i_mode))) | 1331 | || S_ISLNK(inode->i_mode))) |
1294 | invalid &= ~NFS_INO_INVALID_DATA; | 1332 | invalid &= ~NFS_INO_INVALID_DATA; |
1295 | if (!nfs_have_delegation(inode, FMODE_READ) || | 1333 | if (!nfs_have_delegation(inode, FMODE_READ) || |
1296 | (nfsi->cache_validity & NFS_INO_REVAL_FORCED)) | 1334 | (save_cache_validity & NFS_INO_REVAL_FORCED)) |
1297 | nfsi->cache_validity |= invalid; | 1335 | nfsi->cache_validity |= invalid; |
1298 | nfsi->cache_validity &= ~NFS_INO_REVAL_FORCED; | ||
1299 | 1336 | ||
1300 | return 0; | 1337 | return 0; |
1301 | out_changed: | 1338 | out_changed: |
@@ -1442,6 +1479,10 @@ static int __init init_nfs_fs(void) | |||
1442 | { | 1479 | { |
1443 | int err; | 1480 | int err; |
1444 | 1481 | ||
1482 | err = nfs_dns_resolver_init(); | ||
1483 | if (err < 0) | ||
1484 | goto out8; | ||
1485 | |||
1445 | err = nfs_fscache_register(); | 1486 | err = nfs_fscache_register(); |
1446 | if (err < 0) | 1487 | if (err < 0) |
1447 | goto out7; | 1488 | goto out7; |
@@ -1500,6 +1541,8 @@ out5: | |||
1500 | out6: | 1541 | out6: |
1501 | nfs_fscache_unregister(); | 1542 | nfs_fscache_unregister(); |
1502 | out7: | 1543 | out7: |
1544 | nfs_dns_resolver_destroy(); | ||
1545 | out8: | ||
1503 | return err; | 1546 | return err; |
1504 | } | 1547 | } |
1505 | 1548 | ||
@@ -1511,6 +1554,7 @@ static void __exit exit_nfs_fs(void) | |||
1511 | nfs_destroy_inodecache(); | 1554 | nfs_destroy_inodecache(); |
1512 | nfs_destroy_nfspagecache(); | 1555 | nfs_destroy_nfspagecache(); |
1513 | nfs_fscache_unregister(); | 1556 | nfs_fscache_unregister(); |
1557 | nfs_dns_resolver_destroy(); | ||
1514 | #ifdef CONFIG_PROC_FS | 1558 | #ifdef CONFIG_PROC_FS |
1515 | rpc_proc_unregister("nfs"); | 1559 | rpc_proc_unregister("nfs"); |
1516 | #endif | 1560 | #endif |
diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h index 7dd90a6769d0..e21b1bb9972f 100644 --- a/fs/nfs/internal.h +++ b/fs/nfs/internal.h | |||
@@ -49,6 +49,11 @@ struct nfs_clone_mount { | |||
49 | #define NFS_MAX_SECFLAVORS (12) | 49 | #define NFS_MAX_SECFLAVORS (12) |
50 | 50 | ||
51 | /* | 51 | /* |
52 | * Value used if the user did not specify a port value. | ||
53 | */ | ||
54 | #define NFS_UNSPEC_PORT (-1) | ||
55 | |||
56 | /* | ||
52 | * In-kernel mount arguments | 57 | * In-kernel mount arguments |
53 | */ | 58 | */ |
54 | struct nfs_parsed_mount_data { | 59 | struct nfs_parsed_mount_data { |
@@ -63,6 +68,7 @@ struct nfs_parsed_mount_data { | |||
63 | unsigned int auth_flavor_len; | 68 | unsigned int auth_flavor_len; |
64 | rpc_authflavor_t auth_flavors[1]; | 69 | rpc_authflavor_t auth_flavors[1]; |
65 | char *client_address; | 70 | char *client_address; |
71 | unsigned int version; | ||
66 | unsigned int minorversion; | 72 | unsigned int minorversion; |
67 | char *fscache_uniq; | 73 | char *fscache_uniq; |
68 | 74 | ||
@@ -71,7 +77,7 @@ struct nfs_parsed_mount_data { | |||
71 | size_t addrlen; | 77 | size_t addrlen; |
72 | char *hostname; | 78 | char *hostname; |
73 | u32 version; | 79 | u32 version; |
74 | unsigned short port; | 80 | int port; |
75 | unsigned short protocol; | 81 | unsigned short protocol; |
76 | } mount_server; | 82 | } mount_server; |
77 | 83 | ||
@@ -80,7 +86,7 @@ struct nfs_parsed_mount_data { | |||
80 | size_t addrlen; | 86 | size_t addrlen; |
81 | char *hostname; | 87 | char *hostname; |
82 | char *export_path; | 88 | char *export_path; |
83 | unsigned short port; | 89 | int port; |
84 | unsigned short protocol; | 90 | unsigned short protocol; |
85 | } nfs_server; | 91 | } nfs_server; |
86 | 92 | ||
@@ -102,6 +108,7 @@ struct nfs_mount_request { | |||
102 | }; | 108 | }; |
103 | 109 | ||
104 | extern int nfs_mount(struct nfs_mount_request *info); | 110 | extern int nfs_mount(struct nfs_mount_request *info); |
111 | extern void nfs_umount(const struct nfs_mount_request *info); | ||
105 | 112 | ||
106 | /* client.c */ | 113 | /* client.c */ |
107 | extern struct rpc_program nfs_program; | 114 | extern struct rpc_program nfs_program; |
@@ -213,7 +220,6 @@ void nfs_zap_acl_cache(struct inode *inode); | |||
213 | extern int nfs_wait_bit_killable(void *word); | 220 | extern int nfs_wait_bit_killable(void *word); |
214 | 221 | ||
215 | /* super.c */ | 222 | /* super.c */ |
216 | void nfs_parse_ip_address(char *, size_t, struct sockaddr *, size_t *); | ||
217 | extern struct file_system_type nfs_xdev_fs_type; | 223 | extern struct file_system_type nfs_xdev_fs_type; |
218 | #ifdef CONFIG_NFS_V4 | 224 | #ifdef CONFIG_NFS_V4 |
219 | extern struct file_system_type nfs4_xdev_fs_type; | 225 | extern struct file_system_type nfs4_xdev_fs_type; |
@@ -248,6 +254,12 @@ extern void nfs_read_prepare(struct rpc_task *task, void *calldata); | |||
248 | 254 | ||
249 | /* write.c */ | 255 | /* write.c */ |
250 | extern void nfs_write_prepare(struct rpc_task *task, void *calldata); | 256 | extern void nfs_write_prepare(struct rpc_task *task, void *calldata); |
257 | #ifdef CONFIG_MIGRATION | ||
258 | extern int nfs_migrate_page(struct address_space *, | ||
259 | struct page *, struct page *); | ||
260 | #else | ||
261 | #define nfs_migrate_page NULL | ||
262 | #endif | ||
251 | 263 | ||
252 | /* nfs4proc.c */ | 264 | /* nfs4proc.c */ |
253 | extern int _nfs4_call_sync(struct nfs_server *server, | 265 | extern int _nfs4_call_sync(struct nfs_server *server, |
@@ -368,24 +380,3 @@ unsigned int nfs_page_array_len(unsigned int base, size_t len) | |||
368 | return ((unsigned long)len + (unsigned long)base + | 380 | return ((unsigned long)len + (unsigned long)base + |
369 | PAGE_SIZE - 1) >> PAGE_SHIFT; | 381 | PAGE_SIZE - 1) >> PAGE_SHIFT; |
370 | } | 382 | } |
371 | |||
372 | #define IPV6_SCOPE_DELIMITER '%' | ||
373 | |||
374 | /* | ||
375 | * Set the port number in an address. Be agnostic about the address | ||
376 | * family. | ||
377 | */ | ||
378 | static inline void nfs_set_port(struct sockaddr *sap, unsigned short port) | ||
379 | { | ||
380 | struct sockaddr_in *ap = (struct sockaddr_in *)sap; | ||
381 | struct sockaddr_in6 *ap6 = (struct sockaddr_in6 *)sap; | ||
382 | |||
383 | switch (sap->sa_family) { | ||
384 | case AF_INET: | ||
385 | ap->sin_port = htons(port); | ||
386 | break; | ||
387 | case AF_INET6: | ||
388 | ap6->sin6_port = htons(port); | ||
389 | break; | ||
390 | } | ||
391 | } | ||
diff --git a/fs/nfs/mount_clnt.c b/fs/nfs/mount_clnt.c index 38ef9eaec407..0adefc40cc89 100644 --- a/fs/nfs/mount_clnt.c +++ b/fs/nfs/mount_clnt.c | |||
@@ -209,6 +209,71 @@ out_mnt_err: | |||
209 | goto out; | 209 | goto out; |
210 | } | 210 | } |
211 | 211 | ||
212 | /** | ||
213 | * nfs_umount - Notify a server that we have unmounted this export | ||
214 | * @info: pointer to umount request arguments | ||
215 | * | ||
216 | * MOUNTPROC_UMNT is advisory, so we set a short timeout, and always | ||
217 | * use UDP. | ||
218 | */ | ||
219 | void nfs_umount(const struct nfs_mount_request *info) | ||
220 | { | ||
221 | static const struct rpc_timeout nfs_umnt_timeout = { | ||
222 | .to_initval = 1 * HZ, | ||
223 | .to_maxval = 3 * HZ, | ||
224 | .to_retries = 2, | ||
225 | }; | ||
226 | struct rpc_create_args args = { | ||
227 | .protocol = IPPROTO_UDP, | ||
228 | .address = info->sap, | ||
229 | .addrsize = info->salen, | ||
230 | .timeout = &nfs_umnt_timeout, | ||
231 | .servername = info->hostname, | ||
232 | .program = &mnt_program, | ||
233 | .version = info->version, | ||
234 | .authflavor = RPC_AUTH_UNIX, | ||
235 | .flags = RPC_CLNT_CREATE_NOPING, | ||
236 | }; | ||
237 | struct mountres result; | ||
238 | struct rpc_message msg = { | ||
239 | .rpc_argp = info->dirpath, | ||
240 | .rpc_resp = &result, | ||
241 | }; | ||
242 | struct rpc_clnt *clnt; | ||
243 | int status; | ||
244 | |||
245 | if (info->noresvport) | ||
246 | args.flags |= RPC_CLNT_CREATE_NONPRIVPORT; | ||
247 | |||
248 | clnt = rpc_create(&args); | ||
249 | if (unlikely(IS_ERR(clnt))) | ||
250 | goto out_clnt_err; | ||
251 | |||
252 | dprintk("NFS: sending UMNT request for %s:%s\n", | ||
253 | (info->hostname ? info->hostname : "server"), info->dirpath); | ||
254 | |||
255 | if (info->version == NFS_MNT3_VERSION) | ||
256 | msg.rpc_proc = &clnt->cl_procinfo[MOUNTPROC3_UMNT]; | ||
257 | else | ||
258 | msg.rpc_proc = &clnt->cl_procinfo[MOUNTPROC_UMNT]; | ||
259 | |||
260 | status = rpc_call_sync(clnt, &msg, 0); | ||
261 | rpc_shutdown_client(clnt); | ||
262 | |||
263 | if (unlikely(status < 0)) | ||
264 | goto out_call_err; | ||
265 | |||
266 | return; | ||
267 | |||
268 | out_clnt_err: | ||
269 | dprintk("NFS: failed to create UMNT RPC client, status=%ld\n", | ||
270 | PTR_ERR(clnt)); | ||
271 | return; | ||
272 | |||
273 | out_call_err: | ||
274 | dprintk("NFS: UMNT request failed, status=%d\n", status); | ||
275 | } | ||
276 | |||
212 | /* | 277 | /* |
213 | * XDR encode/decode functions for MOUNT | 278 | * XDR encode/decode functions for MOUNT |
214 | */ | 279 | */ |
@@ -258,7 +323,7 @@ static int decode_status(struct xdr_stream *xdr, struct mountres *res) | |||
258 | return -EIO; | 323 | return -EIO; |
259 | status = ntohl(*p); | 324 | status = ntohl(*p); |
260 | 325 | ||
261 | for (i = 0; i <= ARRAY_SIZE(mnt_errtbl); i++) { | 326 | for (i = 0; i < ARRAY_SIZE(mnt_errtbl); i++) { |
262 | if (mnt_errtbl[i].status == status) { | 327 | if (mnt_errtbl[i].status == status) { |
263 | res->errno = mnt_errtbl[i].errno; | 328 | res->errno = mnt_errtbl[i].errno; |
264 | return 0; | 329 | return 0; |
@@ -309,7 +374,7 @@ static int decode_fhs_status(struct xdr_stream *xdr, struct mountres *res) | |||
309 | return -EIO; | 374 | return -EIO; |
310 | status = ntohl(*p); | 375 | status = ntohl(*p); |
311 | 376 | ||
312 | for (i = 0; i <= ARRAY_SIZE(mnt3_errtbl); i++) { | 377 | for (i = 0; i < ARRAY_SIZE(mnt3_errtbl); i++) { |
313 | if (mnt3_errtbl[i].status == status) { | 378 | if (mnt3_errtbl[i].status == status) { |
314 | res->errno = mnt3_errtbl[i].errno; | 379 | res->errno = mnt3_errtbl[i].errno; |
315 | return 0; | 380 | return 0; |
@@ -407,6 +472,13 @@ static struct rpc_procinfo mnt_procedures[] = { | |||
407 | .p_statidx = MOUNTPROC_MNT, | 472 | .p_statidx = MOUNTPROC_MNT, |
408 | .p_name = "MOUNT", | 473 | .p_name = "MOUNT", |
409 | }, | 474 | }, |
475 | [MOUNTPROC_UMNT] = { | ||
476 | .p_proc = MOUNTPROC_UMNT, | ||
477 | .p_encode = (kxdrproc_t)mnt_enc_dirpath, | ||
478 | .p_arglen = MNT_enc_dirpath_sz, | ||
479 | .p_statidx = MOUNTPROC_UMNT, | ||
480 | .p_name = "UMOUNT", | ||
481 | }, | ||
410 | }; | 482 | }; |
411 | 483 | ||
412 | static struct rpc_procinfo mnt3_procedures[] = { | 484 | static struct rpc_procinfo mnt3_procedures[] = { |
@@ -419,6 +491,13 @@ static struct rpc_procinfo mnt3_procedures[] = { | |||
419 | .p_statidx = MOUNTPROC3_MNT, | 491 | .p_statidx = MOUNTPROC3_MNT, |
420 | .p_name = "MOUNT", | 492 | .p_name = "MOUNT", |
421 | }, | 493 | }, |
494 | [MOUNTPROC3_UMNT] = { | ||
495 | .p_proc = MOUNTPROC3_UMNT, | ||
496 | .p_encode = (kxdrproc_t)mnt_enc_dirpath, | ||
497 | .p_arglen = MNT_enc_dirpath_sz, | ||
498 | .p_statidx = MOUNTPROC3_UMNT, | ||
499 | .p_name = "UMOUNT", | ||
500 | }, | ||
422 | }; | 501 | }; |
423 | 502 | ||
424 | 503 | ||
diff --git a/fs/nfs/nfs2xdr.c b/fs/nfs/nfs2xdr.c index c862c9340f9a..5e078b222b4e 100644 --- a/fs/nfs/nfs2xdr.c +++ b/fs/nfs/nfs2xdr.c | |||
@@ -13,7 +13,6 @@ | |||
13 | #include <linux/time.h> | 13 | #include <linux/time.h> |
14 | #include <linux/mm.h> | 14 | #include <linux/mm.h> |
15 | #include <linux/slab.h> | 15 | #include <linux/slab.h> |
16 | #include <linux/utsname.h> | ||
17 | #include <linux/errno.h> | 16 | #include <linux/errno.h> |
18 | #include <linux/string.h> | 17 | #include <linux/string.h> |
19 | #include <linux/in.h> | 18 | #include <linux/in.h> |
diff --git a/fs/nfs/nfs3proc.c b/fs/nfs/nfs3proc.c index d0cc5ce0edfe..3f8881d1a050 100644 --- a/fs/nfs/nfs3proc.c +++ b/fs/nfs/nfs3proc.c | |||
@@ -7,7 +7,6 @@ | |||
7 | */ | 7 | */ |
8 | 8 | ||
9 | #include <linux/mm.h> | 9 | #include <linux/mm.h> |
10 | #include <linux/utsname.h> | ||
11 | #include <linux/errno.h> | 10 | #include <linux/errno.h> |
12 | #include <linux/string.h> | 11 | #include <linux/string.h> |
13 | #include <linux/sunrpc/clnt.h> | 12 | #include <linux/sunrpc/clnt.h> |
@@ -299,7 +298,6 @@ static void nfs3_free_createdata(struct nfs3_createdata *data) | |||
299 | 298 | ||
300 | /* | 299 | /* |
301 | * Create a regular file. | 300 | * Create a regular file. |
302 | * For now, we don't implement O_EXCL. | ||
303 | */ | 301 | */ |
304 | static int | 302 | static int |
305 | nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr, | 303 | nfs3_proc_create(struct inode *dir, struct dentry *dentry, struct iattr *sattr, |
diff --git a/fs/nfs/nfs3xdr.c b/fs/nfs/nfs3xdr.c index 35869a4921f1..5fe5492fbd29 100644 --- a/fs/nfs/nfs3xdr.c +++ b/fs/nfs/nfs3xdr.c | |||
@@ -10,7 +10,6 @@ | |||
10 | #include <linux/time.h> | 10 | #include <linux/time.h> |
11 | #include <linux/mm.h> | 11 | #include <linux/mm.h> |
12 | #include <linux/slab.h> | 12 | #include <linux/slab.h> |
13 | #include <linux/utsname.h> | ||
14 | #include <linux/errno.h> | 13 | #include <linux/errno.h> |
15 | #include <linux/string.h> | 14 | #include <linux/string.h> |
16 | #include <linux/in.h> | 15 | #include <linux/in.h> |
diff --git a/fs/nfs/nfs4namespace.c b/fs/nfs/nfs4namespace.c index 2a2a0a7143ad..2636c26d56fa 100644 --- a/fs/nfs/nfs4namespace.c +++ b/fs/nfs/nfs4namespace.c | |||
@@ -17,6 +17,7 @@ | |||
17 | #include <linux/inet.h> | 17 | #include <linux/inet.h> |
18 | #include "internal.h" | 18 | #include "internal.h" |
19 | #include "nfs4_fs.h" | 19 | #include "nfs4_fs.h" |
20 | #include "dns_resolve.h" | ||
20 | 21 | ||
21 | #define NFSDBG_FACILITY NFSDBG_VFS | 22 | #define NFSDBG_FACILITY NFSDBG_VFS |
22 | 23 | ||
@@ -95,6 +96,20 @@ static int nfs4_validate_fspath(const struct vfsmount *mnt_parent, | |||
95 | return 0; | 96 | return 0; |
96 | } | 97 | } |
97 | 98 | ||
99 | static size_t nfs_parse_server_name(char *string, size_t len, | ||
100 | struct sockaddr *sa, size_t salen) | ||
101 | { | ||
102 | ssize_t ret; | ||
103 | |||
104 | ret = rpc_pton(string, len, sa, salen); | ||
105 | if (ret == 0) { | ||
106 | ret = nfs_dns_resolve_name(string, len, sa, salen); | ||
107 | if (ret < 0) | ||
108 | ret = 0; | ||
109 | } | ||
110 | return ret; | ||
111 | } | ||
112 | |||
98 | static struct vfsmount *try_location(struct nfs_clone_mount *mountdata, | 113 | static struct vfsmount *try_location(struct nfs_clone_mount *mountdata, |
99 | char *page, char *page2, | 114 | char *page, char *page2, |
100 | const struct nfs4_fs_location *location) | 115 | const struct nfs4_fs_location *location) |
@@ -121,11 +136,12 @@ static struct vfsmount *try_location(struct nfs_clone_mount *mountdata, | |||
121 | 136 | ||
122 | if (memchr(buf->data, IPV6_SCOPE_DELIMITER, buf->len)) | 137 | if (memchr(buf->data, IPV6_SCOPE_DELIMITER, buf->len)) |
123 | continue; | 138 | continue; |
124 | nfs_parse_ip_address(buf->data, buf->len, | 139 | mountdata->addrlen = nfs_parse_server_name(buf->data, |
125 | mountdata->addr, &mountdata->addrlen); | 140 | buf->len, |
126 | if (mountdata->addr->sa_family == AF_UNSPEC) | 141 | mountdata->addr, mountdata->addrlen); |
142 | if (mountdata->addrlen == 0) | ||
127 | continue; | 143 | continue; |
128 | nfs_set_port(mountdata->addr, NFS_PORT); | 144 | rpc_set_port(mountdata->addr, NFS_PORT); |
129 | 145 | ||
130 | memcpy(page2, buf->data, buf->len); | 146 | memcpy(page2, buf->data, buf->len); |
131 | page2[buf->len] = '\0'; | 147 | page2[buf->len] = '\0'; |
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c index 6917311f201c..ed7c269e2514 100644 --- a/fs/nfs/nfs4proc.c +++ b/fs/nfs/nfs4proc.c | |||
@@ -36,7 +36,6 @@ | |||
36 | */ | 36 | */ |
37 | 37 | ||
38 | #include <linux/mm.h> | 38 | #include <linux/mm.h> |
39 | #include <linux/utsname.h> | ||
40 | #include <linux/delay.h> | 39 | #include <linux/delay.h> |
41 | #include <linux/errno.h> | 40 | #include <linux/errno.h> |
42 | #include <linux/string.h> | 41 | #include <linux/string.h> |
@@ -61,6 +60,8 @@ | |||
61 | #define NFS4_POLL_RETRY_MIN (HZ/10) | 60 | #define NFS4_POLL_RETRY_MIN (HZ/10) |
62 | #define NFS4_POLL_RETRY_MAX (15*HZ) | 61 | #define NFS4_POLL_RETRY_MAX (15*HZ) |
63 | 62 | ||
63 | #define NFS4_MAX_LOOP_ON_RECOVER (10) | ||
64 | |||
64 | struct nfs4_opendata; | 65 | struct nfs4_opendata; |
65 | static int _nfs4_proc_open(struct nfs4_opendata *data); | 66 | static int _nfs4_proc_open(struct nfs4_opendata *data); |
66 | static int nfs4_do_fsinfo(struct nfs_server *, struct nfs_fh *, struct nfs_fsinfo *); | 67 | static int nfs4_do_fsinfo(struct nfs_server *, struct nfs_fh *, struct nfs_fsinfo *); |
@@ -426,17 +427,19 @@ out: | |||
426 | static int nfs4_recover_session(struct nfs4_session *session) | 427 | static int nfs4_recover_session(struct nfs4_session *session) |
427 | { | 428 | { |
428 | struct nfs_client *clp = session->clp; | 429 | struct nfs_client *clp = session->clp; |
430 | unsigned int loop; | ||
429 | int ret; | 431 | int ret; |
430 | 432 | ||
431 | for (;;) { | 433 | for (loop = NFS4_MAX_LOOP_ON_RECOVER; loop != 0; loop--) { |
432 | ret = nfs4_wait_clnt_recover(clp); | 434 | ret = nfs4_wait_clnt_recover(clp); |
433 | if (ret != 0) | 435 | if (ret != 0) |
434 | return ret; | 436 | break; |
435 | if (!test_bit(NFS4CLNT_SESSION_SETUP, &clp->cl_state)) | 437 | if (!test_bit(NFS4CLNT_SESSION_SETUP, &clp->cl_state)) |
436 | break; | 438 | break; |
437 | nfs4_schedule_state_manager(clp); | 439 | nfs4_schedule_state_manager(clp); |
440 | ret = -EIO; | ||
438 | } | 441 | } |
439 | return 0; | 442 | return ret; |
440 | } | 443 | } |
441 | 444 | ||
442 | static int nfs41_setup_sequence(struct nfs4_session *session, | 445 | static int nfs41_setup_sequence(struct nfs4_session *session, |
@@ -1444,18 +1447,20 @@ static int _nfs4_proc_open(struct nfs4_opendata *data) | |||
1444 | static int nfs4_recover_expired_lease(struct nfs_server *server) | 1447 | static int nfs4_recover_expired_lease(struct nfs_server *server) |
1445 | { | 1448 | { |
1446 | struct nfs_client *clp = server->nfs_client; | 1449 | struct nfs_client *clp = server->nfs_client; |
1450 | unsigned int loop; | ||
1447 | int ret; | 1451 | int ret; |
1448 | 1452 | ||
1449 | for (;;) { | 1453 | for (loop = NFS4_MAX_LOOP_ON_RECOVER; loop != 0; loop--) { |
1450 | ret = nfs4_wait_clnt_recover(clp); | 1454 | ret = nfs4_wait_clnt_recover(clp); |
1451 | if (ret != 0) | 1455 | if (ret != 0) |
1452 | return ret; | 1456 | break; |
1453 | if (!test_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state) && | 1457 | if (!test_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state) && |
1454 | !test_bit(NFS4CLNT_CHECK_LEASE,&clp->cl_state)) | 1458 | !test_bit(NFS4CLNT_CHECK_LEASE,&clp->cl_state)) |
1455 | break; | 1459 | break; |
1456 | nfs4_schedule_state_recovery(clp); | 1460 | nfs4_schedule_state_recovery(clp); |
1461 | ret = -EIO; | ||
1457 | } | 1462 | } |
1458 | return 0; | 1463 | return ret; |
1459 | } | 1464 | } |
1460 | 1465 | ||
1461 | /* | 1466 | /* |
@@ -1997,12 +2002,34 @@ static int _nfs4_server_capabilities(struct nfs_server *server, struct nfs_fh *f | |||
1997 | status = nfs4_call_sync(server, &msg, &args, &res, 0); | 2002 | status = nfs4_call_sync(server, &msg, &args, &res, 0); |
1998 | if (status == 0) { | 2003 | if (status == 0) { |
1999 | memcpy(server->attr_bitmask, res.attr_bitmask, sizeof(server->attr_bitmask)); | 2004 | memcpy(server->attr_bitmask, res.attr_bitmask, sizeof(server->attr_bitmask)); |
2005 | server->caps &= ~(NFS_CAP_ACLS|NFS_CAP_HARDLINKS| | ||
2006 | NFS_CAP_SYMLINKS|NFS_CAP_FILEID| | ||
2007 | NFS_CAP_MODE|NFS_CAP_NLINK|NFS_CAP_OWNER| | ||
2008 | NFS_CAP_OWNER_GROUP|NFS_CAP_ATIME| | ||
2009 | NFS_CAP_CTIME|NFS_CAP_MTIME); | ||
2000 | if (res.attr_bitmask[0] & FATTR4_WORD0_ACL) | 2010 | if (res.attr_bitmask[0] & FATTR4_WORD0_ACL) |
2001 | server->caps |= NFS_CAP_ACLS; | 2011 | server->caps |= NFS_CAP_ACLS; |
2002 | if (res.has_links != 0) | 2012 | if (res.has_links != 0) |
2003 | server->caps |= NFS_CAP_HARDLINKS; | 2013 | server->caps |= NFS_CAP_HARDLINKS; |
2004 | if (res.has_symlinks != 0) | 2014 | if (res.has_symlinks != 0) |
2005 | server->caps |= NFS_CAP_SYMLINKS; | 2015 | server->caps |= NFS_CAP_SYMLINKS; |
2016 | if (res.attr_bitmask[0] & FATTR4_WORD0_FILEID) | ||
2017 | server->caps |= NFS_CAP_FILEID; | ||
2018 | if (res.attr_bitmask[1] & FATTR4_WORD1_MODE) | ||
2019 | server->caps |= NFS_CAP_MODE; | ||
2020 | if (res.attr_bitmask[1] & FATTR4_WORD1_NUMLINKS) | ||
2021 | server->caps |= NFS_CAP_NLINK; | ||
2022 | if (res.attr_bitmask[1] & FATTR4_WORD1_OWNER) | ||
2023 | server->caps |= NFS_CAP_OWNER; | ||
2024 | if (res.attr_bitmask[1] & FATTR4_WORD1_OWNER_GROUP) | ||
2025 | server->caps |= NFS_CAP_OWNER_GROUP; | ||
2026 | if (res.attr_bitmask[1] & FATTR4_WORD1_TIME_ACCESS) | ||
2027 | server->caps |= NFS_CAP_ATIME; | ||
2028 | if (res.attr_bitmask[1] & FATTR4_WORD1_TIME_METADATA) | ||
2029 | server->caps |= NFS_CAP_CTIME; | ||
2030 | if (res.attr_bitmask[1] & FATTR4_WORD1_TIME_MODIFY) | ||
2031 | server->caps |= NFS_CAP_MTIME; | ||
2032 | |||
2006 | memcpy(server->cache_consistency_bitmask, res.attr_bitmask, sizeof(server->cache_consistency_bitmask)); | 2033 | memcpy(server->cache_consistency_bitmask, res.attr_bitmask, sizeof(server->cache_consistency_bitmask)); |
2007 | server->cache_consistency_bitmask[0] &= FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE; | 2034 | server->cache_consistency_bitmask[0] &= FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE; |
2008 | server->cache_consistency_bitmask[1] &= FATTR4_WORD1_TIME_METADATA|FATTR4_WORD1_TIME_MODIFY; | 2035 | server->cache_consistency_bitmask[1] &= FATTR4_WORD1_TIME_METADATA|FATTR4_WORD1_TIME_MODIFY; |
diff --git a/fs/nfs/nfs4state.c b/fs/nfs/nfs4state.c index 1434080aefeb..2ef4fecf3984 100644 --- a/fs/nfs/nfs4state.c +++ b/fs/nfs/nfs4state.c | |||
@@ -638,7 +638,7 @@ static void nfs4_fl_release_lock(struct file_lock *fl) | |||
638 | nfs4_put_lock_state(fl->fl_u.nfs4_fl.owner); | 638 | nfs4_put_lock_state(fl->fl_u.nfs4_fl.owner); |
639 | } | 639 | } |
640 | 640 | ||
641 | static struct file_lock_operations nfs4_fl_lock_ops = { | 641 | static const struct file_lock_operations nfs4_fl_lock_ops = { |
642 | .fl_copy_lock = nfs4_fl_copy_lock, | 642 | .fl_copy_lock = nfs4_fl_copy_lock, |
643 | .fl_release_private = nfs4_fl_release_lock, | 643 | .fl_release_private = nfs4_fl_release_lock, |
644 | }; | 644 | }; |
diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c index 617273e7d47f..83ad47cbdd8a 100644 --- a/fs/nfs/nfs4xdr.c +++ b/fs/nfs/nfs4xdr.c | |||
@@ -39,7 +39,6 @@ | |||
39 | #include <linux/time.h> | 39 | #include <linux/time.h> |
40 | #include <linux/mm.h> | 40 | #include <linux/mm.h> |
41 | #include <linux/slab.h> | 41 | #include <linux/slab.h> |
42 | #include <linux/utsname.h> | ||
43 | #include <linux/errno.h> | 42 | #include <linux/errno.h> |
44 | #include <linux/string.h> | 43 | #include <linux/string.h> |
45 | #include <linux/in.h> | 44 | #include <linux/in.h> |
@@ -702,29 +701,12 @@ struct compound_hdr { | |||
702 | u32 minorversion; | 701 | u32 minorversion; |
703 | }; | 702 | }; |
704 | 703 | ||
705 | /* | 704 | static __be32 *reserve_space(struct xdr_stream *xdr, size_t nbytes) |
706 | * START OF "GENERIC" ENCODE ROUTINES. | 705 | { |
707 | * These may look a little ugly since they are imported from a "generic" | 706 | __be32 *p = xdr_reserve_space(xdr, nbytes); |
708 | * set of XDR encode/decode routines which are intended to be shared by | 707 | BUG_ON(!p); |
709 | * all of our NFSv4 implementations (OpenBSD, MacOS X...). | 708 | return p; |
710 | * | 709 | } |
711 | * If the pain of reading these is too great, it should be a straightforward | ||
712 | * task to translate them into Linux-specific versions which are more | ||
713 | * consistent with the style used in NFSv2/v3... | ||
714 | */ | ||
715 | #define WRITE32(n) *p++ = htonl(n) | ||
716 | #define WRITE64(n) do { \ | ||
717 | *p++ = htonl((uint32_t)((n) >> 32)); \ | ||
718 | *p++ = htonl((uint32_t)(n)); \ | ||
719 | } while (0) | ||
720 | #define WRITEMEM(ptr,nbytes) do { \ | ||
721 | p = xdr_encode_opaque_fixed(p, ptr, nbytes); \ | ||
722 | } while (0) | ||
723 | |||
724 | #define RESERVE_SPACE(nbytes) do { \ | ||
725 | p = xdr_reserve_space(xdr, nbytes); \ | ||
726 | BUG_ON(!p); \ | ||
727 | } while (0) | ||
728 | 710 | ||
729 | static void encode_string(struct xdr_stream *xdr, unsigned int len, const char *str) | 711 | static void encode_string(struct xdr_stream *xdr, unsigned int len, const char *str) |
730 | { | 712 | { |
@@ -749,12 +731,11 @@ static void encode_compound_hdr(struct xdr_stream *xdr, | |||
749 | 731 | ||
750 | dprintk("encode_compound: tag=%.*s\n", (int)hdr->taglen, hdr->tag); | 732 | dprintk("encode_compound: tag=%.*s\n", (int)hdr->taglen, hdr->tag); |
751 | BUG_ON(hdr->taglen > NFS4_MAXTAGLEN); | 733 | BUG_ON(hdr->taglen > NFS4_MAXTAGLEN); |
752 | RESERVE_SPACE(12+(XDR_QUADLEN(hdr->taglen)<<2)); | 734 | p = reserve_space(xdr, 4 + hdr->taglen + 8); |
753 | WRITE32(hdr->taglen); | 735 | p = xdr_encode_opaque(p, hdr->tag, hdr->taglen); |
754 | WRITEMEM(hdr->tag, hdr->taglen); | 736 | *p++ = cpu_to_be32(hdr->minorversion); |
755 | WRITE32(hdr->minorversion); | ||
756 | hdr->nops_p = p; | 737 | hdr->nops_p = p; |
757 | WRITE32(hdr->nops); | 738 | *p = cpu_to_be32(hdr->nops); |
758 | } | 739 | } |
759 | 740 | ||
760 | static void encode_nops(struct compound_hdr *hdr) | 741 | static void encode_nops(struct compound_hdr *hdr) |
@@ -829,55 +810,53 @@ static void encode_attrs(struct xdr_stream *xdr, const struct iattr *iap, const | |||
829 | len += 16; | 810 | len += 16; |
830 | else if (iap->ia_valid & ATTR_MTIME) | 811 | else if (iap->ia_valid & ATTR_MTIME) |
831 | len += 4; | 812 | len += 4; |
832 | RESERVE_SPACE(len); | 813 | p = reserve_space(xdr, len); |
833 | 814 | ||
834 | /* | 815 | /* |
835 | * We write the bitmap length now, but leave the bitmap and the attribute | 816 | * We write the bitmap length now, but leave the bitmap and the attribute |
836 | * buffer length to be backfilled at the end of this routine. | 817 | * buffer length to be backfilled at the end of this routine. |
837 | */ | 818 | */ |
838 | WRITE32(2); | 819 | *p++ = cpu_to_be32(2); |
839 | q = p; | 820 | q = p; |
840 | p += 3; | 821 | p += 3; |
841 | 822 | ||
842 | if (iap->ia_valid & ATTR_SIZE) { | 823 | if (iap->ia_valid & ATTR_SIZE) { |
843 | bmval0 |= FATTR4_WORD0_SIZE; | 824 | bmval0 |= FATTR4_WORD0_SIZE; |
844 | WRITE64(iap->ia_size); | 825 | p = xdr_encode_hyper(p, iap->ia_size); |
845 | } | 826 | } |
846 | if (iap->ia_valid & ATTR_MODE) { | 827 | if (iap->ia_valid & ATTR_MODE) { |
847 | bmval1 |= FATTR4_WORD1_MODE; | 828 | bmval1 |= FATTR4_WORD1_MODE; |
848 | WRITE32(iap->ia_mode & S_IALLUGO); | 829 | *p++ = cpu_to_be32(iap->ia_mode & S_IALLUGO); |
849 | } | 830 | } |
850 | if (iap->ia_valid & ATTR_UID) { | 831 | if (iap->ia_valid & ATTR_UID) { |
851 | bmval1 |= FATTR4_WORD1_OWNER; | 832 | bmval1 |= FATTR4_WORD1_OWNER; |
852 | WRITE32(owner_namelen); | 833 | p = xdr_encode_opaque(p, owner_name, owner_namelen); |
853 | WRITEMEM(owner_name, owner_namelen); | ||
854 | } | 834 | } |
855 | if (iap->ia_valid & ATTR_GID) { | 835 | if (iap->ia_valid & ATTR_GID) { |
856 | bmval1 |= FATTR4_WORD1_OWNER_GROUP; | 836 | bmval1 |= FATTR4_WORD1_OWNER_GROUP; |
857 | WRITE32(owner_grouplen); | 837 | p = xdr_encode_opaque(p, owner_group, owner_grouplen); |
858 | WRITEMEM(owner_group, owner_grouplen); | ||
859 | } | 838 | } |
860 | if (iap->ia_valid & ATTR_ATIME_SET) { | 839 | if (iap->ia_valid & ATTR_ATIME_SET) { |
861 | bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET; | 840 | bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET; |
862 | WRITE32(NFS4_SET_TO_CLIENT_TIME); | 841 | *p++ = cpu_to_be32(NFS4_SET_TO_CLIENT_TIME); |
863 | WRITE32(0); | 842 | *p++ = cpu_to_be32(0); |
864 | WRITE32(iap->ia_mtime.tv_sec); | 843 | *p++ = cpu_to_be32(iap->ia_mtime.tv_sec); |
865 | WRITE32(iap->ia_mtime.tv_nsec); | 844 | *p++ = cpu_to_be32(iap->ia_mtime.tv_nsec); |
866 | } | 845 | } |
867 | else if (iap->ia_valid & ATTR_ATIME) { | 846 | else if (iap->ia_valid & ATTR_ATIME) { |
868 | bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET; | 847 | bmval1 |= FATTR4_WORD1_TIME_ACCESS_SET; |
869 | WRITE32(NFS4_SET_TO_SERVER_TIME); | 848 | *p++ = cpu_to_be32(NFS4_SET_TO_SERVER_TIME); |
870 | } | 849 | } |
871 | if (iap->ia_valid & ATTR_MTIME_SET) { | 850 | if (iap->ia_valid & ATTR_MTIME_SET) { |
872 | bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET; | 851 | bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET; |
873 | WRITE32(NFS4_SET_TO_CLIENT_TIME); | 852 | *p++ = cpu_to_be32(NFS4_SET_TO_CLIENT_TIME); |
874 | WRITE32(0); | 853 | *p++ = cpu_to_be32(0); |
875 | WRITE32(iap->ia_mtime.tv_sec); | 854 | *p++ = cpu_to_be32(iap->ia_mtime.tv_sec); |
876 | WRITE32(iap->ia_mtime.tv_nsec); | 855 | *p++ = cpu_to_be32(iap->ia_mtime.tv_nsec); |
877 | } | 856 | } |
878 | else if (iap->ia_valid & ATTR_MTIME) { | 857 | else if (iap->ia_valid & ATTR_MTIME) { |
879 | bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET; | 858 | bmval1 |= FATTR4_WORD1_TIME_MODIFY_SET; |
880 | WRITE32(NFS4_SET_TO_SERVER_TIME); | 859 | *p++ = cpu_to_be32(NFS4_SET_TO_SERVER_TIME); |
881 | } | 860 | } |
882 | 861 | ||
883 | /* | 862 | /* |
@@ -891,7 +870,7 @@ static void encode_attrs(struct xdr_stream *xdr, const struct iattr *iap, const | |||
891 | len = (char *)p - (char *)q - 12; | 870 | len = (char *)p - (char *)q - 12; |
892 | *q++ = htonl(bmval0); | 871 | *q++ = htonl(bmval0); |
893 | *q++ = htonl(bmval1); | 872 | *q++ = htonl(bmval1); |
894 | *q++ = htonl(len); | 873 | *q = htonl(len); |
895 | 874 | ||
896 | /* out: */ | 875 | /* out: */ |
897 | } | 876 | } |
@@ -900,9 +879,9 @@ static void encode_access(struct xdr_stream *xdr, u32 access, struct compound_hd | |||
900 | { | 879 | { |
901 | __be32 *p; | 880 | __be32 *p; |
902 | 881 | ||
903 | RESERVE_SPACE(8); | 882 | p = reserve_space(xdr, 8); |
904 | WRITE32(OP_ACCESS); | 883 | *p++ = cpu_to_be32(OP_ACCESS); |
905 | WRITE32(access); | 884 | *p = cpu_to_be32(access); |
906 | hdr->nops++; | 885 | hdr->nops++; |
907 | hdr->replen += decode_access_maxsz; | 886 | hdr->replen += decode_access_maxsz; |
908 | } | 887 | } |
@@ -911,10 +890,10 @@ static void encode_close(struct xdr_stream *xdr, const struct nfs_closeargs *arg | |||
911 | { | 890 | { |
912 | __be32 *p; | 891 | __be32 *p; |
913 | 892 | ||
914 | RESERVE_SPACE(8+NFS4_STATEID_SIZE); | 893 | p = reserve_space(xdr, 8+NFS4_STATEID_SIZE); |
915 | WRITE32(OP_CLOSE); | 894 | *p++ = cpu_to_be32(OP_CLOSE); |
916 | WRITE32(arg->seqid->sequence->counter); | 895 | *p++ = cpu_to_be32(arg->seqid->sequence->counter); |
917 | WRITEMEM(arg->stateid->data, NFS4_STATEID_SIZE); | 896 | xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE); |
918 | hdr->nops++; | 897 | hdr->nops++; |
919 | hdr->replen += decode_close_maxsz; | 898 | hdr->replen += decode_close_maxsz; |
920 | } | 899 | } |
@@ -923,10 +902,10 @@ static void encode_commit(struct xdr_stream *xdr, const struct nfs_writeargs *ar | |||
923 | { | 902 | { |
924 | __be32 *p; | 903 | __be32 *p; |
925 | 904 | ||
926 | RESERVE_SPACE(16); | 905 | p = reserve_space(xdr, 16); |
927 | WRITE32(OP_COMMIT); | 906 | *p++ = cpu_to_be32(OP_COMMIT); |
928 | WRITE64(args->offset); | 907 | p = xdr_encode_hyper(p, args->offset); |
929 | WRITE32(args->count); | 908 | *p = cpu_to_be32(args->count); |
930 | hdr->nops++; | 909 | hdr->nops++; |
931 | hdr->replen += decode_commit_maxsz; | 910 | hdr->replen += decode_commit_maxsz; |
932 | } | 911 | } |
@@ -935,30 +914,28 @@ static void encode_create(struct xdr_stream *xdr, const struct nfs4_create_arg * | |||
935 | { | 914 | { |
936 | __be32 *p; | 915 | __be32 *p; |
937 | 916 | ||
938 | RESERVE_SPACE(8); | 917 | p = reserve_space(xdr, 8); |
939 | WRITE32(OP_CREATE); | 918 | *p++ = cpu_to_be32(OP_CREATE); |
940 | WRITE32(create->ftype); | 919 | *p = cpu_to_be32(create->ftype); |
941 | 920 | ||
942 | switch (create->ftype) { | 921 | switch (create->ftype) { |
943 | case NF4LNK: | 922 | case NF4LNK: |
944 | RESERVE_SPACE(4); | 923 | p = reserve_space(xdr, 4); |
945 | WRITE32(create->u.symlink.len); | 924 | *p = cpu_to_be32(create->u.symlink.len); |
946 | xdr_write_pages(xdr, create->u.symlink.pages, 0, create->u.symlink.len); | 925 | xdr_write_pages(xdr, create->u.symlink.pages, 0, create->u.symlink.len); |
947 | break; | 926 | break; |
948 | 927 | ||
949 | case NF4BLK: case NF4CHR: | 928 | case NF4BLK: case NF4CHR: |
950 | RESERVE_SPACE(8); | 929 | p = reserve_space(xdr, 8); |
951 | WRITE32(create->u.device.specdata1); | 930 | *p++ = cpu_to_be32(create->u.device.specdata1); |
952 | WRITE32(create->u.device.specdata2); | 931 | *p = cpu_to_be32(create->u.device.specdata2); |
953 | break; | 932 | break; |
954 | 933 | ||
955 | default: | 934 | default: |
956 | break; | 935 | break; |
957 | } | 936 | } |
958 | 937 | ||
959 | RESERVE_SPACE(4 + create->name->len); | 938 | encode_string(xdr, create->name->len, create->name->name); |
960 | WRITE32(create->name->len); | ||
961 | WRITEMEM(create->name->name, create->name->len); | ||
962 | hdr->nops++; | 939 | hdr->nops++; |
963 | hdr->replen += decode_create_maxsz; | 940 | hdr->replen += decode_create_maxsz; |
964 | 941 | ||
@@ -969,10 +946,10 @@ static void encode_getattr_one(struct xdr_stream *xdr, uint32_t bitmap, struct c | |||
969 | { | 946 | { |
970 | __be32 *p; | 947 | __be32 *p; |
971 | 948 | ||
972 | RESERVE_SPACE(12); | 949 | p = reserve_space(xdr, 12); |
973 | WRITE32(OP_GETATTR); | 950 | *p++ = cpu_to_be32(OP_GETATTR); |
974 | WRITE32(1); | 951 | *p++ = cpu_to_be32(1); |
975 | WRITE32(bitmap); | 952 | *p = cpu_to_be32(bitmap); |
976 | hdr->nops++; | 953 | hdr->nops++; |
977 | hdr->replen += decode_getattr_maxsz; | 954 | hdr->replen += decode_getattr_maxsz; |
978 | } | 955 | } |
@@ -981,11 +958,11 @@ static void encode_getattr_two(struct xdr_stream *xdr, uint32_t bm0, uint32_t bm | |||
981 | { | 958 | { |
982 | __be32 *p; | 959 | __be32 *p; |
983 | 960 | ||
984 | RESERVE_SPACE(16); | 961 | p = reserve_space(xdr, 16); |
985 | WRITE32(OP_GETATTR); | 962 | *p++ = cpu_to_be32(OP_GETATTR); |
986 | WRITE32(2); | 963 | *p++ = cpu_to_be32(2); |
987 | WRITE32(bm0); | 964 | *p++ = cpu_to_be32(bm0); |
988 | WRITE32(bm1); | 965 | *p = cpu_to_be32(bm1); |
989 | hdr->nops++; | 966 | hdr->nops++; |
990 | hdr->replen += decode_getattr_maxsz; | 967 | hdr->replen += decode_getattr_maxsz; |
991 | } | 968 | } |
@@ -1012,8 +989,8 @@ static void encode_getfh(struct xdr_stream *xdr, struct compound_hdr *hdr) | |||
1012 | { | 989 | { |
1013 | __be32 *p; | 990 | __be32 *p; |
1014 | 991 | ||
1015 | RESERVE_SPACE(4); | 992 | p = reserve_space(xdr, 4); |
1016 | WRITE32(OP_GETFH); | 993 | *p = cpu_to_be32(OP_GETFH); |
1017 | hdr->nops++; | 994 | hdr->nops++; |
1018 | hdr->replen += decode_getfh_maxsz; | 995 | hdr->replen += decode_getfh_maxsz; |
1019 | } | 996 | } |
@@ -1022,10 +999,9 @@ static void encode_link(struct xdr_stream *xdr, const struct qstr *name, struct | |||
1022 | { | 999 | { |
1023 | __be32 *p; | 1000 | __be32 *p; |
1024 | 1001 | ||
1025 | RESERVE_SPACE(8 + name->len); | 1002 | p = reserve_space(xdr, 8 + name->len); |
1026 | WRITE32(OP_LINK); | 1003 | *p++ = cpu_to_be32(OP_LINK); |
1027 | WRITE32(name->len); | 1004 | xdr_encode_opaque(p, name->name, name->len); |
1028 | WRITEMEM(name->name, name->len); | ||
1029 | hdr->nops++; | 1005 | hdr->nops++; |
1030 | hdr->replen += decode_link_maxsz; | 1006 | hdr->replen += decode_link_maxsz; |
1031 | } | 1007 | } |
@@ -1052,27 +1028,27 @@ static void encode_lock(struct xdr_stream *xdr, const struct nfs_lock_args *args | |||
1052 | { | 1028 | { |
1053 | __be32 *p; | 1029 | __be32 *p; |
1054 | 1030 | ||
1055 | RESERVE_SPACE(32); | 1031 | p = reserve_space(xdr, 32); |
1056 | WRITE32(OP_LOCK); | 1032 | *p++ = cpu_to_be32(OP_LOCK); |
1057 | WRITE32(nfs4_lock_type(args->fl, args->block)); | 1033 | *p++ = cpu_to_be32(nfs4_lock_type(args->fl, args->block)); |
1058 | WRITE32(args->reclaim); | 1034 | *p++ = cpu_to_be32(args->reclaim); |
1059 | WRITE64(args->fl->fl_start); | 1035 | p = xdr_encode_hyper(p, args->fl->fl_start); |
1060 | WRITE64(nfs4_lock_length(args->fl)); | 1036 | p = xdr_encode_hyper(p, nfs4_lock_length(args->fl)); |
1061 | WRITE32(args->new_lock_owner); | 1037 | *p = cpu_to_be32(args->new_lock_owner); |
1062 | if (args->new_lock_owner){ | 1038 | if (args->new_lock_owner){ |
1063 | RESERVE_SPACE(4+NFS4_STATEID_SIZE+32); | 1039 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+32); |
1064 | WRITE32(args->open_seqid->sequence->counter); | 1040 | *p++ = cpu_to_be32(args->open_seqid->sequence->counter); |
1065 | WRITEMEM(args->open_stateid->data, NFS4_STATEID_SIZE); | 1041 | p = xdr_encode_opaque_fixed(p, args->open_stateid->data, NFS4_STATEID_SIZE); |
1066 | WRITE32(args->lock_seqid->sequence->counter); | 1042 | *p++ = cpu_to_be32(args->lock_seqid->sequence->counter); |
1067 | WRITE64(args->lock_owner.clientid); | 1043 | p = xdr_encode_hyper(p, args->lock_owner.clientid); |
1068 | WRITE32(16); | 1044 | *p++ = cpu_to_be32(16); |
1069 | WRITEMEM("lock id:", 8); | 1045 | p = xdr_encode_opaque_fixed(p, "lock id:", 8); |
1070 | WRITE64(args->lock_owner.id); | 1046 | xdr_encode_hyper(p, args->lock_owner.id); |
1071 | } | 1047 | } |
1072 | else { | 1048 | else { |
1073 | RESERVE_SPACE(NFS4_STATEID_SIZE+4); | 1049 | p = reserve_space(xdr, NFS4_STATEID_SIZE+4); |
1074 | WRITEMEM(args->lock_stateid->data, NFS4_STATEID_SIZE); | 1050 | p = xdr_encode_opaque_fixed(p, args->lock_stateid->data, NFS4_STATEID_SIZE); |
1075 | WRITE32(args->lock_seqid->sequence->counter); | 1051 | *p = cpu_to_be32(args->lock_seqid->sequence->counter); |
1076 | } | 1052 | } |
1077 | hdr->nops++; | 1053 | hdr->nops++; |
1078 | hdr->replen += decode_lock_maxsz; | 1054 | hdr->replen += decode_lock_maxsz; |
@@ -1082,15 +1058,15 @@ static void encode_lockt(struct xdr_stream *xdr, const struct nfs_lockt_args *ar | |||
1082 | { | 1058 | { |
1083 | __be32 *p; | 1059 | __be32 *p; |
1084 | 1060 | ||
1085 | RESERVE_SPACE(52); | 1061 | p = reserve_space(xdr, 52); |
1086 | WRITE32(OP_LOCKT); | 1062 | *p++ = cpu_to_be32(OP_LOCKT); |
1087 | WRITE32(nfs4_lock_type(args->fl, 0)); | 1063 | *p++ = cpu_to_be32(nfs4_lock_type(args->fl, 0)); |
1088 | WRITE64(args->fl->fl_start); | 1064 | p = xdr_encode_hyper(p, args->fl->fl_start); |
1089 | WRITE64(nfs4_lock_length(args->fl)); | 1065 | p = xdr_encode_hyper(p, nfs4_lock_length(args->fl)); |
1090 | WRITE64(args->lock_owner.clientid); | 1066 | p = xdr_encode_hyper(p, args->lock_owner.clientid); |
1091 | WRITE32(16); | 1067 | *p++ = cpu_to_be32(16); |
1092 | WRITEMEM("lock id:", 8); | 1068 | p = xdr_encode_opaque_fixed(p, "lock id:", 8); |
1093 | WRITE64(args->lock_owner.id); | 1069 | xdr_encode_hyper(p, args->lock_owner.id); |
1094 | hdr->nops++; | 1070 | hdr->nops++; |
1095 | hdr->replen += decode_lockt_maxsz; | 1071 | hdr->replen += decode_lockt_maxsz; |
1096 | } | 1072 | } |
@@ -1099,13 +1075,13 @@ static void encode_locku(struct xdr_stream *xdr, const struct nfs_locku_args *ar | |||
1099 | { | 1075 | { |
1100 | __be32 *p; | 1076 | __be32 *p; |
1101 | 1077 | ||
1102 | RESERVE_SPACE(12+NFS4_STATEID_SIZE+16); | 1078 | p = reserve_space(xdr, 12+NFS4_STATEID_SIZE+16); |
1103 | WRITE32(OP_LOCKU); | 1079 | *p++ = cpu_to_be32(OP_LOCKU); |
1104 | WRITE32(nfs4_lock_type(args->fl, 0)); | 1080 | *p++ = cpu_to_be32(nfs4_lock_type(args->fl, 0)); |
1105 | WRITE32(args->seqid->sequence->counter); | 1081 | *p++ = cpu_to_be32(args->seqid->sequence->counter); |
1106 | WRITEMEM(args->stateid->data, NFS4_STATEID_SIZE); | 1082 | p = xdr_encode_opaque_fixed(p, args->stateid->data, NFS4_STATEID_SIZE); |
1107 | WRITE64(args->fl->fl_start); | 1083 | p = xdr_encode_hyper(p, args->fl->fl_start); |
1108 | WRITE64(nfs4_lock_length(args->fl)); | 1084 | xdr_encode_hyper(p, nfs4_lock_length(args->fl)); |
1109 | hdr->nops++; | 1085 | hdr->nops++; |
1110 | hdr->replen += decode_locku_maxsz; | 1086 | hdr->replen += decode_locku_maxsz; |
1111 | } | 1087 | } |
@@ -1115,10 +1091,9 @@ static void encode_lookup(struct xdr_stream *xdr, const struct qstr *name, struc | |||
1115 | int len = name->len; | 1091 | int len = name->len; |
1116 | __be32 *p; | 1092 | __be32 *p; |
1117 | 1093 | ||
1118 | RESERVE_SPACE(8 + len); | 1094 | p = reserve_space(xdr, 8 + len); |
1119 | WRITE32(OP_LOOKUP); | 1095 | *p++ = cpu_to_be32(OP_LOOKUP); |
1120 | WRITE32(len); | 1096 | xdr_encode_opaque(p, name->name, len); |
1121 | WRITEMEM(name->name, len); | ||
1122 | hdr->nops++; | 1097 | hdr->nops++; |
1123 | hdr->replen += decode_lookup_maxsz; | 1098 | hdr->replen += decode_lookup_maxsz; |
1124 | } | 1099 | } |
@@ -1127,21 +1102,21 @@ static void encode_share_access(struct xdr_stream *xdr, fmode_t fmode) | |||
1127 | { | 1102 | { |
1128 | __be32 *p; | 1103 | __be32 *p; |
1129 | 1104 | ||
1130 | RESERVE_SPACE(8); | 1105 | p = reserve_space(xdr, 8); |
1131 | switch (fmode & (FMODE_READ|FMODE_WRITE)) { | 1106 | switch (fmode & (FMODE_READ|FMODE_WRITE)) { |
1132 | case FMODE_READ: | 1107 | case FMODE_READ: |
1133 | WRITE32(NFS4_SHARE_ACCESS_READ); | 1108 | *p++ = cpu_to_be32(NFS4_SHARE_ACCESS_READ); |
1134 | break; | 1109 | break; |
1135 | case FMODE_WRITE: | 1110 | case FMODE_WRITE: |
1136 | WRITE32(NFS4_SHARE_ACCESS_WRITE); | 1111 | *p++ = cpu_to_be32(NFS4_SHARE_ACCESS_WRITE); |
1137 | break; | 1112 | break; |
1138 | case FMODE_READ|FMODE_WRITE: | 1113 | case FMODE_READ|FMODE_WRITE: |
1139 | WRITE32(NFS4_SHARE_ACCESS_BOTH); | 1114 | *p++ = cpu_to_be32(NFS4_SHARE_ACCESS_BOTH); |
1140 | break; | 1115 | break; |
1141 | default: | 1116 | default: |
1142 | WRITE32(0); | 1117 | *p++ = cpu_to_be32(0); |
1143 | } | 1118 | } |
1144 | WRITE32(0); /* for linux, share_deny = 0 always */ | 1119 | *p = cpu_to_be32(0); /* for linux, share_deny = 0 always */ |
1145 | } | 1120 | } |
1146 | 1121 | ||
1147 | static inline void encode_openhdr(struct xdr_stream *xdr, const struct nfs_openargs *arg) | 1122 | static inline void encode_openhdr(struct xdr_stream *xdr, const struct nfs_openargs *arg) |
@@ -1151,29 +1126,29 @@ static inline void encode_openhdr(struct xdr_stream *xdr, const struct nfs_opena | |||
1151 | * opcode 4, seqid 4, share_access 4, share_deny 4, clientid 8, ownerlen 4, | 1126 | * opcode 4, seqid 4, share_access 4, share_deny 4, clientid 8, ownerlen 4, |
1152 | * owner 4 = 32 | 1127 | * owner 4 = 32 |
1153 | */ | 1128 | */ |
1154 | RESERVE_SPACE(8); | 1129 | p = reserve_space(xdr, 8); |
1155 | WRITE32(OP_OPEN); | 1130 | *p++ = cpu_to_be32(OP_OPEN); |
1156 | WRITE32(arg->seqid->sequence->counter); | 1131 | *p = cpu_to_be32(arg->seqid->sequence->counter); |
1157 | encode_share_access(xdr, arg->fmode); | 1132 | encode_share_access(xdr, arg->fmode); |
1158 | RESERVE_SPACE(28); | 1133 | p = reserve_space(xdr, 28); |
1159 | WRITE64(arg->clientid); | 1134 | p = xdr_encode_hyper(p, arg->clientid); |
1160 | WRITE32(16); | 1135 | *p++ = cpu_to_be32(16); |
1161 | WRITEMEM("open id:", 8); | 1136 | p = xdr_encode_opaque_fixed(p, "open id:", 8); |
1162 | WRITE64(arg->id); | 1137 | xdr_encode_hyper(p, arg->id); |
1163 | } | 1138 | } |
1164 | 1139 | ||
1165 | static inline void encode_createmode(struct xdr_stream *xdr, const struct nfs_openargs *arg) | 1140 | static inline void encode_createmode(struct xdr_stream *xdr, const struct nfs_openargs *arg) |
1166 | { | 1141 | { |
1167 | __be32 *p; | 1142 | __be32 *p; |
1168 | 1143 | ||
1169 | RESERVE_SPACE(4); | 1144 | p = reserve_space(xdr, 4); |
1170 | switch(arg->open_flags & O_EXCL) { | 1145 | switch(arg->open_flags & O_EXCL) { |
1171 | case 0: | 1146 | case 0: |
1172 | WRITE32(NFS4_CREATE_UNCHECKED); | 1147 | *p = cpu_to_be32(NFS4_CREATE_UNCHECKED); |
1173 | encode_attrs(xdr, arg->u.attrs, arg->server); | 1148 | encode_attrs(xdr, arg->u.attrs, arg->server); |
1174 | break; | 1149 | break; |
1175 | default: | 1150 | default: |
1176 | WRITE32(NFS4_CREATE_EXCLUSIVE); | 1151 | *p = cpu_to_be32(NFS4_CREATE_EXCLUSIVE); |
1177 | encode_nfs4_verifier(xdr, &arg->u.verifier); | 1152 | encode_nfs4_verifier(xdr, &arg->u.verifier); |
1178 | } | 1153 | } |
1179 | } | 1154 | } |
@@ -1182,14 +1157,14 @@ static void encode_opentype(struct xdr_stream *xdr, const struct nfs_openargs *a | |||
1182 | { | 1157 | { |
1183 | __be32 *p; | 1158 | __be32 *p; |
1184 | 1159 | ||
1185 | RESERVE_SPACE(4); | 1160 | p = reserve_space(xdr, 4); |
1186 | switch (arg->open_flags & O_CREAT) { | 1161 | switch (arg->open_flags & O_CREAT) { |
1187 | case 0: | 1162 | case 0: |
1188 | WRITE32(NFS4_OPEN_NOCREATE); | 1163 | *p = cpu_to_be32(NFS4_OPEN_NOCREATE); |
1189 | break; | 1164 | break; |
1190 | default: | 1165 | default: |
1191 | BUG_ON(arg->claim != NFS4_OPEN_CLAIM_NULL); | 1166 | BUG_ON(arg->claim != NFS4_OPEN_CLAIM_NULL); |
1192 | WRITE32(NFS4_OPEN_CREATE); | 1167 | *p = cpu_to_be32(NFS4_OPEN_CREATE); |
1193 | encode_createmode(xdr, arg); | 1168 | encode_createmode(xdr, arg); |
1194 | } | 1169 | } |
1195 | } | 1170 | } |
@@ -1198,16 +1173,16 @@ static inline void encode_delegation_type(struct xdr_stream *xdr, fmode_t delega | |||
1198 | { | 1173 | { |
1199 | __be32 *p; | 1174 | __be32 *p; |
1200 | 1175 | ||
1201 | RESERVE_SPACE(4); | 1176 | p = reserve_space(xdr, 4); |
1202 | switch (delegation_type) { | 1177 | switch (delegation_type) { |
1203 | case 0: | 1178 | case 0: |
1204 | WRITE32(NFS4_OPEN_DELEGATE_NONE); | 1179 | *p = cpu_to_be32(NFS4_OPEN_DELEGATE_NONE); |
1205 | break; | 1180 | break; |
1206 | case FMODE_READ: | 1181 | case FMODE_READ: |
1207 | WRITE32(NFS4_OPEN_DELEGATE_READ); | 1182 | *p = cpu_to_be32(NFS4_OPEN_DELEGATE_READ); |
1208 | break; | 1183 | break; |
1209 | case FMODE_WRITE|FMODE_READ: | 1184 | case FMODE_WRITE|FMODE_READ: |
1210 | WRITE32(NFS4_OPEN_DELEGATE_WRITE); | 1185 | *p = cpu_to_be32(NFS4_OPEN_DELEGATE_WRITE); |
1211 | break; | 1186 | break; |
1212 | default: | 1187 | default: |
1213 | BUG(); | 1188 | BUG(); |
@@ -1218,8 +1193,8 @@ static inline void encode_claim_null(struct xdr_stream *xdr, const struct qstr * | |||
1218 | { | 1193 | { |
1219 | __be32 *p; | 1194 | __be32 *p; |
1220 | 1195 | ||
1221 | RESERVE_SPACE(4); | 1196 | p = reserve_space(xdr, 4); |
1222 | WRITE32(NFS4_OPEN_CLAIM_NULL); | 1197 | *p = cpu_to_be32(NFS4_OPEN_CLAIM_NULL); |
1223 | encode_string(xdr, name->len, name->name); | 1198 | encode_string(xdr, name->len, name->name); |
1224 | } | 1199 | } |
1225 | 1200 | ||
@@ -1227,8 +1202,8 @@ static inline void encode_claim_previous(struct xdr_stream *xdr, fmode_t type) | |||
1227 | { | 1202 | { |
1228 | __be32 *p; | 1203 | __be32 *p; |
1229 | 1204 | ||
1230 | RESERVE_SPACE(4); | 1205 | p = reserve_space(xdr, 4); |
1231 | WRITE32(NFS4_OPEN_CLAIM_PREVIOUS); | 1206 | *p = cpu_to_be32(NFS4_OPEN_CLAIM_PREVIOUS); |
1232 | encode_delegation_type(xdr, type); | 1207 | encode_delegation_type(xdr, type); |
1233 | } | 1208 | } |
1234 | 1209 | ||
@@ -1236,9 +1211,9 @@ static inline void encode_claim_delegate_cur(struct xdr_stream *xdr, const struc | |||
1236 | { | 1211 | { |
1237 | __be32 *p; | 1212 | __be32 *p; |
1238 | 1213 | ||
1239 | RESERVE_SPACE(4+NFS4_STATEID_SIZE); | 1214 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE); |
1240 | WRITE32(NFS4_OPEN_CLAIM_DELEGATE_CUR); | 1215 | *p++ = cpu_to_be32(NFS4_OPEN_CLAIM_DELEGATE_CUR); |
1241 | WRITEMEM(stateid->data, NFS4_STATEID_SIZE); | 1216 | xdr_encode_opaque_fixed(p, stateid->data, NFS4_STATEID_SIZE); |
1242 | encode_string(xdr, name->len, name->name); | 1217 | encode_string(xdr, name->len, name->name); |
1243 | } | 1218 | } |
1244 | 1219 | ||
@@ -1267,10 +1242,10 @@ static void encode_open_confirm(struct xdr_stream *xdr, const struct nfs_open_co | |||
1267 | { | 1242 | { |
1268 | __be32 *p; | 1243 | __be32 *p; |
1269 | 1244 | ||
1270 | RESERVE_SPACE(4+NFS4_STATEID_SIZE+4); | 1245 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+4); |
1271 | WRITE32(OP_OPEN_CONFIRM); | 1246 | *p++ = cpu_to_be32(OP_OPEN_CONFIRM); |
1272 | WRITEMEM(arg->stateid->data, NFS4_STATEID_SIZE); | 1247 | p = xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE); |
1273 | WRITE32(arg->seqid->sequence->counter); | 1248 | *p = cpu_to_be32(arg->seqid->sequence->counter); |
1274 | hdr->nops++; | 1249 | hdr->nops++; |
1275 | hdr->replen += decode_open_confirm_maxsz; | 1250 | hdr->replen += decode_open_confirm_maxsz; |
1276 | } | 1251 | } |
@@ -1279,10 +1254,10 @@ static void encode_open_downgrade(struct xdr_stream *xdr, const struct nfs_close | |||
1279 | { | 1254 | { |
1280 | __be32 *p; | 1255 | __be32 *p; |
1281 | 1256 | ||
1282 | RESERVE_SPACE(4+NFS4_STATEID_SIZE+4); | 1257 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE+4); |
1283 | WRITE32(OP_OPEN_DOWNGRADE); | 1258 | *p++ = cpu_to_be32(OP_OPEN_DOWNGRADE); |
1284 | WRITEMEM(arg->stateid->data, NFS4_STATEID_SIZE); | 1259 | p = xdr_encode_opaque_fixed(p, arg->stateid->data, NFS4_STATEID_SIZE); |
1285 | WRITE32(arg->seqid->sequence->counter); | 1260 | *p = cpu_to_be32(arg->seqid->sequence->counter); |
1286 | encode_share_access(xdr, arg->fmode); | 1261 | encode_share_access(xdr, arg->fmode); |
1287 | hdr->nops++; | 1262 | hdr->nops++; |
1288 | hdr->replen += decode_open_downgrade_maxsz; | 1263 | hdr->replen += decode_open_downgrade_maxsz; |
@@ -1294,10 +1269,9 @@ encode_putfh(struct xdr_stream *xdr, const struct nfs_fh *fh, struct compound_hd | |||
1294 | int len = fh->size; | 1269 | int len = fh->size; |
1295 | __be32 *p; | 1270 | __be32 *p; |
1296 | 1271 | ||
1297 | RESERVE_SPACE(8 + len); | 1272 | p = reserve_space(xdr, 8 + len); |
1298 | WRITE32(OP_PUTFH); | 1273 | *p++ = cpu_to_be32(OP_PUTFH); |
1299 | WRITE32(len); | 1274 | xdr_encode_opaque(p, fh->data, len); |
1300 | WRITEMEM(fh->data, len); | ||
1301 | hdr->nops++; | 1275 | hdr->nops++; |
1302 | hdr->replen += decode_putfh_maxsz; | 1276 | hdr->replen += decode_putfh_maxsz; |
1303 | } | 1277 | } |
@@ -1306,8 +1280,8 @@ static void encode_putrootfh(struct xdr_stream *xdr, struct compound_hdr *hdr) | |||
1306 | { | 1280 | { |
1307 | __be32 *p; | 1281 | __be32 *p; |
1308 | 1282 | ||
1309 | RESERVE_SPACE(4); | 1283 | p = reserve_space(xdr, 4); |
1310 | WRITE32(OP_PUTROOTFH); | 1284 | *p = cpu_to_be32(OP_PUTROOTFH); |
1311 | hdr->nops++; | 1285 | hdr->nops++; |
1312 | hdr->replen += decode_putrootfh_maxsz; | 1286 | hdr->replen += decode_putrootfh_maxsz; |
1313 | } | 1287 | } |
@@ -1317,26 +1291,26 @@ static void encode_stateid(struct xdr_stream *xdr, const struct nfs_open_context | |||
1317 | nfs4_stateid stateid; | 1291 | nfs4_stateid stateid; |
1318 | __be32 *p; | 1292 | __be32 *p; |
1319 | 1293 | ||
1320 | RESERVE_SPACE(NFS4_STATEID_SIZE); | 1294 | p = reserve_space(xdr, NFS4_STATEID_SIZE); |
1321 | if (ctx->state != NULL) { | 1295 | if (ctx->state != NULL) { |
1322 | nfs4_copy_stateid(&stateid, ctx->state, ctx->lockowner); | 1296 | nfs4_copy_stateid(&stateid, ctx->state, ctx->lockowner); |
1323 | WRITEMEM(stateid.data, NFS4_STATEID_SIZE); | 1297 | xdr_encode_opaque_fixed(p, stateid.data, NFS4_STATEID_SIZE); |
1324 | } else | 1298 | } else |
1325 | WRITEMEM(zero_stateid.data, NFS4_STATEID_SIZE); | 1299 | xdr_encode_opaque_fixed(p, zero_stateid.data, NFS4_STATEID_SIZE); |
1326 | } | 1300 | } |
1327 | 1301 | ||
1328 | static void encode_read(struct xdr_stream *xdr, const struct nfs_readargs *args, struct compound_hdr *hdr) | 1302 | static void encode_read(struct xdr_stream *xdr, const struct nfs_readargs *args, struct compound_hdr *hdr) |
1329 | { | 1303 | { |
1330 | __be32 *p; | 1304 | __be32 *p; |
1331 | 1305 | ||
1332 | RESERVE_SPACE(4); | 1306 | p = reserve_space(xdr, 4); |
1333 | WRITE32(OP_READ); | 1307 | *p = cpu_to_be32(OP_READ); |
1334 | 1308 | ||
1335 | encode_stateid(xdr, args->context); | 1309 | encode_stateid(xdr, args->context); |
1336 | 1310 | ||
1337 | RESERVE_SPACE(12); | 1311 | p = reserve_space(xdr, 12); |
1338 | WRITE64(args->offset); | 1312 | p = xdr_encode_hyper(p, args->offset); |
1339 | WRITE32(args->count); | 1313 | *p = cpu_to_be32(args->count); |
1340 | hdr->nops++; | 1314 | hdr->nops++; |
1341 | hdr->replen += decode_read_maxsz; | 1315 | hdr->replen += decode_read_maxsz; |
1342 | } | 1316 | } |
@@ -1349,20 +1323,20 @@ static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg | |||
1349 | }; | 1323 | }; |
1350 | __be32 *p; | 1324 | __be32 *p; |
1351 | 1325 | ||
1352 | RESERVE_SPACE(12+NFS4_VERIFIER_SIZE+20); | 1326 | p = reserve_space(xdr, 12+NFS4_VERIFIER_SIZE+20); |
1353 | WRITE32(OP_READDIR); | 1327 | *p++ = cpu_to_be32(OP_READDIR); |
1354 | WRITE64(readdir->cookie); | 1328 | p = xdr_encode_hyper(p, readdir->cookie); |
1355 | WRITEMEM(readdir->verifier.data, NFS4_VERIFIER_SIZE); | 1329 | p = xdr_encode_opaque_fixed(p, readdir->verifier.data, NFS4_VERIFIER_SIZE); |
1356 | WRITE32(readdir->count >> 1); /* We're not doing readdirplus */ | 1330 | *p++ = cpu_to_be32(readdir->count >> 1); /* We're not doing readdirplus */ |
1357 | WRITE32(readdir->count); | 1331 | *p++ = cpu_to_be32(readdir->count); |
1358 | WRITE32(2); | 1332 | *p++ = cpu_to_be32(2); |
1359 | /* Switch to mounted_on_fileid if the server supports it */ | 1333 | /* Switch to mounted_on_fileid if the server supports it */ |
1360 | if (readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID) | 1334 | if (readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID) |
1361 | attrs[0] &= ~FATTR4_WORD0_FILEID; | 1335 | attrs[0] &= ~FATTR4_WORD0_FILEID; |
1362 | else | 1336 | else |
1363 | attrs[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; | 1337 | attrs[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; |
1364 | WRITE32(attrs[0] & readdir->bitmask[0]); | 1338 | *p++ = cpu_to_be32(attrs[0] & readdir->bitmask[0]); |
1365 | WRITE32(attrs[1] & readdir->bitmask[1]); | 1339 | *p = cpu_to_be32(attrs[1] & readdir->bitmask[1]); |
1366 | hdr->nops++; | 1340 | hdr->nops++; |
1367 | hdr->replen += decode_readdir_maxsz; | 1341 | hdr->replen += decode_readdir_maxsz; |
1368 | dprintk("%s: cookie = %Lu, verifier = %08x:%08x, bitmap = %08x:%08x\n", | 1342 | dprintk("%s: cookie = %Lu, verifier = %08x:%08x, bitmap = %08x:%08x\n", |
@@ -1378,8 +1352,8 @@ static void encode_readlink(struct xdr_stream *xdr, const struct nfs4_readlink * | |||
1378 | { | 1352 | { |
1379 | __be32 *p; | 1353 | __be32 *p; |
1380 | 1354 | ||
1381 | RESERVE_SPACE(4); | 1355 | p = reserve_space(xdr, 4); |
1382 | WRITE32(OP_READLINK); | 1356 | *p = cpu_to_be32(OP_READLINK); |
1383 | hdr->nops++; | 1357 | hdr->nops++; |
1384 | hdr->replen += decode_readlink_maxsz; | 1358 | hdr->replen += decode_readlink_maxsz; |
1385 | } | 1359 | } |
@@ -1388,10 +1362,9 @@ static void encode_remove(struct xdr_stream *xdr, const struct qstr *name, struc | |||
1388 | { | 1362 | { |
1389 | __be32 *p; | 1363 | __be32 *p; |
1390 | 1364 | ||
1391 | RESERVE_SPACE(8 + name->len); | 1365 | p = reserve_space(xdr, 8 + name->len); |
1392 | WRITE32(OP_REMOVE); | 1366 | *p++ = cpu_to_be32(OP_REMOVE); |
1393 | WRITE32(name->len); | 1367 | xdr_encode_opaque(p, name->name, name->len); |
1394 | WRITEMEM(name->name, name->len); | ||
1395 | hdr->nops++; | 1368 | hdr->nops++; |
1396 | hdr->replen += decode_remove_maxsz; | 1369 | hdr->replen += decode_remove_maxsz; |
1397 | } | 1370 | } |
@@ -1400,14 +1373,10 @@ static void encode_rename(struct xdr_stream *xdr, const struct qstr *oldname, co | |||
1400 | { | 1373 | { |
1401 | __be32 *p; | 1374 | __be32 *p; |
1402 | 1375 | ||
1403 | RESERVE_SPACE(8 + oldname->len); | 1376 | p = reserve_space(xdr, 4); |
1404 | WRITE32(OP_RENAME); | 1377 | *p = cpu_to_be32(OP_RENAME); |
1405 | WRITE32(oldname->len); | 1378 | encode_string(xdr, oldname->len, oldname->name); |
1406 | WRITEMEM(oldname->name, oldname->len); | 1379 | encode_string(xdr, newname->len, newname->name); |
1407 | |||
1408 | RESERVE_SPACE(4 + newname->len); | ||
1409 | WRITE32(newname->len); | ||
1410 | WRITEMEM(newname->name, newname->len); | ||
1411 | hdr->nops++; | 1380 | hdr->nops++; |
1412 | hdr->replen += decode_rename_maxsz; | 1381 | hdr->replen += decode_rename_maxsz; |
1413 | } | 1382 | } |
@@ -1416,9 +1385,9 @@ static void encode_renew(struct xdr_stream *xdr, const struct nfs_client *client | |||
1416 | { | 1385 | { |
1417 | __be32 *p; | 1386 | __be32 *p; |
1418 | 1387 | ||
1419 | RESERVE_SPACE(12); | 1388 | p = reserve_space(xdr, 12); |
1420 | WRITE32(OP_RENEW); | 1389 | *p++ = cpu_to_be32(OP_RENEW); |
1421 | WRITE64(client_stateid->cl_clientid); | 1390 | xdr_encode_hyper(p, client_stateid->cl_clientid); |
1422 | hdr->nops++; | 1391 | hdr->nops++; |
1423 | hdr->replen += decode_renew_maxsz; | 1392 | hdr->replen += decode_renew_maxsz; |
1424 | } | 1393 | } |
@@ -1428,8 +1397,8 @@ encode_restorefh(struct xdr_stream *xdr, struct compound_hdr *hdr) | |||
1428 | { | 1397 | { |
1429 | __be32 *p; | 1398 | __be32 *p; |
1430 | 1399 | ||
1431 | RESERVE_SPACE(4); | 1400 | p = reserve_space(xdr, 4); |
1432 | WRITE32(OP_RESTOREFH); | 1401 | *p = cpu_to_be32(OP_RESTOREFH); |
1433 | hdr->nops++; | 1402 | hdr->nops++; |
1434 | hdr->replen += decode_restorefh_maxsz; | 1403 | hdr->replen += decode_restorefh_maxsz; |
1435 | } | 1404 | } |
@@ -1439,16 +1408,16 @@ encode_setacl(struct xdr_stream *xdr, struct nfs_setaclargs *arg, struct compoun | |||
1439 | { | 1408 | { |
1440 | __be32 *p; | 1409 | __be32 *p; |
1441 | 1410 | ||
1442 | RESERVE_SPACE(4+NFS4_STATEID_SIZE); | 1411 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE); |
1443 | WRITE32(OP_SETATTR); | 1412 | *p++ = cpu_to_be32(OP_SETATTR); |
1444 | WRITEMEM(zero_stateid.data, NFS4_STATEID_SIZE); | 1413 | xdr_encode_opaque_fixed(p, zero_stateid.data, NFS4_STATEID_SIZE); |
1445 | RESERVE_SPACE(2*4); | 1414 | p = reserve_space(xdr, 2*4); |
1446 | WRITE32(1); | 1415 | *p++ = cpu_to_be32(1); |
1447 | WRITE32(FATTR4_WORD0_ACL); | 1416 | *p = cpu_to_be32(FATTR4_WORD0_ACL); |
1448 | if (arg->acl_len % 4) | 1417 | if (arg->acl_len % 4) |
1449 | return -EINVAL; | 1418 | return -EINVAL; |
1450 | RESERVE_SPACE(4); | 1419 | p = reserve_space(xdr, 4); |
1451 | WRITE32(arg->acl_len); | 1420 | *p = cpu_to_be32(arg->acl_len); |
1452 | xdr_write_pages(xdr, arg->acl_pages, arg->acl_pgbase, arg->acl_len); | 1421 | xdr_write_pages(xdr, arg->acl_pages, arg->acl_pgbase, arg->acl_len); |
1453 | hdr->nops++; | 1422 | hdr->nops++; |
1454 | hdr->replen += decode_setacl_maxsz; | 1423 | hdr->replen += decode_setacl_maxsz; |
@@ -1460,8 +1429,8 @@ encode_savefh(struct xdr_stream *xdr, struct compound_hdr *hdr) | |||
1460 | { | 1429 | { |
1461 | __be32 *p; | 1430 | __be32 *p; |
1462 | 1431 | ||
1463 | RESERVE_SPACE(4); | 1432 | p = reserve_space(xdr, 4); |
1464 | WRITE32(OP_SAVEFH); | 1433 | *p = cpu_to_be32(OP_SAVEFH); |
1465 | hdr->nops++; | 1434 | hdr->nops++; |
1466 | hdr->replen += decode_savefh_maxsz; | 1435 | hdr->replen += decode_savefh_maxsz; |
1467 | } | 1436 | } |
@@ -1470,9 +1439,9 @@ static void encode_setattr(struct xdr_stream *xdr, const struct nfs_setattrargs | |||
1470 | { | 1439 | { |
1471 | __be32 *p; | 1440 | __be32 *p; |
1472 | 1441 | ||
1473 | RESERVE_SPACE(4+NFS4_STATEID_SIZE); | 1442 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE); |
1474 | WRITE32(OP_SETATTR); | 1443 | *p++ = cpu_to_be32(OP_SETATTR); |
1475 | WRITEMEM(arg->stateid.data, NFS4_STATEID_SIZE); | 1444 | xdr_encode_opaque_fixed(p, arg->stateid.data, NFS4_STATEID_SIZE); |
1476 | hdr->nops++; | 1445 | hdr->nops++; |
1477 | hdr->replen += decode_setattr_maxsz; | 1446 | hdr->replen += decode_setattr_maxsz; |
1478 | encode_attrs(xdr, arg->iap, server); | 1447 | encode_attrs(xdr, arg->iap, server); |
@@ -1482,17 +1451,17 @@ static void encode_setclientid(struct xdr_stream *xdr, const struct nfs4_setclie | |||
1482 | { | 1451 | { |
1483 | __be32 *p; | 1452 | __be32 *p; |
1484 | 1453 | ||
1485 | RESERVE_SPACE(4 + NFS4_VERIFIER_SIZE); | 1454 | p = reserve_space(xdr, 4 + NFS4_VERIFIER_SIZE); |
1486 | WRITE32(OP_SETCLIENTID); | 1455 | *p++ = cpu_to_be32(OP_SETCLIENTID); |
1487 | WRITEMEM(setclientid->sc_verifier->data, NFS4_VERIFIER_SIZE); | 1456 | xdr_encode_opaque_fixed(p, setclientid->sc_verifier->data, NFS4_VERIFIER_SIZE); |
1488 | 1457 | ||
1489 | encode_string(xdr, setclientid->sc_name_len, setclientid->sc_name); | 1458 | encode_string(xdr, setclientid->sc_name_len, setclientid->sc_name); |
1490 | RESERVE_SPACE(4); | 1459 | p = reserve_space(xdr, 4); |
1491 | WRITE32(setclientid->sc_prog); | 1460 | *p = cpu_to_be32(setclientid->sc_prog); |
1492 | encode_string(xdr, setclientid->sc_netid_len, setclientid->sc_netid); | 1461 | encode_string(xdr, setclientid->sc_netid_len, setclientid->sc_netid); |
1493 | encode_string(xdr, setclientid->sc_uaddr_len, setclientid->sc_uaddr); | 1462 | encode_string(xdr, setclientid->sc_uaddr_len, setclientid->sc_uaddr); |
1494 | RESERVE_SPACE(4); | 1463 | p = reserve_space(xdr, 4); |
1495 | WRITE32(setclientid->sc_cb_ident); | 1464 | *p = cpu_to_be32(setclientid->sc_cb_ident); |
1496 | hdr->nops++; | 1465 | hdr->nops++; |
1497 | hdr->replen += decode_setclientid_maxsz; | 1466 | hdr->replen += decode_setclientid_maxsz; |
1498 | } | 1467 | } |
@@ -1501,10 +1470,10 @@ static void encode_setclientid_confirm(struct xdr_stream *xdr, const struct nfs_ | |||
1501 | { | 1470 | { |
1502 | __be32 *p; | 1471 | __be32 *p; |
1503 | 1472 | ||
1504 | RESERVE_SPACE(12 + NFS4_VERIFIER_SIZE); | 1473 | p = reserve_space(xdr, 12 + NFS4_VERIFIER_SIZE); |
1505 | WRITE32(OP_SETCLIENTID_CONFIRM); | 1474 | *p++ = cpu_to_be32(OP_SETCLIENTID_CONFIRM); |
1506 | WRITE64(client_state->cl_clientid); | 1475 | p = xdr_encode_hyper(p, client_state->cl_clientid); |
1507 | WRITEMEM(client_state->cl_confirm.data, NFS4_VERIFIER_SIZE); | 1476 | xdr_encode_opaque_fixed(p, client_state->cl_confirm.data, NFS4_VERIFIER_SIZE); |
1508 | hdr->nops++; | 1477 | hdr->nops++; |
1509 | hdr->replen += decode_setclientid_confirm_maxsz; | 1478 | hdr->replen += decode_setclientid_confirm_maxsz; |
1510 | } | 1479 | } |
@@ -1513,15 +1482,15 @@ static void encode_write(struct xdr_stream *xdr, const struct nfs_writeargs *arg | |||
1513 | { | 1482 | { |
1514 | __be32 *p; | 1483 | __be32 *p; |
1515 | 1484 | ||
1516 | RESERVE_SPACE(4); | 1485 | p = reserve_space(xdr, 4); |
1517 | WRITE32(OP_WRITE); | 1486 | *p = cpu_to_be32(OP_WRITE); |
1518 | 1487 | ||
1519 | encode_stateid(xdr, args->context); | 1488 | encode_stateid(xdr, args->context); |
1520 | 1489 | ||
1521 | RESERVE_SPACE(16); | 1490 | p = reserve_space(xdr, 16); |
1522 | WRITE64(args->offset); | 1491 | p = xdr_encode_hyper(p, args->offset); |
1523 | WRITE32(args->stable); | 1492 | *p++ = cpu_to_be32(args->stable); |
1524 | WRITE32(args->count); | 1493 | *p = cpu_to_be32(args->count); |
1525 | 1494 | ||
1526 | xdr_write_pages(xdr, args->pages, args->pgbase, args->count); | 1495 | xdr_write_pages(xdr, args->pages, args->pgbase, args->count); |
1527 | hdr->nops++; | 1496 | hdr->nops++; |
@@ -1532,10 +1501,10 @@ static void encode_delegreturn(struct xdr_stream *xdr, const nfs4_stateid *state | |||
1532 | { | 1501 | { |
1533 | __be32 *p; | 1502 | __be32 *p; |
1534 | 1503 | ||
1535 | RESERVE_SPACE(4+NFS4_STATEID_SIZE); | 1504 | p = reserve_space(xdr, 4+NFS4_STATEID_SIZE); |
1536 | 1505 | ||
1537 | WRITE32(OP_DELEGRETURN); | 1506 | *p++ = cpu_to_be32(OP_DELEGRETURN); |
1538 | WRITEMEM(stateid->data, NFS4_STATEID_SIZE); | 1507 | xdr_encode_opaque_fixed(p, stateid->data, NFS4_STATEID_SIZE); |
1539 | hdr->nops++; | 1508 | hdr->nops++; |
1540 | hdr->replen += decode_delegreturn_maxsz; | 1509 | hdr->replen += decode_delegreturn_maxsz; |
1541 | } | 1510 | } |
@@ -1548,16 +1517,16 @@ static void encode_exchange_id(struct xdr_stream *xdr, | |||
1548 | { | 1517 | { |
1549 | __be32 *p; | 1518 | __be32 *p; |
1550 | 1519 | ||
1551 | RESERVE_SPACE(4 + sizeof(args->verifier->data)); | 1520 | p = reserve_space(xdr, 4 + sizeof(args->verifier->data)); |
1552 | WRITE32(OP_EXCHANGE_ID); | 1521 | *p++ = cpu_to_be32(OP_EXCHANGE_ID); |
1553 | WRITEMEM(args->verifier->data, sizeof(args->verifier->data)); | 1522 | xdr_encode_opaque_fixed(p, args->verifier->data, sizeof(args->verifier->data)); |
1554 | 1523 | ||
1555 | encode_string(xdr, args->id_len, args->id); | 1524 | encode_string(xdr, args->id_len, args->id); |
1556 | 1525 | ||
1557 | RESERVE_SPACE(12); | 1526 | p = reserve_space(xdr, 12); |
1558 | WRITE32(args->flags); | 1527 | *p++ = cpu_to_be32(args->flags); |
1559 | WRITE32(0); /* zero length state_protect4_a */ | 1528 | *p++ = cpu_to_be32(0); /* zero length state_protect4_a */ |
1560 | WRITE32(0); /* zero length implementation id array */ | 1529 | *p = cpu_to_be32(0); /* zero length implementation id array */ |
1561 | hdr->nops++; | 1530 | hdr->nops++; |
1562 | hdr->replen += decode_exchange_id_maxsz; | 1531 | hdr->replen += decode_exchange_id_maxsz; |
1563 | } | 1532 | } |
@@ -1571,55 +1540,43 @@ static void encode_create_session(struct xdr_stream *xdr, | |||
1571 | uint32_t len; | 1540 | uint32_t len; |
1572 | struct nfs_client *clp = args->client; | 1541 | struct nfs_client *clp = args->client; |
1573 | 1542 | ||
1574 | RESERVE_SPACE(4); | 1543 | len = scnprintf(machine_name, sizeof(machine_name), "%s", |
1575 | WRITE32(OP_CREATE_SESSION); | 1544 | clp->cl_ipaddr); |
1576 | |||
1577 | RESERVE_SPACE(8); | ||
1578 | WRITE64(clp->cl_ex_clid); | ||
1579 | 1545 | ||
1580 | RESERVE_SPACE(8); | 1546 | p = reserve_space(xdr, 20 + 2*28 + 20 + len + 12); |
1581 | WRITE32(clp->cl_seqid); /*Sequence id */ | 1547 | *p++ = cpu_to_be32(OP_CREATE_SESSION); |
1582 | WRITE32(args->flags); /*flags */ | 1548 | p = xdr_encode_hyper(p, clp->cl_ex_clid); |
1549 | *p++ = cpu_to_be32(clp->cl_seqid); /*Sequence id */ | ||
1550 | *p++ = cpu_to_be32(args->flags); /*flags */ | ||
1583 | 1551 | ||
1584 | RESERVE_SPACE(2*28); /* 2 channel_attrs */ | ||
1585 | /* Fore Channel */ | 1552 | /* Fore Channel */ |
1586 | WRITE32(args->fc_attrs.headerpadsz); /* header padding size */ | 1553 | *p++ = cpu_to_be32(args->fc_attrs.headerpadsz); /* header padding size */ |
1587 | WRITE32(args->fc_attrs.max_rqst_sz); /* max req size */ | 1554 | *p++ = cpu_to_be32(args->fc_attrs.max_rqst_sz); /* max req size */ |
1588 | WRITE32(args->fc_attrs.max_resp_sz); /* max resp size */ | 1555 | *p++ = cpu_to_be32(args->fc_attrs.max_resp_sz); /* max resp size */ |
1589 | WRITE32(args->fc_attrs.max_resp_sz_cached); /* Max resp sz cached */ | 1556 | *p++ = cpu_to_be32(args->fc_attrs.max_resp_sz_cached); /* Max resp sz cached */ |
1590 | WRITE32(args->fc_attrs.max_ops); /* max operations */ | 1557 | *p++ = cpu_to_be32(args->fc_attrs.max_ops); /* max operations */ |
1591 | WRITE32(args->fc_attrs.max_reqs); /* max requests */ | 1558 | *p++ = cpu_to_be32(args->fc_attrs.max_reqs); /* max requests */ |
1592 | WRITE32(0); /* rdmachannel_attrs */ | 1559 | *p++ = cpu_to_be32(0); /* rdmachannel_attrs */ |
1593 | 1560 | ||
1594 | /* Back Channel */ | 1561 | /* Back Channel */ |
1595 | WRITE32(args->fc_attrs.headerpadsz); /* header padding size */ | 1562 | *p++ = cpu_to_be32(args->fc_attrs.headerpadsz); /* header padding size */ |
1596 | WRITE32(args->bc_attrs.max_rqst_sz); /* max req size */ | 1563 | *p++ = cpu_to_be32(args->bc_attrs.max_rqst_sz); /* max req size */ |
1597 | WRITE32(args->bc_attrs.max_resp_sz); /* max resp size */ | 1564 | *p++ = cpu_to_be32(args->bc_attrs.max_resp_sz); /* max resp size */ |
1598 | WRITE32(args->bc_attrs.max_resp_sz_cached); /* Max resp sz cached */ | 1565 | *p++ = cpu_to_be32(args->bc_attrs.max_resp_sz_cached); /* Max resp sz cached */ |
1599 | WRITE32(args->bc_attrs.max_ops); /* max operations */ | 1566 | *p++ = cpu_to_be32(args->bc_attrs.max_ops); /* max operations */ |
1600 | WRITE32(args->bc_attrs.max_reqs); /* max requests */ | 1567 | *p++ = cpu_to_be32(args->bc_attrs.max_reqs); /* max requests */ |
1601 | WRITE32(0); /* rdmachannel_attrs */ | 1568 | *p++ = cpu_to_be32(0); /* rdmachannel_attrs */ |
1602 | 1569 | ||
1603 | RESERVE_SPACE(4); | 1570 | *p++ = cpu_to_be32(args->cb_program); /* cb_program */ |
1604 | WRITE32(args->cb_program); /* cb_program */ | 1571 | *p++ = cpu_to_be32(1); |
1605 | 1572 | *p++ = cpu_to_be32(RPC_AUTH_UNIX); /* auth_sys */ | |
1606 | RESERVE_SPACE(4); /* # of security flavors */ | ||
1607 | WRITE32(1); | ||
1608 | |||
1609 | RESERVE_SPACE(4); | ||
1610 | WRITE32(RPC_AUTH_UNIX); /* auth_sys */ | ||
1611 | 1573 | ||
1612 | /* authsys_parms rfc1831 */ | 1574 | /* authsys_parms rfc1831 */ |
1613 | RESERVE_SPACE(4); | 1575 | *p++ = cpu_to_be32((u32)clp->cl_boot_time.tv_nsec); /* stamp */ |
1614 | WRITE32((u32)clp->cl_boot_time.tv_nsec); /* stamp */ | 1576 | p = xdr_encode_opaque(p, machine_name, len); |
1615 | len = scnprintf(machine_name, sizeof(machine_name), "%s", | 1577 | *p++ = cpu_to_be32(0); /* UID */ |
1616 | clp->cl_ipaddr); | 1578 | *p++ = cpu_to_be32(0); /* GID */ |
1617 | RESERVE_SPACE(16 + len); | 1579 | *p = cpu_to_be32(0); /* No more gids */ |
1618 | WRITE32(len); | ||
1619 | WRITEMEM(machine_name, len); | ||
1620 | WRITE32(0); /* UID */ | ||
1621 | WRITE32(0); /* GID */ | ||
1622 | WRITE32(0); /* No more gids */ | ||
1623 | hdr->nops++; | 1580 | hdr->nops++; |
1624 | hdr->replen += decode_create_session_maxsz; | 1581 | hdr->replen += decode_create_session_maxsz; |
1625 | } | 1582 | } |
@@ -1629,9 +1586,9 @@ static void encode_destroy_session(struct xdr_stream *xdr, | |||
1629 | struct compound_hdr *hdr) | 1586 | struct compound_hdr *hdr) |
1630 | { | 1587 | { |
1631 | __be32 *p; | 1588 | __be32 *p; |
1632 | RESERVE_SPACE(4 + NFS4_MAX_SESSIONID_LEN); | 1589 | p = reserve_space(xdr, 4 + NFS4_MAX_SESSIONID_LEN); |
1633 | WRITE32(OP_DESTROY_SESSION); | 1590 | *p++ = cpu_to_be32(OP_DESTROY_SESSION); |
1634 | WRITEMEM(session->sess_id.data, NFS4_MAX_SESSIONID_LEN); | 1591 | xdr_encode_opaque_fixed(p, session->sess_id.data, NFS4_MAX_SESSIONID_LEN); |
1635 | hdr->nops++; | 1592 | hdr->nops++; |
1636 | hdr->replen += decode_destroy_session_maxsz; | 1593 | hdr->replen += decode_destroy_session_maxsz; |
1637 | } | 1594 | } |
@@ -1655,8 +1612,8 @@ static void encode_sequence(struct xdr_stream *xdr, | |||
1655 | WARN_ON(args->sa_slotid == NFS4_MAX_SLOT_TABLE); | 1612 | WARN_ON(args->sa_slotid == NFS4_MAX_SLOT_TABLE); |
1656 | slot = tp->slots + args->sa_slotid; | 1613 | slot = tp->slots + args->sa_slotid; |
1657 | 1614 | ||
1658 | RESERVE_SPACE(4); | 1615 | p = reserve_space(xdr, 4 + NFS4_MAX_SESSIONID_LEN + 16); |
1659 | WRITE32(OP_SEQUENCE); | 1616 | *p++ = cpu_to_be32(OP_SEQUENCE); |
1660 | 1617 | ||
1661 | /* | 1618 | /* |
1662 | * Sessionid + seqid + slotid + max slotid + cache_this | 1619 | * Sessionid + seqid + slotid + max slotid + cache_this |
@@ -1670,12 +1627,11 @@ static void encode_sequence(struct xdr_stream *xdr, | |||
1670 | ((u32 *)session->sess_id.data)[3], | 1627 | ((u32 *)session->sess_id.data)[3], |
1671 | slot->seq_nr, args->sa_slotid, | 1628 | slot->seq_nr, args->sa_slotid, |
1672 | tp->highest_used_slotid, args->sa_cache_this); | 1629 | tp->highest_used_slotid, args->sa_cache_this); |
1673 | RESERVE_SPACE(NFS4_MAX_SESSIONID_LEN + 16); | 1630 | p = xdr_encode_opaque_fixed(p, session->sess_id.data, NFS4_MAX_SESSIONID_LEN); |
1674 | WRITEMEM(session->sess_id.data, NFS4_MAX_SESSIONID_LEN); | 1631 | *p++ = cpu_to_be32(slot->seq_nr); |
1675 | WRITE32(slot->seq_nr); | 1632 | *p++ = cpu_to_be32(args->sa_slotid); |
1676 | WRITE32(args->sa_slotid); | 1633 | *p++ = cpu_to_be32(tp->highest_used_slotid); |
1677 | WRITE32(tp->highest_used_slotid); | 1634 | *p = cpu_to_be32(args->sa_cache_this); |
1678 | WRITE32(args->sa_cache_this); | ||
1679 | hdr->nops++; | 1635 | hdr->nops++; |
1680 | hdr->replen += decode_sequence_maxsz; | 1636 | hdr->replen += decode_sequence_maxsz; |
1681 | #endif /* CONFIG_NFS_V4_1 */ | 1637 | #endif /* CONFIG_NFS_V4_1 */ |
@@ -2466,68 +2422,53 @@ static int nfs4_xdr_enc_get_lease_time(struct rpc_rqst *req, uint32_t *p, | |||
2466 | } | 2422 | } |
2467 | #endif /* CONFIG_NFS_V4_1 */ | 2423 | #endif /* CONFIG_NFS_V4_1 */ |
2468 | 2424 | ||
2469 | /* | 2425 | static void print_overflow_msg(const char *func, const struct xdr_stream *xdr) |
2470 | * START OF "GENERIC" DECODE ROUTINES. | 2426 | { |
2471 | * These may look a little ugly since they are imported from a "generic" | 2427 | dprintk("nfs: %s: prematurely hit end of receive buffer. " |
2472 | * set of XDR encode/decode routines which are intended to be shared by | 2428 | "Remaining buffer length is %tu words.\n", |
2473 | * all of our NFSv4 implementations (OpenBSD, MacOS X...). | 2429 | func, xdr->end - xdr->p); |
2474 | * | 2430 | } |
2475 | * If the pain of reading these is too great, it should be a straightforward | ||
2476 | * task to translate them into Linux-specific versions which are more | ||
2477 | * consistent with the style used in NFSv2/v3... | ||
2478 | */ | ||
2479 | #define READ32(x) (x) = ntohl(*p++) | ||
2480 | #define READ64(x) do { \ | ||
2481 | (x) = (u64)ntohl(*p++) << 32; \ | ||
2482 | (x) |= ntohl(*p++); \ | ||
2483 | } while (0) | ||
2484 | #define READTIME(x) do { \ | ||
2485 | p++; \ | ||
2486 | (x.tv_sec) = ntohl(*p++); \ | ||
2487 | (x.tv_nsec) = ntohl(*p++); \ | ||
2488 | } while (0) | ||
2489 | #define COPYMEM(x,nbytes) do { \ | ||
2490 | memcpy((x), p, nbytes); \ | ||
2491 | p += XDR_QUADLEN(nbytes); \ | ||
2492 | } while (0) | ||
2493 | |||
2494 | #define READ_BUF(nbytes) do { \ | ||
2495 | p = xdr_inline_decode(xdr, nbytes); \ | ||
2496 | if (unlikely(!p)) { \ | ||
2497 | dprintk("nfs: %s: prematurely hit end of receive" \ | ||
2498 | " buffer\n", __func__); \ | ||
2499 | dprintk("nfs: %s: xdr->p=%p, bytes=%u, xdr->end=%p\n", \ | ||
2500 | __func__, xdr->p, nbytes, xdr->end); \ | ||
2501 | return -EIO; \ | ||
2502 | } \ | ||
2503 | } while (0) | ||
2504 | 2431 | ||
2505 | static int decode_opaque_inline(struct xdr_stream *xdr, unsigned int *len, char **string) | 2432 | static int decode_opaque_inline(struct xdr_stream *xdr, unsigned int *len, char **string) |
2506 | { | 2433 | { |
2507 | __be32 *p; | 2434 | __be32 *p; |
2508 | 2435 | ||
2509 | READ_BUF(4); | 2436 | p = xdr_inline_decode(xdr, 4); |
2510 | READ32(*len); | 2437 | if (unlikely(!p)) |
2511 | READ_BUF(*len); | 2438 | goto out_overflow; |
2439 | *len = be32_to_cpup(p); | ||
2440 | p = xdr_inline_decode(xdr, *len); | ||
2441 | if (unlikely(!p)) | ||
2442 | goto out_overflow; | ||
2512 | *string = (char *)p; | 2443 | *string = (char *)p; |
2513 | return 0; | 2444 | return 0; |
2445 | out_overflow: | ||
2446 | print_overflow_msg(__func__, xdr); | ||
2447 | return -EIO; | ||
2514 | } | 2448 | } |
2515 | 2449 | ||
2516 | static int decode_compound_hdr(struct xdr_stream *xdr, struct compound_hdr *hdr) | 2450 | static int decode_compound_hdr(struct xdr_stream *xdr, struct compound_hdr *hdr) |
2517 | { | 2451 | { |
2518 | __be32 *p; | 2452 | __be32 *p; |
2519 | 2453 | ||
2520 | READ_BUF(8); | 2454 | p = xdr_inline_decode(xdr, 8); |
2521 | READ32(hdr->status); | 2455 | if (unlikely(!p)) |
2522 | READ32(hdr->taglen); | 2456 | goto out_overflow; |
2457 | hdr->status = be32_to_cpup(p++); | ||
2458 | hdr->taglen = be32_to_cpup(p); | ||
2523 | 2459 | ||
2524 | READ_BUF(hdr->taglen + 4); | 2460 | p = xdr_inline_decode(xdr, hdr->taglen + 4); |
2461 | if (unlikely(!p)) | ||
2462 | goto out_overflow; | ||
2525 | hdr->tag = (char *)p; | 2463 | hdr->tag = (char *)p; |
2526 | p += XDR_QUADLEN(hdr->taglen); | 2464 | p += XDR_QUADLEN(hdr->taglen); |
2527 | READ32(hdr->nops); | 2465 | hdr->nops = be32_to_cpup(p); |
2528 | if (unlikely(hdr->nops < 1)) | 2466 | if (unlikely(hdr->nops < 1)) |
2529 | return nfs4_stat_to_errno(hdr->status); | 2467 | return nfs4_stat_to_errno(hdr->status); |
2530 | return 0; | 2468 | return 0; |
2469 | out_overflow: | ||
2470 | print_overflow_msg(__func__, xdr); | ||
2471 | return -EIO; | ||
2531 | } | 2472 | } |
2532 | 2473 | ||
2533 | static int decode_op_hdr(struct xdr_stream *xdr, enum nfs_opnum4 expected) | 2474 | static int decode_op_hdr(struct xdr_stream *xdr, enum nfs_opnum4 expected) |
@@ -2536,18 +2477,23 @@ static int decode_op_hdr(struct xdr_stream *xdr, enum nfs_opnum4 expected) | |||
2536 | uint32_t opnum; | 2477 | uint32_t opnum; |
2537 | int32_t nfserr; | 2478 | int32_t nfserr; |
2538 | 2479 | ||
2539 | READ_BUF(8); | 2480 | p = xdr_inline_decode(xdr, 8); |
2540 | READ32(opnum); | 2481 | if (unlikely(!p)) |
2482 | goto out_overflow; | ||
2483 | opnum = be32_to_cpup(p++); | ||
2541 | if (opnum != expected) { | 2484 | if (opnum != expected) { |
2542 | dprintk("nfs: Server returned operation" | 2485 | dprintk("nfs: Server returned operation" |
2543 | " %d but we issued a request for %d\n", | 2486 | " %d but we issued a request for %d\n", |
2544 | opnum, expected); | 2487 | opnum, expected); |
2545 | return -EIO; | 2488 | return -EIO; |
2546 | } | 2489 | } |
2547 | READ32(nfserr); | 2490 | nfserr = be32_to_cpup(p); |
2548 | if (nfserr != NFS_OK) | 2491 | if (nfserr != NFS_OK) |
2549 | return nfs4_stat_to_errno(nfserr); | 2492 | return nfs4_stat_to_errno(nfserr); |
2550 | return 0; | 2493 | return 0; |
2494 | out_overflow: | ||
2495 | print_overflow_msg(__func__, xdr); | ||
2496 | return -EIO; | ||
2551 | } | 2497 | } |
2552 | 2498 | ||
2553 | /* Dummy routine */ | 2499 | /* Dummy routine */ |
@@ -2557,8 +2503,11 @@ static int decode_ace(struct xdr_stream *xdr, void *ace, struct nfs_client *clp) | |||
2557 | unsigned int strlen; | 2503 | unsigned int strlen; |
2558 | char *str; | 2504 | char *str; |
2559 | 2505 | ||
2560 | READ_BUF(12); | 2506 | p = xdr_inline_decode(xdr, 12); |
2561 | return decode_opaque_inline(xdr, &strlen, &str); | 2507 | if (likely(p)) |
2508 | return decode_opaque_inline(xdr, &strlen, &str); | ||
2509 | print_overflow_msg(__func__, xdr); | ||
2510 | return -EIO; | ||
2562 | } | 2511 | } |
2563 | 2512 | ||
2564 | static int decode_attr_bitmap(struct xdr_stream *xdr, uint32_t *bitmap) | 2513 | static int decode_attr_bitmap(struct xdr_stream *xdr, uint32_t *bitmap) |
@@ -2566,27 +2515,39 @@ static int decode_attr_bitmap(struct xdr_stream *xdr, uint32_t *bitmap) | |||
2566 | uint32_t bmlen; | 2515 | uint32_t bmlen; |
2567 | __be32 *p; | 2516 | __be32 *p; |
2568 | 2517 | ||
2569 | READ_BUF(4); | 2518 | p = xdr_inline_decode(xdr, 4); |
2570 | READ32(bmlen); | 2519 | if (unlikely(!p)) |
2520 | goto out_overflow; | ||
2521 | bmlen = be32_to_cpup(p); | ||
2571 | 2522 | ||
2572 | bitmap[0] = bitmap[1] = 0; | 2523 | bitmap[0] = bitmap[1] = 0; |
2573 | READ_BUF((bmlen << 2)); | 2524 | p = xdr_inline_decode(xdr, (bmlen << 2)); |
2525 | if (unlikely(!p)) | ||
2526 | goto out_overflow; | ||
2574 | if (bmlen > 0) { | 2527 | if (bmlen > 0) { |
2575 | READ32(bitmap[0]); | 2528 | bitmap[0] = be32_to_cpup(p++); |
2576 | if (bmlen > 1) | 2529 | if (bmlen > 1) |
2577 | READ32(bitmap[1]); | 2530 | bitmap[1] = be32_to_cpup(p); |
2578 | } | 2531 | } |
2579 | return 0; | 2532 | return 0; |
2533 | out_overflow: | ||
2534 | print_overflow_msg(__func__, xdr); | ||
2535 | return -EIO; | ||
2580 | } | 2536 | } |
2581 | 2537 | ||
2582 | static inline int decode_attr_length(struct xdr_stream *xdr, uint32_t *attrlen, __be32 **savep) | 2538 | static inline int decode_attr_length(struct xdr_stream *xdr, uint32_t *attrlen, __be32 **savep) |
2583 | { | 2539 | { |
2584 | __be32 *p; | 2540 | __be32 *p; |
2585 | 2541 | ||
2586 | READ_BUF(4); | 2542 | p = xdr_inline_decode(xdr, 4); |
2587 | READ32(*attrlen); | 2543 | if (unlikely(!p)) |
2544 | goto out_overflow; | ||
2545 | *attrlen = be32_to_cpup(p); | ||
2588 | *savep = xdr->p; | 2546 | *savep = xdr->p; |
2589 | return 0; | 2547 | return 0; |
2548 | out_overflow: | ||
2549 | print_overflow_msg(__func__, xdr); | ||
2550 | return -EIO; | ||
2590 | } | 2551 | } |
2591 | 2552 | ||
2592 | static int decode_attr_supported(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *bitmask) | 2553 | static int decode_attr_supported(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *bitmask) |
@@ -2609,8 +2570,10 @@ static int decode_attr_type(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t * | |||
2609 | if (unlikely(bitmap[0] & (FATTR4_WORD0_TYPE - 1U))) | 2570 | if (unlikely(bitmap[0] & (FATTR4_WORD0_TYPE - 1U))) |
2610 | return -EIO; | 2571 | return -EIO; |
2611 | if (likely(bitmap[0] & FATTR4_WORD0_TYPE)) { | 2572 | if (likely(bitmap[0] & FATTR4_WORD0_TYPE)) { |
2612 | READ_BUF(4); | 2573 | p = xdr_inline_decode(xdr, 4); |
2613 | READ32(*type); | 2574 | if (unlikely(!p)) |
2575 | goto out_overflow; | ||
2576 | *type = be32_to_cpup(p); | ||
2614 | if (*type < NF4REG || *type > NF4NAMEDATTR) { | 2577 | if (*type < NF4REG || *type > NF4NAMEDATTR) { |
2615 | dprintk("%s: bad type %d\n", __func__, *type); | 2578 | dprintk("%s: bad type %d\n", __func__, *type); |
2616 | return -EIO; | 2579 | return -EIO; |
@@ -2620,6 +2583,9 @@ static int decode_attr_type(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t * | |||
2620 | } | 2583 | } |
2621 | dprintk("%s: type=0%o\n", __func__, nfs_type2fmt[*type]); | 2584 | dprintk("%s: type=0%o\n", __func__, nfs_type2fmt[*type]); |
2622 | return ret; | 2585 | return ret; |
2586 | out_overflow: | ||
2587 | print_overflow_msg(__func__, xdr); | ||
2588 | return -EIO; | ||
2623 | } | 2589 | } |
2624 | 2590 | ||
2625 | static int decode_attr_change(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *change) | 2591 | static int decode_attr_change(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *change) |
@@ -2631,14 +2597,19 @@ static int decode_attr_change(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t | |||
2631 | if (unlikely(bitmap[0] & (FATTR4_WORD0_CHANGE - 1U))) | 2597 | if (unlikely(bitmap[0] & (FATTR4_WORD0_CHANGE - 1U))) |
2632 | return -EIO; | 2598 | return -EIO; |
2633 | if (likely(bitmap[0] & FATTR4_WORD0_CHANGE)) { | 2599 | if (likely(bitmap[0] & FATTR4_WORD0_CHANGE)) { |
2634 | READ_BUF(8); | 2600 | p = xdr_inline_decode(xdr, 8); |
2635 | READ64(*change); | 2601 | if (unlikely(!p)) |
2602 | goto out_overflow; | ||
2603 | xdr_decode_hyper(p, change); | ||
2636 | bitmap[0] &= ~FATTR4_WORD0_CHANGE; | 2604 | bitmap[0] &= ~FATTR4_WORD0_CHANGE; |
2637 | ret = NFS_ATTR_FATTR_CHANGE; | 2605 | ret = NFS_ATTR_FATTR_CHANGE; |
2638 | } | 2606 | } |
2639 | dprintk("%s: change attribute=%Lu\n", __func__, | 2607 | dprintk("%s: change attribute=%Lu\n", __func__, |
2640 | (unsigned long long)*change); | 2608 | (unsigned long long)*change); |
2641 | return ret; | 2609 | return ret; |
2610 | out_overflow: | ||
2611 | print_overflow_msg(__func__, xdr); | ||
2612 | return -EIO; | ||
2642 | } | 2613 | } |
2643 | 2614 | ||
2644 | static int decode_attr_size(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *size) | 2615 | static int decode_attr_size(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *size) |
@@ -2650,13 +2621,18 @@ static int decode_attr_size(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t * | |||
2650 | if (unlikely(bitmap[0] & (FATTR4_WORD0_SIZE - 1U))) | 2621 | if (unlikely(bitmap[0] & (FATTR4_WORD0_SIZE - 1U))) |
2651 | return -EIO; | 2622 | return -EIO; |
2652 | if (likely(bitmap[0] & FATTR4_WORD0_SIZE)) { | 2623 | if (likely(bitmap[0] & FATTR4_WORD0_SIZE)) { |
2653 | READ_BUF(8); | 2624 | p = xdr_inline_decode(xdr, 8); |
2654 | READ64(*size); | 2625 | if (unlikely(!p)) |
2626 | goto out_overflow; | ||
2627 | xdr_decode_hyper(p, size); | ||
2655 | bitmap[0] &= ~FATTR4_WORD0_SIZE; | 2628 | bitmap[0] &= ~FATTR4_WORD0_SIZE; |
2656 | ret = NFS_ATTR_FATTR_SIZE; | 2629 | ret = NFS_ATTR_FATTR_SIZE; |
2657 | } | 2630 | } |
2658 | dprintk("%s: file size=%Lu\n", __func__, (unsigned long long)*size); | 2631 | dprintk("%s: file size=%Lu\n", __func__, (unsigned long long)*size); |
2659 | return ret; | 2632 | return ret; |
2633 | out_overflow: | ||
2634 | print_overflow_msg(__func__, xdr); | ||
2635 | return -EIO; | ||
2660 | } | 2636 | } |
2661 | 2637 | ||
2662 | static int decode_attr_link_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 2638 | static int decode_attr_link_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -2667,12 +2643,17 @@ static int decode_attr_link_support(struct xdr_stream *xdr, uint32_t *bitmap, ui | |||
2667 | if (unlikely(bitmap[0] & (FATTR4_WORD0_LINK_SUPPORT - 1U))) | 2643 | if (unlikely(bitmap[0] & (FATTR4_WORD0_LINK_SUPPORT - 1U))) |
2668 | return -EIO; | 2644 | return -EIO; |
2669 | if (likely(bitmap[0] & FATTR4_WORD0_LINK_SUPPORT)) { | 2645 | if (likely(bitmap[0] & FATTR4_WORD0_LINK_SUPPORT)) { |
2670 | READ_BUF(4); | 2646 | p = xdr_inline_decode(xdr, 4); |
2671 | READ32(*res); | 2647 | if (unlikely(!p)) |
2648 | goto out_overflow; | ||
2649 | *res = be32_to_cpup(p); | ||
2672 | bitmap[0] &= ~FATTR4_WORD0_LINK_SUPPORT; | 2650 | bitmap[0] &= ~FATTR4_WORD0_LINK_SUPPORT; |
2673 | } | 2651 | } |
2674 | dprintk("%s: link support=%s\n", __func__, *res == 0 ? "false" : "true"); | 2652 | dprintk("%s: link support=%s\n", __func__, *res == 0 ? "false" : "true"); |
2675 | return 0; | 2653 | return 0; |
2654 | out_overflow: | ||
2655 | print_overflow_msg(__func__, xdr); | ||
2656 | return -EIO; | ||
2676 | } | 2657 | } |
2677 | 2658 | ||
2678 | static int decode_attr_symlink_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 2659 | static int decode_attr_symlink_support(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -2683,12 +2664,17 @@ static int decode_attr_symlink_support(struct xdr_stream *xdr, uint32_t *bitmap, | |||
2683 | if (unlikely(bitmap[0] & (FATTR4_WORD0_SYMLINK_SUPPORT - 1U))) | 2664 | if (unlikely(bitmap[0] & (FATTR4_WORD0_SYMLINK_SUPPORT - 1U))) |
2684 | return -EIO; | 2665 | return -EIO; |
2685 | if (likely(bitmap[0] & FATTR4_WORD0_SYMLINK_SUPPORT)) { | 2666 | if (likely(bitmap[0] & FATTR4_WORD0_SYMLINK_SUPPORT)) { |
2686 | READ_BUF(4); | 2667 | p = xdr_inline_decode(xdr, 4); |
2687 | READ32(*res); | 2668 | if (unlikely(!p)) |
2669 | goto out_overflow; | ||
2670 | *res = be32_to_cpup(p); | ||
2688 | bitmap[0] &= ~FATTR4_WORD0_SYMLINK_SUPPORT; | 2671 | bitmap[0] &= ~FATTR4_WORD0_SYMLINK_SUPPORT; |
2689 | } | 2672 | } |
2690 | dprintk("%s: symlink support=%s\n", __func__, *res == 0 ? "false" : "true"); | 2673 | dprintk("%s: symlink support=%s\n", __func__, *res == 0 ? "false" : "true"); |
2691 | return 0; | 2674 | return 0; |
2675 | out_overflow: | ||
2676 | print_overflow_msg(__func__, xdr); | ||
2677 | return -EIO; | ||
2692 | } | 2678 | } |
2693 | 2679 | ||
2694 | static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_fsid *fsid) | 2680 | static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_fsid *fsid) |
@@ -2701,9 +2687,11 @@ static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs | |||
2701 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FSID - 1U))) | 2687 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FSID - 1U))) |
2702 | return -EIO; | 2688 | return -EIO; |
2703 | if (likely(bitmap[0] & FATTR4_WORD0_FSID)) { | 2689 | if (likely(bitmap[0] & FATTR4_WORD0_FSID)) { |
2704 | READ_BUF(16); | 2690 | p = xdr_inline_decode(xdr, 16); |
2705 | READ64(fsid->major); | 2691 | if (unlikely(!p)) |
2706 | READ64(fsid->minor); | 2692 | goto out_overflow; |
2693 | p = xdr_decode_hyper(p, &fsid->major); | ||
2694 | xdr_decode_hyper(p, &fsid->minor); | ||
2707 | bitmap[0] &= ~FATTR4_WORD0_FSID; | 2695 | bitmap[0] &= ~FATTR4_WORD0_FSID; |
2708 | ret = NFS_ATTR_FATTR_FSID; | 2696 | ret = NFS_ATTR_FATTR_FSID; |
2709 | } | 2697 | } |
@@ -2711,6 +2699,9 @@ static int decode_attr_fsid(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs | |||
2711 | (unsigned long long)fsid->major, | 2699 | (unsigned long long)fsid->major, |
2712 | (unsigned long long)fsid->minor); | 2700 | (unsigned long long)fsid->minor); |
2713 | return ret; | 2701 | return ret; |
2702 | out_overflow: | ||
2703 | print_overflow_msg(__func__, xdr); | ||
2704 | return -EIO; | ||
2714 | } | 2705 | } |
2715 | 2706 | ||
2716 | static int decode_attr_lease_time(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 2707 | static int decode_attr_lease_time(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -2721,12 +2712,17 @@ static int decode_attr_lease_time(struct xdr_stream *xdr, uint32_t *bitmap, uint | |||
2721 | if (unlikely(bitmap[0] & (FATTR4_WORD0_LEASE_TIME - 1U))) | 2712 | if (unlikely(bitmap[0] & (FATTR4_WORD0_LEASE_TIME - 1U))) |
2722 | return -EIO; | 2713 | return -EIO; |
2723 | if (likely(bitmap[0] & FATTR4_WORD0_LEASE_TIME)) { | 2714 | if (likely(bitmap[0] & FATTR4_WORD0_LEASE_TIME)) { |
2724 | READ_BUF(4); | 2715 | p = xdr_inline_decode(xdr, 4); |
2725 | READ32(*res); | 2716 | if (unlikely(!p)) |
2717 | goto out_overflow; | ||
2718 | *res = be32_to_cpup(p); | ||
2726 | bitmap[0] &= ~FATTR4_WORD0_LEASE_TIME; | 2719 | bitmap[0] &= ~FATTR4_WORD0_LEASE_TIME; |
2727 | } | 2720 | } |
2728 | dprintk("%s: file size=%u\n", __func__, (unsigned int)*res); | 2721 | dprintk("%s: file size=%u\n", __func__, (unsigned int)*res); |
2729 | return 0; | 2722 | return 0; |
2723 | out_overflow: | ||
2724 | print_overflow_msg(__func__, xdr); | ||
2725 | return -EIO; | ||
2730 | } | 2726 | } |
2731 | 2727 | ||
2732 | static int decode_attr_aclsupport(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 2728 | static int decode_attr_aclsupport(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -2737,12 +2733,17 @@ static int decode_attr_aclsupport(struct xdr_stream *xdr, uint32_t *bitmap, uint | |||
2737 | if (unlikely(bitmap[0] & (FATTR4_WORD0_ACLSUPPORT - 1U))) | 2733 | if (unlikely(bitmap[0] & (FATTR4_WORD0_ACLSUPPORT - 1U))) |
2738 | return -EIO; | 2734 | return -EIO; |
2739 | if (likely(bitmap[0] & FATTR4_WORD0_ACLSUPPORT)) { | 2735 | if (likely(bitmap[0] & FATTR4_WORD0_ACLSUPPORT)) { |
2740 | READ_BUF(4); | 2736 | p = xdr_inline_decode(xdr, 4); |
2741 | READ32(*res); | 2737 | if (unlikely(!p)) |
2738 | goto out_overflow; | ||
2739 | *res = be32_to_cpup(p); | ||
2742 | bitmap[0] &= ~FATTR4_WORD0_ACLSUPPORT; | 2740 | bitmap[0] &= ~FATTR4_WORD0_ACLSUPPORT; |
2743 | } | 2741 | } |
2744 | dprintk("%s: ACLs supported=%u\n", __func__, (unsigned int)*res); | 2742 | dprintk("%s: ACLs supported=%u\n", __func__, (unsigned int)*res); |
2745 | return 0; | 2743 | return 0; |
2744 | out_overflow: | ||
2745 | print_overflow_msg(__func__, xdr); | ||
2746 | return -EIO; | ||
2746 | } | 2747 | } |
2747 | 2748 | ||
2748 | static int decode_attr_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid) | 2749 | static int decode_attr_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid) |
@@ -2754,13 +2755,18 @@ static int decode_attr_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t | |||
2754 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILEID - 1U))) | 2755 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILEID - 1U))) |
2755 | return -EIO; | 2756 | return -EIO; |
2756 | if (likely(bitmap[0] & FATTR4_WORD0_FILEID)) { | 2757 | if (likely(bitmap[0] & FATTR4_WORD0_FILEID)) { |
2757 | READ_BUF(8); | 2758 | p = xdr_inline_decode(xdr, 8); |
2758 | READ64(*fileid); | 2759 | if (unlikely(!p)) |
2760 | goto out_overflow; | ||
2761 | xdr_decode_hyper(p, fileid); | ||
2759 | bitmap[0] &= ~FATTR4_WORD0_FILEID; | 2762 | bitmap[0] &= ~FATTR4_WORD0_FILEID; |
2760 | ret = NFS_ATTR_FATTR_FILEID; | 2763 | ret = NFS_ATTR_FATTR_FILEID; |
2761 | } | 2764 | } |
2762 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); | 2765 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); |
2763 | return ret; | 2766 | return ret; |
2767 | out_overflow: | ||
2768 | print_overflow_msg(__func__, xdr); | ||
2769 | return -EIO; | ||
2764 | } | 2770 | } |
2765 | 2771 | ||
2766 | static int decode_attr_mounted_on_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid) | 2772 | static int decode_attr_mounted_on_fileid(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *fileid) |
@@ -2772,13 +2778,18 @@ static int decode_attr_mounted_on_fileid(struct xdr_stream *xdr, uint32_t *bitma | |||
2772 | if (unlikely(bitmap[1] & (FATTR4_WORD1_MOUNTED_ON_FILEID - 1U))) | 2778 | if (unlikely(bitmap[1] & (FATTR4_WORD1_MOUNTED_ON_FILEID - 1U))) |
2773 | return -EIO; | 2779 | return -EIO; |
2774 | if (likely(bitmap[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)) { | 2780 | if (likely(bitmap[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)) { |
2775 | READ_BUF(8); | 2781 | p = xdr_inline_decode(xdr, 8); |
2776 | READ64(*fileid); | 2782 | if (unlikely(!p)) |
2783 | goto out_overflow; | ||
2784 | xdr_decode_hyper(p, fileid); | ||
2777 | bitmap[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; | 2785 | bitmap[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; |
2778 | ret = NFS_ATTR_FATTR_FILEID; | 2786 | ret = NFS_ATTR_FATTR_FILEID; |
2779 | } | 2787 | } |
2780 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); | 2788 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); |
2781 | return ret; | 2789 | return ret; |
2790 | out_overflow: | ||
2791 | print_overflow_msg(__func__, xdr); | ||
2792 | return -EIO; | ||
2782 | } | 2793 | } |
2783 | 2794 | ||
2784 | static int decode_attr_files_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 2795 | static int decode_attr_files_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -2790,12 +2801,17 @@ static int decode_attr_files_avail(struct xdr_stream *xdr, uint32_t *bitmap, uin | |||
2790 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_AVAIL - 1U))) | 2801 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_AVAIL - 1U))) |
2791 | return -EIO; | 2802 | return -EIO; |
2792 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_AVAIL)) { | 2803 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_AVAIL)) { |
2793 | READ_BUF(8); | 2804 | p = xdr_inline_decode(xdr, 8); |
2794 | READ64(*res); | 2805 | if (unlikely(!p)) |
2806 | goto out_overflow; | ||
2807 | xdr_decode_hyper(p, res); | ||
2795 | bitmap[0] &= ~FATTR4_WORD0_FILES_AVAIL; | 2808 | bitmap[0] &= ~FATTR4_WORD0_FILES_AVAIL; |
2796 | } | 2809 | } |
2797 | dprintk("%s: files avail=%Lu\n", __func__, (unsigned long long)*res); | 2810 | dprintk("%s: files avail=%Lu\n", __func__, (unsigned long long)*res); |
2798 | return status; | 2811 | return status; |
2812 | out_overflow: | ||
2813 | print_overflow_msg(__func__, xdr); | ||
2814 | return -EIO; | ||
2799 | } | 2815 | } |
2800 | 2816 | ||
2801 | static int decode_attr_files_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 2817 | static int decode_attr_files_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -2807,12 +2823,17 @@ static int decode_attr_files_free(struct xdr_stream *xdr, uint32_t *bitmap, uint | |||
2807 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_FREE - 1U))) | 2823 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_FREE - 1U))) |
2808 | return -EIO; | 2824 | return -EIO; |
2809 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_FREE)) { | 2825 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_FREE)) { |
2810 | READ_BUF(8); | 2826 | p = xdr_inline_decode(xdr, 8); |
2811 | READ64(*res); | 2827 | if (unlikely(!p)) |
2828 | goto out_overflow; | ||
2829 | xdr_decode_hyper(p, res); | ||
2812 | bitmap[0] &= ~FATTR4_WORD0_FILES_FREE; | 2830 | bitmap[0] &= ~FATTR4_WORD0_FILES_FREE; |
2813 | } | 2831 | } |
2814 | dprintk("%s: files free=%Lu\n", __func__, (unsigned long long)*res); | 2832 | dprintk("%s: files free=%Lu\n", __func__, (unsigned long long)*res); |
2815 | return status; | 2833 | return status; |
2834 | out_overflow: | ||
2835 | print_overflow_msg(__func__, xdr); | ||
2836 | return -EIO; | ||
2816 | } | 2837 | } |
2817 | 2838 | ||
2818 | static int decode_attr_files_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 2839 | static int decode_attr_files_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -2824,12 +2845,17 @@ static int decode_attr_files_total(struct xdr_stream *xdr, uint32_t *bitmap, uin | |||
2824 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_TOTAL - 1U))) | 2845 | if (unlikely(bitmap[0] & (FATTR4_WORD0_FILES_TOTAL - 1U))) |
2825 | return -EIO; | 2846 | return -EIO; |
2826 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_TOTAL)) { | 2847 | if (likely(bitmap[0] & FATTR4_WORD0_FILES_TOTAL)) { |
2827 | READ_BUF(8); | 2848 | p = xdr_inline_decode(xdr, 8); |
2828 | READ64(*res); | 2849 | if (unlikely(!p)) |
2850 | goto out_overflow; | ||
2851 | xdr_decode_hyper(p, res); | ||
2829 | bitmap[0] &= ~FATTR4_WORD0_FILES_TOTAL; | 2852 | bitmap[0] &= ~FATTR4_WORD0_FILES_TOTAL; |
2830 | } | 2853 | } |
2831 | dprintk("%s: files total=%Lu\n", __func__, (unsigned long long)*res); | 2854 | dprintk("%s: files total=%Lu\n", __func__, (unsigned long long)*res); |
2832 | return status; | 2855 | return status; |
2856 | out_overflow: | ||
2857 | print_overflow_msg(__func__, xdr); | ||
2858 | return -EIO; | ||
2833 | } | 2859 | } |
2834 | 2860 | ||
2835 | static int decode_pathname(struct xdr_stream *xdr, struct nfs4_pathname *path) | 2861 | static int decode_pathname(struct xdr_stream *xdr, struct nfs4_pathname *path) |
@@ -2838,8 +2864,10 @@ static int decode_pathname(struct xdr_stream *xdr, struct nfs4_pathname *path) | |||
2838 | __be32 *p; | 2864 | __be32 *p; |
2839 | int status = 0; | 2865 | int status = 0; |
2840 | 2866 | ||
2841 | READ_BUF(4); | 2867 | p = xdr_inline_decode(xdr, 4); |
2842 | READ32(n); | 2868 | if (unlikely(!p)) |
2869 | goto out_overflow; | ||
2870 | n = be32_to_cpup(p); | ||
2843 | if (n == 0) | 2871 | if (n == 0) |
2844 | goto root_path; | 2872 | goto root_path; |
2845 | dprintk("path "); | 2873 | dprintk("path "); |
@@ -2873,6 +2901,9 @@ out_eio: | |||
2873 | dprintk(" status %d", status); | 2901 | dprintk(" status %d", status); |
2874 | status = -EIO; | 2902 | status = -EIO; |
2875 | goto out; | 2903 | goto out; |
2904 | out_overflow: | ||
2905 | print_overflow_msg(__func__, xdr); | ||
2906 | return -EIO; | ||
2876 | } | 2907 | } |
2877 | 2908 | ||
2878 | static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_fs_locations *res) | 2909 | static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs4_fs_locations *res) |
@@ -2890,8 +2921,10 @@ static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, st | |||
2890 | status = decode_pathname(xdr, &res->fs_path); | 2921 | status = decode_pathname(xdr, &res->fs_path); |
2891 | if (unlikely(status != 0)) | 2922 | if (unlikely(status != 0)) |
2892 | goto out; | 2923 | goto out; |
2893 | READ_BUF(4); | 2924 | p = xdr_inline_decode(xdr, 4); |
2894 | READ32(n); | 2925 | if (unlikely(!p)) |
2926 | goto out_overflow; | ||
2927 | n = be32_to_cpup(p); | ||
2895 | if (n <= 0) | 2928 | if (n <= 0) |
2896 | goto out_eio; | 2929 | goto out_eio; |
2897 | res->nlocations = 0; | 2930 | res->nlocations = 0; |
@@ -2899,8 +2932,10 @@ static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, st | |||
2899 | u32 m; | 2932 | u32 m; |
2900 | struct nfs4_fs_location *loc = &res->locations[res->nlocations]; | 2933 | struct nfs4_fs_location *loc = &res->locations[res->nlocations]; |
2901 | 2934 | ||
2902 | READ_BUF(4); | 2935 | p = xdr_inline_decode(xdr, 4); |
2903 | READ32(m); | 2936 | if (unlikely(!p)) |
2937 | goto out_overflow; | ||
2938 | m = be32_to_cpup(p); | ||
2904 | 2939 | ||
2905 | loc->nservers = 0; | 2940 | loc->nservers = 0; |
2906 | dprintk("%s: servers ", __func__); | 2941 | dprintk("%s: servers ", __func__); |
@@ -2939,6 +2974,8 @@ static int decode_attr_fs_locations(struct xdr_stream *xdr, uint32_t *bitmap, st | |||
2939 | out: | 2974 | out: |
2940 | dprintk("%s: fs_locations done, error = %d\n", __func__, status); | 2975 | dprintk("%s: fs_locations done, error = %d\n", __func__, status); |
2941 | return status; | 2976 | return status; |
2977 | out_overflow: | ||
2978 | print_overflow_msg(__func__, xdr); | ||
2942 | out_eio: | 2979 | out_eio: |
2943 | status = -EIO; | 2980 | status = -EIO; |
2944 | goto out; | 2981 | goto out; |
@@ -2953,12 +2990,17 @@ static int decode_attr_maxfilesize(struct xdr_stream *xdr, uint32_t *bitmap, uin | |||
2953 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXFILESIZE - 1U))) | 2990 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXFILESIZE - 1U))) |
2954 | return -EIO; | 2991 | return -EIO; |
2955 | if (likely(bitmap[0] & FATTR4_WORD0_MAXFILESIZE)) { | 2992 | if (likely(bitmap[0] & FATTR4_WORD0_MAXFILESIZE)) { |
2956 | READ_BUF(8); | 2993 | p = xdr_inline_decode(xdr, 8); |
2957 | READ64(*res); | 2994 | if (unlikely(!p)) |
2995 | goto out_overflow; | ||
2996 | xdr_decode_hyper(p, res); | ||
2958 | bitmap[0] &= ~FATTR4_WORD0_MAXFILESIZE; | 2997 | bitmap[0] &= ~FATTR4_WORD0_MAXFILESIZE; |
2959 | } | 2998 | } |
2960 | dprintk("%s: maxfilesize=%Lu\n", __func__, (unsigned long long)*res); | 2999 | dprintk("%s: maxfilesize=%Lu\n", __func__, (unsigned long long)*res); |
2961 | return status; | 3000 | return status; |
3001 | out_overflow: | ||
3002 | print_overflow_msg(__func__, xdr); | ||
3003 | return -EIO; | ||
2962 | } | 3004 | } |
2963 | 3005 | ||
2964 | static int decode_attr_maxlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxlink) | 3006 | static int decode_attr_maxlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxlink) |
@@ -2970,12 +3012,17 @@ static int decode_attr_maxlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_ | |||
2970 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXLINK - 1U))) | 3012 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXLINK - 1U))) |
2971 | return -EIO; | 3013 | return -EIO; |
2972 | if (likely(bitmap[0] & FATTR4_WORD0_MAXLINK)) { | 3014 | if (likely(bitmap[0] & FATTR4_WORD0_MAXLINK)) { |
2973 | READ_BUF(4); | 3015 | p = xdr_inline_decode(xdr, 4); |
2974 | READ32(*maxlink); | 3016 | if (unlikely(!p)) |
3017 | goto out_overflow; | ||
3018 | *maxlink = be32_to_cpup(p); | ||
2975 | bitmap[0] &= ~FATTR4_WORD0_MAXLINK; | 3019 | bitmap[0] &= ~FATTR4_WORD0_MAXLINK; |
2976 | } | 3020 | } |
2977 | dprintk("%s: maxlink=%u\n", __func__, *maxlink); | 3021 | dprintk("%s: maxlink=%u\n", __func__, *maxlink); |
2978 | return status; | 3022 | return status; |
3023 | out_overflow: | ||
3024 | print_overflow_msg(__func__, xdr); | ||
3025 | return -EIO; | ||
2979 | } | 3026 | } |
2980 | 3027 | ||
2981 | static int decode_attr_maxname(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxname) | 3028 | static int decode_attr_maxname(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *maxname) |
@@ -2987,12 +3034,17 @@ static int decode_attr_maxname(struct xdr_stream *xdr, uint32_t *bitmap, uint32_ | |||
2987 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXNAME - 1U))) | 3034 | if (unlikely(bitmap[0] & (FATTR4_WORD0_MAXNAME - 1U))) |
2988 | return -EIO; | 3035 | return -EIO; |
2989 | if (likely(bitmap[0] & FATTR4_WORD0_MAXNAME)) { | 3036 | if (likely(bitmap[0] & FATTR4_WORD0_MAXNAME)) { |
2990 | READ_BUF(4); | 3037 | p = xdr_inline_decode(xdr, 4); |
2991 | READ32(*maxname); | 3038 | if (unlikely(!p)) |
3039 | goto out_overflow; | ||
3040 | *maxname = be32_to_cpup(p); | ||
2992 | bitmap[0] &= ~FATTR4_WORD0_MAXNAME; | 3041 | bitmap[0] &= ~FATTR4_WORD0_MAXNAME; |
2993 | } | 3042 | } |
2994 | dprintk("%s: maxname=%u\n", __func__, *maxname); | 3043 | dprintk("%s: maxname=%u\n", __func__, *maxname); |
2995 | return status; | 3044 | return status; |
3045 | out_overflow: | ||
3046 | print_overflow_msg(__func__, xdr); | ||
3047 | return -EIO; | ||
2996 | } | 3048 | } |
2997 | 3049 | ||
2998 | static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 3050 | static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -3005,8 +3057,10 @@ static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_ | |||
3005 | return -EIO; | 3057 | return -EIO; |
3006 | if (likely(bitmap[0] & FATTR4_WORD0_MAXREAD)) { | 3058 | if (likely(bitmap[0] & FATTR4_WORD0_MAXREAD)) { |
3007 | uint64_t maxread; | 3059 | uint64_t maxread; |
3008 | READ_BUF(8); | 3060 | p = xdr_inline_decode(xdr, 8); |
3009 | READ64(maxread); | 3061 | if (unlikely(!p)) |
3062 | goto out_overflow; | ||
3063 | xdr_decode_hyper(p, &maxread); | ||
3010 | if (maxread > 0x7FFFFFFF) | 3064 | if (maxread > 0x7FFFFFFF) |
3011 | maxread = 0x7FFFFFFF; | 3065 | maxread = 0x7FFFFFFF; |
3012 | *res = (uint32_t)maxread; | 3066 | *res = (uint32_t)maxread; |
@@ -3014,6 +3068,9 @@ static int decode_attr_maxread(struct xdr_stream *xdr, uint32_t *bitmap, uint32_ | |||
3014 | } | 3068 | } |
3015 | dprintk("%s: maxread=%lu\n", __func__, (unsigned long)*res); | 3069 | dprintk("%s: maxread=%lu\n", __func__, (unsigned long)*res); |
3016 | return status; | 3070 | return status; |
3071 | out_overflow: | ||
3072 | print_overflow_msg(__func__, xdr); | ||
3073 | return -EIO; | ||
3017 | } | 3074 | } |
3018 | 3075 | ||
3019 | static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) | 3076 | static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *res) |
@@ -3026,8 +3083,10 @@ static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32 | |||
3026 | return -EIO; | 3083 | return -EIO; |
3027 | if (likely(bitmap[0] & FATTR4_WORD0_MAXWRITE)) { | 3084 | if (likely(bitmap[0] & FATTR4_WORD0_MAXWRITE)) { |
3028 | uint64_t maxwrite; | 3085 | uint64_t maxwrite; |
3029 | READ_BUF(8); | 3086 | p = xdr_inline_decode(xdr, 8); |
3030 | READ64(maxwrite); | 3087 | if (unlikely(!p)) |
3088 | goto out_overflow; | ||
3089 | xdr_decode_hyper(p, &maxwrite); | ||
3031 | if (maxwrite > 0x7FFFFFFF) | 3090 | if (maxwrite > 0x7FFFFFFF) |
3032 | maxwrite = 0x7FFFFFFF; | 3091 | maxwrite = 0x7FFFFFFF; |
3033 | *res = (uint32_t)maxwrite; | 3092 | *res = (uint32_t)maxwrite; |
@@ -3035,6 +3094,9 @@ static int decode_attr_maxwrite(struct xdr_stream *xdr, uint32_t *bitmap, uint32 | |||
3035 | } | 3094 | } |
3036 | dprintk("%s: maxwrite=%lu\n", __func__, (unsigned long)*res); | 3095 | dprintk("%s: maxwrite=%lu\n", __func__, (unsigned long)*res); |
3037 | return status; | 3096 | return status; |
3097 | out_overflow: | ||
3098 | print_overflow_msg(__func__, xdr); | ||
3099 | return -EIO; | ||
3038 | } | 3100 | } |
3039 | 3101 | ||
3040 | static int decode_attr_mode(struct xdr_stream *xdr, uint32_t *bitmap, umode_t *mode) | 3102 | static int decode_attr_mode(struct xdr_stream *xdr, uint32_t *bitmap, umode_t *mode) |
@@ -3047,14 +3109,19 @@ static int decode_attr_mode(struct xdr_stream *xdr, uint32_t *bitmap, umode_t *m | |||
3047 | if (unlikely(bitmap[1] & (FATTR4_WORD1_MODE - 1U))) | 3109 | if (unlikely(bitmap[1] & (FATTR4_WORD1_MODE - 1U))) |
3048 | return -EIO; | 3110 | return -EIO; |
3049 | if (likely(bitmap[1] & FATTR4_WORD1_MODE)) { | 3111 | if (likely(bitmap[1] & FATTR4_WORD1_MODE)) { |
3050 | READ_BUF(4); | 3112 | p = xdr_inline_decode(xdr, 4); |
3051 | READ32(tmp); | 3113 | if (unlikely(!p)) |
3114 | goto out_overflow; | ||
3115 | tmp = be32_to_cpup(p); | ||
3052 | *mode = tmp & ~S_IFMT; | 3116 | *mode = tmp & ~S_IFMT; |
3053 | bitmap[1] &= ~FATTR4_WORD1_MODE; | 3117 | bitmap[1] &= ~FATTR4_WORD1_MODE; |
3054 | ret = NFS_ATTR_FATTR_MODE; | 3118 | ret = NFS_ATTR_FATTR_MODE; |
3055 | } | 3119 | } |
3056 | dprintk("%s: file mode=0%o\n", __func__, (unsigned int)*mode); | 3120 | dprintk("%s: file mode=0%o\n", __func__, (unsigned int)*mode); |
3057 | return ret; | 3121 | return ret; |
3122 | out_overflow: | ||
3123 | print_overflow_msg(__func__, xdr); | ||
3124 | return -EIO; | ||
3058 | } | 3125 | } |
3059 | 3126 | ||
3060 | static int decode_attr_nlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *nlink) | 3127 | static int decode_attr_nlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t *nlink) |
@@ -3066,16 +3133,22 @@ static int decode_attr_nlink(struct xdr_stream *xdr, uint32_t *bitmap, uint32_t | |||
3066 | if (unlikely(bitmap[1] & (FATTR4_WORD1_NUMLINKS - 1U))) | 3133 | if (unlikely(bitmap[1] & (FATTR4_WORD1_NUMLINKS - 1U))) |
3067 | return -EIO; | 3134 | return -EIO; |
3068 | if (likely(bitmap[1] & FATTR4_WORD1_NUMLINKS)) { | 3135 | if (likely(bitmap[1] & FATTR4_WORD1_NUMLINKS)) { |
3069 | READ_BUF(4); | 3136 | p = xdr_inline_decode(xdr, 4); |
3070 | READ32(*nlink); | 3137 | if (unlikely(!p)) |
3138 | goto out_overflow; | ||
3139 | *nlink = be32_to_cpup(p); | ||
3071 | bitmap[1] &= ~FATTR4_WORD1_NUMLINKS; | 3140 | bitmap[1] &= ~FATTR4_WORD1_NUMLINKS; |
3072 | ret = NFS_ATTR_FATTR_NLINK; | 3141 | ret = NFS_ATTR_FATTR_NLINK; |
3073 | } | 3142 | } |
3074 | dprintk("%s: nlink=%u\n", __func__, (unsigned int)*nlink); | 3143 | dprintk("%s: nlink=%u\n", __func__, (unsigned int)*nlink); |
3075 | return ret; | 3144 | return ret; |
3145 | out_overflow: | ||
3146 | print_overflow_msg(__func__, xdr); | ||
3147 | return -EIO; | ||
3076 | } | 3148 | } |
3077 | 3149 | ||
3078 | static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_client *clp, uint32_t *uid) | 3150 | static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, |
3151 | struct nfs_client *clp, uint32_t *uid, int may_sleep) | ||
3079 | { | 3152 | { |
3080 | uint32_t len; | 3153 | uint32_t len; |
3081 | __be32 *p; | 3154 | __be32 *p; |
@@ -3085,10 +3158,16 @@ static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, struct nf | |||
3085 | if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER - 1U))) | 3158 | if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER - 1U))) |
3086 | return -EIO; | 3159 | return -EIO; |
3087 | if (likely(bitmap[1] & FATTR4_WORD1_OWNER)) { | 3160 | if (likely(bitmap[1] & FATTR4_WORD1_OWNER)) { |
3088 | READ_BUF(4); | 3161 | p = xdr_inline_decode(xdr, 4); |
3089 | READ32(len); | 3162 | if (unlikely(!p)) |
3090 | READ_BUF(len); | 3163 | goto out_overflow; |
3091 | if (len < XDR_MAX_NETOBJ) { | 3164 | len = be32_to_cpup(p); |
3165 | p = xdr_inline_decode(xdr, len); | ||
3166 | if (unlikely(!p)) | ||
3167 | goto out_overflow; | ||
3168 | if (!may_sleep) { | ||
3169 | /* do nothing */ | ||
3170 | } else if (len < XDR_MAX_NETOBJ) { | ||
3092 | if (nfs_map_name_to_uid(clp, (char *)p, len, uid) == 0) | 3171 | if (nfs_map_name_to_uid(clp, (char *)p, len, uid) == 0) |
3093 | ret = NFS_ATTR_FATTR_OWNER; | 3172 | ret = NFS_ATTR_FATTR_OWNER; |
3094 | else | 3173 | else |
@@ -3101,9 +3180,13 @@ static int decode_attr_owner(struct xdr_stream *xdr, uint32_t *bitmap, struct nf | |||
3101 | } | 3180 | } |
3102 | dprintk("%s: uid=%d\n", __func__, (int)*uid); | 3181 | dprintk("%s: uid=%d\n", __func__, (int)*uid); |
3103 | return ret; | 3182 | return ret; |
3183 | out_overflow: | ||
3184 | print_overflow_msg(__func__, xdr); | ||
3185 | return -EIO; | ||
3104 | } | 3186 | } |
3105 | 3187 | ||
3106 | static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, struct nfs_client *clp, uint32_t *gid) | 3188 | static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, |
3189 | struct nfs_client *clp, uint32_t *gid, int may_sleep) | ||
3107 | { | 3190 | { |
3108 | uint32_t len; | 3191 | uint32_t len; |
3109 | __be32 *p; | 3192 | __be32 *p; |
@@ -3113,10 +3196,16 @@ static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, struct nf | |||
3113 | if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER_GROUP - 1U))) | 3196 | if (unlikely(bitmap[1] & (FATTR4_WORD1_OWNER_GROUP - 1U))) |
3114 | return -EIO; | 3197 | return -EIO; |
3115 | if (likely(bitmap[1] & FATTR4_WORD1_OWNER_GROUP)) { | 3198 | if (likely(bitmap[1] & FATTR4_WORD1_OWNER_GROUP)) { |
3116 | READ_BUF(4); | 3199 | p = xdr_inline_decode(xdr, 4); |
3117 | READ32(len); | 3200 | if (unlikely(!p)) |
3118 | READ_BUF(len); | 3201 | goto out_overflow; |
3119 | if (len < XDR_MAX_NETOBJ) { | 3202 | len = be32_to_cpup(p); |
3203 | p = xdr_inline_decode(xdr, len); | ||
3204 | if (unlikely(!p)) | ||
3205 | goto out_overflow; | ||
3206 | if (!may_sleep) { | ||
3207 | /* do nothing */ | ||
3208 | } else if (len < XDR_MAX_NETOBJ) { | ||
3120 | if (nfs_map_group_to_gid(clp, (char *)p, len, gid) == 0) | 3209 | if (nfs_map_group_to_gid(clp, (char *)p, len, gid) == 0) |
3121 | ret = NFS_ATTR_FATTR_GROUP; | 3210 | ret = NFS_ATTR_FATTR_GROUP; |
3122 | else | 3211 | else |
@@ -3129,6 +3218,9 @@ static int decode_attr_group(struct xdr_stream *xdr, uint32_t *bitmap, struct nf | |||
3129 | } | 3218 | } |
3130 | dprintk("%s: gid=%d\n", __func__, (int)*gid); | 3219 | dprintk("%s: gid=%d\n", __func__, (int)*gid); |
3131 | return ret; | 3220 | return ret; |
3221 | out_overflow: | ||
3222 | print_overflow_msg(__func__, xdr); | ||
3223 | return -EIO; | ||
3132 | } | 3224 | } |
3133 | 3225 | ||
3134 | static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rdev) | 3226 | static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rdev) |
@@ -3143,9 +3235,11 @@ static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rde | |||
3143 | if (likely(bitmap[1] & FATTR4_WORD1_RAWDEV)) { | 3235 | if (likely(bitmap[1] & FATTR4_WORD1_RAWDEV)) { |
3144 | dev_t tmp; | 3236 | dev_t tmp; |
3145 | 3237 | ||
3146 | READ_BUF(8); | 3238 | p = xdr_inline_decode(xdr, 8); |
3147 | READ32(major); | 3239 | if (unlikely(!p)) |
3148 | READ32(minor); | 3240 | goto out_overflow; |
3241 | major = be32_to_cpup(p++); | ||
3242 | minor = be32_to_cpup(p); | ||
3149 | tmp = MKDEV(major, minor); | 3243 | tmp = MKDEV(major, minor); |
3150 | if (MAJOR(tmp) == major && MINOR(tmp) == minor) | 3244 | if (MAJOR(tmp) == major && MINOR(tmp) == minor) |
3151 | *rdev = tmp; | 3245 | *rdev = tmp; |
@@ -3154,6 +3248,9 @@ static int decode_attr_rdev(struct xdr_stream *xdr, uint32_t *bitmap, dev_t *rde | |||
3154 | } | 3248 | } |
3155 | dprintk("%s: rdev=(0x%x:0x%x)\n", __func__, major, minor); | 3249 | dprintk("%s: rdev=(0x%x:0x%x)\n", __func__, major, minor); |
3156 | return ret; | 3250 | return ret; |
3251 | out_overflow: | ||
3252 | print_overflow_msg(__func__, xdr); | ||
3253 | return -EIO; | ||
3157 | } | 3254 | } |
3158 | 3255 | ||
3159 | static int decode_attr_space_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 3256 | static int decode_attr_space_avail(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -3165,12 +3262,17 @@ static int decode_attr_space_avail(struct xdr_stream *xdr, uint32_t *bitmap, uin | |||
3165 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_AVAIL - 1U))) | 3262 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_AVAIL - 1U))) |
3166 | return -EIO; | 3263 | return -EIO; |
3167 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_AVAIL)) { | 3264 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_AVAIL)) { |
3168 | READ_BUF(8); | 3265 | p = xdr_inline_decode(xdr, 8); |
3169 | READ64(*res); | 3266 | if (unlikely(!p)) |
3267 | goto out_overflow; | ||
3268 | xdr_decode_hyper(p, res); | ||
3170 | bitmap[1] &= ~FATTR4_WORD1_SPACE_AVAIL; | 3269 | bitmap[1] &= ~FATTR4_WORD1_SPACE_AVAIL; |
3171 | } | 3270 | } |
3172 | dprintk("%s: space avail=%Lu\n", __func__, (unsigned long long)*res); | 3271 | dprintk("%s: space avail=%Lu\n", __func__, (unsigned long long)*res); |
3173 | return status; | 3272 | return status; |
3273 | out_overflow: | ||
3274 | print_overflow_msg(__func__, xdr); | ||
3275 | return -EIO; | ||
3174 | } | 3276 | } |
3175 | 3277 | ||
3176 | static int decode_attr_space_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 3278 | static int decode_attr_space_free(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -3182,12 +3284,17 @@ static int decode_attr_space_free(struct xdr_stream *xdr, uint32_t *bitmap, uint | |||
3182 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_FREE - 1U))) | 3284 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_FREE - 1U))) |
3183 | return -EIO; | 3285 | return -EIO; |
3184 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_FREE)) { | 3286 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_FREE)) { |
3185 | READ_BUF(8); | 3287 | p = xdr_inline_decode(xdr, 8); |
3186 | READ64(*res); | 3288 | if (unlikely(!p)) |
3289 | goto out_overflow; | ||
3290 | xdr_decode_hyper(p, res); | ||
3187 | bitmap[1] &= ~FATTR4_WORD1_SPACE_FREE; | 3291 | bitmap[1] &= ~FATTR4_WORD1_SPACE_FREE; |
3188 | } | 3292 | } |
3189 | dprintk("%s: space free=%Lu\n", __func__, (unsigned long long)*res); | 3293 | dprintk("%s: space free=%Lu\n", __func__, (unsigned long long)*res); |
3190 | return status; | 3294 | return status; |
3295 | out_overflow: | ||
3296 | print_overflow_msg(__func__, xdr); | ||
3297 | return -EIO; | ||
3191 | } | 3298 | } |
3192 | 3299 | ||
3193 | static int decode_attr_space_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) | 3300 | static int decode_attr_space_total(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *res) |
@@ -3199,12 +3306,17 @@ static int decode_attr_space_total(struct xdr_stream *xdr, uint32_t *bitmap, uin | |||
3199 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_TOTAL - 1U))) | 3306 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_TOTAL - 1U))) |
3200 | return -EIO; | 3307 | return -EIO; |
3201 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_TOTAL)) { | 3308 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_TOTAL)) { |
3202 | READ_BUF(8); | 3309 | p = xdr_inline_decode(xdr, 8); |
3203 | READ64(*res); | 3310 | if (unlikely(!p)) |
3311 | goto out_overflow; | ||
3312 | xdr_decode_hyper(p, res); | ||
3204 | bitmap[1] &= ~FATTR4_WORD1_SPACE_TOTAL; | 3313 | bitmap[1] &= ~FATTR4_WORD1_SPACE_TOTAL; |
3205 | } | 3314 | } |
3206 | dprintk("%s: space total=%Lu\n", __func__, (unsigned long long)*res); | 3315 | dprintk("%s: space total=%Lu\n", __func__, (unsigned long long)*res); |
3207 | return status; | 3316 | return status; |
3317 | out_overflow: | ||
3318 | print_overflow_msg(__func__, xdr); | ||
3319 | return -EIO; | ||
3208 | } | 3320 | } |
3209 | 3321 | ||
3210 | static int decode_attr_space_used(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *used) | 3322 | static int decode_attr_space_used(struct xdr_stream *xdr, uint32_t *bitmap, uint64_t *used) |
@@ -3216,14 +3328,19 @@ static int decode_attr_space_used(struct xdr_stream *xdr, uint32_t *bitmap, uint | |||
3216 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_USED - 1U))) | 3328 | if (unlikely(bitmap[1] & (FATTR4_WORD1_SPACE_USED - 1U))) |
3217 | return -EIO; | 3329 | return -EIO; |
3218 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_USED)) { | 3330 | if (likely(bitmap[1] & FATTR4_WORD1_SPACE_USED)) { |
3219 | READ_BUF(8); | 3331 | p = xdr_inline_decode(xdr, 8); |
3220 | READ64(*used); | 3332 | if (unlikely(!p)) |
3333 | goto out_overflow; | ||
3334 | xdr_decode_hyper(p, used); | ||
3221 | bitmap[1] &= ~FATTR4_WORD1_SPACE_USED; | 3335 | bitmap[1] &= ~FATTR4_WORD1_SPACE_USED; |
3222 | ret = NFS_ATTR_FATTR_SPACE_USED; | 3336 | ret = NFS_ATTR_FATTR_SPACE_USED; |
3223 | } | 3337 | } |
3224 | dprintk("%s: space used=%Lu\n", __func__, | 3338 | dprintk("%s: space used=%Lu\n", __func__, |
3225 | (unsigned long long)*used); | 3339 | (unsigned long long)*used); |
3226 | return ret; | 3340 | return ret; |
3341 | out_overflow: | ||
3342 | print_overflow_msg(__func__, xdr); | ||
3343 | return -EIO; | ||
3227 | } | 3344 | } |
3228 | 3345 | ||
3229 | static int decode_attr_time(struct xdr_stream *xdr, struct timespec *time) | 3346 | static int decode_attr_time(struct xdr_stream *xdr, struct timespec *time) |
@@ -3232,12 +3349,17 @@ static int decode_attr_time(struct xdr_stream *xdr, struct timespec *time) | |||
3232 | uint64_t sec; | 3349 | uint64_t sec; |
3233 | uint32_t nsec; | 3350 | uint32_t nsec; |
3234 | 3351 | ||
3235 | READ_BUF(12); | 3352 | p = xdr_inline_decode(xdr, 12); |
3236 | READ64(sec); | 3353 | if (unlikely(!p)) |
3237 | READ32(nsec); | 3354 | goto out_overflow; |
3355 | p = xdr_decode_hyper(p, &sec); | ||
3356 | nsec = be32_to_cpup(p); | ||
3238 | time->tv_sec = (time_t)sec; | 3357 | time->tv_sec = (time_t)sec; |
3239 | time->tv_nsec = (long)nsec; | 3358 | time->tv_nsec = (long)nsec; |
3240 | return 0; | 3359 | return 0; |
3360 | out_overflow: | ||
3361 | print_overflow_msg(__func__, xdr); | ||
3362 | return -EIO; | ||
3241 | } | 3363 | } |
3242 | 3364 | ||
3243 | static int decode_attr_time_access(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time) | 3365 | static int decode_attr_time_access(struct xdr_stream *xdr, uint32_t *bitmap, struct timespec *time) |
@@ -3315,11 +3437,16 @@ static int decode_change_info(struct xdr_stream *xdr, struct nfs4_change_info *c | |||
3315 | { | 3437 | { |
3316 | __be32 *p; | 3438 | __be32 *p; |
3317 | 3439 | ||
3318 | READ_BUF(20); | 3440 | p = xdr_inline_decode(xdr, 20); |
3319 | READ32(cinfo->atomic); | 3441 | if (unlikely(!p)) |
3320 | READ64(cinfo->before); | 3442 | goto out_overflow; |
3321 | READ64(cinfo->after); | 3443 | cinfo->atomic = be32_to_cpup(p++); |
3444 | p = xdr_decode_hyper(p, &cinfo->before); | ||
3445 | xdr_decode_hyper(p, &cinfo->after); | ||
3322 | return 0; | 3446 | return 0; |
3447 | out_overflow: | ||
3448 | print_overflow_msg(__func__, xdr); | ||
3449 | return -EIO; | ||
3323 | } | 3450 | } |
3324 | 3451 | ||
3325 | static int decode_access(struct xdr_stream *xdr, struct nfs4_accessres *access) | 3452 | static int decode_access(struct xdr_stream *xdr, struct nfs4_accessres *access) |
@@ -3331,40 +3458,62 @@ static int decode_access(struct xdr_stream *xdr, struct nfs4_accessres *access) | |||
3331 | status = decode_op_hdr(xdr, OP_ACCESS); | 3458 | status = decode_op_hdr(xdr, OP_ACCESS); |
3332 | if (status) | 3459 | if (status) |
3333 | return status; | 3460 | return status; |
3334 | READ_BUF(8); | 3461 | p = xdr_inline_decode(xdr, 8); |
3335 | READ32(supp); | 3462 | if (unlikely(!p)) |
3336 | READ32(acc); | 3463 | goto out_overflow; |
3464 | supp = be32_to_cpup(p++); | ||
3465 | acc = be32_to_cpup(p); | ||
3337 | access->supported = supp; | 3466 | access->supported = supp; |
3338 | access->access = acc; | 3467 | access->access = acc; |
3339 | return 0; | 3468 | return 0; |
3469 | out_overflow: | ||
3470 | print_overflow_msg(__func__, xdr); | ||
3471 | return -EIO; | ||
3340 | } | 3472 | } |
3341 | 3473 | ||
3342 | static int decode_close(struct xdr_stream *xdr, struct nfs_closeres *res) | 3474 | static int decode_opaque_fixed(struct xdr_stream *xdr, void *buf, size_t len) |
3343 | { | 3475 | { |
3344 | __be32 *p; | 3476 | __be32 *p; |
3477 | |||
3478 | p = xdr_inline_decode(xdr, len); | ||
3479 | if (likely(p)) { | ||
3480 | memcpy(buf, p, len); | ||
3481 | return 0; | ||
3482 | } | ||
3483 | print_overflow_msg(__func__, xdr); | ||
3484 | return -EIO; | ||
3485 | } | ||
3486 | |||
3487 | static int decode_stateid(struct xdr_stream *xdr, nfs4_stateid *stateid) | ||
3488 | { | ||
3489 | return decode_opaque_fixed(xdr, stateid->data, NFS4_STATEID_SIZE); | ||
3490 | } | ||
3491 | |||
3492 | static int decode_close(struct xdr_stream *xdr, struct nfs_closeres *res) | ||
3493 | { | ||
3345 | int status; | 3494 | int status; |
3346 | 3495 | ||
3347 | status = decode_op_hdr(xdr, OP_CLOSE); | 3496 | status = decode_op_hdr(xdr, OP_CLOSE); |
3348 | if (status != -EIO) | 3497 | if (status != -EIO) |
3349 | nfs_increment_open_seqid(status, res->seqid); | 3498 | nfs_increment_open_seqid(status, res->seqid); |
3350 | if (status) | 3499 | if (!status) |
3351 | return status; | 3500 | status = decode_stateid(xdr, &res->stateid); |
3352 | READ_BUF(NFS4_STATEID_SIZE); | 3501 | return status; |
3353 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | 3502 | } |
3354 | return 0; | 3503 | |
3504 | static int decode_verifier(struct xdr_stream *xdr, void *verifier) | ||
3505 | { | ||
3506 | return decode_opaque_fixed(xdr, verifier, 8); | ||
3355 | } | 3507 | } |
3356 | 3508 | ||
3357 | static int decode_commit(struct xdr_stream *xdr, struct nfs_writeres *res) | 3509 | static int decode_commit(struct xdr_stream *xdr, struct nfs_writeres *res) |
3358 | { | 3510 | { |
3359 | __be32 *p; | ||
3360 | int status; | 3511 | int status; |
3361 | 3512 | ||
3362 | status = decode_op_hdr(xdr, OP_COMMIT); | 3513 | status = decode_op_hdr(xdr, OP_COMMIT); |
3363 | if (status) | 3514 | if (!status) |
3364 | return status; | 3515 | status = decode_verifier(xdr, res->verf->verifier); |
3365 | READ_BUF(8); | 3516 | return status; |
3366 | COPYMEM(res->verf->verifier, 8); | ||
3367 | return 0; | ||
3368 | } | 3517 | } |
3369 | 3518 | ||
3370 | static int decode_create(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) | 3519 | static int decode_create(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) |
@@ -3378,10 +3527,16 @@ static int decode_create(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) | |||
3378 | return status; | 3527 | return status; |
3379 | if ((status = decode_change_info(xdr, cinfo))) | 3528 | if ((status = decode_change_info(xdr, cinfo))) |
3380 | return status; | 3529 | return status; |
3381 | READ_BUF(4); | 3530 | p = xdr_inline_decode(xdr, 4); |
3382 | READ32(bmlen); | 3531 | if (unlikely(!p)) |
3383 | READ_BUF(bmlen << 2); | 3532 | goto out_overflow; |
3384 | return 0; | 3533 | bmlen = be32_to_cpup(p); |
3534 | p = xdr_inline_decode(xdr, bmlen << 2); | ||
3535 | if (likely(p)) | ||
3536 | return 0; | ||
3537 | out_overflow: | ||
3538 | print_overflow_msg(__func__, xdr); | ||
3539 | return -EIO; | ||
3385 | } | 3540 | } |
3386 | 3541 | ||
3387 | static int decode_server_caps(struct xdr_stream *xdr, struct nfs4_server_caps_res *res) | 3542 | static int decode_server_caps(struct xdr_stream *xdr, struct nfs4_server_caps_res *res) |
@@ -3466,7 +3621,8 @@ xdr_error: | |||
3466 | return status; | 3621 | return status; |
3467 | } | 3622 | } |
3468 | 3623 | ||
3469 | static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr, const struct nfs_server *server) | 3624 | static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr, |
3625 | const struct nfs_server *server, int may_sleep) | ||
3470 | { | 3626 | { |
3471 | __be32 *savep; | 3627 | __be32 *savep; |
3472 | uint32_t attrlen, | 3628 | uint32_t attrlen, |
@@ -3538,12 +3694,14 @@ static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr, cons | |||
3538 | goto xdr_error; | 3694 | goto xdr_error; |
3539 | fattr->valid |= status; | 3695 | fattr->valid |= status; |
3540 | 3696 | ||
3541 | status = decode_attr_owner(xdr, bitmap, server->nfs_client, &fattr->uid); | 3697 | status = decode_attr_owner(xdr, bitmap, server->nfs_client, |
3698 | &fattr->uid, may_sleep); | ||
3542 | if (status < 0) | 3699 | if (status < 0) |
3543 | goto xdr_error; | 3700 | goto xdr_error; |
3544 | fattr->valid |= status; | 3701 | fattr->valid |= status; |
3545 | 3702 | ||
3546 | status = decode_attr_group(xdr, bitmap, server->nfs_client, &fattr->gid); | 3703 | status = decode_attr_group(xdr, bitmap, server->nfs_client, |
3704 | &fattr->gid, may_sleep); | ||
3547 | if (status < 0) | 3705 | if (status < 0) |
3548 | goto xdr_error; | 3706 | goto xdr_error; |
3549 | fattr->valid |= status; | 3707 | fattr->valid |= status; |
@@ -3633,14 +3791,21 @@ static int decode_getfh(struct xdr_stream *xdr, struct nfs_fh *fh) | |||
3633 | if (status) | 3791 | if (status) |
3634 | return status; | 3792 | return status; |
3635 | 3793 | ||
3636 | READ_BUF(4); | 3794 | p = xdr_inline_decode(xdr, 4); |
3637 | READ32(len); | 3795 | if (unlikely(!p)) |
3796 | goto out_overflow; | ||
3797 | len = be32_to_cpup(p); | ||
3638 | if (len > NFS4_FHSIZE) | 3798 | if (len > NFS4_FHSIZE) |
3639 | return -EIO; | 3799 | return -EIO; |
3640 | fh->size = len; | 3800 | fh->size = len; |
3641 | READ_BUF(len); | 3801 | p = xdr_inline_decode(xdr, len); |
3642 | COPYMEM(fh->data, len); | 3802 | if (unlikely(!p)) |
3803 | goto out_overflow; | ||
3804 | memcpy(fh->data, p, len); | ||
3643 | return 0; | 3805 | return 0; |
3806 | out_overflow: | ||
3807 | print_overflow_msg(__func__, xdr); | ||
3808 | return -EIO; | ||
3644 | } | 3809 | } |
3645 | 3810 | ||
3646 | static int decode_link(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) | 3811 | static int decode_link(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) |
@@ -3662,10 +3827,12 @@ static int decode_lock_denied (struct xdr_stream *xdr, struct file_lock *fl) | |||
3662 | __be32 *p; | 3827 | __be32 *p; |
3663 | uint32_t namelen, type; | 3828 | uint32_t namelen, type; |
3664 | 3829 | ||
3665 | READ_BUF(32); | 3830 | p = xdr_inline_decode(xdr, 32); |
3666 | READ64(offset); | 3831 | if (unlikely(!p)) |
3667 | READ64(length); | 3832 | goto out_overflow; |
3668 | READ32(type); | 3833 | p = xdr_decode_hyper(p, &offset); |
3834 | p = xdr_decode_hyper(p, &length); | ||
3835 | type = be32_to_cpup(p++); | ||
3669 | if (fl != NULL) { | 3836 | if (fl != NULL) { |
3670 | fl->fl_start = (loff_t)offset; | 3837 | fl->fl_start = (loff_t)offset; |
3671 | fl->fl_end = fl->fl_start + (loff_t)length - 1; | 3838 | fl->fl_end = fl->fl_start + (loff_t)length - 1; |
@@ -3676,23 +3843,27 @@ static int decode_lock_denied (struct xdr_stream *xdr, struct file_lock *fl) | |||
3676 | fl->fl_type = F_RDLCK; | 3843 | fl->fl_type = F_RDLCK; |
3677 | fl->fl_pid = 0; | 3844 | fl->fl_pid = 0; |
3678 | } | 3845 | } |
3679 | READ64(clientid); | 3846 | p = xdr_decode_hyper(p, &clientid); |
3680 | READ32(namelen); | 3847 | namelen = be32_to_cpup(p); |
3681 | READ_BUF(namelen); | 3848 | p = xdr_inline_decode(xdr, namelen); |
3682 | return -NFS4ERR_DENIED; | 3849 | if (likely(p)) |
3850 | return -NFS4ERR_DENIED; | ||
3851 | out_overflow: | ||
3852 | print_overflow_msg(__func__, xdr); | ||
3853 | return -EIO; | ||
3683 | } | 3854 | } |
3684 | 3855 | ||
3685 | static int decode_lock(struct xdr_stream *xdr, struct nfs_lock_res *res) | 3856 | static int decode_lock(struct xdr_stream *xdr, struct nfs_lock_res *res) |
3686 | { | 3857 | { |
3687 | __be32 *p; | ||
3688 | int status; | 3858 | int status; |
3689 | 3859 | ||
3690 | status = decode_op_hdr(xdr, OP_LOCK); | 3860 | status = decode_op_hdr(xdr, OP_LOCK); |
3691 | if (status == -EIO) | 3861 | if (status == -EIO) |
3692 | goto out; | 3862 | goto out; |
3693 | if (status == 0) { | 3863 | if (status == 0) { |
3694 | READ_BUF(NFS4_STATEID_SIZE); | 3864 | status = decode_stateid(xdr, &res->stateid); |
3695 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | 3865 | if (unlikely(status)) |
3866 | goto out; | ||
3696 | } else if (status == -NFS4ERR_DENIED) | 3867 | } else if (status == -NFS4ERR_DENIED) |
3697 | status = decode_lock_denied(xdr, NULL); | 3868 | status = decode_lock_denied(xdr, NULL); |
3698 | if (res->open_seqid != NULL) | 3869 | if (res->open_seqid != NULL) |
@@ -3713,16 +3884,13 @@ static int decode_lockt(struct xdr_stream *xdr, struct nfs_lockt_res *res) | |||
3713 | 3884 | ||
3714 | static int decode_locku(struct xdr_stream *xdr, struct nfs_locku_res *res) | 3885 | static int decode_locku(struct xdr_stream *xdr, struct nfs_locku_res *res) |
3715 | { | 3886 | { |
3716 | __be32 *p; | ||
3717 | int status; | 3887 | int status; |
3718 | 3888 | ||
3719 | status = decode_op_hdr(xdr, OP_LOCKU); | 3889 | status = decode_op_hdr(xdr, OP_LOCKU); |
3720 | if (status != -EIO) | 3890 | if (status != -EIO) |
3721 | nfs_increment_lock_seqid(status, res->seqid); | 3891 | nfs_increment_lock_seqid(status, res->seqid); |
3722 | if (status == 0) { | 3892 | if (status == 0) |
3723 | READ_BUF(NFS4_STATEID_SIZE); | 3893 | status = decode_stateid(xdr, &res->stateid); |
3724 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | ||
3725 | } | ||
3726 | return status; | 3894 | return status; |
3727 | } | 3895 | } |
3728 | 3896 | ||
@@ -3737,34 +3905,46 @@ static int decode_space_limit(struct xdr_stream *xdr, u64 *maxsize) | |||
3737 | __be32 *p; | 3905 | __be32 *p; |
3738 | uint32_t limit_type, nblocks, blocksize; | 3906 | uint32_t limit_type, nblocks, blocksize; |
3739 | 3907 | ||
3740 | READ_BUF(12); | 3908 | p = xdr_inline_decode(xdr, 12); |
3741 | READ32(limit_type); | 3909 | if (unlikely(!p)) |
3910 | goto out_overflow; | ||
3911 | limit_type = be32_to_cpup(p++); | ||
3742 | switch (limit_type) { | 3912 | switch (limit_type) { |
3743 | case 1: | 3913 | case 1: |
3744 | READ64(*maxsize); | 3914 | xdr_decode_hyper(p, maxsize); |
3745 | break; | 3915 | break; |
3746 | case 2: | 3916 | case 2: |
3747 | READ32(nblocks); | 3917 | nblocks = be32_to_cpup(p++); |
3748 | READ32(blocksize); | 3918 | blocksize = be32_to_cpup(p); |
3749 | *maxsize = (uint64_t)nblocks * (uint64_t)blocksize; | 3919 | *maxsize = (uint64_t)nblocks * (uint64_t)blocksize; |
3750 | } | 3920 | } |
3751 | return 0; | 3921 | return 0; |
3922 | out_overflow: | ||
3923 | print_overflow_msg(__func__, xdr); | ||
3924 | return -EIO; | ||
3752 | } | 3925 | } |
3753 | 3926 | ||
3754 | static int decode_delegation(struct xdr_stream *xdr, struct nfs_openres *res) | 3927 | static int decode_delegation(struct xdr_stream *xdr, struct nfs_openres *res) |
3755 | { | 3928 | { |
3756 | __be32 *p; | 3929 | __be32 *p; |
3757 | uint32_t delegation_type; | 3930 | uint32_t delegation_type; |
3931 | int status; | ||
3758 | 3932 | ||
3759 | READ_BUF(4); | 3933 | p = xdr_inline_decode(xdr, 4); |
3760 | READ32(delegation_type); | 3934 | if (unlikely(!p)) |
3935 | goto out_overflow; | ||
3936 | delegation_type = be32_to_cpup(p); | ||
3761 | if (delegation_type == NFS4_OPEN_DELEGATE_NONE) { | 3937 | if (delegation_type == NFS4_OPEN_DELEGATE_NONE) { |
3762 | res->delegation_type = 0; | 3938 | res->delegation_type = 0; |
3763 | return 0; | 3939 | return 0; |
3764 | } | 3940 | } |
3765 | READ_BUF(NFS4_STATEID_SIZE+4); | 3941 | status = decode_stateid(xdr, &res->delegation); |
3766 | COPYMEM(res->delegation.data, NFS4_STATEID_SIZE); | 3942 | if (unlikely(status)) |
3767 | READ32(res->do_recall); | 3943 | return status; |
3944 | p = xdr_inline_decode(xdr, 4); | ||
3945 | if (unlikely(!p)) | ||
3946 | goto out_overflow; | ||
3947 | res->do_recall = be32_to_cpup(p); | ||
3768 | 3948 | ||
3769 | switch (delegation_type) { | 3949 | switch (delegation_type) { |
3770 | case NFS4_OPEN_DELEGATE_READ: | 3950 | case NFS4_OPEN_DELEGATE_READ: |
@@ -3776,6 +3956,9 @@ static int decode_delegation(struct xdr_stream *xdr, struct nfs_openres *res) | |||
3776 | return -EIO; | 3956 | return -EIO; |
3777 | } | 3957 | } |
3778 | return decode_ace(xdr, NULL, res->server->nfs_client); | 3958 | return decode_ace(xdr, NULL, res->server->nfs_client); |
3959 | out_overflow: | ||
3960 | print_overflow_msg(__func__, xdr); | ||
3961 | return -EIO; | ||
3779 | } | 3962 | } |
3780 | 3963 | ||
3781 | static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res) | 3964 | static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res) |
@@ -3787,23 +3970,27 @@ static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res) | |||
3787 | status = decode_op_hdr(xdr, OP_OPEN); | 3970 | status = decode_op_hdr(xdr, OP_OPEN); |
3788 | if (status != -EIO) | 3971 | if (status != -EIO) |
3789 | nfs_increment_open_seqid(status, res->seqid); | 3972 | nfs_increment_open_seqid(status, res->seqid); |
3790 | if (status) | 3973 | if (!status) |
3974 | status = decode_stateid(xdr, &res->stateid); | ||
3975 | if (unlikely(status)) | ||
3791 | return status; | 3976 | return status; |
3792 | READ_BUF(NFS4_STATEID_SIZE); | ||
3793 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | ||
3794 | 3977 | ||
3795 | decode_change_info(xdr, &res->cinfo); | 3978 | decode_change_info(xdr, &res->cinfo); |
3796 | 3979 | ||
3797 | READ_BUF(8); | 3980 | p = xdr_inline_decode(xdr, 8); |
3798 | READ32(res->rflags); | 3981 | if (unlikely(!p)) |
3799 | READ32(bmlen); | 3982 | goto out_overflow; |
3983 | res->rflags = be32_to_cpup(p++); | ||
3984 | bmlen = be32_to_cpup(p); | ||
3800 | if (bmlen > 10) | 3985 | if (bmlen > 10) |
3801 | goto xdr_error; | 3986 | goto xdr_error; |
3802 | 3987 | ||
3803 | READ_BUF(bmlen << 2); | 3988 | p = xdr_inline_decode(xdr, bmlen << 2); |
3989 | if (unlikely(!p)) | ||
3990 | goto out_overflow; | ||
3804 | savewords = min_t(uint32_t, bmlen, NFS4_BITMAP_SIZE); | 3991 | savewords = min_t(uint32_t, bmlen, NFS4_BITMAP_SIZE); |
3805 | for (i = 0; i < savewords; ++i) | 3992 | for (i = 0; i < savewords; ++i) |
3806 | READ32(res->attrset[i]); | 3993 | res->attrset[i] = be32_to_cpup(p++); |
3807 | for (; i < NFS4_BITMAP_SIZE; i++) | 3994 | for (; i < NFS4_BITMAP_SIZE; i++) |
3808 | res->attrset[i] = 0; | 3995 | res->attrset[i] = 0; |
3809 | 3996 | ||
@@ -3811,36 +3998,33 @@ static int decode_open(struct xdr_stream *xdr, struct nfs_openres *res) | |||
3811 | xdr_error: | 3998 | xdr_error: |
3812 | dprintk("%s: Bitmap too large! Length = %u\n", __func__, bmlen); | 3999 | dprintk("%s: Bitmap too large! Length = %u\n", __func__, bmlen); |
3813 | return -EIO; | 4000 | return -EIO; |
4001 | out_overflow: | ||
4002 | print_overflow_msg(__func__, xdr); | ||
4003 | return -EIO; | ||
3814 | } | 4004 | } |
3815 | 4005 | ||
3816 | static int decode_open_confirm(struct xdr_stream *xdr, struct nfs_open_confirmres *res) | 4006 | static int decode_open_confirm(struct xdr_stream *xdr, struct nfs_open_confirmres *res) |
3817 | { | 4007 | { |
3818 | __be32 *p; | ||
3819 | int status; | 4008 | int status; |
3820 | 4009 | ||
3821 | status = decode_op_hdr(xdr, OP_OPEN_CONFIRM); | 4010 | status = decode_op_hdr(xdr, OP_OPEN_CONFIRM); |
3822 | if (status != -EIO) | 4011 | if (status != -EIO) |
3823 | nfs_increment_open_seqid(status, res->seqid); | 4012 | nfs_increment_open_seqid(status, res->seqid); |
3824 | if (status) | 4013 | if (!status) |
3825 | return status; | 4014 | status = decode_stateid(xdr, &res->stateid); |
3826 | READ_BUF(NFS4_STATEID_SIZE); | 4015 | return status; |
3827 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | ||
3828 | return 0; | ||
3829 | } | 4016 | } |
3830 | 4017 | ||
3831 | static int decode_open_downgrade(struct xdr_stream *xdr, struct nfs_closeres *res) | 4018 | static int decode_open_downgrade(struct xdr_stream *xdr, struct nfs_closeres *res) |
3832 | { | 4019 | { |
3833 | __be32 *p; | ||
3834 | int status; | 4020 | int status; |
3835 | 4021 | ||
3836 | status = decode_op_hdr(xdr, OP_OPEN_DOWNGRADE); | 4022 | status = decode_op_hdr(xdr, OP_OPEN_DOWNGRADE); |
3837 | if (status != -EIO) | 4023 | if (status != -EIO) |
3838 | nfs_increment_open_seqid(status, res->seqid); | 4024 | nfs_increment_open_seqid(status, res->seqid); |
3839 | if (status) | 4025 | if (!status) |
3840 | return status; | 4026 | status = decode_stateid(xdr, &res->stateid); |
3841 | READ_BUF(NFS4_STATEID_SIZE); | 4027 | return status; |
3842 | COPYMEM(res->stateid.data, NFS4_STATEID_SIZE); | ||
3843 | return 0; | ||
3844 | } | 4028 | } |
3845 | 4029 | ||
3846 | static int decode_putfh(struct xdr_stream *xdr) | 4030 | static int decode_putfh(struct xdr_stream *xdr) |
@@ -3863,9 +4047,11 @@ static int decode_read(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs_ | |||
3863 | status = decode_op_hdr(xdr, OP_READ); | 4047 | status = decode_op_hdr(xdr, OP_READ); |
3864 | if (status) | 4048 | if (status) |
3865 | return status; | 4049 | return status; |
3866 | READ_BUF(8); | 4050 | p = xdr_inline_decode(xdr, 8); |
3867 | READ32(eof); | 4051 | if (unlikely(!p)) |
3868 | READ32(count); | 4052 | goto out_overflow; |
4053 | eof = be32_to_cpup(p++); | ||
4054 | count = be32_to_cpup(p); | ||
3869 | hdrlen = (u8 *) p - (u8 *) iov->iov_base; | 4055 | hdrlen = (u8 *) p - (u8 *) iov->iov_base; |
3870 | recvd = req->rq_rcv_buf.len - hdrlen; | 4056 | recvd = req->rq_rcv_buf.len - hdrlen; |
3871 | if (count > recvd) { | 4057 | if (count > recvd) { |
@@ -3878,6 +4064,9 @@ static int decode_read(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs_ | |||
3878 | res->eof = eof; | 4064 | res->eof = eof; |
3879 | res->count = count; | 4065 | res->count = count; |
3880 | return 0; | 4066 | return 0; |
4067 | out_overflow: | ||
4068 | print_overflow_msg(__func__, xdr); | ||
4069 | return -EIO; | ||
3881 | } | 4070 | } |
3882 | 4071 | ||
3883 | static int decode_readdir(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs4_readdir_res *readdir) | 4072 | static int decode_readdir(struct xdr_stream *xdr, struct rpc_rqst *req, struct nfs4_readdir_res *readdir) |
@@ -3892,17 +4081,17 @@ static int decode_readdir(struct xdr_stream *xdr, struct rpc_rqst *req, struct n | |||
3892 | int status; | 4081 | int status; |
3893 | 4082 | ||
3894 | status = decode_op_hdr(xdr, OP_READDIR); | 4083 | status = decode_op_hdr(xdr, OP_READDIR); |
3895 | if (status) | 4084 | if (!status) |
4085 | status = decode_verifier(xdr, readdir->verifier.data); | ||
4086 | if (unlikely(status)) | ||
3896 | return status; | 4087 | return status; |
3897 | READ_BUF(8); | ||
3898 | COPYMEM(readdir->verifier.data, 8); | ||
3899 | dprintk("%s: verifier = %08x:%08x\n", | 4088 | dprintk("%s: verifier = %08x:%08x\n", |
3900 | __func__, | 4089 | __func__, |
3901 | ((u32 *)readdir->verifier.data)[0], | 4090 | ((u32 *)readdir->verifier.data)[0], |
3902 | ((u32 *)readdir->verifier.data)[1]); | 4091 | ((u32 *)readdir->verifier.data)[1]); |
3903 | 4092 | ||
3904 | 4093 | ||
3905 | hdrlen = (char *) p - (char *) iov->iov_base; | 4094 | hdrlen = (char *) xdr->p - (char *) iov->iov_base; |
3906 | recvd = rcvbuf->len - hdrlen; | 4095 | recvd = rcvbuf->len - hdrlen; |
3907 | if (pglen > recvd) | 4096 | if (pglen > recvd) |
3908 | pglen = recvd; | 4097 | pglen = recvd; |
@@ -3990,8 +4179,10 @@ static int decode_readlink(struct xdr_stream *xdr, struct rpc_rqst *req) | |||
3990 | return status; | 4179 | return status; |
3991 | 4180 | ||
3992 | /* Convert length of symlink */ | 4181 | /* Convert length of symlink */ |
3993 | READ_BUF(4); | 4182 | p = xdr_inline_decode(xdr, 4); |
3994 | READ32(len); | 4183 | if (unlikely(!p)) |
4184 | goto out_overflow; | ||
4185 | len = be32_to_cpup(p); | ||
3995 | if (len >= rcvbuf->page_len || len <= 0) { | 4186 | if (len >= rcvbuf->page_len || len <= 0) { |
3996 | dprintk("nfs: server returned giant symlink!\n"); | 4187 | dprintk("nfs: server returned giant symlink!\n"); |
3997 | return -ENAMETOOLONG; | 4188 | return -ENAMETOOLONG; |
@@ -4015,6 +4206,9 @@ static int decode_readlink(struct xdr_stream *xdr, struct rpc_rqst *req) | |||
4015 | kaddr[len+rcvbuf->page_base] = '\0'; | 4206 | kaddr[len+rcvbuf->page_base] = '\0'; |
4016 | kunmap_atomic(kaddr, KM_USER0); | 4207 | kunmap_atomic(kaddr, KM_USER0); |
4017 | return 0; | 4208 | return 0; |
4209 | out_overflow: | ||
4210 | print_overflow_msg(__func__, xdr); | ||
4211 | return -EIO; | ||
4018 | } | 4212 | } |
4019 | 4213 | ||
4020 | static int decode_remove(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) | 4214 | static int decode_remove(struct xdr_stream *xdr, struct nfs4_change_info *cinfo) |
@@ -4112,10 +4306,16 @@ static int decode_setattr(struct xdr_stream *xdr) | |||
4112 | status = decode_op_hdr(xdr, OP_SETATTR); | 4306 | status = decode_op_hdr(xdr, OP_SETATTR); |
4113 | if (status) | 4307 | if (status) |
4114 | return status; | 4308 | return status; |
4115 | READ_BUF(4); | 4309 | p = xdr_inline_decode(xdr, 4); |
4116 | READ32(bmlen); | 4310 | if (unlikely(!p)) |
4117 | READ_BUF(bmlen << 2); | 4311 | goto out_overflow; |
4118 | return 0; | 4312 | bmlen = be32_to_cpup(p); |
4313 | p = xdr_inline_decode(xdr, bmlen << 2); | ||
4314 | if (likely(p)) | ||
4315 | return 0; | ||
4316 | out_overflow: | ||
4317 | print_overflow_msg(__func__, xdr); | ||
4318 | return -EIO; | ||
4119 | } | 4319 | } |
4120 | 4320 | ||
4121 | static int decode_setclientid(struct xdr_stream *xdr, struct nfs_client *clp) | 4321 | static int decode_setclientid(struct xdr_stream *xdr, struct nfs_client *clp) |
@@ -4124,35 +4324,50 @@ static int decode_setclientid(struct xdr_stream *xdr, struct nfs_client *clp) | |||
4124 | uint32_t opnum; | 4324 | uint32_t opnum; |
4125 | int32_t nfserr; | 4325 | int32_t nfserr; |
4126 | 4326 | ||
4127 | READ_BUF(8); | 4327 | p = xdr_inline_decode(xdr, 8); |
4128 | READ32(opnum); | 4328 | if (unlikely(!p)) |
4329 | goto out_overflow; | ||
4330 | opnum = be32_to_cpup(p++); | ||
4129 | if (opnum != OP_SETCLIENTID) { | 4331 | if (opnum != OP_SETCLIENTID) { |
4130 | dprintk("nfs: decode_setclientid: Server returned operation" | 4332 | dprintk("nfs: decode_setclientid: Server returned operation" |
4131 | " %d\n", opnum); | 4333 | " %d\n", opnum); |
4132 | return -EIO; | 4334 | return -EIO; |
4133 | } | 4335 | } |
4134 | READ32(nfserr); | 4336 | nfserr = be32_to_cpup(p); |
4135 | if (nfserr == NFS_OK) { | 4337 | if (nfserr == NFS_OK) { |
4136 | READ_BUF(8 + NFS4_VERIFIER_SIZE); | 4338 | p = xdr_inline_decode(xdr, 8 + NFS4_VERIFIER_SIZE); |
4137 | READ64(clp->cl_clientid); | 4339 | if (unlikely(!p)) |
4138 | COPYMEM(clp->cl_confirm.data, NFS4_VERIFIER_SIZE); | 4340 | goto out_overflow; |
4341 | p = xdr_decode_hyper(p, &clp->cl_clientid); | ||
4342 | memcpy(clp->cl_confirm.data, p, NFS4_VERIFIER_SIZE); | ||
4139 | } else if (nfserr == NFSERR_CLID_INUSE) { | 4343 | } else if (nfserr == NFSERR_CLID_INUSE) { |
4140 | uint32_t len; | 4344 | uint32_t len; |
4141 | 4345 | ||
4142 | /* skip netid string */ | 4346 | /* skip netid string */ |
4143 | READ_BUF(4); | 4347 | p = xdr_inline_decode(xdr, 4); |
4144 | READ32(len); | 4348 | if (unlikely(!p)) |
4145 | READ_BUF(len); | 4349 | goto out_overflow; |
4350 | len = be32_to_cpup(p); | ||
4351 | p = xdr_inline_decode(xdr, len); | ||
4352 | if (unlikely(!p)) | ||
4353 | goto out_overflow; | ||
4146 | 4354 | ||
4147 | /* skip uaddr string */ | 4355 | /* skip uaddr string */ |
4148 | READ_BUF(4); | 4356 | p = xdr_inline_decode(xdr, 4); |
4149 | READ32(len); | 4357 | if (unlikely(!p)) |
4150 | READ_BUF(len); | 4358 | goto out_overflow; |
4359 | len = be32_to_cpup(p); | ||
4360 | p = xdr_inline_decode(xdr, len); | ||
4361 | if (unlikely(!p)) | ||
4362 | goto out_overflow; | ||
4151 | return -NFSERR_CLID_INUSE; | 4363 | return -NFSERR_CLID_INUSE; |
4152 | } else | 4364 | } else |
4153 | return nfs4_stat_to_errno(nfserr); | 4365 | return nfs4_stat_to_errno(nfserr); |
4154 | 4366 | ||
4155 | return 0; | 4367 | return 0; |
4368 | out_overflow: | ||
4369 | print_overflow_msg(__func__, xdr); | ||
4370 | return -EIO; | ||
4156 | } | 4371 | } |
4157 | 4372 | ||
4158 | static int decode_setclientid_confirm(struct xdr_stream *xdr) | 4373 | static int decode_setclientid_confirm(struct xdr_stream *xdr) |
@@ -4169,11 +4384,16 @@ static int decode_write(struct xdr_stream *xdr, struct nfs_writeres *res) | |||
4169 | if (status) | 4384 | if (status) |
4170 | return status; | 4385 | return status; |
4171 | 4386 | ||
4172 | READ_BUF(16); | 4387 | p = xdr_inline_decode(xdr, 16); |
4173 | READ32(res->count); | 4388 | if (unlikely(!p)) |
4174 | READ32(res->verf->committed); | 4389 | goto out_overflow; |
4175 | COPYMEM(res->verf->verifier, 8); | 4390 | res->count = be32_to_cpup(p++); |
4391 | res->verf->committed = be32_to_cpup(p++); | ||
4392 | memcpy(res->verf->verifier, p, 8); | ||
4176 | return 0; | 4393 | return 0; |
4394 | out_overflow: | ||
4395 | print_overflow_msg(__func__, xdr); | ||
4396 | return -EIO; | ||
4177 | } | 4397 | } |
4178 | 4398 | ||
4179 | static int decode_delegreturn(struct xdr_stream *xdr) | 4399 | static int decode_delegreturn(struct xdr_stream *xdr) |
@@ -4187,6 +4407,7 @@ static int decode_exchange_id(struct xdr_stream *xdr, | |||
4187 | { | 4407 | { |
4188 | __be32 *p; | 4408 | __be32 *p; |
4189 | uint32_t dummy; | 4409 | uint32_t dummy; |
4410 | char *dummy_str; | ||
4190 | int status; | 4411 | int status; |
4191 | struct nfs_client *clp = res->client; | 4412 | struct nfs_client *clp = res->client; |
4192 | 4413 | ||
@@ -4194,36 +4415,45 @@ static int decode_exchange_id(struct xdr_stream *xdr, | |||
4194 | if (status) | 4415 | if (status) |
4195 | return status; | 4416 | return status; |
4196 | 4417 | ||
4197 | READ_BUF(8); | 4418 | p = xdr_inline_decode(xdr, 8); |
4198 | READ64(clp->cl_ex_clid); | 4419 | if (unlikely(!p)) |
4199 | READ_BUF(12); | 4420 | goto out_overflow; |
4200 | READ32(clp->cl_seqid); | 4421 | xdr_decode_hyper(p, &clp->cl_ex_clid); |
4201 | READ32(clp->cl_exchange_flags); | 4422 | p = xdr_inline_decode(xdr, 12); |
4423 | if (unlikely(!p)) | ||
4424 | goto out_overflow; | ||
4425 | clp->cl_seqid = be32_to_cpup(p++); | ||
4426 | clp->cl_exchange_flags = be32_to_cpup(p++); | ||
4202 | 4427 | ||
4203 | /* We ask for SP4_NONE */ | 4428 | /* We ask for SP4_NONE */ |
4204 | READ32(dummy); | 4429 | dummy = be32_to_cpup(p); |
4205 | if (dummy != SP4_NONE) | 4430 | if (dummy != SP4_NONE) |
4206 | return -EIO; | 4431 | return -EIO; |
4207 | 4432 | ||
4208 | /* Throw away minor_id */ | 4433 | /* Throw away minor_id */ |
4209 | READ_BUF(8); | 4434 | p = xdr_inline_decode(xdr, 8); |
4435 | if (unlikely(!p)) | ||
4436 | goto out_overflow; | ||
4210 | 4437 | ||
4211 | /* Throw away Major id */ | 4438 | /* Throw away Major id */ |
4212 | READ_BUF(4); | 4439 | status = decode_opaque_inline(xdr, &dummy, &dummy_str); |
4213 | READ32(dummy); | 4440 | if (unlikely(status)) |
4214 | READ_BUF(dummy); | 4441 | return status; |
4215 | 4442 | ||
4216 | /* Throw away server_scope */ | 4443 | /* Throw away server_scope */ |
4217 | READ_BUF(4); | 4444 | status = decode_opaque_inline(xdr, &dummy, &dummy_str); |
4218 | READ32(dummy); | 4445 | if (unlikely(status)) |
4219 | READ_BUF(dummy); | 4446 | return status; |
4220 | 4447 | ||
4221 | /* Throw away Implementation id array */ | 4448 | /* Throw away Implementation id array */ |
4222 | READ_BUF(4); | 4449 | status = decode_opaque_inline(xdr, &dummy, &dummy_str); |
4223 | READ32(dummy); | 4450 | if (unlikely(status)) |
4224 | READ_BUF(dummy); | 4451 | return status; |
4225 | 4452 | ||
4226 | return 0; | 4453 | return 0; |
4454 | out_overflow: | ||
4455 | print_overflow_msg(__func__, xdr); | ||
4456 | return -EIO; | ||
4227 | } | 4457 | } |
4228 | 4458 | ||
4229 | static int decode_chan_attrs(struct xdr_stream *xdr, | 4459 | static int decode_chan_attrs(struct xdr_stream *xdr, |
@@ -4232,22 +4462,35 @@ static int decode_chan_attrs(struct xdr_stream *xdr, | |||
4232 | __be32 *p; | 4462 | __be32 *p; |
4233 | u32 nr_attrs; | 4463 | u32 nr_attrs; |
4234 | 4464 | ||
4235 | READ_BUF(28); | 4465 | p = xdr_inline_decode(xdr, 28); |
4236 | READ32(attrs->headerpadsz); | 4466 | if (unlikely(!p)) |
4237 | READ32(attrs->max_rqst_sz); | 4467 | goto out_overflow; |
4238 | READ32(attrs->max_resp_sz); | 4468 | attrs->headerpadsz = be32_to_cpup(p++); |
4239 | READ32(attrs->max_resp_sz_cached); | 4469 | attrs->max_rqst_sz = be32_to_cpup(p++); |
4240 | READ32(attrs->max_ops); | 4470 | attrs->max_resp_sz = be32_to_cpup(p++); |
4241 | READ32(attrs->max_reqs); | 4471 | attrs->max_resp_sz_cached = be32_to_cpup(p++); |
4242 | READ32(nr_attrs); | 4472 | attrs->max_ops = be32_to_cpup(p++); |
4473 | attrs->max_reqs = be32_to_cpup(p++); | ||
4474 | nr_attrs = be32_to_cpup(p); | ||
4243 | if (unlikely(nr_attrs > 1)) { | 4475 | if (unlikely(nr_attrs > 1)) { |
4244 | printk(KERN_WARNING "%s: Invalid rdma channel attrs count %u\n", | 4476 | printk(KERN_WARNING "%s: Invalid rdma channel attrs count %u\n", |
4245 | __func__, nr_attrs); | 4477 | __func__, nr_attrs); |
4246 | return -EINVAL; | 4478 | return -EINVAL; |
4247 | } | 4479 | } |
4248 | if (nr_attrs == 1) | 4480 | if (nr_attrs == 1) { |
4249 | READ_BUF(4); /* skip rdma_attrs */ | 4481 | p = xdr_inline_decode(xdr, 4); /* skip rdma_attrs */ |
4482 | if (unlikely(!p)) | ||
4483 | goto out_overflow; | ||
4484 | } | ||
4250 | return 0; | 4485 | return 0; |
4486 | out_overflow: | ||
4487 | print_overflow_msg(__func__, xdr); | ||
4488 | return -EIO; | ||
4489 | } | ||
4490 | |||
4491 | static int decode_sessionid(struct xdr_stream *xdr, struct nfs4_sessionid *sid) | ||
4492 | { | ||
4493 | return decode_opaque_fixed(xdr, sid->data, NFS4_MAX_SESSIONID_LEN); | ||
4251 | } | 4494 | } |
4252 | 4495 | ||
4253 | static int decode_create_session(struct xdr_stream *xdr, | 4496 | static int decode_create_session(struct xdr_stream *xdr, |
@@ -4259,24 +4502,26 @@ static int decode_create_session(struct xdr_stream *xdr, | |||
4259 | struct nfs4_session *session = clp->cl_session; | 4502 | struct nfs4_session *session = clp->cl_session; |
4260 | 4503 | ||
4261 | status = decode_op_hdr(xdr, OP_CREATE_SESSION); | 4504 | status = decode_op_hdr(xdr, OP_CREATE_SESSION); |
4262 | 4505 | if (!status) | |
4263 | if (status) | 4506 | status = decode_sessionid(xdr, &session->sess_id); |
4507 | if (unlikely(status)) | ||
4264 | return status; | 4508 | return status; |
4265 | 4509 | ||
4266 | /* sessionid */ | ||
4267 | READ_BUF(NFS4_MAX_SESSIONID_LEN); | ||
4268 | COPYMEM(&session->sess_id, NFS4_MAX_SESSIONID_LEN); | ||
4269 | |||
4270 | /* seqid, flags */ | 4510 | /* seqid, flags */ |
4271 | READ_BUF(8); | 4511 | p = xdr_inline_decode(xdr, 8); |
4272 | READ32(clp->cl_seqid); | 4512 | if (unlikely(!p)) |
4273 | READ32(session->flags); | 4513 | goto out_overflow; |
4514 | clp->cl_seqid = be32_to_cpup(p++); | ||
4515 | session->flags = be32_to_cpup(p); | ||
4274 | 4516 | ||
4275 | /* Channel attributes */ | 4517 | /* Channel attributes */ |
4276 | status = decode_chan_attrs(xdr, &session->fc_attrs); | 4518 | status = decode_chan_attrs(xdr, &session->fc_attrs); |
4277 | if (!status) | 4519 | if (!status) |
4278 | status = decode_chan_attrs(xdr, &session->bc_attrs); | 4520 | status = decode_chan_attrs(xdr, &session->bc_attrs); |
4279 | return status; | 4521 | return status; |
4522 | out_overflow: | ||
4523 | print_overflow_msg(__func__, xdr); | ||
4524 | return -EIO; | ||
4280 | } | 4525 | } |
4281 | 4526 | ||
4282 | static int decode_destroy_session(struct xdr_stream *xdr, void *dummy) | 4527 | static int decode_destroy_session(struct xdr_stream *xdr, void *dummy) |
@@ -4300,7 +4545,9 @@ static int decode_sequence(struct xdr_stream *xdr, | |||
4300 | return 0; | 4545 | return 0; |
4301 | 4546 | ||
4302 | status = decode_op_hdr(xdr, OP_SEQUENCE); | 4547 | status = decode_op_hdr(xdr, OP_SEQUENCE); |
4303 | if (status) | 4548 | if (!status) |
4549 | status = decode_sessionid(xdr, &id); | ||
4550 | if (unlikely(status)) | ||
4304 | goto out_err; | 4551 | goto out_err; |
4305 | 4552 | ||
4306 | /* | 4553 | /* |
@@ -4309,36 +4556,43 @@ static int decode_sequence(struct xdr_stream *xdr, | |||
4309 | */ | 4556 | */ |
4310 | status = -ESERVERFAULT; | 4557 | status = -ESERVERFAULT; |
4311 | 4558 | ||
4312 | slot = &res->sr_session->fc_slot_table.slots[res->sr_slotid]; | ||
4313 | READ_BUF(NFS4_MAX_SESSIONID_LEN + 20); | ||
4314 | COPYMEM(id.data, NFS4_MAX_SESSIONID_LEN); | ||
4315 | if (memcmp(id.data, res->sr_session->sess_id.data, | 4559 | if (memcmp(id.data, res->sr_session->sess_id.data, |
4316 | NFS4_MAX_SESSIONID_LEN)) { | 4560 | NFS4_MAX_SESSIONID_LEN)) { |
4317 | dprintk("%s Invalid session id\n", __func__); | 4561 | dprintk("%s Invalid session id\n", __func__); |
4318 | goto out_err; | 4562 | goto out_err; |
4319 | } | 4563 | } |
4564 | |||
4565 | p = xdr_inline_decode(xdr, 20); | ||
4566 | if (unlikely(!p)) | ||
4567 | goto out_overflow; | ||
4568 | |||
4320 | /* seqid */ | 4569 | /* seqid */ |
4321 | READ32(dummy); | 4570 | slot = &res->sr_session->fc_slot_table.slots[res->sr_slotid]; |
4571 | dummy = be32_to_cpup(p++); | ||
4322 | if (dummy != slot->seq_nr) { | 4572 | if (dummy != slot->seq_nr) { |
4323 | dprintk("%s Invalid sequence number\n", __func__); | 4573 | dprintk("%s Invalid sequence number\n", __func__); |
4324 | goto out_err; | 4574 | goto out_err; |
4325 | } | 4575 | } |
4326 | /* slot id */ | 4576 | /* slot id */ |
4327 | READ32(dummy); | 4577 | dummy = be32_to_cpup(p++); |
4328 | if (dummy != res->sr_slotid) { | 4578 | if (dummy != res->sr_slotid) { |
4329 | dprintk("%s Invalid slot id\n", __func__); | 4579 | dprintk("%s Invalid slot id\n", __func__); |
4330 | goto out_err; | 4580 | goto out_err; |
4331 | } | 4581 | } |
4332 | /* highest slot id - currently not processed */ | 4582 | /* highest slot id - currently not processed */ |
4333 | READ32(dummy); | 4583 | dummy = be32_to_cpup(p++); |
4334 | /* target highest slot id - currently not processed */ | 4584 | /* target highest slot id - currently not processed */ |
4335 | READ32(dummy); | 4585 | dummy = be32_to_cpup(p++); |
4336 | /* result flags - currently not processed */ | 4586 | /* result flags - currently not processed */ |
4337 | READ32(dummy); | 4587 | dummy = be32_to_cpup(p); |
4338 | status = 0; | 4588 | status = 0; |
4339 | out_err: | 4589 | out_err: |
4340 | res->sr_status = status; | 4590 | res->sr_status = status; |
4341 | return status; | 4591 | return status; |
4592 | out_overflow: | ||
4593 | print_overflow_msg(__func__, xdr); | ||
4594 | status = -EIO; | ||
4595 | goto out_err; | ||
4342 | #else /* CONFIG_NFS_V4_1 */ | 4596 | #else /* CONFIG_NFS_V4_1 */ |
4343 | return 0; | 4597 | return 0; |
4344 | #endif /* CONFIG_NFS_V4_1 */ | 4598 | #endif /* CONFIG_NFS_V4_1 */ |
@@ -4370,7 +4624,8 @@ static int nfs4_xdr_dec_open_downgrade(struct rpc_rqst *rqstp, __be32 *p, struct | |||
4370 | status = decode_open_downgrade(&xdr, res); | 4624 | status = decode_open_downgrade(&xdr, res); |
4371 | if (status != 0) | 4625 | if (status != 0) |
4372 | goto out; | 4626 | goto out; |
4373 | decode_getfattr(&xdr, res->fattr, res->server); | 4627 | decode_getfattr(&xdr, res->fattr, res->server, |
4628 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4374 | out: | 4629 | out: |
4375 | return status; | 4630 | return status; |
4376 | } | 4631 | } |
@@ -4397,7 +4652,8 @@ static int nfs4_xdr_dec_access(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_ac | |||
4397 | status = decode_access(&xdr, res); | 4652 | status = decode_access(&xdr, res); |
4398 | if (status != 0) | 4653 | if (status != 0) |
4399 | goto out; | 4654 | goto out; |
4400 | decode_getfattr(&xdr, res->fattr, res->server); | 4655 | decode_getfattr(&xdr, res->fattr, res->server, |
4656 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4401 | out: | 4657 | out: |
4402 | return status; | 4658 | return status; |
4403 | } | 4659 | } |
@@ -4424,7 +4680,8 @@ static int nfs4_xdr_dec_lookup(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_lo | |||
4424 | goto out; | 4680 | goto out; |
4425 | if ((status = decode_getfh(&xdr, res->fh)) != 0) | 4681 | if ((status = decode_getfh(&xdr, res->fh)) != 0) |
4426 | goto out; | 4682 | goto out; |
4427 | status = decode_getfattr(&xdr, res->fattr, res->server); | 4683 | status = decode_getfattr(&xdr, res->fattr, res->server |
4684 | ,!RPC_IS_ASYNC(rqstp->rq_task)); | ||
4428 | out: | 4685 | out: |
4429 | return status; | 4686 | return status; |
4430 | } | 4687 | } |
@@ -4448,7 +4705,8 @@ static int nfs4_xdr_dec_lookup_root(struct rpc_rqst *rqstp, __be32 *p, struct nf | |||
4448 | if ((status = decode_putrootfh(&xdr)) != 0) | 4705 | if ((status = decode_putrootfh(&xdr)) != 0) |
4449 | goto out; | 4706 | goto out; |
4450 | if ((status = decode_getfh(&xdr, res->fh)) == 0) | 4707 | if ((status = decode_getfh(&xdr, res->fh)) == 0) |
4451 | status = decode_getfattr(&xdr, res->fattr, res->server); | 4708 | status = decode_getfattr(&xdr, res->fattr, res->server, |
4709 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4452 | out: | 4710 | out: |
4453 | return status; | 4711 | return status; |
4454 | } | 4712 | } |
@@ -4473,7 +4731,8 @@ static int nfs4_xdr_dec_remove(struct rpc_rqst *rqstp, __be32 *p, struct nfs_rem | |||
4473 | goto out; | 4731 | goto out; |
4474 | if ((status = decode_remove(&xdr, &res->cinfo)) != 0) | 4732 | if ((status = decode_remove(&xdr, &res->cinfo)) != 0) |
4475 | goto out; | 4733 | goto out; |
4476 | decode_getfattr(&xdr, &res->dir_attr, res->server); | 4734 | decode_getfattr(&xdr, &res->dir_attr, res->server, |
4735 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4477 | out: | 4736 | out: |
4478 | return status; | 4737 | return status; |
4479 | } | 4738 | } |
@@ -4503,11 +4762,13 @@ static int nfs4_xdr_dec_rename(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_re | |||
4503 | if ((status = decode_rename(&xdr, &res->old_cinfo, &res->new_cinfo)) != 0) | 4762 | if ((status = decode_rename(&xdr, &res->old_cinfo, &res->new_cinfo)) != 0) |
4504 | goto out; | 4763 | goto out; |
4505 | /* Current FH is target directory */ | 4764 | /* Current FH is target directory */ |
4506 | if (decode_getfattr(&xdr, res->new_fattr, res->server) != 0) | 4765 | if (decode_getfattr(&xdr, res->new_fattr, res->server, |
4766 | !RPC_IS_ASYNC(rqstp->rq_task)) != 0) | ||
4507 | goto out; | 4767 | goto out; |
4508 | if ((status = decode_restorefh(&xdr)) != 0) | 4768 | if ((status = decode_restorefh(&xdr)) != 0) |
4509 | goto out; | 4769 | goto out; |
4510 | decode_getfattr(&xdr, res->old_fattr, res->server); | 4770 | decode_getfattr(&xdr, res->old_fattr, res->server, |
4771 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4511 | out: | 4772 | out: |
4512 | return status; | 4773 | return status; |
4513 | } | 4774 | } |
@@ -4540,11 +4801,13 @@ static int nfs4_xdr_dec_link(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_link | |||
4540 | * Note order: OP_LINK leaves the directory as the current | 4801 | * Note order: OP_LINK leaves the directory as the current |
4541 | * filehandle. | 4802 | * filehandle. |
4542 | */ | 4803 | */ |
4543 | if (decode_getfattr(&xdr, res->dir_attr, res->server) != 0) | 4804 | if (decode_getfattr(&xdr, res->dir_attr, res->server, |
4805 | !RPC_IS_ASYNC(rqstp->rq_task)) != 0) | ||
4544 | goto out; | 4806 | goto out; |
4545 | if ((status = decode_restorefh(&xdr)) != 0) | 4807 | if ((status = decode_restorefh(&xdr)) != 0) |
4546 | goto out; | 4808 | goto out; |
4547 | decode_getfattr(&xdr, res->fattr, res->server); | 4809 | decode_getfattr(&xdr, res->fattr, res->server, |
4810 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4548 | out: | 4811 | out: |
4549 | return status; | 4812 | return status; |
4550 | } | 4813 | } |
@@ -4573,11 +4836,13 @@ static int nfs4_xdr_dec_create(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_cr | |||
4573 | goto out; | 4836 | goto out; |
4574 | if ((status = decode_getfh(&xdr, res->fh)) != 0) | 4837 | if ((status = decode_getfh(&xdr, res->fh)) != 0) |
4575 | goto out; | 4838 | goto out; |
4576 | if (decode_getfattr(&xdr, res->fattr, res->server) != 0) | 4839 | if (decode_getfattr(&xdr, res->fattr, res->server, |
4840 | !RPC_IS_ASYNC(rqstp->rq_task)) != 0) | ||
4577 | goto out; | 4841 | goto out; |
4578 | if ((status = decode_restorefh(&xdr)) != 0) | 4842 | if ((status = decode_restorefh(&xdr)) != 0) |
4579 | goto out; | 4843 | goto out; |
4580 | decode_getfattr(&xdr, res->dir_fattr, res->server); | 4844 | decode_getfattr(&xdr, res->dir_fattr, res->server, |
4845 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4581 | out: | 4846 | out: |
4582 | return status; | 4847 | return status; |
4583 | } | 4848 | } |
@@ -4609,7 +4874,8 @@ static int nfs4_xdr_dec_getattr(struct rpc_rqst *rqstp, __be32 *p, struct nfs4_g | |||
4609 | status = decode_putfh(&xdr); | 4874 | status = decode_putfh(&xdr); |
4610 | if (status) | 4875 | if (status) |
4611 | goto out; | 4876 | goto out; |
4612 | status = decode_getfattr(&xdr, res->fattr, res->server); | 4877 | status = decode_getfattr(&xdr, res->fattr, res->server, |
4878 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4613 | out: | 4879 | out: |
4614 | return status; | 4880 | return status; |
4615 | } | 4881 | } |
@@ -4716,7 +4982,8 @@ static int nfs4_xdr_dec_close(struct rpc_rqst *rqstp, __be32 *p, struct nfs_clos | |||
4716 | * an ESTALE error. Shouldn't be a problem, | 4982 | * an ESTALE error. Shouldn't be a problem, |
4717 | * though, since fattr->valid will remain unset. | 4983 | * though, since fattr->valid will remain unset. |
4718 | */ | 4984 | */ |
4719 | decode_getfattr(&xdr, res->fattr, res->server); | 4985 | decode_getfattr(&xdr, res->fattr, res->server, |
4986 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4720 | out: | 4987 | out: |
4721 | return status; | 4988 | return status; |
4722 | } | 4989 | } |
@@ -4748,11 +5015,13 @@ static int nfs4_xdr_dec_open(struct rpc_rqst *rqstp, __be32 *p, struct nfs_openr | |||
4748 | goto out; | 5015 | goto out; |
4749 | if (decode_getfh(&xdr, &res->fh) != 0) | 5016 | if (decode_getfh(&xdr, &res->fh) != 0) |
4750 | goto out; | 5017 | goto out; |
4751 | if (decode_getfattr(&xdr, res->f_attr, res->server) != 0) | 5018 | if (decode_getfattr(&xdr, res->f_attr, res->server, |
5019 | !RPC_IS_ASYNC(rqstp->rq_task)) != 0) | ||
4752 | goto out; | 5020 | goto out; |
4753 | if (decode_restorefh(&xdr) != 0) | 5021 | if (decode_restorefh(&xdr) != 0) |
4754 | goto out; | 5022 | goto out; |
4755 | decode_getfattr(&xdr, res->dir_attr, res->server); | 5023 | decode_getfattr(&xdr, res->dir_attr, res->server, |
5024 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4756 | out: | 5025 | out: |
4757 | return status; | 5026 | return status; |
4758 | } | 5027 | } |
@@ -4800,7 +5069,8 @@ static int nfs4_xdr_dec_open_noattr(struct rpc_rqst *rqstp, __be32 *p, struct nf | |||
4800 | status = decode_open(&xdr, res); | 5069 | status = decode_open(&xdr, res); |
4801 | if (status) | 5070 | if (status) |
4802 | goto out; | 5071 | goto out; |
4803 | decode_getfattr(&xdr, res->f_attr, res->server); | 5072 | decode_getfattr(&xdr, res->f_attr, res->server, |
5073 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4804 | out: | 5074 | out: |
4805 | return status; | 5075 | return status; |
4806 | } | 5076 | } |
@@ -4827,7 +5097,8 @@ static int nfs4_xdr_dec_setattr(struct rpc_rqst *rqstp, __be32 *p, struct nfs_se | |||
4827 | status = decode_setattr(&xdr); | 5097 | status = decode_setattr(&xdr); |
4828 | if (status) | 5098 | if (status) |
4829 | goto out; | 5099 | goto out; |
4830 | decode_getfattr(&xdr, res->fattr, res->server); | 5100 | decode_getfattr(&xdr, res->fattr, res->server, |
5101 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
4831 | out: | 5102 | out: |
4832 | return status; | 5103 | return status; |
4833 | } | 5104 | } |
@@ -5001,7 +5272,8 @@ static int nfs4_xdr_dec_write(struct rpc_rqst *rqstp, __be32 *p, struct nfs_writ | |||
5001 | status = decode_write(&xdr, res); | 5272 | status = decode_write(&xdr, res); |
5002 | if (status) | 5273 | if (status) |
5003 | goto out; | 5274 | goto out; |
5004 | decode_getfattr(&xdr, res->fattr, res->server); | 5275 | decode_getfattr(&xdr, res->fattr, res->server, |
5276 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
5005 | if (!status) | 5277 | if (!status) |
5006 | status = res->count; | 5278 | status = res->count; |
5007 | out: | 5279 | out: |
@@ -5030,7 +5302,8 @@ static int nfs4_xdr_dec_commit(struct rpc_rqst *rqstp, __be32 *p, struct nfs_wri | |||
5030 | status = decode_commit(&xdr, res); | 5302 | status = decode_commit(&xdr, res); |
5031 | if (status) | 5303 | if (status) |
5032 | goto out; | 5304 | goto out; |
5033 | decode_getfattr(&xdr, res->fattr, res->server); | 5305 | decode_getfattr(&xdr, res->fattr, res->server, |
5306 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
5034 | out: | 5307 | out: |
5035 | return status; | 5308 | return status; |
5036 | } | 5309 | } |
@@ -5194,7 +5467,8 @@ static int nfs4_xdr_dec_delegreturn(struct rpc_rqst *rqstp, __be32 *p, struct nf | |||
5194 | if (status != 0) | 5467 | if (status != 0) |
5195 | goto out; | 5468 | goto out; |
5196 | status = decode_delegreturn(&xdr); | 5469 | status = decode_delegreturn(&xdr); |
5197 | decode_getfattr(&xdr, res->fattr, res->server); | 5470 | decode_getfattr(&xdr, res->fattr, res->server, |
5471 | !RPC_IS_ASYNC(rqstp->rq_task)); | ||
5198 | out: | 5472 | out: |
5199 | return status; | 5473 | return status; |
5200 | } | 5474 | } |
@@ -5222,7 +5496,8 @@ static int nfs4_xdr_dec_fs_locations(struct rpc_rqst *req, __be32 *p, | |||
5222 | goto out; | 5496 | goto out; |
5223 | xdr_enter_page(&xdr, PAGE_SIZE); | 5497 | xdr_enter_page(&xdr, PAGE_SIZE); |
5224 | status = decode_getfattr(&xdr, &res->fs_locations->fattr, | 5498 | status = decode_getfattr(&xdr, &res->fs_locations->fattr, |
5225 | res->fs_locations->server); | 5499 | res->fs_locations->server, |
5500 | !RPC_IS_ASYNC(req->rq_task)); | ||
5226 | out: | 5501 | out: |
5227 | return status; | 5502 | return status; |
5228 | } | 5503 | } |
diff --git a/fs/nfs/proc.c b/fs/nfs/proc.c index 7be72d90d49d..ef583854d8d0 100644 --- a/fs/nfs/proc.c +++ b/fs/nfs/proc.c | |||
@@ -32,7 +32,6 @@ | |||
32 | #include <linux/slab.h> | 32 | #include <linux/slab.h> |
33 | #include <linux/time.h> | 33 | #include <linux/time.h> |
34 | #include <linux/mm.h> | 34 | #include <linux/mm.h> |
35 | #include <linux/utsname.h> | ||
36 | #include <linux/errno.h> | 35 | #include <linux/errno.h> |
37 | #include <linux/string.h> | 36 | #include <linux/string.h> |
38 | #include <linux/in.h> | 37 | #include <linux/in.h> |
diff --git a/fs/nfs/super.c b/fs/nfs/super.c index 0b4cbdc60abd..29786d3b9326 100644 --- a/fs/nfs/super.c +++ b/fs/nfs/super.c | |||
@@ -73,7 +73,7 @@ enum { | |||
73 | Opt_cto, Opt_nocto, | 73 | Opt_cto, Opt_nocto, |
74 | Opt_ac, Opt_noac, | 74 | Opt_ac, Opt_noac, |
75 | Opt_lock, Opt_nolock, | 75 | Opt_lock, Opt_nolock, |
76 | Opt_v2, Opt_v3, | 76 | Opt_v2, Opt_v3, Opt_v4, |
77 | Opt_udp, Opt_tcp, Opt_rdma, | 77 | Opt_udp, Opt_tcp, Opt_rdma, |
78 | Opt_acl, Opt_noacl, | 78 | Opt_acl, Opt_noacl, |
79 | Opt_rdirplus, Opt_nordirplus, | 79 | Opt_rdirplus, Opt_nordirplus, |
@@ -127,6 +127,7 @@ static const match_table_t nfs_mount_option_tokens = { | |||
127 | { Opt_nolock, "nolock" }, | 127 | { Opt_nolock, "nolock" }, |
128 | { Opt_v2, "v2" }, | 128 | { Opt_v2, "v2" }, |
129 | { Opt_v3, "v3" }, | 129 | { Opt_v3, "v3" }, |
130 | { Opt_v4, "v4" }, | ||
130 | { Opt_udp, "udp" }, | 131 | { Opt_udp, "udp" }, |
131 | { Opt_tcp, "tcp" }, | 132 | { Opt_tcp, "tcp" }, |
132 | { Opt_rdma, "rdma" }, | 133 | { Opt_rdma, "rdma" }, |
@@ -158,7 +159,7 @@ static const match_table_t nfs_mount_option_tokens = { | |||
158 | { Opt_mountvers, "mountvers=%s" }, | 159 | { Opt_mountvers, "mountvers=%s" }, |
159 | { Opt_nfsvers, "nfsvers=%s" }, | 160 | { Opt_nfsvers, "nfsvers=%s" }, |
160 | { Opt_nfsvers, "vers=%s" }, | 161 | { Opt_nfsvers, "vers=%s" }, |
161 | { Opt_minorversion, "minorversion=%u" }, | 162 | { Opt_minorversion, "minorversion=%s" }, |
162 | 163 | ||
163 | { Opt_sec, "sec=%s" }, | 164 | { Opt_sec, "sec=%s" }, |
164 | { Opt_proto, "proto=%s" }, | 165 | { Opt_proto, "proto=%s" }, |
@@ -272,6 +273,10 @@ static const struct super_operations nfs_sops = { | |||
272 | }; | 273 | }; |
273 | 274 | ||
274 | #ifdef CONFIG_NFS_V4 | 275 | #ifdef CONFIG_NFS_V4 |
276 | static int nfs4_validate_text_mount_data(void *options, | ||
277 | struct nfs_parsed_mount_data *args, const char *dev_name); | ||
278 | static int nfs4_try_mount(int flags, const char *dev_name, | ||
279 | struct nfs_parsed_mount_data *data, struct vfsmount *mnt); | ||
275 | static int nfs4_get_sb(struct file_system_type *fs_type, | 280 | static int nfs4_get_sb(struct file_system_type *fs_type, |
276 | int flags, const char *dev_name, void *raw_data, struct vfsmount *mnt); | 281 | int flags, const char *dev_name, void *raw_data, struct vfsmount *mnt); |
277 | static int nfs4_remote_get_sb(struct file_system_type *fs_type, | 282 | static int nfs4_remote_get_sb(struct file_system_type *fs_type, |
@@ -723,6 +728,27 @@ static void nfs_umount_begin(struct super_block *sb) | |||
723 | unlock_kernel(); | 728 | unlock_kernel(); |
724 | } | 729 | } |
725 | 730 | ||
731 | static struct nfs_parsed_mount_data *nfs_alloc_parsed_mount_data(int flags) | ||
732 | { | ||
733 | struct nfs_parsed_mount_data *data; | ||
734 | |||
735 | data = kzalloc(sizeof(*data), GFP_KERNEL); | ||
736 | if (data) { | ||
737 | data->flags = flags; | ||
738 | data->rsize = NFS_MAX_FILE_IO_SIZE; | ||
739 | data->wsize = NFS_MAX_FILE_IO_SIZE; | ||
740 | data->acregmin = NFS_DEF_ACREGMIN; | ||
741 | data->acregmax = NFS_DEF_ACREGMAX; | ||
742 | data->acdirmin = NFS_DEF_ACDIRMIN; | ||
743 | data->acdirmax = NFS_DEF_ACDIRMAX; | ||
744 | data->nfs_server.port = NFS_UNSPEC_PORT; | ||
745 | data->auth_flavors[0] = RPC_AUTH_UNIX; | ||
746 | data->auth_flavor_len = 1; | ||
747 | data->minorversion = 0; | ||
748 | } | ||
749 | return data; | ||
750 | } | ||
751 | |||
726 | /* | 752 | /* |
727 | * Sanity-check a server address provided by the mount command. | 753 | * Sanity-check a server address provided by the mount command. |
728 | * | 754 | * |
@@ -742,127 +768,23 @@ static int nfs_verify_server_address(struct sockaddr *addr) | |||
742 | } | 768 | } |
743 | } | 769 | } |
744 | 770 | ||
771 | dfprintk(MOUNT, "NFS: Invalid IP address specified\n"); | ||
745 | return 0; | 772 | return 0; |
746 | } | 773 | } |
747 | 774 | ||
748 | static void nfs_parse_ipv4_address(char *string, size_t str_len, | ||
749 | struct sockaddr *sap, size_t *addr_len) | ||
750 | { | ||
751 | struct sockaddr_in *sin = (struct sockaddr_in *)sap; | ||
752 | u8 *addr = (u8 *)&sin->sin_addr.s_addr; | ||
753 | |||
754 | if (str_len <= INET_ADDRSTRLEN) { | ||
755 | dfprintk(MOUNT, "NFS: parsing IPv4 address %*s\n", | ||
756 | (int)str_len, string); | ||
757 | |||
758 | sin->sin_family = AF_INET; | ||
759 | *addr_len = sizeof(*sin); | ||
760 | if (in4_pton(string, str_len, addr, '\0', NULL)) | ||
761 | return; | ||
762 | } | ||
763 | |||
764 | sap->sa_family = AF_UNSPEC; | ||
765 | *addr_len = 0; | ||
766 | } | ||
767 | |||
768 | #if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE) | ||
769 | static int nfs_parse_ipv6_scope_id(const char *string, const size_t str_len, | ||
770 | const char *delim, | ||
771 | struct sockaddr_in6 *sin6) | ||
772 | { | ||
773 | char *p; | ||
774 | size_t len; | ||
775 | |||
776 | if ((string + str_len) == delim) | ||
777 | return 1; | ||
778 | |||
779 | if (*delim != IPV6_SCOPE_DELIMITER) | ||
780 | return 0; | ||
781 | |||
782 | if (!(ipv6_addr_type(&sin6->sin6_addr) & IPV6_ADDR_LINKLOCAL)) | ||
783 | return 0; | ||
784 | |||
785 | len = (string + str_len) - delim - 1; | ||
786 | p = kstrndup(delim + 1, len, GFP_KERNEL); | ||
787 | if (p) { | ||
788 | unsigned long scope_id = 0; | ||
789 | struct net_device *dev; | ||
790 | |||
791 | dev = dev_get_by_name(&init_net, p); | ||
792 | if (dev != NULL) { | ||
793 | scope_id = dev->ifindex; | ||
794 | dev_put(dev); | ||
795 | } else { | ||
796 | if (strict_strtoul(p, 10, &scope_id) == 0) { | ||
797 | kfree(p); | ||
798 | return 0; | ||
799 | } | ||
800 | } | ||
801 | |||
802 | kfree(p); | ||
803 | |||
804 | sin6->sin6_scope_id = scope_id; | ||
805 | dfprintk(MOUNT, "NFS: IPv6 scope ID = %lu\n", scope_id); | ||
806 | return 1; | ||
807 | } | ||
808 | |||
809 | return 0; | ||
810 | } | ||
811 | |||
812 | static void nfs_parse_ipv6_address(char *string, size_t str_len, | ||
813 | struct sockaddr *sap, size_t *addr_len) | ||
814 | { | ||
815 | struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)sap; | ||
816 | u8 *addr = (u8 *)&sin6->sin6_addr.in6_u; | ||
817 | const char *delim; | ||
818 | |||
819 | if (str_len <= INET6_ADDRSTRLEN) { | ||
820 | dfprintk(MOUNT, "NFS: parsing IPv6 address %*s\n", | ||
821 | (int)str_len, string); | ||
822 | |||
823 | sin6->sin6_family = AF_INET6; | ||
824 | *addr_len = sizeof(*sin6); | ||
825 | if (in6_pton(string, str_len, addr, | ||
826 | IPV6_SCOPE_DELIMITER, &delim) != 0) { | ||
827 | if (nfs_parse_ipv6_scope_id(string, str_len, | ||
828 | delim, sin6) != 0) | ||
829 | return; | ||
830 | } | ||
831 | } | ||
832 | |||
833 | sap->sa_family = AF_UNSPEC; | ||
834 | *addr_len = 0; | ||
835 | } | ||
836 | #else | ||
837 | static void nfs_parse_ipv6_address(char *string, size_t str_len, | ||
838 | struct sockaddr *sap, size_t *addr_len) | ||
839 | { | ||
840 | sap->sa_family = AF_UNSPEC; | ||
841 | *addr_len = 0; | ||
842 | } | ||
843 | #endif | ||
844 | |||
845 | /* | 775 | /* |
846 | * Construct a sockaddr based on the contents of a string that contains | 776 | * Select between a default port value and a user-specified port value. |
847 | * an IP address in presentation format. | 777 | * If a zero value is set, then autobind will be used. |
848 | * | ||
849 | * If there is a problem constructing the new sockaddr, set the address | ||
850 | * family to AF_UNSPEC. | ||
851 | */ | 778 | */ |
852 | void nfs_parse_ip_address(char *string, size_t str_len, | 779 | static void nfs_set_default_port(struct sockaddr *sap, const int parsed_port, |
853 | struct sockaddr *sap, size_t *addr_len) | 780 | const unsigned short default_port) |
854 | { | 781 | { |
855 | unsigned int i, colons; | 782 | unsigned short port = default_port; |
856 | 783 | ||
857 | colons = 0; | 784 | if (parsed_port != NFS_UNSPEC_PORT) |
858 | for (i = 0; i < str_len; i++) | 785 | port = parsed_port; |
859 | if (string[i] == ':') | ||
860 | colons++; | ||
861 | 786 | ||
862 | if (colons >= 2) | 787 | rpc_set_port(sap, port); |
863 | nfs_parse_ipv6_address(string, str_len, sap, addr_len); | ||
864 | else | ||
865 | nfs_parse_ipv4_address(string, str_len, sap, addr_len); | ||
866 | } | 788 | } |
867 | 789 | ||
868 | /* | 790 | /* |
@@ -904,8 +826,6 @@ static void nfs_set_mount_transport_protocol(struct nfs_parsed_mount_data *mnt) | |||
904 | 826 | ||
905 | /* | 827 | /* |
906 | * Parse the value of the 'sec=' option. | 828 | * Parse the value of the 'sec=' option. |
907 | * | ||
908 | * The flavor_len setting is for v4 mounts. | ||
909 | */ | 829 | */ |
910 | static int nfs_parse_security_flavors(char *value, | 830 | static int nfs_parse_security_flavors(char *value, |
911 | struct nfs_parsed_mount_data *mnt) | 831 | struct nfs_parsed_mount_data *mnt) |
@@ -916,53 +836,43 @@ static int nfs_parse_security_flavors(char *value, | |||
916 | 836 | ||
917 | switch (match_token(value, nfs_secflavor_tokens, args)) { | 837 | switch (match_token(value, nfs_secflavor_tokens, args)) { |
918 | case Opt_sec_none: | 838 | case Opt_sec_none: |
919 | mnt->auth_flavor_len = 0; | ||
920 | mnt->auth_flavors[0] = RPC_AUTH_NULL; | 839 | mnt->auth_flavors[0] = RPC_AUTH_NULL; |
921 | break; | 840 | break; |
922 | case Opt_sec_sys: | 841 | case Opt_sec_sys: |
923 | mnt->auth_flavor_len = 0; | ||
924 | mnt->auth_flavors[0] = RPC_AUTH_UNIX; | 842 | mnt->auth_flavors[0] = RPC_AUTH_UNIX; |
925 | break; | 843 | break; |
926 | case Opt_sec_krb5: | 844 | case Opt_sec_krb5: |
927 | mnt->auth_flavor_len = 1; | ||
928 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5; | 845 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5; |
929 | break; | 846 | break; |
930 | case Opt_sec_krb5i: | 847 | case Opt_sec_krb5i: |
931 | mnt->auth_flavor_len = 1; | ||
932 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5I; | 848 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5I; |
933 | break; | 849 | break; |
934 | case Opt_sec_krb5p: | 850 | case Opt_sec_krb5p: |
935 | mnt->auth_flavor_len = 1; | ||
936 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5P; | 851 | mnt->auth_flavors[0] = RPC_AUTH_GSS_KRB5P; |
937 | break; | 852 | break; |
938 | case Opt_sec_lkey: | 853 | case Opt_sec_lkey: |
939 | mnt->auth_flavor_len = 1; | ||
940 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEY; | 854 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEY; |
941 | break; | 855 | break; |
942 | case Opt_sec_lkeyi: | 856 | case Opt_sec_lkeyi: |
943 | mnt->auth_flavor_len = 1; | ||
944 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEYI; | 857 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEYI; |
945 | break; | 858 | break; |
946 | case Opt_sec_lkeyp: | 859 | case Opt_sec_lkeyp: |
947 | mnt->auth_flavor_len = 1; | ||
948 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEYP; | 860 | mnt->auth_flavors[0] = RPC_AUTH_GSS_LKEYP; |
949 | break; | 861 | break; |
950 | case Opt_sec_spkm: | 862 | case Opt_sec_spkm: |
951 | mnt->auth_flavor_len = 1; | ||
952 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKM; | 863 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKM; |
953 | break; | 864 | break; |
954 | case Opt_sec_spkmi: | 865 | case Opt_sec_spkmi: |
955 | mnt->auth_flavor_len = 1; | ||
956 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKMI; | 866 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKMI; |
957 | break; | 867 | break; |
958 | case Opt_sec_spkmp: | 868 | case Opt_sec_spkmp: |
959 | mnt->auth_flavor_len = 1; | ||
960 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKMP; | 869 | mnt->auth_flavors[0] = RPC_AUTH_GSS_SPKMP; |
961 | break; | 870 | break; |
962 | default: | 871 | default: |
963 | return 0; | 872 | return 0; |
964 | } | 873 | } |
965 | 874 | ||
875 | mnt->auth_flavor_len = 1; | ||
966 | return 1; | 876 | return 1; |
967 | } | 877 | } |
968 | 878 | ||
@@ -1001,7 +911,6 @@ static int nfs_parse_mount_options(char *raw, | |||
1001 | while ((p = strsep(&raw, ",")) != NULL) { | 911 | while ((p = strsep(&raw, ",")) != NULL) { |
1002 | substring_t args[MAX_OPT_ARGS]; | 912 | substring_t args[MAX_OPT_ARGS]; |
1003 | unsigned long option; | 913 | unsigned long option; |
1004 | int int_option; | ||
1005 | int token; | 914 | int token; |
1006 | 915 | ||
1007 | if (!*p) | 916 | if (!*p) |
@@ -1047,10 +956,18 @@ static int nfs_parse_mount_options(char *raw, | |||
1047 | break; | 956 | break; |
1048 | case Opt_v2: | 957 | case Opt_v2: |
1049 | mnt->flags &= ~NFS_MOUNT_VER3; | 958 | mnt->flags &= ~NFS_MOUNT_VER3; |
959 | mnt->version = 2; | ||
1050 | break; | 960 | break; |
1051 | case Opt_v3: | 961 | case Opt_v3: |
1052 | mnt->flags |= NFS_MOUNT_VER3; | 962 | mnt->flags |= NFS_MOUNT_VER3; |
963 | mnt->version = 3; | ||
964 | break; | ||
965 | #ifdef CONFIG_NFS_V4 | ||
966 | case Opt_v4: | ||
967 | mnt->flags &= ~NFS_MOUNT_VER3; | ||
968 | mnt->version = 4; | ||
1053 | break; | 969 | break; |
970 | #endif | ||
1054 | case Opt_udp: | 971 | case Opt_udp: |
1055 | mnt->flags &= ~NFS_MOUNT_TCP; | 972 | mnt->flags &= ~NFS_MOUNT_TCP; |
1056 | mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP; | 973 | mnt->nfs_server.protocol = XPRT_TRANSPORT_UDP; |
@@ -1264,20 +1181,33 @@ static int nfs_parse_mount_options(char *raw, | |||
1264 | switch (option) { | 1181 | switch (option) { |
1265 | case NFS2_VERSION: | 1182 | case NFS2_VERSION: |
1266 | mnt->flags &= ~NFS_MOUNT_VER3; | 1183 | mnt->flags &= ~NFS_MOUNT_VER3; |
1184 | mnt->version = 2; | ||
1267 | break; | 1185 | break; |
1268 | case NFS3_VERSION: | 1186 | case NFS3_VERSION: |
1269 | mnt->flags |= NFS_MOUNT_VER3; | 1187 | mnt->flags |= NFS_MOUNT_VER3; |
1188 | mnt->version = 3; | ||
1189 | break; | ||
1190 | #ifdef CONFIG_NFS_V4 | ||
1191 | case NFS4_VERSION: | ||
1192 | mnt->flags &= ~NFS_MOUNT_VER3; | ||
1193 | mnt->version = 4; | ||
1270 | break; | 1194 | break; |
1195 | #endif | ||
1271 | default: | 1196 | default: |
1272 | goto out_invalid_value; | 1197 | goto out_invalid_value; |
1273 | } | 1198 | } |
1274 | break; | 1199 | break; |
1275 | case Opt_minorversion: | 1200 | case Opt_minorversion: |
1276 | if (match_int(args, &int_option)) | 1201 | string = match_strdup(args); |
1277 | return 0; | 1202 | if (string == NULL) |
1278 | if (int_option < 0 || int_option > NFS4_MAX_MINOR_VERSION) | 1203 | goto out_nomem; |
1279 | return 0; | 1204 | rc = strict_strtoul(string, 10, &option); |
1280 | mnt->minorversion = int_option; | 1205 | kfree(string); |
1206 | if (rc != 0) | ||
1207 | goto out_invalid_value; | ||
1208 | if (option > NFS4_MAX_MINOR_VERSION) | ||
1209 | goto out_invalid_value; | ||
1210 | mnt->minorversion = option; | ||
1281 | break; | 1211 | break; |
1282 | 1212 | ||
1283 | /* | 1213 | /* |
@@ -1352,11 +1282,14 @@ static int nfs_parse_mount_options(char *raw, | |||
1352 | string = match_strdup(args); | 1282 | string = match_strdup(args); |
1353 | if (string == NULL) | 1283 | if (string == NULL) |
1354 | goto out_nomem; | 1284 | goto out_nomem; |
1355 | nfs_parse_ip_address(string, strlen(string), | 1285 | mnt->nfs_server.addrlen = |
1356 | (struct sockaddr *) | 1286 | rpc_pton(string, strlen(string), |
1357 | &mnt->nfs_server.address, | 1287 | (struct sockaddr *) |
1358 | &mnt->nfs_server.addrlen); | 1288 | &mnt->nfs_server.address, |
1289 | sizeof(mnt->nfs_server.address)); | ||
1359 | kfree(string); | 1290 | kfree(string); |
1291 | if (mnt->nfs_server.addrlen == 0) | ||
1292 | goto out_invalid_address; | ||
1360 | break; | 1293 | break; |
1361 | case Opt_clientaddr: | 1294 | case Opt_clientaddr: |
1362 | string = match_strdup(args); | 1295 | string = match_strdup(args); |
@@ -1376,11 +1309,14 @@ static int nfs_parse_mount_options(char *raw, | |||
1376 | string = match_strdup(args); | 1309 | string = match_strdup(args); |
1377 | if (string == NULL) | 1310 | if (string == NULL) |
1378 | goto out_nomem; | 1311 | goto out_nomem; |
1379 | nfs_parse_ip_address(string, strlen(string), | 1312 | mnt->mount_server.addrlen = |
1380 | (struct sockaddr *) | 1313 | rpc_pton(string, strlen(string), |
1381 | &mnt->mount_server.address, | 1314 | (struct sockaddr *) |
1382 | &mnt->mount_server.addrlen); | 1315 | &mnt->mount_server.address, |
1316 | sizeof(mnt->mount_server.address)); | ||
1383 | kfree(string); | 1317 | kfree(string); |
1318 | if (mnt->mount_server.addrlen == 0) | ||
1319 | goto out_invalid_address; | ||
1384 | break; | 1320 | break; |
1385 | case Opt_lookupcache: | 1321 | case Opt_lookupcache: |
1386 | string = match_strdup(args); | 1322 | string = match_strdup(args); |
@@ -1432,8 +1368,11 @@ static int nfs_parse_mount_options(char *raw, | |||
1432 | 1368 | ||
1433 | return 1; | 1369 | return 1; |
1434 | 1370 | ||
1371 | out_invalid_address: | ||
1372 | printk(KERN_INFO "NFS: bad IP address specified: %s\n", p); | ||
1373 | return 0; | ||
1435 | out_invalid_value: | 1374 | out_invalid_value: |
1436 | printk(KERN_INFO "NFS: bad mount option value specified: %s \n", p); | 1375 | printk(KERN_INFO "NFS: bad mount option value specified: %s\n", p); |
1437 | return 0; | 1376 | return 0; |
1438 | out_nomem: | 1377 | out_nomem: |
1439 | printk(KERN_INFO "NFS: not enough memory to parse option\n"); | 1378 | printk(KERN_INFO "NFS: not enough memory to parse option\n"); |
@@ -1445,13 +1384,60 @@ out_security_failure: | |||
1445 | } | 1384 | } |
1446 | 1385 | ||
1447 | /* | 1386 | /* |
1387 | * Match the requested auth flavors with the list returned by | ||
1388 | * the server. Returns zero and sets the mount's authentication | ||
1389 | * flavor on success; returns -EACCES if server does not support | ||
1390 | * the requested flavor. | ||
1391 | */ | ||
1392 | static int nfs_walk_authlist(struct nfs_parsed_mount_data *args, | ||
1393 | struct nfs_mount_request *request) | ||
1394 | { | ||
1395 | unsigned int i, j, server_authlist_len = *(request->auth_flav_len); | ||
1396 | |||
1397 | /* | ||
1398 | * Certain releases of Linux's mountd return an empty | ||
1399 | * flavor list. To prevent behavioral regression with | ||
1400 | * these servers (ie. rejecting mounts that used to | ||
1401 | * succeed), revert to pre-2.6.32 behavior (no checking) | ||
1402 | * if the returned flavor list is empty. | ||
1403 | */ | ||
1404 | if (server_authlist_len == 0) | ||
1405 | return 0; | ||
1406 | |||
1407 | /* | ||
1408 | * We avoid sophisticated negotiating here, as there are | ||
1409 | * plenty of cases where we can get it wrong, providing | ||
1410 | * either too little or too much security. | ||
1411 | * | ||
1412 | * RFC 2623, section 2.7 suggests we SHOULD prefer the | ||
1413 | * flavor listed first. However, some servers list | ||
1414 | * AUTH_NULL first. Our caller plants AUTH_SYS, the | ||
1415 | * preferred default, in args->auth_flavors[0] if user | ||
1416 | * didn't specify sec= mount option. | ||
1417 | */ | ||
1418 | for (i = 0; i < args->auth_flavor_len; i++) | ||
1419 | for (j = 0; j < server_authlist_len; j++) | ||
1420 | if (args->auth_flavors[i] == request->auth_flavs[j]) { | ||
1421 | dfprintk(MOUNT, "NFS: using auth flavor %d\n", | ||
1422 | request->auth_flavs[j]); | ||
1423 | args->auth_flavors[0] = request->auth_flavs[j]; | ||
1424 | return 0; | ||
1425 | } | ||
1426 | |||
1427 | dfprintk(MOUNT, "NFS: server does not support requested auth flavor\n"); | ||
1428 | nfs_umount(request); | ||
1429 | return -EACCES; | ||
1430 | } | ||
1431 | |||
1432 | /* | ||
1448 | * Use the remote server's MOUNT service to request the NFS file handle | 1433 | * Use the remote server's MOUNT service to request the NFS file handle |
1449 | * corresponding to the provided path. | 1434 | * corresponding to the provided path. |
1450 | */ | 1435 | */ |
1451 | static int nfs_try_mount(struct nfs_parsed_mount_data *args, | 1436 | static int nfs_try_mount(struct nfs_parsed_mount_data *args, |
1452 | struct nfs_fh *root_fh) | 1437 | struct nfs_fh *root_fh) |
1453 | { | 1438 | { |
1454 | unsigned int auth_flavor_len = 0; | 1439 | rpc_authflavor_t server_authlist[NFS_MAX_SECFLAVORS]; |
1440 | unsigned int server_authlist_len = ARRAY_SIZE(server_authlist); | ||
1455 | struct nfs_mount_request request = { | 1441 | struct nfs_mount_request request = { |
1456 | .sap = (struct sockaddr *) | 1442 | .sap = (struct sockaddr *) |
1457 | &args->mount_server.address, | 1443 | &args->mount_server.address, |
@@ -1459,15 +1445,19 @@ static int nfs_try_mount(struct nfs_parsed_mount_data *args, | |||
1459 | .protocol = args->mount_server.protocol, | 1445 | .protocol = args->mount_server.protocol, |
1460 | .fh = root_fh, | 1446 | .fh = root_fh, |
1461 | .noresvport = args->flags & NFS_MOUNT_NORESVPORT, | 1447 | .noresvport = args->flags & NFS_MOUNT_NORESVPORT, |
1462 | .auth_flav_len = &auth_flavor_len, | 1448 | .auth_flav_len = &server_authlist_len, |
1449 | .auth_flavs = server_authlist, | ||
1463 | }; | 1450 | }; |
1464 | int status; | 1451 | int status; |
1465 | 1452 | ||
1466 | if (args->mount_server.version == 0) { | 1453 | if (args->mount_server.version == 0) { |
1467 | if (args->flags & NFS_MOUNT_VER3) | 1454 | switch (args->version) { |
1468 | args->mount_server.version = NFS_MNT3_VERSION; | 1455 | default: |
1469 | else | 1456 | args->mount_server.version = NFS_MNT3_VERSION; |
1470 | args->mount_server.version = NFS_MNT_VERSION; | 1457 | break; |
1458 | case 2: | ||
1459 | args->mount_server.version = NFS_MNT_VERSION; | ||
1460 | } | ||
1471 | } | 1461 | } |
1472 | request.version = args->mount_server.version; | 1462 | request.version = args->mount_server.version; |
1473 | 1463 | ||
@@ -1485,23 +1475,25 @@ static int nfs_try_mount(struct nfs_parsed_mount_data *args, | |||
1485 | args->mount_server.addrlen = args->nfs_server.addrlen; | 1475 | args->mount_server.addrlen = args->nfs_server.addrlen; |
1486 | } | 1476 | } |
1487 | request.salen = args->mount_server.addrlen; | 1477 | request.salen = args->mount_server.addrlen; |
1488 | 1478 | nfs_set_default_port(request.sap, args->mount_server.port, 0); | |
1489 | /* | ||
1490 | * autobind will be used if mount_server.port == 0 | ||
1491 | */ | ||
1492 | nfs_set_port(request.sap, args->mount_server.port); | ||
1493 | 1479 | ||
1494 | /* | 1480 | /* |
1495 | * Now ask the mount server to map our export path | 1481 | * Now ask the mount server to map our export path |
1496 | * to a file handle. | 1482 | * to a file handle. |
1497 | */ | 1483 | */ |
1498 | status = nfs_mount(&request); | 1484 | status = nfs_mount(&request); |
1499 | if (status == 0) | 1485 | if (status != 0) { |
1500 | return 0; | 1486 | dfprintk(MOUNT, "NFS: unable to mount server %s, error %d\n", |
1487 | request.hostname, status); | ||
1488 | return status; | ||
1489 | } | ||
1501 | 1490 | ||
1502 | dfprintk(MOUNT, "NFS: unable to mount server %s, error %d\n", | 1491 | /* |
1503 | request.hostname, status); | 1492 | * MNTv1 (NFSv2) does not support auth flavor negotiation. |
1504 | return status; | 1493 | */ |
1494 | if (args->mount_server.version != NFS_MNT3_VERSION) | ||
1495 | return 0; | ||
1496 | return nfs_walk_authlist(args, &request); | ||
1505 | } | 1497 | } |
1506 | 1498 | ||
1507 | static int nfs_parse_simple_hostname(const char *dev_name, | 1499 | static int nfs_parse_simple_hostname(const char *dev_name, |
@@ -1661,22 +1653,11 @@ static int nfs_validate_mount_data(void *options, | |||
1661 | const char *dev_name) | 1653 | const char *dev_name) |
1662 | { | 1654 | { |
1663 | struct nfs_mount_data *data = (struct nfs_mount_data *)options; | 1655 | struct nfs_mount_data *data = (struct nfs_mount_data *)options; |
1656 | struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address; | ||
1664 | 1657 | ||
1665 | if (data == NULL) | 1658 | if (data == NULL) |
1666 | goto out_no_data; | 1659 | goto out_no_data; |
1667 | 1660 | ||
1668 | args->flags = (NFS_MOUNT_VER3 | NFS_MOUNT_TCP); | ||
1669 | args->rsize = NFS_MAX_FILE_IO_SIZE; | ||
1670 | args->wsize = NFS_MAX_FILE_IO_SIZE; | ||
1671 | args->acregmin = NFS_DEF_ACREGMIN; | ||
1672 | args->acregmax = NFS_DEF_ACREGMAX; | ||
1673 | args->acdirmin = NFS_DEF_ACDIRMIN; | ||
1674 | args->acdirmax = NFS_DEF_ACDIRMAX; | ||
1675 | args->mount_server.port = 0; /* autobind unless user sets port */ | ||
1676 | args->nfs_server.port = 0; /* autobind unless user sets port */ | ||
1677 | args->nfs_server.protocol = XPRT_TRANSPORT_TCP; | ||
1678 | args->auth_flavors[0] = RPC_AUTH_UNIX; | ||
1679 | |||
1680 | switch (data->version) { | 1661 | switch (data->version) { |
1681 | case 1: | 1662 | case 1: |
1682 | data->namlen = 0; | 1663 | data->namlen = 0; |
@@ -1697,8 +1678,11 @@ static int nfs_validate_mount_data(void *options, | |||
1697 | if (data->root.size > NFS3_FHSIZE || data->root.size == 0) | 1678 | if (data->root.size > NFS3_FHSIZE || data->root.size == 0) |
1698 | goto out_invalid_fh; | 1679 | goto out_invalid_fh; |
1699 | mntfh->size = data->root.size; | 1680 | mntfh->size = data->root.size; |
1700 | } else | 1681 | args->version = 3; |
1682 | } else { | ||
1701 | mntfh->size = NFS2_FHSIZE; | 1683 | mntfh->size = NFS2_FHSIZE; |
1684 | args->version = 2; | ||
1685 | } | ||
1702 | 1686 | ||
1703 | 1687 | ||
1704 | memcpy(mntfh->data, data->root.data, mntfh->size); | 1688 | memcpy(mntfh->data, data->root.data, mntfh->size); |
@@ -1720,15 +1704,15 @@ static int nfs_validate_mount_data(void *options, | |||
1720 | args->acdirmin = data->acdirmin; | 1704 | args->acdirmin = data->acdirmin; |
1721 | args->acdirmax = data->acdirmax; | 1705 | args->acdirmax = data->acdirmax; |
1722 | 1706 | ||
1723 | memcpy(&args->nfs_server.address, &data->addr, | 1707 | memcpy(sap, &data->addr, sizeof(data->addr)); |
1724 | sizeof(data->addr)); | ||
1725 | args->nfs_server.addrlen = sizeof(data->addr); | 1708 | args->nfs_server.addrlen = sizeof(data->addr); |
1726 | if (!nfs_verify_server_address((struct sockaddr *) | 1709 | if (!nfs_verify_server_address(sap)) |
1727 | &args->nfs_server.address)) | ||
1728 | goto out_no_address; | 1710 | goto out_no_address; |
1729 | 1711 | ||
1730 | if (!(data->flags & NFS_MOUNT_TCP)) | 1712 | if (!(data->flags & NFS_MOUNT_TCP)) |
1731 | args->nfs_server.protocol = XPRT_TRANSPORT_UDP; | 1713 | args->nfs_server.protocol = XPRT_TRANSPORT_UDP; |
1714 | else | ||
1715 | args->nfs_server.protocol = XPRT_TRANSPORT_TCP; | ||
1732 | /* N.B. caller will free nfs_server.hostname in all cases */ | 1716 | /* N.B. caller will free nfs_server.hostname in all cases */ |
1733 | args->nfs_server.hostname = kstrdup(data->hostname, GFP_KERNEL); | 1717 | args->nfs_server.hostname = kstrdup(data->hostname, GFP_KERNEL); |
1734 | args->namlen = data->namlen; | 1718 | args->namlen = data->namlen; |
@@ -1772,12 +1756,18 @@ static int nfs_validate_mount_data(void *options, | |||
1772 | if (nfs_parse_mount_options((char *)options, args) == 0) | 1756 | if (nfs_parse_mount_options((char *)options, args) == 0) |
1773 | return -EINVAL; | 1757 | return -EINVAL; |
1774 | 1758 | ||
1775 | if (!nfs_verify_server_address((struct sockaddr *) | 1759 | if (!nfs_verify_server_address(sap)) |
1776 | &args->nfs_server.address)) | ||
1777 | goto out_no_address; | 1760 | goto out_no_address; |
1778 | 1761 | ||
1779 | nfs_set_port((struct sockaddr *)&args->nfs_server.address, | 1762 | if (args->version == 4) |
1780 | args->nfs_server.port); | 1763 | #ifdef CONFIG_NFS_V4 |
1764 | return nfs4_validate_text_mount_data(options, | ||
1765 | args, dev_name); | ||
1766 | #else | ||
1767 | goto out_v4_not_compiled; | ||
1768 | #endif | ||
1769 | |||
1770 | nfs_set_default_port(sap, args->nfs_server.port, 0); | ||
1781 | 1771 | ||
1782 | nfs_set_mount_transport_protocol(args); | 1772 | nfs_set_mount_transport_protocol(args); |
1783 | 1773 | ||
@@ -1800,7 +1790,7 @@ static int nfs_validate_mount_data(void *options, | |||
1800 | } | 1790 | } |
1801 | 1791 | ||
1802 | #ifndef CONFIG_NFS_V3 | 1792 | #ifndef CONFIG_NFS_V3 |
1803 | if (args->flags & NFS_MOUNT_VER3) | 1793 | if (args->version == 3) |
1804 | goto out_v3_not_compiled; | 1794 | goto out_v3_not_compiled; |
1805 | #endif /* !CONFIG_NFS_V3 */ | 1795 | #endif /* !CONFIG_NFS_V3 */ |
1806 | 1796 | ||
@@ -1825,6 +1815,12 @@ out_v3_not_compiled: | |||
1825 | return -EPROTONOSUPPORT; | 1815 | return -EPROTONOSUPPORT; |
1826 | #endif /* !CONFIG_NFS_V3 */ | 1816 | #endif /* !CONFIG_NFS_V3 */ |
1827 | 1817 | ||
1818 | #ifndef CONFIG_NFS_V4 | ||
1819 | out_v4_not_compiled: | ||
1820 | dfprintk(MOUNT, "NFS: NFSv4 is not compiled into kernel\n"); | ||
1821 | return -EPROTONOSUPPORT; | ||
1822 | #endif /* !CONFIG_NFS_V4 */ | ||
1823 | |||
1828 | out_nomem: | 1824 | out_nomem: |
1829 | dfprintk(MOUNT, "NFS: not enough memory to handle mount options\n"); | 1825 | dfprintk(MOUNT, "NFS: not enough memory to handle mount options\n"); |
1830 | return -ENOMEM; | 1826 | return -ENOMEM; |
@@ -1934,6 +1930,8 @@ static inline void nfs_initialise_sb(struct super_block *sb) | |||
1934 | if (server->flags & NFS_MOUNT_NOAC) | 1930 | if (server->flags & NFS_MOUNT_NOAC) |
1935 | sb->s_flags |= MS_SYNCHRONOUS; | 1931 | sb->s_flags |= MS_SYNCHRONOUS; |
1936 | 1932 | ||
1933 | sb->s_bdi = &server->backing_dev_info; | ||
1934 | |||
1937 | nfs_super_set_maxbytes(sb, server->maxfilesize); | 1935 | nfs_super_set_maxbytes(sb, server->maxfilesize); |
1938 | } | 1936 | } |
1939 | 1937 | ||
@@ -1950,7 +1948,7 @@ static void nfs_fill_super(struct super_block *sb, | |||
1950 | if (data->bsize) | 1948 | if (data->bsize) |
1951 | sb->s_blocksize = nfs_block_size(data->bsize, &sb->s_blocksize_bits); | 1949 | sb->s_blocksize = nfs_block_size(data->bsize, &sb->s_blocksize_bits); |
1952 | 1950 | ||
1953 | if (server->flags & NFS_MOUNT_VER3) { | 1951 | if (server->nfs_client->rpc_ops->version == 3) { |
1954 | /* The VFS shouldn't apply the umask to mode bits. We will do | 1952 | /* The VFS shouldn't apply the umask to mode bits. We will do |
1955 | * so ourselves when necessary. | 1953 | * so ourselves when necessary. |
1956 | */ | 1954 | */ |
@@ -1974,7 +1972,7 @@ static void nfs_clone_super(struct super_block *sb, | |||
1974 | sb->s_blocksize = old_sb->s_blocksize; | 1972 | sb->s_blocksize = old_sb->s_blocksize; |
1975 | sb->s_maxbytes = old_sb->s_maxbytes; | 1973 | sb->s_maxbytes = old_sb->s_maxbytes; |
1976 | 1974 | ||
1977 | if (server->flags & NFS_MOUNT_VER3) { | 1975 | if (server->nfs_client->rpc_ops->version == 3) { |
1978 | /* The VFS shouldn't apply the umask to mode bits. We will do | 1976 | /* The VFS shouldn't apply the umask to mode bits. We will do |
1979 | * so ourselves when necessary. | 1977 | * so ourselves when necessary. |
1980 | */ | 1978 | */ |
@@ -2108,7 +2106,7 @@ static int nfs_get_sb(struct file_system_type *fs_type, | |||
2108 | }; | 2106 | }; |
2109 | int error = -ENOMEM; | 2107 | int error = -ENOMEM; |
2110 | 2108 | ||
2111 | data = kzalloc(sizeof(*data), GFP_KERNEL); | 2109 | data = nfs_alloc_parsed_mount_data(NFS_MOUNT_VER3 | NFS_MOUNT_TCP); |
2112 | mntfh = kzalloc(sizeof(*mntfh), GFP_KERNEL); | 2110 | mntfh = kzalloc(sizeof(*mntfh), GFP_KERNEL); |
2113 | if (data == NULL || mntfh == NULL) | 2111 | if (data == NULL || mntfh == NULL) |
2114 | goto out_free_fh; | 2112 | goto out_free_fh; |
@@ -2120,6 +2118,14 @@ static int nfs_get_sb(struct file_system_type *fs_type, | |||
2120 | if (error < 0) | 2118 | if (error < 0) |
2121 | goto out; | 2119 | goto out; |
2122 | 2120 | ||
2121 | #ifdef CONFIG_NFS_V4 | ||
2122 | if (data->version == 4) { | ||
2123 | error = nfs4_try_mount(flags, dev_name, data, mnt); | ||
2124 | kfree(data->client_address); | ||
2125 | goto out; | ||
2126 | } | ||
2127 | #endif /* CONFIG_NFS_V4 */ | ||
2128 | |||
2123 | /* Get a volume representation */ | 2129 | /* Get a volume representation */ |
2124 | server = nfs_create_server(data, mntfh); | 2130 | server = nfs_create_server(data, mntfh); |
2125 | if (IS_ERR(server)) { | 2131 | if (IS_ERR(server)) { |
@@ -2150,7 +2156,8 @@ static int nfs_get_sb(struct file_system_type *fs_type, | |||
2150 | if (!s->s_root) { | 2156 | if (!s->s_root) { |
2151 | /* initial superblock/root creation */ | 2157 | /* initial superblock/root creation */ |
2152 | nfs_fill_super(s, data); | 2158 | nfs_fill_super(s, data); |
2153 | nfs_fscache_get_super_cookie(s, data); | 2159 | nfs_fscache_get_super_cookie( |
2160 | s, data ? data->fscache_uniq : NULL, NULL); | ||
2154 | } | 2161 | } |
2155 | 2162 | ||
2156 | mntroot = nfs_get_root(s, mntfh); | 2163 | mntroot = nfs_get_root(s, mntfh); |
@@ -2196,8 +2203,8 @@ static void nfs_kill_super(struct super_block *s) | |||
2196 | { | 2203 | { |
2197 | struct nfs_server *server = NFS_SB(s); | 2204 | struct nfs_server *server = NFS_SB(s); |
2198 | 2205 | ||
2199 | bdi_unregister(&server->backing_dev_info); | ||
2200 | kill_anon_super(s); | 2206 | kill_anon_super(s); |
2207 | bdi_unregister(&server->backing_dev_info); | ||
2201 | nfs_fscache_release_super_cookie(s); | 2208 | nfs_fscache_release_super_cookie(s); |
2202 | nfs_free_server(server); | 2209 | nfs_free_server(server); |
2203 | } | 2210 | } |
@@ -2251,6 +2258,7 @@ static int nfs_xdev_get_sb(struct file_system_type *fs_type, int flags, | |||
2251 | if (!s->s_root) { | 2258 | if (!s->s_root) { |
2252 | /* initial superblock/root creation */ | 2259 | /* initial superblock/root creation */ |
2253 | nfs_clone_super(s, data->sb); | 2260 | nfs_clone_super(s, data->sb); |
2261 | nfs_fscache_get_super_cookie(s, NULL, data); | ||
2254 | } | 2262 | } |
2255 | 2263 | ||
2256 | mntroot = nfs_get_root(s, data->fh); | 2264 | mntroot = nfs_get_root(s, data->fh); |
@@ -2317,6 +2325,43 @@ static void nfs4_validate_mount_flags(struct nfs_parsed_mount_data *args) | |||
2317 | args->flags &= ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3); | 2325 | args->flags &= ~(NFS_MOUNT_NONLM|NFS_MOUNT_NOACL|NFS_MOUNT_VER3); |
2318 | } | 2326 | } |
2319 | 2327 | ||
2328 | static int nfs4_validate_text_mount_data(void *options, | ||
2329 | struct nfs_parsed_mount_data *args, | ||
2330 | const char *dev_name) | ||
2331 | { | ||
2332 | struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address; | ||
2333 | |||
2334 | nfs_set_default_port(sap, args->nfs_server.port, NFS_PORT); | ||
2335 | |||
2336 | nfs_validate_transport_protocol(args); | ||
2337 | |||
2338 | nfs4_validate_mount_flags(args); | ||
2339 | |||
2340 | if (args->version != 4) { | ||
2341 | dfprintk(MOUNT, | ||
2342 | "NFS4: Illegal mount version\n"); | ||
2343 | return -EINVAL; | ||
2344 | } | ||
2345 | |||
2346 | if (args->auth_flavor_len > 1) { | ||
2347 | dfprintk(MOUNT, | ||
2348 | "NFS4: Too many RPC auth flavours specified\n"); | ||
2349 | return -EINVAL; | ||
2350 | } | ||
2351 | |||
2352 | if (args->client_address == NULL) { | ||
2353 | dfprintk(MOUNT, | ||
2354 | "NFS4: mount program didn't pass callback address\n"); | ||
2355 | return -EINVAL; | ||
2356 | } | ||
2357 | |||
2358 | return nfs_parse_devname(dev_name, | ||
2359 | &args->nfs_server.hostname, | ||
2360 | NFS4_MAXNAMLEN, | ||
2361 | &args->nfs_server.export_path, | ||
2362 | NFS4_MAXPATHLEN); | ||
2363 | } | ||
2364 | |||
2320 | /* | 2365 | /* |
2321 | * Validate NFSv4 mount options | 2366 | * Validate NFSv4 mount options |
2322 | */ | 2367 | */ |
@@ -2324,36 +2369,24 @@ static int nfs4_validate_mount_data(void *options, | |||
2324 | struct nfs_parsed_mount_data *args, | 2369 | struct nfs_parsed_mount_data *args, |
2325 | const char *dev_name) | 2370 | const char *dev_name) |
2326 | { | 2371 | { |
2327 | struct sockaddr_in *ap; | 2372 | struct sockaddr *sap = (struct sockaddr *)&args->nfs_server.address; |
2328 | struct nfs4_mount_data *data = (struct nfs4_mount_data *)options; | 2373 | struct nfs4_mount_data *data = (struct nfs4_mount_data *)options; |
2329 | char *c; | 2374 | char *c; |
2330 | 2375 | ||
2331 | if (data == NULL) | 2376 | if (data == NULL) |
2332 | goto out_no_data; | 2377 | goto out_no_data; |
2333 | 2378 | ||
2334 | args->rsize = NFS_MAX_FILE_IO_SIZE; | 2379 | args->version = 4; |
2335 | args->wsize = NFS_MAX_FILE_IO_SIZE; | ||
2336 | args->acregmin = NFS_DEF_ACREGMIN; | ||
2337 | args->acregmax = NFS_DEF_ACREGMAX; | ||
2338 | args->acdirmin = NFS_DEF_ACDIRMIN; | ||
2339 | args->acdirmax = NFS_DEF_ACDIRMAX; | ||
2340 | args->nfs_server.port = NFS_PORT; /* 2049 unless user set port= */ | ||
2341 | args->auth_flavors[0] = RPC_AUTH_UNIX; | ||
2342 | args->auth_flavor_len = 0; | ||
2343 | args->minorversion = 0; | ||
2344 | |||
2345 | switch (data->version) { | 2380 | switch (data->version) { |
2346 | case 1: | 2381 | case 1: |
2347 | ap = (struct sockaddr_in *)&args->nfs_server.address; | ||
2348 | if (data->host_addrlen > sizeof(args->nfs_server.address)) | 2382 | if (data->host_addrlen > sizeof(args->nfs_server.address)) |
2349 | goto out_no_address; | 2383 | goto out_no_address; |
2350 | if (data->host_addrlen == 0) | 2384 | if (data->host_addrlen == 0) |
2351 | goto out_no_address; | 2385 | goto out_no_address; |
2352 | args->nfs_server.addrlen = data->host_addrlen; | 2386 | args->nfs_server.addrlen = data->host_addrlen; |
2353 | if (copy_from_user(ap, data->host_addr, data->host_addrlen)) | 2387 | if (copy_from_user(sap, data->host_addr, data->host_addrlen)) |
2354 | return -EFAULT; | 2388 | return -EFAULT; |
2355 | if (!nfs_verify_server_address((struct sockaddr *) | 2389 | if (!nfs_verify_server_address(sap)) |
2356 | &args->nfs_server.address)) | ||
2357 | goto out_no_address; | 2390 | goto out_no_address; |
2358 | 2391 | ||
2359 | if (data->auth_flavourlen) { | 2392 | if (data->auth_flavourlen) { |
@@ -2399,39 +2432,14 @@ static int nfs4_validate_mount_data(void *options, | |||
2399 | nfs_validate_transport_protocol(args); | 2432 | nfs_validate_transport_protocol(args); |
2400 | 2433 | ||
2401 | break; | 2434 | break; |
2402 | default: { | 2435 | default: |
2403 | int status; | ||
2404 | |||
2405 | if (nfs_parse_mount_options((char *)options, args) == 0) | 2436 | if (nfs_parse_mount_options((char *)options, args) == 0) |
2406 | return -EINVAL; | 2437 | return -EINVAL; |
2407 | 2438 | ||
2408 | if (!nfs_verify_server_address((struct sockaddr *) | 2439 | if (!nfs_verify_server_address(sap)) |
2409 | &args->nfs_server.address)) | ||
2410 | return -EINVAL; | 2440 | return -EINVAL; |
2411 | 2441 | ||
2412 | nfs_set_port((struct sockaddr *)&args->nfs_server.address, | 2442 | return nfs4_validate_text_mount_data(options, args, dev_name); |
2413 | args->nfs_server.port); | ||
2414 | |||
2415 | nfs_validate_transport_protocol(args); | ||
2416 | |||
2417 | nfs4_validate_mount_flags(args); | ||
2418 | |||
2419 | if (args->auth_flavor_len > 1) | ||
2420 | goto out_inval_auth; | ||
2421 | |||
2422 | if (args->client_address == NULL) | ||
2423 | goto out_no_client_address; | ||
2424 | |||
2425 | status = nfs_parse_devname(dev_name, | ||
2426 | &args->nfs_server.hostname, | ||
2427 | NFS4_MAXNAMLEN, | ||
2428 | &args->nfs_server.export_path, | ||
2429 | NFS4_MAXPATHLEN); | ||
2430 | if (status < 0) | ||
2431 | return status; | ||
2432 | |||
2433 | break; | ||
2434 | } | ||
2435 | } | 2443 | } |
2436 | 2444 | ||
2437 | return 0; | 2445 | return 0; |
@@ -2448,10 +2456,6 @@ out_inval_auth: | |||
2448 | out_no_address: | 2456 | out_no_address: |
2449 | dfprintk(MOUNT, "NFS4: mount program didn't pass remote address\n"); | 2457 | dfprintk(MOUNT, "NFS4: mount program didn't pass remote address\n"); |
2450 | return -EINVAL; | 2458 | return -EINVAL; |
2451 | |||
2452 | out_no_client_address: | ||
2453 | dfprintk(MOUNT, "NFS4: mount program didn't pass callback address\n"); | ||
2454 | return -EINVAL; | ||
2455 | } | 2459 | } |
2456 | 2460 | ||
2457 | /* | 2461 | /* |
@@ -2507,7 +2511,8 @@ static int nfs4_remote_get_sb(struct file_system_type *fs_type, | |||
2507 | if (!s->s_root) { | 2511 | if (!s->s_root) { |
2508 | /* initial superblock/root creation */ | 2512 | /* initial superblock/root creation */ |
2509 | nfs4_fill_super(s); | 2513 | nfs4_fill_super(s); |
2510 | nfs_fscache_get_super_cookie(s, data); | 2514 | nfs_fscache_get_super_cookie( |
2515 | s, data ? data->fscache_uniq : NULL, NULL); | ||
2511 | } | 2516 | } |
2512 | 2517 | ||
2513 | mntroot = nfs4_get_root(s, mntfh); | 2518 | mntroot = nfs4_get_root(s, mntfh); |
@@ -2618,6 +2623,34 @@ out_err: | |||
2618 | return ret; | 2623 | return ret; |
2619 | } | 2624 | } |
2620 | 2625 | ||
2626 | static int nfs4_try_mount(int flags, const char *dev_name, | ||
2627 | struct nfs_parsed_mount_data *data, | ||
2628 | struct vfsmount *mnt) | ||
2629 | { | ||
2630 | char *export_path; | ||
2631 | struct vfsmount *root_mnt; | ||
2632 | int error; | ||
2633 | |||
2634 | dfprintk(MOUNT, "--> nfs4_try_mount()\n"); | ||
2635 | |||
2636 | export_path = data->nfs_server.export_path; | ||
2637 | data->nfs_server.export_path = "/"; | ||
2638 | root_mnt = nfs_do_root_mount(&nfs4_remote_fs_type, flags, data, | ||
2639 | data->nfs_server.hostname); | ||
2640 | data->nfs_server.export_path = export_path; | ||
2641 | |||
2642 | error = PTR_ERR(root_mnt); | ||
2643 | if (IS_ERR(root_mnt)) | ||
2644 | goto out; | ||
2645 | |||
2646 | error = nfs_follow_remote_path(root_mnt, export_path, mnt); | ||
2647 | |||
2648 | out: | ||
2649 | dfprintk(MOUNT, "<-- nfs4_try_mount() = %d%s\n", error, | ||
2650 | error != 0 ? " [error]" : ""); | ||
2651 | return error; | ||
2652 | } | ||
2653 | |||
2621 | /* | 2654 | /* |
2622 | * Get the superblock for an NFS4 mountpoint | 2655 | * Get the superblock for an NFS4 mountpoint |
2623 | */ | 2656 | */ |
@@ -2625,11 +2658,9 @@ static int nfs4_get_sb(struct file_system_type *fs_type, | |||
2625 | int flags, const char *dev_name, void *raw_data, struct vfsmount *mnt) | 2658 | int flags, const char *dev_name, void *raw_data, struct vfsmount *mnt) |
2626 | { | 2659 | { |
2627 | struct nfs_parsed_mount_data *data; | 2660 | struct nfs_parsed_mount_data *data; |
2628 | char *export_path; | ||
2629 | struct vfsmount *root_mnt; | ||
2630 | int error = -ENOMEM; | 2661 | int error = -ENOMEM; |
2631 | 2662 | ||
2632 | data = kzalloc(sizeof(*data), GFP_KERNEL); | 2663 | data = nfs_alloc_parsed_mount_data(0); |
2633 | if (data == NULL) | 2664 | if (data == NULL) |
2634 | goto out_free_data; | 2665 | goto out_free_data; |
2635 | 2666 | ||
@@ -2638,17 +2669,7 @@ static int nfs4_get_sb(struct file_system_type *fs_type, | |||
2638 | if (error < 0) | 2669 | if (error < 0) |
2639 | goto out; | 2670 | goto out; |
2640 | 2671 | ||
2641 | export_path = data->nfs_server.export_path; | 2672 | error = nfs4_try_mount(flags, dev_name, data, mnt); |
2642 | data->nfs_server.export_path = "/"; | ||
2643 | root_mnt = nfs_do_root_mount(&nfs4_remote_fs_type, flags, data, | ||
2644 | data->nfs_server.hostname); | ||
2645 | data->nfs_server.export_path = export_path; | ||
2646 | |||
2647 | error = PTR_ERR(root_mnt); | ||
2648 | if (IS_ERR(root_mnt)) | ||
2649 | goto out; | ||
2650 | |||
2651 | error = nfs_follow_remote_path(root_mnt, export_path, mnt); | ||
2652 | 2673 | ||
2653 | out: | 2674 | out: |
2654 | kfree(data->client_address); | 2675 | kfree(data->client_address); |
@@ -2724,6 +2745,7 @@ static int nfs4_xdev_get_sb(struct file_system_type *fs_type, int flags, | |||
2724 | if (!s->s_root) { | 2745 | if (!s->s_root) { |
2725 | /* initial superblock/root creation */ | 2746 | /* initial superblock/root creation */ |
2726 | nfs4_clone_super(s, data->sb); | 2747 | nfs4_clone_super(s, data->sb); |
2748 | nfs_fscache_get_super_cookie(s, NULL, data); | ||
2727 | } | 2749 | } |
2728 | 2750 | ||
2729 | mntroot = nfs4_get_root(s, data->fh); | 2751 | mntroot = nfs4_get_root(s, data->fh); |
@@ -2805,6 +2827,7 @@ static int nfs4_remote_referral_get_sb(struct file_system_type *fs_type, | |||
2805 | if (!s->s_root) { | 2827 | if (!s->s_root) { |
2806 | /* initial superblock/root creation */ | 2828 | /* initial superblock/root creation */ |
2807 | nfs4_fill_super(s); | 2829 | nfs4_fill_super(s); |
2830 | nfs_fscache_get_super_cookie(s, NULL, data); | ||
2808 | } | 2831 | } |
2809 | 2832 | ||
2810 | mntroot = nfs4_get_root(s, &mntfh); | 2833 | mntroot = nfs4_get_root(s, &mntfh); |
diff --git a/fs/nfs/write.c b/fs/nfs/write.c index a34fae21fe10..53eb26c16b50 100644 --- a/fs/nfs/write.c +++ b/fs/nfs/write.c | |||
@@ -13,6 +13,7 @@ | |||
13 | #include <linux/file.h> | 13 | #include <linux/file.h> |
14 | #include <linux/writeback.h> | 14 | #include <linux/writeback.h> |
15 | #include <linux/swap.h> | 15 | #include <linux/swap.h> |
16 | #include <linux/migrate.h> | ||
16 | 17 | ||
17 | #include <linux/sunrpc/clnt.h> | 18 | #include <linux/sunrpc/clnt.h> |
18 | #include <linux/nfs_fs.h> | 19 | #include <linux/nfs_fs.h> |
@@ -26,6 +27,7 @@ | |||
26 | #include "internal.h" | 27 | #include "internal.h" |
27 | #include "iostat.h" | 28 | #include "iostat.h" |
28 | #include "nfs4_fs.h" | 29 | #include "nfs4_fs.h" |
30 | #include "fscache.h" | ||
29 | 31 | ||
30 | #define NFSDBG_FACILITY NFSDBG_PAGECACHE | 32 | #define NFSDBG_FACILITY NFSDBG_PAGECACHE |
31 | 33 | ||
@@ -218,24 +220,17 @@ static void nfs_end_page_writeback(struct page *page) | |||
218 | clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); | 220 | clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); |
219 | } | 221 | } |
220 | 222 | ||
221 | /* | 223 | static struct nfs_page *nfs_find_and_lock_request(struct page *page) |
222 | * Find an associated nfs write request, and prepare to flush it out | ||
223 | * May return an error if the user signalled nfs_wait_on_request(). | ||
224 | */ | ||
225 | static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, | ||
226 | struct page *page) | ||
227 | { | 224 | { |
228 | struct inode *inode = page->mapping->host; | 225 | struct inode *inode = page->mapping->host; |
229 | struct nfs_page *req; | 226 | struct nfs_page *req; |
230 | int ret; | 227 | int ret; |
231 | 228 | ||
232 | spin_lock(&inode->i_lock); | 229 | spin_lock(&inode->i_lock); |
233 | for(;;) { | 230 | for (;;) { |
234 | req = nfs_page_find_request_locked(page); | 231 | req = nfs_page_find_request_locked(page); |
235 | if (req == NULL) { | 232 | if (req == NULL) |
236 | spin_unlock(&inode->i_lock); | 233 | break; |
237 | return 0; | ||
238 | } | ||
239 | if (nfs_set_page_tag_locked(req)) | 234 | if (nfs_set_page_tag_locked(req)) |
240 | break; | 235 | break; |
241 | /* Note: If we hold the page lock, as is the case in nfs_writepage, | 236 | /* Note: If we hold the page lock, as is the case in nfs_writepage, |
@@ -247,23 +242,40 @@ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, | |||
247 | ret = nfs_wait_on_request(req); | 242 | ret = nfs_wait_on_request(req); |
248 | nfs_release_request(req); | 243 | nfs_release_request(req); |
249 | if (ret != 0) | 244 | if (ret != 0) |
250 | return ret; | 245 | return ERR_PTR(ret); |
251 | spin_lock(&inode->i_lock); | 246 | spin_lock(&inode->i_lock); |
252 | } | 247 | } |
253 | if (test_bit(PG_CLEAN, &req->wb_flags)) { | ||
254 | spin_unlock(&inode->i_lock); | ||
255 | BUG(); | ||
256 | } | ||
257 | if (nfs_set_page_writeback(page) != 0) { | ||
258 | spin_unlock(&inode->i_lock); | ||
259 | BUG(); | ||
260 | } | ||
261 | spin_unlock(&inode->i_lock); | 248 | spin_unlock(&inode->i_lock); |
249 | return req; | ||
250 | } | ||
251 | |||
252 | /* | ||
253 | * Find an associated nfs write request, and prepare to flush it out | ||
254 | * May return an error if the user signalled nfs_wait_on_request(). | ||
255 | */ | ||
256 | static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, | ||
257 | struct page *page) | ||
258 | { | ||
259 | struct nfs_page *req; | ||
260 | int ret = 0; | ||
261 | |||
262 | req = nfs_find_and_lock_request(page); | ||
263 | if (!req) | ||
264 | goto out; | ||
265 | ret = PTR_ERR(req); | ||
266 | if (IS_ERR(req)) | ||
267 | goto out; | ||
268 | |||
269 | ret = nfs_set_page_writeback(page); | ||
270 | BUG_ON(ret != 0); | ||
271 | BUG_ON(test_bit(PG_CLEAN, &req->wb_flags)); | ||
272 | |||
262 | if (!nfs_pageio_add_request(pgio, req)) { | 273 | if (!nfs_pageio_add_request(pgio, req)) { |
263 | nfs_redirty_request(req); | 274 | nfs_redirty_request(req); |
264 | return pgio->pg_error; | 275 | ret = pgio->pg_error; |
265 | } | 276 | } |
266 | return 0; | 277 | out: |
278 | return ret; | ||
267 | } | 279 | } |
268 | 280 | ||
269 | static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) | 281 | static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) |
@@ -1478,7 +1490,6 @@ static int nfs_write_mapping(struct address_space *mapping, int how) | |||
1478 | .nr_to_write = LONG_MAX, | 1490 | .nr_to_write = LONG_MAX, |
1479 | .range_start = 0, | 1491 | .range_start = 0, |
1480 | .range_end = LLONG_MAX, | 1492 | .range_end = LLONG_MAX, |
1481 | .for_writepages = 1, | ||
1482 | }; | 1493 | }; |
1483 | 1494 | ||
1484 | return __nfs_write_mapping(mapping, &wbc, how); | 1495 | return __nfs_write_mapping(mapping, &wbc, how); |
@@ -1580,6 +1591,41 @@ int nfs_wb_page(struct inode *inode, struct page* page) | |||
1580 | return nfs_wb_page_priority(inode, page, FLUSH_STABLE); | 1591 | return nfs_wb_page_priority(inode, page, FLUSH_STABLE); |
1581 | } | 1592 | } |
1582 | 1593 | ||
1594 | #ifdef CONFIG_MIGRATION | ||
1595 | int nfs_migrate_page(struct address_space *mapping, struct page *newpage, | ||
1596 | struct page *page) | ||
1597 | { | ||
1598 | struct nfs_page *req; | ||
1599 | int ret; | ||
1600 | |||
1601 | if (PageFsCache(page)) | ||
1602 | nfs_fscache_release_page(page, GFP_KERNEL); | ||
1603 | |||
1604 | req = nfs_find_and_lock_request(page); | ||
1605 | ret = PTR_ERR(req); | ||
1606 | if (IS_ERR(req)) | ||
1607 | goto out; | ||
1608 | |||
1609 | ret = migrate_page(mapping, newpage, page); | ||
1610 | if (!req) | ||
1611 | goto out; | ||
1612 | if (ret) | ||
1613 | goto out_unlock; | ||
1614 | page_cache_get(newpage); | ||
1615 | req->wb_page = newpage; | ||
1616 | SetPagePrivate(newpage); | ||
1617 | set_page_private(newpage, page_private(page)); | ||
1618 | ClearPagePrivate(page); | ||
1619 | set_page_private(page, 0); | ||
1620 | page_cache_release(page); | ||
1621 | out_unlock: | ||
1622 | nfs_clear_page_tag_locked(req); | ||
1623 | nfs_release_request(req); | ||
1624 | out: | ||
1625 | return ret; | ||
1626 | } | ||
1627 | #endif | ||
1628 | |||
1583 | int __init nfs_init_writepagecache(void) | 1629 | int __init nfs_init_writepagecache(void) |
1584 | { | 1630 | { |
1585 | nfs_wdata_cachep = kmem_cache_create("nfs_write_data", | 1631 | nfs_wdata_cachep = kmem_cache_create("nfs_write_data", |