224d0aefe4
* First part of multitarget porting * Delete firmware/targets/f7/Inc directory * Delete firmware/targets/f7/Src directory * gpio: cli fixes; about: using version from HAL * sdk: path fixes * gui: include fixes * applications: more include fixes * gpio: ported to new apis * hal: introduced furi_hal_target_hw.h; libs: added one_wire * hal: f18 target * github: also build f18 by default * typo fix * fbt: removed extra checks on app list * api: explicitly bundling select mlib headers with sdk * hal: f18: changed INPUT_DEBOUNCE_TICKS to match f7 * cleaned up commented out code * docs: added info on hw targets * docs: targets: formatting fixes * f18: fixed link error * f18: fixed API version to match f7 * docs: hardware: minor wording fixes * faploader: added fw target check * docs: typo fixes * github: not building komi target by default * fbt: support for `targets` field for built-in apps * github: reworked build flow to exclude app_set; fbt: removed komi-specific appset; added additional target buildset check * github: fixed build; nfc: fixed pvs warnings * attempt to fix target id * f7, f18: removed certain HAL function from public API * apps: debug: enabled bt_debug_app for f18 * Targets: backport input pins configuration routine from F7 to F18 Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
139 lines
4.3 KiB
C
139 lines
4.3 KiB
C
#include <furi_hal_pwm.h>
|
|
#include <core/check.h>
|
|
#include <furi_hal_resources.h>
|
|
|
|
#include <stdint.h>
|
|
#include <stm32wbxx_ll_tim.h>
|
|
#include <stm32wbxx_ll_lptim.h>
|
|
#include <stm32wbxx_ll_rcc.h>
|
|
|
|
#include <furi.h>
|
|
|
|
const uint32_t lptim_psc_table[] = {
|
|
LL_LPTIM_PRESCALER_DIV1,
|
|
LL_LPTIM_PRESCALER_DIV2,
|
|
LL_LPTIM_PRESCALER_DIV4,
|
|
LL_LPTIM_PRESCALER_DIV8,
|
|
LL_LPTIM_PRESCALER_DIV16,
|
|
LL_LPTIM_PRESCALER_DIV32,
|
|
LL_LPTIM_PRESCALER_DIV64,
|
|
LL_LPTIM_PRESCALER_DIV128,
|
|
};
|
|
|
|
void furi_hal_pwm_start(FuriHalPwmOutputId channel, uint32_t freq, uint8_t duty) {
|
|
if(channel == FuriHalPwmOutputIdTim1PA7) {
|
|
furi_hal_gpio_init_ex(
|
|
&gpio_ext_pa7,
|
|
GpioModeAltFunctionPushPull,
|
|
GpioPullNo,
|
|
GpioSpeedVeryHigh,
|
|
GpioAltFn1TIM1);
|
|
|
|
FURI_CRITICAL_ENTER();
|
|
LL_TIM_DeInit(TIM1);
|
|
FURI_CRITICAL_EXIT();
|
|
|
|
LL_TIM_SetCounterMode(TIM1, LL_TIM_COUNTERMODE_UP);
|
|
LL_TIM_SetRepetitionCounter(TIM1, 0);
|
|
LL_TIM_SetClockDivision(TIM1, LL_TIM_CLOCKDIVISION_DIV1);
|
|
LL_TIM_SetClockSource(TIM1, LL_TIM_CLOCKSOURCE_INTERNAL);
|
|
LL_TIM_EnableARRPreload(TIM1);
|
|
|
|
LL_TIM_OC_EnablePreload(TIM1, LL_TIM_CHANNEL_CH1);
|
|
LL_TIM_OC_SetMode(TIM1, LL_TIM_CHANNEL_CH1, LL_TIM_OCMODE_PWM1);
|
|
LL_TIM_OC_SetPolarity(TIM1, LL_TIM_CHANNEL_CH1N, LL_TIM_OCPOLARITY_HIGH);
|
|
LL_TIM_OC_DisableFast(TIM1, LL_TIM_CHANNEL_CH1);
|
|
LL_TIM_CC_EnableChannel(TIM1, LL_TIM_CHANNEL_CH1N);
|
|
|
|
LL_TIM_EnableAllOutputs(TIM1);
|
|
|
|
furi_hal_pwm_set_params(channel, freq, duty);
|
|
|
|
LL_TIM_EnableCounter(TIM1);
|
|
} else if(channel == FuriHalPwmOutputIdLptim2PA4) {
|
|
furi_hal_gpio_init_ex(
|
|
&gpio_ext_pa4,
|
|
GpioModeAltFunctionPushPull,
|
|
GpioPullNo,
|
|
GpioSpeedVeryHigh,
|
|
GpioAltFn14LPTIM2);
|
|
|
|
FURI_CRITICAL_ENTER();
|
|
LL_LPTIM_DeInit(LPTIM2);
|
|
FURI_CRITICAL_EXIT();
|
|
|
|
LL_LPTIM_SetUpdateMode(LPTIM2, LL_LPTIM_UPDATE_MODE_ENDOFPERIOD);
|
|
LL_RCC_SetLPTIMClockSource(LL_RCC_LPTIM2_CLKSOURCE_PCLK1);
|
|
LL_LPTIM_SetClockSource(LPTIM2, LL_LPTIM_CLK_SOURCE_INTERNAL);
|
|
LL_LPTIM_ConfigOutput(
|
|
LPTIM2, LL_LPTIM_OUTPUT_WAVEFORM_PWM, LL_LPTIM_OUTPUT_POLARITY_INVERSE);
|
|
LL_LPTIM_SetCounterMode(LPTIM2, LL_LPTIM_COUNTER_MODE_INTERNAL);
|
|
|
|
LL_LPTIM_Enable(LPTIM2);
|
|
|
|
furi_hal_pwm_set_params(channel, freq, duty);
|
|
|
|
LL_LPTIM_StartCounter(LPTIM2, LL_LPTIM_OPERATING_MODE_CONTINUOUS);
|
|
}
|
|
}
|
|
|
|
void furi_hal_pwm_stop(FuriHalPwmOutputId channel) {
|
|
if(channel == FuriHalPwmOutputIdTim1PA7) {
|
|
furi_hal_gpio_init_simple(&gpio_ext_pa7, GpioModeAnalog);
|
|
FURI_CRITICAL_ENTER();
|
|
LL_TIM_DeInit(TIM1);
|
|
FURI_CRITICAL_EXIT();
|
|
} else if(channel == FuriHalPwmOutputIdLptim2PA4) {
|
|
furi_hal_gpio_init_simple(&gpio_ext_pa4, GpioModeAnalog);
|
|
FURI_CRITICAL_ENTER();
|
|
LL_LPTIM_DeInit(LPTIM2);
|
|
FURI_CRITICAL_EXIT();
|
|
}
|
|
}
|
|
|
|
void furi_hal_pwm_set_params(FuriHalPwmOutputId channel, uint32_t freq, uint8_t duty) {
|
|
furi_assert(freq > 0);
|
|
uint32_t freq_div = 64000000LU / freq;
|
|
|
|
if(channel == FuriHalPwmOutputIdTim1PA7) {
|
|
uint32_t prescaler = freq_div / 0x10000LU;
|
|
uint32_t period = freq_div / (prescaler + 1);
|
|
uint32_t compare = period * duty / 100;
|
|
|
|
LL_TIM_SetPrescaler(TIM1, prescaler);
|
|
LL_TIM_SetAutoReload(TIM1, period - 1);
|
|
LL_TIM_OC_SetCompareCH1(TIM1, compare);
|
|
} else if(channel == FuriHalPwmOutputIdLptim2PA4) {
|
|
uint32_t prescaler = 0;
|
|
uint32_t period = 0;
|
|
|
|
bool clock_lse = false;
|
|
|
|
do {
|
|
period = freq_div / (1UL << prescaler);
|
|
if(period <= 0xFFFF) {
|
|
break;
|
|
}
|
|
prescaler++;
|
|
if(prescaler > 7) {
|
|
prescaler = 0;
|
|
clock_lse = true;
|
|
period = 32768LU / freq;
|
|
break;
|
|
}
|
|
} while(1);
|
|
|
|
uint32_t compare = period * duty / 100;
|
|
|
|
LL_LPTIM_SetPrescaler(LPTIM2, lptim_psc_table[prescaler]);
|
|
LL_LPTIM_SetAutoReload(LPTIM2, period);
|
|
LL_LPTIM_SetCompare(LPTIM2, compare);
|
|
|
|
if(clock_lse) {
|
|
LL_RCC_SetLPTIMClockSource(LL_RCC_LPTIM2_CLKSOURCE_LSE);
|
|
} else {
|
|
LL_RCC_SetLPTIMClockSource(LL_RCC_LPTIM2_CLKSOURCE_PCLK1);
|
|
}
|
|
}
|
|
}
|