В приложении к операционным системам, в частности к Unix, jiffy — это интервал времени между двумя соседними успешно обработанными импульсами системного таймера. Исторически это значение равно 100 ms. Как уже было показано, интервал времени jiffy в операционной системе Linux может иметь разные значения.
Переменная jiffies
определена в файле следующим образом.
extern unsigned long volatile jiffies;
Определение этой переменной достаточно специфичное, и оно будет рассмотрено более подробно в следующем разделе. Сейчас давайте рассмотрим пример кода ядра. Пересчет из секунд в значение переменной jiffies
можно выполнить следующим образом.
(секунды * HZ)
Отсюда следует, что преобразование из значения переменной jiffies
в секунды можно выполнить, как показано ниже.
(jiffies / HZ)
Первый вариант встречается более часто. Например, часто необходимо установить значение некоторого момента времени в будущем.
unsigned long time_stamp = jiffies; /* сейчас */
unsigned long next_tick = jiffies + 1; /* через один импульс таймера
от текущего момента */
unsigned long later = jiffies + 5*HZ; /* через пять секунд от текущего
момента */
Последний пример обычно используется при взаимодействии с пространством пользователя, так как в самом ядре редко используется абсолютное время.
Заметим, что переменная jiffies
имеет тип unsigned long
и использовать какой-либо другой тип будет неправильным.
Внутреннее представление переменной jiffies
Переменная jiffies
исторически всегда представлялась с помощью типа unsigned long
и, следовательно, имеет длину 32 бит для 32-разрядных аппаратных платформ и 64 бит для 64-разрядных. В случае 32-разрядного значения переменной jiffies
и частоты появления временных отметок 100 раз в секунду, переполнение этой переменной будет происходить примерно каждые 497 дней, что является вполне возможным событием. Увеличение значения параметра HZ
до 1000 уменьшает период переполнения до 49.7 дней! В случае 64-разрядного типа переменной jiffies
, переполнение этой переменной невозможно за время существования чего-либо при любых возможных значениях параметра HZ
для любой аппаратной платформы.
Из соображений производительности и по историческим причинам — в основном, для совместимости с уже существующим кодом ядра — разработчики ядра предпочли оставить тип переменной jiffies
— unsigned long
. Для решения проблемы пришлось немного подумать и применить возможности компоновщика.
Как уже говорилось, переменная jiffies
определяется в следующем виде и имеет тип unsigned long
.
extern unsigned long volatile jiffies;
Вторая переменная определяется в файле в следующем виде.
extern u64 jiffies_64;
Директивы компоновщика ld(1)
, которые используются для сборки главного образа ядра (для аппаратной платформы x86 описаны в файле arch/i386/kernel/vmlinux.lds.S
), указывают компоновщику, что переменную jiffies
необходимо совместить с началом переменной jiffies_64
.
jiffies = jiffies_64;
Следовательно, переменная jiffies
— это просто 32 младших разряда полной 64-разрядной переменной jiffies_64
. Так как в большинстве случаев переменная jiffies
используется для измерения промежутков времени, то для большей части кода существенными являются только младшие 32 бит.
В случае применения 64-разрядного значения, переполнение не может возникнуть за время существования чего-либо. В следующем разделе будут рассмотрены проблемы, связанные с переполнением (хотя переполнение счетчика импульсов системного таймера и не желательно, но это вполне нормальное и ожидаемое событие). Код, который используется для управления ходом времени, использует все 64 бит, и это предотвращает возможность переполнения 64-разрядного значения. На рис. 10.1 показана структура переменных jiffies
и jiffies_64
.
Рис. 10.1. Структура переменных jiffies
и jiffies_64
Код, который использует переменную jiffies
, просто получает доступ к тридцати двум младшим битам переменной jiffies_64
. Функция get_jiffies_64()
может быть использована для получения полного 64-разрядного значения [57] Необходима специальная функция, так как на 32-разрядных аппаратных платформах нельзя атомарно обращаться к двум машинным словам 64-разрядного значения, Специальная функция, перед тем как считать значение, блокирует счетчик импульсов системного таймера с помощью блокировки xtime_lock .
. Такая необходимость возникает редко, следовательно большая часть кода просто продолжает считывать младшие 32 разряда непосредственно из переменной jiffies
.
Читать дальше