内联汇编操作数约束

inline assembly operand constraints

我有几个关于我试图理解的代码的问题。我已阅读 the manual here。但它似乎并没有解释或回答这里使用的所有技巧。

代码如下:

#define SAMPLE_OFFSET1 0
#define SAMPLE_OFFSET2 100
#define SAMPLE_OFFSET1 1000
#define STRINGIFY(s) #s

struct data {
    int m1;
    int m2;
    const uint *m3;
    const uint *m4;
};

#define assembly_function(param1, ... param15) \
... body \
... body \
... body \
... body

void testFunction (data* c, uint *const sample) {
int bat, tmp;
    __asm__ volatile(
        assembly_function("%0", "%q0", "(%4)", "%1", "%w1",
                             "%2", "%q2", "%3", "%b3",
                             "%c6(%5)", "%c7(%5)",
                             STRINGIFY(SAMPLE_OFFSET1),
                             STRINGIFY(SAMPLE_OFFSET2),
                             STRINGIFY(SAMPLE_OFFSET3),
                             "%8")
        : "=&r"(bat), "=&r"(c->m1), "=&r"(c->m2), "=&q"(tmp)
        : "r"(sample), "r"(c),
          "i"(offsetof(data, m3)),
          "i"(offsetof(data, m4)),
          "1"(c->m1), "2"(c->m2)
        : "%rcx", "memory"
    );
}

我对以下一些constraints/options的用法进行了大胆的猜测。但是我觉得最好还是向其他人确认一下(或者link一些详细的手册)。

感谢您的帮助。

%c 强制输出为常量地址,这里它用于摆脱通常在 at&t 语法模式中为立即数发出的 $i 约束暗示) .

%b%w%l%q 是大小覆盖修饰符,它们强制使用适当大小的寄存器(即字节、字、长或四) 作为操作数。

括号只是您日常 at&t 风格的有效地址语法的一部分。

i 是一个立即整数操作数(一个具有常数值的操作数)。

"1""2" 是匹配约束,即将它们放在与指示的其他操作数相同的位置。