/* * Copyright (C) 2014 Linaro Ltd. * * Author: Linus Walleij <linus.walleij@linaro.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2, as * published by the Free Software Foundation. * */ #include <linux/init.h> #include <linux/mfd/syscon.h> #include <linux/reboot.h> #include <linux/regmap.h> #include <linux/of.h> #define INTEGRATOR_HDR_CTRL_OFFSET 0x0C #define INTEGRATOR_HDR_LOCK_OFFSET 0x14 #define INTEGRATOR_CM_CTRL_RESET (1 << 3) #define REALVIEW_SYS_LOCK_OFFSET 0x20 #define REALVIEW_SYS_RESETCTL_OFFSET 0x40 /* Magic unlocking token used on all Versatile boards */ #define VERSATILE_LOCK_VAL 0xA05F /* * We detect the different syscon types from the compatible strings. */ enum versatile_reboot { INTEGRATOR_REBOOT_CM, REALVIEW_REBOOT_EB, REALVIEW_REBOOT_PB1176, REALVIEW_REBOOT_PB11MP, REALVIEW_REBOOT_PBA8, REALVIEW_REBOOT_PBX, }; /* Pointer to the system controller */ static struct regmap *syscon_regmap; static enum versatile_reboot versatile_reboot_type; static const struct of_device_id versatile_reboot_of_match[] = { { .compatible = "arm,core-module-integrator", .data = (void *)INTEGRATOR_REBOOT_CM }, { .compatible = "arm,realview-eb-syscon", .data = (void *)REALVIEW_REBOOT_EB, }, { .compatible = "arm,realview-pb1176-syscon", .data = (void *)REALVIEW_REBOOT_PB1176, }, { .compatible = "arm,realview-pb11mp-syscon", .data = (void *)REALVIEW_REBOOT_PB11MP, }, { .compatible = "arm,realview-pba8-syscon", .data = (void *)REALVIEW_REBOOT_PBA8, }, { .compatible = "arm,realview-pbx-syscon", .data = (void *)REALVIEW_REBOOT_PBX, }, {}, }; static int versatile_reboot(struct notifier_block *this, unsigned long mode, void *cmd) { /* Unlock the reset register */ /* Then hit reset on the different machines */ switch (versatile_reboot_type) { case INTEGRATOR_REBOOT_CM: regmap_write(syscon_regmap, INTEGRATOR_HDR_LOCK_OFFSET, VERSATILE_LOCK_VAL); regmap_update_bits(syscon_regmap, INTEGRATOR_HDR_CTRL_OFFSET, INTEGRATOR_CM_CTRL_RESET, INTEGRATOR_CM_CTRL_RESET); break; case REALVIEW_REBOOT_EB: regmap_write(syscon_regmap, REALVIEW_SYS_LOCK_OFFSET, VERSATILE_LOCK_VAL); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x0008); break; case REALVIEW_REBOOT_PB1176: regmap_write(syscon_regmap, REALVIEW_SYS_LOCK_OFFSET, VERSATILE_LOCK_VAL); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x0100); break; case REALVIEW_REBOOT_PB11MP: case REALVIEW_REBOOT_PBA8: regmap_write(syscon_regmap, REALVIEW_SYS_LOCK_OFFSET, VERSATILE_LOCK_VAL); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x0000); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x0004); break; case REALVIEW_REBOOT_PBX: regmap_write(syscon_regmap, REALVIEW_SYS_LOCK_OFFSET, VERSATILE_LOCK_VAL); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x00f0); regmap_write(syscon_regmap, REALVIEW_SYS_RESETCTL_OFFSET, 0x00f4); break; } dsb(); return NOTIFY_DONE; } static struct notifier_block versatile_reboot_nb = { .notifier_call = versatile_reboot, .priority = 192, }; static int __init versatile_reboot_probe(void) { const struct of_device_id *reboot_id; struct device_node *np; int err; np = of_find_matching_node_and_match(NULL, versatile_reboot_of_match, &reboot_id); if (!np) return -ENODEV; versatile_reboot_type = (enum versatile_reboot)reboot_id->data; syscon_regmap = syscon_node_to_regmap(np); if (IS_ERR(syscon_regmap)) return PTR_ERR(syscon_regmap); err = register_restart_handler(&versatile_reboot_nb); if (err) return err; pr_info("versatile reboot driver registered\n"); return 0; } device_initcall(versatile_reboot_probe);