summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTaewan Kim <trunixs.kim@samsung.com>2021-05-17 18:58:12 +0900
committerTaewan Kim <trunixs.kim@samsung.com>2021-05-18 17:22:57 +0900
commit82f48417e88a2c82dce54dfd653aa77e6b019a4b (patch)
treec3fb91b20dd0ae1448e2915340232c352b924f81
parent8e95d3bcc4783cc9647882005b9e61a94e40cf46 (diff)
downloadreset-82f48417e88a2c82dce54dfd653aa77e6b019a4b.tar.gz
power: reset: exynos-reboot: Initialize gs201 reset driver.
- Divide 'exynos_reboot.c' into 'exynos_${target}_reboot.c' - Delete gs101 W/A code Signed-off-by: Taewan Kim <trunixs.kim@samsung.com> Change-Id: I7ba9548814b2d4a7bb522fb0d15735f4300dc9d7
-rw-r--r--Kbuild3
-rw-r--r--Kconfig2
-rw-r--r--exynos-gs101-reboot.c (renamed from exynos-reboot.c)0
-rw-r--r--exynos-gs201-reboot.c288
4 files changed, 292 insertions, 1 deletions
diff --git a/Kbuild b/Kbuild
index 50a7688..3191986 100644
--- a/Kbuild
+++ b/Kbuild
@@ -1 +1,4 @@
+exynos-reboot-$(CONFIG_SOC_GS101) += exynos-gs101-reboot.o
+exynos-reboot-$(CONFIG_SOC_GS201) += exynos-gs201-reboot.o
+
obj-$(CONFIG_POWER_RESET_EXYNOS) += exynos-reboot.o
diff --git a/Kconfig b/Kconfig
index df5c74e..1bc5122 100644
--- a/Kconfig
+++ b/Kconfig
@@ -1,5 +1,5 @@
config POWER_RESET_EXYNOS
» tristate "Samsung Exynos power-off driver"
-» depends on SOC_GS101
+» depends on SOC_GS101 || SOC_GS201
» help
» Reboot support for Samsung Exynos boards.
diff --git a/exynos-reboot.c b/exynos-gs101-reboot.c
index 068ebc2..068ebc2 100644
--- a/exynos-reboot.c
+++ b/exynos-gs101-reboot.c
diff --git a/exynos-gs201-reboot.c b/exynos-gs201-reboot.c
new file mode 100644
index 0000000..9a5387c
--- /dev/null
+++ b/exynos-gs201-reboot.c
@@ -0,0 +1,288 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * exynos-reboot.c - Samsung Exynos SoC reset code
+ *
+ * Copyright (c) 2019-2021 Samsung Electronics Co., Ltd.
+ *
+ * Author: Hyunki Koo <hyunki00.koo@samsung.com>
+ * Youngmin Nam <youngmin.nam@samsung.com>
+ */
+
+#include <linux/delay.h>
+#include <linux/io.h>
+#include <linux/of.h>
+#include <linux/of_gpio.h>
+#include <linux/input.h>
+#include <linux/module.h>
+#include <linux/notifier.h>
+#include <linux/of_address.h>
+#include <linux/regmap.h>
+#include <linux/mfd/syscon.h>
+#include <linux/platform_device.h>
+#include <linux/reboot.h>
+#if IS_ENABLED(CONFIG_GS_ACPM)
+#include <soc/google/acpm_ipc_ctrl.h>
+#endif
+#include <soc/google/exynos-el3_mon.h>
+#include <soc/google/debug-snapshot.h>
+/* TODO: temporary workaround. must remove. see b/169128860 */
+#include <linux/soc/samsung/exynos-smc.h>
+#include "../../bms/google_bms.h"
+
+#define EXYNOS_PMU_SYSIP_DAT0 (0x0810)
+
+#define BMS_RSBM_VALID BIT(31)
+
+static struct regmap *pmureg;
+static u32 warm_reboot_offset, warm_reboot_trigger;
+static u32 cold_reboot_offset, cold_reboot_trigger;
+static u32 reboot_cmd_offset;
+static u32 shutdown_offset, shutdown_trigger;
+static phys_addr_t pmu_alive_base;
+
+enum pon_reboot_mode {
+ REBOOT_MODE_NORMAL = 0x00,
+ REBOOT_MODE_CHARGE = 0x0A,
+
+ REBOOT_MODE_DMVERITY_CORRUPTED = 0x50,
+ REBOOT_MODE_SHUTDOWN_THERMAL = 0x51,
+
+ REBOOT_MODE_RESCUE = 0xF9,
+ REBOOT_MODE_FASTBOOT = 0xFA,
+ REBOOT_MODE_BOOTLOADER = 0xFC,
+ REBOOT_MODE_FACTORY = 0xFD,
+ REBOOT_MODE_RECOVERY = 0xFF,
+};
+
+static void exynos_power_off(void)
+{
+ u32 poweroff_try = 0;
+ int power_gpio = -1;
+ unsigned int keycode = 0;
+ struct device_node *np, *pp;
+ int ret;
+
+ np = of_find_node_by_path("/gpio_keys");
+ if (!np)
+ return;
+
+ for_each_child_of_node(np, pp) {
+ if (!of_find_property(pp, "gpios", NULL))
+ continue;
+ of_property_read_u32(pp, "linux,code", &keycode);
+ if (keycode == KEY_POWER) {
+ pr_info("%s: <%u>\n", __func__, keycode);
+ power_gpio = of_get_gpio(pp, 0);
+ break;
+ }
+ }
+
+ of_node_put(np);
+
+ if (!gpio_is_valid(power_gpio)) {
+ pr_err("Couldn't find power key node\n");
+ return;
+ }
+
+ while (1) {
+ /* wait for power button release */
+ if (gpio_get_value(power_gpio)) {
+#if IS_ENABLED(CONFIG_GS_ACPM)
+ exynos_acpm_reboot();
+#endif
+ pr_emerg("Set PS_HOLD Low.\n");
+ ret = rmw_priv_reg(pmu_alive_base + shutdown_offset, shutdown_trigger, 0);
+ /* TODO: remove following fallback. see b/169128860 */
+ if (ret)
+ regmap_update_bits(pmureg, shutdown_offset, shutdown_trigger, 0);
+
+ ++poweroff_try;
+ pr_emerg("Should not reach here! (poweroff_try:%d)\n", poweroff_try);
+ } else {
+ /*
+ * if power button is not released,
+ * wait and check TA again
+ */
+ pr_info("PWR Key is not released.\n");
+ }
+ mdelay(1000);
+ }
+}
+
+static void exynos_reboot_mode_set(u32 val)
+{
+ int ret;
+ phys_addr_t reboot_cmd_addr = pmu_alive_base + reboot_cmd_offset;
+
+ ret = set_priv_reg(reboot_cmd_addr, val);
+ /* TODO: remove following fallback. see b/169128860 */
+ if (ret) {
+ pr_info("%s(): failed to set addr %pap via set_priv_reg, using regmap\n",
+ __func__, &reboot_cmd_addr);
+ regmap_write(pmureg, reboot_cmd_offset, val);
+ }
+}
+
+static void exynos_reboot_parse(const char *cmd)
+{
+ if (cmd) {
+ u32 value = U32_MAX;
+
+ pr_info("Reboot command: '%s'\n", cmd);
+
+ if (!strcmp(cmd, "charge"))
+ value = REBOOT_MODE_CHARGE;
+ else if (!strcmp(cmd, "bootloader"))
+ value = REBOOT_MODE_BOOTLOADER;
+ else if (!strcmp(cmd, "fastboot"))
+ value = REBOOT_MODE_FASTBOOT;
+ else if (!strcmp(cmd, "recovery"))
+ value = REBOOT_MODE_RECOVERY;
+ else if (!strcmp(cmd, "dm-verity device corrupted"))
+ value = REBOOT_MODE_DMVERITY_CORRUPTED;
+ else if (!strcmp(cmd, "rescue"))
+ value = REBOOT_MODE_RESCUE;
+ else if (!strcmp(cmd, "shutdown-thermal"))
+ value = REBOOT_MODE_SHUTDOWN_THERMAL;
+ else if (!strcmp(cmd, "from_fastboot") ||
+ !strcmp(cmd, "shell") ||
+ !strcmp(cmd, "userrequested") ||
+ !strcmp(cmd, "userrequested,fastboot") ||
+ !strcmp(cmd, "userrequested,recovery") ||
+ !strcmp(cmd, "userrequested,recovery,ui"))
+ value = REBOOT_MODE_NORMAL;
+ else
+ pr_err("Unknown reboot command: '%s'\n", cmd);
+
+ if (value != U32_MAX)
+ exynos_reboot_mode_set(value);
+ }
+}
+
+static int exynos_reboot_handler(struct notifier_block *nb, unsigned long mode, void *cmd)
+{
+ exynos_reboot_parse(cmd);
+
+ return NOTIFY_DONE;
+}
+
+static struct notifier_block exynos_reboot_nb = {
+ .notifier_call = exynos_reboot_handler,
+ .priority = INT_MAX,
+};
+
+static int exynos_restart_handler(struct notifier_block *this, unsigned long mode, void *cmd)
+{
+#if IS_ENABLED(CONFIG_GS_ACPM)
+ exynos_acpm_reboot();
+#endif
+
+ /* Do S/W Reset */
+ pr_emerg("%s: Exynos SoC reset right now\n", __func__);
+
+ pr_emerg("Set PS_HOLD Low.\n");
+ mdelay(2);
+ rmw_priv_reg(pmu_alive_base + cold_reboot_offset, cold_reboot_trigger, 0);
+
+ while (1)
+ wfi();
+
+ return NOTIFY_DONE;
+}
+
+static struct notifier_block exynos_restart_nb = {
+ .notifier_call = exynos_restart_handler,
+ .priority = 128,
+};
+
+static int exynos_reboot_probe(struct platform_device *pdev)
+{
+ struct device *dev = &pdev->dev;
+ struct device_node *np = pdev->dev.of_node;
+ struct device_node *syscon_np;
+ struct resource res;
+ int err;
+
+ pmureg = syscon_regmap_lookup_by_phandle(np, "syscon");
+ if (IS_ERR(pmureg)) {
+ dev_err(dev, "Fail to get regmap of PMU\n");
+ return PTR_ERR(pmureg);
+ }
+
+ syscon_np = of_parse_phandle(np, "syscon", 0);
+ if (!syscon_np) {
+ dev_err(dev, "syscon device node not found\n");
+ return -EINVAL;
+ }
+
+ if (of_address_to_resource(syscon_np, 0, &res)) {
+ dev_err(dev, "failed to get syscon base address\n");
+ return -ENOMEM;
+ }
+
+ pmu_alive_base = res.start;
+
+ if (of_property_read_u32(np, "swreset-system-offset", &warm_reboot_offset) < 0) {
+ dev_err(dev, "failed to find swreset-system-offset property\n");
+ return -EINVAL;
+ }
+
+ if (of_property_read_u32(np, "swreset-system-trigger", &warm_reboot_trigger) < 0) {
+ dev_err(dev, "failed to find swreset-system-trigger property\n");
+ return -EINVAL;
+ }
+
+ if (of_property_read_u32(np, "pshold-control-offset", &cold_reboot_offset) < 0) {
+ dev_err(dev, "failed to find pshold-control-offset property\n");
+ return -EINVAL;
+ }
+
+ if (of_property_read_u32(np, "pshold-control-trigger", &cold_reboot_trigger) < 0) {
+ dev_err(dev, "failed to find shutdown-trigger property\n");
+ return -EINVAL;
+ }
+
+ shutdown_offset = cold_reboot_offset;
+ shutdown_trigger = cold_reboot_trigger;
+
+ if (of_property_read_u32(np, "reboot-cmd-offset", &reboot_cmd_offset) < 0) {
+ dev_info(dev, "failed to find reboot-offset property, using default\n");
+ reboot_cmd_offset = EXYNOS_PMU_SYSIP_DAT0;
+ }
+
+ err = register_reboot_notifier(&exynos_reboot_nb);
+ if (err) {
+ dev_err(dev, "cannot register reboot handler (err=%d)\n", err);
+ return err;
+ }
+
+ err = register_restart_handler(&exynos_restart_nb);
+ if (err) {
+ dev_err(dev, "cannot register restart handler (err=%d)\n", err);
+ unregister_reboot_notifier(&exynos_reboot_nb);
+ return err;
+ }
+
+ pm_power_off = exynos_power_off;
+ dev_info(dev, "register restart handler successfully\n");
+
+ return 0;
+}
+
+static const struct of_device_id exynos_reboot_of_match[] = {
+ { .compatible = "samsung,exynos-reboot" },
+ {}
+};
+
+static struct platform_driver exynos_reboot_driver = {
+ .probe = exynos_reboot_probe,
+ .driver = {
+ .name = "exynos-reboot",
+ .of_match_table = exynos_reboot_of_match,
+ },
+};
+module_platform_driver(exynos_reboot_driver);
+
+MODULE_DESCRIPTION("Exynos Reboot driver");
+MODULE_LICENSE("GPL v2");
+MODULE_ALIAS("platform:exynos-reboot");