Cthulhu  0.2.10
Cthulhu compiler collection
file.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "io/impl.h"
4 #include "io/impl/file.h"
5 
6 #include "base/panic.h"
7 #include "arena/arena.h"
8 
9 #include "os/os.h"
10 
11 static io_file_impl_t *fd_data(io_t *self)
12 {
13  io_file_impl_t *file = io_data(self);
14  return file;
15 }
16 
17 static size_t fd_read(io_t *self, void *dst, size_t size)
18 {
19  io_file_impl_t *file = fd_data(self);
20 
21  size_t read = 0;
22  self->error = os_file_read(&file->file, dst, size, &read);
23  if (self->error != 0) return SIZE_MAX;
24 
25  return read;
26 }
27 
28 static size_t fd_write(io_t *self, const void *src, size_t size)
29 {
30  io_file_impl_t *file = fd_data(self);
31 
32  size_t written = 0;
33  self->error = os_file_write(&file->file, src, size, &written);
34  if (self->error != 0) return SIZE_MAX;
35 
36  return written;
37 }
38 
39 static size_t fd_size(io_t *self)
40 {
41  io_file_impl_t *file = fd_data(self);
42 
43  size_t size = 0;
44  self->error = os_file_size(&file->file, &size);
45  if (self->error != 0) return SIZE_MAX;
46 
47  return size;
48 }
49 
50 static size_t fd_seek(io_t *self, size_t offset)
51 {
52  io_file_impl_t *file = fd_data(self);
53 
54  size_t seek = 0;
55  self->error = os_file_seek(&file->file, offset, &seek);
56  if (self->error != 0) return SIZE_MAX;
57 
58  return seek;
59 }
60 
61 static void *fd_map(io_t *self, os_protect_t protect)
62 {
63  io_file_impl_t *file = fd_data(self);
64 
65  if (!os_mapping_active(&file->mapping))
66  {
67  size_t size = io_size(self);
68  self->error = os_file_map(&file->file, protect, size, &file->mapping);
69  if (self->error != 0) return NULL;
70  }
71 
72  return os_mapping_data(&file->mapping);
73 }
74 
75 static os_error_t fd_close(io_t *self)
76 {
77  io_file_impl_t *file = fd_data(self);
78  os_error_t err = 0;
79 
80  if (os_mapping_active(&file->mapping))
81  {
82  err = os_unmap(&file->mapping);
83  if (err != eOsSuccess)
84  return err;
85  }
86 
87  return os_file_close(&file->file);
88 }
89 
90 static const io_callbacks_t kFileCallbacks = {
91  .fn_read = fd_read,
92  .fn_write = fd_write,
93 
94  .fn_get_size = fd_size,
95  .fn_seek = fd_seek,
96 
97  .fn_map = fd_map,
98  .fn_close = fd_close,
99 
100  .size = sizeof(io_file_impl_t),
101 };
102 
103 static io_t *impl_file_init(void *buffer, const char *path, os_access_t mode, arena_t *arena)
104 {
105  CTASSERT(path != NULL);
106 
107  os_file_t fd = { 0 };
108  os_error_t err = os_file_open(path, mode, &fd);
109 
110  io_file_impl_t impl = {
111  .file = fd,
112  };
113 
114  io_t *io = io_init(buffer, &kFileCallbacks, mode, path, &impl, arena);
115  io->error = err;
116 
117  return io;
118 }
119 
123 
124 STA_DECL
125 io_t *io_file(const char *path, os_access_t mode, arena_t *arena)
126 {
127  void *buffer = ARENA_MALLOC(IO_FILE_SIZE, path, NULL, arena);
128  return impl_file_init(buffer, path, mode, arena);
129 }
130 
134 
135 STA_DECL
136 io_t *io_file_init(void *buffer, const char *path, os_access_t mode)
137 {
138  return impl_file_init(buffer, path, mode, NULL);
139 }
CT_NODISCARD size_t size
Definition: scan.h:128
os_protect_t
file mapping memory protection
Definition: core.h:47
os_access_t
file access mode
Definition: core.h:38
#define IO_FILE_SIZE
Definition: file.h:25
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_PUREFN CT_IO_API void * io_data(io_t *io)
get the user data from an io object
Definition: common.c:10
CT_IO_API io_t * io_init(STA_WRITES(sizeof(io_t)+cb->size) void *buffer, const io_callbacks_t *cb, os_access_t flags, const char *name, STA_READS(cb->size) const void *data, arena_t *arena)
initialize an IO object for a given interface
CT_NODISCARD CT_IO_API size_t io_size(io_t *io)
get the total size of an io objects contents
Definition: io.c:94
#define ARENA_MALLOC(size, name, parent, arena)
allocate memory from a custom allocator
Definition: arena.h:392
#define CTASSERT(expr)
assert a condition, prints the condition as a message
Definition: panic.h:130
STA_DECL io_t * io_file(const char *path, os_access_t mode, arena_t *arena)
create an IO object from a file
Definition: file.c:125
STA_DECL io_t * io_file_init(void *buffer, const char *path, os_access_t mode)
Definition: file.c:136
STA_DECL os_error_t os_file_read(os_file_t *file, void *buffer, size_t size, size_t *actual)
Definition: file.c:82
STA_DECL os_error_t os_file_seek(os_file_t *file, size_t offset, size_t *actual)
Definition: file.c:165
STA_DECL os_error_t os_file_write(os_file_t *file, const void *buffer, size_t size, size_t *actual)
Definition: file.c:104
STA_DECL os_error_t os_file_size(os_file_t *file, size_t *actual)
Definition: file.c:126
CT_OS_API os_error_t os_file_close(STA_RELEASE os_file_t *file)
close a file
CT_NODISCARD CT_PUREFN CT_OS_API bool os_mapping_active(const os_mapping_t *mapping)
does the mapping object contain a valid mapping checks if the mapping data exists,...
Definition: os.c:175
CT_OS_API os_error_t os_file_open(const char *path, os_access_t access, OUT_NOTNULL os_file_t *file)
file api
CT_OS_API os_error_t os_file_map(os_file_t *file, os_protect_t protect, size_t size, OUT_NOTNULL os_mapping_t *mapping)
map a file into memory
CT_OS_API os_error_t os_unmap(INOUT_NOTNULL os_mapping_t *mapping)
unmap a file from memory
CT_NODISCARD CT_OS_API void * os_mapping_data(os_mapping_t *mapping)
get the data of a file mapping
Definition: os.c:166
@ eOsSuccess
Definition: posix.h:24
an allocator object
Definition: arena.h:86
io callback interface
Definition: impl.h:86
io_read_t fn_read
read callback may be NULL on non-readable objects
Definition: impl.h:89
a file descriptor
Definition: file.h:17
os_mapping_t mapping
memory mapping
Definition: file.h:22
os_file_t file
native file descriptor
Definition: file.h:19
io object implementation
Definition: impl.h:122
os_error_t error
the last error set on this object
Definition: impl.h:127
a file handle
Definition: os.h:40