1 /****************************************************************************
2 * fs/nfs/nfs_util.c
3 *
4 * Licensed to the Apache Software Foundation (ASF) under one or more
5 * contributor license agreements. See the NOTICE file distributed with
6 * this work for additional information regarding copyright ownership. The
7 * ASF licenses this file to you under the Apache License, Version 2.0 (the
8 * "License"); you may not use this file except in compliance with the
9 * License. You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
15 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
16 * License for the specific language governing permissions and limitations
17 * under the License.
18 *
19 ****************************************************************************/
20
21 /****************************************************************************
22 * Included Files
23 ****************************************************************************/
24
25 #include <sys/types.h>
26 #include <sys/time.h>
27 #include <stdlib.h>
28 #include <string.h>
29 #include <assert.h>
30 #include <pthread.h>
31 #include "vfs_config.h"
32 #include "dirent.h"
33 #include "rpc.h"
34 #include "nfs.h"
35 #include "nfs_node.h"
36 #include "xdr_subs.h"
37 #include "nfs.h"
38 #undef OK
39 #define OK 0
40
41 /****************************************************************************
42 * Private Functions
43 ****************************************************************************/
44
nfs_pathsegment(const char ** path,char * buffer,char * terminator)45 static inline int nfs_pathsegment(const char **path, char *buffer,
46 char *terminator)
47 {
48 const char *src = *path;
49 char *dest = buffer;
50 int nbytes = 0;
51 char ch;
52
53 /* Loop until the name is successfully parsed or an error occurs */
54
55 for (; ; )
56 {
57 /* Get the next byte from the path */
58
59 ch = *src++;
60
61 /* Check if this the last byte in this segment name */
62
63 if (ch == '\0' || ch == '/')
64 {
65 /* This logic just supports "//" sequences in the path name */
66
67 if (ch == '\0' || nbytes > 0)
68 {
69 /* NULL terminate the parsed path segment */
70
71 *dest = '\0';
72
73 /* Return next path and the terminating character */
74
75 *terminator = ch;
76 *path = src;
77 return OK;
78 }
79
80 /* Just skip over any leading '/' characters */
81 }
82 else if (nbytes >= NAME_MAX)
83 {
84 nfs_debug_error("File name segment is too long: %d\n", *path);
85 return ENAMETOOLONG;
86 }
87 else
88 {
89 /* Save next character in the accumulated name */
90
91 *dest++ = ch;
92 nbytes++;
93 }
94 }
95 }
96
97 /****************************************************************************
98 * Public Functions
99 ****************************************************************************/
100
101 /****************************************************************************
102 * Name: nfs_mux_take
103 ****************************************************************************/
104
nfs_mux_take(struct nfsmount * nmp)105 void nfs_mux_take(struct nfsmount *nmp)
106 {
107 (void)pthread_mutex_lock(&nmp->nm_mux);
108 }
109
110 /****************************************************************************
111 * Name: nfs_mux_release
112 ****************************************************************************/
113
nfs_mux_release(struct nfsmount * nmp)114 void nfs_mux_release(struct nfsmount *nmp)
115 {
116 (void)pthread_mutex_unlock(&nmp->nm_mux);
117 }
118
119 /****************************************************************************
120 * Name: nfs_checkmount
121 *
122 * Description: Check if the mountpoint is still valid.
123 *
124 * The caller should hold the mountpoint semaphore
125 *
126 ****************************************************************************/
127
nfs_checkmount(struct nfsmount * nmp)128 int nfs_checkmount(struct nfsmount *nmp)
129 {
130 struct nfsnode *file;
131
132 /* If the nm_mounted flag is false, then we have already handled the loss
133 * of the mount.
134 */
135
136 DEBUGASSERT(nmp);
137 if (!nmp->nm_mounted)
138 {
139 /* Make sure that this is flagged in every opened file */
140
141 for (file = nmp->nm_head; file; file = file->n_next)
142 {
143 file->n_flags &= ~NFSNODE_OPEN;
144 }
145
146 return ENODEV;
147 }
148
149 return 0;
150 }
151
152 /****************************************************************************
153 * Name: nfs_request
154 *
155 * Description:
156 * Perform the NFS request. On successful receipt, it verifies the NFS level of the
157 * returned values.
158 *
159 * Returned Value:
160 * Zero on success; a positive errno value on failure.
161 *
162 ****************************************************************************/
163
nfs_request(struct nfsmount * nmp,int procnum,void * request,size_t reqlen,void * response,size_t resplen)164 int nfs_request(struct nfsmount *nmp, int procnum,
165 void *request, size_t reqlen,
166 void *response, size_t resplen)
167 {
168 struct rpcclnt *clnt = nmp->nm_rpcclnt;
169 struct nfs_reply_header replyh;
170 int error;
171
172 tryagain:
173 error = rpcclnt_request(clnt, procnum, NFS_PROG, NFS_VER3,
174 request, reqlen, response, resplen);
175 if (error != 0)
176 {
177 nfs_error("rpcclnt_request failed: %d\n", error);
178 return error;
179 }
180
181 memcpy(&replyh, response, sizeof(struct nfs_reply_header));
182
183 if (replyh.nfs_status != 0)
184 {
185 /* NFS_ERRORS are the same as NuttX errno values */
186
187 error = fxdr_unsigned(uint32_t, replyh.nfs_status);
188 return error;
189 }
190
191 if (replyh.rpc_verfi.authtype != 0)
192 {
193 error = fxdr_unsigned(int, replyh.rpc_verfi.authtype);
194
195 if (error == EAGAIN)
196 {
197 error = 0;
198 goto tryagain;
199 }
200
201 nfs_debug_error("NFS error %d from server\n", error);
202 return error;
203 }
204
205 return OK;
206 }
207
208 /****************************************************************************
209 * Name: nfs_lookup
210 *
211 * Description:
212 * Given a directory file handle, and the path to file in the directory,
213 * return the file handle of the path and attributes of both the file and
214 * the directory containing the file.
215 *
216 * NOTE: The LOOKUP call differs from other RPC messages in that the
217 * call message is variable length, depending upon the size of the path
218 * name.
219 *
220 ****************************************************************************/
221
nfs_lookup(struct nfsmount * nmp,const char * filename,struct file_handle * fhandle,struct nfs_fattr * obj_attributes,struct nfs_fattr * dir_attributes)222 int nfs_lookup(struct nfsmount *nmp, const char *filename,
223 struct file_handle *fhandle,
224 struct nfs_fattr *obj_attributes,
225 struct nfs_fattr *dir_attributes)
226 {
227 uint32_t *ptr = NULL;
228 uint32_t value;
229 int reqlen;
230 int namelen;
231 int error = 0;
232
233 DEBUGASSERT(nmp && filename && fhandle);
234
235 /* Get the length of the string to be sent */
236
237 namelen = strlen(filename);
238 if (namelen > NAME_MAX)
239 {
240 nfs_debug_error("Length of the string is too long: %d\n", namelen);
241 return E2BIG;
242 }
243
244 /* Initialize the request */
245
246 ptr = (uint32_t *)&nmp->nm_msgbuffer.lookup.lookup;
247 reqlen = 0;
248
249 /* Copy the variable length, directory file handle */
250
251 *ptr++ = txdr_unsigned(fhandle->length);
252 reqlen += sizeof(uint32_t);
253
254 memcpy(ptr, &fhandle->handle, fhandle->length);
255 reqlen += fhandle->length;
256 ptr += uint32_increment(fhandle->length);
257
258 /* Copy the variable-length file name */
259
260 *ptr++ = txdr_unsigned(namelen);
261 reqlen += sizeof(uint32_t);
262
263 memcpy(ptr, filename, namelen);
264 reqlen += uint32_alignup(namelen);
265
266 /* Request LOOKUP from the server */
267
268 nfs_statistics(NFSPROC_LOOKUP);
269 error = nfs_request(nmp, NFSPROC_LOOKUP,
270 (void *)&nmp->nm_msgbuffer.lookup, reqlen,
271 (void *)nmp->nm_iobuffer, nmp->nm_buflen);
272
273 if (error)
274 {
275 nfs_debug_error("nfs_request failed: %d\n", error);
276 return error;
277 }
278
279 /* Return the data to the caller's buffers. NOTE: Here we ignore the
280 * the exact layout of the rpc_reply_lookup structure. File handles
281 * may differ in size whereas struct rpc_reply_lookup uses a fixed size.
282 */
283
284 ptr = (uint32_t *)&((struct rpc_reply_lookup *)nmp->nm_iobuffer)->lookup;
285
286 /* Get the length of the file handle */
287
288 value = *ptr++;
289 value = fxdr_unsigned(uint32_t, value);
290 if (value > NFSX_V3FHMAX)
291 {
292 nfs_debug_error("Bad file handle length: %d\n", value);
293 return EIO;
294 }
295
296 /* Return the file handle */
297
298 fhandle->length = value;
299 memcpy(&fhandle->handle, ptr, value);
300 ptr += uint32_increment(value);
301
302 /* Check if there are object attributes and, if so, copy them to the user
303 * buffer
304 */
305
306 value = *ptr++;
307 if (value)
308 {
309 if (obj_attributes)
310 {
311 memcpy(obj_attributes, ptr, sizeof(struct nfs_fattr));
312 }
313
314 ptr += uint32_increment(sizeof(struct nfs_fattr));
315 }
316
317 /* Check if there are directory attributes and, if so, copy them to the
318 * user buffer
319 */
320
321 value = *ptr++;
322 if (value && dir_attributes)
323 {
324 memcpy(dir_attributes, ptr, sizeof(struct nfs_fattr));
325 }
326
327 return OK;
328 }
329
330 /****************************************************************************
331 * Name: nfs_findnode
332 *
333 * Description:
334 * Given a path to something that may or may not be in the file system,
335 * return the handle of the directory entry of the requested object.
336 *
337 * Returned Value:
338 * Zero on success; a positive errno value on failure.
339 *
340 ****************************************************************************/
341
nfs_findnode(struct nfsmount * nmp,const char * relpath,struct file_handle * fhandle,struct nfs_fattr * obj_attributes,struct nfs_fattr * dir_attributes)342 int nfs_findnode(struct nfsmount *nmp, const char *relpath,
343 struct file_handle *fhandle,
344 struct nfs_fattr *obj_attributes,
345 struct nfs_fattr *dir_attributes)
346 {
347 const char *path = relpath;
348 char buffer[NAME_MAX + 1];
349 char terminator;
350 uint32_t tmp;
351 int error;
352
353 /* Start with the file handle of the root directory. */
354
355 fhandle->length = nmp->nm_fhsize;
356 memcpy(&fhandle->handle, &nmp->nm_fh, nmp->nm_fhsize);
357
358 /* If no path was provided, then the root directory must be exactly what
359 * the caller is looking for.
360 */
361
362 if (*path == '\0' || strlen(path) == 0)
363 {
364 /* Return the root directory attributes */
365
366 if (obj_attributes)
367 {
368 memcpy(obj_attributes, &nmp->nm_fattr, sizeof(struct nfs_fattr));
369 }
370
371 if (dir_attributes)
372 {
373 memcpy(dir_attributes, &nmp->nm_fattr, sizeof(struct nfs_fattr));
374 }
375
376 return OK;
377 }
378
379 /* This is not the root directory. Loop until the directory entry corresponding
380 * to the path is found.
381 */
382
383 for (; ; )
384 {
385 /* Extract the next path segment name. */
386
387 error = nfs_pathsegment(&path, buffer, &terminator);
388 if (error != OK)
389 {
390 /* The filename segment contains is too long. */
391
392 nfs_debug_error("nfs_pathsegment of \"%s\" failed after \"%s\": %d\n",
393 relpath, buffer, error);
394 return error;
395 }
396
397 /* Look-up this path segment */
398
399 error = nfs_lookup(nmp, buffer, fhandle, obj_attributes, dir_attributes);
400 if (error != OK)
401 {
402 nfs_debug_error("nfs_lookup of \"%s\" failed at \"%s\": %d\n",
403 relpath, buffer, error);
404 return error;
405 }
406
407 /* If the terminator character in the path was the end of the string
408 * then we have successfully found the directory entry that describes
409 * the path.
410 */
411
412 if (!terminator)
413 {
414 /* Return success meaning that the description the matching
415 * directory entry is in fhandle, obj_attributes, and dir_attributes.
416 */
417
418 return OK;
419 }
420
421 /* No.. then we have found one of the intermediate directories on
422 * the way to the final path target. In this case, make sure
423 * the thing that we found is, indeed, a directory.
424 */
425
426 tmp = fxdr_unsigned(uint32_t, obj_attributes->fa_type);
427 if (tmp != NFDIR)
428 {
429 /* Ooops.. we found something else */
430
431 nfs_debug_error("Intermediate segment \"%s\" of \'%s\" is not a directory\n",
432 buffer, path);
433 return ENOTDIR;
434 }
435 }
436 }
437
438 /****************************************************************************
439 * Name: nfs_finddir
440 *
441 * Description:
442 * Given a path to something that may or may not be in the file system,
443 * return the handle of the entry of the directory containing the requested
444 * object.
445 *
446 * Returned Value:
447 * Zero on success; a positive errno value on failure.
448 *
449 ****************************************************************************/
450
nfs_finddir(struct nfsmount * nmp,const char * relpath,struct file_handle * fhandle,struct nfs_fattr * attributes,char * filename)451 int nfs_finddir(struct nfsmount *nmp, const char *relpath,
452 struct file_handle *fhandle,
453 struct nfs_fattr *attributes, char *filename)
454 {
455 const char *path = relpath;
456 uint32_t tmp;
457 char terminator;
458 int error;
459
460 /* Verify that a path was provided */
461
462 if (*path == '\0' || strlen(path) == 0)
463 {
464 /* Return the root directory attributes */
465
466 return ENOENT;
467 }
468
469 /* Start with the file handle of the root directory. */
470
471 fhandle->length = nmp->nm_fhsize;
472 memcpy(&fhandle->handle, &nmp->nm_fh, nmp->nm_fhsize);
473 memcpy(attributes, &nmp->nm_fattr, sizeof(struct nfs_fattr));
474
475 /* Loop until the directory entry containing the path is found. */
476
477 for (; ; )
478 {
479 /* Extract the next path segment name. */
480
481 error = nfs_pathsegment(&path, filename, &terminator);
482 if (error != OK)
483 {
484 /* The filename segment contains is too long. */
485
486 nfs_debug_error("nfs_pathsegment of \"%s\" failed after \"%s\": %d\n",
487 relpath, filename, error);
488 return error;
489 }
490
491 /* If the terminator character in the path was the end of the string
492 * then we have successfully found the directory that contains the name
493 * of interest.
494 */
495
496 if (!terminator)
497 {
498 /* Return success meaning that the description of the directory
499 * containing the object is in fhandle and attributes.
500 */
501
502 return OK;
503 }
504
505 /* Look-up the next path segment */
506
507 error = nfs_lookup(nmp, filename, fhandle, attributes, NULL);
508 if (error != OK)
509 {
510 nfs_debug_error("fs_lookup of \"%s\" failed at \"%s\": %d\n",
511 relpath, filename, error);
512 return error;
513 }
514
515 /* Make sure the thing that we found is, indeed, a directory. */
516
517 tmp = fxdr_unsigned(uint32_t, attributes->fa_type);
518 if (tmp != NFDIR)
519 {
520 /* Ooops.. we found something else */
521
522 nfs_debug_error("Intermediate segment \"%s\" of \'%s\" is not a directory\n",
523 filename, path);
524 return ENOTDIR;
525 }
526 }
527 }
528
529 /****************************************************************************
530 * Name: nfs_attrupdate
531 *
532 * Description:
533 * Update file attributes on write or after the file is modified.
534 *
535 * Returned Value:
536 * None.
537 *
538 ****************************************************************************/
539
nfs_attrupdate(struct nfsnode * np,struct nfs_fattr * attributes)540 void nfs_attrupdate(struct nfsnode *np, struct nfs_fattr *attributes)
541 {
542 struct timespec ts;
543
544 /* Save a few of the files attribute values in file structure (host order) */
545
546 np->n_type = fxdr_unsigned(uint8_t, attributes->fa_type);
547 np->n_mode = fxdr_unsigned(uint16_t, attributes->fa_mode);
548 np->n_size = fxdr_hyper(&attributes->fa_size);
549
550 fxdr_nfsv3time(&attributes->fa_mtime, &ts);
551 np->n_timestamp.tv_sec = ts.tv_sec;
552 np->n_timestamp.tv_nsec = ts.tv_nsec;
553
554 fxdr_nfsv3time(&attributes->fa_ctime, &ts);
555 np->n_ctime = ts.tv_sec;
556 }
557