1 /*
2  * Copyright 2022 The Chromium OS Authors
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT zephyr_usb_c_vbus_adc
8 
9 #include <zephyr/logging/log.h>
10 LOG_MODULE_REGISTER(usbc_vbus_adc, CONFIG_USBC_LOG_LEVEL);
11 
12 #include <zephyr/device.h>
13 #include <zephyr/sys/util.h>
14 #include <zephyr/kernel.h>
15 #include <zephyr/drivers/adc.h>
16 #include <zephyr/drivers/usb_c/usbc_pd.h>
17 #include <zephyr/drivers/usb_c/usbc_vbus.h>
18 #include <soc.h>
19 #include <stddef.h>
20 
21 #include "usbc_vbus_adc_priv.h"
22 
23 /**
24  * @brief Reads and returns VBUS measured in mV
25  *
26  * @retval 0 on success
27  * @retval -EIO on failure
28  */
adc_vbus_measure(const struct device * dev,int * meas)29 static int adc_vbus_measure(const struct device *dev, int *meas)
30 {
31 	const struct usbc_vbus_config *const config = dev->config;
32 	struct usbc_vbus_data *data = dev->data;
33 	int value;
34 	int ret;
35 
36 	__ASSERT(meas != NULL, "ADC VBUS meas must not be NULL");
37 
38 	ret = adc_read(config->adc_channel.dev, &data->sequence);
39 	if (ret != 0) {
40 		LOG_INF("ADC reading failed with error %d.", ret);
41 		return ret;
42 	}
43 
44 	value = data->sample;
45 	ret = adc_raw_to_millivolts_dt(&config->adc_channel, &value);
46 	if (ret != 0) {
47 		LOG_INF("Scaling ADC failed with error %d.", ret);
48 		return ret;
49 	}
50 
51 	if (config->full_ohm > 0) {
52 		/* VBUS is scaled down though a voltage divider */
53 		value = (value * 1000) / ((config->output_ohm * 1000) / config->full_ohm);
54 	}
55 	*meas = value;
56 
57 	return 0;
58 }
59 
60 /**
61  * @brief Checks if VBUS is at a particular level
62  *
63  * @retval true if VBUS is at the level voltage, else false
64  */
adc_vbus_check_level(const struct device * dev,enum tc_vbus_level level)65 static bool adc_vbus_check_level(const struct device *dev,
66 				 enum tc_vbus_level level)
67 {
68 	int meas;
69 	int ret;
70 
71 	ret = adc_vbus_measure(dev, &meas);
72 	if (ret) {
73 		return false;
74 	}
75 
76 	switch (level) {
77 	case TC_VBUS_SAFE0V:
78 		return (meas < PD_V_SAFE_0V_MAX_MV);
79 	case TC_VBUS_PRESENT:
80 		return (meas >= PD_V_SAFE_5V_MIN_MV);
81 	case TC_VBUS_REMOVED:
82 		return (meas < TC_V_SINK_DISCONNECT_MAX_MV);
83 	}
84 
85 	return false;
86 }
87 
88 /**
89  * @brief Sets pin to discharge VBUS
90  *
91  * @retval 0 on success
92  * @retval -EIO on failure
93  * @retval -ENOENT if enable pin isn't defined
94  */
adc_vbus_discharge(const struct device * dev,bool enable)95 static int adc_vbus_discharge(const struct device *dev,
96 			      bool enable)
97 {
98 	const struct usbc_vbus_config *const config = dev->config;
99 	const struct gpio_dt_spec *gcd = &config->discharge_gpios;
100 	int ret = -ENOENT;
101 
102 	if (gcd->port) {
103 		ret = gpio_pin_set_dt(gcd, enable);
104 	}
105 	return ret;
106 }
107 
108 /**
109  * @brief Sets pin to enable VBUS measurments
110  *
111  * @retval 0 on success
112  * @retval -EIO on failure
113  * @retval -ENOENT if enable pin isn't defined
114  */
adc_vbus_enable(const struct device * dev,bool enable)115 static int adc_vbus_enable(const struct device *dev,
116 			   bool enable)
117 {
118 	const struct usbc_vbus_config *const config = dev->config;
119 	const struct gpio_dt_spec *gcp = &config->power_gpios;
120 	int ret = -ENOENT;
121 
122 	if (gcp->port) {
123 		ret = gpio_pin_set_dt(gcp, enable);
124 	}
125 	return ret;
126 }
127 
128 /**
129  * @brief Initializes the ADC VBUS Driver
130  *
131  * @retval 0 on success
132  * @retval -EIO on failure
133  */
adc_vbus_init(const struct device * dev)134 static int adc_vbus_init(const struct device *dev)
135 {
136 	const struct usbc_vbus_config *const config = dev->config;
137 	struct usbc_vbus_data *data = dev->data;
138 	const struct gpio_dt_spec *gcp = &config->power_gpios;
139 	const struct gpio_dt_spec *gcd = &config->discharge_gpios;
140 	int ret;
141 
142 	if (!adc_is_ready_dt(&config->adc_channel)) {
143 		LOG_ERR("ADC controller device is not ready");
144 		return -ENODEV;
145 	}
146 
147 	/* Configure VBUS Measurement enable pin if defined */
148 	if (gcp->port) {
149 		if (!device_is_ready(gcp->port)) {
150 			LOG_ERR("%s: device not ready", gcp->port->name);
151 			return -EIO;
152 		}
153 		ret = gpio_pin_configure_dt(gcp, GPIO_OUTPUT_INACTIVE);
154 		if (ret != 0) {
155 			LOG_ERR("Failed to control feed %s.%u: %d",
156 				gcp->port->name, gcp->pin, ret);
157 			return ret;
158 		}
159 	}
160 
161 	/* Configure VBUS Discharge pin if defined */
162 	if (gcd->port) {
163 		if (!device_is_ready(gcd->port)) {
164 			LOG_ERR("%s: device not ready", gcd->port->name);
165 			return -EIO;
166 		}
167 		ret = gpio_pin_configure_dt(gcd, GPIO_OUTPUT_INACTIVE);
168 		if (ret != 0) {
169 			LOG_ERR("Failed to control feed %s.%u: %d",
170 				gcd->port->name, gcd->pin, ret);
171 			return ret;
172 		}
173 
174 	}
175 
176 	data->sequence.buffer = &data->sample;
177 	data->sequence.buffer_size = sizeof(data->sample);
178 
179 	ret = adc_channel_setup_dt(&config->adc_channel);
180 	if (ret != 0) {
181 		LOG_INF("Could not setup channel (%d)\n", ret);
182 		return ret;
183 	}
184 
185 	ret = adc_sequence_init_dt(&config->adc_channel, &data->sequence);
186 	if (ret != 0) {
187 		LOG_INF("Could not init sequence (%d)\n", ret);
188 		return ret;
189 	}
190 
191 	return 0;
192 }
193 
194 static DEVICE_API(usbc_vbus, driver_api) = {
195 	.measure = adc_vbus_measure,
196 	.check_level = adc_vbus_check_level,
197 	.discharge = adc_vbus_discharge,
198 	.enable = adc_vbus_enable
199 };
200 
201 BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(DT_DRV_COMPAT) > 0,
202 	     "No compatible USB-C VBUS Measurement instance found");
203 
204 #define DRIVER_INIT(inst)								\
205 	static struct usbc_vbus_data drv_data_##inst;					\
206 	static const struct usbc_vbus_config drv_config_##inst = {			\
207 		.output_ohm = DT_INST_PROP(inst, output_ohms),				\
208 		.full_ohm = DT_INST_PROP_OR(inst, full_ohms, 0),			\
209 		.adc_channel = ADC_DT_SPEC_INST_GET(inst),				\
210 		.discharge_gpios = GPIO_DT_SPEC_INST_GET_OR(inst, discharge_gpios, {}), \
211 		.power_gpios = GPIO_DT_SPEC_INST_GET_OR(inst, power_gpios, {}),		\
212 	};										\
213 	DEVICE_DT_INST_DEFINE(inst,							\
214 			      &adc_vbus_init,						\
215 			      NULL,							\
216 			      &drv_data_##inst,						\
217 			      &drv_config_##inst,					\
218 			      POST_KERNEL,						\
219 			      CONFIG_USBC_VBUS_INIT_PRIORITY,				\
220 			      &driver_api);
221 
222 DT_INST_FOREACH_STATUS_OKAY(DRIVER_INIT)
223