共用方式為


在內嵌組譯碼中使用和保留暫存器

Microsoft 特定的

一般而言,您不應該假設暫存器會在 __asm 區塊開始時有指定值。 不保證跨不同的 __asm 區塊會保留暫存器值。 如果您結束內嵌程式碼區塊並開始另一個區塊,您就不能依賴第二個區塊中的暫存器從第一個區塊保有其值。 __asm 區塊會繼承一般控制流程所產生的任何暫存器值。

如果使用 __fastcall 呼叫慣例,編譯器會傳遞暫存器中的函式引數,而非堆疊上的函式引數。 這可能會在包含 __asm 區塊的函式中引起問題,因為函式無法判斷哪一個參數在哪一個暫存器中。 如果函式剛好在 EAX 中接收參數並立即將其他參數儲存在 EAX 中,則會失去原始參數。 此外,您必須將 ECX 暫存器保留在任何以 __fastcall 宣告的函式中。

為避免這種暫存器衝突,請勿將 __asm 慣例用於包含 __fastcall 區塊的函式。 如果使用 /Gr 編譯器選項全域指定 __fastcall 慣例,請以 __cdecl 或 __stdcall 宣告每一個包含 __asm 區塊的函式 (__cdecl 屬性會指示編譯器使用該函式的 C 呼叫慣例)。如果您不使用 /Gr 進行編譯,請避免宣告具有 __fastcall 屬性的函式。

當使用 __asm 在 C/C++ 函式中寫入組合語言時,您不需要保留 EAX、EBX、ECX 和 EDX、ESI 或 EDI 暫存器。 例如,在使用內嵌組譯碼撰寫函式的 POWER2.C 範例中,power2 函式不會保留 EAX 暫存器中的值。 不過,因為暫存器配置器不能利用這些暫存器跨 __asm 區塊儲存值,因此使用這些暫存器會影響程式碼品質。 此外,您可以在內嵌組譯碼中使用 EBX、ESI 或 EDI 來強制編譯器在函式序言和結尾中儲存及還原這些暫存器。

您應該保留用於 __asm 區塊範圍的其他暫存器 (例如 DS、SS、SP、BP 和旗標暫存器)。 除非您有需要變更的理由 (例如切換堆疊),否則您應該保留 ESP 和 EBP 暫存器。 請參閱最佳化內嵌組譯碼

部分 SSE 類型需要有八個位元組的堆疊對齊,強制編譯器發出動態堆疊對齊程式碼。 為了在對齊後能夠存取區域變數和函式參數二者,編譯器會保留兩個框架指標。如果編譯器執行框架指標省略 (FPO),它將使用 EBP 和 ESP。如果編譯器未執行 FPO,則會使用 EBX 和 EBP。 為確保程式碼能夠正確執行,如果函式為了能夠修改框架指標而需要動態堆疊對齊,請勿修改 asm 程式碼中的 EBX。 您可以將八位元組的對齊類型移到函式以外,或者避免使用 EBX。

注意事項注意事項

如果內嵌組譯碼使用 STD 或 CLD 指令變更方向旗標,您必須將旗標還原為其原始值。

END Microsoft 特定的

請參閱

參考

內嵌組譯工具