查看: 4965|回復: 5
打印 上一主題 下一主題

用C++開(kāi)發(fā)STM32,解決了中斷向量問(wèn)題

[復制鏈接]
跳轉到指定樓層
樓主
發(fā)表于 2013-7-8 23:48:50 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
最近喜歡上了C++開(kāi)發(fā)STM32,C++語(yǔ)法上基本是C的擴展,所以習慣用C的人所有的知識和代碼可以繼續發(fā)揮作用。
另外,C++的OO編程,代碼架構更接近人的思考模式,即使我們不需要OO編程,代碼的可維護性也好了很多。

但若要完全C++開(kāi)發(fā),遇到的問(wèn)題是中斷向量的問(wèn)題,用C來(lái)包裝一個(gè)C++ 函數的方式可以解決,但似乎有些不爽。
中斷向量,就是函數入口地址,這個(gè)C++的靜態(tài)函數可以對應起來(lái)。

我們可以從map文件中,把中斷向量的一個(gè)個(gè)的C++函數的入口地址符號找到,寫(xiě)入startup文件中。
但C++的symbol不像C語(yǔ)言那么簡(jiǎn)潔了,而且可能隨著(zhù)源代碼的編輯而改變,包括函數的順序,函數的返回值,函數的參數等。
_ZN10CInterrupt15TIM3_IRQHandlerEv 看看這個(gè)符號,是不是很別扭,C語(yǔ)言的Symbol就一個(gè)TIM3_IRQHandler,跟函數名完全相同。

這時(shí)候,我們應該盡量不要在匯編程序中寫(xiě)向量表,而是到C++文件中寫(xiě)向量表了。

沙發(fā)
 樓主| 發(fā)表于 2013-7-8 23:49:23 | 只看該作者
這是修改的的startup.s文件
-----------------------------華麗麗的分界線(xiàn)---------------------------------------------------

Stack_Size      EQU     0x00000800

                AREA    STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem       SPACE   Stack_Size
__initial_sp
__initial_sp_ex

Heap_Size       EQU     0x00000400

                AREA    HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
Heap_Mem        SPACE   Heap_Size
__heap_limit


                EXPORT  __initial_sp_ex
                EXPORT  __Vectors
                EXPORT  __Vectors_End
                EXPORT  __Vectors_Size
__Vectors            EQU        0x08006000
__Vectors_End        EQU        0x08006040
__Vectors_Size        EQU        0x40


                PRESERVE8
                THUMB

                AREA    |.text|, CODE, READONLY

; Reset handler
Reset_Handler    PROC
                 EXPORT  Reset_Handler             [WEAK]
     IMPORT  __main
     IMPORT  SystemInit
                 LDR     R0, =SystemInit
                 BLX     R0
                 LDR     R0, =__main
                 BX      R0
                 ENDP


;*******************************************************************************
; User Stack and Heap initialization
;*******************************************************************************
                 IF      EF:__MICROLIB           
               
                 EXPORT  __initial_sp
                 EXPORT  __heap_base
                 EXPORT  __heap_limit
               
                 ELSE
               
                 IMPORT  __use_two_region_memory
                 EXPORT  __user_initial_stackheap
                 
__user_initial_stackheap

                 LDR     R0, =  Heap_Mem
                 LDR     R1, =(Stack_Mem + Stack_Size)
                 LDR     R2, = (Heap_Mem +  Heap_Size)
                 LDR     R3, = Stack_Mem
                 BX      LR

                 ALIGN

                 ENDIF

                 END
板凳
 樓主| 發(fā)表于 2013-7-8 23:49:59 | 只看該作者
這個(gè)是Interrupt.hpp文件,可根據需要增添,可以把所有的ISR函數都掛上去,也可以只寫(xiě)自己需要的

class CInterrupt
{
public:
        static void USB_LP_CAN1_RX0_IRQHandler();
        static void TIM3_IRQHandler();
};
地板
 樓主| 發(fā)表于 2013-7-8 23:50:31 | 只看該作者
這個(gè)是Interrupt.cpp文件,包括2個(gè)成員函數的實(shí)現
------------------------------------------------------------------------------
void CInterrupt::USB_LP_CAN1_RX0_IRQHandler()
{
        if(!g_bInterruptPause)        AndyUSB.USB_Istr();
}
void CInterrupt::TIM3_IRQHandler()
{
        if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)
        {
                TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
               
                g_nNowTime++;
               
//                if(0==(g_nNowTime%2))
                {
                        AndyHW.HwCallBackKeyScan();
                }
                AndyHW.HwCallBackOverCurrent();
                AndyOS.sysCallBackTimer10ms();   //10mS
        }
}
地下室
 樓主| 發(fā)表于 2013-7-8 23:51:21 | 只看該作者
下面的這個(gè)部分,可以寫(xiě)在Interrupt.cpp文件中,也可單獨寫(xiě)一個(gè)CPP文件。
-----------------------------------------------------------------------------------------------------------------------------


#ifdef __cplusplus
extern "C" {
#endif

void __initial_sp_ex(void);
void Reset_Handler(void);

#ifdef __cplusplus
}
#endif

void Default_Handler(void);

typedef void(* const ISR_t)(void);

ISR_t IsrVector[] __attribute__ ((section("RESET"))) =
{
                __initial_sp_ex,
                Reset_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                0,
                0,
                0,
                0,
                Default_Handler,
                Default_Handler,
                0,
                Default_Handler,
                Default_Handler,


                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                CInterrupt::USB_LP_CAN1_RX0_IRQHandler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                CInterrupt::TIM3_IRQHandler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler,
                Default_Handler
};

void Default_Handler(void)
{
    while (1);
}
6
 樓主| 發(fā)表于 2013-7-8 23:51:59 | 只看該作者
不過(guò)有些遺憾的是,STM32的FwLib沒(méi)有C++版本的,不過(guò)C++和C混合使用,沒(méi)問(wèn)題的
ST提供的h文件中全加入了兼容C和C++的宏處理
#ifdef __cplusplus
extern "C" {
#endif

當轉到C++后,就可以使用虛函數來(lái)代替一部分函數指針,可讀性更好。
MDK中RVDS編譯器,對C++的優(yōu)化很好,沒(méi)用到的成員函數會(huì )被優(yōu)化掉,所以如果做平臺一定記得對相關(guān)的成員函數禁止優(yōu)化,使用--keep選項

OK,享受C++之旅吧
您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

關(guān)于我們  -  服務(wù)條款  -  使用指南  -  站點(diǎn)地圖  -  友情鏈接  -  聯(lián)系我們
電子工程網(wǎng) © 版權所有   京ICP備16069177號 | 京公網(wǎng)安備11010502021702
快速回復 返回頂部 返回列表
午夜高清国产拍精品福利|亚洲色精品88色婷婷七月丁香|91久久精品无码一区|99久久国语露脸精品|动漫卡通亚洲综合专区48页