ITIMER_REAL | Отслеживает время в терминах настенных часов — в реальном времени (в зависимости от выполнения процесса) — и генерирует сигнал SIGALRM. Несовместим с системным вызовом alarm() , который используется функцией sleep() . Не применяйте ни alarm() , ни sleep() , если имеется реальный интервальный таймер. |
ITIMER_VIRTUAL | Подсчитывает время только при исполнении процесса — не учитывая системные вызовы, которые производит процесс — и генерирует сигнал SIGVTALRM . |
ITIMER_PROF | Подсчитывает время только при выполнении процесса — включая время, за которое ядро посылает исполнительные системные вызовы от имени процесса, и не включая время, потраченное на прерывание процесса по инициативе самого процесса — и генерирует сигнал SIGPROF . Учет времени, затраченного на обработку прерываний, оказывается настолько трудоемким, что даже может изменить настройки таймера. |
Комбинация таймеров ITIMER_VIRTUAL
и ITIMER_PROF
часто используется в профилирующих кодах.
Каждый из этих таймеров генерирует ассоциированный сигнал об истечении таймера в пределах одного хода системных часов (как правило, 1-10 миллисекунд). Если процесс работает в данное время, то сигнал генерируется сразу же; в противном случае сигнал генерируется немного позже (в зависимости от загрузки системы). Поскольку таймер ITIMER_VIRTUAL
следит за временем только во время работы процесса, то сигнал всегда доставляется незамедлительно.
Используйте структуру struct itimerval
для передачи запроса и установки интервальных таймеров.
struct itimerval {
struct timeval it_interval;
struct timeval it_value;
};
Член it_value
показывает количество времени, оставшееся до отправления следующего сигнала. Член it_interval
определяет время между сигналами; каждый раз при истечении таймера это значение присваивается переменной it_value
.
Для взаимодействия с интервальными таймерами предусмотрены два системных вызова. Оба принимают аргумент which
, указывающий обрабатываемый таймер.
int getitimer(int which, struct itimerval *val);
Переменной val
присваивается текущее состояние таймера which
.
int setitimer(int which, struct itimerval *new, struct itimerval *old);
Устанавливает таймер which
на значение new и заменяет old
предыдущей установкой, если она не равна NULL
.
Если параметр таймера it_value
приравнять к нулю, он немедленно заблокируется. Ввод нулевого значения для it_interval
отключает таймер после следующего запуска.
В следующем примере родительский процесс активизирует дочерний процесс, запускает односекундный таймер ITIMER_REAL
, засыпает на 10 секунд, а затем уничтожает дочерний процесс.
1: /* itimer.c */
2:
3: #include
4: #include
5: #include
6: #include
7: #include
8: #include
9: #include
10:
11:
12: void catch_signal(int ignored) {
13: static int iteration=0;
14:
15: printf("получен сигнал интервального таймера, итерация %d\n",
16: iteration++);
17: }
18:
19: pid_t start_timer(int interval) {
20: pid_t child;
21: struct itimerval it;
22: struct sigaction sa;
23:
24: if (!(child = fork())) {
25: memset(&sa, 0, sizeof(sa));
26: sa.sa_handler = catch_signal;
27: sigemptyset(&sa.sa_mask);
28: sa.sa_flags = SA_RESTART;
29:
30: sigaction(SIGALRM, &sa, NULL);
31:
32: memset(, 0, sizeof(it));
33: it.it_interval.tv_sec = interval;