本例子代码参考了STM32库开发实战指南中的代码,由于使用的板子是尚学STM32F103ZET6,为了配合板上已有资源,也参考了其配套代码。为了便于书写文本,我尽量将代码都写到了一个文件中,这种方式是不推荐的,在做具体工程时最好代码分类管理,使工程逻辑清晰。

现在对板上一些资源说明:板上有两个LED灯,引脚为PE5、PE6,均为ResetBits时点亮。有三个按钮,依次为黄色复位,红色PE4(按下接GND)、红色PA0(按下接3.3V,WAKE UP按钮)。ISP口为靠近电源开关的USB,也是USART1口。USART2口为PA3(Rx)、PA2(Tx)。IPD为高电平中断(按键一边接高电平),IPU为低电平中断。

STM32学习笔记:基础例子

接下来例举基本操作:

用GPIO点亮灯(GPIO输出)

#include "stm32f10x.h"
void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

void GPIO_Config(void) //配置LED用到的I/O口
{       
    GPIO_InitTypeDef GPIO_InitStructure; //定义一个GPIO_InitTypeDef类型的结构体
    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOE, ENABLE); //开启GPIOE的外设时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6; //选择要控制的GPIOE引脚,这里选了PE5、PE6
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置引脚模式为:通用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置引脚速率为:50MHz
    GPIO_Init(GPIOE, &GPIO_InitStructure);  //调用库函数,初始化GPIOE
    GPIO_SetBits(GPIOE, GPIO_Pin_5 | GPIO_Pin_6); //关闭所有LED灯
}

int main(void)
{
    GPIO_Config();
    while(1)
    {
    GPIO_SetBits(GPIOE , GPIO_Pin_5); //PE5输出高电平
    GPIO_ResetBits(GPIOE,GPIO_Pin_6); //PE6输出低电平
    Delay(1000000);//1,000,000 六个零以上才有明显闪烁
    GPIO_SetBits(GPIOE , GPIO_Pin_6); //PE6输出高电平
    GPIO_ResetBits(GPIOE,GPIO_Pin_5); //PE5输出低电平
    Delay(1000000);
    }
}

实现效果:PE6、PE5两盏灯闪烁。

按键输入(GPIO输入)

#include "stm32f10x.h"
#define KEY1 GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_4)   //读PE4(GND)
#define KEY2 GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_0)   //读PA0(VCC)

void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

void GPIO_Config(void) //配置LED用到的PE5、PE6
{       
    GPIO_InitTypeDef GPIO_InitStructure; //定义一个GPIO_InitTypeDef类型的结构体
    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOE, ENABLE); //开启GPIOE的外设时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6; //选择要控制的GPIOE引脚,这里选了PE5、PE6
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置引脚模式为:通用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置引脚速率为:50MHz
    GPIO_Init(GPIOE, &GPIO_InitStructure);  //调用库函数,初始化GPIOE
    GPIO_SetBits(GPIOE, GPIO_Pin_5 | GPIO_Pin_6); //关闭所有LED灯
}

void Key_GPIO_Config(void)//按键配置,这里用PE4、PE5输入
{
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE,ENABLE);//开启GPIOE的时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; //设置引脚PE4
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉输入模式(按键按下接GND用这个)
    GPIO_Init(GPIOE, &GPIO_InitStructure);

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
    GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //下拉输入模式(按键按下接VCC用这个)
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}
 
unsigned char KEY1_Scan(void)
{
    static char key_up0=0;//按键按松开标志
    if(KEY1==0)
    {
    Delay(10000);//延时去抖动
        if(KEY1==0)
        {
            key_up0=1;
        }
        
    }
    if(KEY1==1&&key_up0==1)
    {
    key_up0=0;
    return 1;
    }
    return 0;
}

unsigned char KEY2_Scan(void)
{
    static char key_up2=0;//按键按松开标志
    if(KEY2==1)
    {
    Delay(10000);//延时去抖动
        if(KEY2==1)
        {
            key_up2=1;
        }
    }
    if(KEY2==0&&key_up2==1)
    {
        key_up2=0;
        return 1;
    }
    return 0;
}

unsigned char KEY_Scan(void)
{
    unsigned char key_code;
    if(KEY1_Scan()==1) key_code=1;
    else if(KEY2_Scan()==1) key_code=2;
    else key_code=0;
    return key_code;
}

int main(void)
{
    int value=0;
    GPIO_Config();
    Key_GPIO_Config();//配置按键
    while(1)
    {
        value=KEY_Scan();//获取按键值
        if(value==1)
        GPIO_WriteBit(GPIOE, GPIO_Pin_5, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_5))));//LED灯PE5反转
        else if(value==2)
        GPIO_WriteBit(GPIOE, GPIO_Pin_6, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_6))));//LED灯PE6反转
    }
}

实现效果:PE4按下控制PE5灯反转,PA0按下控制PE6灯反转(亮、灭)。

按键中断(EXTI外部中断操作)

#include "stm32f10x.h" 
void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

void GPIO_Config(void) //配置LED用到的PE5、PE6
{       
    GPIO_InitTypeDef GPIO_InitStructure; //定义一个GPIO_InitTypeDef类型的结构体
    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOE, ENABLE); //开启GPIOE的外设时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6; //选择要控制的GPIOE引脚,这里选了PE5、PE6
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置引脚模式为:通用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置引脚速率为:50MHz
    GPIO_Init(GPIOE, &GPIO_InitStructure);  //调用库函数,初始化GPIOE
    GPIO_SetBits(GPIOE, GPIO_Pin_5 | GPIO_Pin_6); //关闭所有LED灯
}

static void NVIC_Configuration(void)//NVIC(中断控制器)初始化配置,这里配PE4、PA0
{
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);//把NVIC中断优先级分组设为第1组
    NVIC_InitStructure.NVIC_IRQChannel = EXTI4_IRQn;//PE4对应EXTI线为EXTI4,填EXTI4_IRQn。(EXTI5~EXTI9使用同一中断向量,则填EXTI9_5_IRQn)
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//抢占优先级2
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;//响应优先级1
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);//向寄存器写入参数

    NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;//PA0对应EXTI线为EXTI0,填EXTI0_IRQn
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//抢占优先级2
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;//响应优先级2
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//IRQ(中断请求)通道使能
    NVIC_Init(&NVIC_InitStructure);//向寄存器写入参数
}

void EXTI_Config(void)//配置PE4、PA0为线中断口,并设置中断优先级
{
    GPIO_InitTypeDef GPIO_InitStructure;
    EXTI_InitTypeDef EXTI_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO,ENABLE);//配置中断线(PA0)时钟和第二功能AFIO时钟,AFIO指GPIO口的复用功能
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO,ENABLE);//配置中断线(PE4)时钟和第二功能AFIO时钟
    
    NVIC_Configuration();//配置NVIC中断控制器
    
    //以下配PE4
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; //选定要配置为EXTI线的gpio口和设置其工作模式
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;//上拉输入
    GPIO_Init(GPIOE, &GPIO_InitStructure);
    //以下配PA0
    GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0;//选PA0
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;//下拉输入
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //以下配PE4中断线、初始化配置
    GPIO_EXTILineConfig(GPIO_PortSourceGPIOE, GPIO_PinSource4); //EXTI中断线(PE5)工作模式配置
    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);
    //以下配PA0中断线、初始化配置
    GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0); //EXTI中断线(PA0)工作模式配置
    EXTI_InitStructure.EXTI_Line = EXTI_Line0;
    EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
    EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //上升沿中断
    EXTI_InitStructure.EXTI_LineCmd = ENABLE;
    EXTI_Init(&EXTI_InitStructure);
}

int main(void)
{
    GPIO_Config();//LED(PE5、PE6)配置
    EXTI_Config(); //外部中断EXTI配置,这里是选PE4、PA0
    while(1)//等待中断
    {
    }
}

在stm32f10x_it.c中加入名为EXTI0_IRQHandler(void)和EXTI4_IRQHandler(void)函数:

void EXTI0_IRQHandler(void)
{
    Delay(10000);//延时消抖
    if(EXTI_GetITStatus(EXTI_Line0) != RESET)//检查指定的EXTI0线路触发请求发生与否
    {
        GPIO_WriteBit(GPIOE, GPIO_Pin_6, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_6))));//控制LED的PE6翻转
    }
    EXTI_ClearITPendingBit(EXTI_Line0);//清除EXTI0线路挂起位
}
 
void EXTI4_IRQHandler(void)
{
    Delay(10000);//延时消抖
    if(EXTI_GetITStatus(EXTI_Line4) != RESET)//确保是否产生了EXTI4 Line中断
    {
        GPIO_WriteBit(GPIOE, GPIO_Pin_5, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_5))));//控制LED的PE5翻转
    }
    EXTI_ClearITPendingBit(EXTI_Line4);//清除中断标志位
}

实现效果:PE4按下触发中断,控制PE5灯反转;PA0按下触发中断,控制PE6灯反转(亮、灭)。

串口打印(用USART1)

#include "stm32f10x.h"
#include "stdio.h"
void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

int fputc(int ch, FILE *f)//重定向c库函数printf到USART1
{
    USART_SendData(USART1, (unsigned char) ch);//将Printf内容发往串口
    while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);
    return (ch);
}

void USART1_Config(unsigned int bound)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);//配置串口1时钟

    //以下串口GPIO端口配置
    //以下配置串口1的Tx(PA9)参数
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;//PA9为Tx
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //以下配置串口1的Rx(PA10)参数
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入模式,Rx不需配Speed
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //以下配置串口1的模式mode
    USART_InitStructure.USART_BaudRate = bound;//USART1的波特率
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;//串口传输的字长
    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); //使能串口1
}


int main(void)
{
    int a=0;
    USART1_Config(115200);
    while(1)
    {
    printf("\r\n Sandeepin poi %d \r\n",a);
    a++;
    Delay(2000000);
    Delay(2000000);
    }
}

实现效果:可在串口调试助手中看到Sandeepin poi 0、Sandeepin poi 1……等信息。

串口中断(用USART1)

#include "stm32f10x.h"
#include "stdio.h"
void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

int fputc(int ch, FILE *f)//重定向c库函数printf到USART1
{
    USART_SendData(USART1, (unsigned char) ch);//将Printf内容发往串口
    while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);
    return (ch);
}

void USART1_Config(unsigned int bound)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);//配置串口1时钟

    //以下串口GPIO端口配置
    //以下配置串口1的Tx(PA9)参数
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;//PA9为Tx
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;//复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //以下配置串口1的Rx(PA10)参数
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入模式,Rx不需配Speed
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    //以下配置串口1的模式mode
    USART_InitStructure.USART_BaudRate = bound;//USART1的波特率
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;//串口传输的字长
    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); //使能串口1
    
    USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启中断
}

void NVIC_Configuration(void)//NVIC(中断控制器)初始化配置,这里配USART1
{
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);//把NVIC中断优先级分组设为第0组
    //以下使能串口1中断
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;//抢占优先级1
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//响应优先级0
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//IRQ(中断请求)通道使能
    NVIC_Init(&NVIC_InitStructure);//根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器USART1
}

int main(void)
{
    USART1_Config(115200);
    NVIC_Configuration();
    while(1)
    {
        Delay(2000000);
    }
}

在stm32f10x_it.c中加入名为USART1_IRQHandler(void)函数:

USART1_IRQHandler(void)
{
    unsigned char code;
    if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
    {
        code=USART_ReceiveData(USART1);
        printf("%c",code);//将接受到的数据直接返回打印
    }
}

实现效果:在串口调试助手中输入一系列字符,STM32接收到后直接将收到字符原样打印出来。

定时器(用TIM3)

#include "stm32f10x.h"
void Delay(__IO u32 nCount)  //简单的延时函数
{
    for(; nCount != 0; nCount--);
}

void LED_GPIO_Config(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE, ENABLE);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_5;//PE5、PE6
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//通用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOE, &GPIO_InitStructure);
    GPIO_SetBits(GPIOE,GPIO_Pin_5);//PE5初始输出高
    GPIO_ResetBits(GPIOE,GPIO_Pin_6);//PE6初始输出低
}

void TIME_NVIC_Configuration(void)//TIM3中断优先级配置
{
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置NVIC中断分组2
    NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;//抢占优先级0
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;//响应优先级3
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
    NVIC_Init(&NVIC_InitStructure);//初始化外设
}

void TIME_Configuration(void)
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //定时器TIM3时钟使能
    TIM_TimeBaseStructure.TIM_Period = 5000; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值,计数到5000为500ms
    TIM_TimeBaseStructure.TIM_Prescaler =(7200-1);//设置用来作为TIMx时钟频率除数的预分频值
    TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
    TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
    TIM_ITConfig(TIM3,TIM_IT_Update|TIM_IT_Trigger,ENABLE);//使能、失能指定的TIM中断
    TIM_Cmd(TIM3, ENABLE);//使能TIMx外设
}

int main(void)
{
    LED_GPIO_Config();//开LED的GPIO
    TIME_NVIC_Configuration();//TIM3定时器中断配置
    TIME_Configuration();//TIM3定时器配置
    while(1)
    {
    }
}

在stm32f10x_it.c中加入名为TIM3_IRQHandler(void)函数:

void TIM3_IRQHandler(void) 
{
    if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否
    {
        TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除TIMx的中断待处理位
        GPIO_WriteBit(GPIOE, GPIO_Pin_5, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_5))));//控制LED灯PE5翻转
        GPIO_WriteBit(GPIOE, GPIO_Pin_6, (BitAction)(1-(GPIO_ReadOutputDataBit(GPIOE, GPIO_Pin_6))));//控制LED灯PE6翻转
    }
}

实现效果:每隔500ms后LED灯PE5、PE6翻转

SysTick(系统滴答定时器)

#include "stm32f10x.h"
u32 TimingDelay;

void LED_GPIO_Config(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE, ENABLE);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_5;//PE5、PE6
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//通用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOE, &GPIO_InitStructure);
    GPIO_SetBits(GPIOE,GPIO_Pin_5);//PE5初始输出高
    GPIO_ResetBits(GPIOE,GPIO_Pin_6);//PE6初始输出低
}

void NVIC_Configuration(void)
{
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断组为2
    NVIC_InitStructure.NVIC_IRQChannel = SysTick_IRQn;//中断线
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //抢占优先级为2
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//响应优先级为0
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //允许SysTick_IRQn中断
    NVIC_Init(&NVIC_InitStructure);
}

void SysTick_Init(void)
{
    /* SystemFrequency / 1000    1ms中断一次
     * SystemFrequency / 100000  10us中断一次,分析:ticks=SystemFrequency / 100000=720,T=ticks/f,f=72000000,T=720/72=10us
     * SystemFrequency / 1000000 1us中断一次*/
    while(SysTick_Config( SystemCoreClock / 1000));//Systick 配置延时n*ms。输入的参数为两个中断之间的脉冲数。
}

void Delay(u32 nTime)//用Systick延时
{ 
    TimingDelay = nTime;
    while(TimingDelay != 0);
}

int main(void)
{
    SysTick_Init();
    LED_GPIO_Config();
    NVIC_Configuration();//中断配置
    while(1)
    {
    GPIO_SetBits(GPIOE,GPIO_Pin_6);
    GPIO_ResetBits(GPIOE,GPIO_Pin_5);
    Delay(200);//Systick 配置延时200*ms
    GPIO_SetBits(GPIOE,GPIO_Pin_5);
    GPIO_ResetBits(GPIOE,GPIO_Pin_6);
    Delay(200);//Systick 配置延时200*ms
    }
}

在stm32f10x_it.c中找SysTick_Handler(void)函数,填入如下内容:

extern u32 TimingDelay;
void SysTick_Handler(void)
{
    if (TimingDelay != 0x00)
    {
    TimingDelay--;
    }
}

实现效果:每隔200ms后LED灯PE5、PE6翻转。

TFTLCD显示(2.4寸液晶ILI9325)

此部分代码复杂,参见我的另一篇文章《STM32操纵2.4寸液晶(ILI9325)》

调用DS18B20温度传感器

#include "stm32f10x.h"
#include "stdio.h"

//DS18B20-DATA引脚为PC2
#define DS18B20_DQ_OUT_Low GPIO_ResetBits(GPIOC,GPIO_Pin_2)  //数据端口PC2
#define DS18B20_DQ_OUT_High GPIO_SetBits(GPIOC,GPIO_Pin_2)  //数据端口PC2
#define DS18B20_DQ_IN GPIO_ReadInputDataBit(GPIOC,GPIO_Pin_2)   //数据端口PC2

void USART1_Config(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    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_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_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;
    USART_Init(USART1, &USART_InitStructure); 
    USART_Cmd(USART1, ENABLE);
}

int fputc(int ch, FILE *f)
{
    USART_SendData(USART1, (unsigned char) ch);
    while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);
    return (ch);
}

void Delay_us(u32 ntimes)
{
    u32 flag;
    SysTick->LOAD=9*ntimes; //时间加载
    SysTick->VAL=0;        //清空计数器
    SysTick->CTRL=0x00000001; //bit2清空,选择外部时钟HCLK/8, bit0位清空,开启倒计时
    do
    {
        flag=SysTick->CTRL;
    }
    while(flag&0x01&&!(flag&(1<<16)));//等待时间到达 
    SysTick->CTRL=0;//关闭计数器
}

void DS18B20_GPIO_Config(void)//DS18B20的GPIO配置
{
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOC, ENABLE );//使能PORTA口时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; //SPI CS
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    GPIO_SetBits(GPIOC,GPIO_Pin_2);
}

void DS18B20_Mode_IPU(void)//使DS18B20-DATA引脚变为输入模式
{
    GPIO_InitTypeDef GPIO_InitStructure;    //GPIO
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;  //SPI CS
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  //复用推挽输出
    GPIO_Init(GPIOC, &GPIO_InitStructure);

}

void DS18B20_Mode_Out(void)//使DS18B20-DATA引脚变为输出模式
{
    GPIO_InitTypeDef GPIO_InitStructure;    //GPIO
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;  //SPI CS
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOC, &GPIO_InitStructure);

}

void DS18B20_Rst(void)//主机给从机发送复位脉冲
{
    DS18B20_Mode_Out(); //IO设置为推挽输出
    DS18B20_DQ_OUT_Low; //产生至少480us的低电平复位信号 
    Delay_us(480);
    DS18B20_DQ_OUT_High; //在产生复位信号后,需将总线拉高
    Delay_us(15);    
}

u8 DS18B20_Answer_Check(void) //检测从机给主机返回的应答脉冲,从机接收到主机的复位信号后,会在15~60us后给主机发一个应答脉冲。0:成功1:失败
{
    u8 delay=0;
    DS18B20_Mode_IPU(); // 主机设置为上拉输入
    //等待应答脉冲(一个60~240us的低电平信号 )的到来
    //如果100us内,没有应答脉冲,退出函数,注意:从机接收到主机的复位信号后,会在15~60us后给主机发一个存在脉冲
    while (DS18B20_DQ_IN&&delay<100)
    {
        delay++;
        Delay_us(1);
    }
    if(delay>=100)//经过100us后,如果没有应答脉冲,退出函数
        return 1;
    else
        delay=0;
    while (!DS18B20_DQ_IN&&delay<240)//有应答脉冲,且存在时间不超过240us 
    {
        delay++;
        Delay_us(1);
    }
    if(delay>=240)
        return 1;       
    return 0;
}

u8 DS18B20_Read_Bit(void) //从DS18B20读取一个位 返回值:1/0
{
    u8 data;
    DS18B20_Mode_Out();
    DS18B20_DQ_OUT_Low; //读时间的起始:必须由主机产生 >1us <15us 的低电平信号
    Delay_us(2);
    DS18B20_DQ_OUT_High; 
    Delay_us(12);
    DS18B20_Mode_IPU();//设置成输入,释放总线,由外部上拉电阻将总线拉高
    if(DS18B20_DQ_IN)
        data=1;
    else
        data=0;
    Delay_us(50);
    return data;
}

u8 DS18B20_Read_Byte(void)//从DS18B20读取一个字节 返回值:读到的数据
{        
    u8 i,j,dat;
    dat=0;
    for(i=0; i<8; i++) 
    {
        j = DS18B20_Read_Bit();
        dat = (dat) | (j<<i);
    }
    return dat;
}

void DS18B20_Write_Byte(u8 dat)//写一个字节到DS18B20
{
    u8 j;
    u8 testb;
    DS18B20_Mode_Out();//SET PA0 OUTPUT;
    for (j=1;j<=8;j++) 
    {
        testb=dat&0x01;
        dat=dat>>1;
        if (testb) 
        {
            DS18B20_DQ_OUT_Low;// Write 1
            Delay_us(10);                            
            DS18B20_DQ_OUT_High;
            Delay_us(50);             
        }
        else 
        {
            DS18B20_DQ_OUT_Low;// Write 0
            Delay_us(60); 
            DS18B20_DQ_OUT_High;   ///释放总线
            Delay_us(2);  
        }
    }
}

u8 DS18B20_Init(void)//初始化DS18B20的IO口 DQ 同时检测DS的存在 返回1:不存在 返回0:存在
{
    DS18B20_GPIO_Config();
    DS18B20_Rst();
    return DS18B20_Answer_Check();
}

short DS18B20_Get_Temp(void)//从ds18b20得到温度值 精度:0.1C 返回值:温度值 (-550~1250)
{

    u8 flag;
    u8 TL,TH;
    short Temperature;
    float Temperature1;
    DS18B20_Rst();
    DS18B20_Answer_Check();
    DS18B20_Write_Byte(0xcc);// skip rom
    DS18B20_Write_Byte(0x44);// ds1820 start convert
    DS18B20_Rst();
    DS18B20_Answer_Check();
    DS18B20_Write_Byte(0xcc);// skip rom
    DS18B20_Write_Byte(0xbe);// convert
    TL=DS18B20_Read_Byte(); // LSB
    TH=DS18B20_Read_Byte(); // MSB
    if( TH&0xfc)
    {
        flag=1;
        Temperature=(TH<<8)|TL;
        Temperature1=(~ Temperature)+1;
        Temperature1*=0.0625;
    }
    else
    {
        flag=0;
        Temperature1=((TH<<8)|TL)*0.0625;
    }
    return Temperature1;
} 

int main(void)
{
    USART1_Config();
    while(DS18B20_Init())//初始化兼检测DS18B20
    {
        printf("DS18B20 Link ERROR \r\n!");
        Delay_us(500000);
    }
    printf("DS18B20 Link OK \r\n!");
    while(1)
    {
        printf("Temperature: %.2f \r\n",DS18B20_Get_Temp());
        Delay_us(500000);
    }
}

实现效果:在PC2接温度传感器,通过串口1打印出温度值。

文章目录