• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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