pipewire/test/test-spa-pod.c

1707 lines
62 KiB
C

/* Simple Plugin API
* Copyright © 2019 Wim Taymans <wim.taymans@gmail.com>
*
* 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 (including the next
* paragraph) 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.
*/
#include <spa/pod/pod.h>
#include <spa/pod/builder.h>
#include <spa/pod/command.h>
#include <spa/pod/event.h>
#include <spa/pod/iter.h>
#include <spa/pod/parser.h>
#include <spa/pod/vararg.h>
#include <spa/debug/pod.h>
#include <spa/param/format.h>
#include <spa/param/video/raw.h>
#include <spa/utils/string.h>
#include "pwtest.h"
PWTEST(pod_abi_sizes)
{
#if defined(__x86_64__) && defined(__LP64__)
spa_assert_se(sizeof(struct spa_pod) == 8);
spa_assert_se(sizeof(struct spa_pod_bool) == 16);
spa_assert_se(sizeof(struct spa_pod_id) == 16);
spa_assert_se(sizeof(struct spa_pod_int) == 16);
spa_assert_se(sizeof(struct spa_pod_long) == 16);
spa_assert_se(sizeof(struct spa_pod_float) == 16);
spa_assert_se(sizeof(struct spa_pod_double) == 16);
spa_assert_se(sizeof(struct spa_pod_string) == 8);
spa_assert_se(sizeof(struct spa_pod_bytes) == 8);
spa_assert_se(sizeof(struct spa_pod_rectangle) == 16);
spa_assert_se(sizeof(struct spa_pod_fraction) == 16);
spa_assert_se(sizeof(struct spa_pod_bitmap) == 8);
spa_assert_se(sizeof(struct spa_pod_array_body) == 8);
spa_assert_se(sizeof(struct spa_pod_array) == 16);
spa_assert_se(sizeof(struct spa_pod_choice_body) == 16);
spa_assert_se(sizeof(struct spa_pod_choice) == 24);
spa_assert_se(sizeof(struct spa_pod_struct) == 8);
spa_assert_se(sizeof(struct spa_pod_object_body) == 8);
spa_assert_se(sizeof(struct spa_pod_object) == 16);
spa_assert_se(sizeof(struct spa_pod_pointer_body) == 16);
spa_assert_se(sizeof(struct spa_pod_pointer) == 24);
spa_assert_se(sizeof(struct spa_pod_fd) == 16);
spa_assert_se(sizeof(struct spa_pod_prop) == 16);
spa_assert_se(sizeof(struct spa_pod_control) == 16);
spa_assert_se(sizeof(struct spa_pod_sequence_body) == 8);
spa_assert_se(sizeof(struct spa_pod_sequence) == 16);
/* builder */
spa_assert_se(sizeof(struct spa_pod_frame) == 24);
spa_assert_se(sizeof(struct spa_pod_builder_state) == 16);
spa_assert_se(sizeof(struct spa_pod_builder) == 48);
/* command */
spa_assert_se(sizeof(struct spa_command_body) == 8);
spa_assert_se(sizeof(struct spa_command) == 16);
/* event */
spa_assert_se(sizeof(struct spa_event_body) == 8);
spa_assert_se(sizeof(struct spa_event) == 16);
/* parser */
spa_assert_se(sizeof(struct spa_pod_parser_state) == 16);
spa_assert_se(sizeof(struct spa_pod_parser) == 32);
return PWTEST_PASS;
#endif
return PWTEST_SKIP;
}
PWTEST(pod_abi)
{
spa_assert_se(SPA_CHOICE_None == 0);
spa_assert_se(SPA_CHOICE_Range == 1);
spa_assert_se(SPA_CHOICE_Step == 2);
spa_assert_se(SPA_CHOICE_Enum == 3);
spa_assert_se(SPA_CHOICE_Flags == 4);
return PWTEST_PASS;
}
PWTEST(pod_init)
{
{
struct spa_pod pod = SPA_POD_INIT(sizeof(int64_t), SPA_TYPE_Long);
int32_t val;
spa_assert_se(SPA_POD_SIZE(&pod) == sizeof(int64_t) + 8);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == sizeof(int64_t));
spa_assert_se(SPA_POD_CONTENTS_SIZE(struct spa_pod, &pod) == sizeof(int64_t));
spa_assert_se(spa_pod_is_long(&pod));
pod = SPA_POD_INIT(sizeof(int32_t), SPA_TYPE_Int);
spa_assert_se(SPA_POD_SIZE(&pod) == sizeof(int32_t) + 8);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Int);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == sizeof(int32_t));
spa_assert_se(SPA_POD_CONTENTS_SIZE(struct spa_pod, &pod) == sizeof(int32_t));
spa_assert_se(spa_pod_is_int(&pod));
/** too small */
pod = SPA_POD_INIT(0, SPA_TYPE_Int);
spa_assert_se(!spa_pod_is_int(&pod));
spa_assert_se(spa_pod_get_int(&pod, &val) < 0);
}
{
struct spa_pod pod = SPA_POD_INIT_None();
spa_assert_se(SPA_POD_SIZE(&pod) == 8);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_None);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 0);
spa_assert_se(SPA_POD_CONTENTS_SIZE(struct spa_pod, &pod) == 0);
spa_assert_se(spa_pod_is_none(&pod));
}
{
struct spa_pod_bool pod = SPA_POD_INIT_Bool(true);
bool val;
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Bool);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_bool, &pod) == true);
spa_assert_se(spa_pod_is_bool(&pod.pod));
spa_assert_se(spa_pod_get_bool(&pod.pod, &val) == 0);
spa_assert_se(val == true);
pod = SPA_POD_INIT_Bool(false);
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Bool);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_bool, &pod) == false);
spa_assert_se(spa_pod_is_bool(&pod.pod));
spa_assert_se(spa_pod_get_bool(&pod.pod, &val) == 0);
spa_assert_se(val == false);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Bool);
spa_assert_se(!spa_pod_is_bool(&pod.pod));
spa_assert_se(spa_pod_get_bool(&pod.pod, &val) < 0);
}
{
struct spa_pod_id pod = SPA_POD_INIT_Id(SPA_TYPE_Int);
uint32_t val;
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Id);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_id, &pod) == SPA_TYPE_Int);
spa_assert_se(spa_pod_is_id(&pod.pod));
spa_assert_se(spa_pod_get_id(&pod.pod, &val) == 0);
spa_assert_se(val == SPA_TYPE_Int);
pod = SPA_POD_INIT_Id(SPA_TYPE_Long);
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Id);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_id, &pod) == SPA_TYPE_Long);
spa_assert_se(spa_pod_is_id(&pod.pod));
spa_assert_se(spa_pod_get_id(&pod.pod, &val) == 0);
spa_assert_se(val == SPA_TYPE_Long);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Id);
spa_assert_se(!spa_pod_is_id(&pod.pod));
spa_assert_se(spa_pod_get_id(&pod.pod, &val) < 0);
}
{
struct spa_pod_int pod = SPA_POD_INIT_Int(23);
int32_t val;
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Int);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_int, &pod) == 23);
spa_assert_se(spa_pod_is_int(&pod.pod));
spa_assert_se(spa_pod_get_int(&pod.pod, &val) == 0);
spa_assert_se(val == 23);
pod = SPA_POD_INIT_Int(-123);
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Int);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_int, &pod) == -123);
spa_assert_se(spa_pod_is_int(&pod.pod));
spa_assert_se(spa_pod_get_int(&pod.pod, &val) == 0);
spa_assert_se(val == -123);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Int);
spa_assert_se(!spa_pod_is_int(&pod.pod));
spa_assert_se(spa_pod_get_int(&pod.pod, &val) < 0);
}
{
struct spa_pod_long pod = SPA_POD_INIT_Long(-23);
int64_t val;
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_long, &pod) == -23);
spa_assert_se(spa_pod_is_long(&pod.pod));
spa_assert_se(spa_pod_get_long(&pod.pod, &val) == 0);
spa_assert_se(val == -23);
pod = SPA_POD_INIT_Long(123);
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_long, &pod) == 123);
spa_assert_se(spa_pod_is_long(&pod.pod));
spa_assert_se(spa_pod_get_long(&pod.pod, &val) == 0);
spa_assert_se(val == 123);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Long);
spa_assert_se(!spa_pod_is_long(&pod.pod));
spa_assert_se(spa_pod_get_long(&pod.pod, &val) < 0);
}
{
struct spa_pod_float pod = SPA_POD_INIT_Float(0.67f);
float val;
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Float);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_float, &pod) == 0.67f);
spa_assert_se(spa_pod_is_float(&pod.pod));
spa_assert_se(spa_pod_get_float(&pod.pod, &val) == 0);
spa_assert_se(val == 0.67f);
pod = SPA_POD_INIT_Float(-134.8f);
spa_assert_se(SPA_POD_SIZE(&pod) == 12);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Float);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 4);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_float, &pod) == -134.8f);
spa_assert_se(spa_pod_is_float(&pod.pod));
spa_assert_se(spa_pod_get_float(&pod.pod, &val) == 0);
spa_assert_se(val == -134.8f);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Float);
spa_assert_se(!spa_pod_is_float(&pod.pod));
spa_assert_se(spa_pod_get_float(&pod.pod, &val) < 0);
}
{
struct spa_pod_double pod = SPA_POD_INIT_Double(0.67);
double val;
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Double);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_double, &pod) == 0.67);
spa_assert_se(spa_pod_is_double(&pod.pod));
spa_assert_se(spa_pod_get_double(&pod.pod, &val) == 0);
spa_assert_se(val == 0.67);
pod = SPA_POD_INIT_Double(-134.8);
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Double);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(SPA_POD_VALUE(struct spa_pod_double, &pod) == -134.8);
spa_assert_se(spa_pod_is_double(&pod.pod));
spa_assert_se(spa_pod_get_double(&pod.pod, &val) == 0);
spa_assert_se(val == -134.8);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Double);
spa_assert_se(!spa_pod_is_double(&pod.pod));
spa_assert_se(spa_pod_get_double(&pod.pod, &val) < 0);
}
{
struct {
struct spa_pod_string pod;
char str[9];
} pod;
char val[12];
pod.pod = SPA_POD_INIT_String(9);
strncpy(pod.str, "test", 9);
spa_assert_se(SPA_POD_SIZE(&pod) == 17);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_String);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 9);
spa_assert_se(spa_pod_is_string(&pod.pod.pod));
spa_assert_se(spa_pod_copy_string(&pod.pod.pod, sizeof(val), val) == 0);
spa_assert_se(spa_streq(pod.str, val));
pod.pod = SPA_POD_INIT_String(6);
memcpy(pod.str, "test123456789", 9);
spa_assert_se(SPA_POD_SIZE(&pod) == 14);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_String);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 6);
spa_assert_se(!spa_pod_is_string(&pod.pod.pod));
spa_assert_se(spa_pod_copy_string(&pod.pod.pod, sizeof(val), val) < 0);
}
{
struct spa_pod_rectangle pod = SPA_POD_INIT_Rectangle(SPA_RECTANGLE(320,240));
struct spa_rectangle val;
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Rectangle);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(memcmp(&SPA_POD_VALUE(struct spa_pod_rectangle, &pod),
&SPA_RECTANGLE(320,240), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(spa_pod_is_rectangle(&pod.pod));
spa_assert_se(spa_pod_get_rectangle(&pod.pod, &val) == 0);
spa_assert_se(memcmp(&val, &SPA_RECTANGLE(320,240), sizeof(struct spa_rectangle)) == 0);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Rectangle);
spa_assert_se(!spa_pod_is_rectangle(&pod.pod));
spa_assert_se(spa_pod_get_rectangle(&pod.pod, &val) < 0);
}
{
struct spa_pod_fraction pod = SPA_POD_INIT_Fraction(SPA_FRACTION(25,1));
struct spa_fraction val;
spa_assert_se(SPA_POD_SIZE(&pod) == 16);
spa_assert_se(SPA_POD_TYPE(&pod) == SPA_TYPE_Fraction);
spa_assert_se(SPA_POD_BODY_SIZE(&pod) == 8);
spa_assert_se(memcmp(&SPA_POD_VALUE(struct spa_pod_fraction, &pod),
&SPA_FRACTION(25,1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(spa_pod_is_fraction(&pod.pod));
spa_assert_se(spa_pod_get_fraction(&pod.pod, &val) == 0);
spa_assert_se(memcmp(&val, &SPA_FRACTION(25,1), sizeof(struct spa_fraction)) == 0);
pod.pod = SPA_POD_INIT(0, SPA_TYPE_Fraction);
spa_assert_se(!spa_pod_is_fraction(&pod.pod));
spa_assert_se(spa_pod_get_fraction(&pod.pod, &val) < 0);
}
return PWTEST_PASS;
}
PWTEST(pod_build)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod *array, *choice, *head, *pod, *it;
const struct spa_pod_prop *prop;
struct spa_pod_control *control;
int64_t longs[] = { 5, 7, 11, 13, 17 }, *al;
uint32_t i, len, yl, *ai;
union {
bool b;
uint32_t I;
int32_t i;
int64_t l;
float f;
double d;
const char *s;
const void *y;
const void *p;
int64_t h;
struct spa_rectangle R;
struct spa_fraction F;
} val;
struct spa_pod_frame f;
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(b.data == buffer);
spa_assert_se(b.size == sizeof(buffer));
spa_assert_se(b.state.offset == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(spa_pod_builder_none(&b) == 0);
spa_assert_se(b.state.offset == 8);
spa_assert_se(spa_pod_builder_bool(&b, true) == 0);
spa_assert_se(b.state.offset == 24);
spa_assert_se(spa_pod_builder_id(&b, SPA_TYPE_Object) == 0);
spa_assert_se(b.state.offset == 40);
spa_assert_se(spa_pod_builder_int(&b, 21) == 0);
spa_assert_se(b.state.offset == 56);
spa_assert_se(spa_pod_builder_float(&b, 0.8f) == 0);
spa_assert_se(b.state.offset == 72);
spa_assert_se(spa_pod_builder_double(&b, -1.56) == 0);
spa_assert_se(b.state.offset == 88);
spa_assert_se(spa_pod_builder_string(&b, "test") == 0);
spa_assert_se(b.state.offset == 104);
spa_assert_se(spa_pod_builder_bytes(&b, "PipeWire", 8) == 0);
spa_assert_se(b.state.offset == 120);
spa_assert_se(spa_pod_builder_pointer(&b, SPA_TYPE_Object, &b) == 0);
spa_assert_se(b.state.offset == 144);
spa_assert_se(spa_pod_builder_fd(&b, 4) == 0);
spa_assert_se(b.state.offset == 160);
spa_assert_se(spa_pod_builder_rectangle(&b, 320, 240) == 0);
spa_assert_se(b.state.offset == 176);
spa_assert_se(spa_pod_builder_fraction(&b, 25, 1) == 0);
spa_assert_se(b.state.offset == 192);
spa_assert_se(spa_pod_builder_push_array(&b, &f) == 0);
spa_assert_se(f.offset == 192);
spa_assert_se(b.state.flags == (SPA_POD_BUILDER_FLAG_BODY | SPA_POD_BUILDER_FLAG_FIRST));
spa_assert_se(b.state.offset == 200);
spa_assert_se(spa_pod_builder_int(&b, 1) == 0);
spa_assert_se(b.state.flags == SPA_POD_BUILDER_FLAG_BODY);
spa_assert_se(b.state.offset == 212);
spa_assert_se(spa_pod_builder_int(&b, 2) == 0);
spa_assert_se(b.state.offset == 216);
spa_assert_se(spa_pod_builder_int(&b, 3) == 0);
array = spa_pod_builder_pop(&b, &f);
spa_assert_se(f.pod.size == 20);
spa_assert_se(array != NULL);
spa_assert_se(SPA_POD_BODY_SIZE(array) == 8 + 12);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 224);
spa_assert_se(spa_pod_builder_array(&b,
sizeof(int64_t), SPA_TYPE_Long,
SPA_N_ELEMENTS(longs), longs) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 280);
spa_assert_se(spa_pod_builder_push_choice(&b, &f, SPA_CHOICE_Enum, 0) == 0);
spa_assert_se(b.state.flags == (SPA_POD_BUILDER_FLAG_BODY | SPA_POD_BUILDER_FLAG_FIRST));
spa_assert_se(b.state.offset == 296);
spa_assert_se(spa_pod_builder_long(&b, 1) == 0);
spa_assert_se(b.state.flags == SPA_POD_BUILDER_FLAG_BODY);
spa_assert_se(b.state.offset == 312);
spa_assert_se(spa_pod_builder_long(&b, 2) == 0);
spa_assert_se(b.state.offset == 320);
spa_assert_se(spa_pod_builder_long(&b, 3) == 0);
choice = spa_pod_builder_pop(&b, &f);
spa_assert_se(choice != NULL);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 328);
spa_assert_se(spa_pod_builder_push_struct(&b, &f) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 336);
spa_assert_se(spa_pod_builder_int(&b, 21) == 0);
spa_assert_se(b.state.offset == 352);
spa_assert_se(spa_pod_builder_float(&b, 0.8f) == 0);
spa_assert_se(b.state.offset == 368);
spa_assert_se(spa_pod_builder_double(&b, -1.56) == 0);
spa_assert_se(spa_pod_builder_pop(&b, &f) != NULL);
spa_assert_se(b.state.offset == 384);
spa_assert_se(spa_pod_builder_push_object(&b, &f, SPA_TYPE_OBJECT_Props, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 400);
spa_assert_se(spa_pod_builder_prop(&b, 1, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 408);
spa_assert_se(spa_pod_builder_int(&b, 21) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 424);
spa_assert_se(spa_pod_builder_prop(&b, 2, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 432);
spa_assert_se(spa_pod_builder_long(&b, 42) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 448);
spa_assert_se(spa_pod_builder_prop(&b, 3, 0) == 0);
spa_assert_se(b.state.offset == 456);
spa_assert_se(spa_pod_builder_string(&b, "test123") == 0);
spa_assert_se(spa_pod_builder_pop(&b, &f) != NULL);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 472);
spa_assert_se(spa_pod_builder_push_sequence(&b, &f, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 488);
spa_assert_se(spa_pod_builder_control(&b, 0, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 496);
spa_assert_se(spa_pod_builder_float(&b, 0.667f) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 512);
spa_assert_se(spa_pod_builder_control(&b, 12, 0) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 520);
spa_assert_se(spa_pod_builder_double(&b, 1.22) == 0);
spa_assert_se(b.state.flags == 0);
spa_assert_se(spa_pod_builder_pop(&b, &f) != NULL);
spa_assert_se(b.state.flags == 0);
spa_assert_se(b.state.offset == 536);
len = b.state.offset;
pod = head = (struct spa_pod *)buffer;
spa_assert_se(spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_none(pod));
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_bool(pod));
spa_assert_se(spa_pod_get_bool(pod, &val.b) == 0);
spa_assert_se(val.b == true);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_id(pod));
spa_assert_se(spa_pod_get_id(pod, &val.I) == 0);
spa_assert_se(val.I == SPA_TYPE_Object);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_int(pod));
spa_assert_se(spa_pod_get_int(pod, &val.i) == 0);
spa_assert_se(val.i == 21);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_float(pod));
spa_assert_se(spa_pod_get_float(pod, &val.f) == 0);
spa_assert_se(val.f == 0.8f);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_double(pod));
spa_assert_se(spa_pod_get_double(pod, &val.d) == 0);
spa_assert_se(val.d == -1.56);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_string(pod));
spa_assert_se(spa_pod_get_string(pod, &val.s) == 0);
spa_assert_se(spa_streq(val.s, "test"));
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_bytes(pod));
spa_assert_se(spa_pod_get_bytes(pod, &val.y, &yl) == 0);
spa_assert_se(yl == 8);
spa_assert_se(memcmp(val.y, "PipeWire", yl) == 0);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_pointer(pod));
spa_assert_se(spa_pod_get_pointer(pod, &yl, &val.p) == 0);
spa_assert_se(yl == SPA_TYPE_Object);
spa_assert_se(val.p == &b);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_fd(pod));
spa_assert_se(spa_pod_get_fd(pod, &val.l) == 0);
spa_assert_se(val.l == 4);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_rectangle(pod));
spa_assert_se(spa_pod_get_rectangle(pod, &val.R) == 0);
spa_assert_se(memcmp(&val.R, &SPA_RECTANGLE(320,240), sizeof(struct spa_rectangle)) == 0);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_fraction(pod));
spa_assert_se(spa_pod_get_fraction(pod, &val.F) == 0);
spa_assert_se(memcmp(&val.F, &SPA_FRACTION(25,1), sizeof(struct spa_fraction)) == 0);
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_array(pod));
spa_assert_se(SPA_POD_ARRAY_VALUE_TYPE(pod) == SPA_TYPE_Int);
spa_assert_se(SPA_POD_ARRAY_VALUE_SIZE(pod) == sizeof(int32_t));
spa_assert_se(SPA_POD_ARRAY_N_VALUES(pod) == 3);
ai = SPA_POD_ARRAY_VALUES(pod);
spa_assert_se(ai != NULL);
spa_assert_se(SPA_POD_ARRAY_CHILD(pod)->type == SPA_TYPE_Int);
spa_assert_se(SPA_POD_ARRAY_CHILD(pod)->size == sizeof(int32_t));
spa_assert_se(ai[0] == 1);
spa_assert_se(ai[1] == 2);
spa_assert_se(ai[2] == 3);
i = 1;
SPA_POD_ARRAY_FOREACH((struct spa_pod_array*)pod, ai) {
spa_assert_se(*ai == i);
i++;
}
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_array(pod));
spa_assert_se(SPA_POD_ARRAY_VALUE_TYPE(pod) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_ARRAY_VALUE_SIZE(pod) == sizeof(int64_t));
spa_assert_se(SPA_POD_ARRAY_N_VALUES(pod) == SPA_N_ELEMENTS(longs));
al = SPA_POD_ARRAY_VALUES(pod);
spa_assert_se(al != NULL);
spa_assert_se(SPA_POD_ARRAY_CHILD(pod)->type == SPA_TYPE_Long);
spa_assert_se(SPA_POD_ARRAY_CHILD(pod)->size == sizeof(int64_t));
for (i = 0; i < SPA_N_ELEMENTS(longs); i++)
spa_assert_se(al[i] == longs[i]);
i = 0;
SPA_POD_ARRAY_FOREACH((struct spa_pod_array*)pod, al) {
spa_assert_se(*al == longs[i++]);
}
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_choice(pod));
spa_assert_se(SPA_POD_CHOICE_TYPE(pod) == SPA_CHOICE_Enum);
spa_assert_se(SPA_POD_CHOICE_FLAGS(pod) == 0);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(pod) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(pod) == sizeof(int64_t));
spa_assert_se(SPA_POD_CHOICE_N_VALUES(pod) == 3);
al = SPA_POD_CHOICE_VALUES(pod);
spa_assert_se(al != NULL);
spa_assert_se(SPA_POD_CHOICE_CHILD(pod)->type == SPA_TYPE_Long);
spa_assert_se(SPA_POD_CHOICE_CHILD(pod)->size == sizeof(int64_t));
spa_assert_se(al[0] == 1);
spa_assert_se(al[1] == 2);
spa_assert_se(al[2] == 3);
i = 1;
SPA_POD_CHOICE_FOREACH((struct spa_pod_choice*)pod, al) {
spa_assert_se(*al == i);
i++;
}
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_struct(pod));
i = 0;
SPA_POD_STRUCT_FOREACH(pod, it) {
switch (i++) {
case 0:
spa_assert_se(spa_pod_is_int(it));
spa_assert_se(spa_pod_get_int(it, &val.i) == 0 && val.i == 21);
break;
case 1:
spa_assert_se(spa_pod_is_float(it));
spa_assert_se(spa_pod_get_float(it, &val.f) == 0 && val.f == 0.8f);
break;
case 2:
spa_assert_se(spa_pod_is_double(it));
spa_assert_se(spa_pod_get_double(it, &val.d) == 0 && val.d == -1.56);
break;
default:
spa_assert_not_reached();
break;
}
}
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_object(pod));
spa_assert_se(spa_pod_is_object_type(pod, SPA_TYPE_OBJECT_Props));
spa_assert_se(spa_pod_is_object_id(pod, 0));
i = 0;
SPA_POD_OBJECT_FOREACH((const struct spa_pod_object*)pod, prop) {
switch (i++) {
case 0:
spa_assert_se(prop->key == 1);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_int(&prop->value, &val.i) == 0 && val.i == 21);
break;
case 1:
spa_assert_se(prop->key == 2);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_long(&prop->value, &val.l) == 0 && val.l == 42);
break;
case 2:
spa_assert_se(prop->key == 3);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_string(&prop->value, &val.s) == 0 &&
spa_streq(val.s, "test123"));
break;
default:
spa_assert_not_reached();
break;
}
}
prop = spa_pod_find_prop(pod, NULL, 3);
spa_assert_se(prop != NULL);
spa_assert_se(prop->key == 3);
spa_assert_se(spa_pod_get_string(&prop->value, &val.s) == 0 &&
spa_streq(val.s, "test123"));
prop = spa_pod_find_prop(pod, prop, 1);
spa_assert_se(prop != NULL);
spa_assert_se(prop->key == 1);
spa_assert_se(spa_pod_get_int(&prop->value, &val.i) == 0 && val.i == 21);
prop = spa_pod_find_prop(pod, prop, 2);
spa_assert_se(prop != NULL);
spa_assert_se(prop->key == 2);
spa_assert_se(spa_pod_get_long(&prop->value, &val.l) == 0 && val.l == 42);
prop = spa_pod_find_prop(pod, prop, 5);
spa_assert_se(prop == NULL);
prop = spa_pod_find_prop(pod, NULL, 3);
spa_assert_se(prop != NULL);
spa_assert_se(prop->key == 3);
spa_assert_se(spa_pod_get_string(&prop->value, &val.s) == 0 &&
spa_streq(val.s, "test123"));
spa_assert_se((pod = spa_pod_next(pod)) != NULL && spa_pod_is_inside(head, len, pod));
spa_assert_se(spa_pod_is_sequence(pod));
i = 0;
SPA_POD_SEQUENCE_FOREACH((const struct spa_pod_sequence*)pod, control) {
switch (i++) {
case 0:
spa_assert_se(control->offset == 0);
spa_assert_se(SPA_POD_CONTROL_SIZE(control) == 20);
spa_assert_se(spa_pod_get_float(&control->value, &val.f) == 0 && val.f == 0.667f);
break;
case 1:
spa_assert_se(control->offset == 12);
spa_assert_se(SPA_POD_CONTROL_SIZE(control) == 24);
spa_assert_se(spa_pod_get_double(&control->value, &val.d) == 0 && val.d == 1.22);
break;
default:
spa_assert_not_reached();
break;
}
}
return PWTEST_PASS;
}
PWTEST(pod_empty)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod *array, *a2, *choice, *ch2;
struct spa_pod_frame f;
uint32_t n_vals, ch;
memset(buffer, 0xab, sizeof(buffer));
/* create empty arrays */
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_array(&b, &f) == 0);
spa_assert_se(spa_pod_builder_child(&b, sizeof(uint32_t), SPA_TYPE_Id) == 0);
array = spa_pod_builder_pop(&b, &f);
spa_assert_se(array != NULL);
spa_debug_mem(0, array, 16);
spa_assert_se(spa_pod_is_array(array));
a2 = spa_pod_get_array(array, &n_vals);
spa_assert_se(a2 != NULL);
spa_assert_se(n_vals == 0);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_array(&b, &f) == 0);
array = spa_pod_builder_pop(&b, &f);
spa_assert_se(array != NULL);
spa_assert_se(spa_pod_is_array(array));
a2 = spa_pod_get_array(array, &n_vals);
spa_assert_se(a2 != NULL);
spa_assert_se(n_vals == 0);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_array(&b, &f) == 0);
spa_assert_se(spa_pod_builder_none(&b) == 0);
array = spa_pod_builder_pop(&b, &f);
spa_assert_se(array != NULL);
spa_assert_se(spa_pod_is_array(array));
a2 = spa_pod_get_array(array, &n_vals);
spa_assert_se(a2 != NULL);
spa_assert_se(n_vals == 0);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_array(&b, 4, SPA_TYPE_Id, 0, NULL) == 0);
array = (struct spa_pod*)buffer;
spa_assert_se(spa_pod_is_array(array));
a2 = spa_pod_get_array(array, &n_vals);
spa_assert_se(a2 != NULL);
spa_assert_se(n_vals == 0);
/* create empty choice */
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_choice(&b, &f, 0, 0) == 0);
spa_assert_se(spa_pod_builder_child(&b, sizeof(uint32_t), SPA_TYPE_Id) == 0);
choice = spa_pod_builder_pop(&b, &f);
spa_assert_se(choice != NULL);
spa_debug_mem(0, choice, 32);
spa_assert_se(spa_pod_is_choice(choice));
ch2 = spa_pod_get_values(choice, &n_vals, &ch);
spa_assert_se(ch2 != NULL);
spa_assert_se(n_vals == 0);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_choice(&b, &f, 0, 0) == 0);
choice = spa_pod_builder_pop(&b, &f);
spa_assert_se(choice != NULL);
spa_assert_se(spa_pod_is_choice(choice));
ch2 = spa_pod_get_values(choice, &n_vals, &ch);
spa_assert_se(ch2 != NULL);
spa_assert_se(n_vals == 0);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_assert_se(spa_pod_builder_push_choice(&b, &f, 0, 0) == 0);
spa_assert_se(spa_pod_builder_none(&b) == 0);
choice = spa_pod_builder_pop(&b, &f);
spa_assert_se(choice != NULL);
spa_assert_se(spa_pod_is_choice(choice));
ch2 = spa_pod_get_values(choice, &n_vals, &ch);
spa_assert_se(ch2 != NULL);
spa_assert_se(n_vals == 0);
return PWTEST_PASS;
}
PWTEST(pod_varargs)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod *pod;
struct spa_pod_prop *prop;
uint32_t i, *aI;
union {
bool b;
uint32_t I;
int32_t i;
int64_t l;
float f;
double d;
const char *s;
const void *y;
const void *p;
int64_t h;
struct spa_rectangle R;
struct spa_fraction F;
} val;
uint32_t media_type, media_subtype, format;
int32_t views;
struct spa_rectangle *aR, size;
struct spa_fraction *aF, framerate;
struct spa_pod *Vformat, *Vsize, *Vframerate;
spa_pod_builder_init(&b, buffer, sizeof(buffer));
pod = spa_pod_builder_add_object(&b,
SPA_TYPE_OBJECT_Format, 0,
SPA_FORMAT_mediaType, SPA_POD_Id(SPA_MEDIA_TYPE_video),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(SPA_MEDIA_SUBTYPE_raw),
SPA_FORMAT_VIDEO_format, SPA_POD_CHOICE_ENUM_Id(3,
SPA_VIDEO_FORMAT_I420,
SPA_VIDEO_FORMAT_I420,
SPA_VIDEO_FORMAT_YUY2),
SPA_FORMAT_VIDEO_size, SPA_POD_CHOICE_RANGE_Rectangle(
&SPA_RECTANGLE(320,242),
&SPA_RECTANGLE(1,1),
&SPA_RECTANGLE(INT32_MAX,INT32_MAX)),
SPA_FORMAT_VIDEO_framerate, SPA_POD_CHOICE_RANGE_Fraction(
&SPA_FRACTION(25,1),
&SPA_FRACTION(0,1),
&SPA_FRACTION(INT32_MAX,1)));
i = 0;
SPA_POD_OBJECT_FOREACH((const struct spa_pod_object*)pod, prop) {
switch (i++) {
case 0:
spa_assert_se(prop->key == SPA_FORMAT_mediaType);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_id(&prop->value, &val.I) == 0 && val.I == SPA_MEDIA_TYPE_video);
break;
case 1:
spa_assert_se(prop->key == SPA_FORMAT_mediaSubtype);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_id(&prop->value, &val.I) == 0 && val.I == SPA_MEDIA_SUBTYPE_raw);
break;
case 2:
spa_assert_se(prop->key == SPA_FORMAT_VIDEO_format);
spa_assert_se(spa_pod_is_choice(&prop->value));
spa_assert_se(SPA_POD_CHOICE_TYPE(&prop->value) == SPA_CHOICE_Enum);
spa_assert_se(SPA_POD_CHOICE_N_VALUES(&prop->value) == 3);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(&prop->value) == SPA_TYPE_Id);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(&prop->value) == sizeof(uint32_t));
aI = SPA_POD_CHOICE_VALUES(&prop->value);
spa_assert_se(aI != NULL);
spa_assert_se(aI[0] == SPA_VIDEO_FORMAT_I420);
spa_assert_se(aI[1] == SPA_VIDEO_FORMAT_I420);
spa_assert_se(aI[2] == SPA_VIDEO_FORMAT_YUY2);
break;
case 3:
spa_assert_se(prop->key == SPA_FORMAT_VIDEO_size);
spa_assert_se(spa_pod_is_choice(&prop->value));
spa_assert_se(SPA_POD_CHOICE_TYPE(&prop->value) == SPA_CHOICE_Range);
spa_assert_se(SPA_POD_CHOICE_N_VALUES(&prop->value) == 3);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(&prop->value) == SPA_TYPE_Rectangle);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(&prop->value) == sizeof(struct spa_rectangle));
aR = SPA_POD_CHOICE_VALUES(&prop->value);
spa_assert_se(aR != NULL);
spa_assert_se(memcmp(&aR[0], &SPA_RECTANGLE(320,242), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&aR[1], &SPA_RECTANGLE(1,1), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&aR[2], &SPA_RECTANGLE(INT32_MAX,INT32_MAX), sizeof(struct spa_rectangle)) == 0);
break;
case 4:
spa_assert_se(prop->key == SPA_FORMAT_VIDEO_framerate);
spa_assert_se(spa_pod_is_choice(&prop->value));
spa_assert_se(SPA_POD_CHOICE_TYPE(&prop->value) == SPA_CHOICE_Range);
spa_assert_se(SPA_POD_CHOICE_N_VALUES(&prop->value) == 3);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(&prop->value) == SPA_TYPE_Fraction);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(&prop->value) == sizeof(struct spa_fraction));
aF = SPA_POD_CHOICE_VALUES(&prop->value);
spa_assert_se(aF != NULL);
spa_assert_se(memcmp(&aF[0], &SPA_FRACTION(25,1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(memcmp(&aF[1], &SPA_FRACTION(0,1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(memcmp(&aF[2], &SPA_FRACTION(INT32_MAX,1), sizeof(struct spa_fraction)) == 0);
break;
default:
spa_assert_not_reached();
break;
}
}
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&media_subtype),
SPA_FORMAT_VIDEO_format, SPA_POD_PodChoice(&Vformat),
SPA_FORMAT_VIDEO_size, SPA_POD_PodChoice(&Vsize),
SPA_FORMAT_VIDEO_framerate, SPA_POD_PodChoice(&Vframerate)) == 5);
spa_assert_se(media_type == SPA_MEDIA_TYPE_video);
spa_assert_se(media_subtype == SPA_MEDIA_SUBTYPE_raw);
spa_assert_se(spa_pod_is_choice(Vformat));
spa_assert_se(SPA_POD_CHOICE_TYPE(Vformat) == SPA_CHOICE_Enum);
spa_assert_se(SPA_POD_CHOICE_N_VALUES(Vformat) == 3);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(Vformat) == SPA_TYPE_Id);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(Vformat) == sizeof(uint32_t));
aI = SPA_POD_CHOICE_VALUES(Vformat);
spa_assert_se(aI != NULL);
spa_assert_se(aI[0] == SPA_VIDEO_FORMAT_I420);
spa_assert_se(aI[1] == SPA_VIDEO_FORMAT_I420);
spa_assert_se(aI[2] == SPA_VIDEO_FORMAT_YUY2);
spa_assert_se(spa_pod_is_choice(Vsize));
spa_assert_se(SPA_POD_CHOICE_TYPE(Vsize) == SPA_CHOICE_Range);
spa_assert_se(SPA_POD_CHOICE_N_VALUES(Vsize) == 3);
spa_assert_se(SPA_POD_CHOICE_VALUE_TYPE(Vsize) == SPA_TYPE_Rectangle);
spa_assert_se(SPA_POD_CHOICE_VALUE_SIZE(Vsize) == sizeof(struct spa_rectangle));
aR = SPA_POD_CHOICE_VALUES(Vsize);
spa_assert_se(aR != NULL);
spa_assert_se(memcmp(&aR[0], &SPA_RECTANGLE(320,242), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&aR[1], &SPA_RECTANGLE(1,1), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&aR[2], &SPA_RECTANGLE(INT32_MAX,INT32_MAX), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(spa_pod_is_choice(Vframerate));
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&media_subtype),
SPA_FORMAT_VIDEO_views, SPA_POD_Int(&views),
SPA_FORMAT_VIDEO_format, SPA_POD_Id(&format),
SPA_FORMAT_VIDEO_size, SPA_POD_Rectangle(&size),
SPA_FORMAT_VIDEO_framerate, SPA_POD_Fraction(&framerate)) == -ESRCH);
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&media_subtype),
SPA_FORMAT_VIDEO_format, SPA_POD_Id(&format),
SPA_FORMAT_VIDEO_size, SPA_POD_Rectangle(&size),
SPA_FORMAT_VIDEO_framerate, SPA_POD_Fraction(&framerate)) == -EPROTO);
spa_debug_pod(0, NULL, pod);
spa_pod_fixate(pod);
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&media_subtype),
SPA_FORMAT_VIDEO_format, SPA_POD_Id(&format),
SPA_FORMAT_VIDEO_views, SPA_POD_OPT_Int(&views),
SPA_FORMAT_VIDEO_size, SPA_POD_Rectangle(&size),
SPA_FORMAT_VIDEO_framerate, SPA_POD_Fraction(&framerate)) == 5);
spa_assert_se(media_type == SPA_MEDIA_TYPE_video);
spa_assert_se(media_subtype == SPA_MEDIA_SUBTYPE_raw);
spa_assert_se(format == SPA_VIDEO_FORMAT_I420);
spa_assert_se(memcmp(&size, &SPA_RECTANGLE(320,242), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&framerate, &SPA_FRACTION(25,1), sizeof(struct spa_fraction)) == 0);
spa_debug_pod(0, NULL, pod);
return PWTEST_PASS;
}
PWTEST(pod_varargs2)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod *pod;
struct spa_pod_prop *prop;
uint32_t i, j;
struct {
bool b;
uint32_t I;
int32_t i;
int64_t l;
float f;
double d;
const char *s;
uint32_t yl;
const void *y;
uint32_t ptype;
const void *p;
uint32_t asize, atype, anvals;
const void *a;
int64_t h;
struct spa_rectangle R;
struct spa_fraction F;
struct spa_pod *P;
} val;
uint8_t bytes[] = { 0x56, 0x00, 0x12, 0xf3, 0xba };
int64_t longs[] = { 1002, 5383, 28944, 1237748 }, *al;
struct spa_pod_int pi = SPA_POD_INIT_Int(77);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
pod = spa_pod_builder_add_object(&b,
SPA_TYPE_OBJECT_Props, 0,
1, SPA_POD_Bool(true),
2, SPA_POD_Id(SPA_TYPE_Id),
3, SPA_POD_Int(3),
4, SPA_POD_Long(4LL),
5, SPA_POD_Float(0.453f),
6, SPA_POD_Double(0.871),
7, SPA_POD_String("test"),
8, SPA_POD_Bytes(bytes, sizeof(bytes)),
9, SPA_POD_Rectangle(&SPA_RECTANGLE(3,4)),
10, SPA_POD_Fraction(&SPA_FRACTION(24,1)),
11, SPA_POD_Array(sizeof(int64_t), SPA_TYPE_Long, SPA_N_ELEMENTS(longs), longs),
12, SPA_POD_Pointer(SPA_TYPE_Object, &b),
13, SPA_POD_Fd(3),
14, SPA_POD_Pod(&pi));
spa_debug_pod(0, NULL, pod);
i = 0;
SPA_POD_OBJECT_FOREACH((const struct spa_pod_object*)pod, prop) {
switch (i++) {
case 0:
spa_assert_se(prop->key == 1);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_bool(&prop->value, &val.b) == 0 && val.b == true);
break;
case 1:
spa_assert_se(prop->key == 2);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_id(&prop->value, &val.I) == 0 && val.I == SPA_TYPE_Id);
break;
case 2:
spa_assert_se(prop->key == 3);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_int(&prop->value, &val.i) == 0 && val.i == 3);
break;
case 3:
spa_assert_se(prop->key == 4);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_long(&prop->value, &val.l) == 0 && val.l == 4);
break;
case 4:
spa_assert_se(prop->key == 5);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_float(&prop->value, &val.f) == 0 && val.f == 0.453f);
break;
case 5:
spa_assert_se(prop->key == 6);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_double(&prop->value, &val.d) == 0 && val.d == 0.871);
break;
case 6:
spa_assert_se(prop->key == 7);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 21);
spa_assert_se(spa_pod_get_string(&prop->value, &val.s) == 0);
spa_assert_se(spa_streq(val.s, "test"));
break;
case 7:
spa_assert_se(prop->key == 8);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 21);
spa_assert_se(spa_pod_get_bytes(&prop->value, &val.y, &val.yl) == 0);
spa_assert_se(val.yl == sizeof(bytes));
spa_assert_se(memcmp(val.y, bytes, val.yl) == 0);
break;
case 8:
spa_assert_se(prop->key == 9);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_rectangle(&prop->value, &val.R) == 0);
spa_assert_se(memcmp(&val.R, &SPA_RECTANGLE(3,4), sizeof(struct spa_rectangle)) == 0);
break;
case 9:
spa_assert_se(prop->key == 10);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_fraction(&prop->value, &val.F) == 0);
spa_assert_se(memcmp(&val.F, &SPA_FRACTION(24,1), sizeof(struct spa_fraction)) == 0);
break;
case 10:
spa_assert_se(prop->key == 11);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 56);
spa_assert_se(spa_pod_is_array(&prop->value));
spa_assert_se(SPA_POD_ARRAY_VALUE_TYPE(&prop->value) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_ARRAY_VALUE_SIZE(&prop->value) == sizeof(int64_t));
spa_assert_se(SPA_POD_ARRAY_N_VALUES(&prop->value) == SPA_N_ELEMENTS(longs));
al = SPA_POD_ARRAY_VALUES(&prop->value);
spa_assert_se(al != NULL);
spa_assert_se(SPA_POD_ARRAY_CHILD(&prop->value)->type == SPA_TYPE_Long);
spa_assert_se(SPA_POD_ARRAY_CHILD(&prop->value)->size == sizeof(int64_t));
for (j = 0; j < SPA_N_ELEMENTS(longs); j++)
spa_assert_se(al[j] == longs[j]);
break;
case 11:
spa_assert_se(prop->key == 12);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == (sizeof(struct spa_pod_prop) +
sizeof(struct spa_pod_pointer_body)));
spa_assert_se(spa_pod_get_pointer(&prop->value, &val.ptype, &val.p) == 0);
spa_assert_se(val.ptype == SPA_TYPE_Object);
spa_assert_se(val.p == &b);
break;
case 12:
spa_assert_se(prop->key == 13);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 24);
spa_assert_se(spa_pod_get_fd(&prop->value, &val.h) == 0);
spa_assert_se(val.h == 3);
break;
case 13:
spa_assert_se(prop->key == 14);
spa_assert_se(SPA_POD_PROP_SIZE(prop) == 20);
spa_assert_se(spa_pod_get_int(&prop->value, &val.i) == 0);
spa_assert_se(val.i == 77);
break;
default:
spa_assert_not_reached();
break;
}
}
spa_assert_se(spa_pod_parse_object(pod, SPA_TYPE_OBJECT_Format, NULL) == -EPROTO);
spa_assert_se(spa_pod_parse_object(pod, SPA_TYPE_OBJECT_Props, NULL) == 0);
spa_zero(val);
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Props, NULL,
1, SPA_POD_Bool(&val.b),
2, SPA_POD_Id(&val.I),
3, SPA_POD_Int(&val.i),
4, SPA_POD_Long(&val.l),
5, SPA_POD_Float(&val.f),
6, SPA_POD_Double(&val.d),
7, SPA_POD_String(&val.s),
8, SPA_POD_Bytes(&val.y, &val.yl),
9, SPA_POD_Rectangle(&val.R),
10, SPA_POD_Fraction(&val.F),
11, SPA_POD_Array(&val.asize, &val.atype, &val.anvals, &val.a),
12, SPA_POD_Pointer(&val.ptype, &val.p),
13, SPA_POD_Fd(&val.h),
14, SPA_POD_Pod(&val.P)) == 14);
spa_assert_se(val.b == true);
spa_assert_se(val.I == SPA_TYPE_Id);
spa_assert_se(val.i == 3);
spa_assert_se(val.l == 4);
spa_assert_se(val.f == 0.453f);
spa_assert_se(val.d == 0.871);
spa_assert_se(spa_streq(val.s, "test"));
spa_assert_se(val.yl == sizeof(bytes));
spa_assert_se(memcmp(val.y, bytes, sizeof(bytes)) == 0);
spa_assert_se(memcmp(&val.R, &SPA_RECTANGLE(3, 4), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&val.F, &SPA_FRACTION(24, 1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(val.asize == sizeof(int64_t));
spa_assert_se(val.atype == SPA_TYPE_Long);
spa_assert_se(val.anvals == SPA_N_ELEMENTS(longs));
spa_assert_se(memcmp(val.a, longs, val.anvals * val.asize) == 0);
spa_assert_se(val.ptype == SPA_TYPE_Object);
spa_assert_se(val.p == &b);
spa_assert_se(val.h == 3);
spa_assert_se(memcmp(val.P, &pi, sizeof(pi)) == 0);
spa_zero(val);
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Props, NULL,
0, SPA_POD_OPT_Bool(&val.b),
0, SPA_POD_OPT_Id(&val.I),
0, SPA_POD_OPT_Int(&val.i),
0, SPA_POD_OPT_Long(&val.l),
0, SPA_POD_OPT_Float(&val.f),
0, SPA_POD_OPT_Double(&val.d),
0, SPA_POD_OPT_String(&val.s),
0, SPA_POD_OPT_Bytes(&val.y, &val.yl),
0, SPA_POD_OPT_Rectangle(&val.R),
0, SPA_POD_OPT_Fraction(&val.F),
0, SPA_POD_OPT_Array(&val.asize, &val.atype, &val.anvals, &val.a),
0, SPA_POD_OPT_Pointer(&val.ptype, &val.p),
0, SPA_POD_OPT_Fd(&val.h),
0, SPA_POD_OPT_Pod(&val.P)) == 0);
for (i = 1; i < 15; i++) {
spa_zero(val);
spa_assert_se(spa_pod_parse_object(pod,
SPA_TYPE_OBJECT_Props, NULL,
i, SPA_POD_OPT_Bool(&val.b),
i, SPA_POD_OPT_Id(&val.I),
i, SPA_POD_OPT_Int(&val.i),
i, SPA_POD_OPT_Long(&val.l),
i, SPA_POD_OPT_Float(&val.f),
i, SPA_POD_OPT_Double(&val.d),
i, SPA_POD_OPT_String(&val.s),
i, SPA_POD_OPT_Bytes(&val.y, &val.yl),
i, SPA_POD_OPT_Rectangle(&val.R),
i, SPA_POD_OPT_Fraction(&val.F),
i, SPA_POD_OPT_Array(&val.asize, &val.atype, &val.anvals, &val.a),
i, SPA_POD_OPT_Pointer(&val.ptype, &val.p),
i, SPA_POD_OPT_Fd(&val.h),
i, SPA_POD_OPT_Pod(&val.P)) == 2);
}
return PWTEST_PASS;
}
PWTEST(pod_parser)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod_parser p;
struct spa_pod_frame f;
struct spa_pod *pod;
struct {
bool b;
uint32_t I;
int32_t i;
int64_t l;
float f;
double d;
const char *s;
uint32_t yl;
const void *y;
uint32_t ptype;
const void *p;
uint32_t asize, atype, anvals;
const void *a;
int64_t h;
struct spa_rectangle R;
struct spa_fraction F;
struct spa_pod *P;
} val;
uint8_t bytes[] = { 0x56, 0x00, 0x12, 0xf3, 0xba };
int64_t longs[] = { 1002, 5383, 28944, 1237748 };
struct spa_pod_int pi = SPA_POD_INIT_Int(77);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
pod = spa_pod_builder_add_object(&b,
SPA_TYPE_OBJECT_Props, 0,
1, SPA_POD_Bool(true),
2, SPA_POD_Id(SPA_TYPE_Id),
3, SPA_POD_Int(3),
4, SPA_POD_Long(4LL),
5, SPA_POD_Float(0.453f),
6, SPA_POD_Double(0.871),
7, SPA_POD_String("test"),
8, SPA_POD_Bytes(bytes, sizeof(bytes)),
9, SPA_POD_Rectangle(&SPA_RECTANGLE(3,4)),
10, SPA_POD_Fraction(&SPA_FRACTION(24,1)),
11, SPA_POD_Array(sizeof(int64_t), SPA_TYPE_Long, SPA_N_ELEMENTS(longs), longs),
12, SPA_POD_Pointer(SPA_TYPE_Object, &b),
13, SPA_POD_Fd(3),
14, SPA_POD_Pod(&pi));
spa_debug_pod(0, NULL, pod);
spa_pod_parser_pod(&p, pod);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_bool(&p, &val.b) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_id(&p, &val.I) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_int(&p, &val.i) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_long(&p, &val.l) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_float(&p, &val.f) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_double(&p, &val.d) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_string(&p, &val.s) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_bytes(&p, &val.y, &val.yl) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_rectangle(&p, &val.R) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_fraction(&p, &val.F) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_pointer(&p, &val.ptype, &val.p) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_fd(&p, &val.h) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_pod(&p, &val.P) == 0);
spa_assert_se(p.state.offset == 392);
spa_assert_se(spa_pod_is_object(val.P));
spa_pod_parser_pod(&p, val.P);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_push_struct(&p, &f) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_push_object(&p, &f, SPA_TYPE_OBJECT_Format, NULL) == -EPROTO);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_push_object(&p, &f, SPA_TYPE_OBJECT_Props, NULL) == 0);
spa_assert_se(p.state.offset == 392);
spa_assert_se(spa_pod_parser_frame(&p, &f) == val.P);
spa_zero(val);
spa_assert_se(spa_pod_parser_get(&p,
1, SPA_POD_OPT_Bool(&val.b),
2, SPA_POD_OPT_Id(&val.I),
3, SPA_POD_OPT_Int(&val.i),
4, SPA_POD_OPT_Long(&val.l),
5, SPA_POD_OPT_Float(&val.f),
6, SPA_POD_OPT_Double(&val.d),
7, SPA_POD_OPT_String(&val.s),
8, SPA_POD_OPT_Bytes(&val.y, &val.yl),
9, SPA_POD_OPT_Rectangle(&val.R),
10, SPA_POD_OPT_Fraction(&val.F),
11, SPA_POD_OPT_Array(&val.asize, &val.atype, &val.anvals, &val.a),
12, SPA_POD_OPT_Pointer(&val.ptype, &val.p),
13, SPA_POD_OPT_Fd(&val.h),
14, SPA_POD_OPT_Pod(&val.P), 0) == 14);
spa_pod_parser_pop(&p, &f);
spa_assert_se(val.b == true);
spa_assert_se(val.I == SPA_TYPE_Id);
spa_assert_se(val.i == 3);
spa_assert_se(val.l == 4);
spa_assert_se(val.f == 0.453f);
spa_assert_se(val.d == 0.871);
spa_assert_se(spa_streq(val.s, "test"));
spa_assert_se(val.yl == sizeof(bytes));
spa_assert_se(memcmp(val.y, bytes, sizeof(bytes)) == 0);
spa_assert_se(memcmp(&val.R, &SPA_RECTANGLE(3, 4), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(memcmp(&val.F, &SPA_FRACTION(24, 1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(val.asize == sizeof(int64_t));
spa_assert_se(val.atype == SPA_TYPE_Long);
spa_assert_se(val.anvals == SPA_N_ELEMENTS(longs));
spa_assert_se(memcmp(val.a, longs, val.anvals * val.asize) == 0);
spa_assert_se(val.ptype == SPA_TYPE_Object);
spa_assert_se(val.p == &b);
spa_assert_se(val.h == 3);
spa_assert_se(memcmp(val.P, &pi, sizeof(pi)) == 0);
spa_assert_se(p.state.offset == 392);
return PWTEST_PASS;
}
PWTEST(pod_parser2)
{
uint8_t buffer[4096];
struct spa_pod_builder b;
struct spa_pod_parser p;
struct spa_pod_frame f;
struct spa_pod *pod;
struct {
bool b;
uint32_t I;
int32_t i;
int64_t l;
float f;
double d;
const char *s;
uint32_t yl;
const void *y;
uint32_t ptype;
const void *p;
uint32_t asize, atype, anvals;
const void *a;
int64_t h;
struct spa_rectangle R;
struct spa_fraction F;
struct spa_pod *P;
} val;
uint8_t bytes[] = { 0x56, 0x00, 0x12, 0xf3, 0xba };
int64_t longs[] = { 1002, 5383, 28944, 1237748 };
struct spa_pod_int pi = SPA_POD_INIT_Int(77);
spa_pod_builder_init(&b, buffer, sizeof(buffer));
pod = spa_pod_builder_add_struct(&b,
SPA_POD_Bool(true),
SPA_POD_Id(SPA_TYPE_Id),
SPA_POD_Int(3),
SPA_POD_Long(4LL),
SPA_POD_Float(0.453f),
SPA_POD_Double(0.871),
SPA_POD_String("test"),
SPA_POD_Bytes(bytes, sizeof(bytes)),
SPA_POD_Rectangle(&SPA_RECTANGLE(3,4)),
SPA_POD_Fraction(&SPA_FRACTION(24,1)),
SPA_POD_Array(sizeof(int64_t), SPA_TYPE_Long, SPA_N_ELEMENTS(longs), longs),
SPA_POD_Pointer(SPA_TYPE_Object, &b),
SPA_POD_Fd(3),
SPA_POD_Pod(&pi));
spa_debug_pod(0, NULL, pod);
spa_pod_parser_pod(&p, pod);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_bool(&p, &val.b) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_id(&p, &val.I) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_int(&p, &val.i) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_long(&p, &val.l) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_float(&p, &val.f) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_double(&p, &val.d) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_string(&p, &val.s) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_bytes(&p, &val.y, &val.yl) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_rectangle(&p, &val.R) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_fraction(&p, &val.F) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_pointer(&p, &val.ptype, &val.p) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_fd(&p, &val.h) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_get_pod(&p, &val.P) == 0);
spa_assert_se(p.state.offset == 272);
spa_assert_se(spa_pod_is_struct(val.P));
spa_pod_parser_pod(&p, val.P);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_push_object(&p, &f, SPA_TYPE_OBJECT_Format, NULL) == -EINVAL);
spa_assert_se(p.state.offset == 0);
spa_assert_se(spa_pod_parser_push_struct(&p, &f) == 0);
spa_assert_se(f.pod.type == SPA_TYPE_Struct);
spa_assert_se(f.pod.size == 264);
spa_assert_se(f.offset == 0);
spa_assert_se(p.state.frame == &f);
spa_assert_se(spa_pod_parser_frame(&p, &f) == val.P);
spa_assert_se(p.state.offset == 8);
spa_assert_se(spa_pod_parser_get_bool(&p, &val.b) == 0 && val.b == true);
spa_assert_se(p.state.offset == 24);
spa_assert_se(spa_pod_parser_get_id(&p, &val.I) == 0 && val.I == SPA_TYPE_Id);
spa_assert_se(p.state.offset == 40);
spa_assert_se(spa_pod_parser_get_int(&p, &val.i) == 0 && val.i == 3);
spa_assert_se(p.state.offset == 56);
spa_assert_se(spa_pod_parser_get_long(&p, &val.l) == 0 && val.l == 4);
spa_assert_se(p.state.offset == 72);
spa_assert_se(spa_pod_parser_get_float(&p, &val.f) == 0 && val.f == 0.453f);
spa_assert_se(p.state.offset == 88);
spa_assert_se(spa_pod_parser_get_double(&p, &val.d) == 0 && val.d == 0.871);
spa_assert_se(p.state.offset == 104);
spa_assert_se(spa_pod_parser_get_string(&p, &val.s) == 0 && spa_streq(val.s, "test"));
spa_assert_se(p.state.offset == 120);
spa_assert_se(spa_pod_parser_get_bytes(&p, &val.y, &val.yl) == 0);
spa_assert_se(val.yl == sizeof(bytes));
spa_assert_se(memcmp(bytes, val.y, sizeof(bytes)) == 0);
spa_assert_se(p.state.offset == 136);
spa_assert_se(spa_pod_parser_get_rectangle(&p, &val.R) == 0);
spa_assert_se(memcmp(&val.R, &SPA_RECTANGLE(3,4), sizeof(struct spa_rectangle)) == 0);
spa_assert_se(p.state.offset == 152);
spa_assert_se(spa_pod_parser_get_fraction(&p, &val.F) == 0);
spa_assert_se(memcmp(&val.F, &SPA_FRACTION(24,1), sizeof(struct spa_fraction)) == 0);
spa_assert_se(p.state.offset == 168);
val.P = spa_pod_parser_next(&p);
spa_assert_se(val.P != NULL);
spa_assert_se(spa_pod_is_array(val.P));
spa_assert_se(p.state.offset == 216);
spa_assert_se(SPA_POD_ARRAY_VALUE_TYPE(val.P) == SPA_TYPE_Long);
spa_assert_se(SPA_POD_ARRAY_VALUE_SIZE(val.P) == sizeof(int64_t));
spa_assert_se(SPA_POD_ARRAY_N_VALUES(val.P) == SPA_N_ELEMENTS(longs));
spa_assert_se(spa_pod_parser_get_pointer(&p, &val.ptype, &val.p) == 0);
spa_assert_se(val.ptype == SPA_TYPE_Object);
spa_assert_se(val.p == &b);
spa_assert_se(p.state.offset == 240);
spa_assert_se(spa_pod_parser_get_fd(&p, &val.h) == 0);
spa_assert_se(val.h == 3);
spa_assert_se(p.state.offset == 256);
spa_assert_se(spa_pod_parser_get_pod(&p, &val.P) == 0);
spa_assert_se(p.state.offset == 272);
spa_assert_se(spa_pod_is_int(val.P));
spa_pod_parser_pop(&p, &f);
spa_assert_se(p.state.offset == 272);
spa_assert_se(p.state.frame == NULL);
return PWTEST_PASS;
}
PWTEST(pod_static)
{
struct _test_format {
struct spa_pod_object fmt;
struct {
struct spa_pod_prop prop_media_type SPA_ALIGNED(8);
uint32_t media_type;
struct spa_pod_prop prop_media_subtype SPA_ALIGNED(8);
uint32_t media_subtype;
struct spa_pod_prop prop_format SPA_ALIGNED(8);
struct {
struct spa_pod_choice_body choice;
uint32_t def_format;
uint32_t enum_format[2];
} format_vals;
struct spa_pod_prop prop_size SPA_ALIGNED(8);
struct {
struct spa_pod_choice_body choice;
struct spa_rectangle def_size;
struct spa_rectangle min_size;
struct spa_rectangle max_size;
} size_vals;
struct spa_pod_prop prop_framerate SPA_ALIGNED(8);
struct {
struct spa_pod_choice_body choice;
struct spa_fraction def_framerate;
struct spa_fraction min_framerate;
struct spa_fraction max_framerate;
} framerate_vals;
} props;
} test_format = {
SPA_POD_INIT_Object(sizeof(test_format.props) + sizeof(struct spa_pod_object_body),
SPA_TYPE_OBJECT_Format, 0),
{
SPA_POD_INIT_Prop(SPA_FORMAT_mediaType, 0,
sizeof(test_format.props.media_type), SPA_TYPE_Id),
SPA_MEDIA_TYPE_video,
SPA_POD_INIT_Prop(SPA_FORMAT_mediaSubtype, 0,
sizeof(test_format.props.media_subtype), SPA_TYPE_Id),
SPA_MEDIA_SUBTYPE_raw,
SPA_POD_INIT_Prop(SPA_FORMAT_VIDEO_format, 0,
sizeof(test_format.props.format_vals), SPA_TYPE_Choice),
{
SPA_POD_INIT_CHOICE_BODY(SPA_CHOICE_Enum, 0,
sizeof(uint32_t), SPA_TYPE_Id),
SPA_VIDEO_FORMAT_I420,
{ SPA_VIDEO_FORMAT_I420, SPA_VIDEO_FORMAT_YUY2 }
},
SPA_POD_INIT_Prop(SPA_FORMAT_VIDEO_size, 0,
sizeof(test_format.props.size_vals), SPA_TYPE_Choice),
{
SPA_POD_INIT_CHOICE_BODY(SPA_CHOICE_Range, 0,
sizeof(struct spa_rectangle), SPA_TYPE_Rectangle),
SPA_RECTANGLE(320,243),
SPA_RECTANGLE(1,1), SPA_RECTANGLE(INT32_MAX, INT32_MAX)
},
SPA_POD_INIT_Prop(SPA_FORMAT_VIDEO_framerate, 0,
sizeof(test_format.props.framerate_vals), SPA_TYPE_Choice),
{
SPA_POD_INIT_CHOICE_BODY(SPA_CHOICE_Range, 0,
sizeof(struct spa_fraction), SPA_TYPE_Fraction),
SPA_FRACTION(25,1),
SPA_FRACTION(0,1), SPA_FRACTION(INT32_MAX,1)
}
}
};
struct {
uint32_t media_type;
uint32_t media_subtype;
uint32_t format;
struct spa_rectangle size;
struct spa_fraction framerate;
} vals;
int res;
spa_debug_pod(0, NULL, &test_format.fmt.pod);
spa_zero(vals);
res = spa_pod_parse_object(&test_format.fmt.pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&vals.media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&vals.media_subtype),
SPA_FORMAT_VIDEO_format, SPA_POD_Id(&vals.format),
SPA_FORMAT_VIDEO_size, SPA_POD_Rectangle(&vals.size),
SPA_FORMAT_VIDEO_framerate, SPA_POD_Fraction(&vals.framerate));
spa_assert_se(res == -EPROTO);
spa_assert_se(vals.media_type == SPA_MEDIA_TYPE_video);
spa_assert_se(vals.media_subtype == SPA_MEDIA_SUBTYPE_raw);
spa_assert_se(vals.format == 0);
spa_assert_se(vals.size.width == 0 && vals.size.height == 0);
spa_assert_se(vals.framerate.num == 0 && vals.framerate.denom == 0);
spa_pod_fixate(&test_format.fmt.pod);
spa_zero(vals);
res = spa_pod_parse_object(&test_format.fmt.pod,
SPA_TYPE_OBJECT_Format, NULL,
SPA_FORMAT_mediaType, SPA_POD_Id(&vals.media_type),
SPA_FORMAT_mediaSubtype, SPA_POD_Id(&vals.media_subtype),
SPA_FORMAT_VIDEO_format, SPA_POD_Id(&vals.format),
SPA_FORMAT_VIDEO_size, SPA_POD_Rectangle(&vals.size),
SPA_FORMAT_VIDEO_framerate, SPA_POD_Fraction(&vals.framerate));
spa_assert_se(res == 5);
spa_assert_se(vals.media_type == SPA_MEDIA_TYPE_video);
spa_assert_se(vals.media_subtype == SPA_MEDIA_SUBTYPE_raw);
spa_assert_se(vals.format == SPA_VIDEO_FORMAT_I420);
spa_assert_se(vals.size.width == 320 && vals.size.height == 243);
spa_assert_se(vals.framerate.num == 25 && vals.framerate.denom == 1);
return PWTEST_PASS;
}
PWTEST(pod_overflow)
{
static const char * const labels[] = {
"640x480p59", "720x480i29", "720x480p59", "720x576i25", "720x576p50",
"1280x720p24", "1280x720p25", "1280x720p30", "1280x720p50", "1280x720p60",
"1920x1080p24", "1920x1080p25", "1920x1080p30", "1920x1080i25", "1920x1080p50",
"1920x1080i30", "1920x1080p60", "640x350p85", "640x400p85", "720x400p85",
"640x480p72", "640x480p75", "640x480p85", "800x600p56", "800x600p60",
"800x600p72", "800x600p75", "800x600p85", "800x600p119", "848x480p60",
"1024x768i43", "1024x768p60", "1024x768p70", "1024x768p75", "1024x768p84",
"1024x768p119", "1152x864p75", "1280x768p59", "1280x768p59", "1280x768p74",
"1280x768p84", "1280x768p119", "1280x800p59", "1280x800p59", "1280x800p74",
"1280x800p84", "1280x800p119", "1280x960p60", "1280x960p85", "1280x960p119",
"1280x1024p60", "1280x1024p75", "1280x1024p85", "1280x1024p119", "1360x768p60",
"1360x768p119", "1366x768p59", "1366x768p60", "1400x1050p59", "1400x1050p59",
"1400x1050p74", "1400x1050p84", "1400x1050p119", "1440x900p59", "1440x900p59",
"1440x900p74", "1440x900p84", "1440x900p119", "1600x900p60", "1600x1200p60",
"1600x1200p65", "1600x1200p70", "1600x1200p75", "1600x1200p85", "1600x1200p119",
"1680x1050p59", "1680x1050p59", "1680x1050p74", "1680x1050p84", "1680x1050p119",
"1792x1344p59", "1792x1344p74", "1792x1344p119", "1856x1392p59", "1856x1392p75",
"1856x1392p119", "1920x1200p59", "1920x1200p59", "1920x1200p74", "1920x1200p84",
"1920x1200p119", "1920x1440p60", "1920x1440p75", "1920x1440p119", "2048x1152p60",
"2560x1600p59", "2560x1600p59", "2560x1600p74", "2560x1600p84", "2560x1600p119",
"3840x2160p24", "3840x2160p25", "3840x2160p30", "3840x2160p50", "3840x2160p60",
"4096x2160p24", "4096x2160p25", "4096x2160p30", "4096x2160p50", "4096x2160p59",
"4096x2160p60", NULL };
uint8_t buffer[1024];
struct spa_pod_builder b = { 0 };
struct spa_pod_builder_state state;
struct spa_pod_frame f[2];
uint32_t idx;
struct spa_pod *pod;
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_pod_builder_push_object(&b, &f[0], SPA_TYPE_OBJECT_PropInfo, SPA_PARAM_PropInfo);
spa_pod_builder_add(&b,
SPA_PROP_INFO_id, SPA_POD_Id(32567359),
SPA_PROP_INFO_type, SPA_POD_CHOICE_ENUM_Int(1, 0),
SPA_PROP_INFO_description, SPA_POD_String("DV Timings"),
0);
spa_pod_builder_get_state(&b, &state),
spa_pod_builder_prop(&b, SPA_PROP_INFO_labels, 0);
spa_pod_builder_push_struct(&b, &f[1]);
for (idx = 0; labels[idx]; idx++) {
spa_pod_builder_int(&b, idx);
spa_pod_builder_string(&b, labels[idx]);
}
spa_assert_se(b.state.offset > sizeof(buffer));
pod = spa_pod_builder_pop(&b, &f[1]);
spa_assert_se(pod == NULL);
spa_pod_builder_reset(&b, &state);
spa_pod_builder_prop(&b, SPA_PROP_INFO_labels, 0);
spa_pod_builder_push_struct(&b, &f[1]);
pod = spa_pod_builder_pop(&b, &f[1]);
spa_assert_se(b.state.offset < sizeof(buffer));
pod = spa_pod_builder_pop(&b, &f[0]);
spa_assert_se(pod != NULL);
spa_debug_pod(0, NULL, pod);
return PWTEST_PASS;
}
static int handle_overflow(void *data, uint32_t size)
{
uint32_t *d = data;
(*d)++;
return -ENOSPC;
}
static struct spa_pod_builder_callbacks overflow_cb = {
SPA_VERSION_POD_BUILDER_CALLBACKS,
.overflow = handle_overflow
};
PWTEST(pod_overflow2)
{
uint8_t buffer[1024];
struct spa_pod_builder b = { 0 };
struct spa_pod_builder_state state;
struct spa_pod_frame f[2];
uint32_t idx, overflow_count = 0;
struct spa_pod *pod;
spa_pod_builder_init(&b, buffer, sizeof(buffer));
spa_pod_builder_set_callbacks(&b, &overflow_cb, &overflow_count);
spa_pod_builder_push_object(&b, &f[0], SPA_TYPE_OBJECT_PropInfo, SPA_PARAM_PropInfo);
spa_pod_builder_add(&b,
SPA_PROP_INFO_id, SPA_POD_Id(32567359),
SPA_PROP_INFO_type, SPA_POD_CHOICE_ENUM_Int(1, 0),
SPA_PROP_INFO_description, SPA_POD_String("DV Timings"),
0);
spa_pod_builder_get_state(&b, &state),
spa_pod_builder_prop(&b, SPA_PROP_INFO_labels, 0);
spa_pod_builder_push_struct(&b, &f[1]);
for (idx = 0; idx < 512; idx++) {
spa_pod_builder_int(&b, idx);
spa_pod_builder_string(&b, "foo");
}
spa_assert_se(b.state.offset > sizeof(buffer));
pod = spa_pod_builder_pop(&b, &f[1]);
spa_assert_se(pod == NULL);
spa_assert_se(overflow_count == 1);
return PWTEST_PASS;
}
PWTEST_SUITE(spa_pod)
{
pwtest_add(pod_abi_sizes, PWTEST_NOARG);
pwtest_add(pod_abi, PWTEST_NOARG);
pwtest_add(pod_init, PWTEST_NOARG);
pwtest_add(pod_empty, PWTEST_NOARG);
pwtest_add(pod_build, PWTEST_NOARG);
pwtest_add(pod_varargs, PWTEST_NOARG);
pwtest_add(pod_varargs2, PWTEST_NOARG);
pwtest_add(pod_parser, PWTEST_NOARG);
pwtest_add(pod_parser2, PWTEST_NOARG);
pwtest_add(pod_static, PWTEST_NOARG);
pwtest_add(pod_overflow, PWTEST_NOARG);
pwtest_add(pod_overflow2, PWTEST_NOARG);
return PWTEST_PASS;
}