STM32CubeMX学习笔记12 ---低功耗模式

在实际使用中很多产品都需要考虑低功耗的问题,STM32F10X提供了三种低功耗模式:睡眠模式(Sleep mode)、停机模式(Stop mode)和待机模式(Standby mode)。这些低功耗模式可以有效减少系统功耗,延长电池寿命,对于需要长时间运行的电池供电设备尤为重要。

注意:在进入低功耗模式时,STlink将无法下载程序,需要重启单片机!!

1、 低功耗模式简介

        系统或电源复位后,微处理器处于运行状态,运行状态下HCLK为CPU提供时钟,内核执行程序代码。当CPU不需继续运行时(例如等待某个外部事件),可以利用多个低功耗模式来节省功耗。用户需要根据最低电源消耗、最快速启动时间和可用的唤醒源等条件,选定一个最佳的低功耗模式。STM32提供了3种低功耗模式,以达到不同层次的降低功耗的目的: - 睡眠模式(CM3内核停止工作,外设仍在运行)(功耗最高) - 停止模式(所有时钟都停止)(典型电流消耗20uA左右) - 待机模式(1.8V内核电源关闭)(最低电流消耗2uA左右)

        在运行模式下,也可以通过如下方式降低功耗: - 降低系统时钟 - 关闭APB和AHB总线上未被使用的外设的时钟

一、睡眠模式

1、睡眠模式简介

        睡眠模式,它使处理器暂停执行,并且可以通过外部事件或中断来唤醒。在睡眠模式下,(Cortex™-M3内核停止,所有外设包括Cortex-M3核心的外设,如NVIC、系统时钟(SysTick)等仍在运行,以便在唤醒时快速恢复正常运行。

睡眠模式的进入:两种睡眠模式,SLEEP-NOW或 SLEEP-ON-EXIT。

SLEEP-NOW:如果SLEEPONEXIT位被清除,当WRI或WFE被执行时,微控制器立即进入睡眠模式。
SLEEP-ON-EXIT:如果SLEEPONEXIT位被置位,系统从最低优先级的中断处理程序中退出时,微控制器就立即进入睡眠模式。
睡眠模式的唤醒:两种唤醒方式,任一中断(WFI)或唤醒事件(WFE)。

WFI:任意一个中断的触发都能唤醒该模式。
WFE:唤醒事件

2、硬件设置

本实验用LED2指示灯提示系统正常运行,指示灯熄灭表示进入睡眠模式,S4按键用外部中断的方式来唤醒待机模式,并使用串口1打印相关调试信息

 3、STM32CubeMX设置

选择芯片STM32F103ZET6,建立工程,设置好工程名称及目录。

RCC设置外接HSE,时钟设置为72M。

设置串口1,并打开串口中断。

将PA0设置成外部中断模式,触发模式选择上升沿触发,GPIO Pull-up/Pull-down中选择Pull-up上拉,并命名为S4。使能外部中断,并将优先级都设置为2.

 输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码

4、程序编程

在main.c文件中添加以下程序

int main(void)
{
while (1)
  {
		printf1("Time: 5\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 4\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 3\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 2\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 1\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("进入睡眠模式!\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);   
		//进入睡眠模式
		// 挂起(暂停)系统时钟中断,这里是主要的减少功耗的地方
		HAL_SuspendTick();
			
		/* 进入睡眠模式, 等待任意中断唤醒 ,程序进入睡眠模式后就会在这里停住,
           往下的程序就不运行了,等待中断触发,就可以唤醒睡眠模式,然后继续运行往下的程序*/
		HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);
			
		/* 恢复系统时钟中断 */
		HAL_ResumeTick();

		printf1("退出睡眠模式!\r\n");
    }
}

编写外部中断函数

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){

	delay(10);  //延时消抖,不建议
					
	if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==1)
        printf1("STM32F103 EXTI test this key is S4\r\n"); //A0
        printf1("中断唤醒睡眠模式\r\n"); //A0
    
    }
}

5、下载验证:

程序编译无误后下载到板子上查看串口输出结果,以看到系统运行时LED2指示灯不断闪烁,5秒钟后进入睡眠模式,此时LDE2指示灯熄灭。当按下S4按键或复位按键时,睡眠模式被唤醒,系统继续运行,同时串口打印提示信息


                        

二、停机模式

1、停机模式简介


  停止模式是在Cortex™-M3的深睡眠模式基础上结合了外设的时钟控制机制,在停止模式下电压 调节器可运行在正常或低功耗模式。此时在1.8V供电区域的的所有时钟都被停止,PLL、HSI和 HSE RC振荡器的功能被禁止,SRAM和寄存器内容被保留下来。在停止模式下,所有的I/O引脚都保持它们在运行模式时的状态。当一个中断或唤醒事件导致退出停止模式时,会自动选用内部高速时钟(HSI RC振荡器)为系统时钟。

2、硬件设计

本实验用LED2指示灯提示系统正常运行,指示灯熄灭表示进入停机模式,S4按键用外部中断的方式来唤醒停机模式,并使用串口1打印相关调试信息。跟睡眠模式一致。

 3、STM32CubeMX设置

选择芯片STM32F103ZET6,建立工程,设置好工程名称及目录。

RCC设置外接HSE,时钟设置为72M。

设置串口1,并打开串口中断。

将PA0设置成外部中断模式,触发模式选择上升沿触发,GPIO Pull-up/Pull-down中选择Pull-up上拉,并命名为S4。使能外部中断,并将优先级都设置为2.

 输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码

4、程序编程

在main.c文件中添加以下程序

int main(void)
{
  //****省略 ****//
    while(1)
    {
        
	printf1("Time: 5\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 4\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 3\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 2\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 1\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("进入停机模式!\r\n");

        // 暂停滴答时钟,防止通过滴答时钟中断唤醒
        HAL_SuspendTick();
        /* 进入停止模式,设置电压调节器为低功耗模式,等待中断唤醒 */
        HAL_PWR_EnterSTOPMode(PWR_MAINREGULATOR_ON,PWR_STOPENTRY_WFI);//在此停止

        //唤醒后需要重新配置RCC时钟才能正常运行
        SYSCLKConfig_STOP();
        // 被唤醒后,恢复滴答时钟
        HAL_ResumeTick();
    printf1("退出停机模式!\r\n");
        
    }
}

//重新配置RCC时钟函数
static void SYSCLKConfig_STOP(void)
{
    RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
    RCC_OscInitTypeDef RCC_OscInitStruct = {0};
    uint32_t pFLatency = 0;
   
    /* 启用电源控制时钟 */
    __HAL_RCC_PWR_CLK_ENABLE();
  
    /* 根据内部 RCC 寄存器获取振荡器配置 */
    HAL_RCC_GetOscConfig(&RCC_OscInitStruct);
  
    /* 从停止模式唤醒后重新配置系统时钟: 启用 HSE 和 PLL */
    RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
    RCC_OscInitStruct.HSEState = RCC_HSE_ON;
    RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
    if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) 
    {
        while (1)
        {;
        }
    }
    /* 根据内部 RCC 寄存器获取时钟配置 */
    HAL_RCC_GetClockConfig(&RCC_ClkInitStruct, &pFLatency);
    /* 选择 PLL 作为系统时钟源, 并配置 HCLK、PCLK1 和 PCLK2 时钟分频系数 */
    RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_SYSCLK;
    RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
    if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, pFLatency) != HAL_OK) 
    {
        while (1) 
        {;
        }
    }
}

编写外部中断函数

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){

	delay(10);  //延时消抖,不建议
					
	if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==1)
        printf1("STM32F103 EXTI test this key is S4\r\n"); //A0
        printf1("中断唤醒睡眠模式\r\n"); //A0
    
    }
}

5、下载验证:

程序编译无误后下载到板子上查看串口输出结果,以看到系统运行时LED2指示灯不断闪烁,5秒钟后进入停止模式,此时LDE2指示灯熄灭。当按下S4按键或复位按键时,停止模式被唤醒,系统继续运行,同时串口打印提示信息

三、待机模式

        本文仅对STM32的最低功耗模式(即待机模式)来做介绍。待机模式可实现STM32的最低功耗,该模式实在CM3深睡眠模式时关闭电压调节器,整个1.8V供电区域被断电,PLL/HSI/HSE振荡器也被断电,SRAM和寄存器内容丢失,仅备份的寄存器和待机电路维持供电 下图为STM32进入及退出待机模式的条件:

2、 硬件设计

本实验用LED2指示灯提示系统正常运行,指示灯熄灭表示进入待机模式,S4按键用来唤醒待机模式,并使用串口1打印相关调试信息

  • LED2指示灯
  • S4按键
  • USART1串口

3、STM32CubeMX设置

  • RCC设置外接HSE,时钟设置为72M
  • PE5(LED2)设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平
  • PA0(S4)设置为GPIO上拉输入模式,F103中指定PA0脚为唤醒引脚。
  • USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位

  • 输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码 

4、程序编程

在main.c中添加一下程序

void Sys_Enter_Standby(void){
    __HAL_RCC_PWR_CLK_ENABLE();     //使能PWR时钟
    __HAL_PWR_CLEAR_FLAG(PWR_FLAG_WU);      //清除Wake_UP标志
    HAL_PWR_EnableWakeUpPin(PWR_WAKEUP_PIN1);   //设置WAKEUP用于唤醒
    HAL_PWR_EnterSTANDBYMode();     //进入待机模式
}

int main(void)
{
///****省略*****///
  while (1)
  {
    printf1("Time: 5\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 4\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 3\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Time: 2\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_SET);       
    HAL_Delay(1000);

    printf1("Time: 1\r\n");
    HAL_GPIO_WritePin(GPIOE,GPIO_PIN_5,GPIO_PIN_RESET);     
    HAL_Delay(1000);

    printf1("Entered Standby Mode...Please press KEY_UP to wakeup system!\r\n");
    Sys_Enter_Standby();		//

    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

 5、下载验证:

程序编译无误后下载到板子上查看串口输出结果,以看到系统运行时LED2指示灯不断闪烁,5秒钟后进入待机模式,此时LDE2指示灯熄灭。当按下S4按键或复位按键时,待机模式被唤醒,系统重新运行,同时串口打印提示信息

总结

1.1 睡眠模式
        在睡眠模式中,仅关闭了内核时钟,内核停止运行,但其片上外设,CM3 核心的外设全都还照常运行。有两种方式进入睡眠模式,它的进入方式决定了从睡眠唤醒的方式,分别是 WFI(wait for interrupt) 和 WFE(wait for event),即由等待“中断”唤醒和由“事件”唤醒。

特性和说明:

  • 立即睡眠: 在执行 WFI 或 WFE 指令时立即进入睡眠模式。
  • 退出时睡眠: 在退出优先级最低的中断服务程序后才进入睡眠模式。
  • 进入方式: 内核寄存器的 SLEEPDEEP=0 ,然后调用 WFI 或 WFE 指令即可进入睡眠模式;SLEEPONEXIT=1 时,进入“退出时睡眠”模式。
  • 唤醒方式: 如果是使用 WFI 指令睡眠的,则可使用任意中断唤醒;如果是使用 WFE 指令睡眠的,则由事件唤醒。
  • 睡眠时: 关闭内核时钟,内核停止,而外设正常运行,在软件上表现为不再执行新的代码。这个状态会保留睡眠前的内核寄存器、内存的数据。
  • 唤醒延迟: 无延迟。
  • 唤醒后: 若由中断唤醒,先进入中断,退出中断服务程序后,接着执行 WFI 指令后的程序;若由事件唤醒,直接接着执行 WFE 后的程序。

1.2 停止模式
        在停止模式中,进一步关闭了其它所有的时钟,于是所有的外设都停止了工作,但由于其 1.8V 区域的部分电源没有关闭,还保留了内核的寄存器、内存的信息,所以从停止模式唤醒,并重新开启时钟后,还可以从上次停止处继续执行代码。停止模式可以由任意一个外部中断(EXTI)唤醒,在停止模式中可以选择电压调节器为开模式或低功耗模式。

特性和说明:

  • 调压器低功耗模式: 在停止模式下调压器可工作在正常模式或低功耗模式,可进一步降低功耗。
  • 进入方式: 内核寄存器的 SLEEPDEEP=1,PWR_CR 寄存器中的 PDDS=0,然后调用 WFI 或 WFE 指令即可进入停止模式;PWR_CR 寄存器的 LPDS=0 时,调压器工作在正常模式,LPDS=1 时工作在低功耗模式。
  • 唤醒方式: 如果是使用 WFI 指令睡眠的,可使用任意 EXTI 线的中断唤醒;如果是使用 WFE 指令睡眠的,可使用任意配置为事件模式的 EXTI 线事件唤醒。
  • 停止时: 内核停止,片上外设也停止。这个状态会保留停止前的内核寄存器、内存的数据。
  • 唤醒延迟: 基础延迟为 HSI 振荡器的启动时间,若调压器工作在低功耗模式,还需要加上调压器从低功耗切换至正常模式下的时间。
  • 唤醒后: 若由中断唤醒,先进入中断,退出中断服务程序后,接着执行 WFI 指令后的程序;若由事件唤醒,直接接着执行 WFE 后的程序。唤醒后,STM32 会使用 HSI 作为系统时钟。

1.3 待机模式
        待机模式,它除了关闭所有的时钟,还把 1.8V 区域的电源也完全关闭了,也就是说,从待机模式唤醒后,由于没有之前代码的运行记录,只能对芯片复位,重新检测 boot 条件,从头开始执行程序。它有四种唤醒方式,分别是 WKUP(PA0)引脚的上升沿,RTC 闹钟事件,NRST 引脚的复位和 IWDG(独立看门狗)复位。

特性和说明:

  • 进入方式: 内核寄存器的 SLEEPDEEP=1,PWR_CR 寄存器中的 PDDS=1,PWR_CR 寄存器中的唤醒状态位 WUF=0,然后调用 WFI 或 WFE 指令即可进入待机模式。
  • 唤醒方式: 通过 WKUP 引脚的上升沿,RTC 闹钟、唤醒、入侵、时间戳事件或 NRST 引脚外部复位及 IWDG 复位唤醒。
  • 待机时: 内核停止,片上外设也停止;内核寄存器、内存的数据会丢失;除复位引脚、RTC_AF1 引脚及 WKUP 引脚,其它 I/O 口均工作在高阻态。
  • 唤醒延迟: 芯片复位的时间。
  • 唤醒后: 相当于芯片复位,在程序表现为从头开始执行代码。

WFI与WFE命令

我们了解到进入各种低功耗模式时都需要调用 WFI 或 WFE 命令,它们实质上都是内核指令,在库文件 core_cm3.h 中把这些指令封装成了函数。

/** brief 等待中断
  等待中断 是一个暂停执行指令
  暂停至任意中断产生后被唤醒
*/
#define __WFI        __wfi 

/** brief 等待事件
  等待事件 是一个暂停执行指令
  暂停至任意事件产生后被唤醒
*/
#define __WFE        __wfe

对于这两个指令,我们应用时一般只需要知道,调用它们都能进入低功耗模式,需要使用函数的格式“__WFI();”和“__WFE();”来调用(因为__wfi 及__wfe 是编译器内置的函数,函数内部调用了相应的汇编指令)。

其中 WFI 指令决定了它需要用中断唤醒,而 WFE 则决定了它可用事件来唤醒。

  • 21
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好!对于STM32F407和ETH LWIP,我可以为您提供一些关于TCP client客户端的开发方面的信息。 首先,您需要使用STM32CubeMX软件来配置STM32F407的ETH和LWIP库。在STM32CubeMX中,您可以配置以太网接口和选择使用LWIP协议栈。确保在配置期间启用TCP/IP协议。 一旦配置完成并生成代码,您可以在生成的代码中找到LWIP库的API函数以及以太网驱动程序的相关函数。在这里,您将能够设置和管理TCP连接。 以下是一个简单的TCP客户端示例代码,用于向服务器发送数据: ```c #include "lwip/api.h" #define SERVER_IP "192.168.0.100" #define SERVER_PORT 8080 void tcp_client_task(void *arg) { struct netconn *conn; err_t err; // 创建TCP连接 conn = netconn_new(NETCONN_TCP); if (conn != NULL) { ip_addr_t server_addr; // 设置服务器IP地址和端口 IP4_ADDR(&server_addr, 192, 168, 0, 100); // 连接服务器 err = netconn_connect(conn, &server_addr, SERVER_PORT); if (err == ERR_OK) { const char *data = "Hello, server!"; struct netbuf *send_buf; // 创建发送数据包 send_buf = netbuf_new(); if (send_buf != NULL) { // 将数据添加到发送数据包中 netbuf_ref(send_buf, data, strlen(data)); // 发送数据包 err = netconn_send(conn, send_buf); // 释放发送数据包 netbuf_delete(send_buf); } } // 关闭连接 netconn_close(conn); netconn_delete(conn); } vTaskDelete(NULL); } ``` 请注意,此示例代码仅用于演示目的,您可能需要根据您的具体需求进行修改。另外,还要确保正确初始化LWIP协议栈和以太网接口。 希望这些信息对您有所帮助!如果您有任何进一步的问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值