1 /*
2  * Copyright (c) 2022, Commonwealth Scientific and Industrial Research
3  * Organisation (CSIRO) ABN 41 687 119 230.
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #define DT_DRV_COMPAT power_domain_gpio
9 
10 #include <zephyr/kernel.h>
11 #include <zephyr/drivers/gpio.h>
12 #include <zephyr/pm/device.h>
13 #include <zephyr/pm/device_runtime.h>
14 
15 #include <zephyr/logging/log.h>
16 LOG_MODULE_REGISTER(power_domain_gpio, CONFIG_POWER_DOMAIN_LOG_LEVEL);
17 
18 struct pd_gpio_config {
19 	struct gpio_dt_spec enable;
20 	uint32_t startup_delay_us;
21 	uint32_t off_on_delay_us;
22 };
23 
24 struct pd_gpio_data {
25 	k_timeout_t next_boot;
26 };
27 
28 struct pd_visitor_context {
29 	const struct device *domain;
30 	enum pm_device_action action;
31 };
32 
33 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
34 
pd_on_domain_visitor(const struct device * dev,void * context)35 static int pd_on_domain_visitor(const struct device *dev, void *context)
36 {
37 	struct pd_visitor_context *visitor_context = context;
38 
39 	/* Only run action if the device is on the specified domain */
40 	if (!dev->pm || (dev->pm->domain != visitor_context->domain)) {
41 		return 0;
42 	}
43 
44 	(void)pm_device_action_run(dev, visitor_context->action);
45 	return 0;
46 }
47 
48 #endif
49 
pd_gpio_pm_action(const struct device * dev,enum pm_device_action action)50 static int pd_gpio_pm_action(const struct device *dev,
51 			     enum pm_device_action action)
52 {
53 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
54 	struct pd_visitor_context context = {.domain = dev};
55 #endif
56 	const struct pd_gpio_config *cfg = dev->config;
57 	struct pd_gpio_data *data = dev->data;
58 	int64_t next_boot_ticks;
59 	int rc = 0;
60 
61 	/* Validate that blocking API's can be used */
62 	if (!k_can_yield()) {
63 		LOG_ERR("Blocking actions cannot run in this context");
64 		return -ENOTSUP;
65 	}
66 
67 	switch (action) {
68 	case PM_DEVICE_ACTION_RESUME:
69 		/* Wait until we can boot again */
70 		k_sleep(data->next_boot);
71 		/* Switch power on */
72 		gpio_pin_set_dt(&cfg->enable, 1);
73 		LOG_INF("%s is now ON", dev->name);
74 		/* Wait for domain to come up */
75 		k_sleep(K_USEC(cfg->startup_delay_us));
76 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
77 		/* Notify devices on the domain they are now powered */
78 		context.action = PM_DEVICE_ACTION_TURN_ON;
79 		(void)device_supported_foreach(dev, pd_on_domain_visitor, &context);
80 #endif
81 		break;
82 	case PM_DEVICE_ACTION_SUSPEND:
83 #ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
84 		/* Notify devices on the domain that power is going down */
85 		context.action = PM_DEVICE_ACTION_TURN_OFF;
86 		(void)device_supported_foreach(dev, pd_on_domain_visitor, &context);
87 #endif
88 		/* Switch power off */
89 		gpio_pin_set_dt(&cfg->enable, 0);
90 		LOG_INF("%s is now OFF", dev->name);
91 		/* Store next time we can boot */
92 		next_boot_ticks = k_uptime_ticks() + k_us_to_ticks_ceil32(cfg->off_on_delay_us);
93 		data->next_boot = K_TIMEOUT_ABS_TICKS(next_boot_ticks);
94 		break;
95 	case PM_DEVICE_ACTION_TURN_ON:
96 		/* Actively control the enable pin now that the device is powered */
97 		gpio_pin_configure_dt(&cfg->enable, GPIO_OUTPUT_INACTIVE);
98 		LOG_DBG("%s is OFF and powered", dev->name);
99 		break;
100 	case PM_DEVICE_ACTION_TURN_OFF:
101 		/* Let the enable pin float while device is not powered */
102 		gpio_pin_configure_dt(&cfg->enable, GPIO_DISCONNECTED);
103 		LOG_DBG("%s is OFF and not powered", dev->name);
104 		break;
105 	default:
106 		rc = -ENOTSUP;
107 	}
108 
109 	return rc;
110 }
111 
pd_gpio_init(const struct device * dev)112 static int pd_gpio_init(const struct device *dev)
113 {
114 	const struct pd_gpio_config *cfg = dev->config;
115 	struct pd_gpio_data *data = dev->data;
116 
117 	if (!gpio_is_ready_dt(&cfg->enable)) {
118 		LOG_ERR("GPIO port %s is not ready", cfg->enable.port->name);
119 		return -ENODEV;
120 	}
121 	/* We can't know how long the domain has been off for before boot */
122 	data->next_boot = K_TIMEOUT_ABS_US(cfg->off_on_delay_us);
123 
124 	/* Boot according to state */
125 	return pm_device_driver_init(dev, pd_gpio_pm_action);
126 }
127 
128 #define POWER_DOMAIN_DEVICE(id)						   \
129 	static const struct pd_gpio_config pd_gpio_##id##_cfg = {	   \
130 		.enable = GPIO_DT_SPEC_INST_GET(id, enable_gpios),	   \
131 		.startup_delay_us = DT_INST_PROP(id, startup_delay_us),	   \
132 		.off_on_delay_us = DT_INST_PROP(id, off_on_delay_us),	   \
133 	};								   \
134 	static struct pd_gpio_data pd_gpio_##id##_data;			   \
135 	PM_DEVICE_DT_INST_DEFINE(id, pd_gpio_pm_action);		   \
136 	DEVICE_DT_INST_DEFINE(id, pd_gpio_init, PM_DEVICE_DT_INST_GET(id), \
137 			      &pd_gpio_##id##_data, &pd_gpio_##id##_cfg,   \
138 			      POST_KERNEL, 75,				   \
139 			      NULL);
140 
141 DT_INST_FOREACH_STATUS_OKAY(POWER_DOMAIN_DEVICE)
142