maug
Quick and dirty C mini-augmentation library.
Loading...
Searching...
No Matches
mfile.h
Go to the documentation of this file.
1
2#ifndef MFILE_H
3#define MFILE_H
4
5#if !defined( MFILE_MMAP ) && \
6 !defined( RETROFLAT_API_WINCE ) && \
7 !defined( RETROFLAT_API_PALM )
8# include <sys/stat.h>
9#endif /* !MFILE_MMAP */
10
11/* TODO: async file_open() call that kicks off download to mem buffer that can
12 * be checked with looped check function.
13 */
14
20
24
25#ifndef MAUG_PATH_MAX
27# define MAUG_PATH_MAX 256
28#endif /* !MAUG_PATH_MAX */
29
38
42#define MFILE_CADDY_TYPE_FILE 0x01
43
47#define MFILE_CADDY_TYPE_MEM_BUFFER 0x80
48 /* maug_mfile_types */
50
51#define MFILE_FLAG_READ_ONLY 0x01
52
60
66#define MFILE_READ_FLAG_LSBF 0x01
67
73#define MFILE_READ_FLAG_MSBF 0x01
74
75#ifndef MFILE_TRACE_LVL
76# define MFILE_TRACE_LVL 0
77#endif /* !MFILE_TRACE_LVL */
78
79#ifndef MFILE_TRACE_CONTENTS_LVL
80# define MFILE_TRACE_CONTENTS_LVL 0
81#endif /* !MFILE_TRACE_CONTENTS_LVL */
82
83struct MFILE_CADDY;
84
85typedef off_t (*mfile_has_bytes_t)( struct MFILE_CADDY* p_file );
86typedef MERROR_RETVAL (*mfile_read_byte_t)(
87 struct MFILE_CADDY* p_file, uint8_t* buf );
88typedef MERROR_RETVAL (*mfile_seek_t)( struct MFILE_CADDY* p_file, off_t pos );
89typedef MERROR_RETVAL (*mfile_read_int_t)(
90 struct MFILE_CADDY* p_file, uint8_t* buf, size_t buf_sz, uint8_t flags );
91typedef MERROR_RETVAL (*mfile_read_line_t)(
92 struct MFILE_CADDY* p_file, char* buf, off_t buf_sz, uint8_t flags );
93
94/* TODO: Don't load platform-specific file stuff if MVFS is enabled. */
95#include <mrapifil.h>
96
99 uint8_t type;
101 union MFILE_HANDLE h;
102 off_t sz;
103 off_t last_read;
107 uint8_t* mem_buffer;
108 uint8_t flags;
109 mfile_has_bytes_t has_bytes;
110 mfile_read_byte_t read_byte;
111 mfile_seek_t seek;
112 mfile_read_int_t read_int;
113 mfile_read_line_t read_line;
114};
115
116typedef struct MFILE_CADDY mfile_t;
117
118off_t mfile_file_has_bytes( struct MFILE_CADDY* p_file );
119
120MERROR_RETVAL mfile_file_read_byte( struct MFILE_CADDY* p_file, uint8_t* buf );
121
122MERROR_RETVAL mfile_file_read_int(
123 struct MFILE_CADDY* p_f, uint8_t* buf, size_t buf_sz, uint8_t flags );
124
125MERROR_RETVAL mfile_file_seek( struct MFILE_CADDY* p_file, off_t pos );
126
127MERROR_RETVAL mfile_file_read_line(
128 struct MFILE_CADDY* p_f, char* buffer, off_t buffer_sz, uint8_t flags );
129
130#define mfile_check_lock( p_file ) (NULL != (p_file)->mem_buffer)
131
132#define mfile_get_sz( p_file ) ((p_file)->sz)
133
137MERROR_RETVAL mfile_lock_buffer( MAUG_MHANDLE, off_t, mfile_t* p_file );
138
143MERROR_RETVAL mfile_open_read( const char* filename, mfile_t* p_file );
144
148void mfile_close( mfile_t* p_file );
149
150#ifdef MFILE_C
151
152#include <mrapifil.h>
153
154#ifdef MVFS_ENABLED
155# include <mvfs.h>
156#endif /* MVFS_ENABLED */
157
158MERROR_RETVAL mfile_file_read_int(
159 struct MFILE_CADDY* p_file, uint8_t* buf, size_t buf_sz, uint8_t flags
160) {
161 MERROR_RETVAL retval = MERROR_OK;
162
163 assert( MFILE_CADDY_TYPE_FILE == p_file->type );
164
165 if(
166#ifdef MAUG_LSBF
168#elif defined( MAUG_MSBF )
170#endif
171 ) {
172 debug_printf( MFILE_TRACE_LVL, "reading integer forward" );
173 /* Shrink the buffer moving right and read into it. */
174 while( 0 < buf_sz ) {
175 retval = p_file->read_byte( p_file, buf );
176 maug_cleanup_if_not_ok();
177 buf++;
178 buf_sz--;
179 }
180
181 } else {
182 debug_printf( MFILE_TRACE_LVL, "reading integer reversed" );
183 /* Move to the end of the output buffer and read backwards. */
184 while( 0 < buf_sz ) {
185 retval = p_file->read_byte( p_file, (buf + (buf_sz - 1)) );
186 maug_cleanup_if_not_ok();
187 buf_sz--;
188 }
189 }
190
191cleanup:
192
193 return retval;
194}
195
196/* === */
197
198off_t mfile_mem_has_bytes( struct MFILE_CADDY* p_file ) {
199 return p_file->sz - p_file->mem_cursor;
200}
201
202/* === */
203
204MERROR_RETVAL mfile_mem_read_byte( struct MFILE_CADDY* p_file, uint8_t* buf ) {
205
206 if( p_file->mem_cursor >= p_file->sz ) {
207 return MERROR_FILE;
208 }
209
210 *buf = p_file->mem_buffer[p_file->mem_cursor];
211 p_file->mem_cursor++;
212
213 return MERROR_OK;
214}
215
216/* === */
217
218MERROR_RETVAL mfile_mem_seek( struct MFILE_CADDY* p_file, off_t pos ) {
219 MERROR_RETVAL retval = MERROR_OK;
220
221 assert( MFILE_CADDY_TYPE_MEM_BUFFER == p_file->type );
222
223 p_file->mem_cursor = pos;
224
225 debug_printf( MFILE_TRACE_LVL,
226 "seeking memory buffer to position " OFF_T_FMT " (" OFF_T_FMT ")",
227 pos, p_file->mem_cursor );
228
229 return retval;
230}
231
232/* === */
233
234MERROR_RETVAL mfile_mem_read_line(
235 struct MFILE_CADDY* p_f, char* buffer, off_t buffer_sz, uint8_t flags
236) {
237 MERROR_RETVAL retval = MERROR_OK;
238 off_t i = 0;
239
240 assert( MFILE_CADDY_TYPE_MEM_BUFFER == p_f->type );
241
242 while( i < buffer_sz - 1 && p_f->has_bytes( p_f ) ) {
243 /* Check for potential overflow. */
244 if( i + 1 >= buffer_sz ) {
245 error_printf( "overflow reading string from file!" );
246 retval = MERROR_FILE;
247 break;
248 }
249
250 p_f->read_int( p_f, (uint8_t*)&(buffer[i]), 1, 0 );
251 if( '\n' == buffer[i] ) {
252 /* Break on newline and overwrite it below. */
253 break;
254 }
255 i++;
256 }
257
258 assert( i < buffer_sz ); /* while() above stops before buffer_sz! */
259
260 /* Append a null terminator. */
261 buffer[i] = '\0';
262
263 return retval;
264}
265
266/* === */
267
269 MAUG_MHANDLE handle, off_t handle_sz, mfile_t* p_file
270) {
271 MERROR_RETVAL retval = MERROR_OK;
272
273 debug_printf( 1,
274 "locking handle %p as file %p (" OFF_T_FMT " bytes)...",
275 handle, p_file, handle_sz );
276
277 maug_mzero( p_file, sizeof( struct MFILE_CADDY ) );
278 maug_mlock( handle, p_file->mem_buffer );
280
281 p_file->has_bytes = mfile_mem_has_bytes;
282 p_file->read_byte = mfile_mem_read_byte;
283 p_file->read_int = mfile_file_read_int;
284 p_file->seek = mfile_mem_seek;
285 p_file->read_line = mfile_mem_read_line;
286
287 p_file->sz = handle_sz;
288
289 return retval;
290}
291
292/* === */
293
294MERROR_RETVAL mfile_open_read( const char* filename, mfile_t* p_file ) {
295 MERROR_RETVAL retval = MERROR_OK;
296# if defined( MVFS_ENABLED )
297 size_t i = 0;
298# elif defined( MFILE_MMAP )
299 uint8_t* bytes_ptr = NULL;
300 struct stat st;
301 int in_file = 0;
302# else
303# endif /* MFILE_MMAP */
304
305 /* MAUG_MHANDLE* p_bytes_ptr_h, off_t* p_bytes_sz */
306
307 /* TODO: Move MVFS stuff into its own platform handler or something? */
308# if defined( MVFS_ENABLED )
309
310 while( NULL != gc_mvfs_data[i] ) {
311 if( 0 == strcmp( filename, gc_mvfs_filenames[i] ) ) {
312 debug_printf( 1, "found file \"%s\" at VFS index: " SIZE_T_FMT
313 " (size: " OFF_T_FMT " bytes)",
314 filename, i, *(gc_mvfs_lens[i]) );
315 break;
316 }
317 i++;
318 }
319
320 if( NULL == gc_mvfs_data[i] ) {
321 retval = MERROR_FILE;
322 error_printf( "file \"%s\" not found in VFS!", filename );
323 goto cleanup;
324 }
325
327
328 p_file->has_bytes = mfile_mem_has_bytes;
329 p_file->read_byte = mfile_mem_read_byte;
330 p_file->read_int = mfile_file_read_int;
331 p_file->seek = mfile_mem_seek;
332 p_file->read_line = mfile_mem_read_line;
333
334 p_file->flags = MFILE_FLAG_READ_ONLY;
335 p_file->mem_buffer = gc_mvfs_data[i];
336 p_file->sz = *(gc_mvfs_lens[i]);
337 p_file->mem_cursor = 0;
338
339cleanup:
340
341# else
342
343 /* Call the platform-specific actual file opener from mrapifil.h. */
344 retval = mfile_plt_open_read( filename, p_file );
345
346# endif /* MVFS_ENABLED */
347
348 return retval;
349}
350
351/* === */
352
353void mfile_close( mfile_t* p_file ) {
354# ifdef MFILE_MMAP
355 munmap( bytes_ptr_h, bytes_sz );
356# else
357 /* maug_mfree( bytes_ptr_h ); */
358 switch( p_file->type ) {
359 case 0:
360 /* Do nothing silently. */
361 break;
362
364 mfile_plt_close( p_file );
365 p_file->type = 0;
366 break;
367
369 if( NULL != p_file->mem_buffer ) {
370 maug_munlock( p_file->h.mem, p_file->mem_buffer );
371 debug_printf( 1, "unlocked handle %p from file %p...",
372 p_file->h.mem, p_file );
373 p_file->type = 0;
374 }
375 break;
376
377 default:
378 error_printf( "unknown file type: %d", (p_file)->type );
379 break;
380 }
381# endif /* MFILE_MMAP */
382}
383
384#endif /* MFILE_C */
385 /* maug_mfile */
387
388#endif /* !MFILE_H */
389
int MERROR_RETVAL
Return type indicating function returns a value from this list.
Definition merror.h:19
#define maug_mzero(ptr, sz)
Zero the block of memory pointed to by ptr.
Definition mmem.h:65
#define MFILE_CADDY_TYPE_FILE
A standard UNIX file.
Definition mfile.h:42
#define MFILE_CADDY_TYPE_MEM_BUFFER
An array of bytes in memory abstracted through this library.
Definition mfile.h:47
#define MFILE_READ_FLAG_MSBF
Flag for mfile_read_int_t() indicating integer should always be read most significant byte first.
Definition mfile.h:73
#define MFILE_READ_FLAG_LSBF
Flag for mfile_read_int_t() indicating integer should always be read least significant byte first.
Definition mfile.h:66
MERROR_RETVAL mfile_open_read(const char *filename, mfile_t *p_file)
Open a file and read it into memory or memory-map it.
void mfile_close(mfile_t *p_file)
Close a file opened with mfile_open_read().
MERROR_RETVAL mfile_lock_buffer(MAUG_MHANDLE, off_t, mfile_t *p_file)
Lock a buffer and assign it to an mfile_t to read/write.
Definition mfile.h:97
off_t mem_cursor
Current position if its type is MFILE_CADDY_TYPE_MEM_BUFFER.
Definition mfile.h:105
uint8_t type
The RetroFile Types flag describing this file.
Definition mfile.h:99
union MFILE_HANDLE h
The physical handle or pointer to access the file by.
Definition mfile.h:101
uint8_t * mem_buffer
Locked pointer for MFILE_HANDLE::mem.
Definition mfile.h:107