#一、端口重映射原理及部分/完全重映射 #二、端口重映射的配置 #三、PWM控制 ##1、通用定时器输出PWM ##2、PWM的工作原理 ##3、PWM的内部运作机制 ##4、PWM的模式 ##5、自动加载的预载寄存器 #四、定时器输出PWM结构体及库函数的配置 #五、项目硬件 #六、项目代码
一、端口重映射原理及部分/完全重映射
每个外设都有若干个输入输出引脚,一般这些引脚也都不是固定不变的,但为了让开发工程师更好的安排引脚的功能和走向,引入了重映射的功能。也就是一个外设的引脚除了具有默认的端口外,也还可以设置重映寄存器来把这个外设映射到其他GPIO端口。方便硬件工程师布线,减少干扰。 部分重映射: 功能外设的部分引脚重新映射,还有一部分引脚是原来的默认引脚,简言之,外设的功能不止能在默认引脚使用,还可以在其他引脚使用。
完全重映射: 功能外设的所有引脚都是重新映射。
二、端口重映射的配置
1.使能GPIO引脚(重映射后的GPIO引脚) 2.使能功能外设 3.使能AFIO时钟,重映射必须使能AFIO时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE); 4.开启重映射: void GPIO_PinRemapConfig(uint32_t GPIO_Remap,FunctionalState NewState);
三、PWM控制
1、通用定时器输出PWM
以TIM3为例,STM32的通用定时器分为TIM2、TIM3、TIM4、TIM5,每个定时器都有独立的四个通道可以用来作为:输入捕获,输出比较,PWM输出,单脉冲模式输出等。
STM32的定时器除了TIM6和TIM7(基本定时器)之外,其他的定时器都可以产生PWM波输出,高级定时器TIM1,TIM8可以同时产生7路PWM输出,而通用定时器可以同时产生4路PWM输出,这样STM32可以最多同时输出30路PWM输出。 2、PWM的工作原理
以向上计数为例,讲述PWM原理:
1.在PWM输出模式下除了CNT(计数器当值),ARR(自动重装载值),CCRx(捕获/比较寄存器值) 2.在CNT小于CCRx时,TIMx_CHx通道输出低电平 3.在CNT大于或等于CCRx时,TIM_CHx通道输出高电平
所谓脉宽调制信号(PWM波),就是一个TIMx_ARR自动重装载寄存器确定频率(由它决定pWM周期),TIM_CCRx寄存器确定占空比信号。
3、PWM的内部运作原理
CCR1:设置捕获比较器寄存器,设置比较值 CCMR1:设置PWM模式1或者PWM模式2 CCER: P位:输出/捕获:设置极性:0高电平有效,1低电平有效 E位:输出/捕获:使能串口 4、PWM的模式
1)模式一:边沿对齐模式
向上计数时:当TIMx_CNT<TIM_CCRx时通道1为有效电平,否则为无效电平; 向下计数时:一旦TIMx_CNT>TIMx_CCRx,CCR1通道1为无效电平,否则为有限电平;
2)模式二:中央对齐模式
向上计数时:当TIMx_CNT<TIMx_CCRx时通道1为无效电平,否则为有效电平; 向下计数时:一旦TIMx_CNT>TIMx_CCRx,CCR1通道1为有效电平,否则为无效电平; 5、自动加载的预加载寄存器
APER=1,ARR立即生效 APER=2,ARR下个周期生效 void TIM_ARRPreloadConfig(TIM_TypeDef * TIMx,FunctionalState NewState);
四、定时器输出PWM结构体及库函数的配置 1)打开时钟:GPIO时钟、TIM定时器时钟、部分重映射时钟 2)配置GPIO结构体 3)配置通用定时器结构体 4)配置定时输出PWM结构体 5)在主函数中配置PWM比较值
五、项目硬件
SG90电机(舵机)
硬件接线: 红线:3.3V/5V 黑线:GND 黄线:信号线
六、项目代码
motor.c
#include "stm32f10x.h"
#include "motor.h"
void motor_config(void)
{
GPIO_InitTypeDef GPIO_Motorinit;
TIM_TimeBaseInitTypeDef TIM_Motorinit;
TIM_OCInitTypeDef TIMPWM_Motorinit;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3,ENABLE);
GPIO_Motorinit.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Motorinit.GPIO_Pin = GPIO_Pin_5;
GPIO_Motorinit.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB,&GPIO_Motorinit);
TIM_Motorinit.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_Motorinit.TIM_CounterMode = TIM_CounterMode_Up;
TIM_Motorinit.TIM_Period = 200-1;
TIM_Motorinit.TIM_Prescaler = 7200-1;
TIM_TimeBaseInit(TIM3,&TIM_Motorinit);
TIMPWM_Motorinit.TIM_OCMode = TIM_OCMode_PWM1;
TIMPWM_Motorinit.TIM_OutputState = TIM_OutputState_Enable;
TIMPWM_Motorinit.TIM_OCPolarity = TIM_OCNPolarity_Low;
TIM_OC2Init(TIM3,&TIMPWM_Motorinit);
TIM_Cmd(TIM3,ENABLE);
TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable);
}
motor.h
#include "stm32f10x.h"
void motor_config(void);
main.c
#include "stm32f10x.h"
#include "main.h"
#include "LED.h"
#include "usart.h"
#include "relay.h"
#include "shake.h"
#include "exti.h"
#include "tim.h"
#include "motor.h"
void delay(uint16_t time)
{
uint16_t i=0;
while(time--)
{
i=12000;
while(i--);
}
}
int main()
{
uint16_t pwmval=155;
usart_init();
tim_config();
LED_Init();
motor_config();
while(1)
{
for(pwmval=195;pwmval>=175;pwmval=-5)
{
TIM_SetCompare2(TIM3,pwmval);
delay(500);
}
}
}
usart.c(串口)
#include "stm32f10x.h"
#include "usart.h"
#include <stdio.h>
void usart_init(void)
{
GPIO_InitTypeDef gpioinstructure;
USART_InitTypeDef usartinstructure;
NVIC_InitTypeDef nvicinstructure;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
gpioinstructure.GPIO_Mode = GPIO_Mode_AF_PP;
gpioinstructure.GPIO_Pin = GPIO_Pin_9 ;
gpioinstructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&gpioinstructure);
gpioinstructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
gpioinstructure.GPIO_Pin = GPIO_Pin_10;
GPIO_Init(GPIOA,&gpioinstructure);
usartinstructure.USART_BaudRate = 115200;
usartinstructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
usartinstructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx ;
usartinstructure.USART_Parity = USART_Parity_No;
usartinstructure.USART_StopBits = USART_StopBits_1;
usartinstructure.USART_WordLength = USART_WordLength_8b;
USART_Init(USART1,&usartinstructure);
USART_Cmd(USART1,ENABLE);
USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
nvicinstructure.NVIC_IRQChannel = USART1_IRQn;
nvicinstructure.NVIC_IRQChannelCmd = ENABLE;
nvicinstructure.NVIC_IRQChannelPreemptionPriority = 1;
nvicinstructure.NVIC_IRQChannelSubPriority = 1;
NVIC_Init(&nvicinstructure);
}
void USARTSendByte(USART_TypeDef* USARTx, uint16_t Data)
{
USART_SendData(USARTx, Data);
while(USART_GetFlagStatus(USARTx,USART_FLAG_TXE)==RESET);
}
void USARTSendStr(USART_TypeDef* USARTx, char *str)
{
uint16_t i=0;
do
{
USARTSendByte(USARTx,*(str+i));
i++;
}while(*(str+i)!='\0');
while(USART_GetFlagStatus(USARTx,USART_FLAG_TC)==RESET);
}
int fputc(int ch,FILE *f)
{
USART_SendData(USART1,(uint8_t)ch);
while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
return (ch);
}
int fgetc(FILE *f)
{
while(USART_GetFlagStatus(USART1,USART_FLAG_RXNE)==RESET);
return (int)USART_ReceiveData(USART1);
}
usart.h
#include "stm32f10x.h"
#include <stdio.h>
void usart_init(void);
void USARTSendByte(USART_TypeDef* USARTx, uint16_t Data);
void USARTSendStr(USART_TypeDef* USARTx, char *str);
|