我是C的装配新手,我不知道如何解决这个错误.我正在创建一个意味着写一个文件的函数.我有的是:
ssize_t mywrite(int fd, const void *buf, size_t count) {
// return write(fd, buf, count);
ssize_t var;
__asm__("movl $4,%%eax\n\t" // Write
"movl %1,%%ebx\n\t"
"movl %2,%%ecx\n\t"
"movl %3,%%edx\n\t"
"int $0x80\n\t" // System call
"movl %%eax,%0"
:"=r"(var)
:"r"(fd),"r"(buf),"r"(count)
:"%eax","%ebx","%ecx","%edx"
);
return var;
}
Run Code Online (Sandbox Code Playgroud)
我的asm应该和write一样(fd,buf,count); 当我编译它时,我得到''asm'操作数有不可能的限制".但是,如果不命名变量并直接从堆栈中获取值,则不会出现错误.这是代码
__asm__("movl $4,%%eax\n\t"
"movl 8(%%ebp),%%ebx\n\t"
"movl 12(%%ebp),%%ecx\n\t"
"movl 16(%%ebp),%%edx\n\t"
"int $0x80\n\t"
"movl %%eax,%0"
:"=r"(var)
:
:"%eax","%ebx","%ecx","%edx"
);
Run Code Online (Sandbox Code Playgroud)
我可以使用第二个代码,ofc,但我需要用优化2编译它.然后%ebp将不会指向我需要的地方.我尝试使用"a","b","c"和"d"代替"r",但没有成功.有人可以帮忙吗?感谢:D
问题是约束r意味着注册,但你的CPU根本就没有那么多寄存器!
您可以使用内存约束m:
:"m"(fd),"m"(buf),"m"(count)
Run Code Online (Sandbox Code Playgroud)
这将产生如下指令:
movl 8(%ebp),%ebx
Run Code Online (Sandbox Code Playgroud)
但我建议在其所有荣耀中使用x86约束:
ssize_t mywrite(int fd, const void *buf, size_t count) {
ssize_t var;
__asm__(
"int $0x80"
:"=a"(var)
:"0"(4), "b"(fd),"c"(buf),"d"(count)
);
return var;
}
Run Code Online (Sandbox Code Playgroud)
那,-Ofast给出:
push %ebx
mov $0x4,%eax
mov 0x10(%esp),%edx
mov 0xc(%esp),%ecx
mov 0x8(%esp),%ebx
int $0x80
pop %ebx
ret
Run Code Online (Sandbox Code Playgroud)
并与-Os:
push %ebp
mov $0x4,%eax
mov %esp,%ebp
push %ebx
mov 0x10(%ebp),%edx
mov 0x8(%ebp),%ebx
mov 0xc(%ebp),%ecx
int $0x80
pop %ebx
pop %ebp
ret
Run Code Online (Sandbox Code Playgroud)
请注意,由于使用约束而不是名称寄存器,编译器能够进一步优化代码.