如何在C51 Keil中找到并调用具有绝对地址的函数? 背景ROM具有一些实用功能。我想调用这些函数以优化闪存的代码空间。
答案 0 :(得分:1)
一个简单的例子:
#pragma SRC
// type of external function
typedef void (code * tExternalFunctionPtr)(void);
// Address of external function in ROM
#define FUNC_ADDR (0x1234)
void callExample(void)
{
// Simple example with call via function pointer in RAM
tExternalFunctionPtr funcPtr;
funcPtr = FUNC_ADDR;
(*funcPtr)();
// Direct call without function pointer in RAM
(* (tExternalFunctionPtr)FUNC_ADDR)();
}
编译为以下汇编代码:
; TEST.SRC generated from: TEST.C
NAME TEST
?PR?callExample?TEST SEGMENT CODE
EXTRN CODE (?C_ICALL)
PUBLIC callExample
; #pragma SRC
;
; // type of external function
; typedef void (code * tExternalFunctionPtr)(void);
;
; // Address of external function in ROM
; #define FUNC_ADDR (0x1234)
;
; void callExample(void)
RSEG ?PR?callExample?TEST
USING 0
callExample:
; SOURCE LINE # 9
; {
; SOURCE LINE # 10
; // Simple example with call via function pointer in RAM
; tExternalFunctionPtr funcPtr;
; funcPtr = FUNC_ADDR;
; SOURCE LINE # 13
;---- Variable 'funcPtr?01' assigned to Register 'R6/R7' ----
MOV R7,#034H
MOV R6,#012H
; (*funcPtr)();
; SOURCE LINE # 14
MOV R2,AR6
MOV R1,AR7
LCALL ?C_ICALL
;
; // Direct call without function pointer in RAM
; (* (tExternalFunctionPtr)FUNC_ADDR)();
; SOURCE LINE # 17
LCALL 01234H
; }
; SOURCE LINE # 18
RET
; END OF callExample
END
这可能有助于说明它原则上如何运作。
但那还不是全部: 您必须知道要调用的ROM中的函数的地址。您还必须知道是否以及如何预期参数以及返回值。 了解ROM功能可能发生变化的寄存器以及确保C51编译器不希望这些寄存器保持不变也很重要。