1 #include "../hardware_api.h"
3 #if defined(__arm__) && defined(__SAM3X8E__)
6 static unsigned long _pwm_frequency;
7 static int _max_pwm_value = 1023;
10 static IRQn_Type irq_type[] = {TC0_IRQn, TC0_IRQn, TC1_IRQn, TC1_IRQn, TC2_IRQn, TC2_IRQn, TC3_IRQn, TC3_IRQn, TC4_IRQn, TC4_IRQn, TC5_IRQn, TC5_IRQn, TC6_IRQn, TC6_IRQn, TC7_IRQn, TC7_IRQn, TC8_IRQn, TC8_IRQn};
12 static volatile uint32_t pwm_counter_vals[] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
16 static uint8_t PWMEnabled = 0;
17 static uint8_t TCChanEnabled[] = {0, 0, 0, 0, 0, 0, 0, 0, 0};
18 static const uint32_t channelToChNo[] = { 0, 0, 1, 1, 2, 2, 0, 0, 1, 1, 2, 2, 0, 0, 1, 1, 2, 2 };
19 static const uint32_t channelToAB[] = { 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0 };
20 static Tc *channelToTC[] = {
21 TC0, TC0, TC0, TC0, TC0, TC0,
22 TC1, TC1, TC1, TC1, TC1, TC1,
23 TC2, TC2, TC2, TC2, TC2, TC2 };
24 static const uint32_t channelToId[] = { 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8 };
28 static void TC_SetCMR_ChannelA(Tc *tc, uint32_t chan, uint32_t v){ tc->TC_CHANNEL[chan].TC_CMR = (tc->TC_CHANNEL[chan].TC_CMR & 0xFFF0FFFF) | v;}
29 static void TC_SetCMR_ChannelB(Tc *tc, uint32_t chan, uint32_t v){ tc->TC_CHANNEL[chan].TC_CMR = (tc->TC_CHANNEL[chan].TC_CMR & 0xF0FFFFFF) | v; }
34 void syncTimers(uint32_t ulPin1,uint32_t ulPin2, uint32_t ulPin3 = -1, uint32_t ulPin4 = -1){
35 uint32_t chNo1,chNo2,chNo3,chNo4;
36 Tc *chTC1 =
nullptr,*chTC2 =
nullptr,*chTC3 =
nullptr,*chTC4 =
nullptr;
39 uint32_t attr = g_APinDescription[ulPin1].ulPinAttribute;
40 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
41 ETCChannel channel1 = g_APinDescription[ulPin1].ulTCChannel;
42 chNo1 = channelToChNo[channel1];
43 chTC1 = channelToTC[channel1];
44 TCChanEnabled[channelToId[channel1]] = 1;
48 attr = g_APinDescription[ulPin2].ulPinAttribute;
49 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
50 ETCChannel channel2 = g_APinDescription[ulPin2].ulTCChannel;
51 chNo2 = channelToChNo[channel2];
52 chTC2 = channelToTC[channel2];
53 TCChanEnabled[channelToId[channel2]] = 1;
57 attr = g_APinDescription[ulPin3].ulPinAttribute;
58 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
59 ETCChannel channel3 = g_APinDescription[ulPin3].ulTCChannel;
60 chNo3 = channelToChNo[channel3];
61 chTC3 = channelToTC[channel3];
62 TCChanEnabled[channelToId[channel3]] = 1;
67 attr = g_APinDescription[ulPin4].ulPinAttribute;
68 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
69 ETCChannel channel4 = g_APinDescription[ulPin4].ulTCChannel;
70 chNo4 = channelToChNo[channel4];
71 chTC4 = channelToTC[channel4];
72 TCChanEnabled[channelToId[channel4]] = 1;
77 TC_Start(chTC1, chNo1);
78 chTC1->TC_BCR = TC_BCR_SYNC;
81 TC_Start(chTC2, chNo2);
82 chTC2->TC_BCR = TC_BCR_SYNC;
85 TC_Start(chTC3, chNo3);
86 chTC3->TC_BCR = TC_BCR_SYNC;
89 TC_Start(chTC4, chNo4);
90 chTC4->TC_BCR = TC_BCR_SYNC;
96 void initPWM(uint32_t ulPin, uint32_t pwm_freq){
98 uint32_t attr = g_APinDescription[ulPin].ulPinAttribute;
99 if ((attr & PIN_ATTR_PWM) == PIN_ATTR_PWM) {
103 pmc_enable_periph_clk(PWM_INTERFACE_ID);
104 PWMC_ConfigureClocks(pwm_freq * _max_pwm_value, 0, VARIANT_MCK);
108 uint32_t chan = g_APinDescription[ulPin].ulPWMChannel;
109 if ((g_pinStatus[ulPin] & 0xF) != PIN_STATUS_PWM) {
111 PIO_Configure(g_APinDescription[ulPin].pPort,
112 g_APinDescription[ulPin].ulPinType,
113 g_APinDescription[ulPin].ulPin,
114 g_APinDescription[ulPin].ulPinConfiguration);
115 PWMC_ConfigureChannel(PWM_INTERFACE, chan, PWM_CMR_CPRE_CLKA, 0, 0);
116 PWMC_SetPeriod(PWM_INTERFACE, chan, _max_pwm_value);
117 PWMC_SetDutyCycle(PWM_INTERFACE, chan, 0);
118 PWMC_EnableChannel(PWM_INTERFACE, chan);
119 g_pinStatus[ulPin] = (g_pinStatus[ulPin] & 0xF0) | PIN_STATUS_PWM;
124 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
126 const uint32_t TC = VARIANT_MCK / 2 / pwm_freq;
128 ETCChannel channel = g_APinDescription[ulPin].ulTCChannel;
129 uint32_t chNo = channelToChNo[channel];
130 uint32_t chA = channelToAB[channel];
131 Tc *chTC = channelToTC[channel];
132 uint32_t interfaceID = channelToId[channel];
134 if (!TCChanEnabled[interfaceID]) {
135 pmc_enable_periph_clk(TC_INTERFACE_ID + interfaceID);
136 TC_Configure(chTC, chNo,
137 TC_CMR_TCCLKS_TIMER_CLOCK1 |
139 TC_CMR_WAVSEL_UP_RC |
141 TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_CLEAR |
142 TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_CLEAR);
143 TC_SetRC(chTC, chNo, TC);
147 TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_SET);
149 TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_SET);
152 if ((g_pinStatus[ulPin] & 0xF) != PIN_STATUS_PWM) {
153 PIO_Configure(g_APinDescription[ulPin].pPort,
154 g_APinDescription[ulPin].ulPinType,
155 g_APinDescription[ulPin].ulPin,
156 g_APinDescription[ulPin].ulPinConfiguration);
157 g_pinStatus[ulPin] = (g_pinStatus[ulPin] & 0xF0) | PIN_STATUS_PWM;
160 chTC->TC_CHANNEL[chNo].TC_IER = TC_IER_CPAS
163 chTC->TC_CHANNEL[chNo].TC_IDR = ~TC_IER_CPAS
167 NVIC_EnableIRQ(irq_type[channel]);
174 void setPwm(uint32_t ulPin, uint32_t ulValue) {
176 uint32_t attr = g_APinDescription[ulPin].ulPinAttribute;
177 if ((attr & PIN_ATTR_PWM) == PIN_ATTR_PWM) {
178 uint32_t chan = g_APinDescription[ulPin].ulPWMChannel;
179 PWMC_SetDutyCycle(PWM_INTERFACE, chan, ulValue);
183 if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) {
185 ETCChannel channel = g_APinDescription[ulPin].ulTCChannel;
186 Tc *chTC = channelToTC[channel];
187 uint32_t chNo = channelToChNo[channel];
190 if (channelToAB[channel])
191 TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_CLEAR);
193 TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_CLEAR);
197 const uint32_t TC = VARIANT_MCK / 2 / _pwm_frequency;
200 ulValue = ulValue * TC;
201 pwm_counter_vals[channel] = ulValue / _max_pwm_value;
203 if (channelToAB[channel])
204 TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_SET);
206 TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_SET);
217 TC_GetStatus(TC0, 0);
219 if(pwm_counter_vals[0]) TC_SetRA(TC0, 0, pwm_counter_vals[0]);
220 if(pwm_counter_vals[1]) TC_SetRB(TC0, 0, pwm_counter_vals[1]);
226 TC_GetStatus(TC0, 1);
228 if(pwm_counter_vals[2]) TC_SetRA(TC0, 1, pwm_counter_vals[2]);
229 if(pwm_counter_vals[3]) TC_SetRB(TC0, 1, pwm_counter_vals[3]);
235 TC_GetStatus(TC0, 2);
237 if(pwm_counter_vals[4]) TC_SetRA(TC0, 2, pwm_counter_vals[4]);
238 if(pwm_counter_vals[5]) TC_SetRB(TC0, 2, pwm_counter_vals[5]);
243 TC_GetStatus(TC1, 0);
245 if(pwm_counter_vals[6]) TC_SetRA(TC1, 0, pwm_counter_vals[6]);
246 if(pwm_counter_vals[7]) TC_SetRB(TC1, 0, pwm_counter_vals[7]);
252 TC_GetStatus(TC1, 1);
254 if(pwm_counter_vals[8]) TC_SetRA(TC1, 1, pwm_counter_vals[8]);
255 if(pwm_counter_vals[9]) TC_SetRB(TC1, 1, pwm_counter_vals[9]);
261 TC_GetStatus(TC1, 2);
263 if(pwm_counter_vals[10]) TC_SetRA(TC1, 2, pwm_counter_vals[10]);
264 if(pwm_counter_vals[11]) TC_SetRB(TC1, 2, pwm_counter_vals[11]);
269 TC_GetStatus(TC2, 0);
271 if(pwm_counter_vals[12]) TC_SetRA(TC2, 0, pwm_counter_vals[12]);
272 if(pwm_counter_vals[13]) TC_SetRB(TC2, 0, pwm_counter_vals[13]);
278 TC_GetStatus(TC2, 1);
280 if(pwm_counter_vals[14]) TC_SetRA(TC2, 1, pwm_counter_vals[14]);
281 if(pwm_counter_vals[15]) TC_SetRB(TC2, 1, pwm_counter_vals[15]);
287 TC_GetStatus(TC2, 2);
289 if(pwm_counter_vals[16]) TC_SetRA(TC2, 2, pwm_counter_vals[16]);
290 if(pwm_counter_vals[17]) TC_SetRB(TC2, 2, pwm_counter_vals[17]);
299 void _configure3PWM(
long pwm_frequency,
const int pinA,
const int pinB,
const int pinC) {
300 if(!pwm_frequency || !
_isset(pwm_frequency) ) pwm_frequency = 35000;
301 else pwm_frequency =
_constrain(pwm_frequency, 0, 50000);
303 _pwm_frequency = pwm_frequency;
305 initPWM(pinA, _pwm_frequency);
306 initPWM(pinB, _pwm_frequency);
307 initPWM(pinC, _pwm_frequency);
309 syncTimers(pinA, pinB, pinC);
316 void _configure2PWM(
long pwm_frequency,
const int pinA,
const int pinB) {
317 if(!pwm_frequency || !
_isset(pwm_frequency)) pwm_frequency = 35000;
318 else pwm_frequency =
_constrain(pwm_frequency, 0, 50000);
320 _pwm_frequency = pwm_frequency;
322 initPWM(pinA, _pwm_frequency);
323 initPWM(pinB, _pwm_frequency);
325 syncTimers(pinA, pinB);
331 void _configure4PWM(
long pwm_frequency,
const int pinA,
const int pinB,
const int pinC,
const int pinD) {
332 if(!pwm_frequency || !
_isset(pwm_frequency)) pwm_frequency = 35000;
333 else pwm_frequency =
_constrain(pwm_frequency, 0, 50000);
335 _pwm_frequency = pwm_frequency;
337 initPWM(pinA, _pwm_frequency);
338 initPWM(pinB, _pwm_frequency);
339 initPWM(pinC, _pwm_frequency);
340 initPWM(pinD, _pwm_frequency);
342 syncTimers(pinA, pinB, pinC, pinD);
348 void _writeDutyCycle3PWM(
float dc_a,
float dc_b,
float dc_c,
int pinA,
int pinB,
int pinC){
350 setPwm(pinA, _max_pwm_value*dc_a);
351 setPwm(pinB, _max_pwm_value*dc_b);
352 setPwm(pinC, _max_pwm_value*dc_c);
358 void _writeDutyCycle4PWM(
float dc_1a,
float dc_1b,
float dc_2a,
float dc_2b,
int pin1A,
int pin1B,
int pin2A,
int pin2B){
360 setPwm(pin1A, _max_pwm_value*dc_1a);
361 setPwm(pin1B, _max_pwm_value*dc_1b);
362 setPwm(pin2A, _max_pwm_value*dc_2a);
363 setPwm(pin2B, _max_pwm_value*dc_2b);
371 setPwm(pinA, _max_pwm_value*dc_a);
372 setPwm(pinB, _max_pwm_value*dc_b);
379 int _configure6PWM(
long pwm_frequency,
float dead_zone,
const int pinA_h,
const int pinA_l,
const int pinB_h,
const int pinB_l,
const int pinC_h,
const int pinC_l){
386 void _writeDutyCycle6PWM(
float dc_a,
float dc_b,
float dc_c,
float dead_zone,
int pinA_h,
int pinA_l,
int pinB_h,
int pinB_l,
int pinC_h,
int pinC_l){