[問題] c51 謎樣的 long call

看板ASM (組合語言)作者 (好人超)時間13年前 (2012/04/18 20:57), 編輯推噓5(5010)
留言15則, 3人參與, 最新討論串1/1
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)

04/18 22:07, , 1F
http://www.8052.com/51lcall 呼叫 0 位址處的副程式?
04/18 22:07, 1F

04/18 22:55, , 2F
呀,我悟了,E 是外部符號這跟 VC 是一樣手法,無法得知的
04/18 22:55, 2F

04/18 22:56, , 3F
外部函數寫成 0 等 linker 改。而 R 是還要重定位址
04/18 22:56, 3F

04/18 22:57, , 4F
應該可以用反組譯、simulator 之類去追蹤進去吧
04/18 22:57, 4F

04/19 00:23, , 5F
這是keil的內部library吧, 算array element的位置用的
04/19 00:23, 5F
我的猜測也是這樣,但很希望能夠消除這個函式呼叫 因為我的環境不允許這個東西....

04/19 01:14, , 6F
DPTRR = DPTRR + 結構大小 * i; temp = DPTRR->SMBAddr;
04/19 01:14, 6F

04/19 01:15, , 7F
能否說明一下為何不允許? 也許把struct大小改為2的乘方
04/19 01:15, 7F

04/19 01:15, , 8F
可避掉該lcall
04/19 01:15, 8F

04/19 01:32, , 9F
看來沒辦法避掉...
04/19 01:32, 9F
其實目前有湊出一個很繞路的方式 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)

04/19 12:49, , 10F
先試試 alignment allocate, 參考 http://ppt.cc/aoSW
04/19 12:49, 10F

04/19 12:49, , 11F
section 9.
04/19 12:49, 11F

04/19 12:51, , 12F
sorry, 補充沒看清楚..
04/19 12:51, 12F

04/19 12:59, , 13F
http://ppt.cc/AWG4 你要的是 SMALL 嗎?
04/19 12:59, 13F

04/19 16:52, , 14F
在loop中用pointer也許可避掉
04/19 16:52, 14F

04/19 16:53, , 15F
如 p=DPTRR; for(i=0;i<n;++i) {t=p->SMBAddr; p++;}
04/19 16:53, 15F
文章代碼(AID): #1FZheU-O (ASM)
文章代碼(AID): #1FZheU-O (ASM)