Как добавить непосредственный байт в длинный регистр с синтаксисом AT & T на x86? - PullRequest
3 голосов
/ 07 декабря 2011

Из того, что я понимаю, читая руководства Intel, можно написать инструкцию типа add $0x7f, %ebx, и она должна быть закодирована как 83 /0 ib, всего три байта.

Однако, когда я делаю это (использую ли я add, addb или addl), он всегда "переводит" непосредственное значение в 32-битное значение и кодируется как 81 /0 id и занимает шесть байт. Та же проблема существует с adc, sub и т. Д. Обратите внимание, что я использую синтаксис AT & T с GNU as.

Я искал решение в течение дня и не нашел его. Кто-нибудь может посоветовать, пожалуйста?

1 Ответ

2 голосов
/ 07 декабря 2011

Удивительно, но у меня нет такой проблемы.

Я взял этот код сборки, созданный gcc (DJGPP):

        .file   "im.c"
.globl _x
        .section        .bss
        .p2align 2
_x:
        .space 4
        .section .text
        .p2align 4,,15
.globl _main
_main:
        pushl   %ebp
        movl    %esp, %ebp
        pushl   %eax
        pushl   %eax
        movl    _x, %eax
        andl    $-16, %esp
        addl    $127, %eax
        movl    %eax, _x
        movl    %ebp, %esp
        xorl    %eax, %eax
        popl    %ebp
        ret
        .ident  "GCC: (GNU) 3.3.4"

И скомпилировал это с as, и это то, что я вижу в a.out:

55                push ebp
89E5              mov ebp,esp
50                push eax
50                push eax
A100000000        mov eax,[0x0]
83E4F0            and esp,byte -0x10
83C07F            add eax,byte +0x7f
A300000000        mov [0x0],eax
89EC              mov esp,ebp
31C0              xor eax,eax
5D                pop ebp
C3                ret

И программа на C была:

volatile int x = 0;

int main(void)
{
  x += 0x7F;
  return 0;
}

Вы уверены, что ваш непосредственный операнд может быть представлен как 8-разрядное целое число со знаком? Если он находится за пределами диапазона от -128 до +127, ассемблеру придется использовать более длинную кодировку.

...