| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * PolarFire SoC (MPFS) Peripheral Clock Reset Controller |
| * |
| * Author: Conor Dooley <conor.dooley@microchip.com> |
| * Copyright (c) 2022 Microchip Technology Inc. and its subsidiaries. |
| * |
| */ |
| #include <linux/auxiliary_bus.h> |
| #include <linux/delay.h> |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/platform_device.h> |
| #include <linux/slab.h> |
| #include <linux/reset-controller.h> |
| #include <dt-bindings/clock/microchip,mpfs-clock.h> |
| #include <soc/microchip/mpfs.h> |
| |
| /* |
| * The ENVM reset is the lowest bit in the register & I am using the CLK_FOO |
| * defines in the dt to make things easier to configure - so this is accounting |
| * for the offset of 3 there. |
| */ |
| #define MPFS_PERIPH_OFFSET CLK_ENVM |
| #define MPFS_NUM_RESETS 30u |
| #define MPFS_SLEEP_MIN_US 100 |
| #define MPFS_SLEEP_MAX_US 200 |
| |
| /* block concurrent access to the soft reset register */ |
| static DEFINE_SPINLOCK(mpfs_reset_lock); |
| |
| struct mpfs_reset { |
| void __iomem *base; |
| struct reset_controller_dev rcdev; |
| }; |
| |
| static inline struct mpfs_reset *to_mpfs_reset(struct reset_controller_dev *rcdev) |
| { |
| return container_of(rcdev, struct mpfs_reset, rcdev); |
| } |
| |
| /* |
| * Peripheral clock resets |
| */ |
| static int mpfs_assert(struct reset_controller_dev *rcdev, unsigned long id) |
| { |
| struct mpfs_reset *rst = to_mpfs_reset(rcdev); |
| unsigned long flags; |
| u32 reg; |
| |
| spin_lock_irqsave(&mpfs_reset_lock, flags); |
| |
| reg = readl(rst->base); |
| reg |= BIT(id); |
| writel(reg, rst->base); |
| |
| spin_unlock_irqrestore(&mpfs_reset_lock, flags); |
| |
| return 0; |
| } |
| |
| static int mpfs_deassert(struct reset_controller_dev *rcdev, unsigned long id) |
| { |
| struct mpfs_reset *rst = to_mpfs_reset(rcdev); |
| unsigned long flags; |
| u32 reg; |
| |
| spin_lock_irqsave(&mpfs_reset_lock, flags); |
| |
| reg = readl(rst->base); |
| reg &= ~BIT(id); |
| writel(reg, rst->base); |
| |
| spin_unlock_irqrestore(&mpfs_reset_lock, flags); |
| |
| return 0; |
| } |
| |
| static int mpfs_status(struct reset_controller_dev *rcdev, unsigned long id) |
| { |
| struct mpfs_reset *rst = to_mpfs_reset(rcdev); |
| u32 reg = readl(rst->base); |
| |
| /* |
| * It is safe to return here as MPFS_NUM_RESETS makes sure the sign bit |
| * is never hit. |
| */ |
| return (reg & BIT(id)); |
| } |
| |
| static int mpfs_reset(struct reset_controller_dev *rcdev, unsigned long id) |
| { |
| mpfs_assert(rcdev, id); |
| |
| usleep_range(MPFS_SLEEP_MIN_US, MPFS_SLEEP_MAX_US); |
| |
| mpfs_deassert(rcdev, id); |
| |
| return 0; |
| } |
| |
| static const struct reset_control_ops mpfs_reset_ops = { |
| .reset = mpfs_reset, |
| .assert = mpfs_assert, |
| .deassert = mpfs_deassert, |
| .status = mpfs_status, |
| }; |
| |
| static int mpfs_reset_xlate(struct reset_controller_dev *rcdev, |
| const struct of_phandle_args *reset_spec) |
| { |
| unsigned int index = reset_spec->args[0]; |
| |
| /* |
| * CLK_RESERVED does not map to a clock, but it does map to a reset, |
| * so it has to be accounted for here. It is the reset for the fabric, |
| * so if this reset gets called - do not reset it. |
| */ |
| if (index == CLK_RESERVED) { |
| dev_err(rcdev->dev, "Resetting the fabric is not supported\n"); |
| return -EINVAL; |
| } |
| |
| if (index < MPFS_PERIPH_OFFSET || index >= (MPFS_PERIPH_OFFSET + rcdev->nr_resets)) { |
| dev_err(rcdev->dev, "Invalid reset index %u\n", index); |
| return -EINVAL; |
| } |
| |
| return index - MPFS_PERIPH_OFFSET; |
| } |
| |
| static int mpfs_reset_probe(struct auxiliary_device *adev, |
| const struct auxiliary_device_id *id) |
| { |
| struct device *dev = &adev->dev; |
| struct reset_controller_dev *rcdev; |
| struct mpfs_reset *rst; |
| |
| rst = devm_kzalloc(dev, sizeof(*rst), GFP_KERNEL); |
| if (!rst) |
| return -ENOMEM; |
| |
| rst->base = (void __iomem *)adev->dev.platform_data; |
| |
| rcdev = &rst->rcdev; |
| rcdev->dev = dev; |
| rcdev->dev->parent = dev->parent; |
| rcdev->ops = &mpfs_reset_ops; |
| rcdev->of_node = dev->parent->of_node; |
| rcdev->of_reset_n_cells = 1; |
| rcdev->of_xlate = mpfs_reset_xlate; |
| rcdev->nr_resets = MPFS_NUM_RESETS; |
| |
| return devm_reset_controller_register(dev, rcdev); |
| } |
| |
| static void mpfs_reset_unregister_adev(void *_adev) |
| { |
| struct auxiliary_device *adev = _adev; |
| |
| auxiliary_device_delete(adev); |
| auxiliary_device_uninit(adev); |
| } |
| |
| static void mpfs_reset_adev_release(struct device *dev) |
| { |
| struct auxiliary_device *adev = to_auxiliary_dev(dev); |
| |
| kfree(adev); |
| } |
| |
| static struct auxiliary_device *mpfs_reset_adev_alloc(struct device *clk_dev) |
| { |
| struct auxiliary_device *adev; |
| int ret; |
| |
| adev = kzalloc(sizeof(*adev), GFP_KERNEL); |
| if (!adev) |
| return ERR_PTR(-ENOMEM); |
| |
| adev->name = "reset-mpfs"; |
| adev->dev.parent = clk_dev; |
| adev->dev.release = mpfs_reset_adev_release; |
| adev->id = 666u; |
| |
| ret = auxiliary_device_init(adev); |
| if (ret) { |
| kfree(adev); |
| return ERR_PTR(ret); |
| } |
| |
| return adev; |
| } |
| |
| int mpfs_reset_controller_register(struct device *clk_dev, void __iomem *base) |
| { |
| struct auxiliary_device *adev; |
| int ret; |
| |
| adev = mpfs_reset_adev_alloc(clk_dev); |
| if (IS_ERR(adev)) |
| return PTR_ERR(adev); |
| |
| ret = auxiliary_device_add(adev); |
| if (ret) { |
| auxiliary_device_uninit(adev); |
| return ret; |
| } |
| |
| adev->dev.platform_data = (__force void *)base; |
| |
| return devm_add_action_or_reset(clk_dev, mpfs_reset_unregister_adev, adev); |
| } |
| EXPORT_SYMBOL_NS_GPL(mpfs_reset_controller_register, MCHP_CLK_MPFS); |
| |
| static const struct auxiliary_device_id mpfs_reset_ids[] = { |
| { |
| .name = "reset_mpfs.reset-mpfs", |
| }, |
| { } |
| }; |
| MODULE_DEVICE_TABLE(auxiliary, mpfs_reset_ids); |
| |
| static struct auxiliary_driver mpfs_reset_driver = { |
| .probe = mpfs_reset_probe, |
| .id_table = mpfs_reset_ids, |
| }; |
| |
| module_auxiliary_driver(mpfs_reset_driver); |
| |
| MODULE_DESCRIPTION("Microchip PolarFire SoC Reset Driver"); |
| MODULE_AUTHOR("Conor Dooley <conor.dooley@microchip.com>"); |
| MODULE_IMPORT_NS(MCHP_CLK_MPFS); |