RetroArch/libretro-db/libretrodb.c

598 lines
13 KiB
C
Raw Normal View History

2015-01-19 22:47:09 +01:00
2015-09-19 03:46:41 +02:00
#include <stdio.h>
2015-01-19 22:47:09 +01:00
#include <sys/types.h>
#ifdef _WIN32
#include <direct.h>
#else
#include <unistd.h>
#endif
#include <string.h>
#include <errno.h>
#include <sys/stat.h>
#include <stdlib.h>
2016-03-20 16:29:14 +01:00
#include <streams/file_stream.h>
#include <retro_endianness.h>
2015-09-19 03:46:41 +02:00
#include <compat/strl.h>
2015-09-19 03:46:41 +02:00
#include "libretrodb.h"
2015-01-19 22:47:09 +01:00
#include "rmsgpack_dom.h"
#include "rmsgpack.h"
#include "bintree.h"
#include "query.h"
2015-09-17 09:46:26 +02:00
#include "libretrodb.h"
2015-01-19 22:47:09 +01:00
#define MAGIC_NUMBER "RARCHDB"
2015-01-24 04:04:56 +01:00
struct node_iter_ctx
{
libretrodb_t *db;
libretrodb_index_t *idx;
2015-01-19 22:47:09 +01:00
};
2015-09-17 20:24:49 +02:00
struct libretrodb
2015-09-17 09:46:26 +02:00
{
RFILE *fd;
2015-09-17 09:46:26 +02:00
uint64_t root;
uint64_t count;
uint64_t first_index_offset;
char path[1024];
2015-09-17 20:24:49 +02:00
};
2015-09-17 09:46:26 +02:00
2015-09-17 20:24:49 +02:00
struct libretrodb_index
2015-09-17 09:46:26 +02:00
{
char name[50];
uint64_t key_size;
uint64_t next;
2015-09-17 20:24:49 +02:00
};
2015-09-17 09:46:26 +02:00
typedef struct libretrodb_metadata
{
uint64_t count;
} libretrodb_metadata_t;
typedef struct libretrodb_header
{
char magic_number[sizeof(MAGIC_NUMBER)-1];
uint64_t metadata_offset;
} libretrodb_header_t;
2015-09-17 20:24:49 +02:00
struct libretrodb_cursor
2015-09-17 09:46:26 +02:00
{
int is_valid;
RFILE *fd;
2015-09-17 09:46:26 +02:00
int eof;
libretrodb_query_t *query;
libretrodb_t *db;
2015-09-17 20:24:49 +02:00
};
2015-09-17 09:46:26 +02:00
2015-01-19 22:47:09 +01:00
static struct rmsgpack_dom_value sentinal;
static int libretrodb_read_metadata(RFILE *fd, libretrodb_metadata_t *md)
2015-01-24 04:04:56 +01:00
{
return rmsgpack_dom_read_into(fd, "count", &md->count, NULL);
2015-01-19 22:47:09 +01:00
}
static int libretrodb_write_metadata(RFILE *fd, libretrodb_metadata_t *md)
2015-01-24 04:04:56 +01:00
{
rmsgpack_write_map_header(fd, 1);
rmsgpack_write_string(fd, "count", strlen("count"));
return rmsgpack_write_uint(fd, md->count);
2015-01-19 22:47:09 +01:00
}
2015-09-17 09:33:24 +02:00
static int validate_document(const struct rmsgpack_dom_value *doc)
2015-01-24 04:04:56 +01:00
{
unsigned i;
2015-09-21 17:47:02 +02:00
struct rmsgpack_dom_value key, value;
2015-01-24 04:04:56 +01:00
int rv = 0;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (doc->type != RDT_MAP)
return -EINVAL;
2015-01-19 22:47:09 +01:00
2015-09-17 06:39:17 +02:00
for (i = 0; i < doc->val.map.len; i++)
2015-01-24 04:04:56 +01:00
{
2015-09-17 06:39:17 +02:00
key = doc->val.map.items[i].key;
value = doc->val.map.items[i].value;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (key.type != RDT_STRING)
return -EINVAL;
2015-09-17 06:39:17 +02:00
if (key.val.string.len <= 0)
2015-01-24 04:04:56 +01:00
return -EINVAL;
2015-09-17 06:39:17 +02:00
if (key.val.string.buff[0] == '$')
2015-01-24 04:04:56 +01:00
return -EINVAL;
if (value.type != RDT_MAP)
continue;
if ((rv == validate_document(&value)) != 0)
return rv;
}
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
return rv;
2015-01-19 22:47:09 +01:00
}
int libretrodb_create(RFILE *fd, libretrodb_value_provider value_provider,
2015-09-17 09:47:48 +02:00
void *ctx)
2015-01-24 04:04:56 +01:00
{
int rv;
libretrodb_metadata_t md;
2015-09-17 09:25:06 +02:00
struct rmsgpack_dom_value item;
uint64_t item_count = 0;
2015-09-17 07:30:32 +02:00
libretrodb_header_t header = {{0}};
2015-09-21 17:47:02 +02:00
ssize_t root = retro_fseek(fd, 0, SEEK_CUR);
2015-01-23 05:14:26 +01:00
2015-01-24 04:04:56 +01:00
memcpy(header.magic_number, MAGIC_NUMBER, sizeof(MAGIC_NUMBER)-1);
2015-01-23 05:14:26 +01:00
2015-01-24 04:04:56 +01:00
/* We write the header in the end because we need to know the size of
2015-01-23 05:59:47 +01:00
* the db first */
retro_fseek(fd, sizeof(libretrodb_header_t), SEEK_CUR);
2015-01-23 05:59:47 +01:00
2016-01-23 01:25:09 +01:00
item.type = RDT_NULL;
2015-01-24 04:04:56 +01:00
while ((rv = value_provider(ctx, &item)) == 0)
{
if ((rv = validate_document(&item)) < 0)
goto clean;
2015-01-19 22:47:09 +01:00
if ((rv = rmsgpack_dom_write(fd, &item)) < 0)
goto clean;
2015-01-19 22:47:09 +01:00
2016-01-23 01:25:09 +01:00
rmsgpack_dom_value_free(&item);
item.type = RDT_NULL;
2015-01-24 04:04:56 +01:00
item_count++;
}
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (rv < 0)
goto clean;
2015-01-19 22:47:09 +01:00
if ((rv = rmsgpack_dom_write(fd, &sentinal)) < 0)
goto clean;
2015-01-19 22:47:09 +01:00
header.metadata_offset = swap_if_little64(retro_fseek(fd, 0, SEEK_CUR));
2015-01-24 04:04:56 +01:00
md.count = item_count;
libretrodb_write_metadata(fd, &md);
retro_fseek(fd, root, SEEK_SET);
retro_fwrite(fd, &header, sizeof(header));
clean:
2015-01-24 04:04:56 +01:00
rmsgpack_dom_value_free(&item);
return rv;
2015-01-19 22:47:09 +01:00
}
static int libretrodb_read_index_header(RFILE *fd, libretrodb_index_t *idx)
2015-01-23 05:59:47 +01:00
{
2015-01-24 04:04:56 +01:00
uint64_t name_len = 50;
return rmsgpack_dom_read_into(fd,
2015-01-24 04:04:56 +01:00
"name", idx->name, &name_len,
"key_size", &idx->key_size,
"next", &idx->next, NULL);
2015-01-19 22:47:09 +01:00
}
static void libretrodb_write_index_header(RFILE *fd, libretrodb_index_t *idx)
2015-01-23 05:59:47 +01:00
{
2015-09-17 09:47:48 +02:00
rmsgpack_write_map_header(fd, 3);
rmsgpack_write_string(fd, "name", strlen("name"));
rmsgpack_write_string(fd, idx->name, strlen(idx->name));
rmsgpack_write_string(fd, "key_size", strlen("key_size"));
rmsgpack_write_uint(fd, idx->key_size);
rmsgpack_write_string(fd, "next", strlen("next"));
rmsgpack_write_uint(fd, idx->next);
2015-01-19 22:47:09 +01:00
}
void libretrodb_close(libretrodb_t *db)
2015-01-23 05:14:26 +01:00
{
if (db->fd)
retro_fclose(db->fd);
db->fd = NULL;
2015-01-19 22:47:09 +01:00
}
int libretrodb_open(const char *path, libretrodb_t *db)
2015-01-23 05:59:47 +01:00
{
2015-01-24 04:04:56 +01:00
libretrodb_header_t header;
libretrodb_metadata_t md;
int rv;
RFILE *fd = retro_fopen(path, RFILE_MODE_READ, -1);
2015-01-23 05:59:47 +01:00
if (!fd)
2015-01-24 04:04:56 +01:00
return -errno;
2015-01-19 22:47:09 +01:00
2015-09-19 03:46:41 +02:00
strlcpy(db->path, path, sizeof(db->path));
db->root = retro_fseek(fd, 0, SEEK_CUR);
2015-01-19 22:47:09 +01:00
if ((rv = retro_fread(fd, &header, sizeof(header))) == -1)
2015-01-24 04:04:56 +01:00
{
rv = -errno;
goto error;
}
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (strncmp(header.magic_number, MAGIC_NUMBER, sizeof(MAGIC_NUMBER)) != 0)
{
rv = -EINVAL;
goto error;
}
header.metadata_offset = swap_if_little64(header.metadata_offset);
2015-09-17 20:10:04 +02:00
retro_fseek(fd, (ssize_t)header.metadata_offset, SEEK_SET);
2015-01-24 04:04:56 +01:00
if (libretrodb_read_metadata(fd, &md) < 0)
2015-01-24 04:04:56 +01:00
{
rv = -EINVAL;
goto error;
}
db->count = md.count;
db->first_index_offset = retro_fseek(fd, 0, SEEK_CUR);
db->fd = fd;
2015-01-24 04:04:56 +01:00
return 0;
2015-09-17 10:14:57 +02:00
2015-01-19 22:47:09 +01:00
error:
if (fd)
retro_fclose(fd);
2015-01-24 04:04:56 +01:00
return rv;
2015-01-19 22:47:09 +01:00
}
2015-01-24 04:04:56 +01:00
static int libretrodb_find_index(libretrodb_t *db, const char *index_name,
libretrodb_index_t *idx)
{
2015-09-21 17:47:02 +02:00
ssize_t eof = retro_fseek(db->fd, 0, SEEK_END);
2016-02-04 11:05:34 +01:00
ssize_t offset = retro_fseek(db->fd,
(ssize_t)db->first_index_offset, SEEK_SET);
2015-01-23 05:59:47 +01:00
2015-01-24 04:04:56 +01:00
while (offset < eof)
2015-01-23 05:59:47 +01:00
{
libretrodb_read_index_header(db->fd, idx);
2015-02-01 15:47:02 +01:00
2015-01-23 05:59:47 +01:00
if (strncmp(index_name, idx->name, strlen(idx->name)) == 0)
return 0;
2015-02-01 15:47:02 +01:00
2015-09-17 20:10:04 +02:00
offset = retro_fseek(db->fd, (ssize_t)idx->next, SEEK_CUR);
2015-01-23 05:59:47 +01:00
}
2015-01-24 04:04:56 +01:00
return -1;
2015-01-19 22:47:09 +01:00
}
2015-09-17 09:33:24 +02:00
static int node_compare(const void *a, const void *b, void *ctx)
2015-01-24 04:04:56 +01:00
{
return memcmp(a, b, *(uint8_t *)ctx);
2015-01-19 22:47:09 +01:00
}
2015-09-17 09:33:24 +02:00
static int binsearch(const void *buff, const void *item,
uint64_t count, uint8_t field_size, uint64_t *offset)
2015-01-24 04:04:56 +01:00
{
int mid = count / 2;
int item_size = field_size + sizeof(uint64_t);
2015-09-17 09:33:24 +02:00
uint64_t *current = (uint64_t *)buff + (mid * item_size);
2015-01-24 04:04:56 +01:00
int rv = node_compare(current, item, &field_size);
if (rv == 0)
{
*offset = *(uint64_t *)(current + field_size);
return 0;
}
2015-02-01 15:47:02 +01:00
if (count == 0)
2015-01-24 04:04:56 +01:00
return -1;
2015-02-01 15:47:02 +01:00
if (rv > 0)
2015-01-24 04:04:56 +01:00
return binsearch(buff, item, mid, field_size, offset);
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
return binsearch(current + item_size, item,
count - mid, field_size, offset);
2015-01-19 22:47:09 +01:00
}
2015-01-24 04:04:56 +01:00
int libretrodb_find_entry(libretrodb_t *db, const char *index_name,
2015-09-17 09:47:48 +02:00
const void *key, struct rmsgpack_dom_value *out)
2015-01-24 04:04:56 +01:00
{
libretrodb_index_t idx;
int rv;
2015-09-17 09:33:24 +02:00
void *buff;
2015-01-24 04:04:56 +01:00
uint64_t offset;
ssize_t bufflen, nread = 0;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (libretrodb_find_index(db, index_name, &idx) < 0)
return -1;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
bufflen = idx.next;
buff = malloc(bufflen);
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (!buff)
return -ENOMEM;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
while (nread < bufflen)
{
2015-09-17 09:33:24 +02:00
void *buff_ = (uint64_t *)buff + nread;
rv = retro_fread(db->fd, buff_, bufflen - nread);
2015-01-24 04:04:56 +01:00
if (rv <= 0)
{
free(buff);
return -errno;
}
nread += rv;
}
2015-01-19 22:47:09 +01:00
2015-09-17 20:10:04 +02:00
rv = binsearch(buff, key, db->count, (ssize_t)idx.key_size, &offset);
2015-01-24 04:04:56 +01:00
free(buff);
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (rv == 0)
2015-09-17 20:10:04 +02:00
retro_fseek(db->fd, (ssize_t)offset, SEEK_SET);
2015-09-17 09:50:34 +02:00
return rmsgpack_dom_read(db->fd, out);
2015-01-19 22:47:09 +01:00
}
/**
2015-01-23 05:59:47 +01:00
* libretrodb_cursor_reset:
* @cursor : Handle to database cursor.
*
* Resets cursor.
*
* Returns: ???.
**/
2015-01-23 05:59:47 +01:00
int libretrodb_cursor_reset(libretrodb_cursor_t *cursor)
{
2015-09-17 09:47:48 +02:00
cursor->eof = 0;
return retro_fseek(cursor->fd,
2015-09-17 20:10:04 +02:00
(ssize_t)(cursor->db->root + sizeof(libretrodb_header_t)),
2015-01-24 04:04:56 +01:00
SEEK_SET);
2015-01-19 22:47:09 +01:00
}
2015-01-24 04:04:56 +01:00
int libretrodb_cursor_read_item(libretrodb_cursor_t *cursor,
2015-09-17 09:33:24 +02:00
struct rmsgpack_dom_value *out)
2015-01-24 04:04:56 +01:00
{
int rv;
if (cursor->eof)
return EOF;
2015-01-19 22:47:09 +01:00
retry:
rv = rmsgpack_dom_read(cursor->fd, out);
2015-01-24 04:04:56 +01:00
if (rv < 0)
return rv;
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (out->type == RDT_NULL)
{
cursor->eof = 1;
return EOF;
}
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
if (cursor->query)
{
if (!libretrodb_query_filter(cursor->query, out))
{
rmsgpack_dom_value_free(out);
2015-01-24 04:04:56 +01:00
goto retry;
}
2015-01-24 04:04:56 +01:00
}
2015-01-19 22:47:09 +01:00
2015-01-24 04:04:56 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
/**
2015-01-23 05:59:47 +01:00
* libretrodb_cursor_close:
* @cursor : Handle to database cursor.
*
* Closes cursor and frees up allocated memory.
**/
2015-01-23 05:59:47 +01:00
void libretrodb_cursor_close(libretrodb_cursor_t *cursor)
{
2015-01-27 04:02:10 +01:00
if (!cursor)
return;
if (cursor->fd)
retro_fclose(cursor->fd);
2015-06-03 16:57:51 +02:00
2015-09-17 09:47:48 +02:00
if (cursor->query)
libretrodb_query_free(cursor->query);
2015-06-03 16:57:51 +02:00
cursor->is_valid = 0;
cursor->eof = 1;
cursor->fd = NULL;
cursor->db = NULL;
cursor->query = NULL;
2015-01-19 22:47:09 +01:00
}
/**
2015-01-23 05:59:47 +01:00
* libretrodb_cursor_open:
* @db : Handle to database.
* @cursor : Handle to database cursor.
* @q : Query to execute.
*
* Opens cursor to database based on query @q.
*
* Returns: 0 if successful, otherwise negative.
**/
2015-01-27 04:02:10 +01:00
int libretrodb_cursor_open(libretrodb_t *db, libretrodb_cursor_t *cursor,
libretrodb_query_t *q)
{
2015-11-14 14:56:00 -03:00
cursor->fd = retro_fopen(db->path, RFILE_MODE_READ | RFILE_HINT_MMAP, -1);
if (!cursor->fd)
2015-01-27 04:02:10 +01:00
return -errno;
cursor->db = db;
2015-01-27 04:02:10 +01:00
cursor->is_valid = 1;
libretrodb_cursor_reset(cursor);
cursor->query = q;
2015-01-27 04:02:10 +01:00
if (q)
libretrodb_query_inc_ref(q);
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-09-17 09:33:24 +02:00
static int node_iter(void *value, void *ctx)
{
2015-09-17 09:47:48 +02:00
struct node_iter_ctx *nictx = (struct node_iter_ctx*)ctx;
2015-01-19 22:47:09 +01:00
if (retro_fwrite(nictx->db->fd, value,
2015-09-17 20:10:04 +02:00
(ssize_t)(nictx->idx->key_size + sizeof(uint64_t))) > 0)
2015-09-17 09:47:48 +02:00
return 0;
2015-01-19 22:47:09 +01:00
2015-09-17 09:47:48 +02:00
return -1;
2015-01-19 22:47:09 +01:00
}
2015-01-23 05:59:47 +01:00
static uint64_t libretrodb_tell(libretrodb_t *db)
{
return retro_fseek(db->fd, 0, SEEK_CUR);
2015-01-19 22:47:09 +01:00
}
2015-01-24 04:04:56 +01:00
int libretrodb_create_index(libretrodb_t *db,
const char *name, const char *field_name)
{
2015-09-17 09:47:48 +02:00
int rv;
struct node_iter_ctx nictx;
struct rmsgpack_dom_value key;
libretrodb_index_t idx;
struct rmsgpack_dom_value item;
struct rmsgpack_dom_value *field;
uint64_t idx_header_offset;
libretrodb_cursor_t cur = {0};
void *buff = NULL;
uint64_t *buff_u64 = NULL;
uint8_t field_size = 0;
uint64_t item_loc = libretrodb_tell(db);
bintree_t *tree = bintree_new(node_compare, &field_size);
2015-09-17 09:25:06 +02:00
2015-09-17 10:21:29 +02:00
if (!tree || (libretrodb_cursor_open(db, &cur, NULL) != 0))
{
2015-09-17 09:47:48 +02:00
rv = -1;
goto clean;
}
2015-01-19 22:47:09 +01:00
2015-09-17 09:47:48 +02:00
key.type = RDT_STRING;
key.val.string.len = strlen(field_name);
2015-02-01 15:47:02 +01:00
2015-09-17 09:47:48 +02:00
/* We know we aren't going to change it */
key.val.string.buff = (char *) field_name;
2015-02-01 15:47:02 +01:00
2015-09-17 09:47:48 +02:00
while (libretrodb_cursor_read_item(&cur, &item) == 0)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
if (item.type != RDT_MAP)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -EINVAL;
printf("Only map keys are supported\n");
goto clean;
}
2015-01-24 04:04:56 +01:00
2015-09-17 09:47:48 +02:00
field = rmsgpack_dom_value_map_value(&item, &key);
2015-01-24 04:04:56 +01:00
2015-09-17 09:47:48 +02:00
if (!field)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -EINVAL;
printf("field not found in item\n");
goto clean;
}
2015-01-24 04:04:56 +01:00
2015-09-17 09:47:48 +02:00
if (field->type != RDT_BINARY)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -EINVAL;
printf("field is not binary\n");
goto clean;
}
2015-01-19 22:47:09 +01:00
2015-09-17 09:47:48 +02:00
if (field->val.binary.len == 0)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -EINVAL;
printf("field is empty\n");
goto clean;
}
if (field_size == 0)
field_size = field->val.binary.len;
else if (field->val.binary.len != field_size)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -EINVAL;
printf("field is not of correct size\n");
goto clean;
}
2015-02-01 15:47:02 +01:00
2015-09-17 09:47:48 +02:00
buff = malloc(field_size + sizeof(uint64_t));
if (!buff)
2015-01-24 04:04:56 +01:00
{
2015-09-17 09:47:48 +02:00
rv = -ENOMEM;
goto clean;
}
2015-06-03 16:57:51 +02:00
2015-09-17 09:47:48 +02:00
memcpy(buff, field->val.binary.buff, field_size);
2015-06-03 16:57:51 +02:00
2015-09-17 09:47:48 +02:00
buff_u64 = (uint64_t *)buff + field_size;
2015-06-03 16:57:51 +02:00
2015-09-17 09:47:48 +02:00
memcpy(buff_u64, &item_loc, sizeof(uint64_t));
2015-09-05 19:51:55 +02:00
2015-09-17 09:47:48 +02:00
if (bintree_insert(tree, buff) != 0)
{
2015-09-17 09:47:48 +02:00
printf("Value is not unique: ");
rmsgpack_dom_value_print(field);
printf("\n");
rv = -EINVAL;
goto clean;
}
buff = NULL;
rmsgpack_dom_value_free(&item);
item_loc = libretrodb_tell(db);
}
2015-09-17 20:10:04 +02:00
idx_header_offset = retro_fseek(db->fd, 0, SEEK_END);
2015-09-17 09:47:48 +02:00
(void)idx_header_offset;
2015-09-17 20:10:04 +02:00
(void)rv;
2015-09-17 09:47:48 +02:00
strncpy(idx.name, name, 50);
idx.name[49] = '\0';
idx.key_size = field_size;
idx.next = db->count * (field_size + sizeof(uint64_t));
libretrodb_write_index_header(db->fd, &idx);
nictx.db = db;
nictx.idx = &idx;
bintree_iterate(tree, node_iter, &nictx);
bintree_free(tree);
2015-09-17 07:09:31 +02:00
clean:
2015-09-17 09:47:48 +02:00
rmsgpack_dom_value_free(&item);
if (buff)
free(buff);
if (cur.is_valid)
libretrodb_cursor_close(&cur);
return 0;
2015-01-19 22:47:09 +01:00
}
2015-09-17 09:46:26 +02:00
libretrodb_cursor_t *libretrodb_cursor_new(void)
{
libretrodb_cursor_t *dbc = (libretrodb_cursor_t*)
calloc(1, sizeof(*dbc));
if (!dbc)
return NULL;
return dbc;
}
void libretrodb_cursor_free(libretrodb_cursor_t *dbc)
{
if (!dbc)
return;
free(dbc);
}
libretrodb_t *libretrodb_new(void)
{
libretrodb_t *db = (libretrodb_t*)calloc(1, sizeof(*db));
if (!db)
return NULL;
return db;
}
void libretrodb_free(libretrodb_t *db)
{
if (!db)
return;
free(db);
}