Вроде все правильно ты понял
меня только смущает число 84000. Это ты в теории или в программе так делаешь? Потому что если в программе, то странно, что у тебя все работает, потому что прескейлер вроде как 16-ти битный. А значит не может быть больше 65535. Пойду посмотрю...
Добавлено спустя 10 минут 10 секунд:http://www.st.com/st-web-ui/static/acti ... 031020.pdfстр 422
16-bit programmable prescaler used to divide (also “on the fly”) the counter clock
frequency by any factor between 1 and 65536
Добавлено спустя 1 час 40 минут 55 секунд:Код:
#include <stdio.h>
#include <stm32f4xx.h>
#include <stm32f4xx_gpio.h>
#include <stm32f4xx_rcc.h>
#include <stm32f4xx_tim.h>
#include <stm32f4xx_exti.h>
#include <misc.h>
#include <stm32f4xx_syscfg.h>
//__IO uint16_t SERVO_0 = 1000; // 1000 us = 1 ms
//__IO uint16_t SERVO_180 = 2000; // 2000 us = 2 ms
#define SERVO_180 3*2200l
#define SERVO_0 3*800l
// Функция устанавливает позицию вала (в градусах)
void set_pos(uint8_t pos)
{
uint16_t tmp= (uint16_t)((SERVO_180 - SERVO_0) * pos / 180l) ;
uint16_t ServoPulse = SERVO_0 + tmp;
TIM_SetCompare1(TIM4, ServoPulse );
}
// Функция задержки
void delay(void)
{
volatile uint32_t i;
for (i=1; i != 0xFFFF; i++)
;
}
int main(void)
{
SystemInit();
// Инициализация портов вводов/вывода
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_StructInit(&GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; // Ножка PD12
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_Init(GPIOD, &GPIO_InitStructure);
//------------------------------------------------------------------------------------------
// Инициализируем таймер 4
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);
#define sPeriod 20000l //20ms = 20000 us
#define tClock (SystemCoreClock/8) //21MHz
#define maxCount 60000l
uint16_t Prescaler = 8; // PCLK1 = 168MHz/8 ~= 21MHz
uint16_t Period = 60000; // 20000; // 20 ms
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_TimeBaseStructInit (&TIM_TimeBaseStructure);
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;
TIM_TimeBaseStructure.TIM_Prescaler = Prescaler;
TIM_TimeBaseStructure.TIM_Period = Period;
TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);
// Разрешаем таймеру обращаться к выходу PD12
GPIO_PinAFConfig(GPIOD, GPIO_PinSource12, GPIO_AF_TIM4);
//------------------------------------------------------------------------------------------
// Инициализируем ШИМ
TIM_OCInitTypeDef TIM_OCInitStructure;
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStructure.TIM_Pulse = 20000;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
TIM_OC1Init(TIM4, &TIM_OCInitStructure);
TIM_Cmd(TIM4, ENABLE);
//delay();
//TIM_SetCompare1(TIM4, 460); // 90
//delay();
//TIM_SetCompare1(TIM4, 760 ); // 180
//Начинаем крутить сервой от 0 до 180 градусов.
uint8_t i;
while(1)
{
for (i=0;i<=180;i++)
{
delay();
set_pos(i);
}
for (i=180;i>0;i--)
{
delay();
set_pos(i);
}
}
}
Серва крутится туда-сюда, импульс меняется от 800 до 2200 мкс.
Единственная ошибка в моих объяснениях была в том, что к рассчетному прескейлеру надо единицу добавлять. Чо-то спросоня не пойму почему, но с единицей тайминги правильные.
Камень работает на 168МГц
Таймер на частоте PCLK1, т.е. 168/8 = 21МГц
С прескейлером 8 получается частота 3МГц.
Поэтому SERVO0 и SERVO180 = это утроенное количество микросекунд, т.е. 3*800 и 3*2200