C 使用英特尔语法的内联汇编编译内核模块

C 使用英特尔语法的内联汇编编译内核模块,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;

我不想编写一个使用大量内联汇编的内核模块。因为我已经习惯了英特尔语法,所以我想完全避免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);
因此,每当我想内联汇编时,我有两个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;
}