STM32实验及笔记 —— (036)STM32重定向printf输出到USART的方法
»
(001)HAL库LED点灯
(002)HAL库LCD帧动画
(003)HAL库FFT动态频谱
(004)HAL库FFT Speed Up
(005)HAL库认识DMA
(006)看门狗总结
(007)GPIO使用步骤
(008)CubeIDE实现USB外设*
(009)为什么C99
(010)Linux UDP广播注意事项
(011)Linux非阻塞TCP
(012)ARM架构防烧脚本
(013)OpenWrt交叉编译
(014)C99基本数据类型
(015)C99 结构体
(016)C99 枚举
(017)C99 的函数及左值右值
(018)C99 的指针*
(019)C99 只有值赋值,没有move*
(020)C99 数组及结构体中的数组存储空间
(021)C99标准函数库没有函数模板
(022)POSIX线程库pthread的同步锁*
(023)POSIX线程库pthread的多线程*
(024)C99的条件语句
(025)C99的选择语句
(026)C99的循环语句
(027)C99标准库中的计时与等待 *
(028)C语言中的高精度计算库GMP *
(029)C语言中的Web服务*
(030)C语言中的字符转码ICU*
(031)从几个方面显然C语言比C++效率高*
(032)C99中的弱函数weak
(033)MTU TCP SOCKET
(034)STM32 UART连续发送数据函数
(035)STM32 UART连续接收数据函数
(036)STM32重定向printf输出到USART的方法
(037)STM32 Normal与Circle
(038)STM32 UART传输错误标志位的清除
(039)STM32 UART传输空闲中断
(040)STM32 串口、RS232是单根线发数据
(041)STM32 RS485差分信号通信
(042)STM32 等待DMA传输完成指定的字节数
(043)STM32进入低功耗模式
(044)DMA传输完成中断需要清空TCx标志位
(045)STM32软件函数命名规范
(046)在单字节完成中断中处理Idle
(047)C99内置宏
(048)使能中断后中断服务程序才起作用
(049)STM32程序启动过程
(050)STM32热加载新的程序固件
(051)中断向量表的指定
(052)STM32程序内的地址空间
(053)STM32修改主程序存放的地方
(054)Bootloader参考
(055)Keil中的下载算法
(056)编译器优化出错点
🍓(001)关闭交换分区
🍓(002)格式化U盘为ext4并设定块大小为1k
🍓(003)获取linux版本信息及硬件架构信息
🍓(004)最新版Ubuntu加速
🍓(005)不稳定网络克隆360浏览器内核脚本
🍓(006)Ubuntu24.04无法运行firefox注意事项
🍓(007)MAC地址
🍓(008)RMII和PHY
🍓(009)Ubuntu Server防火墙配置
🍓(010)Linux主动让出CPU
🍓(011)Linux硬件概念
🍓(012)Linux系统中的进程间通信
🍓(013)Git禁止强制推送
🍓(014)Git禁止reset --hard
🍓(015)将未写入的缓存立即写入硬盘
🍓(016)拷贝pre-receive到每个git仓库的脚本
🍓(017)sshd密码输入错误禁用IP3分钟
🖥️(001)异机备份低功耗自用Git服务器搭建
🖥️(002)OpenWRT配置注意事项
🕹️(001)while(1){}让出CPU后重进
🕹️(002)FreeRTOS主要函数
🕹️(003)FreeRTOS临界区ENTER_CRITICAL
🕹️(004)FreeRTOS NULL未定义问题的解决
🕹️(005)优先级最低的ARM硬件中断PENDSV
🕹️(006)任务优先级层数配置
🕹️(007)不同优先级任务创建注意事项
🕹️(008)关闭同级任务的时间片轮转
🕹️(009)OTA技术
🕹️(010)FreeRTOS静态任务创建
🕹️(011)FreeRTOS列出任务列表
🕹️(012)FreeRTOS堆空间使用量统计
🕹️(013)FreeRTOS通过任务名获取TaskHandle
🕹️(014)FreeRTOS常用任务创建方式
🕹️(015)FreeRTOS解决优先级反转的任务死锁
🕹️(016)FreeRTOS超级中断:SVC
🕹️(017)FreeRTOS中断处理函数运行在特权级
🕹️(018)FreeRTOS SVC进入用户模式的方法
🕹️(019)FreeRTOS实现绝对时间间隔开始任务
🕹️(020)FreeRTOS低功耗通过Tickless实现
🕹️(021)FreeRTOS IdleHook
🕹️(022)FreeRTOS Hooks
🕹️(023)FreeRTOS主动让出CPU
🕹️(024)FreeRTOS信号量
🕹️(025)FreeRTOS解决优先级反转问题的方法
🕹️(026)FreeRTOS获得任务的优先级
🕹️(027)FreeRTOS设置任务的优先级
🕹️(028)FreeRTOS中函数的前缀
🕹️(029)硬件中断中不能调用FreeRTOS的函数
🕹️(030)底半操作的任务接收信号量处理数据
🕹️(031)启用无限期等待
🕹️(032)中断处理函数中调用FreeRTOS的函数
🕹️(033)对于FreeRTOS顶半与底半操作的原则
🕹️(034)STM32中的FATFS支持exFat
🕹️(035)FreeRTOS中的消息队列
🕹️(036)FreeRTOS中的定时任务
🕹️(037)默认生成代码没法ping板子的原因
🕹️(038)LwIP Tcp Server开发例程
🕹️(039)切记别用mutex
🕹️(040)MEMP与Pbuf的关系
🕹️(041)FreeRTOS连接ESP WiFi(待完成)
📟(001)CMSIS标准学习
🕹️(001)函数调用被优化
STM32重定向printf输出到USART1的方法
1、重写_write函数放到syscall.c文件里面
int outputc(int ch){
    while((USART1->SR & USART_SR_TXE) == 0);// 等待 USART1 的发送数据寄存器准备好
    USART1->DR = (uint8_t)ch;                //填充需要发送的字节到USART1的数据寄存器(DR)
    while((USART1->SR & USART_SR_TC) == 0); //USART的状态寄存器(SR),等待直到发送完成
    return ch;
}
int _write(int file, char *ptr, int len) {
    // 判断是否由标准输出调用
    //if (file == STDOUT_FILENO || file == STDERR_FILENO) {//嵌入式不适用
        // 如果是标准输出,则通过串口发送
        for (int i = 0; i < len; i++) {
            //HAL_UART_Transmit(&huart1, (uint8_t *)&ptr[i], 1, HAL_MAX_DELAY);
            outputc(*ptr++);
        }
        return len; // 返回写入的字节数
    //}
    // 如果不是标准输出,返回错误 -1
    return -1;
}

注意:通过改写weak int _write(int file, char *ptr, int len)函数实现的printf没有缓冲区,printf输出的每个字符都会依次直接输出到UART,所以,顺序可能会乱。

调试过程:
如果printf函数TTL没有输出,先判断UART1与TTL是否有问题:
//直接在初始化完成后调用以下函数测试UART1是否正常:
HAL_UART_Transmit(&huart1, (uint8_t *)"Test UART1\n", 11, HAL_MAX_DELAY);


2、功能开发过程注意事项:
1)网口只有在带电烧录后才能使用的问题:
在硬件的开发过程中,需要考虑外设的加电过程、网卡的加电过程以及外设的加电完成时间、网卡的加电完成时间,特别是有振荡电路的外设。必要时使用Delay等待外设加电过程的完成。


技术以外的话:

与我共事:完整交付代码,不留bug,不保留代码,写经得起几年考验的代码。

与我共事,能提升:正确的FreeRTOS的内存配置方法;能提升:裸机while循环外的技巧(说明:裸机开发都只有一个主while循环,来自ChatGPT和DeepSeek)。能提升为什么嵌入式中的printf会导致死机:在FreeRTOS系统任意地方使用带%f号的printf(自我研究过程中总结的使用Queue实现的不阻塞高效printf)

人生格言:招之即来,辞之即去,挥挥衣袖,留下一片云彩。

 
«
——张人杰·www.v-signon.com学习者共勉
返回上一页
备案号:京ICP备19038994号-2
个人作品网站:www.up-task.com 主办:个人 English
网站内容如有侵权,请联系删除:1307776259@qq.com