2016-12-07 14:58:16 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014-2016 Freie Universität Berlin
|
|
|
|
* 2015 Engineering-Spirit
|
|
|
|
* 2016 OTA keys S.A.
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-10-01 21:55:59 +02:00
|
|
|
* @ingroup cpu_stm32_common
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_pwm
|
2016-12-07 14:58:16 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level PWM driver implementation
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Fabian Nack <nack@inf.fu-berlin.de>
|
|
|
|
* @author Nick v. IJzendoorn <nijzendoorn@engineering-spirit.nl>
|
|
|
|
* @author Aurelien Gonce <aurelien.gonce@altran.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "assert.h"
|
|
|
|
#include "periph/pwm.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
|
|
|
|
#define CCMR_LEFT (TIM_CCMR1_OC1M_1 | TIM_CCMR1_OC1M_2 | \
|
|
|
|
TIM_CCMR1_OC2M_1 | TIM_CCMR1_OC2M_2)
|
|
|
|
#define CCMR_RIGHT (TIM_CCMR1_OC1M_0 | TIM_CCMR1_OC1M_1 | \
|
|
|
|
TIM_CCMR1_OC1M_2 | TIM_CCMR1_OC2M_0 | \
|
|
|
|
TIM_CCMR1_OC2M_1 | TIM_CCMR1_OC2M_2);
|
|
|
|
|
|
|
|
static inline TIM_TypeDef *dev(pwm_t pwm)
|
|
|
|
{
|
|
|
|
return pwm_config[pwm].dev;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t pwm_init(pwm_t pwm, pwm_mode_t mode, uint32_t freq, uint16_t res)
|
|
|
|
{
|
2017-11-10 17:25:13 +01:00
|
|
|
uint32_t timer_clk = periph_timer_clk(pwm_config[pwm].bus);
|
2016-12-07 14:58:16 +01:00
|
|
|
|
|
|
|
/* verify parameters */
|
2017-11-10 17:25:13 +01:00
|
|
|
assert((pwm < PWM_NUMOF) && ((freq * res) < timer_clk));
|
2016-12-07 14:58:16 +01:00
|
|
|
|
|
|
|
/* power on the used timer */
|
2017-02-07 15:05:43 +01:00
|
|
|
periph_clk_en(pwm_config[pwm].bus, pwm_config[pwm].rcc_mask);
|
2016-12-07 14:58:16 +01:00
|
|
|
/* reset configuration and CC channels */
|
|
|
|
dev(pwm)->CR1 = 0;
|
|
|
|
dev(pwm)->CR2 = 0;
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
for (unsigned i = 0; i < TIMER_CHAN; ++i) {
|
2016-12-07 14:58:16 +01:00
|
|
|
dev(pwm)->CCR[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* configure the used pins */
|
2017-01-17 14:08:13 +01:00
|
|
|
unsigned i = 0;
|
|
|
|
while ((i < TIMER_CHAN) && (pwm_config[pwm].chan[i].pin != GPIO_UNDEF)) {
|
|
|
|
gpio_init(pwm_config[pwm].chan[i].pin, GPIO_OUT);
|
|
|
|
gpio_init_af(pwm_config[pwm].chan[i].pin, pwm_config[pwm].af);
|
|
|
|
i++;
|
2016-12-07 14:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* configure the PWM frequency and resolution by setting the auto-reload
|
|
|
|
* and prescaler registers */
|
2017-11-10 17:25:13 +01:00
|
|
|
dev(pwm)->PSC = (timer_clk / (res * freq)) - 1;
|
2016-12-07 14:58:16 +01:00
|
|
|
dev(pwm)->ARR = res - 1;
|
|
|
|
|
|
|
|
/* set PWM mode */
|
|
|
|
switch (mode) {
|
|
|
|
case PWM_LEFT:
|
|
|
|
dev(pwm)->CCMR1 = CCMR_LEFT;
|
|
|
|
dev(pwm)->CCMR2 = CCMR_LEFT;
|
|
|
|
break;
|
|
|
|
case PWM_RIGHT:
|
|
|
|
dev(pwm)->CCMR1 = CCMR_RIGHT;
|
|
|
|
dev(pwm)->CCMR2 = CCMR_RIGHT;
|
|
|
|
break;
|
|
|
|
case PWM_CENTER:
|
|
|
|
dev(pwm)->CCMR1 = 0;
|
|
|
|
dev(pwm)->CCMR2 = 0;
|
|
|
|
dev(pwm)->CR1 |= (TIM_CR1_CMS_0 | TIM_CR1_CMS_1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* enable PWM outputs and start PWM generation */
|
|
|
|
#ifdef TIM_BDTR_MOE
|
|
|
|
dev(pwm)->BDTR = TIM_BDTR_MOE;
|
|
|
|
#endif
|
|
|
|
dev(pwm)->CCER = (TIM_CCER_CC1E | TIM_CCER_CC2E |
|
|
|
|
TIM_CCER_CC3E | TIM_CCER_CC4E);
|
|
|
|
dev(pwm)->CR1 |= TIM_CR1_CEN;
|
|
|
|
|
|
|
|
/* return the actual used PWM frequency */
|
2017-11-10 17:25:13 +01:00
|
|
|
return (timer_clk / (res * (dev(pwm)->PSC + 1)));
|
2016-12-07 14:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t pwm_channels(pwm_t pwm)
|
|
|
|
{
|
|
|
|
assert(pwm < PWM_NUMOF);
|
2017-01-17 14:08:13 +01:00
|
|
|
|
|
|
|
unsigned i = 0;
|
|
|
|
while ((i < TIMER_CHAN) && (pwm_config[pwm].chan[i].pin != GPIO_UNDEF)) {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
return (uint8_t)i;
|
2016-12-07 14:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void pwm_set(pwm_t pwm, uint8_t channel, uint16_t value)
|
|
|
|
{
|
2017-01-17 14:08:13 +01:00
|
|
|
assert((pwm < PWM_NUMOF) &&
|
|
|
|
(channel < TIMER_CHAN) &&
|
|
|
|
(pwm_config[pwm].chan[channel].pin != GPIO_UNDEF));
|
2016-12-07 14:58:16 +01:00
|
|
|
|
|
|
|
/* norm value to maximum possible value */
|
|
|
|
if (value > dev(pwm)->ARR) {
|
|
|
|
value = (uint16_t)dev(pwm)->ARR;
|
|
|
|
}
|
|
|
|
/* set new value */
|
2017-01-17 14:08:13 +01:00
|
|
|
dev(pwm)->CCR[pwm_config[pwm].chan[channel].cc_chan] = value;
|
2016-12-07 14:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void pwm_poweron(pwm_t pwm)
|
|
|
|
{
|
|
|
|
assert(pwm < PWM_NUMOF);
|
|
|
|
periph_clk_en(pwm_config[pwm].bus, pwm_config[pwm].rcc_mask);
|
2017-02-07 15:05:43 +01:00
|
|
|
dev(pwm)->CR1 |= TIM_CR1_CEN;
|
2016-12-07 14:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void pwm_poweroff(pwm_t pwm)
|
|
|
|
{
|
|
|
|
assert(pwm < PWM_NUMOF);
|
2017-02-07 15:05:43 +01:00
|
|
|
dev(pwm)->CR1 &= ~TIM_CR1_CEN;
|
2016-12-07 14:58:16 +01:00
|
|
|
periph_clk_dis(pwm_config[pwm].bus, pwm_config[pwm].rcc_mask);
|
|
|
|
}
|