summaryrefslogtreecommitdiff
path: root/zephyr/shim/src/adc.c
blob: c6fbed23c52a31e3f127f05bce84e6a9abea9a61 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/* Copyright 2021 The Chromium OS Authors. All rights reserved.
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */

#include <drivers/adc.h>
#include "adc.h"
#include "zephyr_adc.h"

#define ADC_DEV DT_LABEL(DT_NODELABEL(adc0))
const struct device *adc_dev;

#define HAS_NAMED_ADC_CHANNELS DT_NODE_EXISTS(DT_INST(0, named_adc_channels))

#if HAS_NAMED_ADC_CHANNELS
#define ADC_CHANNEL_COMMA(node_id)                                      \
	[ZSHIM_ADC_ID(node_id)] = {                                     \
		.name = DT_LABEL(node_id),                              \
		.input_ch = DT_PROP(node_id, channel),                  \
		.factor_mul = DT_PROP(node_id, mul),                    \
		.factor_div = DT_PROP(node_id, div),                    \
		.channel_cfg = {                                        \
			.channel_id = DT_PROP(node_id, channel),        \
			.gain = DT_ENUM_TOKEN(node_id, gain),           \
			.reference = DT_ENUM_TOKEN(node_id, reference), \
			.acquisition_time =                             \
				DT_PROP(node_id, acquisition_time),     \
			.differential = DT_PROP(node_id, differential), \
		},                                                      \
	},
const struct adc_t adc_channels[] = { DT_FOREACH_CHILD(
	DT_INST(0, named_adc_channels), ADC_CHANNEL_COMMA) };
#endif /* named_adc_channels */

static int init_device_bindings(const struct device *device)
{
	ARG_UNUSED(device);
	adc_dev = device_get_binding(ADC_DEV);

#if HAS_NAMED_ADC_CHANNELS
	for (int i = 0; i < ARRAY_SIZE(adc_channels); i++)
		adc_channel_setup(adc_dev, &adc_channels[i].channel_cfg);
#endif

	return 0;
}
SYS_INIT(init_device_bindings, POST_KERNEL, 51);

int adc_read_channel(enum adc_channel ch)
{
	int ret = 0, rv;
	struct adc_sequence seq = {
		.options = NULL,
		.channels = BIT(adc_channels[ch].input_ch),
		.buffer = &ret,
		.buffer_size = sizeof(ret),
		.resolution = CONFIG_PLATFORM_EC_ADC_RESOLUTION,
		.oversampling = CONFIG_PLATFORM_EC_ADC_OVERSAMPLING,
		.calibrate = false,
	};

	rv = adc_read(adc_dev, &seq);
	if (rv)
		return rv;

	adc_raw_to_millivolts(adc_ref_internal(adc_dev), ADC_GAIN_1,
			      CONFIG_PLATFORM_EC_ADC_RESOLUTION, &ret);
	ret = (ret * adc_channels[ch].factor_mul) / adc_channels[ch].factor_div;
	return ret;
}