使用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 配置JTAG的初始化配置文件
將下面文件放置在你的工程下,并取任意名稱,這里筆者取名為 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中對JTAG的配置
下圖中注意兩點
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;
}
上一篇:STM32內存之四(CCM)
下一篇:STM32的系統時鐘與SystemInit函數
推薦閱讀
史海拾趣
DAYLIGHT公司成立于XX年,由一群富有遠見卓識的電子工程師和投資者共同創立。當時,電子行業正經歷著快速的技術革新,市場對高效、可靠的電子產品需求日益增加。DAYLIGHT公司憑借其獨特的技術優勢和創新能力,成功開發出一系列高性能的電子產品,迅速在市場中立足。
ASSMANN WSW components公司深知品質是企業生存和發展的關鍵。因此,公司始終堅持品質至上的原則,從原材料采購到生產流程,再到產品出廠,每一個環節都嚴格把控。公司還建立了完善的質量檢測體系,確保每一件產品都符合最高標準。這種對品質的執著追求使得ASSMANN WSW的產品在市場上贏得了良好的口碑。
背景:隨著中國經濟的快速發展和消費者對電子產品需求的不斷增加,Galaxy公司敏銳地捕捉到這一市場機遇。
行動:在2003年初,Galaxy以縝密的市場調研為基礎,全力部署進入大陸市場。同年6月,公司推出了以“影馳”命名的Geforce家族系列產品,迅速吸引了眾多游戲發燒友和電子產品愛好者的關注。隨后,公司又在多個城市設立了銷售和服務網點,為消費者提供更加便捷的產品體驗和服務。
成果:經過多年的深耕細作,Galaxy在大陸市場取得了顯著成績,品牌影響力不斷增強,市場份額穩步提升。
作為一家有社會責任感的企業,創世公司始終關注環境保護和可持續發展。公司積極推廣綠色生產技術和資源循環利用方案,減少生產過程中的能源消耗和廢棄物排放。同時,創世公司還積極參與社會公益活動,為社會和環境貢獻自己的力量。這種對社會責任的關注和擔當使得創世公司在電子行業中樹立了良好的形象。
在電子行業中,品質控制和質量管理是企業生存和發展的關鍵。創世公司深知這一點,始終將品質放在首位。公司建立了嚴格的質量管理體系,從原材料采購到生產制造、產品檢測等各個環節都嚴格把控品質。同時,創世公司還引進了先進的生產設備和檢測儀器,確保產品的可靠性和穩定性。這種對品質的執著追求使得創世公司的產品在市場上獲得了良好的口碑和廣泛的認可。
為了進一步滿足市場需求,Anytek在2005年投建了安尼泰科(中國)生產基地,并成功投產運營。這一舉措不僅大幅提升了公司的生產能力,也為后續的產品研發和市場拓展奠定了堅實基礎。同年,Anytek的產品還通過了SGS ISO 9001(2000)國際質量體系認證,這標志著公司的產品質量和管理水平達到了國際標準,為公司的國際化發展打開了新局面。
在電池供電電路或恒流供電電路中,使繼電器工作的突發電流電涌可導致電源電壓下降。這是由于內部電阻或電流限制影響的結果。圖1所示的電路可克服此問題,此電路在各種條件下從電源吸入恒定1mA電流。 圖1所示電路控制三個Teledyne RF自鎖繼電器7221 ...… 查看全部問答∨ |
本人在“電子制作”08第一期上看到標題為“基于89S52單片機的電子體溫計。”但上面沒有給任何程序。 那位大俠有??可否發上來? 溫度傳感器是:DS18B20.還有個122*64點陣的液晶顯示。。 Help~~~~… 查看全部問答∨ |
cygwin安裝mips編譯器后不識別庫函數,請高手指教,謝謝! cygwin下安裝了mips的編譯器,但不能識別函數,即使寫一個“Hello World”,使用mips-elf-gcc編譯時也會提示“undefined reference to `printf\'”,直接用gcc編譯就沒問題。不知是何原因,望高手指教,謝謝!… 查看全部問答∨ |
|
最近需要寫個讀寫SMBUS設備的寄存器的程序,好象掛在SMBUS上的各類設備都有自己固定的SLAVE ADDRESS,所以想問下是否有相關的資料記錄了各類設備預先分配好的地址,如果有相關資料的話,是否可以提供下哪里下載?比如SPD的SLAVE ADDRESS為A0,A2,A4 ...… 查看全部問答∨ |
|