C 使用英特尔语法的内联汇编编译内核模块
我不想编写一个使用大量内联汇编的内核模块。因为我已经习惯了英特尔语法,所以我想完全避免AT&T语法。下面的示例显示了一种方法: samplemodule.cC 使用英特尔语法的内联汇编编译内核模块,c,kernel-module,inline-assembly,intel-syntax,C,Kernel Module,Inline Assembly,Intel Syntax,我不想编写一个使用大量内联汇编的内核模块。因为我已经习惯了英特尔语法,所以我想完全避免AT&T语法。下面的示例显示了一种方法: samplemodule.c #include <linux/init.h> #include <linux/kernel.h> #include <linux/module.h> MODULE_LICENSE("GPL"); unsigned long foo(void) { unsigned long ret = 0;
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/module.h>
MODULE_LICENSE("GPL");
unsigned long foo(void) {
unsigned long ret = 0;
asm (
".intel_syntax noprefix\n"
"mov rbx, 1337\n"
"mov %0, rbx\n"
".att_syntax noprefix\n"
:"=r"(ret)
:
:"rbx"
);
return ret;
}
static int init_routine(void) {
printk(KERN_INFO "Sample Module init\n");
printk(KERN_INFO "Test: %lu\n", foo());
return 0;
}
static void exit_routine(void) {
printk(KERN_INFO "Sample Module exit\n");
}
module_init(init_routine);
module_exit(exit_routine);
因此,每当我想内联汇编时,我有两个write。intel\u syntax noprefix\n…。\n.att\u syntax noprefix\n
。有没有其他方法可以做到这一点?使用gcc编译时,我通常只将-masm=intel
参数传递给gcc,这样我就可以自由使用英特尔语法。在这种情况下是否可能出现类似情况?根据,有几个变量可以将特定选项传递给gcc工具链:您应该尝试的变量是KBUILD\u CFLAGS\u MODULE
。这样,您仍然可以使用选项-masm=intel
。你的目标应该是
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) KBUILD_CFLAGS_MODULE='-masm=intel' modules
如果包含内核头中的汇编代码(我认为这并不少见),并且使用AT&T语法,就会出现问题:在这种情况下,最终代码将混合使用这两种语法,编译将失败。在最后一种情况下,我认为唯一的方法是在每个asm
指令中手动指定Intel语法,就像您到目前为止所做的那样
变通办法
由于更改编译参数或脚本不是一项干净的工作,而且容易出错,我建议您采取一种简单的解决方法:您的主要问题是避免每次键入“.intel\u syntax noprefix\n”
和“.att\u syntax noprefix\n”
,对吗?您可以将这些指令放在宏中,也可以将宏放在头中,并在需要的任何地方包含该头。例如,可以定义一个宏,如
#define INTEL_ASM(a,b,c,d) \
asm ( \
".intel_syntax noprefix\n" \
a \
".att_syntax noprefix\n" \
:b \
:c \
:d \
)
这样你的代码就变成了
unsigned long foo(void) {
unsigned long ret = 0;
INTEL_ASM (
"mov rbx, 1337\n"
"mov %0, rbx\n"
,"=r"(ret)
,
,"rbx"
);
return ret;
}
或者你可以,你知道的,只是学习AT&T的语法。。。从长远来看,这样做可能会更好…Thx,这几乎奏效了。完全按照您建议的方式使用时,无法再加载模块(insmod
表示模块格式无效
,很可能是因为-DMODULE
被覆盖)。我通过在Makefile的开头添加KBUILD\u CFLAGS\u MODULE+='-masm=intel'
来修复它。
unsigned long foo(void) {
unsigned long ret = 0;
INTEL_ASM (
"mov rbx, 1337\n"
"mov %0, rbx\n"
,"=r"(ret)
,
,"rbx"
);
return ret;
}