| From a9647655a2240148b7b6d29227dc76e9d6f12766 Mon Sep 17 00:00:00 2001 |
| From: Ran Wang <ran.wang_1@nxp.com> |
| Date: Thu, 24 Oct 2019 16:36:14 +0800 |
| Subject: [PATCH] PM: wakeup: Add routine to help fetch wakeup source object. |
| |
| Some user might want to go through all registered wakeup sources |
| and doing things accordingly. For example, SoC PM driver might need to |
| do HW programming to prevent powering down specific IP which wakeup |
| source depending on. So add this API to help walk through all registered |
| wakeup source objects on that list and return them one by one. |
| |
| Signed-off-by: Ran Wang <ran.wang_1@nxp.com> |
| Tested-by: Leonard Crestez <leonard.crestez@nxp.com> |
| Reviewed-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> |
| Acked-by: Anson Huang <Anson.Huang@nxp.com> |
| --- |
| drivers/base/power/wakeup.c | 54 +++++++++++++++++++++++++++++++++++++++++++++ |
| include/linux/pm_wakeup.h | 9 ++++++++ |
| 2 files changed, 63 insertions(+) |
| |
| --- a/drivers/base/power/wakeup.c |
| +++ b/drivers/base/power/wakeup.c |
| @@ -251,6 +251,60 @@ void wakeup_source_unregister(struct wak |
| EXPORT_SYMBOL_GPL(wakeup_source_unregister); |
| |
| /** |
| + * wakeup_sources_read_lock - Lock wakeup source list for read. |
| + * |
| + * Returns an index of srcu lock for struct wakeup_srcu. |
| + * This index must be passed to the matching wakeup_sources_read_unlock(). |
| + */ |
| +int wakeup_sources_read_lock(void) |
| +{ |
| + return srcu_read_lock(&wakeup_srcu); |
| +} |
| +EXPORT_SYMBOL_GPL(wakeup_sources_read_lock); |
| + |
| +/** |
| + * wakeup_sources_read_unlock - Unlock wakeup source list. |
| + * @idx: return value from corresponding wakeup_sources_read_lock() |
| + */ |
| +void wakeup_sources_read_unlock(int idx) |
| +{ |
| + srcu_read_unlock(&wakeup_srcu, idx); |
| +} |
| +EXPORT_SYMBOL_GPL(wakeup_sources_read_unlock); |
| + |
| +/** |
| + * wakeup_sources_walk_start - Begin a walk on wakeup source list |
| + * |
| + * Returns first object of the list of wakeup sources. |
| + * |
| + * Note that to be safe, wakeup sources list needs to be locked by calling |
| + * wakeup_source_read_lock() for this. |
| + */ |
| +struct wakeup_source *wakeup_sources_walk_start(void) |
| +{ |
| + struct list_head *ws_head = &wakeup_sources; |
| + |
| + return list_entry_rcu(ws_head->next, struct wakeup_source, entry); |
| +} |
| +EXPORT_SYMBOL_GPL(wakeup_sources_walk_start); |
| + |
| +/** |
| + * wakeup_sources_walk_next - Get next wakeup source from the list |
| + * @ws: Previous wakeup source object |
| + * |
| + * Note that to be safe, wakeup sources list needs to be locked by calling |
| + * wakeup_source_read_lock() for this. |
| + */ |
| +struct wakeup_source *wakeup_sources_walk_next(struct wakeup_source *ws) |
| +{ |
| + struct list_head *ws_head = &wakeup_sources; |
| + |
| + return list_next_or_null_rcu(ws_head, &ws->entry, |
| + struct wakeup_source, entry); |
| +} |
| +EXPORT_SYMBOL_GPL(wakeup_sources_walk_next); |
| + |
| +/** |
| * device_wakeup_attach - Attach a wakeup source object to a device object. |
| * @dev: Device to handle. |
| * @ws: Wakeup source object to attach to @dev. |
| --- a/include/linux/pm_wakeup.h |
| +++ b/include/linux/pm_wakeup.h |
| @@ -63,6 +63,11 @@ struct wakeup_source { |
| bool autosleep_enabled:1; |
| }; |
| |
| +#define for_each_wakeup_source(ws) \ |
| + for ((ws) = wakeup_sources_walk_start(); \ |
| + (ws); \ |
| + (ws) = wakeup_sources_walk_next((ws))) |
| + |
| #ifdef CONFIG_PM_SLEEP |
| |
| /* |
| @@ -92,6 +97,10 @@ extern void wakeup_source_remove(struct |
| extern struct wakeup_source *wakeup_source_register(struct device *dev, |
| const char *name); |
| extern void wakeup_source_unregister(struct wakeup_source *ws); |
| +extern int wakeup_sources_read_lock(void); |
| +extern void wakeup_sources_read_unlock(int idx); |
| +extern struct wakeup_source *wakeup_sources_walk_start(void); |
| +extern struct wakeup_source *wakeup_sources_walk_next(struct wakeup_source *ws); |
| extern int device_wakeup_enable(struct device *dev); |
| extern int device_wakeup_disable(struct device *dev); |
| extern void device_set_wakeup_capable(struct device *dev, bool capable); |