1 /*
2  * Copyright (c) 2018 Intel Corporation.
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/device.h>
8 #include <zephyr/pm/device.h>
9 #include <zephyr/pm/device_runtime.h>
10 #include <zephyr/sys/iterable_sections.h>
11 
12 #include <zephyr/logging/log.h>
13 LOG_MODULE_REGISTER(pm_device, CONFIG_PM_DEVICE_LOG_LEVEL);
14 
15 static const enum pm_device_state action_target_state[] = {
16 	[PM_DEVICE_ACTION_SUSPEND] = PM_DEVICE_STATE_SUSPENDED,
17 	[PM_DEVICE_ACTION_RESUME] = PM_DEVICE_STATE_ACTIVE,
18 	[PM_DEVICE_ACTION_TURN_OFF] = PM_DEVICE_STATE_OFF,
19 	[PM_DEVICE_ACTION_TURN_ON] = PM_DEVICE_STATE_SUSPENDED,
20 };
21 static const enum pm_device_state action_expected_state[] = {
22 	[PM_DEVICE_ACTION_SUSPEND] = PM_DEVICE_STATE_ACTIVE,
23 	[PM_DEVICE_ACTION_RESUME] = PM_DEVICE_STATE_SUSPENDED,
24 	[PM_DEVICE_ACTION_TURN_OFF] = PM_DEVICE_STATE_SUSPENDED,
25 	[PM_DEVICE_ACTION_TURN_ON] = PM_DEVICE_STATE_OFF,
26 };
27 
pm_device_state_str(enum pm_device_state state)28 const char *pm_device_state_str(enum pm_device_state state)
29 {
30 	switch (state) {
31 	case PM_DEVICE_STATE_ACTIVE:
32 		return "active";
33 	case PM_DEVICE_STATE_SUSPENDED:
34 		return "suspended";
35 	case PM_DEVICE_STATE_OFF:
36 		return "off";
37 	default:
38 		return "";
39 	}
40 }
41 
pm_device_action_run(const struct device * dev,enum pm_device_action action)42 int pm_device_action_run(const struct device *dev,
43 			 enum pm_device_action action)
44 {
45 	struct pm_device_base *pm = dev->pm_base;
46 	int ret;
47 
48 	if (pm == NULL) {
49 		return -ENOSYS;
50 	}
51 
52 	/* Validate action against current state */
53 	if (pm->state == action_target_state[action]) {
54 		return -EALREADY;
55 	}
56 	if (pm->state != action_expected_state[action]) {
57 		return -ENOTSUP;
58 	}
59 
60 	ret = pm->action_cb(dev, action);
61 	if (ret < 0) {
62 		/*
63 		 * TURN_ON and TURN_OFF are actions triggered by a power domain
64 		 * when it is resumed or suspended, which means that the energy
65 		 * to the device will be removed or added. For this reason, if
66 		 * the transition fails or the device does not handle these
67 		 * actions its state still needs to updated to reflect its
68 		 * physical behavior.
69 		 *
70 		 * The function will still return the error code so the domain
71 		 * can take whatever action is more appropriated.
72 		 */
73 		switch (action) {
74 		case PM_DEVICE_ACTION_TURN_ON:
75 			/* Store an error flag when the transition explicitly fails */
76 			if (ret != -ENOTSUP) {
77 				atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_TURN_ON_FAILED);
78 			}
79 			__fallthrough;
80 		case PM_DEVICE_ACTION_TURN_OFF:
81 			pm->state = action_target_state[action];
82 			break;
83 		default:
84 			break;
85 		}
86 		return ret;
87 	}
88 
89 	pm->state = action_target_state[action];
90 	/* Power up flags are no longer relevant */
91 	if (action == PM_DEVICE_ACTION_TURN_OFF) {
92 		atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_PD_CLAIMED);
93 		atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_TURN_ON_FAILED);
94 	}
95 
96 	return 0;
97 }
98 
power_domain_add_or_remove(const struct device * dev,const struct device * domain,bool add)99 static int power_domain_add_or_remove(const struct device *dev,
100 				      const struct device *domain,
101 				      bool add)
102 {
103 #if defined(CONFIG_DEVICE_DEPS_DYNAMIC)
104 	device_handle_t *rv = domain->deps;
105 	device_handle_t dev_handle = -1;
106 	size_t i = 0, region = 0;
107 
108 	/*
109 	 * Supported devices are stored as device handle and not
110 	 * device pointers. So, it is necessary to find what is
111 	 * the handle associated to the given device.
112 	 */
113 	STRUCT_SECTION_FOREACH(device, iter_dev) {
114 		if (iter_dev == dev) {
115 			dev_handle = i + 1;
116 			break;
117 		}
118 
119 		i++;
120 	}
121 
122 	/*
123 	 * The last part is to find an available slot in the
124 	 * supported section of handles array and replace it
125 	 * with the device handle.
126 	 */
127 	while (region != 2) {
128 		if (*rv == Z_DEVICE_DEPS_SEP) {
129 			region++;
130 		}
131 		rv++;
132 	}
133 
134 	i = 0;
135 	while (rv[i] != Z_DEVICE_DEPS_ENDS) {
136 		if (add == false) {
137 			if (rv[i] == dev_handle) {
138 				dev->pm_base->domain = NULL;
139 				rv[i] = DEVICE_HANDLE_NULL;
140 				return 0;
141 			}
142 		} else {
143 			if (rv[i] == DEVICE_HANDLE_NULL) {
144 				dev->pm_base->domain = domain;
145 				rv[i] = dev_handle;
146 				return 0;
147 			}
148 		}
149 		++i;
150 	}
151 
152 	return add ? -ENOSPC : -ENOENT;
153 #else
154 	ARG_UNUSED(dev);
155 	ARG_UNUSED(domain);
156 	ARG_UNUSED(add);
157 
158 	return -ENOSYS;
159 #endif
160 }
161 
pm_device_power_domain_remove(const struct device * dev,const struct device * domain)162 int pm_device_power_domain_remove(const struct device *dev,
163 				  const struct device *domain)
164 {
165 	return power_domain_add_or_remove(dev, domain, false);
166 }
167 
pm_device_power_domain_add(const struct device * dev,const struct device * domain)168 int pm_device_power_domain_add(const struct device *dev,
169 			       const struct device *domain)
170 {
171 	return power_domain_add_or_remove(dev, domain, true);
172 }
173 
174 #ifdef CONFIG_DEVICE_DEPS
175 struct pm_visitor_context {
176 	pm_device_action_failed_cb_t failure_cb;
177 	enum pm_device_action action;
178 };
179 
pm_device_children_visitor(const struct device * dev,void * context)180 static int pm_device_children_visitor(const struct device *dev, void *context)
181 {
182 	struct pm_visitor_context *visitor_context = context;
183 	int rc;
184 
185 	rc = pm_device_action_run(dev, visitor_context->action);
186 	if ((visitor_context->failure_cb != NULL) && (rc < 0)) {
187 		/* Stop the iteration if the callback requests it */
188 		if (!visitor_context->failure_cb(dev, rc)) {
189 			return rc;
190 		}
191 	}
192 	return 0;
193 }
194 
pm_device_children_action_run(const struct device * dev,enum pm_device_action action,pm_device_action_failed_cb_t failure_cb)195 void pm_device_children_action_run(const struct device *dev,
196 				   enum pm_device_action action,
197 				   pm_device_action_failed_cb_t failure_cb)
198 {
199 	struct pm_visitor_context visitor_context = {
200 		.failure_cb = failure_cb,
201 		.action = action
202 	};
203 
204 	(void)device_supported_foreach(dev, pm_device_children_visitor, &visitor_context);
205 }
206 #endif
207 
pm_device_state_get(const struct device * dev,enum pm_device_state * state)208 int pm_device_state_get(const struct device *dev,
209 			enum pm_device_state *state)
210 {
211 	struct pm_device_base *pm = dev->pm_base;
212 
213 	if (pm == NULL) {
214 		return -ENOSYS;
215 	}
216 
217 	*state = pm->state;
218 
219 	return 0;
220 }
221 
pm_device_is_any_busy(void)222 bool pm_device_is_any_busy(void)
223 {
224 	const struct device *devs;
225 	size_t devc;
226 
227 	devc = z_device_get_all_static(&devs);
228 
229 	for (const struct device *dev = devs; dev < (devs + devc); dev++) {
230 		struct pm_device_base *pm = dev->pm_base;
231 
232 		if (pm == NULL) {
233 			continue;
234 		}
235 
236 		if (atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY)) {
237 			return true;
238 		}
239 	}
240 
241 	return false;
242 }
243 
pm_device_is_busy(const struct device * dev)244 bool pm_device_is_busy(const struct device *dev)
245 {
246 	struct pm_device_base *pm = dev->pm_base;
247 
248 	if (pm == NULL) {
249 		return false;
250 	}
251 
252 	return atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
253 }
254 
pm_device_busy_set(const struct device * dev)255 void pm_device_busy_set(const struct device *dev)
256 {
257 	struct pm_device_base *pm = dev->pm_base;
258 
259 	if (pm == NULL) {
260 		return;
261 	}
262 
263 	atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
264 }
265 
pm_device_busy_clear(const struct device * dev)266 void pm_device_busy_clear(const struct device *dev)
267 {
268 	struct pm_device_base *pm = dev->pm_base;
269 
270 	if (pm == NULL) {
271 		return;
272 	}
273 
274 	atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
275 }
276 
pm_device_wakeup_enable(const struct device * dev,bool enable)277 bool pm_device_wakeup_enable(const struct device *dev, bool enable)
278 {
279 	atomic_val_t flags, new_flags;
280 	struct pm_device_base *pm = dev->pm_base;
281 
282 	if (pm == NULL) {
283 		return false;
284 	}
285 
286 	flags = atomic_get(&pm->flags);
287 
288 	if ((flags & BIT(PM_DEVICE_FLAG_WS_CAPABLE)) == 0U) {
289 		return false;
290 	}
291 
292 	if (enable) {
293 		new_flags = flags |
294 			    BIT(PM_DEVICE_FLAG_WS_ENABLED);
295 	} else {
296 		new_flags = flags & ~BIT(PM_DEVICE_FLAG_WS_ENABLED);
297 	}
298 
299 	return atomic_cas(&pm->flags, flags, new_flags);
300 }
301 
pm_device_wakeup_is_enabled(const struct device * dev)302 bool pm_device_wakeup_is_enabled(const struct device *dev)
303 {
304 	struct pm_device_base *pm = dev->pm_base;
305 
306 	if (pm == NULL) {
307 		return false;
308 	}
309 
310 	return atomic_test_bit(&pm->flags,
311 			       PM_DEVICE_FLAG_WS_ENABLED);
312 }
313 
pm_device_wakeup_is_capable(const struct device * dev)314 bool pm_device_wakeup_is_capable(const struct device *dev)
315 {
316 	struct pm_device_base *pm = dev->pm_base;
317 
318 	if (pm == NULL) {
319 		return false;
320 	}
321 
322 	return atomic_test_bit(&pm->flags,
323 			       PM_DEVICE_FLAG_WS_CAPABLE);
324 }
325 
pm_device_on_power_domain(const struct device * dev)326 bool pm_device_on_power_domain(const struct device *dev)
327 {
328 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
329 	struct pm_device_base *pm = dev->pm_base;
330 
331 	if (pm == NULL) {
332 		return false;
333 	}
334 	return pm->domain != NULL;
335 #else
336 	ARG_UNUSED(dev);
337 	return false;
338 #endif
339 }
340 
pm_device_is_powered(const struct device * dev)341 bool pm_device_is_powered(const struct device *dev)
342 {
343 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
344 	struct pm_device_base *pm = dev->pm_base;
345 
346 	/* If a device doesn't support PM or is not under a PM domain,
347 	 * assume it is always powered on.
348 	 */
349 	return (pm == NULL) ||
350 	       (pm->domain == NULL) ||
351 	       (pm->domain->pm_base->state == PM_DEVICE_STATE_ACTIVE);
352 #else
353 	ARG_UNUSED(dev);
354 	return true;
355 #endif
356 }
357 
pm_device_driver_init(const struct device * dev,pm_device_action_cb_t action_cb)358 int pm_device_driver_init(const struct device *dev,
359 			  pm_device_action_cb_t action_cb)
360 {
361 	struct pm_device_base *pm = dev->pm_base;
362 	int rc;
363 
364 	/* Work only needs to be performed if the device is powered */
365 	if (!pm_device_is_powered(dev)) {
366 		/* Start in off mode */
367 		pm_device_init_off(dev);
368 		return 0;
369 	}
370 
371 	/* Run power-up logic */
372 	rc = action_cb(dev, PM_DEVICE_ACTION_TURN_ON);
373 	if ((rc < 0) && (rc != -ENOTSUP)) {
374 		return rc;
375 	}
376 
377 	/* If device has no PM structure */
378 	if (pm == NULL) {
379 		/* Device should always be active */
380 		return action_cb(dev, PM_DEVICE_ACTION_RESUME);
381 	}
382 
383 	/* If device will have PM device runtime enabled */
384 	if (IS_ENABLED(CONFIG_PM_DEVICE_RUNTIME) &&
385 	    atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_RUNTIME_AUTO)) {
386 		/* Init into suspend mode.
387 		 * This saves a SUSPENDED->ACTIVE->SUSPENDED cycle.
388 		 */
389 		pm_device_init_suspended(dev);
390 		return 0;
391 	}
392 
393 	/* Startup into active mode */
394 	return action_cb(dev, PM_DEVICE_ACTION_RESUME);
395 }
396