使用英特尔语法中的内联汇编编译内核模块
Compile Kernel Modules with inline assembly in 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;
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);
生成文件
obj-m += samplemodule.o
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
所以,每当我想内联汇编时,我都会写两个.intel_syntax noprefix\n...\n.att_syntax noprefix\n
。还有其他方法可以做到这一点吗?使用 gcc 编译时,我过去只是将 -masm=intel
参数传递给 gcc,这使我可以自由使用英特尔语法。在这种情况下可能有类似的东西吗?
根据 https://www.kernel.org/doc/Documentation/kbuild/makefiles.txt,有几个变量可以将特定选项传递给 gcc 工具链:您应该尝试的是 KBUILD_CFLAGS_MODULE
。这样,你仍然可以使用选项-masm=intel
。你的 all
目标应该是
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) KBUILD_CFLAGS_MODULE='-masm=intel' modules
如果您包含来自内核 headers 的汇编代码(我认为这种情况并不少见),并且使用 AT&T 语法,则会出现问题:在这种情况下,最终代码将混合两种语法,并且编译会失败。在最后一种情况下,我认为唯一的方法是在每个 asm
指令中手动指定 Intel 语法,就像您目前所做的那样。
解决方法
由于更改编译参数或脚本不是一项干净的工作并且容易出错,我建议一个简单的解决方法:您的主要问题是避免每次都键入 ".intel_syntax noprefix\n"
和 ".att_syntax noprefix\n"
, 正确的?您可以将这些指令放在宏中,也许将宏放在 header 中,然后在您需要的任何地方包含 header。例如,您可以定义一个宏,如
#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 语法。以下最小示例显示了执行此操作的一种方法:
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;
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);
生成文件
obj-m += samplemodule.o
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
所以,每当我想内联汇编时,我都会写两个.intel_syntax noprefix\n...\n.att_syntax noprefix\n
。还有其他方法可以做到这一点吗?使用 gcc 编译时,我过去只是将 -masm=intel
参数传递给 gcc,这使我可以自由使用英特尔语法。在这种情况下可能有类似的东西吗?
根据 https://www.kernel.org/doc/Documentation/kbuild/makefiles.txt,有几个变量可以将特定选项传递给 gcc 工具链:您应该尝试的是 KBUILD_CFLAGS_MODULE
。这样,你仍然可以使用选项-masm=intel
。你的 all
目标应该是
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) KBUILD_CFLAGS_MODULE='-masm=intel' modules
如果您包含来自内核 headers 的汇编代码(我认为这种情况并不少见),并且使用 AT&T 语法,则会出现问题:在这种情况下,最终代码将混合两种语法,并且编译会失败。在最后一种情况下,我认为唯一的方法是在每个 asm
指令中手动指定 Intel 语法,就像您目前所做的那样。
解决方法
由于更改编译参数或脚本不是一项干净的工作并且容易出错,我建议一个简单的解决方法:您的主要问题是避免每次都键入 ".intel_syntax noprefix\n"
和 ".att_syntax noprefix\n"
, 正确的?您可以将这些指令放在宏中,也许将宏放在 header 中,然后在您需要的任何地方包含 header。例如,您可以定义一个宏,如
#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;
}