forked from CodethinkLabs/mt3620-m4-drivers
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathGPIO.c
268 lines (221 loc) · 8.4 KB
/
GPIO.c
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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
/* Copyright (c) Codethink Ltd. All rights reserved.
Licensed under the MIT License. */
#include <stdbool.h>
#include <stdint.h>
#include <stdlib.h>
#include "GPIO.h"
#include "NVIC.h"
#include "mt3620/adc.h"
#include "mt3620/gpio.h"
#include "mt3620/irq.h"
static mt3620_gpio_block pinToBlock(uint32_t pin)
{
if (pin > MT3620_GPIO_COUNT) {
return MT3620_GPIO_BLOCK_NOT_MAPPED;
}
return mt3620_gpioPinMap[pin];
}
static uint32_t getPinMask(uint32_t pin, mt3620_gpio_block block)
{
if (block > MT3620_GPIO_BLOCK_NOT_MAPPED) {
return 0U;
}
return (1U << (pin - mt3620_gpioBlockStart[block]));
}
static int32_t ConfigurePin(uint32_t pin, bool asInput)
{
mt3620_gpio_block block = pinToBlock(pin);
if (block >= MT3620_GPIO_BLOCK_NOT_MAPPED) {
return ERROR_GPIO_NOT_A_PIN;
}
uint32_t pinMask = getPinMask(pin, block);
if (asInput) {
mt3620_gpio[block]->gpio_pwm_grp_ies_set = pinMask;
mt3620_gpio[block]->gpio_pwm_grp_oe_reset = pinMask;
} else {
mt3620_gpio[block]->gpio_pwm_grp_oe_set = pinMask;
mt3620_gpio[block]->gpio_pwm_grp_ies_reset = pinMask;
}
return ERROR_NONE;
}
int32_t GPIO_ConfigurePinForOutput(uint32_t pin)
{
return ConfigurePin(pin, false);
}
int32_t GPIO_ConfigurePinForInput(uint32_t pin)
{
return ConfigurePin(pin, true);
}
int32_t GPIO_Write(uint32_t pin, bool state)
{
mt3620_gpio_block block = pinToBlock(pin);
if (block >= MT3620_GPIO_BLOCK_NOT_MAPPED) {
return ERROR_GPIO_NOT_A_PIN;
}
uint32_t pinMask = getPinMask(pin, block);
if (state) {
mt3620_gpio[block]->gpio_pwm_grp_dout_set = pinMask;
} else {
mt3620_gpio[block]->gpio_pwm_grp_dout_reset = pinMask;
}
return ERROR_NONE;
}
int32_t GPIO_Read(uint32_t pin, bool *state)
{
mt3620_gpio_block block = pinToBlock(pin);
if (block >= MT3620_GPIO_BLOCK_NOT_MAPPED) {
return ERROR_GPIO_NOT_A_PIN;
}
uint32_t pinMask = getPinMask(pin, block);
/* The GPIO register map for ISU is undocumented, but it looks like the DIN
* offset is at 0xc, as opposed to 0x4 (for GPIO). Similarly, the I2S GPIO
* DIN offset is also different, but at 0x0.*/
if (block < MT3620_GPIO_BLOCK_ISU_0) {
*state = pinMask & mt3620_gpio[block]->gpio_pwm_grp_din;
}
else if (block < MT3620_GPIO_BLOCK_I2S_0) {
*state = pinMask & mt3620_gpio[block]->gpio_pwm_grp_din_isu;
}
else {
*state = pinMask & mt3620_gpio[block]->gpio_pwm_grp_global_ctrl__din_i2s;
}
return ERROR_NONE;
}
#define PWM_MAX_DUTY_CYCLE 65535
#define PWM_CLOCK_SEL_DEADZONE 5
int32_t PWM_ConfigurePin(uint32_t pin, uint32_t clockFrequency, uint32_t onTime, uint32_t offTime)
{
mt3620_gpio_block block = pinToBlock(pin);
if ((block < MT3620_GPIO_BLOCK_0) || (block > MT3620_GPIO_BLOCK_2)) {
return ERROR_PWM_NOT_A_PIN;
}
uint32_t pinMask = getPinMask(pin, block);
uint32_t pwmBlock = block - MT3620_GPIO_BLOCK_0;
if (onTime > PWM_MAX_DUTY_CYCLE || offTime > PWM_MAX_DUTY_CYCLE) {
return ERROR_PWM_UNSUPPORTED_DUTY_CYCLE;
}
uint8_t clockSel;
unsigned frequencyLow = (clockFrequency * (100ULL - PWM_CLOCK_SEL_DEADZONE)) / 100U;
unsigned frequencyHigh = (clockFrequency * (100ULL + PWM_CLOCK_SEL_DEADZONE)) / 100U;
if (frequencyLow <= MT3620_PWM_32k && frequencyHigh >= MT3620_PWM_32k ) {
clockSel = MT3620_PWM_CLK_SEL_32K;
}
else if (frequencyLow <= MT3620_PWM_2M && frequencyHigh >= MT3620_PWM_2M) {
clockSel = MT3620_PWM_CLK_SEL_2M;
}
else if (frequencyLow <= MT3620_PWM_XTAL && frequencyHigh >= MT3620_PWM_XTAL) {
clockSel = MT3620_PWM_CLK_SEL_XTAL;
} else {
return ERROR_PWM_UNSUPPORTED_CLOCK_SEL;
}
//Write default values of the registers before starting as recommended in the datasheet
mt3620_pwm[pwmBlock]->pwm_glo_ctrl = MT3620_PWM_GLO_CTRL_DEF;
//Switch statement starts from 1 since pinMask = pwm pwmBlock + 1
switch(pinMask) {
case 1:
mt3620_pwm[pwmBlock]->pwm0_ctrl = MT3620_PWM_CTRL_DEF;
mt3620_pwm[pwmBlock]->pwm0_param_s0 = MT3620_PWM_PARAM_S0_DEF;
mt3620_pwm[pwmBlock]->pwm0_param_s1 = MT3620_PWM_PARAM_S1_DEF;
break;
case 2:
mt3620_pwm[pwmBlock]->pwm1_ctrl = MT3620_PWM_CTRL_DEF;
mt3620_pwm[pwmBlock]->pwm1_param_s0 = MT3620_PWM_PARAM_S0_DEF;
mt3620_pwm[pwmBlock]->pwm1_param_s1 = MT3620_PWM_PARAM_S1_DEF;
break;
case 4:
mt3620_pwm[pwmBlock]->pwm2_ctrl = MT3620_PWM_CTRL_DEF;
mt3620_pwm[pwmBlock]->pwm2_param_s0 = MT3620_PWM_PARAM_S0_DEF;
mt3620_pwm[pwmBlock]->pwm2_param_s1 = MT3620_PWM_PARAM_S1_DEF;
break;
case 8:
mt3620_pwm[pwmBlock]->pwm3_ctrl = MT3620_PWM_CTRL_DEF;
mt3620_pwm[pwmBlock]->pwm3_param_s0 = MT3620_PWM_PARAM_S0_DEF;
mt3620_pwm[pwmBlock]->pwm3_param_s1 = MT3620_PWM_PARAM_S1_DEF;
break;
default:
break;
}
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm_glo_ctrl, pwm_tick_clock_sel, clockSel);
uint32_t pwm_param = ((offTime << 16) | onTime);
switch (pinMask) {
case 1:
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm0_ctrl, pwm_clock_en, 1);
mt3620_pwm[pwmBlock]->pwm0_param_s0 = pwm_param;
mt3620_pwm[pwmBlock]->pwm0_param_s1 = 0;
mt3620_pwm0_ctrl_t pwm0_ctrl = {.mask = mt3620_pwm[pwmBlock]->pwm0_ctrl};
pwm0_ctrl.S0_stay_cycle = 1;
pwm0_ctrl.pwm_io_ctrl = 0;
mt3620_pwm[pwmBlock]->pwm0_ctrl = pwm0_ctrl.mask;
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm0_ctrl, kick, 1);
break;
case 2:
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm1_ctrl, pwm_clock_en, 1);
mt3620_pwm[pwmBlock]->pwm1_param_s0 = pwm_param;
mt3620_pwm[pwmBlock]->pwm1_param_s1 = 0;
mt3620_pwm1_ctrl_t pwm1_ctrl = { .mask = mt3620_pwm[pwmBlock]->pwm1_ctrl };
pwm1_ctrl.S0_stay_cycle = 1;
pwm1_ctrl.pwm_io_ctrl = 0;
mt3620_pwm[pwmBlock]->pwm1_ctrl = pwm1_ctrl.mask;
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm1_ctrl, kick, 1);
break;
case 4:
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm2_ctrl, pwm_clock_en, 1);
mt3620_pwm[pwmBlock]->pwm2_param_s0 = pwm_param;
mt3620_pwm[pwmBlock]->pwm2_param_s1 = 0;
mt3620_pwm2_ctrl_t pwm2_ctrl = { .mask = mt3620_pwm[pwmBlock]->pwm2_ctrl };
pwm2_ctrl.S0_stay_cycle = 1;
pwm2_ctrl.pwm_io_ctrl = 0;
mt3620_pwm[pwmBlock]->pwm2_ctrl = pwm2_ctrl.mask;
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm2_ctrl, kick, 1);
break;
case 8:
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm3_ctrl, pwm_clock_en, 1);
mt3620_pwm[pwmBlock]->pwm3_param_s0 = pwm_param;
mt3620_pwm[pwmBlock]->pwm3_param_s1 = 0;
mt3620_pwm3_ctrl_t pwm3_ctrl = { .mask = mt3620_pwm[pwmBlock]->pwm3_ctrl };
pwm3_ctrl.S0_stay_cycle = 1;
pwm3_ctrl.pwm_io_ctrl = 0;
mt3620_pwm[pwmBlock]->pwm3_ctrl = pwm3_ctrl.mask;
MT3620_PWM_FIELD_WRITE(pwmBlock, pwm3_ctrl, kick, 1);
break;
default:
break;
}
return ERROR_NONE;
}
#define GPIO_EINT_PRIORITY 2
int32_t EINT_ConfigurePin(
uint32_t pin,
gpio_eint_attr_t *attr)
{
if (pin >= GPIO_EINT_PIN_COUNT) {
return ERROR_EINT_NOT_A_PIN;
}
gpio_eint_attr_t attribute = attr ? *attr : gpioEINTAttrDefault;
if (attribute.freq >= GPIO_EINT_DBNC_FREQ_INVALID) {
return ERROR_EINT_ATTRIBUTE;
}
MT3620_IRQ_DBNC_FIELD_WRITE(dbnc_con, en, pin, true);
MT3620_IRQ_DBNC_FIELD_WRITE(dbnc_con, pol, pin, attribute.positive);
MT3620_IRQ_DBNC_FIELD_WRITE(dbnc_con, dual, pin, attribute.dualEdge);
MT3620_IRQ_DBNC_FIELD_WRITE(dbnc_con, prescal, pin, attribute.freq);
NVIC_EnableIRQ(
MT3620_IRQ_EINT_INTERRUPT(pin), GPIO_EINT_PRIORITY);
return ERROR_NONE;
}
int32_t EINT_DeConfigurePin(uint32_t pin)
{
if (pin >= GPIO_EINT_PIN_COUNT) {
return ERROR_EINT_NOT_A_PIN;
}
mt3620_irq->dbnc_con[pin] = 0;
NVIC_DisableIRQ(MT3620_IRQ_EINT_INTERRUPT(pin));
return ERROR_NONE;
}
uint8_t EINT_GetDebounceCounter(uint32_t pin)
{
if (pin >= GPIO_EINT_PIN_COUNT) {
return 0;
}
return MT3620_IRQ_DBNC_FIELD_READ(dbnc_con, cnt, pin);
}