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

发布者:Volare最新更新时间:2016-12-16 来源: eefocus关键字:keil  stm32  jlink  swd方式  printf输出 手机看文章 扫描二维码
随时随地手机看文章

使用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   

  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   

  2.   

  3. #define ITM_Port8(n)    (*((volatile unsigned char *)(0xE0000000+4*n)))  

  4. #define ITM_Port16(n)   (*((volatile unsigned short*)(0xE0000000+4*n)))  

  5. #define ITM_Port32(n)   (*((volatile unsigned long *)(0xE0000000+4*n)))  

  6. #define DEMCR           (*((volatile unsigned long *)(0xE000EDFC)))  

  7. #define TRCENA          0x01000000  

  8.   

  9. struct __FILE { int handle; /* Add whatever you need here */ };  

  10.     FILE __stdout;  

  11.     FILE __stdin;  

  12.       

  13. int fputc(int ch, FILE *f)   

  14. {  

  15.     if (DEMCR & TRCENA)   

  16.     {  

  17.         while (ITM_Port32(0) == 0);  

  18.         ITM_Port8(0) = ch;  

  19.     }  

  20.     return(ch);  

  21. }  


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.   

  13.   

  14. FUNC void DebugSetup (void) {  

  15. //  Debug MCU Configuration  

  16. //       DBG_SLEEP      Debug Sleep Mode  

  17. //       DBG_STOP       Debug Stop Mode  

  18. //       DBG_STANDBY    Debug Standby Mode  

  19. //       TRACE_IOEN     Trace I/O Enable   

  20. //    TRACE_MODE     Trace Mode  

  21. //             <0=> Asynchronous  

  22. //             <1=> Synchronous: TRACEDATA Size 1  

  23. //             <2=> Synchronous: TRACEDATA Size 2  

  24. //             <3=> Synchronous: TRACEDATA Size 4  

  25. //       DBG_IWDG_STOP  Independant Watchdog Stopped when Core is halted  

  26. //       DBG_WWDG_STOP  Window Watchdog Stopped when Core is halted  

  27. //      DBG_TIM1_STOP  Timer 1 Stopped when Core is halted  

  28. //      DBG_TIM2_STOP  Timer 2 Stopped when Core is halted  

  29. //      DBG_TIM3_STOP  Timer 3 Stopped when Core is halted  

  30. //      DBG_TIM4_STOP  Timer 4 Stopped when Core is halted  

  31. //      DBG_CAN_STOP   CAN Stopped when Core is halted  

  32. //   

  33. _WDWORD(0xE0042004, 0x00000027);  // DBGMCU_CR  

  34. _WDWORD(0xE000ED08, 0x20000000);   // Setup Vector Table Offset Register  

  35. }  

  36.   

  37. 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.   

  3. struct __FILE { int handle; /* Add whatever you need here */ };  

  4.     FILE __stdout;  

  5.     FILE __stdin;  

  6.       

  7. int fputc(int ch, FILE *f)   

  8. {  

  9.     return ITM_SendChar(ch);  

  10. }  

  11.   

  12. volatile int32_t ITM_RxBuffer;  

  13. int fgetc(FILE *f)  

  14. {  

  15.   while (ITM_CheckChar() != 1) __NOP();  

  16.   return (ITM_ReceiveChar());  

  17. }  

  18.   

  19. int ferror(FILE *f)  

  20. {  

  21.     /* Your implementation of ferror */  

  22.     return EOF;  

  23. }  

  24.   

  25. void _ttywrch(int c)  

  26. {  

  27.     fputc(c, 0);  

  28. }  

  29.   

  30. int __backspace()  

  31. {  

  32.     return 0;  

  33. }  

  34. void _sys_exit(int return_code)  

  35. {  

  36. label:  

  37.     goto label;  /* endless loop */  

  38. }  


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


关键字:keil  stm32  jlink  swd方式  printf输出 引用地址:keil+stm32+jlink利用swd方式进行printf输出

上一篇:STM32的USART中断死循环,形成死机。
下一篇:STM32定时器输出比较模式中的疑惑

推荐阅读最新更新时间:2024-03-16 15:25

机智云云平台+STM32的智能家居空气检测及净化设计
本文介绍的智能家居空气检测及净化系统是由桂林电子科技大学国际学院陈龙杰等人设计开发完成。通过使用机智云平台、STM32等,实现室内空气监测和排气系统,实时监控室内空气,采集室内温湿度、CO、CO2、PM2.5等各项指标,并将数据上报至APP客户端,用户可根据反馈的数据信息动态控制净化系统。 01 系统设计原理 气体传感器通过与stm32f767的模拟转数字接口 进行数据交互,MCU将获取的数据通过WIFI模组进行智能家居气体与机智云IOT平台物联,并实时上报至云端,用户可根据APP客户端来读取传感器的数据,用户根据反馈开启净化系统(见图1)。 图1 系统设计流程图 02 温湿度及气体浓度获取方法 根据各个传感器的特性曲线
[单片机]
机智云云平台+<font color='red'>STM32</font>的智能家居空气检测及净化设计
Keil C51使用教程---高级编程(四)
本章讨论以下内容: ●绝对地址访问 ●C与汇编的接口 ●C51软件包中的通用文件 ●段名转换与程序优化 第一节 绝对地址访问 C51提供了三种访问绝对地址的方法: 1. 绝对宏: 在程序中,用“#include”即可使用其中定义的宏来访问绝对地址,包括: CBYTE、XBYTE、PWORD、DBYTE、CWORD、XWORD、PBYTE、DWORD 具体使用可看一看absacc.h便知 例如: rval=CBYTE ;指向程序存贮器的0002h地址 rval=XWORD ;指向外RAM的0004h地址 2. _at_关键字 直接在数据定义后加上_at_ const即可,但是注意: (1)绝
[单片机]
基于STM32和CPLD可编程逻辑器件的等精度测频
传统的频率测量方法在实际应用中有较大的局限性,基于传统测频原理的频率计的测量精度将随被测信号频率的变化而变化,传统的直接测频法其测量精度将随被测信号频率的降低而降低,测周法的测量精度将随被测信号频率的升高而降低。本文中提出一种基于ARM与CPLD宽频带的数字频率计的设计,以微控器STM32作为核心控制芯片,利用CPLD可编程逻辑器件,实现闸门测量技术的等精度测频。 本设计的技术指标: 测频范围:1Hz~200MHz,分辨率为0.1Hz,测频相对误差百万分之一。 周期测量:信号测量范围与精度要求与测频功能相同。 占空比测量:准确度99%。 计数范围:0~1000000000,可手动暂停、复位。 功耗大小:5V
[单片机]
基于<font color='red'>STM32</font>和CPLD可编程逻辑器件的等精度测频
STM32家族介绍、区别及命名规则
一、STM32的八大产品及区别 每一种型号的向上延伸,内核部分都是相同的,区别在于外设部分,也就是在于外设功能的增加及丰富。比如F103和F102相比,外设就更丰富,多3相电机定时器、CAN2.0B和SDIO的2个IIS。 二、STM32的Cortex-M产品(所有型号互相兼容) 不同的STM32产品的型号的引脚及闪存容量是不一样的,详情可参考上面的表格,在自己选型时注意结合自己的实际需要来选择合适的芯片,外设太少会不满足,而外设过多优惠导致浪费,选型时可根据下面的命名规则来进行参考或者查看选型手册。 三、STM32命名规则
[单片机]
<font color='red'>STM32</font>家族介绍、区别及命名规则
利用STM32 ST-LINK Utility查看内核运行状态
本文围绕STM32 ST-LINK Utility讲几点主要功能及相关拓展知识: 1.STM32编程下载; 2.利用该编程工具查看内核运行状态; 3.Option Bytes选项字配置; 1STM32编程下载 STM32 ST-LINK Utility一个最重要的功能就是对STM32进行编程。支持常见的hex、bin文件,还有早期摩托罗拉定义的srec和s19格式的文件(说实话,我都不了解这两种格式的文件)。 这里主要想提示一下初学者:hex带有地址,而bin文件不带地址,下载时需要填写起始地址。 下面看两张在STM32 ST-LINK Utility中下载选择文件的图大家就明白了。 1.选择hex下载,地址不
[单片机]
利用<font color='red'>STM32</font> ST-LINK Utility查看内核运行状态
STM32中断优先级--学习心得
刚开始学习STM32单片机时,对中断优先级的规则不是很明白,通过今天的学习,对这方面的内容有了新的了解,记录下来以便日后查看。 首先说一下STM32的中断: STM32有84个中断,包括16个内核中断和68个可屏蔽中断,具有16级可编程的中断优先级。 STM32F103系列上面,又只有60个可屏蔽中断(在107系列才有68个) 中断的管理方法: 1.首先,对STM32中断进行分组,组0~4。同时,对每个中断设置一个抢占优先级和一个响应优先级值。 分组配置是在寄存器SCB- AIRCR中配置: 即设置优先级分组:void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup);
[单片机]
STM32 通用定时器的输出PWM功能 学习笔记
首先来说,你要使用PWM模式你得先选择用哪个定时器来输出PWM吧!除了TIM6、TIM7这两个普通的定时器无法输出PWM外,其余的定时器都可以输出PWM,每个通用定时器可以输出4路PWM,高级定时器TIM1、TIM8每个可输出7路PWM。选好定时器及通道后,下一步就是要使能定时器的时钟。先贴代码,再对每条代码做详细分析。 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; //选择定时器模式:TIM脉冲宽度调制模式2 TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能 TIM_OCI
[单片机]
意法半导体发布全新STM32超值系列微控制器
意法半导体(STMicroelectronics,简称ST)发布最新的STM32F030超值系列微控制器。批量订货最低价仅为0.32美元,STM32F030是低预算项目的理想选择,同时还让设计人员有机会使用型号齐全且软硬件兼容的32位微控制器产品组合,从而提升应用性能,扩大产品系列。 全新STM32F030超值系列基于48MHz的ARM® Cortex™-M0处理器内核。与同类竞争产品不同,STM32F030虽然只有8位微控制器的价格,但性能和特性并没有受到任何影响,如没有降低内核速度,同样保持丰富的外设数量和种类。 意法半导体微控制器产品部总经理Michel Buffa表示:“STM32F0超值型让32位微控制器更加平易
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved