C語言中將絕對地址轉換為函數指針以及跳轉到內存指定位置處執行的技巧 (轉)


一、1、方法一
要對絕對地址0x100000賦值,我們可以用
      (unsigned int  * ) 0x100000 = 1234;
      那么要是想讓程序跳轉到絕對地址是0x100000去執行,應該怎么做?
     *((void (*)( ))0x100000 ) ( ); 

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

 

上述有誤,應該是( *((void (*)( ))0x100000 ) )() ,令pFunction 為一函數指針(已被賦值一函數名,即函數首地址),則pFunction()或(*pFunction)()均表示調用該函數,而*pFunction()有誤

-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
    首先要將0x100000強制轉換成函數指針,即:
     (void (*)())0x100000
     然后再調用它:
   *((void (*)())0x100000)();               //(*((void (*)())0x100000))()
   用typedef可以看得更直觀些:
    typedef void(*)() voidFuncPtr;     
   *((voidFuncPtr)0x100000)();        //( *((voidFuncPtr)0x100000))()

/* typedef void(*pFunction)(void);

  * pFunction = (void(*)(void))0x100000;

*/pFunction(); //調用函數


 
又如
如果用 C 語言,可以像下列示例代碼這樣來調用內核:
void (*theKernel)(int zero, int arch, u32 params_addr)
= (void (*)(int, int, u32))KERNEL_RAM_BASE; 
…… 
theKernel(0, ARCH_NUMBER, (u32) kernel_params_start); 
KERNEL_RAM_BASE 是內核在系統內存中的第一條指令的地址。
2、方法二
C語言使用函數指針跳轉到程序固定地址(0x8000)執行程序的方法

使用函數指針,把一個純數據強制轉換為函數指針類型。

int main(void)

{
void (* my_function)(void);
//int *my_address = 0x8000;
my_function =(void (*)())(0x8000);
my_function();
}

其實更簡單,不適用中間變量,直接一步到位:

(*(void(*)())0x8000)();
--------------------- -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

/******************************以下為從《https://blog.csdn.net/ma57457/article/details/64904313》摘錄及修改**************************************************/

#define jump(TargetAddr )   (*((void(*)())(TargetAddr)))(void)    //使用((void(*)(void))(TargetAddr))()  也是否也可以??

第一個(( void( * )( void)) ,意思為將一常量TargetAddr強制類型轉換為一個無形參,無返回值的函數指針,(*(TargetAddr))()表示跳轉到TargetAddr地址執行,但是函數指針變量不能為常數所以要加((void( * )(  )) 進行強制類型轉換。最后一個()為執行的意思。

整一條指定的目的是為了跳轉到一個絕對地址執行函數。

1、在單片機中可以實現軟件復位,比如跳轉到0地址。

2、如果程序是由多個程序合並的,跳轉到某一個確定的用戶程序地址執行,如果flash空間足夠大的話,可以實現當多份不相同的代碼合並為一份后,在軟件上做邏輯跳轉,好處是新程序不必為舊程序做大量的兼容工作,通常舊程序含有大量的前人的編程習慣,比如宏定義。
————————————————-------------------------------------------------------------------------------------------------------------------------------------------------------


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM