/* * Copyright (c) 2016 Linaro Limited. * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #include "pwm_stm32.h" /* convenience defines */ #define DEV_CFG(dev) \ ((const struct pwm_stm32_config * const)(dev)->config->config_info) #define DEV_DATA(dev) \ ((struct pwm_stm32_data * const)(dev)->driver_data) #define PWM_STRUCT(dev) \ ((TIM_TypeDef *)(DEV_CFG(dev))->pwm_base) #define CHANNEL_LENGTH 4 static u32_t __get_tim_clk(u32_t bus_clk, clock_control_subsys_t *sub_system) { struct stm32_pclken *pclken = (struct stm32_pclken *)(sub_system); u32_t tim_clk, apb_psc; if (pclken->bus == STM32_CLOCK_BUS_APB1) { apb_psc = CONFIG_CLOCK_STM32_APB1_PRESCALER; } #ifndef CONFIG_SOC_SERIES_STM32F0X else { apb_psc = CONFIG_CLOCK_STM32_APB2_PRESCALER; } #endif /* * If the APB prescaler equals 1, the timer clock frequencies * are set to the same frequency as that of the APB domain. * Otherwise, they are set to twice (×2) the frequency of the * APB domain. */ if (apb_psc == 1) { tim_clk = bus_clk; } else { tim_clk = 2 * bus_clk; } return tim_clk; } /* * Set the period and pulse width for a PWM pin. * * Parameters * dev: Pointer to PWM device structure * pwm: PWM channel to set * period_cycles: Period (in timer count) * pulse_cycles: Pulse width (in timer count). * * return 0, or negative errno code */ static int pwm_stm32_pin_set(struct device *dev, u32_t pwm, u32_t period_cycles, u32_t pulse_cycles) { struct pwm_stm32_data *data = DEV_DATA(dev); TIM_HandleTypeDef *TimerHandle = &data->hpwm; TIM_OC_InitTypeDef sConfig; u32_t channel; bool counter_32b; if (period_cycles == 0 || pulse_cycles > period_cycles) { return -EINVAL; } /* configure channel */ channel = (pwm - 1)*CHANNEL_LENGTH; if (!IS_TIM_INSTANCE(PWM_STRUCT(dev)) || !IS_TIM_CHANNELS(channel)) { return -ENOTSUP; } #ifdef CONFIG_SOC_SERIES_STM32F1X /* FIXME: IS_TIM_32B_COUNTER_INSTANCE not available on * SMT32F1 Cube HAL since all timer counters are 16 bits */ counter_32b = 0; #else counter_32b = IS_TIM_32B_COUNTER_INSTANCE(PWM_STRUCT(dev)); #endif /* * The timer counts from 0 up to the value in the ARR register (16-bit). * Thus period_cycles cannot be greater than UINT16_MAX + 1. */ if (!counter_32b && (period_cycles > 0x10000)) { /* 16 bits counter does not support requested period * You might want to adapt PWM output clock to adjust * cycle durations to fit requested period into 16 bits * counter */ return -ENOTSUP; } /* Configure Timer IP */ TimerHandle->Instance = PWM_STRUCT(dev); TimerHandle->Init.Prescaler = data->pwm_prescaler; TimerHandle->Init.ClockDivision = 0; TimerHandle->Init.CounterMode = TIM_COUNTERMODE_UP; TimerHandle->Init.RepetitionCounter = 0; /* Set period value */ TimerHandle->Init.Period = period_cycles - 1; HAL_TIM_PWM_Init(TimerHandle); /* Configure PWM channel */ sConfig.OCMode = TIM_OCMODE_PWM1; sConfig.OCPolarity = TIM_OCPOLARITY_HIGH; sConfig.OCFastMode = TIM_OCFAST_DISABLE; sConfig.OCNPolarity = TIM_OCNPOLARITY_HIGH; sConfig.OCNIdleState = TIM_OCNIDLESTATE_RESET; sConfig.OCIdleState = TIM_OCIDLESTATE_RESET; /* Set the pulse value */ sConfig.Pulse = pulse_cycles; HAL_TIM_PWM_ConfigChannel(TimerHandle, &sConfig, channel); return HAL_TIM_PWM_Start(TimerHandle, channel); } /* * Get the clock rate (cycles per second) for a PWM pin. * * Parameters * dev: Pointer to PWM device structure * pwm: PWM port number * cycles: Pointer to the memory to store clock rate (cycles per second) * * return 0, or negative errno code */ static int pwm_stm32_get_cycles_per_sec(struct device *dev, u32_t pwm, u64_t *cycles) { const struct pwm_stm32_config *cfg = DEV_CFG(dev); struct pwm_stm32_data *data = DEV_DATA(dev); u32_t bus_clk, tim_clk; if (cycles == NULL) { return -EINVAL; } /* Timer clock depends on APB prescaler */ clock_control_get_rate(data->clock, (clock_control_subsys_t *)&cfg->pclken, &bus_clk); tim_clk = __get_tim_clk(bus_clk, (clock_control_subsys_t *)&cfg->pclken); *cycles = (u64_t)(tim_clk / (data->pwm_prescaler + 1)); return 0; } static const struct pwm_driver_api pwm_stm32_drv_api_funcs = { .pin_set = pwm_stm32_pin_set, .get_cycles_per_sec = pwm_stm32_get_cycles_per_sec, }; static inline void __pwm_stm32_get_clock(struct device *dev) { struct pwm_stm32_data *data = DEV_DATA(dev); struct device *clk = device_get_binding(STM32_CLOCK_CONTROL_NAME); __ASSERT_NO_MSG(clk); data->clock = clk; } static int pwm_stm32_init(struct device *dev) { const struct pwm_stm32_config *config = DEV_CFG(dev); struct pwm_stm32_data *data = DEV_DATA(dev); __pwm_stm32_get_clock(dev); /* enable clock */ if (clock_control_on(data->clock, (clock_control_subsys_t *)&config->pclken) != 0) { return -EIO; } return 0; } #define PWM_DEVICE_INIT_STM32(n) \ static struct pwm_stm32_data pwm_stm32_dev_data_ ## n = { \ /* Default case */ \ .pwm_prescaler = DT_PWM_STM32_## n ##_PRESCALER, \ }; \ \ static const struct pwm_stm32_config pwm_stm32_dev_cfg_ ## n = { \ .pwm_base = DT_TIM_STM32_## n ##_BASE_ADDRESS, \ .pclken = { .bus = DT_TIM_STM32_## n ##_CLOCK_BUS, \ .enr = DT_TIM_STM32_## n ##_CLOCK_BITS }, \ }; \ \ DEVICE_AND_API_INIT(pwm_stm32_ ## n, \ DT_PWM_STM32_ ## n ## _DEV_NAME, \ pwm_stm32_init, \ &pwm_stm32_dev_data_ ## n, \ &pwm_stm32_dev_cfg_ ## n, \ POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE,\ &pwm_stm32_drv_api_funcs); #ifdef CONFIG_PWM_STM32_1 /* 16-bit advanced-control timer */ PWM_DEVICE_INIT_STM32(1) #endif /* CONFIG_PWM_STM32_1 */ #ifdef CONFIG_PWM_STM32_2 /* 32-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(2) #endif /* CONFIG_PWM_STM32_2 */ #ifdef CONFIG_PWM_STM32_3 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(3) #endif /* CONFIG_PWM_STM32_3 */ #ifdef CONFIG_PWM_STM32_4 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(4) #endif /* CONFIG_PWM_STM32_4 */ #ifdef CONFIG_PWM_STM32_5 /* 32-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(5) #endif /* CONFIG_PWM_STM32_5 */ #ifdef CONFIG_PWM_STM32_6 /* 16-bit basic timer */ PWM_DEVICE_INIT_STM32(6) #endif /* CONFIG_PWM_STM32_6 */ #ifdef CONFIG_PWM_STM32_7 /* 16-bit basic timer */ PWM_DEVICE_INIT_STM32(7) #endif /* CONFIG_PWM_STM32_7 */ #ifdef CONFIG_PWM_STM32_8 /* 16-bit advanced-control timer */ PWM_DEVICE_INIT_STM32(8) #endif /* CONFIG_PWM_STM32_8 */ #ifdef CONFIG_PWM_STM32_9 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(9) #endif /* CONFIG_PWM_STM32_9 */ #ifdef CONFIG_PWM_STM32_10 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(10) #endif /* CONFIG_PWM_STM32_10 */ #ifdef CONFIG_PWM_STM32_11 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(11) #endif /* CONFIG_PWM_STM32_11 */ #ifdef CONFIG_PWM_STM32_12 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(12) #endif /* CONFIG_PWM_STM32_12 */ #ifdef CONFIG_PWM_STM32_13 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(13) #endif /* CONFIG_PWM_STM32_13 */ #ifdef CONFIG_PWM_STM32_14 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(14) #endif /* CONFIG_PWM_STM32_14 */ #ifdef CONFIG_PWM_STM32_15 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(15) #endif /* CONFIG_PWM_STM32_15 */ #ifdef CONFIG_PWM_STM32_16 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(16) #endif /* CONFIG_PWM_STM32_16 */ #ifdef CONFIG_PWM_STM32_17 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(17) #endif /* CONFIG_PWM_STM32_17 */ #ifdef CONFIG_PWM_STM32_18 /* 16-bit advanced timer */ PWM_DEVICE_INIT_STM32(18) #endif /* CONFIG_PWM_STM32_18 */ #ifdef CONFIG_PWM_STM32_19 /* 16-bit general-purpose timer */ PWM_DEVICE_INIT_STM32(19) #endif /* CONFIG_PWM_STM32_19 */ #ifdef CONFIG_PWM_STM32_20 /* 16-bit advanced timer */ PWM_DEVICE_INIT_STM32(20) #endif /* CONFIG_PWM_STM32_20 */