在调查可疑的索赔时,我编写了这个小测试程序noway.c
int proveit()
{
unsigned int n = 0;
while (1) n++;
return 0;
}
int main()
{
proveit();
return 0;
}
测试一下,我得到:
$ clang -O noway.c
$ ./a.out
zsh: illegal hardware instruction ./a.out
笏。
如果我不进行优化就进行编译,则它会按预期挂起。我看了看程序集,没有所有的花哨main
功能,函数看起来像这样:
_main: ## @main
pushq %rbp
movq %rsp, %rbp
ud2
哪里 ud2
显然是一个指令专门为未定义行为。前面提到的可疑声明“永不返回的函数是UB”得到了增强。我仍然很难相信。真!?您不能安全地编写自旋循环吗?
所以我想我的问题是:
- 这是对正在发生的事情的正确阅读吗?
- 如果是这样,有人可以指出我的官方资源吗?
- 您希望在哪种情况下进行这种优化?
相关资料
$ clang --version
Apple clang version 11.0.0 (clang-1100.0.20.17)
Target: x86_64-apple-darwin18.6.0
Thread model: posix
InstalledDir: /Applications/Xcode-beta.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin
3
IIRC,签名的int溢出是UB。
—
wildplasser
^^^^即
—
。.– WhozCraig
int n = 0
===> unsigned int n = 0;
还是更好while (1);
嗯...编译器资源管理器clang使用-O gcc.godbolt.org/z/NTCQYT获取自我目标跳转指令,并且在没有该指令的情况下进行逐行转换。在许多版本中似乎保持一致。但是我还记得(尽管不会查找)C标准说,如果无副作用,则非终止是ub 。这是Hans Boehm的解释,它允许进行某些其他不可能的编译器优化:open-std.org/jtc1/sc22/wg14/www/docs/n1528.htm
—
基因
未定义的行为是有符号整数溢出,而不是无限循环。您可以使用
—
MM
unsigned int
@Gene我不这么认为。可以假定具有非const控制表达式的无副作用循环终止(port70.net/~nsz/c/c11/n1570.html#6.8.5p6),但是忙循环应该可以。UB处于整数溢出状态,尽管我无法使用UD指令重现该示例。
—
PSkocik