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