PipeWire 0.3.47
hook.h
Go to the documentation of this file.
1/* Simple Plugin API
2 *
3 * Copyright © 2018 Wim Taymans
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the "Software"),
7 * to deal in the Software without restriction, including without limitation
8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 * and/or sell copies of the Software, and to permit persons to whom the
10 * Software is furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice (including the next
13 * paragraph) shall be included in all copies or substantial portions of the
14 * Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
23 */
24
25#ifndef SPA_HOOK_H
26#define SPA_HOOK_H
27
28#ifdef __cplusplus
29extern "C" {
30#endif
31
32#include <spa/utils/defs.h>
33#include <spa/utils/list.h>
34
130struct spa_callbacks {
131 const void *funcs;
132 void *data;
133};
134
136#define SPA_CALLBACK_VERSION_MIN(c,v) ((c) && ((v) == 0 || (c)->version > (v)-1))
139#define SPA_CALLBACK_CHECK(c,m,v) (SPA_CALLBACK_VERSION_MIN(c,v) && (c)->m)
140
145#define SPA_CALLBACKS_INIT(_funcs,_data) (struct spa_callbacks){ _funcs, _data, }
146
149struct spa_interface {
150 const char *type;
151 uint32_t version;
152 struct spa_callbacks cb;
153};
168#define SPA_INTERFACE_INIT(_type,_version,_funcs,_data) \
169 (struct spa_interface){ _type, _version, SPA_CALLBACKS_INIT(_funcs,_data), }
170
176#define spa_callbacks_call(callbacks,type,method,vers,...) \
177({ \
178 const type *_f = (const type *) (callbacks)->funcs; \
179 bool _res = SPA_CALLBACK_CHECK(_f,method,vers); \
180 if (SPA_LIKELY(_res)) \
181 _f->method((callbacks)->data, ## __VA_ARGS__); \
182 _res; \
183})
184
188#define spa_callback_version_min(callbacks,type,vers) \
189({ \
190 const type *_f = (const type *) (callbacks)->funcs; \
191 SPA_CALLBACK_VERSION_MIN(_f,vers); \
192})
193
198#define spa_callback_check(callbacks,type,method,vers) \
199({ \
200 const type *_f = (const type *) (callbacks)->funcs; \
201 SPA_CALLBACK_CHECK(_f,method,vers); \
202})
203
210#define spa_callbacks_call_res(callbacks,type,res,method,vers,...) \
211({ \
212 const type *_f = (const type *) (callbacks)->funcs; \
213 if (SPA_LIKELY(SPA_CALLBACK_CHECK(_f,method,vers))) \
214 res = _f->method((callbacks)->data, ## __VA_ARGS__); \
215 res; \
216})
217
221#define spa_interface_callback_version_min(iface,method_type,vers) \
222 spa_callback_version_min(&(iface)->cb, method_type, vers)
223
228#define spa_interface_callback_check(iface,method_type,method,vers) \
229 spa_callback_check(&(iface)->cb, method_type, method, vers)
236#define spa_interface_call(iface,method_type,method,vers,...) \
237 spa_callbacks_call(&(iface)->cb,method_type,method,vers,##__VA_ARGS__)
238
246#define spa_interface_call_res(iface,method_type,res,method,vers,...) \
247 spa_callbacks_call_res(&(iface)->cb,method_type,res,method,vers,##__VA_ARGS__)
248
322struct spa_hook_list {
323 struct spa_list list;
324};
325
326
333struct spa_hook {
334 struct spa_list link;
335 struct spa_callbacks cb;
338 void (*removed) (struct spa_hook *hook);
339 void *priv;
340};
341
343static inline void spa_hook_list_init(struct spa_hook_list *list)
345 spa_list_init(&list->list);
346}
348static inline bool spa_hook_list_is_empty(struct spa_hook_list *list)
349{
350 return spa_list_is_empty(&list->list);
351}
354static inline void spa_hook_list_append(struct spa_hook_list *list,
355 struct spa_hook *hook,
356 const void *funcs, void *data)
358 spa_zero(*hook);
359 hook->cb = SPA_CALLBACKS_INIT(funcs, data);
360 spa_list_append(&list->list, &hook->link);
361}
362
364static inline void spa_hook_list_prepend(struct spa_hook_list *list,
365 struct spa_hook *hook,
366 const void *funcs, void *data)
367{
368 spa_zero(*hook);
369 hook->cb = SPA_CALLBACKS_INIT(funcs, data);
370 spa_list_prepend(&list->list, &hook->link);
371}
372
374static inline void spa_hook_remove(struct spa_hook *hook)
375{
376 spa_list_remove(&hook->link);
377 if (hook->removed)
378 hook->removed(hook);
379}
380
382static inline void spa_hook_list_clean(struct spa_hook_list *list)
384 struct spa_hook *h;
385 spa_list_consume(h, &list->list, link)
387}
388
389static inline void
391 struct spa_hook_list *save,
392 struct spa_hook *hook,
393 const void *funcs, void *data)
394{
395 /* init save list and move hooks to it */
396 spa_hook_list_init(save);
397 spa_list_insert_list(&save->list, &list->list);
398 /* init hooks and add single hook */
400 spa_hook_list_append(list, hook, funcs, data);
401}
402
403static inline void
405 struct spa_hook_list *save)
406{
407 spa_list_insert_list(&list->list, &save->list);
408}
409
410#define spa_hook_list_call_simple(l,type,method,vers,...) \
411({ \
412 struct spa_hook_list *_l = l; \
413 struct spa_hook *_h, *_t; \
414 spa_list_for_each_safe(_h, _t, &_l->list, link) \
415 spa_callbacks_call(&_h->cb,type,method,vers, ## __VA_ARGS__); \
416})
417
421#define spa_hook_list_do_call(l,start,type,method,vers,once,...) \
422({ \
423 struct spa_hook_list *_list = l; \
424 struct spa_list *_s = start ? (struct spa_list *)start : &_list->list; \
425 struct spa_hook _cursor = { 0 }, *_ci; \
426 int _count = 0; \
427 spa_list_cursor_start(_cursor, _s, link); \
428 spa_list_for_each_cursor(_ci, _cursor, &_list->list, link) { \
429 if (spa_callbacks_call(&_ci->cb,type,method,vers, ## __VA_ARGS__)) { \
430 _count++; \
431 if (once) \
432 break; \
433 } \
434 } \
435 spa_list_cursor_end(_cursor, link); \
436 _count; \
437})
438
443#define spa_hook_list_call(l,t,m,v,...) spa_hook_list_do_call(l,NULL,t,m,v,false,##__VA_ARGS__)
449#define spa_hook_list_call_once(l,t,m,v,...) spa_hook_list_do_call(l,NULL,t,m,v,true,##__VA_ARGS__)
450
451#define spa_hook_list_call_start(l,s,t,m,v,...) spa_hook_list_do_call(l,s,t,m,v,false,##__VA_ARGS__)
452#define spa_hook_list_call_once_start(l,s,t,m,v,...) spa_hook_list_do_call(l,s,t,m,v,true,##__VA_ARGS__)
453
458#ifdef __cplusplus
459}
460#endif
462#endif /* SPA_HOOK_H */
spa/utils/defs.h
static bool spa_hook_list_is_empty(struct spa_hook_list *list)
Definition: hook.h:357
static void spa_hook_list_append(struct spa_hook_list *list, struct spa_hook *hook, const void *funcs, void *data)
Append a hook.
Definition: hook.h:363
static void spa_hook_list_join(struct spa_hook_list *list, struct spa_hook_list *save)
Definition: hook.h:413
static void spa_hook_list_init(struct spa_hook_list *list)
Initialize a hook list to the empty list.
Definition: hook.h:352
static void spa_hook_remove(struct spa_hook *hook)
Remove a hook.
Definition: hook.h:383
static void spa_hook_list_clean(struct spa_hook_list *list)
Remove all hooks from the list.
Definition: hook.h:391
static void spa_hook_list_prepend(struct spa_hook_list *list, struct spa_hook *hook, const void *funcs, void *data)
Prepend a hook.
Definition: hook.h:373
static void spa_hook_list_isolate(struct spa_hook_list *list, struct spa_hook_list *save, struct spa_hook *hook, const void *funcs, void *data)
Definition: hook.h:399
#define SPA_CALLBACKS_INIT(_funcs, _data)
Initialize the set of functions funcs as a spa_callbacks, together with _data.
Definition: hook.h:154
#define spa_list_consume(pos, head, member)
Definition: list.h:107
static void spa_list_init(struct spa_list *list)
Definition: list.h:55
static void spa_list_remove(struct spa_list *elem)
Definition: list.h:80
#define spa_list_prepend(list, item)
Definition: list.h:95
#define spa_list_is_empty(l)
Definition: list.h:60
#define spa_list_append(list, item)
Definition: list.h:92
static void spa_list_insert_list(struct spa_list *list, struct spa_list *other)
Definition: list.h:70
#define spa_zero(x)
Definition: defs.h:385
spa/utils/list.h
Callbacks, contains the structure with functions and the data passed to the functions.
Definition: hook.h:136
const void * funcs
Definition: hook.h:137
void * data
Definition: hook.h:138
A list of hooks.
Definition: hook.h:331
struct spa_list list
Definition: hook.h:332
A hook, contains the structure with functions and the data passed to the functions.
Definition: hook.h:342
void(* removed)(struct spa_hook *hook)
callback and data for the hook list, private to the hook_list implementor
Definition: hook.h:347
struct spa_callbacks cb
Definition: hook.h:344
struct spa_list link
Definition: hook.h:343
void * priv
Definition: hook.h:348
Definition: hook.h:158
uint32_t version
Definition: hook.h:160
const char * type
Definition: hook.h:159
struct spa_callbacks cb
Definition: hook.h:161
Definition: list.h:47