• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /****************************************************************************
2  * fs/vfs/fs_pread.c
3  *
4  * Copyright (c) 2023 Huawei Device Co., Ltd. All rights reserved.
5  * Based on NuttX originally from nuttx source (nuttx/fs/ and nuttx/drivers/)
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * 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,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  ****************************************************************************/
20 
21 /****************************************************************************
22  * Included Files
23  ****************************************************************************/
24 
25 #include "vfs_config.h"
26 
27 #include "sys/types.h"
28 #include "unistd.h"
29 #include "errno.h"
30 
31 #include "fs/file.h"
32 
33 /****************************************************************************
34  * Public Functions
35  ****************************************************************************/
36 
37 /****************************************************************************
38  * Name: file_pread
39  *
40  * Description:
41  *   Equivalent to the standard pread function except that is accepts a
42  *   struct file instance instead of a file descriptor.  Currently used
43  *   only by aio_read();
44  *
45  ****************************************************************************/
46 
file_pread(struct file * filep,void * buf,size_t nbytes,off_t offset)47 ssize_t file_pread(struct file *filep, void *buf, size_t nbytes,
48                    off_t offset)
49 {
50   off_t savepos;
51   off_t pos;
52   ssize_t ret;
53   int errcode;
54 
55   /* Perform the seek to the current position.  This will not move the
56    * file pointer, but will return its current setting
57    */
58 
59   savepos = file_seek(filep, 0, SEEK_CUR);
60   if (savepos == (off_t)-1)
61     {
62       /* file_seek might fail if this if the media is not seekable */
63 
64       return VFS_ERROR;
65     }
66 
67   /* Then seek to the correct position in the file */
68 
69   pos = file_seek(filep, offset, SEEK_SET);
70   if (pos == (off_t)-1)
71     {
72       /* This might fail is the offset is beyond the end of file */
73 
74       return VFS_ERROR;
75     }
76 
77   /* Then perform the read operation */
78 
79   ret = file_read(filep, buf, nbytes);
80   errcode = get_errno();
81 
82   /* Restore the file position */
83 
84   pos = file_seek(filep, savepos, SEEK_SET);
85   if (pos == (off_t)-1 && ret >= 0)
86     {
87       /* This really should not fail */
88 
89       return VFS_ERROR;
90     }
91 
92   if (errcode != 0)
93     {
94       set_errno(errcode);
95     }
96   return ret;
97 }
98 
99 /****************************************************************************
100  * Name: pread
101  *
102  * Description:
103  *   The pread() function performs the same action as read(), except that it
104  *   reads from a given position in the file without changing the file
105  *   pointer. The first three arguments to pread() are the same as read()
106  *   with the addition of a fourth argument offset for the desired position
107  *   inside the file. An attempt to perform a pread() on a file that is
108  *   incapable of seeking results in an error.
109  *
110  *   NOTE: This function could have been wholly implemented within libc but
111  *   it is not.  Why?  Because if pread were implemented in libc, it would
112  *   require four system calls.  If it is implemented within the kernel,
113  *   only three.
114  *
115  * Input Parameters:
116  *   file     File structure instance
117  *   buf      User-provided to save the data
118  *   nbytes   The maximum size of the user-provided buffer
119  *   offset   The file offset
120  *
121  * Returned Value:
122  *   The positive non-zero number of bytes read on success, 0 on if an
123  *   end-of-file condition, or -1 on failure with errno set appropriately.
124  *   See read() return values
125  *
126  ****************************************************************************/
127 
pread(int fd,void * buf,size_t nbytes,off_t offset)128 ssize_t pread(int fd, void *buf, size_t nbytes, off_t offset)
129 {
130   struct file *filep;
131 
132   /* Get the file structure corresponding to the file descriptor. */
133 
134   int ret = fs_getfilep(fd, &filep);
135   if (ret < 0)
136     {
137       /* The errno value has already been set */
138       set_errno(-ret);
139       return (ssize_t)VFS_ERROR;
140     }
141 
142   if (filep->f_oflags & O_DIRECTORY)
143     {
144       set_errno(EBADF);
145       return (ssize_t)VFS_ERROR;
146     }
147 
148   /* Let file_pread do the real work */
149 
150   return file_pread(filep, buf, nbytes, offset);
151 }
152