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

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 6afc9d7 was 6afc9d7, checked in by Jiri Svoboda <jiri@…>, 10 years ago

UNIX-like I/O functions should use errno to return error code for many reasons.

  • Property mode set to 100644
File size: 5.0 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>
40#include <fcntl.h>
[50985c34]41#include <stdio.h>
[5c5c346a]42#include <stdlib.h>
43#include <sys/stat.h>
[da0fef6]44#include <sys/types.h>
45#include <unistd.h>
[6cd10ac]46#include "common.h"
[8fc0f47c]47#include <bithenge/blob.h>
48#include <bithenge/file.h>
[5c5c346a]49
50typedef struct {
51 bithenge_blob_t base;
52 int fd;
53 aoff64_t size; // needed by file_read()
[50985c34]54 bool needs_close;
[5c5c346a]55} file_blob_t;
56
[978ccaf1]57static inline file_blob_t *blob_as_file(bithenge_blob_t *base)
[5c5c346a]58{
59 return (file_blob_t *)base;
60}
61
[978ccaf1]62static inline bithenge_blob_t *file_as_blob(file_blob_t *blob)
[5c5c346a]63{
64 return &blob->base;
65}
66
67static int file_size(bithenge_blob_t *base, aoff64_t *size)
68{
[978ccaf1]69 file_blob_t *blob = blob_as_file(base);
[5c5c346a]70 *size = blob->size;
71 return EOK;
72}
73
74static int file_read(bithenge_blob_t *base, aoff64_t offset, char *buffer,
75 aoff64_t *size)
76{
[978ccaf1]77 file_blob_t *blob = blob_as_file(base);
[5c5c346a]78 if (offset > blob->size)
79 return ELIMIT;
[da0fef6]80 if (lseek(blob->fd, offset, SEEK_SET) < 0)
[1a3b953]81 return errno == EINVAL ? EIO : errno;
[da0fef6]82
83 ssize_t amount_read;
84 aoff64_t remaining_size = *size;
85 *size = 0;
86 do {
87 amount_read = read(blob->fd, buffer, remaining_size);
88 if (amount_read < 0)
89 return errno;
90 buffer += amount_read;
91 *size += amount_read;
92 remaining_size -= amount_read;
93 } while (remaining_size && amount_read);
94 return EOK;
[5c5c346a]95}
96
[978ccaf1]97static void file_destroy(bithenge_blob_t *base)
[5c5c346a]98{
[978ccaf1]99 file_blob_t *blob = blob_as_file(base);
[5c5c346a]100 close(blob->fd);
101 free(blob);
102}
103
104static const bithenge_random_access_blob_ops_t file_ops = {
105 .size = file_size,
106 .read = file_read,
107 .destroy = file_destroy,
108};
109
[5c925ce]110static int new_file_blob(bithenge_node_t **out, int fd, bool needs_close)
[5c5c346a]111{
112 assert(out);
113
114 struct stat stat;
115 int rc = fstat(fd, &stat);
[6afc9d7]116 if (rc != 0) {
[50985c34]117 if (needs_close)
118 close(fd);
[5c5c346a]119 return rc;
120 }
121
122 // Create blob
123 file_blob_t *blob = malloc(sizeof(*blob));
124 if (!blob) {
[50985c34]125 if (needs_close)
126 close(fd);
[5c5c346a]127 return ENOMEM;
128 }
[978ccaf1]129 rc = bithenge_init_random_access_blob(file_as_blob(blob), &file_ops);
[5c5c346a]130 if (rc != EOK) {
131 free(blob);
[50985c34]132 if (needs_close)
133 close(fd);
[5c5c346a]134 return rc;
135 }
136 blob->fd = fd;
[da0fef6]137#ifdef __HELENOS__
[5c5c346a]138 blob->size = stat.size;
[da0fef6]139#else
140 blob->size = stat.st_size;
141#endif
[50985c34]142 blob->needs_close = needs_close;
[978ccaf1]143 *out = bithenge_blob_as_node(file_as_blob(blob));
[5c5c346a]144
145 return EOK;
146}
147
[50985c34]148/** Create a blob for a file. The blob must be freed with @a
[8375d0eb]149 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]150 * @param[out] out Stores the created blob.
151 * @param filename The name of the file.
152 * @return EOK on success or an error code from errno.h. */
[5c925ce]153int bithenge_new_file_blob(bithenge_node_t **out, const char *filename)
[50985c34]154{
155 assert(filename);
156
157 int fd = open(filename, O_RDONLY);
158 if (fd < 0)
[6afc9d7]159 return errno;
[50985c34]160
161 return new_file_blob(out, fd, true);
162}
163
164/** Create a blob for a file descriptor. The blob must be freed with @a
[8375d0eb]165 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]166 * @param[out] out Stores the created blob.
167 * @param fd The file descriptor.
168 * @return EOK on success or an error code from errno.h. */
[5c925ce]169int bithenge_new_file_blob_from_fd(bithenge_node_t **out, int fd)
[50985c34]170{
171 return new_file_blob(out, fd, false);
172}
173
174/** Create a blob for a file pointer. The blob must be freed with @a
[8375d0eb]175 * bithenge_node_t::bithenge_node_destroy after it is used.
[50985c34]176 * @param[out] out Stores the created blob.
177 * @param file The file pointer.
178 * @return EOK on success or an error code from errno.h. */
[5c925ce]179int bithenge_new_file_blob_from_file(bithenge_node_t **out, FILE *file)
[50985c34]180{
181 int fd = fileno(file);
182 if (fd < 0)
183 return errno;
184 return new_file_blob(out, fd, false);
185}
186
[5c5c346a]187/** @}
188 */
Note: See TracBrowser for help on using the repository browser.