2022-02-08 11:08:16 +01:00
|
|
|
/* PipeWire
|
|
|
|
|
*
|
|
|
|
|
* Copyright © 2022 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 <stdio.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <sys/eventfd.h>
|
|
|
|
|
|
|
|
|
|
#include "pwtest.h"
|
|
|
|
|
|
|
|
|
|
#include <pipewire/pipewire.h>
|
|
|
|
|
|
|
|
|
|
struct obj {
|
|
|
|
|
int x;
|
|
|
|
|
struct spa_source source;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct data {
|
|
|
|
|
struct pw_main_loop *ml;
|
|
|
|
|
struct pw_loop *l;
|
|
|
|
|
struct obj *a, *b;
|
2022-02-08 17:21:10 +01:00
|
|
|
int count;
|
2022-02-08 11:08:16 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void on_event(struct spa_source *source)
|
|
|
|
|
{
|
|
|
|
|
struct data *d = source->data;
|
|
|
|
|
|
|
|
|
|
pw_loop_remove_source(d->l, &d->a->source);
|
|
|
|
|
pw_loop_remove_source(d->l, &d->b->source);
|
|
|
|
|
close(d->a->source.fd);
|
|
|
|
|
close(d->b->source.fd);
|
|
|
|
|
free(d->a);
|
|
|
|
|
free(d->b);
|
|
|
|
|
|
|
|
|
|
pw_main_loop_quit(d->ml);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PWTEST(pwtest_loop_destroy2)
|
|
|
|
|
{
|
|
|
|
|
struct data data;
|
|
|
|
|
|
|
|
|
|
pw_init(0, NULL);
|
|
|
|
|
|
2022-02-08 17:21:10 +01:00
|
|
|
spa_zero(data);
|
2022-02-08 11:08:16 +01:00
|
|
|
data.ml = pw_main_loop_new(NULL);
|
|
|
|
|
pwtest_ptr_notnull(data.ml);
|
|
|
|
|
|
|
|
|
|
data.l = pw_main_loop_get_loop(data.ml);
|
|
|
|
|
pwtest_ptr_notnull(data.l);
|
|
|
|
|
|
|
|
|
|
data.a = calloc(1, sizeof(*data.a));
|
|
|
|
|
data.b = calloc(1, sizeof(*data.b));
|
|
|
|
|
|
|
|
|
|
data.a->source.func = on_event;
|
|
|
|
|
data.a->source.fd = eventfd(0, 0);
|
|
|
|
|
data.a->source.mask = SPA_IO_IN;
|
|
|
|
|
data.a->source.data = &data;
|
|
|
|
|
data.b->source.func = on_event;
|
|
|
|
|
data.b->source.fd = eventfd(0, 0);
|
|
|
|
|
data.b->source.mask = SPA_IO_IN;
|
|
|
|
|
data.b->source.data = &data;
|
|
|
|
|
|
|
|
|
|
pw_loop_add_source(data.l, &data.a->source);
|
|
|
|
|
pw_loop_add_source(data.l, &data.b->source);
|
|
|
|
|
|
|
|
|
|
write(data.a->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
write(data.b->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
|
|
|
|
|
pw_main_loop_run(data.ml);
|
|
|
|
|
pw_main_loop_destroy(data.ml);
|
|
|
|
|
|
|
|
|
|
pw_deinit();
|
|
|
|
|
|
|
|
|
|
return PWTEST_PASS;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-08 17:21:10 +01:00
|
|
|
static void
|
|
|
|
|
on_event_recurse1(struct spa_source *source)
|
|
|
|
|
{
|
|
|
|
|
static bool first = true;
|
|
|
|
|
struct data *d = source->data;
|
|
|
|
|
uint64_t val;
|
|
|
|
|
|
|
|
|
|
++d->count;
|
|
|
|
|
pwtest_int_lt(d->count, 3);
|
|
|
|
|
|
|
|
|
|
read(source->fd, &val, sizeof(val));
|
|
|
|
|
|
|
|
|
|
if (first) {
|
|
|
|
|
first = false;
|
|
|
|
|
pw_loop_enter(d->l);
|
|
|
|
|
pw_loop_iterate(d->l, -1);
|
|
|
|
|
pw_loop_leave(d->l);
|
|
|
|
|
}
|
|
|
|
|
pw_main_loop_quit(d->ml);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PWTEST(pwtest_loop_recurse1)
|
|
|
|
|
{
|
|
|
|
|
struct data data;
|
|
|
|
|
|
|
|
|
|
pw_init(0, NULL);
|
|
|
|
|
|
|
|
|
|
spa_zero(data);
|
|
|
|
|
data.ml = pw_main_loop_new(NULL);
|
|
|
|
|
pwtest_ptr_notnull(data.ml);
|
|
|
|
|
|
|
|
|
|
data.l = pw_main_loop_get_loop(data.ml);
|
|
|
|
|
pwtest_ptr_notnull(data.l);
|
|
|
|
|
|
|
|
|
|
data.a = calloc(1, sizeof(*data.a));
|
|
|
|
|
data.b = calloc(1, sizeof(*data.b));
|
|
|
|
|
|
|
|
|
|
data.a->source.func = on_event_recurse1;
|
|
|
|
|
data.a->source.fd = eventfd(0, 0);
|
|
|
|
|
data.a->source.mask = SPA_IO_IN;
|
|
|
|
|
data.a->source.data = &data;
|
|
|
|
|
data.b->source.func = on_event_recurse1;
|
|
|
|
|
data.b->source.fd = eventfd(0, 0);
|
|
|
|
|
data.b->source.mask = SPA_IO_IN;
|
|
|
|
|
data.b->source.data = &data;
|
|
|
|
|
|
|
|
|
|
pw_loop_add_source(data.l, &data.a->source);
|
|
|
|
|
pw_loop_add_source(data.l, &data.b->source);
|
|
|
|
|
|
|
|
|
|
write(data.a->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
write(data.b->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
|
|
|
|
|
pw_main_loop_run(data.ml);
|
|
|
|
|
pw_main_loop_destroy(data.ml);
|
|
|
|
|
|
|
|
|
|
pw_deinit();
|
|
|
|
|
|
2022-02-08 19:04:08 +01:00
|
|
|
free(data.a);
|
|
|
|
|
free(data.b);
|
|
|
|
|
|
2022-02-08 17:21:10 +01:00
|
|
|
return PWTEST_PASS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
on_event_recurse2(struct spa_source *source)
|
|
|
|
|
{
|
|
|
|
|
static bool first = true;
|
|
|
|
|
struct data *d = source->data;
|
|
|
|
|
uint64_t val;
|
|
|
|
|
|
|
|
|
|
++d->count;
|
|
|
|
|
pwtest_int_lt(d->count, 3);
|
|
|
|
|
|
|
|
|
|
read(source->fd, &val, sizeof(val));
|
|
|
|
|
|
|
|
|
|
if (first) {
|
|
|
|
|
first = false;
|
|
|
|
|
pw_loop_enter(d->l);
|
|
|
|
|
pw_loop_iterate(d->l, -1);
|
|
|
|
|
pw_loop_leave(d->l);
|
|
|
|
|
} else {
|
|
|
|
|
pw_loop_remove_source(d->l, &d->a->source);
|
|
|
|
|
pw_loop_remove_source(d->l, &d->b->source);
|
|
|
|
|
close(d->a->source.fd);
|
|
|
|
|
close(d->b->source.fd);
|
|
|
|
|
free(d->a);
|
|
|
|
|
free(d->b);
|
|
|
|
|
}
|
|
|
|
|
pw_main_loop_quit(d->ml);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PWTEST(pwtest_loop_recurse2)
|
|
|
|
|
{
|
|
|
|
|
struct data data;
|
|
|
|
|
|
|
|
|
|
pw_init(0, NULL);
|
|
|
|
|
|
|
|
|
|
spa_zero(data);
|
|
|
|
|
data.ml = pw_main_loop_new(NULL);
|
|
|
|
|
pwtest_ptr_notnull(data.ml);
|
|
|
|
|
|
|
|
|
|
data.l = pw_main_loop_get_loop(data.ml);
|
|
|
|
|
pwtest_ptr_notnull(data.l);
|
|
|
|
|
|
|
|
|
|
data.a = calloc(1, sizeof(*data.a));
|
|
|
|
|
data.b = calloc(1, sizeof(*data.b));
|
|
|
|
|
|
|
|
|
|
data.a->source.func = on_event_recurse2;
|
|
|
|
|
data.a->source.fd = eventfd(0, 0);
|
|
|
|
|
data.a->source.mask = SPA_IO_IN;
|
|
|
|
|
data.a->source.data = &data;
|
|
|
|
|
data.b->source.func = on_event_recurse2;
|
|
|
|
|
data.b->source.fd = eventfd(0, 0);
|
|
|
|
|
data.b->source.mask = SPA_IO_IN;
|
|
|
|
|
data.b->source.data = &data;
|
|
|
|
|
|
|
|
|
|
pw_loop_add_source(data.l, &data.a->source);
|
|
|
|
|
pw_loop_add_source(data.l, &data.b->source);
|
|
|
|
|
|
|
|
|
|
write(data.a->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
write(data.b->source.fd, &(uint64_t){1}, sizeof(uint64_t));
|
|
|
|
|
|
|
|
|
|
pw_main_loop_run(data.ml);
|
|
|
|
|
pw_main_loop_destroy(data.ml);
|
|
|
|
|
|
|
|
|
|
pw_deinit();
|
|
|
|
|
|
|
|
|
|
return PWTEST_PASS;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-23 18:03:46 +01:00
|
|
|
static void
|
|
|
|
|
on_event_fail_if_called(void *data, int fd, uint32_t mask)
|
|
|
|
|
{
|
|
|
|
|
pwtest_fail_if_reached();
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
struct dmsbd_data {
|
|
|
|
|
struct pw_loop *l;
|
|
|
|
|
struct pw_main_loop *ml;
|
|
|
|
|
struct spa_source *source;
|
|
|
|
|
struct spa_hook hook;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void dmsbd_after(void *data)
|
|
|
|
|
{
|
|
|
|
|
struct dmsbd_data *d = data;
|
|
|
|
|
|
|
|
|
|
pw_loop_destroy_source(d->l, d->source);
|
|
|
|
|
pw_main_loop_quit(d->ml);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const struct spa_loop_control_hooks dmsbd_hooks = {
|
|
|
|
|
SPA_VERSION_LOOP_CONTROL_HOOKS,
|
|
|
|
|
.after = dmsbd_after,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
PWTEST(destroy_managed_source_before_dispatch)
|
2022-02-18 19:27:13 +01:00
|
|
|
{
|
|
|
|
|
pw_init(NULL, NULL);
|
|
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
struct dmsbd_data data = {0};
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
data.ml = pw_main_loop_new(NULL);
|
|
|
|
|
pwtest_ptr_notnull(data.ml);
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
data.l = pw_main_loop_get_loop(data.ml);
|
|
|
|
|
pwtest_ptr_notnull(data.l);
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
data.source = pw_loop_add_io(data.l, eventfd(0, 0), SPA_IO_IN, true, on_event_fail_if_called, NULL);
|
|
|
|
|
pwtest_ptr_notnull(data.source);
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
pw_loop_add_hook(data.l, &data.hook, &dmsbd_hooks, &data);
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
pwtest_errno_ok(write(data.source->fd, &(uint64_t){1}, sizeof(uint64_t)));
|
2022-02-18 19:27:13 +01:00
|
|
|
|
2022-02-27 20:18:52 +01:00
|
|
|
pw_main_loop_run(data.ml);
|
|
|
|
|
pw_main_loop_destroy(data.ml);
|
2022-02-18 19:27:13 +01:00
|
|
|
|
|
|
|
|
pw_deinit();
|
|
|
|
|
|
|
|
|
|
return PWTEST_PASS;
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-08 11:08:16 +01:00
|
|
|
PWTEST_SUITE(support)
|
|
|
|
|
{
|
|
|
|
|
pwtest_add(pwtest_loop_destroy2, PWTEST_NOARG);
|
2022-02-08 17:21:10 +01:00
|
|
|
pwtest_add(pwtest_loop_recurse1, PWTEST_NOARG);
|
|
|
|
|
pwtest_add(pwtest_loop_recurse2, PWTEST_NOARG);
|
2022-02-27 20:18:52 +01:00
|
|
|
pwtest_add(destroy_managed_source_before_dispatch, PWTEST_NOARG);
|
2022-02-08 11:08:16 +01:00
|
|
|
|
|
|
|
|
return PWTEST_PASS;
|
|
|
|
|
}
|