/* Copyright (c) 2014-2019, The Linux Foundation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program 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 General Public License for more details. * */ #ifndef __SUBSYS_RESTART_H #define __SUBSYS_RESTART_H #include #include struct subsys_device; extern struct bus_type subsys_bus_type; enum { RESET_SOC = 0, RESET_SUBSYS_COUPLED, RESET_LEVEL_MAX }; enum crash_status { CRASH_STATUS_NO_CRASH = 0, CRASH_STATUS_ERR_FATAL, CRASH_STATUS_WDOG_BITE, }; struct device; struct module; enum ssr_comm { SUBSYS_TO_SUBSYS_SYSMON, SUBSYS_TO_HLOS, HLOS_TO_SUBSYS_SYSMON_SHUTDOWN, NUM_SSR_COMMS, }; /** * struct subsys_notif_timeout - timeout data used by notification timeout hdlr * @comm_type: Specifies if the type of communication being tracked is * through sysmon between two subsystems, subsystem notifier call chain, or * sysmon shutdown. * @dest_name: subsystem to which sysmon notification is being sent to * @source_name: subsystem which generated event that notification is being sent * for * @timer: timer for scheduling timeout */ struct subsys_notif_timeout { enum ssr_comm comm_type; const char *dest_name; const char *source_name; struct timer_list timer; }; /** * struct subsys_desc - subsystem descriptor * @name: name of subsystem * @fw_name: firmware name * @pon_depends_on: subsystem this subsystem wants to power-on first. If the * dependednt subsystem is already powered-on, the framework won't try to power * it back up again. * @poff_depends_on: subsystem this subsystem wants to power-off first. If the * dependednt subsystem is already powered-off, the framework won't try to power * it off again. * @dev: parent device * @owner: module the descriptor belongs to * @shutdown: Stop a subsystem * @powerup: Start a subsystem * @crash_shutdown: Shutdown a subsystem when the system crashes (can't sleep) * @ramdump: Collect a ramdump of the subsystem * @free_memory: Free the memory associated with this subsystem * @is_not_loadable: Indicate if subsystem firmware is not loadable via pil * framework * @no_auth: Set if subsystem does not rely on PIL to authenticate and bring * it out of reset * @ssctl_instance_id: Instance id used to connect with SSCTL service * @sysmon_pid: pdev id that sysmon is probed with for the subsystem * @sysmon_shutdown_ret: Return value for the call to sysmon_send_shutdown * @system_debug: If "set", triggers a device restart when the * subsystem's wdog bite handler is invoked. * @ignore_ssr_failure: SSR failures are usually fatal and results in panic. If * set will ignore failure. * @edge: GLINK logical name of the subsystem */ struct subsys_desc { const char *name; char fw_name[256]; const char *pon_depends_on; const char *poff_depends_on; struct device *dev; struct module *owner; int (*shutdown)(const struct subsys_desc *desc, bool force_stop); int (*powerup)(const struct subsys_desc *desc); void (*crash_shutdown)(const struct subsys_desc *desc); int (*ramdump)(int, const struct subsys_desc *desc); void (*free_memory)(const struct subsys_desc *desc); irqreturn_t (*err_fatal_handler)(int irq, void *dev_id); irqreturn_t (*stop_ack_handler)(int irq, void *dev_id); irqreturn_t (*shutdown_ack_handler)(int irq, void *dev_id); irqreturn_t (*ramdump_disable_handler)(int irq, void *dev_id); irqreturn_t (*wdog_bite_handler)(int irq, void *dev_id); irqreturn_t (*generic_handler)(int irq, void *dev_id); int is_not_loadable; int err_fatal_gpio; unsigned int err_fatal_irq; unsigned int err_ready_irq; unsigned int stop_ack_irq; unsigned int wdog_bite_irq; unsigned int generic_irq; int force_stop_bit; int stop_reason_0_bit; int stop_reason_1_bit; int ramdump_disable_irq; int shutdown_ack_irq; int ramdump_disable; bool no_auth; bool pil_mss_memsetup; int ssctl_instance_id; u32 sysmon_pid; int sysmon_shutdown_ret; bool system_debug; bool ignore_ssr_failure; const char *edge; struct qcom_smem_state *state; #ifdef CONFIG_SETUP_SSR_NOTIF_TIMEOUTS struct subsys_notif_timeout timeout_data; #endif /* CONFIG_SETUP_SSR_NOTIF_TIMEOUTS */ #ifdef CONFIG_SENSORS_SSC bool run_fssr; #endif }; /** * struct notif_data - additional notif information * @crashed: indicates if subsystem has crashed due to wdog bite or err fatal * @enable_ramdump: ramdumps disabled if set to 0 * @enable_mini_ramdumps: enable flag for minimized critical-memory-only * ramdumps * @no_auth: set if subsystem does not use PIL to bring it out of reset * @pdev: subsystem platform device pointer */ struct notif_data { enum crash_status crashed; int enable_ramdump; int enable_mini_ramdumps; bool no_auth; struct platform_device *pdev; }; #if defined(CONFIG_MSM_SUBSYSTEM_RESTART) extern int subsys_get_restart_level(struct subsys_device *dev); extern int subsystem_restart_dev(struct subsys_device *dev); extern int subsystem_restart(const char *name); extern int subsystem_crash(const char *name); extern void subsys_force_stop(const char *name, bool val); extern int subsystem_crashed(const char *name); extern void *subsystem_get(const char *name); extern void *subsystem_get_with_fwname(const char *name, const char *fw_name); extern int subsystem_set_fwname(const char *name, const char *fw_name); extern void subsystem_put(void *subsystem); extern struct subsys_device *subsys_register(struct subsys_desc *desc); extern void subsys_unregister(struct subsys_device *dev); extern void subsys_default_online(struct subsys_device *dev); extern void subsys_set_crash_status(struct subsys_device *dev, enum crash_status crashed); extern enum crash_status subsys_get_crash_status(struct subsys_device *dev); void notify_proxy_vote(struct device *device); void notify_proxy_unvote(struct device *device); void complete_err_ready(struct subsys_device *subsys); void complete_shutdown_ack(struct subsys_device *subsys); struct subsys_device *find_subsys_device(const char *str); extern int wait_for_shutdown_ack(struct subsys_desc *desc); #ifdef CONFIG_SENSORS_SSC extern void subsys_set_fssr(struct subsys_device *dev, bool value); #endif #else static inline int subsys_get_restart_level(struct subsys_device *dev) { return 0; } static inline int subsystem_restart_dev(struct subsys_device *dev) { return 0; } static inline int subsystem_restart(const char *name) { return 0; } static inline subsystem_crash(const char *name) { } static inline void subsys_force_stop(const char *name, bool val) { } static inline int subsystem_crashed(const char *name) { return 0; } static inline void *subsystem_get(const char *name) { return NULL; } static inline void *subsystem_get_with_fwname(const char *name, const char *fw_name) { return NULL; } static inline int subsystem_set_fwname(const char *name, const char *fw_name) { return 0; } static inline void subsystem_put(void *subsystem) { } static inline struct subsys_device *subsys_register(struct subsys_desc *desc) { return NULL; } static inline void subsys_unregister(struct subsys_device *dev) { } static inline void subsys_default_online(struct subsys_device *dev) { } static inline void subsys_set_crash_status(struct subsys_device *dev, enum crash_status crashed) { } static inline enum crash_status subsys_get_crash_status(struct subsys_device *dev) { return false; } static inline void notify_proxy_vote(struct device *device) { } static inline void notify_proxy_unvote(struct device *device) { } static inline int wait_for_shutdown_ack(struct subsys_desc *desc) { return -EOPNOTSUPP; } #ifdef CONFIG_SENSORS_SSC static void subsys_set_fssr(struct subsys_device *dev, bool value) { return; } #endif #endif /* CONFIG_MSM_SUBSYSTEM_RESTART */ #endif