当前位置: 首页 > ds >正文

Keil MDK中禁用半主机(No Semihosting)

ARM 编译器(如 Keil MDK) 中禁用半主机(Semihosting)并实现标准库的基本功能,需要以下步骤:


1. 禁用半主机

#pragma import(__use_no_semihosting)  // 禁用半主机模式
  • 作用:防止标准库函数(如 printfscanf)依赖调试器进行 I/O 操作。
  • 后果:必须手动实现底层函数(如 _sys_exitfputc),否则会链接失败。

2. 定义简化 FILE 结构体

struct __FILE {int handle;  // 占位符,无实际用途(可简化)
};
FILE __stdout, __stdin;  // 标准输入/输出流
  • 说明
    • 标准库需要 FILE 结构体,但禁用半主机后无需复杂实现,仅需满足编译要求。
    • 如果不需要文件操作,可直接定义为空结构体:
      struct __FILE { int dummy; };
      

3. 必须实现的系统函数

(1) 程序退出处理 _sys_exit
#include "stm32f10x.h"  // 假设使用 STM32void _sys_exit(int x) {// NVIC_SystemReset();  // 硬件复位(推荐)// 或 while(1);      // 简单死循环
}
  • 作用:覆盖库的默认退出函数,避免链接错误。
  • 注意
    • 如果调用 exit() 或程序结束,会执行此函数。
(2) 输出重定向 fputc(支持 printf
int fputc(int ch, FILE *f) {USART_SendData(USART1, (uint8_t)ch);  // 发送到 USART1while (!USART_GetFlagStatus(USART1, USART_FLAG_TXE));  // 等待发送完成return ch;
}
  • 关键点
    • printf 依赖此函数输出字符。
    • 需提前初始化 USART(波特率、引脚等)。
(3) 输入重定向 fgetc(支持 scanf
int fgetc(FILE *f) {while (!USART_GetFlagStatus(USART1, USART_FLAG_RXNE));  // 等待接收数据return (int)USART_ReceiveData(USART1);
}
  • 关键点
    • scanf 依赖此函数读取字符。
    • 检查 USART_FLAG_RXNE(接收标志),而非 TC(发送完成)。

4. 完整示例代码

#pragma import(__use_no_semihosting)#include <stdio.h>
#include "stm32f10x.h"  // STM32 头文件// 简化 FILE 结构体
struct __FILE { int dummy; };
FILE __stdout, __stdin;// 系统函数
void _sys_exit(int x) { NVIC_SystemReset(); //也可以为空 }// 输出重定向(printf)
int fputc(int ch, FILE *f) {USART_SendData(USART1, (uint8_t)ch);while (!USART_GetFlagStatus(USART1, USART_FLAG_TXE));return ch;
}// 输入重定向(scanf)
int fgetc(FILE *f) {while (!USART_GetFlagStatus(USART1, USART_FLAG_RXNE));return (int)USART_ReceiveData(USART1);
}int main() {// 初始化 USART1(需自行实现)USART_InitTypeDef USART_InitStruct = { ... };USART_Init(USART1, &USART_InitStruct);USART_Cmd(USART1, ENABLE);printf("Hello, No-Semihosting!\n");  // 通过 USART1 输出int num;scanf("%d", &num);                  // 从 USART1 读取输入return 0;
}

5. 关键注意事项

功能实现要求
禁用半主机#pragma import(__use_no_semihosting)
FILE 结构体定义 struct __FILE__stdout/__stdin(可简化)
_sys_exit必须实现,建议硬件复位(NVIC_SystemReset())或死循环(while(1)
fputc重定向 printf 到硬件(如 UART)
fgetc重定向 scanf 从硬件读取(需检查 USART_FLAG_RXNE
硬件初始化确保 USART/UART 已正确配置(波特率、引脚模式等)

6. 常见问题解决

  • 问题1:printf 无输出
    • 检查 fputc 是否实现,并确认 USART 初始化正确。
  • 问题2:链接错误 undefined _sys_exit
    • 确保所有必需函数(_sys_exitfputc 等)均已实现。
  • 问题3:scanf 无法接收数据
    • 检查 fgetc 是否使用 USART_FLAG_RXNE,而非 TC 标志。

7. 扩展适配其他硬件

  • USB-CDC 重定向:替换 fputc/fgetc 为 USB 通信函数。
  • LCD 输出:修改 fputc 将字符显示到 LCD。
  • GCC 编译器:需实现 _write_read 而非 fputc/fgetc

如果需要针对 特定硬件平台(如 STM32、ESP32、NXP) 的详细配置代码,请提供具体型号!

http://www.xdnf.cn/news/459.html

相关文章:

  • 齐次坐标变换+Unity矩阵变换
  • 【Tauri2】026——Tauri+Webassembly
  • 代谢组数据分析(二十四):基于tidymass包从质谱原始数据到代谢物注释结果的实践指南
  • vue3 watch和watchEffect 的用法和区别
  • 计算机视觉算法实现——智能座椅坐姿识别
  • 基于GRPO将QWEN训练为和deepseek一样的推理模型!
  • linux kernel irq相关函数详解
  • 国产的 Java Solon v3.2.0 发布(央企信创的优选)
  • Day10【基于encoder- decoder架构实现新闻文本摘要的提取】
  • 第3章 垃圾收集器与内存分配策略《深入理解Java虚拟机:JVM高级特性与最佳实践(第3版)》
  • 从0开发一个unibest+vue3项目,使用vscode编辑器开发,总结vue2升vue3项目开始,小白前期遇到的问题
  • Flask应用部署通用指南
  • [Windows] Wireshark 网络抓包工具 v4.4.6
  • 【MATLAB海洋专题】历史汇总
  • C++学习之路,从0到精通的征途:vector类的模拟实现
  • 算法期末复习
  • 基于Fabric.js的选座布局系统开发笔记
  • 如何系统地入门学习stm32?
  • Linux419 三次握手四次挥手抓包 wireshark
  • python程序的流程
  • 移动自动化测试-appium
  • thanos rule组件和prometheus区别?
  • 5G基站设计难题:尺寸、重量、功耗和散热
  • HashMap 初步理解 put 操作流程 HashMap 的线程安全问题
  • 精益数据分析(5/126):解锁创业成功的关键密码
  • FPGA练习———DDS波形发生器
  • 一个项目中多个Composer的使用方法
  • 读文献方法
  • Linux系统之diff3命令详解
  • Nodejs数据库单一连接模式和连接池模式的概述及写法