STM32各模块学习笔记



STm32开发板PCB完整工程链接:http://www.cirmall.com/circuit/7 ... F%EF%BC%81#/details



STM32中断优先级和开关总中断

一,中断优先级:

STM32(Cortex-M3)中的优先级概念
STM32(Cortex-M3)中有两个优先级的概念——抢占式优先级和响应优先级,有人把响应优先级称做'亚优先级'或'副优先级',每一个中断源都须要被指定这两种优先级。

具备高抢占式优先级的中断能够在具备低抢占式优先级的中断处理过程当中被响应,即中断嵌套,或者说高抢占式优先级的中断能够嵌套低抢占式优先级的中断。

当两个中断源的抢占式优先级相同时,这两个中断将没有嵌套关系,当一个中断到来后,若是正在处理另外一个中断,这个后到来的中断就要等到前一个中断处理完以后才能被处理。若是这两个中断同时到达,则中断控制器根据他们的响应优先级高低来决定先处理哪个;若是他们的抢占式优先级和响应优先级都相等,则根据他们在中断表中的排位顺序决定先处理哪个。

既然每一个中断源都须要被指定这两种优先级,就须要有相应的寄存器位记录每一个中断的优先级;在Cortex-M3中定义了8个比特位用于设置中断源的优先级,这8个比特位能够有8种分配方式,以下:

全部8位用于指定响应优先级
最高1位用于指定抢占式优先级,最低7位用于指定响应优先级
最高2位用于指定抢占式优先级,最低6位用于指定响应优先级
最高3位用于指定抢占式优先级,最低5位用于指定响应优先级
最高4位用于指定抢占式优先级,最低4位用于指定响应优先级
最高5位用于指定抢占式优先级,最低3位用于指定响应优先级
最高6位用于指定抢占式优先级,最低2位用于指定响应优先级
最高7位用于指定抢占式优先级,最低1位用于指定响应优先级

这就是优先级分组的概念。




--------------------------------------------------------------------------------
Cortex-M3容许具备较少中断源时使用较少的寄存器位指定中断源的优先级,所以STM32把指定中断优先级的寄存器位减小到4位,这4个寄存器位的分组方式以下: 

第0组:全部4位用于指定响应优先级
第1组:最高1位用于指定抢占式优先级,最低3位用于指定响应优先级
第2组:最高2位用于指定抢占式优先级,最低2位用于指定响应优先级
第3组:最高3位用于指定抢占式优先级,最低1位用于指定响应优先级
第4组:全部4位用于指定抢占式优先级

能够经过调用STM32的固件库中的函数NVIC_PriorityGroupConfig()选择使用哪一种优先级分组方式,这个函数的参数有下列5种:

NVIC_PriorityGroup_0 => 选择第0组
NVIC_PriorityGroup_1 => 选择第1组
NVIC_PriorityGroup_2 => 选择第2组
NVIC_PriorityGroup_3 => 选择第3组
NVIC_PriorityGroup_4 => 选择第4组 

接下来就是指定中断源的优先级,下面以一个简单的例子说明如何指定中断源的抢占式优先级和响应优先级:

// 选择使用优先级分组第1组
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  
// 使能EXTI0中断
NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; // 指定抢占式优先级别1

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; // 指定响应优先级别0
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
  
// 使能EXTI9_5中断
NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; // 指定抢占式优先级别0
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; // 指定响应优先级别1
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

要注意的几点是:

1)若是指定的抢占式优先级别或响应优先级别超出了选定的优先级分组所限定的范围,将可能获得意想不到的结果;

2)抢占式优先级别相同的中断源之间没有嵌套关系;

3)若是某个中断源被指定为某个抢占式优先级别,又没有其它中断源处于同一个抢占式优先级别,则能够为这个中断源指定任意有效的响应优先级别。

二,开关总中断:

在STM32/Cortex-M3中是经过改变CPU的当前优先级来容许或禁止中断。
PRIMASK位:只容许NMI和hard fault异常,其余中断/异常都被屏蔽(当前CPU优先级=0)。
FAULTMASK位:只容许NMI,其余全部中断/异常都被屏蔽(当前CPU优先级=-1)。

在STM32固件库中(stm32f10x_nvic.c和stm32f10x_nvic.h) 定义了四个函数操做PRIMASK位和FAULTMASK位,改变CPU的当前优先级,从而达到控制全部中断的目的。

下面两个函数等效于关闭总中断:
void NVIC_SETPRIMASK(void);
void NVIC_SETFAULTMASK(void);

下面两个函数等效于开放总中断:
void NVIC_RESETPRIMASK(void);
void NVIC_RESETFAULTMASK(void);

上面两组函数要成对使用,不能交叉使用。

例如:

第一种方法:
NVIC_SETPRIMASK();   //关闭总中断
NVIC_RESETPRIMASK();//开放总中断

第二种方法:
NVIC_SETFAULTMASK();   //关闭总中断
NVIC_RESETFAULTMASK();//开放总中断

经常使用

NVIC_SETPRIMASK();                   // Disable Interrupts
NVIC_RESETPRIMASK();                 // Enable Interrupts





STM32时钟系统

STM32资料   2009-09-23 14:53   阅读72   评论0   
字号: 大大  中中  小小 
在STM32中,有五个时钟源,为HSI、HSE、LSI、LSE、PLL。



①、HSI是高速内部时钟,RC振荡器,频率为8MHz。

②、HSE是高速外部时钟,可接石英/陶瓷谐振器,或者接外部时钟源,频率范围为4MHz~16MHz。

③、LSI是低速内部时钟,RC振荡器,频率为40kHz。

④、LSE是低速外部时钟,接频率为32.768kHz的石英晶体。

⑤、PLL为锁相环倍频输出,其时钟输入源可选择为HSI/二、HSE或者HSE/2。倍频可选择为2~16倍,可是其输出频率最大不得超过72MHz。

 



图1 HSE/LSE时钟源

其中40kHz的LSI供独立看门狗IWDG使用,另外它还能够被选择为实时时钟RTC的时钟源。另外,实时时钟RTC的时钟源还能够选择LSE,或者是HSE的128分频。RTC的时钟源经过RTCSEL[1:0]来选择。

STM32中有一个全速功能的USB模块,其串行接口引擎须要一个频率为48MHz的时钟源。该时钟源只能从PLL输出端获取,能够选择为1.5分频或者1分频,也就是,当须要使用USB模块时,PLL必须使能,而且时钟频率配置为48MHz或72MHz。

另外,STM32还能够选择一个时钟信号输出到MCO脚(PA8)上,能够选择为PLL输出的2分频、HSI、HSE、或者系统时钟。

系统时钟SYSCLK,它是供STM32中绝大部分部件工做的时钟源。系统时钟可选择为PLL输出、HSI或者HSE。系统时钟最大频率为72MHz,它经过AHB分频器分频后送给各模块使用,AHB分频器可选择一、二、四、八、1六、6四、12八、25六、512分频。其中AHB分频器输出的时钟送给5大模块使用:

①、送给AHB总线、内核、内存和DMA使用的HCLK时钟。

②、经过8分频后送给Cortex的系统定时器时钟。

③、直接送给Cortex的空闲运行时钟FCLK。

④、送给APB1分频器。APB1分频器可选择一、二、四、八、16分频,其输出一路供APB1外设使用(PCLK1,最大频率36MHz),另外一路送给定时器(Timer)二、三、4倍频器使用。该倍频器可选择1或者2倍频,时钟输出供定时器二、三、4使用。

⑤、送给APB2分频器。APB2分频器可选择一、二、四、八、16分频,其输出一路供APB2外设使用(PCLK2,最大频率72MHz),另外一路送给定时器(Timer)1倍频器使用。该倍频器可选择1或者2倍频,时钟输出供定时器1使用。另外,APB2分频器还有一路输出供ADC分频器使用,分频后送给ADC模块使用。ADC分频器可选择为二、四、六、8分频。

在以上的时钟输出中,有不少是带使能控制的,例如AHB总线时钟、内核时钟、各类APB1外设、APB2外设等等。当须要使用某模块时,记得必定要先使能对应的时钟。

须要注意的是定时器的倍频器,当APB的分频为1时,它的倍频值为1,不然它的倍频值就为2。链接在APB1(低速外设)上的设备有:电源接口、备份接口、CAN、USB、I2C一、I2C二、UART二、UART三、SPI二、窗口看门狗、Timer二、Timer三、Timer4。注意USB模块虽然须要一个单独的48MHz时钟信号,但它应该不是供USB模块工做的时钟,而只是提供给串行接口引擎(SIE)使用的时钟。USB模块工做的时钟应该是由APB1提供的。链接在APB2(高速外设)上的设备有:UART一、SPI一、Timer一、ADC一、ADC二、全部普通IO口(PA~PE)、第二功能IO口。

下图是STM32用户手册中的时钟系统结构图,经过该图能够从整体上掌握STM32的时钟系统。

 



STM32外部中断之二

STM32资料   2009-09-10 21:18   阅读243   评论0   
字号: 大大  中中  小小 
STM32 外部中断配置


1配置中断

一、 分配中断向量表:

/* Set the Vector Table base location at 0x20000000 */ 

NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);

二、 设置中断优先级:

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); //设置中断优先级

三、 初始化外部中断:

/*容许EXTI4中断 */

       NVIC_InitStructure.NVIC_IRQChannel =EXTI4_IRQChannel; //中断通道

       NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority= PreemptionPriorityValue;//强占优先级

       NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//次优先级

       NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //通道中断使能

       NVIC_Init(&NVIC_InitStructure);      //初始化中断

注意:若是咱们配置的外部针脚为PA4,或PB4,或PC4,PD4等,那么采用的外部中断也必须是EXTI4,一样,若是外部中断针脚是PA1,PB1,PC1,PD1 那么中断就要用EXTI1,其余类推。

2配置GPIO针脚做为外部中断的触发事件 

一、 选择IO针脚

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;

注意,若是的针脚是端口的4号针脚,配置的中断必定是EXTI4

二、 配置针脚为输入

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

三、 初始化针脚

GPIO_Init(GPIOD,&GPIO_InitStructure);


3配置EXTI线,使中断线和IO针脚线链接上

一、 将EXTI线链接到IO端口上

将EXTI线4链接到端口GPIOD的第4个针脚上

       GPIO_EXTILineConfig(GPIO_PortSourceGPIOD,GPIO_PinSource4);

       注意:若是配置的针脚是4号,那么参数必须是GPIO_PinSource4

              若是配置的针脚是3号,那么参数必须是GPIO_PinSource3

二、配置中断边沿

       /*配置EXTI线0上出现降低沿,则产生中断*/

       EXTI_InitStructure.EXTI_Line = EXTI_Line4;

       注意:若是配置的4号针脚,那么EXTI_Line4是必须的

       EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

       EXTI_InitStructure.EXTI_Trigger =EXTI_Trigger_Falling; //降低沿触发

       EXTI_InitStructure.EXTI_LineCmd = ENABLE;    //中断线使能

       EXTI_Init(&EXTI_InitStructure);                //初始化中断

       EXTI_GenerateSWInterrupt(EXTI_Line4);        //EXTI_Line4中断容许

到此中断配置完成,能够写中断处理函数。

举例:

配置函数

/*************************************************************************

* 函数名      NVIC_Configration

* 描述          配置各个中断寄存器

* 输入           无 

* 输出          无

* 返回值       无

****************************************************************************/

void NVIC_Configration(void)

{

        NVIC_InitTypeDef NVIC_InitStructure; 

//#ifdef VECT_TAB_RAM 

      /* Set the Vector Table base location at 0x20000000*/ 

      NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); 

//#else /* VECT_TAB_FLASH */

      /* Set the Vector Table base location at 0x08000000*/ 

      //NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);  

//#endif   

       NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); //设置中断优先级

       /*容许EXTI4中断 */

       NVIC_InitStructure.NVIC_IRQChannel =EXTI4_IRQChannel;

       NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority= PreemptionPriorityValue;

       NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

       NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

       NVIC_Init(&NVIC_InitStructure);      

       /*容许EXTI9中断*/

       NVIC_InitStructure.NVIC_IRQChannel =EXTI9_5_IRQChannel;

       NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority= 0;

       NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;

       NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

       NVIC_Init(&NVIC_InitStructure);      

       /*配置SysTick处理优先级:优先级以及子优先级*/

       

}

/************************************************************************

* 函数名      :GPIO_Configuration(void)

* 描述          :配置TIM2阵脚

* 输入           :无

* 输出      :无

* 返回           :无

************************************************************************/

void GPIO_Configuration(void){

/*    GPIO_InitTypeDef GPIO_InitStructure;

       GPIO_InitStructure.GPIO_Pin =GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;

       GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

       GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

       GPIO_Init(GPIOA,&GPIO_InitStructure); */

       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All;

       GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

       GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;

       GPIO_Init(GPIOC,&GPIO_InitStructure);

       /*配置GPIOD的第一个管角为浮动输入*/

       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;

       GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IN_FLOATING;

       GPIO_Init(GPIOD,&GPIO_InitStructure);

       /*配置GPIOB的第9个管脚为浮动输入*/

       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

       GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IN_FLOATING;

       GPIO_Init(GPIOB,&GPIO_InitStructure); 

}


/**************************************************************

* 函数          SysTick_Configuration

* 描述          设置SysTick

* 输入           无

* 输出          无

* 返回值      无

***************************************************************/

void SysTick_Configuration(void)

{

      /*配置 HCLK 时钟作为SysTick 时钟源*/

       SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);//系统时钟8分频 72MHz

      NVIC_SystemHandlerPriorityConfig(SystemHandler_SysTick, 8,2);

       /*SysTick Interrupt each 1000Hz with HCLK equal to72MHz*/

       SysTick_SetReload(9000);//中断周期1ms

       /*Enable the SysTick Interrupt */

       SysTick_ITConfig(ENABLE);//打开中断           

       SysTick_CounterCmd(SysTick_Counter_Enable);

       SysTick_CounterCmd(SysTick_Counter_Clear);                

}


/******************************************************************************

* 函数名       EXTI_Configuration

* 描述           配置EXTI线

* 输入           无 

* 输出          无

* 返回值       无

******************************************************************************/

void EXTI_Configuration(void){

       /*将EXTI线0链接到PA0*/

       GPIO_EXTILineConfig(GPIO_PortSourceGPIOD,GPIO_PinSource4);

       /*配置EXTI线0上出现降低沿,则产生中断*/

       EXTI_InitStructure.EXTI_Line = EXTI_Line4;

       EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

       EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;

       EXTI_InitStructure.EXTI_LineCmd = ENABLE;

       EXTI_Init(&EXTI_InitStructure);

       EXTI_GenerateSWInterrupt(EXTI_Line4);

       

       /*将EXTI线9链接到PB9上*/

      GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource9);

       /*将EXTI线9上出现降低沿产生中断*/

       EXTI_InitStructure.EXTI_Line = EXTI_Line9;

       EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

       EXTI_InitStructure.EXTI_Trigger =EXTI_Trigger_Falling;

       EXTI_InitStructure.EXTI_LineCmd = ENABLE;

       EXTI_Init(&EXTI_InitStructure);

       EXTI_GenerateSWInterrupt(EXTI_Line9);       

}

中断函数:

void EXTI4_IRQHandler(void)

{

       if(EXTI_GetITStatus(EXTI_Line4)!= RESET){

             EXTI_ClearITPendingBit(EXTI_Line4);

              if(Ledflag == 0){

                    Ledflag = 1;

                    GPIOC->ODR |= 0X00000080;

              }

              else{

                    Ledflag = 0;          

                    GPIOC->ODR &= 0XFFFFFF7F;

              }

       }

}

注:时钟设置的时候最好加上这句: 

RCCRCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); // 使能AFIO时钟





STM32中定时器的时钟源

STM32资料   2009-07-24 21:34   阅读277   评论0   
字号: 大大  中中  小小 



STM32中有多达8个定时器,其中TIM1和TIM8是可以产生三对PWM互补输出的高级定时器,经常使用于三相电机的驱动,它们的时钟由APB2的输出产生。其它6个为普通定时器,时钟由APB1的输出产生。

下图是STM32参考手册上时钟分配图中,有关定时器时钟部分的截图:



从图中能够看出,定时器的时钟不是直接来自APB1或APB2,而是来自于输入为APB1或APB2的一个倍频器,图中的蓝色部分。

下面以定时器2~7的时钟说明这个倍频器的做用:当APB1的预分频系数为1时,这个倍频器不起做用,定时器的时钟频率等于APB1的频率;当APB1的预分频系数为其它数值(即预分频系数为二、四、8或16)时,这个倍频器起做用,定时器的时钟频率等于APB1的频率两倍。

假定AHB=36MHz,由于APB1容许的最大频率为36MHz,因此APB1的预分频系数能够取任意数值;当预分频系数=1时,APB1=36MHz,TIM2~7的时钟频率=36MHz(倍频器不起做用);当预分频系数=2时,APB1=18MHz,在倍频器的做用下,TIM2~7的时钟频率=36MHz。

有人会问,既然须要TIM2~7的时钟频率=36MHz,为何不直接取APB1的预分频系数=1?答案是:APB1不但要为TIM2~7提供时钟,并且还要为其它外设提供时钟;设置这个倍频器能够在保证其它外设使用较低时钟频率时,TIM2~7仍能获得较高的时钟频率。

再举个例子:当AHB=72MHz时,APB1的预分频系数必须大于2,由于APB1的最大频率只能为36MHz。若是APB1的预分频系数=2,则由于这个倍频器,TIM2~7仍然可以获得72MHz的时钟频率。可以使用更高的时钟频率,无疑提升了定时器的分辨率,这也正是设计这个倍频器的初衷。

STM32笔记以外部中断GPIO

STM32资料   2009-07-14 13:35   阅读331   评论0   
字号: 大大  中中  小小 
b)        初始化函数定义:
void EXTI_Configuration(void); //定义IO中断初始化函数
c)        初始化函数调用:
EXTI_Configuration();//IO中断初始化函数调用简单应用:
d)        初始化函数:
void EXTI_Configuration(void)
{
  EXTI_InitTypeDef EXTI_InitStructure;       //EXTI初始化结构定义

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_Trigger = EXTI_Trigger_Falling;//触发模式
  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管脚。
//IO输入,GPIOA的4脚输入
  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)));//LED发生明暗交替
}
h)        中断注意事项:
中断发生后必须清除中断位,不然会出现死循环不断发生这个中断。而后须要对中断类型进行判断再执行代码。
使用EXTI的I/O中断,在完成RCC与GPIO硬件设置以后须要作三件事:初始化EXTI、NVIC开中断、编写中断执行代码。 

STM32的USART

STM32资料   2009-07-14 13:33   阅读489   评论4   
字号: 大大  中中  小小 
b)        初始化函数定义:
void USART_Configuration(void);        //定义串口初始化函数
c)        初始化函数调用:
void UART_Configuration(void);        //串口初始化函数调用
初始化代码:
void USART_Configuration(void)                       //串口初始化函数
{
//串口参数初始化  
  USART_InitTypeDef USART_InitStructure;             //串口设置恢复默认参数

//初始化参数设置
  USART_InitStructure.USART_BaudRate = 9600;                                   //波特率9600
   USART_InitStructure.USART_WordLength = USART_WordLength_8b;  //字长8位
  USART_InitStructure.USART_StopBits = USART_StopBits_1;                //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;//打开Rx接收和Tx发送功能

    USART_Init(USART1, &USART_InitStructure);                                      //初始化
  USART_Cmd(USART1, ENABLE);                                                       //启动串口
}

RCC中打开相应串口
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE);

GPIO里面设定相应串口管脚模式
//串口1的管脚初始化  
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;                      //管脚9
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;      //复用推挽输出
  GPIO_Init(GPIOA, &GPIO_InitStructure);                             //TX初始化

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;                   //管脚10
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
  GPIO_Init(GPIOA, &GPIO_InitStructure);                           //RX初始化

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')                                         //循环逐字输出,到结束字'\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(50Hz,AF_PP); 

f)        printf函数重定义(没必要理解,调试经过以备后用)
(1)       须要c标准函数:
#include "stdio.h"
(2)       粘贴函数定义代码
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)  //定义为putchar应用
(3)       RCC中打开相应串口
(4)       GPIO里面设定相应串口管脚模式
(6)       增长为putchar函数。
int putchar(int c)                                            //putchar函数
{
  if (c == '\n'){putchar('\r');}                              //将printf的\n变成\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中。

三、        NVIC串口中断的应用
a)        目的:利用前面调通的硬件基础,和几个函数的代码,进行串口的中断输入练习。由于在实际应用中,不使用中断进行的输入是效率很是低的,这种用法不多见,大部分串口的输入都离不开中断。
b)        初始化函数定义及函数调用:不用添加和调用初始化函数,在指定调试地址的时候已经调用过,在那个NVIC_Configuration里面添加相应开中断代码就好了。
c)        过程:
i.        在串口初始化中USART_Cmd以前加入中断设置:
USART_ITConfig(USART1, USART_IT_TXE, ENABLE);//TXE发送中断,TC传输完成中断,RXNE接收中断,PE奇偶错误中断,能够是多个。
ii.        RCC、GPIO里面打开串口相应的基本时钟、管脚设置
iii.        NVIC里面加入串口中断打开代码:
NVIC_InitTypeDef NVIC_InitStructure;//中断默认参数

NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;//通道设置为串口1中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;   //中断占先等级0
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;             //中断响应优先级0
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;            //打开中断
NVIC_Init(&NVIC_InitStructure);                                             //初始化

iv.        在stm32f10x_it.c文件中找到void USART1_IRQHandler函数,在其中添入执行代码。通常最少三个步骤:先使用if语句判断是发生那个中断,而后清除中断标志位,最后给字符串赋值,或作其余事情。
void USART1_IRQHandler(void)                            //串口1中断
{
char RX_dat;                                                       //定义字符变量
   
  if (USART_GetITStatus(USART1, USART_IT_RXNE) !=RESET)  //判断发生接收中断
  {USART_ClearITPendingBit(USART1,  USART_IT_RXNE);         //清除中断标志
  
   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+定义代码再次定义就能够直接调用了。 

STM32运行的必要硬件库

STM32资料   2009-07-14 13:31   阅读163   评论0   
字号: 大大  中中  小小 
0、        实验以前的准备
a)        接通串口转接器
b)        下载IO与串口的原厂程序,编译经过保证调试所需硬件正常。

一、        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);//写入1
GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x00);//写入0
GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_6) ;//读入IO
k)        简单Delay函数
void Delay(vu32 nCount)//简单延时函数
{for(; nCount != 0; nCount--);} 

基于STM32的PWM输出

STM32资料   2009-07-14 13:30   阅读449   评论2   
字号: 大大  中中  小小 
c)        初始化函数定义:
void TIM_Configuration(void);  //定义TIM初始化函数
d)        初始化函数调用:
TIM_Configuration();  //TIM初始化函数调用
e)        初始化函数,不一样于前面模块,TIM的初始化分为两部分——基本初始化和通道初始化:
void TIM_Configuration(void)//TIM初始化函数

  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;//定时器初始化结构
  TIM_OCInitTypeDef  TIM_OCInitStructure;//通道输出初始化结构

//TIM3初始化
  TIM_TimeBaseStructure.TIM_Period = 0xFFFF;       //周期0~FFFF
  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);//打开中断,中断须要这行代码
  
//TIM3通道初始化
  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);                                                                    //启动TIM3
}

f)        RCC初始化函数中加入TIM时钟开启:
RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM3, ENABLE);
g)        GPIO里面将输入和输出管脚模式进行设置。信号:AF_PP,50MHz。
h)        使用中断的话在NVIC里添加以下代码:

//打开TIM2中断
  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)));//变换LED色彩
    IC4value = TIM_GetCapture4(TIM2);                  //获取捕捉数值
  }  
}

i)        简单应用:
//改变占空比
TIM_SetCompare4(TIM3, 变量);

j)        注意事项:
管脚的IO输出模式是根据应用来定,好比若是用PWM输出驱动LED则应该将相应管脚设为AF_PP,不然单片机没有输出。 

STM32资料一(转载)

STM32资料   2009-06-14 20:15   阅读766   评论1   
字号: 大大  中中  小小 


注:下面是一些经常使用的代码,网上不少可是大多注释不全。高手看没问题,对于咱们这些新手就费劲了……因此我把这些代码集中,进行了逐句注释,但愿对新手们有价值。

阅读flash: 芯片内部存储器flash操做函数

个人理解——对芯片内部flash进行操做的函数,包括读取,状态,擦除,写入等等,能够容许程序去操做flash上的数据。

基础应用1,FLASH时序延迟几个周期,等待总线同步操做。推荐按照单片机系统运行频率,0—24MHz时,取Latency=0;24—48MHz时,取Latency=1;48~72MHz时,取Latency=2。全部程序中必须的

用法:FLASH_SetLatency(FLASH_Latency_2);

位置:RCC初始化子函数里面,时钟起振以后。

基础应用2,开启FLASH预读缓冲功能,加速FLASH的读取。全部程序中必须的

用法:FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

位置:RCC初始化子函数里面,时钟起振以后。

三、        阅读lib:调试全部外设初始化的函数。

个人理解——不理解,也不须要理解。只要知道全部外设在调试的时候,EWRAM须要从这个函数里面得到调试所需信息的地址或者指针之类的信息。

基础应用1,只有一个函数debug。全部程序中必须的。

用法:        #ifdef DEBUG

                 debug();

#endif

        位置:main函数开头,声明变量以后。

四、        阅读nvic:系统中断管理。

个人理解——管理系统内部的中断,负责打开和关闭中断。

基础应用1,中断的初始化函数,包括设置中断向量表位置,和开启所需的中断两部分。全部程序中必须的。

用法:        voidNVIC_Configuration(void)

{

NVIC_InitTypeDef NVIC_InitStructure;                    //中断管理恢复默认参数

#ifdef  VECT_TAB_RAM   //若是C/C++Compiler\Preprocessor\Defined symbols中的定义了VECT_TAB_RAM(见程序库更改内容的表格)

NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); //则在RAM调试

#else                                                                   //若是没有定义VECT_TAB_RAM

NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);//则在Flash里调试

#endif                                                                   //结束判断语句

//如下为中断的开启过程,不是全部程序必须的。

//NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 

//设置NVIC优先级分组,方式。

//注:一共16个优先级,分为抢占式和响应式。两种优先级所占的数量由此代码肯定,NVIC_PriorityGroup_x能够是0、一、二、三、4,分别表明抢占优先级有一、二、四、八、16个和响应优先级有1六、八、四、二、1个。规定两种优先级的数量后,全部的中断级别必须在其中选择,抢占级别高的会打断其余中断优先执行,而响应级别高的会在其余中断执行完优先执行。

//NVIC_InitStructure.NVIC_IRQChannel = 中断通道名;  //开中断,中断名称见函数库

//NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级

//NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;        //响应优先级

//NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;    //启动此通道的中断

//NVIC_Init(&NVIC_InitStructure);                                      //中断初始化

}

五、        阅读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读取缓冲,加速

    FLASH_SetLatency(FLASH_Latency_2);                      //flash操做的延时

RCC_HCLKConfig(RCC_SYSCLK_Div1);                      //AHB使用系统时钟

    RCC_PCLK2Config(RCC_HCLK_Div2);                        //APB2(高速)为HCLK的一半

    RCC_PCLK1Config(RCC_HCLK_Div2);                        //APB1(低速)为HCLK的一半

//注:AHB主要负责外部存储器时钟。PB2负责AD,I/O,高级TIM,串口1。APB1负责DA,USB,SPI,I2C,CAN,串口2345,普通TIM。

    RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9);  //PLLCLK = 8MHz * 9 = 72 MH

    RCC_PLLCmd(ENABLE);                                             //启动PLL

while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){}    //等待PLL启动

RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);  //将PLL设置为系统时钟源

while (RCC_GetSYSCLKSource() != 0x08){}    //等待系统时钟源的启动

  }

//RCC_AHBPeriphClockCmd(ABP2设备1 | ABP2设备2 |, ENABLE); //启动AHP设备

//RCC_APB2PeriphClockCmd(ABP2设备1 | ABP2设备2 |, ENABLE);//启动ABP2设备

  //RCC_APB1PeriphClockCmd(ABP2设备1 | ABP2设备2 |, ENABLE); //启动ABP1设备

}

六、        阅读exti:外部设备中断函数

个人理解——外部设备经过引脚给出的硬件中断,也能够产生软件中断,19个上升、降低或都触发。EXTI0~EXTI15链接到管脚,EXTI线16链接到PVD(VDD监视),EXTI线17链接到RTC(闹钟),EXTI线18链接到USB(唤醒)。

基础应用1,设定外部中断初始化函数。按需求,不是必须代码。

        用法: voidEXTI_Configuration(void)

{

EXTI_InitTypeDef EXTI_InitStructure;                                //外部设备中断恢复默认参数

EXTI_InitStructure.EXTI_Line = 通道1|通道2;  //设定所需产生外部中断的通道,一共19个。

EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;     //产生中断

EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //上升降低沿都触发 

EXTI_InitStructure.EXTI_LineCmd = ENABLE;                  //启动中断的接收 

EXTI_Init(&EXTI_InitStructure);                         //外部设备中断启动

}

        七、       阅读dma:经过总线而越过CPU读取外设数据

个人理解——经过DMA应用能够加速单片机外设、存储器之间的数据传输,并在传输期间不影响CPU进行其余事情。这对于入门开发基本功能来讲没有太大必要,这个内容先行跳过。

八、        阅读systic:系统定时器

个人理解——能够输出和利用系统时钟的计数、状态。

基础应用1,精确计时的延时子函数。推荐使用的代码。

        用法:

static vu32 TimingDelay;                                                                             //全局变量声明

void SysTick_Config(void)                                                                           //systick初始化函数

{

    SysTick_CounterCmd(SysTick_Counter_Disable);                                   //中止系统定时器

    SysTick_ITConfig(DISABLE);                              //中止systick中断

         SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8); //systick使用HCLK做为时钟源,频率值除以8。

    SysTick_SetReload(9000);                                                                   //重置时间1毫秒(以72MHz为基础计算)

    SysTick_ITConfig(ENABLE);                                                                 //开启systic中断

}

void Delay (u32 nTime)                                                                              //延迟一毫秒的函数

{

    SysTick_CounterCmd(SysTick_Counter_Enable);                          //systic开始计时

   TimingDelay = nTime;                                          //计时长度赋值给递减变量

    while(TimingDelay != 0);                              //检测是否计时完成

    SysTick_CounterCmd(SysTick_Counter_Disable);                                          //关闭计数器

    SysTick_CounterCmd(SysTick_Counter_Clear);                          //清除计数值

}

void TimingDelay_Decrement(void)  //递减变量函数,函数名由“stm32f10x_it.c”中的中断响应函数定义好了。

{

    if (TimingDelay != 0x00)                              //检测计数变量是否达到0

            { TimingDelay--;                              //计数变量递减

            }

}

注:建议熟练后使用,所涉及知识和设备太多,新手出错的可能性比较大。新手可用简化的延时函数代替:

void Delay(vu32 nCount)                                                                             //简单延时函数

{

  for(; nCount != 0; nCount--);                                                                      //循环变量递减计数

}

当延时较长,又不须要精确计时的时候可使用嵌套循环:

void Delay(vu32 nCount)                                           //简单的长时间延时函数

{int i;                                              //声明内部递减变量

  for(; nCount != 0; nCount--)                                                                //递减变量计数

{for (i=0; i<0xffff; i++)}                                                                                   //内部循环递减变量计数

}

九、        阅读gpio:I/O设置函数

个人理解——全部输入输出管脚模式设置,能够是上下拉、浮空、开漏、模拟、推挽模式,频率特性为2M,10M,50M。也能够向该管脚直接写入数据和读取数据。

        基础应用1,gpio初始化函数。全部程序必须。

        用法:voidGPIO_Configuration(void)

{

GPIO_InitTypeDef GPIO_InitStructure;                          //GPIO状态恢复默认参数

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_标号 | GPIO_Pin_标号 ;  //管脚位置定义,标号能够是NONE、ALL、0至15。

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;//输出速度2MHz

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;       //模拟输入模式

GPIO_Init(GPIOC, &GPIO_InitStructure);               //C组GPIO初始化

//注:以上四行代码为一组,每组GPIO属性必须相同,默认的GPIO参数为:ALL,2MHz,FLATING。若是其中任意一行与前一组相应设置相同,那么那一行能够省略,由此推论若是前面已经将此行参数设定为默认参数(包括使用GPIO_InitTypeDef GPIO_InitStructure代码),本组应用也是默认参数的话,那么也能够省略。如下重复这个过程直到全部应用的管脚所有被定义完毕。

……

}

基础应用2,向管脚写入0或1

        用法:GPIO_WriteBit(GPIOB,GPIO_Pin_2, (BitAction)0x01);      //写入1

基础应用3,从管脚读入0或1

        用法:GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_6)

STM32笔记之七:让它跑起来,基本硬件功能的创建

0、        实验以前的准备

a)        接通串口转接器

b)        下载IO与串口的原厂程序,编译经过保证调试所需硬件正常。

一、        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);//写入1

GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x00);//写入0

GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_6) ;//读入IO

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初始化代码:

//IO输入,GPIOB的二、十、11脚输出

  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);//写入1

Delay(0xffff);

GPIO_WriteBit(GPIOB, GPIO_Pin_2, (BitAction)0x00);//写入0

Delay(0xffff);

就能够看到链接在PB2脚上的LED闪烁了,单片机就跑起来了。

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;                                   //波特率9600

   USART_InitStructure.USART_WordLength = USART_WordLength_8b;  //字长8位

  USART_InitStructure.USART_StopBits = USART_StopBits_1;                //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;//打开Rx接收和Tx发送功能

USART_Init(USART1, &USART_InitStructure);                                       //初始化

  USART_Cmd(USART1, ENABLE);                                                       //启动串口

}

RCC中打开相应串口

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE);

GPIO里面设定相应串口管脚模式

//串口1的管脚初始化  

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;                      //管脚9

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;      //复用推挽输出

  GPIO_Init(GPIOA, &GPIO_InitStructure);                             //TX初始化

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;                   //管脚10

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入

  GPIO_Init(GPIOA, &GPIO_InitStructure);                           //RX初始化

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')                                         //循环逐字输出,到结束字'\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(50Hz,AF_PP); 

f)        printf函数重定义(没必要理解,调试经过以备后用)

(1)       须要c标准函数:

#include "stdio.h"

(2)       粘贴函数定义代码

#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)  //定义为putchar应用

(3)       RCC中打开相应串口

(4)       GPIO里面设定相应串口管脚模式

(6)       增长为putchar函数。

int putchar(int c)                                            //putchar函数

{

  if (c == '\n'){putchar('\r');}                              //将printf的\n变成\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中。

三、        NVIC串口中断的应用

a)        目的:利用前面调通的硬件基础,和几个函数的代码,进行串口的中断输入练习。由于在实际应用中,不使用中断进行的输入是效率很是低的,这种用法不多见,大部分串口的输入都离不开中断。

b)        初始化函数定义及函数调用:不用添加和调用初始化函数,在指定调试地址的时候已经调用过,在那个NVIC_Configuration里面添加相应开中断代码就好了。

c)        过程:

i.        在串口初始化中USART_Cmd以前加入中断设置:

USART_ITConfig(USART1, USART_IT_TXE, ENABLE);//TXE发送中断,TC传输完成中断,RXNE接收中断,PE奇偶错误中断,能够是多个。

ii.        RCC、GPIO里面打开串口相应的基本时钟、管脚设置

iii.        NVIC里面加入串口中断打开代码:

NVIC_InitTypeDef NVIC_InitStructure;//中断默认参数

NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;//通道设置为串口1中断

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;   //中断占先等级0

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;             //中断响应优先级0

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;            //打开中断

NVIC_Init(&NVIC_InitStructure);                                             //初始化

iv.        在stm32f10x_it.c文件中找到void USART1_IRQHandler函数,在其中添入执行代码。通常最少三个步骤:先使用if语句判断是发生那个中断,而后清除中断标志位,最后给字符串赋值,或作其余事情。

void USART1_IRQHandler(void)                            //串口1中断

{

char RX_dat;                                                       //定义字符变量

    if (USART_GetITStatus(USART1, USART_IT_RXNE) !=RESET)  //判断发生接收中断

  {USART_ClearITPendingBit(USART1,  USART_IT_RXNE);         //清除中断标志

   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+定义代码再次定义就能够直接调用了。

STM32笔记之九:打断它来为我办事,EXIT (外部I/O中断)应用

a)        目的:跟串口输入相似,不使用中断进行的IO输入效率也很低,并且能够经过EXTI插入按钮事件,本节联系EXTI中断。

b)        初始化函数定义:

void EXTI_Configuration(void); //定义IO中断初始化函数

c)        初始化函数调用:

EXTI_Configuration();//IO中断初始化函数调用简单应用:

d)        初始化函数:

void EXTI_Configuration(void)

{ EXTI_InitTypeDef EXTI_InitStructure;       //EXTI初始化结构定义

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_Trigger = EXTI_Trigger_Falling;//触发模式

  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管脚。

//IO输入,GPIOA的4脚输入

  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)));//LED发生明暗交替

}

h)        中断注意事项:

中断发生后必须清除中断位,不然会出现死循环不断发生这个中断。而后须要对中断类型进行判断再执行代码。

使用EXTI的I/O中断,在完成RCC与GPIO硬件设置以后须要作三件事:初始化EXTI、NVIC开中断、编写中断执行代码。

STM32笔记之十:工做工做,PWM输出

a)        目的:基础PWM输出,以及中断配合应用。输出选用PB1,配置为TIM3_CH4,是目标板的LED6控制脚。

b)        对于简单的PWM输出应用,暂时无需考虑TIM1的高级功能之区别。

c)        初始化函数定义:

void TIM_Configuration(void);  //定义TIM初始化函数

d)        初始化函数调用:

TIM_Configuration();  //TIM初始化函数调用

e)        初始化函数,不一样于前面模块,TIM的初始化分为两部分——基本初始化和通道初始化:

void TIM_Configuration(void)//TIM初始化函数



  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;//定时器初始化结构

  TIM_OCInitTypeDef  TIM_OCInitStructure;//通道输出初始化结构

//TIM3初始化

  TIM_TimeBaseStructure.TIM_Period = 0xFFFF;       //周期0~FFFF

  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);//打开中断,中断须要这行代码

  //TIM3通道初始化

  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);                                                                    //启动TIM3

}

f)        RCC初始化函数中加入TIM时钟开启:

RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM3, ENABLE);

g)        GPIO里面将输入和输出管脚模式进行设置。信号:AF_PP,50MHz。

h)        使用中断的话在NVIC里添加以下代码:

//打开TIM2中断

  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)));//变换LED色彩

    IC4value = TIM_GetCapture4(TIM2);                  //获取捕捉数值

  }  

}

i)        简单应用:

//改变占空比

TIM_SetCompare4(TIM3, 变量);

j)        注意事项:

管脚的IO输出模式是根据应用来定,好比若是用PWM输出驱动LED则应该将相应管脚设为AF_PP,不然单片机没有输出

个人测试程序能够发出不断循环三种波长并捕获,对比结果以下:

捕捉的稳定性很好,也就是说,一样的方波捕捉到数值相差在一两个数值。

捕捉的精度跟你设置的滤波器长度有关,在这里

TIM_ICInitStructure.TIM_ICFilter = 0x4;        //滤波设置,经历几个周期跳变认定波形稳定0x0~0xF

这个越长就会捕捉数值越小,可是误差几十个数值,下面是0、四、16个周期滤波的比较,out是输出的数值,in是捕捉到的。

如今有两个疑问:

一、在TIM2的捕捉输入通道初始化里面这句

TIM_SelectInputTrigger(TIM2, TIM_TS_TI2FP2);   //选择时钟触发源

按照硬件框图,4通道应该对应TI4FP4。但是实际使用TI1FP1,TI2FP2都行,其余均编译错误未注册。这是为何?

二、关闭调试器和IAR程序,直接供电跑出来的结果第一个周期很正常,当输出脉宽第二次循环变小后捕捉的数值就差的远了。不知道是为何

STM32笔记之十二:时钟不息工做不止,systic时钟应用

a)        目的:使用系统时钟来进行两项实验——周期执行代码与精肯定时延迟。

b)        初始化函数定义:

void SysTick_Configuration(void);

c)        初始化函数调用:

SysTick_Configuration();

d)        初始化函数:

void SysTick_Configuration(void)

{

  SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);//时钟除8

  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.        定义通用变量:u16Tic_Val=0; //变量用于精确计时

ii.        在stm32f10x_it.c文件中相应定义:

extern u16 Tic_Val;//在本文件引用MAIN.c定义的精确计时变量

iii.        定义函数名称:voidTic_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++;//变量递增

vi.        调用代码:Tic_Delay(10);  //精确延时

vii.        疑问:若是去掉计时行那个没用的printf("");函数将中止工做,这个现象很奇怪

C语言功底问题。是的,那个“注意事项”最后的疑问的缘由就是这个

Tic_Val应该改成vu16

while(Tic_Val != Tic_Count){printf("");}//计时

就能够改成:

while(Tic_Val != Tic_Count);             //检查变量是否计数到位

STM32笔记之十三:恶搞,两只看门狗

a)        目的:

了解两种看门狗(我叫它:系统运行故障探测器和独立系统故障探测器,新手每每被这个并不形象的象形名称搞糊涂)之间的区别和基本用法。

b)        相同:

都是用来探测系统故障,经过编写代码定时发送故障清零信号(高手们都管这个代码叫作“喂狗”),告诉它系统运行正常。一旦系统故障,程序清零代码(“喂狗”)没法执行,其计数器就会计数不止,直到记到零并发生故障中断(狗饿了开始叫唤),控制CPU重启整个系统(不行啦,开始咬人了,快跑……)。

c)        区别:

独立看门狗Iwdg——个人理解是独立于系统以外,由于有独立时钟,因此不受系统影响的系统故障探测器。(这条狗是借来的,见谁偷懒它都咬!)主要用于监视硬件错误。

窗口看门狗wwdg——个人理解是系统内部的故障探测器,时钟与系统相同。若是系统时钟不走了,这个狗也就失去做用了。(这条狗是老板娘养的,老板不干活儿他无论!)主要用于监视软件错误。

d)        初始化函数定义:鉴于两只狗做用差很少,使用过程也差很少初始化函数栓一块儿了,用的时候根据状况删减。

void WDG_Configuration(void);

e)        初始化函数调用:

WDG_Configuration();

f)        初始化函数

void WDG_Configuration()               //看门狗初始化

{

//软件看门狗初始化

  WWDG_SetPrescaler(WWDG_Prescaler_8); //时钟8分频4ms

// (PCLK1/4096)/8= 244 Hz (~4 ms)

   WWDG_SetWindowValue(65);                   //计数器数值

  WWDG_Enable(127);                  //启动计数器,设置喂狗时间

// WWDG timeout = ~4 ms * 64 = 262 ms

   WWDG_ClearFlag();                  //清除标志位

  WWDG_EnableIT();                   //启动中断

//独立看门狗初始化

  IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);//启动寄存器读写

  IWDG_SetPrescaler(IWDG_Prescaler_32);//40K时钟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;           //喂狗循环程序的频率判断变量

第二步:将SysTickHandler中喂狗代码改成下面:

Tic_IWDG++;             //变量递增

if(Tic_IWDG>=100)       //每100个systic周期喂狗

{  IWDG_ReloadCounter();//重启计数器(喂狗)

  Tic_IWDG=0;          //变量清零

}php