0

所以我正在尝试使用STM32L1通过PWM控制伺服。下面是完整的代码,不需要库。当我上传并运行此代码时,伺服器会滴答数次,而不是实际平稳地移动到所需的位置 600(0*)、1100(50*)、1600(100*) 和 2100(150*)。我相信这与预分频值计算有关,而且我不确定以微秒为单位提到 ARR 是否正确,如果不是,我如何将其配置为读取微秒而不是毫秒。有关详细信息,请参阅代码中的注释。

#include <stdio.h>
#include "stm32l1xx.h"             
 // Keil::Device:Startup

 // initialization of GPIOB
void TIM4_Init(){
RCC->AHBRSTR |= RCC_AHBRSTR_GPIOARST;       /* Reset GPIOB clock         */
RCC->AHBENR |= RCC_AHBENR_GPIOBEN;           /* Enable GPIOB clock         */
GPIOB->MODER   &=   ~(0x03 << (2*6));       /* Clear bit 11 & 12 Alternate mode*/
GPIOB->MODER   |=   0x02 << (2*6);          /* set as Alternate mode*/
GPIOB->OSPEEDR &=   ~(0x03<< (2*6));        /* 40 MHz  speed        */
GPIOB->OSPEEDR |=   0x03<< (2*6);           /* 40 MHz  speed        */
GPIOB->PUPDR &=         ~(1<<6);            /* NO PULL-UP PULL-DOWN        */
GPIOB->OTYPER &=        ~(1<<6);            /* PUSH-PULL        */
GPIOB->AFR[0] |=        0x2 << (4*6);       /* Pin6 set as alternate function 2 (TIM4) */

RCC->APB1ENR |= RCC_APB1ENR_TIM4EN;
TIM4->PSC = 16; //prescale value, AHB/APB1 Fmax=32MHz / 2
TIM4->ARR = 20000-1; //motor Freq = 50Hz, Period(ARR)= 1/50 = 20000us


// initialization of TIM & PWM

TIM4->CCMR1 |= TIM_CCMR1_OC1M;  // 111: PWM mode 2 - In upcounting, channel 1 is inactive 
                               //as long as TIMx_CNT<TIMx_CCR1 else active. 
                              // In downcounting, channel 1 is active as long as 
                             //TIMx_CNT>TIMx_CCR1 else inactive.

TIM4->CCMR1 |= TIM_CCMR1_OC1PE;
TIM4->CR1 |= TIM_CR1_ARPE;
TIM4->CCER |= TIM_CCER_CC1E;
TIM4->EGR |= TIM_EGR_UG;
TIM4->SR &= ~TIM_SR_UIF;
TIM4->DIER |= TIM_DIER_UIE;
TIM4->CR1 |= TIM_CR1_CEN;
}

// set servo to 4 positions in sequence

int main(void){

int i;

int position=600; // initial motor position

TIM4_Init();

while (1){  

if ((position >=600)|| (position <=2100))
position = position+500;      // motor positions will be 600(0*), 1100(50*), 1600(100*)
                             //, 2100(150*)
TIM4->CCR1 = position; 

for(i=0;i<1000;i++); // short delay


   }


 }  
4

2 回答 2

1

这取决于您的时钟设置。您需要首先配置您的时钟,因为许多 L1 在复位后只有 2.097MHz 的默认时钟,您设置的周期为 0.152598951 sek,伺服接收到的脉冲宽度为 0.07 到 0.15sek,大约是长的 75 倍(我假设 PCS 应该是 15 以存档 16*20000 时钟周期)。通过您的设置,定时器时钟应该超过 100MHz,这对于 L1 来说是不可能的

计时器计算滴答声,滴答声需要多长时间取决于时钟速度和计时器时钟分频器。

要存档 20 毫秒,您需要将 PSC 设为零值且 ARR = 41939

当然做 1ms 将是 2097 和 2ms - 4194

您需要计算正确的值

配置 PLL 的最简单方法是使用 WYSIWYG CubeMX 时钟编辑器

于 2017-05-02T07:50:15.257 回答
0

正如@PeterJ 提到的,这实际上取决于您的时钟设置。我将在这里更精确一些。

您的 ARR 似乎很好,因为很明显您的周期是 20 毫秒。

但是您的 PSC 应根据 ARR 进行设置以达到 1kHz。为此,您必须知道时钟频率。

附带说明,根据 stm 文档,PSC 值应该比期望的值少 1 个单位(例如,如果您想要 16,则应该将 15 写入寄存器)。

计数器时钟频率 CK_CNT 等于 fCK_PSC / (PSC[15:0] + 1)

于 2017-05-03T09:28:34.727 回答