Повноцінним рішенням планування потоків, яке повинно виходити в однакові рази за кожен тест, є складання вашої програми на незалежність від ОС та завантаження комп'ютера, щоб запустити програму в середовищі, яка не має ОС. Однак це в значній мірі непрактично і в кращому випадку буде складно.
Хороший замінник переходу на ОС, що не потребує ОС, - це просто встановити спорідненість поточного потоку до 1 ядра, а пріоритет - найвищий. Ця альтернатива повинна забезпечити достатньо стійкі результати.
Також слід вимкнути оптимізацію, яка заважатиме налагодженню, що для g ++ або gcc означає додавання -Og
до командного рядка , щоб запобігти оптимізації тестованого коду. -O0
Прапор не слід використовувати , оскільки він вводить додаткові непотрібні накладні витрати , які будуть включені в результатах синхронізації, тим самим спотворюючи таймерну швидкість коду.
Навпаки, обидва припущення, що ви використовуєте -Ofast
(або, принаймні, -O3
) для остаточної збірки виробництва та ігноруєте питання про усунення "мертвого" коду, -Og
виконує дуже мало оптимізацій порівняно з -Ofast
; таким чином -Og
можна неправильно представити реальну швидкість коду в кінцевому продукті.
Крім того, усі випробування на швидкість (певною мірою) підтверджуються: у кінцевому виробничому продукті, складеному -Ofast
, кожен фрагмент / розділ / функція коду не виділений; скоріше, кожен фрагмент коду безперервно переходить у наступний, таким чином, дозволяючи компілятору потенційно приєднатись, об’єднатись та оптимізувати разом фрагменти коду з усього місця.
У той же час, якщо ви орієнтуєте фрагмент коду, який дуже активно використовує realloc()
, фрагмент коду може працювати повільніше у виробничому продукті з достатньо високою фрагментацією пам'яті. Отже, вираз "ціле більше, ніж сума його частин" застосовується до цієї ситуації, оскільки код у кінцевій збірці виробництва може працювати помітно швидше або повільніше, ніж окремий фрагмент, який ви перевіряєте на швидкість.
Часткове рішення, яке може зменшити невідповідність, використовується -Ofast
для тестування швидкості З додаванням asm volatile("" :: "r"(var))
змінних, що беруть участь у тесті для запобігання усуненню мертвого коду / циклу.
Ось приклад того, як орієнтувати функції квадратних коренів на комп'ютері Windows.
// set USE_ASM_TO_PREVENT_ELIMINATION to 0 to prevent `asm volatile("" :: "r"(var))`
// set USE_ASM_TO_PREVENT_ELIMINATION to 1 to enforce `asm volatile("" :: "r"(var))`
#define USE_ASM_TO_PREVENT_ELIMINATION 1
#include <iostream>
#include <iomanip>
#include <cstdio>
#include <chrono>
#include <cmath>
#include <windows.h>
#include <intrin.h>
#pragma intrinsic(__rdtsc)
#include <cstdint>
class Timer {
public:
Timer() : beg_(clock_::now()) {}
void reset() { beg_ = clock_::now(); }
double elapsed() const {
return std::chrono::duration_cast<second_>
(clock_::now() - beg_).count(); }
private:
typedef std::chrono::high_resolution_clock clock_;
typedef std::chrono::duration<double, std::ratio<1> > second_;
std::chrono::time_point<clock_> beg_;
};
unsigned int guess_sqrt32(register unsigned int n) {
register unsigned int g = 0x8000;
if(g*g > n) {
g ^= 0x8000;
}
g |= 0x4000;
if(g*g > n) {
g ^= 0x4000;
}
g |= 0x2000;
if(g*g > n) {
g ^= 0x2000;
}
g |= 0x1000;
if(g*g > n) {
g ^= 0x1000;
}
g |= 0x0800;
if(g*g > n) {
g ^= 0x0800;
}
g |= 0x0400;
if(g*g > n) {
g ^= 0x0400;
}
g |= 0x0200;
if(g*g > n) {
g ^= 0x0200;
}
g |= 0x0100;
if(g*g > n) {
g ^= 0x0100;
}
g |= 0x0080;
if(g*g > n) {
g ^= 0x0080;
}
g |= 0x0040;
if(g*g > n) {
g ^= 0x0040;
}
g |= 0x0020;
if(g*g > n) {
g ^= 0x0020;
}
g |= 0x0010;
if(g*g > n) {
g ^= 0x0010;
}
g |= 0x0008;
if(g*g > n) {
g ^= 0x0008;
}
g |= 0x0004;
if(g*g > n) {
g ^= 0x0004;
}
g |= 0x0002;
if(g*g > n) {
g ^= 0x0002;
}
g |= 0x0001;
if(g*g > n) {
g ^= 0x0001;
}
return g;
}
unsigned int empty_function( unsigned int _input ) {
return _input;
}
unsigned long long empty_ticks=0;
double empty_seconds=0;
Timer my_time;
template<unsigned int benchmark_repetitions>
void benchmark( char* function_name, auto (*function_to_do)( auto ) ) {
register unsigned int i=benchmark_repetitions;
register unsigned long long start=0;
my_time.reset();
start=__rdtsc();
while ( i-- ) {
auto result = (*function_to_do)( i << 7 );
#if USE_ASM_TO_PREVENT_ELIMINATION == 1
asm volatile("" :: "r"(
// There is no data type in C++ that is smaller than a char, so it will
// not throw a segmentation fault error to reinterpret any arbitrary
// data type as a char. Although, the compiler might not like it.
result
));
#endif
}
if ( function_name == nullptr ) {
empty_ticks = (__rdtsc()-start);
empty_seconds = my_time.elapsed();
std::cout<< "Empty:\n" << empty_ticks
<< " ticks\n" << benchmark_repetitions << " repetitions\n"
<< std::setprecision(15) << empty_seconds
<< " seconds\n\n";
} else {
std::cout<< function_name<<":\n" << (__rdtsc()-start-empty_ticks)
<< " ticks\n" << benchmark_repetitions << " repetitions\n"
<< std::setprecision(15) << (my_time.elapsed()-empty_seconds)
<< " seconds\n\n";
}
}
int main( void ) {
void* Cur_Thread= GetCurrentThread();
void* Cur_Process= GetCurrentProcess();
unsigned long long Current_Affinity;
unsigned long long System_Affinity;
unsigned long long furthest_affinity;
unsigned long long nearest_affinity;
if( ! SetThreadPriority(Cur_Thread,THREAD_PRIORITY_TIME_CRITICAL) ) {
SetThreadPriority( Cur_Thread, THREAD_PRIORITY_HIGHEST );
}
if( ! SetPriorityClass(Cur_Process,REALTIME_PRIORITY_CLASS) ) {
SetPriorityClass( Cur_Process, HIGH_PRIORITY_CLASS );
}
GetProcessAffinityMask( Cur_Process, &Current_Affinity, &System_Affinity );
furthest_affinity = 0x8000000000000000ULL>>__builtin_clzll(Current_Affinity);
nearest_affinity = 0x0000000000000001ULL<<__builtin_ctzll(Current_Affinity);
SetProcessAffinityMask( Cur_Process, furthest_affinity );
SetThreadAffinityMask( Cur_Thread, furthest_affinity );
const int repetitions=524288;
benchmark<repetitions>( nullptr, empty_function );
benchmark<repetitions>( "Standard Square Root", standard_sqrt );
benchmark<repetitions>( "Original Guess Square Root", original_guess_sqrt32 );
benchmark<repetitions>( "New Guess Square Root", new_guess_sqrt32 );
SetThreadPriority( Cur_Thread, THREAD_PRIORITY_IDLE );
SetPriorityClass( Cur_Process, IDLE_PRIORITY_CLASS );
SetProcessAffinityMask( Cur_Process, nearest_affinity );
SetThreadAffinityMask( Cur_Thread, nearest_affinity );
for (;;) { getchar(); }
return 0;
}
Також, заслуга Майка Джарвіса за його Таймер.
Зверніть увагу (це дуже важливо), що якщо ви збираєтеся виконувати більші фрагменти коду, тоді ви дійсно повинні зменшити кількість ітерацій, щоб запобігти замерзанню комп'ютера.