1

スタックから stdout に文字列 (例: "Hello") を書き込むにはどうすればよいですか? つまり、データセグメントなしです。

void main() {
    __asm__(
                "movl  $0x4, %eax   \n\t"
                "movl  $0x1, %ebx   \n\t"
            //   put "Hello" on the stack and load its address into %ecx
                "movl  $0x5, %edx   \n\t"
                "int   $0x80        \n\t"

                "movl  $0x1, %eax   \n\t"
                "movl  $0x0, %ebx   \n\t"
                "int   $0x80        \n\t"
             );
}

前もって感謝します

4

3 に答える 3

3

答え 1:

int main()
{
    const char* string = "hello";  // string is not in a data segment, it's in the text segment
    fputs(string, stdout);
    return 0;
}

答え 2:

int main()
{
    char[6] string = "hello";  // Space for string allocated on stack
    fputs(string, stdout);
    return 0;
}

gcc を使用すると、2 番目の回答で次のように表示されます。

main:      
    leal    4(%esp), %ecx
    andl    $-16, %esp
    pushl   -4(%ecx)
    pushl   %ebp
    movl    %esp, %ebp
    pushl   %ecx
    subl    $36, %esp
    movl    $1819043176, -10(%ebp) ;<< hell
    movw    $111, -6(%ebp)         ;<< o\0
    movl    stdout, %eax
    movl    %eax, 4(%esp)
    leal    -10(%ebp), %eax
    movl    %eax, (%esp)
    call    fputs
    movl    $0, %eax
    addl    $36, %esp
    popl    %ecx
    popl    %ebp
    leal    -4(%ecx), %esp

明らかにスタックのみを使用します。

于 2010-07-16T11:00:08.823 に答える
1
int main() {
        char *hello = "Hello world!\n";
        __asm__("\
                movl $4, %%eax\n\
                movl $0, %%ebx\n\
                push %0\n\
                pop %%ecx\n\
                movl $13,%%edx\n\
                int $0x80" : :"g"(hello));
        return 0;
}

スタック部分がわかりません。なぜ「movl %0,%%ecx」を使わないのですか?

于 2010-07-16T10:05:52.173 に答える
0

スタックから stdout に文字列 (例: "Hello") を書き込むにはどうすればよいですか? つまり、データセグメントなしです。

線に沿った何か:

  1. スタックにバッファを割り当てます。たとえば、8 バイトです。( GCC alloca()がそれを行う方法を確認してください。)
  2. バッファに文字列を作成します。「Hello」のバイト単位は 48 65 6c 6c 6f 00 00 00 (整列、ゼロでパディング) です。文字列を表すには、2 つの 32 ビット定数 (0x6c6c6548 と 0x0000006f) で十分です。
  3. バッファのアドレスを write syscall に渡します。
  4. スタック ポインターを以前の場所に復元します (必要な場合)。

私は GCC インライン asm と x86 ABI についてあまり最新ではないので、これ以上正確なことはできません。

于 2010-07-16T10:12:52.037 に答える