Cthulhu  0.2.10
Cthulhu compiler collection
buffer.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/buffer.h"
5 #include "os/os.h"
6 
7 #include "core/macros.h"
8 #include "arena/arena.h"
9 
10 #include "base/panic.h"
11 #include "base/util.h"
12 
13 static io_buffer_impl_t *mem_data(io_t *self)
14 {
15  return io_data(self);
16 }
17 
18 static size_t mem_read(io_t *self, void *dst, size_t size)
19 {
20  io_buffer_impl_t *mem = mem_data(self);
21  size_t len = CT_MIN(size, mem->used - mem->offset);
22  ctu_memcpy(dst, mem->data + mem->offset, len);
23  mem->offset += len;
24  return len;
25 }
26 
27 static size_t mem_write(io_t *self, const void *src, size_t size)
28 {
29  io_buffer_impl_t *mem = mem_data(self);
30  mem->used = CT_MAX(mem->used, mem->offset + size);
31  if (mem->offset + size > mem->total)
32  {
33  mem->data = arena_realloc(mem->data, mem->offset + size, mem->total, self->arena);
34  mem->total = mem->offset + size;
35  }
36 
37  ctu_memcpy(mem->data + mem->offset, src, size);
38  mem->offset += size;
39 
40  return size;
41 }
42 
43 static size_t mem_size(io_t *self)
44 {
45  io_buffer_impl_t *mem = mem_data(self);
46  return mem->used;
47 }
48 
49 static size_t mem_seek(io_t *self, size_t offset)
50 {
51  io_buffer_impl_t *mem = mem_data(self);
52  mem->offset = CT_MIN(offset, mem->used);
53  return mem->offset;
54 }
55 
56 static void *mem_map(io_t *self, os_protect_t protect)
57 {
58  CTASSERTF(!(protect & eOsProtectExecute), "cannot map memory object as executable `%s`", io_name(self));
59 
60  io_buffer_impl_t *mem = mem_data(self);
61 
62  return mem->data;
63 }
64 
65 static os_error_t mem_close(io_t *self)
66 {
67  io_buffer_impl_t *mem = mem_data(self);
68  arena_free(mem->data, mem->total, self->arena);
69 
70  return eOsSuccess;
71 }
72 
73 static const io_callbacks_t kBufferCallbacks = {
74  .fn_read = mem_read,
75  .fn_write = mem_write,
76 
77  .fn_get_size = mem_size,
78  .fn_seek = mem_seek,
79 
80  .fn_map = mem_map,
81  .fn_close = mem_close,
82 
83  .size = sizeof(io_buffer_impl_t)
84 };
85 
86 static io_t *impl_memory_init(void *buffer, const char *name, const void *data, size_t size, os_access_t flags, arena_t *arena)
87 {
88  CTASSERT(data != NULL);
89 
90  io_buffer_impl_t impl = {
91  .data = ARENA_MALLOC(size, "memory", NULL, arena),
92  .total = size,
93  .used = size,
94  .offset = 0
95  };
96 
97  ctu_memcpy(impl.data, data, size);
98 
99  return io_init(buffer, &kBufferCallbacks, flags, name, &impl, arena);
100 }
101 
102 static io_t *impl_blob_init(void *buffer, const char *name, size_t size, os_access_t flags, arena_t *arena)
103 {
104  io_buffer_impl_t impl = {
105  .data = ARENA_MALLOC(size, "blob", NULL, arena),
106  .total = size,
107  .used = 0,
108  .offset = 0
109  };
110 
111  return io_init(buffer, &kBufferCallbacks, flags, name, &impl, arena);
112 }
113 
117 
118 STA_DECL
119 io_t *io_memory(const char *name, const void *data, size_t size, os_access_t flags, arena_t *arena)
120 {
121  void *buffer = ARENA_MALLOC(IO_BUFFER_SIZE, name, NULL, arena);
122  return impl_memory_init(buffer, name, data, size, flags, arena);
123 }
124 
125 STA_DECL
126 io_t *io_blob(const char *name, size_t size, os_access_t flags, arena_t *arena)
127 {
128  void *buffer = ARENA_MALLOC(IO_BUFFER_SIZE, name, NULL, arena);
129  return impl_blob_init(buffer, name, size, flags, arena);
130 }
131 
135 
136 STA_DECL
137 io_t *io_memory_init(void *buffer, const char *name, const void *data, size_t size, os_access_t flags, arena_t *arena)
138 {
139  return impl_memory_init(buffer, name, data, size, flags, arena);
140 }
141 
142 STA_DECL
143 io_t *io_blob_init(void *buffer, const char *name, size_t size, os_access_t flags, arena_t *arena)
144 {
145  return impl_blob_init(buffer, name, size, flags, arena);
146 }
STA_DECL io_t * io_blob(const char *name, size_t size, os_access_t flags, arena_t *arena)
create an IO object in memory of a given size size specifies the initial internal buffer size,...
Definition: buffer.c:126
STA_DECL io_t * io_memory(const char *name, const void *data, size_t size, os_access_t flags, arena_t *arena)
create an IO object from an initial view of memory
Definition: buffer.c:119
STA_DECL io_t * io_blob_init(void *buffer, const char *name, size_t size, os_access_t flags, arena_t *arena)
Definition: buffer.c:143
STA_DECL io_t * io_memory_init(void *buffer, const char *name, const void *data, size_t size, os_access_t flags, arena_t *arena)
Definition: buffer.c:137
#define IO_BUFFER_SIZE
Definition: buffer.h:29
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 STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_NOALIAS CT_BASE_API void * ctu_memcpy(STA_WRITES(size) void *CT_RESTRICT dst, STA_READS(size) const void *CT_RESTRICT src, size_t size)
copy memory from one location to another equivalent to memcpy but with safety checks
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 const char * io_name(const io_t *io)
get the name of an io object
Definition: io.c:112
#define CT_MIN(L, R)
Definition: macros.h:38
#define CT_MAX(L, R)
Definition: macros.h:34
CT_NODISCARD CT_ARENA_API void * arena_realloc(STA_RELEASE void *ptr, size_t new_size, size_t old_size, arena_t *arena)
resize a memory allocation from a custom allocator
#define ARENA_MALLOC(size, name, parent, arena)
allocate memory from a custom allocator
Definition: arena.h:392
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
@ eOsSuccess
Definition: posix.h:24
an allocator object
Definition: arena.h:86
a read/write in memory file
Definition: buffer.h:15
size_t used
used data
Definition: buffer.h:20
size_t total
total size of data
Definition: buffer.h:23
size_t offset
current offset in data
Definition: buffer.h:26
char * data
stored data
Definition: buffer.h:17
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
io object implementation
Definition: impl.h:122