// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (c) 2015 Sascha Hauer , Pengutronix */ #define pr_fmt(fmt) "restart: " fmt #include #include #include #include static LIST_HEAD(restart_handler_list); static unsigned resetidx; /** * restart_handler_register() - register a handler for restarting the system * @rst: The handler struct * * This adds @rst to the list of registered restart handlers. * * return: 0 for success or negative error code */ int restart_handler_register(struct restart_handler *rst) { if (!rst->name) rst->name = basprintf("reset%u", resetidx); if (!rst->priority) rst->priority = RESTART_DEFAULT_PRIORITY; list_add_tail(&rst->list, &restart_handler_list); pr_debug("registering restart handler \"%s\" with priority %d\n", rst->name, rst->priority); resetidx++; return 0; } /** * restart_handler_register_fn() - register a handler function * @name: restart method name or NULL if name should be auto-generated * @restart_fn: The restart function * * convenience wrapper for restart_handler_register() to register a handler * with given function and default values otherwise. * * return: 0 for success or negative error code */ int restart_handler_register_fn(const char *name, void (*restart_fn)(struct restart_handler *)) { struct restart_handler *rst; int ret; rst = xzalloc(sizeof(*rst)); rst->name = xstrdup(name); rst->restart = restart_fn; ret = restart_handler_register(rst); if (ret) free(rst); return ret; } /** * restart_handler_get_by_name() - reset the whole system */ struct restart_handler *restart_handler_get_by_name(const char *name) { struct restart_handler *rst = NULL, *tmp; unsigned int priority = 0; list_for_each_entry(tmp, &restart_handler_list, list) { if (name && tmp->name && strcmp(name, tmp->name)) continue; if (tmp->priority > priority) { priority = tmp->priority; rst = tmp; } } return rst; } /** * restart_machine() - reset the whole system */ void __noreturn restart_machine(void) { struct restart_handler *rst; rst = restart_handler_get_by_name(NULL); if (rst) { pr_debug("%s: using restart handler %s\n", __func__, rst->name); console_flush(); rst->restart(rst); } hang(); } /** * of_get_restart_priority() - get the desired restart priority from device tree * @node: The device_node to read the property from * * return: The priority */ unsigned int of_get_restart_priority(struct device_node *node) { unsigned int priority = RESTART_DEFAULT_PRIORITY; of_property_read_u32(node, "restart-priority", &priority); return priority; } /* * restart_handlers_print - print informations about all restart handlers */ void restart_handlers_print(void) { struct restart_handler *tmp; list_for_each_entry(tmp, &restart_handler_list, list) printf("%-20s %6d\n", tmp->name, tmp->priority); }