系统中断管理:NVIC
我的理解——管理系统内部的中断,负责打开和关闭中断。
基础应用 1,中断的初始化函数,包括设置中断向量表位置,和开启所需的中断两部分。所
有程序中必须的。
用法: void NVIC_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
#ifdef VECT_TAB_RAM
更改内容的表格)
NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);
#else
NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);
#endif
NVIC_PriorityGroup_x 可以是 0、1、2、3、4,分别代表抢占优先级有 1、2、4、8、16 个和
响应优先级有 16、8、4、2、1 个。规定两种优先级的数量后,所有的中断级别必须在其中
选择,抢占级别高的会打断其他中断优先执行,而响应级别高的会在其他中断执行完优先执
行。
}
阅读 rcc:单片机时钟管理。
我的理解——管理外部、内部和外设的时钟,设置、打开和关闭这些时钟。
基础应用 1:时钟的初始化函数过程——
用法:void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus;
RCC_DeInit();
RCC_HSEConfig(RCC_HSE_ON);
HSEStartUpStatus = RCC_WaitForHSEStartUp();
if (HSEStartUpStatus == SUCCESS)
{
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
FLASH_SetLatency(FLASH_Latency_2);
RCC_HCLKConfig(RCC_SYSCLK_Div1);
RCC_PCLK2Config(RCC_HCLK_Div2);
RCC_PCLK1Config(RCC_HCLK_Div2);
USB,SPI,I2C,CAN,串口 2345,普通 TIM。
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
RCC_PLLCmd(ENABLE);
while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){}
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
while (RCC_GetSYSCLKSource() != 0x08){}
}
1、阅读 exti:外部设备中断函数
我的理解——外部设备通过引脚给出的硬件中断,也可以产生软件中断,19 个上升、下降
或都触发。EXTI0~EXTI15 连接到管脚,EXTI 线 16 连接到 PVD(VDD 监视)
,EXTI 线 17 连接
到 RTC(闹钟),EXTI 线 18 连接到 USB(唤醒)。
基础应用 1,设定外部中断初始化函数。按需求,不是必须代码。
用法: void EXTI_Configuration(void)
{
EXTI_InitTypeDef EXTI_InitStructure;
EXTI_InitStructure.EXTI_Line = 通道 1|通道 2;
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
EXTI_InitStructure.EXTI_LineCmd = ENABLE;
EXTI_Init(&EXTI_InitStructure);
}
2、阅读 dma:通过总线而越过 CPU 读取外设数据
我的理解——通过 DMA 应用可以加速单片机外设、存储器之间的数据传输,并在传输期间
不影响 CPU 进行其他事情。这对于入门开发基本功能来说没有太大必要,这个内容先行跳
过。
3、阅读 systic:系统定时器
我的理解——可以输出和利用系统时钟的计数、状态。
基础应用 1,精确计时的延时子函数。推荐使用的代码。
用法:
static vu32 TimingDelay;
void SysTick_Config(void)
{
SysTick_CounterCmd(SysTick_Counter_Disable);
SysTick_ITConfig(DISABLE);
SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
SysTick_SetReload(9000);
SysTick_ITConfig(ENABLE);
}
void Delay (u32 nTime)
{
SysTick_CounterCmd(SysTick_Counter_Enable);
TimingDelay = nTime;
while(TimingDelay != 0);
SysTick_CounterCmd(SysTick_Counter_Disable);
SysTick_CounterCmd(SysTick_Counter_Clear);
}
void TimingDelay_Decrement(void)
{
if (TimingDelay != 0x00)
{
TimingDelay‐‐;
}
}
注:建议熟练后使用,所涉及知识和设备太多,新手出错的可能性比较大。新手可用简化的
延时函数代替:
void Delay(vu32 nCount)
{
for(; nCount != 0; nCount‐‐);(循环变量递减计数)
}
当延时较长,又不需要精确计时的时候可以使用嵌套循环:
void Delay(vu32 nCount)
for(; nCount != 0; nCount‐‐)
{for (i=0; i<0xffff; i++)}
}
4、阅读 gpio:I/O 设置函数
我的理解——所有输入输出管脚模式设置,可以是上下拉、浮空、开漏、模拟、推挽模式,
频率特性为 2M,10M,50M。也可以向该管脚直接写入数据和读取数据。
基础应用 1,gpio 初始化函数。所有程序必须。
用法:void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_标号 | GPIO_Pin_标号 ;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOC, &GPIO_InitStructure);
FLATING。如果其中任意一行与前一组相应设置相同,那么那一行可以省略,由此推论如果
前面已经将此行参数设定为默认参数(包括使用 GPIO_InitTypeDef GPIO_InitStructure 代码),
本组应用也是默认参数的话,那么也可以省略。以下重复这个过程直到所有应用的管脚全部
被定义完毕。
......
}
基础应用 2,向管脚写入 0 或 1
用法:GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x01);
sw 笨笨的 STM32 笔记之七:让它跑起来,基本硬件
功能的建立
SW笨笨 发表于 2009 年 02 月 27 日 09:00 阅读(46) 评论(0) 分类: 个人日记
举报
0、 实验之前的准备
a) 接通串口转接器
b) 下载 IO 与串口的原厂程序,编译通过保证调试所需硬件正常。
1、 flash,lib,nvic,rcc 和 GPIO,基础程序库编写
a) 这几个库函数中有一些函数是关于芯片的初始化的,每个程序中必用。为保障程序品
质,初学阶段要求严格遵守官方习惯。注意,官方程序库例程中有个 platform_config.h 文件,
是专门用来指定同类外设中第几号外设被使用,就是说在 main.c 里面所有外设序号用 x 代
替,比如 USARTx,程序会到这个头文件中去查找到底是用那些外设,初学的时候参考例程
别被这个所迷惑住。
b) 全部必用代码取自库函数所带例程,并增加逐句注释。
c) 习惯顺序——Lib(debug),RCC(包括 Flash 优化),NVIC,GPIO
d) 必用模块初始化函数的定义:
void RCC_Configuration(void);
void GPIO_Configuration(void);
void NVIC_Configuration(void);
void Delay(vu32 nCount);
e) Main 中的初始化函数调用:
RCC_Configuration();
NVIC_Configuration();
GPIO_Configuration();
f) Lib 注意事项:
属于 Lib 的 Debug 函数的调用,应该放在 main 函数最开始,不要改变其位置。
g) RCC 注意事项:
Flash 优化处理可以不做,但是两句也不难也不用改参数......
根据需要开启设备时钟可以节省电能
时钟频率需要根据实际情况设置参数
h) NVIC 注意事项
注意理解占先优先级和响应优先级的分组的概念
i) GPIO 注意事项
注意以后的过程中收集不同管脚应用对应的频率和模式的设置。
作为高低电平的 I/O,所需设置:RCC 初始化里面打开 RCC_APB2
PeriphClockCmd(RCC_APB2Periph_GPIOA);GPIO 里面管脚设定:IO 输出(50MHz,Out_PP);
IO 输入(50MHz,IPU);
j) GPIO 应用 GPIO_WriteBit(GPIOB, GPIO_Pin_2, Bit_RESET);
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x01);
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x00);
GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_6) ;
k) 简单 Delay 函数
void Delay(vu32 nCount)
{for(; nCount != 0; nCount‐‐);}
实验步骤:
RCC
初 始 化 函 数 里 添 加 : RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 |
RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB , ENABLE);
不用其他中断,NVIC 初始化函数不用改
GPIO 初始化代码:
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 ;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOB, &GPIO_InitStructure);
简单的延迟函数:
void Delay(vu32 nCount)
{ for (; nCount != 0; nCount‐‐);}
完成之后再在 main.c 的 while 里面写一段:
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x01);
Delay(0xffff);
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x00);
Delay(0xffff);
就可以看到连接在 PB2 脚上的 LED 闪烁了,单片机就跑起来了。
sw 笨笨的 STM32 笔记之八:来跟 PC 打个招呼,基本串口通讯
a) 目的:在基础实验成功的基础上,对串口的调试方法进行实践。硬件代码顺利完成之
后,对日后调试需要用到的 printf 重定义进行调试,固定在自己的库函数中。
b) 初始化函数定义:
void USART_Configuration(void);
c) 初始化函数调用:
void UART_Configuration(void);
初始化代码:
void USART_Configuration(void)
{
USART_InitTypeDef USART_InitStructure;
USART_InitStructure.USART_BaudRate = 9600;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
流控制
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
x 发送功能
USART_Init(USART1, &USART_InitStructure);
USART_Cmd(USART1, ENABLE);
}
RCC 中打开相应串口
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE);
GPIO 里面设定相应串口管脚模式
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
d) 简单应用:
发送一位字符
USART_SendData(USART1, 数据);
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
{}
接收一位字符
while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET)
{}
变量= (USART_ReceiveData(USART1));
发送一个字符串
先定义字符串:char rx_data[250];
然后在需要发送的地方添加如下代码
int i;
while(rx_data!='\0')
{USART_SendData(USART1, rx_data);
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}
i++;}
e) USART 注意事项:
发动和接受都需要配合标志等待。
只能对一个字节操作,对字符串等大量数据操作需要写函数
使用串口所需设置: RCC 初始化里面打开 RCC_APB2PeriphClockCmd
(RCC_APB2Periph_USARTx);GPIO 里面管脚设定:串口 RX ( 50Hz , IN_FLOATING );串口 TX ( 5
0Hz , AF_PP );
f) printf 函数重定义(不必理解,调试通过以备后用)
( 1 ) 需要 c 标准函数:
#include "stdio.h"
( 2 ) 粘贴函数定义代码
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
( 3 ) RCC 中打开相应串口
( 4 ) GPIO 里面设定相应串口管脚模式
( 6 ) 增加为 putchar 函数。
int putchar(int c)
if (c == '\n'){putchar('\r');}
USART_SendData(USART1, c);
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}
return c;
}
( 8 ) 通过,试验成功。 printf 使用变量输出 :%c 字符, %d 整数, %f 浮点数 ,%s 字符串,
/n 或 /r 为换行。注意:只能用于 main.c 中。
3 、 NVIC 串口中断的应用
a) 目的:利用前面调通的硬件基础,和几个函数的代码,进行串口的中断输入练习。因
为在实际应用中,不使用中断进行的输入是效率非常低的,这种用法很少见,大部分串口的
输入都离不开中断。
b) 初始化函数定义及函数调用:不用添加和调用初始化函数,在指定调试地址的时候已
经调用过,在那个 NVIC_Configuration 里面添加相应开中断代码就行了。
c) 过程:
i. 在串口初始化中 USART_Cmd 之前加入中断设置:
USART_ITConfig(USART1, USART_IT_TXE, ENABLE);
接收中断, PE 奇偶错误中断,可以是多个。
ii. RCC 、 GPIO 里面打开串口相应的基本时钟、管脚设置
iii. NVIC 里面加入串口中断打开代码:
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
iv. 在 stm32f10x_it.c 文件中找到 void USART1_IRQHandler 函数,在其中添入执行代码。
一般最少三个步骤:先使用 if 语句判断是发生那个中断,然后清除中断标志位,最后给字符
串赋值,或做其他事情。
void USART1_IRQHandler(void)
{
char RX_dat;
if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
GPIO_WriteBit(GPIOB, GPIO_Pin_10, (BitAction)0x01);
RX_dat=USART_ReceiveData(USART1) & 0x7F;
USART_SendData(USART1, RX_dat);
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET){}
}
}
d) 中断注意事项:
可以随时在程序中使用 USART_ITConfig(USART1, USART_IT_TXE, DISABLE); 来关闭中断响应。
NVIC_InitTypeDef NVIC_InitStructure 定义一定要加在 NVIC 初始化模块的第一句。
全局变量与函数的定义:在任意 .c 文件中定义的变量或函数,在其它 .c 文件中使用 extern+
定义代码再次定义就可以直接调用了。
sw 笨笨的 STM32 笔记之九:打断它来为我办事,EXI
T (外部 I/O 中断)应用
a) 目的:跟串口输入类似,不使用中断进行的 IO 输入效率也很低,而且可以通过 E
XTI 插入按钮事件,本节联系 EXTI 中断。
b) 初始化函数定义:
void EXTI_Configuration(void);
c) 初始化函数调用:
EXTI_Configuration();
d) 初始化函数:
void EXTI_Configuration(void)
{
EXTI_InitTypeDef EXTI_InitStructure;
EXTI_ClearITPendingBit(EXTI_LINE_KEY_BUTTON);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource3);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource4);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource5);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource6);
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
EXTI_InitStructure.EXTI_Line = EXTI_Line3 | EXTI_Line4;
EXTI_InitStructure.EXTI_LineCmd = ENABLE;
EXTI_Init(&EXTI_InitStructure);
}
e) RCC 初始化函数中开启 I/O 时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);
GPIO 初始化函数中定义输入 I/O 管脚。
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_Init(GPIOA, &GPIO_InitStructure);
f) 在 NVIC 的初始化函数里面增加以下代码打开相关中断:
NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
g) 在 stm32f10x_it.c 文件中找到 void USART1_IRQHandler 函数,在其中添入执行代
码。一般最少三个步骤:先使用 if 语句判断是发生那个中断,然后清除中断标志位,最后给
字符串赋值,或做其他事情。
if(EXTI_GetITStatus(EXTI_Line3) != RESET)
断发生来源
{ EXTI_ClearITPendingBit(EXTI_Line3);
清除中断标志
USART_SendData(USART1, 0x41); /
/发送字符“a”
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)(1‐GPIO_ReadOutputDataBit(GPIOB, GPIO_
Pin_2)));
}
h) 中断注意事项:
中断发生后必须清除中断位,否则会出现死循环不断发生这个中断。然后需要对中断类型进
行判断再执行代码。
使用 EXTI 的 I/O 中断,在完成 RCC 与 GPIO 硬件设置之后需要做三件事:初始化 EXTI、NVIC 开中断、编写中断执行代码。
sw 笨笨的 STM32 笔记之十:工作工作,PWM 输出
a) 目的:基础 PWM 输出,以及中断配合应用。输出选用 PB1,配置为 TIM3_CH4,
是目标板的 LED6 控制脚。
b) 对于简单的 PWM 输出应用,暂时无需考虑 TIM1 的高级功能之区别。
c) 初始化函数定义:
void TIM_Configuration(void);
d) 初始化函数调用:
TIM_Configuration();
e) 初始化函数,不同于前面模块,TIM 的初始化分为两部分——基本初始化和通道
初始化:
void TIM_Configuration(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_OCInitTypeDef TIM_OCInitStructure;
TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
TIM_TimeBaseStructure.TIM_Prescaler = 5;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
TIM_ITConfig(TIM3, TIM_IT_CC4, ENABLE);
TIM_OCStructInit(& TIM_OCInitStructure);
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
状态
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
需要 PWM 输出才需要这行代码
TIM_OCInitStructure.TIM_Pulse = 0x2000;
度
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
TIM_OC4Init(TIM3, &TIM_OCInitStructure);
始化
TIM_Cmd(TIM3, ENABLE);
}
f) RCC 初始化函数中加入 TIM 时钟开启:
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM3, ENABLE);
g) GPIO 里面将输入和输出管脚模式进行设置。信号:AF_PP,50MHz。
h) 使用中断的话在 NVIC 里添加如下代码:
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
化
中断代码:
void TIM2_IRQHandler(void)
{
if (TIM_GetITStatus(TIM2, TIM_IT_CC4) != RESET)
{
TIM_ClearITPendingBit(TIM2, TIM_IT_CC4);
GPIO_WriteBit(GPIOB, GPIO_Pin_11, (BitAction)(1‐GPIO_ReadOutputDataBit(GPIOB, GPIO
_Pin_11)));
IC4value = TIM_GetCapture4(TIM2);
}
}
i) 简单应用:
TIM_SetCompare4(TIM3, 变量);
j) 注意事项:
管脚的 IO 输出模式是根据应用来定,比如如果用 PWM 输出驱动 LED 则应该将相应管脚设
为 AF_PP,否则单片机没有输出。
sw 笨笨的 STM32 笔记之十一:捕捉精彩瞬间,脉冲方波长度捕获
a) 目的:基础 PWM 输入也叫捕获,以及中断配合应用。使用前一章的输出管脚 P
B1(19 脚),直接使用跳线连接输入的 PA3(13 脚),配置为 TIM2_CH4,进行实验。
b) 对于简单的 PWM 输入应用,暂时无需考虑 TIM1 的高级功能之区别,按照目前
我的应用目标其实只需要采集高电平宽度,而不必知道周期,所以并不采用 PWM 输入模式,
而是普通脉宽捕获模式。
c) 初始化函数定义:
void TIM_Configuration(void);
d) 初始化函数调用:
TIM_Configuration();
e) 初始化函数,不同于前面模块,TIM 的 CAP 初始化分为三部分——计时器基本初
始化、通道初始化和时钟启动初始化:
void TIM_Configuration(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;
TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
TIM_TimeBaseStructure.TIM_Prescaler = 5;
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
TIM_ICInitStructure.TIM_Channel = TIM_Channel_4;
TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Falling;
TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;
TIM_ICInitStructure.TIM_ICFilter = 0x4;
稳定 0x0~0xF
TIM_ICInit(TIM2, &TIM_ICInitStructure);
TIM_SelectInputTrigger(TIM2, TIM_TS_TI2FP2);
TIM_SelectSlaveMode(TIM2, TIM_SlaveMode_Reset);
TIM_SelectMasterSlaveMode(TIM2, TIM_MasterSlaveMode_Enable);
触发
TIM_ITConfig(TIM2, TIM_IT_CC4, ENABLE);
TIM_Cmd(TIM2, ENABLE);
}
f) RCC 初始化函数中加入 TIM 时钟开启:
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM3, ENABLE);
g) GPIO 里面将输入和输出管脚模式进行设置。IN_FLOATING,50MHz。
h) 使用中断的话在 NVIC 里添加如下代码:
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
i) 简单应用:
变量 = TIM_GetCapture4(TIM2);
j) 注意事项:
i. 由于我的需求只跟高电平宽度有关,所以避免了使用 PWM 输入模式,这样可以
每个管脚捕捉一路信号。如果使用 PWM 模式,每一路需要占用两个寄存器,所以一个定时
器只能同时使用两路 PWM 输入。
ii. 由于捕捉需要触发启动定时器,所以 PWM 输出与捕捉不容易在同一个 TIM 通道
上实现。如果必须的话只能增加计数溢出的相关代码。
iii. 有些程序省略了捕捉通道的初始化代码,这是不对的
iv. 在基本计时器初始化代码里面注意选择适当的计数器长度,最好让波形长度不要
长于一个计数周期,否则需要增加溢出代码很麻烦。一个计数周期的长度计算跟如下几个参
数有关:
(1) RCC 初始化代码里面的 RCC_PCLKxConfig,这是 TIM 的基础时钟源与系统时钟
的关系。
(2) TIM 初始化的 TIM_Period,这是计数周期的值
(3) TIM 初始化的 TIM_Prescaler,这是计数周期的倍频计数器,相当于调节计数
周期,可以使 TIM_Period 尽量大,提高计数精度。
sw 笨笨的 STM32 笔记之十二:时钟不息工作不止,
systic 时钟应用 a) 目的:使用系统时钟来进行两项实验——周期执行代码与精确定时延迟。
b) 初始化函数定义:
void SysTick_Configuration(void);
c) 初始化函数调用:
SysTick_Configuration();
d) 初始化函数:
void SysTick_Configuration(void)
{
SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
SysTick_SetReload(250000);
计数周期长度
SysTick_CounterCmd(SysTick_Counter_Enable);
SysTick_ITConfig(ENABLE);
打开中断
}
e) 在 NVIC 的初始化函数里面增加以下代码打开相关中断:
NVIC_SystemHandlerPriorityConfig(SystemHandler_SysTick, 1, 0);
高一些会少受其他影响
f) 在 stm32f10x_it.c 文件中找到 void SysTickHandler 函数
void SysTickHandler(void)
{
执行代码
}
g) 简单应用:精确延迟函数,因为 systic 中断往往被用来执行周期循环代码,所以
一些例程中使用其中断的启动和禁止来编写的精确延时函数实际上不实用,我自己编写了精
确计时函数反而代码更精简,思路更简单。思路是调用后,变量清零,然后使用时钟来的曾
变量,不断比较变量与延迟的数值,相等则退出函数。代码和步骤如下:
i. 定义通用变量:u16 Tic_Val=0;
ii. 在 stm32f10x_it.c 文件中相应定义:
extern u16 Tic_Val;
iii. 定义函数名称:void Tic_Delay(u16 Tic_Count);
iv. 精确延时函数:
void Tic_Delay(u16 Tic_Count)
{ Tic_Val=0;
while(Tic_Val != Tic_Count){printf("");}
}
v. 在 stm32f10x_it.c 文件中 void SysTickHandler 函数里面添加
Tic_Val++;
疑问:如果去掉计时行那个没用的 printf("");函数将停止工作,这个现象很奇
vii.
怪
sw 笨笨的 STM32 笔记之十三:恶搞,两只看门狗
a) 目的:
了解两种看门狗(我叫它:系统运行故障探测器和独立系统故障探测器,新手往往被这个并
不形象的象形名称搞糊涂)之间的区别和基本用法。
b) 相同:
都是用来探测系统故障,通过编写代码定时发送故障清零信号(高手们都管这个代码叫做“喂
狗” ),告诉它系统运行正常。一旦系统故障,程序清零代码( “喂狗” )无法执行,其计数器就
会计数不止,直到记到零并发生故障中断(狗饿了开始叫唤),控制 CPU 重启整个系统(不
行啦,开始咬人了,快跑......)。
c) 区别:
独立看门狗 Iwdg——我的理解是独立于系统之外,因为有独立时钟,所以不受系统影响的系
统故障探测器。(这条狗是借来的,见谁偷懒它都咬!)主要用于监视硬件错误。
窗口看门狗 wwdg——我的理解是系统内部的故障探测器,时钟与系统相同。如果系统时钟
不走了,这个狗也就失去作用了。
(这条狗是老板娘养的,老板不干活儿他不管!)主要用于
监视软件错误。
d) 初始化函数定义:鉴于两只狗作用差不多,使用过程也差不多初始化函数栓一起了,
用的时候根据情况删减。
void WDG_Configuration(void);
e) 初始化函数调用:
WDG_Configuration();
f) 初始化函数
void WDG_Configuration()
{
WWDG_SetPrescaler(WWDG_Prescaler_8);
WWDG_SetWindowValue(65);
WWDG_Enable(127);
WWDG_ClearFlag();
WWDG_EnableIT();
IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);
IWDG_SetPrescaler(IWDG_Prescaler_32);
IWDG_SetReload(349);
IWDG_ReloadCounter();
IWDG_Enable();
}
g) RCC 初始化:只有软件看门狗需要时钟初始化,独立看门狗有自己的时钟不需要但是
需要 systic 工作相关设置。
RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE);
h) 独立看门狗使用 systic 的中断来喂狗,所以添加 systic 的中断打开代码就行了。软件看
门狗需要在 NVIC 打开中断添加如下代码:
NVIC_InitStructure.NVIC_IRQChannel = WWDG_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_Init(&NVIC_InitStructure);
i) 中断程序,软件看门狗在自己的中断中喂狗,独立看门狗需要使用 systic 的定时中断来
喂狗。以下两个程序都在 stm32f10x_it.c 文件中。
void WWDG_IRQHandler(void)
{
WWDG_SetCounter(0x7F);
WWDG_ClearFlag();
}
void SysTickHandler(void)
{ IWDG_ReloadCounter();
}
j) 注意事项:
i. 有狗平常没事情可以不理,但是千万别忘了喂它,否则死都不知道怎么死的!
ii. 初始化程序的调用一定要在 systic 的初始化之后。
iii. 独立看门狗需要 systic 中断来喂,但是 systic 做别的用处不能只做这件事,所以我写
了如下几句代码,可以不影响 systic 的其他应用,其他 systic 周期代码也可参考:
第一步:在 stm32f10x_it.c 中定义变量
int Tic_IWDG;
Tic_IWDG++;
if(Tic_IWDG>=100)
{ IWDG_ReloadCounter();
Tic_IWDG=0;
}