使用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口就可以调试了,不用再接串口线了

参考文献
[1] MDK help. Indirect semihosting C library function dependencies
[2] MDK help ARM Development Tools.
         Debugger Adapter User's Guides
             J-Link/J-Trace User's Guide
         Libraries and Floating Point Support Referencee
         Libraries and Floating Point Support Guide

Linker Reference Guide

转载于:https://www.cnblogs.com/wuhh123/p/11208569.html

keil+stm32+jlink利用swd ITM 方式进行printf输出相关推荐

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

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

  2. keil+stm32+jlink 用swd方式printf输出

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

  3. 关于STM32的ITM 方式输出走了一些弯路

    最简单直接且有效的方式就是这篇文章:使用SWO代替UART,实现printf打印功能. 实现STM32的printf输出的四种方式   四种实现方式,我认为第二种最简单可行. 方式一二三通用配置 方式 ...

  4. STM32的SWD调试方式

    http://www.ichanging.org/stm32-swd.html 找到一篇比较好的 关于stm32  SWD模式 下载 调试 配置文章 整理如下: 我们比较常用的是Jlink下载器 ,这 ...

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

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

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

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

  7. linux stm32 swd,stm32 JTAG和SWD的使用区别

    二.仿真器对 SWD 模式支持情况 1. 市面上的常用仿真器对 SWD 模式支持情况 (1) JTAGV6 支持 SWD 仿真模式, 速度较慢. (2) JTAGV7 比较好的支持 SWD 仿真模式, ...

  8. JLINK与 SWD接口

    JLINK与 SWD接口 1.使用Jlink连接 Jlink驱动:SEGGER - The Embedded Experts - Downloads - J-Link / J-Trace pylink ...

  9. ICP、ISP、IAP、JTAG、SWD下载方式

    目录 ICP.ISP.IAP.JTAG.SWD下载方式 概述 JTAG SWD ICP ISP IAP ISP与ICP的差别 ISP和IAP的差别 ICP.ISP.IAP.JTAG.SWD下载方式 概 ...

最新文章

  1. 8.11. Migrating MySQL Data into Elasticsearch using logstash
  2. linux run文件夹,Linux下运行run文件
  3. PHP读取excel表格内容 PHP-ExcelReader
  4. 快速幂(Fast_Power)
  5. Shell(6)——array的删改unset
  6. ajax 中文乱码问题 主要是IE浏览器
  7. 脸部螨虫和肥皂洗脸的问题
  8. 探索Julia(part2)--关于IDE
  9. 番外篇01:angularJS最佳实战
  10. LeetCode 309. 最佳买卖股票时机含冷冻期(动态规划)
  11. 推荐:26种NLP练手项目(代码+数据)
  12. 重构,体现一个工程师的基本素养和底蕴
  13. jupyter notebook python环境_jupyter Notebook环境搭建
  14. lucene 全文检索引擎的架构
  15. SQL 查找重复记录
  16. altiumdesigner的基本你操作
  17. linux安装yum
  18. Microsoft DirectX组件v11.0完整版更新啦
  19. mysql服务启动失败
  20. windows系统下itunes的备份路径修改

热门文章

  1. 利用Javascript制作网页特效(其他常见特效)
  2. 【一起学d3.js】(一)
  3. HDU2097:Sky数
  4. 淘宝/天猫API:item_search_pro-高级关键字搜索淘宝商品
  5. (三)DepthAI-python相关接口:OAK Nodes
  6. 【愚公系列】2022年11月 微信小程序-本地生活项目-首页
  7. wifi 6路由器如何增强穿墙能力
  8. 美团安卓面试,阿里巴巴Android面试都问些什么?含小米、腾讯、阿里
  9. 图文并茂使用CocosBuilder制作Cocos2D游戏
  10. node各个版本的下载地址