Windows 上来自 VS C++ 的非法指令

use*_*840 2 c++ assembly avx visual-c++ illegal-instruction

对于某些 Windows 7 用户,我有一个 C++ 应用程序在启动时崩溃。我无法在自己的机器上重现该错误,但使用 breakpad 生成了一个 .dmp 文件,这表明代码因初始化静态 std::vector 数组的“非法指令”而崩溃。这可能意味着什么?

例外: Unhandled exception at 0x000000013F121362 (myApp.exe) in myApp.exe.4328.dmp: 0xC000001D:

Illegal Instruction.myApp.exe! dynamic initializer for Keyboard::key_freqs_() Line 11 C++

拆卸:

const std::vector<double> Keyboard::key_freqs_ = std::vector<double>({
**Crashed here->** 000000013F121362  vmovaps     ymm0,ymmword ptr [__ymm@404059fbe76c8b44403ede353f7ced91403d228f5c28f5c3403b800000000000 (01408B7DE0h)]  
000000013F12136A  movzx       r9d,byte ptr [rsp+20h]  
000000013F121370  lea         r8,[rbp+1F0h]  
000000013F121377  vmovups     ymmword ptr [rsp+30h],ymm0  
000000013F12137D  vmovaps     ymm0,ymmword ptr [__ymm@404499fbe76c8b44404371eb851eb85240425a9fbe76c8b4404152f1a9fbe76d (01408B7E00h)]  
000000013F121385  vmovups     ymmword ptr [rsp+50h],ymm0  
000000013F12138B  vmovaps     ymm0,ymmword ptr [__ymm@4049f4dd2f1a9fbe40487fdf3b645a1d40471fdf3b645a1d4045d3b645a1cac1 (01408B7E20h)]  
000000013F121393  vmovups     ymmword ptr [rsp+70h],ymm0  
000000013F121399  vmovaps     ymm0,ymmword ptr [__ymm@405059fbe76c8b44404ede147ae147ae404d22b020c49ba6404b800000000000 (01408B7E40h)]  
000000013F1213A1  vmovups     ymmword ptr [rbp-70h],ymm0  
000000013F1213A6  vmovaps     ymm0,ymmword ptr [__ymm@40549a0c49ba5e354053720c49ba5e3540525a9fbe76c8b4405152f1a9fbe76d (01408B7E60h)]  
000000013F1213AE  vmovups     ymmword ptr [rbp-50h],ymm0  
000000013F1213B3  vmovaps     ymm0,ymmword ptr [__ymm@4059f47ae147ae1440587fef9db22d0e40571fef9db22d0e4055d3a5e353f7cf (01408B7E80h)]  
000000013F1213BB  vmovups     ymmword ptr [rbp-30h],ymm0  
000000013F1213C0  vmovaps     ymm0,ymmword ptr [__ymm@406059eb851eb852405ede147ae147ae405d228f5c28f5c3405b800000000000 (01408B7EA0h)]  
000000013F1213C8  vmovups     ymmword ptr [rbp-10h],ymm0  
...
Run Code Online (Sandbox Code Playgroud)

编辑:正如答案所暗示的那样,我使用的是 /arch:AVX 编译标志,它不适用于所有机器。

rus*_*tyx 7

vmovups是一条AVX指令。从 Intel Sandy Bridge(第二代 i3/i5/i7 系列(2xxx 及更高版本,2011 年左右推出)和 Bulldozer 的 AMD 开始支持 AVX。

在较旧的 CPU(或当前的 Pentium/Celeron 或低功耗 Atom/Silvermont)上,指令将陷入非法指令错误。

您应该重新编译你的应用程序不支持AVX和/或使用__cpuid检查CPU的能力,并委托执行,以“慢”的版本,如果需要的话。


默认情况下,虚拟机可能不会通过 AVX 支持,但这是可配置的。如果客户操作系统在 CPUID 标志中没有看到 AVX,它不会设置控制寄存器位,让 AVX 指令无故障执行。(这种必须启用的机制可以防止使用 AVX 的用户空间的状态被 AVX 不感知操作系统上的上下文切换破坏。)

因此,与 BMI2 等其他扩展不同,如果真正的硬件支持它们(无论虚拟化 CPUID),您就可以使用它们,而引入新架构状态的扩展则不是这种情况:AVX 和 AVX-512。

VM 可能默认为没有 AVX 的来宾,因此可以将 VM 迁移到没有 AVX 的主机。