#include #include #include #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)); } static int _expand_fcn(snd_config_t **dst, const char *s, void *private_data ATTRIBUTE_UNUSED) { if (strcmp(s, "var10") == 0) return snd_config_imake_integer(dst, NULL, 10); if (strcmp(s, "var50") == 0) return snd_config_imake_integer(dst, NULL, 50); return snd_config_imake_string(dst, NULL, ""); } static void test_evaluate_string(void) { struct { const char *expr; long long result; } *p, e[] = { { .expr = "$var10", .result = 10 }, { .expr = "$var50", .result = 50 }, { .expr = "$[1+1]", .result = 2 }, { .expr = "$[10-5]", .result = 5 }, { .expr = "$[10*5]", .result = 50 }, { .expr = "$[15/5]", .result = 3 }, { .expr = "$[12%5]", .result = 2 }, { .expr = "$[0xaa|0x55]", .result = 0xff }, { .expr = "$[0xff&0xfc]", .result = 0xfc }, { .expr = "$[4294967296+10]", .result = 4294967306LL }, { .expr = "$[$var10+1]", .result = 11 }, { .expr = "$[$var10 + $var50]", .result = 60 }, { .expr = "$[ $var10 + $[ $var50 + 10 ] ]", .result = 70 }, { .expr = "$[ ( $var10 + ( $var50 + 112 ) ) + 5 ]", .result = 177 }, { .expr = NULL, .result = 0 }, }; snd_config_t *dst; long l; long long ll; for (p = e; p->expr; p++) { ALSA_CHECK(snd_config_evaluate_string(&dst, p->expr, _expand_fcn, NULL)); if (snd_config_get_type(dst) == SND_CONFIG_TYPE_INTEGER) { ALSA_CHECK(snd_config_get_integer(dst, &l)); TEST_CHECK(l == p->result); } else if (snd_config_get_type(dst) == SND_CONFIG_TYPE_INTEGER64) { ALSA_CHECK(snd_config_get_integer64(dst, &ll)); TEST_CHECK(ll == p->result); } else { ALSA_CHECK(0); } ALSA_CHECK(snd_config_delete(dst)); } } static void test_load_string(void) { const char **cfg, *configs[] = { "a=1,b=2", "j 3;z 15;", "x 0 y -1", NULL }; snd_config_t *dst; for (cfg = configs; *cfg; cfg++) { ALSA_CHECK(snd_config_load_string(&dst, *cfg, 0)); ALSA_CHECK(snd_config_delete(dst)); } } 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(); test_evaluate_string(); test_load_string(); return TEST_EXIT_CODE(); }