1 /*
2 * Copyright (c) 2022 Nordic Semiconductor ASA
3 * SPDX-License-Identifier: Apache-2.0
4 */
5
6 #define DT_DRV_COMPAT nordic_npm6001_wdt
7
8 #include <errno.h>
9
10 #include <zephyr/device.h>
11 #include <zephyr/drivers/i2c.h>
12 #include <zephyr/drivers/watchdog.h>
13 #include <zephyr/sys/util_macro.h>
14 #include <zephyr/toolchain.h>
15
16 /* nPM6001 Watchdog related registers */
17 #define NPM6001_WDARMEDVALUE 0x54U
18 #define NPM6001_WDARMEDSTROBE 0x55U
19 #define NPM6001_WDTRIGGERVALUE0 0x56U
20 #define NPM6001_WDTRIGGERVALUE1 0x57U
21 #define NPM6001_WDTRIGGERVALUE2 0x58U
22 #define NPM6001_WDDATASTROBE 0x5DU
23 #define NPM6001_WDPWRUPVALUE 0x5EU
24 #define NPM6001_WDPWRUPSTROBE 0x5FU
25 #define NPM6001_WDKICK 0x60U
26 #define NPM6001_WDREQPOWERDOWN 0x62U
27
28 /* nPM6001 WDTRIGGERVALUEx ms/LSB, min/max values */
29 #define NPM6001_WDTRIGGERVALUE_MS_LSB 4000U
30 #define NPM6001_WDTRIGGERVALUE_MIN 0x2U
31 #define NPM6001_WDTRIGGERVALUE_MAX 0xFFFFFFU
32
33 /* nPM6001 WDPWRUPVALUE fields */
34 #define NPM6001_WDPWRUPVALUE_OSC_ENABLE BIT(0)
35 #define NPM6001_WDPWRUPVALUE_COUNTER_ENABLE BIT(1)
36 #define NPM6001_WDPWRUPVALUE_LS_ENABLE BIT(2)
37
38 struct wdt_npm6001_config {
39 struct i2c_dt_spec bus;
40 };
41
wdt_npm6001_setup(const struct device * dev,uint8_t options)42 static int wdt_npm6001_setup(const struct device *dev, uint8_t options)
43 {
44 ARG_UNUSED(dev);
45 ARG_UNUSED(options);
46
47 return 0;
48 }
49
wdt_npm6001_disable(const struct device * dev)50 static int wdt_npm6001_disable(const struct device *dev)
51 {
52 const struct wdt_npm6001_config *config = dev->config;
53 uint8_t buf[4] = {NPM6001_WDARMEDVALUE, 1U, NPM6001_WDARMEDSTROBE, 1U};
54
55 return i2c_write_dt(&config->bus, buf, sizeof(buf));
56 }
57
wdt_npm6001_install_timeout(const struct device * dev,const struct wdt_timeout_cfg * timeout)58 static int wdt_npm6001_install_timeout(const struct device *dev,
59 const struct wdt_timeout_cfg *timeout)
60 {
61 const struct wdt_npm6001_config *config = dev->config;
62 uint32_t window;
63 uint8_t buf[2];
64 int ret;
65
66 if (timeout->window.min != 0U) {
67 return -EINVAL;
68 }
69
70 /* round-up timeout, e.g. 5s -> 8s */
71 window = (((timeout->window.max + NPM6001_WDTRIGGERVALUE_MS_LSB - 1U) /
72 NPM6001_WDTRIGGERVALUE_MS_LSB) +
73 1U);
74 if ((window < NPM6001_WDTRIGGERVALUE_MIN) ||
75 (window > NPM6001_WDTRIGGERVALUE_MAX)) {
76 return -EINVAL;
77 }
78
79 /* enable OSC/COUNTER/LS */
80 buf[0] = NPM6001_WDPWRUPVALUE;
81 buf[1] = NPM6001_WDPWRUPVALUE_OSC_ENABLE |
82 NPM6001_WDPWRUPVALUE_COUNTER_ENABLE |
83 NPM6001_WDPWRUPVALUE_LS_ENABLE;
84 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
85 if (ret < 0) {
86 return ret;
87 }
88
89 buf[0] = NPM6001_WDPWRUPSTROBE;
90 buf[1] = 1U;
91 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
92 if (ret < 0) {
93 return ret;
94 }
95
96 /* write trigger value */
97 buf[0] = NPM6001_WDTRIGGERVALUE0;
98 buf[1] = (uint8_t)window;
99 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
100 if (ret < 0) {
101 return ret;
102 }
103
104 buf[0] = NPM6001_WDTRIGGERVALUE1;
105 buf[1] = (uint8_t)(window >> 8U);
106 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
107 if (ret < 0) {
108 return ret;
109 }
110
111 buf[0] = NPM6001_WDTRIGGERVALUE2;
112 buf[1] = (uint8_t)(window >> 16U);
113 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
114 if (ret < 0) {
115 return ret;
116 }
117
118 buf[0] = NPM6001_WDDATASTROBE;
119 buf[1] = 1U;
120 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
121 if (ret < 0) {
122 return ret;
123 }
124
125 /* arm watchdog & kick */
126 buf[0] = NPM6001_WDARMEDVALUE;
127 buf[1] = 1U;
128 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
129 if (ret < 0) {
130 return ret;
131 }
132
133 buf[0] = NPM6001_WDARMEDSTROBE;
134 buf[1] = 1U;
135 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
136 if (ret < 0) {
137 return ret;
138 }
139
140 buf[0] = NPM6001_WDKICK;
141 buf[1] = 1U;
142 ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
143 if (ret < 0) {
144 return ret;
145 }
146
147 return 0;
148 }
149
wdt_npm6001_feed(const struct device * dev,int channel_id)150 static int wdt_npm6001_feed(const struct device *dev, int channel_id)
151 {
152 const struct wdt_npm6001_config *config = dev->config;
153 uint8_t buf[2] = {NPM6001_WDKICK, 1U};
154
155 ARG_UNUSED(channel_id);
156
157 return i2c_write_dt(&config->bus, buf, sizeof(buf));
158 }
159
160 static DEVICE_API(wdt, wdt_npm6001_api) = {
161 .setup = wdt_npm6001_setup,
162 .disable = wdt_npm6001_disable,
163 .install_timeout = wdt_npm6001_install_timeout,
164 .feed = wdt_npm6001_feed,
165 };
166
wdt_npm6001_init(const struct device * dev)167 static int wdt_npm6001_init(const struct device *dev)
168 {
169 const struct wdt_npm6001_config *config = dev->config;
170
171 if (!device_is_ready(config->bus.bus)) {
172 return -ENODEV;
173 }
174
175 return 0;
176 }
177
178 #define WDT_NPM6001_DEFINE(n) \
179 static const struct wdt_npm6001_config wdt_npm6001_config##n = { \
180 .bus = I2C_DT_SPEC_GET(DT_INST_PARENT(n)), \
181 }; \
182 \
183 DEVICE_DT_INST_DEFINE(n, &wdt_npm6001_init, NULL, NULL, \
184 &wdt_npm6001_config##n, POST_KERNEL, \
185 CONFIG_WDT_NPM6001_INIT_PRIORITY, \
186 &wdt_npm6001_api);
187
188 DT_INST_FOREACH_STATUS_OKAY(WDT_NPM6001_DEFINE)
189