请注意,Cortex M0和Cortex M0 +器件(例如STM32L053)的核心没有ITM en SWO(参见数据手册),即cortex-M0和cortex-M0+不能使用swd printf

转至http://www.stmcu.org.cn/module/forum/thread-606866-1-1.html

使用ITM机制实现调试stm32单片机,实现printf与scanf。

1. ITM简介
ITM机制是一种调试机制,是新一代调试方式,在这之前,有一种比较出名的调试方式,称为半主机(semihosting)方式。

在pc上编写过C语言的人都知道,printf可以向控制台输出,scanf可以从控制台获取输入,这里的printf/scanf都是标准库函数,利用操作系统的这些函数,我们可以很方便的调试程序。在嵌入式设备上(如stm32单片机平台上)开发工具(如MDK/IAR)也都提供了标准库函,自然也提供了printf/scanf函数,那么这些函数是否可以使用呢? 问题来了,printf向哪里输出呢?并且大部分情况下,也没有键盘,又如何使用scanf实现输入呢?

我们都知道,嵌入式设备一般的使用仿真器,如常见Jlink/ulink,可以实现烧录,单步,下断点,查看变量,等等。仿真器将PC机和单片机连接器来。聪明的设计者们就在考虑是否可以借助仿真器,使得单片机可以借助PC机的屏幕以及PC机的键盘实现printf的输出和scanf的按键获取。
也就是说,如下的hello,world程序

  1. #include <stdio.h>
  2. int main()
  3. {
  4. //硬件初始化
  5. //....
  6. printf("hello, world");
  7. for(;;);
  8. }

复制代码

这个程序烧录到单片机中后,仿真器连接接单片机与PC,开始在线调试后,那么这个程序会将"Hello, world"输出到PC机上,在开发工具(MDK/IAR等)的某个窗口中显示。

这就相当于,单片机借助了PC机的显示/输入设备实现了自己的输出/输入。这种方式无疑可以方便程序开发者调试。

这种机制有多种实现方式,比较著名的就是semihosting(半主机机制)和ITM机制。
ITM是ARM在推出semihosting之后推出的新一代调试机制。现在我们来尝试一下这种方式调试。

2. stm32使用ITM调试
MCU:stm32f207VG
仿真器:Jlink V8
IDE:MDK4.50

2.1 硬件连接
ITM机制要求使用SWD方式接口,并需要连接SWO线,一般的四线SWD方式(VCC SDCLK,SDIO,GND)是不行的。标准的20针JTAG接口是可以的,只需要在MDK里设置使用SWD接口即可。

2.2 添加重定向文件
将下面的文件保存成任意C文件,并添加到工程中。这里对这个文件简单说明一下,要知道我们的程序是在单片机上运行的,为什么printf可以输出到MDK窗口里去呢?这是因为 标准库中的printf实际上调用 fputc实现输出,所以我们需要自己编写一个fputc函数,这个函数会借助ITM(类似于USART)提供的寄存器,实现数据的发送,仿真器会收到这些数据,并发往PC机。

实际上,如果你的单片机和一块LCD连接,那么你只需要重新实现fputc函数,并向LCD上输出即可,那么你调用printf时就会输出到LCD上了。这中机制,就是所谓的重定向机制。

  1. #include <stdio.h>
  2. #define ITM_Port8(n)    (*((volatile unsigned char *)(0xE0000000+4*n)))
  3. #define ITM_Port16(n)   (*((volatile unsigned short*)(0xE0000000+4*n)))
  4. #define ITM_Port32(n)   (*((volatile unsigned long *)(0xE0000000+4*n)))
  5. #define DEMCR           (*((volatile unsigned long *)(0xE000EDFC)))
  6. #define TRCENA          0x01000000
  7. struct __FILE { int handle; /* Add whatever you need here */ };
  8. FILE __stdout;
  9. FILE __stdin;
  10. int fputc(int ch, FILE *f)
  11. {
  12. if (DEMCR & TRCENA)
  13. {
  14. while (ITM_Port32(0) == 0);
  15. ITM_Port8(0) = ch;
  16. }
  17. return(ch);
  18. }

复制代码

2.2 配置JLINK的初始化配置文件

将下面文件放置在你的工程下,并取任意名称,这里笔者取名为 STM32DBG.ini

  1. /******************************************************************************/
  2. /* STM32DBG.INI: STM32 Debugger Initialization File                           */
  3. /******************************************************************************/
  4. // <<< Use Configuration Wizard in Context Menu >>>                           //
  5. /******************************************************************************/
  6. /* This file is part of the uVision/ARM development tools.                    */
  7. /* Copyright (c) 2005-2007 Keil Software. All rights reserved.                */
  8. /* This software may only be used under the terms of a valid, current,        */
  9. /* end user licence from KEIL for a compatible version of KEIL software       */
  10. /* development tools. Nothing else gives you the right to use this software.  */
  11. /******************************************************************************/
  12. FUNC void DebugSetup (void) {
  13. // <h> Debug MCU Configuration
  14. //   <o1.0>    DBG_SLEEP     <i> Debug Sleep Mode
  15. //   <o1.1>    DBG_STOP      <i> Debug Stop Mode
  16. //   <o1.2>    DBG_STANDBY   <i> Debug Standby Mode
  17. //   <o1.5>    TRACE_IOEN    <i> Trace I/O Enable
  18. //   <o1.6..7> TRACE_MODE    <i> Trace Mode
  19. //             <0=> Asynchronous
  20. //             <1=> Synchronous: TRACEDATA Size 1
  21. //             <2=> Synchronous: TRACEDATA Size 2
  22. //             <3=> Synchronous: TRACEDATA Size 4
  23. //   <o1.8>    DBG_IWDG_STOP <i> Independant Watchdog Stopped when Core is halted
  24. //   <o1.9>    DBG_WWDG_STOP <i> Window Watchdog Stopped when Core is halted
  25. //   <o1.10>   DBG_TIM1_STOP <i> Timer 1 Stopped when Core is halted
  26. //   <o1.11>   DBG_TIM2_STOP <i> Timer 2 Stopped when Core is halted
  27. //   <o1.12>   DBG_TIM3_STOP <i> Timer 3 Stopped when Core is halted
  28. //   <o1.13>   DBG_TIM4_STOP <i> Timer 4 Stopped when Core is halted
  29. //   <o1.14>   DBG_CAN_STOP  <i> CAN Stopped when Core is halted
  30. // </h>
  31. _WDWORD(0xE0042004, 0x00000027);  // DBGMCU_CR
  32. _WDWORD(0xE000ED08, 0x20000000);   // Setup Vector Table Offset Register
  33. }
  34. DebugSetup();                       // Debugger Setup

复制代码

这里对这个文件做简单的解释,
_WDWORD(0xE0042004, 0x00000027); // DBGMCU_CR
这一句表示想 0xE0042004地址处写入 0x000000027,这个寄存器是各个位表示的含义在注释中给出了详细的解释。 0x27即表示
        BIT0 DBG_SLEEP
        BIT1 DBG_STOP
        BIT2 DBG_STANDBY
        BIT5 TRACE_IOEN
注意,要使用ITM机制,必须要打开BIT5。

打开MDK工程,按照下图修改。

2.3 MDK中对JLINK的配置

下图中注意两点
1). 这里的CoreClock是120M,因为笔者使用的是stm32F207VG这款芯片,并且时钟配置为120M,所以这里填入120M,如果你使用stm32F10x,时钟配置成72M,那么这里需要填入72M。即需要跟实际情况保持一致。
2). 最后一定要将 0处打勾,并将其他bit位上的勾去掉,最好与此图保持一致,除CoreClock外。

2.4 烧录程序,并启动调试。可以看到,笔者在程序源码中插入了一句printf语句输出,然后按照下图,就可以看到程序的输出了。

3. 综合版本使用scanf和printf
3.1 添加retarget文件
将如下代码保存成retarget.c,然后加入到工程中。

  1. #pragma import(__use_no_semihosting_swi)
  2. struct __FILE { int handle; /* Add whatever you need here */ };
  3. FILE __stdout;
  4. FILE __stdin;
  5. int fputc(int ch, FILE *f)
  6. {
  7. return ITM_SendChar(ch);
  8. }
  9. volatile int32_t ITM_RxBuffer;
  10. int fgetc(FILE *f)
  11. {
  12. while (ITM_CheckChar() != 1) __NOP();
  13. return (ITM_ReceiveChar());
  14. }
  15. int ferror(FILE *f)
  16. {
  17. /* Your implementation of ferror */
  18. return EOF;
  19. }
  20. void _ttywrch(int c)
  21. {
  22. fputc(c, 0);
  23. }
  24. int __backspace()
  25. {
  26. return 0;
  27. }
  28. void _sys_exit(int return_code)
  29. {
  30. label:
  31. goto label;  /* endless loop */
  32. }

复制代码

3.2 编译运行
编译,烧录,运行,打开Debug (printf) viewer,就可以看到输入,参看下图

这里对retarget.c文件做几点说明.
1). 上面的代码实际是在X:\Keil\ARM\Startup\Retarget.c上修改而成的,scanf依赖的函数共有两个,fgetc和__backspace都需要实现,如果缺少__backespace函数,则scanf胡无法从Debug Viewer Dialog 窗口获取输入。另外上面提供的代码只是个demo,用于演示效果,用于生产时应该处理的更完善一些。见参考文献[1]

2). 函数ITM_SendChar,ITM_CheckChar,ITM_ReceiveChar在库文件CMSIS\Include\core_cm3.h中。

3) 查看函数的符号引用关系,可以通过生成详细的map文件来查看。命令行增加 --verbose --list rtt.map选项即可生成名为rtt.map的文件。

4. ITM与RTT结合(待实现)
grissiom 写道:
忽然想到,或许可以把这个半主机做成 device,然后 rt_console_set_device("semi") 就可以直接用半主机做 finsh/rt_kprintf 了…… 不知可行不可行……

prife: ITM的接收不知道是否支持中断,目前接收字符使用是轮询方式。如果是中断才有意义。这样可以把ITM设备做成一个 rtt 的device了,让finsh跑在 Debug printf Viewer窗口上。以后只要接一个jtag/SWD口就可以调试了,不用再接串口线了

keil+stm32+jlink 用swd方式printf输出相关推荐

  1. keil+stm32+jlink利用swd方式进行printf输出

    出处:http://www.douban.com/note/248637026/ ----------------------------------------------------------- ...

  2. keil+stm32+jlink利用swd ITM 方式进行printf输出

    使用ITM机制实现调试stm32单片机,实现printf与scanf. 1. ITM简介 ITM机制是一种调试机制,是新一代调试方式,在这之前,有一种比较出名的调试方式,称为半主机(semihosti ...

  3. 解决烧录问题:Jlink的三线制SWD方式连接STM32芯片无法识别的解决方案

    网上找到此文章解决了j-flash只能连接后不能烧录的问题,希望遇到相同问题的小伙伴,可以用同样的办法解决: 前段时间,做了一块板子,调试接口只留了三个口SWD.SWCLK.GND,在使用Jlink的 ...

  4. 2022-09-09 STM32 Jlink SWD接口SEGGER J-Flash烧录调试记录

    一.SWD接口 串行调试(Serial Wire Debug),与JTAG的20个引脚相比,SWD只需要4个(或者5个)引脚,结构简单,但是使用范围没有JTAG广泛,主流调试器上也是后来才加的SWD调 ...

  5. STM32串口通信中使用printf发送数据配置方法 开发环境 Keil

    STM32串口通信中使用printf发送数据配置方法(开发环境 Keil RVMDK) 已有 12456 次阅读2011-6-29 23:29 | 在STM32串口通信程序中使用printf发送数据, ...

  6. STM32 printf 输出到usart1

    MDK4.7环境 #include "stm32f10x.h" //在STM32中使用printf输出到USART1,在配置好USART1后,再点"Project-> ...

  7. C方式格式化输出(printf 函数详解)

    本篇将重点介绍C语言的格式化输出函数 printf.当然该函数在C++程序中也可以使用. 目录 printf 函数构成及标准函数语句 格式字符种类的介绍 printf 函数的具体细节实现 格式符数量与 ...

  8. ARM调试(2):在keil利用指令跟踪宏单元(ITM)重定向printf,并完成scanf实现数据双向交互

    在keil利用指令跟踪宏单元(ITM)重定向printf,并完成scanf实现数据双向交互 文章目录 在keil利用指令跟踪宏单元(ITM)重定向printf,并完成scanf实现数据双向交互 1. ...

  9. STM32 J-LINK、ST-Link、CMSIS-DAP

    1.J-Link J-Link是德国SEGGER公司为支持仿真ARM内核芯片推出的JTAG仿真器,很多ARM芯片的接口协议是JTAG,JLink一端接电脑USB接口,一端接CPU的JTAG接口,JLi ...

最新文章

  1. 数据架构简史:转换中的范式
  2. 再话单元测试unittest
  3. 安卓APP动态调试技术
  4. 【2020.12.30更新】信号处理常用公式(一)
  5. Python学习 Day 040 - css选择器
  6. 十多年前国内的三大杀毒软件瑞星、江民和金山毒霸的现状如何?
  7. 你不知道的接口测试之简单的开始
  8. VMware Workstation Pro 无法在Windows上运行的解决方法
  9. Oracle and MS SQL Server 2005
  10. Redis应用(四)——在Spring框架中的应用
  11. myEclipse-svn的安装使用
  12. 通过ip地址定位计算机,局域网通过IP地址如何找到电脑的位置
  13. python开发ps插件_【UI/UE】22款设计师必备的PS插件【附教程】
  14. Python头歌合集(题集附解)
  15. 【2020 Java基础快速学习路线】写了很久,这是一份最适合普通大众、非科班的路线
  16. SpringMvc下载文件损坏
  17. Python xlwt 操作 excel 表格基础(二):冻结窗口、设置加密保护、打印设置等
  18. 基于AIX VIOS常用命令整理
  19. 如何区分2G/3G/4G基站
  20. 绿色智慧档案馆构想之智慧档案馆环境综合管控一体化平台

热门文章

  1. 10 万阿里人都爱用的网红工具,语雀如何“用保护钱包的技术在保护文档安全​”?...
  2. 在线文本编辑器实现原理
  3. 怎样选择在线文本编辑器
  4. python3 bytes与str转换
  5. sysbase powerdesigner 过期 需要lic文件 解决方法
  6. SD-WAN 系列(5)SD-WAN = SDN + Internet线路 +专线 + WAN加速 + IPsec + DPI + ?
  7. easyui filebox 上传图片预览
  8. Vista Home Basic 安装Sqlserver 2000个人版
  9. GIS数据下载,全国省市县乡行政区划分
  10. 将cl-home的磁盘空间压缩分配给cl-root