Разница в производительности между Windows и Linux с помощью компилятора intel: просмотр сборки
я запускаю программу как на Windows, так и на Linux (x86-64). Он был скомпилирован с тем же компилятором (Intel Parallel Studio XE 2017) с теми же параметрами, и версия Windows в 3 раза быстрее, чем Linux. Виновником является звонок в std:: erf который разрешен в библиотеке Intel math для обоих случаев (по умолчанию он связан динамически в Windows и статически в Linux, но использование динамической компоновки в Linux дает то же самое производительность.)
вот простая программа для воспроизведения проблемы.
#include <cmath>
#include <cstdio>
int main() {
int n = 100000000;
float sum = 1.0f;
for (int k = 0; k < n; k++) {
sum += std::erf(sum);
}
std::printf("%7.2fn", sum);
}
когда я профилирую эту программу с помощью vTune, я нахожу, что сборка немного отличается между версией Windows и Linux. Вот сайт вызова (цикл) на Windows
Block 3:
"vmovaps xmm0, xmm6"
call 0x1400023e0 <erff>
Block 4:
inc ebx
"vaddss xmm6, xmm6, xmm0"
"cmp ebx, 0x5f5e100"
jl 0x14000103f <Block 3>
и начало функции erf вызывается на Windows
Block 1:
push rbp
"sub rsp, 0x40"
"lea rbp, ptr [rsp+0x20]"
"lea rcx, ptr [rip-0xa6c81]"
"movd edx, xmm0"
"movups xmmword ptr [rbp+0x10], xmm6"
"movss dword ptr [rbp+0x30], xmm0"
"mov eax, edx"
"and edx, 0x7fffffff"
"and eax, 0x80000000"
"add eax, 0x3f800000"
"mov dword ptr [rbp], eax"
"movss xmm6, dword ptr [rbp]"
"cmp edx, 0x7f800000"
...
в Linux код немного отличается. Сайт вызова:
Block 3
"vmovaps %xmm1, %xmm0"
"vmovssl %xmm1, (%rsp)"
callq 0x400bc0 <erff>
Block 4
inc %r12d
"vmovssl (%rsp), %xmm1"
"vaddss %xmm0, %xmm1, %xmm1" <-------- hotspot here
"cmp x5f5e100, %r12d"
jl 0x400b6b <Block 3>
и начала вызываемая функция (erf):
"movd %xmm0, %edx"
"movssl %xmm0, -0x10(%rsp)" <-------- hotspot here
"mov %edx, %eax"
"and x7fffffff, %edx"
"and x80000000, %eax"
"add x3f800000, %eax"
"movl %eax, -0x18(%rsp)"
"movssl -0x18(%rsp), %xmm0"
"cmp x7f800000, %edx"
jnl 0x400dac <Block 8>
...
Я показал 2 точки, где время теряется на Linux.
кто-нибудь понимает сборку достаточно, чтобы объяснить мне разницу в кодах 2 и почему версия Linux в 3 раза медленнее?
2 ответа:
в обоих случаях Аргументы и результаты передаются только в регистрах, согласно соответствующим соглашениям о вызовах в Windows и GNU/Linux.
в варианте GNU / Linux,
xmm1
используется для накопления суммы. Поскольку это регистр Call-clobbered (a.k.a caller-saved), он хранится (и восстанавливается) в кадре стека вызывающего абонента при каждом вызове.в варианте Windows,
xmm6
используется для накопления суммы. Этот регистр является callee-сохранено в соглашении о вызовах Windows (но не в GNU / Linux one).Итак, в целом, версия GNU / Linux сохраняет / восстанавливает оба
xmm0
(в вызываемом[1]) иxmm1
(в вызывающем), тогда как версия Windows сохраняет / восстанавливает толькоxmm6
(на абонента).[1] нужно посмотреть
std::errf
чтобы выяснить, почему.
используя Visual Studio 2015, Win 7 64-разрядный режим, я нахожу следующий код для некоторых путей, используемых в erf() (не все пути показаны). Каждый путь включает до 8 (возможно, больше для других путей) констант, считанных из памяти, поэтому одно хранилище / загрузка для сохранения регистра вряд ли приведет к 3-кратной разнице скоростей между Linux и Windows. Что касается сохранения / восстановления, этот пример сохраняет и восстанавливает xmm6 и xmm7. Что касается времени, то программа в исходном посте занимает около 0.86 секунд на Intel 3770K (3.5 ghz cpu) (VS2015 / Win 7 64 бит). Обновление-позже я определил, что накладные расходы для сохранения и восстановления регистра xmm составляют около 0,03 секунды в случае программ 10^8 циклов (около 3 наносекунд на цикл).
000007FEEE25CF90 mov rax,rsp 000007FEEE25CF93 movss dword ptr [rax+8],xmm0 000007FEEE25CF98 sub rsp,48h 000007FEEE25CF9C movaps xmmword ptr [rax-18h],xmm6 000007FEEE25CFA0 lea rcx,[rax+8] 000007FEEE25CFA4 movaps xmmword ptr [rax-28h],xmm7 000007FEEE25CFA8 movaps xmm6,xmm0 000007FEEE25CFAB call 000007FEEE266370 000007FEEE25CFB0 movsx ecx,ax 000007FEEE25CFB3 test ecx,ecx 000007FEEE25CFB5 je 000007FEEE25D0AF 000007FEEE25CFBB sub ecx,1 000007FEEE25CFBE je 000007FEEE25D08F 000007FEEE25CFC4 cmp ecx,1 000007FEEE25CFC7 je 000007FEEE25D0AF 000007FEEE25CFCD xorps xmm7,xmm7 000007FEEE25CFD0 movaps xmm2,xmm6 000007FEEE25CFD3 comiss xmm7,xmm6 000007FEEE25CFD6 jbe 000007FEEE25CFDF 000007FEEE25CFD8 xorps xmm2,xmmword ptr [7FEEE2991E0h] 000007FEEE25CFDF movss xmm0,dword ptr [7FEEE298E50h] 000007FEEE25CFE7 comiss xmm0,xmm2 000007FEEE25CFEA jbe 000007FEEE25D053 000007FEEE25CFEC movaps xmm2,xmm6 000007FEEE25CFEF mulss xmm2,xmm6 000007FEEE25CFF3 movaps xmm0,xmm2 000007FEEE25CFF6 movaps xmm1,xmm2 000007FEEE25CFF9 mulss xmm0,dword ptr [7FEEE298B34h] 000007FEEE25D001 mulss xmm1,dword ptr [7FEEE298B5Ch] 000007FEEE25D009 addss xmm0,dword ptr [7FEEE298B8Ch] 000007FEEE25D011 addss xmm1,dword ptr [7FEEE298B9Ch] 000007FEEE25D019 mulss xmm0,xmm2 000007FEEE25D01D mulss xmm1,xmm2 000007FEEE25D021 addss xmm0,dword ptr [7FEEE298BB8h] 000007FEEE25D029 addss xmm1,dword ptr [7FEEE298C88h] 000007FEEE25D031 mulss xmm0,xmm2 000007FEEE25D035 mulss xmm1,xmm2 000007FEEE25D039 addss xmm0,dword ptr [7FEEE298DC8h] 000007FEEE25D041 addss xmm1,dword ptr [7FEEE298D8Ch] 000007FEEE25D049 divss xmm0,xmm1 000007FEEE25D04D mulss xmm0,xmm6 000007FEEE25D051 jmp 000007FEEE25D0B2 000007FEEE25D053 movss xmm1,dword ptr [7FEEE299028h] 000007FEEE25D05B comiss xmm1,xmm2 000007FEEE25D05E jbe 000007FEEE25D076 000007FEEE25D060 movaps xmm0,xmm2 000007FEEE25D063 call 000007FEEE25CF04 000007FEEE25D068 movss xmm1,dword ptr [7FEEE298D8Ch] 000007FEEE25D070 subss xmm1,xmm0 000007FEEE25D074 jmp 000007FEEE25D07E 000007FEEE25D076 movss xmm1,dword ptr [7FEEE298D8Ch] 000007FEEE25D07E comiss xmm7,xmm6 000007FEEE25D081 jbe 000007FEEE25D08A 000007FEEE25D083 xorps xmm1,xmmword ptr [7FEEE2991E0h] 000007FEEE25D08A movaps xmm0,xmm1 000007FEEE25D08D jmp 000007FEEE25D0B2 000007FEEE25D08F mov eax,8000h 000007FEEE25D094 test word ptr [rsp+52h],ax 000007FEEE25D099 je 000007FEEE25D0A5 000007FEEE25D09B movss xmm0,dword ptr [7FEEE2990DCh] 000007FEEE25D0A3 jmp 000007FEEE25D0B2 000007FEEE25D0A5 movss xmm0,dword ptr [7FEEE298D8Ch] 000007FEEE25D0AD jmp 000007FEEE25D0B2 000007FEEE25D0AF movaps xmm0,xmm6 000007FEEE25D0B2 movaps xmm6,xmmword ptr [rsp+30h] 000007FEEE25D0B7 movaps xmm7,xmmword ptr [rsp+20h] 000007FEEE25D0BC add rsp,48h 000007FEEE25D0C0 ret