历史上的今天

今天是:2024年09月08日(星期日)

2020年09月08日 | semihost/ITM机制浅析以及使用JLINK通过ITM调试stm32单片机

发布者:NexusDream 来源: eefocus关键字:semihost  ITM机制  JLINK  ITM  stm32 手机看文章 扫描二维码
随时随地手机看文章

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


#include

int main()

{

        //硬件初始化

        //....

        printf("hello, world");

        for(;;);

}


这个程序烧录到单片机中后,仿真器连接接单片机与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上了。这中机制,就是所谓的重定向机制。


#include

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

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

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

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

#define TRCENA 0x01000000

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

    FILE __stdout;

    FILE __stdin;

int fputc(int ch, FILE *f) 

{

    if (DEMCR & TRCENA) 

    {

        while (ITM_Port32(0) == 0);

        ITM_Port8(0) = ch;

    }

    return(ch);

}


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

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


/******************************************************************************/

/* STM32DBG.INI: STM32 Debugger Initialization File */

/******************************************************************************/

// <<< Use Configuration Wizard in Context Menu >>> // 

/******************************************************************************/

/* This file is part of the uVision/ARM development tools. */

/* Copyright (c) 2005-2007 Keil Software. All rights reserved. */

/* This software may only be used under the terms of a valid, current, */

/* end user licence from KEIL for a compatible version of KEIL software */

/* development tools. Nothing else gives you the right to use this software. */

/******************************************************************************/

FUNC void DebugSetup (void) {

// Debug MCU Configuration

// DBG_SLEEP Debug Sleep Mode

// DBG_STOP Debug Stop Mode

// DBG_STANDBY Debug Standby Mode

// TRACE_IOEN Trace I/O Enable 

// TRACE_MODE Trace Mode

// <0=> Asynchronous

// <1=> Synchronous: TRACEDATA Size 1

// <2=> Synchronous: TRACEDATA Size 2

// <3=> Synchronous: TRACEDATA Size 4

// DBG_IWDG_STOP Independant Watchdog Stopped when Core is halted

// DBG_WWDG_STOP Window Watchdog Stopped when Core is halted

// DBG_TIM1_STOP Timer 1 Stopped when Core is halted

// DBG_TIM2_STOP Timer 2 Stopped when Core is halted

// DBG_TIM3_STOP Timer 3 Stopped when Core is halted

// DBG_TIM4_STOP Timer 4 Stopped when Core is halted

// DBG_CAN_STOP CAN Stopped when Core is halted

//

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

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

}

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,然后加入到工程中。


#pragma import(__use_no_semihosting_swi)

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

    FILE __stdout;

    FILE __stdin;

int fputc(int ch, FILE *f) 

{

    return ITM_SendChar(ch);

}

volatile int32_t ITM_RxBuffer;

int fgetc(FILE *f)

{

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

  return (ITM_ReceiveChar());

}

int ferror(FILE *f)

{

    /* Your implementation of ferror */

    return EOF;

}

void _ttywrch(int c)

{

    fputc(c, 0);

}

int __backspace()

{

    return 0;

}

void _sys_exit(int return_code)

{

label:

    goto label; /* endless loop */

}


3.2 编译运行

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

这里对retarget.c文件做几点说明.

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

2). 函数ITM_SendChar,ITM_CheckChar,ITM_ReceiveChar在库文件CMSISIncludecore_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


关键字:semihost  ITM机制  JLINK  ITM  stm32 引用地址:semihost/ITM机制浅析以及使用JLINK通过ITM调试stm32单片机

上一篇:用Eclipse开发stm32(GNU ARM Eclipse Plug-in)(编码+编译+下载+调试)
下一篇:如何通过stm32驱动电源检测芯片cs5463

推荐阅读

一、RAM空间在内存映射图上可以看到STM8L的RAM划分为2部分,如下:2K的RAM空间包含513个字节的Stack,经过深入研究发现,其实RAM被划分为:short range、long range、stack这3个部分。如下:可以在STVD中通过project setting设置上述3部分的大小,具体如下:Project->Settings->linker->input:Short range空间的最大值是256个字节,库函数有时也会占用...
昨天,小米产品总监、MIUI体验总负责人@MIUI小凡发微博宣布在MIUI10最后一个内测的开发版中,新增了一个开关:一键关闭MIUI所有系统工具广告的开关。一些铁杆小米用户看到这条微博感动的痛哭流涕:“小米的MIUI终于要摘掉ADUI的帽子呢”。那么问题来了,小米为什么要选择在此刻放弃MIUI的广告呢?对于该问题,魅族前员工@真黄轶轩发表长文进行了一番解析。...
要做一个报警功能的东东,要求可以通过遥控来改变遥控内容。由于对系统的稳定性要求很高,所以用了看门狗。可是看门狗复位会引起所有寄存器重置,恢复到默认状态。遥控要改变的内容也被复位了,所以只能借助EEPROM来存储报警标志了。我用的是MPLAB X IDE +XC8 编译器,XC8在语法上跟PICC有点不一样EEPROM操作有两种方法,在该存储空间中定义命名变量,...
日前,好利来(中国)电子科技股份有限公司(以下简称“好利来”)发布公告称,公司近日已与戴斌、北京锦远科技有限公司、浙江曲速科技有限公司(以下简称“曲速科技”)签署《合作意向书》。根据协议,公司拟向曲速科技增资5000万元,增资完成后将持有曲速科技3.125%股权。且公司后续有权要求戴斌及北京锦远科技有限公司向公司转让所持有的全部或部分股权...

史海拾趣

问答坊 | AI 解惑

基于EM-LPC1700开发板CAN测试通过了,请高手指点

前段时间使用英蓓特公司开发的EM-LPC1700开发板http://arm.embedinfo.com/list.asp?id=317(采用NXP公司的最新的Cortex-M3内核,主频有100M)做CAN的测试,没通过,最近采用他们提供的新的测试例程终于通过了。在这里跟大家分享一下 首先,将他们 ...…

查看全部问答∨

XILINX,ALTERA开发套件及资料

XILINX,ALTERA开发套件,下载线,并长期备有SPARTAN,VIRTEX系列常用芯片,并可免费提供ISE11.1,QUARTUS9.0软件及相关资料! QQ:379073540 MB:13910408597…

查看全部问答∨

射频识别PK无线互联 哪个更“泛在“?

“泛在”即“无所不在” ,也有人称之为“普适”即“普遍适用” 。20世纪80年代8位台式电脑开始普及,但有些办公设备的厂商认为计算不应只限于电脑,而是存在于社会的所有方面,也就是说任何电子设备里都应该包含计算技术。本世纪初日本东京工业大 ...…

查看全部问答∨

时钟设计程序问题

利用10M的时钟,设计一个周期为50微秒单周期的输出时钟,其中波峰为10微秒…

查看全部问答∨

混合信号FPGA促进临床医疗应用发展

  由于医疗服务成本的不断攀升、慢性病的流行、人口的老龄化,以及中国、印度和巴西等大规模新兴市场的崛起,对价格低廉又稳健可靠的医疗设备的需求非常庞大,以期改善全球数百万病患的治疗和护理条件,并扩大疾病治疗范围。另一方面,医疗设备设 ...…

查看全部问答∨

LED

有没有关于1W到10WLED的简单驱动电路图,和简单开关电源…

查看全部问答∨

wince 5.0 + 2440 ,无法同时支持三种USB 设备同时工作??

有没有人测试过,2440+wince5.0 的usb host 同时只能接两种设备,当第三种设备插上的时候,就会提示需要安装驱动。 例如:我现在设备上插上 usb 鼠标 + csr 的蓝牙适配器,都工作正常,但是,这个时候,我再把U 盘插上,就会跳出一个要我指定驱动 ...…

查看全部问答∨

关于ROM 和 RAM 编址问题

系统上电的时候,cs:ip 指向0xffff0,这个位置应该为ROM地址,这时RAM是不可用的,当ROM bios检测到RAM时,是否执行了代码拷贝?将ROM bios拷到1M高端的64k,如果bios大于64k怎么办呢?或者根本没有什么代码拷贝,只是进行了内存映射? video bios ...…

查看全部问答∨

怎么在tcpmp里注册一种新类型的视频格式

我现在想使tcpmp能够播放一种新的视频文件(.avs格式),想请教高手怎么才能在里面注册这种格式,使tcpmp能够认识.avs格式呢??…

查看全部问答∨
小广播
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved