freertos下printf(“hello\r\n“)和printf(“hello %d\r\n“,i)任务堆栈消耗有何区别

一 背景

原先做的项目,TCP通讯采用stm32f103+freertos+w5500方案,实现的tcp单连接通信,同事提了个需求让实现tcp多连接,于是改了一版代码。测试发现根本跑不起来,程序应该是卡死崩溃了,相关问题代码如下:

c 复制代码
void do_tcp_server(void)
{
         switch(wizGetSnSR(tcp_socket[i].sock_num))                                                /*获取socket的状态*/
         {
            case SOCK_CLOSED: 
               printf("SOCK %d CLOSED\r\n",tcp_socket[i].sock_num);                                            /*socket处于关闭状态*/
               socket(tcp_socket[i].sock_num ,Sn_MR_TCP,local_port,Sn_MR_ND);         /*打开socket*/
            break;

            case SOCK_INIT:   
               printf("SOCK %d INIT\r\n",tcp_socket[i].sock_num);                                                       /*socket已初始化状态*/
               listen(tcp_socket[i].sock_num);                                                   /*socket建立监听*/
            break;
            ...
         }
}

在上面这段代码里面,如果我将打印

c 复制代码
 printf("SOCK %d CLOSED\r\n",tcp_socket[i].sock_num);
 printf("SOCK %d INIT\r\n",tcp_socket[i].sock_num);

改为

c 复制代码
 printf("SOCK CLOSED\r\n");
 printf("SOCK INIT\r\n");

则程序运行正常。

二 定位分析过程

  1. 大方向是堆栈溢出 问了下chatgpt是说printf("hello")和printf("%d")格式化打印,消耗的堆栈是不一样的,前者纯字符串,压栈的时候只是个地址,而后者则要压入更多内容。这么说来应该是堆栈溢出问题。于是我将打印改为printf("SOCK CLOSED\r\n"")和printf("SOCK INIT\r\n")在另一个任务里面调用vTasklist打印出任务堆栈信息如下:
c 复制代码
Name          State  Prio  Stack  Num
greenLedTask   	R	3	86	3
yellowLedTask  	R	3	86	4
beepLedTask    	R	3	86	6
tcpServerTask  	R	3	0	1
redLedTask     	X	3	0	5
mail_station   	R	3	88	2
IDLE           	R	0	107	7

关联的任务函数tcpServerTask剩余堆栈大小是0,也就是说此时该任务已经没有堆栈空间了,printf改为%d打印,多一点就可能导致堆栈溢出;

2.堆栈消耗能差多少呢 我将tcpServerTask任务堆栈大小由128word改为256word,然后打印两种情况下的剩余对战情况:

c 复制代码
//printf("SOCK CLOSED\r\n"")和printf("SOCK INIT\r\n")
Name          State  Prio  Stack  Num
greenLedTask   	R	3	86	3
yellowLedTask  	R	3	86	4
beepLedTask    	R	3	86	6
tcpServerTask  	R	3	106	1
redLedTask     	X	3	0	5
mail_station   	R	3	88	2
IDLE           	R	0	107	7
c 复制代码
//printf("SOCK %d CLOSED\r\n",tcp_socket[i].sock_num);
//printf("SOCK %d INIT\r\n",tcp_socket[i].sock_num);
 Name          State  Prio  Stack  Num
redLedTask     	X	3	0	5
tcpServerTask  	R	3	57	1
IDLE           	R	0	109	7
greenLedTask   	B	3	86	3
yellowLedTask  	B	3	86	4
beepLedTask    	B	3	86	6
mail_station   	B	3	88	2

根据打印结果看printf%d格式化输出比单纯打印纯字符串,任务堆栈要多消耗50个word大小左右,所以在你任务本身剩余堆栈很少的情况下,一个printf打印就可能导致溢出,程序崩溃。

  1. 最后解决方案

由于多链接,这里打印最好能打印出socket编号,所以

  1. 增加总的堆内存大小5000-》8192
  2. 增加printf%d打印任务堆栈大小128-》256
c 复制代码
configTOTAL_HEAP_SIZE                    ((size_t)8192)
osThreadDef(tcpServerTask, StartTcpServerTask, osPriorityNormal, 0, 256);
相关推荐
chem41113 分钟前
STM32 ISP下载
stm32·单片机·接口隔离原则
say_fall14 分钟前
微机原理:微型计算机基础
服务器·网络·单片机·微机原理
BreezeJuvenile26 分钟前
ADC_案例练习:独立模式单通道转换
stm32·单片机·adc·hal·寄存器·单通道采集
zd84510150027 分钟前
stm32f407 电机多轴联动算法
stm32·单片机·算法
电子工程师成长日记-C5130 分钟前
基于51单片机的乒乓球计分器
单片机·嵌入式硬件·51单片机
weixin_6695452033 分钟前
单通道 2.7-12.0V 持续电流 2.3A H 桥驱动芯片 智能锁马达驱动IC XR8313
单片机·嵌入式硬件·硬件工程·信息与通信
先知后行。1 小时前
【无标题】
单片机·嵌入式硬件
boneStudent1 小时前
STM32 CAN总线数据采集与转发系统完整代码
stm32·单片机·嵌入式硬件
深耕AI10 小时前
【时钟周期 vs 指令】为什么51单片机需要12个时钟周期?
单片机·嵌入式硬件·51单片机
清风66666614 小时前
基于单片机的多功能智能婴儿车设计
单片机·嵌入式硬件·毕业设计·课程设计·期末大作业