Как установить разрешение таймера до 0,5 мс?

Я хочу установить разрешение таймера машины на 0,5 мс.

Утилита Sysinternal сообщает, что минимальное разрешение часов составляет 0,5 мс, так что это можно сделать.

PS Я знаю, как установить его на 1 мс.

PPS Я изменил его с C# на более общий вопрос (спасибо Гансу)

Разрешение системного таймера

6 ответов

Решение

NtSetTimerResolution

Пример кода:

#include <windows.h>

extern "C" NTSYSAPI NTSTATUS NTAPI NtSetTimerResolution(ULONG DesiredResolution, BOOLEAN SetResolution, PULONG CurrentResolution);

...

ULONG currentRes;
NtSetTimerResolution(5000, TRUE, &currentRes);

Связать с ntdll.lib,

Вы можете получить разрешение 0,5 мс с помощью скрытого API NtSetTimerResolution(), NtSetTimerResolution экспортируется родной библиотекой Windows NT NTDLL.DLL. См. Как установить разрешение таймера до 0,5 мс? на MSDN. Тем не менее, истинно достижимое разрешение определяется базовым оборудованием. Современное оборудование поддерживает разрешение 0,5 мс. Еще больше подробностей можно найти во внутренних таймерах высокого разрешения Windows NT. Поддерживаемые разрешения могут быть получены путем вызова NtQueryTimerResolution().

Как сделать:

#define STATUS_SUCCESS 0
#define STATUS_TIMER_RESOLUTION_NOT_SET 0xC0000245

// after loading NtSetTimerResolution from ntdll.dll:

// The requested resolution in 100 ns units:
ULONG DesiredResolution = 5000;  
// Note: The supported resolutions can be obtained by a call to NtQueryTimerResolution()

ULONG CurrentResolution = 0;

// 1. Requesting a higher resolution
// Note: This call is similar to timeBeginPeriod.
// However, it to to specify the resolution in 100 ns units.
if (NtSetTimerResolution(DesiredResolution ,TRUE,&CurrentResolution) != STATUS_SUCCESS) {
    // The call has failed
}

printf("CurrentResolution [100 ns units]: %d\n",CurrentResolution);
// this will show 5000 on more modern platforms (0.5ms!)

//       do your stuff here at 0.5 ms timer resolution

// 2. Releasing the requested resolution
// Note: This call is similar to timeEndPeriod 
switch (NtSetTimerResolution(DesiredResolution ,FALSE,&CurrentResolution) {
    case STATUS_SUCCESS:
        printf("The current resolution has returned to %d [100 ns units]\n",CurrentResolution);
        break;
    case STATUS_TIMER_RESOLUTION_NOT_SET:
        printf("The requested resolution was not set\n");   
        // the resolution can only return to a previous value by means of FALSE 
        // when the current resolution was set by this application      
        break;
    default:
        // The call has failed

}

Примечание. Функциональность NtSetTImerResolution в основном сопоставлена ​​с функциями timeBeginPeriod а также timeEndPeriod используя значение bool Set (См. Внутри таймеров высокого разрешения Windows NT для более подробной информации о схеме и всех ее последствиях). Однако мультимедийный пакет ограничивает степень детализации до миллисекунд, а NtSetTimerResolution позволяет устанавливать значения менее миллисекунды.

Если вы используете python, я написал библиотеку с именем wres, которая внутренне вызывает NtSetTimerResolution.

pip install wres

import wres

# Set resolution to 0.5 ms (in 100-ns units)
# Automatically restore previous resolution when exit with statement
with wres.set_resolution(5000):
    pass

Лучшее, что вы можете получить из Win32 API - это одна миллисекунда с timeBeginPeriod и timeSetEvent. Может быть, ваш HAL может работать лучше, но это академично, вы не можете написать код драйвера устройства на C#.

Для этого вам понадобится таймер высокого разрешения. Вы можете найти некоторую информацию здесь: http://www.codeproject.com/KB/cs/highperformancetimercshar.aspx

РЕДАКТИРОВАТЬ: Более подробную информацию можно найти здесь: установить десятой доли миллисекунды; http://msdn.microsoft.com/en-us/library/aa964692(VS.80).aspx

Timer.Interval = 500;

Другие вопросы по тегам