source: mainline/uspace/lib/bithenge/src/file.c@ d8023313

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since d8023313 was b7fd2a0, checked in by Jiří Zárevúcky <zarevucky.jiri@…>, 7 years ago

Use errno_t in all uspace and kernel code.

Change type of every variable, parameter and return value that holds an
<errno.h> constant to either errno_t (the usual case), or sys_errno_t
(some places in kernel). This is for the purpose of self-documentation,
as well as for type-checking with a bit of type definition hackery.

Although this is a massive commit, it is a simple text replacement, and thus
is very easy to verify. Simply do the following:

`
git checkout <this commit's hash>
git reset HEAD
git add .
tools/srepl '\berrno_t\b' int
git add .
tools/srepl '\bsys_errno_t\b' sysarg_t
git reset
git diff
`

While this doesn't ensure that the replacements are correct, it does ensure
that the commit doesn't do anything except those replacements. Since errno_t
is typedef'd to int in the usual case (and sys_errno_t to sysarg_t), even if
incorrect, this commit cannot change behavior.

  • Property mode set to 100644
File size: 4.8 KB
RevLine 
[5c5c346a]1/*
2 * Copyright (c) 2012 Sean Bartell
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup bithenge
30 * @{
31 */
32/**
33 * @file
34 * Access files as blobs.
35 * @todo Provide more information about the file.
36 */
37
38#include <assert.h>
39#include <errno.h>
[50985c34]40#include <stdio.h>
[5c5c346a]41#include <stdlib.h>
[23a0368]42#include <vfs/vfs.h>
[8d2dd7f2]43#include <stddef.h>
[6cd10ac]44#include "common.h"
[8fc0f47c]45#include <bithenge/blob.h>
46#include <bithenge/file.h>
[5c5c346a]47
48typedef struct {
49 bithenge_blob_t base;
50 int fd;
51 aoff64_t size; // needed by file_read()
[50985c34]52 bool needs_close;
[5c5c346a]53} file_blob_t;
54
[978ccaf1]55static inline file_blob_t *blob_as_file(bithenge_blob_t *base)
[5c5c346a]56{
57 return (file_blob_t *)base;
58}
59
[978ccaf1]60static inline bithenge_blob_t *file_as_blob(file_blob_t *blob)
[5c5c346a]61{
62 return &blob->base;
63}
64
[b7fd2a0]65static errno_t file_size(bithenge_blob_t *base, aoff64_t *size)
[5c5c346a]66{
[978ccaf1]67 file_blob_t *blob = blob_as_file(base);
[5c5c346a]68 *size = blob->size;
69 return EOK;
70}
71
[b7fd2a0]72static errno_t file_read(bithenge_blob_t *base, aoff64_t offset, char *buffer,
[5c5c346a]73 aoff64_t *size)
74{
[978ccaf1]75 file_blob_t *blob = blob_as_file(base);
[5c5c346a]76 if (offset > blob->size)
77 return ELIMIT;
[da0fef6]78
[8e3498b]79 size_t amount_read;
[b7fd2a0]80 errno_t rc = vfs_read(blob->fd, &offset, buffer, *size, &amount_read);
[8e3498b]81 if (rc != EOK)
[58898d1d]82 return errno;
83 *size += amount_read;
[da0fef6]84 return EOK;
[5c5c346a]85}
86
[978ccaf1]87static void file_destroy(bithenge_blob_t *base)
[5c5c346a]88{
[978ccaf1]89 file_blob_t *blob = blob_as_file(base);
[9c4cf0d]90 vfs_put(blob->fd);
[5c5c346a]91 free(blob);
92}
93
94static const bithenge_random_access_blob_ops_t file_ops = {
95 .size = file_size,
96 .read = file_read,
97 .destroy = file_destroy,
98};
99
[b7fd2a0]100static errno_t new_file_blob(bithenge_node_t **out, int fd, bool needs_close)
[5c5c346a]101{
102 assert(out);
103
104 struct stat stat;
[b7fd2a0]105 errno_t rc = vfs_stat(fd, &stat);
[23a0368]106 if (rc != EOK) {
[50985c34]107 if (needs_close)
[9c4cf0d]108 vfs_put(fd);
[5c5c346a]109 return rc;
110 }
111
112 // Create blob
113 file_blob_t *blob = malloc(sizeof(*blob));
114 if (!blob) {
[50985c34]115 if (needs_close)
[9c4cf0d]116 vfs_put(fd);
[5c5c346a]117 return ENOMEM;
118 }
[978ccaf1]119 rc = bithenge_init_random_access_blob(file_as_blob(blob), &file_ops);
[5c5c346a]120 if (rc != EOK) {
121 free(blob);
[50985c34]122 if (needs_close)
[9c4cf0d]123 vfs_put(fd);
[5c5c346a]124 return rc;
125 }
126 blob->fd = fd;
[da0fef6]127#ifdef __HELENOS__
[5c5c346a]128 blob->size = stat.size;
[da0fef6]129#else
130 blob->size = stat.st_size;
131#endif
[50985c34]132 blob->needs_close = needs_close;
[978ccaf1]133 *out = bithenge_blob_as_node(file_as_blob(blob));
[5c5c346a]134
135 return EOK;
136}
137
[50985c34]138/** Create a blob for a file. The blob must be freed with @a
[8375d0eb]139 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]140 * @param[out] out Stores the created blob.
141 * @param filename The name of the file.
142 * @return EOK on success or an error code from errno.h. */
[b7fd2a0]143errno_t bithenge_new_file_blob(bithenge_node_t **out, const char *filename)
[50985c34]144{
145 assert(filename);
146
[f77c1c9]147 int fd;
[b7fd2a0]148 errno_t rc = vfs_lookup_open(filename, WALK_REGULAR, MODE_READ, &fd);
[f77c1c9]149 if (rc != EOK)
150 return rc;
[50985c34]151
152 return new_file_blob(out, fd, true);
153}
154
155/** Create a blob for a file descriptor. The blob must be freed with @a
[8375d0eb]156 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]157 * @param[out] out Stores the created blob.
158 * @param fd The file descriptor.
159 * @return EOK on success or an error code from errno.h. */
[b7fd2a0]160errno_t bithenge_new_file_blob_from_fd(bithenge_node_t **out, int fd)
[50985c34]161{
162 return new_file_blob(out, fd, false);
163}
164
165/** Create a blob for a file pointer. The blob must be freed with @a
[8375d0eb]166 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]167 * @param[out] out Stores the created blob.
168 * @param file The file pointer.
169 * @return EOK on success or an error code from errno.h. */
[b7fd2a0]170errno_t bithenge_new_file_blob_from_file(bithenge_node_t **out, FILE *file)
[50985c34]171{
172 int fd = fileno(file);
173 if (fd < 0)
174 return errno;
175 return new_file_blob(out, fd, false);
176}
177
[5c5c346a]178/** @}
179 */
Note: See TracBrowser for help on using the repository browser.