我正在寻找用于进行系统调用的 linux 内核中的 __kernel_vsyscall 方法。我想观察它的代码以了解更多关于它的信息,但我的 grep 搜索似乎无法找出它,而且我在互联网上找不到它的位置。谁能指出我的确切位置?它可以被操纵吗?
谢谢你的帮助!
我正在寻找用于进行系统调用的 linux 内核中的 __kernel_vsyscall 方法。我想观察它的代码以了解更多关于它的信息,但我的 grep 搜索似乎无法找出它,而且我在互联网上找不到它的位置。谁能指出我的确切位置?它可以被操纵吗?
谢谢你的帮助!
假设您的当前目录位于 linux 内核源代码的头部,这里是定义 __kernel_vsyscall 符号的文件的位置。(下面显示的所有内容仅适用于 x86,在许多其他硬件架构中尚不存在)。
./arch/x86/vdso/vdso32/syscall.S:
__kernel_vsyscall:
./arch/x86/vdso/vdso32/sysenter.S:
__kernel_vsyscall:
./arch/x86/vdso/vdso32/int80.S:
__kernel_vsyscall:
如您所见,它本质上是在三个文件中声明和实现的:int80.S、sysenter.S 和 syscall.S。
以系统调用.S:
__kernel_vsyscall:
.LSTART_vsyscall:
push %ebp
.Lpush_ebp:
movl %ecx, %ebp
syscall
movl $__USER32_DS, %ecx
movl %ecx, %ss
movl %ebp, %ecx
popl %ebp
如果您阅读上述文件和“arch/x86/vdso/vdso32/sigreturn.S”的组合,则上面的“syscall”实际上解析为“int 0x80”。
而对于sysenter.S,它使用intel汇编指令“sysenter”来实现系统调用转换。
而对于 int80.S,它使用“int 0x80”进行系统调用转换。
如果你问哪个是用于系统调用实现的方法,请查看 arch/x86/vdso/vdso32-setup.c:
int __init sysenter_setup(void)
{
void *syscall_page = (void *)get_zeroed_page(GFP_ATOMIC);
const void *vsyscall;
size_t vsyscall_len;
vdso32_pages[0] = virt_to_page(syscall_page);
#ifdef CONFIG_X86_32
gate_vma_init();
#endif
if (vdso32_syscall()) {
vsyscall = &vdso32_syscall_start;
vsyscall_len = &vdso32_syscall_end - &vdso32_syscall_start;
} else if (vdso32_sysenter()){
vsyscall = &vdso32_sysenter_start;
vsyscall_len = &vdso32_sysenter_end - &vdso32_sysenter_start;
} else {
vsyscall = &vdso32_int80_start;
vsyscall_len = &vdso32_int80_end - &vdso32_int80_start;
}
memcpy(syscall_page, vsyscall, vsyscall_len);
relocate_vdso(syscall_page);
return 0;
}
如您所见,现代操作系统更喜欢 sysenter 方法,因为它比 int80 方法快。(象征性地,“vds32_syscall_start”将回退到 int80)。
有趣的是,在您的互联网搜索中,您没想到要搜索 stackoverflow.com:__kernel_vsyscall是什么?
要回答您更具体的问题,看起来符号本身是在(对于 x86)arch/x86/vdso 中定义的。它在汇编中,而不是 C.