导入二进制文件gnu-assembler时检查标签之间的距离
Check distance between labels when importing binary file gnu-assembler
我有以下宏来嵌入文件名中的二进制数据:
#define INCBIN(identifier, filename) \
asm(".pushsection .rodata\n" \
"\t.local " #identifier "_begin\n" \
"\t.type " #identifier "_begin, @object\n" \
"\t.align 16\n" #identifier "_begin:\n" \
"\t.incbin \"" filename "\"\n\n" \
\
"\t.local " #identifier "_end\n" \
"\t.type " #identifier "_end, @object\n" \
"\t.align 1\n" #identifier "_end:\n" \
"\t.byte 0\n" \
"\t.popsection\n"); \
\
extern std::byte const identifier##_begin[]; \
extern std::byte const identifier##_end[]
我希望能够声明它是 uint32_t
而不是 std::byte
。这将要求文件具有 4 字节的整数倍。对齐已经由宏处理(对齐到 16)。如果文件大小错误,我可以触发编译时错误吗?即在程序集中添加一些编译时断言。
如果你只是 .p2align 2
在文件之后,它会将总大小四舍五入到 4 字节的倍数,而不管有多少字节 .incbin
assembled 到。 (因为文件从对齐的位置开始)。
如果您想检查而不是 pad,可以在 assemble 时使用 assemble r 指令。显然不在 compile-time 正确的位置,因为它只是将 C++ 转换为 asm,以便稍后提供给 assembler。 (在 GCC 中,这些步骤实际上是分开的,并不是同一个编译器进程的所有部分。但在其他实现中逻辑上仍然是顺序的。)
同一文件中两个标签之间的距离是一个 assemble-time 常量,您可以在 (b-a)&3
.
等 GAS 表达式中使用它
如果你使用 .rept -((b-a)&3)
,你将有一个 0 或负的重复计数,后者是一个 assemble-time 错误,根据文件大小 % 4 是 non-zero. (有趣的事实:Linux 内核在 C 中使用这样的技巧,通过生成大小为 0 或负数的数组来执行静态断言)。
或者更好,.error
can be controlled by .rept
or other GAS directives like .ifgt
.(.rept
会重复错误消息 (b-a)%4
次,而 if-greater-than-zero只重复一次。)
.p2align 2
a:
.incbin "bin"
# .p2align 2
b:
.ifgt (b-a)&3 # true if remainder > 0
.error "asm static_assert failed: file size not a multiple of 4"
.endif
$ echo xyz > bin # size 4
$ gcc -c foo.s
$ echo >> bin # size 5
$ gcc -c foo.s
foo.s: Assembler messages:
foo.s:9: Error: asm static_assert failed: file size not a multiple of 4
使用 GNU assembler (GNU Binutils) 2.36.1
测试
我会留给你把它填回你的内联 asm 宏,文件名在 .error
字符串中。
我有以下宏来嵌入文件名中的二进制数据:
#define INCBIN(identifier, filename) \
asm(".pushsection .rodata\n" \
"\t.local " #identifier "_begin\n" \
"\t.type " #identifier "_begin, @object\n" \
"\t.align 16\n" #identifier "_begin:\n" \
"\t.incbin \"" filename "\"\n\n" \
\
"\t.local " #identifier "_end\n" \
"\t.type " #identifier "_end, @object\n" \
"\t.align 1\n" #identifier "_end:\n" \
"\t.byte 0\n" \
"\t.popsection\n"); \
\
extern std::byte const identifier##_begin[]; \
extern std::byte const identifier##_end[]
我希望能够声明它是 uint32_t
而不是 std::byte
。这将要求文件具有 4 字节的整数倍。对齐已经由宏处理(对齐到 16)。如果文件大小错误,我可以触发编译时错误吗?即在程序集中添加一些编译时断言。
如果你只是 .p2align 2
在文件之后,它会将总大小四舍五入到 4 字节的倍数,而不管有多少字节 .incbin
assembled 到。 (因为文件从对齐的位置开始)。
如果您想检查而不是 pad,可以在 assemble 时使用 assemble r 指令。显然不在 compile-time 正确的位置,因为它只是将 C++ 转换为 asm,以便稍后提供给 assembler。 (在 GCC 中,这些步骤实际上是分开的,并不是同一个编译器进程的所有部分。但在其他实现中逻辑上仍然是顺序的。)
同一文件中两个标签之间的距离是一个 assemble-time 常量,您可以在 (b-a)&3
.
如果你使用 .rept -((b-a)&3)
,你将有一个 0 或负的重复计数,后者是一个 assemble-time 错误,根据文件大小 % 4 是 non-zero. (有趣的事实:Linux 内核在 C 中使用这样的技巧,通过生成大小为 0 或负数的数组来执行静态断言)。
或者更好,.error
can be controlled by .rept
or other GAS directives like .ifgt
.(.rept
会重复错误消息 (b-a)%4
次,而 if-greater-than-zero只重复一次。)
.p2align 2
a:
.incbin "bin"
# .p2align 2
b:
.ifgt (b-a)&3 # true if remainder > 0
.error "asm static_assert failed: file size not a multiple of 4"
.endif
$ echo xyz > bin # size 4
$ gcc -c foo.s
$ echo >> bin # size 5
$ gcc -c foo.s
foo.s: Assembler messages:
foo.s:9: Error: asm static_assert failed: file size not a multiple of 4
使用 GNU assembler (GNU Binutils) 2.36.1
测试我会留给你把它填回你的内联 asm 宏,文件名在 .error
字符串中。