1 /*
2  * Copyright (c) 2023 The ChromiumOS Authors
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #ifndef ZEPHYR_INCLUDE_DRIVERS_ADC_CURRENT_SENSE_AMPLIFIER_H_
8 #define ZEPHYR_INCLUDE_DRIVERS_ADC_CURRENT_SENSE_AMPLIFIER_H_
9 
10 #include <zephyr/drivers/adc.h>
11 #include <zephyr/drivers/gpio.h>
12 
13 struct current_sense_amplifier_dt_spec {
14 	const struct adc_dt_spec port;
15 	uint32_t sense_milli_ohms;
16 	uint16_t sense_gain_mult;
17 	uint16_t sense_gain_div;
18 	struct gpio_dt_spec power_gpio;
19 };
20 
21 /**
22  * @brief Get current sensor information from devicetree.
23  *
24  * This returns a static initializer for a @p current_sense_amplifier_dt_spec structure
25  * given a devicetree node.
26  *
27  * @param node_id Devicetree node identifier.
28  *
29  * @return Static initializer for an current_sense_amplifier_dt_spec structure.
30  */
31 #define CURRENT_SENSE_AMPLIFIER_DT_SPEC_GET(node_id)                                               \
32 	{                                                                                          \
33 		.port = ADC_DT_SPEC_GET(node_id),                                                  \
34 		.sense_milli_ohms = DT_PROP(node_id, sense_resistor_milli_ohms),                   \
35 		.sense_gain_mult = DT_PROP(node_id, sense_gain_mult),                              \
36 		.sense_gain_div = DT_PROP(node_id, sense_gain_div),                                \
37 		.power_gpio = GPIO_DT_SPEC_GET_OR(node_id, power_gpios, {0}),                      \
38 	}
39 
40 /**
41  * @brief Calculates the actual amperage from the measured voltage
42  *
43  * @param[in] spec current sensor specification from Devicetree.
44  * @param[in,out] v_to_i Pointer to the measured voltage in millivolts on input, and the
45  * corresponding scaled current value in milliamps on output.
46  */
47 static inline void
current_sense_amplifier_scale_dt(const struct current_sense_amplifier_dt_spec * spec,int32_t * v_to_i)48 current_sense_amplifier_scale_dt(const struct current_sense_amplifier_dt_spec *spec,
49 				 int32_t *v_to_i)
50 {
51 	/* store in a temporary 64 bit variable to prevent overflow during calculation */
52 	int64_t tmp = *v_to_i;
53 
54 	/* (INT32_MAX * 1000 * UINT16_MAX) < INT64_MAX
55 	 * Therefore all multiplications can be done before divisions, preserving resolution.
56 	 */
57 	tmp = tmp * 1000 * spec->sense_gain_div / spec->sense_milli_ohms / spec->sense_gain_mult;
58 
59 	*v_to_i = (int32_t)tmp;
60 }
61 
62 #endif /* ZEPHYR_INCLUDE_DRIVERS_ADC_CURRENT_SENSE_AMPLIFIER_H_ */
63