xref: /reactos/base/services/nfsd/mount.c (revision c2c66aff)
1 /* NFSv4.1 client for Windows
2  * Copyright � 2012 The Regents of the University of Michigan
3  *
4  * Olga Kornievskaia <aglo@umich.edu>
5  * Casey Bodley <cbodley@umich.edu>
6  *
7  * This library is free software; you can redistribute it and/or modify it
8  * under the terms of the GNU Lesser General Public License as published by
9  * the Free Software Foundation; either version 2.1 of the License, or (at
10  * your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful, but
13  * without any warranty; without even the implied warranty of merchantability
14  * or fitness for a particular purpose.  See the GNU Lesser General Public
15  * License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public License
18  * along with this library; if not, write to the Free Software Foundation,
19  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
20  */
21 
22 #include <windows.h>
23 #include <strsafe.h>
24 #include <stdio.h>
25 
26 #include "daemon_debug.h"
27 #include "nfs41_ops.h"
28 #include "upcall.h"
29 #include "util.h"
30 
31 
32 /* NFS41_MOUNT */
parse_mount(unsigned char * buffer,uint32_t length,nfs41_upcall * upcall)33 static int parse_mount(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall)
34 {
35     int status;
36     mount_upcall_args *args = &upcall->args.mount;
37 
38     status = get_name(&buffer, &length, &args->hostname);
39     if(status) goto out;
40     status = get_name(&buffer, &length, &args->path);
41     if(status) goto out;
42     status = safe_read(&buffer, &length, &args->sec_flavor, sizeof(DWORD));
43     if (status) goto out;
44     status = safe_read(&buffer, &length, &args->rsize, sizeof(DWORD));
45     if (status) goto out;
46     status = safe_read(&buffer, &length, &args->wsize, sizeof(DWORD));
47     if (status) goto out;
48 
49     dprintf(1, "parsing NFS14_MOUNT: srv_name=%s root=%s sec_flavor=%s "
50         "rsize=%d wsize=%d\n", args->hostname, args->path,
51         secflavorop2name(args->sec_flavor), args->rsize, args->wsize);
52 out:
53     return status;
54 }
55 
handle_mount(nfs41_upcall * upcall)56 static int handle_mount(nfs41_upcall *upcall)
57 {
58     int status;
59     mount_upcall_args *args = &upcall->args.mount;
60     nfs41_abs_path path;
61     multi_addr4 addrs;
62     nfs41_root *root;
63     nfs41_client *client;
64     nfs41_path_fh file;
65 
66     // resolve hostname,port
67     status = nfs41_server_resolve(args->hostname, 2049, &addrs);
68     if (status) {
69         eprintf("nfs41_server_resolve() failed with %d\n", status);
70         goto out;
71     }
72 
73     if (upcall->root_ref != INVALID_HANDLE_VALUE) {
74         /* use an existing root from a previous mount, but don't take an
75          * extra reference; we'll only get one UNMOUNT upcall for each root */
76         root = upcall->root_ref;
77     } else {
78         // create root
79         status = nfs41_root_create(args->hostname, args->sec_flavor,
80             args->wsize + WRITE_OVERHEAD, args->rsize + READ_OVERHEAD, &root);
81         if (status) {
82             eprintf("nfs41_root_create() failed %d\n", status);
83             goto out;
84         }
85         root->uid = upcall->uid;
86         root->gid = upcall->gid;
87     }
88 
89     // find or create the client/session
90     status = nfs41_root_mount_addrs(root, &addrs, 0, 0, &client);
91     if (status) {
92         eprintf("nfs41_root_mount_addrs() failed with %d\n", status);
93         goto out_err;
94     }
95 
96     // make a copy of the path for nfs41_lookup()
97     InitializeSRWLock(&path.lock);
98     if (FAILED(StringCchCopyA(path.path, NFS41_MAX_PATH_LEN, args->path))) {
99         status = ERROR_FILENAME_EXCED_RANGE;
100         goto out_err;
101     }
102     path.len = (unsigned short)strlen(path.path);
103 
104     // look up the mount path, and fail if it doesn't exist
105     status = nfs41_lookup(root, client->session,
106         &path, NULL, &file, NULL, NULL);
107     if (status) {
108         eprintf("nfs41_lookup('%s') failed with %d\n", path.path, status);
109         status = ERROR_BAD_NETPATH;
110         goto out_err;
111     }
112 
113     nfs41_superblock_fs_attributes(file.fh.superblock, &args->FsAttrs);
114 
115     if (upcall->root_ref == INVALID_HANDLE_VALUE)
116         nfs41_root_ref(root);
117     upcall->root_ref = root;
118     args->lease_time = client->session->lease_time;
119 out:
120     return status;
121 
122 out_err:
123     if (upcall->root_ref == INVALID_HANDLE_VALUE)
124         nfs41_root_deref(root);
125     goto out;
126 }
127 
marshall_mount(unsigned char * buffer,uint32_t * length,nfs41_upcall * upcall)128 static int marshall_mount(unsigned char *buffer, uint32_t *length, nfs41_upcall *upcall)
129 {
130     mount_upcall_args *args = &upcall->args.mount;
131     int status;
132     dprintf(2, "NFS41_MOUNT: writing pointer to nfs41_root %p, version %d, "
133         "lease_time %d\n", upcall->root_ref, NFS41D_VERSION, args->lease_time);
134     status = safe_write(&buffer, length, &upcall->root_ref, sizeof(HANDLE));
135     if (status) goto out;
136     status = safe_write(&buffer, length, &NFS41D_VERSION, sizeof(DWORD));
137     if (status) goto out;
138     status = safe_write(&buffer, length, &args->lease_time, sizeof(DWORD));
139     if (status) goto out;
140     status = safe_write(&buffer, length, &args->FsAttrs, sizeof(args->FsAttrs));
141 out:
142     return status;
143 }
144 
cancel_mount(IN nfs41_upcall * upcall)145 static void cancel_mount(IN nfs41_upcall *upcall)
146 {
147     if (upcall->root_ref != INVALID_HANDLE_VALUE)
148         nfs41_root_deref(upcall->root_ref);
149 }
150 
151 const nfs41_upcall_op nfs41_op_mount = {
152     parse_mount,
153     handle_mount,
154     marshall_mount,
155     cancel_mount
156 };
157 
158 
159 /* NFS41_UNMOUNT */
parse_unmount(unsigned char * buffer,uint32_t length,nfs41_upcall * upcall)160 static int parse_unmount(unsigned char *buffer, uint32_t length, nfs41_upcall *upcall)
161 {
162     dprintf(1, "parsing NFS41_UNMOUNT: root=%p\n", upcall->root_ref);
163     return ERROR_SUCCESS;
164 }
165 
handle_unmount(nfs41_upcall * upcall)166 static int handle_unmount(nfs41_upcall *upcall)
167 {
168     /* release the original reference from nfs41_root_create() */
169     nfs41_root_deref(upcall->root_ref);
170     return ERROR_SUCCESS;
171 }
172 
173 const nfs41_upcall_op nfs41_op_unmount = {
174     parse_unmount,
175     handle_unmount
176 };
177