| // SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) |
| /* |
| * SP7021 reset driver |
| * |
| * Copyright (C) Sunplus Technology Co., Ltd. |
| * All rights reserved. |
| */ |
| |
| #include <linux/io.h> |
| #include <linux/init.h> |
| #include <linux/mod_devicetable.h> |
| #include <linux/platform_device.h> |
| #include <linux/reset-controller.h> |
| #include <linux/reboot.h> |
| |
| /* HIWORD_MASK_REG BITS */ |
| #define BITS_PER_HWM_REG 16 |
| |
| /* resets HW info: reg_index_shift */ |
| static const u32 sp_resets[] = { |
| /* SP7021: mo_reset0 ~ mo_reset9 */ |
| 0x00, |
| 0x02, |
| 0x03, |
| 0x04, |
| 0x05, |
| 0x06, |
| 0x07, |
| 0x08, |
| 0x09, |
| 0x0a, |
| 0x0b, |
| 0x0d, |
| 0x0e, |
| 0x0f, |
| 0x10, |
| 0x12, |
| 0x14, |
| 0x15, |
| 0x16, |
| 0x17, |
| 0x18, |
| 0x19, |
| 0x1a, |
| 0x1b, |
| 0x1c, |
| 0x1d, |
| 0x1e, |
| 0x1f, |
| 0x20, |
| 0x21, |
| 0x22, |
| 0x23, |
| 0x24, |
| 0x25, |
| 0x26, |
| 0x2a, |
| 0x2b, |
| 0x2d, |
| 0x2e, |
| 0x30, |
| 0x31, |
| 0x32, |
| 0x33, |
| 0x3d, |
| 0x3e, |
| 0x3f, |
| 0x42, |
| 0x44, |
| 0x4b, |
| 0x4c, |
| 0x4d, |
| 0x4e, |
| 0x4f, |
| 0x50, |
| 0x55, |
| 0x60, |
| 0x61, |
| 0x6a, |
| 0x6f, |
| 0x70, |
| 0x73, |
| 0x74, |
| 0x86, |
| 0x8a, |
| 0x8b, |
| 0x8d, |
| 0x8e, |
| 0x8f, |
| 0x90, |
| 0x92, |
| 0x93, |
| 0x94, |
| 0x95, |
| 0x96, |
| 0x97, |
| 0x98, |
| 0x99, |
| }; |
| |
| struct sp_reset { |
| struct reset_controller_dev rcdev; |
| struct notifier_block notifier; |
| void __iomem *base; |
| }; |
| |
| static inline struct sp_reset *to_sp_reset(struct reset_controller_dev *rcdev) |
| { |
| return container_of(rcdev, struct sp_reset, rcdev); |
| } |
| |
| static int sp_reset_update(struct reset_controller_dev *rcdev, |
| unsigned long id, bool assert) |
| { |
| struct sp_reset *reset = to_sp_reset(rcdev); |
| int index = sp_resets[id] / BITS_PER_HWM_REG; |
| int shift = sp_resets[id] % BITS_PER_HWM_REG; |
| u32 val; |
| |
| val = (1 << (16 + shift)) | (assert << shift); |
| writel(val, reset->base + (index * 4)); |
| |
| return 0; |
| } |
| |
| static int sp_reset_assert(struct reset_controller_dev *rcdev, |
| unsigned long id) |
| { |
| return sp_reset_update(rcdev, id, true); |
| } |
| |
| static int sp_reset_deassert(struct reset_controller_dev *rcdev, |
| unsigned long id) |
| { |
| return sp_reset_update(rcdev, id, false); |
| } |
| |
| static int sp_reset_status(struct reset_controller_dev *rcdev, |
| unsigned long id) |
| { |
| struct sp_reset *reset = to_sp_reset(rcdev); |
| int index = sp_resets[id] / BITS_PER_HWM_REG; |
| int shift = sp_resets[id] % BITS_PER_HWM_REG; |
| u32 reg; |
| |
| reg = readl(reset->base + (index * 4)); |
| |
| return !!(reg & BIT(shift)); |
| } |
| |
| static const struct reset_control_ops sp_reset_ops = { |
| .assert = sp_reset_assert, |
| .deassert = sp_reset_deassert, |
| .status = sp_reset_status, |
| }; |
| |
| static int sp_restart(struct notifier_block *nb, unsigned long mode, |
| void *cmd) |
| { |
| struct sp_reset *reset = container_of(nb, struct sp_reset, notifier); |
| |
| sp_reset_assert(&reset->rcdev, 0); |
| sp_reset_deassert(&reset->rcdev, 0); |
| |
| return NOTIFY_DONE; |
| } |
| |
| static int sp_reset_probe(struct platform_device *pdev) |
| { |
| struct device *dev = &pdev->dev; |
| struct sp_reset *reset; |
| struct resource *res; |
| int ret; |
| |
| reset = devm_kzalloc(dev, sizeof(*reset), GFP_KERNEL); |
| if (!reset) |
| return -ENOMEM; |
| |
| reset->base = devm_platform_get_and_ioremap_resource(pdev, 0, &res); |
| if (IS_ERR(reset->base)) |
| return PTR_ERR(reset->base); |
| |
| reset->rcdev.ops = &sp_reset_ops; |
| reset->rcdev.owner = THIS_MODULE; |
| reset->rcdev.of_node = dev->of_node; |
| reset->rcdev.nr_resets = resource_size(res) / 4 * BITS_PER_HWM_REG; |
| |
| ret = devm_reset_controller_register(dev, &reset->rcdev); |
| if (ret) |
| return ret; |
| |
| reset->notifier.notifier_call = sp_restart; |
| reset->notifier.priority = 192; |
| |
| return register_restart_handler(&reset->notifier); |
| } |
| |
| static const struct of_device_id sp_reset_dt_ids[] = { |
| {.compatible = "sunplus,sp7021-reset",}, |
| { /* sentinel */ }, |
| }; |
| |
| static struct platform_driver sp_reset_driver = { |
| .probe = sp_reset_probe, |
| .driver = { |
| .name = "sunplus-reset", |
| .of_match_table = sp_reset_dt_ids, |
| .suppress_bind_attrs = true, |
| }, |
| }; |
| builtin_platform_driver(sp_reset_driver); |