2015-01-06 00:19:24 +01:00
|
|
|
/*
|
2016-12-14 20:09:45 +01:00
|
|
|
* Copyright (C) 2014-2016 Freie Universität Berlin
|
2015-01-06 00:19:24 +01:00
|
|
|
* Copyright (C) 2014 PHYTEC Messtechnik GmbH
|
2016-02-14 08:38:15 +01:00
|
|
|
* Copyright (C) 2015-2016 Eistec AB
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
2016-12-14 20:09:45 +01:00
|
|
|
* 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.
|
2015-01-06 00:19:24 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-10-27 01:26:26 +02:00
|
|
|
* @ingroup cpu_kinetis
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_pwm
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level PWM driver implementation
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Johann Fischer <j.fischer@phytec.de>
|
|
|
|
* @author Jonas Remmert <j.remmert@phytec.de>
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2016-12-14 20:09:45 +01:00
|
|
|
#include "assert.h"
|
2015-01-06 00:19:24 +01:00
|
|
|
#include "periph/pwm.h"
|
|
|
|
|
2017-03-22 08:56:25 +01:00
|
|
|
/* This driver uses the FlexTimer module (FTM) for generating PWM signals, but
|
|
|
|
* not all Kinetis CPUs have such a module. This preprocessor condition prevents
|
|
|
|
* compilation errors when the CPU header lacks the register definitions for the
|
|
|
|
* FTM */
|
|
|
|
#ifdef FTM0
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
#define PRESCALER_MAX (7U)
|
2015-05-16 00:15:31 +02:00
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
static inline FTM_Type *ftm(pwm_t pwm)
|
2015-01-06 00:19:24 +01:00
|
|
|
{
|
2016-12-14 20:09:45 +01:00
|
|
|
return pwm_config[pwm].ftm;
|
|
|
|
}
|
2015-05-16 00:15:31 +02:00
|
|
|
|
2017-02-07 15:05:43 +01:00
|
|
|
static void poweron(pwm_t pwm)
|
|
|
|
{
|
|
|
|
int ftm = pwm_config[pwm].ftm_num;
|
|
|
|
|
|
|
|
#ifdef SIM_SCGC6_FTM2_SHIFT
|
|
|
|
BITBAND_REG32(SIM->SCGC6, SIM_SCGC6_FTM0_SHIFT + ftm) = 1;
|
|
|
|
#else
|
|
|
|
if (ftm < 2) {
|
|
|
|
BITBAND_REG32(SIM->SCGC6, SIM_SCGC6_FTM0_SHIFT + ftm) = 1;
|
|
|
|
}
|
|
|
|
else if (ftm == 2) {
|
|
|
|
BITBAND_REG32(SIM->SCGC3, SIM_SCGC3_FTM2_SHIFT) = 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
uint32_t pwm_init(pwm_t pwm, pwm_mode_t mode, uint32_t freq, uint16_t res)
|
|
|
|
{
|
|
|
|
uint8_t pre = 0;
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
if (pwm >= PWM_NUMOF || ((res * freq) > CLOCK_BUSCLOCK)) {
|
2015-10-21 13:33:12 +02:00
|
|
|
return 0;
|
2015-05-16 00:15:31 +02:00
|
|
|
}
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
/* figure out the clock settings
|
|
|
|
* the resulting frequency is calculated by
|
|
|
|
* ticks := BUS_CLK / 2 ^ pre
|
|
|
|
* where `ticks` is `freq * res`
|
|
|
|
* and `pre` must be between [0, 7].
|
|
|
|
*
|
|
|
|
* The resulting prescaler yields a timer frequency, which is the closest
|
|
|
|
* possible frequency requested. */
|
|
|
|
while ((CLOCK_BUSCLOCK >> pre) > (res * freq)) {
|
|
|
|
++pre;
|
2015-05-16 00:15:31 +02:00
|
|
|
}
|
2016-12-14 20:09:45 +01:00
|
|
|
/* make sure the calculated prescaler is valid, else return */
|
|
|
|
if (pre > PRESCALER_MAX) {
|
|
|
|
return 0;
|
2015-05-16 00:15:31 +02:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
/* configure the used timer */
|
2017-02-07 15:05:43 +01:00
|
|
|
poweron(pwm);
|
2015-05-16 00:15:31 +02:00
|
|
|
/* disable write protect for changing settings */
|
2016-12-14 20:09:45 +01:00
|
|
|
ftm(pwm)->MODE = FTM_MODE_WPDIS_MASK;
|
|
|
|
/* clear any existing configuration */
|
|
|
|
ftm(pwm)->COMBINE = 0;
|
|
|
|
ftm(pwm)->CNTIN = 0;
|
|
|
|
ftm(pwm)->SWOCTRL = 0;
|
|
|
|
/* apply prescaler and set resolution */
|
|
|
|
ftm(pwm)->SC = FTM_SC_PS(pre);
|
|
|
|
ftm(pwm)->MOD = (res - 1);
|
|
|
|
|
|
|
|
/* set CPWMS bit in the SC register in case of center aligned mode */
|
|
|
|
if (mode == PWM_CENTER) {
|
|
|
|
BITBAND_REG32(ftm(pwm)->SC, FTM_SC_CPWMS_SHIFT) = 1;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
/* setup the configured channels */
|
|
|
|
for (int i = 0; i < (int)pwm_config[pwm].chan_numof; i++) {
|
|
|
|
/* configure the used pin */
|
|
|
|
gpio_init_port(pwm_config[pwm].chan[i].pin,
|
|
|
|
PORT_PCR_MUX(pwm_config[pwm].chan[i].af));
|
|
|
|
/* set the given mode */
|
|
|
|
ftm(pwm)->CONTROLS[pwm_config[pwm].chan[i].ftm_chan].CnSC = mode;
|
|
|
|
/* and reset the PWM to 0% duty cycle */
|
|
|
|
ftm(pwm)->CONTROLS[pwm_config[pwm].chan[i].ftm_chan].CnV = 0;
|
2015-05-16 00:15:31 +02:00
|
|
|
}
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
/* and now we start the actual waveform generation */
|
2017-02-07 15:05:43 +01:00
|
|
|
ftm(pwm)->SC |= FTM_SC_CLKS(1);
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
/* finally we need to return the actual applied PWM frequency */
|
|
|
|
return (CLOCK_BUSCLOCK >> pre) / res;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
uint8_t pwm_channels(pwm_t pwm)
|
2015-10-21 13:33:12 +02:00
|
|
|
{
|
2016-12-14 20:09:45 +01:00
|
|
|
assert(pwm < PWM_NUMOF);
|
|
|
|
return pwm_config[pwm].chan_numof;
|
2015-10-21 13:33:12 +02:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
void pwm_set(pwm_t pwm, uint8_t channel, uint16_t value)
|
2015-01-06 00:19:24 +01:00
|
|
|
{
|
2016-12-14 20:09:45 +01:00
|
|
|
assert((pwm < PWM_NUMOF) && (channel < pwm_config[pwm].chan_numof));
|
|
|
|
ftm(pwm)->CONTROLS[pwm_config[pwm].chan[channel].ftm_chan].CnV = value;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
void pwm_poweron(pwm_t pwm)
|
2015-01-06 00:19:24 +01:00
|
|
|
{
|
2016-12-14 20:09:45 +01:00
|
|
|
assert(pwm < PWM_NUMOF);
|
2017-02-07 15:05:43 +01:00
|
|
|
poweron(pwm);
|
|
|
|
ftm(pwm)->SC |= FTM_SC_CLKS(1);
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
2016-12-14 20:09:45 +01:00
|
|
|
void pwm_poweroff(pwm_t pwm)
|
2015-01-06 00:19:24 +01:00
|
|
|
{
|
2016-12-14 20:09:45 +01:00
|
|
|
assert(pwm < PWM_NUMOF);
|
2017-02-07 15:05:43 +01:00
|
|
|
int ftm_num = pwm_config[pwm].ftm_num;
|
|
|
|
|
|
|
|
/* disable PWM generation */
|
|
|
|
ftm(pwm)->SC &= ~(FTM_SC_CLKS_MASK);
|
2016-12-14 20:09:45 +01:00
|
|
|
|
2017-02-07 15:05:43 +01:00
|
|
|
/* and power of the peripheral */
|
2016-12-14 20:09:45 +01:00
|
|
|
#ifdef SIM_SCGC6_FTM2_SHIFT
|
2017-02-07 15:05:43 +01:00
|
|
|
BITBAND_REG32(SIM->SCGC6, SIM_SCGC6_FTM0_SHIFT + ftm_num) = 0;
|
2016-12-14 20:09:45 +01:00
|
|
|
#else
|
2017-02-07 15:05:43 +01:00
|
|
|
if (ftm_num < 2) {
|
|
|
|
BITBAND_REG32(SIM->SCGC6, SIM_SCGC6_FTM0_SHIFT + ftm_num) = 0;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
2017-02-07 15:05:43 +01:00
|
|
|
else if (ftm_num == 2) {
|
2016-12-14 20:09:45 +01:00
|
|
|
BITBAND_REG32(SIM->SCGC3, SIM_SCGC3_FTM2_SHIFT) = 0;
|
|
|
|
}
|
|
|
|
#endif
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
2017-03-22 08:56:25 +01:00
|
|
|
|
|
|
|
#endif /* defined(FTM0) */
|