【问题标题】:inline assembly operand constraints内联汇编操作数约束
【发布时间】:2015-02-18 18:07:41
【问题描述】:

关于我想要理解的一段代码,我有几个问题。我读过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"
    );
}

我对以下一些约束/选项的用法有很大的猜测。但我觉得最好从其他人那里确认(或获取一些详细手册的链接)。

  • “%c6(%5)”和“%c7(%5)”中的%c
  • “%q0”和“%q2”中的%q,
  • “%b3”中的%b
  • %w in "%w1"
  • “(%4)”中的括号
  • “i”表示两个输入参数。
  • “1”和“2”表示两个输入参数。

感谢您的帮助。

【问题讨论】:

  • 远离诡计。坚持简单明了。无论如何,事情都会变得很棘手。

标签: c++ c gcc assembly


【解决方案1】:

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

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

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

i 是立即整数操作数(具有常量值)。

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

【讨论】:

  • 感谢您的评论。关于 %c,理解 "%c6(%5)" == %5 和 %6 的值的总和并将其用作内存中的常量地址是否正确,我们只能从中读取但不能写入?
  • 不,地址是常量而不是值。这是foo* const 而不是const foo*
猜你喜欢
  • 1970-01-01
  • 2011-04-23
  • 2015-12-23
  • 1970-01-01
  • 1970-01-01
  • 2012-11-12
  • 1970-01-01
  • 2014-10-24
  • 1970-01-01
相关资源
最近更新 更多