1

我不想编写一个使用大量内联汇编的内核模块。因为我习惯了 Intel 语法,所以我想完全避免 AT&T 语法。以下最小示例显示了执行此操作的一种方法:

示例模块.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

所以,每当我想内联汇编时,我都有两个 write .intel_syntax noprefix\n...\n.att_syntax noprefix\n。还有其他方法可以做到这一点吗?使用 gcc 编译时,我过去只是将-masm=intel参数传递给 gcc,这使我可以自由使用 Intel 语法。在这种情况下是否有类似的可能?

4

1 回答 1

3

根据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

如果您包含来自内核头文件的汇编代码(我认为这并不少见),它使用 AT&T 语法,就会出现问题:在这种情况下,最终代码将混合两种语法,编译将失败。在最后一种情况下,我认为唯一的方法是在每个asm指令中手动指定 Intel 语法,就像您目前所做的那样。

解决方法

由于更改编译参数或脚本不是一项干净的工作并且容易出错,我建议一个简单的解决方法:您的主要问题是避免每次都输入".intel_syntax noprefix\n"和,对吗?".att_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;
}
于 2015-01-09T22:09:02.320 回答