ast_wdt.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright 2017 Google, Inc
  4. */
  5. #include <common.h>
  6. #include <dm.h>
  7. #include <errno.h>
  8. #include <wdt.h>
  9. #include <asm/io.h>
  10. #include <asm/arch/wdt.h>
  11. #define WDT_AST2500 2500
  12. #define WDT_AST2400 2400
  13. struct ast_wdt_priv {
  14. struct ast_wdt *regs;
  15. };
  16. static int ast_wdt_start(struct udevice *dev, u64 timeout, ulong flags)
  17. {
  18. struct ast_wdt_priv *priv = dev_get_priv(dev);
  19. ulong driver_data = dev_get_driver_data(dev);
  20. u32 reset_mode = ast_reset_mode_from_flags(flags);
  21. clrsetbits_le32(&priv->regs->ctrl,
  22. WDT_CTRL_RESET_MASK << WDT_CTRL_RESET_MODE_SHIFT,
  23. reset_mode << WDT_CTRL_RESET_MODE_SHIFT);
  24. if (driver_data >= WDT_AST2500 && reset_mode == WDT_CTRL_RESET_SOC)
  25. writel(ast_reset_mask_from_flags(flags),
  26. &priv->regs->reset_mask);
  27. writel((u32) timeout, &priv->regs->counter_reload_val);
  28. writel(WDT_COUNTER_RESTART_VAL, &priv->regs->counter_restart);
  29. /*
  30. * Setting CLK1MHZ bit is just for compatibility with ast2400 part.
  31. * On ast2500 watchdog timer clock is fixed at 1MHz and the bit is
  32. * read-only
  33. */
  34. setbits_le32(&priv->regs->ctrl,
  35. WDT_CTRL_EN | WDT_CTRL_RESET | WDT_CTRL_CLK1MHZ);
  36. return 0;
  37. }
  38. static int ast_wdt_stop(struct udevice *dev)
  39. {
  40. struct ast_wdt_priv *priv = dev_get_priv(dev);
  41. clrbits_le32(&priv->regs->ctrl, WDT_CTRL_EN);
  42. return 0;
  43. }
  44. static int ast_wdt_reset(struct udevice *dev)
  45. {
  46. struct ast_wdt_priv *priv = dev_get_priv(dev);
  47. writel(WDT_COUNTER_RESTART_VAL, &priv->regs->counter_restart);
  48. return 0;
  49. }
  50. static int ast_wdt_expire_now(struct udevice *dev, ulong flags)
  51. {
  52. struct ast_wdt_priv *priv = dev_get_priv(dev);
  53. int ret;
  54. ret = ast_wdt_start(dev, 1, flags);
  55. if (ret)
  56. return ret;
  57. while (readl(&priv->regs->ctrl) & WDT_CTRL_EN)
  58. ;
  59. return ast_wdt_stop(dev);
  60. }
  61. static int ast_wdt_ofdata_to_platdata(struct udevice *dev)
  62. {
  63. struct ast_wdt_priv *priv = dev_get_priv(dev);
  64. priv->regs = devfdt_get_addr_ptr(dev);
  65. if (IS_ERR(priv->regs))
  66. return PTR_ERR(priv->regs);
  67. return 0;
  68. }
  69. static const struct wdt_ops ast_wdt_ops = {
  70. .start = ast_wdt_start,
  71. .reset = ast_wdt_reset,
  72. .stop = ast_wdt_stop,
  73. .expire_now = ast_wdt_expire_now,
  74. };
  75. static const struct udevice_id ast_wdt_ids[] = {
  76. { .compatible = "aspeed,wdt", .data = WDT_AST2500 },
  77. { .compatible = "aspeed,ast2500-wdt", .data = WDT_AST2500 },
  78. { .compatible = "aspeed,ast2400-wdt", .data = WDT_AST2400 },
  79. {}
  80. };
  81. static int ast_wdt_probe(struct udevice *dev)
  82. {
  83. debug("%s() wdt%u\n", __func__, dev->seq);
  84. ast_wdt_stop(dev);
  85. return 0;
  86. }
  87. U_BOOT_DRIVER(ast_wdt) = {
  88. .name = "ast_wdt",
  89. .id = UCLASS_WDT,
  90. .of_match = ast_wdt_ids,
  91. .probe = ast_wdt_probe,
  92. .priv_auto_alloc_size = sizeof(struct ast_wdt_priv),
  93. .ofdata_to_platdata = ast_wdt_ofdata_to_platdata,
  94. .ops = &ast_wdt_ops,
  95. .flags = DM_FLAG_PRE_RELOC,
  96. };