RetroArch/libretro-db/rmsgpack_dom.c

514 lines
14 KiB
C
Raw Normal View History

2016-03-21 18:16:05 +01:00
/* Copyright (C) 2010-2016 The RetroArch team
*
* ---------------------------------------------------------------------------------------
* The following license statement only applies to this file (rmsgpack_dom.c).
* ---------------------------------------------------------------------------------------
*
* Permission is hereby granted, free of charge,
* to any person obtaining a copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
* INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
2015-01-19 22:47:09 +01:00
#include "rmsgpack_dom.h"
#include <stdint.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include "rmsgpack.h"
#define MAX_DEPTH 128
2015-01-27 04:02:10 +01:00
struct dom_reader_state
{
int i;
struct rmsgpack_dom_value *stack[MAX_DEPTH];
2015-01-19 22:47:09 +01:00
};
static struct rmsgpack_dom_value *dom_reader_state_pop(struct dom_reader_state *s)
2015-01-27 04:02:10 +01:00
{
struct rmsgpack_dom_value *v = s->stack[s->i];
s->i--;
return v;
2015-01-19 22:47:09 +01:00
}
static int dom_reader_state_push(struct dom_reader_state *s, struct rmsgpack_dom_value *v)
{
if ((s->i + 1) == MAX_DEPTH)
return -ENOMEM;
s->i++;
s->stack[s->i] = v;
return 0;
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
static int dom_read_nil(void *data)
2015-01-27 04:02:10 +01:00
{
2015-02-01 15:47:02 +01:00
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_NULL;
return 0;
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
static int dom_read_bool(int value, void *data)
2015-01-27 04:02:10 +01:00
{
2015-02-01 15:47:02 +01:00
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_BOOL;
2015-09-17 06:39:17 +02:00
v->val.bool_ = value;
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-01-27 04:02:10 +01:00
static int dom_read_int(int64_t value, void *data)
{
2015-02-01 15:47:02 +01:00
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_INT;
2015-09-17 06:39:17 +02:00
v->val.int_ = value;
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
static int dom_read_uint(uint64_t value, void *data)
2015-01-27 04:02:10 +01:00
{
2015-02-01 15:47:02 +01:00
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_UINT;
2015-09-17 06:39:17 +02:00
v->val.uint_ = value;
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-01-27 04:02:10 +01:00
static int dom_read_string(char *value, uint32_t len, void *data)
{
2015-02-01 15:47:02 +01:00
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_STRING;
2015-09-17 06:39:17 +02:00
v->val.string.len = len;
v->val.string.buff = value;
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-01-27 04:02:10 +01:00
static int dom_read_bin(void *value, uint32_t len, void *data)
{
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v =
2015-02-01 15:47:02 +01:00
(struct rmsgpack_dom_value*)dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_BINARY;
2015-09-17 06:39:17 +02:00
v->val.binary.len = len;
v->val.binary.buff = (char *)value;
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-01-27 04:02:10 +01:00
static int dom_read_map_start(uint32_t len, void *data)
{
unsigned i;
2015-02-01 15:47:02 +01:00
struct rmsgpack_dom_pair *items = NULL;
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v = dom_reader_state_pop(dom_state);
2015-01-27 04:02:10 +01:00
v->type = RDT_MAP;
2015-09-17 06:39:17 +02:00
v->val.map.len = len;
v->val.map.items = NULL;
2015-01-27 04:02:10 +01:00
items = (struct rmsgpack_dom_pair *)calloc(len,
sizeof(struct rmsgpack_dom_pair));
if (!items)
return -ENOMEM;
2015-09-17 06:39:17 +02:00
v->val.map.items = items;
2015-02-01 15:47:02 +01:00
2015-01-27 04:02:10 +01:00
for (i = 0; i < len; i++)
{
if (dom_reader_state_push(dom_state, &items[i].value) < 0)
return -ENOMEM;
if (dom_reader_state_push(dom_state, &items[i].key) < 0)
return -ENOMEM;
}
2015-02-01 15:47:02 +01:00
2015-01-27 04:02:10 +01:00
return 0;
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
static int dom_read_array_start(uint32_t len, void *data)
{
unsigned i;
struct dom_reader_state *dom_state = (struct dom_reader_state *)data;
struct rmsgpack_dom_value *v = dom_reader_state_pop(dom_state);
struct rmsgpack_dom_value *items = NULL;
2015-01-19 22:47:09 +01:00
v->type = RDT_ARRAY;
2015-09-17 06:39:17 +02:00
v->val.array.len = len;
v->val.array.items = NULL;
2015-01-19 22:47:09 +01:00
2016-05-02 16:57:28 +02:00
items = (struct rmsgpack_dom_value *)calloc(len, sizeof(*items));
2015-01-19 22:47:09 +01:00
if (!items)
return -ENOMEM;
2015-01-19 22:47:09 +01:00
2015-09-17 06:39:17 +02:00
v->val.array.items = items;
2015-01-19 22:47:09 +01:00
for (i = 0; i < len; i++)
2015-02-01 15:47:02 +01:00
{
if (dom_reader_state_push(dom_state, &items[i]) < 0)
return -ENOMEM;
}
return 0;
2015-01-19 22:47:09 +01:00
}
static struct rmsgpack_read_callbacks dom_reader_callbacks = {
dom_read_nil,
dom_read_bool,
dom_read_int,
dom_read_uint,
dom_read_string,
dom_read_bin,
dom_read_map_start,
dom_read_array_start
2015-01-19 22:47:09 +01:00
};
2015-02-01 15:47:02 +01:00
void rmsgpack_dom_value_free(struct rmsgpack_dom_value *v)
{
unsigned i;
2015-01-19 22:47:09 +01:00
2015-02-01 15:47:02 +01:00
switch (v->type)
{
case RDT_STRING:
2015-09-17 06:39:17 +02:00
free(v->val.string.buff);
2015-02-01 15:47:02 +01:00
break;
case RDT_BINARY:
2015-09-17 06:39:17 +02:00
free(v->val.binary.buff);
2015-02-01 15:47:02 +01:00
break;
case RDT_MAP:
2015-09-17 06:39:17 +02:00
for (i = 0; i < v->val.map.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
rmsgpack_dom_value_free(&v->val.map.items[i].key);
rmsgpack_dom_value_free(&v->val.map.items[i].value);
2015-02-01 15:47:02 +01:00
}
2015-09-17 06:39:17 +02:00
free(v->val.map.items);
2015-02-01 15:47:02 +01:00
break;
case RDT_ARRAY:
2015-09-17 06:39:17 +02:00
for (i = 0; i < v->val.array.len; i++)
rmsgpack_dom_value_free(&v->val.array.items[i]);
free(v->val.array.items);
2015-02-01 15:47:02 +01:00
break;
case RDT_NULL:
case RDT_INT:
case RDT_BOOL:
case RDT_UINT:
2015-09-17 07:23:36 +02:00
/* Do nothing */
2015-02-01 15:47:02 +01:00
break;
}
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
struct rmsgpack_dom_value *rmsgpack_dom_value_map_value(
const struct rmsgpack_dom_value *map,
const struct rmsgpack_dom_value *key)
{
unsigned i;
if (map->type != RDT_MAP)
return NULL;
2015-09-17 06:39:17 +02:00
for (i = 0; i < map->val.map.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
if (rmsgpack_dom_value_cmp(key, &map->val.map.items[i].key) == 0)
return &map->val.map.items[i].value;
2015-02-01 15:47:02 +01:00
}
return NULL;
2015-01-19 22:47:09 +01:00
}
int rmsgpack_dom_value_cmp(
2015-02-01 15:47:02 +01:00
const struct rmsgpack_dom_value *a,
const struct rmsgpack_dom_value *b
)
2015-02-01 15:47:02 +01:00
{
int rv;
unsigned i;
if (a == b)
return 1;
if (a->type != b->type)
return 1;
switch (a->type)
{
case RDT_NULL:
return 0;
case RDT_BOOL:
2015-09-17 06:39:17 +02:00
return a->val.bool_ == b->val.bool_ ? 0 : 1;
2015-02-01 15:47:02 +01:00
case RDT_INT:
2015-09-17 06:39:17 +02:00
return a->val.int_ == b->val.int_ ? 0 : 1;
2015-02-01 15:47:02 +01:00
case RDT_UINT:
2015-09-17 06:39:17 +02:00
return a->val.uint_ == b->val.uint_ ? 0 : 1;
2015-02-01 15:47:02 +01:00
case RDT_STRING:
2015-09-17 06:39:17 +02:00
if (a->val.string.len != b->val.string.len)
2015-02-01 15:47:02 +01:00
return 1;
2015-09-17 06:39:17 +02:00
return strncmp(a->val.string.buff, b->val.string.buff, a->val.string.len);
2015-02-01 15:47:02 +01:00
case RDT_BINARY:
2015-09-17 06:39:17 +02:00
if (a->val.binary.len != b->val.binary.len)
2015-02-01 15:47:02 +01:00
return 1;
2015-09-17 06:39:17 +02:00
return memcmp(a->val.binary.buff, b->val.binary.buff, a->val.binary.len);
2015-02-01 15:47:02 +01:00
case RDT_MAP:
2015-09-17 06:39:17 +02:00
if (a->val.map.len != b->val.map.len)
2015-02-01 15:47:02 +01:00
return 1;
2015-09-17 06:39:17 +02:00
for (i = 0; i < a->val.map.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_value_cmp(&a->val.map.items[i].key,
&b->val.map.items[i].key)) != 0)
2015-02-01 15:47:02 +01:00
return rv;
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_value_cmp(&a->val.map.items[i].value,
&b->val.map.items[i].value)) != 0)
2015-02-01 15:47:02 +01:00
return rv;
}
break;
case RDT_ARRAY:
2015-09-17 06:39:17 +02:00
if (a->val.array.len != b->val.array.len)
2015-02-01 15:47:02 +01:00
return 1;
2015-09-17 06:39:17 +02:00
for (i = 0; i < a->val.array.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_value_cmp(&a->val.array.items[i],
&b->val.array.items[i])) != 0)
2015-02-01 15:47:02 +01:00
return rv;
}
break;
}
2015-01-19 22:47:09 +01:00
2015-02-01 15:47:02 +01:00
return 1;
2015-01-19 22:47:09 +01:00
}
2015-02-01 15:47:02 +01:00
void rmsgpack_dom_value_print(struct rmsgpack_dom_value *obj)
{
unsigned i;
2015-01-19 22:47:09 +01:00
2015-02-01 15:47:02 +01:00
switch (obj->type)
{
case RDT_NULL:
printf("nil");
break;
case RDT_BOOL:
2015-09-17 06:39:17 +02:00
if (obj->val.bool_)
2015-02-01 15:47:02 +01:00
printf("true");
else
printf("false");
break;
case RDT_INT:
#ifdef _WIN32
2015-09-17 06:39:17 +02:00
printf("%I64d", (signed long long)obj->val.int_);
#else
2015-09-17 06:39:17 +02:00
printf("%lld", (signed long long)obj->val.int_);
#endif
2015-02-01 15:47:02 +01:00
break;
case RDT_UINT:
#ifdef _WIN32
2015-09-17 06:39:17 +02:00
printf("%I64u", (unsigned long long)obj->val.uint_);
#else
2015-09-17 06:39:17 +02:00
printf("%llu", (unsigned long long)obj->val.uint_);
#endif
2015-02-01 15:47:02 +01:00
break;
case RDT_STRING:
2015-09-17 06:39:17 +02:00
printf("\"%s\"", obj->val.string.buff);
break;
case RDT_BINARY:
2015-02-01 15:47:02 +01:00
printf("\"");
2015-09-17 06:39:17 +02:00
for (i = 0; i < obj->val.binary.len; i++)
printf("%02X", (unsigned char) obj->val.binary.buff[i]);
2015-02-01 15:47:02 +01:00
printf("\"");
break;
case RDT_MAP:
printf("{");
2015-09-17 06:39:17 +02:00
for (i = 0; i < obj->val.map.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
rmsgpack_dom_value_print(&obj->val.map.items[i].key);
2015-02-01 15:47:02 +01:00
printf(": ");
2015-09-17 06:39:17 +02:00
rmsgpack_dom_value_print(&obj->val.map.items[i].value);
if (i < (obj->val.map.len - 1))
2015-02-01 15:47:02 +01:00
printf(", ");
}
printf("}");
break;
case RDT_ARRAY:
printf("[");
2015-09-17 06:39:17 +02:00
for (i = 0; i < obj->val.array.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
rmsgpack_dom_value_print(&obj->val.array.items[i]);
if (i < (obj->val.array.len - 1))
2015-02-01 15:47:02 +01:00
printf(", ");
}
printf("]");
}
2015-01-19 22:47:09 +01:00
}
int rmsgpack_dom_write(RFILE *fd, const struct rmsgpack_dom_value *obj)
2015-02-01 15:47:02 +01:00
{
unsigned i;
int rv = 0;
2015-02-01 15:47:02 +01:00
int written = 0;
switch (obj->type)
{
case RDT_NULL:
return rmsgpack_write_nil(fd);
2015-02-01 15:47:02 +01:00
case RDT_BOOL:
2015-09-17 06:39:17 +02:00
return rmsgpack_write_bool(fd, obj->val.bool_);
2015-02-01 15:47:02 +01:00
case RDT_INT:
2015-09-17 06:39:17 +02:00
return rmsgpack_write_int(fd, obj->val.int_);
2015-02-01 15:47:02 +01:00
case RDT_UINT:
2015-09-17 06:39:17 +02:00
return rmsgpack_write_uint(fd, obj->val.uint_);
2015-02-01 15:47:02 +01:00
case RDT_STRING:
2015-09-17 06:39:17 +02:00
return rmsgpack_write_string(fd, obj->val.string.buff, obj->val.string.len);
2015-02-01 15:47:02 +01:00
case RDT_BINARY:
2015-09-17 06:39:17 +02:00
return rmsgpack_write_bin(fd, obj->val.binary.buff, obj->val.binary.len);
2015-02-01 15:47:02 +01:00
case RDT_MAP:
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_write_map_header(fd, obj->val.map.len)) < 0)
2015-02-01 15:47:02 +01:00
return rv;
written += rv;
2015-09-17 06:39:17 +02:00
for (i = 0; i < obj->val.map.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_write(fd, &obj->val.map.items[i].key)) < 0)
2015-02-01 15:47:02 +01:00
return rv;
written += rv;
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_write(fd, &obj->val.map.items[i].value)) < 0)
2015-02-01 15:47:02 +01:00
return rv;
written += rv;
}
break;
case RDT_ARRAY:
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_write_array_header(fd, obj->val.array.len)) < 0)
2015-02-01 15:47:02 +01:00
return rv;
written += rv;
2015-09-17 06:39:17 +02:00
for (i = 0; i < obj->val.array.len; i++)
2015-02-01 15:47:02 +01:00
{
2015-09-17 06:39:17 +02:00
if ((rv = rmsgpack_dom_write(fd, &obj->val.array.items[i])) < 0)
2015-02-01 15:47:02 +01:00
return rv;
written += rv;
}
}
return written;
2015-01-19 22:47:09 +01:00
}
int rmsgpack_dom_read(RFILE *fd, struct rmsgpack_dom_value *out)
2015-02-01 15:47:02 +01:00
{
struct dom_reader_state s;
int rv = 0;
2015-02-01 15:47:02 +01:00
s.i = 0;
2015-02-01 15:47:02 +01:00
s.stack[0] = out;
rv = rmsgpack_read(fd, &dom_reader_callbacks, &s);
2015-01-19 22:47:09 +01:00
2015-02-01 15:47:02 +01:00
if (rv < 0)
rmsgpack_dom_value_free(out);
2015-01-19 22:47:09 +01:00
2015-02-01 15:47:02 +01:00
return rv;
2015-01-19 22:47:09 +01:00
}
int rmsgpack_dom_read_into(RFILE *fd, ...)
2015-01-27 04:02:10 +01:00
{
va_list ap;
2015-06-13 00:33:31 +02:00
struct rmsgpack_dom_value map;
int rv;
const char *key_name;
2015-01-27 04:02:10 +01:00
struct rmsgpack_dom_value key;
2015-02-01 15:47:02 +01:00
struct rmsgpack_dom_value *value;
int64_t *int_value;
uint64_t *uint_value;
int *bool_value;
char *buff_value;
2015-01-27 04:02:10 +01:00
uint64_t min_len;
int value_type = 0;
va_start(ap, fd);
2015-01-27 04:02:10 +01:00
rv = rmsgpack_dom_read(fd, &map);
2015-02-01 15:47:02 +01:00
(void)value_type;
2015-01-27 04:02:10 +01:00
if (rv < 0)
2015-01-28 09:26:22 -05:00
{
va_end(ap);
2015-01-27 04:02:10 +01:00
return rv;
2015-01-28 09:26:22 -05:00
}
2015-01-27 04:02:10 +01:00
if (map.type != RDT_MAP)
goto clean;
2015-01-27 04:02:10 +01:00
while (1)
{
key_name = va_arg(ap, const char *);
2015-02-01 15:47:02 +01:00
if (!key_name)
goto clean;
2015-01-27 04:02:10 +01:00
2015-02-01 15:47:02 +01:00
key.type = RDT_STRING;
2015-09-17 06:39:17 +02:00
key.val.string.len = strlen(key_name);
key.val.string.buff = (char *) key_name;
2015-01-27 04:02:10 +01:00
value = rmsgpack_dom_value_map_value(&map, &key);
switch (value->type)
{
case RDT_INT:
2015-02-01 15:47:02 +01:00
int_value = va_arg(ap, int64_t *);
2015-09-17 06:39:17 +02:00
*int_value = value->val.int_;
2015-01-27 04:02:10 +01:00
break;
case RDT_BOOL:
2015-02-01 15:47:02 +01:00
bool_value = va_arg(ap, int *);
2015-09-17 06:39:17 +02:00
*bool_value = value->val.bool_;
2015-01-27 04:02:10 +01:00
break;
case RDT_UINT:
2015-02-01 15:47:02 +01:00
uint_value = va_arg(ap, uint64_t *);
2015-09-17 06:39:17 +02:00
*uint_value = value->val.uint_;
2015-01-27 04:02:10 +01:00
break;
case RDT_BINARY:
2015-02-01 15:47:02 +01:00
buff_value = va_arg(ap, char *);
uint_value = va_arg(ap, uint64_t *);
2015-09-17 06:39:17 +02:00
*uint_value = value->val.binary.len;
min_len = (value->val.binary.len > *uint_value) ?
*uint_value : value->val.binary.len;
2015-01-27 04:02:10 +01:00
2015-09-17 20:10:04 +02:00
memcpy(buff_value, value->val.binary.buff, (size_t)min_len);
2015-01-27 04:02:10 +01:00
break;
case RDT_STRING:
buff_value = va_arg(ap, char *);
uint_value = va_arg(ap, uint64_t *);
2015-09-17 06:39:17 +02:00
min_len = (value->val.string.len + 1 > *uint_value) ?
*uint_value : value->val.string.len + 1;
2015-01-27 04:02:10 +01:00
*uint_value = min_len;
2015-02-01 15:47:02 +01:00
2015-09-17 20:10:04 +02:00
memcpy(buff_value, value->val.string.buff, (size_t)min_len);
2015-01-27 04:02:10 +01:00
break;
default:
goto clean;
2015-01-27 04:02:10 +01:00
}
}
2015-01-19 22:47:09 +01:00
clean:
2015-01-27 04:02:10 +01:00
va_end(ap);
rmsgpack_dom_value_free(&map);
return 0;
2015-01-19 22:47:09 +01:00
}