1 /*
2 * Copyright (c) 2021 BrainCo Inc.
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT gd_gd32_dac
8
9 #include <errno.h>
10
11 #include <zephyr/drivers/clock_control.h>
12 #include <zephyr/drivers/clock_control/gd32.h>
13 #include <zephyr/drivers/pinctrl.h>
14 #include <zephyr/drivers/reset.h>
15 #include <zephyr/drivers/dac.h>
16
17 #include <gd32_dac.h>
18
19 #include <zephyr/logging/log.h>
20 LOG_MODULE_REGISTER(dac_gd32, CONFIG_DAC_LOG_LEVEL);
21
22 /**
23 * For some gd32 series which only have 1 DAC, their HAL name may not same as others.
24 * Below definitions help to unify the HAL name.
25 */
26 #if defined(CONFIG_SOC_SERIES_GD32A50X)
27 #define DAC_CTL_DEN0 DAC_CTL_DEN
28 #define DAC0_R8DH OUT_R8DH
29 #define DAC0_R12DH OUT_R12DH
30 #elif defined(CONFIG_SOC_SERIES_GD32F3X0)
31 #define DAC_CTL_DEN0 DAC_CTL_DEN
32 #define DAC0_R8DH DAC_R8DH
33 #define DAC0_R12DH DAC_R12DH
34 #endif
35
36 struct dac_gd32_config {
37 uint32_t reg;
38 uint16_t clkid;
39 struct reset_dt_spec reset;
40 const struct pinctrl_dev_config *pcfg;
41 uint32_t num_channels;
42 uint32_t reset_val;
43 };
44
45 struct dac_gd32_data {
46 uint8_t resolutions[2];
47 };
48
dac_gd32_enable(uint8_t dacx)49 static void dac_gd32_enable(uint8_t dacx)
50 {
51 switch (dacx) {
52 case 0U:
53 DAC_CTL |= DAC_CTL_DEN0;
54 break;
55 #if DT_INST_PROP(0, num_channels) == 2
56 case 1U:
57 DAC_CTL |= DAC_CTL_DEN1;
58 break;
59 #endif
60 }
61 }
62
dac_gd32_disable(uint8_t dacx)63 static void dac_gd32_disable(uint8_t dacx)
64 {
65 switch (dacx) {
66 case 0U:
67 DAC_CTL &= ~DAC_CTL_DEN0;
68 break;
69 #if DT_INST_PROP(0, num_channels) == 2
70 case 1U:
71 DAC_CTL &= ~DAC_CTL_DEN1;
72 break;
73 #endif
74 }
75 }
76
dac_gd32_write(struct dac_gd32_data * data,uint8_t dacx,uint32_t value)77 static void dac_gd32_write(struct dac_gd32_data *data,
78 uint8_t dacx, uint32_t value)
79 {
80 switch (dacx) {
81 case 0U:
82 if (data->resolutions[dacx] == 8U) {
83 DAC0_R8DH = value;
84 } else {
85 DAC0_R12DH = value;
86 }
87 break;
88 #if DT_INST_PROP(0, num_channels) == 2
89 case 1U:
90 if (data->resolutions[dacx] == 8U) {
91 DAC1_R8DH = value;
92 } else {
93 DAC1_R12DH = value;
94 }
95 break;
96 #endif
97 }
98 }
99
dac_gd32_channel_setup(const struct device * dev,const struct dac_channel_cfg * channel_cfg)100 static int dac_gd32_channel_setup(const struct device *dev,
101 const struct dac_channel_cfg *channel_cfg)
102 {
103 struct dac_gd32_data *data = dev->data;
104 const struct dac_gd32_config *config = dev->config;
105 uint8_t dacx = channel_cfg->channel_id;
106
107 if (dacx >= config->num_channels) {
108 return -ENOTSUP;
109 }
110
111 /* GD32 DAC only support 8 or 12 bits resolution */
112 if ((channel_cfg->resolution != 8U) &&
113 (channel_cfg->resolution != 12U)) {
114 LOG_ERR("Only 8 and 12 bits resolutions are supported!");
115 return -ENOTSUP;
116 }
117
118 if (channel_cfg->internal) {
119 LOG_ERR("Internal channels not supported");
120 return -ENOTSUP;
121 }
122
123 data->resolutions[dacx] = channel_cfg->resolution;
124
125 dac_gd32_disable(dacx);
126 dac_gd32_write(data, dacx, config->reset_val);
127 dac_gd32_enable(dacx);
128
129 return 0;
130 }
131
dac_gd32_write_value(const struct device * dev,uint8_t dacx,uint32_t value)132 static int dac_gd32_write_value(const struct device *dev,
133 uint8_t dacx, uint32_t value)
134 {
135 struct dac_gd32_data *data = dev->data;
136 const struct dac_gd32_config *config = dev->config;
137
138 if (dacx >= config->num_channels) {
139 return -ENOTSUP;
140 }
141
142 dac_gd32_write(data, dacx, value);
143
144 return 0;
145 }
146
147 DEVICE_API(dac, dac_gd32_driver_api) = {
148 .channel_setup = dac_gd32_channel_setup,
149 .write_value = dac_gd32_write_value
150 };
151
dac_gd32_init(const struct device * dev)152 static int dac_gd32_init(const struct device *dev)
153 {
154 const struct dac_gd32_config *cfg = dev->config;
155 int ret;
156
157 ret = pinctrl_apply_state(cfg->pcfg, PINCTRL_STATE_DEFAULT);
158 if (ret < 0) {
159 LOG_ERR("Failed to apply pinctrl state");
160 return ret;
161 }
162
163 (void)clock_control_on(GD32_CLOCK_CONTROLLER,
164 (clock_control_subsys_t)&cfg->clkid);
165
166 (void)reset_line_toggle_dt(&cfg->reset);
167
168 return 0;
169 }
170
171 PINCTRL_DT_INST_DEFINE(0);
172
173 static struct dac_gd32_data dac_gd32_data_0;
174
175 static const struct dac_gd32_config dac_gd32_cfg_0 = {
176 .reg = DT_INST_REG_ADDR(0),
177 .clkid = DT_INST_CLOCKS_CELL(0, id),
178 .reset = RESET_DT_SPEC_INST_GET(0),
179 .pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(0),
180 .num_channels = DT_INST_PROP(0, num_channels),
181 .reset_val = DT_INST_PROP(0, reset_val),
182 };
183
184 DEVICE_DT_INST_DEFINE(0, &dac_gd32_init, NULL, &dac_gd32_data_0,
185 &dac_gd32_cfg_0, POST_KERNEL, CONFIG_DAC_INIT_PRIORITY,
186 &dac_gd32_driver_api);
187