07-CubeMx+Keil+Proteus仿真STM32 - EXTI

本文例子参考《STM32单片机开发实例——基于Proteus虚拟仿真与HAL/LL库》
源代码:https://github.com/LanLinnet/STM32F103R6

项目要求

04节,电路常态为流水灯状态,当按下按钮BTN0时,8个LED灯全亮全灭闪烁3次后恢复到常态;当按下按钮BTN1时,8个LED灯间隔交替闪烁3次后恢复常态;当BTN0和BTN1同时按下时,系统优先相应BTN1。

硬件设计

  1. 第一节的基础上,在Proteus中添加电路如下图所示,其中我们添加了一个排阻RX8、一组8个LED灯、两组由按钮BUTTON构成的按键电路。

    根据电路图和芯片技术手册,我们知道PB0可用作外部中断0(EXTI0),PB1可用作外部中断1(EXTI1),当按下按键时,PB会输入低电平,所以这两个外部中断都是通过下降沿触发的。

  2. 打开CubeMX,按照建立工程,配置PC0-PC7引脚为GPIO_Output,PB0和PB1分别为GPIO_EXTI0和GPIO_EXTI1。
    随后选中“System Core”中的GPIO,展开“Configuration”列表,如图中4所示,选中PB0和PB1,将两个GPIO管脚的“GPIO mode”都选为下降沿触发External Interrupt Mode with Falling edge trigger detection

  3. 接下来进行中断优先级配置。在“System core”中选中“NVIC”(Nested Vectored Interrupt Controller, 嵌套向量中断控制器),勾选列表中“EXTI line0 interrupt”和“EXTI line1 interrupt”两项的Enable。将页面上方的优先级组“Priority Group”选为2 bits for pre-emption priority 2 bits for subpriority,即抢占优先级和响应优先级都用2bit来设定。

  4. 中断回调函数可以使用HAL库也可以使用LL库,所以我们要设置GPIO的库:点击“Project Manager”--“Advanced Settings”,可设置库为LL或HAL,这里我们先设置为HAL库。点击“Generator Code”生成Keil工程。

软件编写

(一)基于HAL库的程序

  1. 本次我们需要实现外部中断,由前面电路知道,当按下按键时,会生成下降沿,只要将相应的GPIO设置为EXTI模式,就会自动触发外部中断。进入中断后实现的功能,不是写在主函数中,而是写在外部中断的回调函数中。

  2. 点击“Open Project”在Keil中打开工程,双击“main.c”文件。

  3. 本次仿真我们用到EXTI线侦测回调函数HAL_GPIO_EXTI_Callback(),其官方文档API介绍如下图所示。

    同时这个回调函数可以在“stm32f1xx_hal_gpio.c”程序中找到,这里的回调函数前面有一个“弱函数”的关键字“_weak”,该关键字的作用是,如果工程的任何一个源文件中都没有与该“弱函数”同名的函数,则编译器会编译该“弱函数”;但是当工程中有另一个同名函数定义出现时,编译器会忽略“弱函数”而编译另一个没有标注“_weak”关键字的同名函数。

  4. 我们需要更方便地独立地控制PC0-PC7的管脚输出,所以这里我们自定义一个函数ByteOut2PC,用于将1字节数据输出到PC端口的PC0-PC7引脚。我们先在程序最开头声明它。

    /* USER CODE BEGIN PFP */
    void ByteOut2PC(uint8_t dat);  //声明函数
    /* USER CODE END PFP */
    
  5. 然后我们在/* USER CODE BEGIN 4 *//* USER CODE END 4 */间添加这个自定义函数。同时,因为回调函数不会在生成初始化代码的时候自动生成,需要手动添加到main.c中,所以我们在这里也添加一个回调函数。

    /* USER CODE BEGIN 4 */
    //自定义函数,将1字节数据输出到PC端口的PC0-PC7引脚
    void ByteOut2PC(uint8_t dat)
    {
      if(dat & 0x01)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_0, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_0, 0);
      if(dat & 0x02)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_1, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_1, 0);
      if(dat & 0x04)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_2, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_2, 0);
      if(dat & 0x08)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_3, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_3, 0);
      if(dat & 0x10)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_4, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_4, 0);
      if(dat & 0x20)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_5, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_5, 0);
      if(dat & 0x40)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_6, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_6, 0);
      if(dat & 0x80)
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_7, 1);
      else 
        HAL_GPIO_WritePin(GPIOC, GPIO_PIN_7, 0);
    }
    
    //中断回调函数
    void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
    {
      int8_t i;  //循环变量
      if(GPIO_Pin == GPIO_PIN_0)  //检测到EXTI0线产生外部中断事件
      {
        for(i=0; i<3; i++)  //全亮全灭闪烁3次
        {
          ByteOut2PC(0xff);  //全灭
          HAL_Delay(500);
          ByteOut2PC(0);  //全亮
          HAL_Delay(500);
        }
      }
      else if(GPIO_Pin == GPIO_PIN_1) //检测到EXTI1线产生外部中断事件
      {
        for(i=0; i<3; i++)  //间隔交替闪烁3次
        {
          ByteOut2PC(0x55);  
          HAL_Delay(500);
          ByteOut2PC(0xaa);  
          HAL_Delay(500);
        }
      }
    }
    /* USER CODE END 4 */
    
  6. 因为常态呈现流水灯状态,我们首先在main函数中声明一个循环变量。

    /* USER CODE BEGIN 1 */
    int8_t i;   //循环变量i
    /* USER CODE END 1 */
    
  7. 最后,我们在while循环中添加下面的代码

    /* USER CODE BEGIN WHILE */
    while (1)
    {
      for(i=0; i<8; i++)
      {
        ByteOut2PC((0xfe<<i)|(0xfe>>(8-i)));  //正常流水灯状态
        HAL_Delay(500);
      }
    /* USER CODE END WHILE */		
    /* USER CODE BEGIN 3 */
    }
    

(二)基于LL库的程序

  1. 在上面的基础上,使用CubeMX将GPIO的库改为LL库。
  2. LL库没有提供回调函数,我们要在“stm32f1xx_it.c”程序中找到相应的外部中断库函数void EXTI0_IRQHandler(void)void EXTI1_IRQHandler(void)并填写功能代码。
  3. 同理我们在“main.c”文件的main函数中添加流水灯程序,这里就不再赘述了。
  4. 在“stm32f1xx_it.c”文件中,首先在外部中断库函数void EXTI0_IRQHandler(void)中添加代码如下
    /* USER CODE BEGIN EXTI0_IRQn 0 */
    int8_t i;  //循环变量
    for(i=0;i<3;i++)
    {
      LL_GPIO_WriteOutputPort(GPIOC, 0xff);  //全灭
      HAL_Delay(500);
      LL_GPIO_WriteOutputPort(GPIOC, 0);  //全亮
      HAL_Delay(500);
    }
    /* USER CODE END EXTI0_IRQn 0 */
    
  5. 同理我们在void EXTI1_IRQHandler(void)中添加代码如下
    /* USER CODE BEGIN EXTI1_IRQn 0 */
    int8_t i;  //循环变量
    for(i=0;i<3;i++)  //交替闪烁3次
    {
      LL_GPIO_WriteOutputPort(GPIOC, 0x55);  
      HAL_Delay(500);
      LL_GPIO_WriteOutputPort(GPIOC, 0xaa);  
      HAL_Delay(500);
    }
    /* USER CODE END EXTI1_IRQn 0 */
    

联合调试

  1. 点击运行,生成HEX文件。
  2. 在Proteus中加载相应HEX文件,点击运行,正常显示流水灯状态;当按下按钮BTN0时,8个LED灯全亮全灭闪烁3次后恢复到常态;当按下按钮BTN1时,8个LED灯间隔交替闪烁3次后恢复常态。
posted @ 2022-05-07 10:29  Sheepeach  阅读(2399)  评论(0编辑  收藏  举报