SimpleFOClibrary  2.1
due_mcu.cpp
Go to the documentation of this file.
1 #include "../hardware_api.h"
2 
3 #if defined(__arm__) && defined(__SAM3X8E__)
4 
5 // pwm frequency and max duty cycle
6 static unsigned long _pwm_frequency;
7 static int _max_pwm_value = 1023;
8 
9 // array mapping the timer values to the interrupt handlers
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};
11 // current counter values
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};
13 
14 
15 // variables copied from wiring_analog.cpp for arduino due
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 };
25 
26 
27 // function setting the CMR register
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; }
30 
31 
32 // function which starts and syncs the timers
33 // if the pin is the true PWM pin this function does not do anything
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;
37 
38  // configure timer channel for the first pin if it is a timer pin
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;
45  }
46 
47  // configure timer channel for the first pin if it is a timer pin
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;
54  }
55  if(ulPin3 > 0 ){
56  // configure timer channel for the first pin if it is a timer pin
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;
63  }
64  }
65  if(ulPin4 > 0 ){
66  // configure timer channel for the first pin if it is a timer pin
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;
73  }
74  }
75  // start timers and make them synced
76  if(chTC1){
77  TC_Start(chTC1, chNo1);
78  chTC1->TC_BCR = TC_BCR_SYNC;
79  }
80  if(chTC2){
81  TC_Start(chTC2, chNo2);
82  chTC2->TC_BCR = TC_BCR_SYNC;
83  }
84  if(chTC3 && ulPin3){
85  TC_Start(chTC3, chNo3);
86  chTC3->TC_BCR = TC_BCR_SYNC;
87  }
88  if(chTC4 && ulPin4){
89  TC_Start(chTC4, chNo4);
90  chTC4->TC_BCR = TC_BCR_SYNC;
91  }
92 }
93 
94 // function configuring the pwm frequency for given pin
95 // possible to supply the pwm pin and the timer pin
96 void initPWM(uint32_t ulPin, uint32_t pwm_freq){
97  // check which pin type
98  uint32_t attr = g_APinDescription[ulPin].ulPinAttribute;
99  if ((attr & PIN_ATTR_PWM) == PIN_ATTR_PWM) { // if pwm pin
100 
101  if (!PWMEnabled) {
102  // PWM Startup code
103  pmc_enable_periph_clk(PWM_INTERFACE_ID);
104  PWMC_ConfigureClocks(pwm_freq * _max_pwm_value, 0, VARIANT_MCK);
105  PWMEnabled = 1;
106  }
107 
108  uint32_t chan = g_APinDescription[ulPin].ulPWMChannel;
109  if ((g_pinStatus[ulPin] & 0xF) != PIN_STATUS_PWM) {
110  // Setup PWM for this pin
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;
120  }
121  return;
122  }
123 
124  if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) { // if timer pin
125  // We use MCLK/2 as clock.
126  const uint32_t TC = VARIANT_MCK / 2 / pwm_freq;
127  // Setup Timer for this pin
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];
133 
134  if (!TCChanEnabled[interfaceID]) {
135  pmc_enable_periph_clk(TC_INTERFACE_ID + interfaceID);
136  TC_Configure(chTC, chNo,
137  TC_CMR_TCCLKS_TIMER_CLOCK1 |
138  TC_CMR_WAVE | // Waveform mode
139  TC_CMR_WAVSEL_UP_RC | // Counter running up and reset when equals to RC
140  TC_CMR_EEVT_XC0 | // Set external events from XC0 (this setup TIOB as output)
141  TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_CLEAR |
142  TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_CLEAR);
143  TC_SetRC(chTC, chNo, TC);
144  }
145  // disable the counter on start
146  if (chA){
147  TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_SET);
148  }else{
149  TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_SET);
150  }
151  // configure input-ouput structure
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;
158  }
159  // enable interrupts
160  chTC->TC_CHANNEL[chNo].TC_IER = TC_IER_CPAS // interrupt on RA compare match
161  | TC_IER_CPBS // interrupt on RB compare match
162  | TC_IER_CPCS; // interrupt on RC compare match
163  chTC->TC_CHANNEL[chNo].TC_IDR = ~TC_IER_CPAS // interrupt on RA compare match
164  & ~TC_IER_CPBS // interrupt on RB compare match
165  & ~ TC_IER_CPCS; // interrupt on RC compare match
166  // enable interrupts for this timer
167  NVIC_EnableIRQ(irq_type[channel]);
168  return;
169  }
170 }
171 
172 // pwm setting function
173 // it sets the duty cycle for pwm pin or timer pin
174 void setPwm(uint32_t ulPin, uint32_t ulValue) {
175  // check pin type
176  uint32_t attr = g_APinDescription[ulPin].ulPinAttribute;
177  if ((attr & PIN_ATTR_PWM) == PIN_ATTR_PWM) { // if pwm
178  uint32_t chan = g_APinDescription[ulPin].ulPWMChannel;
179  PWMC_SetDutyCycle(PWM_INTERFACE, chan, ulValue);
180  return;
181  }
182 
183  if ((attr & PIN_ATTR_TIMER) == PIN_ATTR_TIMER) { // if timer pin
184  // get the timer variables
185  ETCChannel channel = g_APinDescription[ulPin].ulTCChannel;
186  Tc *chTC = channelToTC[channel];
187  uint32_t chNo = channelToChNo[channel];
188  if(!ulValue) {
189  // if the value 0 disable counter
190  if (channelToAB[channel])
191  TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_CLEAR);
192  else
193  TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_CLEAR);
194  }else{
195  // if the value not zero
196  // calculate clock
197  const uint32_t TC = VARIANT_MCK / 2 / _pwm_frequency;
198  // Map value to Timer ranges 0..max_duty_cycle => 0..TC
199  // Setup Timer for this pin
200  ulValue = ulValue * TC;
201  pwm_counter_vals[channel] = ulValue / _max_pwm_value;
202  // enable counter
203  if (channelToAB[channel])
204  TC_SetCMR_ChannelA(chTC, chNo, TC_CMR_ACPA_CLEAR | TC_CMR_ACPC_SET);
205  else
206  TC_SetCMR_ChannelB(chTC, chNo, TC_CMR_BCPB_CLEAR | TC_CMR_BCPC_SET);
207  }
208 
209  return;
210  }
211 }
212 
213 // interrupt handlers for seamless pwm duty-cycle setting
214 void TC0_Handler()
215 {
216  // read/clear interrupt status
217  TC_GetStatus(TC0, 0);
218  // update the counters
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]);
221 }
222 
223 void TC1_Handler()
224 {
225  // read/clear interrupt status
226  TC_GetStatus(TC0, 1);
227  // update the counters
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]);
230 }
231 
232 void TC2_Handler()
233 {
234  // read/clear interrupt status
235  TC_GetStatus(TC0, 2);
236  // update the counters
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]);
239 }
240 void TC3_Handler()
241 {
242  // read/clear interrupt status
243  TC_GetStatus(TC1, 0);
244  // update the counters
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]);
247 }
248 
249 void TC4_Handler()
250 {
251  // read/clear interrupt status
252  TC_GetStatus(TC1, 1);
253  // update the counters
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]);
256 }
257 
258 void TC5_Handler()
259 {
260  // read/clear interrupt status
261  TC_GetStatus(TC1, 2);
262  // update the counters
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]);
265 }
266 void TC6_Handler()
267 {
268  // read/clear interrupt status
269  TC_GetStatus(TC2, 0);
270  // update the counters
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]);
273 }
274 
275 void TC7_Handler()
276 {
277  // read/clear interrupt status
278  TC_GetStatus(TC2, 1);
279  // update the counters
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]);
282 }
283 
284 void TC8_Handler()
285 {
286  // read/clear interrupt status
287  TC_GetStatus(TC2, 2);
288  // update the counters
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]);
291 }
292 
293 // implementation of the hardware_api.cpp
294 // ---------------------------------------------------------------------------------------------------------------------------------
295 
296 // function setting the high pwm frequency to the supplied pins
297 // - BLDC motor - 3PWM setting
298 // - hardware specific
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; // default frequency 50khz
301  else pwm_frequency = _constrain(pwm_frequency, 0, 50000); // constrain to 50kHz max
302  // save the pwm frequency
303  _pwm_frequency = pwm_frequency;
304  // cinfigure pwm pins
305  initPWM(pinA, _pwm_frequency);
306  initPWM(pinB, _pwm_frequency);
307  initPWM(pinC, _pwm_frequency);
308  // sync the timers if possible
309  syncTimers(pinA, pinB, pinC);
310 }
311 
312 
313 // Configuring PWM frequency, resolution and alignment
314 //- Stepper driver - 2PWM setting
315 // - hardware specific
316 void _configure2PWM(long pwm_frequency, const int pinA, const int pinB) {
317  if(!pwm_frequency || !_isset(pwm_frequency)) pwm_frequency = 35000; // default frequency 50khz
318  else pwm_frequency = _constrain(pwm_frequency, 0, 50000); // constrain to 50kHz max
319  // save the pwm frequency
320  _pwm_frequency = pwm_frequency;
321  // cinfigure pwm pins
322  initPWM(pinA, _pwm_frequency);
323  initPWM(pinB, _pwm_frequency);
324  // sync the timers if possible
325  syncTimers(pinA, pinB);
326 }
327 
328 // function setting the high pwm frequency to the supplied pins
329 // - Stepper motor - 4PWM setting
330 // - hardware speciffic
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; // default frequency 50khz
333  else pwm_frequency = _constrain(pwm_frequency, 0, 50000); // constrain to 50kHz max
334  // save the pwm frequency
335  _pwm_frequency = pwm_frequency;
336  // cinfigure pwm pins
337  initPWM(pinA, _pwm_frequency);
338  initPWM(pinB, _pwm_frequency);
339  initPWM(pinC, _pwm_frequency);
340  initPWM(pinD, _pwm_frequency);
341  // sync the timers if possible
342  syncTimers(pinA, pinB, pinC, pinD);
343 }
344 
345 // function setting the pwm duty cycle to the hardware
346 // - BLDC motor - 3PWM setting
347 // - hardware speciffic
348 void _writeDutyCycle3PWM(float dc_a, float dc_b, float dc_c, int pinA, int pinB, int pinC){
349  // transform duty cycle from [0,1] to [0,_max_pwm_value]
350  setPwm(pinA, _max_pwm_value*dc_a);
351  setPwm(pinB, _max_pwm_value*dc_b);
352  setPwm(pinC, _max_pwm_value*dc_c);
353 }
354 
355 // function setting the pwm duty cycle to the hardware
356 // - Stepper motor - 4PWM setting
357 // - hardware speciffic
358 void _writeDutyCycle4PWM(float dc_1a, float dc_1b, float dc_2a, float dc_2b, int pin1A, int pin1B, int pin2A, int pin2B){
359  // transform duty cycle from [0,1] to [0,_max_pwm_value]
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);
364 }
365 
366 // Function setting the duty cycle to the pwm pin (ex. analogWrite())
367 // - Stepper driver - 2PWM setting
368 // - hardware specific
369 void _writeDutyCycle2PWM(float dc_a, float dc_b, int pinA, int pinB){
370  // transform duty cycle from [0,1] to [0,_max_pwm_value]
371  setPwm(pinA, _max_pwm_value*dc_a);
372  setPwm(pinB, _max_pwm_value*dc_b);
373 }
374 
375 
376 // Configuring PWM frequency, resolution and alignment
377 // - BLDC driver - 6PWM setting
378 // - hardware specific
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){
380  return -1;
381 }
382 
383 // Function setting the duty cycle to the pwm pin (ex. analogWrite())
384 // - BLDC driver - 6PWM setting
385 // - hardware specific
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){
387  return;
388 }
389 
390 
391 #endif
_configure3PWM
void _configure3PWM(long pwm_frequency, const int pinA, const int pinB, const int pinC)
Definition: drivers/hardware_specific/generic_mcu.cpp:31
_configure4PWM
void _configure4PWM(long pwm_frequency, const int pin1A, const int pin1B, const int pin2A, const int pin2B)
Definition: drivers/hardware_specific/generic_mcu.cpp:40
_constrain
#define _constrain(amt, low, high)
Definition: foc_utils.h:9
_isset
#define _isset(a)
Definition: foc_utils.h:11
_writeDutyCycle3PWM
void _writeDutyCycle3PWM(float dc_a, float dc_b, float dc_c, int pinA, int pinB, int pinC)
Definition: drivers/hardware_specific/generic_mcu.cpp:64
_writeDutyCycle2PWM
void _writeDutyCycle2PWM(float dc_a, float dc_b, int pinA, int pinB)
Definition: drivers/hardware_specific/generic_mcu.cpp:55
_writeDutyCycle4PWM
void _writeDutyCycle4PWM(float dc_1a, float dc_1b, float dc_2a, float dc_2b, int pin1A, int pin1B, int pin2A, int pin2B)
Definition: drivers/hardware_specific/generic_mcu.cpp:74
_writeDutyCycle6PWM
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)
Definition: drivers/hardware_specific/generic_mcu.cpp:86
_configure6PWM
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)
Definition: drivers/hardware_specific/generic_mcu.cpp:47
_configure2PWM
void _configure2PWM(long pwm_frequency, const int pinA, const int pinB)
Definition: drivers/hardware_specific/generic_mcu.cpp:23