cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

svc_watchdog.c (4474B)


      1// SPDX-License-Identifier: GPL-2.0
      2/*
      3 * SVC Greybus "watchdog" driver.
      4 *
      5 * Copyright 2016 Google Inc.
      6 */
      7
      8#include <linux/delay.h>
      9#include <linux/suspend.h>
     10#include <linux/workqueue.h>
     11#include <linux/greybus.h>
     12
     13#define SVC_WATCHDOG_PERIOD	(2 * HZ)
     14
     15struct gb_svc_watchdog {
     16	struct delayed_work	work;
     17	struct gb_svc		*svc;
     18	bool			enabled;
     19	struct notifier_block pm_notifier;
     20};
     21
     22static struct delayed_work reset_work;
     23
     24static int svc_watchdog_pm_notifier(struct notifier_block *notifier,
     25				    unsigned long pm_event, void *unused)
     26{
     27	struct gb_svc_watchdog *watchdog =
     28		container_of(notifier, struct gb_svc_watchdog, pm_notifier);
     29
     30	switch (pm_event) {
     31	case PM_SUSPEND_PREPARE:
     32		gb_svc_watchdog_disable(watchdog->svc);
     33		break;
     34	case PM_POST_SUSPEND:
     35		gb_svc_watchdog_enable(watchdog->svc);
     36		break;
     37	default:
     38		break;
     39	}
     40
     41	return NOTIFY_DONE;
     42}
     43
     44static void greybus_reset(struct work_struct *work)
     45{
     46	static char const start_path[] = "/system/bin/start";
     47	static char *envp[] = {
     48		"HOME=/",
     49		"PATH=/sbin:/vendor/bin:/system/sbin:/system/bin:/system/xbin",
     50		NULL,
     51	};
     52	static char *argv[] = {
     53		(char *)start_path,
     54		"unipro_reset",
     55		NULL,
     56	};
     57
     58	pr_err("svc_watchdog: calling \"%s %s\" to reset greybus network!\n",
     59	       argv[0], argv[1]);
     60	call_usermodehelper(start_path, argv, envp, UMH_WAIT_EXEC);
     61}
     62
     63static void do_work(struct work_struct *work)
     64{
     65	struct gb_svc_watchdog *watchdog;
     66	struct gb_svc *svc;
     67	int retval;
     68
     69	watchdog = container_of(work, struct gb_svc_watchdog, work.work);
     70	svc = watchdog->svc;
     71
     72	dev_dbg(&svc->dev, "%s: ping.\n", __func__);
     73	retval = gb_svc_ping(svc);
     74	if (retval) {
     75		/*
     76		 * Something went really wrong, let's warn userspace and then
     77		 * pull the plug and reset the whole greybus network.
     78		 * We need to do this outside of this workqueue as we will be
     79		 * tearing down the svc device itself.  So queue up
     80		 * yet-another-callback to do that.
     81		 */
     82		dev_err(&svc->dev,
     83			"SVC ping has returned %d, something is wrong!!!\n",
     84			retval);
     85
     86		if (svc->action == GB_SVC_WATCHDOG_BITE_PANIC_KERNEL) {
     87			panic("SVC is not responding\n");
     88		} else if (svc->action == GB_SVC_WATCHDOG_BITE_RESET_UNIPRO) {
     89			dev_err(&svc->dev, "Resetting the greybus network, watch out!!!\n");
     90
     91			INIT_DELAYED_WORK(&reset_work, greybus_reset);
     92			schedule_delayed_work(&reset_work, HZ / 2);
     93
     94			/*
     95			 * Disable ourselves, we don't want to trip again unless
     96			 * userspace wants us to.
     97			 */
     98			watchdog->enabled = false;
     99		}
    100	}
    101
    102	/* resubmit our work to happen again, if we are still "alive" */
    103	if (watchdog->enabled)
    104		schedule_delayed_work(&watchdog->work, SVC_WATCHDOG_PERIOD);
    105}
    106
    107int gb_svc_watchdog_create(struct gb_svc *svc)
    108{
    109	struct gb_svc_watchdog *watchdog;
    110	int retval;
    111
    112	if (svc->watchdog)
    113		return 0;
    114
    115	watchdog = kmalloc(sizeof(*watchdog), GFP_KERNEL);
    116	if (!watchdog)
    117		return -ENOMEM;
    118
    119	watchdog->enabled = false;
    120	watchdog->svc = svc;
    121	INIT_DELAYED_WORK(&watchdog->work, do_work);
    122	svc->watchdog = watchdog;
    123
    124	watchdog->pm_notifier.notifier_call = svc_watchdog_pm_notifier;
    125	retval = register_pm_notifier(&watchdog->pm_notifier);
    126	if (retval) {
    127		dev_err(&svc->dev, "error registering pm notifier(%d)\n",
    128			retval);
    129		goto svc_watchdog_create_err;
    130	}
    131
    132	retval = gb_svc_watchdog_enable(svc);
    133	if (retval) {
    134		dev_err(&svc->dev, "error enabling watchdog (%d)\n", retval);
    135		unregister_pm_notifier(&watchdog->pm_notifier);
    136		goto svc_watchdog_create_err;
    137	}
    138	return retval;
    139
    140svc_watchdog_create_err:
    141	svc->watchdog = NULL;
    142	kfree(watchdog);
    143
    144	return retval;
    145}
    146
    147void gb_svc_watchdog_destroy(struct gb_svc *svc)
    148{
    149	struct gb_svc_watchdog *watchdog = svc->watchdog;
    150
    151	if (!watchdog)
    152		return;
    153
    154	unregister_pm_notifier(&watchdog->pm_notifier);
    155	gb_svc_watchdog_disable(svc);
    156	svc->watchdog = NULL;
    157	kfree(watchdog);
    158}
    159
    160bool gb_svc_watchdog_enabled(struct gb_svc *svc)
    161{
    162	if (!svc || !svc->watchdog)
    163		return false;
    164	return svc->watchdog->enabled;
    165}
    166
    167int gb_svc_watchdog_enable(struct gb_svc *svc)
    168{
    169	struct gb_svc_watchdog *watchdog;
    170
    171	if (!svc->watchdog)
    172		return -ENODEV;
    173
    174	watchdog = svc->watchdog;
    175	if (watchdog->enabled)
    176		return 0;
    177
    178	watchdog->enabled = true;
    179	schedule_delayed_work(&watchdog->work, SVC_WATCHDOG_PERIOD);
    180	return 0;
    181}
    182
    183int gb_svc_watchdog_disable(struct gb_svc *svc)
    184{
    185	struct gb_svc_watchdog *watchdog;
    186
    187	if (!svc->watchdog)
    188		return -ENODEV;
    189
    190	watchdog = svc->watchdog;
    191	if (!watchdog->enabled)
    192		return 0;
    193
    194	watchdog->enabled = false;
    195	cancel_delayed_work_sync(&watchdog->work);
    196	return 0;
    197}