wdg_acts.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <kernel.h>
  8. #include <drivers/watchdog.h>
  9. #include <soc.h>
  10. #include <board_cfg.h>
  11. #define WD_CTL_EN (1 << 4)
  12. #define WD_CTL_CLKSEL_MASK (0x7 << 1)
  13. #define WD_CTL_CLKSEL_SHIFT (1)
  14. #define WD_CTL_CLKSEL(n) ((n) << WD_CTL_CLKSEL_SHIFT)
  15. #define WD_CTL_CLR (1 << 0)
  16. #define WD_CTL_IRQ_PD (1 << 6)
  17. #define WD_CTL_IRQ_EN (1 << 5)
  18. /* timeout: 176ms * 2^n */
  19. #define WD_TIMEOUT_CLKSEL_TO_MS(n) (176 << (n))
  20. #define WD_TIMEOUT_MS_TO_CLKSEL(ms) (find_msb_set(((ms) + 1) / 176))
  21. static wdt_callback_t g_irq_handler_wdg;
  22. DEVICE_DECLARE(wdg0);
  23. static void wdg_acts_enable(const struct device *dev)
  24. {
  25. ARG_UNUSED(dev);
  26. sys_write32(sys_read32(WD_CTL) | WD_CTL_EN, WD_CTL);
  27. }
  28. static int wdg_acts_disable(const struct device *dev)
  29. {
  30. ARG_UNUSED(dev);
  31. sys_write32(sys_read32(WD_CTL) & ~WD_CTL_EN, WD_CTL);
  32. return 0;
  33. }
  34. static void wdg_acts_clear_irq_pd(void)
  35. {
  36. sys_write32(sys_read32(WD_CTL), WD_CTL);
  37. }
  38. static void wdg_irq_handler(void *arg)
  39. {
  40. wdg_acts_clear_irq_pd();
  41. if (g_irq_handler_wdg)
  42. g_irq_handler_wdg(arg, 0);
  43. }
  44. static int wdg_acts_feed(const struct device *dev, int channel_id)
  45. {
  46. ARG_UNUSED(dev);
  47. ARG_UNUSED(channel_id);
  48. sys_write32(sys_read32(WD_CTL) | WD_CTL_CLR, WD_CTL);
  49. return 0;
  50. }
  51. static int wdg_acts_setup(const struct device *dev, uint8_t options)
  52. {
  53. ARG_UNUSED(options);
  54. wdg_acts_enable(dev);
  55. return 0;
  56. }
  57. static int wdg_acts_install_timeout(const struct device *dev,
  58. const struct wdt_timeout_cfg *cfg)
  59. {
  60. int clksel;
  61. if (cfg->flags != WDT_FLAG_RESET_SOC) {
  62. return -ENOTSUP;
  63. }
  64. if (cfg->window.max == 0U)
  65. return -EINVAL;
  66. ARG_UNUSED(dev);
  67. clksel = WD_TIMEOUT_MS_TO_CLKSEL(cfg->window.max);
  68. if (cfg->callback == NULL) {
  69. g_irq_handler_wdg = NULL;
  70. irq_disable(IRQ_ID_WD2HZ);
  71. sys_write32((sys_read32(WD_CTL) & ~WD_CTL_CLKSEL_MASK)
  72. | WD_CTL_CLKSEL(clksel), WD_CTL);
  73. sys_write32((sys_read32(WD_CTL) & ~WD_CTL_IRQ_EN), WD_CTL);
  74. } else {
  75. g_irq_handler_wdg = cfg->callback;
  76. IRQ_CONNECT(IRQ_ID_WD2HZ, CONFIG_WDT_0_IRQ_PRI, wdg_irq_handler, DEVICE_GET(wdg0), 0);
  77. irq_enable(IRQ_ID_WD2HZ);
  78. sys_write32((sys_read32(WD_CTL) & ~WD_CTL_CLKSEL_MASK)
  79. | WD_CTL_CLKSEL(clksel) | WD_CTL_IRQ_EN, WD_CTL);
  80. }
  81. return 0;
  82. }
  83. static const struct wdt_driver_api wdg_acts_driver_api = {
  84. .setup = wdg_acts_setup,
  85. .disable = wdg_acts_disable,
  86. .install_timeout = wdg_acts_install_timeout,
  87. .feed = wdg_acts_feed,
  88. };
  89. static int wdg_acts_init(const struct device *dev)
  90. {
  91. //sys_write32(0x0, WD_CTL);
  92. #ifdef CONFIG_WDT_ACTS_START_AT_BOOT
  93. wdg_acts_enable(dev);
  94. #endif
  95. return 0;
  96. }
  97. DEVICE_DEFINE(wdg0, CONFIG_WDT_ACTS_NAME, wdg_acts_init, NULL, NULL, NULL,
  98. PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEVICE,
  99. &wdg_acts_driver_api);