Blob Blame History Raw
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include "test.h"

static int configs_equal(snd_config_t *c1, snd_config_t *c2);

/* checks if all children of c1 also occur in c2 */
static int subset_of(snd_config_t *c1, snd_config_t *c2)
{
	snd_config_iterator_t i, next;
	snd_config_t *e1, *e2;
	const char *id;

	snd_config_for_each(i, next, c1) {
		e1 = snd_config_iterator_entry(i);
		if (snd_config_get_id(e1, &id) < 0 || !id)
			return 0;
		if (snd_config_search(c2, id, &e2) < 0)
			return 0;
		if (!configs_equal(e1, e2))
			return 0;
	}
	return 1;
}

/* checks if two configuration nodes are equal */
static int configs_equal(snd_config_t *c1, snd_config_t *c2)
{
	long i1, i2;
	long long i641, i642;
	const char *s1, *s2;

	if (snd_config_get_type(c1) != snd_config_get_type(c2))
		return 0;
	switch (snd_config_get_type(c1)) {
	case SND_CONFIG_TYPE_INTEGER:
		return snd_config_get_integer(c1, &i1) >= 0 &&
			snd_config_get_integer(c2, &i2) >= 0 &&
			i1 == i2;
	case SND_CONFIG_TYPE_INTEGER64:
		return snd_config_get_integer64(c1, &i641) >= 0 &&
			snd_config_get_integer64(c2, &i642) >= 0 &&
			i641 == i642;
	case SND_CONFIG_TYPE_STRING:
		return snd_config_get_string(c1, &s1) >= 0 &&
			snd_config_get_string(c2, &s2) >= 0 &&
			!s1 == !s2 &&
			(!s1 || !strcmp(s1, s2));
	case SND_CONFIG_TYPE_COMPOUND:
		return subset_of(c1, c2) && subset_of(c2, c1);
	default:
		fprintf(stderr, "unknown configuration node type %d\n",
			(int)snd_config_get_type(c1));
		return 0;
	}
}

static void test_top(void)
{
	snd_config_t *top;
	const char *id;

	if (ALSA_CHECK(snd_config_top(&top)) < 0)
		return;

	TEST_CHECK(snd_config_get_type(top) == SND_CONFIG_TYPE_COMPOUND);
	TEST_CHECK(snd_config_iterator_first(top) == snd_config_iterator_end(top));
	TEST_CHECK(snd_config_get_id(top, &id) >= 0 && id == NULL);

	ALSA_CHECK(snd_config_delete(top));
}

static void test_load(void)
{
	const char *config_text1 = "s='world';";
	const char *config_text2 = "c.elem 0";
	snd_config_t *loaded, *made, *c, *c2;
	snd_input_t *input;

	ALSA_CHECK(snd_config_top(&loaded));
	ALSA_CHECK(snd_config_imake_integer(&c, "i", 42));
	ALSA_CHECK(snd_config_add(loaded, c));
	ALSA_CHECK(snd_config_imake_string(&c, "s", "hello"));
	ALSA_CHECK(snd_config_add(loaded, c));

	ALSA_CHECK(snd_config_top(&made));
	ALSA_CHECK(snd_config_imake_string(&c, "s", "world"));
	ALSA_CHECK(snd_config_add(made, c));
	ALSA_CHECK(snd_config_imake_integer(&c, "i", 42));
	ALSA_CHECK(snd_config_add(made, c));

	ALSA_CHECK(snd_input_buffer_open(&input, config_text1, strlen(config_text1)));
	ALSA_CHECK(snd_config_load(loaded, input));
	ALSA_CHECK(snd_input_close(input));
	TEST_CHECK(configs_equal(loaded, made));

	ALSA_CHECK(snd_config_make_compound(&c, "c", 0));
	ALSA_CHECK(snd_config_add(made, c));
	ALSA_CHECK(snd_config_imake_integer(&c2, "elem", 0));
	ALSA_CHECK(snd_config_add(c, c2));

	ALSA_CHECK(snd_input_buffer_open(&input, config_text2, strlen(config_text2)));
	ALSA_CHECK(snd_config_load(loaded, input));
	ALSA_CHECK(snd_input_close(input));
	TEST_CHECK(configs_equal(loaded, made));

	ALSA_CHECK(snd_config_delete(loaded));
	ALSA_CHECK(snd_config_delete(made));
}

static void test_save(void)
{
	const char *text =
		"a.b.c 'x.y.z'\n"
		"xxx = yyy;\n"
		"q { qq=qqq }\n"
		"a [ 1 2 3 4 5 '...' ]\n";
	snd_config_t *orig, *saved;
	snd_input_t *input;
	snd_output_t *output;
	char *buf;
	size_t buf_size;

	ALSA_CHECK(snd_input_buffer_open(&input, text, strlen(text)));
	ALSA_CHECK(snd_config_top(&orig));
	ALSA_CHECK(snd_config_load(orig, input));
	ALSA_CHECK(snd_input_close(input));
	ALSA_CHECK(snd_output_buffer_open(&output));
	ALSA_CHECK(snd_config_save(orig, output));
	buf_size = snd_output_buffer_string(output, &buf);
	ALSA_CHECK(snd_input_buffer_open(&input, buf, buf_size));
	ALSA_CHECK(snd_config_top(&saved));
	ALSA_CHECK(snd_config_load(saved, input));
	ALSA_CHECK(snd_input_close(input));
	ALSA_CHECK(snd_output_close(output));
	TEST_CHECK(configs_equal(orig, saved));
	ALSA_CHECK(snd_config_delete(orig));
	ALSA_CHECK(snd_config_delete(saved));
}

static void test_update(void)
{
	ALSA_CHECK(snd_config_update_free_global());
	TEST_CHECK(snd_config == NULL);
	ALSA_CHECK(snd_config_update());
	TEST_CHECK(snd_config_get_type(snd_config) == SND_CONFIG_TYPE_COMPOUND);
	ALSA_CHECK(snd_config_update());
	TEST_CHECK(snd_config_get_type(snd_config) == SND_CONFIG_TYPE_COMPOUND);
	ALSA_CHECK(snd_config_update_free_global());
	TEST_CHECK(snd_config == NULL);
}

static void test_search(void)
{
	const char *text =
		"a 42\n"
		"b {\n"
		"    c cee\n"
		"    d {\n"
		"        e 2.71828\n"
		"    }\n"
		"}\n";
	snd_input_t *input;
	snd_config_t *top, *c;
	const char *id;

	ALSA_CHECK(snd_input_buffer_open(&input, text, strlen(text)));
	ALSA_CHECK(snd_config_top(&top));
	ALSA_CHECK(snd_config_load(top, input));
	ALSA_CHECK(snd_input_close(input));

	ALSA_CHECK(snd_config_search(top, "a", &c));
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "a"));
	ALSA_CHECK(snd_config_search(top, "b.d.e", &c));
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "e"));
	ALSA_CHECK(snd_config_search(top, "b.c", NULL));
	TEST_CHECK(snd_config_search(top, "x", NULL) == -ENOENT);
	TEST_CHECK(snd_config_search(top, "b.y", &c) == -ENOENT);
	TEST_CHECK(snd_config_search(top, "a.z", &c) == -ENOENT);

	ALSA_CHECK(snd_config_delete(top));
}

static void test_searchv(void)
{
	const char *text =
		"a 42\n"
		"b {\n"
		"    c cee\n"
		"    d {\n"
		"        e 2.71828\n"
		"    }\n"
		"}\n";
	snd_input_t *input;
	snd_config_t *top, *c;
	const char *id;

	ALSA_CHECK(snd_input_buffer_open(&input, text, strlen(text)));
	ALSA_CHECK(snd_config_top(&top));
	ALSA_CHECK(snd_config_load(top, input));
	ALSA_CHECK(snd_input_close(input));

	ALSA_CHECK(snd_config_searchv(top, &c, "a", NULL));
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "a"));
	ALSA_CHECK(snd_config_searchv(top, &c, "b", "d.e", NULL));
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "e"));
	ALSA_CHECK(snd_config_searchv(top, NULL, "b.c", NULL));
	TEST_CHECK(snd_config_searchv(top, NULL, "x", NULL) == -ENOENT);
	TEST_CHECK(snd_config_searchv(top, &c, "b.y", NULL) == -ENOENT);
	TEST_CHECK(snd_config_searchv(top, &c, "a", "z", NULL) == -ENOENT);

	ALSA_CHECK(snd_config_delete(top));
}

static void test_add(void)
{
	snd_config_t *c1, *c2, *c3, *c4, *c5;
	snd_config_iterator_t i;
	unsigned int count = 0;

	ALSA_CHECK(snd_config_top(&c1));
	ALSA_CHECK(snd_config_imake_integer(&c2, "c2", 0xc2));
	ALSA_CHECK(snd_config_add(c1, c2));
	ALSA_CHECK(snd_config_imake_string(&c3, "c3", "c3"));
	ALSA_CHECK(snd_config_add(c1, c3));
	for (i = snd_config_iterator_first(c1);
	     i != snd_config_iterator_end(c1);
	     i = snd_config_iterator_next(i))
		++count;
	TEST_CHECK(count == 2);
	ALSA_CHECK(snd_config_search(c1, "c2", &c2));
	ALSA_CHECK(snd_config_search(c1, "c3", &c3));
	ALSA_CHECK(snd_config_top(&c4));
	TEST_CHECK(snd_config_add(c1, c4) == -EINVAL);
	ALSA_CHECK(snd_config_imake_integer(&c5, "c5", 5));
	ALSA_CHECK(snd_config_add(c4, c5));
	TEST_CHECK(snd_config_add(c1, c5) == -EINVAL);
	ALSA_CHECK(snd_config_delete(c4));
	ALSA_CHECK(snd_config_imake_integer(&c3, "c3", 333));
	TEST_CHECK(snd_config_add(c1, c3) == -EEXIST);
	ALSA_CHECK(snd_config_delete(c3));
	ALSA_CHECK(snd_config_delete(c1));
}

static void test_delete(void)
{
	snd_config_t *c;

	ALSA_CHECK(snd_config_top(&c));
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_imake_string(&c, "s", "..."));
	ALSA_CHECK(snd_config_delete(c));
}

static void test_copy(void)
{
	snd_config_t *c1, *c2, *c3;
	long value;

	ALSA_CHECK(snd_config_imake_integer(&c1, "c1", 123));
	ALSA_CHECK(snd_config_copy(&c2, c1));
	ALSA_CHECK(snd_config_set_integer(c1, 456));
	TEST_CHECK(snd_config_get_type(c2) == SND_CONFIG_TYPE_INTEGER);
	ALSA_CHECK(snd_config_get_integer(c2, &value));
	TEST_CHECK(value == 123);
	ALSA_CHECK(snd_config_delete(c1));
	ALSA_CHECK(snd_config_delete(c2));
	ALSA_CHECK(snd_config_top(&c1));
	ALSA_CHECK(snd_config_imake_integer(&c2, "a", 1));
	ALSA_CHECK(snd_config_add(c1, c2));
	ALSA_CHECK(snd_config_copy(&c3, c1));
	ALSA_CHECK(snd_config_set_integer(c2, 2));
	TEST_CHECK(!configs_equal(c1, c3));
	ALSA_CHECK(snd_config_search(c3, "a", &c2));
	ALSA_CHECK(snd_config_set_integer(c2, 2));
	TEST_CHECK(configs_equal(c1, c3));
	ALSA_CHECK(snd_config_delete(c1));
	ALSA_CHECK(snd_config_delete(c3));
}

static void test_make_integer(void)
{
	snd_config_t *c;
	const char *id;
	long value;

	ALSA_CHECK(snd_config_make_integer(&c, "i"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_INTEGER);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "i"));
	ALSA_CHECK(snd_config_get_integer(c, &value));
	TEST_CHECK(value == 0);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_make_integer64(void)
{
	snd_config_t *c;
	const char *id;
	long long value;

	ALSA_CHECK(snd_config_make_integer64(&c, "i"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_INTEGER64);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "i"));
	ALSA_CHECK(snd_config_get_integer64(c, &value));
	TEST_CHECK(value == 0);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_make_string(void)
{
	snd_config_t *c;
	const char *id;
	const char *value;

	ALSA_CHECK(snd_config_make_string(&c, "s"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_STRING);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "s"));
	ALSA_CHECK(snd_config_get_string(c, &value));
	TEST_CHECK(value == NULL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_make_compound(void)
{
	snd_config_t *c;
	const char *id;

	ALSA_CHECK(snd_config_make_compound(&c, "c", 0));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_COMPOUND);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "c"));
	TEST_CHECK(snd_config_iterator_first(c) == snd_config_iterator_end(c));
	ALSA_CHECK(snd_config_delete(c));
}

static void test_imake_integer(void)
{
	snd_config_t *c;
	const char *id;
	long value;

	ALSA_CHECK(snd_config_imake_integer(&c, "i", 123));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_INTEGER);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "i"));
	ALSA_CHECK(snd_config_get_integer(c, &value));
	TEST_CHECK(value == 123);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_imake_integer64(void)
{
	snd_config_t *c;
	const char *id;
	long long value;

	ALSA_CHECK(snd_config_imake_integer64(&c, "i", 123456789012345LL));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_INTEGER64);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "i"));
	ALSA_CHECK(snd_config_get_integer64(c, &value));
	TEST_CHECK(value == 123456789012345LL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_imake_string(void)
{
	snd_config_t *c;
	const char *id;
	const char *value;

	ALSA_CHECK(snd_config_imake_string(&c, "s", "xyzzy"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_STRING);
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "s"));
	ALSA_CHECK(snd_config_get_string(c, &value));
	TEST_CHECK(!strcmp(value, "xyzzy"));
	ALSA_CHECK(snd_config_delete(c));
}

static void test_get_type(void)
{
	snd_config_t *c;

	ALSA_CHECK(snd_config_top(&c));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_COMPOUND);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_integer(&c, "i"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_INTEGER);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_string(&c, "s"));
	TEST_CHECK(snd_config_get_type(c) == SND_CONFIG_TYPE_STRING);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_set_integer(void)
{
	snd_config_t *c;
	long value;

	ALSA_CHECK(snd_config_make_integer(&c, "i"));
	ALSA_CHECK(snd_config_set_integer(c, 123));
	ALSA_CHECK(snd_config_get_integer(c, &value));
	TEST_CHECK(value == 123);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_string(&c, "s"));
	TEST_CHECK(snd_config_set_integer(c, 123) == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_set_integer64(void)
{
	snd_config_t *c;
	long long value;

	ALSA_CHECK(snd_config_make_integer64(&c, "i"));
	ALSA_CHECK(snd_config_set_integer64(c, 123456789012345LL));
	ALSA_CHECK(snd_config_get_integer64(c, &value));
	TEST_CHECK(value == 123456789012345LL);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_string(&c, "s"));
	TEST_CHECK(snd_config_set_integer64(c, 123) == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_set_string(void)
{
	snd_config_t *c;
	const char *value;

	ALSA_CHECK(snd_config_make_string(&c, "s"));
	ALSA_CHECK(snd_config_set_string(c, "string"));
	ALSA_CHECK(snd_config_get_string(c, &value));
	TEST_CHECK(!strcmp(value, "string"));
	ALSA_CHECK(snd_config_set_string(c, NULL));
	ALSA_CHECK(snd_config_get_string(c, &value));
	TEST_CHECK(value == NULL);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_integer(&c, "i"));
	TEST_CHECK(snd_config_set_string(c, "") == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_set_ascii(void)
{
	snd_config_t *c;
	const char *s;
	long i;

	ALSA_CHECK(snd_config_make_string(&c, "s"));
	ALSA_CHECK(snd_config_set_ascii(c, "foo"));
	ALSA_CHECK(snd_config_get_string(c, &s));
	TEST_CHECK(!strcmp(s, "foo"));
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_make_integer(&c, "i"));
	ALSA_CHECK(snd_config_set_ascii(c, "23"));
	ALSA_CHECK(snd_config_get_integer(c, &i));
	TEST_CHECK(i == 23);
	TEST_CHECK(snd_config_set_ascii(c, "half blue") == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_top(&c));
	TEST_CHECK(snd_config_set_ascii(c, "0") == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_get_id(void)
{
	snd_config_t *c;
	const char *id;

	ALSA_CHECK(snd_config_make_integer(&c, "my_id"));
	ALSA_CHECK(snd_config_get_id(c, &id));
	TEST_CHECK(!strcmp(id, "my_id"));
	ALSA_CHECK(snd_config_delete(c));
}

#define test_get_integer test_set_integer
#define test_get_integer64 test_set_integer64
#define test_get_string test_set_string

static void test_get_ascii(void)
{
	snd_config_t *c;
	char *value;

	ALSA_CHECK(snd_config_imake_integer(&c, "i", 123));
	ALSA_CHECK(snd_config_get_ascii(c, &value));
	TEST_CHECK(!strcmp(value, "123"));
	free(value);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_imake_string(&c, "s", "bar"));
	ALSA_CHECK(snd_config_get_ascii(c, &value));
	TEST_CHECK(!strcmp(value, "bar"));
	free(value);
	ALSA_CHECK(snd_config_delete(c));
	ALSA_CHECK(snd_config_top(&c));
	TEST_CHECK(snd_config_get_ascii(c, &value) == -EINVAL);
	ALSA_CHECK(snd_config_delete(c));
}

static void test_iterators(void)
{
	snd_config_t *c, *c2;
	snd_config_iterator_t i;
	long v;

	ALSA_CHECK(snd_config_top(&c));
	i = snd_config_iterator_first(c);
	TEST_CHECK(i == snd_config_iterator_end(c));
	ALSA_CHECK(snd_config_imake_integer(&c2, "one", 1));
	ALSA_CHECK(snd_config_add(c, c2));
	i = snd_config_iterator_first(c);
	TEST_CHECK(i != snd_config_iterator_end(c));
	c2 = snd_config_iterator_entry(i);
	ALSA_CHECK(snd_config_get_integer(c2, &v));
	TEST_CHECK(v == 1);
	i = snd_config_iterator_next(i);
	TEST_CHECK(i == snd_config_iterator_end(c));
	ALSA_CHECK(snd_config_delete(c));
}

static void test_for_each(void)
{
	snd_config_t *c, *c2;
	snd_config_iterator_t i, next;
	long v;
	unsigned int count = 0;

	ALSA_CHECK(snd_config_top(&c));
	ALSA_CHECK(snd_config_imake_integer(&c2, "one", 1));
	ALSA_CHECK(snd_config_add(c, c2));
	snd_config_for_each(i, next, c) {
		TEST_CHECK(i != snd_config_iterator_end(c));
		c2 = snd_config_iterator_entry(i);
		ALSA_CHECK(snd_config_get_integer(c2, &v));
		TEST_CHECK(v == 1);
		++count;
	}
	TEST_CHECK(count == 1);
	ALSA_CHECK(snd_config_delete(c));
}

int main(void)
{
	test_top();
	test_load();
	test_save();
	test_update();
	test_search();
	test_searchv();
	test_add();
	test_delete();
	test_copy();
	test_make_integer();
	test_make_integer64();
	test_make_string();
	test_make_compound();
	test_imake_integer();
	test_imake_integer64();
	test_imake_string();
	test_get_type();
	test_set_integer();
	test_set_integer64();
	test_set_string();
	test_set_ascii();
	test_get_id();
	test_get_integer();
	test_get_integer64();
	test_get_string();
	test_get_ascii();
	test_iterators();
	test_for_each();
	return TEST_EXIT_CODE();
}