2018-11-05 17:48:52 +01:00
|
|
|
/* Simple Plugin API
|
2017-08-08 16:56:29 +02:00
|
|
|
*
|
2018-11-05 17:48:52 +01:00
|
|
|
* Copyright © 2018 Wim Taymans
|
2017-08-08 16:56:29 +02:00
|
|
|
*
|
2018-11-05 17:48:52 +01:00
|
|
|
* 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:
|
2017-08-08 16:56:29 +02:00
|
|
|
*
|
2018-11-05 17:48:52 +01:00
|
|
|
* 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.
|
2017-08-08 16:56:29 +02:00
|
|
|
*/
|
|
|
|
|
|
2019-01-14 12:58:23 +01:00
|
|
|
#ifndef SPA_HOOK_H
|
|
|
|
|
#define SPA_HOOK_H
|
2017-08-08 16:56:29 +02:00
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
|
extern "C" {
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-10-23 13:13:01 +02:00
|
|
|
#include <spa/utils/defs.h>
|
2017-11-10 13:36:14 +01:00
|
|
|
#include <spa/utils/list.h>
|
2017-08-08 16:56:29 +02:00
|
|
|
|
2021-05-21 14:03:07 +10:00
|
|
|
/** \defgroup spa_hook SPA Hooks
|
2017-08-08 18:22:44 +02:00
|
|
|
*
|
|
|
|
|
* \brief a list of hooks
|
|
|
|
|
*
|
|
|
|
|
* The hook list provides a way to keep track of hooks.
|
|
|
|
|
*/
|
2021-05-21 14:03:07 +10:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* \addtogroup spa_hook
|
|
|
|
|
* \{
|
|
|
|
|
*/
|
|
|
|
|
|
2017-08-08 18:22:44 +02:00
|
|
|
/** A list of hooks */
|
2017-08-08 16:56:29 +02:00
|
|
|
struct spa_hook_list {
|
|
|
|
|
struct spa_list list;
|
|
|
|
|
};
|
|
|
|
|
|
2019-05-15 12:17:52 +02:00
|
|
|
/** Callbacks, contains the structure with functions and the data passed
|
|
|
|
|
* to the functions. The structure should also contain a version field that
|
|
|
|
|
* is checked. */
|
|
|
|
|
struct spa_callbacks {
|
|
|
|
|
const void *funcs;
|
|
|
|
|
void *data;
|
|
|
|
|
};
|
|
|
|
|
|
2021-05-21 10:35:43 +10:00
|
|
|
/** Check if a callback \a c has method \a m of version \a v */
|
2020-03-17 11:37:56 +01:00
|
|
|
#define SPA_CALLBACK_CHECK(c,m,v) ((c) && ((v) == 0 || (c)->version > (v)-1) && (c)->m)
|
|
|
|
|
|
2019-05-15 12:17:52 +02:00
|
|
|
#define SPA_CALLBACKS_INIT(_funcs,_data) (struct spa_callbacks){ _funcs, _data, }
|
|
|
|
|
|
2019-05-20 16:11:23 +02:00
|
|
|
struct spa_interface {
|
2019-12-19 13:15:10 +01:00
|
|
|
const char *type;
|
2019-05-20 16:11:23 +02:00
|
|
|
uint32_t version;
|
|
|
|
|
struct spa_callbacks cb;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#define SPA_INTERFACE_INIT(_type,_version,_funcs,_data) \
|
|
|
|
|
(struct spa_interface){ _type, _version, SPA_CALLBACKS_INIT(_funcs,_data), }
|
|
|
|
|
|
2017-08-08 18:22:44 +02:00
|
|
|
/** A hook, contains the structure with functions and the data passed
|
|
|
|
|
* to the functions. */
|
2017-08-08 16:56:29 +02:00
|
|
|
struct spa_hook {
|
|
|
|
|
struct spa_list link;
|
2019-05-15 12:17:52 +02:00
|
|
|
struct spa_callbacks cb;
|
2020-11-20 09:40:24 +01:00
|
|
|
/** callback and data for the hook list, private to the
|
|
|
|
|
* hook_list implementor */
|
2018-08-01 21:41:25 +02:00
|
|
|
void (*removed) (struct spa_hook *hook);
|
2019-09-20 13:04:14 +02:00
|
|
|
void *priv;
|
2017-08-08 16:56:29 +02:00
|
|
|
};
|
|
|
|
|
|
2017-08-08 18:22:44 +02:00
|
|
|
/** Initialize a hook list */
|
2017-08-08 16:56:29 +02:00
|
|
|
static inline void spa_hook_list_init(struct spa_hook_list *list)
|
|
|
|
|
{
|
|
|
|
|
spa_list_init(&list->list);
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-20 13:04:14 +02:00
|
|
|
static inline bool spa_hook_list_is_empty(struct spa_hook_list *list)
|
|
|
|
|
{
|
|
|
|
|
return spa_list_is_empty(&list->list);
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-21 14:03:07 +10:00
|
|
|
/** Append a hook. */
|
2017-08-08 16:56:29 +02:00
|
|
|
static inline void spa_hook_list_append(struct spa_hook_list *list,
|
|
|
|
|
struct spa_hook *hook,
|
|
|
|
|
const void *funcs, void *data)
|
|
|
|
|
{
|
2020-11-20 09:40:24 +01:00
|
|
|
spa_zero(*hook);
|
2019-05-15 12:17:52 +02:00
|
|
|
hook->cb = SPA_CALLBACKS_INIT(funcs, data);
|
2017-08-08 16:56:29 +02:00
|
|
|
spa_list_append(&list->list, &hook->link);
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-21 14:03:07 +10:00
|
|
|
/** Prepend a hook */
|
2017-08-08 16:56:29 +02:00
|
|
|
static inline void spa_hook_list_prepend(struct spa_hook_list *list,
|
|
|
|
|
struct spa_hook *hook,
|
|
|
|
|
const void *funcs, void *data)
|
|
|
|
|
{
|
2020-11-20 09:40:24 +01:00
|
|
|
spa_zero(*hook);
|
2019-05-15 12:17:52 +02:00
|
|
|
hook->cb = SPA_CALLBACKS_INIT(funcs, data);
|
2017-08-08 16:56:29 +02:00
|
|
|
spa_list_prepend(&list->list, &hook->link);
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-21 14:03:07 +10:00
|
|
|
/** Remove a hook */
|
2017-08-08 16:56:29 +02:00
|
|
|
static inline void spa_hook_remove(struct spa_hook *hook)
|
|
|
|
|
{
|
|
|
|
|
spa_list_remove(&hook->link);
|
2018-08-01 21:41:25 +02:00
|
|
|
if (hook->removed)
|
|
|
|
|
hook->removed(hook);
|
2017-08-08 16:56:29 +02:00
|
|
|
}
|
|
|
|
|
|
2020-11-06 15:32:49 +01:00
|
|
|
static inline void spa_hook_list_clean(struct spa_hook_list *list)
|
|
|
|
|
{
|
|
|
|
|
struct spa_hook *h;
|
|
|
|
|
spa_list_consume(h, &list->list, link)
|
|
|
|
|
spa_hook_remove(h);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-01 12:00:42 +01:00
|
|
|
static inline void
|
|
|
|
|
spa_hook_list_isolate(struct spa_hook_list *list,
|
|
|
|
|
struct spa_hook_list *save,
|
|
|
|
|
struct spa_hook *hook,
|
|
|
|
|
const void *funcs, void *data)
|
|
|
|
|
{
|
|
|
|
|
/* init save list and move hooks to it */
|
|
|
|
|
spa_hook_list_init(save);
|
2019-03-08 11:34:28 +01:00
|
|
|
spa_list_insert_list(&save->list, &list->list);
|
2019-03-01 12:00:42 +01:00
|
|
|
/* init hooks and add single hook */
|
|
|
|
|
spa_hook_list_init(list);
|
|
|
|
|
spa_hook_list_append(list, hook, funcs, data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
|
spa_hook_list_join(struct spa_hook_list *list,
|
|
|
|
|
struct spa_hook_list *save)
|
|
|
|
|
{
|
2019-03-08 11:34:28 +01:00
|
|
|
spa_list_insert_list(&list->list, &save->list);
|
2019-03-01 12:00:42 +01:00
|
|
|
}
|
|
|
|
|
|
2019-05-15 12:17:52 +02:00
|
|
|
#define spa_callbacks_call(callbacks,type,method,vers,...) \
|
2018-09-24 11:35:46 +02:00
|
|
|
({ \
|
2019-05-15 12:17:52 +02:00
|
|
|
const type *_f = (const type *) (callbacks)->funcs; \
|
2020-03-17 11:37:56 +01:00
|
|
|
if (SPA_LIKELY(SPA_CALLBACK_CHECK(_f,method,vers))) \
|
2019-05-15 12:17:52 +02:00
|
|
|
_f->method((callbacks)->data, ## __VA_ARGS__); \
|
2019-05-14 18:04:34 +02:00
|
|
|
})
|
|
|
|
|
|
2019-05-15 12:17:52 +02:00
|
|
|
#define spa_callbacks_call_res(callbacks,type,res,method,vers,...) \
|
2019-05-14 18:04:34 +02:00
|
|
|
({ \
|
2019-05-15 12:17:52 +02:00
|
|
|
const type *_f = (const type *) (callbacks)->funcs; \
|
2020-03-17 11:37:56 +01:00
|
|
|
if (SPA_LIKELY(SPA_CALLBACK_CHECK(_f,method,vers))) \
|
2019-05-15 12:17:52 +02:00
|
|
|
res = _f->method((callbacks)->data, ## __VA_ARGS__); \
|
2019-05-14 18:04:34 +02:00
|
|
|
res; \
|
2018-09-24 11:35:46 +02:00
|
|
|
})
|
|
|
|
|
|
2019-05-20 16:11:23 +02:00
|
|
|
#define spa_interface_call(iface,type,method,vers,...) \
|
|
|
|
|
spa_callbacks_call(&(iface)->cb,type,method,vers,##__VA_ARGS__)
|
|
|
|
|
|
|
|
|
|
#define spa_interface_call_res(iface,type,res,method,vers,...) \
|
|
|
|
|
spa_callbacks_call_res(&(iface)->cb,type,res,method,vers,##__VA_ARGS__)
|
|
|
|
|
|
2019-03-01 12:00:42 +01:00
|
|
|
#define spa_hook_list_call_simple(l,type,method,vers,...) \
|
|
|
|
|
({ \
|
|
|
|
|
struct spa_hook_list *_l = l; \
|
|
|
|
|
struct spa_hook *_h, *_t; \
|
|
|
|
|
spa_list_for_each_safe(_h, _t, &_l->list, link) \
|
2019-05-15 12:17:52 +02:00
|
|
|
spa_callbacks_call(&_h->cb,type,method,vers, ## __VA_ARGS__); \
|
2019-03-01 12:00:42 +01:00
|
|
|
})
|
|
|
|
|
|
2017-08-08 18:22:44 +02:00
|
|
|
/** Call all hooks in a list, starting from the given one and optionally stopping
|
2018-03-14 11:17:51 +01:00
|
|
|
* after calling the first non-NULL function, returns the number of methods
|
|
|
|
|
* called */
|
2018-08-01 21:41:25 +02:00
|
|
|
#define spa_hook_list_do_call(l,start,type,method,vers,once,...) \
|
2018-05-17 17:21:30 +02:00
|
|
|
({ \
|
2017-08-08 16:56:29 +02:00
|
|
|
struct spa_hook_list *list = l; \
|
|
|
|
|
struct spa_list *s = start ? (struct spa_list *)start : &list->list; \
|
2018-08-15 11:17:12 +02:00
|
|
|
struct spa_hook cursor = { 0 }, *ci; \
|
2018-03-14 11:17:51 +01:00
|
|
|
int count = 0; \
|
2018-08-15 11:17:12 +02:00
|
|
|
spa_list_cursor_start(cursor, s, link); \
|
|
|
|
|
spa_list_for_each_cursor(ci, cursor, &list->list, link) { \
|
2019-05-15 12:17:52 +02:00
|
|
|
const type *_f = (const type *)ci->cb.funcs; \
|
2020-03-17 11:37:56 +01:00
|
|
|
if (SPA_LIKELY(SPA_CALLBACK_CHECK(_f,method,vers))) { \
|
2019-05-15 12:17:52 +02:00
|
|
|
_f->method(ci->cb.data, ## __VA_ARGS__); \
|
2018-03-14 11:17:51 +01:00
|
|
|
count++; \
|
2017-08-08 16:56:29 +02:00
|
|
|
if (once) \
|
|
|
|
|
break; \
|
|
|
|
|
} \
|
|
|
|
|
} \
|
2018-08-15 11:17:12 +02:00
|
|
|
spa_list_cursor_end(cursor, link); \
|
2018-03-14 11:17:51 +01:00
|
|
|
count; \
|
2017-09-21 18:57:41 +02:00
|
|
|
})
|
2017-08-08 16:56:29 +02:00
|
|
|
|
2018-08-01 21:41:25 +02:00
|
|
|
#define spa_hook_list_call(l,t,m,v,...) spa_hook_list_do_call(l,NULL,t,m,v,false,##__VA_ARGS__)
|
|
|
|
|
#define spa_hook_list_call_once(l,t,m,v,...) spa_hook_list_do_call(l,NULL,t,m,v,true,##__VA_ARGS__)
|
2017-08-08 16:56:29 +02:00
|
|
|
|
2018-08-01 21:41:25 +02:00
|
|
|
#define spa_hook_list_call_start(l,s,t,m,v,...) spa_hook_list_do_call(l,s,t,m,v,false,##__VA_ARGS__)
|
|
|
|
|
#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__)
|
2017-08-08 16:56:29 +02:00
|
|
|
|
2021-05-21 14:03:07 +10:00
|
|
|
/**
|
|
|
|
|
* \}
|
|
|
|
|
*/
|
|
|
|
|
|
2017-08-08 16:56:29 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-01-14 12:58:23 +01:00
|
|
|
#endif /* SPA_HOOK_H */
|