@@ -23,8 +23,6 @@
#define to_smd_subdev(d) container_of(d, struct qcom_rproc_subdev, subdev)
#define to_ssr_subdev(d) container_of(d, struct qcom_rproc_ssr, subdev)
-static BLOCKING_NOTIFIER_HEAD(ssr_notifiers);
-
static int glink_subdev_start(struct rproc_subdev *subdev)
{
struct qcom_rproc_glink *glink = to_glink_subdev(subdev);
@@ -180,27 +178,52 @@ EXPORT_SYMBOL_GPL(qcom_remove_smd_subdev);
/**
* qcom_register_ssr_notifier() - register SSR notification handler
+ * @rproc: pointer to the remoteproc structure
* @nb: notifier_block to notify for restart notifications
*
- * Returns 0 on success, negative errno on failure.
+ * Returns pointer to srcu notifier head on success, ERR_PTR on failure.
*
- * This register the @notify function as handler for restart notifications. As
- * remote processors are stopped this function will be called, with the SSR
- * name passed as a parameter.
+ * This registers the @notify function as handler for restart notifications. As
+ * remote processors are stopped this function will be called, with the rproc
+ * pointer passed as a parameter.
*/
-int qcom_register_ssr_notifier(struct notifier_block *nb)
+void *qcom_register_ssr_notifier(struct rproc *rproc, struct notifier_block *nb)
{
- return blocking_notifier_chain_register(&ssr_notifiers, nb);
+ struct rproc_subdev *subdev;
+ struct qcom_rproc_ssr *ssr;
+ int ret;
+
+ if (!rproc)
+ return ERR_PTR(-EINVAL);
+
+ mutex_lock(&rproc->lock);
+ list_for_each_entry(subdev, &rproc->subdevs, node) {
+ ret = strcmp(subdev->name, "ssr_notifs");
+ if (!ret)
+ break;
+ }
+ mutex_unlock(&rproc->lock);
+ if (ret)
+ return ERR_PTR(-ENOENT);
+
+ ssr = to_ssr_subdev(subdev);
+ srcu_notifier_chain_register(ssr->rproc_notif_list, nb);
+
+ return ssr->rproc_notif_list;
}
EXPORT_SYMBOL_GPL(qcom_register_ssr_notifier);
/**
* qcom_unregister_ssr_notifier() - unregister SSR notification handler
+ * @notify: pointer to srcu notifier head
* @nb: notifier_block to unregister
*/
-void qcom_unregister_ssr_notifier(struct notifier_block *nb)
+int qcom_unregister_ssr_notifier(void *notify, struct notifier_block *nb)
{
- blocking_notifier_chain_unregister(&ssr_notifiers, nb);
+ if (!notify)
+ return -EINVAL;
+
+ return srcu_notifier_chain_unregister(notify, nb);
}
EXPORT_SYMBOL_GPL(qcom_unregister_ssr_notifier);
@@ -208,7 +231,7 @@ static void ssr_notify_unprepare(struct rproc_subdev *subdev)
{
struct qcom_rproc_ssr *ssr = to_ssr_subdev(subdev);
- blocking_notifier_call_chain(&ssr_notifiers, 0, (void *)ssr->name);
+ srcu_notifier_call_chain(ssr->rproc_notif_list, 0, (void *)ssr->name);
}
/**
@@ -226,6 +249,9 @@ void qcom_add_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr,
ssr->name = ssr_name;
ssr->subdev.name = kstrdup("ssr_notifs", GFP_KERNEL);
ssr->subdev.unprepare = ssr_notify_unprepare;
+ ssr->rproc_notif_list = kzalloc(sizeof(struct srcu_notifier_head),
+ GFP_KERNEL);
+ srcu_init_notifier_head(ssr->rproc_notif_list);
rproc_add_subdev(rproc, &ssr->subdev);
}
@@ -239,6 +265,7 @@ EXPORT_SYMBOL_GPL(qcom_add_ssr_subdev);
void qcom_remove_ssr_subdev(struct rproc *rproc, struct qcom_rproc_ssr *ssr)
{
kfree(ssr->subdev.name);
+ kfree(ssr->rproc_notif_list);
rproc_remove_subdev(rproc, &ssr->subdev);
}
EXPORT_SYMBOL_GPL(qcom_remove_ssr_subdev);
@@ -27,6 +27,7 @@ struct qcom_rproc_subdev {
struct qcom_rproc_ssr {
struct rproc_subdev subdev;
+ struct srcu_notifier_head *rproc_notif_list;
const char *name;
};
@@ -7,6 +7,7 @@
#include <linux/completion.h>
#include <linux/module.h>
#include <linux/notifier.h>
+#include <linux/remoteproc.h>
#include <linux/rpmsg.h>
#include <linux/remoteproc/qcom_rproc.h>
@@ -49,6 +50,7 @@ struct glink_ssr {
struct rpmsg_endpoint *ept;
struct notifier_block nb;
+ void *notifier_head;
u32 seq_num;
struct completion completion;
@@ -112,6 +114,7 @@ static int qcom_glink_ssr_notify(struct notifier_block *nb, unsigned long event,
static int qcom_glink_ssr_probe(struct rpmsg_device *rpdev)
{
struct glink_ssr *ssr;
+ struct rproc *rproc;
ssr = devm_kzalloc(&rpdev->dev, sizeof(*ssr), GFP_KERNEL);
if (!ssr)
@@ -125,14 +128,27 @@ static int qcom_glink_ssr_probe(struct rpmsg_device *rpdev)
dev_set_drvdata(&rpdev->dev, ssr);
- return qcom_register_ssr_notifier(&ssr->nb);
+ rproc = rproc_get_by_child(&rpdev->dev);
+ if (!rproc) {
+ dev_err(&rpdev->dev, "glink device not child of rproc\n");
+ return -EINVAL;
+ }
+
+ ssr->notifier_head = qcom_register_ssr_notifier(rproc, &ssr->nb);
+ if (IS_ERR(ssr->notifier_head)) {
+ dev_err(&rpdev->dev,
+ "failed to register for ssr notifications\n");
+ return PTR_ERR(ssr->notifier_head);
+ }
+
+ return 0;
}
static void qcom_glink_ssr_remove(struct rpmsg_device *rpdev)
{
struct glink_ssr *ssr = dev_get_drvdata(&rpdev->dev);
- qcom_unregister_ssr_notifier(&ssr->nb);
+ qcom_unregister_ssr_notifier(ssr->notifier_head, &ssr->nb);
}
static const struct rpmsg_device_id qcom_glink_ssr_match[] = {
@@ -2,20 +2,27 @@
#define __QCOM_RPROC_H__
struct notifier_block;
+struct rproc;
#if IS_ENABLED(CONFIG_QCOM_RPROC_COMMON)
-int qcom_register_ssr_notifier(struct notifier_block *nb);
-void qcom_unregister_ssr_notifier(struct notifier_block *nb);
+void *qcom_register_ssr_notifier(struct rproc *rproc,
+ struct notifier_block *nb);
+int qcom_unregister_ssr_notifier(void *notify, struct notifier_block *nb);
#else
-static inline int qcom_register_ssr_notifier(struct notifier_block *nb)
+static inline void *qcom_register_ssr_notifier(struct rproc *rproc,
+ struct notifier_block *nb)
{
- return 0;
+ return NULL;
}
-static inline void qcom_unregister_ssr_notifier(struct notifier_block *nb) {}
+static inline int qcom_unregister_ssr_notifier(void *notify,
+ struct notifier_block *nb)
+{
+ return 0;
+}
#endif