2016-09-14 07:16:04 +02:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Wireless daemon for Linux
|
|
|
|
*
|
2019-10-25 00:43:08 +02:00
|
|
|
* Copyright (C) 2016-2019 Intel Corporation. All rights reserved.
|
2016-09-14 07:16:04 +02:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
typedef void (*watchlist_item_destroy_func_t)(void *data);
|
|
|
|
|
|
|
|
struct watchlist_item {
|
|
|
|
unsigned int id;
|
|
|
|
void *notify;
|
|
|
|
void *notify_data;
|
|
|
|
watchlist_item_destroy_func_t destroy;
|
|
|
|
};
|
|
|
|
|
2017-09-01 00:57:16 +02:00
|
|
|
struct watchlist_ops {
|
|
|
|
void (*item_free)(struct watchlist_item *item);
|
|
|
|
};
|
|
|
|
|
2016-09-14 07:16:04 +02:00
|
|
|
struct watchlist {
|
|
|
|
int next_id;
|
|
|
|
struct l_queue *items;
|
2016-09-22 22:28:29 +02:00
|
|
|
bool in_notify : 1;
|
|
|
|
bool stale_items : 1;
|
2020-03-18 15:45:27 +01:00
|
|
|
bool pending_destroy : 1;
|
2017-09-01 00:57:16 +02:00
|
|
|
const struct watchlist_ops *ops;
|
2016-09-14 07:16:04 +02:00
|
|
|
};
|
|
|
|
|
2017-09-01 00:57:16 +02:00
|
|
|
struct watchlist *watchlist_new(const struct watchlist_ops *ops);
|
|
|
|
void watchlist_init(struct watchlist *watchlist,
|
|
|
|
const struct watchlist_ops *ops);
|
2016-09-14 07:16:04 +02:00
|
|
|
unsigned int watchlist_add(struct watchlist *watchlist, void *notify,
|
|
|
|
void *notify_data,
|
|
|
|
watchlist_item_destroy_func_t destroy);
|
2017-09-01 00:57:16 +02:00
|
|
|
unsigned int watchlist_link(struct watchlist *watchlist,
|
|
|
|
struct watchlist_item *item,
|
|
|
|
void *notify, void *notify_data,
|
|
|
|
watchlist_item_destroy_func_t destroy);
|
2016-09-14 07:16:04 +02:00
|
|
|
bool watchlist_remove(struct watchlist *watchlist, unsigned int id);
|
|
|
|
void watchlist_destroy(struct watchlist *watchlist);
|
|
|
|
void watchlist_free(struct watchlist *watchlist);
|
|
|
|
|
2016-09-22 22:28:29 +02:00
|
|
|
void __watchlist_prune_stale(struct watchlist *watchlist);
|
|
|
|
|
2020-04-30 15:48:44 +02:00
|
|
|
#define WATCHLIST_NOTIFY(list, type, args...) \
|
2016-09-14 07:16:04 +02:00
|
|
|
do { \
|
2020-04-30 15:48:44 +02:00
|
|
|
struct watchlist *watchlist = (list); \
|
2016-09-14 07:16:04 +02:00
|
|
|
const struct l_queue_entry *entry = \
|
2020-04-30 15:48:44 +02:00
|
|
|
l_queue_get_entries(watchlist->items); \
|
2016-09-14 07:16:04 +02:00
|
|
|
\
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = true; \
|
2016-09-14 07:16:04 +02:00
|
|
|
for (; entry; entry = entry->next) { \
|
|
|
|
struct watchlist_item *item = entry->data; \
|
|
|
|
type t = item->notify; \
|
2018-09-19 21:14:53 +02:00
|
|
|
if (item->id == 0) \
|
|
|
|
continue; \
|
2016-09-14 07:16:04 +02:00
|
|
|
t(args, item->notify_data); \
|
2020-04-30 15:48:44 +02:00
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
break; \
|
2016-09-14 07:16:04 +02:00
|
|
|
} \
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = false; \
|
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
watchlist_destroy(watchlist); \
|
2020-04-30 15:48:44 +02:00
|
|
|
else if (watchlist->stale_items) \
|
2016-09-22 22:28:29 +02:00
|
|
|
__watchlist_prune_stale(watchlist); \
|
2016-09-14 07:16:04 +02:00
|
|
|
} while (false) \
|
|
|
|
|
2020-04-30 15:48:44 +02:00
|
|
|
#define WATCHLIST_NOTIFY_MATCHES(list, match, match_data, type, args...) \
|
2017-09-01 00:58:36 +02:00
|
|
|
do { \
|
2020-04-30 15:48:44 +02:00
|
|
|
struct watchlist *watchlist = (list); \
|
2017-09-01 00:58:36 +02:00
|
|
|
const struct l_queue_entry *entry = \
|
2020-04-30 15:48:44 +02:00
|
|
|
l_queue_get_entries(watchlist->items); \
|
2017-09-01 00:58:36 +02:00
|
|
|
\
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = true; \
|
2017-09-01 00:58:36 +02:00
|
|
|
for (; entry; entry = entry->next) { \
|
|
|
|
struct watchlist_item *item = entry->data; \
|
|
|
|
type t = item->notify; \
|
|
|
|
\
|
2018-09-19 21:14:53 +02:00
|
|
|
if (item->id == 0) \
|
|
|
|
continue; \
|
2017-09-07 23:04:43 +02:00
|
|
|
if (!match(item, match_data)) \
|
2017-09-01 00:58:36 +02:00
|
|
|
continue; \
|
|
|
|
\
|
|
|
|
t(args, item->notify_data); \
|
2020-03-18 15:45:27 +01:00
|
|
|
\
|
2020-04-30 15:48:44 +02:00
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
break; \
|
2017-09-01 00:58:36 +02:00
|
|
|
} \
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = false; \
|
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
watchlist_destroy(watchlist); \
|
2020-04-30 15:48:44 +02:00
|
|
|
else if (watchlist->stale_items) \
|
2017-09-01 00:58:36 +02:00
|
|
|
__watchlist_prune_stale(watchlist); \
|
|
|
|
} while (false)
|
2017-11-14 18:25:32 +01:00
|
|
|
|
2020-04-30 15:48:44 +02:00
|
|
|
#define WATCHLIST_NOTIFY_NO_ARGS(list, type) \
|
2017-11-14 18:25:32 +01:00
|
|
|
do { \
|
2020-04-30 15:48:44 +02:00
|
|
|
struct watchlist *watchlist = (list); \
|
2017-11-14 18:25:32 +01:00
|
|
|
const struct l_queue_entry *entry = \
|
2020-04-30 15:48:44 +02:00
|
|
|
l_queue_get_entries(watchlist->items); \
|
2017-11-14 18:25:32 +01:00
|
|
|
\
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = true; \
|
2017-11-14 18:25:32 +01:00
|
|
|
for (; entry; entry = entry->next) { \
|
|
|
|
struct watchlist_item *item = entry->data; \
|
|
|
|
type t = item->notify; \
|
2018-09-19 21:14:53 +02:00
|
|
|
if (item->id == 0) \
|
|
|
|
continue; \
|
2020-03-18 15:45:27 +01:00
|
|
|
t(item->notify_data); \
|
2020-04-30 15:48:44 +02:00
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
break; \
|
2017-11-14 18:25:32 +01:00
|
|
|
} \
|
2020-04-30 15:48:44 +02:00
|
|
|
watchlist->in_notify = false; \
|
|
|
|
if (watchlist->pending_destroy) \
|
2020-03-18 15:45:27 +01:00
|
|
|
watchlist_destroy(watchlist); \
|
2020-04-30 15:48:44 +02:00
|
|
|
else if (watchlist->stale_items) \
|
2017-11-14 18:25:32 +01:00
|
|
|
__watchlist_prune_stale(watchlist); \
|
|
|
|
} while (false)
|