作者james732 (好人超)
看板ASM
标题[问题] c51 谜样的 long call
时间Wed Apr 18 20:57:00 2012
struct ExtRAM
{
unsigned short SMBAddr;
unsigned char Parameter;
};
以上是我的 struct 定义,内容很简单
sfr DPHH = 0x83; // Data Pointer High Register(DPHR)
sfr DPLL = 0x82; // Data Pointer Low Register(DPLR)
void function()
{
unsigned char i=0;
struct ExtRAM code *DPTRR = (struct ExtRAM code*)((DPHH<<8)+DPLL);
volatile unsigned char temp = DPTRR[i].SMBAddr;
// Line 74
temp = DPTRR[0].SMBAddr;
// Line 75
}
以上是我的 function
变数 DPTRR 是从 DPTR 暂存器里取出来的
但我的问题是,编译出来的 LST 档里
第74行出现了不知哪里来的 lcall
而作为对照组,唯一差别只有把 [i] 改成 [0] 的第75行
却没有这个 lcall
;---- Variable 'DPTRR' assigned to Register 'R6/R7' ----
; SOURCE LINE # 74
0011 8F82 MOV DPL,R7
0013 F583 MOV DPH,A
0015 75F003 MOV B,#03H
0018 ED MOV A,R5
0019 120000 E LCALL ?C?OFFXADD ←这是什麽?
001C 7401 MOV A,#01H
001E 93 MOVC A,@A+DPTR
001F F500 R MOV temp,A
; SOURCE LINE # 75
0021 8F82 MOV DPL,R7
0023 8E83 MOV DPH,R6
0025 7401 MOV A,#01H
0027 93 MOVC A,@A+DPTR
0028 F500 R MOV temp,A
想请问一下,为什麽会出现这个 Lcall,它的用途是什麽?
又要怎麽去避免它呢?(我不希望程式里出现这个)
--
※ 发信站: 批踢踢实业坊(ptt.cc)
◆ From: 101.12.28.191
※ 编辑: james732 来自: 101.12.36.30 (04/18 21:47)
2F:推 purpose:呀,我悟了,E 是外部符号这跟 VC 是一样手法,无法得知的 04/18 22:55
3F:→ purpose:外部函数写成 0 等 linker 改。而 R 是还要重定位址 04/18 22:56
4F:→ purpose:应该可以用反组译、simulator 之类去追踪进去吧 04/18 22:57
5F:→ ksmrt0123:这是keil的内部library吧, 算array element的位置用的 04/19 00:23
我的猜测也是这样,但很希望能够消除这个函式呼叫
因为我的环境不允许这个东西....
6F:推 purpose:DPTRR = DPTRR + 结构大小 * i; temp = DPTRR->SMBAddr; 04/19 01:14
7F:→ ksmrt0123:能否说明一下为何不允许? 也许把struct大小改为2的乘方 04/19 01:15
8F:→ ksmrt0123:可避掉该lcall 04/19 01:15
9F:→ ksmrt0123:看来没办法避掉... 04/19 01:32
其实目前有凑出一个很绕路的方式
void function()
{
unsigned char i=0;
unsigned char *ptr = (unsigned char *)((DPHH<<8)+DPLL);
unsigned int offset = i * sizeof(struct ExtRAM);
struct ExtRAM code *DPTRR = (struct ExtRAM code*)(ptr + i);
unsigned char temp = DPTRR->SMBAddr;
}
(以上程式是凭印象打的,没有验证,可能有错误)
自己把 offset 算出来,就不会冒出那个 lcall 了
但我会想知道,有没有什麽方式(譬如修改c51编译参数)
在不修改程式码的前提下,避开这个问题
毕竟这样写起来,程式码变得又臭又长...
至於为什麽不允许那个 lcall,则是工作上的需求
※ 编辑: james732 来自: 114.32.92.122 (04/19 02:12)
11F:→ EdisonX:section 9. 04/19 12:49
12F:→ EdisonX:sorry, 补充没看清楚.. 04/19 12:51
14F:推 ksmrt0123:在loop中用pointer也许可避掉 04/19 16:52
15F:→ ksmrt0123:如 p=DPTRR; for(i=0;i<n;++i) {t=p->SMBAddr; p++;} 04/19 16:53