Cthulhu  0.2.10
Cthulhu compiler collection
io.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "io/io.h"
4 #include "io/impl.h"
5 
6 #include "os/os.h"
7 #include "base/panic.h"
8 #include "std/str.h"
9 #include "arena/arena.h"
10 
11 static os_error_t impl_close(io_t *io)
12 {
13  CTASSERT(io != NULL);
14 
15  os_error_t err = io_error(io);
16  if (err != eOsSuccess)
17  return err;
18 
19  if (io->cb->fn_close != NULL)
20  return io->cb->fn_close(io);
21 
22  return eOsSuccess;
23 }
24 
26 os_error_t io_free(io_t *io)
27 {
28  os_error_t err = impl_close(io);
29  if (err != eOsSuccess)
30  return err;
31 
32  arena_free(io, sizeof(io_t) + io->cb->size, io->arena);
33 
34  return eOsSuccess;
35 }
36 
38 os_error_t io_close(io_t *io)
39 {
40  return impl_close(io);
41 }
42 
44 size_t io_read(io_t *io, void *dst, size_t size)
45 {
46  CTASSERT(io != NULL);
47  CTASSERTF(io->flags & eOsAccessRead, "cannot io_read(%s). flags did not include eOsAccessRead", io_name(io));
48  CTASSERTF(io->cb->fn_read, "fn_read not provided for `%s`", io->name);
49 
50  return io->cb->fn_read(io, dst, size);
51 }
52 
54 size_t io_write(io_t *io, const void *src, size_t size)
55 {
56  CTASSERT(io != NULL);
57  CTASSERTF(io->flags & eOsAccessWrite, "cannot io_write(%s). flags did not include eOsAccessWrite", io_name(io));
58  CTASSERTF(io->cb->fn_write, "fn_write not provided for `%s`", io->name);
59 
60  return io->cb->fn_write(io, src, size);
61 }
62 
64 size_t io_printf(io_t *io, const char *fmt, ...)
65 {
66  CTASSERT(io != NULL);
67 
68  va_list args;
69  va_start(args, fmt);
70 
71  size_t size = io_vprintf(io, fmt, args);
72 
73  va_end(args);
74 
75  return size;
76 }
77 
79 size_t io_vprintf(io_t *io, const char *fmt, va_list args)
80 {
81  CTASSERT(io != NULL);
82 
83  const io_callbacks_t *cb = io->cb;
84  if (cb->fn_fwrite != NULL)
85  {
86  return cb->fn_fwrite(io, fmt, args);
87  }
88 
89  text_t text = text_vformat(io->arena, fmt, args);
90  return io_write(io, text.text, text.length);
91 }
92 
94 size_t io_size(io_t *io)
95 {
96  CTASSERT(io != NULL);
97  CTASSERTF(io->cb->fn_get_size, "fn_get_size not provided for `%s`", io->name);
98 
99  return io->cb->fn_get_size(io);
100 }
101 
102 STA_DECL
103 size_t io_seek(io_t *io, size_t offset)
104 {
105  CTASSERT(io != NULL);
106  CTASSERTF(io->cb->fn_seek, "fn_seek not provided for `%s`", io->name);
107 
108  return io->cb->fn_seek(io, offset);
109 }
110 
111 STA_DECL
112 const char *io_name(const io_t *io)
113 {
114  CTASSERT(io != NULL);
115 
116  return io->name;
117 }
118 
119 STA_DECL
120 void *io_map(io_t *io, os_protect_t protect)
121 {
122  CTASSERT(io != NULL);
123  CTASSERTF(io->cb->fn_map, "fn_map not provided for `%s`", io->name);
124 
125  CTASSERTF(protect != eOsProtectNone, "cannot io_map(%s). protect is eOsProtectNone", io_name(io));
126 
127  // validate protect flags against access flags
128  if (protect & eOsProtectRead)
129  {
130  CTASSERTF(io->flags & eOsAccessRead, "io_map(%s) flags not readable", io_name(io));
131  }
132 
133  if (protect & eOsProtectWrite)
134  {
135  CTASSERTF(io->flags & eOsAccessWrite, "io_map(%s) flags not writable", io_name(io));
136  }
137 
138  if (io_size(io) == 0) { return ""; }
139 
140  return io->cb->fn_map(io, protect);
141 }
142 
143 STA_DECL
144 os_error_t io_error(const io_t *io)
145 {
146  CTASSERT(io != NULL);
147 
148  return io->error;
149 }
CT_NODISCARD size_t size
Definition: scan.h:128
os_protect_t
file mapping memory protection
Definition: core.h:47
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_ARENA_API void arena_free(STA_RELEASE void *ptr, size_t size, arena_t *arena)
release memory from a custom allocator
#define CTASSERT(expr)
assert a condition, prints the condition as a message
Definition: panic.h:130
#define CTASSERTF(expr,...)
assert a condition with a message and optional format arguments
Definition: panic.h:116
CT_NODISCARD STA_FORMAT_STRING const char CT_NODISCARD CT_STD_API text_t text_vformat(arena_t *arena, const char *fmt, va_list args)
format a string
Definition: str.c:56
CT_NODISCARD STA_FORMAT_STRING const char * fmt
Definition: str.h:68
STA_DECL size_t io_seek(io_t *io, size_t offset)
seek to an absolute offset in a file
Definition: io.c:103
STA_DECL size_t io_vprintf(io_t *io, const char *fmt, va_list args)
vprintf to an io object
Definition: io.c:79
STA_DECL size_t io_read(io_t *io, void *dst, size_t size)
Definition: io.c:44
STA_DECL const char * io_name(const io_t *io)
get the name of an io object
Definition: io.c:112
STA_DECL os_error_t io_error(const io_t *io)
get the last error from the io object
Definition: io.c:144
STA_DECL os_error_t io_free(io_t *io)
Definition: io.c:26
STA_DECL size_t io_size(io_t *io)
get the total size of an io objects contents
Definition: io.c:94
STA_DECL void * io_map(io_t *io, os_protect_t protect)
map an io object into memory maps an io objects entire contents into memory.
Definition: io.c:120
STA_DECL os_error_t io_close(io_t *io)
Definition: io.c:38
STA_DECL size_t io_printf(io_t *io, const char *fmt,...)
Definition: io.c:64
STA_DECL size_t io_write(io_t *io, const void *src, size_t size)
Definition: io.c:54
@ eOsSuccess
Definition: posix.h:24
io callback interface
Definition: impl.h:86
io_seek_t fn_seek
absolute seek callback must always be provided
Definition: impl.h:106
io_map_t fn_map
file map callback must always be provided
Definition: impl.h:110
io_close_t fn_close
close callback optional if backing data does not require lifetime management
Definition: impl.h:114
io_read_t fn_read
read callback may be NULL on non-readable objects
Definition: impl.h:89
io_size_t fn_get_size
total size callback must always be provided
Definition: impl.h:102
size_t size
the size of the io objects private data
Definition: impl.h:117
io_write_t fn_write
write callback may be NULL on non-writable objects
Definition: impl.h:93
io_fwrite_t fn_fwrite
write format callback may be NULL on non-writable objects
Definition: impl.h:98
io object implementation
Definition: impl.h:122
arena_t * arena
the arena this object was allocated from
Definition: impl.h:133
const char * name
the name of this object
Definition: impl.h:136
os_error_t error
the last error set on this object
Definition: impl.h:127
const io_callbacks_t * cb
callback struct
Definition: impl.h:124
os_access_t flags
the access flags for this object
Definition: impl.h:130
a range of text
Definition: text.h:14
size_t length
the number of characters in the text
Definition: text.h:19