Add memory callback functions (besides macros)

This commit is contained in:
Tiago 2024-10-06 14:31:27 -04:00
parent 42629f7442
commit 481ef00066

View file

@ -1,7 +1,7 @@
/*
* fast_obj
*
* Version 1.3
* Version 1.4
*
* MIT License
*
@ -31,7 +31,7 @@
#define FAST_OBJ_HDR
#define FAST_OBJ_VERSION_MAJOR 1
#define FAST_OBJ_VERSION_MINOR 3
#define FAST_OBJ_VERSION_MINOR 4
#define FAST_OBJ_VERSION ((FAST_OBJ_VERSION_MAJOR << 8) | FAST_OBJ_VERSION_MINOR)
#include <stdlib.h>
@ -168,10 +168,14 @@ typedef struct
unsigned long (*file_size)(void* file, void* user_data);
} fastObjCallbacks;
typedef void* (*fastObjRealloc)(void* ptr, size_t new_size, size_t old_size, void* context);
typedef void (*fastObjFree)(void* ptr, size_t size, void* context);
#ifdef __cplusplus
extern "C" {
#endif
void fast_obj_set_memory_functions(fastObjRealloc realloc_fn, fastObjFree free_fn, void* user_data);
fastObjMesh* fast_obj_read(const char* path);
fastObjMesh* fast_obj_read_with_callbacks(const char* path, const fastObjCallbacks* callbacks, void* user_data);
void fast_obj_destroy(fastObjMesh* mesh);
@ -247,20 +251,34 @@ double POWER_10_NEG[MAX_POWER] =
};
static void* memory_realloc(void* ptr, size_t bytes)
static
void* default_memory_realloc(void* ptr, size_t new_size, size_t old_size, void* context)
{
return FAST_OBJ_REALLOC(ptr, bytes);
(void*)(old_size);
(void*)(context);
return FAST_OBJ_REALLOC(ptr, new_size);
}
static
void memory_dealloc(void* ptr)
void default_memory_dealloc(void* ptr, size_t size, void* context)
{
(void*)(size);
(void*)(context);
FAST_OBJ_FREE(ptr);
}
#define array_clean(_arr) ((_arr) ? memory_dealloc(_array_header(_arr)), 0 : 0)
static
fastObjRealloc memory_realloc = default_memory_realloc;
static
fastObjFree memory_dealloc = default_memory_dealloc;
static
void* memory_context = 0;
#define array_clean(_arr) ((_arr) ? array_dealloc(_array_header(_arr), sizeof(*(_arr))), 0 : 0)
#define array_push(_arr, _val) (_array_mgrow(_arr, 1) ? ((_arr)[_array_size(_arr)++] = (_val), _array_size(_arr) - 1) : 0)
#define array_size(_arr) ((_arr) ? _array_size(_arr) : 0)
#define array_capacity(_arr) ((_arr) ? _array_capacity(_arr) : 0)
@ -274,7 +292,8 @@ void memory_dealloc(void* ptr)
#define _array_grow(_arr, _n) (*((void**)&(_arr)) = array_realloc(_arr, _n, sizeof(*(_arr))))
static void* array_realloc(void* ptr, fastObjUInt n, fastObjUInt b)
static
void* array_realloc(void* ptr, fastObjUInt n, fastObjUInt b)
{
fastObjUInt sz = array_size(ptr);
fastObjUInt nsz = sz + n;
@ -286,7 +305,9 @@ static void* array_realloc(void* ptr, fastObjUInt n, fastObjUInt b)
ncap = nsz;
ncap = (ncap + 15) & ~15u;
r = (fastObjUInt*)(memory_realloc(ptr ? _array_header(ptr) : 0, (size_t)b * ncap + 2 * sizeof(fastObjUInt)));
size_t bsz = (size_t)b * cap + 2 * sizeof(fastObjUInt);
size_t nbsz = (size_t)b * ncap + 2 * sizeof(fastObjUInt);
r = (fastObjUInt*)(memory_realloc(ptr ? _array_header(ptr) : 0, nbsz, bsz, memory_context));
if (!r)
return 0;
@ -297,6 +318,17 @@ static void* array_realloc(void* ptr, fastObjUInt n, fastObjUInt b)
}
static
void array_dealloc(void* ptr, fastObjUInt b)
{
fastObjUInt cap = array_capacity(ptr);
size_t bsz = (size_t)b * cap + 2 * sizeof(fastObjUInt);
if (ptr)
memory_dealloc(_array_header(ptr), bsz, memory_context);
}
static
void* file_open(const char* path, void* user_data)
{
@ -310,7 +342,7 @@ void file_close(void* file, void* user_data)
{
FILE* f;
(void)(user_data);
f = (FILE*)(file);
fclose(f);
}
@ -321,7 +353,7 @@ size_t file_read(void* file, void* dst, size_t bytes, void* user_data)
{
FILE* f;
(void)(user_data);
f = (FILE*)(file);
return fread(dst, 1, bytes, f);
}
@ -354,9 +386,9 @@ char* string_copy(const char* s, const char* e)
{
size_t n;
char* p;
n = (size_t)(e - s);
p = (char*)(memory_realloc(0, n + 1));
p = (char*)(memory_realloc(0, n + 1, 0, memory_context));
if (p)
{
memcpy(p, s, n);
@ -367,23 +399,16 @@ char* string_copy(const char* s, const char* e)
}
static
char* string_substr(const char* s, size_t a, size_t b)
{
return string_copy(s + a, s + b);
}
static
char* string_concat(const char* a, const char* s, const char* e)
{
size_t an;
size_t sn;
char* p;
an = a ? strlen(a) : 0;
sn = (size_t)(e - s);
p = (char*)(memory_realloc(0, an + sn + 1));
p = (char*)(memory_realloc(0, an + sn + 1, 0, memory_context));
if (p)
{
if (a)
@ -497,7 +522,9 @@ fastObjGroup object_default(void)
static
void object_clean(fastObjGroup* object)
{
memory_dealloc(object->name);
size_t namesz = object->name ? strlen(object->name) + 1 : 0;
memory_dealloc(object->name, namesz, memory_context);
}
@ -535,7 +562,9 @@ fastObjGroup group_default(void)
static
void group_clean(fastObjGroup* group)
{
memory_dealloc(group->name);
size_t namesz = group->name ? strlen(group->name) + 1 : 0;
memory_dealloc(group->name, namesz, memory_context);
}
@ -947,15 +976,20 @@ const char* parse_usemtl(fastObjData* data, const char* ptr)
static
void map_clean(fastObjTexture* map)
{
memory_dealloc(map->name);
memory_dealloc(map->path);
size_t namesz = map->name ? strlen(map->name) + 1 : 0;
size_t pathsz = map->path ? strlen(map->path) + 1 : 0;
memory_dealloc(map->name, namesz, memory_context);
memory_dealloc(map->path, pathsz, memory_context);
}
static
void mtl_clean(fastObjMaterial* mtl)
{
memory_dealloc(mtl->name);
size_t namesz = mtl->name ? strlen(mtl->name) + 1 : 0;
memory_dealloc(mtl->name, namesz, memory_context);
}
@ -1045,7 +1079,7 @@ int read_mtllib(fastObjData* data, void* file, const fastObjCallbacks* callbacks
/* Read entire file */
n = callbacks->file_size(file, user_data);
contents = (char*)(memory_realloc(0, n + 1));
contents = (char*)(memory_realloc(0, n + 1, 0, memory_context));
if (!contents)
return 0;
@ -1212,7 +1246,7 @@ int read_mtllib(fastObjData* data, void* file, const fastObjCallbacks* callbacks
if (mtl.name)
array_push(data->mesh->materials, mtl);
memory_dealloc(contents);
memory_dealloc(contents, n + 1, memory_context);
return 1;
}
@ -1224,6 +1258,7 @@ const char* parse_mtllib(fastObjData* data, const char* ptr, const fastObjCallba
const char* s;
const char* e;
char* lib;
size_t libsz;
void* file;
@ -1245,7 +1280,8 @@ const char* parse_mtllib(fastObjData* data, const char* ptr, const fastObjCallba
callbacks->file_close(file, user_data);
}
memory_dealloc(lib);
libsz = strlen(lib) + 1;
memory_dealloc(lib, libsz, memory_context);
}
return ptr;
@ -1256,8 +1292,8 @@ static
void parse_buffer(fastObjData* data, const char* ptr, const char* end, const fastObjCallbacks* callbacks, void* user_data)
{
const char* p;
p = ptr;
while (p != end)
{
@ -1375,6 +1411,14 @@ void parse_buffer(fastObjData* data, const char* ptr, const char* end, const fas
}
void fast_obj_set_memory_functions(fastObjRealloc realloc_fn, fastObjFree free_fn, void* user_data)
{
memory_realloc = realloc_fn;
memory_dealloc = free_fn;
memory_context = user_data;
}
void fast_obj_destroy(fastObjMesh* m)
{
unsigned int ii;
@ -1404,7 +1448,7 @@ void fast_obj_destroy(fastObjMesh* m)
array_clean(m->materials);
array_clean(m->textures);
memory_dealloc(m);
memory_dealloc(m, sizeof(fastObjMesh), memory_context);
}
@ -1444,7 +1488,7 @@ fastObjMesh* fast_obj_read_with_callbacks(const char* path, const fastObjCallbac
/* Empty mesh */
m = (fastObjMesh*)(memory_realloc(0, sizeof(fastObjMesh)));
m = (fastObjMesh*)(memory_realloc(0, sizeof(fastObjMesh), 0, memory_context));
if (!m)
return 0;
@ -1494,12 +1538,12 @@ fastObjMesh* fast_obj_read_with_callbacks(const char* path, const fastObjCallbac
const char* sep = sep2 && (!sep1 || sep1 < sep2) ? sep2 : sep1;
if (sep)
data.base = string_substr(path, 0, sep - path + 1);
data.base = string_copy(path, path + (sep - path) + 1);
}
/* Create buffer for reading file */
buffer = (char*)(memory_realloc(0, 2 * BUFFER_SIZE * sizeof(char)));
buffer = (char*)(memory_realloc(0, 2 * BUFFER_SIZE * sizeof(char), 0, memory_context));
if (!buffer)
return 0;
@ -1572,8 +1616,8 @@ fastObjMesh* fast_obj_read_with_callbacks(const char* path, const fastObjCallbac
/* Clean up */
memory_dealloc(buffer);
memory_dealloc(data.base);
memory_dealloc(buffer, 2 * BUFFER_SIZE * sizeof(char), memory_context);
memory_dealloc(data.base, data.base ? strlen(data.base) + 1 : 0, memory_context);
callbacks->file_close(file, user_data);