stm32-reset.c 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2017, STMicroelectronics - All Rights Reserved
  4. * Author(s): Patrice Chotard, <patrice.chotard@foss.st.com> for STMicroelectronics.
  5. */
  6. #define LOG_CATEGORY UCLASS_RESET
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <errno.h>
  10. #include <log.h>
  11. #include <malloc.h>
  12. #include <reset-uclass.h>
  13. #include <stm32_rcc.h>
  14. #include <asm/io.h>
  15. #include <dm/device_compat.h>
  16. #include <linux/bitops.h>
  17. /* offset of register without set/clear management */
  18. #define RCC_MP_GCR_OFFSET 0x10C
  19. /* reset clear offset for STM32MP RCC */
  20. #define RCC_CL 0x4
  21. struct stm32_reset_priv {
  22. fdt_addr_t base;
  23. };
  24. static int stm32_reset_assert(struct reset_ctl *reset_ctl)
  25. {
  26. struct stm32_reset_priv *priv = dev_get_priv(reset_ctl->dev);
  27. int bank = (reset_ctl->id / (sizeof(u32) * BITS_PER_BYTE)) * 4;
  28. int offset = reset_ctl->id % (sizeof(u32) * BITS_PER_BYTE);
  29. dev_dbg(reset_ctl->dev, "reset id = %ld bank = %d offset = %d)\n",
  30. reset_ctl->id, bank, offset);
  31. if (dev_get_driver_data(reset_ctl->dev) == STM32MP1)
  32. if (bank != RCC_MP_GCR_OFFSET)
  33. /* reset assert is done in rcc set register */
  34. writel(BIT(offset), priv->base + bank);
  35. else
  36. clrbits_le32(priv->base + bank, BIT(offset));
  37. else
  38. setbits_le32(priv->base + bank, BIT(offset));
  39. return 0;
  40. }
  41. static int stm32_reset_deassert(struct reset_ctl *reset_ctl)
  42. {
  43. struct stm32_reset_priv *priv = dev_get_priv(reset_ctl->dev);
  44. int bank = (reset_ctl->id / (sizeof(u32) * BITS_PER_BYTE)) * 4;
  45. int offset = reset_ctl->id % (sizeof(u32) * BITS_PER_BYTE);
  46. dev_dbg(reset_ctl->dev, "reset id = %ld bank = %d offset = %d)\n",
  47. reset_ctl->id, bank, offset);
  48. if (dev_get_driver_data(reset_ctl->dev) == STM32MP1)
  49. if (bank != RCC_MP_GCR_OFFSET)
  50. /* reset deassert is done in rcc clr register */
  51. writel(BIT(offset), priv->base + bank + RCC_CL);
  52. else
  53. setbits_le32(priv->base + bank, BIT(offset));
  54. else
  55. clrbits_le32(priv->base + bank, BIT(offset));
  56. return 0;
  57. }
  58. static const struct reset_ops stm32_reset_ops = {
  59. .rst_assert = stm32_reset_assert,
  60. .rst_deassert = stm32_reset_deassert,
  61. };
  62. static int stm32_reset_probe(struct udevice *dev)
  63. {
  64. struct stm32_reset_priv *priv = dev_get_priv(dev);
  65. priv->base = dev_read_addr(dev);
  66. if (priv->base == FDT_ADDR_T_NONE) {
  67. /* for MFD, get address of parent */
  68. priv->base = dev_read_addr(dev->parent);
  69. if (priv->base == FDT_ADDR_T_NONE)
  70. return -EINVAL;
  71. }
  72. return 0;
  73. }
  74. U_BOOT_DRIVER(stm32_rcc_reset) = {
  75. .name = "stm32_rcc_reset",
  76. .id = UCLASS_RESET,
  77. .probe = stm32_reset_probe,
  78. .priv_auto = sizeof(struct stm32_reset_priv),
  79. .ops = &stm32_reset_ops,
  80. };