offset指令
offset指令是什麼,怎麼使用呢?不知道的小夥伴來看看小編今天的分享吧!
1、offset指令簡介:
偽指令offset是組合語言中編譯器處理的符號,它的功能是取得標號的偏移地址。
assume cs:code
code segment
start: mov ax,offset start ;相當於mov ax,0
;start所標記的是程式碼段的第一條指令,偏移地址為0
s: mov ax,offset s ;相當於mov ax,3
;s所標記的指令是程式碼段中的第二條指令,第一條指令的長度為3byte,則s的偏移地址為3
code ends
end start
2、問題:
有如下程式段,填寫兩條指令,使改程式在執行中將s處的第一條指令複製到s0處:
程式碼如下:
;問題:有如下程式段,填寫兩條指令,使改程式在執行中將s處的第一條指令複製到s0處:
assume cs:code
code segment
s: mov ax,bx ;mov ax,bx機器碼佔兩個位元組
mov si,offset s
mov di,offset s0
mov dx,cs:[si] ;資料從哪裡來
mov cs:[di],dx ;資料到哪裡去
s0: nop ;cpu遇到nop指令什麼都不做,nop指令佔一個位元組
nop
code ends
end s
拓展資料:
addr和offset指令的區別:
一、相同點
1、addr 和 offset 操作符都是獲得運算元的偏移地址;
2、addr 和 offset 的處理都是先檢查處理的是全域性還是區域性變數,若是全域性變數則把其地址放到目標檔案中。
二、不同點
1、 addr 偽操作符,只能用在 invoke 偽指令語句中; (本來就是為了在invoke指令中,使用區域性變數的地址)
在其他例如mov指令中,可以先使用lea指令,來取得區域性變數的地址
2、 offset 偽操作符可以用在任何可能涉及偏移地址的指令(當然包括 invoke 偽指令)並想獲取運算元偏移地址的場合中;
3、addr 不能處理向前引用(即 addr 引用的運算元必須在使用 addr 前就得定義或宣告),而offset 則能(不管引用的運算元是
其前或其後定義或宣告);
所謂向前引用是指:標號的定義是在invoke 語句之後,比如在如下的例子:
invoke MessageBox,NULL, addr MsgBoxText,addr MsgBoxCaption,MB_OK //引用MsgBoxText、MsgBoxCaption 在先
......
MsgBoxCaption db "Iczelion Tutorial No.2",0 //定義或宣告 MsgBoxCaption 在 addr 後
MsgBoxText db "Win32 Assembly is Great!",0 //定義或宣告 MsgBoxText 在 addr 後
如果您是用 addr 而不是 offset 的話,那 MASM 就會報
4、addr 是執行階段在堆疊中分配記憶體空間,offset 是編譯階段由編譯器解釋。因此,addr 可以處理區域性變數而 offset 則不能。
5、addr 如果檢查到待處理的變數是區域性變數,就在執行 invoke 語句前產生如下指令序列:
lea eax,operand
push eax
因為 lea 指令能夠在執行時決定標號的有效地址,所以有了上述指令序列,就可以保證 invoke 的正確執行了。
總結:為了避免出現錯誤,建議除在區域性變數中引用 addr 操作符外,其它場合使用 offset。
說明:某些文章中對 addr 和 offset 所引用的物件僅用了“變數或標號”,我是用“運算元”來闡述的,本人的觀點是:
變數或標號感覺上包含的概念過窄,比如結構、函式等等,因此,覺得使用運算元好像感覺準確些。
以上就是小編今天的分享了,希望可以幫助到大家。