2020-04-10 10:47:05 +08:00
|
|
|
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
|
2020-05-11 13:17:22 +02:00
|
|
|
@@ -250,6 +250,60 @@ void wakeup_source_unregister(struct wak
|
2020-04-10 10:47:05 +08:00
|
|
|
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);
|