Производительность систем 9785446118182, 9780136820154


251 42 17MB

Russian Pages 992 Year 2023

Report DMCA / Copyright

DOWNLOAD PDF FILE

Table of contents :
Предисловие
Об этом издании
Об этой книге
Благодарности
Об авторе
ОТ ИЗДАТЕЛЬСТВА
Глава 1. Введение
1.1. Производительность системы
1.2. Роли
1.3. Действия
1.4. Перспективы
1.5. Сложности оценки производительности
1.5.1. Субъективность
1.5.2. Сложность
1.5.3. Множественные причины
1.5.4. Множественные проблемы с производительностью
1.6. Задержка
1.7. Наблюдаемость
1.7.1. Счетчики, статистики и метрики
1.7.2. Профилирование
1.7.3. Трассировка
1.8. Эксперименты
1.9. Облачные вычисления
1.10. Методологии
1.10.1. Анализ производительности Linux за 60 секунд
1.11. Практические примеры
1.11.1. Медленные диски
1.11.2. Изменение в программном обеспечении
1.11.3. Дополнительное чтение
1.12. Ссылки
Глава 2. Методологии
2.1. Терминология
2.2. Модели
2.2.1. Тестируемая система
2.2.2. Система массового обслуживания
2.3. Основные понятия
2.3.1. Задержка
2.3.2. Шкалы времени
2.3.3. Компромиссы
2.3.4. Настройка производительности
2.3.5. Целесообразность
2.3.6. Когда лучше остановить анализ
2.3.7. Рекомендации действительны на данный момент времени
2.3.8. Нагрузка и архитектура
2.3.9. Масштабируемость
2.3.10. Метрики
2.3.11. Потребление
2.3.12. Насыщенность
2.3.13. Профилирование
2.3.14. Кэширование
2.3.15. Известные неизвестные
2.4. Точки зрения
2.4.1. Анализ ресурсов
2.4.2. Анализ рабочей нагрузки
2.5. Методология
2.5.1. Антиметодология «уличный фонарь»
2.5.2. Антиметодология «случайное изменение»
2.5.3. Антиметодология «виноват кто-то другой»
2.5.4. Специальный чек-лист
2.5.5. Формулировка проблемы
2.5.6. Научный метод
2.5.7. Цикл диагностики
2.5.8. Метод инструментов
2.5.9. Метод USE
2.5.10. Метод RED
2.5.11. Определение характеристик рабочей нагрузки
2.5.12. Анализ с увеличением детализации
2.5.13. Анализ задержек
2.5.14. Метод R
2.5.15. Трассировка событий
2.5.16. Базовые статистики
2.5.17. Статическая настройка производительности
2.5.18. Настройка кэширования
2.5.19. Микробенчмаркинг производительности
2.5.20. Мантры производительности
2.6. Моделирование
2.6.1. Корпоративные и облачные среды
2.6.2. Визуальная идентификация
2.6.3. Закон Амдала
2.6.4. Универсальный закон масштабируемости
2.6.5. Теория массового обслуживания
2.7. Планирование емкости
2.7.1. Пределы ресурсов
2.7.2. Факторный анализ
2.7.3. Решения масштабирования
2.8. Статистики
2.8.1. Количественная оценка прироста производительности
2.8.2. Усреднение
2.8.3. Стандартное отклонение, процентили, медиана
2.8.4. Коэффициент вариации
2.8.5. Мультимодальные распределения
2.8.6. Выбросы
2.9. Мониторинг
2.9.1. Временные закономерности
2.9.2. Инструменты мониторинга
2.9.3. Сводная статистика, накопленная с момента загрузки
2.10. Визуализация
2.10.1. Линейная диаграмма
2.10.2. Диаграммы рассеяния
2.10.3. Тепловые карты
2.10.4. Временная шкала
2.10.5. График поверхности
2.10.6. Инструменты визуализации
2.11. Упражнения
2.12. Ссылки
Глава 3. Операционные системы
3.1. Терминология
3.2. Основы
3.2.1. Ядро
3.2.2. Ядро и пользовательские режимы
3.2.3. Системные вызовы
3.2.4. Прерывания
3.2.5. Часы и бездействие
3.2.6. Процессы
3.2.7. Стеки
3.2.8. Виртуальная память
3.2.9. Планировщики
3.2.10. Файловая система
3.2.11. Кэширование
3.2.12. Сеть
3.2.13. Драйверы устройств
3.2.14. Многопроцессорность
3.2.15. Вытеснение
3.2.16. Управление ресурсами
3.2.17. Наблюдаемость
3.3. Ядра
3.3.1. Unix
3.3.2. BSD
3.3.3. Solaris
3.4. Linux
3.4.1. Новые разработки в ядре Linux
3.4.2. systemd
3.4.3. KPTI (Meltdown)
3.4.4. Расширенный BPF
3.5. Другие темы
3.5.1. Ядра PGO
3.5.2. Одноцелевые ядра
3.5.3. Микроядра и гибридные ядра
3.5.4. Распределенные операционные системы
3.6. Сравнение ядер
3.7. Упражнения
3.8. Ссылки
3.8.1. Дополнительное чтение
Глава 4. Инструменты наблюдения
4.1. Покрытие инструментами
4.1.1. Инструменты статического анализа производительности
4.1.2. Инструменты анализа кризисных ситуаций
4.2. Типы инструментов
4.2.1. Фиксированные счетчики
4.2.2. Профилирование
4.2.3. Трассировка
4.2.4. Мониторинг
4.3. Источники информации
4.3.1. /proc
4.3.2. /sys
4.3.3. Учет задержек
4.3.4. netlink
4.3.5. Точки трассировки
4.3.6. kprobes
4.3.7. uprobes
4.3.8. USDT
4.3.9. Аппаратные счетчики (PMC)
4.3.10. Другие источники информации для наблюдения
4.4. sar
4.4.1. Область покрытия sar(1)
4.4.2. Мониторинг с sar(1)
4.4.3. Вывод оперативной информации с помощью sar(1)
4.4.4. Документация с описанием sar(1)
4.5. Инструменты трассировки
4.6. Наблюдение за наблюдаемостью
4.7. Упражнения
4.8. Ссылки
Глава 5. Приложения
5.1. Основы приложений
5.1.1. Цель
5.1.2. Оптимизация общего случая
5.1.3. Наблюдаемость
5.1.4. Нотация «О-большое»
5.2. Методы повышения производительности приложений
5.2.1. Выбор размеров блоков ввода/вывода
5.2.2. Кэширование
5.2.3. Буферизация
5.2.4. Опрос
5.2.5. Конкурентность и параллелизм
5.2.6. Неблокирующий ввод/вывод
5.2.7. Привязка к процессору
5.2.8. Мантры производительности
5.3. Языки программирования
5.3.1. Компилируемые языки
5.3.2. Интерпретируемые языки
5.3.3. Виртуальные машины
5.3.4. Сборка мусора
5.4. Методология
5.4.1. Профилирование процессора
5.4.2. Анализ времени ожидания вне процессора
5.4.3. Анализ системных вызовов
5.4.4. Метод USE
5.4.5. Анализ состояния потока
5.4.6. Анализ блокировок
5.4.7. Настройка статической производительности
5.4.8. Распределенная трассировка
5.5. Инструменты наблюдения
5.5.1. perf
5.5.2. profile
5.5.3. offcputime
5.5.4. strace
5.5.5. execsnoop
5.5.6. syscount
5.5.7. bpftrace
5.6. Проблемы
5.6.1. Отсутствие символов
5.6.2. Отсутствие стеков
5.7. Упражнения
5.8. Ссылки
Глава 6. Процессоры
6.1. Терминология
6.2. Модели
6.2.1. Архитектура процессора
6.2.2. Кэш-память процессора
6.2.3. Очереди на выполнение
6.3. Понятия
6.3.1. Тактовая частота
6.3.2. Инструкции
6.3.3. Вычислительный конвейер
6.3.4. Ширина инструкций
6.3.5. Размеры инструкций
6.3.6. SMT
6.3.7. IPC, CPI
6.3.8. Потребление
6.3.9. Время в режиме пользователя/ядра
6.3.10. Насыщенность
6.3.11. Вытеснение
6.3.12. Инверсия приоритета
6.3.13. Несколько процессов, несколько потоков
6.3.14. Размер слова
6.3.15. Оптимизации компилятора
6.4. Архитектура
6.4.1. Аппаратное обеспечение
6.4.2. Программное обеспечение
6.5. Методология
6.5.1. Метод инструментов
6.5.2. Метод USE
6.5.3. Определение характеристик рабочей нагрузки
6.5.4. Профилирование
6.5.5. Анализ тактов
6.5.6. Мониторинг производительности
6.5.7. Статическая настройка производительности
6.5.8. Настройка приоритетов
6.5.9. Управление ресурсами
6.5.10. Привязка к процессору
6.5.11. Микробенчмаркинг
6.6. Инструменты наблюдения
6.6.1. uptime
6.6.2. vmstat
6.6.3. mpstat
6.6.4. sar
6.6.5. ps
6.6.6. top
6.6.7. pidstat
6.6.8. time, ptime
6.6.9. turbostat
6.6.10. showboost
6.6.11. pmcarch
6.6.12. tlbstat
6.6.13. perf
6.6.14. profile
6.6.15. cpudist
6.6.16. runqlat
6.6.17. runqlen
6.6.18. softirqs
6.6.19. hardirqs
6.6.20. bpftrace
6.6.21. Другие инструменты
6.7. Методы визуализации
6.7.1. Тепловая карта потребления
6.7.2. Тепловая карта с субсекундным смещением
6.7.3. Флейм-графики
6.7.4. FlameScope
6.8. Эксперименты
6.8.1. Ad hoc
6.8.2. SysBench
6.9. Настройка
6.9.1. Параметры компилятора
6.9.2. Приоритет и класс планирования
6.9.3. Параметры планировщика
6.9.4. Режимы масштабирования частоты
6.9.5. Состояния энергопотребления
6.9.6. Привязка к процессору
6.9.7. Исключительные процессорные наборы
6.9.8. Управление ресурсами
6.9.9. Параметры безопасной загрузки
6.9.10. Параметры процессора (настройка BIOS)
6.10. Упражнения
6.11. Ссылки
Глава 7. Память
7.1. Терминология
7.2. Основные понятия
7.2.1. Виртуальная память
7.2.2. Подкачка страниц
7.2.3. Подкачка страниц по требованию
7.2.4. Чрезмерное выделение памяти
7.2.5. Подкачка процессов
7.2.6. Использование кэша файловой системы
7.2.7. Потребление и насыщение
7.2.8. Распределители
7.2.9. Разделяемая память
7.2.10. Размер рабочего набора
7.2.11. Размер слова
7.3. Архитектура
7.3.1. Аппаратное обеспечение
7.3.2. Программное обеспечение
7.3.3. Адресное пространство процесса
7.4. Методология
7.4.1. Метод инструментов
7.4.2. Метод USE
7.4.3. Определение характеристик потребления памяти
7.4.4. Анализ тактов
7.4.5. Мониторинг производительности
7.4.6. Выявление утечек
7.4.7. Статическая настройка производительности
7.4.8. Управление ресурсами
7.4.9. Микробенчмаркинг
7.4.10. Уменьшение потребления памяти
7.5. Инструменты наблюдения
7.5.1. vmstat
7.5.2. PSI
7.5.3. swapon
7.5.4. sar
7.5.5. slabtop
7.5.6. numastat
7.5.7. ps
7.5.8. top
7.5.9. pmap
7.5.10. perf
7.5.11. drsnoop
7.5.12. wss
7.5.13. bpftrace
7.5.14. Другие инструменты
7.6. Настройка
7.6.1. Настраиваемые параметры
7.6.2. Несколько размеров страниц
7.6.3. Распределители
7.6.4. Привязка NUMA
7.6.5. Управление ресурсами
7.7. Упражнения
7.8. Ссылки
Глава 8. Файловые системы
8.1. Терминология
8.2. Модели
8.2.1. Интерфейсы файловых систем
8.2.2. Кэш файловой системы
8.2.3. Кэш второго уровня
8.3. Основные понятия
8.3.1. Задержки в файловой системе
8.3.2. Кэширование
8.3.3. Произвольный и последовательный ввод/вывод
8.3.4. Предварительная выборка
8.3.5. Упреждающее чтение
8.3.6. Кэширование с отложенной записью
8.3.7. Синхронная запись
8.3.8. Прямой и низкоуровневый ввод/вывод
8.3.9. Неблокирующий ввод/вывод
8.3.10. Файлы, отображаемые в память
8.3.11. Метаданные
8.3.12. Логический и физический ввод/вывод
8.3.13. Неравноценность операций
8.3.14. Специальные файловые системы
8.3.15. Доступ к отметкам времени
8.3.16. Емкость
8.4. Архитектура
8.4.1. Стек ввода-вывода файловой системы
8.4.2. VFS
8.4.3. Кэши файловой системы
8.4.4. Особенности файловых систем
8.4.5. Типы файловых систем
8.4.6. Тома и пулы
8.5. Методология
8.5.1. Анализ дисков
8.5.2. Анализ задержек
8.5.3. Определение характеристик рабочей нагрузки
8.5.4. Мониторинг производительности
8.5.5. Статическая настройка производительности
8.5.6. Настройка кэша
8.5.7. Разделение рабочей нагрузки
8.5.8. Микробенчмаркинг
8.6. Инструменты наблюдения
8.6.1. mount
8.6.2. free
8.6.3. top
8.6.4. vmstat
8.6.5. sar
8.6.6. slabtop
8.6.7. strace
8.6.8. fatrace
8.6.9. LatencyTOP
8.6.10. opensnoop
8.6.11. filetop
8.6.12. cachestat
8.6.13. ext4dist (xfs, zfs, btrfs, nfs)
8.6.14. ext4slower (xfs, zfs, btrfs, nfs)
8.6.15. bpftrace
8.6.17. Другие инструменты
8.6.18. Визуализация
8.7. Эксперименты
8.7.1. Ad hoc
8.7.2. Инструменты микробенчмаркинга
8.7.3. Очистка кэша
8.8. Настройка
8.8.1. Функции
8.8.2. ext4
8.8.3. ZFS
8.9. Упражнения
8.10. Ссылки
Глава 9. Диски
9.1. Терминология
9.2. Модели
9.2.1. Простой диск
9.2.2. Кэширующие диски
9.2.3. Контроллер
9.3. Основные понятия
9.3.1. Измерение времени
9.3.2. Масштаб времени
9.3.3. Кэширование
9.3.4. Произвольный и последовательный ввод/вывод
9.3.5. Соотношение операций чтения и записи
9.3.6. Размер ввода/вывода
9.3.7. Несопоставимость количества операций в секунду
9.3.8. Дисковые команды, не связанные с передачей данных
9.3.9. Потребление
9.3.10. Насыщенность
9.3.11. Ожидание ввода/вывода
9.3.12. Синхронный и асинхронный ввод/вывод
9.3.13. Дисковый ввод/вывод и ввод/вывод в приложении
9.4. Архитектура
9.4.1. Типы дисков
9.4.2. Интерфейсы
9.4.3. Типы хранилищ
9.4.4. Стек дискового ввода/вывода в операционной системе
9.5. Методология
9.5.1. Метод инструментов
9.5.2. Метод USE
9.5.3. Мониторинг производительности
9.5.4. Определение характеристик рабочей нагрузки
9.5.5. Анализ задержек
9.5.6. Статическая настройка производительности
9.5.7. Настройка кэширования
9.5.8. Управление ресурсами
9.5.9. Микробенчмаркинг
9.5.10. Масштабирование
9.6. Инструменты наблюдения
9.6.1. iostat
9.6.2. sar
9.6.3. PSI
9.6.4. pidstat
9.6.5. perf
9.6.6. biolatency
9.6.7. biosnoop
9.6.8. iotop, biotop
9.6.9. biostacks
9.6.10. blktrace
9.6.11. bpftrace
9.6.12. MegaCli
9.6.13. smartctl
9.6.14. Журналирование SCSI
9.6.15. Другие инструменты
9.7. Визуализация
9.7.1. Линейные диаграммы
9.7.2. Диаграммы рассеяния задержек
9.7.3. Тепловые карты задержек
9.7.4. Тепловые карты смещений
9.7.5. Тепловые карты потребления
9.8. Эксперименты
9.8.1. Ad hoc
9.8.2. Пользовательские генераторы нагрузки
9.8.3. Инструменты микробенчмаркинга
9.8.4. Пример произвольного чтения
9.8.5. ioping
9.8.6. fio
9.8.7. blkreplay
9.9. Настройка
9.9.1. Параметры операционной системы
9.9.2. Настраиваемые параметры дисковых устройств
9.9.3. Настраиваемые параметры контроллера диска
9.10. Упражнения
9.11. Ссылки
Глава 10. Сеть
10.1. Терминология
10.2. Модели
10.2.1. Сетевой интерфейс
10.2.2. Контроллер
10.2.3. Стек протоколов
10.3. Основные понятия
10.3.1. Сети и маршрутизация
10.3.2. Протоколы
10.3.3. Инкапсуляция
10.3.4. Размер пакета
10.3.5. Задержка
10.3.6. Буферизация
10.3.7. Очередь запросов на подключение
10.3.8. Согласование интерфейса
10.3.9. Предотвращение перегрузки
10.3.10. Потребление
10.3.11. Локальные подключения
10.4. Архитектура
10.4.1. Протоколы
10.4.2. Оборудование
10.4.3. Программное обеспечение
10.5. Методология
10.5.1. Метод инструментов
10.5.2. Метод USE
10.5.3. Определение характеристик рабочей нагрузки
10.5.4. Анализ задержек
10.5.5. Мониторинг производительности
10.5.6. Перехват пакетов
10.5.7. Анализ TCP
10.5.8. Статическая настройка производительности
10.5.9. Управление ресурсами
10.5.10. Микробенчмаркинг
10.6. Инструменты наблюдения
10.6.1. ss
10.6.2. ip
10.6.3. ifconfig
10.6.4. nstat
10.6.5 netstat
10.6.6. sar
10.6.7. nicstat
10.6.8. ethtool
10.6.9. tcplife
10.6.10. tcptop
10.6.11. tcpretrans
10.6.12. bpftrace
10.6.13. tcpdump
10.6.14. Wireshark
10.6.15. Другие инструменты
10.7. Эксперименты
10.7.1. ping
10.7.2. traceroute
10.7.3. pathchar
10.7.4. iperf
10.7.5. netperf
10.7.6. tc
10.7.7. Другие инструменты
10.8. Настройка
10.8.1. Общесистемные
10.8.2. Параметры сокетов
10.8.3. Конфигурация
10.9. Упражнения
10.10. Ссылки
Глава 11. Облачные вычисления
11.1. Основы
11.1.1. Типы экземпляров
11.1.2. Масштабируемая архитектура
11.1.3. Планирование мощности
11.1.4. Хранилище
11.1.5. Мультиарендность
11.1.6. Оркестровка (Kubernetes)
11.2. Виртуализация оборудования
11.2.1. Реализация
11.2.2. Оверхед
11.2.3. Управление ресурсами
11.2.4. Наблюдаемость
11.3. Виртуализация операционной системы
11.3.1. Реализация
11.3.2. Оверхед
11.3.3. Управление ресурсами
11.3.4. Наблюдаемость
11.4. Легковесная виртуализация
11.4.1. Реализация
11.4.2. Оверхед
11.4.3. Управление ресурсами
11.4.4. Наблюдаемость
11.5. Другие типы виртуализации
11.6. Сравнение
11.7. Упражнения
11.8. Ссылки
Глава 12. Бенчмаркинг
12.1. Основы
12.1.1. Причины
12.1.2. Эффективный бенчмаркинг
12.1.3. Проблемы бенчмаркинга
12.2. Типы бенчмаркинга
12.2.1. Микробенчмаркинг
12.2.2. Моделирование
12.2.3. Воспроизведение
12.2.4. Отраслевые стандарты
12.3. Методология
12.3.1 Пассивный бенчмаркинг
12.3.2. Активный бенчмаркинг
12.3.3. Профилирование процессора
12.3.4. Метод USE
12.3.5. Определение характеристик рабочей нагрузки
12.3.6. Собственный бенчмаркинг
12.3.7. Пиковая нагрузка
12.3.8. Проверка правильности
12.3.9. Статистический анализ
12.3.10. Чек-лист бенчмаркинга
12.4. Вопросы о бенчмаркинге
12.5. Упражнения
12.6. Ссылки
Глава 13. perf
13.1. Обзор подкоманд
13.2. Однострочные сценарии
13.3. События perf
13.4. Аппаратные события
13.4.1. Дискретная выборка
13.5. Программные события
13.6. События точек трассировки
13.7. События зондов
13.7.1. kprobes
13.7.2. uprobes
13.7.3. USDT
13.8. perf stat
13.8.1. Параметры
13.8.2. Интервальные статистики
13.8.3. Баланс между процессорами
13.8.4. Фильтрация событий
13.8.5. Теневые статистики
13.9. perf record
13.9.1. Параметры
13.9.2. Профилирование процессора
13.9.3. Обход стека
13.10. perf report
13.10.1. TUI
13.10.2. STDIO
13.11. perf script
13.11.1. Флейм-графики
13.11.2. Сценарии обработки трассировок
13.12. perf trace
13.12.1. Версии ядра
13.13. Другие команды
13.14. Документация perf
13.15. Ссылки
Глава 14. Ftrace
14.1. Обзор возможностей
14.2. tracefs (/sys)
14.2.1. Содержимое tracefs
14.3. Профилировщик функций
14.4. Трассировщик function
14.4.1. Использование файла trace
14.4.2. Использование файла trace_pipe
14.4.3. Параметры
14.5. Точки трассировки
14.5.1. Фильтрация
14.5.2. Триггеры
14.6. Зонды kprobes
14.6.1. Трассировка событий
14.6.2. Аргументы
14.6.3. Возвращаемые значения
14.6.4. Фильтры и триггеры
14.6.5. Профилировщик kprobe
14.7. Зонды uprobes
14.7.1. Трассировка событий
14.7.2. Аргументы и возвращаемые значения
14.7.3. Фильтры и триггеры
14.7.4. Профилировщик uprobe
14.8. Трассировщик function_graph
14.8.1. Трассировка графа
14.8.2. Параметры
14.9. Трассировщик hwlat
14.10. Триггеры hist
14.10.1. Гистограмма с единственным ключом
14.10.2. Поля
14.10.3. Модификаторы
14.10.4. Фильтры PID
14.10.5. Гистограмма с несколькими ключами
14.10.6 Трассировки стека в роли ключей
14.10.7. Синтетические события
14.11. trace-cmd
14.11.1. Обзор подкоманд
14.11.2. Однострочные сценарии для trace-cmd
14.11.3. Сравнение trace-cmd и perf(1)
14.11.4. trace-cmd function_graph
14.11.5. KernelShark
14.11.6. Документация для trace-cmd
14.12. perf ftrace
14.13. perf-tools
14.13.1. Покрытие инструментами
14.13.2. Специализированные инструменты
14.13.3. Многоцелевые инструменты
14.13.4. Однострочные сценарии для perf-tools
14.13.5. Пример
14.13.6. Сравнение perf-tools и BCC/BPF
14.13.7. Документация
14.14. Документация для Ftrace
14.15. Ссылки
Глава 15. BPF
15.1. BCC
15.1.1. Установка
15.1.2. Покрытие инструментами
15.1.3. Специализированные инструменты
15.1.4. Многоцелевые инструменты
15.1.5. Однострочные сценарии
15.1.6. Пример многоцелевого инструмента
15.1.7. Сравнение BCC и bpftrace
15.1.8. Документация
15.2. bpftrace
15.2.1. Установка
15.2.2. Инструменты
15.2.3. Однострочные сценарии
15.2.4. Программирование
15.2.5. Справочник
15.2.6. Документация
15.3. Ссылки
Глава 16. Пример из практики
16.1. Необъяснимый выигрыш
16.1.1. Постановка задачи
16.1.2. Стратегия анализа
16.1.3. Статистики
16.1.4. Конфигурация
16.1.5. Счетчики PMC
16.1.6. Программные события
16.1.7. Трассировка
16.1.8. Заключение
16.2. Дополнительная информация
16.3. Ссылки
Приложение A. Метод USE: Linux
Приложение B. Краткий справочник по sar
Приложение C. Однострочные сценарии для bpftrace
Приложение D. Решения некоторых упражнений
Приложение E. Производительность систем, кто есть кто
Глоссарий
Recommend Papers

Производительность систем
 9785446118182, 9780136820154

  • 0 0 0
  • Like this paper and download? You can publish your own PDF file online for free in a few minutes! Sign Up
File loading please wait...
Citation preview

Производительность систем Âòîðîå èçäàíèå

Áðåíäàí Ãðåãã

2023

ББК 32.973.2-018.2 УДК 004.451 Г79



Грегг Брендан

Г79 Производительность систем. — СПб.: Питер, 2023. — 992 с.: ил. — (Серия «Для профессионалов»).

ISBN 978-5-4461-1818-2

Книга посвящена концепциям, стратегиям, инструментам и настройке операционных систем и приложений на примере систем на базе Linux. Понимание этих инструментов и методов критически важно при разработке современного ПО. Применение стратегий, изложенных в обновленном и переработанном издании, позволит перформанс-инженерам улучшить взаимодействие с конечными пользователями и снизить затраты, особенно для облачных сред. Брендан Грегг — эксперт в области производительности систем и автор нескольких бестселлеров — лаконично, но емко излагает наиболее важные сведения о работе операционных систем, оборудования и приложений, которые позволят специалистам быстро добиться результатов, даже если раньше они никогда не занимались анализом производительности. Далее автор дает детальные объяснения по применению современных инструментов и методов, включая расширенный BPF, и показывает, как добиться максимальной эффективности ваших систем в облачных, веб- и крупных корпоративных средах.

16+ (В соответствии с Федеральным законом от 29 декабря 2010 г. № 436-ФЗ.)

ББК 32.973.2-018.2 УДК 004.451

Права на издание получены по соглашению с Pearson Education, Inc. Все права защищены. Никакая часть данной книги не может быть воспроизведена в какой бы то ни было форме без письменного разрешения владельцев авторских прав. Информация, содержащаяся в данной книге, получена из источников, рассматриваемых издательством как надежные. Тем не менее, имея в виду возможные человеческие или технические ошибки, издательство не может гарантировать абсолютную точность и полноту приводимых сведений и не несет ответственности за возможные ошибки, связанные с использованием книги. В книге возможны упоминания организаций, деятельность которых запрещена на территории Российской Федерации, таких как Meta Platforms Inc., Facebook, Instagram и др. Издательство не несет ответственности за доступность материалов, ссылки на которые вы можете найти в этой книге. На момент подготовки книги к изданию все ссылки на интернет-ресурсы были действующими.

ISBN 978-0136820154 англ. ISBN 978-5-4461-1818-2

Authorized translation from the English language edition, entitled Systems Performance, 2nd Edition by Brendan Gregg, published by Pearson Education, Inc, publishing as Addison Wesley Professional. © 2021 Pearson Education, Inc. © Перевод на русский язык ООО «Прогресс книга», 2023 © Издание на русском языке, оформление ООО «Прогресс книга», 2023 © Серия «Для профессионалов», 2023

КРАТКОЕ СОДЕРЖАНИЕ

Предисловие................................................................................................................. 30 Благодарности.............................................................................................................. 38 Об авторе....................................................................................................................... 41 Глава 1. Введение......................................................................................................... 42 Глава 2. Методологии................................................................................................... 66 Глава 3. Операционные системы............................................................................... 146 Глава 4. Инструменты наблюдения........................................................................... 195 Глава 5. Приложения.................................................................................................. 241 Глава 6. Процессоры................................................................................................... 299 Глава 7. Память........................................................................................................... 396 Глава 8. Файловые системы...................................................................................... 460 Глава 9. Диски............................................................................................................. 533 Глава 10. Сеть.............................................................................................................. 622 Глава 11. Облачные вычисления............................................................................... 714 Глава 12. Бенчмаркинг............................................................................................... 786 Глава 13. perf............................................................................................................... 821 Глава 14. Ftrace........................................................................................................... 857

6  Краткое содержание

Глава 15. BPF............................................................................................................... 904 Глава 16. Пример из практики.................................................................................. 939 Приложение A. Метод USE: Linux ............................................................................ 950 Приложение B. Краткий справочник по sar............................................................ 957 Приложение C. Однострочные сценарии для bpftrace........................................... 959 Приложение D. Решения некоторых упражнений.................................................. 966 Приложение E. Производительность систем, кто есть кто..................................... 969 Глоссарий.................................................................................................................... 974

ОГЛАВЛЕНИЕ

Предисловие................................................................................................................. 30 Об этом издании........................................................................................................................... 30 Об этой книге................................................................................................................................ 31

Благодарности............................................................................................................... 38 Об авторе....................................................................................................................... 41 От издательства............................................................................................................................ 41

Глава 1. Введение......................................................................................................... 42 1.1. Производительность системы.......................................................................................... 42 1.2. Роли........................................................................................................................................... 43 1.3. Действия.................................................................................................................................. 44 1.4. Перспективы.......................................................................................................................... 46 1.5. Сложности оценки производительности..................................................................... 46 1.5.1. Субъективность................................................................................................... 47 1.5.2. Сложность.............................................................................................................. 47 1.5.3. Множественные причины................................................................................ 48 1.5.4. Множественные проблемы с производительностью.............................. 48 1.6. Задержка.................................................................................................................................. 49 1.7. Наблюдаемость..................................................................................................................... 50 1.7.1. Счетчики, статистики и метрики................................................................... 50 1.7.2. Профилирование................................................................................................. 52 1.7.3. Трассировка........................................................................................................... 54 1.8. Эксперименты....................................................................................................................... 56 1.9. Облачные вычисления....................................................................................................... 58 1.10. Методологии........................................................................................................................ 58 1.10.1. Анализ производительности Linux за 60 секунд.................................... 59

8  Оглавление 1.11. Практические примеры.................................................................................................... 60 1.11.1. Медленные диски.............................................................................................. 60 1.11.2. Изменение в программном обеспечении.................................................. 63 1.11.3. Дополнительное чтение.................................................................................. 65 1.12. Ссылки................................................................................................................................... 65

Глава 2. Методологии.................................................................................................... 66 2.1. Терминология........................................................................................................................ 67 2.2. Модели..................................................................................................................................... 68 2.2.1. Тестируемая система.......................................................................................... 68 2.2.2. Система массового обслуживания................................................................ 69 2.3. Основные понятия............................................................................................................... 69 2.3.1. Задержка................................................................................................................. 70 2.3.2. Шкалы времени................................................................................................... 71 2.3.3. Компромиссы ...................................................................................................... 72 2.3.4. Настройка производительности.................................................................... 73 2.3.5. Целесообразность................................................................................................ 74 2.3.6. Когда лучше остановить анализ..................................................................... 75 2.3.7. Рекомендации действительны на данный момент времени................. 76 2.3.8. Нагрузка и архитектура.................................................................................... 77 2.3.9. Масштабируемость............................................................................................. 78 2.3.10. Метрики............................................................................................................... 80 2.3.11. Потребление........................................................................................................ 81 2.3.12. Насыщенность.................................................................................................... 82 2.3.13. Профилирование............................................................................................... 83 2.3.14. Кэширование...................................................................................................... 83 2.3.15. Известные неизвестные.................................................................................. 86 2.4. Точки зрения.......................................................................................................................... 86 2.4.1. Анализ ресурсов................................................................................................... 86 2.4.2. Анализ рабочей нагрузки................................................................................. 88 2.5. Методология.......................................................................................................................... 89 2.5.1. Антиметодология «Уличный фонарь»......................................................... 91 2.5.2. Антиметодология «Случайное изменение»............................................... 92 2.5.3. Антиметодология «Виноват кто-то другой».............................................. 92

Оглавление  9 2.5.4. Специальный чек-лист...................................................................................... 93 2.5.5. Формулировка проблемы................................................................................. 94 2.5.6. Научный метод..................................................................................................... 94 2.5.7. Цикл диагностики............................................................................................... 96 2.5.8. Метод инструментов.......................................................................................... 96 2.5.9. Метод USE............................................................................................................. 97 2.5.10. Метод RED........................................................................................................ 105 2.5.11. Определение характеристик рабочей нагрузки................................... 105 2.5.12. Анализ с увеличением детализации......................................................... 107 2.5.13. Анализ задержек.............................................................................................. 109 2.5.14. Метод R.............................................................................................................. 109 2.5.15. Трассировка событий..................................................................................... 110 2.5.16. Базовые статистики........................................................................................ 112 2.5.17. Статическая настройка производительности....................................... 113 2.5.18. Настройка кэширования.............................................................................. 113 2.5.19. Микробенчмаркинг производительности.............................................. 114 2.5.20. Мантры производительности..................................................................... 115 2.6. Моделирование................................................................................................................... 116 2.6.1. Корпоративные и облачные среды.............................................................. 117 2.6.2. Визуальная идентификация.......................................................................... 117 2.6.3. Закон Амдала...................................................................................................... 119 2.6.4. Универсальный закон масштабируемости............................................... 120 2.6.5. Теория массового обслуживания................................................................. 121 2.7. Планирование емкости..................................................................................................... 125 2.7.1. Пределы ресурсов.............................................................................................. 125 2.7.2. Факторный анализ............................................................................................ 127 2.7.3. Решения масштабирования........................................................................... 128 2.8. Статистики........................................................................................................................... 129 2.8.1. Количественная оценка прироста производительности..................... 129 2.8.2. Усреднение........................................................................................................... 131 2.8.3. Стандартное отклонение, процентили, медиана.................................... 132 2.8.4. Коэффициент вариации.................................................................................. 133 2.8.5. Мультимодальные распределения.............................................................. 133 2.8.6. Выбросы................................................................................................................ 134

10  Оглавление 2.9. Мониторинг.......................................................................................................................... 135 2.9.1. Временные закономерности.......................................................................... 135 2.9.2. Инструменты мониторинга........................................................................... 137 2.9.3. Сводная статистика, накопленная с момента загрузки....................... 137 2.10. Визуализация.................................................................................................................... 137 2.10.1. Линейная диаграмма...................................................................................... 137 2.10.2. Диаграммы рассеяния................................................................................... 139 2.10.3. Тепловые карты............................................................................................... 140 2.10.4. Временная шкала............................................................................................ 141 2.10.5. График поверхности....................................................................................... 142 2.10.6. Инструменты визуализации....................................................................... 143 2.11. Упражнения....................................................................................................................... 144 2.12. Ссылки................................................................................................................................. 144

Глава 3. Операционные системы..................................................................................146 3.1. Терминология...................................................................................................................... 147 3.2. Основы................................................................................................................................... 148 3.2.1. Ядро........................................................................................................................ 148 3.2.2. Ядро и пользовательские режимы............................................................... 150 3.2.3. Системные вызовы........................................................................................... 152 3.2.4. Прерывания......................................................................................................... 154 3.2.5. Часы и бездействие........................................................................................... 158 3.2.6. Процессы.............................................................................................................. 159 3.2.7. Стеки...................................................................................................................... 162 3.2.8. Виртуальная память......................................................................................... 164 3.2.9. Планировщики................................................................................................... 165 3.2.10. Файловая система........................................................................................... 167 3.2.11. Кэширование.................................................................................................... 169 3.2.12. Сеть...................................................................................................................... 170 3.2.13. Драйверы устройств....................................................................................... 171 3.2.14. Многопроцессорность................................................................................... 171 3.2.15. Вытеснение........................................................................................................ 172 3.2.16. Управление ресурсами.................................................................................. 172 3.2.17. Наблюдаемость................................................................................................ 173

Оглавление  11 3.3. Ядра......................................................................................................................................... 173 3.3.1. Unix........................................................................................................................ 174 3.3.2. BSD......................................................................................................................... 175 3.3.3. Solaris..................................................................................................................... 176 3.4. Linux........................................................................................................................................ 177 3.4.1. Новые разработки в ядре Linux.................................................................... 178 3.4.2. systemd.................................................................................................................. 184 3.4.3. KPTI (Meltdown).............................................................................................. 185 3.4.4. Расширенный BPF............................................................................................ 186 3.5. Другие темы......................................................................................................................... 187 3.5.1. Ядра PGO............................................................................................................. 187 3.5.2. Одноцелевые ядра............................................................................................. 188 3.5.3. Микроядра и гибридные ядра....................................................................... 188 3.5.4. Распределенные операционные системы.................................................. 189 3.6. Сравнение ядер................................................................................................................... 189 3.7. Упражнения.......................................................................................................................... 190 3.8. Ссылки................................................................................................................................... 191 3.8.1. Дополнительное чтение.................................................................................. 193

Глава 4. Инструменты наблюдения..............................................................................195 4.1. Покрытие инструментами............................................................................................... 196 4.1.1. Инструменты статического анализа производительности................. 197 4.1.2. Инструменты анализа кризисных ситуаций........................................... 198 4.2. Типы инструментов........................................................................................................... 199 4.2.1. Фиксированные счетчики.............................................................................. 200 4.2.2. Профилирование............................................................................................... 202 4.2.3. Трассировка......................................................................................................... 203 4.2.4. Мониторинг......................................................................................................... 204 4.3. Источники информации.................................................................................................. 206 4.3.1. /proc....................................................................................................................... 208 4.3.2. /sys.......................................................................................................................... 212 4.3.3. Учет задержек..................................................................................................... 213 4.3.4. netlink.................................................................................................................... 214 4.3.5. Точки трассировки............................................................................................ 214

12  Оглавление 4.3.6. kprobes................................................................................................................... 219 4.3.7. uprobes................................................................................................................... 222 4.3.8. USDT..................................................................................................................... 224 4.3.9. Аппаратные счетчики (PMC)....................................................................... 225 4.3.10. Другие источники информации для наблюдения............................... 229 4.4. sar............................................................................................................................................. 231 4.4.1. Область покрытия sar(1)................................................................................ 232 4.4.2. Мониторинг с sar(1)......................................................................................... 232 4.4.3. Вывод оперативной информации с помощью sar(1)............................ 236 4.4.4. Документация с описанием sar(1)............................................................... 236 4.5. Инструменты трассировки............................................................................................. 237 4.6. Наблюдение за наблюдаемостью.................................................................................. 237 4.7. Упражнения.......................................................................................................................... 239 4.8. Ссылки................................................................................................................................... 240

Глава 5. Приложения...................................................................................................241 5.1. Основы приложений......................................................................................................... 242 5.1.1. Цель........................................................................................................................ 243 5.1.2. Оптимизация общего случая......................................................................... 245 5.1.3. Наблюдаемость.................................................................................................. 245 5.1.4. Нотация «О-большое»..................................................................................... 245 5.2. Методы повышения производительности приложений....................................... 247 5.2.1. Выбор размеров блоков ввода/вывода...................................................... 247 5.2.2. Кэширование....................................................................................................... 248 5.2.3. Буферизация....................................................................................................... 248 5.2.4. Опрос..................................................................................................................... 248 5.2.5. Конкурентность и параллелизм................................................................... 249 5.2.6. Неблокирующий ввод/вывод....................................................................... 254 5.2.7. Привязка к процессору.................................................................................... 255 5.2.8. Мантры производительности....................................................................... 255 5.3. Языки программирования............................................................................................... 256 5.3.1. Компилируемые языки................................................................................... 256 5.3.2. Интерпретируемые языки.............................................................................. 258 5.3.3. Виртуальные машины...................................................................................... 259 5.3.4. Сборка мусора.................................................................................................... 259

Оглавление  13 5.4. Методология........................................................................................................................ 260 5.4.1. Профилирование процессора........................................................................ 261 5.4.2. Анализ времени ожидания вне процессора.............................................. 264 5.4.3. Анализ системных вызовов........................................................................... 268 5.4.4. Метод USE........................................................................................................... 269 5.4.5. Анализ состояния потока............................................................................... 270 5.4.6. Анализ блокировок........................................................................................... 275 5.4.7. Настройка статической производительности......................................... 276 5.4.8. Распределенная трассировка........................................................................ 277 5.5. Инструменты наблюдения.............................................................................................. 277 5.5.1. perf.......................................................................................................................... 278 5.5.2. profile...................................................................................................................... 281 5.5.3. offcputime.............................................................................................................. 282 5.5.4. strace....................................................................................................................... 284 5.5.5. execsnoop............................................................................................................... 286 5.5.6. syscount................................................................................................................. 287 5.5.7. bpftrace.................................................................................................................. 288 5.6. Проблемы.............................................................................................................................. 292 5.6.1. Отсутствие символов....................................................................................... 293 5.6.2. Отсутствие стеков............................................................................................. 294 5.7. Упражнения.......................................................................................................................... 295 5.8. Ссылки................................................................................................................................... 297

Глава 6. Процессоры....................................................................................................299 6.1. Терминология...................................................................................................................... 300 6.2. Модели................................................................................................................................... 301 6.2.1. Архитектура процессора................................................................................. 301 6.2.2. Кэш-память процессора.................................................................................. 302 6.2.3. Очереди на выполнение.................................................................................. 303 6.3. Понятия................................................................................................................................. 303 6.3.1. Тактовая частота................................................................................................ 304 6.3.2. Инструкции......................................................................................................... 304 6.3.3. Вычислительный конвейер............................................................................ 305 6.3.4. Ширина инструкций........................................................................................ 305 6.3.5. Размеры инструкций........................................................................................ 306

14  Оглавление 6.3.6. SMT........................................................................................................................ 306 6.3.7. IPC, CPI................................................................................................................ 306 6.3.8. Потребление........................................................................................................ 307 6.3.9. Время в режиме пользователя/ядра........................................................... 308 6.3.10. Насыщенность.................................................................................................. 308 6.3.11. Вытеснение........................................................................................................ 309 6.3.12. Инверсия приоритета.................................................................................... 309 6.3.13. Несколько процессов, несколько потоков.............................................. 310 6.3.14. Размер слова..................................................................................................... 312 6.3.15. Оптимизации компилятора......................................................................... 312 6.4. Архитектура......................................................................................................................... 312 6.4.1. Аппаратное обеспечение................................................................................. 312 6.4.2. Программное обеспечение............................................................................. 326 6.5. Методология........................................................................................................................ 330 6.5.1. Метод инструментов........................................................................................ 331 6.5.2. Метод USE........................................................................................................... 332 6.5.3. Определение характеристик рабочей нагрузки...................................... 333 6.5.4. Профилирование............................................................................................... 335 6.5.5. Анализ тактов..................................................................................................... 338 6.5.6. Мониторинг производительности............................................................... 339 6.5.7. Статическая настройка производительности.......................................... 340 6.5.8. Настройка приоритетов.................................................................................. 340 6.5.9. Управление ресурсами..................................................................................... 341 6.5.10. Привязка к процессору................................................................................. 342 6.5.11. Микробенчмаркинг........................................................................................ 342 6.6. Инструменты наблюдения.............................................................................................. 343 6.6.1. uptime..................................................................................................................... 344 6.6.2. vmstat..................................................................................................................... 347 6.6.3. mpstat..................................................................................................................... 348 6.6.4. sar............................................................................................................................ 349 6.6.5. ps.............................................................................................................................. 350 6.6.6. top............................................................................................................................ 351 6.6.7. pidstat..................................................................................................................... 352 6.6.8. time, ptime............................................................................................................. 353

Оглавление  15 6.6.9. turbostat................................................................................................................ 354 6.6.10. showboost............................................................................................................ 355 6.6.11. pmcarch............................................................................................................... 355 6.6.12. tlbstat................................................................................................................... 356 6.6.13. perf........................................................................................................................ 357 6.6.14. profile................................................................................................................... 367 6.6.15. cpudist.................................................................................................................. 369 6.6.16. runqlat.................................................................................................................. 370 6.6.17. runqlen................................................................................................................. 371 6.6.18. softirqs.................................................................................................................. 372 6.6.19. hardirqs................................................................................................................ 373 6.6.20. bpftrace................................................................................................................ 373 6.6.21. Другие инструменты...................................................................................... 376 6.7. Методы визуализации...................................................................................................... 379 6.7.1. Тепловая карта потребления......................................................................... 379 6.7.2. Тепловая карта с субсекундным смещением........................................... 380 6.7.3. Флейм-графики................................................................................................. 381 6.7.4. FlameScope........................................................................................................... 384 6.8. Эксперименты..................................................................................................................... 385 6.8.1. Ad hoc..................................................................................................................... 386 6.8.2. SysBench............................................................................................................... 386 6.9. Настройка............................................................................................................................. 387 6.9.1. Параметры компилятора................................................................................ 387 6.9.2. Приоритет и класс планирования............................................................... 387 6.9.3. Параметры планировщика............................................................................. 388 6.9.4. Режимы масштабирования частоты........................................................... 389 6.9.5. Состояния энергопотребления..................................................................... 390 6.9.6. Привязка к процессору.................................................................................... 390 6.9.7. Исключительные процессорные наборы.................................................. 390 6.9.8. Управление ресурсами..................................................................................... 391 6.9.9. Параметры безопасной загрузки.................................................................. 391 6.9.10. Параметры процессора (настройка BIOS)............................................. 392 6.10. Упражнения....................................................................................................................... 392 6.11. Ссылки................................................................................................................................. 393

16  Оглавление

Глава 7. Память............................................................................................................396 7.1. Терминология...................................................................................................................... 397 7.2. Основные понятия............................................................................................................. 398 7.2.1. Виртуальная память......................................................................................... 398 7.2.2. Подкачка страниц.............................................................................................. 399 7.2.3. Подкачка страниц по требованию............................................................... 401 7.2.4. Чрезмерное выделение памяти..................................................................... 402 7.2.5. Подкачка процессов......................................................................................... 403 7.2.6. Использование кэша файловой системы.................................................. 403 7.2.7. Потребление и насыщение............................................................................. 403 7.2.8. Распределители.................................................................................................. 404 7.2.9. Разделяемая память.......................................................................................... 404 7.2.10. Размер рабочего набора................................................................................ 405 7.2.11. Размер слова..................................................................................................... 405 7.3. Архитектура......................................................................................................................... 406 7.3.1. Аппаратное обеспечение................................................................................. 406 7.3.2. Программное обеспечение............................................................................. 411 7.3.3. Адресное пространство процесса................................................................. 416 7.4. Методология........................................................................................................................ 421 7.4.1. Метод инструментов........................................................................................ 422 7.4.2. Метод USE........................................................................................................... 422 7.4.3. Определение характеристик потребления памяти................................ 424 7.4.4. Анализ тактов..................................................................................................... 425 7.4.5. Мониторинг производительности............................................................... 425 7.4.6. Выявление утечек.............................................................................................. 426 7.4.7. Статическая настройка производительности.......................................... 427 7.4.8. Управление ресурсами..................................................................................... 427 7.4.9. Микробенчмаркинг.......................................................................................... 428 7.4.10. Уменьшение потребления памяти............................................................. 428 7.5. Инструменты наблюдения.............................................................................................. 428 7.5.1. vmstat..................................................................................................................... 429 7.5.2. PSI........................................................................................................................... 431 7.5.3. swapon.................................................................................................................... 431 7.5.4. sar............................................................................................................................ 431

Оглавление  17 7.5.5. slabtop.................................................................................................................... 434 7.5.6. numastat................................................................................................................ 435 7.5.7. ps.............................................................................................................................. 436 7.5.8. top............................................................................................................................ 437 7.5.9. pmap........................................................................................................................ 438 7.5.10. perf........................................................................................................................ 439 7.5.11. drsnoop................................................................................................................ 443 7.5.12. wss......................................................................................................................... 443 7.5.13. bpftrace................................................................................................................ 444 7.5.14. Другие инструменты...................................................................................... 449 7.6. Настройка............................................................................................................................. 451 7.6.1. Настраиваемые параметры............................................................................ 452 7.6.2. Несколько размеров страниц........................................................................ 454 7.6.3. Распределители.................................................................................................. 455 7.6.4. Привязка NUMA............................................................................................... 455 7.6.5. Управление ресурсами..................................................................................... 456 7.7. Упражнения.......................................................................................................................... 456 7.8. Ссылки................................................................................................................................... 458

Глава 8. Файловые системы.........................................................................................460 8.1. Терминология...................................................................................................................... 461 8.2. Модели................................................................................................................................... 462 8.2.1. Интерфейсы файловых систем..................................................................... 462 8.2.2. Кэш файловой системы................................................................................... 463 8.2.3. Кэш второго уровня.......................................................................................... 464 8.3. Основные понятия............................................................................................................. 464 8.3.1. Задержки в файловой системе...................................................................... 464 8.3.2. Кэширование....................................................................................................... 465 8.3.3. Произвольный и последовательный ввод/вывод.................................. 465 8.3.4. Предварительная выборка............................................................................. 466 8.3.5. Упреждающее чтение....................................................................................... 467 8.3.6. Кэширование с отложенной записью......................................................... 468 8.3.7. Синхронная запись........................................................................................... 468 8.3.8. Прямой и низкоуровневый ввод/вывод.................................................... 469 8.3.9. Неблокирующий ввод/вывод....................................................................... 470

18  Оглавление 8.3.10. Файлы, отображаемые в память................................................................ 470 8.3.11. Метаданные....................................................................................................... 471 8.3.12. Логический и физический ввод/вывод................................................... 472 8.3.13. Неравноценность операций......................................................................... 474 8.3.14. Специальные файловые системы.............................................................. 475 8.3.15. Доступ к отметкам времени......................................................................... 475 8.3.16. Емкость............................................................................................................... 476 8.4. Архитектура......................................................................................................................... 476 8.4.1. Стек ввода-вывода файловой системы...................................................... 476 8.4.2. VFS......................................................................................................................... 477 8.4.3. Кэши файловой системы................................................................................ 478 8.4.4. Особенности файловых систем.................................................................... 481 8.4.5. Типы файловых систем................................................................................... 482 8.4.6. Тома и пулы......................................................................................................... 490 8.5. Методология........................................................................................................................ 491 8.5.1. Анализ дисков..................................................................................................... 492 8.5.2. Анализ задержек................................................................................................ 492 8.5.3. Определение характеристик рабочей нагрузки...................................... 495 8.5.4. Мониторинг производительности............................................................... 497 8.5.5. Статическая настройка производительности.......................................... 498 8.5.6. Настройка кэша................................................................................................. 498 8.5.7. Разделение рабочей нагрузки....................................................................... 498 8.5.8. Микробенчмаркинг.......................................................................................... 499 8.6. Инструменты наблюдения.............................................................................................. 501 8.6.1. mount...................................................................................................................... 502 8.6.2. free........................................................................................................................... 502 8.6.3. top............................................................................................................................ 503 8.6.4. vmstat..................................................................................................................... 503 8.6.5. sar............................................................................................................................ 503 8.6.6. slabtop.................................................................................................................... 504 8.6.7. strace....................................................................................................................... 505 8.6.8. fatrace..................................................................................................................... 505 8.6.9. LatencyTOP......................................................................................................... 506 8.6.10. opensnoop........................................................................................................... 507 8.6.11. filetop................................................................................................................... 508

Оглавление  19 8.6.12. cachestat.............................................................................................................. 509 8.6.13. ext4dist (xfs, zfs, btrfs, nfs)............................................................................. 510 8.6.14. ext4slower (xfs, zfs, btrfs, nfs)........................................................................ 511 8.6.15. bpftrace................................................................................................................ 512 8.6.17. Другие инструменты...................................................................................... 519 8.6.18. Визуализация................................................................................................... 520 8.7. Эксперименты..................................................................................................................... 521 8.7.1. Ad hoc..................................................................................................................... 522 8.7.2. Инструменты микробенчмаркинга............................................................. 522 8.7.3. Очистка кэша...................................................................................................... 524 8.8. Настройка............................................................................................................................. 525 8.8.1. Функции............................................................................................................... 525 8.8.2. ext4.......................................................................................................................... 526 8.8.3. ZFS.......................................................................................................................... 529 8.9. Упражнения.......................................................................................................................... 530 8.10. Ссылки................................................................................................................................. 532

Глава 9. Диски..............................................................................................................533 9.1. Терминология...................................................................................................................... 534 9.2. Модели................................................................................................................................... 535 9.2.1. Простой диск....................................................................................................... 535 9.2.2. Кэширующие диски.......................................................................................... 536 9.2.3. Контроллер.......................................................................................................... 536 9.3. Основные понятия............................................................................................................. 537 9.3.1. Измерение времени.......................................................................................... 537 9.3.2. Масштаб времени.............................................................................................. 540 9.3.3. Кэширование....................................................................................................... 541 9.3.4. Произвольный и последовательный ввод/вывод.................................. 542 9.3.5. Соотношение операций чтения и записи.................................................. 543 9.3.6. Размер ввода/вывода....................................................................................... 543 9.3.7. Несопоставимость количества операций в секунду.............................. 544 9.3.8. Дисковые команды, не связанные с передачей данных....................... 544 9.3.9. Потребление........................................................................................................ 545 9.3.10. Насыщенность.................................................................................................. 546 9.3.11. Ожидание ввода/вывода.............................................................................. 546

20  Оглавление 9.3.12. Синхронный и асинхронный ввод/вывод.............................................. 547 9.3.13. Дисковый ввод/вывод и ввод/вывод в приложении......................... 548 9.4. Архитектура......................................................................................................................... 548 9.4.1. Типы дисков........................................................................................................ 548 9.4.2. Интерфейсы........................................................................................................ 557 9.4.3. Типы хранилищ.................................................................................................. 559 9.4.4. Стек дискового ввода/вывода в операционной системе..................... 563 9.5. Методология........................................................................................................................ 567 9.5.1. Метод инструментов........................................................................................ 568 9.5.2. Метод USE........................................................................................................... 568 9.5.3. Мониторинг производительности............................................................... 570 9.5.4. Определение характеристик рабочей нагрузки...................................... 571 9.5.5. Анализ задержек................................................................................................ 573 9.5.6. Статическая настройка производительности.......................................... 574 9.5.7. Настройка кэширования................................................................................. 575 9.5.8. Управление ресурсами..................................................................................... 576 9.5.9. Микробенчмаркинг.......................................................................................... 576 9.5.10. Масштабирование........................................................................................... 577 9.6. Инструменты наблюдения.............................................................................................. 578 9.6.1. iostat....................................................................................................................... 579 9.6.2. sar............................................................................................................................ 584 9.6.3. PSI........................................................................................................................... 585 9.6.4. pidstat..................................................................................................................... 586 9.6.5. perf.......................................................................................................................... 587 9.6.6. biolatency.............................................................................................................. 590 9.6.7. biosnoop................................................................................................................. 592 9.6.8. iotop, biotop.......................................................................................................... 594 9.6.9. biostacks................................................................................................................ 596 9.6.10. blktrace................................................................................................................ 597 9.6.11. bpftrace................................................................................................................ 601 9.6.12. MegaCli............................................................................................................... 605 9.6.13. smartctl................................................................................................................ 606 9.6.14. Журналирование SCSI.................................................................................. 607 9.6.15. Другие инструменты...................................................................................... 608

Оглавление  21 9.7. Визуализация....................................................................................................................... 609 9.7.1. Линейные диаграммы...................................................................................... 609 9.7.2. Диаграммы рассеяния задержек.................................................................. 609 9.7.3. Тепловые карты задержек............................................................................... 610 9.7.4. Тепловые карты смещений............................................................................. 611 9.7.5. Тепловые карты потребления....................................................................... 612 9.8. Эксперименты..................................................................................................................... 612 9.8.1. Ad hoc..................................................................................................................... 613 9.8.2. Пользовательские генераторы нагрузки................................................... 613 9.8.3. Инструменты микробенчмаркинга............................................................. 614 9.8.4. Пример произвольного чтения..................................................................... 614 9.8.5. ioping...................................................................................................................... 615 9.8.6. fio............................................................................................................................. 615 9.8.7. blkreplay................................................................................................................ 615 9.9. Настройка............................................................................................................................. 616 9.9.1. Параметры операционной системы............................................................ 616 9.9.2. Настраиваемые параметры дисковых устройств................................... 617 9.9.3. Настраиваемые параметры контроллера диска...................................... 618 9.10. Упражнения....................................................................................................................... 618 9.11. Ссылки................................................................................................................................. 619

Глава 10. Сеть...............................................................................................................622 10.1. Терминология.................................................................................................................... 623 10.2. Модели................................................................................................................................. 624 10.2.1. Сетевой интерфейс......................................................................................... 624 10.2.2. Контроллер........................................................................................................ 624 10.2.3. Стек протоколов.............................................................................................. 625 10.3. Основные понятия........................................................................................................... 626 10.3.1. Сети и маршрутизация.................................................................................. 626 10.3.2. Протоколы......................................................................................................... 627 10.3.3. Инкапсуляция.................................................................................................. 628 10.3.4. Размер пакета................................................................................................... 628 10.3.5. Задержка............................................................................................................ 629 10.3.6. Буферизация..................................................................................................... 631 10.3.7. Очередь запросов на подключение........................................................... 632

22  Оглавление 10.3.8. Согласование интерфейса............................................................................ 632 10.3.9. Предотвращение перегрузки....................................................................... 633 10.3.10. Потребление................................................................................................... 633 10.3.11. Локальные подключения........................................................................... 634 10.4. Архитектура....................................................................................................................... 635 10.4.1. Протоколы......................................................................................................... 635 10.4.2. Оборудование................................................................................................... 643 10.4.3. Программное обеспечение........................................................................... 646 10.5. Методология...................................................................................................................... 655 10.5.1. Метод инструментов...................................................................................... 655 10.5.2. Метод USE........................................................................................................ 656 10.5.3. Определение характеристик рабочей нагрузки................................... 657 10.5.4. Анализ задержек.............................................................................................. 659 10.5.5. Мониторинг производительности............................................................ 660 10.5.6. Перехват пакетов............................................................................................. 661 10.5.7. Анализ TCP....................................................................................................... 662 10.5.8. Статическая настройка производительности....................................... 663 10.5.9. Управление ресурсами.................................................................................. 664 10.5.10. Микробенчмаркинг...................................................................................... 665 10.6. Инструменты наблюдения............................................................................................ 665 10.6.1. ss............................................................................................................................ 666 10.6.2. ip............................................................................................................................ 668 10.6.3. ifconfig................................................................................................................. 670 10.6.4. nstat...................................................................................................................... 670 10.6.5. netstat.................................................................................................................. 671 10.6.6. sar.......................................................................................................................... 675 10.6.7. nicstat................................................................................................................... 678 10.6.8. ethtool.................................................................................................................. 679 10.6.9. tcplife.................................................................................................................... 680 10.6.10. tcptop................................................................................................................. 681 10.6.11. tcpretrans.......................................................................................................... 682 10.6.12. bpftrace.............................................................................................................. 683 10.6.13. tcpdump............................................................................................................ 691 10.6.14. Wireshark.......................................................................................................... 693 10.6.15. Другие инструменты.................................................................................... 694

Оглавление  23 10.7. Эксперименты................................................................................................................... 695 10.7.1. ping....................................................................................................................... 695 10.7.2. traceroute............................................................................................................ 696 10.7.3. pathchar............................................................................................................... 697 10.7.4. iperf....................................................................................................................... 698 10.7.5. netperf.................................................................................................................. 699 10.7.6. tc............................................................................................................................ 699 10.7.7. Другие инструменты...................................................................................... 700 10.8. Настройка........................................................................................................................... 701 10.8.1. Общесистемные............................................................................................... 701 10.8.2. Параметры сокетов......................................................................................... 707 10.8.3. Конфигурация.................................................................................................. 708 10.9. Упражнения....................................................................................................................... 708 10.10. Ссылки............................................................................................................................... 709

Глава 11. Облачные вычисления..................................................................................714 11.1. Основы................................................................................................................................. 715 11.1.1. Типы экземпляров.......................................................................................... 716 11.1.2. Масштабируемая архитектура................................................................... 717 11.1.3. Планирование мощности............................................................................. 718 11.1.4. Хранилище........................................................................................................ 721 11.1.5. Мультиарендность.......................................................................................... 722 11.1.6. Оркестровка (Kubernetes)........................................................................... 723 11.2. Виртуализация оборудования..................................................................................... 724 11.2.1. Реализация........................................................................................................ 726 11.2.2. Оверхед............................................................................................................... 727 11.2.3. Управление ресурсами.................................................................................. 734 11.2.4. Наблюдаемость................................................................................................ 738 11.3. Виртуализация операционной системы................................................................... 746 11.3.1. Реализация........................................................................................................ 748 11.3.2. Оверхед............................................................................................................... 751 11.3.3. Управление ресурсами.................................................................................. 755 11.3.4. Наблюдаемость................................................................................................ 760 11.4. Легковесная виртуализация......................................................................................... 775 11.4.1. Реализация........................................................................................................ 776

24  Оглавление 11.4.2. Оверхед............................................................................................................... 777 11.4.3. Управление ресурсами.................................................................................. 777 11.4.4. Наблюдаемость................................................................................................ 777 11.5. Другие типы виртуализации........................................................................................ 779 11.6. Сравнение........................................................................................................................... 780 11.7. Упражнения....................................................................................................................... 782 11.8. Ссылки................................................................................................................................. 783

Глава 12. Бенчмаркинг.................................................................................................786 12.1. Основы................................................................................................................................. 787 12.1.1. Причины............................................................................................................. 787 12.1.2. Эффективный бенчмаркинг........................................................................ 788 12.1.3. Проблемы бенчмаркинга.............................................................................. 790 12.2. Типы бенчмаркинга......................................................................................................... 798 12.2.1. Микробенчмаркинг........................................................................................ 799 12.2.2. Моделирование................................................................................................ 801 12.2.3. Воспроизведение............................................................................................. 802 12.2.4. Отраслевые стандарты.................................................................................. 803 12.3. Методология...................................................................................................................... 805 12.3.1. Пассивный бенчмаркинг.............................................................................. 805 12.3.2. Активный бенчмаркинг................................................................................ 806 12.3.3. Профилирование процессора..................................................................... 809 12.3.4. Метод USE........................................................................................................ 811 12.3.5. Определение характеристик рабочей нагрузки................................... 811 12.3.6. Собственный бенчмаркинг.......................................................................... 811 12.3.7. Пиковая нагрузка............................................................................................ 812 12.3.8. Проверка правильности................................................................................ 814 12.3.9. Статистический анализ................................................................................. 815 12.3.10. Чек-лист бенчмаркинга.............................................................................. 816 12.4. Вопросы о бенчмаркинге............................................................................................... 817 12.5. Упражнения....................................................................................................................... 819 12.6. Ссылки................................................................................................................................. 819

Глава 13. perf................................................................................................................821 13.1. Обзор подкоманд.............................................................................................................. 822 13.2. Однострочные сценарии................................................................................................ 824

Оглавление  25 13.3. События perf...................................................................................................................... 830 13.4. Аппаратные события....................................................................................................... 832 13.4.1. Дискретная выборка...................................................................................... 833 13.5. Программные события................................................................................................... 834 13.6. События точек трассировки......................................................................................... 835 13.7. События зондов................................................................................................................ 836 13.7.1. kprobes................................................................................................................. 836 13.7.2. uprobes................................................................................................................. 838 13.7.3. USDT................................................................................................................... 840 13.8. perf stat................................................................................................................................. 842 13.8.1. Параметры......................................................................................................... 843 13.8.2. Интервальные статистики........................................................................... 844 13.8.3. Баланс между процессорами . .................................................................... 844 13.8.4. Фильтрация событий..................................................................................... 844 13.8.5. Теневые статистики........................................................................................ 845 13.9. perf record............................................................................................................................ 845 13.9.1. Параметры......................................................................................................... 846 13.9.2. Профилирование процессора..................................................................... 846 13.9.3. Обход стека....................................................................................................... 847 13.10. perf report.......................................................................................................................... 848 13.10.1. TUI..................................................................................................................... 848 13.10.2. STDIO............................................................................................................... 849 13.11. perf script........................................................................................................................... 850 13.11.1. Флейм-графики............................................................................................. 852 13.11.2. Сценарии обработки трассировок.......................................................... 852 13.12. perf trace............................................................................................................................ 852 13.12.1. Версии ядра..................................................................................................... 853 13.13. Другие команды............................................................................................................. 854 13.14. Документация perf......................................................................................................... 855 13.15. Ссылки............................................................................................................................... 855

Глава 14. Ftrace............................................................................................................857 14.1. Обзор возможностей....................................................................................................... 858 14.2. tracefs (/sys)....................................................................................................................... 861 14.2.1. Содержимое tracefs......................................................................................... 861

26  Оглавление 14.3. Профилировщик функций........................................................................................... 863 14.4. Трассировщик function................................................................................................... 865 14.4.1. Использование файла trace......................................................................... 866 14.4.2. Использование файла trace_pipe............................................................... 867 14.4.3. Параметры......................................................................................................... 868 14.5. Точки трассировки........................................................................................................... 869 14.5.1. Фильтрация....................................................................................................... 870 14.5.2. Триггеры............................................................................................................. 871 14.6. Зонды kprobes.................................................................................................................... 871 14.6.1. Трассировка событий..................................................................................... 871 14.6.2. Аргументы......................................................................................................... 872 14.6.3. Возвращаемые значения............................................................................... 873 14.6.4. Фильтры и триггеры...................................................................................... 874 14.6.5. Профилировщик kprobe............................................................................... 874 14.7. Зонды uprobes.................................................................................................................... 875 14.7.1. Трассировка событий..................................................................................... 875 14.7.2. Аргументы и возвращаемые значения..................................................... 876 14.7.3. Фильтры и триггеры...................................................................................... 876 14.7.4. Профилировщик uprobe............................................................................... 876 14.8. Трассировщик function_graph..................................................................................... 876 14.8.1. Трассировка графа.......................................................................................... 877 14.8.2. Параметры......................................................................................................... 878 14.9. Трассировщик hwlat........................................................................................................ 878 14.10. Триггеры hist.................................................................................................................... 879 14.10.1. Гистограмма с единственным ключом................................................... 880 14.10.2. Поля................................................................................................................... 881 14.10.3. Модификаторы . ........................................................................................... 881 14.10.4. Фильтры PID................................................................................................. 882 14.10.5. Гистограмма с несколькими ключами................................................... 882 14.10.6. Трассировки стека в роли ключей........................................................... 883 14.10.7. Синтетические события............................................................................. 884 14.11. trace-cmd........................................................................................................................... 887 14.11.1. Обзор подкоманд.......................................................................................... 887 14.11.2. Однострочные сценарии для trace-cmd................................................ 889

Оглавление  27 14.11.3. Сравнение trace-cmd и perf(1).................................................................. 891 14.11.4. trace-cmd function_graph............................................................................ 892 14.11.5. KernelShark..................................................................................................... 892 14.11.6. Документация для trace-cmd.................................................................... 894 14.12. perf ftrace........................................................................................................................... 894 14.13. perf-tools............................................................................................................................ 895 14.13.1. Покрытие инструментами......................................................................... 895 14.13.2. Специализированные инструменты...................................................... 896 14.13.3. Многоцелевые инструменты.................................................................... 898 14.13.4. Однострочные сценарии для perf-tools................................................. 898 14.13.5. Пример.............................................................................................................. 901 14.13.6. Сравнение perf-tools и BCC/BPF........................................................... 901 14.13.7. Документация................................................................................................ 902 14.14. Документация для Ftrace............................................................................................ 902 14.15. Ссылки............................................................................................................................... 903

Глава 15. BPF................................................................................................................904 15.1. BCC....................................................................................................................................... 907 15.1.1. Установка........................................................................................................... 908 15.1.2. Покрытие инструментами........................................................................... 908 15.1.3. Специализированные инструменты......................................................... 909 15.1.4. Многоцелевые инструменты....................................................................... 911 15.1.5. Однострочные сценарии............................................................................... 911 15.1.6. Пример многоцелевого инструмента....................................................... 913 15.1.7. Сравнение BCC и bpftrace........................................................................... 914 15.1.8. Документация.................................................................................................. 914 15.2. bpftrace................................................................................................................................. 915 15.2.1. Установка........................................................................................................... 917 15.2.2. Инструменты.................................................................................................... 917 15.2.3. Однострочные сценарии............................................................................... 918 15.2.4. Программирование......................................................................................... 920 15.2.5. Справочник....................................................................................................... 929 15.2.6. Документация.................................................................................................. 937 15.3. Ссылки................................................................................................................................. 937

28  Оглавление

Глава 16. Пример из практики.....................................................................................939 16.1. Необъяснимый выигрыш.............................................................................................. 939 16.1.1. Постановка задачи.......................................................................................... 939 16.1.2. Стратегия анализа.......................................................................................... 940 16.1.3. Статистики........................................................................................................ 941 16.1.4. Конфигурация.................................................................................................. 943 16.1.5. Счетчики PMC................................................................................................. 944 16.1.6. Программные события.................................................................................. 945 16.1.7. Трассировка....................................................................................................... 946 16.1.8. Заключение....................................................................................................... 948 16.2. Дополнительная информация..................................................................................... 949 16.3. Ссылки................................................................................................................................. 949

Приложение A. Метод USE: Linux ...............................................................................950 Приложение B. Краткий справочник по sar................................................................957 Приложение C. Однострочные сценарии для bpftrace................................................959 Приложение D. Решения некоторых упражнений.......................................................966 Приложение E. Производительность систем, кто есть кто..........................................969 Глоссарий.....................................................................................................................974

Посвящается Дейрдре Страуган, удивительному специалисту и удивительному человеку, — мы сделали это!

ПРЕДИСЛОВИЕ Есть известные известные — вещи, о которых мы знаем, что знаем их. Есть также известные неизвестные — вещи, о которых мы знаем, что не знаем. Но еще есть неизвестные неизвестные — это вещи, о которых мы не знаем, что не знаем их. Министр обороны США Дональд Рамсфельд, 12 февраля 2002 г.

Это заявление на пресс-брифинге встретили с усмешками. Однако оно определяет принцип, одинаково важный и для сложных технических систем, и для геополитики: проблемы с эффективностью могут возникать где угодно, включая те области системы, о которых вы ничего не знаете и поэтому не проверяете (неизвестные неизвестные). Эта книга поможет раскрыть многие из таких областей, а также предоставит методики и инструменты для их анализа.

ОБ ЭТОМ ИЗДАНИИ Первое издание я написал восемь лет назад и рассчитывал, что оно будет актуально достаточно долго. Главы структурированы так, чтобы сначала охватить то, что постоянно (модели, архитектуры и методологии), а затем то, что быстро меняется (инструменты и настройки). Инструменты и приемы настройки устаревают, но знание базовых вещей поможет всегда оставаться в курсе последних изменений. За последние восемь лет в Linux появилось большое дополнение: Extended BPF — технология ядра, которая поддерживает новое поколение инструментов анализа производительности и используется в Netflix и Facebook. В это новое издание я включил главу о BPF и инструментах BPF, а также опубликовал подробный справочник по BPF [Gregg 19]. Инструменты perf и Ftrace в ОС Linux также претерпели множество изменений, и я добавил для них отдельные главы. Ядро Linux получило множество новых технологий и параметров оценки производительности, которые тоже рассматриваются в этом издании книги. Гипервизоры, управляющие облачными виртуальными машинами, и контейнерные технологии тоже существенно изменились, и главы, посвященные им, были обновлены и дополнены. Первое издание в равной степени охватывало Linux и Solaris. Однако доля рынка Solaris за эти годы сильно сократилась [ITJobsWatch 20], поэтому я  почти полностью убрал из этого издания все, что касалось Solaris, освободив место для

Об этой книге  31 дополнительной информации о Linux. Но как мне кажется, возможность сравнения с альтернативами укрепит общее понимание операционной системы или ядра. По этой причине я включил в это издание некоторые упоминания о Solaris и других ОС. Последние шесть лет я работал старшим перформанс-инженером в Netflix, используя свои знания для оценки производительности микросервисов в Netflix и устранения проблем. Я занимался вопросами производительности гипервизоров, контейнеров, библиотек, ядер, баз данных и приложений. По мере необходимости я разрабатывал новые методологии и инструменты и обменивался опытом с экспертами в области производительности облачных систем и разработки ядра Linux. Все это в немалой степени улучшило второе издание книги.

ОБ ЭТОЙ КНИГЕ Итак, добро пожаловать во второе издание «Производительности систем»! Книга посвящена производительности (performance) ОС и приложений в контексте операционной системы и охватывает корпоративные серверы и облачные среды. Большая часть информации из книги может пригодиться также для анализа производительности клиентских устройств и ОС настольных компьютеров. Моя цель — помочь вам получить максимальную отдачу от ваших систем, какими бы они ни были. При работе с прикладным программным обеспечением, находящимся в постоянном развитии, может возникнуть соблазн думать об эффективности ОС, ядро которой разрабатывалось и настраивалось десятилетиями, как о решенной проблеме. Но это не так! Операционная система — это сложный комплекс ПО, управляющего множеством постоянно меняющихся физических устройств и различными прикладными рабочими нагрузками. Ядра тоже находятся в постоянном развитии, в них добавляются новые возможности увеличения производительности определенных рабочих нагрузок, а вновь возникающие узкие места устраняются по мере масштабирования системы. Изменения в ядре, например, устраняющие уязвимость Meltdown и представленные в 2018 году, тоже могут отрицательно сказываться на производительности. Анализ и работа над улучшением производительности ОС — это непрерывный процесс. Производительность приложений тоже можно проанализировать в контексте операционной системы, чтобы отыскать подсказки, которые можно упустить при использовании только инструментов для приложений; об этом я тоже расскажу.

Рассматриваемые операционные системы Эта книга изучает производительность систем. В роли основного представителя выступают ОС на базе Linux для процессоров Intel. Но книга организована так, чтобы вы могли изучить и другие ядра для других аппаратных архитектур. Для представленных примеров конкретный дистрибутив Linux не важен, если явно не указано иное. Основная масса примеров была получена в дистрибутиве

32   Предисловие Ubuntu, и там, где это важно, в текст включены примечания, объясняющие отличия от других дистрибутивов. В книге приводятся примеры, полученные в системах различных типов: без операционной системы (на «голом железе») и в виртуальных средах, на производственных и тестовых машинах, на серверах и клиентских устройствах. В своей работе я сталкивался со множеством разных операционных систем и ядер, что углубило мое понимание их дизайна. Чтобы вы могли во всем разобраться, в книгу включены некоторые упоминания о Unix, BSD, Solaris и Windows.

Другие материалы Примеры скриншотов инструментов анализа производительности включены не только чтобы показать данные, но и для иллюстрации видов доступных данных. Инструменты часто представляют данные простым и понятным способом, многие из них реализованы в стиле, знакомом по более ранним инструментам для Unix. Учитывая это, скриншоты могут быть мощным средством показать назначение этих инструментов почти без дополнительного описания. (Если инструмент требует подробного объяснения, это может быть признаком неудачного дизайна.) Там, где это уместно, я затрагиваю историю появления определенных технологий. Также полезно узнать немного о ключевых людях в этой отрасли: вы наверняка сталкивались с ними или их работой в сфере эффективности и в других контекстах. Их имена вы найдете в приложении E. Некоторые темы, рассматриваемые в этом издании, также были освещены в моей предыдущей книге, «BPF Performance Tools»1 [Gregg 19]: в частности, BPF, BCC, bpftrace, tracepoints, kprobes, uprobes и множество других инструментов на основе BPF. В этой книге вы найдете дополнительную информацию. Краткое изложение этих тем здесь часто основано на той, более ранней книге. Иногда я использую тот же текст и примеры.

О чем здесь не рассказывается Эта книга посвящена производительности. Для выполнения всех приведенных примеров потребуются некоторые действия по администрированию системы, включая установку или компиляцию программного обеспечения (которые здесь не рассматриваются). Также здесь кратко описывается внутреннее устройство операционной системы, которое более подробно рассматривается в специализированных изданиях. Вопросы, касающиеся углубленного анализа производительности, здесь рассмотрены в общих чертах — лишь для того, чтобы вы знали об их существовании и могли заняться их Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

Об этой книге  33 изучением по другим источникам. См. раздел «Дополнительные источники, ссылки и библиография» в конце предисловия.

Структура Глава 1 «Введение» — это введение в анализ производительности системы. Глава обобщает ключевые идеи и приводит примеры действий, направленных на улучшение производительности. Глава 2 «Методологии» формирует основу для анализа и настройки производительности и описывает терминологию, основные понятия, модели, методологии для наблюдения и экспериментов, планирование емкости, анализ и статистику. Глава 3 «Операционные системы» описывает внутреннее устройство ядра с точки зрения анализа производительности. Глава закладывает основы, необходимые для интерпретации и понимания действий операционной системы. Глава 4 «Инструменты наблюдения» знакомит с доступными средствами наблюдения за системой, а также интерфейсами и фреймворками, на которых они построены. Глава 5 «Приложения» обсуждает вопросы производительности приложений и наблюдение за ними из операционной системы. Глава 6 «Процессоры» рассматривает процессоры, ядра, аппаратные потоки выполнения, кэш-память процессора, взаимосвязи между процессорами, взаимосвязи между устройствами и механизмы планирования в ядре. Глава 7 «Память» посвящена виртуальной памяти, страничной организации, механизму подкачки, архитектурам памяти, шинам, адресным пространствам и механизмам распределения памяти. Глава 8 «Файловые системы» посвящена производительности операций ввода/ вывода с файловой системой, включая различные механизмы кэширования. Глава 9 «Диски» описывает устройства хранения, рабочие нагрузки дискового ввода/вывода, контроллеры хранилищ, дисковые массивы RAID и подсистему ввода/вывода ядра. Глава 10 «Сеть» посвящена сетевым протоколам, сокетам, интерфейсам и физическим соединениям. Глава 11 «Облачные вычисления» знакомит с методами виртуализации операционных систем и оборудования, которые обычно используются для организации облачных вычислений, а также с их характеристиками производительности, изоляции и наблюдаемости. В этой главе также рассматриваются гипервизоры и контейнеры. Глава 12 «Бенчмаркинг» показывает, как правильно проводить сравнительный анализ и как интерпретировать результаты бенчмаркинга. Это на удивление сложная тема, и в этой главе я покажу, как избежать типичных ошибок, и попробую объяснить их смысл.

34   Предисловие Глава 13 «perf» кратко описывает стандартный профилировщик Linux, perf(1), и его многочисленные возможности. Ссылки на этот справочник по perf(1) вы будете встречать по всей книге. Глава 14 «Ftrace» кратко описывает стандартное средство трассировки Li­ nux, Ftrace, которое особенно подходит для исследования особенностей работы ядра. Глава 15 «BPF» кратко описывает стандартные внешние интерфейсы BPF: BCC и bpftrace. Глава 16 «Пример из практики» содержит пример исследования производительности системы, проводившегося в Netflix. Он показывает, как проводился анализ производительности. Главы 1–4 содержат важную базовую информацию. Прочитав их, вы будете готовы перейти к любой из остальных глав в книге, в частности к главам 5–12, в которых рассматриваются конкретные цели для анализа. Главы 13–15 посвящены продвинутому профилированию и трассировке и являются факультативным чтением для тех, кто хочет подробнее изучить один или несколько трассировщиков. В главе 16 я привожу истории, которые позволят сформировать более широкое представление о работе перформанс-инженера. Если вы только начинаете заниматься анализом производительности, то можете прочитать сначала эту главу. Она покажет пример анализа производительности с использованием множества различных инструментов. Вы можете вернуться к ней после прочтения других глав.

Применимость в будущем Я писал эту книгу так, чтобы ее полезность сохранялась на долгие годы. Основное внимание в ней уделяется опыту и методологиям анализа производительности систем. Для этого многие главы были разделены на две части. Первая часть включает определения, описание понятий и методологий (часто соответствующие разделы именно так и называются), которые должны оставаться актуальными на протяжении многих лет. Во второй части приводятся примеры реализации: архитектура, инструменты анализа и настраиваемые параметры. Они могут устареть, но все же останутся полезными в качестве примеров.

Примеры трассировки Часто требуется глубоко изучить операционную систему, чтобы знать, что можно сделать с помощью инструментов трассировки.

Об этой книге  35 Уже после выхода первого издания была разработана и внедрена в ядро Linux расширенная технология BPF, в результате чего появилось новое поколение инструментов трассировки, использующих внешние интерфейсы BCC и bpftrace. Эта книга посвящена BCC и bpftrace, а также встроенному в ядро Linux трассировщику Ftrace. BPF, BCC и bpftrace более подробно описаны в моей предыдущей книге [Gregg 19]. В книге рассматривается еще один инструмент трассировки Linux — perf. Но perf здесь в основном используется для получения и анализа счетчиков контроля производительности (performance monitoring counter, PMC), а не для трассировки. Возможно, вы захотите использовать другие инструменты трассировки, и это нормально. Представленные в этой книге инструменты показывают, какие вопросы можно задать системе. Часто именно эти вопросы и методологии, которые их ставят, труднее всего понять.

Для кого эта книга В первую очередь книга адресована системным администраторам и операторам корпоративных и облачных вычислительных сред. Она послужит справочником для разработчиков, администраторов баз данных и администраторов веб-серверов, которые должны понимать, из чего складывается производительность операционных систем и приложений. Как перформанс-инженеру в компании с гигантской вычислительной инфраструктурой (Netflix), мне часто приходится работать с SRE-инженерами и разработчиками, которым просто физически не хватает времени для решения сразу нескольких проблем с производительностью. Мне тоже доводилось работать дежурным инженером в Netflix CORE SRE, и я знаком с этой нехваткой времени на собственном опыте. Для многих людей обеспечение производительности не является их основной работой, и им достаточно знать ровно столько, чтобы решать текущие проблемы. Я понимаю, что у вас может быть мало времени, и я постарался сделать эту книгу как можно короче и проще по структуре. Другая целевая аудитория — студенты. Книга поможет при изучении курса производительности систем. Я вел подобные курсы раньше и знаю, что лучше всего помогает студентам решать проблемы с успеваемостью. Этими знаниями я руководствовался при работе над книгой. Кем бы вы ни были, упражнения в главах позволят проверить себя и надежнее усвоить материал. Среди них вы найдете особенно сложные упражнения, которые необязательно решать. (Они могут представлять проблемы, не имеющие решения, главная их цель — заставить задуматься.) Наконец, с точки зрения размера компании эта книга содержит достаточно подробностей, чтобы удовлетворить потребности компаний от мала до велика. Для многих небольших компаний эта книга послужит справочником, в котором лишь некоторые части используются ежедневно.

36   Предисловие

Условные обозначения В этой книге используются следующие условные обозначения: Пример

Описание

netif_receive_skb()

Имя функции

iostat(1)

Команда со ссылкой на раздел в справочном руководстве man с ее описанием

read(2)

Системный вызов со ссылкой на раздел в справочном руководстве man с его описанием

malloc(3)

Имя функции из библиотеки языка C со ссылкой на раздел в справочном руководстве man с ее описанием

vmstat(8)

Команда администрирования со ссылкой на раздел в справочном руководстве man с ее описанием

Documentation/...

Каталог с документацией в дереве исходных текстов ядра Linux

kernel/...

Каталог в дереве исходных текстов ядра Linux

fs/...

Каталог с реализацией файловой системы в дереве исходных текстов ядра Linux

CONFIG_...

Параметры настройки ядра Linux (Kconfig)

r_await

Команда в командной строке и ее вывод

mpstat 1

Команда или ключевая деталь, на которую следует обратить особое внимание

#

Приглашение к вводу в командной оболочке суперпользователя (root)

$

Приглашение к вводу в командной оболочке обычного пользователя (не root)

^C

Прерывание выполнения команды (комбинацией клавиш Ctrl-C)

[...]

Усечение

Дополнительные источники, ссылки и библиография Список литературы приводится в конце каждой главы, а не в конце всей книги. Вы сразу можете смотреть источники, относящиеся к теме каждой главы. В списке ниже перечислены книги, из которых можно почерпнуть информацию об ОС и анализе производительности: [Jain 91] Jain, R., «The Art of Computer Systems Performance Analysis: Techniques for Experimental Design, Measurement, Simulation, and Modeling», Wiley, 1991. [Vahalia 96] Vahalia, U., «UNIX Internals: The New Frontiers», Prentice Hall, 1996.1 [Cockcroft 98] Cockcroft, A., and Pettit, R., «Sun Performance and Tuning: Java and the Internet, Prentice Hall», 1998. Вахалия Ю. «UNIX изнутри». СПб.: Издательство «Питер».

1

Об этой книге  37 [Musumeci 02] Musumeci, G. D., and Loukides, M., «System Performance Tuning, 2nd Edition», O’Reilly, 2002.1 [Bovet 05] Bovet, D., and Cesati, M., «Understanding the Linux Kernel, 3rd Edition», O’Reilly, 2005.2 [McDougall 06a] McDougall, R., Mauro, J., and Gregg, B., «Solaris Performance and Tools: DTrace and MDB Techniques for Solaris 10 and OpenSolaris», Prentice Hall, 2006. [Gove 07] Gove, D., «Solaris Application Programming», Prentice Hall, 2007. [Love 10] Love, R., «Linux Kernel Development, 3rd Edition», Addison-Wesley, 2010.3 [Gregg 11a] Gregg, B., and Mauro, J., «DTrace: Dynamic Tracing in Oracle Solaris, Mac OS X and FreeBSD», Prentice Hall, 2011. [Gregg 13a] Gregg, B., «Systems Performance: Enterprise and the Cloud», Prentice Hall, 2013 (first edition). [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»4, Addison-Wesley, 2019. [ITJobsWatch 20] ITJobsWatch, «Solaris Jobs», https://www.itjobswatch.co.uk/jobs/uk/solaris. do#demand_trend, ссылка была действительна в феврале 2021.

Мусумеси Дж.-П. Д. Лукидес М. «Настройка производительности UNIX-систем».

1

Бовет Д., Чезати М. «Ядро Linux».

2

Лав Р. «Ядро Linux. Описание процесса разработки».

3

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

4

БЛАГОДАРНОСТИ

Спасибо всем купившим первое издание, и особенно тем, кто рекомендовал прочитать его своим коллегам. Поддержка первой книги способствовала созданию второй. Спасибо вам. Это моя последняя книга, посвященная производительности систем, но не первая. Я хотел бы поблагодарить авторов других книг по этой же тематике, на которые я опирался и на которые неоднократно ссылаюсь здесь. В частности, хотел бы поблагодарить Адриана Кокрофта (Adrian Cockcroft), Джима Мауро (Jim Mauro), Ричарда Макдугалла (Richard McDougall), Майка Лукидеса (Mike Loukides) и Раджа Джейна (Raj Jain). Все вы здорово помогли мне, и я надеюсь, что смогу помочь вам. Я благодарен всем, кто давал мне обратную связь: Дейрдра Страуган (Deirdré Straughan) всячески поддерживала меня и использовала свой богатый опыт редактирования технических книг, чтобы каждая страница этой книги стала лучше. Слова, которые вы читаете, принадлежат нам обоим. Нам нравится не только вместе проводить время (сейчас мы женаты), но и работать. Спасибо. Филиппу Мареку (Philipp Marek) — эксперту в информационных технологиях, ИТ-архитектору и перформанс-инженеру в Австрийском федеральном вычислительном центре. Он одним из первых делился мнением по каждой теме этой книги (настоящий подвиг) и даже обнаружил проблемы в тексте первого издания. Филипп начал программировать в 1983 году, еще для микропроцессора 6502, и практически сразу стал искать способы экономии тактов процессора. Спасибо, Филипп, за твой опыт и неустанную работу! Дейл Хэмел (Dale Hamel, Shopify) тоже внимательно прочитал каждую главу, предоставил важные сведения о различных облачных технологиях и помог взглянуть на книгу другими глазами. Спасибо тебе, Дейл, что взял на себя этот труд! Это случилось сразу после того, как ты помог мне с книгой о BPF. Даниэль Боркманн (Daniel Borkmann, Isovalent) тщательно прорецензировал некоторые главы, в частности главы о сетях. Это помогло мне лучше понять имеющиеся сложности и сопутствующие технологии. Даниэль уже много лет занимается сопровождением ядра Linux и обладает огромным опытом работы над сетевым стеком ядра и расширенным BPF. Спасибо тебе, Даниэль, за профессионализм и строгость оценок. Я особенно благодарен мейнтейнеру инструмента perf Арнальдо Карвалью де Мело (Arnaldo Carvalho de Melo; Red Hat) за помощь с главой 13 «perf» и Стивену Ростедту (Steven Rostedt; VMware), создателю Ftrace, за помощь с главой 14

Благодарности  39 «Ftrace» — двумя темами, которые я недостаточно полно рассмотрел в первом издании. Я высоко ценю их не только за помощь в написании этой книги, но также за их работу над этими инструментами повышения производительности, которые я использовал для решения бесчисленных проблем в Netflix. Было очень приятно, что Доминик Кэй (Dominic Kay) пролистал несколько глав и оставил множество советов по улучшению читаемости и повышению технической точности. Доминик также помогал мне с первым изданием (а еще раньше мы вместе работали в Sun Microsystems, где занимались вопросами производительности). Спасибо, Доминик. Мой нынешний коллега по Netflix, Амер Атер (Amer Ather), оставил очень ценные отзывы к нескольким главам. Амер — инженер, разбирающийся в сложных технологиях. Захарий Джонс (Zachary Jones, Verizon) тоже дал обратную связь по особенно сложным вопросам и поделился опытом в области производительности, чем очень помог улучшить книгу. Спасибо вам, Амер и Захарий. Несколько рецензентов, взяв несколько глав, участвовали в обсуждении конкретных тем: Алехандро Проаньо (Alejandro Proaño, Amazon), Бикаш Шарма (Bikash Sharma, Facebook), Кори Луенингонер (Cory Lueninghoener, Национальная лаборатория в Лос-Аламосе), Грег Данн (Greg Dunn, Amazon), Джон Аррасджид (John Arrasjid, Ottometric), Джастин Гаррисон (Justin Garrison, Amazon), Майкл Хаузенблас (Michael Hausenblas, Amazon) и Патрик Кейбл (Patrick Cable, Threat Stack). Спасибо всем за вашу помощь и энтузиазм. Также спасибо Адитье Сарваде (Aditya Sarwade, Facebook), Эндрю Галлатину (Andrew Gallatin, Netflix), Басу Смиту (Bas Smit), Джорджу Невиллу-Нилу (George Neville-Neil, JUUL Labs), Йенсу Аксбоу (Jens Axboe, Facebook), Джоэлю Фернандесу (Joel Fernandes, Google), Рэндаллу Стюарту (Randall Stewart, Netflix), Стефану Эраниану (Stephane Eranian, Google) и Токе Хойланд-Йоргенсену (Toke HøilandJørgensen, Red Hat) за ответы на вопросы и своевременную техническую помощь. Те, кто участвовал в создании моей предыдущей книги — «BPF Performance Tools», тоже косвенно помогли мне, потому что некоторые материалы этого издания основаны на предыдущем. Улучшению той книги в немалой степени способствовали Аластер Робертсон (Alastair Robertson, Yellowbrick Data), Алексей Старовойтов (Alexei Starovoitov, Facebook), Дэниел Боркманн (Daniel Borkmann), Джейсон Кох (Jason Koch, Netflix), Мэри Марчини (Mary Marchini, Netflix), Масами Хирамацу (Masami Hiramatsu, Linaro), Мэтью Дезнойерс (Mathieu Desnoyers, EfficiOS), Йонгхонг Сонг (Yonghong Song, Facebook) и многие другие. Полный список вы найдете в разделе «Благодарности» предыдущего издания. Многие, помогавшие мне в работе над первым изданием, помогли в работе и над этим. В частности, я получил техническую поддержку по нескольким главам от Адама Левенталя (Adam Leventhal), Карлоса Карденаса (Carlos Cardenas), Дэррила Гоува (Darryl Gove), Доминика Кэя (Dominic Kay), Джерри Елинека (Jerry Jelinek), Джима Мауро (Jim Mauro), Макса Брунинга (Max Bruning), Ричарда Лоу (Richard Lowe) и Роберта Мустаччи (Robert Mustacchi). Я также получил отзывы

40   Благодарности и поддержку от Адриана Кокрофта (Adrian Cockcroft), Брайана Кантрилла (Bryan Cantrill), Дэна Макдональда (Dan McDonald), Дэвида Пачеко (David Pacheco), Кита Весоловски (Keith Wesolowski), Марселля Кукульевича-Пирса (Marsell KukuljevicPearce) и Пола Эгглтона (Paul Eggleton). Рох Бурбоннис (Roch Bourbonnais) и Ричард Макдугалл (Richard McDougall) многому научили меня на моей предыдущей работе, где мы занимались проблемами производительности, и тем самым оказали косвенную помощь в работе над этой книгой, как и Джейсон Хоффман (Jason Hoffman), косвенно помогавший в работе над первым изданием. Ядро Linux — сложный и постоянно меняющийся программный продукт, и я ценю труд Джонатана Корбета (Jonathan Corbet) и Джейка Эджа (Jake Edge) из lwn.net по обобщению большого числа сложнейших тем. Многие из их статей упоминаются в этой книге. Отдельное спасибо Грегу Доенчу (Greg Doench) — выпускающему редактору издательства Pearson за гибкость и поддержку, благодаря которым процесс двигался особенно эффективно. Спасибо продюсеру информационного наполнения Джулии Нахил (Julie Nahil; Pearson) и менеджеру проекта Рэйчел Пол (Rachel Paul) за внимание к деталям и помощь в создании качественного продукта. Спасибо редактору Киму Уимпсетту (Kim Wimpsett) за работу над еще одной из моих длинных и глубоко технических книг, за множество предложений по улучшению текста. И спасибо тебе, Митчелл, за терпение и понимание. Начиная с первого издания, я продолжал работать перформанс-инженером, устраняя проблемы по всему программно-аппаратному стеку. Теперь у меня еще больше опыта в работе с гипервизорами, в настройке производительности, в анализе среды выполнения (включая JVM), в применении трассировщиков, включая Ftrace и BPF, а также в реагировании на быстро меняющиеся микросервисы Netflix и ядро Linux. Многое из этого недостаточно хорошо задокументировано, и порой было очень сложно определить, что отразить в книге. Но я люблю сложности.

ОБ АВТОРЕ Брендан Грегг — эксперт в области производительности и облачных вычислений. Работает старшим перформанс-инженером в Netflix, где занимается проектированием, оценкой, анализом и настройкой производительности. Автор нескольких книг, в том числе «BPF Performance Tools1». Обладатель награды USENIX LISA за выдающиеся достижения в системном администрировании. Работал инженером по поддержке ядра, руководил командой обеспечения производительности и профессионально занимался преподаванием технических дисциплин, был сопредседателем конференции USENIX LISA 2018. Создал множество инструментов оценки производительности для разных операционных систем, а также разработал средства и методы визуализации для анализа производительности, включая флейм-графики.

ОТ ИЗДАТЕЛЬСТВА Ваши замечания, предложения, вопросы отправляйте по адресу [email protected] (­издательство «Питер», компьютерная редакция). Мы будем рады узнать ваше мнение! На веб-сайте издательства www.piter.com вы найдете подробную информацию о наших книгах.

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

Глава 1

ВВЕДЕНИЕ Производительность компьютеров — увлекательная, многообразная и сложная дисциплина. В этой главе вы познакомитесь с понятием эффективности и производительности систем. Цели главы:

yy познакомить с понятием производительности системы, кто и как ее обеспечивает и какие сложности встречаются на этом пути;

yy показать разницу между инструментами наблюдения и инструментами проведения экспериментов;

yy дать общее представление о способах и средствах оценки производительности,

таких как параметры, профилирование, флейм-графики, трассировка, статические и динамические инструменты;

yy представить роль методологий и короткий чек-лист для исследования производительности Linux.

Здесь будут даны ссылки на последующие главы, поэтому данную главу можно считать введением и в дисциплину производительности систем, и в книгу в целом. Глава заканчивается реальными примерами, показывающими, насколько важно уделять внимание производительности систем.

1.1. ПРОИЗВОДИТЕЛЬНОСТЬ СИСТЕМЫ Под оценкой производительности системы понимается изучение производительности всей компьютерной системы, включая основные программные и аппаратные компоненты, — все, что находится на пути к данным, от устройств хранения до прикладного программного обеспечения, — потому что все они могут влиять на производительность. В случае с распределенными системами в этот список также входят все серверы и приложения. Если у вас нет схемы вашего окружения, отражающей путь к данным, найдите ее или нарисуйте сами; она поможет увидеть взаимосвязи между компонентами и не упустить из виду целые области. Типичная цель оценки производительности системы — улучшить взаимодействие с конечным пользователем за счет уменьшения задержек и снижения затрат на

1.2. Роли  43 вычисления. Снижения затрат можно достигнуть за счет устранения неэффективности, повышения пропускной способности и общей настройки системы. На рис. 1.1 показан обобщенный стек системного ПО на одном сервере, включая ядро операционной системы (ОС), базу данных и приложение. Иногда термин фуллстек используется для описания только прикладного окружения, включающего базы данных, приложения и веб-серверы. Но говоря о производительности системы, под словом фуллстек мы подразумеваем весь программный стек — от приложений до железа (оборудования), включая системные библиотеки, ядро и само оборудование. При оценке производительности системы рассматривается фуллстек. Приложение База данных Компиляторы

Системные библиотеки Системные вызовы

Уровень пользователя Уровень ядра

Ядро Планировщик потоков выполнения

Файловые системы

Сетевой стек

Виртуальная память

Драйверы устройств

Устройства

Рис. 1.1. Обобщенный стек системного программного обеспечения Компиляторы включены в обобщенный стек на рис. 1.1, потому что играют важную роль в производительности системы. Этот стек мы обсудим в главе 3 «Операционные системы» и подробно рассмотрим в последующих главах. В следующих разделах будут более подробно описаны особенности оценки производительности системы.

1.2. РОЛИ Производительность системы обеспечивается различными специалистами, в том числе системными администраторами, SRE-инженерами, разработчиками приложений, сетевыми инженерами, администраторами баз данных, веб-администраторами и др. Для многих из этих специалистов обеспечение производительности является лишь

44  Глава 1. Введение частью их работы, поэтому при оценке производительности они фокусируются только на своей сфере ответственности: группа сетевых администраторов проверяет производительность сетевого стека, группа администраторов баз данных проверяет базу данных, и т. д. Однако иногда для выяснения причин низкой производительности или факторов, способствующих этому, требуются совместные усилия нескольких команд. В некоторых компаниях работают перформанс-инженеры, для которых обеспечение высокой производительности — основная деятельность. Они могут взаимодействовать с несколькими командами для проведения комплексного исследования среды, что нередко очень важно для решения сложных проблем с производительностью. Они также могут выступать в качестве центрального звена, осуществляющего поиск и разработку инструментов для анализа производительности и планирования ресурсов во всей среде. Например, в Netflix есть команда по производительности облачных вычислений, в которую вхожу и я. Мы помогаем командам микросервисов и обеспечения надежности анализировать производительность и создаем инструменты оценки производительности для всех остальных. Компании, нанимающие несколько перформанс-инженеров, могут позволить им специализироваться в одной или нескольких областях и обеспечивать более глубокую поддержку. Например, большая группа перформанс-инженеров может включать специалистов по производительности ядра, клиентских приложений, языка (например, Java), среды выполнения (например, JVM), по разработке инструментов для оценки производительности и т. д.

1.3. ДЕЙСТВИЯ Анализ и увеличение производительности системы предполагают выполнение множества действий. Ниже приводится список таких действий, которые одновременно представляют этапы идеального жизненного цикла программного проекта — от идеи до разработки и развертывания в продакшене. В этой книге описаны методологии и инструменты, помогающие выполнять эти действия. 1. Определение целей и моделирование производительности будущего продукта. 2. Определение характеристик производительности прототипа программного и аппаратного обеспечения. 3. Анализ производительности разрабатываемых продуктов в тестовой среде. 4. Регрессионное тестирование новых версий продукта. 5. Бенчмаркинг производительности разных версий продуктов. 6. Проверка концепции в целевой промышленной среде. 7. Оптимизация производительности в промышленной среде. 8. Мониторинг ПО, действующего в промышленной среде. 9. Анализ производительности промышленных задач.

1.3. Действия  45 10. Ревью инцидентов, возникающих в промышленной среде. 11. Разработка инструментов для повышения эффективности анализа производительности в промышленной среде. Шаги с 1-го по 5-й охватывают традиционный процесс разработки продукта, будь то продукт, продаваемый клиентам или используемый внутри компании. После разработки продукт вводится в эксплуатацию, иногда сначала проверяется пригодность использования продукта в целевой среде (клиента или внутри компании), а иногда сразу же производится развертывание и настройка. Если в целевой среде обнаружится проблема (шаги с 6-го по 9-й), это говорит только о том, что она не была обнаружена или исправлена на этапах разработки. В идеале проектирование производительности должно начинаться до выбора какого-либо оборудования или создания программного обеспечения: первым шагом должны быть постановка целей и создание модели производительности. Однако часто продукты разрабатываются, минуя этот шаг, из-за чего работы по проектированию производительности откладываются на более позднее время, когда проблемы уже возникнут. С каждым следующим этапом процесса разработки становится все труднее устранять проблемы с производительностью, обусловленные ранее принятыми архитектурными решениями. Облачные вычисления предлагают новые методы проверки концепции (шаг 6), которые побуждают пропускать более ранние шаги (с 1-го по 5-й). Один из таких методов — тестирование нового ПО на единственном экземпляре с небольшой рабочей нагрузкой: это называется канареечным тестированием. Другой метод превращает его в обычный шаг при развертывании: трафик постепенно перемещается в новый пул экземпляров, при этом старый пул остается в горячем резерве; этот метод известен как сине-зеленое развертывание1. Применение таких приемов защиты от сбоев в новом ПО позволяет проводить тестирование в продакшене без всякого предварительного анализа производительности и при необходимости быстро возвращаться к исходному состоянию. Но я рекомендую всегда, когда это возможно, выполнить также первые этапы, чтобы достичь максимальной производительности (даже при том, что иногда могут быть веские причины миновать их, например, быстрый выход на рынок). Некоторые из перечисленных этапов охватываются термином планирование мощности (capacity planning). На этапе проектирования он подразумевает изучение объема ресурсов, необходимых разрабатываемому ПО, чтобы увидеть, насколько полно его архитектура может удовлетворить целевые потребности. После развертывания он подразумевает мониторинг использования ресурсов для прогнозирования проблем до их возникновения. В анализ производительности в промышленной среде (шаг 9) могут также вовлекаться SRE-инженеры. Далее следует шаг по ревью инцидентов в промышленной среде (шаг 10), когда производится анализ произошедшего, обмен опытом отладки В Netflix используется термин красно-черное развертывание.

1

46  Глава 1. Введение и поиск способов избежать аналогичных инцидентов в будущем. Эти ревью похожи на ретроспективы разработчиков (см. [Corry 20], где рассказывается, что такое ретроспективы и их антипаттерны). Среды и мероприятия различаются для разных компаний и продуктов, и во многих случаях выполняются не все десять шагов. Ваша работа также может быть сосредоточена только на некоторых или только на одном из этих мероприятий.

1.4. ПЕРСПЕКТИВЫ Помимо выполнения различных мероприятий, привлечение различных специалистов можно рассматривать как использование разных точек зрения — перспектив. На рис. 1.2 обозначены две точки зрения на анализ производительности: анализ рабочей нагрузки и анализ ресурсов — двух подходов к оценке программного стека. Рабочая нагрузка Приложение

Анализ рабочей нагрузки

Системные библиотеки Стек программного обеспечения операционной системы

Системные вызовы Ядро Устройства

Анализ ресурсов

Рис. 1.2. Анализ с двух перспектив Перспектива анализа ресурсов обычно используется системными администраторами, отвечающими за системные ресурсы. Разработчики приложений, отвечающие за производительность под рабочей нагрузкой, обычно сосредоточиваются на перспективе анализа рабочей нагрузки. Каждая перспектива имеет свои сильные стороны, которые подробно обсуждаются в главе 2 «Методологии». При решении сложных вопросов полезно попытаться проанализировать ситуацию с обеих сторон.

1.5. СЛОЖНОСТИ ОЦЕНКИ ПРОИЗВОДИТЕЛЬНОСТИ Проектирование производительности системы — сложная область по многим причинам, в том числе из-за субъективности, природной сложности, отсутствия какой-то единственной основной причины и наличия множества связанных проблем.

1.5. Сложности оценки производительности  47

1.5.1. Субъективность Технические дисциплины тяготеют к объективности, причем настолько, что занимающиеся ими люди видят мир в черно-белом цвете. Это может быть верно в отношении неполадок в софте, когда ошибка либо есть, либо ее нет и либо она исправлена, либо не исправлена. Такие ошибки часто проявляются в виде сообщений, которые легко интерпретировать и идентифицировать как сообщения об ошибках. Производительность, напротив, часто бывает субъективной. В отношении проблем с производительностью часто неочевидно, имела ли место проблема изначально, и если да, то когда она была устранена. Один пользователь может считать производительность «плохой» и рассматривать ее как проблему, а другой может считать ее «хорошей». Например, представьте, что вам поступила такая информация: Среднее время отклика подсистемы дискового ввода/вывода составляет 1 мс. Это «хорошо» или «плохо»? Время отклика, или задержка, является одним из лучших доступных показателей, но интерпретировать информацию о задержке сложно. Часто выбор между оценками «хорошая» или «плохая» зависит от ожиданий разработчиков приложений и конечных пользователей. Субъективную оценку производительности можно сделать объективной, определив четкие цели, например целевое среднее время отклика или попадание определенного процента запросов в некоторый диапазон задержек. Другие способы борьбы с субъективностью будут представлены в главе 2 «Методологии», в том числе и анализ задержки.

1.5.2. Сложность Помимо субъективности, сложность оценки производительности может быть связана со свойственной системам сложностью и отсутствием очевидной отправной точки для анализа. В облачных средах порой даже нет возможности узнать, на какой экземпляр обратить внимание в первую очередь. Иногда мы начинаем с выдвижения гипотезы, например, обвиняя сеть или базу данных, а аналитик производительности должен выяснить, является ли эта гипотеза верной. Проблемы с производительностью также могут возникать из-за сложных взаимодействий между подсистемами, которые показывают хорошие характеристики при их анализе в изоляции друг от друга. Падение производительности может произойти из-за каскадного сбоя, когда один отказавший компонент вызывает проблемы с производительностью в других компонентах. Чтобы понять причину возникшей проблемы, необходимо распутать клубок взаимосвязей между компонентами и понять, какой вклад они вносят. Сложность также может быть обусловлена наличием неожиданных взаимосвязей, когда устранение проблемы в одном месте может привести к появлению проблемы

48  Глава 1. Введение в другом месте системы, при этом общая производительность улучшится не так сильно, как ожидалось. Помимо сложности системы, проблемы с производительностью также могут быть вызваны сложным характером производственной нагрузки. Некоторые случаи могут просто не воспроизводиться в лабораторных условиях или возникать лишь периодически. Решение сложных проблем производительности часто требует комплексного подхода. Возможно, потребуется исследовать всю систему — и ее внутренние элементы, и внешние взаимодействия. Для этого необходимо обладать широким спектром навыков, что может сделать проектирование производительности многообразным и интеллектуально сложным делом. Для преодоления этих сложностей можно использовать разные методологии, как описано в главе 2. В главах 6–10 вы найдете описания конкретных методологий анализа конкретных системных ресурсов: процессоры, память, файловые системы, диски и сеть. (Комплексный анализ сложных систем, включая разливы нефти и крах финансовых систем, описан в [Dekker 18].) В некоторых случаях проблема производительности может быть вызвана взаимодействием этих ресурсов.

1.5.3. Множественные причины Некоторые проблемы с производительностью не имеют единственной первопричины и обусловлены множеством факторов. Представьте сценарий, когда одновременно происходят три вполне обычных события, которые в совокупности вызывают проблему с производительностью: каждое из этих событий — обычное и само по себе не является первопричиной. Множественными могут быть не только причины, но и сами проблемы с производительностью.

1.5.4. Множественные проблемы с производительностью В сложном программном обеспечении обычно бывает немало проблем с производительностью. Для примера попробуйте найти базу данных ошибок для вашей ОС или приложений и поищите по слову performance (производительность). Результаты могут удивить вас! Как правило, в таких базах данных даже для зрелого ПО, считающегося высокопроизводительным, есть множество известных, но пока не исправленных проблем с производительностью. Это создает еще одну трудность при анализе: настоящая задача не в том, чтобы найти проблему, а в том, чтобы определить, какая проблема или проблемы наиболее важны. Для этого специалист по анализу производительности должен количественно оценить масштаб проблемы. Некоторые проблемы с производительностью могут быть не свойственны вашей рабочей нагрузке или проявляться в весьма незначительной степени. В идеале вы должны не только количественно оценить проблемы, но также

1.6. Задержка  49 оценить потенциальное ускорение, которое можно получить за счет устранения каждой из них. Эта информация может пригодиться, когда менеджеры будут искать оправдание расходам на инженерные или операционные ресурсы. Один из показателей, хорошо подходящих для количественной оценки производительности, если он доступен, — это задержка.

1.6. ЗАДЕРЖКА Задержка характеризует время, затраченное на ожидание, и является важной метрикой производительности. В широком смысле под задержкой понимается время до завершения любой операции, такой как обработка запроса приложением или базой данных, операция файловой системы и т. д. Например, задержка может выражать время полной загрузки веб-страницы от щелчка на ссылке до появления полного изображения страницы на экране. Это важный показатель как для клиента, так и для владельца сайта: большая задержка может вызвать разочарование и желание у клиентов разместить заказ в другом месте. В роли метрики задержка позволяет оценить максимальное ускорение. Например, на рис. 1.3 изображена временная диаграмма обработки запроса к базе данных, на что уходит 100 мс (это время и является задержкой), из которых 80 мс тратится на ожидание завершения операции чтения с диска. В данном случае за счет исключения операций чтения с диска (например, путем кэширования) можно добиться уменьшения времени обработки со 100 мс до 20 мс (100–80), то есть добиться пятикратного (в 5 раз) улучшения производительности. Это оценочное ускорение, и вычисления также помогли количественно оценить масштаб проблемы производительности: чтение с диска в 5 раз замедляет обработку запросов. Обработка запроса к базе данных: 100 мс

Поток выполнения ЦП

Вне ЦП Чтение с диска: 80 мс

Рис. 1.3. Пример задержки, вызванной дисковым вводом/выводом Подобные вычисления невозможны при использовании других метрик. Например, количество операций ввода/вывода в секунду (IOPS) зависит от типа ввода/вывода и часто напрямую несопоставимо для разных ситуаций. Если какое-то изменение приведет к уменьшению IOPS на 80 %, то трудно предсказать, как это отразится на производительности. Операций в секунду может быть в 5 раз меньше, но что, если каждая из этих операций обрабатывает в 10 раз больше данных? Значение задержки также может быть неоднозначным без уточняющих терминов. Например, задержка в сети может означать время, необходимое для установления

50  Глава 1. Введение соединения, но не время передачи данных; или общую продолжительность соединения, включая передачу данных (например, так обычно измеряется задержка DNS). По мере возможности в этой книге я буду использовать уточняющие термины: эти примеры лучше описать как задержку соединения и задержку обработки запроса. Терминология, связанная с задержкой, также приводится в начале каждой главы. Задержка — полезная метрика, но она не всегда доступна. Некоторые системные области позволяют измерить только среднюю задержку; некоторые вообще не дают возможности измерения. С появлением новых инструментов наблюдения на основе BPF1 задержку теперь можно измерять практически в любых точках и получить данные, описывающие полное распределение задержки.

1.7. НАБЛЮДАЕМОСТЬ Под наблюдаемостью подразумевается исследование системы через наблюдение и инструменты, предназначенные для этого. Сюда входят инструменты, использующие счетчики, профилирование и трассировку, но не входят инструменты тестирования производительности, которые изменяют состояние системы, выполняя эксперименты с рабочей нагрузкой. В промышленных средах желательно сначала попробовать применить инструменты наблюдения, где это возможно, потому что инструменты для экспериментов могут препятствовать обработке промышленных рабочих нагрузок из-за конкуренции за ресурсы. В тестовых средах, которые большую часть времени простаивают, можно сразу начать с инструментов тестирования производительности для определения быстродействия оборудования. В этом разделе я расскажу о счетчиках, метриках, профилировании и трассировке. Более подробно о наблюдаемости речь пойдет в главе 4, где рассмотрены общесистемная и индивидуальная наблюдаемость, инструменты наблюдения Linux и их внутреннее устройство. Кроме того, в главах 5–11 есть разделы, посвященные наблюдаемости, например, раздел 6.6 описывает инструменты наблюдения за процессором.

1.7.1. Счетчики, статистики и метрики Приложения и ядро обычно предоставляют данные c информацией о своем состоянии и активности: счетчики операций, счетчики байтов, измеренные задержки, использованный объем ресурсов и частоту ошибок. Обычно эти данные доступны в виде целочисленных переменных, называемых счетчиками; они жестко «вшиты» в программное обеспечение, часть из них являются кумулятивными и постоянно увеличиваются. Эти кумулятивные счетчики можно читать в разное время инструментами оценки производительности для вычисления таких статистик, как скорость изменения во времени, среднее значение, процентили и т. д. 1

В настоящее время BPF является названием, а не аббревиатурой от Berkeley Packet Filter, как было первоначально.

1.7. Наблюдаемость  51 Например, утилита vmstat(8) выводит общесистемную статистику по виртуальной памяти и другие данные на основе счетчиков ядра, доступных в файловой системе /proc. Вот пример вывода vmstat(8) на производственном сервере с 48 процессорами: $ vmstat 1 5 procs ---------memory---------- ---swap-r b swpd free buff cache si so 19 0 0 6531592 42656 1672040 0 26 0 0 6533412 42656 1672064 0 62 0 0 6533856 42656 1672088 0 34 0 0 6532972 42656 1672088 0 31 0 0 6534876 42656 1672088 0

-----io---bi bo 0 1 0 0 0 0 0 0 0 0

-system-- ------cpu----in cs us sy id wa st 7 21 33 51 4 46 0 0 81262 188942 54 4 43 8 80865 180514 53 4 43 0 81250 180651 53 4 43 0 74389 168210 46 3 51

0 0 0 0 0

0 0 0 0

Судя по этому примеру, доля занятости процессора в системе составляет около 57 % (столбцы cpu us + sy). Более подробно значение столбцов объясняется в главах 6 и 7. Метрика — это статистика, выбранная для оценки или мониторинга цели. Большинство компаний используют агенты мониторинга для записи выбранных статистик (метрик) через регулярные промежутки времени и построения графиков их изменения в графическом интерфейсе, чтобы видеть, как они меняются с течением времени. Программное обеспечение для мониторинга также может поддерживать создание специальных предупреждений на основе этих метрик, например отправку электронных писем для уведомления персонала об обнаружении проблем. Эта иерархия от счетчиков до предупреждений изображена на рис. 1.4, который поможет вам понять эти термины, но их использование в отрасли не является жестко фиксированным. Термины счетчики, статистики и метрики часто используются как синонимы. Кроме того, оповещения могут генерироваться на любом уровне, а не только специальной системой оповещения. В качестве примера графического представления метрик на рис. 1.5 показан скриншот инструмента на основе Grafana, который наблюдает за тем же сервером, на котором был получен предыдущий вывод vmstat(8). Эти линейные графики удобно использовать для планирования мощности — они помогают предсказать, когда наступит момент исчерпания ресурсов. Ваша интерпретация статистик производительности улучшится, если вы поймете, как они вычисляются. Статистики, включая средние значения, распределения, моды и выбросы, более подробно описываются в главе 2 «Методологии», в разделе 2.8 «Статистики». Иногда для решения проблемы с производительностью достаточно получить временные ряды с метриками. Время, когда проявилась проблема, может коррелировать с известным изменением в ПО или конфигурации, которое можно отменить. Иногда метрики лишь подсказывают направление, сообщая о проблеме с процессором или диском, но без объяснения причин. В таких случаях, чтобы копнуть глубже и отыскать причину, необходимо использовать инструменты профилирования.

52  Глава 1. Введение

Обработка событий Графический интерфейс мониторинга Инструменты/агенты для получения данных о производительности Приложения/ Ядро

Примеры Предупреждения

Prometheus

Метрики

Grafana

Статистики

vmstat collectd

Счетчики

/proc

Рис. 1.4. Терминология, связанная с оценкой производительности

Рис. 1.5. Графический интерфейс с метриками (Grafana)

1.7.2. Профилирование В контексте производительности систем термин профилирование обычно означает использование инструментов для выборки подмножества измерений и создания на их основе приблизительной картины, характеризующей цель. Часто целью профилирования является доля занятости процессора. Обычно для этого используется

1.7. Наблюдаемость  53 метод профилирования, основанный на выборке путей в коде, выполняющемся на процессоре, с заданным интервалом времени. Один из эффективных способов визуализации результатов профилирования процессора — флейм-графики (или графики пламени — flame graphs). Флейм-графики помогают добиться большего выигрыша в производительности, чем любой другой инструмент, после метрик. Они позволяют выявить не только проблемы с занятостью процессора, но и другие типы проблем, обнаруживаемые по характерным особенностям использования процессора. Проблемы, связанные с конфликтом блокировок, например, можно обнаружить по большим затратам процессорного времени в циклах ожидания блокировок; проблемы с памятью легко обнаруживаются по чрезмерным затратам процессорного времени в функциях распределения памяти (malloc()) и в коде, вызывающем эти функции; проблемы, связанные с ошибками в настройках сети, можно обнаружить по затратам процессорного времени в медленных или устаревших путях в коде; и т. д. На рис. 1.6 приведен пример флейм-графика, показывающего процессорное время, потраченное инструментом тестирования сети iperf(1).

Рис. 1.6. Флейм-график расходования процессорного времени

54  Глава 1. Введение На этом флейм-графике видно, насколько больше процессорного времени тратится на копирование байтов (путь в коде, заканчивающийся функцией copy_user_ enhanced_fast_string()) по сравнению с передачей TCP-пакетов (второй большой пик слева, включающий функцию tcp_write_xmit()). Ширина пиков (или пирамид, как их еще называют) пропорциональна потраченному процессорному времени, а вдоль вертикальной оси откладывается путь в коде. Подробнее приемы профилирования обсуждаются в главах 4, 5 и 6, а об использовании флейм-графиков рассказывается в главе 6 «Процессор», в разделе 6.7.3 «Флейм-графики».

1.7.3. Трассировка Трассировка — это запись событий, когда данные о событиях фиксируются и сохраняются для последующего анализа или используются на лету для обобщения и выполнения других действий. Есть специальные инструменты трассировки для системных вызовов (например, strace(1) в Linux) и сетевых пакетов (например, tcpdump(8) в Linux), а также инструменты трассировки общего назначения, способные анализировать все программные и аппаратные события (например, Ftrace, BCC и bpftrace в Linux). Эти всевидящие трассировщики используют различные источники событий, в частности точки статической и динамической инструментации, а также механизм BPF, поддерживающий возможность программного управления.

Точки статической инструментации Под точками статической инструментации подразумеваются жестко закодированные точки в ПО. В ядре Linux сотни таких точек, позволяющих выполнять трассировку дискового ввода/вывода, событий планировщика, системных вызовов и многого другого. Технология статической инструментации ядра Linux называется tracepoints (точки трассировки). Есть технология статической инструментации ПО в пространстве пользователя, которая называется статически определяемой трассировкой на уровне пользователя (User-level Statically Defined Tracing, USDT). Точки USDT используются во многих библиотеках (например, libc) для инструментации библиотечных вызовов и в приложениях для инструментации функций обработки запросов. Примером утилиты, использующей статическую инструментацию, может служить execsnoop(8), которая выводит информацию о процессах, запущенных во время трассировки, получаемую путем инструментации точки трассировки в системном вызове execve(2). Ниже показано, как execsnoop(8) трассирует вход по SSH: # execsnoop PCOMM ssh sshd sh env

PID 30656 30657 30660 30661

PPID RET 20063 0 1401 0 30657 0 30660 0

ARGS /usr/bin/ssh 0 /usr/sbin/sshd -D -R /usr/bin/env -i PATH=/usr/local/sbin:/usr/local...

1.7. Наблюдаемость  55 run-parts 00-header uname uname uname 10-help-text 50-motd-news cat cut tr head 80-esm lsb_release [...]

30661 30662 30663 30664 30665 30666 30667 30668 30671 30670 30669 30672 30673

30660 30661 30662 30662 30662 30661 30661 30667 30667 30667 30667 30661 30672

0 0 0 0 0 0 0 0 0 0 0 0 0

/bin/run-parts --lsbsysinit /etc/update-motd.d /etc/update-motd.d/00-header /bin/uname -o /bin/uname -r /bin/uname -m /etc/update-motd.d/10-help-text /etc/update-motd.d/50-motd-news /bin/cat /var/cache/motd-news /usr/bin/cut -c -80 /usr/bin/tr -d \000-\011\013\014\016-\037 /usr/bin/head -n 10 /etc/update-motd.d/80-esm /usr/bin/lsb_release -cs

Этот прием особенно удобен для трассировки короткоживущих процессов, которые могут остаться незамеченными другими инструментами наблюдения, такими как top(1). Эти короткоживущие процессы могут быть источником проблем с производительностью. Дополнительную информацию о точках трассировки и зондах USDT вы найдете в главе 4.

Динамическая инструментация Технология динамической инструментации создает точки трассировки уже после запуска ПО путем подмены инструкций в памяти процесса вызовами процедур трассировки. Примерно так отладчики вставляют точки останова в любые функции в запущенном ПО. Разница лишь в том, что когда при отладке поток выполнения достигает точки останова, управление передается интерактивному отладчику, а при динамической инструментации вызывается процедура трассировки, по окончании которой целевое ПО продолжает работу как ни в чем не бывало. Динамическая инструментация позволяет собирать необходимую статистику производительности из любого выполняющегося ПО. Проблемы, которые раньше было невозможно или очень сложно решить из-за недостаточной наблюдаемости, теперь можно исправить. Динамическая инструментация настолько отличается от традиционного наблюдения и мониторинга, что поначалу трудно понять ее роль. Возьмем для примера ядро операционной системы: анализ внутреннего устройства ядра часто похож на блуждание в темной комнате со свечами (системными счетчиками), установленными там, где разработчики ядра посчитали необходимым. Динамическую инструментацию я бы сравнил с фонариком, луч которого можно направить куда угодно. Первые методы динамической инструментации были разработаны в 1990-х годах [Hollingsworth 94] вместе с инструментами, которые их используют. Эти инструменты называют динамическими трассировщиками (например, kerninst [Tamches 99]). Поддержка динамической инструментации для ядра Linux была разработана в 2000 году [Kleen 08] и начала внедряться в 2004 году (kprobes). Но эти технологии были малоизвестны и сложны в использовании. Ситуация изменилась, когда в 2005 году Sun Microsystems выпустила свою версию DTrace — простую

56  Глава 1. Введение в использовании и безопасную для применения в производственной среде. Я разработал множество инструментов на основе DTrace, которые доказали свою важность для оценки производительности системы, получили широкое распространение и принесли широкую известность DTrace и динамической инструментации.

BPF Механизм BPF, название которого первоначально произошло от Berkeley Packet Filter, поддерживает новейшие инструменты динамической трассировки для Linux. BPF создавался как миниатюрная виртуальная машина в ядре для ускорения выполнения выражений tcpdump(8). Но в 2013 году был расширен (поэтому иногда его называют eBPF1) и превратился в универсальную среду выполнения в ядре, обеспечивающую безопасный и быстрый доступ к ресурсам. Среди многочисленных новых применений обновленного механизма BPF — инструменты трассировки, для которых он обеспечивает поддержку программирования операций с применением коллекции компиляторов BPF (BPF Compiler Collection, BCC), и внешний интерфейс bpftrace. Например, инструмент execsnoop(8), показанный выше, является инструментом BCC2. Подробнее о BPF рассказывается в главе 3, а глава 15 знакомит с интерфейсами трассировки BPF: BCC и bpftrace. В других главах, в разделах о наблюдаемости, будут представлены многие инструменты трассировки на основе BPF; например, инструменты трассировки процессора описываются в главе 6 «Процессор» в разделе 6.6 «Инструменты наблюдения». Также ранее я опубликовал книги по инструментам трассировки (для DTrace [Gregg 11a] и BPF [Gregg 19]). Оба инструмента, perf(1) и Ftrace, тоже являются трассировщиками и обладают возможностями, аналогичными интерфейсам BPF. Подробнее о perf(1) и Ftrace рассказывается в главах 13 и 14.

1.8. ЭКСПЕРИМЕНТЫ Помимо инструментов наблюдения, также есть инструменты для экспериментов, большинство из которых применяется для бенчмаркинга. Они позволяют ставить эксперименты, применяя искусственную рабочую нагрузку к системе и измеряя ее производительность. Такие эксперименты следует проводить с осторожностью, потому что они могут ухудшать производительность тестируемых систем. Есть инструменты макробенчмарикинга, которые имитируют реальную рабочую нагрузку, например действия клиентов, посылающих запросы приложениям, а есть 1

Первое время для описания расширенного BPF (extended BPF) использовалось название eBPF; однако в настоящее время эта технология называется просто BPF.

2

Сначала я разработал версию для DTrace, а потом и для других трассировщиков, включая BCC и bpftrace.

1.8. Эксперименты  57 инструменты микробенчмаркинга, которые тестируют конкретный компонент системы, например процессоры, диски или сети. В качестве аналогии: определение времени, необходимого для преодоления круга на трассе Laguna Seca Raceway, можно считать макробенчмаркингом, а определение времени разгона от 0 до 100 км/ч можно считать микробенчмаркингом. Оба типа тестов важны, но микробенчмарки обычно проще в разработке, отладке, использовании и понимании, и они более стабильны. Следующий пример показывает использование iperf(1) на простаивающем сервере для микробенчмаркинга пропускной способности канала TCP с удаленным неактивным сервером. Этот бенчмарк выполнялся в течение 10 с (-t 10) и выводит средние значения в секунду (-i 1): # iperf -c 100.65.33.90 -i 1 -t 10 -----------------------------------------------------------Client connecting to 100.65.33.90, TCP port 5001 TCP window size: 12.0 MByte (default) -----------------------------------------------------------[ 3] local 100.65.170.28 port 39570 connected with 100.65.33.90 port 5001 [ ID] Interval Transfer Bandwidth [ 3] 0.0- 1.0 sec 582 MBytes 4.88 Gbits/sec [ 3] 1.0- 2.0 sec 568 MBytes 4.77 Gbits/sec [ 3] 2.0- 3.0 sec 574 MBytes 4.82 Gbits/sec [ 3] 3.0- 4.0 sec 571 MBytes 4.79 Gbits/sec [ 3] 4.0- 5.0 sec 571 MBytes 4.79 Gbits/sec [ 3] 5.0- 6.0 sec 432 MBytes 3.63 Gbits/sec [ 3] 6.0- 7.0 sec 383 MBytes 3.21 Gbits/sec [ 3] 7.0- 8.0 sec 388 MBytes 3.26 Gbits/sec [ 3] 8.0- 9.0 sec 390 MBytes 3.28 Gbits/sec [ 3] 9.0-10.0 sec 383 MBytes 3.22 Gbits/sec [ 3] 0.0-10.0 sec 4.73 GBytes 4.06 Gbits/sec

Как показывают результаты эксперимента, пропускная способность1 в первые 5 с находилась на уровне около 4,8 Гбит/с, а затем упала до примерно 3,2 Гбит/с. Этот интересный результат демонстрирует бимодальную пропускную способность. Чтобы увеличить производительность, можно сосредоточиться на моде 3,2 Гбит/с и поискать другие метрики, объясняющие ее. Рассмотрим недостатки отладки подобной проблемы производительности на рабочем сервере с использованием только инструментов наблюдения. Пропускная способность сети может меняться с течением времени из-за естественной разницы в рабочей нагрузке клиента, и подобное бимодальное поведение сети может оставаться незамеченным. Используя инструмент iperf(1), генерирующий фиксированную рабочую нагрузку, можно избавиться от влияния изменчивости поведения клиентов и обнаружить отклонения, вызванные другими факторами (например, наличием ограничений во внешней сети, использованием буферов и т. д.). В этом выводе можно видеть термин «Bandwidth» (полоса пропускания), который часто используется не по назначению. Под полосой пропускания подразумевается максимально возможная пропускная способность, которую iperf(1) не измеряет. iperf(1) измеряет текущую скорость передачи данных по сети: ее пропускную способность.

1

58  Глава 1. Введение Как я рекомендовал выше, в производственных системах следует сначала попробовать применить инструменты наблюдения. Но их так много, что с ними можно работать часами, тогда как инструмент для экспериментов позволит быстрее прий­ ти к результатам. Много лет назад старший перформанс-инженер Рох Бурбоннис (Roch Bourbonnais) рассказал мне о такой аналогии: у вас есть две руки, наблюдение и эксперименты. Использование инструментов только одного типа похоже на попытку решить проблему одной рукой. Главы 6–10 включают разделы об инструментах для экспериментирования. Например, инструменты для экспериментирования с процессором описаны в главе 6 «Процессор» в разделе 6.8 «Эксперименты».

1.9. ОБЛАЧНЫЕ ВЫЧИСЛЕНИЯ Облачные вычисления как способ развертывания вычислительных ресурсов по запросу позволяют быстро масштабировать приложения, развертывая их во все большем количестве небольших виртуальных систем, называемых экземплярами (instances). Этот подход избавляет от необходимости тщательно планировать емкость, потому что в кратчайшие сроки можно добавить дополнительные ресурсы из облака. В некоторых случаях это также увеличило потребность в анализе производительности, потому что использование меньшего количества ресурсов может означать меньшее количество систем. Поскольку за использование облачных ресурсов обычно взимается поминутная или часовая оплата, выигрыш в производительности, способствующий использованию меньшего количества систем, дает прямую экономию средств. Сравните этот сценарий с услугами корпоративного вычислительного центра, с которым вы можете быть связаны фиксированным договором на поддержку в течение многих лет без возможности экономии, пока срок действия договора не истечет. Облачные вычисления и виртуализация принесли новые трудности, включая управление влиянием других подписчиков на производительность (иногда это называют изоляцией производительности) и возможность мониторинга физической системы со стороны каждого подписчика. Например, в отсутствие должного управления системой производительность дискового ввода/вывода может упасть из-за конфликта с соседом. В некоторых средах информация о фактической нагрузке на физические диски может быть недоступна подписчикам, что затрудняет выявление таких проблем. Более подробно эти вопросы рассматриваются в главе 11 «Облачные вычисления».

1.10. МЕТОДОЛОГИИ Методология — это способ задокументировать рекомендуемые шаги для решения различных задач по оценке производительности системы. Без методологии

1.10. Методологии  59 исследование производительности может превратиться в рыбалку, когда рыбак пробует разные наживки в надежде на удачу. Это неэффективный и потенциально долгий путь с риском упустить из виду важные области. В главе 2 «Методологии» я привожу перечень методологий для оценки производительности систем. В следующем подразделе я покажу первое, что использую для решения любой проблемы с производительностью: чек-лист инструментов.

1.10.1. Анализ производительности Linux за 60 секунд Это чек-лист инструментов анализа производительности для Linux, который можно выполнить за 60 секунд в самом начале исследования проблемы производительности. Здесь перечислены традиционные инструменты, доступные в большинстве дистрибутивов Linux [Gregg 15a]. В табл. 1.1 показаны конкретные команды, которые следует выполнить, а также разделы в этой книге, где соответствующие команды рассматриваются более подробно. Таблица 1.1. Чек-лист для анализа производительности Linux за 60 секунд №

Инструмент

Проверяет

Раздел

 1

uptime

Средние значения нагрузки, чтобы определить, увеличивается или уменьшается нагрузка (можно сравнить средние значения за 1, 5 и 15 минут)

6.6.1

 2

dmesg -T | tail

Ошибки в ядре, включая события OOM (нехватки памяти)

7.5.11

 3

vmstat -SM 1

Общесистемные статистики: длина очереди на выполнение, подкачка (swapping), общая доля занятости процессора

7.5.1

 4

mpstat -P ALL 1

Баланс нагрузки по процессорам: увеличенная нагрузка на один из процессоров может указывать на плохое масштабирование потоков выполнения

6.6.3

 5

pidstat 1

Потребление процессора каждым процессом: позволяет выявить непредвиденных потребителей процессорного времени и определить, сколько процессорного времени потрачено каждым процессом в пространстве ядра и в пространстве пользователя

6.6.7

 6

iostat -sxz 1

Статистики дискового ввода/вывода: количество операций ввода/вывода в секунду (IOPS) и пропускная способность, среднее время ожидания, процент занятости

9.6.1

 7

free –m

Потребление памяти, включая кэши файловой системы

8.6.2

 8

sar -n DEV 1

Статистики сетевого устройства ввода/вывода: количество пакетов и пропускная способность

10.6.6

 9

sar -n TCP,ETCP

Статистики TCP: частота приема соединений, частота повторных передач

10.6.6

10

top

Общий обзор

6.6.6

60  Глава 1. Введение Этот чек-лист также можно использовать в графическом интерфейсе мониторинга, если в нем доступны те же метрики1. Глава 2 «Методологии» и следующие за ней содержат описание множества методологий анализа производительности, включая метод USE, определение характеристик рабочей нагрузки, анализ задержки и многие другие.

1.11. ПРАКТИЧЕСКИЕ ПРИМЕРЫ Если вы новичок в вопросах оценки производительности систем, то практические примеры, показывающие, когда и почему выполняются различные действия, помогут вам провести аналогию с вашим текущим окружением. В этом разделе представлены два гипотетических примера. Первый показывает проблему производительности, связанную с дисковым вводом/выводом, а второй — тестирование производительности после изменения ПО. Примеры описывают действия, о которых подробно рассказывается в других главах этой книги. Основная цель этих примеров — показать не правильный или единственный способ, а скорее один из способов, которым можно выполнить эти действия.

1.11.1. Медленные диски Сумит — системный администратор в компании среднего размера. Группа обслуживания базы данных добавила тикет с жалобой на «медленные диски» на одном из серверов баз данных. Первая задача Сумита — узнать как можно больше о проблеме, собрать необходимую информацию и сформулировать проблему. В тикете утверждается, что диски работают медленно, но не объясняется, действительно ли это является причиной проблемы в базе данных. В ответ Сумит задает следующие вопросы:

yy Наблюдаются ли проблемы с производительностью базы данных сейчас? Как она измеряется?

yy Как давно существует эта проблема? yy Изменилось ли что-нибудь в базе данных за последнее время? yy Почему подозрение пало на диски? В ответ команда базы данных пишет: «В нашем отделе ведется журнал, в котором фиксируются запросы, продолжительность обработки которых превышает 1000 мс. Обычно такие запросы встречаются редко, но за последнюю неделю их число 1

Можно даже создать свой дашборд для такого чек-листа, но имейте в виду, что этот чек-лист составлен с целью максимально использовать доступные инструменты командной строки, тогда как средства мониторинга могут иметь множество других (и более информативных) метрик. Я более склонен создавать дашборды для методологии USE и др.

1.11. Практические примеры  61 выросло до нескольких десятков в час. Анализ с применением AcmeMon показал большую загруженность дисков». Этот ответ подтверждает наличие проблемы с базой данных, но также показывает, что гипотеза о том, что причиной является низкая производительность диска, скорее всего, является предположением. Сумит решает проверить диски, а также другие ресурсы на тот случай, если гипотеза окажется неверной. AcmeMon — это базовая система мониторинга серверов компании, предоставляющая исторические графики изменения стандартных метрик операционной системы, которые можно получить с помощью mpstat(1), iostat(1) и других системных утилит. Сумит входит в AcmeMon, чтобы выполнить задуманные проверки. На первом шаге Сумит применяет методологию USE (описывается в главе 2 «Методологии» в разделе 2.5.9), чтобы быстро проверить наличие узких мест в ресурсах. Как сообщила группа обслуживания базы данных, доля загруженности дисков достигла высокого уровня, около 80 %, тогда как потребление других ресурсов (процессор, сеть) намного ниже. По историческим данным выяснилось, что загруженность дисков неуклонно росла в течение последней недели, в то время как потребление процессора оставалось постоянным. Система AcmeMon не предоставляет статистики, характеризующие насыщенность или частоту ошибок для дисков, поэтому для применения методологии USE Сумит должен выполнить некоторые команды на самом сервере. Он проверил счетчики ошибок дискового ввода/вывода в /sys — они оказались равны нулю. Запустил iostat(1), задав интервал равным одной секунде, и понаблюдал за изменением метрик потребления и насыщенности с течением времени. Система AcmeMon сообщала об уровне загруженности 80 %, но проводила измерения с интервалом в одну минуту. Используя интервал измерений в одну секунду, Сумит увидел, что загруженность диска колеблется, часто достигая 100 % и вызывая увеличение уровня насыщенности и задержки дискового ввода/вывода. Чтобы еще раз убедиться, что загруженность диска является причиной блокировки базы данных и возрастает синхронно с запросами к ней, он решил использовать инструмент трассировки BCC/BPF под названием offcputime(8) и с его помощью захватывать трассировки стека всякий раз, когда база приостанавливается ядром, а также определять продолжительность приостановки. Трассировки стека показали, что база данных часто блокируется на время чтения файловой системы в процессе обработки запроса. Для Сумита этого было достаточно. Следующий вопрос — почему. Статистики производительности диска соответствуют высокой нагрузке. Сумит решил выяснить характеристики рабочей нагрузки, измерив с помощью iostat(1) частоту операций ввода/вывода, пропускную способность, среднюю задержку дискового ввода/вывода и соотношение операций чтения/ записи. Для получения дополнительной информации Сумит мог бы использовать трассировку дискового ввода/вывода, однако ему достаточно информации, указывающей, что имеет место высокая нагрузка на диск, а не проблема с производительностью дисков.

62  Глава 1. Введение Сумит добавляет дополнительные детали в тикет, указав, что было проверено, и включив скриншоты команд, использовавшихся для анализа работы дисков. На данный момент он пришел к выводу, что диски работают под высокой нагрузкой, из-за чего увеличивается задержка ввода/вывода и медленно обрабатываются запросы. Но судя по имеющимся данным, диски вполне справляются с нагрузкой, и он задает вопрос: есть ли простое объяснение случившемуся; увеличилась ли нагрузка на базу данных? Команда обслуживания БД ответила, что простого объяснения нет и количество запросов (о которых не сообщает AcmeMon) остается постоянным. Похоже, это согласуется с более ранним выводом о неизменности нагрузки на процессор. Сумит размышляет о том, какие еще причины могут вызвать увеличение нагрузки на дисковый ввод/вывод без заметного увеличения потребления процессора, и консультируется со своими коллегами. Один из них выдвигает предположение о сильной фрагментированности файловой системы, что вполне ожидаемо, когда заполненность файловой система приближается к 100 %. Но, как выяснил Сумит, файловая система заполнена всего на 30 %. Сумит знает, как провести более детальный анализ1, чтобы выяснить точные причины, но на это требуется много времени. Основываясь на своем знании стека ввода/вывода в ядре, он пытается придумать другие простые объяснения, которые можно быстро проверить. Он помнит, что часто дисковые операции обусловлены промахами кэша файловой системы (кэша страниц). Сумит проверяет коэффициент попаданий в кэш файловой системы с помощью cachestat(8)2 и обнаруживает, что в данный момент он составляет 91 %. Выглядит неплохо (и даже очень хорошо), но у него нет исторических данных для сравнения. Тогда он заходит на другие серверы баз данных, обслуживающие аналогичные рабочие нагрузки, и обнаруживает, что в них коэффициент попадания в кэш превышает 98 %. Он также обнаруживает, что размер кэша файловой системы на других серверах намного больше. Обратив внимание на размер кэша файловой системы и потребление памяти сервера, он замечает, что было упущено из виду: в разрабатываемом проекте имеется прототип приложения, потребляющий все больше памяти, хотя пока работает вхолостую. В результате для кэша файловой системы остается меньше свободной памяти, из-за чего снижается частота попаданий в кэш и увеличивается количество дисковых операций чтения с диска. Сумит связался с командой разработчиков приложения и попросил их остановить приложение и переместить его на другой сервер, сославшись на проблему с базой данных. После того как они выполнили его просьбу, Сумит увидел в AcmeMon, как 1

Об этом рассказывается в главе 2 «Методологии» в разделе 2.5.12 «Анализ с увеличением детализации».

2

Инструмент трассировки BCC, описывается в главе 8 «Файловые системы» в разделе 8.6.12 «cachestat».

1.11. Практические примеры  63 по мере восстановления кэша файловой системы до исходного объема нагрузка на диск стала постепенно снижаться. Количество медленно обрабатываемых запросов упало до нуля, и он закрыл тикет как разрешенный.

1.11.2. Изменение в программном обеспечении Памела — перформанс-инженер в небольшой компании, она занимается всем, что так или иначе связано с производительностью. Разработчики приложений реализовали новую возможность, но не уверены, что ее внедрение не ухудшит производительность. Памела решает провести регрессионное тестирование новой версии приложения перед внедрением в продакшен. Для тестирования Памела выбирает простаивающий сервер, и теперь ей нужен имитатор рабочей нагрузки клиента. В недавнем прошлом группа разработчиков приложений написала такой имитатор, но у него есть различные ограничения и известные ошибки. Памела решает попробовать его, но прежде хочет убедиться, что он способен создавать рабочую нагрузку, адекватную текущей. Она настраивает сервер в соответствии с текущей конфигурацией развертывания и запускает имитатор в другой системе для создания рабочей нагрузки на сервер. Нагрузку, имитирующую действия клиентов, можно оценить, изучив журнал доступа, и в компании уже есть подходящий для этого инструмент. Она запускает этот инструмент, передает ему журнал с рабочего сервера с данными за разные периоды в течение суток и сравнивает рабочие нагрузки. Как оказалось, имитатор создает среднюю рабочую нагрузку без всяких отклонений. Она отмечает это и продолжает анализ. Памела знает несколько подходов, которые можно применить на этом этапе. Она выбирает самый простой: увеличивать нагрузку до достижения предела (иногда этот подход называют стресс-тестированием). Имитатор клиента можно настроить на отправку определенного количества клиентских запросов в секунду со значением по умолчанию 1000. Она решает начать со 100 запросов в секунду и постепенно увеличивать нагрузку с шагом 100, пока не будет достигнут предел, при этом на каждом уровне нагрузки тестирование продолжается в течение одной минуты. Она пишет сценарий на языке командной оболочки, который собирает результаты в файл для анализа другими инструментами. При действующей нагрузке она проводит активный анализ производительности, чтобы выявить ограничивающие факторы. Ресурсы сервера кажутся свободными, а потоки выполнения по большей части бездействующими. Как показал имитатор, пропускная способность составила примерно 700 запросов в секунду. После этого она запускает новую версию приложения и повторяет тесты. Достигается тот же уровень 700 запросов в секунду, и никаких ограничивающих факторов не выявляется. Она наносит результаты на графики, отражающие зависимость количества обработанных запросов в секунду от нагрузки, чтобы визуально определить профиль

64  Глава 1. Введение масштабируемости, и отмечает присутствие на обоих графиках резкой верхней границы. По всей видимости, обе версии ПО имеют схожие характеристики производительности, но Памела разочарована, потому что не смогла выявить ограничивающий фактор, определяющий потолок масштабируемости. Она знает, что проверяла только ресурсы сервера, а ограничение может скрываться в логике приложения, в работе сети, в имитаторе клиента или где-то еще. Памела решает использовать другой подход, например, запустить имитацию клиента с фиксированной частотой следования запросов и оценить потребление ресурсов (процессора, дискового ввода/вывода, сетевого ввода/вывода), чтобы потом выразить затраты ресурсов на обработку одного клиентского запроса. Она запускает имитатор с частотой 700 запросов в секунду и измеряет потребление ресурсов при использовании текущего и нового программного обеспечения. С текущим ПО в системе с 32 процессорами загрузка процессоров составила 20 %, а с новым программным обеспечением при той же нагрузке она увеличилась до 30 %. Похоже, что в новом ПО действительно имеется регресс производительности из-за увеличенного потребления вычислительных ресурсов. Но было бы интересно понять причину предела в 700 запросов в секунду. Памела настраивает имитатор на более высокую нагрузку и исследует все компоненты на пути к данным, включая сеть, клиентскую систему и генератор клиентской рабочей нагрузки. Она также выполняет детальный анализ серверного и клиентского ПО, документируя проверенные метрики и скриншоты для справки. Для исследования клиентского ПО она проанализировала состояния потоков выполнения и обнаружила, что оно однопоточное! Этот единственный поток потребляет все 100 % процессорного времени, что, по всей видимости, и является причиной столь резкой границы в 700 запросов в секунду, обнаруженной во время тестирования. Чтобы подтвердить догадку, она запускает имитатор сразу на нескольких клиентских системах и доводит потребление процессора на сервере до 100 % как с текущим, так и с новым программным обеспечением. По результатам этого тестирования производительность текущей версии достигла 3500 запросов в секунду, а новой версии — 2300 запросов в секунду, что согласуется с более ранними данными о потреблении ресурсов. Памела сообщает разработчикам, что в новой версии ПО наблюдается регресс, и приступает к профилированию потребления процессора с помощью флеймграфика, чтобы понять, какие пути в коде вносят наибольший вклад. Она отмечает, что тестирование выполнялось с нагрузкой, сопоставимой со средней на производственном сервере, и другие нагрузки не проверялись. Она также сообщает об ошибке в имитаторе клиентской рабочей нагрузки — что он однопоточный и может стать узким местом.

1.12. Ссылки  65

1.11.3. Дополнительное чтение Подробный практический пример представлен в главе 16 «Пример из практики», где описывается решение конкретной проблемы с производительностью в облачной среде. В следующей главе будут представлены методологии, используемые для анализа производительности, а в последующих главах дается вся необходимая информация.

1.12. ССЫЛКИ [Hollingsworth 94] Hollingsworth, J., Miller, B., and Cargille, J., «Dynamic Program Instrumentation for Scalable Performance Tools», Scalable High-Performance Computing Conference (SHPCC), May 1994. [Tamches 99] Tamches, A., and Miller, B., «Fine-Grained Dynamic Instrumentation of Commodity Operating System Kernels», Proceedings of the 3rd Symposium on Operating Systems Design and Implementation, February 1999. [Kleen 08] Kleen, A., «On Submitting Kernel Patches», Intel Open Source Technology Center, http://halobates.de/on-submitting-patches.pdf, 2008. [Gregg 11a] Gregg, B., and Mauro, J., «DTrace: Dynamic Tracing in Oracle Solaris, Mac OS X and FreeBSD», Prentice Hall, 2011. [Gregg 15a] Gregg, B., «Linux Performance Analysis in 60,000 Milliseconds», Netflix Technology Blog, http://techblog.netflix.com/2015/11/linux-performance-analysis-in-60s.html, 2015. [Dekker 18] Dekker, S., «Drift into Failure: From Hunting Broken Components to Understanding Complex Systems», CRC Press, 2018. [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»1, Addison-Wesley, 2019. [Corry 20] Corry, A., «Retrospectives Antipatterns», Addison-Wesley, 2020.

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

Глава 2

МЕТОДОЛОГИИ Дай человеку рыбу, и он будет сыт один день. Научи его ловить рыбу, и он будет сыт до конца жизни. Китайская пословица Свою карьеру я начинал младшим системным администратором и думал тогда, что смогу освоить оценку производительности, изучая только метрики и инструменты командной строки. Как же я ошибался! Я прочитал страницы справочного руководства от корки до корки и изучил, как определять сбои страниц (page faults), переключения контекста и множество других системных показателей, но я не знал, что с ними делать: как перейти от сигналов к решениям. Я заметил, что всякий раз, когда возникала проблема с производительностью, старшие системные администраторы применяли свои методики использования инструментов и метрик, чтобы как можно быстрее добраться до первопричины. Они понимали, какие метрики наиболее информативны, когда они указывают на проблему, и как их использовать для сужения исследуемой области. Именно этого ноу-хау не хватало на страницах справочного руководства, и мы, молодежь, перенимали этот опыт, заглядывая через плечо старшего администратора или инженера. С тех пор я собрал, разработал, задокументировал и опубликовал собственные методологии оценки эффективности и включил их в эту главу вместе с другой важной информацией о производительности системы: понятиями, терминологией, статистиками и приемами визуализации. Здесь та теория, которая пригодится в последующих главах, где мы перейдем к практическому применению этих методологий. Цели главы:

yy Познакомить с ключевыми метриками производительности: задержкой, коэффициентом использования (потреблением) и насыщенностью.

yy Развить интуицию в выборе масштаба измеряемого времени вплоть до наносекунд.

yy Научить правильно выстраивать компромиссы, определять цели и момент прекращения анализа.

yy Показать, как выявлять проблемы рабочей нагрузки в зависимости от архитектуры.

2.1. Терминология  67

yy Научить анализировать ресурсы и рабочие нагрузки. yy Представить различные методологии оценки производительности, в том

числе метод USE, определение характеристик рабочей нагрузки, анализ задержки, статическую настройку производительности и мантры производительности.

yy Познакомить с основами статистики и теории массового обслуживания. Из всех глав моей книги эта меньше всего изменилась по сравнению с первым изданием. Программное и аппаратное обеспечение, инструменты оценки и настройки производительности — все это не раз менялось в течение моей карьеры, но теория и методология, о которых идет речь в этой главе, оставались неизменными. Глава делится на три части:

yy Основы: знакомит с терминологией, базовыми моделями, ключевыми поняти-

ями и перспективами. Практически вся оставшаяся часть книги предполагает знание этих основ.

yy Методологии: рассматривает методологии анализа производительности, основанные как на наблюдениях, так и на экспериментах, моделирование, а также планирование мощности.

yy Метрики: представляет статистики производительности, приемы мониторинга и визуализации.

Многие из представленных здесь методологий мы рассмотрим подробнее в последующих главах, в том числе в разделах, посвященных методологиям, в главах 5–10.

2.1. ТЕРМИНОЛОГИЯ Ниже перечислены ключевые термины, используемые в сфере анализа производительности систем. В последующих главах будут представлены дополнительные термины с описаниями в различных контекстах.

yy IOPS (input/output operations per second): количество операций ввода/выво-

да в секунду; является мерой частоты следования операций передачи данных. Для дискового ввода/вывода под количеством операций ввода/вывода (IOPS) подразумевается сумма операций чтения и записи в секунду.

yy Пропускная способность (throughput): скорость выполненной работы. В част-

ности, в области связи этот термин используется для обозначения скорости передачи данных (байтов или битов в секунду). В некоторых контекстах (например, в базах данных) под пропускной способностью может подразумеваться скорость работы (количество операций или транзакций в секунду).

yy Время отклика (response time): время от начала до завершения операции. Вклю-

чает в себя время на ожидание, на обслуживание (время обслуживания — service time) и на передачу результата.

68  Глава 2. Методологии

yy Задержка (latency): время, в течение которого операция ожидает обслуживания.

В некоторых контекстах под задержкой подразумевается все время выполнения операции, то есть время отклика. См. примеры в разделе 2.3 «Основные понятия».

yy Потребление (utilization): для ресурсов, необходимых для обслуживания, по-

требление определяет меру занятости ресурса, исходя из продолжительности в данном интервале времени, когда этот ресурс активно выполнял работу. Для ресурсов, которые обеспечивают хранение, под коэффициентом использования (потреблением) может подразумеваться потребленный объем (например, потребленный объем памяти).

yy Насыщенность (saturation): степень заполнения очереди заданий, которые ресурс не может обслужить немедленно.

yy Узкое место (bottleneck). В сфере анализа производительности систем под

узким местом подразумевается ресурс, ограничивающий производительность системы. Основная работа по повышению производительности системы как раз заключается в выявлении и устранении узких мест.

yy Рабочая нагрузка (workload): рабочей нагрузкой называются входные данные для системы или приложенная к ней нагрузка. Для базы данных рабочей нагрузкой является поток запросов и команд, отправляемых клиентами.

yy Кэш (cache): область быстродействующего хранилища, в которой могут хра-

ниться копии или буферы ограниченного объема данных. Кэш используется, чтобы предотвратить операции с более медленными хранилищами и тем самым повысить производительность. По экономическим причинам кэш часто меньше основного и более медленного хранилища.

В глоссарии можно найти описания дополнительных терминов.

2.2. МОДЕЛИ Следующие простые модели показывают некоторые основные принципы оценки производительности системы.

2.2.1. Тестируемая система На рис. 2.1 показаны характеристики тестируемой системы (System Under Test, SUT). Важно помнить, что на результат могут влиять посторонние возмущения (помехи), в том числе вызванные запланированной активностью системы, пользователями системы и другими рабочими нагрузками. Происхождение возмущений не всегда очевидно, и для их определения может потребоваться тщательное изучение характеристик системы. Это особенно сложно в некоторых облачных средах, где другие действия (со стороны подписчиков) в физической системе хоста могут быть недоступны для наблюдения из гостевой SUT.

2.3. Основные понятия  69

Возмущения

Вход

Тестируемая система

Рабочая нагрузка

Полученная оценка производительности

Рис. 2.1. Схема тестируемой системы Другая сложность современных окружений заключается в том, что они могут включать несколько сетевых компонентов, обслуживающих входную рабочую нагрузку, в том числе балансировщики нагрузки, прокси-серверы, веб-серверы, серверы кэширования, серверы приложений, серверы баз данных и системы хранения. Простое картографирование окружения может помочь выявить ранее упущенные из виду источники возмущений. Для аналитических исследований окружение также можно представить как сеть систем массового обслуживания (queueing systems).

2.2.2. Система массового обслуживания Некоторые компоненты и ресурсы можно смоделировать как систему массового обслуживания и получить возможность предсказывать их производительность в различных ситуациях. Диски, например, часто моделируют как систему массового обслуживания, которая может предсказать, насколько ухудшится время отклика под нагрузкой. На рис. 2.2 показана простая система массового обслуживания. Очередь

Выход

Время отклика

Вход Центр обслуживания

Рис. 2.2. Простая модель системы массового обслуживания Теория массового обслуживания, представленная в разделе 2.6 «Моделирование», изучает системы массового обслуживания и сети систем массового обслуживания.

2.3. ОСНОВНЫЕ ПОНЯТИЯ Далее приводится описание важных концепций в сфере оценки производительности систем, понимание которых понадобится в оставшейся части этой главы

70  Глава 2. Методологии и всей книги. Здесь приводится лишь общее описание понятий, а более подробные разъяснения, связанные с реализацией, будут даны в последующих главах — в разделах, посвященных архитектуре.

2.3.1. Задержка В некоторых средах задержка — единственный фактор, позволяющий хоть как-то оценить производительность. В других средах это одна из двух основных метрик наряду с пропускной способностью. В качестве примера на рис. 2.3 показана передача данных по сети, в данном случае запрос HTTP GET, с разделением времени на задержку и передачу данных. Время отклика

Завершение

Сетевой запрос на обслуживание Задержка на установку соединения

Время передачи данных

Рис. 2.3. Задержка при передаче данных по сети Задержка — это время ожидания до завершения операции. В этом примере операцией является запрос к сетевой службе на передачу данных. Прежде чем эта операция завершится, система должна дождаться, пока будет установлено сетевое соединение, что является задержкой для этой операции. Время отклика включает в себя эту задержку и время выполнения самой операции. Поскольку задержку можно измерить в разных местах, ее часто выражают с использованием цели измерений. Например, время загрузки страницы веб-сайта может включать в себя три разных значения времени, измеренных в разных местах: задержка DNS, задержка соединения TCP и время передачи данных TCP. Задержка DNS охватывает всю операцию разрешения имен в DNS. Задержка соединения TCP охватывает только инициализацию соединения («рукопожатие» TCP). На более высоком уровне все эти времена, включая время передачи данных TCP, можно рассматривать как задержку чего-то еще. Например, время с момента щелчка на ссылке до полной загрузки веб-страницы можно назвать задержкой, включающей время получения браузером страницы из сети и ее отображение на экране. Поскольку само слово «задержка» может быть неоднозначным, его лучше сопровождать уточняющими терминами, объясняющими, что именно измеряет эта задержка: задержка запроса, задержка соединения TCP и т. д. Поскольку задержка является метрикой времени, ее можно использовать в различных вычислениях. Проблемы производительности можно количественно оценить с помощью задержки и затем ранжировать, потому что они выражаются в одних и тех же единицах (времени). Также, опираясь на задержки, можно рассчитать прогнозируемое ускорение, определив степень уменьшения задержки. Подобные

2.3. Основные понятия  71 вычисления и прогнозы невозможны с использованием, например, только метрики IOPS. В табл. 2.1 приводятся краткие обозначения единиц измерения времени и их масштаб. Таблица 2.1. Единицы измерения времени Единица

Краткое обозначение

Величина в секундах

Минута

мин

60

Секунда

с

1

Миллисекунда

мс

0,001, или 1/1000, или 1×10–3

Микросекунда

мкс

0,000001, или 1/1000000, или 1×10–6

Наносекунда

нс

0,000000001, или 1/1000000000, или 1×10–9

Пикосекунда

пс

0,000000000001, или 1/1000000000000, или 1×10–12

Преобразование других типов метрик в задержку или время, если это возможно, позволяет сравнивать их. Если вам нужно было бы выбирать между 100 сетевыми и 50 дисковыми операциями ввода/вывода, как бы вы узнали, что лучше? Это сложный вопрос, требующий учета множества факторов: количества сетевых переходов, частоты сброса сетевых пакетов и повторных передач, объема ввода/вывода, характера ввода/вывода — произвольный или последовательный, типов дисков и т. д. Но если сравнивать 100 мс сетевого ввода/вывода и 50 мс дискового ввода/ вывода, разница очевидна.

2.3.2. Шкалы времени Время можно сравнивать численно, но у нас также есть интуитивное понимание времени, и мы можем выдвигать разумные предположения относительно задержек из разных источников. Компоненты системы работают в совершенно разных временных масштабах (отличающихся порой на порядки), разных до такой степени, что порой трудно понять, насколько велики эти различия. В табл. 2.2 представлены примеры задержек, начиная с доступа к регистру процессора с тактовой частотой 3,5 ГГц. Чтобы наглядно показать разницу во временных масштабах, с которыми приходится работать, в таблице показано также среднее время выполнения каждой операции в масштабе воображаемой системы, в которой один такт процессора длительностью в 0,3 нс (примерно одна треть от одной миллиардной доли секунды) занимает целую секунду. Как видите, один такт процессора длится очень короткое время. Чтобы преодолеть расстояние 0,5 м — примерно столько же, сколько от ваших глаз до этой страницы, — свету требуется около 1,7 нс. За это время современный процессор может выполнить пять тактов и обработать несколько инструкций.

72  Глава 2. Методологии Таблица 2.2. Масштабы задержек в системе Событие

Задержка

В масштабе

1 такт процессора

0,3 нс



Доступ к кэшу 1-го уровня

0,9 нс



Доступ к кэшу 2-го уровня

3 нс

10 с

Доступ к кэшу 3-го уровня

10 нс

33 с

Доступ к оперативной памяти (DRAM из процессора)

100 нс

6 мин

Продолжительность ввода/вывода для твердотельного накопителя (флеш-память)

10–100 мкс

9–90 часов

Продолжительность ввода/вывода для вращающегося жесткого диска

1–10 мс

1–12 месяцев

Интернет: время передачи сигнала от Сан-Франциско до Нью-Йорка

40 мс

4 года

Интернет: время передачи сигнала от Сан-Франциско до Великобритании

81 мс

8 лет

Облегченная загрузка с аппаратной виртуализацией

100 мс

11 лет

Интернет: время передачи сигнала от Сан-Франциско до Австралии

183 мс

19 лет

Загрузка системы с виртуализацией на уровне ОС

10.2.0.2.33986: Flags [P.], seq 1182098726:1182098918, ack 4234203806, win 132, options [nop,nop,TS val 1751498743 ecr 1751639660], length 192 00:00:00.000392 IP 10.2.0.2.33986 > 10.2.203.2.22: Flags [.], ack 192, win 501, options [nop,nop,TS val 1751639684 ecr 1751498743], length 0 00:00:00.009561 IP 10.2.203.2.22 > 10.2.0.2.33986: Flags [P.], seq 192:560, ack 1,

2.5. Методология  111 win 132, options [nop,nop,TS val 1751498744 ecr 1751639684], length 368 00:00:00.000351 IP 10.2.0.2.33986 > 10.2.203.2.22: Flags [.], ack 560, win 501, options [nop,nop,TS val 1751639685 ecr 1751498744], length 0 00:00:00.010489 IP 10.2.203.2.22 > 10.2.0.2.33986: Flags [P.], seq 560:896, ack 1, win 132, options [nop,nop,TS val 1751498745 ecr 1751639685], length 336 00:00:00.000369 IP 10.2.0.2.33986 > 10.2.203.2.22: Flags [.], ack 896, win 501, options [nop,nop,TS val 1751639686 ecr 1751498745], length 0 [...]

При необходимости tcpdump(8) может выводить больше или меньше информации (см. главу 10 «Сеть»). Трассировку событий ввода/вывода для блочных устройств хранения можно выполнить с помощью инструмента biosnoop(8) (основанного на BCC/BPF): # biosnoop TIME(s) 0.000004 0.000178 0.001469 0.001588 1.022346 [...]

COMM supervise supervise supervise supervise supervise

PID 1950 1950 1956 1956 1950

DISK xvda1 xvda1 xvda1 xvda1 xvda1

T W W W W W

SECTOR 13092560 13092432 13092440 13115128 13115272

BYTES 4096 4096 4096 4096 4096

LAT(ms) 0.74 0.61 1.24 1.09 0.98

В этом выводе команды biosnoop(8) можно видеть время завершения ввода/вывода (TIME(s)), информацию о процессе-инициаторе (COMM, PID), задействованное дисковое устройство (DISK), тип ввода/вывода (T), объем ввода/вывода (BYTES) и продолжительность (LAT(ms)). Дополнительную информацию об этом инструменте вы найдете в главе 9 «Диски». Системные вызовы — еще одна распространенная цель для трассировки. В Linux трассировку системных вызовов можно выполнить с помощью команды trace (см. главу 5 «Приложения») инструментов strace(1) и perf(1). Эти инструменты также поддерживают возможность вывода отметок времени. Трассируя события, ищите следующую информацию:

yy вход: все атрибуты события запроса: тип, направление, размер и т. д.; yy время: время начала, время конца, задержка (разница); yy результат: наличие ошибки, результат события (например, объем успешно переданных данных).

Иногда причины проблем с производительностью можно понять, изучив атрибуты события запроса или результата. Отметки времени событий позволяют анализировать задержки, и их вывод поддерживается многими инструментами трассировки событий. В предыдущем примере tcpdump(8) я включил вывод разности отметок времени между пакетами, добавив параметр -ttt. Изучение предшествующих событий дает дополнительную информацию. Событие с необычно высокой задержкой, известное как выброс задержки, может быть обу­ словлено предыдущими событиями, а не им самим. Например, событие в конце

112  Глава 2. Методологии очереди может иметь высокую задержку, обусловленную не его собственными свойствами, а событиями, поставленными в очередь ранее. Эту ситуацию можно определить по трассируемым событиям.

2.5.16. Базовые статистики Во многих средах используются решения мониторинга для записи метрик производительности и их визуализации в виде линейных графиков с временем на оси X (см. раздел 2.9 «Мониторинг»). Такие графики могут показать, менялась ли метрика в последнее время и как она менялась. Иногда добавляются дополнительные графики с историческими данными, такими как средние значения или просто соответствующие другим временным диапазонам, для сравнения с текущим диапазоном. Например, многие дашборды в Netflix отображают дополнительный график, соответствующий тому же временному диапазону, но за предыдущую неделю, благодаря чему характер работы системы в 15:00 во вторник можно сравнить с характером работы в 15:00 в предыдущий вторник. Такой подход хорошо зарекомендовал себя в анализе уже отслеживаемых метрик, графики изменений которых отображаются в GUI. Но в командной строке доступна масса системных метрик и подробностей, которые не отслеживаются решением мониторинга. Вы можете столкнуться с незнакомыми системными статистиками и задаться вопросом, являются ли они «нормальными» для сервера или свидетельствуют о проблеме. Это не новая задача, и для ее решения есть своя методология, которая появилась раньше широкого распространения решений мониторинга с использованием линейных графиков. Она заключается в сборе базовых статистик. Под этим подразумевается сбор всех системных метрик, когда система находится под «нормальной» нагрузкой, и их запись в текстовый файл или базу данных для дальнейшего использования. Программное обеспечение для сбора базовых статистик может быть реализовано как сценарий командной оболочки, который запускает инструменты наблюдения и собирает информацию из других источников (например, извлекая содержимое файлов в /proc с помощью cat(1)). Применение профилировщиков и инструментов трассировки позволяет зафиксировать гораздо больше деталей, чем это возможно с помощью обычных средств мониторинга продуктов (но будьте осторожны: эти инструменты имеют высокий оверхед и могут ухудшить производительность системы). Базовые статистики могут собираться на регулярной основе (например, ежедневно), а также до и после изменений в системе или приложении, чтобы можно было проанализировать различия в производительности. Если базовые статистики не собирались и мониторинг не производится, то для оценки текущей активности можно использовать некоторые инструменты наблюдения (основанные на счетчиках ядра), отображающие средние значения, накопленные с момента загрузки. Этот прием не дает достаточной точности, но все же лучше, чем ничего.

2.5. Методология  113

2.5.17. Статическая настройка производительности Статическая настройка производительности фокусируется на проблемах сконфигурированной архитектуры, в отличие от других методологий, которые сосредоточены на оценке динамической производительности — производительности при обработке приложенной нагрузки [Elling 00]. Статический анализ производительности выполняется, когда система находится в состоянии покоя и без нагрузки. Для статического анализа производительности и ее настройки нужно перечислить все компоненты системы и для каждого ответить на следующие вопросы:

yy Есть ли смысл использовать данный компонент? (Устаревший, низкопроизводительный и т. д.)

yy Имеет ли смысл данная конфигурация для предполагаемой рабочей нагрузки? yy Обеспечивает ли автоматически выбранная конфигурация компонента наилучшее соответствие предполагаемой рабочей нагрузке?

yy Возникла ли в компоненте какая-то ошибка, из-за которой пострадала производительность?

Вот несколько примеров проблем, которые можно обнаружить при статической настройке производительности:

yy yy yy yy

согласование сетевого интерфейса: выбор 1 Гбит/с вместо 10 Гбит/с; неисправный диск в пуле RAID; старая версия операционной системы, приложений или прошивки; файловая система почти заполнена (что может вызвать проблемы с производительностью);

yy несоответствие размера записи в файловой системе и размера блока ввода/вывода рабочей нагрузки;

yy приложение выполняется в случайно оставленном включенным режиме отладки; yy сервер по ошибке настроен как сетевой маршрутизатор (включена переадресация трафика);

yy сервер настроен на использование ресурсов, например аутентификации, удаленного вычислительного центра вместо локального.

К счастью, такие проблемы легко выявляются. Самое сложное — не забыть это сделать!

2.5.18. Настройка кэширования Приложения и операционные системы могут использовать несколько кэшей для увеличения производительности ввода/вывода, от приложения до дисков. Полный список вы найдете в главе 3 «Операционные системы», в разделе 3.2.11 «Кэширование». Вот общая стратегия настройки кэширования на каждом уровне:

114  Глава 2. Методологии 1. Старайтесь хранить данные в кэше как можно более высокого уровня, то есть как можно ближе к месту, где выполняется работа. Это уменьшит оверхед при попадании в кэш. В этом местоположении также должно быть доступно как можно больше метаданных, которые можно использовать для оптимизации политики хранения в кэше. 2. Убедитесь, что кэширование включено и работает. 3. Проверьте соотношение попаданий/промахов и частоту промахов. 4. Если размер кэша настраивается динамически, проверьте текущий размер. 5. Настройте кэш в соответствии с характером рабочей нагрузки. Эта задача зависит от доступных параметров настройки кэша. 6. Настройте рабочую нагрузку в соответствии с особенностями кэша, например, сократите количество потребителей кэша, чтобы освободить больше места для целевой рабочей нагрузки. Проверьте, нет ли двойного кэширования, когда существуют два разных кэша, которые потребляют оперативную память и хранят одни и те же данные. Также учитывайте общий прирост производительности, который дает настройка кэширования на каждом уровне. Настройка кэша процессора уровня 1 может сэко­ номить наносекунды, потому что промахи в этом кэше могут компенсироваться кэшем уровня 2. Но оптимизация кэширования в кэше процессора уровня 3 может дать гораздо больший прирост производительности за счет исключения обращений к более медленной оперативной памяти. (Кэши процессора описываются в главе 6 «Процессоры».)

2.5.19. Микробенчмаркинг производительности Микробенчмаркинг производительности позволяет оценить производительность простых и искусственных рабочих нагрузок. Этим он отличается от макробенчмаркинга производительности (или отраслевого эталонного тестирования), который обычно направлен на оценку фактической и естественной рабочей нагрузки. Макробенчмаркинг выполняется путем моделирования рабочих нагрузок и нередко оказывается сложным в реализации и трудным для понимания делом. Благодаря меньшему числу факторов, микробенчмаркинг проще в реализации и для понимания. Часто для микробенчмаркинга производительности в Linux используется инструмент iperf(1), который оценивает пропускную способность стека TCP: он может выявить узкие места во внешней сети (которые иначе трудно обнаружить) путем проверки счетчиков TCP во время приложения промышленной нагрузки. Микробенчмаркинг может выполняться с помощью инструмента микробенчмаркинга, который применяет рабочую нагрузку и оценивает производительность системы при ее обработке. Также можно использовать генератор нагрузки, который просто применяет рабочую нагрузку, оставляя измерения производительности другим

2.5. Методология  115 инструментам наблюдения (примеры генераторов нагрузки приводятся в главе 12 «Бенчмаркинг» в разделе 12.2.2 «Моделирование»). Оба подхода имеют право на жизнь, но инструменты микробенчмаркинга обычно безопаснее и позволяют перепроверить производительность с помощью других инструментов. Вот некоторые примеры целей для микробенчмаркинга, включая второе измерение для тестов:

yy время выполнения системного вызова: для fork(2), execve(2), open(2), read(2), close(2);

yy операции чтения из файловой системы: из кэшированного файла с разными размерами блоков от одного байта до одного мегабайта;

yy пропускная способность сети: передача данных между конечными точками TCP с разными размерами буфера сокета.

В процессе микробенчмаркинга обычно целевая операция выполняется с максимальной частотой и измеряется время, необходимое для выполнения большого числа этих операций. После этого вычисляется среднее время (среднее время = = общее время выполнения / количество операций). В последующих главах будут описаны конкретные методологии микробенчмаркинга, перечислены цели и атрибуты. Более подробно тема тестирования рассматривается в главе 12 «Бенчмаркинг».

2.5.20. Мантры производительности Цель этой методологии — показать, как лучше всего повысить производительность. Она перечисляет правила в порядке от наиболее эффективных к наименее эффективным: 1. Не делай этого. 2. Сделай это только один раз. 3. Делай меньше. 4. Делай позже. 5. Делай это, когда нет других дел. 6. Делай это параллельно. 7. Делай это оптимально. Вот несколько примеров реализации этих правил: 1. Не делай этого: избавьтесь от ненужной работы. 2. Сделай это только один раз: кэширование. 3. Делай меньше: настройте извлечение и обновление данных так, чтобы они выполнялись как можно реже.

116  Глава 2. Методологии 4. Делай позже: кэширование с отложенной записью. 5. Делай это, когда нет других дел: запланируйте выполнение работы в непиковые часы. 6. Делай это параллельно: используйте многопоточный режим вместо однопоточного. 7. Делай это оптимально: приобретите более производительное оборудование. Это одна из моих любимых методологий, с которой меня познакомил Скотт Эммонс (Scott Emmons) в Netflix. Он приписывает ее авторство Крейгу Хэнсону (Craig Hanson) и Пэту Крейну (Pat Crain), но мне не удалось найти подтверждение его слов.

2.6. МОДЕЛИРОВАНИЕ Аналитическое моделирование системы можно использовать для разных целей, в частности для анализа масштабируемости: изучения возможности масштабирования производительности с увеличением нагрузки или ресурсов. Ресурсы могут быть аппаратными (например, ядра процессора) или программными (процессы или потоки выполнения). Аналитическое моделирование можно рассматривать как третий способ оценки производительности наряду с наблюдением за промышленной системой («измерение») и экспериментальным тестированием («симуляция») [Jain 91]. Оценка производительности точнее, когда применяются по крайней мере два из этих способов: аналитическое моделирование и симуляция или симуляция и измерение. Анализ существующей системы можно начать с измерения: определения характера нагрузки и фактической производительности. Экспериментальный анализ путем симуляции рабочей нагрузки можно использовать, если в системе нет промышленной нагрузки или требуется оценить производительность при рабочих нагрузках, которые пока не наблюдаются в продакшене. Аналитическое моделирование можно использовать для прогнозирования производительности на основе результатов измерений или симуляции. Анализ масштабируемости может показать, что производительность перестает линейно масштабироваться в определенной точке из-за ограниченности ресурсов. Эта точка называется точкой перегиба: в ней одна функция сменяется другой, в данном случае линейное масштабирование сменяется конкуренцией за ресурсы. Выявление таких точек поможет сфокусироваться на исследовании проблем производительности, препятствующих масштабируемости, — чтобы исправить их до того, как они проявятся в промышленной среде. Более подробно эти шаги описываются в разделах 2.5.11 «Определение характеристик рабочей нагрузки» и 2.5.19 «Микробенчмаркинг производительности».

2.6. Моделирование  117

2.6.1. Корпоративные и облачные среды Моделирование позволяет имитировать крупномасштабные корпоративные системы без затрат на владение ими, но производительность крупномасштабных сред складывается из очень большого числа факторов и ее трудно смоделировать точно. С развитием облачных вычислений появилась возможность арендовать на короткий срок (на время тестирования) среду любого масштаба. Вместо создания математической модели для прогнозирования производительности можно определить характер рабочей нагрузки, смоделировать ее и затем протестировать в облачных средах разного масштаба. Некоторые результаты, такие как точки перегиба, могут остаться прежними, но теперь будут подтверждены фактическими измерениями, а не только теоретическими моделями. Кроме того, при тестировании в реальной среде можно обнаружить ограничители, не учтенные в вашей модели.

2.6.2. Визуальная идентификация Когда есть возможность получить достаточный объем экспериментальных данных, построение графика зависимости производительности от параметра масштабирования может выявить закономерности. На рис. 2.15 показана зависимость пропускной способности приложения от числа потоков выполнения. Как видно на графике, точка перегиба находится в районе восьми потоков, где наклон линии графика меняется. Теперь можно исследовать причины появления этой точки перегиба, например, изучив конфигурацию приложения и системы при количестве потоков, близком к восьми. В данном случае система была оснащена восьмиядерным процессором, каждое ядро ​​имело два аппаратных потока. Чтобы убедиться, что точка перегиба связана с количеством ядер процессора, можно исследовать производительность процессоров с меньшим или большим числом ядер (например, инструкций на такт — IPC; см. главу 6 «Процессоры»). Или можно повторить тестирование масштабируемости в системе с другим количеством ядер и подтвердить, что точка перегиба перемещается, как ожидалось. Есть ряд профилей масштабируемости, которые можно идентифицировать визуально, без использования формальной модели. Они показаны на рис. 2.16. Для каждого из них ось X определяет масштаб, а ось Y — итоговую производительность (пропускную способность, количество транзакций в секунду и т. д.). Вот эти профили:

yy Линейная масштабируемость: производительность увеличивается пропорцио­

нально масштабированию ресурса. Линейный рост не может продолжаться вечно и часто является ранней стадией другого профиля масштабируемости.

yy Масштабируемость с конкуренцией: некоторые компоненты архитектуры являются общими и могут использоваться только последовательно; конкуренция за эти общие ресурсы снижает эффективность масштабирования.

118  Глава 2. Методологии

yy Масштабируемость с когерентностью: затраты на поддержание когерентности

данных, включая распространение изменений, начинают перевешивать преимущества масштабирования.

yy Точка перегиба: имеет место фактор, который меняет профиль масштабируемости.

yy Потолок масштабируемости: достигнут жесткий предел. Это может быть устрой-

Пропускная способность

ство, являющееся узким местом, например шина с ограниченной пропускной способностью или предел, установленный программно (в системе управления ресурсами).

Потоки выполнения

Рис. 2.15. Результаты тестирования масштабируемости Визуальная идентификация может быть простой и эффективной, тем не менее использование математической модели позволит узнать больше о масштабируемости системы. Модель может иметь неожиданные отклонения от фактических данных, которые полезно исследовать и выяснить, заключается ли причина в модели и, следовательно, в вашем понимании системы или проблема в реальной масштабируемости. В следующих разделах представлены закон Амдала, универсальный закон масштабируемости и теория массового обслуживания.

2.6. Моделирование  119

Линейная

С конкуренцией

Точка перегиба

С когерентностью

Потолок

Рис. 2.16. Профили масштабируемости

2.6.3. Закон Амдала Названный в честь компьютерного архитектора Джина Амдала (Gene Amdahl) [Amdahl 67], этот закон моделирует масштабируемость системы с учетом компонентов рабочих нагрузок, имеющих последовательный характер, которые нельзя обрабатывать параллельно. Его можно использовать для изучения масштабирования процессоров, потоков выполнения, рабочих нагрузок и т. д. Закон Амдала можно наблюдать в профиле масштабируемости «с конкуренцией», показанном выше, который иллюстрирует конкуренцию за последовательный ресурс или компонент рабочей нагрузки. Его можно определить как [Gunther 97]: C(N) = N/(1 + α(N – 1)), где C(N) — относительная емкость, а N — размерность масштабирования, например количество процессоров или пользовательская нагрузка. Параметр α (0 zw\321\353..."...,

65536) 65536) 65536) 65536) 65536) 65536)

= = = = = =

65536 65536 65536 65536 65536 65536





Параметр -tt обеспечивает вывод отметок времени слева, а -T — продолжительность выполнения системных вызовов справа. Все вызовы read(2) прочитали по 64 Кбайт, продолжительность выполнения первого вызова составила 18 мс, второго 56 мкс (вероятно, данные были получены из кэша) и третьего 5 мс. Операции чтения выполнялись с файловым дескриптором 9. Чтобы проверить, относится ли этот дескриптор к файловой системе (а не к сокету), поищите в выводе выше вызов open(2) либо воспользуйтесь другим инструментом, таким как lsof(8). Также информацию о дескрипторе 9 можно найти в файловой системе /proc: /proc/845/ fd{,info}/9 . Учитывая оверхед, характерный для текущей реализации strace(1), измеренная задержка может быть искажена из-за эффекта наблюдателя. Обратите внимание на новые инструменты трассировки, включая ext4slower(8), которые для уменьшения оверхеда и более точного измерения задержек используют трассировку с буферизацией для каждого процессора и BPF.

8.6.8. fatrace fatrace(1) — специализированный трассировщик, использующий прикладной интерфейс fanotify (file access notify — уведомление о попытках доступа к файлу) в Linux. Вот пример вывода этой команды: # fatrace sar(25294): sar(25294): sar(25294): sar(25294): sar(25294): sar(25294):

O /etc/ld.so.cache RO /lib/x86_64-linux-gnu/libc-2.27.so C /etc/ld.so.cache O /usr/lib/locale/locale-archive O /usr/share/zoneinfo/America/Los_Angeles RC /usr/share/zoneinfo/America/Los_Angeles

506  Глава 8. Файловые системы sar(25294): RO /var/log/sysstat/sa09 sar(25294): R /var/log/sysstat/sa09 [...]

В каждой строке выводятся: имя процесса, идентификатор процесса PID, тип события, полный путь и необязательный статус. Поддерживаются типы событий: (O) — открытие (open), (R) — чтение (read), (W) — запись (write) и (C) — закрытие (close). fatrace(1) можно использовать для определения таких характеристик рабочей нагрузки, как имена файлов, к которым осуществляется доступ, и поиска случаев, когда выполняется ненужная работа, которой можно избежать. Но для высоконагруженных файловых систем fatrace(1) может выводить десятки тысяч строк в секунду и расходовать значительные вычислительные ресурсы. Это немного смягчается фильтрацией по типу события. Также можно использовать инструменты трассировки на основе BPF, включая opensnoop(8) (раздел 8.6.10), отличающиеся незначительным оверхедом.

8.6.9. LatencyTOP LatencyTOP — это инструмент для составления отчетов об источниках задержки по системе в целом и по отдельным процессам. LatencyTOP сообщает и о задержках в файловой системе. Например: Cause Reading from file synchronous write Marking inode dirty Waiting for a process to die Waiting for event (select) Page fault Process gzip (10969) Reading from file synchronous write Marking inode dirty

Maximum 209.6 msec 82.6 msec 7.9 msec 4.6 msec 3.6 msec 0.2 msec Total: 442.4 209.6 82.6 7.9

Percentage 61.9 % 24.0 % 2.2 % 1.5 % 10.1 % 0.2 %

msec msec msec msec

70.2 % 27.2 % 2.5 %

В верхней части выводится сводная информация для системы в целом, а в нижней — для единственного процесса gzip(1), который в данный момент сжимает файл. Больше всего задержек вносят операции чтения исходного файла (Reading from file) — 70,2 % и операции синхронной записи (synchronous write) — 27,2 % в новый сжатый файл. Инструмент LatencyTOP был разработан в Intel, но давно не обновлялся, и сайт проекта больше не доступен. Кроме того, для поддержки этого инструмента ядро должно быть скомпилировано с определенными параметрами, которые обычно выключены1. Возможно, будет проще измерить задержку в файловой системе с помощью инструментов трассировки BPF, см. разделы 8.6.13–8.6.15. CONFIG_LATENCYTOP и CONFIG_HAVE_LATENCYTOP_SUPPORT.

1

8.6. Инструменты наблюдения  507

8.6.10. opensnoop opensnoop(8)1 — это инструмент для BCC и bpftrace, который трассирует операции открытия файлов. Может использоваться для определения местоположения файлов данных, журналов и конфигураций. Он также обнаруживает проблемы с производительностью, вызванные частым открытием, и помогает устранить проблемы с отсутствующими файлами. Вот пример вывода, полученный в производственной системе, с параметром -T для включения в вывод отметок времени: # opensnoop -T TIME(s) PID 0.000000000 26447 [...] 1.961686000 25983 1.961715000 25983 1.961770000 25983 1.961799000 25983 1.961818000 25983 1.961843000 25983 1.961862000 25983 [...] 2.438417000 25983 [...] 2.816953000 25983 2.818827000 25983 2.820621000 25983 [...]

COMM sshd

FD ERR PATH 5 0 /var/log/btmp

mysqld mysqld mysqld mysqld mysqld mysqld mysqld

4 5 5 5 5 5 5

0 0 0 0 0 0 0

/etc/mysql/my.cnf /etc/mysql/conf.d/ /etc/mysql/conf.d/mysql.cnf /etc/mysql/conf.d/mysqldump.cnf /etc/mysql/mysql.conf.d/ /etc/mysql/mysql.conf.d/mysql.cnf /etc/mysql/mysql.conf.d/mysqld.cnf

mysqld

4

0 /var/log/mysql/error.log

mysqld mysqld mysqld

30 31 4

0 ./binlog.000024 0 ./binlog.index_crash_safe 0 ./binlog.index

В этом выводе зафиксирован запуск базы данных MySQL, и как видите, opensnoop(2) обнаружил файлы конфигурации, файл журнала, файлы данных (двоичные журналы) и многое другое. opensnoop(8) трассирует варианты системного вызова open(2): open(2) и openat(2). Оверхед на трассировку обычно незначителен, потому что open(2) вызывается нечасто. В числе параметров версии для BCC можно назвать:

yy yy yy yy

-T: добавить колонку с отметками времени; -x: показать только неудачные вызовы open; -p PID: трассировать только процесс с этим PID; -n NAME: выводить только имена процессов, содержащие NAME.

Параметр -x можно использовать при решении проблем: он выводит информацию о неудачных попытках открыть файл.

Немного истории: первую версию opensnoop я создал в 2004 году, версию для BCC — 17 сентября 2015 года, а для bpftrace — 8 сентября 2018 года.

1

508  Глава 8. Файловые системы

8.6.11. filetop filetop(8)1 — это инструмент для BCC, действующий как утилита top(1), только для файлов. Он показывает имена наиболее часто используемых файлов. Вот пример вывода: # filetop Tracing... Output every 1 secs. Hit Ctrl-C to end 19:16:22 loadavg: 0.11 0.04 0.01 3/189 23035 TID 23033 23033 23032 23031 23032 23032 23035 [...]

COMM mysqld mysqld oltp_read_only. oltp_read_only. oltp_read_only. oltp_read_only. systemd-udevd

READS 481 3 3 3 1 4 4

WRITES 0 0 0 0 0 0 0

R_Kb 7681 48 20 20 19 16 16

W_Kb 0 0 0 0 0 0 0

T R R R R R R R

FILE sb1.ibd mysql.ibd oltp_common.lua oltp_common.lua Index.xml openssl.cnf sys_vendor

По умолчанию выводятся двадцать наиболее часто используемых файлов, отсор­тированных по столбцу прочитанных байтов. Как показывает верхняя строка, mysqld выполнил 481 операцию чтения из файла sb1.ibd общим объемом 7681 Кбайт. Этот инструмент используют для определения характера рабочей нагрузки и наблюдения за файловой системой в целом. Как top(1) выявляет процессы с неожиданно большим потреблением процессора, этот инструмент помогает обнаружить файлы, использующиеся необычно интенсивно. По умолчанию filetop выводит информацию только об обычных файлах. Но если добавить параметр -a, то инструмент выведет информацию обо всех файлах, включая сокеты TCP: # filetop -a [...] TID COMM 21701 sshd 23033 mysqld 23335 sshd 1 systemd [...]

READS 1 1 1 4

WRITES 0 0 0 0

R_Kb 16 16 8 4

W_Kb 0 0 0 0

T O R S R

FILE ptmx sbtest1.ibd TCP comm

Этот вывод содержит информацию о дополнительных типах файлов: другой (O, other) и сокет (S, socket). В этом случае к другому типу отнесен файл ptmx — специальный файл символьного устройства в каталоге /dev.

1

Немного истории: версию для BCC я создал 6 февраля 2016 года, вдохновившись утилитой top(1) Уильяма Лефевра.

8.6. Инструменты наблюдения  509 В числе поддерживаемых параметров можно назвать:

yy yy yy yy

-C: не очищать экран — выводить данные с прокруткой; -a: показать файлы всех типов; -r ROWS: выводить указанное число строк (по умолчанию 20); -p PID: трассировать только процесс с этим PID.

Экран обновляется каждую секунду (как в top(1)), если не задан параметр -C. Я предпочитаю использовать команду с параметром -C, чтобы вывод сохранялся в буфере прокрутки терминала на тот случай, если мне понадобится исследовать данные.

8.6.12. cachestat cachestat(8)1 — это инструмент для BCC, который выводит статистику попаданий и промахов кэша страниц. Его можно использовать для проверки доли попаданий и эффективности кэша страниц, а также для исследования влияния разных настроек системы и приложений и получения информации об эффективности кэша. Вот пример вывода: $ cachestat -T 1 TIME HITS 21:00:48 586 21:00:49 125 21:00:50 113 21:00:51 23 21:00:52 134 [...]

MISSES 0 0 0 0 0

DIRTIES HITRATIO 1870 100.00% 1775 100.00% 1644 100.00% 1389 100.00% 1906 100.00%

BUFFERS_MB 208 208 208 208 208

CACHED_MB 775 776 776 776 777

Этот вывод показывает рабочую нагрузку чтения, которая получает все данные исключительно из кэша (количество попаданий выводится в столбце HITS, а доля попаданий — 100 % в данном примере — в столбце HITRATIO), и более высокую рабочую нагрузку записи (столбец DIRTIES). В идеале желательно, чтобы доля попаданий была как можно ближе к 100 %, чтобы операции чтения не блокировали приложения для выполнения дискового ввода/вывода. Столкнувшись с низкой долей попаданий в кэш, из-за чего может наблюдаться снижение производительности, попробуйте немного уменьшить объем памяти, выделяемой приложению, чтобы для кэша страниц осталось больше места. Если настроены устройства подкачки, то настройте подкачку, чтобы предпочтение отдавалось удалению страниц из кэша вместо подкачки. Из параметров можно назвать -T, включающий вывод отметок времени.

Немного истории: первую экспериментальную версию инструмента на основе Ftrace я создал 28 декабря 2014 года. На ее основе Аллан Макаливи (Allan McAleavy) реализовал версию для BCC 6 ноября 2015 года.

1

510  Глава 8. Файловые системы Этот инструмент дает ценную информацию о доле попаданий в кэш страниц, но пока остается экспериментальным и использует зонды kprobes для трассировки определенных функций ядра. Может потребоваться изменить его исходный код для работы с разными версиями ядра. Еще лучше было бы переписать этот инструмент и использовать в нем точки трассировки или статистики из /proc, чтобы улучшить стабильность. Лучшее применение этого инструмента на сегодняшний день — демонстрация того, что такое вообще возможно.

8.6.13. ext4dist (xfs, zfs, btrfs, nfs) ext4dist(8)1 — это инструмент для BCC и bpftrace, позволяющий инструментировать файловую систему ext4 и исследовать гистограммы распределения задержек типичных операций: чтения, записи, открытия и синхронизации. Есть версии и для других файловых систем: xfsdist(8), zfsdist(8), btrfsdist(8) и nfsdist(8). Вот пример вывода: # ext4dist 10 1 Tracing ext4 operation latency... Hit Ctrl-C to end. 21:09:46: operation = read usecs 0 -> 1 2 -> 3 4 -> 7 8 -> 15 16 -> 31 32 -> 63 64 -> 127 128 -> 255 256 -> 511 512 -> 1023 1024 -> 2047

: : : : : : : : : : : :

count 783 88 449 1306 48 12 39 11 158 110 33

distribution |*********************** | |** | |************* | |****************************************| |* | | | |* | | | |**** | |*** | |* |

operation = write usecs 0 -> 1 2 -> 3 4 -> 7 8 -> 15 16 -> 31 32 -> 63 64 -> 127 128 -> 255

: : : : : : : : :

count 1073 324 1378 1505 183 37 11 9

distribution |**************************** | |******** | |************************************ | |****************************************| |**** | | | | | | |

operation = open usecs 0 -> 1

: count : 672

distribution |****************************************|

1

Немного истории: версию для BCC я создал 12 февраля 2016 года, а версию для bpftrace — 2 февраля 2019 года [Gregg 19]. За основу был взят аналогичный инструмент для ZFS, который я написал в 2012 году.

8.6. Инструменты наблюдения  511 2 -> 3 operation = fsync usecs 256 -> 511 512 -> 1023 1024 -> 2047 2048 -> 4095 4096 -> 8191 8192 -> 16383

: 10

|

: : : : : : :

distribution |********** | |****** | |****************************************| |* | | | | |

count 485 308 1779 79 26 4

|

Параметры команды в этом примере определяют 10-секундный интервал и однократный вывод результатов трассировки, накопленных в течение этого интервала. Как показывает вывод, у задержек чтения бимодальное распределение: одна мода находится в интервале от 0 до 15 мкс и, вероятно, связана с попаданиями в кэш, а другая — в интервале от 256 до 2048 мкс и, вероятно, соответствует операциям чтения с диска. Можно исследовать и распределение задержек в других операциях. Запись выполняется быстро, вероятно, из-за сохранения данных в буферы, которые позже выталкиваются на диск с помощью более медленной операции синхронизации (fsync). Этот и сопутствующий ему инструмент ext4slower(8) (см. следующий раздел) показывают задержки, с которыми могут столкнуться приложения. Измерение задержек на уровне диска возможно и показано в главе 9, но приложения могут не блокироваться операциями дискового ввода/вывода, что затрудняет интерпретацию таких измерений. По возможности я использую инструменты ext4dist(8) и ext4slower(8) перед инструментами измерения задержки дискового ввода/вывода. См. раздел 8.3.12, где описываются различия между логическим вводом/выводом на уровне файловых систем, продолжительность которого измеряется этим инструментом, и физическим вводом/выводом на уровне дисков. В числе поддерживаемых параметров можно назвать:

yy -m: выводит время в миллисекундах; yy -p PID: трассировать только процесс с этим PID. Вывод этого инструмента можно представить в виде тепловой карты задержек. Чтобы получить больше информации о медленном вводе/выводе файловой системы, используйте ext4slower(8) и его варианты.

8.6.14. ext4slower (xfs, zfs, btrfs, nfs) ext4slower(8)1 трассирует выполнение обычных операций в файловой системе ext4 (чтение, запись, открытие и синхронизация) и сообщает о тех, продолжительность которых превышает определенный порог. Вот пример вывода: Немного истории: этот инструмент я  создал 11 февраля 2016 года, взяв за основу аналогичный инструмент для ZFS, который написал в 2011 году.

1

512  Глава 8. Файловые системы # ext4slower Tracing ext4 operations TIME COMM 21:36:03 mysqld 21:36:15 mysqld 21:36:15 mysqld 21:36:15 mysqld 21:36:15 mysqld [...]

slower PID 22935 22935 22935 22935 22935

than 10 ms T BYTES OFF_KB S 0 0 S 0 0 S 0 0 S 0 0 S 0 0

LAT(ms) 12.81 12.13 10.46 13.66 11.79

FILENAME sbtest1.ibd ib_logfile1 binlog.000026 ib_logfile1 ib_logfile1

В столбцах выводятся: текущее время (TIME), имя процесса (COMM), идентификатор процесса (PID), тип операции (T: R — чтение, W — запись, O — открытие и S — синхронизация), смещение в килобайтах (OFF_KB), задержка в миллисекундах (LAT(ms)) и имя файла (FILENAME). В этом примере видно несколько операций синхронизации (S), длившихся дольше 10 мс (пороговое значение по умолчанию). Порог можно передать в виде аргумента. Если указать 0 мс, то в выводе будут показаны все операции: # ext4slower 0 Tracing ext4 operations 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld 21:36:50 mysqld [...]

22935 22935 22935 22935 22935 22935 22935 22935 22935 22935 22935 22935

W W W S W S W W W W W W

917504 1024 512 0 1746 0 16384 16384 16384 16384 16384 16384

2048 14165 14166 0 21714 0 4640 11504 13248 11808 1328 6768

0.42 0.00 0.00 3.21 0.02 5.56 0.01 0.01 0.01 0.01 0.01 0.01

ibdata1 ib_logfile1 ib_logfile1 ib_logfile1 binlog.000026 ibdata1 undo_001 sbtest1.ibd sbtest1.ibd sbtest1.ibd undo_001 undo_002

В этом примере видна закономерность: mysqld выполняет синхронизацию после записи в файлы. При трассировке всех операций можно получить большой объем вывода с соответствующим оверхедом. Я выполняю такую трассировку только короткое время (например, не дольше 10 с), чтобы понять закономерности работы файловой системы, которые не видны в выводе других инструментов (ext4dist(8)). В числе поддерживаемых параметров можно назвать -p PID для трассировки только одного процесса и -j для вывода результатов в формате CSV.

8.6.15. bpftrace bpftrace — это трассировщик на основе BPF, который предоставляет язык программирования высокого уровня, позволяющий создавать мощные однострочные и короткие сценарии. Он хорошо подходит для анализа работы файловой системы на основе подсказок, полученных с помощью других инструментов.

8.6. Инструменты наблюдения  513 Подробнее о bpftrace я рассказываю в главе 15. В этом разделе показаны некоторые примеры анализа файловой системы: однострочные сценарии и приемы трассировки системных вызовов, VFS и внутренних механизмов файловой системы.

Однострочные сценарии Однострочные сценарии ниже показывают различные возможности bpftrace. Трассировка операций открытия файлов через вызов openat(2) с именами процессов: bpftrace -e 't:syscalls:sys_enter_openat { printf("%s %s\n", comm, str(args->filename)); }'

Подсчет обращений к системным вызовам из семейства read: bpftrace -e 'tracepoint:syscalls:sys_enter_*read* { @[probe] = count(); }'

Подсчет обращений к системным вызовам из семейства write: bpftrace -e 'tracepoint:syscalls:sys_enter_*write* { @[probe] = count(); }'

Выводит гистограмму с распределением вызовов read() по запрошенным размерам блоков: bpftrace -e 'tracepoint:syscalls:sys_enter_read { @ = hist(args->count); }'

Выводит гистограмму с распределением вызовов read() по размерам прочитанных блоков (и кодам ошибок): bpftrace -e 'tracepoint:syscalls:sys_exit_read { @ = hist(args->ret); }'

Подсчитывает количество вызовов read(), завершившихся ошибкой, для каждого кода ошибки: bpftrace -e 't:syscalls:sys_exit_read /args->ret < 0/ { @[- args->ret] = count(); }'

Подсчитывает количество вызовов VFS: bpftrace -e 'kprobe:vfs_* { @[probe] = count(); }'

Подсчитывает количество вызовов VFS для процесса с PID 181: bpftrace -e 'kprobe:vfs_* /pid == 181/ { @[probe] = count(); }'

Подсчитывает количество прохождений через точки трассировки в ext4: bpftrace -e 'tracepoint:ext4:* { @[probe] = count(); }'

Подсчитывает количество прохождений через точки трассировки в xfs: bpftrace -e 'tracepoint:xfs:* { @[probe] = count(); }'

514  Глава 8. Файловые системы Подсчитывает число операций чтения файлов в файловой системе в ext4 по именам процессов и трассировкам стека в пространстве пользователя: bpftrace -e 'kprobe:ext4_file_read_iter { @[ustack, comm] = count(); }'

Определяет время выполнения spa_sync() в ZFS: bpftrace -e 'kprobe:spa_sync { time("%H:%M:%S ZFS spa_sync()\n"); }'

Подсчитывает число обращений к кэшу каталогов по именам и идентификаторам (PID) процессов: bpftrace -e 'kprobe:lookup_fast { @[comm, pid] = count(); }'

Трассировка системных вызовов Системные вызовы — отличная цель для многих инструментов трассировки. Но некоторым системным вызовам не хватает контекста файловой системы, что затрудняет их анализ. Я покажу примеры того, что возможно осуществить с предлагаемыми средствами (трассировка openat(2)), а что может вызвать затруднения (трассировка read(2)).

openat(2) Выполняя трассировку семейства системных вызовов open(2), можно узнать, какие файлы открываются. Сейчас наиболее часто используется вариант openat(2). Вот пример его трассировки: # bpftrace -e 't:syscalls:sys_enter_openat { printf("%s %s\n", comm, str(args->filename)); }' Attaching 1 probe... sa1 /etc/sysstat/sysstat sadc /etc/ld.so.cache sadc /lib/x86_64-linux-gnu/libsensors.so.5 sadc /lib/x86_64-linux-gnu/libc.so.6 sadc /lib/x86_64-linux-gnu/libm.so.6 sadc /sys/class/i2c-adapter sadc /sys/bus/i2c/devices sadc /sys/class/hwmon sadc /etc/sensors3.conf [...]

Здесь мне удалось поймать запуск sar(1) для архивирования статистик, и теперь я могу видеть, какие файлы он открывал. Этот сценарий использует аргумент filename точки трассировки. Все доступные аргументы можно получить с помощью параметров -lv: # bpftrace -lv t:syscalls:sys_enter_openat tracepoint:syscalls:sys_enter_openat int __syscall_nr; int dfd;

8.6. Инструменты наблюдения  515 const char * filename; int flags; umode_t mode;

Как видите, эта точка трассировки получает в аргументах номер системного вызова, дескриптор файла, имя файла, флаги открытия и режим открытия. Этой информации достаточно для однострочных сценариев и инструментов, например opensnoop(8).

read(2) read(2) мог бы быть полезной целью трассировки для понимания задержек операций чтения в файловой системе. Но взгляните на аргументы точки трассировки (сможете ли вы определить проблему?): # bpftrace -lv t:syscalls:sys_enter_read tracepoint:syscalls:sys_enter_read int __syscall_nr; unsigned int fd; char * buf; size_t count;

read(2) может вызываться для выполнения операций с файловыми системами, сокетами, /proc и другими целями, но имеющиеся аргументы не позволяют различить их. Чтобы понять, какие это вызывает сложности, взгляните на результаты сценария, подсчитывающего обращения к системному вызову read(2) по именам процессов: # bpftrace -e 't:syscalls:sys_enter_read { @[comm] = count(); }' Attaching 1 probe... ^C @[systemd-journal]: 13 @[sshd]: 141 @[java]: 3472

В период трассировки Java-процесс 3472 раза обратился к системному вызову read(2), но с какой целью — для чтения данных из файловой системы, сокета или откуда-то еще? (Процесс sshd, вероятнее всего, читал данные из сокета.) При трассировке системного вызова read(2) можно узнать дескриптор файла (FD), представленный целым числом, но это всего лишь число, которое не отражает тип FD (bpftrace работает в ограниченном режиме ядра и не может выполнить поиск информации о FD в /proc). У этой проблемы минимум четыре решения:

yy Вывести значения PID и FD в сценарии bpftrace, а затем отыскать FD с помощью lsof(8) или /proc, чтобы узнать, что он представляет.

yy Вспомогательная функция get_fd_path() в BPF, внедрение которой ожидается в ближайшее время, может возвращать путь к файлу, соответствующему дескриптору FD. Это поможет отличить операции чтения из файловой системы (имеющие путь) от других типов чтения.

516  Глава 8. Файловые системы

yy Использовать трассировку VFS, где доступно больше структур данных. yy Напрямую трассировать функции файловой системы, что исключает другие виды ввода/вывода. Этот подход используется в ext4dist(8) и ext4slower(8).

В разделе, посвященном анализу задержек в VFS, показано решение на основе VFS.

Трассировка VFS Виртуальная файловая система (VFS) абстрагирует все файловые системы (и другие устройства) и обеспечивает единую точку для наблюдения за всеми файловыми системами.

Количество вызовов VFS Подсчет вызовов VFS позволяет получить общее представление о типах используемых операций. Пример ниже показывает подсчет с помощью kprobes вызовов функций ядра, имена которых начинаются с «vfs_»: # bpftrace -e 'kprobe:vfs_* { @[func] = count(); }' Attaching 65 probes... ^C [...] @[vfs_statfs]: 36 @[vfs_readlink]: 164 @[vfs_write]: 364 @[vfs_lock_file]: 516 @[vfs_iter_read]: 2551 @[vfs_statx]: 3141 @[vfs_statx_fd]: 4214 @[vfs_open]: 5271 @[vfs_read]: 5602 @[vfs_getattr_nosec]: 7794 @[vfs_getattr]: 7795

Этот сценарий выводит количество операций, выполняемых в масштабе всей системы. За время трассировки функция vfs_read(), например, была вызвана 7795 раз.

Измерение задержек в VFS Как и в случае с системными вызовами, операции чтения в VFS выполняются с файловыми системами, сокетами и другими источниками. Сценарий ниже для bpftrace извлекает тип операции из структуры ядра (имя суперблока индексного узла) и разбивает задержки в vfs_read() по типам: # vfsreadlat.bt Tracing vfs_read() by type... Hit Ctrl-C to end. ^C [...] @us[sockfs]: [0] 141 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| [1] 91 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ |

8.6. Инструменты наблюдения  517 [2, 4) [4, 8) [8, 16) [16, 32) [...]

57 53 86 2

|@@@@@@@@@@@@@@@@@@@@@ |@@@@@@@@@@@@@@@@@@@ |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ |

| | | |

@us[proc]: [0] [1] [2, 4) [4, 8) [8, 16) [16, 32) [32, 64) [64, 128)

242 41 40 61 44 40 6 3

|@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@ | |@@@@@@@@ | |@@@@@@@@@@@@@ | |@@@@@@@@@ | |@@@@@@@@ | |@ | | |

@us[ext4]: [0] [1] [2, 4) [4, 8) [8, 16) [16, 32) [32, 64) [64, 128) [128, 256) [256, 512) [512, 1K) [1K, 2K)

653 447 70 774 417 25 7 170 55 59 118 3

|@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@ | | | |@@@@@@@@@@@ | |@@@ | |@@@ | |@@@@@@@ | |@@ |

Вывод (усеченный) также включает гистограммы распределения задержек для: sysfs, devpts, pipefs, devtmpfs, tmpfs и anon_inodefs. Вот исходный код сценария: #!/usr/local/bin/bpftrace #include BEGIN { printf("Tracing vfs_read() by type... Hit Ctrl-C to end.\n"); } kprobe:vfs_read { @file[tid] = ((struct file *)arg0)->f_inode->i_sb->s_type->name; @ts[tid] = nsecs; } kretprobe:vfs_read /@ts[tid]/ { @us[str(@file[tid])] = hist((nsecs - @ts[tid]) / 1000); delete(@file[tid]); delete(@ts[tid]); }

518  Глава 8. Файловые системы END { }

clear(@file); clear(@ts);

При желании этот инструмент можно расширить, включив в него другие операции — vfs_readv(), vfs_write(), vfs_writev() и т. д. Чтобы понять этот код, прочитайте раздел 15.2.4 «Программирование», в котором объясняются основы измерения задержек в vfs_read(). Обратите внимание, что эта задержка может и не влиять напрямую на производительность приложения, как говорилось в разделе 8.3.1 «Задержки в файловой системе». Многое зависит от того, чем обусловлена задержка — запросом от приложения или работой асинхронной фоновой задачи. Чтобы ответить на этот вопрос, можно добавить в гистограмму дополнительный ключ — трассировку стека в пространстве пользователя (ustack), который может показать, был ли вызов vfs_read() произведен самим приложением.

Внутренние механизмы файловой системы При необходимости можно разработать собственные инструменты, показывающие поведение внутренних компонентов файловой системы. Прежде всего проверьте имеющиеся точки трассировки. Вот как получить список точек трассировки для ext4: # bpftrace -l 'tracepoint:ext4:*' tracepoint:ext4:ext4_other_inode_update_time tracepoint:ext4:ext4_free_inode tracepoint:ext4:ext4_request_inode tracepoint:ext4:ext4_allocate_inode tracepoint:ext4:ext4_evict_inode tracepoint:ext4:ext4_drop_inode [...]

Каждая из них имеет свои аргументы, список которых можно получить с помощью параметров -lv. Если точек трассировки недостаточно (или они недоступны для целевой файловой системы), воспользуйтесь динамической инструментацией с помощью kprobes. Вот список зондов kprobe для ext4: # bpftrace -lv 'kprobe:ext4_*' kprobe:ext4_has_free_clusters kprobe:ext4_validate_block_bitmap kprobe:ext4_get_group_number kprobe:ext4_get_group_no_and_offset kprobe:ext4_get_group_desc kprobe:ext4_wait_block_bitmap [...]

В этом ядре (5.3) файловая система ext4 имеет 105 точек трассировки и 538 зондов kprobe.

8.6. Инструменты наблюдения  519

8.6.17. Другие инструменты В табл. 8.7 перечислены инструменты наблюдения за работой файловой системы, представленные в других главах этой книги и в «BPF Performance Tools» [Gregg 19]. Таблица 8.7. Другие инструменты наблюдения за работой файловой системы Раздел

Инструмент

Описание

5.5.6

syscount

Подсчитывает обращения к системным вызовам, включая связанные с файловыми системами

[Gregg 19]

statsnoop

Трассирует вызовы вариантов функции stat(2)

[Gregg 19]

syncsnoop

Трассирует вызовы вариантов функции sync(2) с отметками времени

[Gregg 19]

mmapfiles

Подсчитывает вызовы mmap(2) для файлов

[Gregg 19]

scread

Подсчитывает вызовы read(2) для файлов

[Gregg 19]

fmapfault

Подсчитывает сбои отображения файлов

[Gregg 19]

filelife

Трассирует короткоживущие файлы и определяет продолжительность их существования в секундах

[Gregg 19]

vfsstat

Возвращает статистики, характеризующие работу VFS

[Gregg 19]

vfscount

Подсчитывает все операции с VFS

[Gregg 19]

vfssize

Выводит распределение объемов данных в операциях чтения/записи

[Gregg 19]

fsrwstat

Подсчитывает операции чтения/записи по типам файловых систем

[Gregg 19]

fileslower

Определяет медленные операции чтения/записи

[Gregg 19]

filetype

Подсчитывает операции чтения/записи по типам файлов и процессам

[Gregg 19]

ioprofile

Подсчитывает трассировки стека для операций ввода/вывода

[Gregg 19]

writesync

Подсчитывает операции записи с флагом sync в обычные файлы

[Gregg 19]

writeback

Выводит события отложенной записи и задержки

[Gregg 19]

dcstat

Выводит статистику попаданий в кэш каталогов

[Gregg 19]

dcsnoop

Трассирует операции поиска в кэше каталогов

[Gregg 19]

mountsnoop

Трассирует события монтирования и размонтирования файловых систем в системе в целом

[Gregg 19]

icstat

Выводит статистику попаданий в кэш индексных узлов

[Gregg 19]

bufgrow

Выводит события увеличения буферов в байтах по процессам

[Gregg 19]

readahead

Показывает попадания в кэш после операций опережающего чтения и их эффективность

Другие инструменты для анализа файловых систем в Linux:

yy df(1): сообщает статистики потребления и мощности файловой системы; yy inotify: фреймворк для мониторинга событий файловой системы в Linux.

520  Глава 8. Файловые системы Некоторые типы файловых систем имеют собственные инструменты анализа производительности, дополняющие набор инструментов, которые есть в ОС. Одна из таких файловых систем — ZFS.

ZFS В состав ZFS входит инструмент zpool(1M), имеющий подкоманду iostat. Она возвращает статистики пула ZFS, в том числе частоту операций (чтение и запись) и пропускную способность пула. Популярное дополнение — инструмент arcstat.pl, который сообщает размер ARC и L2ARC, а также частоту попаданий и промахов. Например: $ arcstat 1 time read 04:45:47 0 04:45:49 15K 04:45:50 23K 04:45:51 65K [...]

miss 0 10 81 25

miss% 0 0 0 0

dmis 0 10 81 25

dm% 0 0 0 0

pmis 0 0 0 0

pm% 0 0 0 0

mmis 0 1 1 4

mm% 0 0 0 0

arcsz 14G 14G 14G 14G

c 14G 14G 14G 14G

Статистики относятся к текущему интервалу:

yy read, miss: общее количество обращений к первичному кэшу (ARC) и промахов;

yy miss%, dm%, pm%, mm%: общий процент промахов ARC, процент промахов данных, предварительной выборки и метаданных;

yy dmis, pmis, mmis: промахи данных, предварительной выборки и метаданных; yy arcsz, c: размер первичного кэша (ARC), целевой размер первичного кэша. arcstat.pl — это программа на языке Perl, которая читает статистики из kstat.

8.6.18. Визуализация Изменение нагрузки, приложенной к файловым системам, можно изобразить в виде линейного графика. Такой график помогает выявить временные закономерности изменения нагрузки. Иногда полезно строить отдельные графики для разных операций с файловой системой — чтение, запись и т. д. Распределение задержек в файловой системе обычно имеет бимодальный вид: одна мода с низкой задержкой соответствует попаданиям в кэш файловой системы, а другая — с высокой задержкой — промахам кэша (когда выполняется чтение с устройства хранения). По этой причине представление распределения единственным значением, таким как среднее, мода или медиана, может вводить в заблуждение. Одно из решений этой проблемы — использовать визуализацию, показывающую полное распределение, например тепловую карту. Тепловые карты были представлены в главе 2 «Методологии» в разделе 2.10.3 «Тепловые карты». На рис. 8.13

8.7. Эксперименты  521 показан пример тепловой карты задержек в файловой системе: здесь течение времени отображается по оси Х, а задержка ввода/вывода — по оси Y [Gregg 09a]. Эта тепловая карта показывает разницу в величине задержек до и после включения устройства L2ARC для NFSv3. Устройство L2ARC — это вторичный кэш ZFS, обычно основанный на флеш-памяти (об этом шла речь в разделе 8.3.2 «Кэширование»). Система, представленная на рис. 8.13, имеет 128 Гбайт DRAM и устройство L2ARC (твердотельный диск SSD с оптимизацией чтения) емкостью 600 Гбайт. Левая половина тепловой карты показывает распределение задержек в отсутствие устройства L2ARC (L2ARC был отключен), а правая половина — распределение задержек с включенным устройством L2ARC.

Рис. 8.13. Тепловая карта задержек в файловой системе На тепловой карте слева четко просматриваются две области — низких и высоких задержек, — разделенные промежутком. Низкие задержки — это синяя полоса внизу около 0 мс. Вероятнее всего, они обусловлены попаданием в кэш в основной памяти. Высокие задержки начинаются примерно с 3 мс и простираются до самого верха в виде «облака». Они, скорее всего, обусловлены задержками в устройствах с вращающимися дисками. Такое двухмодальное распределение задержек типично для файловых систем на вращающихся дисках. Картина справа соответствует распределению задержек с включенным устройством L2ARC. Теперь задержки часто оказываются ниже 3 мс, и количество задержек, обу­ словленных вращающимися дисками, значительно меньше. Как видите, задержки в L2ARC заполнили диапазон, где раньше был промежуток, и задержки в файловой системе в целом уменьшились.

8.7. ЭКСПЕРИМЕНТЫ В этом разделе описаны инструменты для активного тестирования производительности файловой системы. Дополнительную информацию см. в разделе 8.5.8 «Микробенчмаркинг», где представлена предлагаемая стратегия тестирования.

522  Глава 8. Файловые системы При использовании представленных ниже инструментов рекомендуется оставить постоянно работающим iostat (1). Это позволит убедиться, что рабочая нагрузка достигает или не достигает дисков в соответствии с ожиданиями. Например, при тестировании с рабочим набором, размер которого позволяет ему уместиться в кэше файловой системы, ожидается, что 100 % операций чтения будут попадать в кэш. Поэтому iostat(1) не должен показывать значительный объем дискового ввода/ вывода. Подробнее о iostat(1) идет речь в главе 9 «Диски».

8.7.1. Ad hoc Команду dd(1) (копирует с устройства на устройство) используют для специальных тестов производительности последовательных операций с файловой системой. Коман­ды ниже записывают, а затем читают файл с именем file1 и размером в 1 Гбайт блоками по 1 Мбайт: write: dd if=/dev/zero of=file1 bs=1024k count=1k read: dd if=file1 of=/dev/null bs=1024k

Версия dd(1) в Linux по завершении выводит статистики. Например: $ dd if=/dev/zero of=file1 bs=1024k count=1k 1024+0 records in 1024+0 records out 1073741824 bytes (1.1 GB, 1.0 GiB) copied, 0.76729 s, 1.4 GB/s

Здесь видно, что у файловой системы пропускная способность записи 1,4 Гбайт/с (используется кэширование с отложенной записью, поэтому фактическая запись на диск производится позже, в зависимости от параметров vm.dirty_*: см. главу 7 «Память», раздел 7.6.1 «Настраиваемые параметры»).

8.7.2. Инструменты микробенчмаркинга Для тестирования файловой системы есть множество инструментов, включая Bonnie, Bonnie++, iozone, tiobench, SysBench, fio и FileBench. Некоторые из них приводятся ниже в порядке возрастания сложности. См. также главу 12 «Бенчмаркинг». Я рекомендую использовать fio.

Bonnie, Bonnie++ Bonnie — это простая программа, написанная на языке C и предназначенная для проверки нескольких рабочих нагрузок с одним файлом из одного потока. Она была написана Тимом Бреем (Tim Bray) в 1989 году [Bray 90]. Ее можно запускать без аргументов (в этом случае будут использоваться значения по умолчанию): $ ./Bonnie File './Bonnie.9598', size: 104857600 [...] -------Sequential Output-------- ---Sequential Input-- --Random--

8.7. Эксперименты  523 -Per Char- --Block--- -Rewrite-- -Per Char- --Block--- --Seeks--MB K/sec %CPU K/sec %CPU K/sec %CPU K/sec %CPU K/sec %CPU /sec %CPU 100 123396 100.0 1258402 100.0 996583 100.0 126781 100.0 2187052 100.0 164190.1 299.0 Machine

Среди прочих значений выводится величина потребления процессора в каждом тесте: 100 % показывает, что программа Bonnie не блокировалась дисковым вводом/ выводом и всегда оставалась на процессоре, получая данные из кэша. Причина в том, что размер целевого файла в этом примере составляет 100 Мбайт и полностью умещается в кэше в этой системе. Размер файла можно изменить, передав параметр -s . Есть 64-разрядная версия Bonnie-64, которая позволяет тестировать файлы большего размера. Есть версия Bonnie++, переписанная на C++ Расселом Кокером (Russell Coker) [Coker 01]. К сожалению, инструменты бенчмаркинга файловых систем, такие как Bonnie, могут приводить в замешательство, если вы не понимаете, что они тестируют. Первый результат, тест putc(3), зависит от реализации системной библиотеки, и именно она становится предметом тестирования, а не файловая система. См. пример в главе 12 «Бенчмаркинг», в разделе 12.3.2 «Активный бенчмаркинг».

fio fio (Flexible IO Tester) Йенса Аксбо (Jens Axboe) — это гибкий инструмент для бенчмаркинга файловых систем, обладающий множеством дополнительных возможностей [Axboe 20]. Вот две из них — именно они побудили меня использовать его вместо других:

yy неоднородные случайные распределения, которые более точно моделируют реальные нагрузки (например, -random_distribution=pareto:0.9);

yy вывод информации о распределении задержек в процентилях, включая 99,00; 99,50; 99,90; 99,95; 99,99.

Вот результаты имитации рабочей нагрузки произвольного чтения с размером блока ввода/вывода 8 Кбайт, размером рабочего набора 5 Гбайт и неоднородной закономерностью доступа (pareto:0.9): # fio --runtime=60 --time_based --clocksource=clock_gettime --name=randread -numjobs=1 --rw=randread --random_distribution=pareto:0.9 --bs=8k --size=5g -filename=fio.tmp randread: (g=0): rw=randread, bs=8K-8K/8K-8K/8K-8K, ioengine=sync, iodepth=1 fio-2.0.13-97-gdd8d Starting 1 process Jobs: 1 (f=1): [r] [100.0% done] [3208K/0K/0K /s] [401 /0 /0 iops] [eta 00m:00s] randread: (groupid=0, jobs=1): err= 0: pid=2864: Tue Feb 5 00:13:17 2013 read : io=247408KB, bw=4122.2KB/s, iops=515 , runt= 60007msec clat (usec): min=3 , max=67928 , avg=1933.15, stdev=4383.30 lat (usec): min=4 , max=67929 , avg=1934.40, stdev=4383.31 clat percentiles (usec):

524  Глава 8. Файловые системы | 1.00th=[ 5], 5.00th=[ 5], 10.00th=[ 5], 20.00th=[ 6], | 30.00th=[ 6], 40.00th=[ 6], 50.00th=[ 7], 60.00th=[ 620], | 70.00th=[ 692], 80.00th=[ 1688], 90.00th=[ 7648], 95.00th=[10304], | 99.00th=[19584], 99.50th=[24960], 99.90th=[39680], 99.95th=[51456], | 99.99th=[63744] bw (KB/s) : min= 1663, max=71232, per=99.87%, avg=4116.58, stdev=6504.45 lat (usec) : 4=0.01%, 10=55.62%, 20=1.27%, 50=0.28%, 100=0.13% lat (usec) : 500=0.01%, 750=15.21%, 1000=4.15% lat (msec) : 2=3.72%, 4=2.57%, 10=11.50%, 20=4.57%, 50=0.92% lat (msec) : 100=0.05% cpu : usr=0.18%, sys=1.39%, ctx=13260, majf=0, minf=42 IO depths : 1=100.0%, 2=0.0%, 4=0.0%, 8=0.0%, 16=0.0%, 32=0.0%, >=64=0.0% submit : 0=0.0%, 4=100.0%, 8=0.0%, 16=0.0%, 32=0.0%, 64=0.0%, >=64=0.0% complete : 0=0.0%, 4=100.0%, 8=0.0%, 16=0.0%, 32=0.0%, 64=0.0%, >=64=0.0% issued : total=r=30926/w=0/d=0, short=r=0/w=0/d=0

Процентили задержек (clat ) показывают очень низкие задержки вплоть до 50-го процентиля, которые, если судить по их величине (от 5 до 7 мс), обусловлены попаданиями в кэш. Остальные процентили показывают эффект промахов кэша, включая хвост очереди. В этом случае 99,99-й процентиль показывает задержку 63 мс. Только одних процентилей недостаточно, чтобы понять, что распределение задержек многомодальное. Но при этом они показывают не менее интересный аспект: хвост более медленной моды (дисковый ввод/вывод). В качестве альтернативы можно использовать похожий, но более простой инструмент SysBench (пример использования SysBench для анализа потребления процессора представлен в главе 6 в разделе 6.8.2 «SysBench»). Но если вам нужен еще больший контроль, попробуйте FileBench.

FileBench FileBench — это инструмент для бенчмаркинга файловых систем, позволяющий моделировать рабочие нагрузки, описывая их на языке моделирования рабочих нагрузок Workload Model Language. С помощью FileBench можно моделировать разные шаблоны поведения потоков выполнения, в том числе и синхронный. Вместе с инструментом распространяется множество конфигураций, называемых характерами (personalities), в том числе для имитации модели ввода/вывода в базе данных Oracle. К сожалению, FileBench сложен для изучения и использования и может быть интересен только тем, кто работает с файловыми системами на постоянной основе.

8.7.3. Очистка кэша Linux предоставляет возможность очищать кэши файловой системы (удалять запи­ си), что может пригодиться для бенчмаркинга из согласованного и «холодного» состояния кэша, как бывает после загрузки системы. В исходном коде ядра этот механизм описан просто (Documentation/sysctl/vm.txt):

8.8. Настройка  525 Чтобы очистить echo 1 Чтобы очистить echo 2 Чтобы очистить echo 3

кэш страниц: > /proc/sys/vm/drop_caches кэши объектов в ядре (в том числе кэши каталогов и индексных узлов): > /proc/sys/vm/drop_caches кэши страниц и объектов в ядре: > /proc/sys/vm/drop_caches

Иногда полезно очистить все кэши (3) перед запуском других бенчмарков, чтобы бенчмаркинг начинался с известного состояния (холодный кэш). Это поможет обеспечить согласованность результатов бенчмаркинга.

8.8. НАСТРОЙКА В разделе 8.5 «Методология» мы рассмотрели множество подходов к настройке, включая настройку кэша и определение характеристик рабочей нагрузки. В последнем случае, выявляя и устраняя ненужную работу, можно добиться наивысших результатов. В этот раздел включено описание конкретных параметров, доступных для настройки. Конкретные настройки — доступные параметры и оптимальные значения — зависят от версии ОС и предполагаемой рабочей нагрузки. В следующих разделах, организованных по типам настроек, приводятся примеры доступных настраиваемых параметров и описываются причины, почему может потребоваться их настройка. В них я представлю функции, доступные из приложений, и два примера для файловых систем ext4 и ZFS. Описание настройки кэша страниц вы найдете в главе 7 «Память».

8.8.1. Функции В разделе 8.3.7 «Синхронная запись» говорилось о том, как можно повысить производительность операций синхронной записи, используя fsync(2) для сбрасывания на диск данных группами, а не по отдельности, с флагами O_DSYNC/O_RSYNC при обращении к системному вызову open(2). Ниже перечислены другие функции, вызов которых повышает производительность, — posix_fadvise() и madvise(2). Они предоставляют файловой системе подсказки по требованиям к кэшированию.

posix_fadvise() Эта библиотечная функция (обертка вокруг системного вызова fadvise64(2)) работает с областью файла и имеет сигнатуру: int posix_fadvise(int fd, off_t offset, off_t len, int advice);

Флаги, которые можно передать в аргументе advice, перечислены в табл. 8.8.

526  Глава 8. Файловые системы Таблица 8.8. Флаги для аргумента advice функции posix_fadvise() в ядре Linux Флаг

Описание

POSIX_FADV_SEQUENTIAL

Доступ к указанному диапазону данных будет последовательным

POSIX_FADV_RANDOM

Доступ к указанному диапазону данных будет произвольным

POSIX_FADV_NOREUSE

Данные не будут использоваться повторно

POSIX_FADV_WILLNEED

Данные будут использоваться повторно в ближайшем будущем

POSIX_FADV_DONTNEED

Данные не будут использоваться повторно в ближайшем будущем

Ядро может использовать эту информацию для повышения производительности. Она помогает ему решить, когда стоит производить предварительную выборку данных и когда есть смысл кэшировать данные. Эти подсказки могут повысить коэффициент попадания в кэш для данных с высоким приоритетом. Полный список флагов для аргумента advice см. на странице справочного руководства man для вашей системы.

madvise() Этот системный вызов работает с файлами, отображенными в память, и имеет такую сигнатуру: int madvise(void *addr, size_t length, int advice);

Флаги, которые можно передать в аргументе advice, перечислены в табл. 8.9. Таблица 8.9. Флаги для аргумента advice системного вызова madvise(2) в ядре Linux Флаг

Описание

MADV_RANDOM

Доступ к данным в файле будет произвольным

MADV_SEQUENTIAL

Доступ к данным в файле будет последовательным

MADV_WILLNEED

Данные будут использоваться повторно (кэширование желательно)

MADV_DONTNEED

Данные не будут использоваться повторно (кэширование нежелательно)

Как и в случае с posix_fadvise(), ядро может использовать эту информацию для увеличения производительности, в том числе для принятия более эффективных решений по кэшированию.

8.8.2. ext4 Файловые системы ext2, ext3 и ext4 в Linux можно настроить с помощью:

yy параметров монтирования; yy команды tune2fs(8);

8.8. Настройка  527

yy файлов свойств /sys/fs/ext4; yy команды e2fsck(8).

Параметры монтирования и команда tune2fs Параметры монтирования можно установить во время монтирования с помощью команды mount(8) или во время загрузки в /boot/grub/menu.lst и /etc/fstab. Список доступных параметров ищите на странице справочного руководства man для mount(8). Вот несколько вариантов: # man mount [...] ПАРАМЕТРЫ МОНТИРОВАНИЯ, НЕ ЗАВИСЯЩИЕ ОТ ТИПА ФАЙЛОВОЙ СИСТЕМЫ [...] atime Не использовать поддержку noatime. В этом случае поддержка обновления времени последнего доступа к индексным узлам управляется настройками по умолчанию в ядре. См. также описание параметров монтирования relatime и strictatime. noatime Не обновлять время последнего доступа к индексным узлам в этой файловой системе (например, для более быстрого доступа к файлам на серверах новостей). [...] relatime Обновлять время последнего доступа к индексным узлам одновременно с обновлением времени последнего изменения. Время последнего доступа обновляется, только если предыдущее время доступа было раньше, чем текущее время последнего изменения. (Напоминает noatime, но не нарушает работу mutt и других приложений, которым необходимо знать, выполнялось ли чтение файла с момента последнего его изменения.)

[...]

Начиная с Linux 2.6.30, поведение ядра по умолчанию соответствует этому параметру (если не указан параметр noatime), а для возврата к традиционной семантике следует использовать параметр strictatime. Кроме того, начиная с Linux 2.6.30, время последнего доступа к файлу всегда обновляется, если оно оказалось в прошлом больше чем на 1 день.1

Параметр noatime исторически использовался для повышения производительности за счет предотвращения обновления отметок времени последнего доступа и связанных с этим операций дискового ввода/вывода. На странице справочного руководства описано, что в настоящее время по умолчанию используется параметр relatime, который тоже уменьшает число этих обновлений. На странице справочного руководства для mount(8) описаны не только общие параметры монтирования, но и параметры монтирования, характерные для файловой системы. Но для параметров монтирования файловой системы ext4 есть своя страница справочного руководства ext4(5): Это перевод текста из справочного руководства. — Примеч. пер.

1

528  Глава 8. Файловые системы # man ext4 [...] Параметры монтирования для ext4 [...] Параметры journal_dev, journal_path, norecovery, noload, data, commit, orlov, oldalloc, [no]user_xattr, [no]acl, bsddf, minixdf, debug, errors, data_err, grpid, bsdgroups, nogrpid, sysvgroups, resgid, resuid, sb, quota, noquota, nouid32, grpquota, usrquota, usrjquota, grpjquota и jqfmt обратно совместимы с ext3 или ext2.

[...]

journal_checksum | nojournal_checksum Параметр journal_checksum включает подсчет контрольных сумм транзакций с журналом. Это обеспечивает поддержку кода восстановления в e2fsck и…1

Текущие настройки монтирования можно узнать с помощью tune2fs -l device и mount (без параметров). tune2fs (8) может устанавливать или сбрасывать различные параметры монтирования, как описано на соответствующей странице справочного руководства man. Чаще всего для повышения производительности используется параметр монтирования noatime: он позволяет отключить обновление времени последнего доступа к файлам, если оно не требуется пользователям файловой системы, и за счет этого сократить количество внутренних операций ввода/вывода.

Файлы свойств /sys/fs Некоторые дополнительные параметры можно установить в реальном времени через файловую систему /sys. Вот пример для ext4: # cd /sys/fs/ext4/nvme0n1p1 # ls delayed_allocation_blocks last_error_time err_ratelimit_burst lifetime_write_kbytes err_ratelimit_interval_ms max_writeback_mb_bump errors_count mb_group_prealloc extent_max_zeroout_kb mb_max_to_scan first_error_time mb_min_to_scan inode_goal mb_order2_req inode_readahead_blks mb_stats journal_task mb_stream_req # cat inode_readahead_blks 32

msg_ratelimit_burst msg_ratelimit_interval_ms reserved_clusters session_write_kbytes trigger_fs_error warning_ratelimit_burst warning_ratelimit_interval_ms

Как показывает этот вывод, ext4 будет читать не более 32 блоков из таблицы индексных узлов. Не все эти файлы позволяют изменять их: некоторые доступны только для чтения. Они описаны в исходном коде Linux в файле Documentation/ admin-guide/ext4.rst [Linux 20h], где также приведены параметры монтирования. Это перевод текста из справочного руководства. — Примеч. пер.

1

8.8. Настройка  529

e2fsck Команда e2fsck(8) поддерживает возможность переиндексации каталогов в файловой системе ext4, что помогает повысить производительность. Например: e2fsck -D -f /dev/hdX

Другие параметры e2fsck(8) связаны с проверкой и восстановлением файловой системы.

8.8.3. ZFS ZFS поддерживает большое количество настраиваемых параметров (называемых свойствами) для каждой файловой системы и несколько параметров — для системы в целом. Список параметров можно получить командой zfs(1). Например: # zfs get all zones/var NAME PROPERTY [...] zones/var recordsize zones/var mountpoint zones/var sharenfs zones/var checksum zones/var compression zones/var atime [...]

VALUE

SOURCE

128K legacy off on off off

default local default default inherited from zones inherited from zones

Вывод включает столбцы с именами свойств, текущими значениями и источниками. Источник сообщает, как тот или иной параметр был установлен: унаследован из набора данных ZFS более высокого уровня, выбрано значение по умолчанию или настроен локально для данной файловой системы. Параметры также можно установить с помощью команды zfs(1M), они описаны на ее странице справочного руководства man. Ключевые параметры, связанные с производительностью, перечислены в табл. 8.10. Таблица 8.10. Ключевые настраиваемые параметры ZFS Параметр

Варианты значений

Описание

recordsize

от 512 до 128 K

Желаемый размер блока для файлов

compression

on | off | lzjb | gzip | gzip-[1–9] | zle | lz4

Использование легковесных алгоритмов (например, lzjb) может улучшить производительность в некоторых ситуациях за счет уменьшения объема внутреннего ввода/вывода

atime

on | off

Обновление времени последнего доступа (при включении будет вызывать операции записи после операций чтения)

primarycache

all | none | metadata

Политика первичного кэша ARC. Загрязнение кэша данными из файловых систем с низким приоритетом (например, архивов) можно уменьшить, установив значение «no» или «metadata»

530  Глава 8. Файловые системы Таблица 8.10 (окончание) Параметр

Варианты значений

Описание

secondarycache

all | none | metadata

Политика вторичного кэша L2ARC

logbias

latency | throughput

Подсказка для операций синхронной записи: «latency» означает использовать устройства журналов, а «throughput» — устройства пула

sync

standard | always | disabled

Поведение операций синхронной записи

Наиболее важный параметр для настройки — recordsize (размер записи). Желательно, чтобы его значение соответствовало характеру ввода/вывода в приложении. Обычно он имеет значение по умолчанию 128 Кбайт, что может быть неэффективно для небольших случайных операций ввода/вывода. Обратите внимание, что это не относится к файлам, размер которых меньше размера записи и которые сохраняются с использованием динамического размера записи, равного длине файла. Отключение atime тоже может улучшить производительность, если время последнего доступа не нужно. ZFS также предоставляет общесистемные настройки, включая настройки времени (TXG) синхронизации группы транзакций (zfs_txg_synctime_ms, zfs_txg_timeout), а также порогового значения метаблоков для переключения между алгоритмами распределения, оптимизированными по памяти и по времени (metaslab_df_free_pct). Уменьшение TXG может улучшить производительность за счет уменьшения числа конфликтов и очередей с другими операциями ввода/вывода. Обязательно ознакомьтесь с описанием настраиваемых параметров в документации, где можно найти исчерпывающий список, подробное описание и преду­ преждения.

8.9. УПРАЖНЕНИЯ 1. Ответьте на следующие вопросы, касающиеся файловых систем: • В чем разница между логическим и физическим вводом/выводом? • В чем разница между произвольным и последовательным вводом/выводом? • Что такое прямой ввод/вывод? • Что такое неблокирующий ввод/вывод? • Что такое размер рабочего набора? 2. Ответьте на следующие концептуальные вопросы: • Какую роль играет VFS? • Что такое задержка в файловой системе и как ее можно измерить?

8.9. Упражнения  531 • Какова цель предварительной выборки (упреждающего чтения)? • Какова цель прямого ввода/вывода? 3. Ответьте на следующие более сложные вопросы: • Опишите преимущества fsync(2) перед O_SYNC. • Опишите достоинства и недостатки использования mmap(2) вместо read(2)/ write(2). • Опишите причины, почему объем логического ввода/вывода может быть больше объема физического ввода/вывода. • Опишите причины, почему объем логического ввода/вывода может быть меньше объема физического ввода/вывода. • Объясните, как использование алгоритма копирования при записи может повысить производительность файловой системы. 4. Разработайте следующие процедуры для своей среды: • Чек-лист настройки кэша файловой системы. Он должен включать способы получения списка существующих кэшей файловой системы, определения их текущих размеров и потребления, а также коэффициентов попадания. • Чек-лист определения характеристик рабочей нагрузки на файловую систему. Он должен включать способы получения всех необходимых деталей, и в первую очередь с использованием инструментов ОС. 5. Решите следующие задачи: • Выберите приложение и измерьте задержки в операциях с файловой системой, включая:

ƒƒ полное распределение задержек, а не только среднее значение; ƒƒ какую часть каждой секунды каждый поток выполнения в приложении тратит на операции с файловой системой.

• С помощью инструмента микробенчмаркинга определите экспериментально размер кэша файловой системы. Объясните свой выбор инструмента. Также покажите снижение производительности (с использованием любой метрики) в случаях, когда рабочий набор не умещается в кэше. 6. (опционально) Разработайте инструмент, позволяющий оценить производительность синхронной и асинхронной записи. Он должен измерять скорость операций и задержку, а также поддерживать возможность определения идентификаторов процессов, выполнивших эти операции; это позволит использовать инструмент для определения характеристик рабочей нагрузки. 7. (опционально) Разработайте инструмент для получения статистик косвенного и увеличенного ввода/вывода: количество дополнительных байтов и операций ввода/вывода, не инициированных приложениями непосредственно. Инструмент должен разбивать дополнительные операции ввода/вывода по типам, чтобы можно было объяснить их причину.

532  Глава 8. Файловые системы

8.10. ССЫЛКИ [Ritchie 74] Ritchie, D. M., and Thompson, K., «The UNIX Time-Sharing System», Communications of the ACM 17, no. 7, pp. 365–75, July 1974. [Lions 77] Lions, J., A «Commentary on the Sixth Edition UNIX Operating System», University of New South Wales, 1977. [McKusick 84] McKusick, M. K., Joy, W. N., Leffler, S. J., and Fabry, R. S., «A Fast File System for UNIX». ACM Transactions on Computer Systems (TOCS) 2, no. 3, August 1984. [Bach 86] Bach, M. J., «The Design of the UNIX Operating System», Prentice Hall, 1986. [Bray 90] Bray, T., «Bonnie», http://www.textuality.com/bonnie, 1990. [Sweeney 96] Sweeney, A., «Scalability in the XFS File System», USENIX Annual Technical Conference, https://www.cs.princeton.edu/courses/archive/fall09/cos518/papers/xfs.pdf, 1996. [Vahalia 96] Vahalia, U., «UNIX Internals: The New Frontiers», Prentice Hall, 19961. [Coker 01] Coker, R., «bonnie++», https://www.coker.com.au/bonnie++, 2001. [Gregg 09a] Gregg, B., «L2ARC Screenshots», http://www.brendangregg.com/blog/2009-01-30/ l2arc-screenshots.html, 2009. [Corbet 10] Corbet, J., «Dcache scalability and RCU-walk», LWN.net, http://lwn.net/Articles/419811, 2010. [Doeppner 10] Doeppner, T., «Operating Systems in Depth: Design and Programming», Wiley, 2010. [XFS 10] «Runtime Stats», https://xfs.org/index.php/Runtime_Stats, 2010. [Oracle 12] «ZFS Storage Pool Maintenance and Monitoring Practices», Oracle Solaris Administration: ZFS File Systems, https://docs.oracle.com/cd/E36784_01/html/E36835/storage-9. html, 2012. [Ahrens 19] Ahrens, M., «State of OpenZFS», OpenZFS Developer Summit 2019, https://drive. google.com/file/d/197jS8_MWtfdW2LyvIFnH58uUasHuNszz/view, 2019. [Axboe 19] Axboe, J., «Efficient IO with io_uring», https://kernel.dk/io_uring.pdf, 2019. [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»2, Addison-Wesley, 2019. [Axboe 20] Axboe, J., «Flexible I/O Tester», https://github.com/axboe/fio, последнее обновление 2020. [Linux 20h] «ext4 General Information», Linux documentation, https://www.kernel.org/doc/html/ latest/admin-guide/ext4.html, по состоянию на 2020. [Torvalds 20a] Torvalds, L., «Re: Do not blame anyone. Please give polite, constructive criticism», https://www.realworldtech.com/forum/?threadid=189711&curpostid=189841, 2020.

Вахалия Ю. «UNIX изнутри». СПб., издательство «Питер».

1

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

2

Глава 9

ДИСКИ

Дисковый ввод/вывод может вызывать значительные задержки в работе приложений, поэтому он важная цель для анализа производительности. При высокой нагрузке диски становятся узким местом, в результате чего процессоры простаивают, пока система ожидает завершения дискового ввода/вывода. Выявление и устранение этих узких мест увеличивает производительность и пропускную способность приложений на порядки. Термин диски относится к основным устройствам хранения в системе, в том числе к твердотельным накопителям (SSD) на основе флеш-памяти и магнитным вращающимся дискам. Твердотельные накопители изначально предназначались для повышения производительности дискового ввода/вывода, с чем они прекрасно справляются. Но требования к емкости, скорости ввода/вывода и пропускной способности тоже растут, и устройства на основе флеш-памяти не лишены своих проблем с производительностью. Цели этой главы:

yy yy yy yy yy yy yy yy yy

представить модели дисков и основные понятия; показать, как модели доступа к диску влияют на производительность; описать опасности интерпретации потребления диска; познакомить с характеристиками и «внутренностями» дисковых устройств; познакомить с путями в коде ядра между файловыми системами и устройствами; познакомить с уровнями RAID и их производительностями; представить различные методологии анализа производительности дисков; охарактеризовать общесистемный и межпроцессный дисковый ввод/вывод; познакомить с приемами измерения задержек дискового ввода/вывода и выявления выбросов;

yy показать, как определять пути в коде приложения, заканчивающиеся дисковым вводом/выводом;

yy детально изучить дисковый ввод/вывод с помощью трассировщиков; yy познакомить с параметрами настройки дисков.

534  Глава 9. Диски Глава состоит из шести частей. В первых трех обсуждаются основы анализа производительности дискового ввода/вывода, а в следующих трех показано их практическое применение в системах на базе Linux:

yy Основы представляет терминологию, связанную с хранением данных, основные модели дисковых устройств и ключевые идеи производительности дисков.

yy Архитектура описывает обобщенную аппаратную и программную архитектуру хранилища.

yy Методология описывает методологии анализа эффективности, как наблюдательные, так и экспериментальные.

yy Инструменты наблюдения описывает инструменты наблюдения за работой дисков в Linux, включая трассировку и визуализацию.

yy Экспериментирование обобщает возможности инструментов тестирования дисков.

yy Настройка описывает параметры настройки дисков. В предыдущей главе рассматривалась производительность файловых систем, построенных на дисках, так что она заслуживает особого внимания для понимания производительности приложений.

9.1. ТЕРМИНОЛОГИЯ Ниже даны основные термины этой главы, связанные с дисками:

yy Виртуальный диск: имитация запоминающего устройства. Система видит его как один физический диск, но он может состоять из нескольких физических дисков или части диска.

yy Транспорт: физическая шина, через которую происходит обмен информацией, включая передачу данных (ввод/вывод) и другие дисковые команды.

yy Сектор: блок хранения на диске, обычно размером 512 байт, но сейчас секторы нередко имеют размер 4 Кбайт.

yy Ввод/вывод: строго говоря, ввод/вывод подразумевает только чтение и запись

на диск и не включает другие дисковые команды. Ввод/вывод можно описать как минимум направлением (чтение или запись), адресом на диске (местоположением) и размером (в байтах).

yy Дисковые команды: дискам можно передавать команды на выполнение других операций, не связанных с передачей данных (например, очистка кэша).

yy Пропускная способность (throughput): под пропускной способностью дисков

обычно понимается текущая скорость передачи данных, измеряемая в байтах в секунду.

yy Полоса пропускания (bandwidth): максимально возможная скорость передачи данных для транспорта или контроллера. Ширина полосы пропускания ограничена аппаратными возможностями.

9.2. Модели  535

yy Задержка ввода/вывода: время выполнения операции ввода/вывода от начала до конца. Более точно понятие времени определяется в разделе 9.3.1 «Измерение времени». Имейте в виду, что в сетевых взаимодействиях термин задержка обозначает время, необходимое для инициирования ввода/вывода и последующей передачи данных.

yy Выбросы задержки: дисковый ввод/вывод с необычно высокой задержкой. Глоссарий включает базовую терминологию для справки, в том числе: диск, контроллер диска, дисковый массив, локальные диски, удаленные диски и IOPS (операций ввода/вывода в секунду). См. также разделы «Терминология» в главах 2 и 3.

9.2. МОДЕЛИ Далее описываются простые модели, иллюстрирующие некоторые базовые принципы устройства дисков.

9.2.1. Простой диск Современные диски включают в себя очередь для запросов ввода/вывода (рис. 9.1). Диск Запросы на ввод/вывод Очередь ввода/вывода Ввод/вывод завершен

Рис. 9.1. Простой диск с очередью Операции ввода/вывода, полученные диском, либо находятся в очереди, либо обслуживаются. Эта простая модель напоминает кассу в супермаркете, где клиенты выстраиваются в очередь для оплаты покупок. Модель хорошо подходит и для анализа с использованием теории массового обслуживания. Часто такая очередь действует по принципу «первым пришел — первым обслужился», но дисковый контроллер может использовать также другие алгоритмы для оптимизации производительности. Эти алгоритмы могут учитывать время, необходимое для перемещения магнитных головок во вращающихся дисках (см. обсуждение в разделе 9.4.1 «Типы дисков») или организовывать отдельные очереди для операций чтения и записи (это особенно характерно для дисков на основе флеш-памяти).

536  Глава 9. Диски

9.2.2. Кэширующие диски Наличие кэша в дисковых устройствах позволяет удовлетворять некоторые запросы на чтение из более быстрой памяти, как показано на рис. 9.2. Такой кэш может быть реализован с использованием микросхем памяти небольшого объема (ОЗУ) внутри физического дискового устройства. При попадании в кэш данные возвращаются с очень низкой задержкой (что хорошо), но промахи все еще случаются достаточно часто и приводят к высокой задержке из-за необходимости чтения данных с физического диска. Кэш в дисковом устройстве также может использоваться для увеличения производительности записи — он используется для хранения буферов отложенной записи. В таких случаях дисковое устройство сообщает о завершении записи сразу после сохранения данных в кэше, то есть до фактической их записи на диск. Иногда кэши используются в режиме сквозной записи, когда признак завершения записи возвращается только после полной передачи данных на следующий уровень. На практике кэши отложенной записи часто комплектуются батареями, чтобы буферизованные данные можно было сохранить позже в случае сбоя питания. Такие батареи могут встраиваться в дисковое устройство или в контроллер. Диск Запросы на ввод/вывод Попадание в кэш

Встроенный кэш

Очередь ввода/вывода Промах кэша

Рис. 9.2. Простой диск со встроенным кэшем

9.2.3. Контроллер На рис. 9.3 показан простой контроллер диска, соединяющий транспорт ввода/вывода процессора с транспортом системы хранения и подключенными дисковыми устройствами. Контроллеры также называют адаптерами главной шины (host bus adapter, HBA). Производительность может ограничиваться любой из этих шин, контроллером диска или самими дисками. Подробности о контроллерах дисков ищите в разделе 9.4 «Архитектура».

9.3. Основные понятия  537

Процессор Процессор Транспорт ввода/вывода

Контроллер дисков Транспорт системы хранения

Диск

Диск

Диск

Диск

Диск

Диск

Рис. 9.3. Простой контроллер дисков с подключенными к нему транспортами

9.3. ОСНОВНЫЕ ПОНЯТИЯ Ниже даны некоторые важные понятия, касающиеся дисков и их производительности.

9.3.1. Измерение времени Время ввода/вывода можно измерить как:

yy время обработки запроса ввода/вывода (также называется временем отклика ввода/вывода): все время от запуска ввода/вывода до его завершения;

yy время ожидания ввода/вывода: время ожидания в очереди; yy время обслуживания ввода/вывода: время, в течение которого производился ввод/вывод (без ожидания).

Соответствующие интервалы изображены на рис. 9.4. Запрос на ввод/вывод

Время бездействия

Время ожидания ввода/вывода

Ввод/вывод завершен Время обслуживания ввода/вывода

Время обработки запроса ввода/вывода

Рис. 9.4. Понятия времени ввода/вывода (обобщенные)

538  Глава 9. Диски Термин время обслуживания возник во времена, когда диски были устроены проще и управлялись непосредственно операционной системой, которая знала, когда диск активно обслуживал ввод/вывод. Современные диски организуют собственные внутренние очереди, а время обслуживания включает время ожидания в очередях в ядре. По возможности я буду использовать поясняющие термины, чтобы указать, что измеряется, от какого начального события и до какого конечного события. Начальные и конечные события могут происходить в ядре или в дисковом устройстве. Время в ядре при этом измеряется с помощью интерфейса блочного ввода/вывода для дисковых устройств (как показано на рис. 9.5).

Попадание в кэш

Завершение

Очередь планировщика ввода/вывода и очередь отправки

Время обслуживания блочного ввода/вывода (время отклика диска)

Завершение

Время ожидания блочного ввода/вывода

Отправка

Вставка

Время обработки запроса блочного ввода/вывода

Промах кэша

Диск DST

DWT

DST

Встроенный кэш Очередь ввода/вывода в диске

Рис. 9.5. Понятия времени в ядре и диске В ядре:

yy Время ожидания блочного ввода/вывода (или время ожидания ОС) — это

время, прошедшее от момента создания и добавления запроса на ввод/вывод в очередь в ядре до момента, когда запрос покинул очередь в ядре и был передан дисковому устройству. Этот интервал может охватывать ожидание в нескольких очередях в ядре, включая очередь блочного ввода/вывода и очередь дискового устройства.

yy Время обслуживания блочного ввода/вывода — это время от отправки запроса

устройству до получения прерывания от устройства, сигнализирующего о завершении его обработки.

yy Время обработки запроса блочного ввода/вывода — это время ожидания запроса на блочный ввод/вывод плюс время обслуживания блочного ввода/вывода: полное время от создания запроса до завершения его обработки.

9.3. Основные понятия  539 В дисковом устройстве:

yy Время ожидания диска — это время, проведенное запросом в очереди внутри диска.

yy Время обслуживания диском — это время, необходимое на активную обработку запроса после того, как он покинул очередь в диске.

yy Время обработки запроса диском (также называется временем отклика диска

и задержкой дискового ввода/вывода) — это время ожидания диска плюс время обслуживания диском. Оно равно времени обслуживания блочного ввода/вывода.

Соответствующие интервалы изображены на рис. 9.5, где DWT  — это время ожидания диска (disk wait time), а DST — время обслуживания диском (disk service time). На схеме также показан кэш в диске и то, как попадание в кэш диска способствует значительному уменьшению времени обслуживания диском (DST). Задержка ввода/вывода — еще один широко используемый термин, представленный в главе 1. Значение этого термина, как и многих других, зависит от места проведения измерений. Часто под задержкой ввода/вывода подразумевается время обработки запроса блочного ввода/вывода, то есть все время, потраченное на выполнение операции ввода/вывода. В приложениях и инструментах анализа производительности для обозначения времени обработки запроса диском обычно используется термин задержка дискового ввода/вывода, обозначающий все время обработки запроса устройством. Инженеры, занимающиеся аппаратным обеспечением, могут использовать термин задержка дискового ввода/вывода для обозначения времени ожидания диска. Время обслуживания блочного ввода/вывода обычно интерпретируется как мера текущей производительности диска (его показывают старые версии iostat(1)), но все же это слишком упрощенное представление. На рис. 9.7 изображен обобщенный стек ввода/вывода с тремя возможными уровнями драйверов под интерфейсом блочного устройства. Любой из них может иметь свою очередь или блокировать обработку запросов на мьютексах, увеличивая задержку ввода/вывода. Эта задержка входит во время обслуживания блочного ввода/вывода.

Расчет времени Время обслуживания диском обычно фиксируется ядром напрямую, но его среднее значение можно оценить по значениям IOPS (операций ввода/вывода в секунду) и потребления: Время обслуживания диском = потребление / IOPS. Например, при уровне потребления 60 % и 300 операциях ввода/вывода в секунду получаем среднее время обслуживания 2 мс (600 мс / 300 операций в секунду). Эта формула предполагает, что величина потребления относится к единственному устройству (или центру обслуживания), которое обрабатывает запросы ввода/вывода последовательно, по одному. Современные диски могут обрабатывать сразу несколько операций ввода/вывода, что делает эту формулу неточной.

540  Глава 9. Диски Чтобы получить точное время обслуживания диском, с помощью трассировки событий можно фиксировать моменты времени, соответствующие передаче запросов в устройство и получения ответов. Это можно сделать с помощью инструментов, описанных в этой главе (например, biolatency(8) в разделе 9.6.6 «biolatency»).

9.3.2. Масштаб времени Время выполнения дискового ввода/вывода широко варьируется — от десятков микросекунд до тысяч миллисекунд. На самом медленном конце диапазона большое время отклика может быть обусловлено единичным медленным дисковым вводом/выводом, на самом быстром дисковый ввод/вывод может стать проблемой только при большом количестве операций (сумма времени выполнения большого количества быстрых операций ввода/вывода сопоставима с временем медленной операции). Таблица 9.1 дает общее представление о возможном диапазоне задержек дискового ввода/вывода. Чтобы узнать точные текущие значения, обратитесь к документации поставщика дисков и проведите свой микробенчмаркинг. Чтобы получить представление о масштабах времени, отличных от дискового ввода/вывода, см. главу 2 «Методологии». Для более точного представления о порядках величин в столбце «В масштабе» приводятся значения, масштабированные по величине воображаемой задержки попадания в кэш на диске, равной одной секунде. Таблица 9.1. Масштабы задержек дискового ввода/вывода

1

Событие

Задержка

В масштабе

Попадание в кэш на диске

< 100 мкс

1 с

Чтение из флеш-памяти

от ~100 до 1000 мкс (в зависимости от объема ввода/вывода)

от 1 до 10 с

Последовательное чтение с вращающегося диска

~1 мс

10 с

Произвольное чтение с вращающегося диска (7200 об/мин)

~8 мс

1,3 мин

Произвольное чтение с вращающегося диска (медленное с очередью)

> 10 мс

1,7 мин

Произвольное чтение с вращающегося диска (с десятками запросов в очереди)

> 100 мс

17 мин

Худший случай с виртуальным диском (аппаратный контроллер, RAID-5, очередь, произвольный ввод/вывод)

> 1000 мс

2,8 ч

1

От 10 до 20 мкс для запоминающих устройств с энергонезависимой памятью (Non-Volatile Memory, NVMe): обычно это флеш-память, подключенная через адаптер к шине PCIe.

9.3. Основные понятия  541 Эти задержки можно интерпретировать по-разному, в зависимости от среды. Работая в сфере корпоративных систем хранения данных, я считал необычно медленным любой дисковый ввод/вывод, занимающий более 10 мс (и, следовательно, рассматривал это как потенциальный источник проблем). В сфере облачных вычислений к большим задержкам относятся терпимее, особенно это относится к веб-приложениям, для которых большие задержки в сети нормальны. В таких средах дисковый ввод/вывод будет проблемой, только если его продолжительность превышает 50 мс (отдельный случай или в целом в течение всего времени обработки запроса). Эта таблица также показывает, что задержки дискового ввода/вывода могут быть двух типов: при попадании в кэш на диске (задержка менее 100 мкс) и при промахе (1–8 мс и выше, в зависимости от характера доступа и типа устройства). Поскольку при работе диска будут наблюдаться задержки обоих типов, то выражение их в виде средней задержки (как это делает iostat(1)) может ввести в заблуждение, так как такое распределение по сути бимодальное. См. рис. 2.23 в главе 2 «Методологии», где показан пример распределения задержки дискового ввода/вывода в виде гистограммы.

9.3.3. Кэширование Наилучшая производительность дискового ввода/вывода — при полном отсутствии такового. Многие уровни программного стека пытаются избежать дискового ввода/вывода путем кэширования операций чтения и буферизации операций записи вплоть до самого диска. Полный список кэшей, включающий кэши на уровне приложений и файловой системы, был приведен в табл. 3.2 в главе 3 «Операционные системы». На уровне драйверов дисковых устройств и ниже могут быть свои кэши, перечисленные в табл. 9.2. Таблица 9.2. Кэши, используемые при выполнении дискового ввода/вывода Кэш

Пример

Кэш устройства

ZFS vdev

Блочный кэш

Кэш буферов

Кэш контроллера дисков

Кэш на карте RAID

Кэш дискового массива

Кэш массива

Кэш внутри дискового устройства

Контроллер данных в диске (Disk Data Controller, DDC) с микросхемами DRAM

Блочный кэш буферов описан в главе 8 «Файловые системы». Эти кэши, используемые при выполнении дискового ввода/вывода, особенно важны для увеличения производительности рабочих нагрузок с произвольным вводом/выводом.

542  Глава 9. Диски

9.3.4. Произвольный и последовательный ввод/вывод Последовательность операций дискового ввода/вывода можно описать как произвольную или последовательную, в зависимости от смещения каждой последующей операции от предыдущей (имеется в виду смещение на диске). Эти термины мы обсуждали в главе 8 «Файловые системы», когда знакомились с особенностями доступа к файлам. Последовательные рабочие нагрузки также часто называют потоковыми. Термин потоковый обычно используется на уровне приложений для описания потокового чтения и записи «на диск» (в файловую систему). Произвольный и последовательный характер дискового ввода/вывода было важно изучить в эпоху магнитных вращающихся дисков. В них произвольный ввод/вывод вызывает дополнительную задержку из-за необходимости позиционирования головок и ожидания, пока диск повернется на нужный угол и требуемый сектор на диске окажется под головкой. Это показано на рис. 9.6, где для перемещения от сектора 1 к сектору 2 требуется выполнить позиционирование головок и дождаться, пока нужный сектор окажется под головкой (фактический путь будет более прямым). Настройка производительности включала выявление таких операций произвольного ввода/вывода и попытки устранить их, например, кэшированием, распределением произвольных операций ввода/вывода между разными дисками и размещением данных на диске так, чтобы уменьшить расстояние, которое должна преодолеть головка при позиционировании.

Пере

меще н

ие

Вращающийся диск

е ени щ а Вр

Рис. 9.6. Вращающийся диск Для других типов дисков, в том числе твердотельных накопителей на основе флешпамяти (SSD), производительность произвольного и последовательного ввода/ вывода обычно одна и та же. Впрочем, в зависимости от диска, бывает небольшая разница, обусловленная другими факторами, например, кэш поиска адреса может охватывать последовательный доступ, но не охватывать произвольный. В операциях записи, объем которых меньше размера блока, иногда наблюдается потеря

9.3. Основные понятия  543 производительности из-за цикла «чтение-изменение-запись», что особенно характерно для произвольных операций записей. Обратите внимание, что смещения на диске, которые видно в ОС, могут не совпадать со смещениями на физическом диске. Например, аппаратный виртуальный диск может отображать непрерывный диапазон смещений на несколько дисков. Диски могут переназначать смещения по-своему (через контроллер данных). Иногда произвольный ввод/вывод не определяется по смещениям, но может определяться по увеличению времени обслуживания диском.

9.3.5. Соотношение операций чтения и записи Помимо характера рабочих нагрузок (произвольных и последовательных), еще одна характерная мера, тесно связанная с количеством операций в секунду (IOPS) и пропускной способностью, — это соотношение операций чтения и записи. Она может быть выражена как отношение во времени в процентах, например: «Из всех операций ввода/вывода, выполненных системой с момента загрузки, 80 % приходится на операции чтения». Знание этого соотношения помогает при проектировании и настройке систем. Система с большой долей операций чтения может получить наибольшую выгоду от добавления кэша, а система с большой долей операций записи — от добавления дополнительных дисков, способствующих увеличению пропускной способности и количества операций ввода/вывода в секунду. Операции чтения и записи сами по себе могут иметь разный характер: чтение может быть произвольным, а запись — последовательной (что особенно характерно для файловых систем с поддержкой копирования при записи). Они также могут иметь разные размеры.

9.3.6. Размер ввода/вывода Еще одна характеристика рабочей нагрузки — средний размер ввода/вывода (количество байтов) или распределение размеров ввода/вывода. Чем больше размер ввода/вывода, тем обычно больше пропускная способность, но при этом увеличивается задержка каждой отдельной операции ввода/вывода. Размер ввода/вывода может изменяться подсистемой дискового устройства (например, делиться на блоки по 512 байт, соответствующие размерам секторов). Размер также может увеличиваться или уменьшаться после запуска операции приложением компонентами ядра, такими как файловые системы, диспетчеры томов и драйверы устройств. См. подразделы «Увеличенный ввод/вывод» и «Уменьшенный ввод/ вывод» в главе 8 «Файловые системы» в разделе 8.3.12 «Логический и физический ввод/вывод». Некоторые дисковые устройства, особенно на основе флеш-памяти, работают по-разному с разными объемами чтения и записи. Например, флеш-накопитель

544  Глава 9. Диски может показывать наибольшую эффективность при чтении данных блоками по 4 Кбайт и записи блоками по 1 Мбайт. Идеальные размеры ввода/вывода могут упоминаться производителем диска в документации, либо их можно определить с помощью микробенчмаркинга. Текущий размер ввода/вывода можно узнать с помощью инструментов наблюдения (см. раздел 9.6 «Инструменты наблюдения»).

9.3.7. Несопоставимость количества операций в секунду Из-за трех последних характеристик количество операций ввода/вывода в секунду (IOPS) для разных устройств и рабочих нагрузок нельзя сравнивать напрямую. Величина IOPS сама по себе мало что значит. Например, вращающийся диск может потратить на выполнение 5000 последовательных операций ввода/вывода значительно меньше времени, чем на 1000 произвольных операций. IOPS для твердотельных накопителей тоже трудно сравнивать, потому что их производительность часто зависит от размера и направления ввода/ вывода (чтение или запись). Более того, IOPS может вообще не иметь значения для приложения. Рабочая нагрузка, состоящая из произвольных запросов, обычно более чувствительна к задержкам, и в этом случае желательно высокое значение IOPS. Потоковая (последовательная) рабочая нагрузка зависит от пропускной способности, из-за чего может оказаться желательной низкая величина IOPS при более объемных операциях ввода/вывода. Чтобы понять значение IOPS, добавьте в уравнение другие детали: случайный или последовательный характер ввода/вывода, размер, отношение чтение/запись, буферизованный/прямой ввод/вывод и количество операций ввода/вывода, выполняемых параллельно. Также рассмотрите возможность использования метрик на основе времени — уровня потребления и времени обслуживания, которые отражают итоговую производительность и лучше подходят для сравнения.

9.3.8. Дисковые команды, не связанные с передачей данных Дискам можно посылать множество команд, не связанных с чтением и записью данных. Например, дискам с внутренним кэшем можно послать команду вытолкнуть содержимое кэша на диск. Такая команда не является передачей данных. Данные были отправлены раньше, во время операции записи. Еще один пример — команды для уничтожения данных: ATA TRIM или SCSI UNMAP. Они сообщают диску, что диапазон секторов больше не понадобится. Это помогает накопителям SSD поддерживать высокую производительность записи. Эти дисковые команды могут влиять на производительность и приводить к тому, что диск будет занят их выполнением, вынуждая другие операции ввода/вывода ожидать.

9.3. Основные понятия  545

9.3.9. Потребление Потребление может быть рассчитано как время, в течение которого диск был занят выполнением работы. Диск с уровнем потребления 0 % «простаивает», а диск с уровнем потребления 100 % — постоянно занят выполнением операций ввода/вывода (и других команд). Диски со 100 %-ным уровнем потребления — это почти всегда источник проблем с производительностью, особенно если их потребление остается на уровне 100 % некоторое время. Но к снижению производительности может приводить любой уровень потребления, потому что дисковый ввод/вывод — это, как правило, медленное действие. Между 0 % и 100 % также может быть уровень (например, 60 %), при пересечении которого производительность диска становится неудовлетворительной из-за повышенной вероятности постановки запросов в очередь, будь то очереди внутри диска или в ОС. Точное значение такого уровня потребления зависит от требований к диску, рабочей нагрузке и задержке. См. подраздел «M/D/1 и 60 % нагрузка» в главе 2 «Методологии» в разделе 2.6.5 «Теория массового обслуживания». Чтобы убедиться, что проблемы в приложении вызывает высокое потребление, изучите время отклика диска и определите, блокируется ли приложение на время выполнения ввода/вывода. Приложение или ОС могут выполнять ввод/вывод асинхронно, и в таких случаях медленный ввод/вывод не вынуждает приложение простаивать. Обратите внимание, что потребление — интервальная метрика. Дисковый ввод/ вывод может выполняться пакетами, особенно когда происходит выталкивание буферов записи, что может остаться незамеченным при вычислении метрики за более длительные интервалы. Более подробно особенности измерения потребления рассматриваются в главе 2 «Методологии» в разделе 2.3.11 «Потребление».

Потребление виртуального диска Для виртуальных дисков, организованных аппаратно (например, контроллером диска или сетевым хранилищем), ОС может знать, когда виртуальный диск был занят, но ничего не знать о производительности физических дисков, из которых он состоит. Из-за этого потребление виртуального диска, которое сообщается ОС, значительно отличается от фактического положения дел на физических дисках (и выглядит нелогично):

yy Виртуальный диск с уровнем потребления 100 %, состоящий из нескольких физических дисков, может выполнить больше работы. В этом случае величина 100 % может означать, что лишь некоторые диски были заняты на 100 %, а некоторые простаивали.

yy Виртуальные диски, содержащие кэш отложенной записи, могут казаться не

сильно нагруженными во время записи, потому что контроллер диска немедленно возвращает признак завершения записи, даже если после этого физические диски какое-то время остаются занятыми.

546  Глава 9. Диски

yy Диски могут оставаться занятыми из-за перестройки аппаратного RAID, при этом ОС может не видеть соответствующих операций ввода/вывода.

По тем же причинам трудно интерпретировать потребление виртуальных дисков, созданных программно (программный RAID). Но ОС должна также предоставлять доступ к информации о потреблении физических дисков. Когда потребление физического диска достигает 100 % и запрашиваются дополнительные операции ввода/вывода, наступает состояние насыщения.

9.3.10. Насыщенность Насыщенность — это мера, характеризующая объем работы, поставленной в очередь, который превышает возможности ресурса. Для дисковых устройств насыщенность рассчитывается как средняя длина очереди ожидания устройства в ОС (если она поддерживается). Насыщенность обеспечивает оценку производительности, когда потребление превышает отметку 100 %. Диск со 100 %-ным потреблением может не быть насыщенным (с пустой очередью) или иметь значительный уровень насыщенности, отрицательно влияющий на производительность из-за того, что запросы долгое время ожидают обработки в очереди. Вы можете предположить, что для дисков с потреблением менее 100 % насыщение не наступает, но на самом деле многое зависит от интервала измерения: 50 %-ное потребление диска долгое время может означать, что половину этого времени было 100 %-ное потребление, а остальное время диск простаивал. Любая статистика, измеряемая за интервал, может скрывать подобные проблемы. Когда важно иметь точное представление о происходящем, можно использовать инструменты трассировки и ими анализировать события ввода/вывода.

9.3.11. Ожидание ввода/вывода Ожидание ввода/вывода — это метрика производительности, измеряемая для каждого процессора отдельно. Она показывает время бездействия процессора, когда в очереди планировщика находились потоки, заблокированные в ожидании завершения дискового ввода/вывода. Она вычисляется как отношение времени бездействия процессора к времени отсутствия любых заданий и ожидания завершения операций дискового ввода/вывода. Большая величина ожидания ввода/вывода для процессора может сигнализировать, что диски — это узкое место и процессор вынужден простаивать, ожидая их. Ожидание ввода/вывода бывает очень запутанной метрикой. Когда появляется другой процесс с высоким потреблением ресурсов процессора, значение ожидания ввода/вывода может упасть, ведь теперь процессору есть чем заняться. Но дисковый ввод/вывод никуда не исчез и все так же блокирует потоки выполнения, несмотря на уменьшение метрики ожидания ввода/вывода. Бывает и наоборот, когда

9.3. Основные понятия  547 системные администраторы после установки обновленного и более эффективного прикладного ПО обнаруживают, что ожидание ввода/вывода увеличилось. Здесь можно подумать, что в новой версии есть проблема с диском и худшая производительность, хотя на самом деле потребление диска осталось прежним, но потребление процессора уменьшилось. Более надежной метрикой является время, за которое прикладные потоки ожидают завершения дискового ввода/вывода. Она отражает падение производительности потоков, вызванное дисками, независимо от того, какую другую работу делают процессоры в это время. Эту метрику можно измерить путем статической или динамической инструментации. Ожидание ввода/вывода по-прежнему популярная метрика в системах Linux, и, несмотря на ее запутанный характер, она успешно используется для выявления узкого места, которое характеризуется высокой занятостью дисков и низкой занятостью процессоров. Один из вариантов ее интерпретации — рассматривать любое ожидание ввода/вывода как признак наличия узкого места в системе, а затем настроить систему так, чтобы минимизировать его, даже если при увеличении нагрузки на процессор будет выполняться тот же объем ввода/вывода. Параллельный ввод/вывод с большей вероятностью будет неблокирующим и с меньшей вероятностью будет вызывать проблемы непосредственно. Непараллельный ввод/вывод, который вызывает ожидание ввода/вывода, с большей вероятностью заблокирует приложение.

9.3.12. Синхронный и асинхронный ввод/вывод Важно понимать, что задержка дискового ввода/вывода может не влиять напрямую на производительность приложения, если операции ввода/вывода в приложении и дисковый ввод/вывод выполняются асинхронно. Обычно это происходит при кэшировании с отложенной записью, когда операция ввода/вывода в приложении завершается раньше, а дисковый ввод/вывод выполняется позже. Приложения могут использовать упреждающее чтение для асинхронного чтения, которое может не блокировать приложение, пока диск выполняет ввод/вывод. Файловая система может инициировать этот процесс сама, чтобы нагреть кэш (предварительная выборка). Даже если приложение выполняет ввод/вывод синхронно, этот путь в коде приложения может быть некритическим и асинхронным по отношению к запросам клиентов. Это может быть рабочий поток ввода/вывода, созданный приложением для управления вводом/выводом, в то время как другие потоки продолжают выполнять основную работу. Ядра ОС тоже часто поддерживают асинхронный или неблокирующий ввод/вывод, предоставляя приложениям API для запуска ввода/вывода и получения уведомления о его завершении через некоторое время. Дополнительную информацию по этим темам ищите в главе 8 «Файловые системы», в разделах 8.3.9 «Неблокирующий ввод/вывод», 8.3.5 «Упреждающее чтение», 8.3.4 «Предварительная выборка» и 8.3.7 «Синхронная запись».

548  Глава 9. Диски

9.3.13. Дисковый ввод/вывод и ввод/вывод в приложении Дисковый ввод/вывод — это конечный результат работы различных компонентов ядра, в том числе файловых систем и драйверов устройств. Есть много причин, по которым частота и объем дискового ввода/вывода могут не соответствовать вводу/ выводу, выполняемому приложением:

yy Уменьшенный, увеличенный и несвязанный ввод/вывод файловой системы. См. главу 8 «Файловые системы», раздел 8.3.12 «Логический и физический ввод/вывод».

yy Подкачка страниц из-за нехватки системной памяти. См. главу 7 «Память», раздел 7.2.2 «Подкачка страниц».

yy Размер блока ввода/вывода в драйвере устройства: драйвер округляет объем ввода/вывода в большую сторону или фрагментирует ввод/вывод.

yy Запись зеркальных блоков и контрольных сумм или проверка прочитанных данных в RAID.

Это несоответствие может сбивать с толку, если не знать о нем. Его можно исследовать, изучив архитектуру и проведя анализ.

9.4. АРХИТЕКТУРА В этом разделе я описываю архитектуру диска, которую обычно изучают при планировании емкости с целью определить ограничения для различных компонентов и конфигураций. Ее также важно изучить при исследовании более поздних проблем с производительностью, потому что проблема может быть связана с архитектурным выбором, а не с текущей нагрузкой и настройками.

9.4.1. Типы дисков Сейчас в основном используются два типа дисков: магнитные вращающиеся и твердотельные накопители (SSD) на основе флеш-памяти. Оба обеспечивают постоянное хранение данных. В отличие от энергозависимой памяти, их содержимое сохраняется и после выключения питания.

9.4.1.1. Магнитные вращающиеся диски Диски этого типа, которые также называют приводами жестких дисков (hard disk drive, HDD), состоят из одного или нескольких дисков — пластин, — покрытых частицами оксида железа. Небольшой объем этих частиц может быть намагничен в одном из двух направлений — эта ориентация используется для хранения битов информации. Пластины вращаются, а над ними перемещается механический рычаг со схемой для чтения и записи данных. Эта схема включает магнитные головки, и на рычаге бывает несколько головок, что позволяет одновременно читать и записывать

9.4. Архитектура  549 несколько битов. Данные хранятся на пластине в виде концентрических дорожек, каждая из которых разделена на секторы. Как всякое механическое устройство, жесткие диски работают довольно медленно, особенно когда производится произвольный ввод/вывод. С развитием технологии производства флеш-памяти твердотельные накопители начали вытеснять вращающиеся диски, и вполне вероятно, что однажды вращающиеся диски выйдут из употребления. Тем не менее вращающиеся диски остаются вполне конкурентоспособными в некоторых сценариях, например, когда требуются недорогие хранилища высокой плотности (с низкой стоимостью хранения мегабайта), особенно для хранилищ данных1. Ниже кратко описаны некоторые факторы, влияющие на производительность вращающихся дисков.

Позиционирование и вращение Невысокая скорость ввода/вывода магнитных вращающихся дисков обычно обусловлена затратами времени на позиционирование головок и ожиданием, пока диск повернется на требуемый угол. То и другое может исчисляться миллисекундами. В лучшем случае следующий ввод/вывод должен производиться в конце текущего ввода/вывода, обслуживаемого в данный момент, поэтому диску не требуется позиционировать головки или ждать поворота пластин на дополнительный угол. Как отмечалось выше, такой ввод/вывод называется последовательным, а ввод/ вывод, требующий позиционирования головки или ожидания поворота диска на требуемый угол, — произвольным. Есть множество стратегий сокращения времени ожидания позиционирования и поворота, в том числе:

yy кэширование: полное устранение дискового ввода/вывода; yy размещение и поведение файловой системы, включая копирование при записи (делающее запись последовательной, но последующие чтения могут стать произвольными);

yy распределение разных рабочих нагрузок по разным дискам, чтобы избежать излишних затрат времени на позиционирование головок между рабочими нагрузками;

yy распределение разных рабочих нагрузок по разным системам (некоторые об-

лачные среды дают такую возможность, чтобы уменьшить влияние разных клиентов друг на друга);

Серверы Open Connect Appliance (OCA) в Netflix, на которых размещается видео для потоковой передачи, могут выглядеть вполне подходящими для использования жестких дисков, но поддержка большого количества клиентов, одновременно просматривающих разные видеофильмы, может вызвать произвольный ввод/вывод. По этой причине некоторые OCA были переведены на использование флеш-накопителей [Netflix 20].

1

550  Глава 9. Диски

yy использование внутри дисков алгоритмов, сокращающих затраты времени на позиционирование;

yy использование дисков с более высокой плотностью хранения данных, чтобы ограничить рабочую нагрузку;

yy настройка разделов (или «срезов»), например, с коротким ходом. Еще одна стратегия сокращения времени ожидания поворота — использование дисков с высокими скоростями вращения. Сейчас доступны жесткие диски с разными скоростями вращения, включая 5400, 7200, 10 000 и 15 000 оборотов в минуту. Но учтите: более высокие скорости могут привести к сокращению срока службы дисков из-за повышенного нагрева и износа.

Теоретическая максимальная пропускная способность Если известно максимальное количество секторов на дорожку диска, то его пропускную способность можно рассчитать по формуле: Максимальная пропускная способность = максимальное количество секторов на дорожку × размер сектора × об/мин/60 с. Эта формула более подходила для старых дисков, которые точно отображали исходную информацию. Современные диски предоставляют ОС виртуальный образ диска и искусственные значения этих атрибутов.

Короткий ход Короткий ход имеет место, когда рабочая нагрузка использует только внешние дорожки диска, а остальные либо не используются, либо используются рабочими нагрузками с низкими требованиями к пропускной способности (например, архивами). Это сокращает время позиционирования, потому что головки перемещаются на короткие расстояния и диск может оставить их в состоянии покоя на внешнем крае, уменьшая время на первое позиционирование после простоя. Кроме того, внешние дорожки обычно имеют более высокую пропускную способность из-за зонирования секторов (см. следующий подраздел). Следите за короткими ходами при изучении опубликованных результатов тестирования производительности дисков, особенно результатов, в которых не указаны цены и при получении которых могли использоваться несколько дисков с коротким ходом.

Зонирование секторов Все дорожки на диске имеют разную длину, самые короткие находятся ближе к центру диска, а самые длинные — у внешнего края. Зонирование секторов (также называемое многозонной записью) позволяет не фиксировать количество секторов (и битов), приходящееся на дорожку, а увеличивать его для более длинных дорожек, которые физически позволяют разбить их на большее число секторов. Поскольку скорость вращения постоянна, более длинные внешние дорожки обеспечивают более высокую пропускную способность (мегабайт в секунду) по сравнению с внутренними.

9.4. Архитектура  551

Размер сектора В индустрии хранения данных был разработан новый стандарт для дисковых устройств, получивший название «Advanced Format». Он определяет поддержку секторов большего размера, в частности 4 Кбайт. Это снижает оверхед на операции ввода/вывода, увеличивает пропускную способность, а также сокращает оверхед на хранение метаданных в каждом секторе диска. Секторы размером 512 байт по-прежнему могут поддерживаться микропрограммой диска через стандарт эмуляции «Advanced Format 512e». В некоторых дисках это может увеличить оверхед на ­запись, вызывая цикл чтения-изменения-записи для отображения 512 байт в сектор размером 4 Кбайт. К другим проблемам с производительностью, о которых следует знать, относятся смещенные 4-килобайтные операции ввода/ вывода, охватывающие два сектора, которые увеличивают объем ввода/вывода для их обслуживания.

Кэш внутри дискового устройства Общим компонентом современных жестких дисков является небольшой объем памяти, используемый для кэширования результатов чтения и буферизации записи. Эта память позволяет помещать операции ввода/вывода (команды) в очередь на устройстве и переупорядочивать их для большей эффективности. В SCSI эта очередь называется Tagged Command Queueing (TCQ), в SATA — Native Command Queuing (NCQ).

Алгоритм лифта для позиционирования Алгоритм лифта — это один из способов повышения эффективности работы за счет переупорядочения команд в очереди. Этот алгоритм переупорядочивает операции ввода/вывода с учетом их местоположения на диске так, чтобы минимизировать перемещение головок. Своим действием алгоритм напоминает строительный лифт, который обслуживает этажи не в порядке нажатия кнопок этажей, а движется вверх и вниз, останавливаясь на ближайшем этаже из запрошенных в данный момент. Такое поведение легко выявляется при исследовании трассировки дискового ввода/вывода — сортировка операций ввода/вывода по времени их начала не соответствует сортировке по времени окончания, то есть операции завершаются не по порядку. Кажется, что это дает очевидный выигрыш в производительности, но представьте следующий сценарий: на диск отправлен пакет операций ввода/вывода со смещением 1000 и одна операция со смещением 2000. Головки диска находятся в позиции со смещением 1000. Когда будет обслужен ввод/вывод со смещением 2000? Теперь представьте, что при обслуживании ввода/вывода со смещением, близким к 1000, поступает еще множество операций со смещением, близким к 1000, и такие операции продолжают поступать в достаточном количестве, чтобы удерживать диск занятым работой около смещения 1000 в течение 10 с. Когда

552  Глава 9. Диски будет обслужен ввод/вывод со смещением 2000 и какова будет его окончательная задержка ввода/вывода?

Целостность данных Диски хранят код исправления ошибок (error-correcting code, ECC) в конце каждого сектора. Это обеспечивает целостность данных и то, что диск может проверить их правильность при чтении или исправить любые возникающие ошибки. Если сектор был прочитан неправильно, диск может попытаться повторно прочитать его на следующем обороте (и повторить попытку несколько раз, немного меняя положение головки). Иногда этим объясняется необычно медленный ввод/вывод. Накопитель может возвращать ОС программные ошибки, объясняющие происходящее. Часто полезно наблюдать за частотой программных ошибок, так как ее увеличение может указывать на то, что вскоре диск может выйти из строя. Одно из преимуществ перехода с 512-байтных секторов на 4-килобайтные в том, что для одного и того же объема данных требуется меньше битов ECC, потому что ECC более эффективен для секторов большого размера [Smith 09]. Обратите внимание, что для проверки данных также могут использоваться другие контрольные суммы. Например, для проверки передачи данных в шину может использоваться циклический избыточный код (cyclic redundancy check, CRC), а файловые системы могут использовать другие контрольные суммы.

Вибрация В отличие от поставщиков дисковых устройств, хорошо знакомых с проблемой вибрации, в программной индустрии не всегда знали об этой проблеме и не воспринимали ее всерьез. В 2008 году, изучая загадочную проблему с производительностью, я провел эксперимент, подвергнув вибрационному воздействию дисковый массив во время выполнения теста записи. Это вызвало всплеск увеличения времени выполнения операций ввода/вывода. Мой эксперимент был снят на видео и размещен на YouTube. Он стал вирусным и был описан как первая демонстрация влияния вибрации на производительность диска [Turner 10]. Видео набрало более 1 700 000 просмотров, поспособствовав повышению осведомленности о проблемах влияния вибрации на диски [Gregg 08]. Судя по электронным письмам, я также случайно дал толчок к развитию индустрии звукоизоляции центров обработки данных: теперь вы можете нанять профессионалов, которые проанализируют уровни шумов и выполнят мероприятия, способствующие улучшению производительности дисков за счет гашения вибраций.

Ленивые диски Актуальная проблема производительности некоторых вращающихся дисков связана с открытием так называемых ленивых дисков (sloth disks). Эти диски иногда выполняют ввод/вывод очень медленно, тратя более одной секунды и не сообщая при этом ни о каких ошибках. Это время намного превышает время, необходимое

9.4. Архитектура  553 для повторных попыток чтения, когда получен неверный код ECC. Было бы лучше, если бы такие диски возвращали сообщение об отказе, а не занимали так много времени, чтобы ОС или контроллеры дисков могли предпринять корректирующие действия, например отключение диска в избыточных средах и отправку сообщения о сбое. Ленивые диски доставляют неудобства, особенно когда это часть виртуального диска, представленного дисковым массивом, и ОС не видит отдельные диски напрямую, что затрудняет их идентификацию1.

Черепичная магнитная запись Приводы с поддержкой черепичной магнитной записи (Shingled Magnetic Recording, SMR) обеспечивают более высокую плотность хранения данных за счет использования более узких дорожек. Такие дорожки слишком узкие для записывающей головки, но достаточно широкие для читающей, поэтому запись выполняется с частичным перекрытием дорожек, что напоминает укладку кровельной черепицы (отсюда и название). Диски, использующие SMR, обеспечивают увеличение плотности примерно на 25 % за счет снижения производительности записи, потому что перекрывающиеся данные уничтожаются и должны быть записаны повторно. Эти диски подходят для хранения архивов, которые записываются один раз и затем только читаются, но не подходят для рабочих нагрузок с большим объемом записи в конфигурациях RAID [Mellor 20].

Дисковый контроллер данных Механические диски представляют простой интерфейс для систем, предполагающих фиксированное количество секторов на дорожке и непрерывный диапазон адресуемых смещений. Что происходит на диске, зависит от контроллера данных — внут­ реннего микропроцессора, выполняющего микропрограмму. На дисках могут быть реализованы алгоритмы, включая зонирование секторов, влияющее на определение смещений. Об этом нужно знать, но такие тонкости сложно анализировать, потому что дисковый контроллер данных недоступен ОС.

9.4.1.2. Твердотельные накопители Твердотельные накопители (solid-state drives, SSD) также называют твердотельными дисками (solid-state disks). Термин твердотельный означает использование твердотельной электроники — программируемой энергонезависимой памяти, обладающей гораздо более высокой производительностью, чем вращающиеся диски. Эти диски, не имеющие движущихся частей, более прочные физически и не подвержены проблемам, связанным с вибрацией. Производительность дисков этого типа обычно одинакова для разных смещений (нет задержки, необходимой для позиционирования и ожидания поворота Если используется система Linux Distributed Replicated Block Device (DRBD), то она предоставляет параметр «disk-timeout».

1

554  Глава 9. Диски на нужный угол) и предсказуема для известного объема ввода/вывода. Произвольный или последовательный характер рабочей нагрузки здесь менее важен, чем для вращающихся дисков. Все это упрощает их анализ и планирование мощностей. Но при появлении проблем с производительностью из-за особенностей внут­реннего устройства разобраться в них так же сложно, как и с вращающимися дисками. Некоторые твердотельные накопители используют энергонезависимую память (NV-DRAM), но в большинстве конструкций применяется флеш-память.

Флеш-память Твердотельные накопители на основе флеш-памяти обеспечивают высокую производительность чтения, в том числе и произвольного чтения, которая может в разы превосходить производительность вращающихся дисков. Большинство из них основаны на использовании флеш-памяти NAND, в которой применяются электронные носители, захватывающие заряд и способные хранить электроны долгое время1, даже в отсутствие питания [Cornwell 12]. Название «флеш» относится к способу записи данных, требующему одновременного удаления всего блока памяти (до нескольких страниц, обычно 8 или 64 Кбайт на страницу) и перезаписи содержимого. Из-за оверхеда на запись флеш-память отличается асимметричной производительностью операций чтения и записи: чтение выполняется быстро, а запись — медленно. Накопители обычно сглаживают это различие, используя кэши с отложенной записью для повышения производительности записи и небольшой конденсатор в качестве резервного источника питания на случай сбоя. Есть несколько типов флеш-памяти:

yy с одноуровневыми ячейками (single-level cell, SLC): хранит по одному биту в каждой ячейке;

yy с многоуровневыми ячейками (multi-level cell, MLC): хранит несколько битов в каждой ячейке (обычно два, что требует четырех уровней напряжения);

yy с многоуровневыми ячейками промышленного уровня (enterprise multi-level cell, eMLC): флеш-память MLC с расширенным микропрограммным обеспечением, предназначенная для использования в промышленных условиях;

yy с трехуровневыми ячейками (tri-level cell, TLC): хранит три бита (восемь уровней напряжения) в каждой ячейке;

yy с четырехуровневыми ячейками (quad-level cell, QLC): хранит четыре бита в каждой ячейке;

yy 3D NAND/Vertical NAND (V-NAND): многослойная архитектура, включаю-

щая несколько слоев флеш-памяти (например, TLC) для увеличения плотности и емкости хранилища.

1

Но не до бесконечности. Ошибки хранения данных в современных MLC могут возникнуть уже через несколько месяцев после выключении питания [Cassidy 12], [Cai 15].

9.4. Архитектура  555 Этот список составлен в приблизительном хронологическом порядке, последними перечислены новейшие технологии: 3D NAND стала доступна для коммерческого распространения с 2013 года. Флеш-память SLC, как правило, имеет более высокую производительность и надежность по сравнению с другими типами. Она предпочтительнее для промышленного использования, хотя и стоит дороже. На предприятиях часто используется флешпамять MLC, обладающая более высокой плотностью, несмотря на ее не самую высокую надежность. Надежность флеш-памяти часто измеряется как количество операций записи блока (циклов программирования/стирания). Для SLC это количество составляет от 50 000 до 100 000 циклов, для MLC от 5000 до 10 000 циклов, для ТLC около 3000 циклов, а для QLC около 1000 циклов [Liu 20].

Контроллер Контроллер для SSD решает следующие задачи [Leventhal 13]:

yy ввод: читает и записывает страницы (обычно по 8 Кбайт), запись может происходить только в стертые страницы, страницы стираются блоками от 32 до 64 штук (256–512 Кбайт);

yy вывод: имитирует интерфейс жесткого диска: читает или записывает произвольные секторы (по 512 байт или 4 Кбайт).

Преобразование между вводом и выводом выполняется уровнем трансляции флешпамяти (flash translation layer, FTL) контроллера, который также должен следить за свободными блоками. По сути, для этой цели используется собственная файловая система, например журналируемая файловая система. Особенности записи могут вызывать проблемы с производительностью в пишущих рабочих нагрузках, особенно когда данные записываются блоками размером меньше размера блока флеш-памяти (который может достигать 512 Кбайт). Это может вызвать увеличение объема записи, когда оставшаяся часть блока копируется в другое место перед стиранием, а также задержку, по крайней мере, на продолжительность цикла стирания/записи. Некоторые флеш-накопители стараются ослабить проблему задержки, предоставляя внутренний буфер (на основе ОЗУ) с резервной батареей, чтобы записываемые данные можно было сохранить в буфер и записать позже даже в случае сбоя питания. Самый распространенный флеш-накопитель промышленного уровня, который я использовал, оптимально работал при чтении данных блоками по 4 Кбайт и записи блоками по 1 Мбайт, что объяснялось структурой флеш-памяти. Эти значения различаются для разных приводов, и их можно определить путем микробенчмаркинга размеров ввода/вывода. Учитывая несоответствие между внутренними операциями и блочным интерфейсом флеш-памяти, есть место для совершенствования операционной системы и ее файловых систем. Примером может служить команда TRIM: она сообщает твердотельному накопителю SSD, что некоторая область больше не используется; это

556  Глава 9. Диски упрощает для SSD сборку своего пула свободных блоков и ослабляет проблему увеличения объема записи. (Для SCSI то же самое можно реализовать с помощью команд UNMAP или WRITE SAME, для ATA — с помощью команд DATA SET MANAGEMENT. Поддержка Linux включает параметр discard команды mount и команду fstrim(8).)

Продолжительность жизни Использование флеш-памяти NAND в качестве носителя информации сопряжено с разными проблемами, включая выгорание, затухание данных и нарушение чтения [Cornwell 12]. Их можно решить с помощью контроллера SSD, который способен перемещать данные, чтобы избежать этих проблем. Обычно контроллер использует алгоритм выравнивания износа, который распределяет записываемые данные по разным блокам, чтобы сократить количество циклов записи для отдельных блоков, и избыточное выделение памяти, резервирующее дополнительную память, которую можно было бы использовать для обслуживания. Все эти методы увеличивают срок службы, но блоки в SSD по-прежнему имеют ограниченное количество циклов записи, в зависимости от типа флеш-памяти и алгоритмов выравнивания износа, используемых накопителем. В накопителях промышленного уровня используется избыточное выделение памяти и самый надежный тип флеш-памяти SLC, допускающий от 1 миллиона циклов записи и выше. Приводы потребительского уровня на основе MLC могут допускать всего 1000 циклов.

Патологии Твердотельные накопители с флеш-памятью страдают некоторыми патологиями, о которых следует знать и помнить:

yy выбросы по величине задержки из-за старения, обусловленные тем, что SSD пытается извлечь правильные данные (проверяя их с помощью ECC);

yy высокая задержка из-за фрагментации (эту проблему можно исправить, выполнив форматирование и очистив карты блоков FTL);

yy низкая пропускная способность, если в SSD реализовано внутреннее сжатие. Проверьте наличие других разработок, влияющих на производительность SSD, и выявленных проблем.

9.4.1.3. Энергонезависимая память Для поддержки кэшей отложенной записи в контроллерах устройств хранения используется энергонезависимая память в форме DRAM с питанием от батарей1. Производительность памяти этого типа на порядки выше производительности

Батарей или суперконденсатора.

1

9.4. Архитектура  557 флеш-памяти, но стоимость и ограниченный срок службы батарей позволяют использовать их только в особых случаях. Новый тип энергонезависимой памяти под названием 3D XPoint, разработанный компаниями Intel и Micron, расширяет круг использования памяти этого типа благодаря привлекательному соотношению цена/производительность, по которому она занимает промежуточное положение между DRAM и флеш-памятью. 3D XPoint хранит биты в наращиваемом массиве в виде сетки и имеет байтовую адресацию. По результатам тестирования, проведенного в Intel, память 3D XPoint показала задержку доступа в 14 мкс против 200 мкс для твердотельных накопителей 3D NAND [Hady 18]. 3D XPoint также показала стабильность задержки, тогда как для 3D NAND был отмечен разброс задержек в более широком диапазоне, достигающем 3 мс. Для коммерческого использования память 3D XPoint стала доступна в 2017 году. Intel выпускает модули энергонезависимой памяти и твердотельные накопители под маркой Intel Optane.

9.4.2. Интерфейсы Интерфейс — это протокол, поддерживаемый приводом для взаимодействия с системой, обычно через контроллер диска. Ниже приводится краткое описание интерфейсов SCSI, SAS, SATA, FC и NVMe. Проверьте, какие интерфейсы используются прямо сейчас, и изучите их пропускную способность — они имеют свойство меняться по мере развития существующих и появления новых спецификаций.

SCSI Интерфейс малых вычислительных систем (Small Computer System Interface, SCSI) первоначально имел вид параллельной транспортной шины, использующей несколько электрических разъемов для параллельной передачи битов. Первая версия SCSI-1, появившаяся в 1986 году, имела ширину шины данных 8 бит и позволяла передавать один байт, обеспечивая пропускную способность 5 Мбайт/с. Она подключалась с помощью 50-контактного разъема Centronics C50. В поздних версиях SCSI использовались более широкие шины данных с разъемами до 80 контактов, пропускная способность достигала сотен мегабайт. Поскольку параллельный интерфейс SCSI — это общая шина, он может испытывать проблемы с производительностью из-за конфликтов на шине, например, когда запланированное резервное копирование системы насыщает шину низкоприоритетным вводом/выводом. Для решения этой проблемы устройства с низким приоритетом часто подключались к отдельной шине SCSI или контроллеру. На высоких скоростях синхронизация параллельных шин становится проблемой. Наряду с другими проблемами (включая ограниченное количество устройств и необходимость использования терминаторов SCSI) это привело к переходу на последовательную версию: SAS.

558  Глава 9. Диски

SAS Последовательный интерфейс SCSI (Serial Attached SCSI) проектировался как высокоскоростной двухточечный транспорт, позволяющий избежать конфликтов на шине параллельного интерфейса SCSI. Первоначальная спецификация SAS-1 поддерживала пропускную способность 3 Гбит/с (выпущена в 2003 году). За ней последовали спецификации SAS-2 с пропускной способностью 6 Гбит/с (2009), SAS-3 — 12 Гбит/с (2012) и SAS-4 — 22,5 Гбит/с (2017). Спецификация SAS поддерживает агрегирование каналов, что позволяет объединить несколько портов для достижения более высокой пропускной способности. Фактическая скорость передачи данных составляет 80 % полосы пропускания из-за кодирования 8-битных данных в 10-битные. В числе других особенностей SAS можно назвать поддержку приводов с двумя портами для подключения резервных разъемов и резервных архитектур, многоканальный ввод/вывод, домены SAS, горячую замену и поддержку совместимости с устройствами SATA. Благодаря этому SAS стал популярен в промышленном секторе, особенно с резервными архитектурами.

SATA По тем же причинам, что SCSI и SAS, стандарт параллельного интерфейса ATA (он же IDE) превратился в последовательный интерфейс обмена данными с накопителями информации (Serial ATA). Первоначальная спецификация SATA 1.0 (2003) поддерживала пропускную способность 1,5 Гбит/с. Более поздняя версия SATA 2.0 — 3,0 Гбит/с (2004) и SATA 3.0 — 6,0 Гбит/с (2008). В основные и промежуточные версии спецификации были добавлены такие особенности, как встроенная поддержка очереди команд. SATA использует кодирование 8-битных данных в 10-битные, поэтому скорость передачи составляет 80 % от пропускной способности. Интерфейс SATA широко используется в настольных компьютерах и ноутбуках.

FC Оптоволоконный канал (Fibre Channel, FC) — это стандарт высокоскоростного интерфейса для передачи данных, изначально предназначавшийся только для использования с оптоволоконным кабелем (отсюда и название), но позднее в него была добавлена поддержка медных кабелей. FC обычно используется в промышленных средах для создания сетей хранения данных (storage area networks, SAN), в которых несколько устройств хранения могут быть подключены к нескольким серверам через оптоволоконную сеть (Fibre Channel Fabric). Он обеспечивает большую масштабируемость и доступность, чем другие интерфейсы, и действует подобно хостам в сети. Как и в обычной сети, FC позволяет использовать коммутаторы для соединения нескольких локальных конечных точек (сервера и хранилища). Разработка стандарта Fibre Channel началась в 1988 году, и первая версия, одобренная в ANSI, вышла в 1994 году [FICA 20]. С тех пор появилось множество вариантов

9.4. Архитектура  559 и улучшений: в недавнем стандарте Gen 7 256GFC пропускная способность достигла 51 200 Мбайт/с в полнодуплексном режиме [FICA 18].

NVMe Высокопроизводительная энергонезависимая память (Non-Volatile Memory express, NVMe) — это спецификация шины PCIe для устройств хранения. В отличие от устройств хранения, подключающихся к карте контроллера хранилища, устройство NVMe само по себе — это карта, которая подключается непосредственно к шине PCIe. Работы над спецификацией NVMe начались в 2011 году. Первая версия 1.0e вышла в 2013 году, а последняя версия 1.4 — в 2019-м [NVMe 20]. В новые спецификации были добавлены различные особенности: управление температурным режимом и команды для самотестирования, проверка и очистка данных (исключающая возможность восстановления). Пропускная способность карт NVMe ограничена пропускной способностью шины PCIe. PCIe версии 4.0, широко используемая сегодня, имеет однонаправленную пропускную способность 31,5 Гбайт/с для карты x16 (ширина канала). Главное преимущество NVMe перед традиционными SAS и SATA — поддержка нескольких аппаратных очередей. Эти очереди могут использоваться одним и тем же процессором для «разогрева» кэша (а с поддержкой нескольких очередей в Linux можно даже избежать использования общих блокировок в ядре). Эти очереди дают обширные возможности буферизации, позволяя сохранять в каждой очереди до 64 тысяч команд, тогда как типичные SAS и SATA ограничены 256 и 32 командами соответственно. NVMe также поддерживает виртуализацию ввода/вывода с единым корнем (singleroot I/O virtualization, SR-IOV) для повышения производительности хранилища виртуальных машин (см. главу 11 «Облачные вычисления», раздел 11.2 «Виртуализация оборудования»). NVMe используется для флеш-устройств с малой задержкой ввода/вывода менее 20 мкс.

9.4.3. Типы хранилищ Хранилище может быть организовано на сервере несколькими способами. В подразделах ниже описаны четыре наиболее распространенные архитектуры: дисковые устройства, RAID, массивы хранения и сетевые хранилища (NAS).

Дисковые устройства Самая простая архитектура — это сервер с внутренними дисками, которыми ОС управляет по отдельности. Диски подключаются к контроллеру дисков, реализованному в виде схемы на материнской плате или отдельной карты расширения, который обеспечивает доступ к дисковым устройствам. В этой архитектуре контроллер дисков действует просто как канал связи, позволяющий системе обмениваться данными с дисками. В типичном персональном компьютере или ноутбуке диск, играющий роль основного хранилища, подключен именно таким образом.

560  Глава 9. Диски Эту архитектуру проще всего анализировать с помощью инструментов оценки производительности, потому что каждый диск известен ОС и за ними можно наблюдать по отдельности. В терминологии некоторых контроллеров дисков такая архитектура называется простой связкой дисков (just a bunch of disks, JBOD).

RAID Усовершенствованные контроллеры дисков могут поддерживать архитектуру избыточного массива независимых дисков (redundant array of independent disks, RAID) для дисковых устройств (первоначально эта архитектура называлась избыточным массивом недорогих дисков — redundant array of inexpensive disks [Patterson 88]). RAID может представлять несколько дисков как один большой, быстрый и надежный виртуальный диск. Эти контроллеры часто имеют встроенный кэш для увеличения производительности чтения и записи. Если дисковый массив формируется с использованием карты контроллера RAID, то такой массив называется аппаратным RAID. Дисковый массив может быть организован и программно операционной системой, но традиционно предпочтение отдавалось аппаратному RAID, поскольку дорогостоящие вычисления контрольных сумм и четности предпочтительнее выполнять на выделенном оборудовании. Также такое оборудование часто включает блок резервного питания от батареи для повышения отказоустойчивости. Но с появлением процессоров с большим количеством ядер оверхед на вычисление четности стал менее ощутимым. В ряде решений для хранения данных снова стали использовать программный RAID (например, с применением ZFS), что помогло снизить сложность и стоимость оборудования и улучшило наблюдаемость со стороны ОС. В случае серьезного сбоя программный RAID также часто проще восстановить, чем аппаратный RAID (представьте вышедшую из строя карту RAID). В подразделах ниже описаны характеристики производительности RAID. Говоря о RAID, часто можно услышать термин чередование (stripe): он относится к случаям, когда данные, сгруппированные в блоки, записываются на несколько дисков (как бы прочерчивая полосу — stripe — через все диски).

Типы Для разных потребностей в смысле емкости, производительности и надежности доступны разные типы RAID. В этом подразделе основное внимание уделим характеристикам, приведенным в табл. 9.3. Уровень RAID-0 с чередованием обладает самой высокой производительностью, но не имеет избыточности, что делает его малопригодным для промышленного использования. Возможные исключения — отказоустойчивые облачные среды, которые не хранят важные данные и где отказавшие экземпляры автоматически замещаются новыми, а также серверы хранения, используемые только для кэширования.

9.4. Архитектура  561 Таблица 9.3. Типы RAID Уровень

Описание

Производительность

0 (последовательный)

Диски заполняются последовательно, друг за другом

Увеличивает производительность произвольного чтения, когда в работе может участвовать несколько дисков

0 (с чередованием)

Диски заполняются параллельно, данные распределяются (чередуются) сразу по нескольким дискам

Как предполагается, такая организация обеспечивает наилучшую производительность произвольного и последовательного ввода/вывода (зависит от размера полосы чередования и характера рабочей нагрузки)

1 (зеркалирование)

Диски объединяются в группы (обычно по два) и хранят одни и те же данные, обеспечивая резервирование

Хорошая производительность произвольного и последовательного чтения (чтение может выполняться со всех дисков одновременно, в зависимости от реализации). Скорость записи ограничена самым медленным диском в зеркале, а оверхед удваивается (если диски группируются по два)

10

Комбинация RAID-0 с чередованием и RAID-1 с группировкой дисков, обеспечивающая емкость и избыточность

Характеристики производительности аналогичны уровню RAID-1, но допускается участие большего количества групп дисков для увеличения пропускной способности как в RAID-0

5

Данные хранятся с чередованием на нескольких дисках вместе с дополнительной информацией о четности для избыточности

Невысокая производительность записи из-за цикла «чтение-изменение-запись» и вычислений четности

6

RAID-5 с двумя дисками для хранения с чередованием информации о четности

Близкая к RAID-5, но немного хуже

Наблюдаемость Как описывалось выше в разделе о потреблении виртуальных дисков, использование аппаратных виртуальных дисков может затруднить наблюдение за ними из ОС, которая не знает, чем заняты физические диски. Если дисковый массив RAID организован программно, то обычно есть возможность наблюдать за работой отдельных дисковых устройств, потому что ОС сама управляет ими.

Чтение-изменение-запись Когда данные хранятся с чередованием и с информацией о четности, как в RAID-5, каждая операция записи может повлечь за собой дополнительные операции чтения и затраты времени на вычисления. Это связано с тем, что для записи блока данных, размер которого меньше размера полосы, может потребоваться прочитать всю полосу, изменить в ней требуемые байты, повторно вычислить четности и записать полосу обратно. Чтобы избежать этого, можно использовать оптимизацию для RAID-5: вместо всей полосы прочитать только часть полосы (полос), включающую

562  Глава 9. Диски измененные данные, вместе с информацией о четности, выполнить последовательность операций XOR, чтобы вычислить обновленное значение четности, и записать его вместе с измененной частью полосы. Запись блока, охватывающего всю полосу, можно произвести без предварительного чтения. Производительность в таких средах можно улучшить, выбрав размер полосы равным среднему размеру операций записи, чтобы уменьшить дополнительный оверхед на чтение.

Кэши Контроллеры дисков, реализующие RAID-5, могут ослабить проблему падения производительности из-за цикла «чтение-изменение-запись» за счет использования кэша отложенной записи. Эти кэши должны иметь резервное питание от батареи, чтобы в случае пропажи питания они могли завершить буферизованную запись.

Дополнительные возможности Имейте в виду, что усовершенствованные контроллеры дисков могут поддерживать дополнительные возможности, влияющие на производительность. Загляните в документацию производителя, чтобы получить хотя бы общее представление об имеющихся возможностях. Например, вот пара возможностей, которыми обладает контроллер Dell PERC 5 [Dell 20]:

yy Patrol read: каждые несколько дней контроллер читает все блоки диска и про-

веряет их контрольные суммы. Если диски заняты обслуживанием запросов, то на patrol read выделяется меньше ресурсов, чтобы исключить конкуренцию с рабочей нагрузкой.

yy Интервал выталкивания кэша: время в секундах между принудительным вытал-

киванием измененных данных из кэша на диск. Установка большего интервала может уменьшить объем дискового ввода/вывода за счет сокращения операций записи и улучшения агрегирования записываемых данных. При этом может увеличиться задержка операций чтения в моменты, когда на диск выталкиваются значительные объемы данных из кэша.

Оба этих фактора могут существенно повлиять на производительность.

Массивы хранения Массивы хранения позволяют подключать к системе множество дисков. Для их организации используются усовершенствованные контроллеры дисков, позволяющие настроить RAID и обычно имеющие кэш большой емкости (измеряемой гигабайтами) для увеличения производительности чтения и записи. Также эти кэши обычно питаются от батарей, что позволяет им работать в режиме отложенной записи. Распространенная тактика — переключение в режим сквозной записи в случае отказа батареи, что может быть заметно по внезапному падению производительности записи из-за цикла «чтение-изменение-запись».

9.4. Архитектура  563 На производительность влияет и способ подключения к системе массива хранения — обычно для этой цели используется внешний контроллер хранения. Контроллер и транспорт, соединяющий его с массивом хранения, оба будут иметь свои ограничения на количество операций ввода/вывода в секунду и пропускную способность. Для увеличения производительности и надежности массивы хранения часто поддерживают двойное подключение, то есть их можно подключать двумя физическими кабелями к одному или двум контроллерам хранения.

Сетевые хранилища Сетевые хранилища (network attached storage, NAS) подключаются к системе по сети с использованием сетевого протокола, такого как NFS, SMB/CIFS или iSCSI. Обычно они организуются как выделенные системы, которые называют устройствами NAS. Это отдельные системы, и их следует анализировать как таковые. Однако и на стороне клиента можно выполнить некоторый анализ производительности, например оценить прикладываемую рабочую нагрузку и измерить задержки ввода/вывода. Важным фактором в таких средах становится производительность сети, и проблемы могут возникать из-за перегрузки сети и задержек с несколькими сетевыми переходами.

9.4.4. Стек дискового ввода/вывода в операционной системе Точный состав компонентов и слоев в стеке дискового ввода/вывода зависит от ОС, версии, а также используемых программных и аппаратных технологий. На рис. 9.7 изображена обобщенная модель. Аналогичная модель, включающая приложения, описывается в главе 3 «Операционные системы». Файловая система / Диспетчер томов

Низкоуровневый ввод/вывод с блочным устройством

Интерфейс блочного устройства Кэш буферов Целевой драйвер ввода/вывода Драйвер многоканального ввода/вывода (если имеется) Драйвер контроллера шины Дисковые устройства

Рис. 9.7. Обобщенный стек дискового ввода/вывода

564  Глава 9. Диски

Интерфейс блочного устройства Интерфейс блочного устройства был разработан на ранних этапах развития Unix для доступа к устройствам с единицами хранения в виде блоков по 512 байт и поддержки кэша буферов для повышения производительности. Этот интерфейс перекочевал в Linux, хотя важность кэша буферов уменьшилась с появлением других кэшей в файловых системах, о которых рассказывается в главе 8 «Файловые системы». Unix поддерживает возможность выполнения ввода/вывода в обход кэша буферов, которая называется низкоуровневым вводом/выводом с блочным устройством (или просто низкоуровневым вводом/выводом), через специальные файлы символьных устройств (см. главу 3 «Операционные системы»). Эти файлы больше недоступны по умолчанию в Linux. Низкоуровневый ввод/вывод с блочным устройством отличается от функции «прямого ввода/вывода» в файловой системе, описанной в главе 8 «Файловые системы», но имеет некоторые схожие черты. Интерфейс блочного ввода/вывода обычно можно исследовать с помощью инструментов ОС (iostat(1)). Он поддерживает широкий выбор точек для статической инструментации, а в последнее время появилась возможность исследования с помощью динамической инструментации. Linux расширил эту область ядра дополнительными возможностями.

Linux На рис. 9.8 показаны основные компоненты стека блочного ввода/вывода в Linux. В Linux усовершенствовали поддержку блочного ввода/вывода. В нее была добавлена возможность слияния ввода/вывода, более эффективные планировщики, поддержка группировки нескольких устройств в диспетчерах томов и средства отображения физических устройств для создания виртуальных устройств.

Слияние ввода/вывода При создании новых запросов на выполнение ввода/вывода Linux может объединять и группировать их, как показано на рис. 9.9. Такая группировка ввода/вывода уменьшает оверхед процессора на каждый ввод/ вывод в подсистеме хранения ядра и оверхед дисков, а также повышает пропускную способность. Статистики слияний спереди и сзади доступны в iostat(1). После слияния выполняется планирование передачи запросов ввода/вывода на диски.

Планировщики ввода/вывода Запросы на ввод/вывод ставятся в очередь и планируются на уровне блочного интерфейса либо классическими планировщиками (присутствующими только в версиях Linux старше 5.0), либо более новыми планировщиками с несколькими

9.4. Архитектура  565 очередями. Планировщики могут переупорядочивать (или перепланировать) запросы для оптимизации. Такое поведение может улучшить производительность и более справедливо распределить потребление ресурсов, особенно при использовании устройств с высокими задержками ввода/вывода (вращающиеся диски). Кэш страниц

Файловая система

Кэш буферов

Низкоуровневый ввод/вывод с блочным устройством

Интерфейс блочного устройства Диспетчер томов (если используется) Модуль отображения устройств (если используется)

Блочный уровень Планировщики с неКлассические сколькими очередями планировщики

Драйвер контроллера шины (SCSI)

Дисковые устройства

Рис. 9.8. Стек ввода/вывода в Linux

99

Добавления Добавление в начало в конец

Очередь ввода/вывода Дисковый адрес

Группировка

Рис. 9.9. Типы слияния ввода/вывода

566  Глава 9. Диски К классическим планировщикам относятся:

yy Noop: этот планировщик не выполняет никакого планирования (noop на языке процессоров означает no-operation — ничего не делать) и может использоваться, когда оверхед на планирование считается избыточным (например, для RAMдиска).

yy Deadline: пытается установить крайний срок задержки. Используя этот плани-

ровщик, можно, например, выбрать предельное время в миллисекундах, в течение которого должна быть выполнена операция чтения или записи. Это может пригодиться в системах реального времени, когда определенность является одним из важных качеств. Также этот планировщик может решить проблему голодания: когда запрос на выполнение ввода/вывода испытывает нехватку дисковых ресурсов из-за того, что вновь запущенный ввод/вывод перескакивает через очередь, вследствие чего происходит выброс по задержке. Голодание может быть обусловлено проблемой задержки чтения из-за выполнения записи (writes starving reads), удерживающей головки в одной области диска из-за интенсивного ввода/вывода, из-за чего другие операции ввода/вывода испытывают голодание. Планировщик deadline частично решает эту проблему, используя три отдельные очереди ввода/вывода: очередь для чтения (FIFO), очередь для записи (FIFO) и отсортированную очередь [Love 10].

yy CFQ: полностью справедливый планировщик очередей (completely fair queueing) распределяет временные интервалы ввода/вывода между процессами, по аналогии с планированием процессорного времени, для справедливого распределения дисковых ресурсов. Он также позволяет устанавливать приоритеты и классы для пользовательских процессов с помощью команды ionice(1).

Проблема классических планировщиков заключается в использовании единственной очереди запросов, защищенной единственной блокировкой, которая часто оказывалась узким местом при высоких скоростях ввода/вывода. Драйвер с несколькими очередями (blk-mq, добавленный в Linux 3.13) решает эту проблему, используя отдельные очереди передачи для каждого процессора и несколько очередей для устройств. Это обеспечивает лучшую производительность и меньшую задержку ввода/вывода по сравнению с классическими планировщиками, потому что запросы могут обрабатываться параллельно и на том же процессоре, на котором был инициирован ввод/вывод. Это было нужно для поддержки устройств на основе флеш-памяти и других типов устройств, способных выполнять миллионы операций ввода/вывода в секунду [Corbet 13b]. К планировщикам с несколькими очередями относятся:

yy None: планировщик без очереди. yy BFQ: планировщик очередей со справедливым распределением бюджета (budget fair queueing), действующий как и планировщик CFQ, но вдобавок к времени ввода/вывода распределяющий еще и полосу пропускания. Он создает очередь для каждого процесса, выполняющего дисковый ввод/вывод, и поддерживает бюджет для каждой очереди, измеряемый в секторах. Также есть общесистемный

9.5. Методология  567 бюджет тайм-аутов, чтобы ни один процесс не мог удерживать устройство слишком долго. Кроме того, планировщик BFQ поддерживает контрольные группы cgroups.

yy mq-deadline: версия планировщика deadline (описан выше) с драйвером blk-mq. yy Kyber: планировщик, регулирующий длину очереди на чтение и запись в зависи-

мости от производительности, чтобы обеспечить достижение целевых задержек при чтении или записи. Этот простой планировщик имеет всего две настройки: целевая задержка чтения (read_lat_nsec) и целевая задержка синхронной записи (write_lat_nsec). Планировщик Kyber хорошо зарекомендовал ctmz в облаке Netflix, где используется по умолчанию.

Начиная с Linux 5.0, планировщики с несколькими очередями используются по умолчанию (классические планировщики отключены). Планировщики ввода/вывода подробно описываются в исходном коде Linux в каталоге Documentation/block. После этапа планирования запрос помещается в очередь для отправки блочному устройству.

9.5. МЕТОДОЛОГИЯ В этом разделе описаны методологии анализа и настройки дискового ввода/вывода. Краткий перечень методологий приводится в табл. 9.4. Таблица 9.4. Методологии анализа производительности дисков Раздел

Методология

Тип

9.5.1

Метод инструментов

Анализ наблюдения

9.5.2

Метод USE

Анализ наблюдения

9.5.3

Мониторинг производительности

Анализ наблюдения, планирование мощности

9.5.4

Определение характеристик рабочей нагрузки

Анализ наблюдения, планирование мощности

9.5.5

Анализ задержек

Анализ результатов наблюдения

9.5.6

Статическая настройка производительности

Анализ наблюдения, планирование мощности

9.5.7

Настройка кэширования

Анализ наблюдения, настройка

9.5.8

Управление ресурсами

Настройка

9.5.9

Микробенчмаркинг

Анализ результатов экспериментов

9.5.10

Масштабирование

Планирование мощности, настройка

Список дополнительных методологий и краткое введение в них ищите в главе 2 «Методологии».

568  Глава 9. Диски Эти методологии можно применять по отдельности или в сочетании друг с другом. При устранении проблем с дисками предлагаю начать с использования следующих стратегий в таком порядке: метод USE, мониторинг производительности, определение характеристик рабочей нагрузки, анализ задержек, микробенчмаркинг, статический анализ и трассировка событий. В разделе 9.6 «Инструменты наблюдения» показаны приемы применения этих методологий с использованием инструментов ОС.

9.5.1. Метод инструментов Метод инструментов — это процесс перебора доступных инструментов с целью изу­ чения ключевых метрик, которые они возвращают. Это очень простая методология, но при ее использовании могут оставаться незамеченными некоторые проблемы, которые плохо обнаруживаются или вообще не обнаруживаются инструментами, и на ее применение может уйти много времени. При анализе дисков можно использовать следующие инструменты (для Linux):

yy iostat: при использовании в расширенном режиме помогает выявить высокое

потребление дисков (более 60 %), высокое среднее время обслуживания (скажем, более 10 мс) и большое количество операций ввода/вывода в секунду (в зависимости от обстоятельств);

yy iotop/biotop: помогает определить, какой процесс генерирует дисковый ввод/ вывод;

yy biolatency: с помощью этого инструмента можно получить распределение задержек ввода/вывода в виде гистограммы и выявить мультимодальные распределения и выбросы задержек (более 100 мс);

yy biosnoop: позволяет исследовать отдельные операции ввода/вывода; yy perf(1)/BCC/bpftrace: позволяет исследовать стеки в пространствах пользователя и ядра, ведущих к дисковому вводу/выводу;

yy инструменты для исследования конкретных контроллеров дисков (от поставщика).

При обнаружении проблемы изучите все поля в выводе доступных инструментов, чтобы получить больше информации. Подробности о каждом инструменте ищите в разделе 9.6 «Инструменты наблюдения». Для выявления других типов проблем используйте и другие методологии.

9.5.2. Метод USE Метод USE можно использовать на ранних этапах исследования производительности для выявления узких мест и ошибок во всех компонентах, прежде чем переходить к более глубоким и трудоемким стратегиям. В подразделах ниже описано,

9.5. Методология  569 как можно применить метод USE к дисковым устройствам и контроллерам, а в разделе 9.6 «Инструменты наблюдения» перечислены инструменты для оценки конкретных показателей.

Дисковые устройства Для каждого диска проверьте:

yy Потребление: время, в течение которого устройство было занято. yy Насыщенность: время, в течение которого запросы на ввод/вывод ждали своей очереди.

yy Ошибки: ошибки устройства. Сначала можно проверить ошибки. Иногда на них не обращают внимания, потому что система работает нормально, хотя и медленнее, несмотря на сбои дисков: диски обычно объединяются в пул с целью увеличения отказоустойчивости. Помимо стандартных счетчиков ошибок в ОС, дисковые устройства могут поддерживать более широкий спектр своих счетчиков, значения которых можно получить с помощью специальных инструментов (например, SMART data1). Если дисковые устройства — это физические диски, то оценка их потребления не должна быть сложной задачей. Если это виртуальные диски, то оценка потребления может не отражать характера работы физических дисков. См. раздел 9.3.9 «Потребление», где этот вопрос обсуждается более подробно.

Контроллеры дисков Для каждого контроллера дисков проверьте:

yy Потребление: текущую и максимальную пропускную способность, время и аналогичные метрики для частоты операций.

yy Насыщенность: время, в течение которого запросы ждали своей очереди из-за насыщения контроллера.

yy Ошибки: ошибки контроллера. Здесь метрика потребления определяется не с точки зрения времени, а с точки зрения ограничений контроллера диска: пропускной способности (байтов в секунду) и скорости работы (операций в секунду). Под операциями подразумеваются чтение/ запись и другие дисковые команды. Пропускная способность или скорость работы также могут ограничиваться транспортом, соединяющим контроллер с системой или контроллер с отдельными дисками. Каждый транспорт нужно проверять одинаково: ошибки, потребление, насыщение.

В Linux см. такие инструменты, как MegaCLI и smartctl (рассматриваются ниже), cciss-vol-status, cpqarrayd, varmon и dpt-i2o-raidutils.

1

570  Глава 9. Диски Вы можете обнаружить, что инструменты наблюдения (например, iostat(1) в Linux) предоставляют метрики не для контроллеров, а для дисков. Эта проблема имеет обходное решение: если в системе только один контроллер, то можно определить количество операций ввода/вывода в секунду и пропускную способность контроллера, суммируя эти метрики для всех дисков. Если в системе несколько контроллеров, то нужно определить, какие диски к какому контроллеру подключены, и просуммировать соответствующие метрики. Производительность контроллеров и транспортов часто упускается из виду. К счастью, они редко становятся узким местом в системе, потому что их пропускная способность обычно превышает пропускную способность подключенных дисков. Если общая пропускная способность диска или количество операций ввода/вывода в секунду всегда снижаются с определенной скоростью, даже при разных рабочих нагрузках, то это может указывать на то, что источником проблем действительно являются контроллеры или транспорты.

9.5.3. Мониторинг производительности Мониторинг производительности помогает обнаружить проблемы и модели поведения, проявляющиеся со временем. Ключевые метрики для дискового ввода/ вывода:

yy потребление диска; yy время отклика. Потребление диска на уровне 100 % в течение нескольких секунд, скорее всего, признак проблемы. В зависимости от среды уровень потребления более 60 % тоже может вызвать снижение производительности из-за увеличения очереди. «Нормальный» или «высокий» уровень потребления зависит от рабочей нагрузки, среды и требований к задержкам. Если вы не уверены, то можно провести микробенчмаркинг с использованием заведомо умеренных и тяжелых рабочих нагрузок, чтобы понять, как их можно определить, опираясь на метрики потребления диска. См. раздел 9.8 «Эксперименты». Эти метрики следует оценивать для каждого диска, чтобы выявить несбалансированные рабочие нагрузки и отдельные плохо работающие диски. Метрика времени отклика может вычисляться как среднее значение в секунду и включать другие значения, например максимальное и стандартное отклонение. В идеале нужно исследовать полное распределение времени отклика, например построить гистограмму или тепловую карту, чтобы не пропустить выбросы и другие закономерности. Если в системе используются средства управления дисковыми ресурсами, то можно также собрать статистику, показывающую, как и когда эти ресурсы использовались. Дисковый ввод/вывод может быть узким местом из-за наложенных ограничений, а не из-за высокой активности самого диска.

9.5. Методология  571 Потребление и время отклика показывают результирующую производительность диска. Для определения характеристик рабочей нагрузки можно добавить дополнительные метрики, включая количество операций ввода/вывода в секунду и пропускную способность, которые служат важными данными при планировании мощности (см. следующий раздел и раздел 9.5.10 «Масштабирование»).

9.5.4. Определение характеристик рабочей нагрузки Определение характеристик приложенной нагрузки — важный шаг при планировании мощностей, сравнительном анализе и моделировании рабочих нагрузок. Это также помогает значительно увеличить производительность за счет выявления и устранения ненужной работы. Основные атрибуты, характеризующие нагрузку на дисковый ввод/вывод:

yy yy yy yy yy

частота операций ввода/вывода; пропускная способность ввода/вывода; объем ввода/вывода; соотношение операций чтения/записи; произвольный и последовательный доступ к данным на дисках.

Влияние на производительность произвольного и последовательного ввода/вывода, соотношения операций чтения/записи и объема ввода/вывода было описано в разделе 9.3 «Основные понятия». Частота операций ввода/вывода (IOPS) и пропускная способность были определены в разделе 9.1 «Терминология». Эти характеристики могут быстро меняться со временем, особенно когда в системе выполняются приложения, активно использующие файловые системы и периодически буферизующие и выталкивающие на диск блоки данных. Чтобы точнее охарактеризовать рабочую нагрузку, фиксируйте как максимальные, так и средние значения. Еще лучше изучить полное распределение значений во времени. Пример ниже показывает, как все эти атрибуты можно выразить в одном описании: Системные диски испытывают небольшую рабочую нагрузку произвольного чтения, в среднем 350 операций в секунду с пропускной способностью 3 Мбайт/с. Количество операций чтения составляет 96 %. Время от времени бывают короткие всплески последовательных операций записи, продолжительностью от 2 до 5 с, в результате частота операций дискового ввода/вывода достигает максимального значения 4800 в секунду, а пропускная способность возрастает до 560 Мбайт/с. В среднем чтение выполняется блоками по 8 Кбайт, а запись — по 128 Кбайт. Эти характеристики можно измерить не только в масштабе всей системы, но также для отдельных дисков и контроллеров.

572  Глава 9. Диски

Чек-лист для определения дополнительных характеристик рабочей нагрузки Для характеристики рабочей нагрузки можно включить дополнительные сведения. Они перечислены ниже в форме вопросов, которые могут послужить чек-листом при исследовании проблем с дисками:

yy Какова общесистемная частота операций ввода/вывода в секунду? Для каждого диска? Для каждого контроллера?

yy Какова пропускная способность всей системы? Каждого диска? Каждого контроллера?

yy Какие приложения или пользователи используют диски? yy К каким файловым системам или файлам осуществляется доступ? yy Были ли обнаружены ошибки? Эти ошибки обусловлены неверными запросами или проблемами с диском?

yy Насколько сбалансирован ввод/вывод между доступными дисками? yy Какова частота операций ввода/вывода для каждой задействованной транспортной шины?

yy Какая доля пропускной способности приходится на каждую задействованную транспортную шину?

yy Какие выполняются команды, не связанные с передачей данных? yy Почему выполняются операции дискового ввода/вывода (пути в коде, ведущие к системным вызовам в ядре)?

yy Какая доля дискового ввода/вывода приходится на операции, выполняемые приложениями синхронно?

yy Как распределяется время поступления запросов на ввод/вывод? Вопросы, затрагивающие частоту операций и пропускную способность, можно задать отдельно для операций чтения и записи. Любые из этих метрик можно исследовать во времени, чтобы найти максимальные и минимальные значения, а также характер изменения во времени. Cм. также раздел 2.5.11 «Определение характеристик рабочей нагрузки» в главе 2 «Методологии», где приводится обобщенное описание этих и других характеристик (кто, почему, зачем, как).

Определение характеристик производительности В предыдущих чек-листах перечисляются метрики, определяющие характеристики применяемой рабочей нагрузки. В списке ниже перечислены метрики результирующей производительности:

yy Какова величина потребления каждого диска (нагрузка)? yy Насколько насыщен каждый диск операциями ввода/вывода (величина очереди ожидания)?

9.5. Методология  573

yy yy yy yy yy

Каково среднее время обслуживания ввода/вывода? Каково среднее время ожидания ввода/вывода? Имеются ли выбросы ввода/вывода с большой задержкой? Каково полное распределение задержки ввода/вывода? Имеются ли и активны ли системные средства управления ресурсами, в частности ресурсами ввода/вывода?

yy Какова задержка выполнения дисковых команд, не связанных с передачей данных?

Трассировка событий Инструменты трассировки можно использовать для записи деталей всех операций с файловой системой в журнал для последующего анализа (см., например, раздел 9.6.7 «biosnoop»). В число деталей могут входить: идентификатор дискового устройства, тип операции ввода/вывода или команды, смещение, размер, отметки времени начала и окончания, код завершения, а также идентификатор и имя процесса (если возможно). На основе отметок времени начала и окончания можно вычислять задержку ввода/вывода (или записывать ее в журнал напрямую). Изу­ чая последовательность отметок времени начала и окончания обработки запроса, можно идентифицировать их переупорядочивание устройством. Трассировка может быть идеальным подходом к определению характеристик рабочей нагрузки, но на практике она часто сопряжена со значительным оверхедом из-за высокой частоты следования дисковых операций. Если запись данных трассировки на диск событий происходит в период трассировки, они могут не только загрязнять данные для анализа, но также создать петлю обратной связи и проблемы с производительностью.

9.5.5. Анализ задержек Анализ задержек предполагает более глубокое изучение системы для поиска источника задержек. В случае с дисками исследование часто заканчивается на интерфейсе диска, где измеряется время между передачей запроса на ввод/вывод и прерыванием, уведомляющим о его выполнении. Если это время соответствует задержке ввода/вывода на уровне приложения, то можно с уверенностью предположить, что задержка ввода/вывода обусловлена особенностями работы дисков, и сосредоточить внимание на них. Если задержка отличается, ее измерение на разных уровнях стека ОС позволит определить источник. На рис. 9.10 показан общий стек ввода/вывода с задержками на разных уровнях, соответствующих двум выбросам, A и B. Задержка A одинакова на всех уровнях от приложения до драйверов диска. Это указывает на то, что ее причиной являются диски (или драйверы дисков). Об этом нетрудно догадаться по схожим задержкам на разных уровнях, измеряемым независимо.

574  Глава 9. Диски Задержка B, по всей видимости, возникает на уровне файловой системы (блокировка или ожидание в очереди), потому что уровни ниже вносят гораздо меньший вклад. Имейте в виду, что на разных уровнях стека количество операций ввода/вывода может уменьшаться или увеличиваться, а это означает, что объем ввода/вывода, количество операций и  величина задержки будут отличаться на разных уровнях. Пример с задержкой B может соответствовать ситуации, когда на нижних уровнях наблюдается одна задержка (10 мс), но не учитываются другие связанные операции ввода/вывода, которые обусловлены обслуживанием того же самого ввода/вывода в файловой системе (например, для обновления метаданных). Задержки на каждом уровне можно представить в виде:

yy средних значений за интервал: обычно сообщаются инструментами ОС; yy полных распределений: в виде гистограмм или тепловых карт, см. раздел 9.7.3 «Тепловые карты задержек»;

yy значений задержек: см. предыдущий подраздел «Трассировка событий». Последние два представления помогают определить источник выбросов и выявить случаи, когда запросы на ввод/вывод разделялись или объединялись. Приложение

мс

мс

мс

мс

мс

мс

мс

мс

Системные вызовы VFS Файловая система Интерфейс блочного устройства Драйверы

мс

мс

мс

мс

Задержка ввода/вывода A

Задержка ввода/вывода B

Диски

Рис. 9.10. Анализ задержек в стеке ввода/вывода

9.5.6. Статическая настройка производительности Статическая настройка производительности фокусируется на проблемах сконфигурированной архитектуры. Анализируя производительность дисков, изучите следующие аспекты статической конфигурации:

9.5. Методология  575

yy yy yy yy yy yy

Сколько имеется дисков? Каких типов (например, SMR, MLC)? Какой емкости?

yy yy yy yy yy

Версия прошивки контроллера диска?

Версия прошивки диска? Сколько имеется контроллеров дисков? Какие типы интерфейсов они имеют? Подключены ли контроллеры дисков к высокоскоростным слотам? Сколько дисков подключено к каждому контроллеру? Имеются ли резервные батареи для питания дисков/контроллеров и какой мощности? Настроен ли RAID? Как именно, включая ширину полосы? Доступен и настроен ли режим многоканального доступа? Версия драйвера дискового устройства? Каков объем основной памяти сервера? Какой объем используется для кэшей страниц и буферов?

yy Есть ли известные ошибки и исправления для каких-либо драйверов устройств хранения?

yy Используются ли средства управления ресурсами дискового ввода/вывода? Имейте в виду, что в драйверах устройств и прошивке могут быть ошибки, влияющие на производительность, которые в идеале исправляются обновлениями от поставщика. Эти ответы помогут выявить ранее упущенные из виду варианты конфигурации. Иногда бывает так, что систему настроили для одной рабочей нагрузки, а затем перепрофилировали для другой. Этот метод напомнит о необходимости вернуться к выбору параметров. Когда я работал ведущим специалистом по производительности хранилищ на ZFS в Sun, то наиболее частые жалобы на производительность, которые я получал, были вызваны неправильной конфигурацией: использованием половины JBOD (12 дисков) в RAID-Z2 (с широкими полосами). Эта конфигурация обеспечивает высокую надежность, но уступает по производительности отдельным дискам. Я научился сначала запрашивать у клиентов детали конфигурации и только потом входить к ним в систему и тратить время на изучение задержек ввода/вывода.

9.5.7. Настройка кэширования В системе может быть много разных кэшей, включая кэши в приложениях, в файловых системах, в контроллерах дисков и в самих дисках. Они были перечислены в разделе 9.3.3 «Кэширование». Их можно настроить, как описано в главе 2 «Методологии», в разделе 2.5.18 «Настройка кэширования». Поэтому проверьте, какие кэши есть, убедитесь, что они работают, и проверьте, насколько хорошо они работают, а затем настройте рабочую нагрузку для кэша и настройте кэш для рабочей нагрузки.

576  Глава 9. Диски

9.5.8. Управление ресурсами Операционная система может предоставлять средства управления для распределения ресурсов дискового ввода/вывода между процессами или группами процессов, позволяющие устанавливать фиксированные ограничения по частоте операций ввода/вывода и пропускной способности или ее доли. Как это работает, зависит от реализации и обсуждается в разделе 9.9 «Настройка».

9.5.9. Микробенчмаркинг Микробенчмаркинг дискового ввода/вывода был представлен в главе 8 «Файловые системы», где объяснялась разница между тестированием ввода/вывода файловой системы и тестированием дискового ввода/вывода. Под тестированием дискового ввода/вывода обычно подразумевается тестирование через пути к устройствам, которые есть в ОС. В частности, через низкоуровневый интерфейс к устройствам, если он доступен, позволяющий исключить из тестирования поведение файловой системы (включая кэширование, буферизацию, разделение и объединение ввода/ вывода, оверхед на выполнение кода и отображение смещений). Вот основные факторы, которые можно проверить:

yy направление: чтение или запись; yy характер доступа к диску: произвольный или последовательный; yy диапазон смещений: весь диск или узкие диапазоны (например, тестируется только смещение 0); yy размер ввода/вывода: от 512 байт (типичный минимум) до 1 Мбайт; yy конкуренция: количество операций ввода/вывода, выполняемых одновременно, или количество потоков, выполняющих операции ввода/вывода; yy количество устройств: один диск или несколько дисков (для изучения ограничений контроллера и шины). В подразделах ниже показано, как можно комбинировать эти факторы для исследования производительности диска и контроллера. Подробную информацию о конкретных инструментах для проведения этих тестов ищите в разделе 9.8 «Эксперименты».

Диски Микробенчмаркинг может быть выполнен для каждого диска для определения следующих метрик и предлагаемых рабочих нагрузок:

yy максимальная пропускная способность диска (мегабайт в секунду): последовательное чтение блоками по 128 Кбайт или 1 Мбайт;

yy максимальная частота операций с диском (IOPS): чтение блоками по 512 байт1, только из смещения 0;

Этот размер должен соответствовать наименьшему размеру блока на диске. Во многих современных дисках используются блоки размером 4 Кбайт.

1

9.5. Методология  577

yy максимальное количество произвольных чтений (IOPS): чтение блоками по 512 байт из случайно выбираемых смещений;

yy профиль задержки чтения (средние значения в микросекундах): многократное

последовательное чтение блоками по 512 байт, 1 Кбайт, 2 Кбайт, 4 Кбайт и т. д.;

yy профиль задержки произвольного ввода/вывода (средние значения в микро-

секундах): многократное чтение блоками по 512 байт в полном диапазоне смещений, только в области начальных смещений, только в области конечных смещений.

Эти тесты можно повторить для операций записи. Случай «только из смещения 0» позволяет протестировать кэширование данных на диске и оценить время доступа к кэшу1.

Контроллеры дисков Для микробенчмаркинга контроллеров дисков можно применить рабочую нагрузку к нескольким дискам, чтобы достичь предельных возможностей контроллера. При тестировании контроллеров можно определить следующие метрики и характер рекомендуемых рабочих нагрузок:

yy максимальная пропускная способность контроллера (мегабайт в секунду): чтение блоками по 128 Кбайт, только из смещения 0;

yy максимальная частота выполнения операций контроллером (IOPS): чтение блоками по 512 байт, только из смещения 0.

Примените рабочую нагрузку, последовательно увеличивая количество охватываемых дисков, и следите за ограничениями. Чтобы найти предел в контроллере дисков, может потребоваться приложить нагрузку к десятку дисков. Этот размер должен соответствовать наименьшему размеру блока на диске. Во многих современных дисках используются блоки размером 4 Кбайт.

9.5.10. Масштабирование Диски и контроллеры дисков имеют ограничения по пропускной способности и количеству операций ввода/вывода в секунду (IOPS), что можно показать с помощью микробенчмаркинга, как было описано выше. Настройка может улучшить производительность только до этих пределов. Если требуется повышенная производительность диска, а кэширование, например, не дает желаемого результата, то дисковую подсистему нужно масштабировать.

Поговаривают, что некоторые производители накопителей оптимизируют прошивки для ускорения операций ввода/вывода с сектором 0, что завышает результаты такого теста. Это можно проверить, сравнив производительность операций с сектором 0 и производительность операций с любым другим сектором по вашему выбору.

1

578  Глава 9. Диски Вот простой метод, основанный на планировании емкости ресурсов: 1. Определите целевую рабочую нагрузку в терминах пропускной способности и количества операций ввода/вывода в секунду. Если это новая система, то см. главу 2 «Методологии», раздел 2.7 «Планирование мощности». Если система уже имеет рабочую нагрузку, выразите количество пользователей в терминах текущей дисковой пропускной способности и IOPS и масштабируйте эти числа для целевой группы пользователей. (Если параллельно нельзя масштабировать кэш, то рабочая нагрузка на диск может увеличиться, потому что уменьшается доля объема кэша, приходящаяся на одного пользователя.) 2. Рассчитайте количество дисков, необходимых для поддержки этой рабочей нагрузки. Учтите фактор конфигурации RAID. Не используйте в расчетах максимальную пропускную способность и IOPS на диск, так как это приведет к 100 %-ной нагрузке на диски, что тут же повлечет проблемы с производительностью из-за насыщения и увеличения очередей. Выберите целевое потребление (скажем, 50 %) и соответственно масштабируйте значения. 3. Рассчитайте количество контроллеров дисков, которое нужно для поддержки этой рабочей нагрузки. 4. Убедитесь, что не превышены пределы возможностей транспортов, и при необходимости масштабируйте транспорты. 5. Рассчитайте количество тактов процессора, расходуемых на дисковый ввод/ вывод и нужное количество процессоров (для этого может потребоваться несколько процессоров и организация параллельного ввода/вывода). Максимальная пропускная способность каждого диска и величина IOPS будут зависеть от типа ввода/вывода и типов дисков. См. раздел 9.3.7 «Несопоставимость количества операций в секунду». Микробенчмаркинг можно использовать для определения конкретных ограничений для заданного размера и типа ввода/вывода, а определение характеристик рабочей нагрузки — на существующих рабочих нагрузках, чтобы увидеть, какие размеры и типы имеют значение. Для удовлетворения требований, предъявляемых рабочей нагрузкой, нередко могут понадобиться серверы с десятками дисков, объединенных в массивы хранения. Раньше мы говорили: «Добавляйте больше шпинделей». Теперь же говорим: «Добавляйте больше флеш-памяти».

9.6. ИНСТРУМЕНТЫ НАБЛЮДЕНИЯ В этом разделе представлены инструменты наблюдения за работой дисков, доступные в ОС на базе Linux. Стратегии их использования описаны в предыдущем разделе. Инструменты перечислены в табл. 9.5.

9.6. Инструменты наблюдения  579 Таблица 9.5. Инструменты наблюдения за работой дисков Раздел

Инструмент

Описание

9.6.1

iostat

Различные статистики, характеризующие дисковый ввод/вывод

9.6.2

sar

Исторические статистики

9.6.3

PSI

Информация о времени, затраченном на ожидание доступности диска

9.6.4

pidstat

Разбивка потребления дисков по процессам

9.6.5

perf

Точки трассировки для анализа производительности блочного ввода/ вывода

9.6.6

biolatency

Распределение задержек дискового ввода/вывода в виде гистограммы

9.6.7

biosnoop

Трассировка дискового ввода/вывода, генерируемого процессами, с выводом информации о задержках

9.6.8

iotop, biotop

Аналоги команды top для дисков: обобщают характеристики блочного ввода/вывода по процессам

9.6.9

biostacks

Отображает дисковый ввод/вывод со стеками инициализации

9.6.10

blktrace

Трассировка событий дискового ввода/вывода

9.6.11

bpftrace

Настраиваемая трассировка дискового ввода/вывода

9.6.12

MegaCli

Инструмент компании LSI, выводит статистики работы контроллера

9.6.13

smartctl

Статистики работы контроллера

Описание этого набора инструментов и возможностей дополняет раздел 9.5 «Методология». Он начинается с описания традиционных инструментов, затем переходит к инструментам трассировки и, наконец, к инструментам для получения статистик работы контроллера диска. Некоторые из традиционных инструментов, вероятно, доступны (а иногда первоначально созданы) в других Unix-подобных ОС, в том числе iostat(8) и sar(1). Многие из инструментов трассировки основаны на BPF и используют интерфейсы BCC и bpftrace (глава 15), в том числе biolatency(8), biosnoop(8), biotop(8) и biostacks(8). Полный перечень возможностей каждого инструмента вы найдете в соответствующей документации, включая страницы справочного руководства man.

9.6.1. iostat iostat(1) выводит сводную статистику ввода/вывода для каждого диска, сообщая метрики, характеризующие рабочую нагрузку, потребление и насыщение. Эта коман­да может запускаться рядовыми пользователями и обычно используется при исследовании проблем дискового ввода/вывода. Статистики, возвращаемые этим инструментом, также можно получить с помощью ПО для мониторинга, поэтому полезно поближе познакомиться с iostat(1), чтобы углубить понимание статистик

580  Глава 9. Диски мониторинга. Кроме того, предоставляемые статистики по умолчанию собираются ядром1, поэтому iostat(1) не оказывает большой нагрузки на систему. Название «iostat» — это сокращение от «I/O statistics» (статистики ввода/вывода). Но правильнее, пожалуй, было бы назвать этот инструмент «diskiostat», чтобы подчеркнуть тип ввода/вывода, характеристики которого он сообщает. Иногда такое название могло запутать, когда пользователь знал, что приложение выполняет ввод/ вывод (в файловую систему), но не видел его через iostat(1) (диски). Утилита iostat(1) была написана в начале 1980-х для Unix, и сейчас доступны разные ее версии для разных ОС. Ее можно добавить в системы на базе Linux, установив пакет sysstat. Ниже описана версия для Linux.

Вывод iostat по умолчанию После запуска без аргументов и параметров команда выводит сводку о потреблении процессоров и дисков, накопленную с момента загрузки. Ниже приведен пример вывода по умолчанию для первичного знакомства с инструментом. Но чаще приходится использовать расширенный режим, описанный ниже, который дает больше полезной информации. $ iostat Linux 5.3.0-1010-aws (ip-10-1-239-218) avg-cpu:

%user 0.29

Device loop0 [...] nvme0n1

02/12/20

%nice %system %iowait 0.01 0.18 0.03

%steal 0.21

_x86_64_

(2 CPU)

%idle 99.28

tps 0.00

kB_read/s 0.05

kB_wrtn/s 0.00

kB_read 1232

kB_wrtn 0

3.40

17.11

36.03

409902

863344

В первой строке дана общая информация о системе, включая версию ядра, имя хоста, дату, архитектуру и количество процессоров. В следующих строках выводятся статистики, накопленные с момента загрузки системы, в том числе среднее потребление процессора (avg-cpu, эта статистика была описана в главе 6 «Процессоры») и дисковых устройств (в разделе Device:). Каждому дисковому устройству соответствует отдельная строка с основными сведениями в столбцах. Заголовки столбцов в листинге выделены жирным. Среди них:

yy tps: количество транзакций в секунду (IOPS); yy kB_read/s, kB_wrtn/s: объемы чтения и записи в килобайтах в секунду соответственно;

yy kB_read, kB_wrtn: суммарный объем прочитанных и записанных данных в килобайтах.

Сбор статистик можно отключить через файл /sys/ block//queue/iostats, но я еще не встречал никого, кто сделал бы это.

1

9.6. Инструменты наблюдения  581 Некоторые устройства SCSI, включая CD-ROM, могут не отображаться командой iostat(1). Информацию о ленточных накопителях SCSI можно получить с помощью утилиты tapestat(1) из того же пакета sysstat. Также обратите внимание, что iostat(1) сообщает информацию об операциях чтения и записи с блочными устройствами, но может исключать некоторые другие типы команд, посылаемых дисковым устройствам, в зависимости от ядра (см. реализацию функции ядра blk_do_io_stat()). При использовании в расширенном режиме iostat(1) выводит дополнительные поля с информацией об этих командах.

Параметры iostat Команде iostat(1) можно передавать различные параметры, за которыми следуют необязательные значения интервала и счетчик. Например: # iostat 1 10

10 раз выведет статистики с интервалом в 1 с. А: # iostat 1

будет выводить статистики бесконечно (до нажатия Ctrl-C) с интервалом в 1 с. Вот некоторые наиболее часто используемые параметры:

yy yy yy yy yy yy yy yy yy

-c: выводит отчет о потреблении процессора; -d: выводит отчет о потреблении дисков; -k: выводит объемы в килобайтах вместо блоков (по 512 байт); -m: выводит объемы в мегабайтах вместо блоков (по 512 байт); -p: добавляет статистики по разделам; -t: выводит отметки времени; -x: выводит расширенные статистики; -s: режим узкого вывода; -z: пропускает вывод информации об устройствах с нулевой активностью.

Есть переменная среды POSIXLY_CORRECT=1 для вывода объемов в блоках (по 512 байт каждый) вместо килобайтов. Некоторые старые версии поддерживали параметр -n для получения статистик о NFS. Начиная с версии sysstat 9.1.3, эта возможность была реализована в виде команды nfsiostat.

Режим расширенного узкого вывода iostat В режиме расширенного вывода (-x) iostat(1) выводит дополнительные столбцы, которые пригодятся при выполнении описанных выше действий. В этих дополнительных столбцах выводятся: метрики IOPS и пропускной способности для определения характеристик рабочей нагрузки, потребление дисков и размеры очередей

582  Глава 9. Диски для метода USE, а также время отклика диска для определения характеристик производительности и анализа задержек. Со временем в расширенный вывод добавлялось все больше полей, и в последней версии (12.3.1, декабрь 2019-го) вывод был шириной 197 символов. Он не умещается не только по ширине книжной страницы, но и во многих терминалах, что затрудняет чтение вывода из-за переноса строк. В 2017 году было добавлено решение, параметр -s, обеспечивающий «узкий» вывод и предназначенный для отображения в терминалах с шириной строк в 80 символов. Вот пример узкого (-s ) и расширенного (-x ) вывода статистик с пропуском устройств с нулевой активностью (-z): $ iostat -sxz 1 [...] avg-cpu: %user 15.82 Device nvme0n1 [...]

%nice %system %iowait 0.00 10.71 31.63 tps 1642.00

kB/s 9064.00

%steal 1.53

rqm/s 664.00

%idle 40.31

await aqu-sz 0.44 0.00

areq-sz 5.52

%util 100.00

Значение столбцов:

yy yy yy yy

tps: количество транзакций в секунду (IOPS); kB/s: килобайт в секунду; rqm/s: количество запросов в секунду, добавленных в очередь и объединенных; await: среднее время отклика ввода/вывода, включая время ожидания в очереди

драйвера и время отклика устройства ввода/вывода (миллисекунд);

yy aqu-sz: среднее количество запросов, ожидающих в очереди запросов драйвера и обрабатываемых устройством;

yy areq-sz: средний размер запроса в килобайтах; yy %util: процент времени, в течение которого устройство было занято обработкой запросов ввода/вывода (потребление).

Наиболее важная метрика производительности — это await, показывающая общее среднее время ожидания ввода/вывода. Какое значение считать «хорошим» или «плохим», зависит от ваших потребностей. Как показано в примере вывода, среднее время ожидания (await) составило 0,44 мс, что вполне удовлетворительно для этого сервера базы данных. Это время может увеличиваться по разным причинам: из-за ожидания в очередях (из-за высокой нагрузки), больших размеров ввода/вывода, произвольного ввода/вывода на вращающихся дисках и ошибок устройств. Метрика %util важна для планирования мощности и потребления ресурсов, но учтите, что это всего лишь оценка потребления (доля времени, в течение которого устройство было занято обработкой запросов) и она может почти ничего не значить для виртуальных устройств, охватывающих несколько дисков. Такие устройства

9.6. Инструменты наблюдения  583 лучше исследовать по значению приложенной нагрузки: tps (IOPS) и  kB/s (пропускная способность). Ненулевые значения в столбце rqm/s показывают, что последовательные запросы объединялись перед отправкой на устройство для повышения производительности. Этот показатель также является признаком последовательной нагрузки. Поскольку величина areq-sz измеряется после слияния запросов, небольшие значения (8 Кбайт или меньше) могут служить признаком рабочей нагрузки произвольного ввода/вывода, запросы которой не удалось объединить. Большие размеры могут соответствовать либо большим операциям ввода/вывода, либо объединенным запросам рабочей нагрузки последовательного ввода/вывода (как в примере выше).

Режим расширенного вывода iostat Без параметра -s команда iostat(1) с параметром -x выводит намного больше столбцов. Вот сводная информация, накопленная с момента загрузки системы (однократный вывод), которую выводит iostat(1) из пакета sysstat версии 12.3.2 (апрель 2020-го): $ iostat -x [...] Device r/s rkB/s rrqm/s wrqm/s %wrqm w_await wareq-sz d/s f/s f_await aqu-sz %util nvme0n1 0.23 9.91 0.16 0.92 22.91 0.89 10.66 0.00 0.00 0.00 0.00 0.12

%rrqm r_await rareq-sz w/s wkB/s dkB/s drqm/s %drqm d_await dareq-sz 40.70 0.00

0.56 0.00

43.01 0.00

3.10 0.00

33.09 0.00

Многие метрики, доступные с параметрами -sx, разбиты на компоненты, соответствующие операциям чтения, записи, синхронизации (flush) и усечения (discard). Вот краткое описание дополнительных столбцов:

yy r/s, w/s, d/s, f/s: запросов на чтение, запись, усечение и синхронизацию, выполняемых дисковым устройством в секунду (после слияния);

yy rkB/s, wkB/s, dkB/s: объем чтения, записи и усечения в килобайтах в секунду; yy %rrqm/s, %wrqm/s, %drqm/s: количество запросов в секунду на чтение, запись и усечение, добавленных в очередь и объединенных, в процентах от общего количества запросов каждого типа;

yy r_await, w_await, d_await, f_await: среднее время отклика для чтения, записи, усечения и синхронизации, включая время ожидания в очереди драйвера и время отклика устройства (миллисекунд);

yy reduq-sz, wareq-sz, dareq-sz: средний размер чтения, записи и усечения (килобайт).

Возможность исследовать операции чтения и записи по отдельности очень важна. Приложения и файловые системы обычно используют приемы уменьшения

584  Глава 9. Диски задержки записи (например, кэширование с отложенной записью) и тем самым уменьшают вероятность блокировки приложения на время выполнения записи на диск. Это означает, что любые метрики, суммирующие операции чтения и записи, искажаются компонентом, который может прямо не влиять на производительность (операции записи). Благодаря разделению вы можете начать изучение метрики r_wait, которая показывает среднюю задержку чтения и, вероятно, является наиболее важным показателем производительности приложения. Метрики, оценивающие производительность чтения и записи в виде количества операций ввода/вывода в секунду (r/s, w/s) и пропускной способности (rkB/s, wkB/s), можно использовать для определения характеристик рабочей нагрузки. Поддержка вывода статистик отмены и синхронизации появилась в iostat(1) недавно. Операции усечения (discard) освобождают блоки на диске (команда ATA TRIM), а соответствующая статистика была добавлена в ядро Linux 4.19. Статистика для операций синхронизации (flush) была добавлена в Linux 5.5. Их изучение помогает сузить поле поиска причин задержки диска. Вот еще одна полезная комбинация параметров iostat(1): $ iostat -dmstxz -p ALL 1 Linux 5.3.0-1010-aws (ip-10-1-239-218) 02/12/20 17:39:29 Device tps nvme0n1 3.33 nvme0n1p1 3.31 02/12/20 17:39:30 Device tps nvme0n1 1730.00 nvme0n1p1 1538.00 [...]

02/12/20

_x86_64_

(2 CPU)

MB/s 0.04 0.04

rqm/s 1.09 1.09

await 0.87 0.87

areq-sz 12.84 12.91

aqu-sz 0.00 0.00

%util 0.12 0.12

MB/s 14.97 14.97

rqm/s 709.00 709.00

await 0.54 0.61

areq-sz 8.86 9.97

aqu-sz 0.02 0.02

%util 99.60 99.60

Первый вывод — это сводная информация о системе, за которой следуют блоки статистик, выводящиеся с интервалом в одну секунду. Параметр -d требует выводить только статистики потребления диска (без статистик процессора), -m — значения объемов в мегабайтах и -t — значения отметок времени, что может пригодиться при сравнении результатов из других источников с отметками времени, а параметр -p ALL включает вывод статистик по разделам. К сожалению, текущая версия iostat(1) не выводит информацию об ошибках диска. В противном случае все метрики метода USE можно было бы получить с помощью одного инструмента.

9.6.2. sar Генератор отчетов о работе системы (system activity reporter) sar(1) можно использовать для наблюдения за текущей активностью или настроить для архивирования статистик и генерирования хронологических отчетов. Он был представлен в главе 4

9.6. Инструменты наблюдения  585 «Инструменты наблюдения», в разделе 4.4 «sar» и упоминается в других главах, где используется для получения соответствующих статистик. При вызове с параметром -d инструмент sar(1) выводит сводную информацию о потреблении диска с интервалом в одну секунду, как показано в следующих примерах. Вывод довольно широкий, поэтому здесь я разбил его на две части (sysstat 12.3.2): $ sar -d 1 Linux 5.3.0-1010-aws (ip-10-0-239-218) 02/13/20 09:10:22 DEV tps rkB/s wkB/s 09:10:23 dev259-0 1509.00 11100.00 12776.00 [...]

_x86_64_ (2 CPU) dkB/s areq-sz \ ... 0.00 15.82 / ...

и остальные столбцы: $ sar -d 1 09:10:22 09:10:23 [...]

\ ... \ / ... /

aqu-sz 0.02

await 0.60

%util 94.00

Эти столбцы есть в выводе iostat(1) с параметром -x, они были описаны в предыдущем разделе. В этом выводе мы видим смешанную рабочую нагрузку чтения/записи со значением await 0,6 мс, что обусловлено потреблением диска на уровне 94 %. Предыдущие версии sar(1) включали столбец svctm (service time — время обслуживания): среднее (расчетное) время отклика диска в миллисекундах. Информацию о времени обслуживания см. в разделе 9.3.1 «Измерение времени». Но так как упрощенный алгоритм вычисления времени обслуживания давал неверные результаты для современных дисков, поддерживающих возможность параллельного ввода/ вывода, в более поздних версиях столбец svctm удалили.

9.6.3. PSI Информация о задержках доступа в Linux (pressure stall information, PSI), добавленная в Linux 4.20, включает статистики, характеризующие насыщение ввода/ вывода. Они не только показывают наличие задержек, но и то, как они менялись за последние 5 минут. Вот пример вывода: # cat /proc/pressure/io some avg10=63.11 avg60=32.18 avg300=8.62 total=667212021 full avg10=60.76 avg60=31.13 avg300=8.35 total=622722632

Как показывает этот вывод, насыщение ввода/вывода увеличивается, потому что среднее значение за 10 с (63,11) выше среднего за 300 с (8,62). Эти средние значения представляют процент времени, в течение которого задача была заблокирована в ожидании ввода/вывода. Строка some (некоторые) обобщает случаи, когда задержкой были затронуты некоторые задачи (потоки), а строка full (все) обобщает случаи, когда задержкой были затронуты все выполняемые задачи. Так же как со средними нагрузками, эти значения могут служить высокоуровневыми метриками для рассылки оповещений. Узнав о появлении проблемы

586  Глава 9. Диски с производительностью диска, вы можете использовать другие инструменты для поиска основных причин, включая pidstat(8) для получения статистик потребления диска по процессам.

9.6.4. pidstat По умолчанию инструмент pidstat(1) в Linux выводит данные о потреблении процессора по процессам, но еще поддерживает параметр -d для вывода статистик потребления дискового ввода/вывода. Эта возможность доступна в ядрах 2.6.20 и выше. Например: $ pidstat -d 1 Linux 5.3.0-1010-aws (ip-10-0-239-218) 09:47:41 UID PID kB_rd/s 09:47:42 0 2705 32468.00 09:47:42 0 2706 0.00 [...] 09:47:56 UID PID kB_rd/s 09:47:57 0 229 0.00 09:47:57 0 380 0.00 09:47:57 0 2699 4.00 u4:1-flush-259:0 09:47:57 0 2705 15104.00 09:47:57 0 2706 0.00

02/13/20 _x86_64_ (2 CPU) kB_wr/s kB_ccwr/s iodelay Command 0.00 0.00 5 tar 8192.00 0.00 0 gzip kB_wr/s kB_ccwr/s iodelay Command 72.00 0.00 0 systemd-journal 4.00 0.00 0 auditd 0.00 0.00 10 kworker/ 0.00 6912.00

0.00 0.00

0 tar 0 gzip

Вот значение некоторых столбцов:

yy kB_rd/s: прочитано килобайт в секунду; yy kB_wd/s: записано килобайт в секунду; yy kB_ccwr/s: отменено для записи килобайт в секунду (например, при повторной записи или при удалении перед выталкиванием на диск);

yy iodelay: время, в течение которого процесс оставался заблокированным в ожидании завершения дискового ввода/вывода (в тактах часов), включая подкачку.

Роль рабочей нагрузки для этого примера играла команда tar, читающая файлы и передающая данные через конвейер команде gzip, которая, в свою очередь, читала данные из конвейера, сжимала их и записывала в файл архива. Операциям чтения в tar в столбце iodelay соответствует значение 5 (тактов), тогда операции записи в  gzip не блокировались благодаря кэшированию с отложенной записью в кэше страниц. Некоторое время спустя кэш страницы был вытолкнут на диск, о чем можно судить по задержке iodelay, возникшей в процессе kworker/u4:1-flush-259:0. iodelay — достаточно новое дополнение, помогающее оценить масштабы проблем

с производительностью: время, в течение которого приложение ожидало ввода/вывода. Остальные столбцы отражают характеристики приложенной рабочей нагрузки. Обратите внимание: чтобы получить доступ к статистикам потребления диска для процессов, запущенных от лица других пользователей, нужно обладать привилегиями суперпользователя (root). Они доступны также в /proc/PID io.

9.6. Инструменты наблюдения  587

9.6.5. perf Инструмент perf(1) (глава 13) для Linux поддерживает возможность инструментации точек трассировки блочного вывода. Вот как можно получить их список: # perf list 'block:*' List of pre-defined events (to be used in -e): block:block_bio_backmerge block:block_bio_bounce block:block_bio_complete block:block_bio_frontmerge block:block_bio_queue block:block_bio_remap block:block_dirty_buffer block:block_getrq block:block_plug block:block_rq_complete block:block_rq_insert block:block_rq_issue block:block_rq_remap block:block_rq_requeue block:block_sleeprq block:block_split block:block_touch_buffer block:block_unplug

[Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint

event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event]

Вот пример регистрации событий передачи запросов блочному устройству с трассировками стека. Команда sleep 10 задает продолжительность трассировки. # perf record -e block:block_rq_issue -a -g sleep 10 [ perf record: Woken up 22 times to write data ] [ perf record: Captured and wrote 5.701 MB perf.data (19267 samples) ] # perf script --header [...] mysqld 1965 [001] 160501.158573: block:block_rq_issue: 259,0 WS 12288 () 10329704 + 24 [mysqld] ffffffffb12d5040 blk_mq_start_request+0xa0 ([kernel.kallsyms]) ffffffffb12d5040 blk_mq_start_request+0xa0 ([kernel.kallsyms]) ffffffffb1532b4c nvme_queue_rq+0x16c ([kernel.kallsyms]) ffffffffb12d7b46 __blk_mq_try_issue_directly+0x116 ([kernel.kallsyms]) ffffffffb12d87bb blk_mq_request_issue_directly+0x4b ([kernel.kallsyms]) ffffffffb12d8896 blk_mq_try_issue_list_directly+0x46 ([kernel.kallsyms]) ffffffffb12dce7e blk_mq_sched_insert_requests+0xae ([kernel.kallsyms]) ffffffffb12d86c8 blk_mq_flush_plug_list+0x1e8 ([kernel.kallsyms]) ffffffffb12cd623 blk_flush_plug_list+0xe3 ([kernel.kallsyms]) ffffffffb12cd676 blk_finish_plug+0x26 ([kernel.kallsyms]) ffffffffb119771c ext4_writepages+0x77c ([kernel.kallsyms]) ffffffffb10209c3 do_writepages+0x43 ([kernel.kallsyms]) ffffffffb1017ed5 __filemap_fdatawrite_range+0xd5 ([kernel.kallsyms]) ffffffffb10186ca file_write_and_wait_range+0x5a ([kernel.kallsyms]) ffffffffb118637f ext4_sync_file+0x8f ([kernel.kallsyms]) ffffffffb1105869 vfs_fsync_range+0x49 ([kernel.kallsyms])

588  Глава 9. Диски ffffffffb11058fd do_fsync+0x3d ([kernel.kallsyms]) ffffffffb1105944 __x64_sys_fsync+0x14 ([kernel.kallsyms]) ffffffffb0e044ca do_syscall_64+0x5a ([kernel.kallsyms]) ffffffffb1a0008c entry_SYSCALL_64_after_hwframe+0x44 ([kernel.kallsyms]) 7f2285d1988b fsync+0x3b (/usr/lib/x86_64-linux-gnu/libpthread-2.30.so) 55ac10a05ebe Fil_shard::redo_space_flush+0x44e (/usr/sbin/mysqld) 55ac10a06179 Fil_shard::flush_file_redo+0x99 (/usr/sbin/mysqld) 55ac1076ff1c [unknown] (/usr/sbin/mysqld) 55ac10777030 log_flusher+0x520 (/usr/sbin/mysqld) 55ac10748d61 std::thread::_State_impl::_M_run+0xc1 (/usr/sbin/mysql 7f228559df74 [unknown] (/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.28) 7f226c3652c0 [unknown] ([unknown]) 55ac107499f0 std::thread::_State_impl::~_State_impl+0x0 (/usr/sbin/ 5441554156415741 [unknown] ([unknown]) [...]

Для каждого события выводится строка со сводной информацией о событии с трассировкой стека, которая привела к этому событию. Строка со сводной информацией начинается с полей, которые идут по умолчанию: имени процесса, идентификатора потока, идентификатора процессора, отметки времени и имени события (см. главу 13 «perf», раздел 13.11 «Сценарий perf»). Остальные поля относятся к точке трассировки. В частности, для точки трассировки block:block_ rq_issue выводятся:

yy yy yy yy yy yy yy

старший и младший номера диска: 259,0; тип ввода/вывода: WS (synchronous writes — синхронная запись); объем ввода/вывода: 12288 (байт); командная строка ввода/вывода: (); адрес сектора: 10329704; количество секторов: 24; процесс: mysqld.

Эти поля извлекаются из строки формата точки трассировки (см. главу 4 «Инструменты наблюдения», раздел 4.3.5 «Точки трассировки», подраздел «Аргументы и строка формата точки трассировки»). Трассировка стека помогает объяснить природу дискового ввода/вывода. Здесь инициатором является подпрограмма log_flusher () в mysqld, которая вызвала fsync(2). Путь к коду ядра показывает, что ввод/вывод обрабатывался файловой системой ext4 и превратился в проблему дискового ввода/вывода из-за вызова blk_mq_try_issue_list_directly(). Операции ввода/вывода часто ставятся в очередь, а затем выполняются потоком ядра, поэтому при наблюдении за точкой трассировки block:block_rq_issue

9.6. Инструменты наблюдения  589 информация об исходном процессе или трассировка стека в пространстве пользователя могут оказаться недоступными. В таких случаях понаблюдайте за точкой трассировки block:block_rq_insert, которая находится в операции постановки в очередь. Обратите внимание, что она не замечает ввода/вывода, который производится в обход очереди.

Однострочные сценарии Однострочные сценарии ниже показывают использование фильтров с точками трассировки block. Регистрация всех завершенных блочных операций размером не менее 100 Кбайт до нажатия комбинации Ctrl-C1: perf record -e block:block_rq_complete --filter 'nr_sector > 200'

Регистрация всех завершенных блочных операций синхронной записи до нажатия Ctrl-C: perf record -e block:block_rq_complete --filter 'rwbs == "WS"

Регистрация всех завершенных блочных операций записи любых типов до нажатия Ctrl-C: perf record -e block:block_rq_complete --filter 'rwbs ~ "*W*"'

Задержка дискового ввода/вывода Задержку дискового ввода/вывода (описанную выше как время обработки запроса диском) также можно определить путем регистрации событий отправки и завершения обработки запроса для последующего анализа. Пример ниже регистрирует эти события в течение 60 с, а затем выводит в файл out.disk01.txt: perf record -e block:block_rq_issue,block:block_rq_complete -a sleep 60 perf script --header > out.disk01.txt

Обработать выходной файл можно с помощью любого удобного вам инструмента: awk(1), Perl, Python, R, Google Spreadsheets и т. д. Нужно найти соответствующие друг другу события отправки и завершения запросов и использовать отметки времени для вычисления задержки. Следующие инструменты, biolatency(8) и biosnoop(8), эффективно вычисляют задержку дискового ввода/вывода в пространстве ядра с помощью BPF и включают величину задержки непосредственно в вывод.

При размере сектора 512 байт объему 100 Кбайт соответствует 200 секторов.

1

590  Глава 9. Диски

9.6.6. biolatency biolatency(8)1 — это инструмент для BCC и bpftrace, отображающий задержки дискового ввода/вывода в виде гистограммы. Под задержкой ввода/вывода здесь понимается время между передачей запроса устройству и завершением его выполнения (также известное как время обработки запроса диском). Ниже приведен пример использования biolatency(8) для трассировки блочного ввода/вывода в BCC в течение 10 с: # biolatency 10 1 Tracing block device I/O... Hit Ctrl-C to end. usecs 0 2 4 8 16 32 64 128 256 512 1024 2048 4096 8192 16384

-> -> -> -> -> -> -> -> -> -> -> -> -> -> ->

1 3 7 15 31 63 127 255 511 1023 2047 4095 8191 16383 32767

: : : : : : : : : : : : : : : :

count 0 0 0 0 2 0 0 1065 2462 1949 373 1815 591 397 50

distribution | | | | | | | | | | | | | | |***************** | |****************************************| |******************************* | |****** | |***************************** | |********* | |****** | | |

В этом выводе мы видим бимодальное распределение, в котором одна мода охватывает интервал от 128 до 1023 мкс, а вторая — от 2048 до 4095 мкс (от 2,0 до 4,1 мс). Теперь, зная, что распределение задержек в устройстве имеет бимодальной характер, я могу заняться настройкой, перемещающей больше операций ввода/вывода в более быстрый интервал. Например, медленно может выполняться произвольный ввод/вывод или ввод/вывод большого объема (источник которого можно идентифицировать с помощью других инструментов BPF) или операции, выполняемые с другими флагами (можно узнать с помощью параметра  -F). Самые медленные операции ввода/вывода в этом примере выполнялись от 16 до 32 мс: похоже, что на устройстве образовалась очень длинная очередь. Версия biolatency(8) для BCC поддерживает следующие параметры:

yy -m: выводить время в миллисекундах; yy -Q: включать время, проведенное в очередях ОС (время обработки запроса операционной системой);

1

Немного истории: версию biolatency для BCC я создал 20 сентября 2015 года, а версию для bpftrace — 13 сентября 2018 года, взяв за основу свой ранний инструмент iolatency. Буква «b» была добавлена, чтобы подчеркнуть, что инструмент трассирует блочный (block) ввод/ вывод.

9.6. Инструменты наблюдения  591

yy -F: показать гистограмму отдельно для каждого установленного флага ввода/ вывода;

yy -D: показать гистограмму отдельно для каждого дискового устройства. Параметр -Q заставляет biolatency(8) сообщать полное время ввода/вывода, от создания и добавления запроса в очередь ядра до завершения его обработки устройством, которое выше было описано как время обработки запроса блочного ввода/вывода. biolatency(8) для BCC также принимает необязательные аргументы со значениями интервала в секундах и счетчика.

Флаги Параметр -F особенно полезен — он позволяет разбить распределение задержек по флагам ввода/вывода. Вот пример гистограмм, полученных с дополнительным параметром -m для вывода времени в миллисекундах: # biolatency -Fm 10 1 Tracing block device I/O... Hit Ctrl-C to end. flags = Sync-Write msecs 0 -> 1

: count : 2

distribution |****************************************|

flags = Flush msecs 0 -> 1

: count : 1

distribution |****************************************|

flags = Write msecs 0 -> 2 -> 4 -> 8 -> 16 ->

: : : : : :

count 14 1 10 11 11

distribution |****************************************| |** | |**************************** | |******************************* | |******************************* |

flags = NoMerge-Write msecs 0 -> 1 2 -> 3 4 -> 7 8 -> 15 16 -> 31

: : : : : :

count 95 152 266 350 298

distribution |********** | |***************** | |****************************** | |****************************************| |********************************** |

flags = Read msecs 0 -> 1

: count : 11

distribution |****************************************|

flags = ReadAhead-Read msecs 0 -> 1 2 -> 3 4 -> 7 8 -> 15 16 -> 31

: : : : : :

distribution |****************************************| |********* | |*** | | | | |

1 3 7 15 31

count 5261 1238 481 5 2

592  Глава 9. Диски Флаги могут обрабатываться устройствами хранения по-разному. Такое разделение помогает исследовать их по отдельности. Гистограммы выше показывают, что ­запись выполнялась медленнее, чем чтение, и могут объяснить выявленное раннее бимодальное распределение. biolatency(8) обобщает информацию о задержках дискового ввода/вывода. Для исследования задержек отдельных операций ввода/вывода используйте biosnoop(8).

9.6.7. biosnoop biosnoop(8)1 — это инструмент для BCC и bpftrace, который выводит короткие однострочные сводки для каждой операции дискового ввода/вывода. Например: # biosnoop TIME(s) 0.009165000 0.009612000 0.011836000 0.012363000 0.012844000 0.016809000 0.017184000 0.017679000 0.018056000 0.018264000 0.018657000 0.018954000 0.019053000 0.019731000 0.020243000 0.020593000 [...]

COMM PID jbd2/nvme0n1p1 174 jbd2/nvme0n1p1 174 mysqld 1948 jbd2/nvme0n1p1 174 jbd2/nvme0n1p1 174 mysqld 1948 mysqld 1948 mysqld 1948 mysqld 1948 mysqld 1948 mysqld 1948 mysqld 1948 mysqld 1948 jbd2/nvme0n1p1 174 jbd2/nvme0n1p1 174 mysqld 1948

DISK nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1

T W W W W W W W W W W W W W W W R

SECTOR 2116272 2116288 10434672 2116296 2116312 10227712 10228224 10228736 10229248 10229760 10230272 10230784 10231296 2116320 2116336 4495352

BYTES 8192 4096 4096 8192 4096 262144 262144 262144 262144 262144 262144 262144 131072 8192 4096 4096

LAT(ms) 0.43 0.39 0.45 0.49 0.43 1.82 2.19 2.68 3.05 3.25 3.64 3.93 4.03 0.49 0.46 0.26

Как показывает этот вывод, основную рабочую нагрузку записи на диск nvme0n1 создает процесс mysqld с PID 174, выполняющий ввод/вывод разного объема. Вот значения столбцов:

yy yy yy yy yy yy yy 1

TIME(s): время выполнения операции в секундах; COMM: имя процесса (если известно этому инструменту); PID: идентификатор процесса (если известен этому инструменту); DISK: имя устройства хранения; T: тип: R == чтение (reads), W == запись (writes); SECTOR: адрес на диске в 512-байтных секторах; BYTES: объем ввода/вывода;

Немного истории: версию для BCC я написал 16 сентября 2015 года, а версию для bpftrace — 15 ноября 2017 года, взяв за основу свой же инструмент, созданный в 2003 году. Полная история происхождения biosnoop(8) приводится в моей книге [Gregg 19].

9.6. Инструменты наблюдения  593

yy LAT(ms): продолжительность ввода/вывода от передачи запроса устройству до завершения его обработки устройством (время отклика диска).

Примерно в середине этого примера можно видеть серию из операций записи по 262 144 байта, при выполнении которых задержка постепенно увеличивается с 1,82 до 4,03 мс. Я часто наблюдаю такую закономерность, а ее вероятную причину можно определить по другому столбцу: TIME(s). Если вычесть значение в столбце LAT(ms) из значения в столбце TIME(s), то получится время начала ввода/вывода. В данном случае все эти операции были инициированы примерно в одно и то же время. Похоже, что эта группа операций записи была запущена почти одновременно, затем они были поставлены в очередь на устройстве, а после этого выполнены по очереди, что объясняет увеличение задержки каждой последующей операции. Путем тщательного изучения времени начала и конца можно также определить переупорядочение операций на устройстве. Поскольку вывод инструмента может состоять из многих тысяч строк, я часто прибегал к помощи программного обеспечения на R для статистических расчетов, чтобы преобразовать полученный вывод в диаграмму рассеяния и упростить идентификацию таких закономерностей (см. раздел 9.7 «Визуализация»).

Анализ выбросов Ниже описан метод поиска и анализа выбросов по задержке с помощью biosnoop(8). 1. Сохраните вывод в файл: # biosnoop > out.biosnoop01.txt

2. Отсортируйте вывод по столбцу со значением задержки и выведите последние пять записей (с наибольшей задержкой): # sort -n -k 8,8 out.biosnoop01.txt | tail -5 31.344175 logger 10994 nvme0n1 W 15218056 31.344401 logger 10994 nvme0n1 W 15217544 31.344757 logger 10994 nvme0n1 W 15219080 31.345260 logger 10994 nvme0n1 W 15218568 46.059274 logger 10994 nvme0n1 W 15198896

262144 262144 262144 262144 4096

30.92 31.15 31.49 32.00 64.86

3. Откройте получившийся файл в текстовом редакторе (например, vi(1) или vim(1)): # vi out.biosnoop01.txt

4. Просмотрите выбросы от самого медленного к самому быстрому, сравнивая время в первом столбце. Самый медленный здесь составил 64,86 мс, время обработки — 46,059274 (в секундах от начала трассировки). Ищем 46.059274: [...] 45.992419 45.992988 46.059274 [...]

jbd2/nvme0n1p1 174 jbd2/nvme0n1p1 174 logger 10994

nvme0n1 W 2107232 nvme0n1 W 2107248 nvme0n1 W 15198896

8192 4096 4096

0.45 0.50 64.86

594  Глава 9. Диски Посмотрите, какие события происходили до выброса, и сравните их задержки. Если величины задержек сопоставимы, значит, задержка — это результат ожидания в очереди (аналогично линейному изменению задержек от 1,82 до 4,03 мс, наблюдаемому в первом примере вывода biosnoop(8)). Вероятно, вам удастся обнаружить какие-то другие подсказки. Здесь ситуация иная: предыдущее событие произошло примерно на 6 мс раньше и имело задержку 0,5 мс. Возможно, устройство переупорядочило запросы и первыми обработало другие. Если бы предыдущее событие завершения обработки отстояло от текущего на 64 мс в прошлом, то разрыв можно было бы объяснить другими факторами: например, эта система — экземпляр виртуальной машины и была перепланирована гипервизором во время ввода/вывода, что добавило дополнительное время к времени ввода/вывода.

Время в очереди Параметр -Q в версии biosnoop(8) для BCC можно использовать для отображения времени между созданием запроса на выполнение ввода/вывода и его передачи в устройство (выше это время было определено как время ожидания блочного ввода/ вывода, или время ожидания ОС). Большую часть этого времени запрос проводит в очередях ОС, но оно также может включать время, необходимое для выделения памяти и получения блокировок. Например: # biosnoop -Q TIME(s) COMM 0.000000 kworker/u4:0 0.000039 kworker/u4:0 0.000084 kworker/u4:0 0.000138 kworker/u4:0 0.000231 kworker/u4:0 [...]

PID 9491 9491 9491 9491 9491

DISK nvme0n1 nvme0n1 nvme0n1 nvme0n1 nvme0n1

T W W W W W

SECTOR 5726504 8128536 8128584 8128632 8128664

BYTES 4096 4096 4096 4096 4096

QUE(ms) 0.06 0.05 0.05 0.05 0.05

LAT(ms) 0.60 0.64 0.68 0.74 0.83

Время в очереди выводится в столбце QUE(ms).

9.6.8. iotop, biotop Первую версию iotop я написал в 2005 году для систем на основе Solaris [McDou­ gall 06a]. Сейчас есть множество версий, включая инструмент iotop(1) для Linux, основанный на статистиках ядра1 [Chazarain 13], и мой собственный инструмент biotop(8), основанный на BPF.

iotop Обычно iotop устанавливается в составе одноименного пакета iotop. При запуске без аргументов команда обновляет экран каждую секунду и выводит список iotop(1) требует, чтобы ядро было собрано с  параметрами CONFIG_TASK_ DELAY_ACCT, CONFIG_TASK_IO_ACCOUNTING, CONFIG_TASKSTATS и CONFIG_VM_EVENT_COUNTERS.

1

9.6. Инструменты наблюдения  595 процессов, являющихся основными потребителями дискового ввода/вывода. Для непрерывного вывода (без очистки экрана) можно использовать пакетный режим (-b). Он показан в примере ниже, где используется для вывода только процессов, осуществляющих ввод/вывод (-o), с интервалом 5 с (-d5): # iotop -bod5 Total DISK READ: TID PRIO USER 22400 be/4 root 279 be/3 root 22446 be/4 root Total DISK READ: TID PRIO USER 279 be/3 root 22446 be/4 root 646 be/4 root [...]

4.78 K/s | Total DISK DISK READ DISK WRITE 4.78 K/s 0.00 B/s 0.00 B/s 1657.27 K/s 0.00 B/s 10.16 M/s 0.00 B/s | Total DISK DISK READ DISK WRITE 0.00 B/s 9.55 M/s 0.00 B/s 10.37 M/s 0.00 B/s 272.71 B/s

WRITE: 15.04 M/s SWAPIN IO COMMAND 0.00 % 13.76 % [flush-252:0] 0.00 % 9.25 % [jbd2/vda2-8] 0.00 % 0.00 % beam.smp -K true ... WRITE: 10.75 M/s SWAPIN IO COMMAND 0.00 % 0.01 % [jbd2/vda2-8] 0.00 % 0.00 % beam.smp -K true ... 0.00 % 0.00 % rsyslogd -n -c 5

Как показывает этот вывод, процесс beam.smp (Riak) выполняет запись на диск со скоростью около 10 Мбайт/с. Вот значения столбцов:

yy DISK READ: скорость чтения в килобайтах в секунду; yy DISK WRITE: скорость записи в килобайтах в секунду; yy SWAPIN: процент времени, в течение которого поток ожидал завершения ввода/ вывода подкачки;

yy IO: процент времени, в течение которого поток ожидал завершения ввода/вывода. iotop(8) поддерживает еще несколько параметров, включая -a для вывода накопленной статистики (вместо интервальной), -p PID для отображения данных, касающихся только определенного процесса, и  -d SEC для установки интервала обновления экрана. Рекомендую проверить работу утилиты iotop(8) с известной рабочей нагрузкой и убедиться, что она отображает верные числа. Я только что попробовал сделать это у себя (iotop версии 0.6) и обнаружил, что она сильно недооценивает рабочие нагрузки записи. Также можно использовать инструмент biotop(8), который использует другой источник информации и у меня, например, выводит значения, соответствующие моей тестовой рабочей нагрузке.

biotop biotop(8) — это инструмент для BCC и еще одна версия утилиты top(1) для дисков. Вот пример вывода: # biotop Tracing... Output every 1 secs. Hit Ctrl-C to end 08:04:11 loadavg: 1.48 0.87 0.45 1/287 14547 PID

COMM

D MAJ MIN

DISK

I/O

Kbytes

AVGms

596  Глава 9. Диски 14501 6961 13855 326 1880 1873 1871 1876 [...]

cksum dd dd jbd2/xvda1-8 supervise supervise supervise supervise

R R R W W W W W

202 202 202 202 202 202 202 202

1 1 1 1 1 1 1 1

xvda1 xvda1 xvda1 xvda1 xvda1 xvda1 xvda1 xvda1

361 1628 1627 3 2 2 2 2

28832 13024 13016 168 8 8 8 8

3.39 0.59 0.59 3.00 6.71 2.51 1.57 1.22

В этом выводе можно видеть команды cksum(1) и dd(1), производящие чтение, и процессы supervise, производящие запись. Это быстрый способ определить, какие процессы и в каком объеме выполняют дисковый ввод/вывод. Вот значения столбцов:

yy PID: кэшированный идентификатор процесса (может не соответствовать истинному процессу);

yy COMM: кэшированное имя процесса (может не соответствовать истинному процессу);

yy D: направление (R == чтение (read), W == запись (write)); yy MAJ MIN: старший и младший номера дискового устройства (идентификатор в ядре);

yy yy yy yy

DISK: имя диска; I/O: количество операций дискового ввода/вывода за интервал; Kbytes: общая пропускная способность диска за интервал (килобайты); AVGms: среднее время ввода/вывода (задержка) от передачи запроса устройству

до завершения его выполнения (миллисекунды).

К моменту передачи запроса дисковому устройству процесс, пославший этот запрос, может находиться вне процессора и его идентификация может быть затруднена. biotop(8) делает все возможное, чтобы отобразить верные значения в столбцах PID и COMM, но не гарантирует абсолютной точности. biotop(8) поддерживает необязательные аргументы, определяющие интервал и счетчик (по умолчанию интервал равен одной секунде). Параметр -C предотвращает очистку экрана перед выводом очередной порции данных, а с помощью параметра -r MAXROWS можно задать максимальное количество отображаемых процессов.

9.6.9. biostacks biostacks(8)1 — это инструмент для bpftrace, который трассирует запросы блочного ввода/вывода (от постановки в очередь в ОС до завершения обработки устройством) и выводит соответствующие трассировки стека. Например:

Немного истории: я создал этот инструмент 19 марта 2019 года [Gregg 19].

1

9.6. Инструменты наблюдения  597 # biostacks.bt Attaching 5 probes... Tracing block I/O with init stacks. Hit Ctrl-C to end. ^C [...] @usecs[ blk_account_io_start+1 blk_mq_make_request+1069 generic_make_request+292 submit_bio+115 submit_bh_wbc+384 ll_rw_block+173 ext4_bread+102 __ext4_read_dirblock+52 ext4_dx_find_entry+145 ext4_find_entry+365 ext4_lookup+129 lookup_slow+171 walk_component+451 path_lookupat+132 filename_lookup+182 user_path_at_empty+54 sys_access+175 do_syscall_64+115 entry_SYSCALL_64_after_hwframe+61 ]: [2K, 4K) 2 |@@ | [4K, 8K) 37 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| [8K, 16K) 15 |@@@@@@@@@@@@@@@@@@@@@ | [16K, 32K) 9 |@@@@@@@@@@@@ | [32K, 64K) 1 |@ |

В выводе отображается гистограмма распределения задержек (в микросекундах) дискового ввода/вывода, а также стек, инициировавший ввод/вывод через системный вызов access(2), filename_lookup() и ext4_lookup(). Этот ввод/вывод обусловлен поиском путей при проверке прав доступа к файлам. В выводе было много таких стеков, которые показывают, что дисковый ввод/вывод вызван действиями, отличными от чтения и записи. Я видел случаи, когда происходил загадочный дисковый ввод/вывод, не инициированный никаким выполняющимся приложением. Причина была в выполнении фоновых задач файловой системы. (В одном случае это был фоновый скруббер ZFS, периодически проверяющий контрольные суммы.) biostacks(8) способен определить истинную причину дискового ввода/вывода, отображая трассировку стека ядра.

9.6.10. blktrace blktrace(8) — это инструмент для трассировки событий блочного ввода/вывода в Linux, который использует трассировщик blktrace в ядре. Это специализированный трассировщик, управляемый через системные вызовы BLKTRACE ioctl(2)

598  Глава 9. Диски к файлам дисковых устройств. К инструментам, составляющим внешний интерфейс, относятся blktrace(8), blkparse(1) и btrace(8). blktrace(8) запускает трассировку драйвера блочного устройства в ядре и возвращает данные трассировки, которые можно обработать с помощью blkparse(1). Для удобства был также создан инструмент btrace(8), который запускает blktrace(8) и blkparse(1), то есть следующие команды эквивалентны: # blktrace -d /dev/sda -o - | blkparse -i # btrace /dev/sda

blktrace(8) — низкоуровневый инструмент, отображающий множество событий для каждой операции ввода/вывода.

Вывод по умолчанию Ниже дан пример вывода btrace(8) по умолчанию для одной операции чтения с диска командой cksum(1): # btrace 8,16 8,16 8,16 8,16 8,16 8,16 8,16 8,16 [...]

/dev/sdb 3 3 3 3 3 3 3 1

1 2 3 4 5 6 7 1

0.429604145 0.429604569 0.429606014 0.429607624 0.429608804 0.429610501 0.429611912 0.440227144

20442 20442 20442 20442 20442 20442 20442 0

A Q G P I U D C

R R R N R N R R

184773879 184773879 184773879 [cksum] 184773879 [cksum] 1 184773879 184773879

+ 8 rwbs, то результаты будут дополнительно группироваться по типу ввода/вывода: # bpftrace -e 't:block:block_rq_insert /args->bytes/ { @[comm, args->rwbs] = hist(args->bytes); }' Attaching 1 probe... ^C [...] @[dmcrypt_write, WS]: [4K, 8K) [8K, 16K) [16K, 32K) [32K, 64K) [64K, 128K) [128K, 256K) @[dmcrypt_write, W]: [512K, 1M) [1M, 2M)

4 1 0 1 1 1

|@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@@@@@@ | | | |@@@@@@@@@@@@@ | |@@@@@@@@@@@@@ | |@@@@@@@@@@@@@ |

8 |@@@@@@@@@@ | 38 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@|

Теперь результаты разделены по типам операций: «W» (запись), «WS» (синхронная запись) и т. д. См. выше подраздел «Описание RWBS», где объясняются буквенные обозначения типов.

Задержка дискового ввода/вывода Время отклика диска, которое часто называют задержкой дискового ввода/вывода, можно измерить, инструментировав события передачи запроса устройству и завершения его обработки. Именно это и делает инструмент biolatency.bt, сообщая задержки дискового ввода/вывода в виде гистограммы. Например:

604  Глава 9. Диски # biolatency.bt Attaching 4 probes... Tracing block device I/O... Hit Ctrl-C to end. ^C @usecs: [32, 64) [64, 128) [128, 256) [256, 512) [512, 1K) [1K, 2K) [2K, 4K) [4K, 8K) [8K, 16K) [16K, 32K)

2 1 1 27 43 54 41 47 16 4

|@ | | | | | |@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@ | |@@@ |

Как показывают эти результаты, в большинстве случаев на выполнение ввода/ вывода уходило от 256 мкс до 16 мс (16K мкс). Вот исходный код инструмента: #!/usr/local/bin/bpftrace BEGIN { printf("Tracing block device I/O... Hit Ctrl-C to end.\n"); } tracepoint:block:block_rq_issue { }

@start[args->dev, args->sector] = nsecs;

tracepoint:block:block_rq_complete /@start[args->dev, args->sector]/ { @usecs = hist((nsecs - @start[args->dev, args->sector]) / 1000); delete(@start[args->dev, args->sector]); } END { }

clear(@start);

Измерение задержки требует сохранить время начала каждого ввода/вывода, а затем извлечь ее, когда ввод/вывод будет завершен, чтобы вычислить прошедшее время. При измерении задержки в VFS (в главе 8 «Файловые системы», в разделе 8.6.15 «bpftrace») время начала сохранялось в карте BPF с идентификатором потока в роли ключа. Это было оправданно, потому что поток с тем идентификатором будет выполняться на процессоре в моменты начала и окончания обработки запроса. С дисковым вводом/выводом ситуация иная, потому что к моменту появления события завершения на процессоре может выполняться другой поток. По этой причине в роли уникального идентификатора в biolatency.bt была выбрана комбинация из

9.6. Инструменты наблюдения  605 номеров устройства и сектора: предполагается, что только одна операция ввода/ вывода будет ссылаться на этот сектор в данный момент времени. По аналогии с однострочным сценарием для анализа размера ввода/вывода можно добавить в карту ключ args->rwbs, чтобы дополнительно сгруппировать время задержки по типам ввода/вывода.

Ошибки дискового ввода/вывода Код ошибки ввода/вывода доступен как аргумент точки трассировки block:block_ rq_complete, и следующий инструмент bioerr(8)1 использует его для вывода информации об операциях ввода/вывода, в которых возникла ошибка (однострочная версия этого инструмента была представлена выше): #!/usr/local/bin/bpftrace BEGIN { printf("Tracing block I/O errors. Hit Ctrl-C to end.\n"); } tracepoint:block:block_rq_complete /args->error != 0/ { time("%H:%M:%S "); printf("device: %d,%d, sector: %d, bytes: %d, flags: %s, error: %d\n", args->dev >> 20, args->dev & ((1 sector, args->nr_sector * 512, args->rwbs, args->error); }

Найти дополнительную информацию об ошибке дискового ввода/вывода помогут более низкоуровневые инструменты, например MegaCli, smartctl и журналирование SCSI, которые описываются ниже.

9.6.12. MegaCli Контроллеры дисков (адаптеры главной шины) состоят из аппаратного и программного обеспечения, внешних по отношению к системе. Инструменты анализа ОС и даже динамические трассировщики не могут напрямую наблюдать за их работой изнутри. Но иногда об особенностях этой работы можно судить, внимательно анализируя результаты реакции контроллера диска на серию операций ввода/вывода (в том числе полученные методом статической или динамической инструментации ядра). Но сейчас появились инструменты для анализа конкретных контроллеров дисков, например MegaCli от LSI. Ниже показаны события контроллера: # MegaCli -AdpEventLog -GetLatest 50 -f lsi.log -aALL # more lsi.log

Немного истории: я создал его 19 марта 2019 года для своей книги [Gregg 19].

1

606  Глава 9. Диски seqNum: 0x0000282f Time: Sat Jun 16 05:55:05 2012 Code: 0x00000023 Class: 0 Locale: 0x20 Event Description: Patrol Read complete Event Data: =========== None seqNum: 0x000027ec Time: Sat Jun 16 03:00:00 2012 Code: 0x00000027 Class: 0 Locale: 0x20 Event Description: Patrol Read started [...]

Последние два события показывают, что между 3:00 и 5:55 утра произошло Patrol Read (влияющее на производительность). Patrol Read упоминалось в разделе 9.4.3 «Типы хранилищ». Контроллер производит Patrol Read блоков на диске с целью проверки их контрольных сумм. MegaCli имеет множество других параметров для отображения информации об адаптере, о дисковом устройстве, о виртуальном устройстве, о среде, о состоянии батареи и о физических ошибках. Это помогает выявить проблемы в конфигурации и ошибки. Но даже при наличии этой информации бывает трудно выявить некоторые проблемы, например, почему именно тот или иной ввод/вывод продолжался сотни миллисекунд. Чтобы узнать, какой интерфейс, если он есть, используется для анализа контроллера диска, обращайтесь к документации производителя.

9.6.13. smartctl Дисковое устройство выполняет определенную программную логику, управляя своей работой, включая организацию очереди, кэширование и обработку ошибок. Как и в случае с контроллерами дисков, внутреннее поведение дискового устройство недоступно для наблюдения со стороны ОС, но о нем обычно можно судить по результатам наблюдения за запросами ввода/вывода и их задержкой. Многие современные диски предоставляют данные SMART (Self-Monitoring, Analysis and Reporting Technology — технология самоконтроля, анализа и отчетности) с разнообразными статистиками, характеризующими состояние устройства. Следующий вывод команды smartctl(8) в Linux показывает, какие данные доступны (получен обращением к первому диску в виртуальном RAID с использованием параметров -d megaraid, 0): # smartctl --all -d megaraid,0 /dev/sdb smartctl 5.40 2010-03-16 r3077 [x86_64-unknown-linux-gnu] (local build) Copyright (C) 2002-10 by Bruce Allen, http://smartmontools.sourceforge.net

9.6. Инструменты наблюдения  607 Device: SEAGATE ST3600002SS Version: ER62 Serial number: 3SS0LM01 Device type: disk Transport protocol: SAS Local Time is: Sun Jun 17 10:11:31 2012 UTC Device supports SMART and is Enabled Temperature Warning Disabled or Not Supported SMART Health Status: OK Current Drive Temperature: 23 C Drive Trip Temperature: 68 C Elements in grown defect list: 0 Vendor (Seagate) cache information Blocks sent to initiator = 3172800756 Blocks received from initiator = 2618189622 Blocks read from cache and sent to initiator = 854615302 Number of read and write commands whose size segment size = 0 Vendor (Seagate/Hitachi) factory information number of hours powered up = 12377.45 number of minutes until next internal SMART test = 56 Error counter log: Errors Corrected by ECC rereads/ fast | delayed rewrites read: 7416197 0 0 write: 0 0 0 verify: 142475069 0 0 Non-medium error count: SMART Self-test log Num Test Description # 1 Background long # 2 Background short

Total errors corrected 7416197 0 142475069

Correction Gigabytes Total algorithm processed uncorrected invocations [10^9 bytes] errors 7416197 1886.494 0 0 1349.999 0 142475069 22222.134 0

2661 Status Completed Completed

segment number 16 16

LifeTime (hours) 3 0

LBA_first_err [SK ASC ASQ] - [- [-

-

-] -]

Long (extended) Self Test duration: 6400 seconds [106.7 minutes]

Несмотря на всю полезность, эта информация не дает ответа на вопросы об отдельных медленных операциях дискового ввода/вывода, как это делают фреймворки трассировки ядра. Информация об исправленных ошибках пригодится для мониторинга: опираясь на нее, можно предсказать сбой диска до того, как он произойдет, а также подтвердить выход диска из строя или состояние, близкое к выходу из строя.

9.6.14. Журналирование SCSI В Linux есть встроенное средство для журналирования событий SCSI. Его можно включить через sysctl(8) или /proc. Например, обе команды ниже устанавливают максимальный уровень журналирования для всех типов событий (будьте осторожны:

608  Глава 9. Диски при высокой нагрузке на диск это может привести к переполнению системного журнала): # sysctl -w dev.scsi.logging_level=03333333333 # echo 03333333333 > /proc/sys/dev/scsi/logging_level

Числа в этих примерах представляют битовую маску, которая устанавливает уровень журналирования от 1 до 7 для десяти различных типов событий (здесь числа записаны в восьмеричном формате, в шестнадцатеричном формате они будут иметь вид 0x1b6db6db). Эта битовая маска определена в файле drivers/scsi/scsi_logging.h. Для ее настройки в пакете sg3-utils есть инструмент scsi_logging_level(8). Например: # scsi_logging_level -s --all 3

Примеры событий: # dmesg [...] [542136.259412] sd 0:0:0:0: [542136.259422] sd 0:0:0:0: [542136.261103] sd 0:0:0:0: driverbyte=DRIVER_OK [542136.261110] sd 0:0:0:0: [542136.261115] sd 0:0:0:0: [542136.261121] sd 0:0:0:0: [542136.261127] sd 0:0:0:0: [...]

tag#0 Send: scmd 0x0000000001fb89dc tag#0 CDB: Test Unit Ready 00 00 00 00 00 00 tag#0 Done: SUCCESS Result: hostbyte=DID_OK tag#0 tag#0 tag#0 tag#0

CDB: Test Unit Ready 00 00 00 00 00 00 Sense Key : Not Ready [current] Add. Sense: Medium not present 0 sectors total, 0 bytes done.

Поддержку журналирования можно использовать для выявления ошибок и таймаутов. Несмотря на наличие отметок времени (первый столбец), их едва ли можно использовать для вычисления задержек ввода/вывода, когда нет уникальных идентификационных данных.

9.6.15. Другие инструменты В табл. 9.6 перечислены инструменты наблюдения за работой дисков, представленные в других главах этой книги и в «BPF Performance Tools» [Gregg 19]. Таблица 9.6. Другие инструменты наблюдения за работой дисков Раздел

Инструмент

Описание

7.5.1

vmstat

Статистики виртуальной памяти, включая подкачку

7.5.3

swapon

Использование устройства подкачки

[Gregg 19]

seeksize

Отображает расстояния для перемещения головок

[Gregg 19]

biopattern

Идентифицирует закономерность произвольного/последовательного обращения к диску

[Gregg 19]

bioerr

Трассирует дисковые ошибки

9.7. Визуализация  609

Раздел

Инструмент

Описание

[Gregg 19]

mdflush

Трассирует запросы на выталкивание в драйверах для групп устройств

[Gregg 19]

iosched

Обобщает информацию о задержках планировщика ввода/вывода

[Gregg 19]

scsilatency

Отображает распределение задержек при выполнении команд SCSI

[Gregg 19]

scsiresult

Отображает коды завершения команд SCSI

[Gregg 19]

nvmelatency

Обобщает задержки выполнения команд в драйвере NVME (твердотельного накопителя)

Другие инструменты наблюдения за дисками и источников информации в Linux:

yy /proc/diskstats: высокоуровневые статистики по дискам; yy seekwatcher: визуализирует закономерности доступа к диску [Mason 08]. Поставщики дисков могут предлагать дополнительные инструменты для доступа к статистикам микропрограммы или для установки отладочной версии прошивки.

9.7. ВИЗУАЛИЗАЦИЯ Для анализа производительности дискового ввода/вывода есть множество разных приемов визуализации. Некоторые из них показаны в этом разделе вместе со скриншотами. Обсуждение визуализации вы найдете в главе 2 «Методологии», в разделе 2.10 «Визуализация».

9.7.1. Линейные диаграммы Решения для мониторинга производительности обычно отображают изменение частоты операций ввода/вывода (IOPS), пропускной способности и потребления диска со временем в виде линейных графиков. Это помогает увидеть временные закономерности — изменение нагрузки в течение дня или повторяющиеся события, например интервалы очистки кэшей файловой системы. Учитывайте особенности метрики, изображенной на графике. Средняя задержка может скрывать мультимодальные распределения и выбросы. Средние значения по всем дисковым устройствам могут скрыть несбалансированное поведение, включая чрезмерную нагрузку на одно устройство. Средние значения за длительные периоды времени могут скрывать краткосрочные колебания.

9.7.2. Диаграммы рассеяния задержек Диаграмма рассеяния позволяет визуализировать задержки ввода/вывода для всех событий, а их могут быть тысячи. Вдоль оси X такой диаграммы откладывается время завершения операции, а по оси Y — время отклика (задержка). На рис. 9.11 дан

610  Глава 9. Диски пример диаграммы рассеяния, построенной с использованием R на основе 1400 событий ввода/вывода, зафиксированных с помощью iosnoop(8) на промышленном сервере базы данных MySQL.

Рис. 9.11. Диаграмма рассеяния задержек дисковых операций чтения и записи События чтения и записи на этой диаграмме рассеяния изображены разными значками: + — чтение, ° — запись. Также можно построить диаграмму, откладывая вдоль оси Y другие метрики, например адрес блока диска. Здесь можно заметить пару выбросов задержек чтения с величинами больше 150 мс. Причина этих выбросов ранее не была известна. Эта диаграмма рассеяния и другие, включающие аналогичные выбросы, показали, что они возникают после большого количества операций записи. Операции записи имеют низкую задержку, потому что используют кэш отложенной записи в RAID-контроллере, который выполняет фактическую запись на устройство чуть позже. Я подозреваю, что эти операции чтения оказались позади операций записи в очереди на устройстве. На этой диаграмме показана работа одного сервера в течение нескольких секунд. Если увеличить массив данных, добавив несколько серверов или использовав более длинные интервалы, то события на графике могут начать сливаться и его будет трудно читать. В таких случаях предпочтительнее использовать тепловую карту задержек.

9.7.3. Тепловые карты задержек Тепловую карту можно использовать для визуализации задержек, откладывая время вдоль оси X, задержку ввода/вывода вдоль оси Y и количество операций ввода/вывода в конкретное время и диапазон задержек вдоль оси Z, которая отображается цветом (чем темнее, тем больше). О тепловых картах шла речь в главе 2 «Методологии», в разделе 2.10.3 «Тепловые карты». На рис. 9.12 показан один интересный пример. Рабочая нагрузка здесь была получена экспериментальным путем: я организовал последовательное чтение с нескольких дисков один за другим, чтобы исследовать ограничения шины и контроллера. Получившаяся тепловая карта оказалась неожиданной (была описана как «птеродактиль»), и она показывает информацию, которая была бы упущена при анализе только средних значений. Каждая из рассматриваемых

9.7. Визуализация  611 деталей имеет техническое объяснение: например, «клюв» охватывает восемь дисков, что равно количеству подключенных портов SAS (два порта x4), а «голова» начинается с девятого диска, когда возникает конкуренция за эти порты. Я придумал создавать тепловые карты для визуализации изменения задержек во времени, вдохновившись инструментом taztool, описанным в следующем разделе. Рисунок 9.12 взят из раздела «Analytics» в описании устройства Sun Microsystems ZFS Storage [Gregg 10a]: я построил эту и другие интересные тепловые карты задержки, чтобы поделиться ими и популяризовать их применение.

Рис. 9.12. Тепловая карта задержек дискового ввода/вывода в форме птеродактиля Оси X и Y аналогичны одноименным осям на диаграмме рассеяния задержек. Основное преимущество тепловых карт в том, что они могут масштабироваться до миллионов событий, когда диаграмма рассеяния превращается в полностью закрашенные полосы, на которых невозможно ничего различить. Эта проблема обсуждалась в разделах 2.10.2 «Диаграммы рассеяния» и 2.10.3 «Тепловые карты».

9.7.4. Тепловые карты смещений Местоположение ввода/вывода, или смещение, тоже можно визуализировать в виде тепловой карты (желательно перед созданием тепловых карт задержек). Пример такой тепловой карты показан на рис. 9.13. Смещение на диске (адрес блока) отображается вдоль оси Y, а время — вдоль оси X. Каждый пиксель окрашивается в зависимости от количества операций ввода/вывода, попавших в этот диапазон времени и задержки. Чем больше значение, тем темнее цвет. В данном примере визуализирована рабочая нагрузка, созданная процедурой архивирования файловой системы, которая перемещается по всему диску, начиная с блока 0. Темные линии соответствуют последовательному вводу/выводу, а светлые — произвольному.

612  Глава 9. Диски

Рис. 9.13. DTraceTazTool Эта диаграмма создана Ричардом Макдугаллом (Richard McDougall) в 1995 году с помощью инструмента taztool. Скриншот получен с помощью DTraceTazTool, версии инструмента taztool, которую я написал в 2006 году. Тепловые карты смещений дискового ввода/вывода можно получить с помощью других инструментов, включая seekwatcher (Linux).

9.7.5. Тепловые карты потребления Потребление каждого устройства тоже можно показать в виде тепловой карты, чтобы выявить баланс потребления устройств и отдельные выбросы [Gregg 11b]. В этом случае вдоль оси Y откладывается процент потребления, а более темный цвет означает большее количество дисков с этим уровнем потребления. Тепловые карты этого вида пригодятся для выявления отдельных «горячих» дисков, в том числе «ленивых», в виде линий в верхней части тепловой карты (100 %). Пример тепловой карты потребления приводится в главе 6 «Процессоры», в разделе 6.7.1 «Тепловая карта потребления».

9.8. ЭКСПЕРИМЕНТЫ В этом разделе описываются инструменты для активного тестирования производительности дискового ввода/вывода. Общее описание предлагаемой методологии приводится в разделе 9.5.9 «Микробенчмаркинг».

9.8. Эксперименты  613 При использовании этих инструментов рекомендуется оставить постоянно работающим iostat(1), чтобы любой результат можно было немедленно перепроверить. Некоторым инструментам микробенчмаркинга может потребоваться режим «прямого» доступа в обход кэша файловой системы, чтобы сосредоточить внимание на производительности дискового устройства.

9.8.1. Ad hoc Для выполнения бенчмаркинга последовательного доступа к дискам можно использовать команду dd(1) (выполняет копирование с устройства на устройство). Вот пример тестирования последовательного чтения с размером ввода/вывода 1 Мбайт: # dd if=/dev/sda1 of=/dev/null bs=1024k count=1k 1024+0 records in 1024+0 records out 1073741824 bytes (1.1 GB) copied, 7.44024 s, 144 MB/s

Поскольку ядро может кэшировать и буферизовать данные, измеренная с помощью dd(1) пропускная способность может характеризовать производительность кэша и диска вместе, а не только диска. Чтобы протестировать только производительность диска, используйте специальное символьное устройство для диска: в Linux его можно создавать в /dev/raw командой raw(8) (если доступна). Последовательную запись тестируют аналогично. Остерегайтесь уничтожения всех данных на диске, включая главную загрузочную запись и таблицу разделов! Более безопасный подход — использовать флаг прямого ввода/вывода с командой dd(1) и файлы в файловой системе вместо дисковых устройств. Но имейте в виду, что в этом случае тест включает некоторый оверхед файловой системы. Вот пример тестирования записи в файл с именем out1: # dd if=/dev/zero of=out1 bs=1024k count=1000 oflag=direct 1000+0 records in 1000+0 records out 1048576000 bytes (1.0 GB, 1000 MiB) copied, 1.79189 s, 585 MB/s

Утилита iostat(1), запущенная в другом терминале, подтвердила, что пропускная способность дискового ввода/вывода при записи составила около 585 Мбайт/с. Используйте iflag=direct для прямого чтения их входных файлов.

9.8.2. Пользовательские генераторы нагрузки Для создания своих рабочих нагрузок можно написать собственный генератор нагрузки и измерить итоговую производительность с помощью iostat(1). Роль такого генератора нагрузки может играть короткая программа на языке C, которая открывает устройство и применяет предполагаемую рабочую нагрузку. В Linux специальные файлы блочных устройств можно открывать с флагом O_DIRECT, чтобы исключить буферизацию. Если вы пользуетесь языками более высокого

614  Глава 9. Диски уровня, попробуйте использовать системные интерфейсы, чтобы как минимум обойти механизмы буферизации в стандартных библиотеках (например, sysread() в Perl) и, желательно, также механизм буферизации в ядре (например, применив флаг O_DIRECT).

9.8.3. Инструменты микробенчмаркинга Доступные инструменты бенчмаркинга дисков включают, например, hdparm(8) в Linux: # hdparm -Tt /dev/sdb /dev/sdb: Timing cached reads: 16718 MB in 2.00 seconds = 8367.66 MB/sec Timing buffered disk reads: 846 MB in 3.00 seconds = 281.65 MB/sec

Параметр -T тестирует чтение из кэша, а -t — чтение с дискового устройства. Полученные результаты показывают, насколько велика разница между попаданиями и промахами дискового кэша. Изучите документацию по инструменту, чтобы понимать все достоинства и недостатки, и загляните в главу 12 «Бенчмаркинг» — там приводятся дополнительные сведения о микробенчмаркинге. См. также главу 8 «Файловые системы», где описываются инструменты для проверки производительности диска через файловую систему (которых доступно гораздо больше).

9.8.4. Пример произвольного чтения В качестве примера эксперимента я написал инструмент, выполняющий произвольное чтение блоками по 8 Кбайт из файла дискового устройства. Одновременно выполнялись от одного до пяти экземпляров инструмента и утилита iostat(1). Столбцы, отражающие нулевой объем записи, были удалены: Device: sda [...] Device: sda [...] Device: sda [...] Device: sda [...] Device: sda

rrqm/s 878.00

r/s 234.00

rkB/s 2224.00

avgrq-sz 19.01

aqu-sz r_await 1.00 4.27

svctm %util 4.27 100.00

rrqm/s 1233.00

r/s 311.00

rkB/s 3088.00

avgrq-sz 19.86

aqu-sz r_await 2.00 6.43

svctm 3.22

%util 100.00

rrqm/s 1366.00

r/s 358.00

rkB/s 3448.00

avgrq-sz 19.26

aqu-sz r_await 3.00 8.44

svctm 2.79

%util 100.00

rrqm/s 1775.00

r/s 413.00

rkB/s 4376.00

avgrq-sz 21.19

aqu-sz r_await 4.01 9.66

svctm 2.42

%util 100.00

rrqm/s 1977.00

r/s 423.00

rkB/s 4800.00

avgrq-sz 22.70

aqu-sz r_await 5.04 12.08

svctm 2.36

%util 100.00

Обратите внимание на ступенчатое увеличение aqu-sz и увеличивающуюся задержку r_await.

9.8. Эксперименты  615

9.8.5. ioping ioping(1) — это интересный инструмент для микробенчмаркинга дисков, напоминающий утилиту ping(8). Вот пример запуска ioping(1) на дисковом устройстве nvme0n1: # ioping /dev/nvme0n1 4 KiB 100.2.6.189:50010 ESTABLISHED 00:20:12 60695 4 100.1.58.46:52346 R> 100.2.6.189:50010 ESTABLISHED 00:20:13 60695 6 ::ffff:100.1.58.46:13562 R> ::ffff:100.2.51.209:47356 FIN_WAIT1 00:20:13 60695 6 ::ffff:100.1.58.46:13562 R> ::ffff:100.2.51.209:47356 FIN_WAIT1 [...]

Как показывает этот вывод, в период трассировки повторные передачи выполнялись нечасто, всего несколько раз в секунду (об этом можно судить по столбцу TIME), и в основном в сеансах с состоянием ESTABLISHED. Высокая частота повторных передач в состоянии ESTABLISHED может указывать на проблемы с внешней сетью. Высокая частота в состоянии SYN_SENT может указывать на большую нагрузку на серверное приложение, которое не успевает обслуживать свою очередь запросов на соединение. Немного истории: в 2011 году я написал несколько похожих инструментов. Версию tcpretrans(8) на основе Ftrace я написал в 2014 году, а версию для BCC — 14 февраля 2016 года. Версию для bpftrace написал Дейл Хамел (Dale Hamel) 23 ноября 2018 года.

1

10.6. Инструменты наблюдения  683 Этот инструмент трассирует события повторной передачи TCP в ядре. Так как обычно они происходят нечасто, оверхед должен быть незначительным. Для сравнения, в традиционном подходе с использованием инструментов для перехвата всех пакетов и последующей их обработкой в поисках повторных передач оба шага могут приводить к значительным затратам вычислительных ресурсов. Кроме того, подход на основе перехвата пакетов может наблюдать только детали, имеющие отношение к сети, тогда как tcpretrans(8) выводит состояние TCP-соединений непосредственно из ядра и его можно расширить для вывода большего количества информации, если потребуется. Версия для BCC поддерживает следующие параметры:

yy -l: включать попытки повторной отправки «хвостового» сегмента (Tail Loss Probe, добавляет трассировку зонда ядра для tcp_send_loss_probe());

yy -c: подсчитывать повторные передачи по потокам данных. Параметр -c меняет поведение tcpretrans(8), заставляя выводить суммарные счетчики вместо подробностей о каждом событии.

10.6.12. bpftrace bpftrace — это трассировщик, основанный на BPF, который поддерживает высокоуровневый язык программирования, позволяющий создавать мощные и короткие сценарии. Хорошо подходит для анализа работы сети на основе подсказок, полученных с помощью других инструментов. Позволяет исследовать сетевые события в ядре и в приложениях, в том числе подключение к сокетам, ввод/вывод в сокетах, события TCP, передачу пакетов, отбрасывание запросов на соединение, повторные передачи TCP и другие детали. Эти возможности помогают определять характер рабочей нагрузки и анализировать задержки. Подробнее о bpftrace речь пойдет в главе 15. В этом разделе показано лишь несколько примеров однострочных сценариев для анализа производительности сети, трассировки сокетов и протокола TCP.

Однострочные сценарии Следующие однострочные сценарии показывают различные возможности bpftrace. Подсчет обращений к системному вызову accept(2) с группировкой по PID и именам процессов: bpftrace -e 't:syscalls:sys_enter_accept* { @[pid, comm] = count(); }'

Подсчет обращений к системному вызову connect(2) с группировкой по PID и именам процессов: bpftrace -e 't:syscalls:sys_enter_connect { @[pid, comm] = count(); }'

684  Глава 10. Сеть Подсчет обращений к системному вызову connect(2) с группировкой по трассировкам стека в пространстве пользователя: bpftrace -e 't:syscalls:sys_enter_connect { @[ustack, comm] = count(); }'

Подсчет операций приема/передачи с группировкой по направлениям, а также PID и именам процессов, выполняющихся на процессоре1: bpftrace -e 'k:sock_sendmsg,k:sock_recvmsg { @[func, pid, comm] = count(); }'

Подсчет принятых/отправленных байтов с группировкой по PID и именам процессов на процессоре: bpftrace -e 'kr:sock_sendmsg,kr:sock_recvmsg /(int32)retval > 0/ { @[pid, comm] = sum((int32)retval); }'

Подсчет инициированных соединений TCP с группировкой по PID и именам процессов на процессоре: bpftrace -e 'k:tcp_v*_connect { @[pid, comm] = count(); }'

Подсчет принятых соединений TCP с группировкой по PID и именам процессов на процессоре: bpftrace -e 'k:inet_csk_accept { @[pid, comm] = count(); }'

Подсчет операций приема/передачи с группировкой по PID и именам процессов на процессоре: bpftrace -e 'k:tcp_sendmsg,k:tcp_recvmsg { @[func, pid, comm] = count(); }'

Гистограмма с количествами отправленных байтов по протоколу TCP: bpftrace -e 'k:tcp_sendmsg { @send_bytes = hist(arg2); }'

Гистограмма с количествами полученных байтов по протоколу TCP: bpftrace -e 'kr:tcp_recvmsg /retval >= 0/ { @recv_bytes = hist(retval); }'

Подсчет повторных передач TCP с группировкой по типам и удаленным хостам (предполагается, что используется протокол IPv4): bpftrace -e 't:tcp:tcp_retransmit_* { @[probe, ntop(2, args->saddr)] = count(); }'

Системные вызовы, которые трассируются предыдущими сценариями, выполняются в контексте процесса, где надежно определяются PID и имя процесса. Зонды kprobes, используемые в этом сценарии, находятся глубже в ядре и к моменту их срабатывания процесс, инициировавший соединение, может уже оставить процессор. Это означает, что значения pid и comm, доступные bpftrace, могут быть связаны с другим процессом. Такое случается редко, но все же случается.

1

10.6. Инструменты наблюдения  685 Подсчет вызовов всех функций TCP (добавляет значительный оверхед при большом количестве операций TCP): bpftrace -e 'k:tcp_* { @[func] = count(); }'

Подсчет операций приема/передачи по протоколу UDP с группировкой по PID и именам процессов на процессоре: bpftrace -e 'k:udp*_sendmsg,k:udp*_recvmsg { @[func, pid, comm] = count(); }'

Гистограмма с количествами отправленных байтов по протоколу UDP: bpftrace -e 'k:udp_sendmsg { @send_bytes = hist(arg2); }'

Гистограмма с количествами принятых байтов по протоколу UDP: bpftrace -e 'kr:udp_recvmsg /retval >= 0/ { @recv_bytes = hist(retval); }'

Подсчет операций передачи с группировкой по трассировкам стека в пространстве ядра: bpftrace -e 't:net:net_dev_xmit { @[kstack] = count(); }'

Выводит гистограмму потребления процессорного времени на прием для каждого устройства: bpftrace -e 't:net:netif_receive_skb { @[str(args->name)] = lhist(cpu, 0, 128, 1); }'

Подсчитывает вызовы функций ieee80211 (добавляет значительный оверхед в операции с пакетами): bpftrace -e 'k:ieee80211_* { @[func] = count(); }'

Подсчитывает вызовы всех функций драйвера устройства ixgbevf (добавляет значительный оверхед в работу ixgbevf): bpftrace -e 'k:ixgbevf_* { @[func] = count(); }'

Подсчитывает срабатывания всех точек трассировки драйвера устройства iwl (добавляет значительный оверхед в работу iwl): bpftrace -e 't:iwlwifi:*,t:iwlwifi_io:* { @[probe] = count(); }'

Трассировка сокетов Трассировка сетевых событий на уровне сокетов имеет то преимущество, что соответствующий процесс по-прежнему выполняется на процессоре. Это упрощает определение приложения и путей в коде. Вот пример подсчета обращений к системному вызову accept(2) с группировкой по PID и именам процессов: # bpftrace -e 't:syscalls:sys_enter_accept { @[pid, comm] = count(); }' Attaching 1 probe... ^C

686  Глава 10. Сеть @[573, sshd]: 2 @[1948, mysqld]: 41

Как показывает этот вывод, в период трассировки сервер mysqld вызвал accept(2) 41 раз, а sshd — 2 раза. Есть возможность включить в вывод трассировки стека, чтобы показать путь в коде, который привел к вызову accept(2). Вот пример подсчета обращений к accept(2) с группировкой по трассировкам стека в пространстве пользователя и именам процессов: # bpftrace -e 't:syscalls:sys_enter_accept { @[ustack, comm] = count(); }' Attaching 1 probe... ^C @[

accept+79 Mysqld_socket_listener::listen_for_connection_event()+283 mysqld_main(int, char**)+15577 __libc_start_main+243 0x49564100fe8c4b3d , mysqld]: 22

Как показывает этот вывод, сервер mysqld принимал входящие соединения в функции Mysqld_socket_listener::listen_for_connection_event(). Заменив «accept» на «connect», с помощью этого однострочного сценария можно определить пути в коде, ведущие к вызову connect(2). Мне доводилось использовать такие однострочные сценарии, чтобы объяснить загадочные сетевые соединения, анализируя пути в коде, ведущие к ним.

Точки трассировки sock Помимо системных вызовов есть и точки трассировки сокетов. Вот пример для ядра 5.3: # bpftrace -l 't:sock:*' tracepoint:sock:sock_rcvqueue_full tracepoint:sock:sock_exceed_buf_limit tracepoint:sock:inet_sock_set_state

Точка трассировки sock:inet_sock_set_state используется инструментом tcplife(8), описанным выше. Вот пример однострочного сценария, использующего эту точку трассировки для подсчета IPv4-адресов отправителя и получателя в новых соединениях: # bpftrace -e 't:sock:inet_sock_set_state /args->newstate == 1 && args->family == 2/ { @[ntop(args->saddr), ntop(args->daddr)] = count() }' Attaching 1 probe... ^C @[127.0.0.1, 127.0.0.1]: 2 @[10.1.239.218, 10.29.225.81]: 18

10.6. Инструменты наблюдения  687 Этот однострочный сценарий довольно длинный, и его было бы проще сохранить в файл с программой на языке bpftrace (.bt), чтобы его легче было редактировать и запускать. Сохранив программу в файле, можно также подключить соответствующие заголовочные файлы из исходного кода ядра, чтобы строку фильтра можно было переписать с использованием имен констант, а не жестко закодированных чисел (которые ненадежны), например: /args->newstate == TCP_ESTABLISHED && args->family == AF_INET/ {

Далее приводится пример такой программы в файле: socketio.bt.

socketio.bt В качестве более сложного примера ниже приведен исходный код инструмента socketio(8), подсчитывающий операции ввода/вывода с сокетом и добавляющий информацию о процессе, направлении, протоколе и портах. Например: # ./socketio.bt Attaching 2 probes... ^C [...] @io[sshd, 21925, read, UNIX, 0]: 40 @io[sshd, 21925, read, TCP, 37408]: 41 @io[systemd, 1, write, UNIX, 0]: 51 @io[systemd, 1, read, UNIX, 0]: 57 @io[systemd-udevd, 241, write, NETLINK, 0]: 65 @io[systemd-udevd, 241, read, NETLINK, 0]: 75 @io[dbus-daemon, 525, write, UNIX, 0]: 98 @io[systemd-logind, 526, read, UNIX, 0]: 105 @io[systemd-udevd, 241, read, UNIX, 0]: 127 @io[snapd, 31927, read, NETLINK, 0]: 150 @io[dbus-daemon, 525, read, UNIX, 0]: 160 @io[mysqld, 1948, write, TCP, 55010]: 8147 @io[mysqld, 1948, read, TCP, 55010]: 24466

Как показывает этот вывод, большинство операций ввода/вывода с сокетами выполнял сервер mysqld, который читал и записывал данные в TCP-порт 55010 — временный порт, использовавшийся для подключения клиента. Исходный код socketio(8): #!/usr/local/bin/bpftrace #include kprobe:sock_recvmsg { $sock = (struct socket *)arg0; $dport = $sock->sk->__sk_common.skc_dport; $dport = ($dport >> 8) | (($dport sk->__sk_common.skc_prot->name, $dport] = count(); }

688  Глава 10. Сеть kprobe:sock_sendmsg { $sock = (struct socket *)arg0; $dport = $sock->sk->__sk_common.skc_dport; $dport = ($dport >> 8) | (($dport sk->__sk_common.skc_prot->name, $dport] = count(); }

В этом сценарии пример извлечения информации из структур в ядра — из struct socket, которая содержит название протокола и порт назначения. Порт назначения представлен целым числом с прямым порядком следования байтов (big-endian) и преобразуется в число с обратным порядком байтов (little-endian, для процессора x86) перед добавлением в карту @io1. Этот сценарий можно изменить и подсчитывать байты вместо операций ввода/вывода.

Трассировка TCP Трассировка на уровне TCP позволяет получить представление о событиях и внутренних компонентах протокола TCP, а также о событиях, не связанных с сокетами (например, сканирование портов TCP).

Точки трассировки TCP Инструментация внутренних механизмов TCP часто возможна только через зонды kprobes, но есть несколько точек трассировки TCP. Вот пример для ядра 5.3: # bpftrace -l 't:tcp:*' tracepoint:tcp:tcp_retransmit_skb tracepoint:tcp:tcp_send_reset tracepoint:tcp:tcp_receive_reset tracepoint:tcp:tcp_destroy_sock tracepoint:tcp:tcp_rcv_space_adjust tracepoint:tcp:tcp_retransmit_synack tracepoint:tcp:tcp_probe

Точка трассировки tcp:tcp_retransmit_skb используется инструментом tcpretrans(8), описанным выше. Точки трассировки предпочтительнее из-за их стабильности, но если они не позволяют решить поставленную задачу, то для инструментации функций TCP в ядре можно использовать зонды kprobes. Например: # bpftrace -e 'k:tcp_* { @[func] = count(); }' Attaching 336 probes... ^C @[tcp_try_keep_open]: 1 @[tcp_ooo_try_coalesce]: 1

1

Для правильной работы на процессорах с прямым порядком следования байтов инструмент должен проверять аппаратный порядок байтов и применять преобразование, только если это необходимо, например, используя #ifdef LITTLE_ENDIAN.

10.6. Инструменты наблюдения  689 @[tcp_reset]: 1 [...] @[tcp_push]: 3191 @[tcp_established_options]: 3584 @[tcp_wfree]: 4408 @[tcp_small_queue_check.isra.0]: 4617 @[tcp_rate_check_app_limited]: 7022 @[tcp_poll]: 8898 @[tcp_release_cb]: 18330 @[tcp_send_mss]: 28168 @[tcp_sendmsg]: 31450 @[tcp_sendmsg_locked]: 31949 @[tcp_write_xmit]: 33276 @[tcp_tx_timestamp]: 33485

Как показывает этот вывод, в период трассировки чаще других вызывалась функция tcp_tx_timestamp() — 33 485 раз. Подсчитывая вызовы функций, можно определить цели для более детальных исследований. Обратите внимание, что подсчет вызовов всех функций TCP может добавить заметный оверхед из-за количества и частоты вызова трассируемых функций. Для этой конкретной задачи я бы использовал профилирование функций с помощью моего инструмента funccount(8), основанного на Ftrace, — оверхед и время инициализации у него намного меньше. См. главу 14 «Ftrace».

tcpsynbl.bt Инструмент tcpsynbl(8)1 — это пример инструментации TCP с использованием зондов kprobes. Он показывает размеры очередей с поступившими запросами на соединение, используемыми системным вызовом listen(2), с разбивкой по размерам, чтобы можно было определить, насколько близки очереди к переполнению (что вызывает отбрасывание пакетов TCP SYN). Вот пример вывода: # tcpsynbl.bt Attaching 4 probes... Tracing SYN backlog size. Ctrl-C to end. 04:44:31 dropping a SYN. 04:44:31 dropping a SYN. 04:44:31 dropping a SYN. 04:44:31 dropping a SYN. 04:44:31 dropping a SYN. [...] ^C @backlog[backlog limit]: histogram of backlog size @backlog[128]: [0] [1] [2, 4) [4, 8) [8, 16)

473 502 1001 1996 3943

|@ |@ |@@@ |@@@@@@ |@@@@@@@@@@@

Немного истории: я написал tcpsynbl.bt 19 апреля 2019 года для книги [Gregg 19].

1

| | | | |

690  Глава 10. Сеть [16, 32) [32, 64) [64, 128) [128, 256)

7718 14460 17246 1844

|@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@ |

Во время работы tcpsynbl.bt выводит отметки времени, когда происходило отбрасывание пакетов SYN, если это происходило. После прерывания (нажатием Ctrl-C) выводится гистограмма с размерами очередей запросов на соединение для каждого используемого предела длины очереди. В этом выводе можно видеть, что в 4:44:31 было отброшено несколько пакетов SYN, а результирующая гистограмма показывает распределение размеров очередей для предела 128, согласно которой этот предел был достигнут 1844 раза (интервал от 128 до 256). Это распределение сообщает размеры очередей, бывших в момент получения пакета SYN. Трассировка размеров очередей запросов на соединение позволяет проверить, растет ли очередь со временем, и предупредить заранее о неизбежности потери пакетов SYN. Такой анализ можно использовать при планировании мощностей. Исходный код tcpsynbl(8): #!/usr/local/bin/bpftrace #include BEGIN { printf("Tracing SYN backlog size. Ctrl-C to end.\n"); } kprobe:tcp_v4_syn_recv_sock, kprobe:tcp_v6_syn_recv_sock { $sock = (struct sock *)arg0; @backlog[$sock->sk_max_ack_backlog & 0xffffffff] = hist($sock->sk_ack_backlog); if ($sock->sk_ack_backlog > $sock->sk_max_ack_backlog) { time("%H:%M:%S dropping a SYN.\n"); } } END { }

printf("\n@backlog[backlog limit]: histogram of backlog size\n");

Форма распределения, представленного выше, во многом обусловлена масштабом log2, используемым функцией hist(), в соответствии с которым каждый следующий интервал охватывает более широкий диапазон. Вместо hist() можно использовать lhist(): lhist($sock->sk_ack_backlog, 0, 1000, 10);

В результате выводится линейная гистограмма с одинаковыми диапазонами в интервалах: в данном случае гистограмма охватывает диапазон от 0 до 1000 с шириной

10.6. Инструменты наблюдения  691 интервала 10. Подробнее о программировании на bpftrace рассказывается в главе 15 «BPF».

Источники событий bpftrace может инструментировать не только точки трассировки и зонды kprobes. В табл. 10.6 перечислены все источники сетевых событий, доступные для инструментации. Таблица 10.6. Сетевые события и их источники События

Источник

Прикладные протоколы

Зонды uprobes

Сокеты

Точки трассировки системных вызовов

TCP

Точки трассировки TCP, зонды kprobes

UDP

Зонды kprobes

IP и ICMP

Зонды kprobes

Пакеты

Точки трассировки ckb, зонды kprobes

Дисциплины очередей и очереди драйверов

Точки трассировки qdisc и net, зонды kprobes

XDP

Точки трассировки xdp

Драйверы сетевых устройств

Зонды kprobes, некоторые драйверы имеют точки трассировки

По возможности используйте точки трассировки, так как они представляют более стабильный интерфейс.

10.6.13. tcpdump Перехватывать и исследовать пакеты в Linux можно с помощью утилиты tcpdump(8). Она может выводить сводную информацию на стандартный вывод или записывать сведения о пакетах в файл для последующего анализа. Последний способ предпочтительнее с практической точки зрения: скорость передачи пакетов может быть слишком высокой, чтобы следить за ними в реальном времени. Вот пример вывода пакетов из интерфейса eth4 в файл в каталоге /tmp: # tcpdump -i eth4 -w /tmp/out.tcpdump tcpdump: listening on eth4, link-type EN10MB (Ethernet), capture size 65535 bytes ^C273893 packets captured 275752 packets received by filter 1859 packets dropped by kernel

В выводе указывается, сколько пакетов отбросило ядро вместо передачи в tcpdump(8). Такое случается, когда частота следования пакетов слишком высокая.

692  Глава 10. Сеть Обратите внимание, что утилиту можно запустить с параметром -i any, чтобы заставить ее перехватывать пакеты со всех интерфейсов. Вот пример содержимого файла с информацией о пакетах: # tcpdump -nr /tmp/out.tcpdump reading from file /tmp/out.tcpdump, link-type EN10MB (Ethernet) 02:24:46.160754 IP 10.2.124.2.32863 > 10.2.203.2.5001: Flags [.], seq 3612664461:3612667357, ack 180214943, win 64436, options [nop,nop,TS val 692339741 ecr 346311608], length 2896 02:24:46.160765 IP 10.2.203.2.5001 > 10.2.124.2.32863: Flags [.], ack 2896, win 18184, options [nop,nop,TS val 346311610 ecr 692339740], length 0 02:24:46.160778 IP 10.2.124.2.32863 > 10.2.203.2.5001: Flags [.], seq 2896:4344, ack 1, win 64436, options [nop,nop,TS val 692339741 ecr 346311608], length 1448 02:24:46.160807 IP 10.2.124.2.32863 > 10.2.203.2.5001: Flags [.], seq 4344:5792, ack 1, win 64436, options [nop,nop,TS val 692339741 ecr 346311608], length 1448 02:24:46.160817 IP 10.2.203.2.5001 > 10.2.124.2.32863: Flags [.], ack 5792, win 18184, options [nop,nop,TS val 346311610 ecr 692339741], length 0 [...]

Каждая строка в выводе сообщает время передачи пакета (с разрешением до микросекунд), IP-адреса отправителя и получателя, а также значения полей в заголовке TCP. Изучая их, можно понять, как работает TCP, в том числе насколько хорошо расширенные возможности справляются с вашей рабочей нагрузкой. Параметр -n отключает разрешение IP-адресов в имена хостов. В числе других параметров можно назвать вывод подробных сведений, если они доступны (-v), вывод заголовков канального уровня (-e) и вывод содержимого в шестнадцатеричном формате (-x или -X). Например: # tcpdump -enr /tmp/out.tcpdump -vvv -X reading from file /tmp/out.tcpdump, link-type EN10MB (Ethernet) 02:24:46.160754 80:71:1f:ad:50:48 > 84:2b:2b:61:b6:ed, ethertype IPv4 (0x0800), length 2962: (tos 0x0, ttl 63, id 46508, offset 0, flags [DF], proto TCP (6), length 2948) 10.2.124.2.32863 > 10.2.203.2.5001: Flags [.], cksum 0x667f (incorrect -> 0xc4da), seq 3612664461:3612667357, ack 180214943, win 64436, options [nop,nop,TS val 692339741 ecr 346311608], length 289 6 0x0000: 4500 0b84 b5ac 4000 3f06 1fbf 0a02 7c02 E.....@.?.....|. 0x0010: 0a02 cb02 805f 1389 d754 e28d 0abd dc9f ....._...T...... 0x0020: 8010 fbb4 667f 0000 0101 080a 2944 441d ....f.......)DD. 0x0030: 14a4 4bb8 3233 3435 3637 3839 3031 3233 ..K.234567890123 0x0040: 3435 3637 3839 3031 3233 3435 3637 3839 4567890123456789 [...]

При анализе производительности иногда полезно изменить столбец с отметкой времени так, чтобы в нем отображалась разность времени между пакетами (-ttt) или время, прошедшее с момента первого пакета (-ttttt). Также есть возможность определить выражение для фильтрации пакетов (см. описание pcap-filter(7)), чтобы сосредоточиться только на интересующих пакетах. Выражение вычисляется в ядре с использованием BPF для большей эффективности (кроме Linux 2.0 и старше).

10.6. Инструменты наблюдения  693 Перехват пакетов требует больших затрат процессорного времени и памяти. Если возможно, используйте tcpdump(8) только в течение коротких периодов времени, чтобы ограничить отрицательное влияние на производительность. Ищите возможности использовать более эффективные инструменты на основе BPF, например bpftrace. tshark(1) — аналогичный инструмент командной строки для перехвата пакетов, который предлагает улучшенные возможности фильтрации и вывода. Это версия Wireshark с интерфейсом командной строки.

10.6.14. Wireshark tcpdump(8) отлично подходит для кратковременных и бессистемных исследований, но его использование для более глубокого анализа, тем более в командной строке, может занять много времени. На этом фоне намного предпочтительнее выглядит инструмент с графическим интерфейсом Wireshark (ранее Ethereal). Он позволяет перехватывать и исследовать пакеты, а также импортировать файлы с пакетами, созданными tcpdump(8) [Wireshark 20]. В числе его полезных возможностей — идентификация сетевых подключений и связанных с ними пакетов, чтобы их можно было изучать отдельно, а также перевод сотен заголовков протоколов. На рис. 10.12 показано, как выглядит окно Wireshark. Оно разделено по горизонтали на три части. В верхней таблице отображаются строки с информацией о пакетах.

Рис. 10.12. Окно Wireshark

694  Глава 10. Сеть В средней части выводятся сведения о протоколе: в этом примере было раскрыто описание протокола TCP и выбран порт назначения. В нижней части слева показано содержимое пакета в шестнадцатеричном виде слева и справа — то же содержимое, но в текстовом виде: здесь выделены ячейки с номером порта назначения TCP.

10.6.15. Другие инструменты В табл. 10.7 перечислены инструменты наблюдения за работой сети, представленные в других главах этой книги и в «BPF Performance Tools» [Gregg 19]. Таблица 10.7. Другие инструменты наблюдения за работой сети Раздел

Инструмент

Описание

5.5.3

offcputime

Профилирование времени вне процессора, может профилировать операции сетевого ввода/вывода

[Gregg 19]

sockstat

Выводит обобщенную статистику по сокетам

[Gregg 19]

sofamily

Подсчет новых сокетов по семействам адресов и процессам

[Gregg 19]

soprotocol

Подсчет новых сокетов по транспортным протоколам и процессам

[Gregg 19]

soconnect

Трассирует запросы на соединение по протоколу IP

[Gregg 19]

soaccept

Трассирует прием соединений по протоколу IP

[Gregg 19]

socketio

Отображает сводную информацию о сокетах со счетчиками ввода/вывода

[Gregg 19]

socksize

Отображает объемы ввода/вывода через сокеты по процессам в виде гистограмм

[Gregg 19]

sormem

Отображает статистику использования и переполнения буферов чтения в сокетах

[Gregg 19]

soconnlat

Обобщает информацию о задержках соединения с IP-сокетами с трассировками стека

[Gregg 19]

so1stbyte

Обобщает информацию о задержках до получения первого байта

[Gregg 19]

tcpconnect

Трассирует создание активных TCP-соединений (connect())

[Gregg 19]

tcpaccept

Трассирует создание пассивных TCP-соединений (accept())

[Gregg 19]

tcpwin

Трассирует изменение параметра, определяющего размер окна насыщения

[Gregg 19]

tcpnagle

Трассирует работу алгоритма Нейгла и задержки передачи

[Gregg 19]

udpconnect

Трассирует новые локальные соединения UDP

[Gregg 19]

gethostlatency

Трассирует задержки поиска в DNS через библиотечные вызовы

[Gregg 19]

ipecn

Трассирует уведомления о перегрузке входящего IP-трафика

[Gregg 19]

superping

Измеряет время выполнения запросов ICMP ECHO

[Gregg 19]

qdisc-fq (...)

Отображает задержку в очереди с дисциплиной FQ

[Gregg 19]

netsize

Отображает объемы ввода/вывода через сетевые устройства

10.7. Эксперименты  695

Раздел

Инструмент

Описание

[Gregg 19]

nettxlat

Отображает задержку передачи через сетевые устройства

[Gregg 19]

skbdrop

Трассирует события сброса буферов sk_buff и отображает соответствующие им трассировки стека ядра

[Gregg 19]

skblife

Измеряет продолжительность существования буферов sk_buff как задержки между вызовами функций сетевого стека

[Gregg 19]

ieee80211scan

Трассирует сканирование Wi-Fi согласно IEEE 802.11

Среди других инструментов для анализа работы сети в Linux можно назвать:

yy strace(1) : для трассировки системных вызовов, обслуживающих сокеты, и исследования их параметров (имейте в виду, что strace(1) может создавать существенную нагрузку);

yy lsof(8): для получения списка файлов с некоторыми деталями, открытых процессами, в том числе и сокетов;

yy nfsstat(8): для получения статистик сервера и клиента NFS; yy ifpps(8): для получения статистик сети и системы в стиле top(1); yy iftop(8): для получения обобщенной информации о пропускной способности сетевых интерфейсов (анализатор пакетов);

yy perf(1): для подсчета срабатываний и получения аргументов точек трассировки в сетевой подсистеме ядра;

yy /proc/net: содержит множество файлов с информацией о сети; yy итератор BPF: позволяет программам для BPF экспортировать нестандартные статистики в /sys/fs/bpf.

Кроме того, есть множество решений для мониторинга сети, основанных либо на SNMP, либо на собственных нестандартных агентах.

10.7. ЭКСПЕРИМЕНТЫ Производительность сети обычно исследуется с помощью инструментов, которые выполняют эксперименты, а не просто наблюдают за состоянием системы. К таким инструментам относятся ping(8), traceroute(8) и средства микробенчмаркинга — iperf(8). Их можно использовать для определения работоспособности сети между узлами и для выяснения того, обусловлена ли низкая производительность приложений недостаточной пропускной способностью сети.

10.7.1. ping Команда ping(8) проверяет сетевое соединение, отправляя пакеты эхо-запроса ICMP. Например:

696  Глава 10. Сеть # ping www.netflix.com PING www.netflix.com(2620:108:700f::3423:46a1 (2620:108:700f::3423:46a1)) 56 data bytes 64 bytes from 2620:108:700f::3423:46a1 (2620:108:700f::3423:46a1): icmp_seq=1 ttl=43 time=32.3 ms 64 bytes from 2620:108:700f::3423:46a1 (2620:108:700f::3423:46a1): icmp_seq=2 ttl=43 time=34.3 ms 64 bytes from 2620:108:700f::3423:46a1 (2620:108:700f::3423:46a1): icmp_seq=3 ttl=43 time=34.0 ms ^C --- www.netflix.com ping statistics --3 packets transmitted, 3 received, 0% packet loss, time 2003ms rtt min/avg/max/mdev = 32.341/33.579/34.389/0.889 ms

Команда выводит время приема/передачи (time) для каждого пакета и некоторую сводную информацию. Старые версии ping(8) измеряли время приема/передачи в пространстве пользователя, немного увеличивая его из-за задержки планировщика. Новые ядра и версии ping(8) используют поддержку отметок времени ядра (SIOCGSTAMP или SO_TIMESTAMP), чтобы повысить точность отображаемых значений времени. Пакеты ICMP могут обрабатываться маршрутизаторами с более низким приоритетом, чем прикладные протоколы, поэтому задержка может иметь более высокую дисперсию, чем обычно1.

10.7.2. traceroute Команда traceroute(8) отправляет серию тестовых пакетов для экспериментального определения текущего маршрута к хосту. Для этого в каждом следующем пакете на единицу увеличивается время жизни (time to live, TTL) в заголовке IP, в результате чего цепочка шлюзов на пути к хосту раскрывает себя, отправляя ICMP-сообщения о превышении времени (при условии, что брандмауэр не блокирует их). Вот пример тестирования текущего маршрута от хоста в Калифорнии до моего сайта: # traceroute www.brendangregg.com traceroute to www.brendangregg.com (184.168.188.1), 30 hops max, 60 byte packets 1 _gateway (10.0.0.1) 3.453 ms 3.379 ms 4.769 ms 2 196.120.89.153 (196.120.89.153) 19.239 ms 19.217 ms 13.507 ms 3 be-10006-rur01.sanjose.ca.sfba.comcast.net (162.151.1.145) 19.141 ms 19.102 ms 19.050 ms 4 be-231-rar01.santaclara.ca.sfba.comcast.net (162.151.78.249) 19.018 ms 18.987 ms 18.941 ms 5 be-299-ar01.santaclara.ca.sfba.comcast.net (68.86.143.93) 21.184 ms 18.849 ms 21.053 ms 6 lag-14.ear3.SanJose1.Level3.net (4.68.72.105) 18.717 ms 11.950 ms 16.471 ms 7 4.69.216.162 (4.69.216.162) 24.905 ms 4.69.216.158 (4.69.216.158) 21.705 ms 28.043 ms 1

Хотя в некоторых сетях ICMP может иметь более высокий приоритет, чтобы лучше работать в бенчмарках на основе ping.

10.7. Эксперименты  697 8 4.53.228.238 (4.53.228.238) 35.802 ms 37.202 ms 37.137 ms 9 ae0.ibrsa0107-01.lax1.bb.godaddy.com (148.72.34.5) 24.640 ms 24.610 ms 24.579 ms 10 148.72.32.16 (148.72.32.16) 33.747 ms 35.537 ms 33.598 ms 11 be38.trmc0215-01.ars.mgmt.phx3.gdg (184.168.0.69) 33.646 ms 33.590 ms 35.220 ms 12 * * * 13 * * * [...]

Для каждого перехода выводится три значения времени приема/передачи (round-trip time, RTT), которые можно использовать для приблизительного выявления источников задержек в сети. Как и в случае с ping(8), отправляемые пакеты имеют низкий приоритет и могут показывать завышенную задержку по сравнению с другими протоколами. В некоторых попытках выводятся звездочки («*»): они соответствуют случаям, когда сообщение ICMP о превышении времени жизни не было возвращено отправителю. Все три случая, в которых выводятся звездочки, могут быть результатом того, что шлюз, обслуживающий переход, вообще не возвращает сообщения ICMP или эти сообщения заблокированы брандмауэром. Более надежное решение — переключиться на TCP вместо ICMP, добавив в команду параметр -T. Команда с этим параметром часто доступна как tcptraceroute(1). Есть более совершенная версия — astraceroute(8), которая позволяет настраивать флаги. Выявленный маршрут также можно исследовать в рамках статической настройки производительности. Сети проектируются так, чтобы могли быстро реагировать на перебои в работе и динамически менять маршруты при необходимости, поэтому производительность может снижаться из-за изменения маршрута. Обратите внимание, что маршрут может измениться даже во время выполнения traceroute(8): ответ от перехода 7 в предыдущем выводе сначала возвращается с адреса 4.69.216.162, а затем с адреса 4.69.216.158. Адрес выводится, только когда меняется, в противном случае для последующих попыток выводится только время RTT. Дополнительные сведения об интерпретации вывода traceroute(8) см. в [Steenbergen 09]. traceroute(8) был написан Ваном Якобсоном (Van Jacobson). Позже он создал удивительный инструмент под названием pathchar.

10.7.3. pathchar pathchar — инструмент, похожий на traceroute(8), но дополнительно он определяет ширину полосы пропускания между переходами. Для этого он многократно отправляет серии сетевых пакетов разного размера и выполняет статистический анализ. Вот пример вывода: # pathchar 192.168.1.10 pathchar to 192.168.1.1 (192.168.1.1) doing 32 probes at each of 64 to 1500 by 32

698  Глава 10. Сеть 0 localhost | 30 Mb/s, 79 us (562 us) 1 neptune.test.com (192.168.2.1) | 44 Mb/s, 195 us (1.23 ms) 2 mars.test.com (192.168.1.1) 2 hops, rtt 547 us (1.23 ms), bottleneck 30 Mb/s, pipe 7555 bytes

К сожалению, pathchar не пользуется популярностью (возможно, из-за того, что не был открыт исходный код), и у него имеются сложности с запуском оригинальной версии (самый свежий двоичный файл для Linux 2.0.30 был выложен на сайте pathchar в 1997 году [Jacobson 97]). Новая версия, созданная Брюсом А. Ма (Bruce A. Mah) и названная pchar(8), более доступна. Кроме того, запуск pathchar занимал очень много времени, порой десятки минут, в зависимости от количества переходов, но позднее были предложены методы, позволяющие сократить это время [Downey 99].

10.7.4. iperf iperf(1) — это инструмент с открытым исходным кодом для тестирования максимальной пропускной способности TCP и UDP. Он поддерживает множество возможностей, включая параллельный режим, в котором запускается несколько клиентских потоков, что может потребоваться для определения пределов возможностей сети. iperf(1) должен выполняться как на сервере, так и на клиенте. Вот пример запуска iperf(1) на сервере: $ iperf -s -l 128k -----------------------------------------------------------Server listening on TCP port 5001 TCP window size: 85.3 KByte (default) ------------------------------------------------------------

Эта команда увеличила размер буфера сокета до 128 Кбайт (-l 128k) со значения по умолчанию 8 Кбайт. А вот результаты, полученные на стороне клиента: # iperf -c 10.2.203.2 -l 128k -P 2 -i 1 -t 60 -----------------------------------------------------------Client connecting to 10.2.203.2, TCP port 5001 TCP window size: 48.0 KByte (default) -----------------------------------------------------------[ 4] local 10.2.124.2 port 41407 connected with 10.2.203.2 port 5001 [ 3] local 10.2.124.2 port 35830 connected with 10.2.203.2 port 5001 [ ID] Interval Transfer Bandwidth [ 4] 0.0- 1.0 sec 6.00 MBytes 50.3 Mbits/sec [ 3] 0.0- 1.0 sec 22.5 MBytes 189 Mbits/sec [SUM] 0.0- 1.0 sec 28.5 MBytes 239 Mbits/sec [ 3] 1.0- 2.0 sec 16.1 MBytes 135 Mbits/sec [ 4] 1.0- 2.0 sec 12.6 MBytes 106 Mbits/sec [SUM] 1.0- 2.0 sec 28.8 MBytes 241 Mbits/sec [...]

10.7. Эксперименты  699 [ 4] [ 3] [SUM]

0.0-60.0 sec 0.0-60.0 sec 0.0-60.0 sec

748 MBytes 996 MBytes 1.70 GBytes

105 Mbits/sec 139 Mbits/sec 244 Mbits/sec

В этой команде использованы следующие параметры:

yy yy yy yy yy

-c host: имя хоста или IP-адрес для подключения; -l 128k: использовать для сокета буфер размером 128 Кбайт; -P 2: запустить в параллельном режиме два клиентских потока; -i 1: выводить статистики с интервалом в одну секунду; -t 60: общая продолжительность тестирования: 60 с.

Последняя строка сообщает среднюю пропускную способность за все время тестирования по всем параллельным потокам: 244 Мбит/с. Чтобы увидеть, как меняются статистики с течением времени, можно выводить их через определенный интервал. Параметр --reportstyle C позволяет вывести данные в формате CSV, чтобы затем их можно было импортировать в другие инструменты, например средства для построения графиков.

10.7.5. netperf netperf(1) — усовершенствованный инструмент для микробенчмаркинга, позволяющий оценить производительность запросов/ответов [HP 18]. Я использую netperf(1) для измерения задержки передачи пакетов TCP туда и обратно. Вот пример вывода: server$ netserver -D -p 7001 Starting netserver with host 'IN(6)ADDR_ANY' port '7001' and family AF_UNSPEC [...] client$ netperf -v 100 -H 100.66.63.99 -t TCP_RR -p 7001 MIGRATED TCP REQUEST/RESPONSE TEST from 0.0.0.0 (0.0.0.0) port 0 AF_INET to 100.66.63.99 () port 0 AF_INET : demo : first burst 0 Alignment Offset RoundTrip Trans Throughput Local Remote Local Remote Latency Rate 10^6bits/s Send Recv Send Recv usec/Tran per sec Outbound Inbound 8 0 0 0 98699.102 10.132 0.000 0.000

Как показывает этот вывод, задержка приема/передачи пакетов TCP составляет 98,7 мс.

10.7.6. tc Утилита управления трафиком tc(8) позволяет выбирать различные дисциплины очередей (qdiscs) для управления производительностью. Для экспериментов есть дисциплины, которые могут ограничивать или случайным образом менять производительность, что может пригодиться для тестирования и моделирования. В этом разделе представлена дисциплина netem (сетевой эмулятор).

700  Глава 10. Сеть Для начала представлю команду, которая отображает текущую дисциплину очередей для интерфейса eth0: # tc qdisc show dev eth0 qdisc noqueue 0: root refcnt 2

Теперь добавим дисциплину netem. Каждая дисциплина имеет свои настраиваемые параметры. В этом примере я использую параметр для дисциплины netem, управляющий количеством теряемых пакетов, и установлю его равным 1 %: # tc qdisc add dev eth0 root netem loss 1% # tc qdisc show dev eth0 qdisc netem 8001: root refcnt 2 limit 1000 loss 1%

После этого операции сетевого ввода/вывода с eth0 будут терять 1 % пакетов. Параметр -s показывает статистики: # tc -s qdisc show dev eth0 qdisc netem 8001: root refcnt 2 limit 1000 loss 1% Sent 75926119 bytes 89538 pkt (dropped 917, overlimits 0 requeues 0) backlog 0b 0p requeues 0

Здесь видно количество отброшенных пакетов. Чтобы удалить дисциплину, достаточно выполнить команду: # tc qdisc del dev eth0 root # tc qdisc show dev eth0 qdisc noqueue 0: root refcnt 2

Полный перечень параметров можно найти на странице справочного руководства man для соответствующей дисциплины (для netem страница называется tc-netem(8)).

10.7.7. Другие инструменты Другие инструменты для экспериментов включают:

yy pktgen: генератор пакетов, включенный в ядро Linux [Linux 20l]; yy Flent: FLExible Network Tester запускает несколько микробенчмарков и выводит результаты в виде графика [Høiland-Jørgensen 20];

yy mtr(8): инструмент, похожий на traceroute, который включает статистики проверки связи (ping);

yy tcpreplay(1): воспроизводит ранее перехваченный (с помощью tcpdump (8))

сетевой трафик, вплоть до интервалов времени между пакетами. Этот инструмент предназначен скорее для общей отладки, чем для тестирования производительности, но иногда проблемы с производительностью возникают только с определенными последовательностями пакетов или битовыми шаблонами, и этот инструмент помогает воспроизвести их.

10.8. Настройка  701

10.8. НАСТРОЙКА Настраиваемые параметры сети обычно уже имеют значения, обеспечивающие высокую производительность. Кроме того, предполагается, что сетевой стек должен динамически реагировать на различные рабочие нагрузки, обеспечивая оптимальную производительность. Прежде чем пробовать изменять настраиваемые параметры, полезно сначала понять, как используется сеть. Этот шаг помогает выявлять ненужную работу и исключать ее, что часто дает гораздо больший выигрыш в производительности. Попробуйте применить методологии определения характеристик рабочей нагрузки и статической настройки производительности с помощью инструментов, описанных в предыдущем разделе. Круг доступных для настройки параметров зависит от версии ОС, поэтому дополнительную информацию ищите в соответствующей документации. Разделы ниже дают общее представление о доступных параметрах и особенностях их настройки. Рассматривайте их как отправную точку для исследований в зависимости от конкретной рабочей нагрузки и среды.

10.8.1. Общесистемные Общесистемные настраиваемые параметры в Linux можно просмотреть и изменить с помощью команды sysctl(8) и сохранить в /etc/sysctl.conf. Их также можно читать и изменять с помощью файловой системы /proc — в каталоге /proc/sys/net. Например, чтобы увидеть текущие настройки TCP, выполните поиск в выводе sysctl(8) по тексту «tcp»: # sysctl -a | grep tcp net.ipv4.tcp_abort_on_overflow = 0 net.ipv4.tcp_adv_win_scale = 1 net.ipv4.tcp_allowed_congestion_control = reno cubic net.ipv4.tcp_app_win = 31 net.ipv4.tcp_autocorking = 1 net.ipv4.tcp_available_congestion_control = reno cubic net.ipv4.tcp_available_ulp = net.ipv4.tcp_base_mss = 1024 net.ipv4.tcp_challenge_ack_limit = 1000 net.ipv4.tcp_comp_sack_delay_ns = 1000000 net.ipv4.tcp_comp_sack_nr = 44 net.ipv4.tcp_congestion_control = cubic net.ipv4.tcp_dsack = 1 [...]

В этом ядре (5.3) 70 параметров, в названиях которых есть символы «tcp», и еще множество параметров в разделе «net», в том числе параметры настройки для IP, Ethernet, маршрутизации и сетевых интерфейсов. Некоторые из них можно настраивать отдельно для каждого сокета. Например, net.ipv4.tcp_congestion_control — это общесистемный алгоритм по умолчанию

702  Глава 10. Сеть управления перегрузкой, который можно назначить для каждого сокета с помощью параметра TCP_CONGESTION (см. раздел 10.8.2 «Параметры сокетов»).

Пример из промышленной среды Ниже показаны настройки, используемые в Netflix для облачных экземпляров [Gregg 19c]. Они устанавливаются в сценарии запуска во время загрузки: net.core.default_qdisc = fq net.core.netdev_max_backlog = 5000 net.core.rmem_max = 16777216 net.core.somaxconn = 1024 net.core.wmem_max = 16777216 net.ipv4.ip_local_port_range = 10240 65535 net.ipv4.tcp_abort_on_overflow = 1 net.ipv4.tcp_congestion_control = bbr net.ipv4.tcp_max_syn_backlog = 8192 net.ipv4.tcp_rmem = 4096 12582912 16777216 net.ipv4.tcp_slow_start_after_idle = 0 net.ipv4.tcp_syn_retries = 2 net.ipv4.tcp_tw_reuse = 1 net.ipv4.tcp_wmem = 4096 12582912 16777216

Здесь перечислены настройки только 14 параметров из доступных, и это лишь пример на определенный момент времени, а не рецепт. В течение 2020 года в Netflix предполагается изменить два из них (параметру net.core.netdev_max_backlog присвоить значение 1000 и параметру net.core.somaxconn — значение 4096)1, но пока эти изменения ожидают нерегрессионного тестирования. Далее мы более конкретно рассмотрим некоторые параметры.

Буферы сокетов и TCP Максимальный размер буфера сокета для всех протоколов, как для чтения (rmem_ max), так и для записи (wmem_max), можно установить с помощью параметров: net.core.rmem_max = 16777216 net.core.wmem_max = 16777216

Значения задаются в байтах. Размер, равный 16 Мбайт или выше, может потребоваться установить для высокоскоростных подключений 10 GbE. Включение автоматической настройки буфера приема для TCP: net.ipv4.tcp_moderate_rcvbuf = 1

Спасибо Даниэлю Боркманну (Daniel Borkmann) за предложения, сделанные во время рецензирования этой книги. Эти новые значения уже используются в Google [Dumazet 17b], [Dumazet 19].

1

10.8. Настройка  703 Установка параметров автоматической настройки для буферов чтения и записи TCP: net.ipv4.tcp_rmem = 4096 87380 16777216 net.ipv4.tcp_wmem = 4096 65536 16777216

Оба параметра имеют три значения: минимальный размер, размер по умолчанию и максимальный размер в байтах. Первоначально автоматически устанавливается размер по умолчанию. Чтобы увеличить пропускную способность TCP, можно попробовать увеличить максимальный размер. Увеличение минимального размера и размера по умолчанию приведет к выделению большего объема памяти для каждого соединения, чего может не потребоваться.

Очередь входящих запросов на соединение TCP Всего есть две очереди. Первая предназначена для полуоткрытых соединений: net.ipv4.tcp_max_syn_backlog = 4096

Вторая — для передачи запросов в accept(2): net.core.somaxconn = 1024

Иногда нужно увеличить значения по умолчанию для обоих параметров, например, до 4096 и 1024 или выше, чтобы система могла лучше справляться со всплесками нагрузки.

Очередь устройства Увеличение длины очереди входящих запросов на соединение для сетевого устройства на каждый процессор: net.core.netdev_max_backlog = 10000

Для сетевых карт 10 GbE значение по умолчанию может потребоваться увеличить до 10 000.

Управление перегрузкой TCP Linux поддерживает возможность выбора алгоритмов управления перегрузкой. Вот список алгоритмов, доступных сейчас: # sysctl net.ipv4.tcp_available_congestion_control net.ipv4.tcp_available_congestion_control = reno cubic

Некоторые из них могут быть доступны, но в текущий момент не загружены. Вот пример добавления алгоритма htcp: # modprobe tcp_htcp # sysctl net.ipv4.tcp_available_congestion_control net.ipv4.tcp_available_congestion_control = reno cubic htcp

704  Глава 10. Сеть Назначить алгоритм текущим можно с помощью параметра: net.ipv4.tcp_congestion_control = cubic

Параметры TCP Вот еще несколько параметров TCP, которые можно изменить: net.ipv4.tcp_sack = 1 net.ipv4.tcp_fack = 1 net.ipv4.tcp_tw_reuse = 1 net.ipv4.tcp_tw_recycle = 0

Расширения SACK и FACK могут улучшить пропускную способность в сетях с высокой задержкой за счет некоторой нагрузки на процессор. Параметр tcp_tw_reuse разрешает повторно использовать сеансы в состоянии TIME_WAIT, когда это кажется безопасным. Это может ускорить создание соединений между двумя хостами, например между веб-сервером и базой данных, без риска исчерпать 16-битный диапазон эфемерных портов с сеансами в состоянии TIME_WAIT. tcp_tw_recycle — еще один параметр, позволяющий повторно использовать сеансы в состоянии TIME_WAIT, хотя и не такой безопасный, как tcp_tw_reuse.

ECN Управлять явными уведомлениями о перегрузке (explicit congestion notification, ECN) можно с помощью параметра: net.ipv4.tcp_ecn = 1

Значение 0 отключает ECN, значение 1 разрешает входящие соединения и требует запрашивать ECN для исходящих соединений, значение 2 разрешает входящие соединения и не требует запрашивать ECN для исходящих. Значение по умолчанию: 2. Есть и параметр net.ipv4.tcp_ecn_fallback со значением по умолчанию 1 (true), который отключает ECN для соединений, если ядро обнаружит, что механизм уведомлений работает некорректно.

Byte Queue Limits Этот алгоритм поддерживает настройку через /sys. Вот как можно просмотреть содержимое управляющих файлов для этого алгоритма (имя каталога, усеченное в этом выводе, у вас будет отличаться): # grep . /sys/devices/pci.../net/ens5/queues/tx-0/byte_queue_limits/limit* /sys/devices/pci.../net/ens5/queues/tx-0/byte_queue_limits/limit:16654 /sys/devices/pci.../net/ens5/queues/tx-0/byte_queue_limits/limit_max:1879048192 /sys/devices/pci.../net/ens5/queues/tx-0/byte_queue_limits/limit_min:0

10.8. Настройка  705 Ограничение для этого интерфейса составляет 16 654 байта и установлено автоматической настройкой. Чтобы ограничить допустимый диапазон, установите параметры limit_ min и limit_max.

Управление ресурсами В системе контрольных групп (cgroups) есть подсистема управления приоритетом сетевого трафика (net_prio), которая позволяет задавать приоритет исходящего сетевого трафика для процессов или групп процессов. С ее помощью можно дать преимущество высокоприоритетному сетевому трафику, например промышленной нагрузке, перед низкоприоритетным, таким как резервное копирование или мониторинг. Есть и подсистема сетевого классификатора (net_cls) для маркировки пакетов, принадлежащих контрольной группе, идентификатором класса: эти идентификаторы могут использоваться дисциплиной очереди для ограничения пакетов или полосы пропускания, а также программами BPF. Программы BPF могут также использовать другую информацию, такую как идентификатор cgroup v2, для получения сведений о контейнерах и могут улучшить масштабируемость за счет переноса классификации, оценки и перемаркировки в обработчик исходящего трафика tc и уменьшения давления на корневую блокировку дисциплины очереди [Fomichev 20]. Дополнительную информацию об управлении ресурсами ищите в подразделе «Сетевой ввод/вывод» в разделе 11.3.3 «Управление ресурсами», в главе 11 «Облачные вычисления».

Дисциплины очередей Дисциплины очередей (qdiscs), описанные в разделе 10.4.3 «Программное обеспечение» и изображенные на рис. 10.8, — это алгоритмы планирования, управления, фильтрации и формирования сетевых пакетов. В разделе 10.7.6 «tc» было показано применение дисциплины netem для обеспечения потери пакетов. Но есть дисциплины, помогающие улучшить производительность разных рабочих нагрузок. Получить список дисциплин, доступных в системе, можно командой: # man -k tc-

Для каждой дисциплины есть своя страница в справочном руководстве man. Дисциплины можно использовать для управления частотой передачи пакетов или шириной полосы пропускания, для установки флагов IP ECN и т. д. Узнать, какая дисциплина используется по умолчанию, можно командой: # sysctl net.core.default_qdisc net.core.default_qdisc = fq_codel

Во многих дистрибутивах Linux уже выполнен переход на использование по умолчанию дисциплины fq_codel, которая в большинстве случаев обеспечивает хорошую производительность.

706  Глава 10. Сеть

Проект Tuned Из-за большого количества доступных настраиваемых параметров работать с ними сложно. Проект Tuned позволяет автоматизировать настройку некоторых из них на основе выбираемых профилей и поддерживает дистрибутивы Linux, включая RHEL, Fedora, Ubuntu и CentOS [Tuned Project 20]. После установки Tuned можно получить список доступных профилей, выполнив команду: # tuned-adm list Доступные профили: [...] - balanced - Общий неспециализированный профиль tuned [...] - network-latency - Оптимизированный для получения предсказуемой производительности за счет увеличения энергопотребления с упором на низкую задержку в сети - network-throughput - Оптимизированный для получения высокой пропускной способности, обычно требуется только на старых процессорах или в сетях 40G+1 [...]

Это неполный список, всего есть 28 профилей. Активировать профиль networklatency, например, можно командой: # tuned-adm profile network-latency

Чтобы узнать, какие настройки определены в этом профиле, можно заглянуть в файл конфигурации в исходных кодах tuned [Škarvada 20]: $ more tuned/profiles/network-latency/tuned.conf [...] [main] summary=Optimize for deterministic performance at the cost of increased power consumption, focused on low latency network performance include=latency-performance [vm] transparent_hugepages=never [sysctl] net.core.busy_read=50 net.core.busy_poll=50 net.ipv4.tcp_fastopen=3 kernel.numa_balancing=0 [bootloader] cmdline_network_latency=skew_tick=1

Обратите внимание на директиву include, которая подключает настройки из профиля latency-performance.

Это перевод вывода, который вы получите. — Примеч. пер.

1

10.8. Настройка  707

10.8.2. Параметры сокетов Сокеты можно настраивать по отдельности внутри приложений с помощью системного вызова setsockopt(2). Но это реально, только если приложение ваше или вам доступен его исходный код и вы можете изменять его1. setsockopt(2) позволяет настраивать разные уровни (например, сокеты, протокол TCP). Некоторые из возможностей, доступных в Linux, перечислены в табл. 10.8. Таблица 10.8. Настраиваемые параметры сокетов Параметр

Описание

SO_SNDBUF, SO_RCVBUF

Размеры буфера передачи и приема (можно указать значения вплоть до системных пределов, описанных выше. Есть и параметр SO_SNDBUFFORCE, настраивающий возможность превышения ограничения на размер буфера передачи)

SO_REUSEPORT

Позволяет нескольким процессам или потокам использовать один и тот же порт, поэтому ядро может распределять нагрузку между ними для масштабируемости (начиная с Linux 3.9)

SO_MAX_PACING_RATE

Позволяет установить максимальную скорость обработки пактов на транспортном уровне в байтах в секунду (см. описание tc-fq(8))

SO_LINGER

Может использоваться для уменьшения времени пребывания сеансов в состоянии TIME_WAIT

SO_TXTIME

Позволяет запрашивать передачу пакетов в определенные интервалы времени (начиная с Linux 4.19) [Corbet 18c] (используется для регулирования UDP [Bruijn 18])

TCP_NODELAY

Отключает алгоритм Нейгла, отправляя сегменты при первой возможности. Это может уменьшить задержку за счет более высокого потребления сети (большего количества пакетов)

TCP_CORK

Приостанавливает передачу до тех пор, пока не будут отправлены полные пакеты, что повышает пропускную способность. (Есть и общесистемная настройка ядра для автоматической приостановки: net.ipv4. tcp_autocorking)

TCP_QUICKACK

Отправляет пакеты ACK немедленно (это может увеличить пропускную способность отправки)

TCP_CONGESTION

Алгоритм управления перегрузкой для сокетов

Описание доступных параметров сокетов ищите на страницах справочного руководства man для socket(7), tcp(7), udp(7) и т. д. Системные вызовы ввода/вывода через сокеты поддерживают некоторые флаги, которые тоже могут влиять на производительность. Например, в Linux 4.14 Есть несколько опасных способов изменить выполняемый двоичный файл, но с моей стороны было бы безответственно приводить их здесь.

1

708  Глава 10. Сеть в системный вызов send(2) добавили поддержку флага MSG_ZEROCOPY: он позволяет использовать во время передачи буфер в пространстве пользователя, чтобы избежать затрат на его копирование в пространство ядра1 [Linux 20c].

10.8.3. Конфигурация Для настройки производительности сети могут быть доступны следующие параметры конфигурации:

yy Jumbo-фреймы Ethernet: увеличение размера MTU до ~9000 может повысить производительность сети, если сетевая инфраструктура поддерживает jumboфреймы.

yy Агрегирование каналов: несколько сетевых интерфейсов можно сгруппировать, чтобы они работали как один интерфейс с объединенной полосой пропускания. Для этого требуется поддержка и настройка коммутатора.

yy Конфигурация брандмауэра: правила iptables или программы BPF, обслужи-

вающие исходящий трафик, можно использовать для установки уровня IP ToS (DSCP) в заголовках IP и, например, задавать приоритеты трафика на основе номера порта или использовать для других целей.

10.9. УПРАЖНЕНИЯ 1. Ответьте на следующие вопросы, касающиеся терминологии: • В чем разница между шириной полосы пропускания и пропускной способностью? • Что такое задержка подключения TCP? • Что такое задержка первого байта? • Что такое время приема/передачи? 2. Ответьте на следующие концептуальные вопросы: • Опишите такие характеристики сетевого интерфейса, как потребление и насыщение. • Что такое очередь установленных соединений TCP и как она используется? • Опишите плюсы и минусы объединения прерываний. 3. Ответьте на следующие, более сложные вопросы: • Объясните, как ошибка сетевого фрейма (или пакета) может повлиять на скорость создания TCP-соединения. 1

При использовании MSG_ZEROCOPY недостаточно просто установить флаг: системный вызов send(2) может вернуть управление еще до того, как данные будут отправлены, поэтому отправляющее приложение должно дождаться уведомления ядра, прежде чем освободить или повторно использовать память, занятую буфером передачи.

10.10. Ссылки  709 • Опишите, что происходит, когда сетевой интерфейс оказывается перегружен, и как происходящее влияет на производительность приложения. 4. Разработайте следующие процедуры для своей ОС: • Чек-лист анализа потребления сетевых ресурсов (интерфейсов и контроллеров) для метода USE. Опишите, как получить каждую метрику (например, какую команду выполнить) и как интерпретировать результаты. Перед установкой или использованием дополнительных программных продуктов постарайтесь ограничиться инструментами наблюдения, которые есть в ОС. • Чек-лист для определения характеристик рабочей нагрузки, потребляющей сетевые ресурсы. Опишите, как получить каждую метрику, и для начала попробуйте ограничиться инструментами наблюдения, имеющимися в ОС. 5. Решите следующие задачи (может потребоваться использовать средства и методы динамической трассировки): • Измерьте задержку первого байта для исходящих (активных) TCPсоединений. • Измерьте задержку подключения TCP. Сценарий должен учитывать неблокирующие вызовы connect(2). 6. (опционально) Измерьте задержку стека TCP/IP для приема и передачи. Для приема — время от прерывания до чтения из сокета; для передачи — от записи в сокет до передачи в устройство. Протестируйте под нагрузкой. Можно ли использовать какую-либо дополнительную информацию, чтобы объяснить причину выбросов по задержке?

10.10. ССЫЛКИ [Postel 80] Postel, J., «RFC 768: User Datagram Protocol», Information Sciences Institute, https:// tools.ietf.org/html/rfc768, 19801. [Postel 81] Postel, J., «RFC 793: Transmission Control Protocol», Information Sciences Institute, https://tools.ietf.org/html/rfc793, 19812. [Nagle 84] Nagle, J., «RFC 896: Congestion Control in IP/TCP Internetworks», https://tools.ietf. org/html/rfc896,19843. [Saltzer 84] Saltzer, J., Reed, D., and Clark, D., «End-to-End Arguments in System Design», ACM TOCS, November 1984. [Braden 89] Braden, R., «RFC 1122: Requirements for Internet Hosts — Communication Layers», https://tools.ietf.org/html/rfc1122, 19894. Перевод на русский язык: https://www.protocols.ru/WP/rfc768/. — Примеч. пер.

1

Перевод на русский язык: https://www.protocols.ru/WP/rfc793/. — Примеч. пер.

2

Перевод на русский язык: https://www.protocols.ru/WP/rfc896/. — Примеч. пер.

3

Перевод на русский язык: https://www.protocols.ru/WP/rfc1122/. — Примеч. пер.

4

710  Глава 10. Сеть [Jacobson 92] Jacobson, V., et al., «TCP Extensions for High Performance», Network Working Group, https://tools.ietf.org/html/rfc1323, 19921. [Stevens 93] Stevens, W. R., TCP/IP Illustrated, Volume 1, Addison-Wesley, 19932. [Mathis 96] Mathis, M., and Mahdavi, J., «Forward Acknowledgement: Refining TCP Congestion Control», ACM SIGCOMM, 1996. [Jacobson 97] Jacobson, V., «pathchar-a1-linux-2.0.30.tar.gz», ftp://ftp.ee.lbl.gov/pathchar, 1997. [Nichols 98] Nichols, K., Blake, S., Baker, F., and Black, D., «Definition of the Differentiated Services Field (DS Field) in the IPv4 and IPv6 Headers», Network Working Group, https:// tools.ietf.org/html/rfc2474, 19983. [Downey 99] Downey, A., «Using pathchar to Estimate Internet Link Characteristics», ACM SIGCOMM, October 1999. [Ramakrishnan 01] Ramakrishnan, K., Floyd, S., and Black, D., «The Addition of Explicit Congestion Notification (ECN) to IP», Network Working Group, https://tools.ietf.org/html/ rfc3168, 20014. [Corbet 03] Corbet, J., «Driver porting: Network drivers», LWN.net, https://lwn.net/Articles/30107, 2003. [Hassan 03] Hassan, M., and R. Jain., High Performance TCP/IP Networking, Prentice Hall, 2003. [Deri 04] Deri, L., «Improving Passive Packet Capture: Beyond Device Polling», Proceedings of SANE, 2004. [Corbet 06b] Corbet, J., «Reworking NAPI», LWN.net, https://lwn.net/Articles/214457, 2006. [Cook 09] Cook, T., «nicstat — the Solaris and Linux Network Monitoring Tool You Did Not Know You Needed», https://blogs.oracle.com/timc/entry/nicstat_the_solaris_and_linux, 2009. [Steenbergen 09] Steenbergen, R., «A Practical Guide to (Correctly) Troubleshooting with Traceroute», https://archive.nanog.org/meetings/nanog47/presentations/Sunday/RAS_Traceroute_ N47_Sun.pdf, 2009. [Paxson 11] Paxson, V., Allman, M., Chu, J., and Sargent, M., «RFC 6298: Computing TCP’s Retransmission Timer», Internet Engineering Task Force (IETF), https://tools.ietf.org/html/ rfc6298, 20115. [Corbet 12] «TCP friends», LWN.net, https://lwn.net/Articles/511254, 2012. [Fritchie 12] Fritchie, S. L., «TCP incast: What is it?», http://www.snookles.com/slf-blog/slfblog/2012/01/05/tcp-incast-what-is-it/, 2012. [Hrubý 12] Hrubý, T., «Byte Queue Limits», Linux Plumber’s Conference, https://blog. linuxplumbersconf.org/2012/wp-content/uploads/2012/08/bql_slide.pdf, 2012. Перевод на русский язык: https://www.protocols.ru/WP/rfc1323/. — Примеч. пер.

1

У. Ричард Стивенс. «Протоколы TCP/IP. Практическое руководство».

2

Перевод на русский язык: https://www.protocols.ru/WP/rfc2474/. — Примеч. пер.

3

Перевод на русский язык: https://www.protocols.ru/WP/rfc3168/. — Примеч. пер.

4

Перевод на русский язык: https://www.protocols.ru/WP/rfc6298/. — Примеч. пер.

5

10.10. Ссылки  711 [Nichols 12] Nichols, K., and Jacobson, V., «Controlling Queue Delay», Communications of the ACM, July 2012. [Roskind 12] Roskind, J., «QUIC: Quick UDP Internet Connections», https://docs.google.com/ document/d/1RNHkx_VvKWyWg6Lr8SZ-saqsQx7rFV-ev2jRFUoVD34/edit#, 2012. [Dukkipati 13] Dukkipati, N., Cardwell, N., Cheng, Y., and Mathis, M., «Tail Loss Probe (TLP): An Algorithm for Fast Recovery of Tail Losses», TCP Maintenance Working Group, https:// tools.ietf.org/html/draft-dukkipati-tcpm-tcp-loss-probe-01, 2013. [Siemon 13] Siemon, D., «Queueing in the Linux Network Stack», https://www.coverfire.com/ articles/queueing-in-the-linux-network-stack, 2013. [Cheng 16] Cheng, Y., and Cardwell, N., «Making Linux TCP Fast», netdev 1.2, https://netdevconf. org/1.2/papers/bbr-netdev-1.2.new.new.pdf, 2016. [Linux 16] «TCP Offload Engine (TOE)», https://wiki.linuxfoundation.org/networking/toe, 2016. [Ather 17] Ather, A., «BBR TCP congestion control offers higher network utilization and throughput during network congestion (packet loss, latencies)», https://twitter.com/amernetflix/ status/892787364598132736, 2017. [Bensley 17] Bensley, S., et al., «Data Center TCP (DCTCP): TCP Congestion Control for Data Centers», Internet Engineering Task Force (IETF), https://tools.ietf.org/html/rfc8257, 2017. [Dumazet 17a] Dumazet, E., «Busy Polling: Past, Present, Future», netdev 2.1, https://netdevconf. info/2.1/slides/apr6/dumazet-BUSY-POLLING-Netdev-2.1.pdf, 2017. [Dumazet 17b] Dumazet, E., «Re: Something hitting my total number of connections to the server», netdev mailing list, https://lore.kernel.org/netdev/1503423863.2499.39.camel@edumazet-glaptop3. roam.corp.google.com, 2017. [Gallatin 17] Gallatin, D., «Serving 100 Gbps from an Open Connect Appliance», Netflix Technology Blog, https://netflixtechblog.com/serving-100-gbps-from-an-open-connect-appliancecdb51dda3b99, 2017. [Bruijn 18] Bruijn, W., and Dumazet, E., «Optimizing UDP for Content Delivery: GSO, Pacing and Zerocopy», Linux Plumber’s Conference, http://vger.kernel.org/lpc_net2018_talks/willemdebruijnlpc2018-udpgso-paper-DRAFT-1.pdf, 2018. [Corbet 18b] Corbet, J., «Zero-copy TCP receive», LWN.net, https://lwn.net/Articles/752188, 2018. [Corbet 18c] Corbet, J., «Time-based packet transmission», LWN.net, https://lwn.net/Articles/748879, 2018. [Deepak 18] Deepak, A., «eBPF / XDP firewall and packet filtering», Linux Plumber’s Conference, http://vger.kernel.org/lpc_net2018_talks/ebpf-firewall-LPC.pdf, 2018. [Jacobson 18] Jacobson, V., «Evolving from AFAP: Teaching NICs about Time», netdev 0x12, July 2018, https://www.files.netdevconf.org/d/4ee0a09788fe49709855/files/?p=/Evolving%20from%20 AFAP%20%E2%80%93%20Teaching%20NICs%20about%20time.pdf, 2018. [Høiland-Jørgensen 18] Høiland-Jørgensen, T., et al., «The eXpress Data Path: Fast Programmable Packet Processing in the Operating System Kernel», Proceedings of the 14th International Conference on emerging Networking EXperiments and Technologies, 2018. [HP 18] «Netperf», https://github.com/HewlettPackard/netperf, 2018.

712  Глава 10. Сеть [Majkowski 18] Majkowski, M., «How to Drop 10 Million Packets per Second», https://blog. cloudflare.com/how-to-drop-10-million-packets, 2018. [Stewart 18] Stewart, R., «This commit brings in a new refactored TCP stack called Rack», https:// reviews.freebsd.org/rS334804, 2018. [Amazon 19] «Announcing Amazon VPC Traffic Mirroring for Amazon EC2 Instances», https://

aws.amazon.com/about-aws/whats-new/2019/06/announcing-amazon-vpc-traffic-mirroring-foramazon-ec2-instances, 2019.

[Dumazet 19] Dumazet, E., «Re: [LKP] [net] 19f92a030c: apachebench.requests_per_second -37.9% regression», netdev mailing list, https://lore.kernel.org/lkml/[email protected], 2019. [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»1, Addison-Wesley, 2019. [Gregg 19b] Gregg, B., «BPF Theremin, Tetris, and Typewriters», http://www.brendangregg.com/ blog/2019-12-22/bpf-theremin.html, 2019. [Gregg 19c] Gregg, B., «LISA2019 Linux Systems Performance», USENIX LISA, http://www. brendangregg.com/blog/2020-03-08/lisa2019-linux-systems-performance.html, 2019. [Gregg 19d] Gregg, B., «udplife.bt», https://github.com/brendangregg/bpf-perf-tools-book/blob/master/ exercises/Ch10_Networking/udplife.bt, 2019. [Hassas Yeganeh 19] Hassas Yeganeh, S., and Cheng, Y., «TCP SO_TIMESTAMPING with OPT_STATS for Performance Analytics», netdev 0x13, https://netdevconf.info/0x13/session. html?talk-tcp-timestamping, 2019. [Bufferbloat 20] «Bufferbloat», https://www.bufferbloat.net, 2020. [Cheng 20] Cheng, Y., Cardwell, N., Dukkipati, N., and Jha, P., «RACK-TLP: A Time-Based Efficient Loss Detection for TCP», TCP Maintenance Working Group, https://tools.ietf.org/ html/draft-ietf-tcpm-rack-09, 2020. [Cilium 20a] «API-aware Networking and Security», https://cilium.io, по состоянию на 2020. [Corbet 20] Corbet, J., «Kernel operations structures in BPF», LWN.net, https://lwn.net/ Articles/811631, 2020. [DPDK 20] «AF_XDP Poll Mode Driver», DPDK documentation, http://doc.dpdk.org/guides/ index.html, по состоянию на 2020. [Fomichev 20] Fomichev, S., et al., «Replacing HTB with EDT and BPF», netdev 0x14, https:// netdevconf.info/0x14/session.html?talk-replacing-HTB-with-EDT-and-BPF, 2020. [Google 20b] «Packet Mirroring Overview», https://cloud.google.com/vpc/docs/packetmirroring, по состоянию на 2020. [Høiland-Jørgensen 20] Høiland-Jørgensen, T., «The FLExible Network Tester», https://flent.org, по состоянию на 2020.

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

10.10. Ссылки  713 [Linux 20i] «Segmentation Offloads», Linux documentation, https://www.kernel.org/doc/ Documentation/networking/segmentation-offloads.rst, по состоянию на 2020. [Linux 20c] «MSG_ZEROCOPY», Linux documentation, https://www.kernel.org/doc/html/latest/ networking/msg_zerocopy.html, по состоянию на 2020. [Linux 20j] «timestamping.txt», Linux documentation, https://www.kernel.org/doc/Documentation/ networking/timestamping.txt, по состоянию на 2020. [Linux 20k] «AF_XDP», Linux documentation, https://www.kernel.org/doc/html/latest/networking/ af_xdp.html, по состоянию на 2020. [Linux 20l] «HOWTO for the Linux Packet Generator», Linux documentation, https://www.kernel. org/doc/html/latest/networking/pktgen.html, по состоянию на 2020. [Nosachev 20] Nosachev, D., «How 1500 Bytes Became the MTU of the Internet», https://blog. benjojo.co.uk/post/why-is-ethernet-mtu-1500, 2020. [Škarvada 20] Škarvada, J., «network-latency/tuned.conf», https://github.com/redhatperformance/ tuned/blob/master/profiles/network-latency/tuned.conf, последнее обновление 2020. [Tuned Project 20] «The Tuned Project», https://tuned-project.org, по состоянию на 2020. [Wireshark 20] «Wireshark», https://www.wireshark.org, по состоянию на 2020.

Глава 11

ОБЛАЧНЫЕ ВЫЧИСЛЕНИЯ

Развитие облачных вычислений решает некоторые старые проблемы производительности, создавая при этом новые. Облачную среду можно создать практически мгновенно и масштабировать по мере необходимости без обычного оверхеда на создание и управление локальным центром обработки данных. Облака также обес­печивают лучшую детализацию для развертываний — разным клиентам могут выделяться разные доли сервера. Но это создает свои проблемы: оверхед на работу технологий виртуализации и конкуренцию за ресурсы между арендаторами. Цели этой главы:

yy познакомить с архитектурой облачных вычислений и ее влиянием на производительность;

yy представить типы виртуализации: оборудования, операционной системы и легковесную виртуализацию оборудования;

yy познакомить с внутренними функциями виртуализации, включая использование агентов (прокси-объектов) ввода/вывода, и с методами настройки;

yy показать на примерах типичный оверхед при различных рабочих нагрузках для каждого типа виртуализации;

yy научить диагностировать проблемы производительности гостевых и хост-систем

и показать, как может меняться набор используемых инструментов в зависимости от типа виртуализации.

Все, о чем говорится в книге, применимо к анализу производительности облачных сред, но именно в этой главе основное внимание уделяется их уникальным особенностям: работе гипервизоров и виртуализации, влиянию средств управления ресурсами на гостевые системы и работе инструментов наблюдения на стороне гостевой и хост-системы. Облачные провайдеры обычно предлагают свои службы и API, которые здесь не рассматриваются: см. документацию с описанием набора сервисов, которые предоставляет каждый провайдер. В главе четыре части:

yy Основы представляет общую архитектуру облачных вычислений и ее влияние на производительность.

11.1. Основы  715

yy Виртуализация оборудования, когда гипервизор управляет несколькими виртуальными машинами с экземплярами гостевых ОС, в каждой из которых выполняется свое ядро с виртуализированными устройствами. В качестве примеров используются гипервизоры Xen, KVM и Amazon Nitro.

yy Виртуализация операционной системы, когда системой управляет единое ядро

и создает виртуальные экземпляры ОС, изолированные друг от друга. В качестве примеров используются контейнеры Linux.

yy Легковесная виртуализация оборудования вобрала в себя лучшее из обоих

подходов: легковесные виртуализированные экземпляры оборудования запускаются с собственными ядрами и при этом сохраняются преимущества быстрой загрузки и плотности размещения, аналогичные контейнерам. В качестве примера гипервизора используется AWS Firecracker.

Разделы, посвященные виртуализации, упорядочены по времени широкого распространения этих технологий. Например, Amazon Elastic Compute Cloud (EC2) предлагал экземпляры виртуального оборудования еще в 2006 году, контейнеры с виртуальными ОС — в 2017 году (Amazon Fargate), а легковесные виртуальные машины — в 2019 году (Amazon Firecracker).

11.1. ОСНОВЫ Облачные вычисления позволяют предоставлять вычислительные ресурсы как услугу и масштабировать их от небольшой доли одного сервера до систем со множеством серверов. Устройство облака зависит от выбора части программного стека для установки и настройки. В этой главе основное внимание уделяется облачным предложениям, предоставляющим экземпляры серверов, которыми могут быть:

yy Экземпляры оборудования (иначе называются инфраструктура как услуга —

infrastructure as a service, IaaS): предоставляются с использованием виртуализации оборудования. Каждый экземпляр сервера — это виртуальная машина.

yy Экземпляры операционной системы: предоставляются легковесные экземпляры, обычно посредством виртуализации ОС.

Вместе они могут называться экземплярами серверов, облачными экземплярами или просто экземплярами. В числе облачных провайдеров, которые их поддерживают, можно назвать Amazon Web Services (AWS), Microsoft Azure и Google Cloud Platform (GCP). Есть и другие типы облачных примитивов, включая «функции как услуга» (functions as a service, FaaS; см. раздел 11.5 «Другие типы»). В общем случае термин облачные вычисления описывает структуру динамического выделения ресурсов для экземпляров. В одной физической хост-системе действует один или несколько экземпляров гостевых систем. Иногда гостевые системы называют арендаторами (tenants), а если одна хост-система служит для нужд нескольких арендаторов, то используют термин мультиарендность (multitenancy). Хост может находиться под управлением внешнего провайдера, поддерживающего

716  Глава 11. Облачные вычисления публичное облако, или вашей компании, поддерживающей частное облако только для внутреннего пользования. Некоторые компании создают гибридное облако, охватывающее общедоступное и частное облака1. Гостевые (арендуемые) системы в облаке управляются их конечными пользователями. Для виртуализации оборудования используется технология под названием гипервизор (или монитор виртуальных машин — virtual machine monitor, VMM). Она позволяет создавать и управлять экземплярами виртуальных машин, которые извне видны как отдельные компьютеры, и устанавливать полноценные ОС и ядра. Обычно экземпляры могут создаваться (и уничтожаться) за минуты или даже секунды и немедленно запускаться в эксплуатацию. Нередко предлагаемый облачный API позволяет автоматизировать подготовку экземпляров с помощью другой программы. Для обсуждения различных вопросов, связанных с производительностью, важно иметь некоторое представление о типах экземпляров, архитектуре, планировании мощности, организации хранилищ и мультиарендности. Кратко об этом в следующих разделах.

11.1.1. Типы экземпляров Облачные провайдеры часто предлагают экземпляры разных типов и размеров. Некоторые типы экземпляров универсальны и сбалансированы по ресурсам, другие могут быть оптимизированы по определенному ресурсу: памяти, процессорам, дискам и т. д. Например, AWS объединяет типы в семейства (обозначаемые буквами) и поколения (обозначаемые числами). Сейчас предлагаются следующие экземпляры:

yy yy yy yy yy

m5: общего назначения (сбалансированные); c5: оптимизированные для вычислительных задач; i3, d2: оптимизированные для хранения; r4, x1: оптимизированные по объему памяти; p1, g3, f1: с поддержкой ускорения вычислений (графические процессоры, FPGA и т. д.).

В каждом семействе есть экземпляры разных размеров. Семейство m5 в AWS, например, включает экземпляры от m5.large (2 виртуальных процессора и 8 Гбайт основной памяти) до m5.24xlarge (96 виртуальных процессоров и 384 Гбайт основной памяти). Обычно соотношение цена/производительность остается неизменным для экземпляров всех размеров — это позволяет клиентам выбирать экземпляры, лучше подходящие для их рабочей нагрузки. Например, Google Anthos — это платформа управления приложениями, которая поддерживает локальный вариант движка Google Kubernetes Engine (GKE) с экземплярами GCP, а также другие облака.

1

11.1. Основы  717 Google Cloud Platform, например, дополнительно предлагает настраиваемые экземпляры, позволяя клиентам самим определять количество доступных ресурсов. Благодаря такому разнообразию вариантов и простоте повторного развертывания, тип экземпляра стал больше похожим на настраиваемый параметр, который можно изменять по мере необходимости. Это значительное усовершенствование по сравнению с традиционной корпоративной моделью выбора и заказа физического оборудования, которое компания может не менять годами.

11.1.2. Масштабируемая архитектура Чтобы справиться со все возрастающей нагрузкой, в корпоративных средах исторически использовалось вертикальное масштабирование, заключающееся в создании все более крупных и мощных систем (мейнфреймов). Этот подход имеет свои ограничения. Есть реальный предел физического размера компьютеров (он может ограничиваться размерами дверей лифта или транспортных контейнеров), а кроме того, по мере увеличения процессоров возникает все больше трудностей с синхронизацией их кэшей, а также с электропитанием и охлаждением. Обойти эти ограничения позволяет распределение нагрузки между несколькими (возможно, небольшими) системами. Это называется горизонтальным масштабированием. На предприятиях этот подход использовался для создания компьютерных ферм и кластеров, обеспечивавших, в частности, высокопроизводительные вычисления (предшествовавшие облачным вычислениям). Облачные вычисления тоже основаны на идее горизонтального масштабирования. На рис. 11.1 показан пример такой среды, включающей балансировщики нагрузки, веб-серверы, серверы приложений и базы данных. На каждом уровне среды есть один или несколько экземпляров серверов, работающих параллельно, и их число увеличивается с добавлением нагрузки. Архитектура может поддерживать возможность добавления экземпляров в каждый слой независимо либо делиться на вертикальные сегменты, каждый из которых состоит из серверов баз данных, серверов приложений и веб-серверов и добавляется как единое целое1. Наибольшую сложность в этой модели представляет развертывание традиционных серверов баз данных, из которых один экземпляр должен быть главным. Данные для этих БД, например MySQL, можно логически разделить на группы, которые называются шардами (shards), каждая из которых управляется своим собственным сервером БД (или парой серверов главный/вторичный). Архитектуры распределенных баз данных, например Riak, могут динамически распределять нагрузку между доступным экземплярами. Сейчас есть полноценные облачные базы данных, предназначенные для использования в облачных средах, включая Cassandra, CockroachDB, Amazon Aurora и Amazon DynamoDB.

В Shopify, например, эти сегменты называются «подами» [Denis 18].

1

718  Глава 11. Облачные вычисления

Интернет Балансировщики нагрузки

Веб-серверы

Серверы приложений

Масштабирование

Серверы баз данных Хранилище

Рис. 11.1. Архитектура облака: горизонтальное масштабирование Поскольку размер экземпляра обычно невелик, например 8 Гбайт (на физических хостах с 512 Гбайт DRAM и более), масштабирование можно делать небольшими приращениями для достижения оптимального соотношения цена/производительность, а не заранее вкладывать деньги в огромные системы, которые долгое время могут оставаться практически простаивающими.

11.1.3. Планирование мощности Покупка отдельных серверов может потребовать значительных затрат на инфраструктуру и оплату многолетних контрактов на обслуживание. Кроме того, запуск нового сервера в промышленную эксплуатацию может занять месяцы, например, из-за необходимости согласования, ожидания появления деталей, доставки, установки, сборки и тестирования. Планирование мощностей — критически важный аспект в этом подходе, потому что помогает приобретать системы подходящего размера: слишком маленькие системы могут не справиться с возрастающей нагрузкой, а слишком большие стоят неоправданно дорого (а с учетом контрактов на обслуживание система может не окупиться за многие годы). Планирование мощности нужно и для прогнозирования роста потребностей, чтобы можно было вовремя завершить длительные процедуры закупок. Отдельные серверы и вычислительные центры были нормой для корпоративных сред. Облачные вычисления — это совсем другое. Экземпляры серверов обходятся недорого и могут создаваться и уничтожаться практически мгновенно. Вместо того чтобы тратить время на планирование будущих потребностей, компании могут просто увеличивать количество используемых экземпляров по мере необходимости. Это можно делать автоматически через облачный API, опираясь на значения

11.1. Основы  719 метрик, которые собирает ПО мониторинга производительности. Малый бизнес или стартап может вырасти из одного небольшого экземпляра до нескольких тысяч, не затратив ни копейки на подробные исследования с целью планирования емкости1. Для растущих стартапов еще одним важным фактором является скорость изменения кода. Обычно сайты обновляют свой продакшен-код еженедельно, ежедневно или даже несколько раз в день. Исследования для планирования мощности могут занять недели, а поскольку результаты таких исследований основаны на моментальном снимке метрик производительности, они могут устареть к моменту их завершения. Этим они отличаются от корпоративных сред с коммерческим ПО, которое может меняться не чаще нескольких раз в год. Вот что делается в облаке для планирования мощности:

yy динамическое изменение размера: автоматическое добавление и удаление экземпляров серверов;

yy тестирование масштабируемости: приобретение значительных облачных мощно-

стей на короткий срок, чтобы протестировать масштабируемость с применением синтетической нагрузки (фактически это бенчмаркинг).

Принимая во внимание временные ограничения, можно также смоделировать масштабируемость (аналогично исследованиям на предприятиях), чтобы оценить, насколько фактическая масштабируемость отстает от теоретической.

Динамическое изменение размера (автомасштабирование) Облачные провайдеры обычно поддерживают развертывание групп экземпляров серверов, которые могут автоматически масштабироваться с увеличением нагрузки (например, группа автоматического масштабирования AWS — auto scaling group, ASG). Они также поддерживают архитектуру микросервисов, когда приложение разделено на небольшие фрагменты, взаимодействующие между собой по сети, и их можно масштабировать индивидуально. Автомасштабирование решает проблему быстрого реагирования на изменения нагрузки, но при этом может привести к избыточному выделению ресурсов, как показано на рис. 11.2. Например, увеличение нагрузки может быть вызвано DoSатакой и привести к выделению избыточного количества дорогостоящих экземпляров сервера. Аналогичный риск несут изменения в приложениях, снижающие производительность и приводящие к необходимости выделения большего числа экземпляров для обслуживания той же нагрузки. Важно постоянно делать мониторинг, чтобы понимать, что такое масштабирование имеет смысл. Ситуация может усложниться, когда потребности компании вырастут до сотен тысяч экземпляров, потому что из-за высокого спроса у облачных провайдеров могут временно закончиться доступные экземпляры. Если вы дорастете до такого масштаба, обсудите с представителем вашего провайдера способы смягчения ситуации (например, покупку резервных мощностей).

1

720  Глава 11. Облачные вычисления

избыточные мощности

Автоматическое добавление экземпляров

Масштаб

нагрузка ресурсы

Время

Рис. 11.2. Динамическое изменение размера Облачные провайдеры выставляют счет по часам, минутам или даже секундам, что позволяет пользователям быстро увеличивать и уменьшать масштаб. Стоимость услуги снижается одновременно с уменьшением количества используемых экземпляров. Это снижение можно автоматизировать, выделяя достаточное количество экземпляров для каждой минуты в соответствии с шаблоном изменения рабочей нагрузки в течение дня1. В Netflix реализовали такой подход в своем облаке, добавляя и удаляя десятки тысяч экземпляров ежедневно в соответствии со своим посекундным шаблоном изменения рабочей нагрузки, пример которого показан на рис. 11.3 [Gregg 14b].

Рис. 11.3. Суточная закономерность изменения рабочей нагрузки в Netflix Вот еще несколько примеров. В декабре 2012 года Pinterest сообщила о сокращении удельных расходов с $54 в час до $20 в час за счет автоматического отключения своих облачных систем в нерабочее время в ответ на уменьшение трафика [Hoff 12]. В 2018 году Shopify отметила большую экономию на инфраструктуре за счет перехода в облако: вместо серверов со средним временем простоя 61 % они начали использовать облачные экземпляры со средним временем простоя 19 % [Kwiatkowski 19]. Немедленная экономия также может быть результатом настрой1

Обратите внимание, что автоматизация увеличения и уменьшения вычислительной мощности может стать сложной задачей. Масштабирование вниз может включать ожидание не только завершения обработки текущих запросов, но и завершения длительных пакетных заданий и передачи локальных данных между базами данных.

11.1. Основы  721 ки производительности, когда для обработки той же нагрузки требуется меньшее количество экземпляров. Некоторые облачные архитектуры (см. раздел 11.3 «Виртуализация операционной системы») могут динамически выделять больше процессоров, если они доступны, используя стратегию увеличения доступной вычислительной мощности (bursting). Эта стратегия не требует дополнительных затрат и предназначена для предотвращения избыточного выделения ресурсов: резервная вычислительная мощность предоставляется на период, за который можно определить факт увеличения нагрузки и проверить, реальное ли это увеличение и продолжится ли нарастание нагрузки в будущем. В этом случае можно подготовить дополнительные экземпляры, чтобы гарантировать достаточность ресурсов в будущем. Любой из этих методов обычно намного эффективнее, чем организация корпоративной среды, особенно с фиксированным размером, выбранным для обработки ожидаемой пиковой нагрузки в течение всего срока службы сервера: такие серверы могут большую часть времени работать в режиме ожидания.

11.1.4. Хранилище Облачному экземпляру требуется хранилище для ОС, прикладного ПО и временных файлов. В системах Linux это корневой и другие тома. Хранилище может быть локальным физическим или сетевым. Хранилища экземпляров являются временными и уничтожаются вместе со своими экземплярами (поэтому они называются эфемерными дисками). В качестве постоянного хранилища обычно используется независимый сервис, который предоставляет экземплярам такие хранилища, как:

yy хранилища файлов: например, через NFS; yy блочные хранилища: например, через iSCSI; yy хранилища объектов: через API, обычно на основе HTTP. Они работают в сети, а сетевая инфраструктура и устройства хранения совместно используются несколькими арендаторами. По этим причинам производительность хранилища может быть менее предсказуемой, чем производительность локальных дисков, но с другой стороны, облачный провайдер может улучшить постоянство производительности за счет использования средств управления ресурсами. Облачные провайдеры часто предлагают собственные услуги хранилищ. Например, Amazon предлагает для хранения файлов Amazon Elastic File System (EFS), блочное хранилище Amazon Elastic Block Store (EBS) и хранилище объектов Amazon Simple Storage Service (S3). На рис. 11.4 изображены два вида хранилищ — локальное и сетевое. Увеличенную задержку, характерную для сетевых хранилищ, обычно можно уменьшить за счет хранения часто используемых данных в кэше памяти.

722  Глава 11. Облачные вычисления

Сетевое хранилище (постоянное)

NFS/ iSCSI/ API

Хост Гость

Частная сеть

Гость

Интернет

Локальное хранилище

Рис. 11.4. Облачное хранилище Некоторые сервисы хранения позволяют покупать гарантированное быстродействие IOPS, когда требуется постоянная производительность (например, том Amazon EBS Provisioned IOPS).

11.1.5. Мультиарендность Unix — многозадачная операционная система, поддерживающая одновременную работу с несколькими пользователями и процессами, которые обращаются к одним и тем же ресурсам. Позднее в Linux появилась возможность ограничивать потребление ресурсов и средства управления справедливым их распределением, а также инструменты наблюдения для выявления и количественной оценки проблем производительности, связанных с конкуренцией за ресурсы. Облачные вычисления отличаются тем, что все экземпляры ОС могут сосуществовать в одной физической системе. Каждая гостевая система — это отдельная, изолированная ОС: гостевые системы (обычно1) не имеют возможности наблюдать за пользователями и процессами других гостевых систем на том же хосте, потому что это будет считаться утечкой информации, даже если они используют одни и те же физические ресурсы. Поскольку ресурсы распределяются между арендаторами, проблемы с производительностью могут быть вызваны так называемыми шумными соседями. Например, другая гостевая система на том же хосте может производить резервное копирование объемной базы данных в то же самое время, когда ваша система испытывает пиковую нагрузку, препятствуя вашим операциям с дисками и сетью. Хуже того, сосед может оценивать облачного провайдера, выполняя микробенчмаркинг и намеренно насыщая ресурсы, чтобы определить их предел. 1

Контейнеры Linux могут собираться разными способами из пространств имен и контрольных групп. Также можно создавать контейнеры, совместно использующие пространство имен процессов, что может использоваться для создания контейнеров-прицепов («sidecar»), которые могут отлаживать процессы в другом контейнере. В Kubernetes основной абстракцией является под (Pod) — группа контейнеров, разделяющих общее сетевое пространство имен.

11.1. Основы  723 Есть несколько вариантов решения подобных проблем. Эффекты мультиарендности можно контролировать с помощью средств управления ресурсами: настроек, управляющих распределением ресурсов ОС, которые обеспечивают изоляцию производительности (также называемую изоляцией ресурсов). В этом случае для каждого арендатора устанавливаются ограничения или приоритеты на использование системных ресурсов: процессоров, памяти, дискового или файлового ввода/ вывода, а также пропускной способности сети. Кроме того, команда, обслуживающая облако, может наблюдать за конкуренцией арендаторов и настраивать ограничения, чтобы точнее сбалансировать распределение имеющихся ресурсов. Степень наблюдаемости зависит от типа виртуализации.

11.1.6. Оркестровка (Kubernetes) Многие компании создают свои частные облака, используя программное обеспечение оркестровки, работающее на их собственном железе или в облачных системах. Самым популярным таким ПО является Kubernetes (сокращенно K8s), изначально созданное в Google. Kubernetes, по-гречески «рулевой», — это система с открытым исходным кодом, управляющая развертыванием приложений с использованием контейнеров (обычно контейнеров Docker, но вообще может использоваться любая среда выполнения, реализующая интерфейс Open Container, например containerd) [Kubernetes 20b]. Общедоступные облачные провайдеры тоже предлагают поддержку Kubernetes для упрощения развертывания контейнеров, в том числе Google Kubernetes Engine (GKE), Amazon Elastic Kubernetes Service (Amazon EKS) и Microsoft Azure Kubernetes Service (AKS). Kubernetes развертывает контейнеры в виде объединенных групп — подов (Pods), в которых контейнеры могут совместно использовать ресурсы и взаимодействовать друг с другом локально. Каждому поду присваивается собственный IP-адрес, который можно использовать для взаимодействий (через сеть) с другими подами. Сервис в Kubernetes — это абстракция конечных точек, предоставляющая группу подов с метаданными, включая IP-адрес, которая служит постоянным и стабильным интерфейсом для этих конечных точек, в то время как сами поды можно добавлять и удалять, что позволяет считать их одноразовыми. Сервисы Kubernetes поддерживают архитектуру микросервисов. Kubernetes реализует стратегии автоматического масштабирования, такие как «Horizontal Pod Autoscaler», способные создавать дополнительные копии подов, исходя из уровня потребления ресурсов или других метрик. В Kubernetes физические машины называются узлами, а группы узлов объединяются в кластеры Kubernetes, если они подключены к одному и тому же серверу Kubernetes API. Проблемы производительности в Kubernetes обычно обусловлены сложностями планирования (выбора узла в кластере с максимальной производительностью для запуска контейнера) и оверхедом на работу сети из-за использования дополнительных компонентов, реализующих сети контейнеров и осуществляющих балансировку нагрузки.

724  Глава 11. Облачные вычисления При планировании Kubernetes учитывает потребление и ограничения процессоров и памяти, а также метаданные, такие как непригодность узлов (node taints, когда узлы отмечаются как непригодные для планирования) и селекторы меток (настраиваемые метаданные). Сейчас Kubernetes не ограничивает блочный ввод/вывод (но такая поддержка, с помощью контрольной группы blkio, может быть добавлена в будущем [Xu 20]), что делает конкуренцию за диск еще одним возможным источником проблем с производительностью. Для сетевых взаимодействий Kubernetes предлагает различные сетевые компоненты, поэтому определение наиболее подходящих для использования — важный шаг в обеспечении максимальной производительности. Сеть контейнеров может быть реализована с помощью подключаемого сетевого интерфейса контейнера (container network interface, CNI). Пример такого программного интерфейса — Calico, основанный на netfilter или iptables, и Cilium, основанный на BPF. Оба имеют открытый исходный код [Calico 20], [Cilium 20b]. Для балансировки нагрузки Cilium предлагает замену kube-proxy на основе BPF [Borkmann 19].

11.2. ВИРТУАЛИЗАЦИЯ ОБОРУДОВАНИЯ При виртуализации оборудования создается виртуальная машина (virtual machine, VM), на которой может работать ОС, включая ее собственное ядро. Виртуальные машины создаются гипервизорами, которые также называют диспетчерами виртуальных машин (virtual machine manager, VMM). Распространенная классификация гипервизоров делит их на два типа: тип 1 и тип 2 [Goldberg 73]:

yy Тип 1 выполняется непосредственно на процессорах. Администрирование

гипервизора может производиться привилегированной гостевой системой, которая создает и запускает новые гостевые системы. Тип 1 также называется низкоуровневым (native) гипервизором или гипервизором на железе (bare-metal). Этот гипервизор имеет собственный планировщик процессоров для гостевых виртуальных машин. Популярный пример — гипервизор Xen.

yy Тип 2 выполняется внутри хост-системы, обладающей привилегиями на администрирование гипервизора и запуск новых гостевых систем. В этом случае загружается обычная операционная система, которая затем запускает гипервизор. Этот гипервизор управляется планировщиком процессоров в ядре хост-системы, а гостевые виртуальные машины обслуживаются как процессы хост-системы.

Сейчас все еще можно встретить термины тип 1 и тип 2, но с развитием технологии гипервизоров эта классификация больше неприменима в строгом смысле [Liguori 07]: за счет использования модулей ядра, благодаря которым гипервизор получил прямой доступ к оборудованию, тип 2 практически превратился в тип 1. На рис. 11.5 показана классификация, более близкая к реальности, где представлены две типичные конфигурации, которые я назвал конфигурацией A и конфигурацией B [Gregg 19].

11.2. Виртуализация оборудования  725

Конфигурация A

Конфигурация B

Гостевая ОС #0

Гостевая ОС

Гостевая ОС

Хостадминистратор

Приложения

Приложения

Гостевое ядро

Гостевое ядро

Планировщик

Гипервизор

Хост-система

Гостевая ОС

Гостевая ОС

Приложения

Приложения

Гостевое ядро

Гостевое ядро

Гостевое ядро

Оборудование (процессоры)

Гипервизор Планировщик

Хост-ядро

Модуль поддержки гипервизора

Оборудование (процессоры)

Рис. 11.5. Типичные конфигурации гипервизоров Вот краткое описание этих конфигураций:

yy Конфигурация A: также называется низкоуровневым гипервизором или

гипервизором на железе. ПО гипервизора выполняется непосредственно на процессорах, создает домены для запуска гостевых виртуальных машин и распределяет виртуальные гостевые процессоры между физическими процессорами. Привилегированный домен (с номером 0 на рис. 11.5) может использоваться для администрирования остальных доменов. Популярный пример — гипервизор Xen.

yy Конфигурация B: программное обеспечение гипервизора выполняется ядром

хост-системы и может состоять из модулей ядра и процессов пользователя. Хост-система обладает привилегиями, необходимыми для администрирования гипервизора, а ее ядро распределяет процессорное время между виртуальными машинами и другими процессами в хост-системе. Благодаря использованию модулей ядра эта конфигурация обеспечивает прямой доступ к оборудованию. Популярный пример — гипервизор KVM.

Обе конфигурации могут предусматривать запуск агентов ввода/вывода (например, программного обеспечения QEMU) в домене 0 (Xen) или в хост-системе (KVM) для обслуживания гостевого ввода/вывода. Это решение увеличивает оверхед на ввод/вывод, но с годами оно было оптимизировано за счет добавления транспортов с разделяемой памятью и других технологий. Первоначальный гипервизор оборудования, впервые предложенный компанией VMware в 1998 году, использовал двоичную трансляцию для полной виртуализации оборудования [VMware 07], включая переадресацию привилегированных инструкций, таких как системные вызовы и операции с таблицами страниц перед выполнением. Непривилегированные инструкции могут выполняться непосредственно на процессоре. Такой подход позволил создать полноценную виртуальную систему, состоящую из

726  Глава 11. Облачные вычисления виртуализированных аппаратных компонентов, на которую можно было установить немодифицированную ОС. Присущий ему оверхед часто оказывался вполне приемлемым с учетом экономии, которую давала консолидация серверов. С тех пор были добавлены следующие улучшения:

yy Поддержка виртуализации процессоров: в 2005–2006 годах были добавлены

расширения AMD-V и Intel VT-x, обеспечившие ускорение процессорных операций на виртуальной машине. Эти расширения повысили скорость выполнения виртуализированных привилегированных инструкций и работу модуля управления памятью (MMU).

yy Паравиртуализация (paravirtualization, PV): предлагает виртуальную систему, ко-

торая включает интерфейс для гостевых ОС с целью эффективного использования ресурсов хоста (через гипервызовы) без полной виртуализации всех компонентов. Например, включение таймера обычно предполагает выполнение нескольких привилегированных инструкций, которые должен эмулировать гипервизор. Эти инструкции могут объединяться паравиртуализированной гостевой системой в один гипервызов, что позволяет гипервизору более эффективно обрабатывать его. Для достижения еще большей эффективности гипервизор Xen группирует гипервызовы в мультивызовы (multicall). Паравиртуализация может допускать использование гостевой системой драйвера паравиртуального сетевого устройства для эффективной передачи пакетов физическим сетевым интерфейсам. В общем случае производительность увеличивается, но многое зависит от поддержки паравиртуализации гостевой системой (которой, например, традиционно нет в Windows).

yy Аппаратная поддержка устройств: для еще большей оптимизации производи-

тельности VM в аппаратные устройства, кроме процессоров, была добавлена поддержка виртуальных машин. Сюда входит виртуализация ввода/вывода с одним корнем (SR-IOV) для сетевых устройств и устройств хранения, что позволяет гостевым виртуальным машинам напрямую обращаться к оборудованию. Для этого требуется поддержка драйверов (например, ixgbe, ena, hv_netvsc и nvme).

С годами гипервизор Xen развился и улучшил производительность. Современные виртуальные машины Xen часто загружаются в режиме виртуализации оборудования (hardware VM, HVM), а затем используют драйверы PV с поддержкой HVM для повышения производительности: эта конфигурация получила название PVHVM. Ее можно усовершенствовать еще больше, если использовать полную аппаратную виртуализацию для некоторых драйверов, например SR-IOV, сетевых устройств и устройств хранения.

11.2.1. Реализация Есть множество разных реализаций виртуализации оборудования, часть из которых уже упоминались выше (Xen и KVM). Например:

yy VMware ESX (выпущен в 2001 году) — это корпоративный продукт для кон-

солидации серверов и ключевой компонент VMware vSphere, продукта для

11.2. Виртуализация оборудования  727 организации облачных вычислений. Его гипервизор представляет собой микроядро, работающее на «голом железе», а первая виртуальная машина называется служебной консолью, которая может управлять гипервизором и новыми виртуальными машинами.

yy Xen (выпущен в 2003 году) начинался как исследовательский проект в Кем-

бриджском университете, а затем был приобретен компанией Citrix. Xen — это гипервизор типа 1, который запускает паравиртуализированные гостевые системы для обеспечения высокой производительности. Позднее была добавлена поддержка немодифицированных гостевых систем (Windows). Виртуальные машины называются доменами, наиболее привилегированный из них — dom0, из которого производится администрирование гипервизора и запуск новых доменов. Xen распространяется с открытым исходным кодом и может запускаться из Linux. Ранее на Xen было основано вычислительное облако Amazon Elastic Compute Cloud (EC2).

yy Hyper-V (появился вместе с Windows Server 2008) — это гипервизор типа 1,

который создает разделы для запуска гостевых ОС. В общедоступном облаке Microsoft Azure есть возможность запускать свои версии Hyper-V (но детали этой возможности не раскрываются широкой публике).

yy KVM (был разработан стартапом Qumranet, купленным Red Hat в 2008 году) —

это гипервизор типа 2, работающий как модуль ядра. Поддерживает аппаратные расширения для обеспечения высокой производительности и использует паравиртуализацию для некоторых устройств, поддерживаемых гостевой системой. Чтобы создать полноценный экземпляр виртуальной машины с аппаратной поддержкой, он связывается с пользовательским процессом QEMU (Quick Emulator) — гипервизором, который может создавать виртуальные машины и управлять ими. QEMU изначально был высококачественным гипервизором типа 2 с открытым исходным кодом, который использовал двоичную трансляцию; он был написан Фабрисом Белларом (Fabrice Bellard). KVM распространяется с открытым исходным кодом и используется компанией Google в Google Compute Engine [Google 20c].

yy Nitro (создан компанией AWS в 2017 году). Этот гипервизор использует ком-

поненты на основе KVM с аппаратной поддержкой всех основных ресурсов: процессоров, сетевых устройств, устройств хранения, прерываний и таймеров [Gregg 17e]. Он не использует прокси-сервер QEMU. Nitro обеспечивает производительность гостевых виртуальных машин практически на уровне голого железа.

В разделах ниже рассмотрены вопросы производительности, связанные с виртуализацией оборудования: оверхед, управление ресурсами и наблюдаемость. Они различаются в зависимости от реализации и конфигурации.

11.2.2. Оверхед При исследовании проблем с производительностью облачных сред важно понимать, какой оверхед можно ожидать от виртуализации.

728  Глава 11. Облачные вычисления Виртуализация оборудования производится по-разному. Для доступа к ресурсам может потребоваться проксирование и преобразование со стороны гипервизора с присущим ему оверхедом, но также могут использоваться аппаратные технологии, позволяющие избежать оверхеда. В разделах ниже в общих чертах описывается оверхед, обусловленный выполнением на процессоре, отображением памяти, размером памяти, операциями ввода/вывода и конкуренцией со стороны других арендаторов.

Процессор Как правило, гостевые приложения выполняются непосредственно на процессорах, поэтому преимущественно вычислительные приложения могут иметь практически такую же производительность, что и в обычной системе. Оверхед может возникать при выполнении привилегированных машинных инструкций, обращении к оборудованию и отображении основной памяти в зависимости от того, как эти операции обрабатываются гипервизором. Ниже описано, как обрабатываются машинные инструкции различными типами механизмов виртуализации оборудования:

yy Двоичная трансляция: гипервизор идентифицирует и транслирует инструкции,

которые выполняются гостевым ядром и работают с физическими ресурсами. Двоичная трансляция применялась до появления виртуализации с аппаратной поддержкой. Схема виртуализации без аппаратной поддержки, используемая в VMware, включала запуск монитора виртуальных машин (virtual machine monitor, VMM) в кольце 0 процессора и перемещение гостевого ядра в кольцо 1, которое ранее не использовалось (приложения выполняются в кольце 3, а большинство процессоров поддерживают четыре кольца. О кольцах защиты шла речь в главе 3 «Операционные системы», в разделе 3.2.2 «Ядро и пользовательские режимы»). Поскольку некоторые инструкции гостевого ядра предполагают выполнение в кольце 0, то для выполнения из кольца 1 их необходимо транслировать в вызовы VMM, чтобы применить виртуализацию. Эта трансляция выполняется во время выполнения и требует значительного оверхеда.

yy Паравиртуализация: инструкции в гостевой ОС, которые необходимо вирту-

ализировать, заменяются гипервызовами к гипервизору. Производительность можно повысить, если модифицировать гостевую ОС для оптимизации гипервызовов, чтобы она знала, что работает на виртуализированном оборудовании.

yy Аппаратная поддержка: немодифицированные инструкции гостевого ядра,

обращающиеся к оборудованию, обрабатываются гипервизором, который использует монитор VMM на уровне ниже кольца 0. Вместо двоичной трансляции привилегированные инструкции гостевого ядра перехватываются более привилегированным монитором VMM, который затем может эмулировать привилегии для поддержки виртуализации [Adams 06].

Виртуализация с аппаратной поддержкой обычно предпочтительнее, в зависимости от реализации и рабочей нагрузки, но для увеличения производительности некоторых рабочих нагрузок (особенно ввода/вывода) можно использовать паравиртуализацию, если гостевая система ее поддерживает.

11.2. Виртуализация оборудования  729 В качестве примера различий в реализации приведу модель двоичной трансляции в VMware, которая оптимизировалась на протяжении многих лет. Вот как о ней писали в 2007 году [VMware 07]: Из-за высокого оверхеда на переходы между гипервизором и гостевой системой и жесткой модели программирования подход к двоичной трансляции в VMware сейчас превосходит большинство аппаратных реализаций первого поколения. Жесткая модель программирования в реализациях первого поколения оставляет мало места для гибкости в управлении частотой или оверхедом на переходы между гипервизором и гостевыми системами. Частоту переходов между гостевой системой и гипервизором, а также время, проведенное в гипервизоре, можно интерпретировать как метрику, характеризующую оверхед. Эти события обычно называют гостевыми выходами, так как виртуальный процессор должен прекратить выполнение гостевого кода, когда происходит переход. На рис. 11.6 показан оверхед, связанный с гостевыми выходами внутри KVM. Рисунок представляет поток гостевых выходов между пользовательским процессом, хост-ядром и гостевой системой. Время, потраченное на обработку гостевых выходов, — это оверхед при виртуализации оборудования. Чем больше времени тратится на обработку выходов, тем выше оверхед. Когда гость выходит, часть событий может обрабатываться непосредственно в ядре. События, которые не могут быть обработаны ядром, должны покинуть его и вернуться в пользовательский процесс. Это еще больше увеличивает оверхед по сравнению с выходами, которые могут обрабатываться ядром. Оверхед на потребление CPU Пространство пользователя

Запуск ioctl(2) на виртуальном CPU

Обработка выхода в пространстве пользователя

Ядро

Без оверхеда на потребление CPU Гостевая VM

Гостевой CPU готов к выполнению

Обработка выхода в ядре

Гостевой CPU выполняет инструкции

Гостевой выход

Рис. 11.6. Оверхед на потребление процессора при виртуализации оборудования

730  Глава 11. Облачные вычисления При использовании реализации KVM в Linux оверхед можно исследовать с помощью функций, выполняющих гостевой выход, которые отображаются в исходном коде вот так (фрагмент взят из файла arch/x86/kvm/vmx/vmx.c в исходном коде Linux 5.2): /* * Обработчики выхода возвращают 1, если выход обработан полностью и гость * может продолжить выполнение. В противном случае они устанавливают * параметр kvm_run, чтобы показать, что нужно сделать в пространстве пользователя, * и возвращают 0. */ static int (*kvm_vmx_exit_handlers[])(struct kvm_vcpu *vcpu) = { [EXIT_REASON_EXCEPTION_NMI] = handle_exception, [EXIT_REASON_EXTERNAL_INTERRUPT] = handle_external_interrupt, [EXIT_REASON_TRIPLE_FAULT] = handle_triple_fault, [EXIT_REASON_NMI_WINDOW] = handle_nmi_window, [EXIT_REASON_IO_INSTRUCTION] = handle_io, [EXIT_REASON_CR_ACCESS] = handle_cr, [EXIT_REASON_DR_ACCESS] = handle_dr, [EXIT_REASON_CPUID] = handle_cpuid, [EXIT_REASON_MSR_READ] = handle_rdmsr, [EXIT_REASON_MSR_WRITE] = handle_wrmsr, [EXIT_REASON_PENDING_INTERRUPT] = handle_interrupt_window, [EXIT_REASON_HLT] = handle_halt, [EXIT_REASON_INVD] = handle_invd, [EXIT_REASON_INVLPG] = handle_invlpg, [EXIT_REASON_RDPMC] = handle_rdpmc, [EXIT_REASON_VMCALL] = handle_vmcall, [...] [EXIT_REASON_XSAVES] = handle_xsaves, [EXIT_REASON_XRSTORS] = handle_xrstors, [EXIT_REASON_PML_FULL] = handle_pml_full, [EXIT_REASON_INVPCID] = handle_invpcid, [EXIT_REASON_VMFUNC] = handle_vmx_instruction, [EXIT_REASON_PREEMPTION_TIMER] = handle_preemption_timer, [EXIT_REASON_ENCLS] = handle_encls, };

Несмотря на краткость, имена могут дать некоторое представление о причинах, почему гость может вызвать гипервизор и тем самым способствовать увеличению оверхеда. Одним из распространенных гостевых выходов является инструкция остановки halt, которая обычно вызывается потоком бездействия, когда задания для выполнения нет (что позволяет перевести процессор в режим с низким энергопотреблением до получения прерывания). Эта инструкция обрабатывается функцией handle_halt() (см. элемент EXIT_REASON_HLT в предыдущем листинге), которая в итоге вызывает kvm_vcpu_halt() (arch/x86/kvm/x86.c): int kvm_vcpu_halt(struct kvm_vcpu *vcpu) { ++vcpu->stat.halt_exits; if (lapic_in_kernel(vcpu)) {

11.2. Виртуализация оборудования  731

}

vcpu->arch.mp_state = KVM_MP_STATE_HALTED; return 1; } else { vcpu->run->exit_reason = KVM_EXIT_HLT; return 0; }

Как и во многих других обработчиках гостевых выходов, код этого обработчика очень короткий, чтобы минимизировать нагрузку на процессор. Этот обработчик начинается с приращения статистики виртуального процессора, подсчитывающей количество остановок. Остальной код выполняет аппаратную эмуляцию этой привилегированной инструкции. Эти функции можно инструментировать в Linux с помощью зондов kprobes в хост-ядре гипервизора для трассировки типов и продолжительности выходов. Выходы также можно трассировать глобально с помощью точки трассировки kvm:kvm_exit, пример использования которой показан в разделе 11.2.4 «Наблюдаемость». Виртуализация оборудования, такого как контроллер прерываний и таймеры с высоким разрешением, тоже влечет за собой некоторый оверхед на процессор (и небольшой объем памяти).

Отображение памяти Как рассказывалось в главе 7 «Память», операционная система использует блок управления памятью MMU для отображения страниц виртуальной памяти в физические адреса, кэшируя их в TLB для эффективности. В случае виртуализации отображение новой страницы памяти (отказ страницы) в гостевой системе выполняется в два этапа: 1. Гостевое ядро выполняет преобразование виртуального адреса в гостевое физическое пространство. 2. Гипервизор VMM преобразует гостевой физический адрес в физический адрес хоста (фактический). Затем отображение гостевого виртуального адреса в физический адрес хоста может быть кэшировано в TLB, чтобы последующие обращения обрабатывались с нормальной скоростью — без дополнительных затрат на трансляцию. Современные процессоры поддерживают виртуализацию MMU, благодаря чему отображения, остающиеся в TLB, могут обслуживаться быстрее (обход страниц) без вызова гипервизора. Эта особенность называется расширенными таблицами страниц (extended page tables, EPT) в Intel и вложенными таблицами страниц (nested page tables, NPT) в AMD [Milewski 11]. Другой подход, позволяющий увеличить производительность без поддержки EPT/ NPT, заключается в организации теневых таблиц страниц (shadow page tables) для хранения соответствий между гостевыми виртуальными адресами и физическими адресами хоста, которые управляются гипервизором и доступны гостю

732  Глава 11. Облачные вычисления через гостевой регистр CR3. В этой стратегии гостевое ядро поддерживает свои собственные таблицы страниц, которые, как обычно, отображают гостевые виртуальные адреса в гостевые физические адреса. Гипервизор перехватывает изменения в этих таблицах страниц и создает в теневой таблице эквивалентные отображения в физические страницы хоста. Затем, передавая управление гостю, гипервизор перезаписывает регистр CR3 так, чтобы он указывал на теневые страницы.

Размер памяти В отличие от виртуализации ОС, виртуализация оборудования добавляет дополнительных потребителей памяти. Для каждой гостевой системы запускается ее собственное ядро, которое потребляет некоторый объем памяти. Архитектура хранилища тоже может приводить к двойному кэшированию, когда и гость и хост кэшируют одни и те же данные. Гипервизоры в стиле KVM также запускают дополнительный процесс VMM для каждой виртуальной машины, например QEMU, который сам потребляет некоторый объем основной памяти.

Ввод/вывод Ввод/вывод традиционно отличался самым большим оверхедом в виртуализации оборудования. Это было обусловлено необходимостью трансляции гипервизором всех операций с устройствами ввода/вывода. В высокоскоростных сетях, например в сети 10 Гбит/с, даже небольшой оверхед на каждую операцию ввода/вывода (пакет) может повлечь значительное снижение общей производительности. Но со временем были созданы технологии для уменьшения оверхеда, кульминацией которых стала аппаратная поддержка, полностью устраняющая такой оверхед. Такая аппаратная поддержка включает виртуализацию ввода/вывода MMU (AMD-Vi и Intel VT-d). Одним из методов повышения производительности ввода/вывода является использование паравиртуализированных драйверов, которые могут объединять операции ввода/вывода и тем самым уменьшать количество прерываний, чтобы сократить оверхед на работу гипервизора. Другой метод — сквозной канал PCI, когда устройство PCI отдается гостю под прямое управление и может использоваться гостевой системой, как если бы она работала непосредственно на железе. Сквозной канал PCI часто обеспечивает лучшую производительность по сравнению с другими доступными вариантами, но снижает гибкость при настройке системы с несколькими арендаторами, потому что некоторые устройства напрямую управляются гостевыми системами и не могут использоваться совместно. Это также может усложнять оперативную миграцию [Xen 19]. Есть несколько технологий повышения гибкости в использовании устройств PCI с виртуализацией, включая виртуализацию ввода/вывода с одним корнем (SRIOV, упоминалась выше) и виртуализацию ввода/вывода с несколькими корнями (MR-IOV). Эти термины ссылаются на сложные корневые топологии PCI, поразному обеспечивающие виртуализацию оборудования. В облаке Amazon EC2

11.2. Виртуализация оборудования  733 эти технологии применялись для ускорения сначала сетевых взаимодействий, а затем дискового ввода/вывода и по умолчанию используются с гипервизором Nitro [Gregg 17e]. На рис. 11.7 изображены типичные конфигурации гипервизоров Xen, KVM и Nitro. Xen Гостевая ОС (domU)

KVM

dom0

Процесс

Внешний ИД

Хостсистема

Гостевая ОС

Агент ввода/вывода

Процесс

Ядро

ГЯ Драйвер

ГЯ

Nitro

Внутр. ИД

Системный драйвер

Агент ввода/вывода

Гипервизор Гипервизор

Хост-ядро

Оборудование

Гостевая ОС Процесс ГЯ Драйвер

Системный драйвер

Оборудование

Гипервизор Оборудование

ГЯ — Гостевое ядро Внешний ИД — Внешний интерфейсный драйвер Внутр. ИД — Внутренний интерфейсный драйвер

Рис. 11.7. Пути ввода/вывода в Xen, KVM и Nitro Пунктирные стрелки показывают поток управления — порядок, в котором компоненты информируют друг друга, синхронно или асинхронно, о готовности к передаче дополнительных данных. Поток данных (сплошные стрелки) иногда может быть реализован с помощью разделяемой памяти и кольцевых буферов. Для Nitro поток управления не показан, потому что он совпадает с потоком данных благодаря прямому доступу к оборудованию. Есть несколько вариантов настройки Xen и KVM, которые здесь не показаны. На этом рисунке представлен вариант с использованием агентов (прокси-процессов) ввода/вывода (обычно QEMU), которые создаются для каждой гостевой виртуальной машины. Но их также можно настроить на использование технологии SR-IOV, позволяющей гостевым виртуальным машинам напрямую обращаться к оборудованию (не показано для Xen или KVM на рис. 11.7). Nitro требует такой аппаратной поддержки, что избавляет от необходимости использовать агенты ввода/вывода. Xen увеличивает производительность ввода/вывода за счет использования канала устройства — асинхронного транспорта на основе разделяемой памяти между dom0 и гостевыми доменами (domU). Это позволяет избежать оверхеда на создание дополнительных копий данных ввода/вывода при их передаче между доменами.

734  Глава 11. Облачные вычисления Кроме того, Xen может использовать отдельные домены для выполнения операций ввода/вывода, как описано в разделе 11.2.3 «Управление ресурсами». Количество шагов на пути ввода/вывода, как в потоке управления, так и в потоке данных, очень важно для производительности: чем их меньше, тем лучше. В 2006 году разработчики KVM провели сравнение своего гипервизора с привилегированной гостевой системой — Xen и обнаружили, что в KVM ввод/вывод выполняется за вдвое меньшее количество шагов (пять против десяти; правда, тестирование проводилось без паравиртуализации, поэтому не отражает состояния дел в большинстве современных конфигураций) [Qumranet 06]. Поскольку гипервизор Nitro исключает дополнительные операции ввода/вывода, можно ожидать, что все крупные облачные провайдеры, стремящиеся к максимальной производительности, последуют их примеру и начнут использовать аппаратную поддержку, чтобы отказаться от использования агентов ввода/вывода.

Конкуренция между арендаторами В зависимости от конфигурации гипервизора и количества совместно используемых процессоров арендаторы могут недополучать процессорное время и страдать от загрязнения кэшей процессоров другими арендаторами, что будет снижать производительность их приложений. Обычно эта проблема более характерна для конфигураций с контейнерами, чем с виртуальными машинами, потому что контейнеры особенно способствуют такому совместному использованию процессоров. Другие арендаторы, производящие ввод/вывод, могут способствовать появлению прерываний, прерывающих работу приложений в зависимости от конфигурации гипервизора. Конкуренцией за ресурсы можно управлять с помощью средств управления ресурсами.

11.2.3. Управление ресурсами Процессоры и основная память, доступные гостевой системе, обычно определяются настройками ограничений. ПО гипервизора может также предоставлять средства управления ресурсами для сетевого и дискового ввода/вывода. При использовании гипервизоров, подобных KVM, распределение физических ресурсов контролируется хост-системой, и кроме средств управления ресурсами в гипервизоре она может использовать свои средства управления для распределения ресурсов между гостевым системами. В Linux к таким средствам относятся контрольные группы, наборы задач и т. д. Дополнительную информацию о средствах управления ресурсами, доступных хост-системе, ищите в разделе 11.3 «Виртуализация операционной системы». В следующих разделах в качестве примеров описаны средства управления ресурсами в гипервизорах Xen и KVM.

11.2. Виртуализация оборудования  735

Процессоры Процессоры обычно выделяются гостевым системам в виде виртуальных процессоров (vCPU), которые управляются гипервизором. Количество выделенных ­виртуальных процессоров ограничивает потребление вычислительных ресурсов. В Xen поддерживается выделение точных квот на потребление процессоров гостевыми системами, которые контролируются планировщиком процессора в гипервизоре. В числе доступных планировщиков можно назвать [Cherkasova 07], [Matthews 08]:

yy Заимствованное виртуальное время (borrowed virtual time, BVT): справедли-

вый планировщик распределения виртуального времени, которое может быть заимствовано заранее для обеспечения выполнения с малой задержкой интерактивных приложений и приложений реального времени.

yy Самый ранний дедлайн первым (simple earliest deadline first, SEDF): планиров-

щик реального времени, который позволяет настраивать гарантии выполнения и при этом отдает приоритет задачам с самым ранним дедлайном.

yy На основе квот (credit-based): поддерживает приоритеты (веса) и ограничения потребления процессора, а также балансировку нагрузки между несколькими процессорами.

При использовании KVM точные квоты на потребление процессора могут применяться хост-системой, например, при использовании справедливого планировщика в ядре хост-системы, описанного ранее. В Linux того же эффекта можно добиться применением контрольных групп, управляющих полосой пропускания процессоров. Но обе технологии лишь ограниченно могут учитывать приоритеты гостевых систем. Распределение потребления процессоров в гостевой системе обычно непрозрачно для гипервизора, и приоритеты потоков гостевого ядра обычно невозможно определить или соблюсти. Например, низкоприоритетный демон ротации журналов в одной гостевой системе может иметь тот же приоритет гипервизора, что и критически важный сервер приложений в другой гостевой системе. При использовании Xen управление распределением вычислительных ресурсов может дополнительно осложняться рабочими нагрузками, выполняющими большое количество операций ввода/вывода, которые потребляют дополнительные вычислительные ресурсы в dom0. Внутренний интерфейсный драйвер и агент ввода/вывода в гостевом домене могут потреблять больше процессорного времени, чем им выделено, но эту разницу невозможно учесть [Cherkasova 05]. Решение было найдено в создании изолированных доменов драйверов (Isolated Driver Domains, IDD), которые разделяют обслуживание ввода/вывода для обеспечения безопасности, изоляции производительности и учета, как показано на рис. 11.8.

736  Глава 11. Облачные вычисления

Процесс

Администратор

Внешний интерфейсный драйвер

Агент ввода/вывода Внутренний интерфейсный Системный драйвер драйвер

Гипервизор Оборудование

Рис. 11.8. Гипервизор Xen с изолированным доменом драйверов Потребление процессоров изолированным доменом драйверов доступно для наблюдения и поэтому может учитываться при распределении вычислительных ресурсов между гостями. Вот что говорится в [Gupta 06]: Наш модифицированный планировщик, SEDF-DC (SEDF-Debt Collector), периодически получает от XenMon информацию о потреблении процессоров изолированным доменом драйверов для обработки ввода/вывода от имени гостевых доменов. Используя эту информацию, SEDF-DC ограничивает выделение процессорного времени гостевым доменам в соответствии с заданным комбинированным пределом потребления. Позднее для Xen был разработан еще один метод — фиктивный домен (stub domain), в котором выполняется мини-ОС.

Кэши процессоров Помимо выделения виртуальных процессоров, можно управлять потреблением кэшей процессоров с помощью технологии распределения кэша (cache allocation technology, CAT), разработанной в Intel. Она позволяет разделить кэш последнего уровня (LLC) между гостями и предотвратить загрязнение кэша одного гостя другим гостем, но также может ухудшить производительность из-за ограничения размера доступного кэша.

Объем памяти Ограничения на потребление памяти задаются в конфигурации гостевой системы, то есть гость видит только заданный объем памяти. А гостевое ядро предпринимает все необходимое (в том числе подкачку страниц), чтобы остаться в пределах своего лимита.

11.2. Виртуализация оборудования  737 Для повышения гибкости статической конфигурации в VMware был разработан драйвер воздушного шара (balloon driver) [Waldspurger 02], который может уменьшить объем памяти, доступный запущенной гостевой системе, за счет «раздувания» модуля шара внутри нее, потребляющего гостевую память. Эта память затем может использоваться гипервизором для распределения между другими гостями. Шар также можно спустить и вернуть память гостевому ядру. Во время этого процесса гостевое ядро выполняет свои обычные процедуры управления памятью, чтобы освободить память (например, подкачку страниц). VMware, Xen и KVM поддерживают драйверы воздушного шара. Когда используется драйвер шара (для проверки этого факта в гостевой системе можно выполнить поиск слова «balloon» в выводе dmesg(1)), я внимательно слежу за проблемами производительности, которые он может вызвать.

Объем файловой системы Гости получают виртуальные дисковые тома от хоста. При использовании гипервизоров, подобных KVM, это могут быть программные тома, созданные хост-системой и имеющие соответствующий размер. Например, файловая система ZFS способна создавать виртуальные тома желаемого размера.

Дисковый ввод/вывод Управление ресурсами с помощью ПО виртуализации оборудования традиционно было сосредоточено на потреблении процессоров, контролируя которое можно косвенно управлять потреблением ввода/вывода. Пропускная способность сети может регулироваться внешними выделенными устройствами, а в случае гипервизоров вроде KVM функциями хост-ядра. Например, в Linux есть средства управления шириной полосы пропускания сети на основе контрольных групп, а также различных дисциплин очередей, которые могут применяться к интерфейсам гостевой сети. После изучения изоляции производительности сети в Xen был сделан следующий вывод [Adamczyk 12]: ...в плане виртуализации сети слабое место Xen — это отсутствие должной изоляции производительности. Авторы [Adamczyk 12] также предлагают решение для планирования сетевого ввода/вывода в Xen, которое добавляет параметры настройки приоритета и скорости сетевого ввода/вывода. Если вы используете Xen, проверьте: возможно, эта или подобная технология уже доступна. Для гипервизоров с полной аппаратной поддержкой (например, Nitro) ограничения ввода/вывода могут поддерживаться оборудованием или внешними устройствами. В облаке Amazon EC2 сетевой и дисковый ввод/вывод для устройств, подключенных к сети, регулируются квотами с использованием внешних систем.

738  Глава 11. Облачные вычисления

11.2.4. Наблюдаемость Что именно доступно для наблюдения в виртуализированных системах, зависит от гипервизора и от того, где запускаются инструменты наблюдения:

yy В привилегированной гостевой системе (Xen) или хост-системе (KVM): все

физические ресурсы должны быть доступны для наблюдения с помощью стандартных инструментов ОС, описанных в предыдущих главах. Гостевой ввод/ вывод можно изучать, наблюдая за агентами ввода/вывода, если они используются. Статистики потребления ресурсов каждой гостевой системой должны быть доступны в гипервизоре. Внутренние детали работы гостевой системы, включая их процессы, недоступны для наблюдения напрямую. Некоторые операции ввода/вывода могут быть недоступны для наблюдения, если устройство использует сквозной канал или SR-IOV.

yy В хост-системе с аппаратной поддержкой (Nitro): использование SR-IOV

может затруднить наблюдение из гипервизора за операциями ввода/вывода с устройствами, потому что гость обращается к оборудованию напрямую, а не через агента или хост-ядро. (Информация о том, как в Amazon на самом деле обеспечивают наблюдаемость на уровне гипервизора Nitro, не разглашается.)

yy В гостевой системе: можно наблюдать за виртуализированными ресурсами и их потреблением в гостевой системе, а также делать выводы о проблемах с физическими ресурсами. Поскольку виртуальная машина имеет собственное выделенное ядро, можно изучить внутреннюю работу ядра и использовать для этого все инструменты трассировки, включая инструменты на основе BPF.

В привилегированной гостевой или хост-системе (гипервизоры Xen или KVM) можно получить общую информацию об использовании физических ресурсов: потребление, насыщение, ошибки, IOPS, пропускная способность, тип ввода/вывода. Обычно эта информация доступна для каждого гостя в отдельности, что позволяет быстро выявить активных пользователей. Подробная информация о том, какие гостевые процессы производят ввод/вывод, и их трассировки стеков на этом уровне недоступны. Но все это можно получить, войдя в гостевую систему (при условии, что настроены необходимые средства, например SSH) и используя инструменты наблюдения, которые она предлагает. Когда используется сквозной канал или SR-IOV, гость может выполнять ввод/вывод, обращаясь непосредственно к оборудованию, в обход гипервизора и статистик, которые тот обычно собирает. В результате операции ввода/вывода могут стать невидимыми для гипервизора и не отображаться в iostat(1) или других инструментах. Одно из возможных решений — использовать счетчики производительности PMC, связанные с вводом/выводом, и делать выводы на основе их значений. Чтобы определить корень проблемы с производительностью гостевой системы, оператору облака может потребоваться войти в обе системы — гостевую и хост-систему — и запустить инструменты наблюдения в обеих. Трассировка путей ввода/вывода осложняется из-за задействованных шагов и может также

11.2. Виртуализация оборудования  739 включать анализ внутренней работы гипервизора и агента ввода/вывода, если он используется. В рядовой гостевой системе использование физических ресурсов может быть вообще недоступно для наблюдения. Из-за этого клиенты склонны обвинять в загадочных проблемах с производительностью и нехваткой ресурсов «шумных соседей». Для спокойствия клиентов (и сокращения количества обращений в службу поддержки) информация об использовании физических ресурсов (отредактированная) может предоставляться другими путями, включая SNMP или облачный API. Чтобы упростить наблюдение за работой контейнеров, есть различные решения мониторинга, которые предоставляют графики, дашборды и ориентированные графы для представления информации о контейнерной среде. Среди них Google cAdvisor [Google 20d] и Cilium Hubble [Cilium 19] (оба распространяются с открытым исходным кодом). В разделах ниже показаны простые инструменты наблюдения, которые можно использовать в разных местах, и описана стратегия анализа производительности. Для примера используются Xen и KVM (Nitro не рассматривается, потому что это собственность Amazon).

11.2.4.1. Привилегированный гость/хост Все ресурсы системы (процессоры, память, файловая система, диск, сеть) должны быть доступными для наблюдения с использованием инструментов, описанных в предыдущих главах (за исключением ввода/вывода через сквозной канал или SR-IOV).

Xen При использовании Xen-подобных гипервизоров гостевые виртуальные процессоры есть в гипервизоре и не видны для стандартных инструментов ОС из привилегированного гостя (dom0). В Xen вместо них используйте xentop(1): # xentop xentop - 02:01:05 Xen 3.3.2-rc1-xvm 2 domains: 1 running, 1 blocked, 0 paused, 0 crashed, 0 dying, 0 shutdown Mem: 50321636k total, 12498976k used, 37822660k free CPUs: 16 @ 2394MHz NAME STATE CPU(sec) CPU(%) MEM(k) MEM(%) MAXMEM(k) MAXMEM(%) VCPUS NETS NETTX(k) NETRX(k) VBDS VBD_OO VBD_RD VBD_WR SSID Domain-0 -----r 6087972 2.6 9692160 19.3 no limit n/a 16 0 0 0 0 0 0 0 0 Doogle_Win --b--172137 2.0 2105212 4.2 2105344 4.2 1 2 0 0 2 0 0 0 0 [...]

Особый интерес представляют эти столбцы:

yy CPU(%): потребление процессора в процентах (в сумме для всех процессоров); yy MEM(k): потребление основной памяти (в килобайтах);

740  Глава 11. Облачные вычисления

yy MEM(%): потребление основной памяти в процентах от общего объема системной памяти;

yy MAXMEM(k): максимальный объем основной памяти (в килобайтах); yy MAXMEM(%): максимальный объем основной памяти в процентах от общего объема системной памяти;

yy yy yy yy yy yy

VCPUS: количество виртуальных процессоров, выделенных гостю; NETS: количество виртуализированных сетевых интерфейсов; NETTX(k): передано в сеть (в килобайтах); NETRX(k): принято из сети (в килобайтах); VBDS: количество виртуальных блочных устройств; VBD_OO: количество обращений к виртуальному блочному устройству, заблоки-

рованных и поставленных в очередь (насыщение);

yy VBD_RD: запросов на чтение из виртуального блочного устройства; yy VBD_WR: запросов на запись в виртуальное блочное устройство. По умолчанию xentop(1) обновляет вывод каждые 3 с, но этот интервал можно изменить с помощью параметра -d delay_secs. Для расширенного анализа особенностей работы Xen есть инструмент xentrace(8), который может извлекать из гипервизора журнал с фиксированными типами событий. Этот журнал можно проанализировать с помощью xenanalyze на наличие проблем планирования в гипервизоре и узнать, какой планировщик процессора используется. В исходном коде Xen также есть инструмент xenoprof — профилировщик для Xen (MMU и гости).

KVM При использовании KVM-подобных гипервизоров гостевые экземпляры доступны для наблюдения из хост-системы. Например: host$ top top - 15:27:55 up 26 days, 22:04, 1 user, load average: 0.26, 0.24, 0.28 Tasks: 499 total, 1 running, 408 sleeping, 2 stopped, 0 zombie %Cpu(s): 19.9 us, 4.8 sy, 0.0 ni, 74.2 id, 1.1 wa, 0.0 hi, 0.1 si, 0.0 st KiB Mem : 24422712 total, 6018936 free, 12767036 used, 5636740 buff/cache KiB Swap: 32460792 total, 31868716 free, 592076 used. 8715220 avail Mem PID USER 24881 libvirt+ 21897 23445 15476 23038

root root root root

22784 root [...]

PR 20 0 0 0 0

NI VIRT RES 0 6161864 1.051g

SHR S %CPU %MEM 19448 S 171.9 4.5

TIME+ COMMAND 0:25.88 qemu-system-x86

-20 -20 -20 -20

0 0 0 0

0 0 0 0

0 0 0 0

I I I I

2.3 2.3 2.0 2.0

0.0 0.0 0.0 0.0

0:00.47 0:00.24 0:01.23 0:00.28

kworker/u17:8 kworker/u17:7 kworker/u17:2 kworker/u17:0

0 -20

0

0

0 I

1.7

0.0

0:00.36 kworker/u17:1

11.2. Виртуализация оборудования  741 Процесс qemu-system-x86 — это гостевой гипервизор KVM, в рамках которого выполняются потоки для каждого vCPU и потоки агентов ввода/вывода. Общее потребление процессора гостем можно увидеть в верхней части вывода top(1), а потребление каждого виртуального процессора определить с помощью других инструментов, например pidstat(1): host$ pidstat -tp 24881 1 03:40:44 PM UID TGID TID %usr %system %guest %wait 03:40:45 PM 64055 24881 - 17.00 17.00 147.00 0.00 03:40:45 PM 64055 - 24881 9.00 5.00 0.00 0.00 03:40:45 PM 64055 - 24889 0.00 0.00 0.00 0.00 03:40:45 PM 64055 - 24897 1.00 3.00 69.00 1.00 03:40:45 PM 64055 - 24899 1.00 4.00 79.00 0.00 03:40:45 PM 64055 - 24901 0.00 0.00 0.00 0.00 03:40:45 PM 64055 - 25811 0.00 0.00 0.00 0.00 03:40:45 PM 64055 - 25812 0.00 0.00 0.00 0.00 [...]

%CPU CPU Command 181.00 0 qemu-system-x86 14.00 0 |__qemu-system-x86 0.00 6 |__qemu-system-x86 73.00 4 |__CPU 0/KVM 84.00 5 |__CPU 1/KVM 0.00 2 |__vnc_worker 0.00 7 |__worker 0.00 6 |__worker

Как показывает этот вывод, потоки процессоров с именами CPU 0/KVM и CPU 1/ KVM потребляют 73 % и 84 % процессорного времени соответственно. Для отображения процессов QEMU в имена соответствующих гостевых экземпляров обычно достаточно изучить аргументы процесса (ps -wwfp PID) и прочитать параметр -name. Еще одна важная область для анализа — выходы в гипервизор на гостевых виртуальных процессорах. Типы выходов могут подсказать, что делает гость: бездействует, выполняет ввод/вывод или производит вычисления. В Linux команда perf(1) имеет подкоманду kvm, которая сообщает высокоуровневые статистики по выходам в KVM, например: host# perf kvm stat live 11:12:07.687968 Analyze events for all VMs, all VCPUs: VM-EXIT Samples Samples% Time% Min Time Max Time Avg time MSR_WRITE 1668 68.90% 0.28% 0.67us 31.74us 3.25us ( +- 2.20% ) HLT 466 19.25% 99.63% 2.61us 100512.98us 4160.68us ( +- 14.77% ) PREEMPTION_TIMER 112 4.63% 0.03% 2.53us 10.42us 4.71us ( +- 2.68% ) PENDING_INTERRUPT 82 3.39% 0.01% 0.92us 18.95us 3.44us ( +- 6.23% ) EXTERNAL_INTERRUPT 53 2.19% 0.01% 0.82us 7.46us 3.22us ( +- 6.57% ) IO_INSTRUCTION 37 1.53% 0.04% 5.36us 84.88us 19.97us ( +- 11.87% ) MSR_READ 2 0.08% 0.00% 3.33us 4.80us 4.07us ( +- 18.05% ) EPT_MISCONFIG 1 0.04% 0.00% 19.94us 19.94us 19.94us ( +- 0.00% ) Total Samples:2421, Total events handled time:1946040.48us. [...]

Этот вывод показывает причины выхода виртуальной машины в гипервизор и статистику по каждой причине. Наиболее продолжительные выходы в этом примере были обусловлены операцией HLT (остановка), выполняя которую виртуальные процессоры переходят в состояние ожидания. В столбцах выводятся:

742  Глава 11. Облачные вычисления

yy yy yy yy yy yy yy

VM-EXIT: тип выхода; Samples: число выходов, зафиксированных за период трассировки; Samples%: доля от общего числа выходов в процентах; Time%: продолжительность от общего времени всех выходов в процентах; Min Time: минимальная продолжительность выхода; Max Time: максимальная продолжительность выхода; Avg time: средняя продолжительность выхода.

Даже при том, что оператору непросто наблюдать за внутренней работой гостевой виртуальной машины, изучение статистик выходов позволяет охарактеризовать то, как оверхед на виртуализацию оборудования влияет на арендатора. Если наблюдается небольшое количество выходов и большинство обусловлено выполнением инструкций HLT, то можно быть уверенными, что гостевой процессор бездействует долгое время. С другой стороны, если выполняется большое количество операций ввода/вывода с прерываниями, как сгенерированными, так и внедренными в гостевую систему, то весьма вероятно, что гость активно выполняет ввод/вывод через свои виртуальные сетевые адаптеры и диски. Для расширенного анализа KVM есть множество точек трассировки: host# perf list | grep kvm kvm:kvm_ack_irq kvm:kvm_age_page kvm:kvm_apic kvm:kvm_apic_accept_irq kvm:kvm_apic_ipi kvm:kvm_async_pf_completed kvm:kvm_async_pf_doublefault kvm:kvm_async_pf_not_present kvm:kvm_async_pf_ready kvm:kvm_avic_incomplete_ipi kvm:kvm_avic_unaccelerated_access kvm:kvm_cpuid kvm:kvm_cr kvm:kvm_emulate_insn kvm:kvm_enter_smm kvm:kvm_entry kvm:kvm_eoi kvm:kvm_exit [...]

[Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint [Tracepoint

event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event] event]

Особый интерес представляют kvm:kvm_exit (упоминалась выше) и kvm:kvm_entry. Вот как можно получить список аргументов kvm:kvm_exit с помощью bpftrace: host# bpftrace -lv t:kvm:kvm_exit tracepoint:kvm:kvm_exit unsigned int exit_reason; unsigned long guest_rip; u32 isa; u64 info1; u64 info2;

11.2. Виртуализация оборудования  743 Как видите, на основании аргументов можно узнать причину выхода (exit_reason), указатель гостевой инструкции для возврата (guest_rip) и другие детали. При трассировке kvm:kvm_exit в паре с kvm:kvm_entry, которая срабатывает при выполнении входа в гостевую систему из KVM (по завершении выхода в гипервизор), можно измерить продолжительность и узнать причину выхода. В книге «BPF Performance Tools» [Gregg 19] я опубликовал инструмент kvmexits.bt для bpftrace, обобщающий причины выхода в виде гистограммы (он доступен в моем репозитории [Gregg 19e]). Вот пример вывода: host# kvmexits.bt Attaching 4 probes... Tracing KVM exits. Ctrl-C to end ^C [...] @exit_ns[30, IO_INSTRUCTION]: [1K, 2K) 1 | | [2K, 4K) 12 |@@@ | [4K, 8K) 71 |@@@@@@@@@@@@@@@@@@ | [8K, 16K) 198 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| [16K, 32K) 129 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | [32K, 64K) 94 |@@@@@@@@@@@@@@@@@@@@@@@@ | [64K, 128K) 37 |@@@@@@@@@ | [128K, 256K) 12 |@@@ | [256K, 512K) 23 |@@@@@@ | [512K, 1M) 2 | | [1M, 2M) 0 | | [2M, 4M) 1 | | [4M, 8M) 2 | | @exit_ns[48, EPT_VIOLATION]: [512, 1K) 6160 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | [1K, 2K) 6885 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | [2K, 4K) 7686 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| [4K, 8K) 2220 |@@@@@@@@@@@@@@@ | [8K, 16K) 582 |@@@ | [16K, 32K) 244 |@ | [32K, 64K) 47 | | [64K, 128K) 3 | |

Вывод включает гистограммы для всех типов выходов, но здесь показаны только две. Судя по этим гистограммам, выходы IO_INSTRUCTION длятся обычно менее 512 мкс, но иногда случаются выбросы от 2 до 8 мс. Другой пример расширенного анализа — профилирование содержимого регистра CR3. Каждый процесс в гостевой системе имеет собственное адресное пространство и набор таблиц страниц для преобразования виртуальных адресов в физические. Ссылка на эту таблицу страниц хранится в регистре CR3. Путем выборки регистра CR3 из хост-системы (например, с помощью bpftrace) можно определить, активен ли какой-то один процесс в гостевой системе (одно и то же значение в CR3) или она переключается между несколькими процессами (разные значения CR3). Для получения дополнительной информации нужно авторизоваться в гостевой системе.

744  Глава 11. Облачные вычисления

11.2.4.2. Гость При виртуализации оборудования из гостевой системы можно наблюдать только виртуальные устройства (если не используется сквозной канал/SR-IOV), включая количество виртуальных процессоров, выделенных гостю. Вот пример проверки процессоров гостя KVM с помощью mpstat(1): kvm-guest$ mpstat -P ALL 1 Linux 4.15.0-91-generic (ubuntu0) 03/22/2020 _x86_64_ (2 CPU) 10:51:34 PM CPU %usr %nice %sys %iowait %irq %soft %steal %guest %gnice 10:51:35 PM all 14.95 0.00 35.57 0.00 0.00 0.00 0.00 0.00 0.00 10:51:35 PM 0 11.34 0.00 28.87 0.00 0.00 0.00 0.00 0.00 0.00 10:51:35 PM 1 17.71 0.00 42.71 0.00 0.00 0.00 0.00 0.00 0.00 10:51:35 10:51:36 10:51:36 10:51:36 [...]

PM PM PM PM

CPU all 0 1

%usr 11.56 8.05 15.04

%nice %sys %iowait 0.00 37.19 0.00 0.00 22.99 0.00 0.00 48.67 0.00

%irq 0.00 0.00 0.00

%soft %steal %guest %gnice 0.00 0.50 0.00 0.00 0.00 0.00 0.00 0.00 0.00 0.00 0.00 0.00

%idle 49.48 59.79 39.58 %idle 50.75 68.97 36.28

Этот вывод показывает состояние всего двух гостевых процессоров. Команда vmstat(8) в Linux выводит столбец с процентом недополученного (stolen) процессорного времени (st), и это редкий пример статистики виртуализации. Недополученное время — это процессорное время, недополученное гостем: оно могло быть использовано другими арендаторами или функциями гипервизора (например, для обслуживания собственного ввода/вывода или из-за ограничений, наложенных на экземпляр): xen-guest$ vmstat 1 procs -----------memory---------- ---swap-- -----io---- -system-r b swpd free buff cache si so bi bo in cs 1 0 0 107500 141348 301680 0 0 0 0 1006 9 1 0 0 107500 141348 301680 0 0 0 0 1006 11 1 0 0 107500 141348 301680 0 0 0 0 978 9 3 0 0 107500 141348 301680 0 0 0 4 912 15 2 0 0 107500 141348 301680 0 0 0 0 33 7 3 0 0 107500 141348 301680 0 0 0 0 34 6 5 0 0 107500 141348 301680 0 0 0 0 35 7 2 0 0 107500 141348 301680 0 0 0 48 38 16 [...]

------cpu----us sy id wa st 99 0 0 0 1 97 0 0 0 3 95 0 0 0 5 99 0 0 0 1 3 0 0 0 97 100 0 0 0 0 1 0 0 0 99 2 0 0 0 98

В этом примере тестировался гость Xen с агрессивной политикой ограничения потребления процессора. За первые 4 с более 90 % процессорного времени было потрачено в пользовательском режиме гостя, при этом недополучено было всего несколько процентов. Затем поведение начинает резко меняться, и гость недополучает большую часть процессорного времени. Анализ потребления процессора на уровне тактов часто требует использования аппаратных счетчиков (см. главу 4 «Инструменты наблюдения», раздел 4.3.9 «Аппаратные счетчики (PMC)»). Они могут быть доступны или недоступны гостю в зависимости от конфигурации гипервизора. В Xen, например, есть модуль мониторинга

11.2. Виртуализация оборудования  745 производительности виртуальной среды (virtual performance monitoring unit, vpmu), обеспечивающий доступность счетчиков PMC для гостевых систем и поддерживающий настройки, позволяющие определить, какие счетчики PMC будут доступны [Gregg 17f]. Поскольку дисковые и сетевые устройства виртуализированы, важной метрикой для анализа является задержка. Она показывает, как устройство реагирует на виртуализацию, ограничения и действия других арендаторов. Такие показатели, как процент занятости, трудно интерпретировать, не зная характеристик фактического устройства. Задержку устройства можно подробно изучить с помощью инструментов трассировки ядра, включая perf(1), Ftrace и BPF (главы 13, 14 и 15). К счастью, все они должны работать в гостевых системах, потому что имеют собственные ядра, а пользователь root обладает неограниченным доступом к ядру. Вот пример запуска biosnoop(8) для BPF в гостевой системе KVM: kvm-guest# biosnoop TIME(s) COMM 0.000000000 systemd-journa 0.001647000 jbd2/vda2-8 0.011814000 jbd2/vda2-8 1.711989000 jbd2/vda2-8 1.718005000 jbd2/vda2-8 [...]

PID 389 319 319 319 319

DISK vda vda vda vda vda

T W W W W W

SECTOR 13103112 8700872 8701576 8701584 8701624

BYTES 4096 360448 4096 20480 4096

LAT(ms) 3.41 0.77 0.20 0.72 0.67

Этот вывод показывает задержку устройства виртуального диска. Обратите внимание, что с контейнерами (раздел 11.3 «Виртуализация операционной системы») эти инструменты трассировки ядра могут не работать, из-за чего конечный пользователь не сможет детально анализировать работу устройств ввода/вывода и др.

11.2.4.3. Стратегия В предыдущих главах мы рассмотрели методы анализа потребления физических ресурсов, которые могут использоваться администраторами физических систем для поиска узких мест и ошибок. Также изучают средства управления, выделяющие ресурсы для гостей. Так можно проверить, как часто гости достигают установленных пределов, и при необходимости проинформировать их и побудить к обновлению. Без входа в гостевую систему, что может потребоваться для любого серьезного исследования производительности, администраторы мало что могут определить1. В гостевых системах можно использовать инструменты и стратегии анализа потребления ресурсов, описанные в предыдущих главах. Но важно помнить, что Рецензент подсказал еще один возможный метод (обратите внимание, что это не рекомендация): можно проанализировать моментальный снимок гостевого хранилища (при условии, что он не шифруется). Например, с учетом журналирования адресов предыдущих операций дискового ввода/вывода моментальный снимок состояния файловой системы можно использовать, чтобы определить, к каким файлам происходили обращения.

1

746  Глава 11. Облачные вычисления ресурсы в этом случае обычно виртуальные. Потребление некоторых ресурсов невозможно довести до предела из-за невидимого контроля со стороны гипервизора или конкуренции с другими арендаторами. В идеальном случае облачное ПО или облачный провайдер предоставляют клиентам возможность проверять потребление физических ресурсов (с некоторыми ограничениями), чтобы они могли сами анализировать проблемы с производительностью. В других случаях выводы о конфликтах и ограничениях можно сделать на основе увеличения задержки ввода/ вывода и планирования процессора. Такую задержку можно измерить на уровне системных вызовов или в гостевом ядре. Стратегия, которую я использую для выявления конкуренции за дисковые и сетевые ресурсы из гостевой системы, заключается в тщательном анализе закономерностей ввода/вывода. Сюда может входить журналирование вывода biosnoop(8) (см. предыдущий пример), изучение последовательностей ввода/вывода на наличие какихлибо выбросов по задержке и их причин (ввод/вывод больших объемов данных выполняется медленнее), их закономерностей доступа (например, когда операция чтение ставится в очередь после выталкивания буферов, заполненных операциями записи). Если нет явных причин, то, возможно, мы имеем дело с физической конкуренцией или с проблемой в устройстве.

11.3. ВИРТУАЛИЗАЦИЯ ОПЕРАЦИОННОЙ СИСТЕМЫ Виртуализация операционной системы делит ее на экземпляры, которые в Linux называются контейнерами. Контейнеры действуют как отдельные гостевые серверы и могут управляться и перезагружаться независимо от хост-системы. Эта технология позволяет получить небольшие, эффективные, быстро загружаемые экземпляры для облачных клиентов и серверы высокой плотности для облачных операторов. На рис. 11.9 показаны гостевые экземпляры в технологии виртуализации ОС. Хост-система Контейнер Администратор

Приложения

Контейнер

Контейнер

Приложения

Приложения

Хост-ядро Хост-система

Рис. 11.9. Виртуализация операционной системы Этот подход уходит корнями в команду Unix chroot(8), которая ограничивает процесс поддеревом глобальной файловой системы Unix (изменяет каталог верхнего уровня, «/», видимый процессом). В 1998 году в FreeBSD было предложено дальнейшее развитие этой технологии в виде клеток FreeBSD (FreeBSD jails),

11.3. Виртуализация операционной системы  747 предоставлявшее защищенные ячейки, действующие как отдельные серверы. В 2005 году в Solaris 10 появилась версия Solaris Zones со средствами управления ресурсами. В то же время в Linux была добавлена возможность частичной изоляции процессов: в 2002 году в Linux 2.4.19 были добавлены пространства имен, и в 2008 году в Linux 2.6.24 — группы управления (cgroups) [Corbet 07a], [Corbet 07b], [Linux 20m]. Пространства имен и контрольные группы объединяются для создания контейнеров, которые обычно используются механизмом seccomp-bpf, а также средствами управления доступом к системным вызовам. Ключевое отличие от технологий виртуализации оборудования в том, что работает только одно ядро. Ниже перечислены преимущества контейнеров по сравнению с виртуальными машинами при виртуализации оборудования (раздел 11.2 «Виртуализация оборудования»):

yy короткое время инициализации: обычно не превышает нескольких миллисекунд; yy гости могут использовать всю память для приложений (без затрат на запуск собственного ядра);

yy единый кэш файловой системы — это позволяет избежать двойного кэширования хостом и гостем;

yy более точное управление совместным использованием ресурсов (контрольные группы);

yy для операторов хоста: улучшенная наблюдаемость производительности благодаря возможности наблюдать за гостевыми процессами и их взаимодействиями;

yy контейнеры могут совместно использовать страницы памяти для общих файлов, что обеспечивает более эффективное использование кэша страниц и увеличивает коэффициент попаданий в кэш процессора;

yy процессоры — это настоящие физические процессоры; предположения, сделанные на основе адаптивных блокировок мьютексов, остаются в силе.

Есть и минусы:

yy выше конкуренция за ресурсы ядра (блокировки, кэши, буферы, очереди); yy для гостей: хуже наблюдаемость производительности, так как обычно нет возможности проанализировать работу ядра;

yy yy yy yy

любой фатальный сбой в ядре оказывает фатальное влияние на всех гостей; гости не могут запускать собственные модули ядра; гости не могут запускать профильные ядра PGO (см. раздел 3.5.1 «Ядра PGO»); гости не могут запускать другие версии ядра1.

Есть технологии эмуляции других интерфейсов системных вызовов, чтобы другие ОС могли работать под управлением ядра, но на практике такая эмуляция отрицательно влияет на производительность. Кроме того, такие технологии обычно эмулируют только базовый набор возможностей системных вызовов и при попытке обратиться к расширенным возможностям возвращают код ошибки ENOTSUP (не поддерживается).

1

748  Глава 11. Облачные вычисления Рассмотрим два первых недостатка: после переноса из виртуальной машины в контейнер гость почти наверняка столкнется с проблемой конкуренции за ядро, а также потеряет возможность анализировать его работу. Они станут более зависимыми от оператора хоста для такого рода анализа. Недостаток контейнеров, не связанный с производительностью, заключается в том, что контейнеры считаются менее безопасными, так как используют одно общее ядро. Все эти недостатки решаются легковесной виртуализацией, описанной в разделе 11.4 «Легковесная виртуализация», хотя и за счет некоторых преимуществ. В разделах ниже описаны особенности виртуализации ОС в Linux: реализация, оверхед, управление ресурсами и наблюдаемость.

11.3.1. Реализация В ядре Linux нет понятия контейнера. Но есть пространства имен и контрольные группы (cgroups), которые используются ПО, действующим в пространстве пользователя (например, Docker), для создания так называемых контейнеров1. На рис. 11.10 показана типичная конфигурация контейнера. Host Контейнер 1

Контейнер 2

PID 1

PID 1

Пространства имен

Пространства имен

Контейнер 3 PID 1

Пространства имен

Общие cgroups cgroups для контейнера 1

cgroups для контейнера 2

cgroups для контейнера 3

Ядро

Рис. 11.10. Контейнеры в Linux Для связи процесса с контрольными группами в ядре используется структура nsproxy. Поскольку эта структура определяет, как контейнеризуется процесс, ее можно считать лучшим представлением понятия контейнера в ядре.

1

11.3. Виртуализация операционной системы  749 Несмотря на то что в каждом контейнере есть процесс с идентификатором PID 1, это разные процессы, потому что принадлежат разным пространствам имен. Поскольку чаще всего для развертывания контейнеров используется Kubernetes, его архитектура изображена на рис. 11.11. О Kubernetes шла речь в разделе 11.1.6 «Оркестровка (Kubernetes)». Интернет Нода Kubernetes

Прокси-сервер kube-proxy Под

Под

Пространства имен

Пространства имен

Контейнер

Контейнер

Контейнер

Контейнер

Сетевой интерфейс

Сетевой интерфейс

CNI

Конфигурация сети

Рис. 11.11. Нода Kubernetes На рис. 11.11 также показана связь по сети между подами (группами контейнеров) через прокси-сервер kube-proxy и между контейнерами через сетевой интерфейс контейнеров (container network interface, CNI). Преимущество Kubernetes — это возможность с легкостью создать несколько контейнеров для совместного использования одних и тех же пространств имен, включив их в один под (группу). Это позволяет использовать более быстрые методы связи между контейнерами.

Пространства имен Пространство имен фильтрует представление системы, поэтому контейнеры могут видеть и администрировать только собственные процессы, точки монтирования и другие ресурсы. Это основной механизм изоляции контейнеров друг от друга. В табл. 11.1 перечислены некоторые из пространств имен.

750  Глава 11. Облачные вычисления Таблица 11.1. Некоторые пространства имен в Linux Пространство имен

Описание

cgroup

Для видимости контрольных групп

ipc

Для видимости межпроцессных взаимодействий

mnt

Для монтирования файловых систем

net

Для изоляции сетевого стека: управляет видимостью интерфейсов, сокетов, маршрутов и т. д.

pid

Для видимости процессов: фильтрует содержимое /proc

time

Для поддержки отдельных часов в каждом контейнере

user

Для идентификаторов пользователей

uts

Для информации о хосте; системный вызов uname(2)

Список текущих пространств имен в системе можно получить с помощью lsns(8): # lsns

NS 4026531835 4026531836 4026531837 4026531838 4026531839 4026531840 4026531860 4026531992 4026532166 4026532167 [...]

TYPE NPROCS cgroup 105 pid 105 user 105 uts 102 ipc 105 mnt 98 mnt 1 net 105 mnt 1 uts 1

PID 1 1 1 1 1 1 19 1 241 241

USER root root root root root root root root root root

COMMAND /sbin/init /sbin/init /sbin/init /sbin/init /sbin/init /sbin/init kdevtmpfs /sbin/init /lib/systemd/systemd-udevd /lib/systemd/systemd-udevd

Как показывает вывод команды lsns(8), процесс init имеет шесть разных пространств имен, используемых более чем 100 процессами. Описание пространств имен можно найти в исходном коде Linux, а также на страницах справочного руководства man, начиная с namespaces(7).

Контрольные группы Контрольные группы (cgroups) ограничивают использование ресурсов. В ядре Linux есть две версии контрольных групп: 1 и 21. Многие проекты, такие как Kubernetes, все еще используют версию 1 (версия 2 находится в разработке). В табл. 11.2 перечислены некоторые контрольные группы версии 1. Эти контрольные группы можно настроить для ограничения конкуренции за ресурсы между контейнерами, например, установив жесткие ограничения на потребление процессора и памяти или более мягкие (на основе относительных долей) Есть и смешанные конфигурации, в которых одновременно используются обе версии.

1

11.3. Виртуализация операционной системы  751 ограничения на потребление процессора и диска. Кроме того, контрольные группы можно организовать в иерархию, включающую системные контрольные группы, которые совместно используются контейнерами (см. рис. 11.10). Таблица 11.2. Некоторые контрольные группы в Linux Контрольная группа

Описание

bikio

Ограничивает блочный (дисковый) ввод/вывод: байты и частоту операций

cpu

Ограничивает потребление процессора на основе относительных долей

cpuacct

Учитывает потребление процессора процессами группы

cpuset

Назначает контейнерам процессоры и узлы памяти

devices

Управляет доступом к устройствам

hugetlb

Ограничивает использование огромных страниц

memory

Ограничивает объем памяти для процессов и ядра, а также использование подкачки

net_cls

Присваивает сетевым пакетам идентификатор класса (classid) для использования дисциплинами очередей и брандмауэрами

net_prio

Устанавливает приоритеты для сетевых интерфейсов

perf_event

Позволяет утилите perf наблюдать за процессами в контрольной группе

pids

Ограничивает количество процессов, которое можно запустить

rdma

Ограничивает потребление ресурсов RDMA (Remote Direct Memory Access — прямой доступ к удаленной памяти) и InfiniBand (высокоскоростная коммутируемая сеть)

Контрольные группы версии 2 изначально основаны на иерархии и свободны от различных недостатков версии 1. Ожидается, что в ближайшие годы контейнерные технологии будут переведены на версию 2, а версия 1 будет считаться устаревшей. Дистрибутив Fedora в версии 31, выпущенной в 2019 году, уже перешел на использование контрольных групп версии 2. Документацию по контрольным группам можно найти в исходном коде Linux в Documentation/cgroup-v1 и Documentation/admin-guide/cgroup-v2.rst, а также на странице справочного руководства cgroups(7). В разделах ниже описаны такие темы, связанные с виртуализацией контейнеров, как оверхед, управление ресурсами и наблюдаемость. Они различаются для разных реализаций контейнеров и конфигураций.

11.3.2. Оверхед Контейнеры обычно дают незначительный оверхед. Потребление процессоров и потребление памяти приложениями в контейнере не должны отличаться от случая, когда те же приложения выполняются на «голом железе». Однако в ядре

752  Глава 11. Облачные вычисления могут производиться некоторые дополнительные вызовы при вводе/выводе из-за дополнительных слоев на пути к файловой системе и сети. Намного большую угрозу для производительности представляет конкуренция между несколькими арендаторами, так как контейнеры способствуют интенсивному совместному использованию ядра и физических ресурсов. В разделах ниже в общих чертах описывается оверхед, обусловленный выполнением на процессоре, отображением памяти, операциями ввода/вывода и конкуренцией со стороны других арендаторов.

Процессор Когда поток контейнера выполняется в пользовательском режиме, то прямых оверхедов на потребление процессора нет: потоки выполняются непосредственно на процессоре, пока либо добровольно не уступят процессор, либо не будут вытеснены планировщиком. Также в Linux нет дополнительного оверхеда процессорного времени на запуск процессов в пространствах имен и контрольных группах: все процессы уже выполняются в пространстве имен и контрольных группах по умолчанию, независимо от того, используются контейнеры или нет. Чаще всего снижение вычислительной производительности объясняется конкуренцией с другими арендаторами (см. следующий раздел «Конкуренция между арендаторами»). При использовании ПО оркестровки, такой как Kubernetes, дополнительные сетевые компоненты могут использовать немного больше вычислительных ресурсов для обработки сетевых пакетов. Например, при наличии большого количества служб (измеряемого тысячами), Kube-proxy тратит довольно много процессорного времени на обслуживание первого пакета, когда требуется проверить его соответствие большим наборам правил iptables. Этот оверхед можно уменьшить, заменив kube-proxy на BPF [Borkmann 20].

Отображение памяти Отображение памяти, загрузка и сохранение должны выполняться без дополнительного оверхеда.

Размер памяти Приложения могут использовать весь объем памяти, выделенной контейнеру. Сравните это с виртуализацией оборудования, когда для каждого арендатора запускается отдельная виртуальная машина со своим ядром, причем каждому ядру требуется для работы некоторый объем оперативной памяти. Обычно конфигурация контейнеров (с использованием OverlayFS) позволяет всем вместе использовать один кэш страниц при обращении к тем же файлам.

11.3. Виртуализация операционной системы  753 Это помогает эффективнее использовать память по сравнению с виртуальными машинами, которые загружают в память свои копии общих файлов (например, системных библиотек).

Ввод/вывод Оверхед ввода/вывода зависит от конфигурации контейнера, потому что может включать дополнительные уровни для большей изоляции:

yy ввода/вывода через файловую систему: например, overlayfs; yy сетевого ввода/вывода: например, доступ в сеть через мост. Ниже показана трассировка стека ядра для операции записи в файловую систему из контейнера, которая была обработана overlayfs (и файловой системой XFS): blk_mq_make_request+1 generic_make_request+420 submit_bio+108 _xfs_buf_ioapply+798 __xfs_buf_submit+226 xlog_bdstrat+48 xlog_sync+703 __xfs_log_force_lsn+469 xfs_log_force_lsn+143 xfs_file_fsync+244 xfs_file_buffered_aio_write+629 do_iter_readv_writev+316 do_iter_write+128 ovl_write_iter+376 __vfs_write+274 vfs_write+173 ksys_write+90 do_syscall_64+85 entry_SYSCALL_64_after_hwframe+68

Обращение к overlayfs можно увидеть в этой трассировке стека в виде вызова функции ovl_write_iter(). Насколько это важно, зависит от рабочей нагрузки и частоты операций ввода/ вывода. Для серверов с невысокой частотой операций ввода/вывода (скажем, 'pid:[4026532216]' host# ls -lh /proc/4915/ns/pid lrwxrwxrwx 1 root root 0 Mar 15 20:46 /proc/4915/ns/pid -> 'pid:[4026532216]'

Обратите внимание на совпадение идентификаторов пространств имен (4026532216): это подтверждает, что идентификатор PID 4915 в хост-системе соответствует гостевому идентификатору PID 753. Пространства имен точек монтирования могут создавать аналогичные проблемы. Например, если запустить команду perf(1) на уровне хоста, она будет искать дополнительные файлы символов в /tmp/perf-PID.map, но приложения в контейнерах сохраняют их в /tmp внутри контейнера, а это совсем другой каталог, отличный от каталога /tmp в хост-системе. Кроме того, идентификатор PID с большой долей вероятности будет отличаться из-за использования разных пространств имен процессов. Элис Голдфасс (Alice Goldfuss) первой опубликовала обходное решение, предусматривающее перемещение и переименование файлов символов, чтобы они были доступны на хосте [Goldfuss 17]. Но спустя некоторое время в perf(1) была добавлена поддержка пространств имен, устраняющая эту проблему, а в ядро — отображение пространства имен /proc/PID/root для прямого доступа к корневому каталогу («/») контейнера. Например: host# ls -lh /proc/4915/root/tmp total 0 -rw-r--r-- 1 root root 0 Mar 15 20:54 I_am_in_the_container.txt

Это список файлов в каталоге /tmp контейнера. Кроме чтения файлов в каталоге /proc, команда nsenter(1) может запускать другие команды в выбранных пространствах имен. Эта команда запускает top(1) на уровне хоста в пространствах имен точек монтирования (-m) и процессов (-p) внутри PID 4915 (-t 4915): # nsenter -t 4915 -m -p top top - 21:14:24 up 32 days, 23:23, 0 users, load average: 0.32, 0.09, 0.02 Tasks: 3 total, 2 running, 1 sleeping, 0 stopped, 0 zombie

768  Глава 11. Облачные вычисления %Cpu(s): 0.2 us, 0.1 sy, KiB Mem : 1996844 total, KiB Swap: 0 total, PID 753 1 766

USER root root root

PR 20 20 20

0.0 ni, 99.4 id, 0.0 wa, 0.0 hi, 0.0 si, 0.2 st 98400 free, 858060 used, 1040384 buff/cache 0 free, 0 used. 961564 avail Mem

NI VIRT 0 818504 0 18504 0 38364

RES 93428 3212 3420

SHR 11996 2796 2968

S %CPU R 100.0 S 0.0 R 0.0

%MEM 0.2 0.2 0.2

TIME+ 0:27.88 0:17.57 0:00.00

COMMAND java bash top

Здесь видно, что самым активным является процесс java с идентификатором PID 753.

Трассировка с помощью BPF Некоторые инструменты трассировки на основе BPF уже поддерживают контейнеры, но многие еще не имеют этой поддержки. К счастью, добавить такую поддержку в инструменты bpftrace обычно несложно, как показано ниже. См. главу 15, где объясняются приемы программирования bpftrace. Инструмент forks.bt подсчитывает количество новых процессов, запущенных в период, пока выполнялась трассировка, наблюдая за обращениями к системным вызовам clone(2), fork(2) и vfork(2). Вот его исходный код: #!/usr/local/bin/bpftrace tracepoint:syscalls:sys_enter_clone, tracepoint:syscalls:sys_enter_fork, tracepoint:syscalls:sys_enter_vfork { @new_processes = count(); }

Пример вывода: # ./forks.bt Attaching 3 probes... ^C @new_processes: 590

Как показывает вывод, за время трассировки было запущено 590 новых процессов. Чтобы получить разбивку по контейнерам, можно вывести имя узла (хоста) из пространства имен uts. Такая возможность зависит от реализации ПО контейнера, настраивающего это пространство имен, но обычно она поддерживается. Вот исходный код того же инструмента с дополнениями, выделенными полужирным шрифтом: #!/usr/local/bin/bpftrace #include #include #include tracepoint:syscalls:sys_enter_clone, tracepoint:syscalls:sys_enter_fork, tracepoint:syscalls:sys_enter_vfork

11.3. Виртуализация операционной системы  769 {

}

$task = (struct task_struct *)curtask; $nodename = $task->nsproxy->uts_ns->name.nodename; @new_processes[$nodename] = count();

Этот дополнительный код использует текущую структуру task_struct ядра, чтобы получить имя узла из пространства имен uts, и добавляет его как ключ в выходную карту @new_processes. Пример вывода: # ./forks.bt Attaching 3 probes... ^C @new_processes[ip-10-1-239-218]: 171 @new_processes[efe9f9be6185]: 743

Теперь результаты разбиты по контейнерам, и по ним видно, что узел 6280172ea7b9 (контейнер) в период трассировки запустил 252 процесса. Другой узел, ip-10-1239-218, — это хост-система. Такое возможно только потому, что системные вызовы выполняются в контексте задачи (процесса), поэтому curtask возвращает соответствующую структуру task_struct, через которую мы получаем имя узла. При трассировке асинхронных событий, например прерываний по завершении дискового ввода/вывода, процесс, запустивший операцию ввода/вывода, к моменту прерывания может покинуть процессор. Тогда curtask будет ссылаться на другое имя узла. Поскольку извлечение имени узла из uts может превратиться в часто используемую операцию в bpftrace, я собираюсь добавить встроенную переменную nodename, чтобы единственным был следующий код: @new_processes[nodename] = count();

Проверьте обновления bpftrace: возможно, эта переменная уже появится, когда вы будете читать эти строки.

Управление ресурсами Наблюдение за средствами управления ресурсами из раздела 11.3.3 «Управление ресурсами» необходимо, чтобы определить, влияют ли на контейнер накладываемые ими ограничения. Традиционные инструменты анализа производительности и документация в основном уделяют внимание физическим ресурсам и не видят ограничений, накладываемых программно. Проверка средств управления ресурсами описана в методе USE (глава 2 «Методологии», раздел 2.5.9 «Метод USE»), который предполагает наблюдение за ресурсами и проверку потребления, насыщения и ошибок. При наличии средств управления ресурсами их тоже нужно проверять для каждого ресурса.

770  Глава 11. Облачные вычисления В подразделе «Статистики контрольных групп» упоминался файл /sys/fs/cgroup/.../ cpu.stat со статистиками, отражающими, сколько раз накладывалось ограничение на потребление процессора (nr_throttled) и общую продолжительность действия ограничений в наносекундах (throttled_time). Это регулирование заключается в ограничении пропускной способности процессоров. Определить, регулируется ли пропускная способность для контейнера, довольно легко: значение throttled_time будет увеличиваться. Если регулирование выполняется с помощью контрольной группы cpuset, то потребление процессорного времени можно проверить с помощью инструментов и метрик, включая mpstat(1). Распределением вычислительных ресурсов можно также управлять с помощью назначения долей, как описано в подразделе «Доли и пропускная способность». Ограниченный по производительности из-за долей контейнер обнаружить сложнее — у него нет соответствующих статистик. Я разработал блок-схему, показанную на рис. 11.13, которая поможет определить, выполняется ли регулирование вычислительных ресурсов доступных контейнеров и как это происходит [Gregg 17g]: Начало

Время действия ограничений увеличивается?

Да

Нет Количество принудительных переключений контекста увеличивается?

Нет

Да

Не ограничивается Да

На уровне хоста имеются бездействующие CPU?

Потребление CPU в группе cpuset составляет 100%? Нет Не ограничивается, но требуются дальнейшие исследования

Ограничивается полоса пропускания

Да Действуют ограничения в группе cpuset

Нет

Все остальные арендаторы бездействуют? Да Ограничиваются физические CPU

Нет Ограничение по назначенной доле

Рис. 11.13. Анализ регулирования вычислительных ресурсов, доступных контейнеру

11.3. Виртуализация операционной системы  771 Процедура анализа, представленная на рис. 11.13, помогает определить, ограничиваются ли вычислительные ресурсы контейнера и как. Для этого используются пять статистик:

yy Продолжительность действия ограничений: значение throttled_time для контрольной группы.

yy Принудительные переключения контекста: идентифицируются по увеличению значения nonvolvention_ctxt_switches в файле /proc/PID/status.

yy Бездействие процессоров на уровне хоста: можно определить по значению столбца %idle в выводе mpstat(1), в /proc/stat и с помощью других инструментов.

yy Потребление процессоров составляет 100 %: в группе cpuset потребление процессоров можно узнать из вывода mpstat(1), из файла /proc/stat и т. д.

yy Бездействие всех остальных арендаторов: можно определить с помощью

специализированного инструмента для конкретного контейнера (например, docker stat) или системных инструментов, которые показывают отсутствие конкуренции за вычислительные ресурсы (например, если top(1) показывает только один контейнер с ненулевым значением в столбце %CPU).

Аналогичную процедуру можно разработать для других ресурсов, а вспомогательная статистика, включая статистики контрольных групп, должна быть доступна в ПО и инструментах мониторинга. Идеальный продукт или инструмент мониторинга определит все автоматически и сообщит, как происходит регулирование ресурсов каждого контейнера, если оно есть.

11.3.4.3. Гость (контейнер) Многие ожидают, что инструменты анализа производительности, которые запускаются в контейнере, будут отображать только статистики контейнера, но часто это не так. Вот пример запуска iostat(1) в неактивном контейнере: container# iostat -sxz 1 [...] avg-cpu: %user %nice %system %iowait 57.29 0.00 8.54 33.17 Device nvme0n1 avg-cpu: Device nvme0n1 [...]

tps 2578.00 %user 51.78

kB/s 12436.00

rqm/s 331.00

%steal 0.00

await aqu-sz 0.33 0.00

%nice %system %iowait 0.00 7.61 40.61 tps 2664.00

kB/s 11020.00

%idle 1.01

rqm/s 88.00

%steal 0.00

areq-sz %util 4.82 100.00

%idle 0.00

await aqu-sz 0.32 0.00

areq-sz 4.14

%util 98.80

Судя по этому выводу, есть существенная нагрузка на процессор и диск, но контейнер полностью бездействует. Это может сбивать с толку тех, кто плохо знаком с виртуализацией ОС, — может возникнуть ощущение, что контейнер работает под

772  Глава 11. Облачные вычисления нагрузкой. Причина в том, что инструменты показывают статистики хоста, которые суммируют активность других арендаторов. Об этой особенности я кратко рассказал в разделе 11.3.4 «Наблюдаемость», в подразделе «Традиционные инструменты». Со временем в этих инструментах появляется поддержка контейнеров, и они начинают отображать статистики контрольных групп только по контейнерам, когда используются в контейнере. Есть статистики контрольных групп для блочного ввода/вывода. Вот пример, полученный в том же контейнере: container# cat /sys/fs/cgroup/blkio/blkio.throttle.io_serviced 259:0 Read 452 259:0 Write 352 259:0 Sync 692 259:0 Async 112 259:0 Discard 0 259:0 Total 804 Total 804 container# sleep 10 container# cat /sys/fs/cgroup/blkio/blkio.throttle.io_serviced 259:0 Read 452 259:0 Write 352 259:0 Sync 692 259:0 Async 112 259:0 Discard 0 259:0 Total 804 Total 804

Здесь показаны статистики, отражающие количество операций разных типов. Я вывел их дважды, выполнив команду sleep 10 в промежутке, чтобы выждать некоторое время. Значения статистик не увеличились за это время, следовательно, контейнер не выполнял операции дискового ввода/вывода. Есть еще один файл со счетчиками байтов: blkio.throttle.io_service_bytes. К сожалению, эти счетчики не обеспечивают всей полноты информации, необходимой инструменту iostat(1). Для контрольных групп нужно организовать поддержку большего количества счетчиков, чтобы iostat(1) могла работать с контейнерами.

Поддержка контейнеров Добавить в инструмент поддержку контейнеров не обязательно означает ограничивать видимую ему область рамками контейнера: возможность видеть состояние физических ресурсов из контейнера можно считать дополнительным преимуществом. Многое зависит от целей использования контейнеров: A. Контейнер как изолированный сервер: если контейнер используется с этой целью, как обычно бывает с облачными провайдерами, то инструменты с поддержкой контейнеров должны показывать только текущую активность контейнера. Например, утилита iostat(1) должна показывать только дисковый ввод/ вывод, инициированный контейнером и никем другим. Это можно реализовать,

11.3. Виртуализация операционной системы  773 изолировав все источники статистик, и отображать только метрики для текущего контейнера (/proc, /sys, netlink и т. д.). Но такая изоляция может как помогать, так и мешать анализу производительности на уровне гостя: диагностика проблем, связанных с конкуренцией за ресурсы со стороны других арендаторов, займет больше времени и будет основываться на наблюдении необъяснимого увеличения задержки устройства. B. Контейнер как пакетное решение: компаниям, использующим собственные облака контейнеров, не обязательно отделять статистики, характеризующие работу контейнеров. Возможность видеть статистики хоста из контейнеров (как это часто бывает и как было показано выше на примере с iostat(1)) помогает конечным пользователям лучше понимать состояние аппаратных устройств и проблемы, вызванные шумными соседями. Под поддержкой контейнеров в iostat(1) в таком сценарии может пониматься разбивка, показывающая характеристики текущего контейнера в сопоставлении с характеристиками хоста или других контейнеров. В обоих сценариях инструменты должны также поддерживать наблюдаемость средств управления ресурсами, где это необходимо, как часть поддержки контейнеров. Продолжая пример iostat(1): помимо статистики %util для устройства (недоступна в сценарии A), также можно вывести ограничение %cap, исходя из пределов пропускной способности и IOPS в blkio, чтобы из контейнера можно было определить, ограничивается ли дисковый ввод/вывод средствами управления ресурсами1. Если наблюдение за физическими ресурсами разрешено, то гости смогут исключить из рассмотрения некоторые проблемы, в том числе проблему шумных соседей. А это может облегчить бремя поддержки для операторов контейнеров: люди склонны винить в проблемах то, что не могут наблюдать. Кроме того, в этом заключается важное отличие от виртуализации оборудования, при котором физические ресурсы скрываются от гостей и не имеют никакой возможности получить информацию о них (без использования внешних средств). В идеале в Linux должна появиться настройка для управления видимостью статистик хоста, чтобы любая контейнерная среда могла выбирать между сценариями A и B.

Инструменты трассировки У продвинутых инструментов трассировки, основанных на использовании механизмов ядра — perf(1), Ftrace и BPF, — такие же проблемы, и их создатели работают над реализацией поддержки контейнеров. Сейчас они не могут выполняться в контейнерах из-за отсутствия разрешений, необходимых для доступа к различным системным вызовам (perf_event_open(2), bpf(2) и т. д.) и файлам в /proc и /sys. Кратко опишу их будущее для сценариев A и B, представленных выше: Вывод iostat(1) с параметром -x содержит так много полей, что не умещается по ширине ни в одном моем самом широком терминале, и я не решаюсь прямо предложить добавить дополнительные. Я бы предпочел добавить еще один параметр, например -l, для вывода столбцов со значениями программных пределов.

1

774  Глава 11. Облачные вычисления C. Требуется изоляция: разрешить контейнерам использовать инструменты трассировки можно через: • Фильтрацию в ядре: события и их аргументы могут фильтроваться ядром, чтобы, например, в точке трассировки block:block_rq_issue контейнеру была доступна только информация о его текущем дисковом вводе/выводе. • API хоста: хост может предоставлять доступ к определенным инструментам трассировки через безопасный API или графический интерфейс. Например, контейнер может запросить выполнение обычных инструментов BCC, таких как execsnoop(8) и biolatency(8), а хост может после проверки запроса выполнить ограниченную версию инструмента и вернуть его вывод. D. Изоляция не требуется: ресурсы (системные вызовы, /proc и /sys) делаются доступными контейнеру, чтобы инструменты трассировки работали без ограничений. Инструменты трассировки при этом могут иметь свою поддержку контейнеров, чтобы упростить фильтрацию посторонних событий, не имеющих отношения к текущему контейнеру. Создание инструментов и статистик для контейнеров в Linux и других ядрах — медленный процесс, и, вероятно, потребуется много лет, прежде чем все они будут реализованы. Особенно это удручает опытных пользователей, которые входят в систему, чтобы воспользоваться инструментами командной строки для анализа производительности. Многие пользователи применяют продукты для мониторинга с агентами, которые уже поддерживают контейнеры (в той или иной степени).

11.3.4.4. Стратегия Предыдущие главы охватывали методы анализа физических ресурсов системы и включали описание различных методологий. Они могут использоваться в хостсистеме и, в некоторой степени, в гостевой, с учетом упоминавшихся выше ограничений. Гостям обычно доступны для наблюдения высокоуровневые статистики потребления ресурсов, но детальный анализ на уровне ядра, как правило, невозможен. Помимо физических ресурсов, операторы хостов и арендаторы гостевых систем должны также проверять ограничения, накладываемые средствами управления ресурсами в облачных средах. Эти ограничения проявляются задолго до физических ограничений, поэтому доступность ресурсов гостю, скорее всего, будет ограничиваться ими, и они должны проверяться в первую очередь. Многие традиционные инструменты наблюдения создавались до появления контейнеров и средств управления ресурсами (например, top(1), iostat(1)), поэтому не включают по умолчанию информацию об управлении ресурсами, из-за чего пользователи могут упустить это из виду. Вот несколько комментариев и стратегий для проверки средств управления каждого ресурса:

yy Процессор: см. схему на рис. 11.13. Проверьте настройки контрольной группы cpuset, ширину полосы пропускания и количество выделенных долей.

11.4. Легковесная виртуализация  775

yy Память: проверьте текущее потребление и его соответствие ограничениям контрольной группы memory.

yy Объем файловой системы: эта характеристика должна быть доступна для наблю-

дения, как и для любой другой файловой системы (включая использование df(1)).

yy Дисковый ввод/вывод: проверьте настройки контрольной группы blkio (/sys/

fs/cgroup/blkio) и статистики из файлов blkio.throttle.io_serviced и blkio.throttle. io_service_bytes: если они увеличиваются с течением времени, то это говорит о регулировании дискового ввода/вывода. Если доступна трассировка средствами BPF, то для подтверждения предположения о регулировании используйте инструмент blkthrot(8) [Gregg 19].

yy Сетевой ввод/вывод: определите текущую пропускную способность сети

и сравните с любым известным пределом полосы пропускания, который только можно наблюдать на уровне хоста. Достижение предела будет приводить к увеличению задержек сетевого ввода/вывода из-за регулирования сетевой активности арендаторов.

И заключительный комментарий: большая часть этого раздела описывает контрольные группы версии 1 и текущее состояние поддержки контейнеров в Linux. Возможности ядра быстро меняются, поэтому документация и поддержка контейнеров в инструментах оказываются в роли догоняющих. Чтобы быть в курсе последних достижений в области поддержки контейнеров в Linux, следите за изменениями в новых выпусках ядра и просматривайте каталог Documentation в исходном коде Linux. Рекомендую обращать внимание на любую документацию, написанную Теджуном Хео (Tejun Heo), ведущим разработчиком контрольных групп версии 2, включая Documentation/admin-guide/cgroup-v2.rst в исходном коде Linux [Heo 15].

11.4. ЛЕГКОВЕСНАЯ ВИРТУАЛИЗАЦИЯ Легковесная виртуализация оборудования была разработана с целью объединить лучшее из двух подходов: безопасность виртуализации оборудования, с одной стороны, и эффективность с высокой скоростью загрузки контейнеров — с другой. На рис. 11.14 изображена технология легковесной виртуализации на основе Firecracker в сравнении с контейнерами. В легковесной виртуализации оборудования используется легковесный гипервизор, основанный на виртуализации процессора и поддерживающий минимальное количество эмулируемых устройств. В этом его главное отличие от полноценных гипервизоров оборудования (раздел 11.2 «Виртуализация оборудования»), которые создавались на основе виртуальных машин для настольных компьютеров и включают поддержку видео, звука, BIOS, шины PCI и других устройств, а также различные уровни поддержки процессора. Гипервизор, предназначенный только для серверных вычислений, не нуждается в поддержке всех этих устройств. По поводу современных гипервизоров можно быть уверенными только в том, что в них доступны современные возможности виртуализации процессоров.

776  Глава 11. Облачные вычисления

Контейнеры Хост-система

Планировщик

Гостевая ОС

Легковесные VM Гостевая ОС

Приложения

Приложения

Пространства имен

Пространства имен

Хост-ядро

Оборудование (процессоры)

Хост-система

Гостевая ОС

Гостевая ОС

Приложения

Приложения

Пространства имен

Пространства имен

Гостевое ядро

Гостевое ядро

Легковесный гипервизор Планировщик

Хост-ядро

KVM

Оборудование (процессоры)

Рис. 11.14. Легковесная виртуализация Для сравнения: QEMU (quick EMUlator) — это полноценный гипервизор оборудования для KVM, исходный код которого насчитывает более 1,4 миллиона строк (QEMU версии 4.2). Amazon Firecracker — это легковесный гипервизор, исходный код которого содержит всего 50 тысяч строк [Agache 20]. Легковесные виртуальные машины действуют как и виртуальные машины в конфигурации B, описанной в разделе 11.2. По сравнению с традиционными виртуальными машинами легковесные загружаются намного быстрее, имеют меньший оверхед на память и повышенную безопасность. Легковесные гипервизоры позволяют еще больше повысить безопасность за счет настройки пространства имен — еще одного уровня безопасности, — как показано на рис. 11.14. В одних реализациях легковесные виртуальные машины называются контейнерами, в других используется термин MicroVM. Я предпочитаю использовать термин «MicroVM», так как термин «контейнеры» обычно ассоциируется с виртуализацией ОС.

11.4.1. Реализация Есть несколько проектов легковесной виртуализации оборудования, в том числе:

yy Intel Clear Containers: в этом проекте, запущенном в 2015 году, используются возможности Intel VT для создания легковесных виртуальных машин. Он доказал потенциал легковесных контейнеров, сократив время загрузки до 45 мс [Kadera 16]. В 2017 году проект Intel Clear Containers присоединился к проекту Kata Containers, в рамках которого продолжает развитие.

yy Kata Containers: этот проект, запущенный в 2017 году, основан на Intel Clear Containers и Hyper.sh RunV и находится под управлением OpenStack Foundation.

11.4. Легковесная виртуализация  777 Его девиз: «Скорость контейнеров, безопасность виртуальных машин» [Kata Containers 20].

yy Google gVisor: этот проект с  открытым исходным кодом, запущенный

в 2018 году, использует специализированное ядро пользовательского пространства для гостевых систем, написанное на Go, которое улучшает безопасность контейнеров.

yy Amazon Firecracker: этот проект с открытым исходным кодом был запущен в 2019 году [Firecracker 20]. Он использует KVM с новым легковесным диспетчером виртуальных машин (VMM) вместо QEMU и обеспечивает время загрузки системы около 100 мс [Agache 20].

В разделах ниже описана широко используемая реализация: легковесный гипервизор оборудования (Intel Clear Containers, Kata Containers и Firecracker). gVisor реализует другой подход, основанный на собственном легковесном ядре, и имеет характеристики, более схожие с характеристиками контейнеров (раздел 11.3 «Виртуализация операционной системы»).

11.4.2. Оверхед Оверхед примерно такой же, как и на виртуализацию KVM (см. раздел 11.2.2 «Оверхед»), но с меньшим оверхедом памяти, потому что диспетчер виртуальных машин намного меньше. Так, оверхед памяти в Intel Clear Containers 2.0 составляет около 48–50 Мбайт на контейнер [Kadera 16]. Amazon Firecracker сообщает об оверхеде меньше 5 Мбайт [Agache 20].

11.4.3. Управление ресурсами Поскольку процессы диспетчера виртуальных машин выполняются на хосте, ими можно управлять с помощью средств управления ресурсами ОС: контрольные группы, дисциплины очередей и т. д., по аналогии с виртуализацией KVM, как было описано в разделе 11.2.3 «Управление ресурсами». Эти средства управления ресурсами ОС подробно обсуждались с точки зрения контейнеров в разделе 11.3.3 «Управление ресурсами».

11.4.4. Наблюдаемость Степень наблюдаемости аналогична наблюдаемости виртуализации KVM, описанной в разделе 11.2.4 «Наблюдаемость», то есть:

yy В хост-системе: можно наблюдать все физические ресурсы, используя стан-

дартные инструменты ОС, как было описано в предыдущих главах. Гостевые виртуальные машины отображаются как процессы. Внутренние детали работы гостевой системы, включая их процессы и файловые системы, недоступны для наблюдения напрямую. Чтобы оператор гипервизора мог изучить внутреннюю работу гостя, ему должен быть предоставлен доступ (например, через SSH).

778  Глава 11. Облачные вычисления

yy В гостевой системе: можно наблюдать виртуализированные ресурсы и их по-

требление гостем, а также делать выводы о физических проблемах. Работают все инструменты трассировки ядра, включая инструменты на основе BPF, потому что у виртуальной машины собственное выделенное ядро.

В качестве примера наблюдаемости показана виртуальная машина Firecracker, наблюдаемая с помощью top(1) на уровне хоста1: host# top top - 15:26:22 up 25 days, 22:03, 2 users, load average: 4.48, 2.10, 1.18 Tasks: 495 total, 1 running, 398 sleeping, 2 stopped, 0 zombie %Cpu(s): 25.4 us, 0.1 sy, 0.0 ni, 74.4 id, 0.0 wa, 0.0 hi, 0.0 si, 0.0 st KiB Mem : 24422712 total, 8268972 free, 10321548 used, 5832192 buff/cache KiB Swap: 32460792 total, 31906152 free, 554640 used. 11185060 avail Mem PID 30785 31568 31437 30719 1 [..]

USER root bgregg bgregg root root

PR 20 20 20 20 20

NI VIRT RES SHR S %CPU %MEM 0 1057360 297292 296772 S 200.0 1.2 0 110076 3336 2316 R 45.7 0.0 0 57028 8052 5436 R 22.8 0.0 0 120320 16348 10756 S 0.3 0.1 0 227044 7140 3540 S 0.0 0.0

TIME+ 0:22.03 0:01.93 0:01.09 0:00.83 15:32.13

COMMAND firecracker sshd ssh ignite-spawn systemd

Вся виртуальная машина выглядит как единый процесс с именем firecracker. Как показывает этот вывод, она потребляет 200 % процессорного времени (два процессора). На уровне хоста нельзя узнать, какие гостевые процессы потребляют столько процессорного времени. Ниже показаны результаты запуска top(1) в гостевой системе: guest# top top - 22:26:30 up 16 min, 1 user, load average: 1.89, 0.89, 0.38 Tasks: 67 total, 3 running, 35 sleeping, 0 stopped, 0 zombie %Cpu(s): 81.0 us, 19.0 sy, 0.0 ni, 0.0 id, 0.0 wa, 0.0 hi, 0.0 si, 0.0 st KiB Mem : 1014468 total, 793660 free, 51424 used, 169384 buff/cache KiB Swap: 0 total, 0 free, 0 used. 831400 avail Mem PID USER 1104 root 1105 root 1106 root 1 root 3 root [...]

PR NI 20 0 20 0 20 0 20 0 0 -20

VIRT 18592 18592 38916 77224 0

RES 1232 1232 3468 8352 0

SHR 700 700 2944 6648 0

S %CPU %MEM R 100.0 0.1 R 100.0 0.1 R 4.8 0.3 S 0.0 0.8 I 0.0 0.0

TIME+ 0:05.77 0:05.59 0:00.01 0:00.38 0:00.00

COMMAND bash bash top systemd rcu_gp

Как показывает этот вывод, практически все процессорное время потребляется двумя программами bash. Также сравните различие в сводной информации в начале вывода: на уровне хоста средняя нагрузка за минуту составляет 4,48, а на уровне гостя — 1,89. Другие детали Эта виртуальная машина была создана с помощью Weave Ignite, диспетчера контейнеров MicroVM [Weaveworks 20].

1

11.5. Другие типы виртуализации  779 тоже отличаются, потому что в гостевой системе выполняется собственное ядро, которое накапливает статистики только для этого гостя. Как описано в разделе 11.3.4 «Наблюдаемость» в контейнерах, в отличие от виртуальных машин, статистики, полученные на уровне гостя, могут неожиданно оказаться общесистемными статистиками хоста. В качестве другого примера показан вывод команды mpstat(1), запущенной в гостевой системе: guest# mpstat -P ALL 1 Linux 4.19.47 (cd41e0d846509816) 03/21/20 _x86_64_ (2 CPU) 22:11:07 CPU %usr %nice %sys %iowait %irq %soft %steal %guest %gnice 22:11:08 all 81.50 0.00 18.50 0.00 0.00 0.00 0.00 0.00 0.00 22:11:08 0 82.83 0.00 17.17 0.00 0.00 0.00 0.00 0.00 0.00 22:11:08 1 80.20 0.00 19.80 0.00 0.00 0.00 0.00 0.00 0.00 [...]

%idle 0.00 0.00 0.00

В этом выводе отображается информация только о двух процессорах, потому что гостевой системе выделено только два процессора.

11.5. ДРУГИЕ ТИПЫ ВИРТУАЛИЗАЦИИ В числе других примитивов и технологий облачных вычислений можно назвать:

yy Функции как услуга (functions as a service, FaaS): разработчик отправляет функцию приложения в облако, которая вызывается по запросу. Эта технология упрощает процесс разработки ПО, но отсутствие управляющего сервера (потому она и называется «бессерверной») сказывается на производительности. Время запуска функции может быть значительным, и в отсутствие сервера конечный пользователь не может запускать традиционные инструменты наблюдения из командной строки. Анализ производительности обычно ограничивается отметками времени, которые фиксируются в приложении.

yy Программное обеспечение как услуга (software as a service, SaaS): предостав-

ляет ПО высокого уровня, при этом конечному пользователю не требуется настраивать серверы или приложения. Анализ производительности доступен только операторам: без доступа к серверам конечные пользователи мало что могут оценить, кроме времени на стороне клиента.

yy Одноцелевые ядра (unikernels): эта технология предполагает компиляцию

приложения вместе с минимальным ядром в единый двоичный файл, который может выполняться аппаратными гипервизорами напрямую без операционной системы. Несмотря на возможность увеличения производительности за счет минимизации выполняемого кода и, следовательно, загрязнения кэша процессора, а также повышения безопасности за счет удаления неиспользуемого кода, одноцелевые ядра создают проблемы наблюдаемости, обусловленные отсутствием ОС, где можно запускать инструменты наблюдения. Статистики ядра, подобные тем, что находятся в /proc, могут отсутствовать. К счастью, многие реализации

780  Глава 11. Облачные вычисления позволяют запускать одноцелевые ядра как обычные процессы, предоставляя один путь для анализа (хотя и в другой среде, отличной от гипервизора). Гипервизоры также могут поддерживать возможности исследования одноцелевых ядер, например профилирование стека: я разработал прототип, который генерирует флейм-графики для MirageOS Unikernel [Gregg 16a]. Во всех этих технологиях нет ОС, в которую конечный пользователь мог бы войти (или контейнера для входа) и провести традиционный анализ производительности. Анализ FaaS и SaaS доступен только операторам. Для анализа производительности одноцелевых ядер требуются специализированные инструменты и статистики, а в идеале — поддержки профилирования в гипервизоре.

11.6. СРАВНЕНИЕ Сравнение технологий помогает лучше понять их, даже если вы не в состоянии изменить технологию, которая используется в вашей компании. В табл. 11.7 приведено сравнение характеристик производительности трех технологий из этой главы1. Таблица 11.7. Сравнение характеристик производительности технологий виртуализации Характеристика

Виртуализация оборудования

Виртуализация операционной системы (контейнеры)

Легковесная виртуализация

Пример

KVM

Контейнеры

FireCracker

Производительность процессора

Высокая (поддержка процессоров)

Высокая

Высокая (поддержка процессоров)

Распределение процессоров

Фиксированное в виде виртуальных процессоров

Гибкое (доли и полоса пропускания)

Фиксированное в виде виртуальных процессоров

Пропускная способность ввода/вывода

Высокая (с SR-IOV)

Высокая (без внутреннего оверхеда)

Высокая (с SR-IOV)

Задержка ввода/вывода

Низкая (с SR-IOV и без QEMU)

Низкая (без внутреннего оверхеда)

Низкая (с SR-IOV)

Оверхед на доступ к памяти

Имеется (на EPT/NPT или теневые таблицы страниц)

Нет

Имеется (на EPT/NPT или теневые таблицы страниц)

Потери памяти

Имеются (на дополнительные ядра, таблицы страниц)

Нет

Имеются (на дополнительные ядра, таблицы страниц)

Обратите внимание, что в этой таблице сравнивается только производительность. Есть и другие отличия, например, контейнеры были описаны как имеющие более слабую защиту [Agache 20].

1

11.6. Сравнение  781

Характеристика

Виртуализация оборудования

Виртуализация операционной системы (контейнеры)

Легковесная виртуализация

Распределение памяти

Фиксированное (возможно двойное кэширование)

Гибкое (память, неиспользуемая гостем, используется кэшем файловой системы)

Фиксированное (возможно двойное кэширование)

Управление ресурсами

Большинством (ядро плюс элементы управления гипервизором)

Многими (зависит от ядра)

Большинством (ядро плюс элементы управления гипервизором)

Наблюдаемость: на уровне хоста

Средняя (потребление ресурсов, статистики гипервизора, ОС для KVM-подобных гипервизоров, но нет возможности наблюдать внутреннюю работу гостя)

Высокая (доступно все)

Средняя (потребление ресурсов, статистики гипервизора, ОС для KVMподобных гипервизоров, но нет возможности наблюдать внутреннюю работу гостя)

Наблюдаемость: на уровне гостя

Высокая (полная наблюдаемость ядра и виртуальных устройств)

Средняя (доступно только пространство пользователя, счетчики ядра, наблюдаемость ядра ограниченна) с дополнительными метриками, характеризующими хост (например, iostat(1))

Высокая (полная наблюдаемость ядра и виртуальных устройств)

Наблюдаемость благоприятствует

Конечным пользователям

Операторам хостов

Конечным пользователям

Сложность гипервизора

Наивысшая (вдобавок к сложному гипервизору)

Средняя (ОС)

Высокая (вдобавок к легковесному гипервизору)

Разные гостевые системы

Да

Обычно нет (иногда возможно при использовании механизма трансляции системных вызовов, который может вносить свой оверхед)

Да

Эта таблица будет устаревать по мере появления новых возможностей в технологиях виртуализации, но перечисленные в ней характеристики можно использовать для сравнения даже совершенно новых технологий виртуализации, которые не подходят ни к одной из категорий. Технологии виртуализации часто сравнивают по результатам микробенчмаркинга, помогающим определить, какая из них работает лучше всего. К сожалению, при

782  Глава 11. Облачные вычисления этом упускается из виду такая важная характеристика, как наблюдаемость системы, благодаря которой можно получить наибольший выигрыш в производительности. Хорошая наблюдаемость часто позволяет выявить и устранить ненужную работу, что увеличивает потенциальный выигрыш в производительности намного больше, чем незначительные различия в гипервизоре. Наибольший уровень наблюдаемости для облачных операторов предлагают контейнеры, потому что, находясь на уровне хоста, они могут видеть все процессы и их взаимодействия. Для конечных пользователей предпочтительнее виртуальные машины, потому что они предоставляют пользователям доступ к ядру и позволяют запускать любые инструменты производительности на основе ядра, включая описанные в главах 13, 14 и 15. Другой вариант — контейнеры с доступом к ядру, дающие операторам и пользователям полную наблюдаемость всех ресурсов. Но этот вариант возможен, только если клиент сам управляет хостом для размещения контейнеров, — такой способ не обеспечивает должной изоляции безопасности между контейнерами. Виртуализация все еще продолжает развиваться, например, легковесные гипервизоры оборудования появились совсем недавно. Учитывая их преимущества, особенно с точки зрения наблюдаемости для конечного пользователя, я ожидаю, что их популярность будет расти.

11.7. УПРАЖНЕНИЯ 1. Ответьте на следующие вопросы по терминологии виртуализации: • В чем разница между хост-системой и гостевой системой? • Что такое арендатор (tenant)? • Что такое гипервизор? • Что такое виртуализация оборудования? • Что такое виртуализация ОС? 2. Ответьте на следующие концептуальные вопросы: • Опишите роль изоляции производительности. • Опишите, какой оверхед с точки зрения производительности сопровождает современную виртуализацию оборудования (например, Nitro). • Опишите, какой оверхед с точки зрения производительности сопровождает виртуализацию ОС (например, контейнеры Linux). • Опишите наблюдаемость физической системы с точки зрения гостя в виртуализации оборудования (в XEN или KVM). • Опишите наблюдаемость физической системы с точки зрения гостя в виртуализации ОС. • Объясните разницу между виртуализацией оборудования (например, Xen или KVM) и легковесной виртуализацией оборудования (например, Firecracker).

11.8. Ссылки  783 3. Выберите технологию виртуализации и ответьте на следующие вопросы с позиции гостя: • Опишите, как применяется ограничение на объем доступной памяти и как это ограничение проявляется на стороне гостя. (Что наблюдает системный администратор, когда гостевая система исчерпывает доступную ей память?) • Опишите, как применяется ограничение на вычислительные ресурсы и как это ограничение проявляется на стороне гостя. • Опишите, как применяется ограничение на дисковый ввод/вывод и как это ограничение проявляется на стороне гостя. • Опишите, как применяется ограничение на сетевой ввод/вывод и как это ограничение проявляется на стороне гостя. 4. Разработайте чек-лист для метода USE, определяющий процедуру анализа средств управления ресурсами. Опишите, как получить каждую метрику (например, какую команду выполнить) и как интерпретировать результаты. Перед установкой или использованием дополнительных программных продуктов постарайтесь ограничиться инструментами наблюдения, имеющимися в ОС.

11.8. ССЫЛКИ [Goldberg 73] Goldberg, R. P., «Architectural Principles for Virtual Computer Systems», Harvard University (Thesis), 1972. [Waldspurger 02] Waldspurger, C., «Memory Resource Management in VMware ESX Server», Proceedings of the 5th Symposium on Operating Systems Design and Implementation, 2002. [Cherkasova 05] Cherkasova, L., and Gardner, R., «Measuring CPU Overhead for I/O Processing in the Xen Virtual Machine Monitor», USENIX ATEC, 2005. [Adams 06] Adams, K., and Agesen, O., «A Comparison of Software and Hardware Techniques for x86 Virtualization», ASPLOS, 2006. [Gupta 06] Gupta, D., Cherkasova, L., Gardner, R., and Vahdat, A., «Enforcing Performance Isolation across Virtual Machines in Xen», ACM/IFIP/USENIX Middleware, 2006. [Qumranet 06] «KVM: Kernel-based Virtualization Driver», Qumranet Whitepaper, 2006. [Cherkasova 07] Cherkasova, L., Gupta, D., and Vahdat, A., «Comparison of the Three CPU Schedulers in Xen», ACM SIGMETRICS, 2007. [Corbet 07a] Corbet, J., «Process containers», LWN.net, https://lwn.net/Articles/236038, 2007. [Corbet 07b] Corbet, J., «Notes from a container», LWN.net, https://lwn.net/Articles/256389, 2007. [Liguori, 07] Liguori, A., «The Myth of Type I and Type II Hypervisors», http://blog.codemonkey. ws/2007/10/myth-of-type-i-and-type-ii-hypervisors.html, 2007. [VMware 07] «Understanding Full Virtualization, Paravirtualization, and Hardware Assist», https:// www.vmware.com/techpapers/2007/understanding-full-virtualization-paravirtualizat-1008.html, 2007. [Matthews 08] Matthews, J., et al. Running Xen: «A Hands-On Guide to the Art of Virtualization», Prentice Hall, 2008.

784  Глава 11. Облачные вычисления [Milewski 11] Milewski, B., «Virtual Machines: Virtualizing Virtual Memory», http://corensic. wordpress.com/2011/12/05/virtual-machines-virtualizing-virtual-memory, 2011. [Adamczyk 12] Adamczyk, B., and Chydzinski, A., «Performance Isolation Issues in Network Virtualization in Xen», International Journal on Advances in Networks and Services, 2012. [Hoff 12] Hoff, T., «Pinterest Cut Costs from $54 to $20 Per Hour by Automatically Shutting Down Systems», http://highscalability.com/blog/2012/12/12/pinterest-cut-costs-from-54-to-20-perhour-by-automatically.html, 2012. [Gregg 14b] Gregg, B., «From Clouds to Roots: Performance Analysis at Netflix», http://www. brendangregg.com/blog/2014-09-27/from-clouds-to-roots.html, 2014. [Heo 15] Heo, T., «Control Group v2», Linux documentation, https://www.kernel.org/doc/ Documentation/cgroup-v2.txt, 2015. [Gregg 16a] Gregg, B., «Unikernel Profiling: Flame Graphs from dom0», http://www.brendangregg. com/blog/2016-01-27/unikernel-profiling-from-dom0.html, 2016. [Kadera 16] Kadera, M., «Accelerating the Next 10,000 Clouds», https://www.slideshare.net/Docker/ accelerating-the-next-10000-clouds-by-michael-kadera-intel, 2016. [Borello 17] Borello, G., «Container Isolation Gone Wrong», Sysdig blog, https://sysdig.com/blog/ container-isolation-gone-wrong, 2017. [Goldfuss 17] Goldfuss, A., «Making FlameGraphs with Containerized Java», https://blog. alicegoldfuss.com/making-flamegraphs-with-containerized-java, 2017. [Gregg 17e] Gregg, B., «AWS EC2 Virtualization 2017: Introducing Nitro», http://www.brendangregg. com/blog/2017-11-29/aws-ec2-virtualization-2017.html, 2017. [Gregg 17f] Gregg, B., «The PMCs of EC2: Measuring IPC», http://www.brendangregg.com/ blog/2017-05-04/the-pmcs-of-ec2.html, 2017. [Gregg 17g] Gregg, B., «Container Performance Analysis at DockerCon 2017», http://www. brendangregg.com/blog/2017-05-15/container-performance-analysis-dockercon-2017.html, 2017. [CNI 18] «bandwidth plugin», https://github.com/containernetworking/plugins/blob/master/plugins/ meta/bandwidth/README.md, 2018. [Denis 18] Denis, X., «A Pods Architecture to Allow Shopify to Scale», https://engineering.shopify. com/blogs/engineering/a-pods-architecture-to-allow-shopify-to-scale, 2018. [Leonovich 18] Leonovich, M., «Another reason why your Docker containers may be slow», https:// hackernoon.com/another-reason-why-your-docker-containers-may-be-slowd37207dec27f, 2018. [Borkmann 19] Borkmann, D., and Pumputis, M., «Kube-proxy Removal», https://cilium.io/ blog/2019/08/20/cilium-16/#kubeproxy-removal, 2019. [Cilium 19] «Announcing Hubble — Network, Service & Security Observability for Kubernetes», https://cilium.io/blog/2019/11/19/announcing-hubble/, 2019. [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»1, Addison-Wesley, 2019. Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

11.8. Ссылки  785 [Gregg 19e] Gregg, B., «kvmexits.bt», https://github.com/brendangregg/bpf-perf-tools-book/blob/ master/originals/Ch16_Hypervisors/kvmexits.bt, 2019. [Kwiatkowski 19] Kwiatkowski, A., «Autoscaling in Reality: Lessons Learned from Adaptively Scaling Kubernetes», https://conferences.oreilly.com/velocity/vl-eu/public/schedule/detail/78924, 2019. [Xen 19] «Xen PCI Passthrough», http://wiki.xen.org/wiki/Xen_PCI_Passthrough, 2019. [Agache 20] Agache, A., et al., «Firecracker: Lightweight Virtualization for Serverless Applications», https://www.amazon.science/publications/firecracker-lightweight-virtualization-for-serverlessapplications, 2020.

[Calico 20] «Cloud Native Networking and Network Security», https://github.com/projectcalico/ calico, последнее обновление 2020. [Cilium 20a] «API-aware Networking and Security», https://cilium.io, по состоянию на 2020. [Cilium 20b] «eBPF-based Networking, Security, and Observability», https://github.com/cilium/ cilium, последнее обновление 2020. [Firecracker 20] «Secure and Fast microVMs for Serverless Computing», https://github.com/ firecracker-microvm/firecracker, последнее обновление 2020. [Google 20c] «Google Compute Engine FAQ», https://developers.google.com/compute/docs/ faq#whatis, по состоянию на 2020. [Google 20d] «Analyzes Resource Usage and Performance Characteristics of Running containers», https://github.com/google/cadvisor, последнее обновление 2020. [Kata Containers 20] «Kata Containers», https://katacontainers.io, по состоянию на 2020. [Linux 20m] «mount_namespaces(7)», http://man7.org/linux/man-pages/man7/mount_ namespaces.7.html, по состоянию на 2020. [Weaveworks 20] «Ignite a Firecracker microVM», https://github.com/weaveworks/ignite, последнее обновление 2020. [Kubernetes 20b] «Production-Grade Container Orchestration», https://kubernetes.io, по состоянию на 2020. [Kubernetes 20c] «Tools for Monitoring Resources», https://kubernetes.io/docs/tasks/debugapplication-cluster/resource-usage-monitoring, последнее обновление 2020. [Torvalds 20b] Torvalds, L., «Re: [GIT PULL] x86/mm changes for v5.8», https://lkml.org/ lkml/2020/6/1/1567, 2020. [Xu 20] Xu, P., «iops limit for pod/pvc/pv #92287», https://github.com/kubernetes/kubernetes/ issues/92287, 2020.

Глава 12

БЕНЧМАРКИНГ Есть ложь, есть наглая ложь, и есть метрики производительности. Anon et al., «A Measure of Transaction Processing Power» [Anon 85] Контролируемый бенчмаркинг позволяет сравнивать выбранные варианты, выявлять регрессии и оценивать ограничения производительности еще до того, как они обнаружатся в промышленной среде. Ограничения могут быть обусловлены системными ресурсами, программными ограничениями в виртуализированной среде (облачные вычисления) или ограничениями в целевом приложении. В предыдущих главах были рассмотрены эти компоненты, описаны типы ограничений и инструменты, используемые для их анализа. Кроме того, я показал инструменты для микробенчмаркинга, использующие простые искусственные рабочие нагрузки, — файловый ввод/вывод для тестирования компонентов. Есть также макробенчмаркинг, имитирующий рабочие нагрузки для проверки всей системы. Макробенчмаркинг может включать имитацию рабочей нагрузки или воспроизведение рабочей нагрузки по результатам трассировки. Независимо от выбранного типа бенчмаркинга важно проанализировать полученные результаты и убедиться, что вы понимаете, что именно было оценено. Бенчмарки сообщают лишь то, насколько быстро система может выполнять их, поэтому важно проанализировать результат и определить, насколько он применим к вашей среде. Цели этой главы:

yy yy yy yy yy

познакомить с микро- и макробенчмаркингом; обозначить многочисленные подводные камни бенчмаркинга; познакомить с методологией активного бенчмаркинга; представить чек-лист для проверки результатов бенчмаркинга; показать, как наращивать точность при бенчмаркинге и интерпретации результатов.

В главе рассмотрен бенчмаркинг в целом. Даны рекомендации и методологии, которые помогут избежать распространенных ошибок и точно протестировать свои системы. Информация из этой главы пригодится вам при интерпретации других результатов, в том числе из промышленных бенчмарков или бенчмарков вендора.

12.1. Основы  787

12.1. ОСНОВЫ В этом разделе описаны приемы эффективного бенчмаркинга и обобщены распространенные ошибки.

12.1.1. Причины Бенчмаркинг можно проводить по следующим причинам:

yy Проектирование системы: сравнение различных систем, компонентов или

приложений. Бенчмаркинг коммерческих продуктов позволяет получить данные, помогающие принять решение о покупке, в частности соотношение цена/ производительность для имеющихся вариантов1. В некоторых случаях могут использоваться опубликованные бенчмарки, что позволит избежать самостоятельного проведения бенчмаркинга.

yy Проверка концепции: для проверки производительности программного или аппаратного обеспечения под нагрузкой, перед покупкой или развертыванием в промышленной среде.

yy Настройка: тестирование настраиваемых параметров и конфигураций с целью определения тех из них, которые заслуживают дальнейшего изучения с учетом рабочей нагрузки.

yy Разработка: для нерегрессионного тестирования и определения пределов при

разработке продукта. Нерегрессионное тестирование можно проводить с помощью пакета бенчмарков, и делать это регулярно, чтобы любое ухудшение производительности обнаруживалось на ранней стадии. Это позволит быстро сопоставить результаты с изменениями в продукте. В случае определения пределов бенчмаркинг можно использовать для доведения нагрузки на продукт до предела во время разработки и выяснить, куда направить усилия инженеров, чтобы повысить производительности продукта.

yy Планирование емкости: определение ограничений системы и приложений для планирования емкости либо получение данных для моделирования производительности или прямого определения пределов емкости.

yy Устранение неисправностей (troubleshooting): для проверки способности компо-

нентов работать с максимальной производительностью, например, тестирование максимальной пропускной способности сети между хостами, чтобы проверить, не возникнет ли проблема с сетью.

yy Маркетинг: определение максимальной производительности продукта для использования в рекламе (также называется бенчмаркетингом — benchmarketing).

Несмотря на широкое использование соотношения цена/производительность, мне кажется, что соотношение производительность/цена выглядит понятнее. С точки зрения математики (а не просто привычного высказывания) такое соотношение имеет более привычную интерпретацию «чем больше, тем лучше» для людей, оценивающих показатели производительности.

1

788  Глава 12. Бенчмаркинг Бенчмаркинг оборудования в корпоративных средах для проверки концепции (proofof-concept, PoC) может стать важным этапом перед крупной закупкой оборудования и длиться неделями или даже месяцами. Сюда входит время на доставку, монтаж и прокладку кабельных линий, а также на установку ОС перед тестированием. Эта процедура может проводиться раз в год или в два, с выпуском нового оборудования. В облачных средах ресурсы доступны по требованию без дорогостоящих инвестиций в оборудование, и при необходимости их количество можно быстро изменить (повторное развертывание на экземплярах различных типов). Эти среды тоже требуют долгосрочных инвестиций при выборе языка программирования и ОС, баз данных, веб-серверов и балансировщиков нагрузки. Некоторые из этих компонентов бывает сложно заменить на ходу. Бенчмаркинг помогает выяснить, насколько хорошо различные компоненты масштабируются при необходимости. Модель облачных вычислений упрощает бенчмаркинг: она позволяет за считаные минуты создать крупномасштабную среду для запуска бенчмарка, а затем уничтожить его, и все это обойдется очень недорого. Обратите внимание, что отказоустойчивая и распределенная облачная среда также упрощает экспериментирование: с появлением новых типов экземпляров среда может быть немедленно подвергнута бенчмаркингу с помощью промышленных рабочих нагрузок, минуя традиционный бенчмаркинг. В этом сценарии бенчмаркинг можно использовать для объяснения различий путем сравнения производительности компонентов. Например, команда поддержки производительности в Netflix имеет автоматизированное ПО для анализа новых типов экземпляров с помощью различных микробенчмарков. Оно автоматически собирает системные статистики и профили процессоров, что позволяет проанализировать и объяснить любые обнаруженные различия.

12.1.2. Эффективный бенчмаркинг Бенчмаркинг — удивительно сложная задача с массой возможностей для ошибок и упущений. Как отмечено в статье «A Nine Year Study of File System and Storage Benchmarking» [Traeger 08]: Работая над этой статьей, мы изучили 415 бенчмарков файловых систем и хранилищ из 106 недавних статей. Мы обнаружили, что большинство популярных бенчмарков дают неверную картину, и многие исследовательские работы не позволяют получить четкое представление об истинной производительности. В статье есть рекомендации относительно того, что можно сделать. В частности, оценки бенчмаркинга должны объяснять, что было протестировано и почему, и они должны включать некоторый анализ ожидаемого поведения системы. Вот суть хорошего бенчмаркинга [Smaalders 06]:

yy повторяемость: чтобы упростить сравнение; yy наблюдаемость: чтобы проанализировать и объяснить результаты;

12.1. Основы  789

yy переносимость: чтобы провести сравнительный анализ конкурентов и различных версий продукта;

yy простота представления результатов: чтобы каждый мог понять их; yy реалистичность: чтобы измерения отражали реальную картину, которую видят клиенты;

yy возможность запуска: чтобы разработчики могли быстро протестировать свои изменения.

При сравнении различных систем с целью их приобретения стоит добавить еще одну характеристику: соотношение цена/производительность. Цену можно количественно определить как пятилетние капиталовложения в оборудование [Anon 85]. Эффективность также зависит от того, как используются бенчмарки: нужно проводить анализ и на его основе делать выводы.

Анализ бенчмаркинга При использовании бенчмарков вы должны понимать:

yy yy yy yy

что тестируется; какие есть ограничивающие факторы; какие есть возмущающие факторы, которые могут повлиять на результаты; какие выводы можно сделать на основе результатов.

Для этого нужно глубокое понимание того, что делает ПО, осуществляющее бенчмарки, как реагирует система на его действия и как результаты соотносятся с целевой средой. При наличии инструмента бенчмаркинга и доступа к системе эти потребности лучше всего удовлетворяются путем анализа производительности системы во время бенчмаркинга. Распространенная ошибка — когда бенчмарки выполняются младшим персоналом, а затем для интерпретации результатов приглашают экспертов по производительности. Намного лучше привлекать экспертов уже на этапе бенчмаркинга — так они смогут проанализировать работу системы в обычном режиме, в том числе сделать углубленный анализ для объяснения и количественной оценки ограничивающего фактора (факторов). Ниже приведен интересный пример анализа: Изучая производительность получившейся реализации TCP/IP, мы провели эксперимент и передали 4 Мбайт данных между двумя пользовательскими процессами, действующими на разных машинах. Передача выполнялась 1024-байтными записями, инкапсулированными в 1068-байтные пакеты Ethernet. Отправка данных с 11/750 на 11/780 по протоколу TCP/IP заняла 28 с. Сюда вошло все время, необходимое для установки и разрыва соединений и для обеспечения

790  Глава 12. Бенчмаркинг пропускной способности 1,2 Мбод. Во время эксперимента процессор на 11/750 был нагружен на 100 %, а на 11/780 — на 70 %. Время, затраченное системой на обработку данных, распределилось между обработкой пакетов Ethernet (20 %), IP-пакетов (10 %), TCP-пакетов (30 %), вычислением контрольных сумм (25 %) и обработкой обращений пользователя к системным вызовам (15 %). При этом никакой этап обработки не оказался доминирующим по затраченному процессорному времени. Здесь описаны ограничивающие факторы («процессор на 11/750 был нагружен на 100 %»1), а затем объясняется, какие компоненты ядра внесли вклад в это ограничение. Кроме того, выполнить такой анализ и обобщить затраты процессорного времени разными компонентами ядра стало проще лишь недавно благодаря появлению флейм-графиков. Эти слова написаны задолго до появления флеймграфиков и принадлежат Биллу Джою, описавшему оригинальный стек TCP/IP BSD в 1981 году [Joy 81]. Нередко эффективнее разработать и использовать собственное ПО для бенчмаркинга вместо стандартных инструментов или, по крайней мере, генераторы нагрузки. Их можно сделать специализированными, сосредоточив внимание только на том, что важно для вашего теста; это ускорит их анализ и отладку. Иногда инструменты бенчмаркинга или система могут быть недоступны, как, например, при изучении бенчмарков, проведенных другими пользователями. В таких случаях пройдите по пунктам списка из начала этого подраздела. Основываясь на доступных материалах, ответьте на вопросы: какая системная среда используется? как она настроена? Рассмотрите и другие вопросы, которые приведены в разделе 12.4 «Вопросы о бенчмаркинге».

12.1.3. Проблемы бенчмаркинга В подразделах ниже представлен чек-лист различных ошибок при бенчмаркинге — оплошностей, заблуждений и неправильных действий. Также я рассказал о том, как их избежать. О том, как проводить бенчмаркинг, речь идет в разделе 12.3 «Методология».

1. Случайное тестирование Бенчмаркинг — это не то занятие, о котором можно сразу забыть. Инструменты бенчмаркинга сообщают цифры, которые могут отражать совсем не то, что вы думаете, и ваши выводы на их основе могут быть ложными. Вот пример: Случайный бенчмаркинг: вы тестируете A, но на самом деле получаете характеристики B и делаете вывод, что протестировали C. 11/750 — это сокращение от VAX-11/750, мини-компьютера, произведенного компанией DEC в 1980 году.

1

12.1. Основы  791 Хороший бенчмаркинг требует тщательности, чтобы проверить именно то, что предполагалось, и понимания того, что было протестировано, чтобы можно было сделать обоснованные выводы. Например, многие инструменты заявляют или подразумевают, что они оценивают производительность диска, но на самом деле тестируют производительность файловой системы. Разница в оценках может достигать порядков, потому что файловые системы используют кэширование и буферизацию, подменяя дисковый ввод/вывод операциями с памятью. Даже если инструмент бенчмаркинга работает правильно и тестирует файловую систему, ваши выводы о дисках будут совершенно неверными. Бенчмарки сложны для понимания, особенно новичками, у которых недостаточно опыта, чтобы понять, подозрительны ли цифры или нет. Если вдруг комнатный термометр покажет температуру в 100 градусов, вы сразу поймете, что что-то не так. Но так нельзя сказать о бенчмарках, сообщающих числа, которые ни о чем вам не говорят.

2. Слепая вера Заманчиво довериться популярному инструменту бенчмаркинга, особенно если он давний и у него открытый исходный код. Это распространенное заблуждение, что популярность равна достоверности. Оно известно как argumentsum ad populum (лат. «аргумент к народу»1). Анализ используемых бенчмарков требует времени и специальных знаний. А если какой-то бенчмарк популярен, то анализ того, что обязательно должно быть достоверным, может показаться пустой тратой времени и сил. Если бы популярный бенчмарк продвигался одной из ведущих компаний в индустрии IT, вы бы доверились ему? В прошлом уже были случаи, когда рекламируемый микробенчмарк, широко известный среди опытных перформанс-инженеров, оказывался ошибочным. К сожалению, простого способа избежать этого нет (я пробовал). И проблема здесь даже не в ПО для бенчмаркинга (хотя ошибки случаются), а в интерпретации результатов.

3. Числа без анализа Простые результаты бенчмаркинга, не содержащие аналитических деталей, могут служить признаком неопытности автора, наивно полагающего, что результаты заслуживают доверия и окончательные. Часто получение результатов — лишь начальный этап исследований, когда можно обнаружить, что результаты ошибочны или запутывают.

1

Вид заведомо ошибочной логической аргументации, основанной на мнении, что большинство всегда право. — Примеч. пер.

792  Глава 12. Бенчмаркинг Каждое число бенчмарка должно сопровождаться описанием обнаруженного предела и выполненного анализа: Если вы потратили на изучение бенчмарка меньше недели, то, скорее всего, поступили неправильно. Большая часть этой книги посвящена анализу производительности, который следует проводить во время бенчмаркинга. Когда у вас нет времени на тщательный анализ, рекомендую перечислить предположения, на проверку которых у вас не было времени, и включить этот перечень в результаты, например:

yy предполагается, что инструмент бенчмаркинга не содержит ошибок; yy предполагается, что тест дискового ввода/вывода действительно измеряет характеристики дискового ввода/вывода;

yy предполагается, что инструмент бенчмаркинга довел нагрузку дискового ввода/ вывода до предела;

yy предполагается, что этот вид дискового ввода/вывода подходит для этого приложения.

Такой перечень вполне может стать списком заданий для дальнейшей проверки, если позже результаты бенчмарка будут признаны достаточно важными, чтобы провести дополнительные исследования.

4. Сложные инструменты бенчмаркинга Важно, чтобы инструмент бенчмаркинга не мешал анализу производительности своей сложностью. В идеале программа должна иметь открытый исходный код, который можно изучить, и быть достаточно короткой, чтобы ее можно было быстро прочитать и понять. На роль микробенчмарков рекомендую выбирать программы, написанные на языке C. Для имитации поведения клиентского ПО рекомендуется использовать тот же язык программирования, на котором написан клиент, чтобы минимизировать различия. Распространенная проблема — бенчмаркинг бенчмарка, когда сообщаемый результат ограничивается самим бенчмарком. Особенно часто это объясняется однопоточной природой ПО для бенчмаркинга. Сложные наборы бенчмарков могут затруднить анализ из-за большого объема кода, который нужно понять и проанализировать.

5. Тестирование не того, что надо Несмотря на большое количество инструментов бенчмаркинга различных рабочих нагрузок, многие из них могут не подходить для целевого приложения. Например, распространенная ошибка — бенчмаркинг диска с использованием доступных инструментов, даже при том, что потребности рабочей нагрузки в целевой

12.1. Основы  793 среде будут полностью удовлетворяться из кэша файловой системы и она не будет зависеть от дискового ввода/вывода. Команда инженеров, разрабатывающая продукт, может взять за эталон конкретный бенчмарк и все силы направить на повышение производительности, измеряемой им. Но если бенчмарк имеет мало общего с фактической рабочей нагрузкой, то усилия инженеров будут направлены на оптимизацию не того поведения [Smaalders 06]. В реальной промышленной среде можно использовать методологию определения характеристик фактической рабочей нагрузки (описанную в предыдущих главах), чтобы оценить ее структуру и измерить показатели от устройства ввода/вывода до обработки запросов в приложении. Эти измерения помогут выбрать наиболее подходящие бенчмарки. Если промышленная среда недоступна для анализа, то можно сымитировать ее или смоделировать предполагаемую рабочую нагрузку. Также познакомьте с данными бенчмарков целевую аудиторию и узнайте, согласна ли она с результатами тестов. Возможно, бенчмарк когда-то соответствовал рабочей нагрузке, но из-за того, что не обновлялся годами, сейчас может тестировать совсем не то.

6. Игнорирование особенностей среды Сопоставимы ли промышленная и тестовая среды? Представьте, что нужно оценить новую базу данных. Вы подготавливаете тестовый сервер и запускаете бенчмарк базы данных, а потом узнаете, что упустили из виду один важный аспект: тестовый сервер действует с настройками по умолчанию, с файловой системой по умолчанию и т. д. Обычно промышленные серверы баз данных настраиваются на выполнение максимального количества операций дискового ввода/вывода в секунду, поэтому тестирование в системе с настройками по умолчанию даст результаты, далекие от реальности: вы не сделали первый важный шаг — не изучили особенности промышленной среды.

7. Игнорирование ошибок Тот факт, что инструмент бенчмаркинга дает какой-то результат, еще не означает, что результат отражает успешное выполнение теста. Некоторые или даже все запросы могли привести к ошибке. Эта проблема связана с предыдущими ошибками и упущениями, но она настолько распространена, что ее стоит упомянуть отдельно. Об этом мне напомнили во время бенчмаркинга веб-сервера. Команда, проводившая тестирование, сообщила, что средняя задержка веб-сервера оказалась слишком высокой — более одной секунды в среднем. Одна секунда? В ходе короткого анализа удалось выяснить, что пошло не так: во время тестирования веб-сервер вообще ничего не делал, так как все запросы блокировались брандмауэром. Все запросы. Получившаяся задержка — это время, которое требовалось тестовому клиенту, чтобы прервать ожидание ответа по тайм-ауту!

794  Глава 12. Бенчмаркинг

8. Игнорирование изменчивости Инструменты бенчмаркинга, особенно микробенчмаркинга, часто применяют стабильную и последовательную рабочую нагрузку, основанную на среднем значении, вычисленном по серии измерений характеристик, произведенных в разное время суток или за определенный интервал. Например, рабочая нагрузка может выполнять в среднем 500 операций чтения с диска в секунду и 50 операций записи. Инструмент бенчмаркинга, опираясь на эти значения, может смоделировать эту частоту или соотношение 10:1 операций чтения/записи, чтобы протестировать более высокую нагрузку. Такой подход игнорирует дисперсию: скорость операций может меняться. Меняться могут и типы операций, а некоторые из них могут выполняться независимо друг от друга. Например, запись данных может производиться пакетами раз в 10 с (асинхронное выталкивание буферов отложенной записи), тогда как характер синхронных операций чтения почти не меняется. Операции пакетной записи могут вызвать проблемы в промышленной среде, например, вынуждая операции чтения простаивать долгое время в очереди, которые не воспроизводятся бенчмарком, соблюдающим постоянную среднюю нагрузку. Одно из возможных решений для моделирования дисперсии — использовать в бенчмарках модели Маркова, отражающие вероятность того, что за одной операцией записи может последовать другая.

9. Игнорирование возмущающих факторов Подумайте, какие внешние возмущающие факторы могут повлиять на результаты. Будут ли во время бенчмаркинга выполняться какие-либо запланированные по времени действия — резервное копирование или работа агентов мониторинга, собирающих статистику раз в минуту? В облачных средах таким возмущающим фактором также могут быть действия невидимых арендаторов в той же хост-системе. На практике для сглаживания возмущений часто используется стратегия увеличения продолжительности выполнения бенчмарка — до нескольких минут вместо секунд. Обычно продолжительность бенчмаркинга не должна быть меньше одной секунды. Кратковременные тесты могут не столкнуться с прерываниями устройств (когда поток приостанавливается на время выполнения процедуры обработки прерывания), с решениями планировщика процессоров (ожидание в очереди перед переносом для сохранения привязки к процессору) и с эффектами разогрева кэша процессора. Попробуйте запустить бенчмарк несколько раз и проверьте стандартное отклонение — оно должно быть как можно меньше. Также соберите данные, которые помогут изучить возмущения, если они есть. Пример таких данных — распределение задержек операций в дополнение к общему времени выполнения бенчмарка, чтобы можно было увидеть выбросы и выяснить их причины.

12.1. Основы  795

10. Изменение множества факторов Сравнивая результаты двух бенчмарков, будьте внимательны и попробуйте опознать все факторы, отличающие их друг от друга. Например, тестируя два хоста в сети, выясните: идентична ли сеть между ними? Не находится ли один хост на большем расстоянии и в более медленной или более нагруженной сети? Любые такие дополнительные факторы могут усложнить анализ результатов тестирования. В облачных средах для бенчмарков иногда создают отдельные экземпляры, которые затем уничтожаются. При этом не учитываются многие факторы: экземпляры могут создаваться в более быстрых или медленных системах или в системах с более высокой нагрузкой и конкуренцией со стороны других арендаторов. Рекомендую протестировать несколько экземпляров и взять медианное значение (или, что еще лучше, записать распределение), чтобы исключить выбросы, вызванные тестированием в необычно быстрой или необычно медленной системе.

11. Парадокс бенчмарка Потенциальные покупатели часто используют бенчмарки для оценки продукта перед покупкой, и они бывают настолько неточны, что с таким же успехом можно просто подбросить монетку. Один продавец как-то сказал мне, что был бы доволен таким шансом: выигрыш в половине случаев оценки его продукта вполне соответствовал бы его планам продаж. Но игнорирование бенчмарков — это подводный камень бенчмаркинга, и на практике шансы гораздо хуже: Если шансы вашего продукта на победу в бенчмарке равны 50/50, то это означает, что в большинстве случаев вы будете проигрывать. [Gregg 14c] Этот кажущийся парадокс можно объяснить простой вероятностью. Покупая продукт, ориентируясь на производительность, клиенты хотят быть уверены в том, что он действительно обладает высокой производительностью. Они могут запустить несколько бенчмарков, желая во всех них увидеть продукт на первом месте. Если бенчмарк имеет вероятность выигрыша 50 %, то: Вероятность выигрыша в трех тестах = 0,5 × 0,5 × 0,5 = 0,125 = 12,5 %. Чем больше бенчмарков, тем меньше шансов выиграть их все.

12. Бенчмаркинг конкурентов Ваш отдел маркетинга наверняка хотел бы иметь результаты бенчмарков, показывающие, насколько ваш продукт превосходит конкурентов. Это намного сложнее, чем кажется.

796  Глава 12. Бенчмаркинг Когда покупатели выбирают товар, они пробуют его не пять минут, а анализируют и настраивают месяцами, чтобы добиться максимальной производительности. Они устраняют самые серьезные проблемы в первые несколько недель. У вас нет нескольких недель, чтобы проанализировать и настроить продукт, конкурирующий с вашим. В отведенное время вы можете опробовать только ненастроенный продукт и, следовательно, получить результаты, далекие от реальности. Клиенты вашего конкурента, на которых направлена эта маркетинговая активность, вполне могут увидеть, что вы опубликовали результаты, полученные на ненастроенном продукте. Тогда ваша компания потеряет доверие у тех самых людей, на которых пыталась произвести впечатление. Если понадобится провести сравнительный анализ с конкурирующим продуктом, то придется потратить довольно много времени на его настройку. Проанализируйте производительность, используя методы, описанные в предыдущих главах. Поищите рекомендации по настройке на форумах клиентов и в базах данных ошибок. Возможно, вы даже решите привлечь сторонних экспертов для настройки системы. Затем приложите те же усилия для настройки своего продукта и только потом проводите сравнительный бенчмаркинг.

13. Огонь по своим При бенчмаркинге собственных продуктов приложите все силы, чтобы убедиться, что тестирование выполняется в самой эффективной системе и с оптимальной конфигурацией, а производительность системы доведена до предела. Поделитесь результатами с командой инженеров перед публикацией. Они могут заметить ошибки в конфигурации, которые вы пропустили. А если вы работаете в такой команде инженеров, внимательно следите за результатами бенчмаркинга — независимо от того, делает ли его ваша компания или подрядчик, — и помогайте коллегам. Однажды я наблюдал, как команда инженеров упорно разрабатывала высокопроизводительный продукт, основанный на применении новой технологии, которая еще не была задокументирована. Для запуска продукта команде по бенчмаркингу предложили представить цифры. Они не понимали новой технологии (так как она не была задокументирована) и использовали неправильные настройки, а затем опубликовали цифры, не отражающие все преимущества продукта. Иногда система может быть настроена правильно, но работает не на пределе возможностей. Задайте вопрос: что является узким местом для этого бенчмарка? Это может быть физический ресурс (например, процессоры, диски или внутренние соединения) с уровнем потребления, достигшим 100 %, который можно выявить при анализе. См. раздел 12.3.2 «Активный бенчмаркинг». Еще одна проблема «огня по своим» возникает при бенчмаркинге старых версий ПО, имеющих проблемы с производительностью, которые были исправлены в более поздних версиях, или при тестировании на оборудовании с ограниченными возможностями, доступном на тот момент, но дающем не лучший результат. В большинстве случаев потенциальные клиенты склонны предполагать, что любые результаты,

12.1. Основы  797 опубликованные компанией, показывают наилучшую возможную производительность. Мало кто допускает, что эти результаты могли быть получены не в лучших условиях.

14. Бенчмарки, вводящие в заблуждение Вводящие в заблуждение бенчмарки — обычное дело в индустрии. Это может быть обусловлено непреднамеренным ограничением информации о том, что на самом деле измеряет бенчмарк, или намеренным исключением некоторой информации из публикации. Часто результаты бенчмарка остаются технически правильными, но представляются клиенту в ложном свете. Рассмотрим такую гипотетическую ситуацию: производитель достигает фантастического результата, создав нестандартный продукт, который получился настолько дорогим, что ни один реальный покупатель не купит его. Цена продукта не раскрывается вместе с результатами бенчмарков, в которых основное внимание уделяется показателям, не связанным с соотношением цена/производительность. Отдел маркетинга публикует неоднозначное обобщение результатов («Мы в два раза быстрее!»), связывая его в сознании клиентов либо с компанией в целом, либо с линейкой продуктов. Это случай исключения деталей с целью произвести благоприятное впечатление. Это нельзя назвать обманом — цифры не поддельные, но это ложь из-за неполноты информации. Впрочем, даже такие результаты могут быть полезны как ориентир, показывающий верхнюю границу производительности. Это значения, превышения которых не следует ожидать (за исключением случаев огня по своим). Рассмотрим другую гипотетическую ситуацию: у отдела маркетинга есть бюджет, который можно потратить на рекламную кампанию, и они хотят получить хорошие результаты бенчмарков. Они привлекают несколько подрядчиков для бенчмаркинга своего продукта и выбирают наилучший результат из полученных. Эти подрядчики выбираются не на основе опыта, а по способности дать результаты быстро и недорого. Фактически отсутствие опыта в такой ситуации оказывается выгоднее: чем больше результаты отклоняются от реальности, тем лучше — в идеале один из них будет сильно отклоняться в положительную сторону! При использовании результатов, публикуемых производителем, внимательно проверьте, в какой системе производилось тестирование, какие типы дисков и какие сетевые интерфейсы использовались, какие настройки были выбраны, а также другие факторы. Чтобы узнать, чего следует опасаться, см. раздел 12.4 «Вопросы о бенчмаркинге».

15. Специализация под бенчмарки Специализация под бенчмарки — это когда поставщик изучает популярный или отраслевой бенчмарк, а затем проектирует продукт так, чтобы тот показал хорошие результаты в этом бенчмарке, не принимая во внимание фактическую производительность клиента. Это также называется оптимизацией для бенчмарка.

798  Глава 12. Бенчмаркинг Термин «специализация под бенчмарки» вошел в употребление в 1993 году вместе с бенчмарком TPC-A, как описано на странице с историей Transaction Processing Performance Council (TPC) [Shanley 98]: Консалтинговая компания Standish Group из Массачусетса обвинила Oracle в добавлении специальной возможности (дискретных транзакций) в программное обеспечение баз данных с единственной целью — завысить результаты TPC-A. Standish Group заявила, что Oracle «нарушила дух TPC», потому что дискретные транзакции не используются типичными заказчиками, и следовательно, являются специализацией под бенчмарк. Oracle отвергла это обвинение, заявив, не без оснований, что они следовали букве закона при составлении спецификаций бенчмарков. Компания утверждала, что поскольку специализация под бенчмарки, не говоря уже о духе TPC, никак не оговаривается в спецификациях TPC, то несправедливо обвинять их в каких-либо нарушениях. TPC добавила специальный пункт, запрещающий специализацию: Запрещены все «специализированные» реализации, показывающие повышенную производительность при бенчмаркинге, но не реальную производительность или цены. Поскольку TPC сосредоточена на соотношении цена/производительность, другой стратегией завышения цифр может быть использование специальных цен — больших скидок, которые на самом деле не получит ни один покупатель. Подобно специализации ПО, специализация цен не соответствует действительности, когда реальный клиент покупает систему. Компания TPC учла это в своих требованиях [TPC 19a]: Согласно спецификациям TPC, общая цена не должна отличаться больше чем на 2 % от цены, которую покупатель заплатит за конфигурацию. Эти примеры объясняют понятие специализации под бенчмарки, но TPC учла их в своих спецификациях много лет назад, и сейчас об этом уже можно не беспо­ коиться.

16. Мошенничество Последняя проблема в бенчмаркинге — мошенничество: публикация фальшивых результатов. К счастью, это случается очень редко, по крайней мере мне не приходилось сталкиваться с выдуманными цифрами, даже в самой кровожадной конкурентной борьбе.

12.2. ТИПЫ БЕНЧМАРКИНГА На рис. 12.1 показан спектр типов бенчмарков, в зависимости от тестируемой рабочей нагрузки. Промышленная нагрузка также включена в этот спектр.

12.2. Типы бенчмаркинга  799

Моделирование Микробенчмарк Искусственная

Промышленная

Воспроизведение Рабочая нагрузка

Естественная

Рис. 12.1. Типы бенчмарков В разделах ниже описаны три типа бенчмаркинга: микробенчмаркинг, моделирование и трассировка/воспроизведение. Рассмотрены и стандартные отраслевые бенчмарки.

12.2.1. Микробенчмаркинг В микробенчмаркинге используются искусственные рабочие нагрузки, тестирующие операции определенного типа, например одного типа операций ввода/вывода с файловой системой, запросов к базе данных, машинных инструкций или одного системного вызова. Преимущество микробенчмарков в их простоте: сужение круга задействованных компонентов и путей в коде упрощает изучение цели и позволяет быстро устранить коренные причины уменьшения производительности. Микробенчмарки обычно легко повторить, потому что влияние других компонентов сведено к минимуму и их легко использовать в разных системах. А поскольку микробенчмарки намеренно сделаны искусственными, их трудно спутать с моделированием реальных рабочих нагрузок. Чтобы использовать результаты микробенчмаркинга, их нужно сопоставить с целевой рабочей нагрузкой. Микробенчмарк может тестировать несколько параметров, но только один или два могут быть актуальными. Анализ производительности или моделирование целевой системы помогают определить, какие результаты микробенчмаркинга подходят и в какой степени. Вот примеры инструментов для микробенчмаркинга по типам ресурсов, упомянутые в предыдущих главах:

yy yy yy yy yy

процессоры: SysBench; ввод/вывод с памятью: lmbench (упоминается в главе 6 «Процессоры»); файловые системы: fio; диски: hdparm, dd или fio с прямым вводом/выводом; сети: iperf.

Есть множество других инструментов для бенчмаркинга, но помните предупреждение [Traeger 08]: «Большинство популярных бенчмарков несовершенны». При желании можно разработать свои инструменты. Старайтесь делать их как можно более простыми и определяйте атрибуты рабочей нагрузки, которые можно тестировать индивидуально. (Подробнее об этом рассказывается в разделе 12.3.6

800  Глава 12. Бенчмаркинг «Собственные тесты».) Используйте сторонние инструменты, чтобы убедиться, что они выполняют те операции, которые должны выполнять.

Пример дизайна бенчмарка Рассмотрим разработку микробенчмарка файловой системы для тестирования следующих характеристик: производительности последовательного и произвольного ввода/вывода; ввода/вывода с блоками разных размеров; и наконец, в разных направлениях (чтение и запись). В табл. 12.1 перечислены пять примеров тестов и их назначение. Таблица 12.1. Примеры тестов для микробенчмаркинга файловой системы №

Тест

Назначение

1

Последовательное чтение блоками по 512 байт1

Протестировать максимальное (реалистичное) количество операций ввода/вывода в секунду

2

Последовательное чтение блоками по 1 Мбайт2

Протестировать максимальную пропускную способность чтения

3

Последовательная запись блоками по 1 Мбайт

Протестировать максимальную пропускную способность записи

4

Произвольное чтение блоками по 512 байт

Протестировать эффект произвольного ввода/ вывода

5

Произвольная запись блоками по 512 байт

Протестировать эффект повторной записи

При желании можно добавить другие тесты. Эти тесты можно разнообразить применением двух дополнительных факторов:

yy Размер рабочего набора. Объем данных, к которым осуществляется доступ (например, общий размер файла):

Цель — максимизировать количество операций ввода/вывода в секунду за счет использования блоков небольшого размера. Для этой цели лучше было бы использовать блоки размером в 1 байт, но диски по крайней мере округляют размеры блоков до размеров секторов (512 байт или 4 Кбайт).

1

2

Цель — максимизировать пропускную способность за счет уменьшения количества операций ввода/вывода и использования блоков большого размера (это позволяет уменьшить время, расходуемое на инициализацию ввода/вывода). Теоретически чем больше размер блока, тем лучше, но файловая система может иметь свою «золотую середину», обусловленную особенностями механизма распределения памяти в ядре, размерами страниц памяти и другими деталями. Например, ядро Solaris показывает наивысшую производительность при работе с блоками ввода/вывода 128 Кбайт, так как это самый большой размер блоков в кэше ядра (блоки большего размера размещаются в специальной области для блоков очень большого размера, которая обслуживается с меньшей производительностью).

12.2. Типы бенчмаркинга  801 • когда рабочий размер намного меньше объема основной памяти, данные могут полностью уместиться в кэше файловой системы, и в этом случае проводится бенчмаркинг производительности ПО файловой системы; • когда рабочий размер намного больше объема основной памяти, влияние кэша файловой системы минимально, и в этом случае проводится бенчмаркинг дискового ввода/вывода.

yy Количество потоков выполнения. Предполагается небольшой размер рабочего набора:

• с одним потоком выполнения проверяется производительность файловой системы с учетом текущей тактовой частоты процессора; • с несколькими потоками, когда их количество достаточно велико для насыщения всех процессоров, проверяется максимальная производительность системы, файловой системы и процессоров. Их можно быстро комбинировать и масштабировать, чтобы сформировать большую матрицу тестов. Также можно использовать методы статистического анализа, чтобы сократить требуемый набор для тестирования. Создание бенчмарков, ориентированных на максимальную скорость, называется «солнечным» тестированием производительности. Чтобы не упустить из виду проблемы, также можно выполнить «пасмурное» и «дождливое» тестирование — они предусматривают тестирование в неидеальных ситуациях, включая конфликты, возмущения и переменчивость рабочей нагрузки.

12.2.2. Моделирование Многие бенчмарки имитируют рабочие нагрузки, создаваемые клиентскими приложениями. Иногда их называют макробенчмарками производительности. Они могут создаваться на основе характеристик рабочей нагрузки в промышленной среде (см. главу 2 «Методологии»). Например, вы можете выяснить, что рабочая нагрузка на NFS в промышленной среде складывается из следующих типов операций: чтение — 40 %; запись — 7 %; чтение атрибутов файлов — 19 %; чтение содержимого каталогов — 1 %; и т. д. Аналогично можно измерить и сымитировать другие характеристики. Моделирование позволяет воспроизвести рабочую нагрузку, достаточно похожую на создаваемую клиентами в реальности, чтобы из этого можно было извлечь пользу. Моделирование может учитывать множество факторов, на изучение которых с помощью микробенчмаркинга потребуется много времени. Кроме того, моделирование может включать эффекты сложных взаимодействий в системе, которые могут быть полностью упущены из виду при микробенчмаркинге. Примеры моделирования — бенчмарки Whetstone и Dhrystone для оценки производительности процессоров, представленные в главе 6 «Процессоры». Whetstone был разработан в 1972 году для моделирования научных вычислительных нагрузок того времени. Dhrystone, созданный в 1984 году, тоже моделирует рабочие нагрузки того времени, связанные с целочисленными вычислениями.

802  Глава 12. Бенчмаркинг Многие компании моделируют HTTP-нагрузку, создаваемую клиентами, с помощью собственного или стороннего ПО (в числе примеров можно назвать wrk [Glozer 19], siege [Fulmer 12] и hey [Dogan 20]). Их можно использовать для оценки влияния изменений в программном или аппаратном обеспечении, а также для моделирования пиковой нагрузки (например, наплыва покупателей в периоды распродаж на маркетплейсе), чтобы выявить узкие места, проанализировать их и устранить. Моделирование рабочей нагрузки может проводиться без учета сохранения состояния, когда каждый следующий запрос к серверу не связан с предыдущим. Например, описанную выше рабочую нагрузку на сервер NFS можно смоделировать путем запроса серии операций, когда тип каждой операции выбирается случайным образом на основе измеренной вероятности. Моделирование также может учитывать сохранение состояния, когда каждый следующий запрос зависит от состояния клиента или, по меньшей мере, от предыдущего запроса. Вы можете обнаружить, что операции чтения и записи NFS, как правило, поступают группами и вероятность записи после записи намного выше, чем записи после чтения. Такую рабочую нагрузку лучше всего моделировать с помощью модели Маркова, представив запросы как состояния и измерив вероятность переходов между состояниями [Jain 91]. Проблема заключается в том, что моделирование может не учитывать дисперсию, как описано в разделе 12.1.3 «Проблемы бенчмаркинга». Закономерности работы клиентов тоже могут меняться со временем, что требует обновления и корректировки моделей, чтобы они оставались актуальными. Но такая корректировка может вызвать сопротивление, если уже есть опубликованные результаты, полученные с использованием более старой версии бенчмарка, которые нельзя использовать для сравнения с новой версией.

12.2.3. Воспроизведение Третий тип бенчмаркинга основан на воспроизведении журнала трассировки в целевой системе для оценки ее производительности на реальных зафиксированных клиентских операциях. Звучит захватывающе — казалось бы, такой способ ничуть не хуже, чем тестирование в промышленной среде, не так ли? Но не все так гладко: когда характеристики сервера и величины задержек меняются, зафиксированная рабочая нагрузка вряд ли естественным образом отреагирует на эти различия, и в результате воспроизведение может оказаться не лучше моделирования. Если возлагать слишком большие надежды на этот метод, можно сильно разочароваться. Рассмотрим гипотетическую ситуацию: заказчик рассматривает возможность модернизации инфраструктуры хранения. Он выполнил трассировку текущей промышленной нагрузки и воспроизвел ее на новом оборудовании. К сожалению, производительность оказалась хуже, что может привести к снижению продаж. Проблема: трассировка/воспроизведение выполнялись на уровне дискового ввода/вывода. В старой системе использовались диски со скоростью вращения 10 000 об/мин, а в новой — более медленные диски со скоростью вращения 7200 об/мин. Но в новой

12.2. Типы бенчмаркинга  803 системе объем кэша файловой системы в 16 раз больше и процессоры намного быстрее. Фактическая производительность должна вырасти, потому что данные извлекались бы в основном из кэша, чего не учитывает воспроизведение событий обращения к дискам. Это пример «тестирования не того, что надо», но могут быть и другие тонкие временные эффекты, способные ухудшить ситуацию даже при правильно проведенных трассировке/воспроизведении. Как и во всех бенчмарках, очень важно анализировать и понимать происходящее.

12.2.4. Отраслевые стандарты Стандартные отраслевые бенчмарки доступны в независимых организациях, которые стремятся добиться максимальной согласованности и актуальности бенчмарков. Обычно они имеют вид набора из различных микробенчмарков и симуляций рабочих нагрузок, которые четко определены, задокументированы и должны выполняться в соответствии с конкретными правилами, чтобы результаты были такими, как задумано. Вендоры могут участвовать в таких организациях (обычно за плату) и получать ПО для выполнения бенчмарков. Для публикации полученных результатов обычно требуется полное раскрытие настроек среды, которые можно проверить. Эти бенчмарки помогают клиентам экономить массу времени благодаря возможному наличию результатов бенчмарка разных продуктов и разных вендоров. В таком случае вам остается лишь найти бенчмарк, наиболее точно соответствующий вашей будущей или текущей промышленной нагрузке. Соответствие текущей рабочей нагрузке можно определить, выяснив характеристики этой нагрузки. Потребность в стандартных отраслевых бенчмарках была четко обозначена в статье «A Measure of Transaction Processing Power», написанной Джимом Греем (Jim Gray) и другими в 1985 году [Anon 85]. В ней обосновывается необходимость измерения соотношения цена/производительность и подробно описываются три бенчмарка, которые могут использовать производители: Sort, Scan и DebitCredit. В статье была предложена стандартная метрика — количество транзакций в секунду (transactions per second, TPS) на основе DebitCredit, — которую можно было бы использовать так же, как метрику расхода топлива в литрах на сто километров для автомобилей. Позже наработки Джима Грея способствовали созданию TPC [DeWitt 08]. Помимо TPS были предложены другие метрики для использования в качестве стандартных:

yy MIPS: миллионов инструкций в секунду (millions of instructions per second). Это

определенно показатель производительности, но его величина сильно зависит от типов инструкций, которые могут быть несопоставимыми для процессоров с разными архитектурами.

yy FLOPS: количество операций с плавающей точкой в секунду (floating-point

operations per second). Это метрика, подобная MIPS, но для рабочих нагрузок, в которых интенсивно используются вычисления с плавающей точкой.

804  Глава 12. Бенчмаркинг Отраслевые бенчмарки обычно измеряют специальную метрику, которую можно сравнивать только с самой собой.

TPC Совет по производительности обработки транзакций (Transaction Processing Performance Council, TPC) создает и администрирует отраслевые бенчмарки, уделяя особое внимание производительности баз данных, в том числе:

yy TPC-C: моделирование полной вычислительной среды, в которой группа пользователей выполняет транзакции с базой данных.

yy TPC-DS: моделирование системы поддержки принятия решений, включая запросы и обслуживание данных.

yy TPC-E: рабочая нагрузка обработки транзакций в реальном времени (online transaction processing, OLTP), моделирующая базу данных брокерской компании с клиентами, которые генерируют транзакции, связанные с торговыми сделками, запросами по счетам и исследованиями рынка.

yy TPC-H: бенчмарк для системы поддержки принятия решений, моделирующий специальные запросы и одновременное изменение данных.

yy TPC-VMS: единая система виртуальных измерений TPC (TPC virtual measurement single system) позволяет собирать другие бенчмарки для виртуализированных баз данных.

yy TPCx-HS: бенчмарк для оценки производительности обработки больших данных с использованием Hadoop.

yy TPCx-V: тестирует рабочие нагрузки баз данных на виртуальных машинах. Результаты TPC публикуются в интернете [TPC 19b] и включают соотношение цена/производительность.

SPEC Корпорация по стандартной оценке производительности (Standard Performance Evaluation Corporation, SPEC) разрабатывает и публикует стандартизированный набор отраслевых бенчмарков, в том числе:

yy SPEC Cloud IaaS 2018: для тестирования выделения вычислительных и сетевых ресурсов и ресурсов хранения путем создания нескольких рабочих нагрузок на нескольких экземплярах.

yy SPEC CPU 2017: оценивает производительность рабочих нагрузок с интенсивными вычислениями; сообщает скорость вычислений с целыми числами и с числами с плавающей точкой, а также меру энергопотребления.

yy SPECjEnterprise 2018 Web Profile: оценивает производительность всей системы для серверов приложений, баз данных и вспомогательной инфраструктуры Java Enterprise Edition (Java EE) Web Profile версии 7 или выше.

12.3. Методология  805

yy SPECsfs2014: моделирует рабочие нагрузки доступа к файлам для серверов NFS, серверов с общей межсетевой файловой системой (common internet file system, CIFS) и с другими подобными файловыми системами.

yy SPECvirt_sc2013: предназначен для виртуализированных сред. Оценивает

сквозную производительность виртуализированного оборудования, платформы, гостевой ОС и прикладного ПО.

Результаты SPEC публикуются в интернете [SPEC 20] и включают подробную информацию о настройках системы и список компонентов, но их стоимость публикуется редко.

12.3. МЕТОДОЛОГИЯ В этом разделе представлены методологии и примеры, описывающие приемы микробенчмаркинга, моделирования и воспроизведения. Краткий перечень методологий приводится в табл. 12.2. Таблица 12.2. Методологии бенчмаркинга Раздел

Методология

Тип

12.3.1

Пассивный бенчмаркинг

Анализ результатов экспериментов

12.3.2

Активный бенчмаркинг

Анализ наблюдения

12.3.3

Профилирование процессора

Анализ наблюдения

12.3.4

Метод USE

Анализ наблюдения

12.3.5

Определение характеристик рабочей нагрузки

Анализ наблюдения

12.3.6

Нестандартный бенчмаркинг

Разработка ПО

12.3.7

Пиковая нагрузка

Анализ результатов экспериментов

12.3.8

Проверка правильности

Анализ наблюдения

12.3.9

Статистический анализ

Статистический анализ

12.3.1. Пассивный бенчмаркинг Эта стратегия бенчмаркинга относится к категории «запустил и забыл», когда бенчмарк запускается и игнорируется до его завершения. Основная задача таких бенчмарков — сбор данных, характеризующих производительность. Именно так обычно выполняются бенчмарки, но при сравнении с активным бенчмаркингом этот подход описывается как антиметодология. Вот пример списка шагов, которые может включать пассивный бенчмаркинг: 1. Выбор инструмента бенчмаркинга. 2. Запуск со множеством параметров.

806  Глава 12. Бенчмаркинг 3. Создание сравнительных таблиц с результатами. 4. Передача таблиц руководству. Проблемы этого подхода обсуждались выше. В общем случае результаты могут:

yy быть недействительными из-за ошибок в ПО, реализующем бенчмарк; yy ограничиваться ПО бенчмарка (например, из-за однопоточной природы); yy ограничиваться компонентом, не связанным с целью бенчмарка (например,

перегруженной сетью); yy ограничиваться конфигурацией (отключены параметры, обеспечивающие высокую производительность); yy иметь отклонения (и не воспроизводиться); yy отражать тестирование не того, что нужно. Пассивный бенчмаркинг легко выполнить, но он подвержен ошибкам. Когда такой бенчмарк выполняется поставщиком, то может давать ошибочно заниженные результаты, которые напрасно расходуют инженерные ресурсы или приводят к потере продаж. В случае, когда это делает клиент, это может способствовать неправильному выбору продукта, который впоследствии будет сказываться на работе компании.

12.3.2. Активный бенчмаркинг При активном бенчмаркинге производительность анализируется во время выполнения бенчмарка, а не после, для чего применяются инструменты наблюдения [Gregg 14d]. В этом случае вы можете убедиться, что бенчмарк проверяет именно то, что должен проверять, и что вы понимаете его суть. Активный бенчмаркинг также помогает определить истинные ограничения тестируемой системы или самого бенчмарка. Иногда очень полезно включить конкретную информацию о встреченных ограничениях при публикации результатов бенчмарка. Кроме того, активный бенчмаркинг способствует развитию ваших навыков владения инструментами наблюдения за производительностью. Теоретически вы исследуете известную нагрузку и можете увидеть, как она проявляется в данных, возвращаемых этими инструментами. В идеале бенчмарк можно настроить и оставить работающим постоянно, чтобы получить возможность анализировать производительность в течение нескольких часов или даже дней.

Пример анализа В качестве примера рассмотрим применение инструмента микробенчмаркинга bonnie++. На странице справочного руководства man он описан как (выделено мной): NAME

bonnie++ - это программа для тестирования производительности жесткого диска1.

Это перевод текста из справочного руководства. — Примеч. пер.

1

12.3. Методология  807 А на домашней странице проекта приводится такое описание [Coker 01]: Bonnie++ — это набор простых бенчмарков для оценки производительности жесткого диска и файловой системы. Вот пример запуска bonnie++ в Ubuntu Linux: # bonnie++ [...] Version 1.97 ------Sequential Output------ --Sequential InputConcurrency 1 -Per Chr- --Block-- -Rewrite- -Per Chr- --Block-Machine Size K/sec %CP K/sec %CP K/sec %CP K/sec %CP K/sec %CP ip-10-1-239-21 4G 739 99 549247 46 308024 37 1845 99 1156838 Latency 18699us 983ms 280ms 11065us 4505us [...]

--Random--Seeks-/sec %CP 38 +++++ +++ 7762us

Судя по этому выводу, первый тест — «Sequential Output» (последовательный вывод) и «Per Chr» (посимвольно) — достиг скорости 739 Кбайт/с. Проверка правильности: если это действительно посимвольный ввод/вывод, то полученный результат означает, что система выполняет до 739 000 операций ввода/ вывода в секунду. Бенчмарк описывается как оценивающий производительность жесткого диска, но я сомневаюсь, что эта система способна обеспечить такую частоту операций ввода/вывода. Во время выполнения первого теста я использовал iostat(1), чтобы проверить частоту следования дисковых операций: $ iostat -sxz 1 [...] avg-cpu: %user 11.44 Device [...]

%nice %system %iowait 0.00 38.81 0.00 tps

kB/s

rqm/s

%steal 0.00

%idle 49.75

await aqu-sz

areq-sz

%util

Никаких операций дискового ввода/вывода не наблюдалось. Теперь используем bpftrace, чтобы подсчитать события блочного ввода/вывода (см. главу 9 «Диски», раздел 9.6.11 «bpftrace»): # bpftrace -e 'tracepoint:block:* { @[probe] = count(); }' Attaching 18 probes... ^C @[tracepoint:block:block_dirty_buffer]: 808225 @[tracepoint:block:block_touch_buffer]: 1025678

Этот однострочный сценарий тоже сообщает, что блочный ввод/вывод не запускался (событие block:block_rq_issue) и не завершался (событие block:block_rq_ complete). Но буферы изменялись. Используем cachestat(8) (глава 8 «Файловые системы», раздел 8.6.12 «cachestat»), чтобы посмотреть состояние кэша файловой системы:

808  Глава 12. Бенчмаркинг # cachestat 1 HITS MISSES 0 0 293 0 658 0 250 0 [...]

DIRTIES HITRATIO 0 0.00% 54299 100.00% 298748 100.00% 602499 100.00%

BUFFERS_MB 49 49 49 49

CACHED_MB 361 361 361 362

bonnie++ был запущен перед появлением второй строки, и последующий вывод подтверждает, что рабочая нагрузка действительно изменяла содержимое буфера, то есть запись производилась в кэш файловой системы. Проверим выполнение операций выше в стеке ввода/вывода — на уровне VFS (см. главу 8 «Файловые системы», раздел 8.6.15 «bpftrace»): # bpftrace -e 'kprobe:vfs_* /comm == "bonnie++"/ { @[probe] = count(); }' Attaching 65 probes... ^C @[kprobe:vfs_fsync_range]: 2 @[kprobe:vfs_statx_fd]: 6 @[kprobe:vfs_open]: 7 @[kprobe:vfs_read]: 13 @[kprobe:vfs_write]: 1176936

Здесь видно, что действительно была большая нагрузка на vfs_write(). Углубимся еще дальше и с помощью bpftrace проверим размеры ввода/вывода: # bpftrace -e 'k:vfs_write /comm == "bonnie++"/ { @bytes = hist(arg2); }' Attaching 1 probe... ^C @bytes: [1] [2, 4) [4, 8) [8, 16) [16, 32)

668839 0 0 0 1

|@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| | | | | | | | |

Третий аргумент vfs_write() — это счетчик байтов, и обычно он равен 1 байту (единственная операция записи блока размером от 16 до 31 байта, вероятно, является сообщением bonnie++ о начале бенчмарка). Проанализировав работу бенчмарка (активный бенчмаркинг), мы узнали, что первый тест bonnie++ выполняет запись одиночных символов в файловую систему, которые помещаются в буфер файловой системы. То есть он не тестирует дисковый ввод/вывод, как утверждается в описании bonnie ++. Согласно странице справочного руководства man, bonnie++ поддерживает параметр  -b для «запрета буферизации записи», при наличии которого после каждой операции записи вызывается fsync(2). Для проверки этого поведения можно использовать strace(1), потому что strace(1) выводит все использовавшиеся системные вызовы в удобочитаемом виде. У strace(1) есть существенный оверхед,

12.3. Методология  809 поэтому результаты бенчмарка при использовании strace(1) не следует принимать во внимание. $ strace [...] write(3, write(3, write(3, write(3, write(3, [...]

bonnie++ -b "6", "7", "8", "9", ":",

1) 1) 1) 1) 1)

= = = = =

1 1 1 1 1

Как показывает вывод, bonnie++ не вызывает fsync(2) после каждой операции запи­ си. Этот инструмент поддерживает также параметр -D для прямого ввода/вывода, но не работает в моей системе. Фактически у меня не получилось провести тест посимвольной записи на диск. Кто-то может возразить, что bonnie++ действительно работает и выполняет тесты «Sequential Output» (последовательный вывод) и «Per Chr» (посимвольно), названия которых не говорят о том, что тестируется дисковый ввод/вывод. Но как мне кажется, для бенчмарка, в описании которого утверждается, что он тестирует производительность «жесткого диска», это по меньшей мере странно. Bonnie++ — неплохой инструмент для бенчмаркинга, во многих случаях он хорошо служит людям. Я выбрал его (а также самый подозрительный из его тестов) для этого примера, потому что он хорошо известен, я изучал его раньше, и подобные выводы не редкость. Но это всего лишь один пример. Более старые версии bonnie++ имели еще одну проблему с этим тестом: он позволял библиотеке libc буферизовать записываемые данные до передачи их в файловую систему, поэтому размер записи в VFS составлял 4 Кбайт или больше, в зависимости от версии libc и ОС1. Это делало невозможным сравнение результатов bonnie++, полученных в разных ОС, в которых в libc использовались буферы разного размера. Эта проблема исправлена в последних версиях bonnie++, но возникла другая: результаты новой версии bonnie++ нельзя сравнивать с результатами старой версии. Подробнее об анализе производительности с помощью bonnie ++ идет речь в статье Роха Бурбонне (Roch Bourbonnais) «Decoding Bonnie++» [Bourbonnais 08].

12.3.3. Профилирование процессора Профилирование процессора для бенчмаркинга его производительности стоит выделить особо, потому что такой подход может привести к некоторым быстрым открытиям. Профилирование часто выполняется в рамках активного бенчмаркинга. 1

Если вам интересно, то размер буфера в libc можно настроить с помощью setbuffer(3). Этот буфер использовался из-за того, что запись в bonnie++ выполнялась вызовом функции putc(3) из библиотеки libc.

810  Глава 12. Бенчмаркинг Цель состоит в том, чтобы быстро проверить, что делает ПО, и увидеть, не обнаружится ли что-нибудь интересное. Профилирование также помогает сузить поле поиска до наиболее важных программных компонентов из числа участвующих в бенчмаркинге. Профилировать можно стеки как на уровне пользователя, так и на уровне ядра. Профилирование процессора на уровне пользователя было описано в главе 5 «Приложения». Затем обе разновидности профилирования были рассмотрены в главе 6 «Процессоры» с примерами в разделе 6.6 «Инструменты наблюдения», включая флейм-графики.

Пример В новой системе был проведен микробенчмаркинг диска и получены некоторые неутешительные результаты: пропускная способность диска оказалась хуже, чем в старой системе. Меня попросили выяснить причину, высказав предположение, что проблема либо в дисках, либо в контроллере дисков, и их, возможно, следует заменить. Я начал с метода USE (глава 2 «Методологии») и обнаружил, что диски работают с довольно низкой нагрузкой, но возникла несколько увеличенная нагрузка на процессор на уровне ядра (системное время). Мало кто ожидает, что процессоры могут быть интересной целью для анализа при бенчмаркинге дисков. Увидев повышенную нагрузку на процессор на уровне ядра, я подумал, что стоит проверить, не обнаружится ли что-нибудь интересное, хотя, честно говоря, не ожидал увидеть ничего особенного. Я провел профилирование и построил флейм-график (рис. 12.2).

Рис. 12.2. Флейм-график профилирования процессора на уровне ядра

12.3. Методология  811 Обзор фреймов стека показал, что 62,17 % образцов включали функцию zfs_zone_ io_throttle(). Мне не нужно было читать код этой функции, так как само ее имя было ключом к разгадке: средства управления ресурсами ввода/вывода в ZFS искусственно ограничивали бенчмарк! Это регулирование было включено по умолчанию в новой системе (и его не было в старой системе), и оно не учитывалось при бенчмаркинге.

12.3.4. Метод USE Метод USE был представлен в главе 2 «Методологии» и описан в главах, посвященных конкретным ресурсам. Применение метода USE во время бенчмаркинга гарантирует обнаружение предела: либо вы обнаружите компонент (аппаратный или программный), потребление которого достигнет 100 %, либо вам не удастся довести систему до предела.

12.3.5. Определение характеристик рабочей нагрузки Методология определения характеристик рабочей нагрузки была представлена в главе 2 «Методологии» и обсуждалась в следующих за ней главах. Эта методология позволяет определить, насколько хорошо бенчмарк соотносится с текущей промышленной средой, путем выяснения характеристик промышленной рабочей нагрузки для сравнения.

12.3.6. Собственный бенчмаркинг Для простых бенчмарков иногда удобнее написать свою программу. Постарайтесь сделать такую программу как можно короче, чтобы избежать сложностей, затрудняющих анализ. Лучший выбор для создания микробенчмарков — язык C, он позволяет достаточно точно выразить желаемое. При этом следует оценить, как оптимизации компилятора повлияют на ваш код: компилятор может исключить вызовы простых процедур бенчмарка, если посчитает, что их результаты не используются и они не нужны для расчетов. Выполняя бенчмарк, всегда проверяйте его работу с помощью других инструментов. Возможно, стоит дизассемблировать скомпилированный двоичный файл, чтобы увидеть, какие операции в действительности будут выполнены. Программы на языках, включающих виртуальные машины, асинхронную сборку мусора и динамическую компиляцию, могут быть сложнее в отладке и менее точными. Возможно, вам все равно придется использовать эти языки, если при проведении макробенчмаркинга потребуется смоделировать работу клиентского ПО, написанного на таких языках. Разработка собственных бенчмарков также поможет раскрыть тонкие особенности цели, которые могут оказаться полезными в дальнейшем. Например, при разработке

812  Глава 12. Бенчмаркинг бенчмарка для базы данных можно обнаружить, что API поддерживает некоторые возможности увеличения производительности, которые сейчас не используются в промышленной среде, разработанной до появления этих возможностей. Ваше ПО может просто генерировать нагрузку (генератор нагрузки) и перекладывать задачу измерения на другие инструменты. Этот подход часто используется для создания пиковых нагрузок.

12.3.7. Пиковая нагрузка Это простой метод определения максимальной пропускной способности системы. Он предполагает последовательное увеличение нагрузки до предела и измерение получившейся пропускной способности. Результаты могут быть отображены в виде графика, показывающего профиль масштабируемости. Этот профиль можно исследовать визуально или с помощью моделей масштабируемости (см. главу 2 «Методологии»). В качестве примера на рис. 12.3 показано масштабирование файловой системы и сервера с помощью потоков выполнения. Каждый поток производит произвольное чтение из кэшированного файла блоками размером 8 Кбайт, и они добавляются по одному.

Операций чтения блоками по 8 Кбайт (×1000)

Эта система достигла пика на уровне почти в полмиллиона операций чтения в секунду. Результаты были проверены с использованием статистик VFS, которые подтвердили, что размер блока ввода/вывода составлял 8 Кбайт, а на пике данные передавались со скоростью более 3,5 Гбайт/с. Тест масштабируемости файловой системы

Потоков выполнения

Рис. 12.3. Создание пиковой нагрузки на файловую систему Генератор нагрузки для этого теста был написан на Perl, и он достаточно короткий, поэтому приведу его полностью:

12.3. Методология  813 #!/usr/bin/perl -w # # randread.pl - выполняет произвольное чтение из указанного файла. use strict; my $IOSIZE = 8192; my $QUANTA = $IOSIZE;

# размер блока ввода/вывода в байтах # шаг перемещения в файле

die "USAGE: randread.pl filename\n" if @ARGV != 1 or not -e $ARGV[0]; my $file = $ARGV[0]; my $span = -s $file; my $junk;

# размер файла для выбора произвольного # блока, в байтах

open FILE, "$file" or die "ERROR: reading $file: $!\n"; while (1) { seek(FILE, int(rand($span / $QUANTA)) * $QUANTA, 0); sysread(FILE, $junk, $IOSIZE); } close FILE;

Для предотвращения буферизации используется функция sysread(), которая напрямую обращается к системному вызову read(2). Этот генератор был написан для микробенчмаркинга сервера NFS и запускался одновременно на нескольких клиентах, каждый из которых выполнял произвольное чтение файла из смонтированного тома NFS. Результаты микробенчмаркинга (операций чтения в секунду) измерялись на сервере NFS с использованием nfsstat(8) и других инструментов. Количество используемых файлов и их общий размер контролировались (из них складывается размер рабочего набора) так, чтобы некоторые тесты могли получать данные из кэша на сервере, а другие — с диска. (См. подраздел «Пример дизайна бенчмарка» в разделе 12.2.1 «Микробенчмаркинг».) Количество экземпляров, играющих роль клиентов, постепенно увеличивалось, чтобы довести нагрузку до предела. Как и в предыдущем примере, был построен график профиля масштабируемости с кривыми потребления ресурсов (метод USE), показавшими, какие ресурсы были исчерпаны. Здесь это оказался вычислительный ресурс (процессор) на сервере, что побудило провести еще одно исследование с целью дальнейшего повышения производительности. Я использовал эту программу и этот подход, чтобы найти пределы для Sun ZFS Storage Appliance [Gregg 09b]. Впоследствии эти пределы были представлены как официальные результаты, которые, насколько нам известно, установили мировые рекорды. Кроме того, у меня был аналогичный набор ПО, написанного на C, который я обычно использую, но в этом случае он не пригодился — было множество клиентских процессоров. Хотя переход на C и уменьшил нагрузку на них, это

814  Глава 12. Бенчмаркинг никак не повлияло на результат и было получено то же самое узкое место. Были опробованы и другие, более сложные бенчмарки и другие языки, но они также не смогли изменить результаты. Следуя этому подходу, измерьте задержку, пропускную способность и обязательно распределение задержек. Когда система приближается к своему пределу, ожидание операций в очереди может стать значительным, что приведет к увеличению задержек. Если приложить слишком большую нагрузку, задержка будет настолько большой, что результат можно считать недействительным. Спросите себя: устроит ли заказчика получившаяся задержка? Например: вы используете большой массив клиентов, чтобы довести нагрузку на целевую систему до уровня 990 000 операций ввода/вывода в секунду, соответствующего средней задержке ввода/вывода 5 мс. В действительности хотелось бы превысить уровень 1 миллион операций ввода/вывода в секунду, но система уже достигла насыщения. Добавляя все больше и больше клиентов, вам удается преодолеть планку в 1 миллион операций ввода/вывода в секунду, но теперь все операции много времени проводят в очереди, и средняя задержка превышает 50 мс, что неприемлемо. Какой результат вы опубликуете для рекламы? (Ответ: 990 000 операций ввода/вывода в секунду.)

12.3.8. Проверка правильности Цель этой методологии — проверка результатов бенчмарка путем выяснения, имеет ли смысл та или иная характеристика. Также проверяется, не потребовал ли результат превышения известных пределов какого-либо компонента, таких как пропускная способность сети, пропускная способность контроллера, пропускная способность внутренних соединений или максимального количества дисковых операций ввода/вывода в секунду. Если какой-либо предел оказался превышен, стоит изучить его более подробно. В большинстве случаев эта методология обнаруживает, что результат бенчмарка ложный. Рассмотрим пример: по результатам бенчмаркинга сервера NFS операциями чтения блоками по 8 Кбайт было сообщено, что он способен выдержать нагрузку до 50 000 операций ввода/вывода в секунду. Сервер подключен к сети через единственный порт Ethernet со скоростью передачи 1 Гбит/с. Необходимая пропускная способность сети составляет 50 000 операций в секунду × 8 Кбайт = 400 000 Кбайт/с плюс заголовки протокола. Это более 3,2 Гбит/с, что значительно превышает известный предел в 1 Гбит/с. Здесь явно что-то не так. Такие результаты обычно означают, что фактически бенчмарк проверил работу механизма кэширования на стороне клиента и не вся рабочая нагрузка достигла сервера NFS. Я использовал подобные расчеты для выявления множества ошибочных результатов бенчмарков, которые включали следующие значения пропускной способности через один интерфейс 1 Гбит/с [Gregg 09c]:

12.3. Методология  815

yy yy yy yy yy

120 Мбайт/с (0,96 Гбит/с); 200 Мбайт/с (1,6 Гбит/с); 350 Мбайт/с (2,8 Гбит/с); 800 Мбайт/с (6,4 Гбит/с); 1,15 Гбайт/с (9,2 Гбит/с).

Все это — пропускная способность в одном направлении. Результат 120 Мбайт/с кажется вполне правдоподобным, но на практике интерфейс 1 Гбит/с способен пропускать только где-то около 119 Мбайт/с. Результат 200 Мбайт/с возможен только при суммировании трафика в обоих направлениях, но в этом случае утверждалось, что измерялся однонаправленный трафик. Результаты 350 Мбайт/с и выше явно фальшивые. Когда вам предложат результат бенчмарка для проверки, поищите любые простые суммы, которые сможете вычислить с указанными числами, чтобы определить такие пределы. При наличии доступа к системе, вероятно, вы сможете получить дополнительные результаты, проведя новые наблюдения или эксперименты. Для этого используйте научный метод: ответьте на вопрос, верен ли результат бенчмарка. Исходя из этого можно делать гипотезы и прогнозы, а затем проверить их для ответа на вопрос.

12.3.9. Статистический анализ Для изучения результатов бенчмарка можно использовать статистический анализ. Он состоит из трех этапов. 1. Выбор инструмента бенчмарка, настроек и метрик производительности системы для сбора. 2. Выполнение бенчмарка, сбор большого набора данных с результатами и метриками. 3. Интерпретация данных со статистическим анализом, составление отчета. В отличие от активного бенчмаркинга, основное внимание в котором уделяется анализу системы во время выполнения бенчмарка, статистический анализ сосредоточен на анализе результатов. Он также отличается от пассивного бенчмаркинга, при котором анализ вообще не проводится. Этот подход используется в средах, где доступ к крупномасштабной системе может ограничиваться по времени и стоить очень дорого. Например, может быть доступна только одна система в «максимальной конфигурации», и многим командам одновременно потребовалось запустить свои тесты, в том числе:

yy Отделу продаж: для моделирования клиентской нагрузки в процессе проверки

концепции, чтобы определить, что может дать система в максимальной конфигурации.

816  Глава 12. Бенчмаркинг

yy Отделу маркетинга: для получения наилучших результатов маркетинговой кампании.

yy Отделу поддержки: для исследования патологий, возникающих только в системе с максимальной конфигурацией при серьезной нагрузке.

yy Отделу разработки: для проверки производительности новых функций и измененного кода.

yy Отделу QA: для проведения нерегрессионного тестирования и сертификации. Каждой команде выделяется ограниченное время для запуска бенчмарка в системе и намного больше времени для анализа полученных результатов. Поскольку сбор метрик обходится довольно дорого, важно приложить все силы, чтобы обеспечить их надежность и достоверность и избежать повторного тестирования. Кроме проверки того, как они генерируются технически, также можно собрать дополнительные статистические характеристики, которые помогут ускорить обнаружение проблемы, в том числе статистики изменчивости, полные распределения, пределы ошибок и др. (см. главу 2 «Методологии», раздел 2.8 «Статистика»). При бенчмаркинге изменений в коде или проведении нерегрессионного тестирования важно понимать пределы вариаций и ошибок, чтобы осмыслить пары результатов. Также желательно собрать как можно больше данных о производительности действующей системы (без ущерба для результатов из-за оверхеда на сбор), чтобы впоследствии можно было провести ретроспективный анализ этих данных. Сбор данных производят с помощью sar(1), продуктов для мониторинга и собственных инструментов, которые собирают все доступные статистики. Например, в Linux с помощью собственного сценария командной оболочки можно копировать содержимое файлов-счетчиков в /proc до и после запуска1. При необходимости можно включить копирование всего, что доступно. Такой сценарий также может выполняться во время бенчмаркинга через определенные промежутки времени при условии, что он имеет приемлемый оверхед. Для сбора данных можно использовать и другие статистические инструменты. Статистический анализ результатов и метрик может включать анализ масштабируемости и теорию очередей для моделирования системы как многоканальной системы обслуживания. Эти темы были представлены в главе 2 «Методологии». Также они рассмотрены в книгах [Jain 91], [Gunther 97], [Gunther 07].

12.3.10. Чек-лист бенчмаркинга Вдохновившись чек-листом мантр производительности (глава 2 «Методологии», раздел 2.5.20 «Мантры производительности»), я создал чек-лист вопросов, ответы 1

Не используйте утилиту tar(1) для этой цели — ее сбивают с толку файлы в /proc, которые по мнению stat(2) имеют нулевой размер, и она не читает их содержимое.

12.4. Вопросы о бенчмаркинге  817 на которые можно найти в результатах бенчмаркинга, чтобы проверить их точность [Gregg 18d]:

yy yy yy yy yy yy

А почему не в два раза? Нарушены ли какие-то ограничения? Были ли ошибки? Можно ли воспроизвести результаты? Имеют ли результаты значение? Бенчмаркинг вообще выполнялся?

Точнее:

yy А почему не в два раза? Почему скорость работы получилась именно такой, а не

в два раза больше? На самом деле этот вопрос касается выявления ограничивающих компонентов. Ответ на него может решить многие проблемы бенчмаркинга, если обнаружится, что ограничивающий компонент — не предполагаемая цель тестирования.

yy Нарушены ли какие-то ограничения? Это проверка правильности (раздел 12.3.8 «Проверка правильности»).

yy Были ли ошибки? Столкнувшись с ошибкой, код работает иначе, и высокая частота ошибок может исказить результаты бенчмарка.

yy Можно ли воспроизвести результаты? Насколько стабильны результаты бенчмарка?

yy Имеют ли результаты значение? Рабочая нагрузка, которую оценивает конкретный бенчмарк, может не соответствовать вашим потребностям. Некоторые микробенчмарки тестируют отдельные системные вызовы и вызовы библиотечных функций, но ваше приложение может даже не использовать их.

yy Бенчмаркинг вообще выполнялся? В разделе 12.1.3 «Проблемы бенчмаркинга», в подразделе «Игнорирование ошибок» описан случай, когда брандмауэр заблокировал запросы, посылаемые бенчмарком, поэтому тот фактически оценивал величину тайм-аутов.

В разделе ниже приведен гораздо более длинный список вопросов, который также охватывает сценарии, когда у вас может не быть доступа к целевой системе для самостоятельного анализа бенчмарка.

12.4. ВОПРОСЫ О БЕНЧМАРКИНГЕ Получив от поставщика результаты бенчмарка, вы можете задать ему вопросы, чтобы лучше понять эти результаты и применить к своей среде, даже в отсутствие доступа к работающему бенчмарку для его анализа. Главная задача — определить, что в действительности измеряется и насколько результат воспроизводим и близок к реальности.

818  Глава 12. Бенчмаркинг

yy В целом: • Бенчмарк как-то связан с рабочей нагрузкой в моей промышленной среде? • Как была настроена тестируемая система? • Была ли протестирована одна система или результат описывает работу группы систем? • Сколько стоит тестируемая система? • Как были настроены клиенты бенчмарка? • Как долго выполнялось тестирование? Сколько результатов было собрано? • Результат описывает среднюю или пиковую нагрузку? Какова была средняя нагрузка? • Известны ли другие параметры распределения (стандартное отклонение, процентили или полные распределения)? • Что было ограничивающим фактором бенчмарка? • Каково соотношение успешных/неудачных операций? • С какими атрибутами выполнялись операции? • Использовались ли атрибуты, имитирующие рабочую нагрузку? Какие? • Имитирует ли бенчмарк переменчивость рабочей нагрузки или оказывал усредненную нагрузку на систему? • Подтверждаются ли результаты бенчмарка другими инструментами анализа? (Предоставьте скриншоты.) • Можно ли оценить погрешность по результатам бенчмарка? • Можно ли воспроизвести результат бенчмарка? yy Для бенчмарков, связанных с процессором/памятью: • Какие процессоры использовались? • Работали ли процессоры на повышенной частоте? Использовалось ли нестандартное охлаждение (например, водяное)? • Сколько модулей памяти (например, DIMM) использовалось? Как они крепятся в разъемах? • Были ли отключены какие-либо процессоры? • Каким был уровень потребления процессоров в масштабе всей системы? (Ненагруженные системы могут работать быстрее из-за доступности более высоких уровней турбоускорения.) • Что подразумевается под процессорами — ядра или гиперпотоки? • Какой объем основной памяти был установлен? Какого типа? • Использовались ли какие-либо нестандартные настройки BIOS? yy Для бенчмарков, связанных с устройствами хранения: • Как были настроены устройства хранения (сколько устройств использовалось, их типы, протокол хранения, конфигурация RAID, размер кэша, обратная или сквозная запись и т. д.)?

12.6. Ссылки  819 • Как были настроены файловые системы (типы, сколько файловых систем использовалось, их настройки, например поддержка журналирования)? • Размер рабочего набора? • Какая доля рабочего набора умещается в кэш? Где размещался кэш? • Ко скольким файлам осуществлялся доступ?

yy Для бенчмарков, связанных с сетью: • Конфигурация сети (сколько интерфейсов использовалось, их типы и настройки)? • Топология использовавшейся сети? • Какие протоколы использовались? Параметры сокетов? • Какие параметры сетевого стека настраивались? Параметры TCP/UDP? Ответы на многие из этих вопросов при изучении отраслевых бенчмарков можно найти в сопроводительном описании.

12.5. УПРАЖНЕНИЯ 1. Ответьте на следующие вопросы, касающиеся ключевых понятий: • Что такое микробенчмаркинг? • Что такое размер рабочего набора и как он может повлиять на результаты бенчмарка хранилища? • Почему важно оценивать соотношение цена/производительность? 2. Выберите микробенчмарк и выполните следующие задания: • Масштабируйте какое-либо из измерений (количество потоков, размер блоков ввода/вывода) и проверьте, как меняется производительность. • Постройте график масштабируемости по результатам. • Используйте микробенчмарк, чтобы достичь максимальной производительности, и проанализируйте ограничивающий фактор.

12.6. ССЫЛКИ [Joy 81] Joy, W., «tcp-ip digest contribution», http://www.rfc-editor.org/rfc/museum/tcp-ip-digest/ tcp-ip-digest.v1n6.1, 1981. [Anon 85] Anon et al., «A Measure of Transaction Processing Power», Datamation, April 1, 1985. [Jain 91] Jain, R., «The Art of Computer Systems Performance Analysis: Techniques for Experimental Design, Measurement, Simulation, and Modeling», Wiley, 1991. [Gunther 97] Gunther, N., «The Practical Performance Analyst», McGraw-Hill, 1997.

820  Глава 12. Бенчмаркинг [Shanley 98] Shanley, K., «History and Overview of the TPC», http://www.tpc.org/information/ about/history.asp, 1998. [Coker 01] Coker, R., «bonnie++», https://www.coker.com.au/bonnie++, 2001. [Smaalders 06] Smaalders, B., «Performance Anti-Patterns», ACM Queue 4, no. 1, February 2006. [Gunther 07] Gunther, N., «Guerrilla Capacity Planning», Springer, 2007. [Bourbonnais 08] Bourbonnais, R., «Decoding Bonnie++», https://blogs.oracle.com/roch/entry/ decoding_bonnie, 2008. [DeWitt 08] DeWitt, D., and Levine, C., «Not Just Correct, but Correct and Fast», SIGMOD Record, 2008. [Traeger 08] Traeger, A., Zadok, E., Joukov, N., and Wright, C., «A Nine Year Study of File System and Storage Benchmarking», ACM Transactions on Storage, 2008. [Gregg 09b] Gregg, B., «Performance Testing the 7000 series, Part 3 of 3», http://www.brendangregg. com/blog/2009-05-26/performance-testing-the-7000-series3.html, 2009. [Gregg 09c] Gregg, B., and Straughan, D., «Brendan Gregg at FROSUG, Oct 2009», http://www. beginningwithi.com/2009/11/11/brendan-gregg-at-frosug-oct-2009, 2009. [Fulmer 12] Fulmer, J., «Siege Home», https://www.joedog.org/siege-home, 2012. [Gregg 14c] Gregg, B., «The Benchmark Paradox», http://www.brendangregg.com/blog/2014-05-03/ the-benchmark-paradox.html, 2014. [Gregg 14d] Gregg, B., «Active Benchmarking», http://www.brendangregg.com/activebenchmarking. html, 2014. [Gregg 18d] Gregg, B., «Evaluating the Evaluation: A Benchmarking Checklist», http://www. brendangregg.com/blog/2018-06-30/benchmarking-checklist.html, 2018. [Glozer 19] Glozer, W., «Modern HTTP Benchmarking Tool», https://github.com/wg/wrk, 2019. [TPC 19a] «Third Party Pricing Guideline», http://www.tpc.org/information/other/pricing_guidelines. asp, 2019. [TPC 19b] «TPC», http://www.tpc.org, 2019. [Dogan 20] Dogan, J., «HTTP load generator, ApacheBench (ab) replacement, formerly known as rakyll/boom», https://github.com/rakyll/hey, последнее обновление 2020. [SPEC 20] «Standard Performance Evaluation Corporation», https://www.spec.org, по состоянию на 2020.

Глава 13

PERF perf(1) — это официальный профилировщик Linux, который находится в исходном коде ядра Linux в каталоге tools/perf1. Это многофункциональный инструмент для профилирования, трассировки и создания сценариев, а также интерфейс для подсистемы наблюдения ядра perf_events, которая известна под названиями Performance Counters for Linux (PCL) и Linux Performance Events (LPE). Подсистема perf_events и интерфейс perf(1) начинались как поддержка счетчиков мониторинга производительности (performance monitoring counter, PMC), но постепенно развились и теперь поддерживают механизмы трассировки на основе событий: точки трассировки и зонды kprobes, uprobes и USDT. Эта глава наряду с главами 14 «Ftrace» и 15 «BPF» необязательна для чтения и предназначена для желающих изучить один или несколько системных трассировщиков подробнее. По сравнению с другими трассировщиками perf(1) особенно хорошо подходит для анализа потребления процессорного времени: профилирования методом выборки трассировок стека, трассировки поведения планировщика процессора и изучения счетчиков PMC для анализа производительности процессора на уровне микроархитектуры, включая анализ выполнения отдельных инструкций и тактов. Но имеющиеся в perf(1) возможности позволяют анализировать и другие цели, включая дисковый ввод/вывод и программные функции. С помощью perf(1) можно ответить на такие вопросы:

yy yy yy yy

Какие пути в коде потребляют больше всего процессорного времени? Не приостанавливаются ли процессоры при выполнении операций с памятью? По каким причинам потоки оставляют процессор? Как выглядит типичный шаблон дискового ввода/вывода?

Профилировщик perf(1) необычен тем, что это большая сложная программа, действующая в пространстве пользователя и находящаяся в дереве исходных текстов ядра Linux. Ее мейнтейнер Арнальдо Карвалью де Мело (Arnaldo Carvalho de Melo) описал мне эту ситуацию как «эксперимент». Такое положение вещей было выгодно для perf(1) и Linux, так как они разрабатывались синхронно, но некоторым не нравится включение perf(1) в исходный код ядра, и он остается единственным сложным программным продуктом пользовательского уровня, когда-либо включенным в исходный код Linux.

1

822  Глава 13. perf В разделах ниже дается введение в perf(1), перечисляются источники событий, а затем обсуждаются подкоманды, которые их используют:

yy 13.1: Обзор подкоманд yy 13.2: Примеры однострочных сценариев yy События: • 13.3: Обзор событий • 13.4: Аппаратные события • 13.5: Программные события • 13.6: Точки трассировки • 13.7: События зондов

yy Команды: • 13.8: perf stat • 13.9: perf record • 13.10: perf report • 13.11: perf script • 13.12: perf trace • 13.13: Другие команды

yy 13.14: Документация yy 13.15: Ссылки В предыдущих главах не раз было показано, как использовать perf(1) для анализа конкретных целей. В этой главе основное внимание уделяется самому профилировщику perf(1).

13.1. ОБЗОР ПОДКОМАНД Разные возможности perf(1) доступны посредством подкоманд. Ниже демонстрируется типичный пример, использующий две подкоманды: record — для инструментации событий и записи полученных данных в файл и report — для обобщения содержимого файла. Эти подкоманды объясняются в разделах 13.9 «perf record» и 13.10 «perf report». # perf record -F 99 -a -- sleep 30 [ perf record: Woken up 193 times to write data ] [ perf record: Captured and wrote 48.916 MB perf.data (11880 samples) ] # perf report --stdio [...] # Overhead Command Shared Object Symbol # ........ ............... ......................... ............................ # 21.10% swapper [kernel.vmlinux] [k] native_safe_halt

13.1. Обзор подкоманд  823 6.39% mysqld 4.66% mysqld 2.64% mysqld

[...]

[kernel.vmlinux] mysqld [kernel.vmlinux]

[k] _raw_spin_unlock_irqrest [.] _Z8ut_delaym [k] finish_task_switch

В этом примере производится выборка трассировок стека любых программ, выполняющихся на любом процессоре, в течение 30 с с частотой 99 Гц, а затем выводятся функции, наиболее часто встречающиеся в выборке. В табл. 13.1 перечислены подкоманды, которые поддерживаются последней версией perf(1) (начиная с Linux 5.6). Таблица 13.1. Некоторые подкоманды perf Раздел

Команда

Описание



annotate

Читает файл perf.data (созданный подкомандой perf record) и выводит аннотированный код



archive

Создает переносимую версию файла perf.data, содержащую отладочную информацию и символы



bench

Системные микробенчмарки



buildid-cache

Управляет кэшем идентификаторов сборки (используется зондами USDT)



cdc

Инструмент анализа кэша процессоров



diff

Читает два файла perf.data и отображает различия между профилями



evlist

Выводит список имен событий, зафиксированных в файле perf.data

14.12

ftrace

Интерфейс perf(1) к трассировщику Ftrace



inject

Фильтр для добавления дополнительной информации в поток событий



kmem

Трассировка/оценка свойств памяти (slab) ядра

11.3.3

kvm

Трассировка/оценка гостевых экземпляров kvm

13.3

list

Список типов событий



lock

Анализ событий блокировки



mem

Доступ к памяти профиля

13.7

probe

Определение новой динамической точки трассировки

13.9

record

Запускает заданную команду и записывает ее профиль в perf.data

13.10

report

Читает файл perf.data (созданный командой perf record) и выводит профиль

6.6.13

sched

Трассировка/оценка свойств планировщика (задержек)

5.5.1

script

Читает файл perf.data (созданный командой perf record) и выводит трассировку

13.8

stat

Запускает заданную команду и собирает статистики из счетчиков производительности

824  Глава 13. perf Таблица 13.1 (окончание) Раздел

Команда

Описание



timechart

Визуализирует поведение системы под нагрузкой



top

Инструмент профилирования системы, динамически обновляющий вывод на экран

13.12

trace

Оперативный трассировщик (по умолчанию трассирует системные вызовы)

На рис. 13.1 показаны часто используемые подкоманды perf с их источниками данных и типами вывода. Ядро События Счетчики Кольцевые буферы

perf list

список событий (текст)

perf stat

счетчики (текст)

perf trace

образцы (текст)

perf record

perf.data perf report perf script

обобщенная информация (текст) образцы (текст)

stackcollapse-perf.pl flamegraph.pl

флейм-график (SVG)

Рис. 13.1. Часто используемые подкоманды perf Многие из этих и других подкоманд описаны в следующих разделах. Некоторые подкоманды рассматривались в предыдущих главах (см. табл. 13.1). В будущих версиях perf(1) могут появиться дополнительные возможности: запуск perf без аргументов для получения полного списка подкоманд, доступных в системе.

13.2. ОДНОСТРОЧНЫЕ СЦЕНАРИИ На примере следующих однострочных сценариев показаны различные возможности perf(1). Это лишь выдержка из большого списка, опубликованного мной

13.2. Однострочные сценарии  825 в интернете [Gregg 20h], который оказался эффективным способом объяснить возможности perf(1). Их синтаксис рассматривается в следующих разделах и на страницах справочного руководства man для perf (1). Обратите внимание, что многие из этих сценариев используют параметр -a для указания всех процессоров. Но начиная с Linux 4.11, этот параметр подразумевается по умолчанию и его можно опустить в этой и более поздних версиях ядра.

Список событий Список всех известных событий: perf list

Список точек трассировки sched: perf list 'sched:*'

Список событий с именами, содержащими подстроку «block»: perf list block

Список доступных в данный момент динамических зондов: perf probe -l

Подсчет событий Выводит статистики из счетчиков PMC для указанной команды: perf stat команда

Выводит статистики из счетчиков PMC для процесса с указанным PID до нажатия Ctrl-C: perf stat -p PID

Выводит статистики из счетчиков PMC для всей системы в целом в течение 5 с: perf stat -a sleep 5

Выводит статистики, характеризующие работу кэша последнего уровня (LLC) процессора для указанной команды: perf stat -e LLC-loads,LLC-load-misses,LLC-stores,LLC-prefetches команда

Посчитывает непрерванные такты ядра, опираясь на спецификацию PMC (Intel): perf stat -e r003c -a sleep 5

Подсчитывает инструкции, простаивавшие в интерфейсе, опираясь на спецификацию PMC (Intel): perf stat -e cpu/event=0x0e,umask=0x01,inv,cmask=0x01/ -a sleep 5

826  Глава 13. perf Подсчитывает количество обращений к системным вызовам в секунду для всей системы: perf stat -e raw_syscalls:sys_enter -I 1000 -a

Подсчитывает количество обращений к системным вызовам по типам для процесса с указанным PID: perf stat -e 'syscalls:sys_enter_*' -p PID

Подсчитывает события блочного устройства ввода/вывода для всей системы в течение 10 с: perf stat -e 'block:*' -a sleep 10

Профилирование Выборка трассировок стека указанной команды с частотой 99 Гц: perf record -F 99 команда

Выборка трассировок стека (через указатели фреймов) в масштабе всей системы в течение 10 с: perf record -F 99 -a -g sleep 10

Выборка трассировок стека для процесса с заданным идентификатором PID с использованием отладочной информации в формате dwarf для раскручивания стека: perf record -F 99 -p PID --call-graph dwarf sleep 10

Выборка трассировок стека для контейнера по его контрольной группе из /sys/fs/ cgroup/perf_event: perf record -F 99 -e cpu-clock --cgroup=docker/1d567f439319...etc... -a sleep 10

Выборка трассировок стека в масштабе всей системы с использованием метода записи последней ветви (Last Branch Record, LBR; Intel): perf record -F 99 -a --call-graph lbr sleep 10

Выборка трассировок стека, по одной на каждые 100 промахов в кэше последнего уровня в течение 5 с: perf record -e LLC-load-misses -c 100 -ag sleep 5

Точная выборка пользовательских инструкций (например, с использованием Intel PEBS) в течение 5 с: perf record -e cycles:up -a sleep 5

Выборка трассировок стека с частотой 49 Гц и с выводом на экран имен и сегментов наиболее часто встречающихся процессов: perf top -F 49 -ns comm,dso

13.2. Однострочные сценарии  827

Статическая трассировка Регистрация событий запуска новых процессов до нажатия Ctrl-C: perf record -e sched:sched_process_exec -a

Регистрация подмножества событий переключения контекста с трассировками стека в течение 1 с: perf record -e context-switches -a -g sleep 1

Регистрация всех переключений контекста с трассировками стека в течение 1 с: perf record -e sched:sched_switch -a -g sleep 1

Регистрация всех переключений контекста с трассировками стека глубиной до пяти уровней в течение 1 с: perf record -e sched:sched_switch/max-stack=5/ -a sleep 1

Регистрация вызовов connect(2) (исходящие соединения) с трассировками стека до нажатия Ctrl-C: perf record -e syscalls:sys_enter_connect -a -g

Регистрация событий передачи запросов блочным устройствам не чаще 100 раз в секунду до нажатия Ctrl-C: perf record -F 100 -e block:block_rq_issue -a

Регистрация всех событий передачи запросов блочным устройствам и завершения их обработки (с отметками времени) до нажатия комбинации Ctrl-C: perf record -e block:block_rq_issue,block:block_rq_complete -a

Регистрация всех событий передачи запросов блочным устройствам, размер которых больше или равен 64 Кбайт, до нажатия Ctrl-C: perf record -e block:block_rq_issue --filter 'bytes >= 65536'

Регистрация всех обращений к ext4 с записью в файловую систему, отличную от ext4, до нажатия Ctrl-C: perf record -e 'ext4:*' -o /tmp/perf.data -a

Регистрация USDT-событий http__server__request (из Node.js; Linux 4.10+): perf record -e sdt_node:http__server__request -a

Регистрация событий передачи запросов блочным устройствам с выводом на экран (не в perf.data) до нажатия Ctrl-C: perf trace -e block:block_rq_issue

828  Глава 13. perf Регистрация всех событий передачи запросов блочным устройствам и завершения их обработки с выводом на экран: perf trace -e block:block_rq_issue,block:block_rq_complete

Регистрация обращений к системным вызовам в масштабе всей системы с выводом (подробной информации) на экран: perf trace

Динамическая трассировка Вставить зонд на входе в функцию ядра tcp_sendmsg() (--add можно опустить): perf probe --add tcp_sendmsg

Удалить зонд из функции tcp_sendmsg() (вместо --del можно использовать -d): perf probe --del tcp_sendmsg

Список переменных, доступных функции tcp_sendmsg(), включая внешние (требуется отладочная информация для ядра): perf probe -V tcp_sendmsg --externs

Список строк в tcp_sendmsg(), где можно установить зонд (требуется отладочная информация для ядра): perf probe -L tcp_sendmsg

Список переменных, доступных в строке 81 функции tcp_sendmsg() (требуется отладочная информация для ядра): perf probe -V tcp_sendmsg:81

Вставить зонд на входе в функцию ядра tcp_sendmsg(), фиксирующий аргументы в регистрах (зависит от модели процессора): perf probe 'tcp_sendmsg %ax %dx %cx'

Вставить зонд на входе в функцию ядра tcp_sendmsg(), фиксирующий значение регистра %cx под псевдонимом «bytes»: perf probe 'tcp_sendmsg bytes=%cx'

Регистрировать срабатывание предыдущего зонда, когда значение bytes (псевдоним) больше 100: perf record -e probe:tcp_sendmsg --filter 'bytes > 100'

Вставить зонд на выходе из tcp_sendmsg() и фиксировать возвращаемое значение: perf probe 'tcp_sendmsg%return $retval'

13.2. Однострочные сценарии  829 Вставить зонд на входе в tcp_sendmsg() и фиксировать объем отправляемых данных и состояние сокета (требуется отладочная информация): perf probe 'tcp_sendmsg size sk->__sk_common.skc_state'

Вставить зонд на входе в do_sys_open() и фиксировать имя файла в виде строки (требуется отладочная информация): perf probe 'do_sys_open filename:string'

Вставить зонд на входе в функцию пользовательского уровня fopen(3) из библио­ теки libc: perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --add fopen

Отчеты Вывод содержимого perf.data в браузере ncurses (TUI), если возможно: perf report

Вывод содержимого perf.data в виде текстового отчета с объединенными данными, суммами и процентами: perf report -n --stdio

Вывод всех событий, зафиксированных в perf.data, с заголовками (рекомендуется): perf script --header

Вывод всех событий, зафиксированных в perf.data, с рекомендуемыми мной полями (файл должен создаваться подкомандой record -a. В Linux < 4.1 следует использовать параметр -f вместо -F): perf script --header -F comm,pid,tid,cpu,time,event,ip,sym,dso

Создание флейм-графика (Linux 5.8+): perf script report flamegraph

Дизассемблирование и аннотирование инструкций с процентами (требуется отладочная информация): perf annotate --stdio

Это был перечень однострочных сценариев, которые я сам часто использую, но есть и другие возможности, которые здесь не рассматриваются. Дополнительные подкоманды perf(1) можно найти в предыдущем разделе, а также в последующих разделах этой главы и в других главах.

830  Глава 13. perf

13.3. СОБЫТИЯ PERF Список всех поддерживаемых событий можно получить командой perf list. Далее приводится неполный список событий, полученный в Linux 5.8, чтобы показать, какие типы событий существуют (выделены полужирным шрифтом): # perf list List of pre-defined events (to be used in -e): branch-instructions OR branches [Hardware event] branch-misses [Hardware event] bus-cycles [Hardware event] cache-misses [Hardware event] [...] context-switches OR cs [Software event] cpu-clock [Software event] [...] L1-dcache-load-misses [Hardware cache event] L1-dcache-loads [Hardware cache event] [...] branch-instructions OR cpu/branch-instructions/ [Kernel PMU event] branch-misses OR cpu/branch-misses/ [Kernel PMU event] [...] cache: l1d.replacement [L1D data line replacements] [...] floating point: fp_arith_inst_retired.128b_packed_double [Number of SSE/AVX computational 128-bit packed double precision [...] frontend: dsb2mite_switches.penalty_cycles [Decode Stream Buffer (DSB)-to-MITE switch true penalty cycles] [...] memory: cycle_activity.cycles_l3_miss [Cycles while L3 cache miss demand load is outstanding] [...] offcore_response.demand_code_rd.l3_miss.any_snoop [DEMAND_CODE_RD & L3_MISS & ANY_SNOOP] [...] other: hw_interrupts.received [Number of hardware interrupts received by the processor] pipeline: arith.divider_active [Cycles when divide unit is busy executing divide or square root [...] uncore: unc_arb_coh_trk_requests.all [Unit: uncore_arb Number of entries allocated. Account for Any type: e.g. Snoop, Core aperture, etc] [...] rNNN [Raw hardware event descriptor] cpu/t1=v1[,t2=v2,t3 ...]/modifier [Raw hardware event descriptor] (see 'man perf-list' on how to encode it)

13.3. События perf  831 mem:[/len][:access] [Hardware breakpoint] alarmtimer:alarmtimer_cancel [Tracepoint event] alarmtimer:alarmtimer_fired [Tracepoint event] [...] probe:do_nanosleep [Tracepoint event] [...] sdt_hotspot:class__initialization__clinit [SDT event] sdt_hotspot:class__initialization__concurrent [SDT event] [...] List of pre-defined events (to be used in --pfm-events): ix86arch: UNHALTED_CORE_CYCLES [count core clock cycles whenever the clock signal on the specific core is running (not halted)] INSTRUCTION_RETIRED [...]

Список сильно сокращен во многих местах, так как полный список в этой тестовой системе насчитывает 4402 строки. Вот основные типы событий:

yy Hardware event (аппаратное событие): большинство событий этого типа генерируются процессором (с помощью счетчиков мониторинга PMC).

yy Software event (программное событие): событие счетчика ядра. yy Hardware cache event (событие аппаратного кэша): события кэша процессора (PMC).

yy Kernel PMU event (событие модуля мониторинга производительности ядра): события, генерируемые модулем мониторинга производительности (PMC).

yy cache, floating point... (кэш, плавающая точка...): события, предусмотренные производителем процессора (PMC), с кратким описанием.

yy Raw hardware event descriptor (низкоуровневый дескриптор аппаратного события): счетчики PMC, определяемые низкоуровневыми кодами.

yy Hardware breakpoint (аппаратная точка останова): событие точки останова процессора.

yy Tracepoint event (событие точки трассировки): события механизма статической инструментации ядра.

yy SDT event (событие SDT): события механизма статической инструментации в пространстве пользователя (USDT).

yy pfm-events: события библиотеки libpfm (добавлено в Linux 5.8). В списке событий точек трассировки и SDT перечисляются точки, доступные для статической инструментации, но если вы создали несколько динамических зондов, они тоже будут перечислены в этом списке. Я включил в вывод один такой пример: probe:do_nanosleep — это установленный мной зонд kprobe, который описывается как «Tracepoint event».

832  Глава 13. perf Команда perf list может принимать аргумент со строкой поиска. Например, вот список событий, содержащих «mem_load_l3», с названиями, выделенными полужирным: # perf list mem_load_l3 List of pre-defined events (to be used in -e): cache: mem_load_l3_hit_retired.xsnp_hit [Retired load instructions which data sources were L3 and cross-core snoop hits in on-pkg core cache Supports address when precise (Precise event)] mem_load_l3_hit_retired.xsnp_hitm [Retired load instructions which data sources were HitM responses from shared L3 Supports address when precise (Precise event)] mem_load_l3_hit_retired.xsnp_miss [Retired load instructions which data sources were L3 hit and cross-core snoop missed in on-pkg core cache Supports address when precise (Precise event)] mem_load_l3_hit_retired.xsnp_none [Retired load instructions which data sources were hits in L3 without snoops required Supports address when precise (Precise event)] [...]

Это аппаратные события (на основе PMC), и вывод содержит их краткие описания. Аннотация (Precise event — точное событие) отмечает события, поддерживающие возможность точной выборки инструкций на основе событий (precise event-based sampling, PEBS).

13.4. АППАРАТНЫЕ СОБЫТИЯ Об аппаратных событиях шла речь в главе 4 «Инструменты наблюдения», в разделе 4.3.9 «Аппаратные счетчики (PMC)». Обычно они реализуются посредством счетчиков PMC, которые настраиваются с помощью кодов, уникальных для разных процессоров. Например, инструкции ветвления в процессорах Intel обычно можно инструментировать в perf(1), используя низкоуровневый дескриптор аппаратного события «r00c4», код которого формируется из значения umask 0x0 и селектора события 0xc4. Эти коды опубликованы в руководствах по процессорам [Intel 16], [AMD 18], [ARM 19]. Intel также публикует их списки в JSON [Intel 20c]. Не нужно запоминать эти коды, всегда сверяйте их с руководством по процессору. Для простоты в perf(1) поддерживаются удобочитаемые псевдонимы, которые можно использовать вместо кодов. Например, событие «branch-instructions» в вашей системе должно соответствовать счетчикам PMC для инструкций перехода1. Раньше мне доводилось сталкиваться с проблемами несоответствия между удобочитаемыми псевдонимами и правильными кодами PMC. Их сложно диагностировать только по выводу perf(1): для этого нужно иметь опыт работы с PMC и представления о том, какие значения нормальные. Имейте в виду возможность таких проблем. Учитывая скорость обновления процессора, я бы не исключал появления ошибок и в будущих определениях псевдонимов.

1

13.4. Аппаратные события  833 Некоторые из этих удобочитаемых псевдонимов можно заметить в предыдущем списке (аппаратные события и события PMU). Сейчас есть огромное разнообразие типов процессоров, и регулярно выпускаются новые версии. Может быть так, что удобочитаемые псевдонимы еще не будут определены для вашего процессора в perf(1) или определены в более новой версии ядра. Кроме того, для некоторых PMC вообще нет таких псевдонимов. Мне постоянно приходится переключаться с удобочитаемых имен на низкоуровневые дескрипторы, когда требуется изучать более глубокие счетчики PMC, для которых псевдонимов нет. В псевдонимах также могут быть ошибки, и если вам доведется столкнуться с подозрительным результатом, то используйте низкоуровневый дескриптор, чтобы подтвердить или опровергнуть подозрения.

13.4.1. Дискретная выборка При использовании команды perf record со счетчиками PMC выборка их значений происходит с определенной частотой по умолчанию, поэтому фиксируется далеко не каждое событие. Вот пример записи событий тактов: # perf record -vve cycles -a sleep 1 Using CPUID GenuineIntel-6-8E intel_pt default config: tsc,mtc,mtc_period=3,psb_period=3,pt,branch -----------------------------------------------------------perf_event_attr: size 112 { sample_period, sample_freq } 4000 sample_type IP|TID|TIME|CPU|PERIOD disabled 1 inherit 1 mmap 1 comm 1 freq 1 [...] [ perf record: Captured and wrote 3.360 MB perf.data (3538 samples) ]

Как показывает этот вывод, здесь включена дискретная выборка (freq 1) с частотой 4000 Гц. Эти настройки требуют от ядра установить такую частоту выборки, чтобы фиксировалось примерно 4000 событий в секунду для каждого процессора. Дискретизация совершено необходима, потому что некоторые события PMC могут происходить миллиарды раз в секунду (например, такты процессора), и оверхед на запись каждого события может оказаться недопустимым1. Но проблема в том, что в выводе по умолчанию (без параметра очень подробного вывода: -vv) perf(1) не сообщает о том, что включена дискретизация, и вы могли бы рассчитывать на ­запись всех событий. Частота выборки событий влияет только на подкоманду record, подкоманда stat считает все события. Конечно, в таких случаях ядро будет защищать себя и уменьшать частоту выборки, отбрасывая события. Всегда проверяйте потерю событий, чтобы узнать, произошло ли это (например, проверьте сводные счетчики из: perf report -D | tail -20).

1

834  Глава 13. perf Частоту выборки событий можно изменить с помощью параметра -F, определяющего частоту в герцах, или с помощью параметра -c, определяющего период — количество событий, после накопления которых должна производиться одна выборка (такой способ выборки известен также как выборка при переполнении). Вот пример использования параметра -F: perf record -F 99 -e cycles -a sleep 1

Эта команда будет производить выборку с частотой 99 Гц (событий в секунду). Она похожа на однострочные сценарии из раздела 13.2 «Однострочные сценарии»: они не определяют тип события (без -e cycles), что заставляет perf(1) по умолчанию использовать такты при доступности счетчиков PMC, или программное событие хода процессорных часов. Более подробно об этом идет речь в разделе 13.9.2 «Профилирование процессора». Обратите внимание, что для perf(1) есть определенные ограничения на частоту и уровень потребления процессора. Эти ограничения можно просмотреть и установить с помощью sysctl(8): # sysctl kernel.perf_event_max_sample_rate kernel.perf_event_max_sample_rate = 15500 # sysctl kernel.perf_cpu_time_max_percent kernel.perf_cpu_time_max_percent = 25

На примере видно, что максимальная частота выборки в этой системе составляет 15 500 Гц, а максимальная нагрузка на процессор, которую может создавать perf(1) (в частности, прерыванием PMU), — 25 %.

13.5. ПРОГРАММНЫЕ СОБЫТИЯ Эти события обычно соответствуют аппаратным событиями, но инструментируются программно. По аналогии с аппаратными событиями, для них может устанавливаться частота выборки по умолчанию, обычно 4000, из-за чего подкоманда record будет фиксировать только подмножество событий. Обратите внимание на следующее различие между программным событием переключения контекста и эквивалентной точкой трассировки. Сначала программные события: # perf record -vve context-switches -a -- sleep 1 [...] -----------------------------------------------------------perf_event_attr: type 1 size 112 config 0x3 { sample_period, sample_freq } 4000 sample_type IP|TID|TIME|CPU|PERIOD [...]

13.6. События точек трассировки  835 freq 1 [...] [ perf record: Captured and wrote 3.227 MB perf.data (660 samples) ]

Как показывает этот вывод, для регистрации программного события по умолчанию выбрана частота 4000 Гц. А теперь эквивалентная точка трассировки: # perf record -vve sched:sched_switch -a sleep 1 [...] -----------------------------------------------------------perf_event_attr: type 2 size 112 config 0x131 { sample_period, sample_freq } 1 sample_type IP|TID|TIME|CPU|PERIOD|RAW [...] [ perf record: Captured and wrote 3.360 MB perf.data (3538 samples) ]

В этом случае используется периодическая выборка (нет признака freq 1) с периодом 1 (эквивалент -c 1) и фиксируется каждое событие. То же самое можно сделать с программными событиями, добавив параметр -c 1, например: perf record -vve context-switches -a -c 1 -- sleep 1

Но будьте внимательны: учитывайте объем данных, сохраняемых для каждого события, и связанный с этим оверхед, особенно в случае с переключениями контекста, которые могут следовать очень часто. Проверить их частоту можно с помощью perf stat: см. раздел 13.8 «perf stat».

13.6. СОБЫТИЯ ТОЧЕК ТРАССИРОВКИ Точки трассировки были представлены в главе 4 «Инструменты наблюдения», в разделе 4.3.5 «Точки трассировки», с примерами их инструментации с помощью perf(1). Напомню, что я использовал точку трассировки block:block_rq_issue и показал следующие примеры. Трассировка всей системы в течение 10 с и вывод событий: perf record -e block:block_rq_issue -a sleep 10; perf script

Вывод аргументов этой точки трассировки и ее строки формата (метаданных): cat /sys/kernel/debug/tracing/events/block/block_rq_issue/format

Регистрация событий блочного ввода/вывода с размером больше 65 536 байт: perf record -e block:block_rq_issue --filter 'bytes > 65536' -a sleep 10

Дополнительные примеры использования perf(1) и точек трассировки приведены в разделе 13.2 «Однострочные сценарии» и в других главах этой книги.

836  Глава 13. perf Обратите внимание, что perf list показывает все инициализированные зонды, включая kprobes (механизм динамической инструментации ядра), как «Tracepoint event» (события точек трассировки); см. раздел 13.7 «События зондов».

13.7. СОБЫТИЯ ЗОНДОВ perf(1) использует термин события зондов (probe events) для обозначения зондов kprobes, uprobes и USDT. Они являются «динамическими» и поэтому должны инициализироваться перед трассировкой: они не присутствуют в выводе perf list по умолчанию (некоторые зонды USDT могут присутствовать, потому что были инициализированы автоматически). После инициализации они отображаются как «Tracepoint event» (события точек трассировки).

13.7.1. kprobes Зонды kprobes были представлены в главе 4 «Инструменты наблюдения», в разделе 4.3.6 «kprobes». Вот типичный порядок инициализации и использования зонда kprobe; в этом примере инструментируется функция ядра do_nanosleep(): perf perf perf perf

probe --add do_nanosleep record -e probe:do_nanosleep -a sleep 5 script probe --del do_nanosleep

Инициализируется зонд kprobe с помощью подкоманды probe с параметром --add (--add можно опустить), а когда он становится ненужным, то удаляется с помощью подкоманды probe с параметром --del. Вот вывод этой последовательности команд, включая сами события: # perf probe --add do_nanosleep Added new event: probe:do_nanosleep (on do_nanosleep) You can now use it in all perf tools, such as: perf record -e probe:do_nanosleep -aR sleep 1 # perf list probe:do_nanosleep List of pre-defined events (to be used in -e): probe:do_nanosleep # [ [ #

perf perf perf perf

[Tracepoint event]

record -e probe:do_nanosleep -aR sleep 1 record: Woken up 1 times to write data ] record: Captured and wrote 3.368 MB perf.data (604 samples) ] script sleep 11898 [002] 922215.458572: probe:do_nanosleep: (ffffffff83dbb6b0) SendControllerT 15713 [002] 922215.459871: probe:do_nanosleep: (ffffffff83dbb6b0)

13.7. События зондов  837 SendControllerT 5460 [001] 922215.459942: probe:do_nanosleep: (ffffffff83dbb6b0) [...] # perf probe --del probe:do_nanosleep Removed event: probe:do_nanosleep

Вывод perf script показывает вызовы do_nanosleep(), произошедшие во время трассировки, сначала из команды sleep(1) (скорее всего, из команды sleep(1), которую запустила команда perf(1)), за которыми следуют вызовы SendControllerT (имя усечено). Инструментация точек выхода из функций выполняется добавлением %return: perf probe --add do_nanosleep%return

В этом случае будет инициализирован зонд kretprobe.

Аргументы зондов kprobe Есть минимум четыре разных способа получить значения аргументов функций ядра. Во-первых, если доступна отладочная информация для ядра, то для perf(1) будет доступна информация о переменных функции, включая аргументы. Вот как можно получить список переменных в do_nanosleep() с помощью параметра --vars: # perf probe --vars do_nanosleep Available variables at do_nanosleep @ enum hrtimer_mode mode struct hrtimer_sleeper* t

Согласно этому выводу, в функции do_nanosleep() доступны переменные с именами mode и t, которые одновременно являются входными аргументами. Их можно добавить в команду инициализации зонда, чтобы включить запись их значений. Вот как можно добавить переменную mode: # perf probe 'do_nanosleep mode' [...] # perf record -e probe:do_nanosleep -a [...] # perf script svscan 1470 [012] 4731125.216396: probe:do_nanosleep: (ffffffffa8e4e440) mode=0x1

В этом примере видно, что mode=0x1. Во-вторых, если отладочная информация для ядра недоступна (как это часто бывает в промышленных средах), то аргументы можно получить через соответствующие регистры. Один из приемов — использовать идентичную систему (на таком же оборудовании и с той же версией ядра) и установить на нее отладочную информацию для ядра. Затем в этой справочной системе определить соответствие регистров и аргументов, использовав параметры -n (пробный запуск) и -v (подробный вывод):

838  Глава 13. perf # perf probe -nv 'do_nanosleep mode' [...] Writing event: p:probe/do_nanosleep _text+10806336 mode=%si:x32 [...]

Поскольку это пробный запуск, он не создает событий, но показывает, в каком регистре хранится значение переменной mode (выделено жирным): оно хранится в регистре %si как 32-битное шестнадцатеричное число (x32). (Этот синтаксис объясняется в следующем разделе, посвященном зондам uprobes.) Теперь эту информацию можно использовать в системе без отладочной информации, скопировав и вставив строку объявления переменной (mode =% si: x32): # perf probe 'do_nanosleep mode=%si:x32' [...] # perf record -e probe:do_nanosleep -a [...] # perf script svscan 1470 [000] 4732120.231245: probe:do_nanosleep: (ffffffffa8e4e440) mode=0x1

Этот прием работает, только если в системах используются процессоры с одинаковым ABI и одинаковые версии ядра, в противном случае аргументы могут оказаться в других регистрах. В-третьих, если ABI процессора известен, вы можете сами определить размещение аргументов в регистрах. Пример такого подхода показан в следующем разделе о зондах uprobes. В-четвертых, появился новый источник отладочной информации для ядра: формат представления типов BPF (BTF). Скорее всего, он будет использоваться по умолчанию и будущие версии perf(1) будут поддерживать его в качестве альтернативного источника отладочной информации. Чтобы получить возвращаемое значение do_nanosleep, инструментируйте зонд kretprobe и прочтите специальную переменную $retval: perf probe 'do_nanosleep%return $retval'

См. исходный код ядра, чтобы определить, что может содержать возвращаемое значение.

13.7.2. uprobes Зонды uprobes были представлены в главе 4 «Инструменты наблюдения», в разделе 4.3.7 «uprobes». Зонды uprobes инициализируются аналогично зондам kprobes. Например, чтобы инициализировать зонд uprobe на входе в функцию fopen(3) из библиотеки libc: # perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --add fopen Added new event: probe_libc:fopen (on fopen in /lib/x86_64-linux-gnu/libc-2.27.so)

13.7. События зондов  839 Теперь этот зонд с именем probe_libc:fopen можно использовать в команде perf record для регистрации событий: perf record -e probe_libc:fopen -aR sleep 1

Когда зонд uprobe станет ненужным, удалите его с помощью параметра --del: # perf probe --del probe_libc:fopen Removed event: probe_libc:fopen

С помощью параметра –x можно указать путь к двоичному файлу для инструментации. Для получения возвращаемого значения добавьте переменную %return: perf probe –x /lib/x86_64-linux-gnu/libc.so.6 –add fopen%return

В этом случае будет использоваться зонд uretprobe.

Аргументы зондов uprobe При наличии в системе отладочной информации для целевого двоичного файла может быть доступна информация о переменных и аргументах функций. Получить их список можно с помощью --vars: # perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --vars fopen Available variables at fopen @ char* filename char* mode

Как показывает вывод, в функции fopen(3) доступны переменные filename и mode. Их можно добавить в команду инициализации зонда: perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --add 'fopen filename mode'

Отладочная информация может быть предоставлена через пакет -dbg или -dbgsym. Если он недоступен в целевой, но есть в другой системе, можно воспользоваться этой другой системой для справки, как было показано в предыдущем разделе о зондах kprobes. Если отладочная информация вообще недоступна, варианты все равно есть. Один из них — скомпилировать ПО с включенной отладочной информацией (если это ПО с открытым исходным кодом). Другой вариант — самостоятельно определить использование регистров на основе ABI процессора. Вот пример для x86_64: # perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --add 'fopen filename=+0(%di):string mode=%si:u8' [...] # perf record -e probe_libc:fopen -a [...] # perf script run 28882 [013] 4503285.383830: probe_libc:fopen: (7fbe130e6e30) filename="/etc/nsswitch.conf" mode=147

840  Глава 13. perf run 28882 [013] 4503285.383997: probe_libc:fopen: (7fbe130e6e30) filename="/etc/passwd" mode=17 setuidgid 28882 [013] 4503285.384447: probe_libc:fopen: (7fed1ad56e30) filename="/etc/nsswitch.conf" mode=147 setuidgid 28882 [013] 4503285.384589: probe_libc:fopen: (7fed1ad56e30) filename="/etc/passwd" mode=17 run 28883 [014] 4503285.392096: probe_libc:fopen: (7f9be2f55e30) filename="/etc/nsswitch.conf" mode=147 run 28883 [014] 4503285.392251: probe_libc:fopen: (7f9be2f55e30) filename="/etc/passwd" mode=17 mkdir 28884 [015] 4503285.392913: probe_libc:fopen: (7fad6ea0be30) filename="/proc/filesystems" mode=22 chown 28885 [015] 4503285.393536: probe_libc:fopen: (7efcd22d5e30) filename="/etc/nsswitch.conf" mode=147 [...]

В этом выводе можно видеть несколько вызовов fopen(3) с именами файлов /etc/ nsswitch.conf, /etc/passwd и т. д. Разберем синтаксис, который я использовал:

yy filename= : это псевдоним («filename»), который будет использоваться для аннотирования вывода.

yy %di, %si: это регистры в архитектуре x86_64, которые, согласно AMD64 ABI, содержат первые два аргумента функции [Matz 13].

yy +0(...): разыменовать содержимое по нулевому смещению. Без этого вместо строки, находящейся по заданному адресу, были бы напечатаны сами адреса.

yy :string: вывести как строку. yy :u8: вывести как 8-битное целое число без знака. Синтаксис задокументирован на странице справочного руководства man для perfprobe(1). Прочитать возвращаемое значение при использовании зонда uretprobe можно с помощью $retval: perf probe -x /lib/x86_64-linux-gnu/libc.so.6 --add 'fopen%return $retval'

См. исходный код приложения, чтобы узнать, что содержится в возвращаемом значении. Зонды uprobes позволяют исследовать внутреннюю работу приложения, но у них нестабильный интерфейс, потому что они напрямую используют двоичный файл, который может меняться от версии к версии ПО. Намного предпочтительнее использовать зонды USDT, если они доступны.

13.7.3. USDT О зондах USDT шла речь в главе 4 «Инструменты наблюдения», в разделе 4.3.8 «USDT». Они обеспечивают стабильный интерфейс для трассировки событий.

13.7. События зондов  841 При наличии в двоичном файле зондов USDT1 их список можно получить с помощью подкоманды buildid-cache. Вот пример для двоичного файла Node.js, скомпилированного с зондами USDT (сборка настраивалась командой: ./configure --with-dtrace): # perf buildid-cache --add $(which node)

После этого список доступных зондов USDT можно получить командой perf list: # perf list | grep sdt_node sdt_node:gc__done sdt_node:gc__start sdt_node:http__client__request sdt_node:http__client__response sdt_node:http__server__request sdt_node:http__server__response sdt_node:net__server__connection sdt_node:net__stream__end

[SDT [SDT [SDT [SDT [SDT [SDT [SDT [SDT

event] event] event] event] event] event] event] event]

В данном случае это всего лишь метаданные, описывающие местоположение статически определенных событий трассировки (statically defined tracing, SDT) в коде программы. Чтобы фактически их инструментировать, нужно инициализировать зонды примерно так же, как мы инициализировали зонды uprobes в предыдущем разделе (для инициализации зондов USDT используется все тот же механизм uprobes)2. Вот пример инициализации зонда sdt_node:http__ server_request: # perf probe sdt_node:http__server__request Added new event: sdt_node:http__server__request (on %http__server__request in /home/bgregg/Build/node-v12.4.0/out/Release/node)

После инициализации зонд можно использовать в инструментах perf, например: perf record -e sdt_node:http__server__request -aR sleep 1 # perf list | grep http__server__request sdt_node:http__server__request sdt_node:http__server__request

[Tracepoint event] [SDT event]

Обратите внимание, что событие отображается как событие SDT («SDT event», из метаданных USDT) и как событие точки трассировки («Tracepoint event», которое можно инструментировать с помощью perf(1) и других инструментов). Немного странно видеть две записи для одного и того же события, но такое представление согласуется с поддержкой других событий. Также есть кортежи для точек трассировки, Также список зондов USDT, доступных в двоичном файле, можно получить командой readelf -n: они будут перечислены в разделе примечаний ELF.

1

В будущем этот шаг может отпасть: при необходимости команда perf record может автоматически интерпретировать события SDT как точки трассировки.

2

842  Глава 13. perf за исключением того, что perf(1) выводит не точки трассировки, а только соответствующие им события (если они существуют1). Вот пример регистрации событий USDT: # perf record -e sdt_node:http__server__request -a ^C[ perf record: Woken up 1 times to write data ] [ perf record: Captured and wrote 3.924 MB perf.data (2 samples) ] # perf script node 16282 [006] 510375.595203: sdt_node:http__server__request: (55c3d8b03530) arg1=140725176825920 arg2=140725176825888 arg3=140725176829208 arg4=39090 arg5=140725176827096 arg6=140725176826040 arg7=20 node 16282 [006] 510375.844040: sdt_node:http__server__request: (55c3d8b03530) arg1=140725176825920 arg2=140725176825888 arg3=140725176829208 arg4=39092 arg5=140725176827096 arg6=140725176826040 arg7=20

Как показывает вывод, в период трассировки зонд sdt_node:http__server__request сработал дважды. Здесь также видны аргументы для зонда USDT, но некоторые из них — это структуры и строки, поэтому perf(1) вывел их как адреса указателей. Чтобы получить фактические значения аргументов, должна быть какая-то возможность указывать типы аргументов при инициализации зонда. Например, чтобы преобразовать третий аргумент в строку с именем «address»: perf probe --add 'sdt_node:http__server__request address=+0(arg3):string'

Но на момент написания этих строк такая возможность не поддерживалась. Еще одна типичная проблема, которая была исправлена в Linux 4.20, заключается в том, что некоторые зонды USDT требуют наращивания семафора в адресном пространстве процесса для правильной активации. sdt_node:http__server__request — один из таких зондов, и без наращивания семафора он не будет регистрировать никакие события.

13.8. PERF STAT Подкоманда perf stat подсчитывает количество событий. Ее можно использовать для измерения частоты событий или просто для проверки наличия каких-либо событий. perf stat имеет эффективную реализацию: она подсчитывает программные события в контексте ядра, а аппаратные события с использованием регистров PMC. Это позволяет прикинуть оверхед более дорогостоящей подкоманды perf record, предварительно измерив частоту событий с помощью perf stat. Вот пример подсчета событий точки трассировки sched:sched_switch (с парамет­ ром -e, в котором указывается событие для подсчета) в системе в целом (-a) в течение одной секунды (sleep 1: фиктивная команда): В документации ядра действительно указано, что некоторые точки трассировки могут не иметь соответствующих им событий, хотя я еще не сталкивался с таким.

1

13.8. perf stat  843 # perf stat -e sched:sched_switch -a -- sleep 1 Performance counter stats for 'system wide': 5,705 sched:sched_switch 1.001892925 seconds time elapsed

В данном случае за одну секунду точка трассировки sched:sched switch сработала 5705 раз. Я часто использую разделитель «--» командной оболочки, чтобы отделить параметры команды perf(1) от фиктивной команды, которую она запускает, хотя здесь это не нужно. В разделах ниже объясняются параметры подкоманды и демонстрируются примеры ее использования.

13.8.1. Параметры Подкоманда stat поддерживает множество параметров, в том числе:

yy -a: регистрировать события на всех процессорах (начиная с Linux 4.11 этот параметр подразумевается по умолчанию);

yy yy yy yy yy

-e event: регистрировать указанное событие event; --filter filter: установить логическое выражение фильтра filter для события; -p PID: регистрировать события только для процесса с этим PID; -t TID: регистрировать события только для потока с этим TID; -G cgroup: регистрировать события только для этой контрольной группы cgroup

(используется для контейнеров);

yy -A: вести подсчет для каждого процессора отдельно; yy -I interval_ms : выводить результаты через каждый интервал interval_ms (в миллисекундах);

yy -v: выводить подробные сообщения; -vv: выводить дополнительные сообщения. Событиями могут быть точки трассировки, программные события, аппаратные события, зонды kprobes, uprobes и USDT (см. разделы 13.3–13.7). Для перечисления нескольких событий можно использовать шаблонные символы («*» соответствует любому количеству любых символов, а «?» соответствует любому одному символу). Например, следующая команда подсчитает все события для всех точек трассировки типа sched: # perf stat -e 'sched:*' -a

Также для перечисления нескольких описаний событий можно использовать несколько параметров -e. Например, события точек трассировки sched и block можно подсчитать любым из этих способов:

844  Глава 13. perf # perf stat -e 'sched:*' -e 'block:*' -a # perf stat -e 'sched:*,block:*' -a

Если события не указаны, по умолчанию perf stat будет использовать счетчики PMC из архитектурного набора: соответствующий пример см. в главе 4 «Инструменты наблюдения», в разделе 4.3.9 «Аппаратные счетчики (PMC)».

13.8.2. Интервальные статистики Интервальные статистики можно вывести с помощью параметра -I. Вот пример вывода количества событий sched:sched_switch каждые 1000 мс: # perf stat -e sched:sched_switch -a -I 1000 # time counts unit events 1.000791768 5,308 sched:sched_switch 2.001650037 4,879 sched:sched_switch 3.002348559 5,112 sched:sched_switch 4.003017555 5,335 sched:sched_switch 5.003760359 5,300 sched:sched_switch ^C 5.217339333 1,256 sched:sched_switch

Столбец counts сообщает количество событий, происходивших в истекший интервал. При просмотре этого столбца можно заметить изменения во времени. Последняя строка показывает количество событий от начала текущего интервала до момента, когда я нажал Ctrl-C, чтобы завершить perf(1). Это время составило 0,214 с, как нетрудно подсчитать по значениям в столбце time.

13.8.3. Баланс между процессорами Баланс между процессорами можно оценить параметром -A: # perf stat -e sched:sched_switch -a -A -I 1000 # time CPU counts unit events 1.000351429 CPU0 1,154 sched:sched_switch 1.000351429 CPU1 555 sched:sched_switch 1.000351429 CPU2 492 sched:sched_switch 1.000351429 CPU3 925 sched:sched_switch [...]

В этом случае команда выводит количество событий за интервал отдельно для каждого логического процессора. Есть также параметры --per-socket и --per-core для подсчета событий по физическим процессорам и ядрам.

13.8.4. Фильтрация событий Для некоторых типов событий (событий точек трассировки) поддерживается возможность фильтрации по значениям аргументов. Событие будет зарегистрировано, только если выражение, определяющее фильтр, истинно. Вот пример подсчета события sched:sched_switch, когда предыдущим был процесс с PID 25467:

13.9. perf record  845 # perf stat -e sched:sched_switch --filter 'prev_pid == 25467' -a -I 1000 # time counts unit events 1.000346518 131 sched:sched_switch 2.000937838 145 sched:sched_switch 3.001370500 11 sched:sched_switch 4.001905444 217 sched:sched_switch [...]

Описание всех этих аргументов вы найдете в главе 4 «Инструменты наблюдения», в разделе 4.3.5 «Точки трассировки», в подразделе «Аргументы и строка формата точки трассировки». Они уникальны для каждого события и перечисляются в соответствующих файлах format в /sys/kernel/debug/tracing/events.

13.8.5. Теневые статистики perf(1) имеет множество теневых статистик, которые будут выводиться при инструментации определенных комбинаций событий. Например, при инструментации счетчиков PMC для циклов и инструкций выводится статистика количество инструкций на такт (instructions per cycle, IPC): # perf stat -e cycles,instructions -a ^C Performance counter stats for 'system wide': 2,895,806,892 cycles 6,452,798,206 instructions

# 2.23 insn per cycle

1.040093176 seconds time elapsed

Как показывает этот вывод, на один такт приходится 2,23 инструкции. Теневые статистики выводятся справа от знака решетки. При вызове без событий perf stat выводит несколько таких теневых статистик (см. главу 4 «Инструменты наблюдения», раздел 4.3.9 «Аппаратные счетчики (PMC)»). Для более подробного изучения событий их можно регистрировать с помощью perf record.

13.9. PERF RECORD Подкоманда perf record записывает события в файл для последующего анализа. Требуемое событие указывается в параметре -e, поддерживается возможность регистрации нескольких событий одновременно (для этого команде можно передать несколько параметров -e или перечислить события через запятую). По умолчанию события сохраняются в файл с именем perf.data. Например: # perf record -e sched:sched_switch -a ^C[ perf record: Woken up 9 times to write data ] [ perf record: Captured and wrote 6.060 MB perf.data (23526 samples) ]

846  Глава 13. perf Обратите внимание, что вывод включает размер файла perf.data (6,060 Мбайт), количество образцов (23 526) и число случаев, когда выполнение perf (1) возобновлялось для записи данных (9 раз). Данные передаются из ядра в пользовательское пространство через кольцевые буферы, по одному для каждого процессора; чтобы уменьшить оверхед на переключение контекста, perf(1) периодически возобновляется для их чтения. Предыдущая команда продолжала запись до нажатия Ctrl-C. Чтобы установить определенную продолжительность трассировки, используйте фиктивную команду sleep(1) (или любую другую), как было показано на примере команды perf stat выше: perf record -e tracepoint -a -- sleep 1

Эта команда регистрирует события tracepoint в масштабе всей системы (-a) в течение одной секунды.

13.9.1. Параметры Подкоманда record поддерживает множество параметров, в том числе:

yy -a: регистрировать события на всех процессорах (начиная с Linux 4.11, этот параметр подразумевается по умолчанию);

yy yy yy yy yy

-e event: регистрировать указанное событие event; --filter filter: установить логическое выражение фильтра filter для события; -p PID: регистрировать события только для процесса с этим PID; -t TID: регистрировать события только для потока с этим TID; -G cgroup: регистрировать события только для этой контрольной группы cgroup

(используется для контейнеров);

yy -g: записывать трассировки стека; yy --call-graph mode: записывать трассировки стека с использованием данного метода (fp, dwarf или lbr);

yy -o file: записывать события в файл с именем file; yy -v: выводить подробные сообщения; -vv: выводить дополнительные сообщения. Те же события можно регистрировать с помощью подкоманды perf stat и выводить на экран в реальном времени (по мере их появления) с помощью perf trace.

13.9.2. Профилирование процессора Команда perf(1) часто используется для профилирования процессора. Следующая команда будет записывать трассировки стека для всех процессоров с частотой 99 Гц в течение 30 с: perf record -F 99 -a -g -- sleep 30

13.9. perf record  847 Когда событие не указано (нет параметра -e), perf(1) по умолчанию использует первое из доступных событий (многие используют точные события, представленные в главе 4 «Инструменты наблюдения», в разделе 4.3.9 «Аппаратные счетчики (PMC)»): 1. cycles:ppp: выборка частоты на основе тактов процессора с точной настройкой на нулевой занос (skid); 2. cycles:pp: выборка частоты на основе тактов процессора с точной настройкой на запрос нулевого заноса (который на практике может отличаться от нуля); 3. cycles:p: выборка частоты на основе тактов процессора с точной настройкой на запрос постоянного заноса; 4. cycles: выборка частоты на основе тактов ЦП (неточно); 5. cpu-clock: программная выборка частоты процессора. При этом выбирается наиболее точный механизм профилирования процессора. Синтаксис :ppp, :pp и  :p активирует режим выборки точных событий и может применяться к другим событиям (не только к тактам), которые его поддерживают. Кроме того, события могут поддерживать разные уровни точности. В процессорах Intel для точной выборки событий используется механизм PEBS, в процессорах AMD — механизм IBS. Они были кратко описаны в подразделе «Проблемы PMC» раздела 4.3.9.

13.9.3. Обход стека Для записи трассировок стека вместо параметра -g можно использовать параметр конфигурации max-stack. У такого подхода два преимущества: он позволяет указать максимальную глубину стека и использовать разные настройки для разных событий. Например: # perf record -e sched:sched_switch/max-stack=5/,sched:sched_wakeup/max-stack=1/ \ -a -- sleep 1

Это команда будет регистрировать события sched_switch с трассировками стека глубиной до пяти фреймов и события sched_wakeup с трассировками стека глубиной до одного фрейма. Обратите внимание: если трассировки стека выглядят поврежденными, то это может быть связано с тем, что программное обеспечение не поддерживает регистр указателя фрейма. Эта проблема обсуждалась в главе 5 «Приложения», в разделе 5.6.2 «Отсутствующие стеки». Кроме перекомпиляции ПО с поддержкой указателей фреймов (например, gcc(1) -fno-omit-frame-pointer), можно также попробовать другой метод обхода стека, выбрав его с помощью параметра --callgraph, например:

yy --call-graph dwarf : выбирает механизм обхода стека, основанный на использовании отладочной информации, который требует наличия отладочной

848  Глава 13. perf информации для выполняемого файла (для некоторых программ, например, можно установить пакет с именем, заканчивающимся на «-dbgsym» или «-dbg»);

yy --call-graph lbr: выбирает механизм обхода стека, реализованный для про-

цессоров Intel и основанный на записи последней ветви (Last Branch Record, LBR), который, впрочем, ограничивает глубину стека 16 фреймами1 и поэтому имеет ограниченную полезность;

yy --call-graph fp: выбирает механизм обхода стека, основанный на указателях фреймов (по умолчанию).

Обход стека на основе указателей фреймов описан в главе 3 «Операционные системы», в разделе 3.2.7 «Стеки». Другие механизмы (dwarf, LBR и ORC) описаны в главе 2 «Tech» («Основы технологии»), в разделе 2.4 «Stack Trace Walking» («Обход трассировки стека»), в книге «BPF Performance Tools » [Gregg 19]. После записи событий их можно исследовать с помощью perf report или perf script.

13.10. PERF REPORT Подкоманда perf report обобщает содержимое файла perf.data и поддерживает следующие параметры:

yy yy yy yy yy

--tui: использовать интерфейс TUI (по умолчанию); --stdio: вывести отчет в текстовом виде; -i file: входной файл; -n: включить столбец со счетчиком образцов; -g options: установить параметры отображения трассировки стека.

Исследовать содержимое perf.data можно с помощью сторонних инструментов. Эти инструменты могут обрабатывать вывод команды perf script, описанной в разделе 13.11 «perf script». Команды perf report обычно достаточно во многих ситуациях, и сторонние инструменты нужны нечасто. Команда perf report выводит результаты либо с использованием интерактивного текстового пользовательского интерфейса (text user interface, TUI), либо в виде простого текстового отчета (STDIO).

13.10.1. TUI Вот пример профилирования регистра указателя инструкций процессора с частотой 99 Гц в течение 10 с (без записи трассировок стека) и вывода результатов с использованием TUI:

Для процессоров с микроархитектурой Haswell глубина стека составляет 16, для процессоров с микроархитектурой Skylake — 32.

1

13.10. perf report  849 # perf record -F 99 -a -- sleep 30 [ perf record: Woken up 193 times to write data ] [ perf record: Captured and wrote 48.916 MB perf.data (11880 samples) ] # perf report Samples: 11K of event 'cpu-clock:pppH', Event count (approx.): 119999998800 Overhead Command Shared Object Symbol 21.10% swapper [kernel.vmlinux] [k] native_safe_halt 6.39% mysqld [kernel.vmlinux] [k] _raw_spin_unlock_irqrestor 4.66% mysqld mysqld [.] _Z8ut_delaym 2.64% mysqld [kernel.vmlinux] [k] finish_task_switch 2.59% oltp_read_write [kernel.vmlinux] [k] finish_task_switch 2.03% mysqld [kernel.vmlinux] [k] exit_to_usermode_loop 1.68% mysqld mysqld [.] _Z15row_search_mvccPh15pag 1.40% oltp_read_write [kernel.vmlinux] [k] _raw_spin_unlock_irqrestor [...]

perf report — это интерактивный интерфейс, позволяющий перемещаться по дан-

ным и выбирать функции и потоки для получения более подробной информации.

13.10.2. STDIO Как выглядит тот же профиль процессора при выводе в виде текстового отчета (--stdio), было показано в разделе 13.1 «Обзор подкоманд». Этот отчет не имеет интерактивных функций, зато его удобно использовать для вывода отчета в файл в виде текста. Такие отчеты могут быть полезны для обмена с другими людьми через чат-системы, электронную почту и тикет-системы поддержки. Я обычно добавляю параметр -n, чтобы включить столбец со счетчиком образцов. В качестве еще одного примера текстового отчета ниже показан профиль процессора, полученный с включенной трассировкой стека (-g): # [ [ #

perf record -F 99 -a -g -- sleep 30 perf record: Woken up 8 times to write data ] perf record: Captured and wrote 2.282 MB perf.data (11880 samples) ] perf report --stdio [...] # Children Self Command Shared Object Symbol # ........ ........ ............... .......................... ................. # 50.45% 0.00% mysqld libpthread-2.27.so [.] start_thread | ---start_thread | |--44.75%--pfs_spawn_thread | | | --44.70%--handle_connection | | | --44.55%--_Z10do_commandP3THD | | | |--42.93%--_Z16dispatch_commandP3THD | | | | | --40.92%--_Z19mysqld_stm | | | [...]

850  Глава 13. perf Трассировки стека объединяются в иерархию, начиная с корневой функции слева и двигаясь вниз и вправо. Самая правая функция — это функция, породившая событие (в данном случае функция, выполняющаяся на процессоре), а слева от нее — родительская. Путь в этом примере показывает, что процесс mysqld (демон) вызвал функцию start_thread(), которая вызывала pfs_spawn_thread(), которая вызывала handle_connection(), и т. д. Крайние правые функции в этом выводе не показаны. Такой порядок — слева направо — в терминологии perf(1) называется прямым, от вызывающей к вызываемой (caller). Его можно изменить на обратный, от вызываемой к вызывающей (callee), когда функция, породившая событие, находится слева, а ее предки — внизу и справа. Для этого используйте параметр -g callee (раньше он подразумевался по умолчанию, но с версии Linux 4.4 в perf(1) стал использоваться прямой порядок).

13.11. PERF SCRIPT Подкоманда perf script по умолчанию выводит каждый образец из perf.data и позволяет выявлять временные закономерности, которые могут затеряться в сводном отчете. Выходные данные можно использовать для создания флейм-графиков. Также можно запускать сценарии обработки трассировок, автоматизирующие формирование и вывод отчетов о событиях разными способами. Об этом пойдет речь в данном разделе. Для начала посмотрите, как выглядит вывод, сгенерированный на основе профиля процессора, собранного без трассировок стека: # perf script mysqld 8631 [000] 4142044.582702: 10101010 cpu-clock:pppH: c08fd9 _Z19close_thread_tablesP3THD+0x49 (/usr/sbin/mysqld) mysqld 8619 [001] 4142044.582711: 10101010 cpu-clock:pppH: 79f81d _ZN5Field10make_fieldEP10Send_field+0x1d (/usr/sbin/mysqld) mysqld 22432 [002] 4142044.582713: 10101010 cpu-clock:pppH: ffffffff95530302 get_futex_key_refs.isra.12+0x32 (/lib/modules/5.4.0-rc8-virtua... [...]

Ниже перечислены поля этого вывода вместе с содержимым из первой строки:

yy yy yy yy yy

имя процесса: mysqld; идентификатор потока (TID): 8631; идентификатор процессора: [000]; отметка времени: 4142044.582702 (секунды); период: 10101010 (это значение — следствие использования параметра -F 99). Включается в некоторые режимы выборки;

yy имя события: cpu-clock:pppH;

13.11. perf script  851

yy аргументы события: это и последующие поля являются аргументами собы-

тия. В аргументах события cpu-clock передаются указатель инструкции, имя и смещение функции, а также имя сегмента. Их описание приведено в главе 4, в разделе 4.3.5, в подразделе «Аргументы и строка формата точки трассировки».

Это перечень полей по умолчанию для этого события, но он может измениться в следующих версиях perf(1). Другие события не включают поле периода. Иногда важно обеспечить согласованность вывода, особенно когда требуется произвести дополнительную обработку. В таких случаях можно передать команде параметр -F с перечнем полей. Я часто использую его, чтобы включить идентификатор процесса, отсутствующий в наборе полей по умолчанию. Рекомендую добавлять параметр --header, чтобы включить метаданные из perf.data. Например, ниже показан профиль процессора с трассировкой стека: # perf script --header -F comm,pid,tid,cpu,time,event,ip,sym,dso,trace # ======== # captured on : Sun Jan 5 23:43:56 2020 # header version : 1 # data offset : 264 # data size : 2393000 # feat offset : 2393264 # hostname : bgregg-mysql # os release : 5.4.0 # perf version : 5.4.0 # arch : x86_64 # nrcpus online : 4 # nrcpus avail : 4 # cpudesc : Intel(R) Xeon(R) Platinum 8175M CPU @ 2.50GHz # cpuid : GenuineIntel,6,85,4 # total memory : 15923672 kB # cmdline : /usr/bin/perf record -F 99 -a -g -- sleep 30 # event : name = cpu-clock:pppH, , id = { 5997, 5998, 5999, 6000 }, type = 1, size = 112, { sample_period, sample_freq } = 99, sample_ty [...] # ======== # mysqld 21616/8583 [000] 4142769.671581: cpu-clock:pppH: c36299 [unknown] (/usr/sbin/mysqld) c3bad4 _ZN13QEP_tmp_table8end_sendEv (/usr/sbin/mysqld) c3c1a5 _Z13sub_select_opP4JOINP7QEP_TABb (/usr/sbin/mysqld) c346a8 _ZN4JOIN4execEv (/usr/sbin/mysqld) ca735a _Z12handle_queryP3THDP3LEXP12Query_resultyy [...]

Вывод включает заголовок с префиксом «#», описывающий систему и команду perf(1), с помощью которой был создан файл perf.data. Сохранив этот вывод в файл, вы поблагодарите себя за предусмотрительность, потому что он содержит обширную информацию, которая понадобится позже. Эти файлы можно прочитать с помощью других инструментов визуализации, включая инструмент создания флейм-графиков.

852  Глава 13. perf

13.11.1. Флейм-графики Обычно флейм-графики используются для визуализации трассировок стека, полученных при профилировании процессора. Но с их помощью можно визуализировать коллекции трассировок стека для любых событий, собранные командой perf(1), в том числе для событий переключения контекста, чтобы узнать, почему потоки оставляют процессор, либо для событий блочного ввода/вывода, чтобы увидеть, какие пути в коде ведут к дисковому вводу/выводу. Две широко используемые реализации флейм-графиков (моя собственная и версия для d3) визуализируют вывод perf script. В perf (1) поддержка флейм-графиков была добавлена в Linux 5.8. Шаги, описывающие порядок создания флейм-графиков с использованием perf(1), перечислены в главе 6 «Процессоры», в разделе 6.6.13 «perf», в подразделе «Флейм-графики процессора». Сама идея визуализации объясняется в разделе 6.7.3 «Флейм-графики». FlameScope — еще один инструмент визуализации выходных данных perf script, объединяющий флейм-графики и тепловую карту с субсекундным смещением для изучения временных вариаций. Он также описан в главе 6 «Процессоры», в разделе 6.7.4 «FlameScope».

13.11.2. Сценарии обработки трассировок Список доступных в perf(1) сценариев обработки трассировок можно получить с помощью параметра -l: # perf script -l List of available trace scripts: [...] event_analyzing_sample analyze all perf samples mem-phys-addr resolve physical address samples intel-pt-events print Intel PT Power Events and PTWRITE sched-migration sched migration overview net_dropmonitor display a table of dropped frames syscall-counts-by-pid [comm] system-wide syscall counts, by pid failed-syscalls-by-pid [comm] system-wide failed syscalls, by pid export-to-sqlite [database name] [columns] [calls] export perf data to a sqlite3 database stackcollapse produce callgraphs in short form for scripting use

Их можно передавать в виде аргументов команде perf script. Также можно разрабатывать свои сценарии обработки трассировок на Perl или Python.

13.12. PERF TRACE Подкоманда perf trace по умолчанию трассирует системные вызовы и выводит полученные данные на экран в реальном времени (без вывода в файл perf.data).

13.12. perf trace  853 Этот способ был представлен в главе 5 «Приложения», в разделе 5.5.1 «perf» как альтернатива strace(1) с меньшим оверхедом, способная выполнять трассировку в масштабе всей системы. perf trace также может перехватывать любые события, используя синтаксис, аналогичный синтаксису perf record. Вот пример трассировки проблемы с дисковым вводом/выводом: # perf trace -e block:block_rq_issue,block:block_rq_complete 0.000 auditd/391 block:block_rq_issue:259,0 WS 8192 () 16046032 + 16 [auditd] 0.566 systemd-journa/28651 block:block_rq_complete:259,0 WS () 16046032 + 16 [0] 0.748 jbd2/nvme0n1p1/174 block:block_rq_issue:259,0 WS 61440 () 2100744 + 120 [jbd2/nvme0n1p1-] 1.436 systemd-journa/28651 block:block_rq_complete:259,0 WS () 2100744 + 120 [0] 1.515 kworker/0:1H-k/365 block:block_rq_issue:259,0 FF 0 () 0 + 0 [kworker/0:1H] 1.543 kworker/0:1H-k/365 block:block_rq_issue:259,0 WFS 4096 () 2100864 + 8 [kworker/0:1H] 2.074 sshd/6463 block:block_rq_complete:259,0 WFS () 2100864 + 8 [0] 2.077 sshd/6463 block:block_rq_complete:259,0 WFS () 2100864 + 0 [0] 1087.562 kworker/0:1H-k/365 block:block_rq_issue:259,0 W 4096 () 16046040 + 8 [kworker/0:1H] [...]

Как и в случае с perf record, можно фильтровать события, используя некоторые строковые константы, сгенерированные из заголовков ядра. Вот пример трассировки обращений к системному вызову mmap(2) с флагом MAP_SHARED с использованием строки «SHARED» в качестве фильтра: # perf trace -e syscalls:*enter_mmap --filter='flags==SHARED' 0.000 env/14780 syscalls:sys_enter_mmap(len: 27002, prot: READ, flags: SHARED, fd: 3) 16.145 grep/14787 syscalls:sys_enter_mmap(len: 27002, prot: READ, flags: SHARED, fd: 3) 18.704 cut/14791 syscalls:sys_enter_mmap(len: 27002, prot: READ, flags: SHARED, fd: 3) [...]

Обратите внимание, что для большей удобочитаемости строки формата perf(1) использует строки: вместо "prot: 1" команда вывела "prot: READ". В perf(1) эта возможность называется «украшательством».

13.12.1. Версии ядра До Linux 4.19 команда perf trace по умолчанию инструментировала все системные вызовы (параметр --syscalls) в дополнение к указанным событиям (-e). Чтобы отключить трассировку других системных вызовов, нужно передать параметр --nosyscalls (теперь он подразумевается по умолчанию). Например: # perf trace -e block:block_rq_issue,block:block_rq_complete --no-syscalls

Обратите внимание, что по умолчанию трассировка выполняется по всем процессорам (-a), начиная с Linux 3.8. Фильтры (--filter) были добавлены в Linux 5.5.

854  Глава 13. perf

13.13. ДРУГИЕ КОМАНДЫ perf(1) поддерживает и другие подкоманды, часть которых используется в других главах. Напомню эти дополнительные подкоманды (полный список см. в табл. 13.1):

yy perf c2c (Linux 4.10+): анализ ложного совместного использования строк кэша и операций копирования из кэша в кэш;

yy yy yy yy yy yy

perf kmem: анализирует распределение памяти ядра; perf kvm: анализ гостевой системы при использовании гипервизора KVM; perf lock: анализ блокировок; perf mem: анализ обращений к памяти; perf sched: статистики планировщика ядра; perf script: настраиваемый набор инструментов для perf.

В числе дополнительных возможностей можно назвать запуск программ BPF по событиям и использование механизмов аппаратной трассировки, таких как трассировка процессоров Intel (processor trace, PT) или ARM CoreSight, для анализа на уровне отдельных инструкций [Hunter 20]. Ниже приводится простой пример трассировки процессора Intel. Здесь perf record записывает такты в пользовательском режиме для команды date(1): # perf record -e intel_pt/cyc/u date Sat Jul 11 05:52:40 PDT 2020 [ perf record: Woken up 1 times to write data ] [ perf record: Captured and wrote 0.049 MB perf.data ]

Полученные результаты можно вывести как трассировку инструкций (инструкции выделены полужирным): # perf script --insn-trace date 31979 [003] 653971.670163672: 7f3bfbf4d090 _start+0x0 (/lib/x86_64linux-gnu/ld-2.27.so) insn: 48 89 e7 date 31979 [003] 653971.670163672: 7f3bfbf4d093 _start+0x3 (/lib/x86_64linux-gnu/ld-2.27.so) insn: e8 08 0e 00 00 [...]

Инструкции выводятся в виде машинного кода. После установки Intel X86 Encoder Decoder (XED) инструкции можно вывести в виде мнемоник на языке Ассемблера [Intelxed 19]: # perf script --insn-trace --xed date 31979 [003] 653971.670163672: ... (/lib/x86_64-linux-gnu/ld-2.27.so) mov %rsp, %rdi date 31979 [003] 653971.670163672: ... (/lib/x86_64-linux-gnu/ld-2.27.so) callq 0x7f3bfbf4dea0 date 31979 [003] 653971.670163672: ... (/lib/x86_64-linux-gnu/ld-2.27.so) pushq %rbp [...]

13.15. Ссылки  855 date 31979 date 31979 date 31979 date 31979 date 31979 [...]

[003] [003] [003] [003] [003]

653971.670439432: 653971.670439432: 653971.670439432: 653971.670439432: 653971.670439432:

... ... ... ... ...

(/bin/date) (/bin/date) (/bin/date) (/bin/date) (/bin/date)

xor %ebp, %ebp mov %rdx, %r9 popq %rsi mov %rsp, %rdx and $0xfffffffffffffff0, %rsp

Вывод получается не только подробным, но и очень объемным. Полный вывод в этом примере составил 266 105 строк, и это только для команды date(1). Другие примеры см. в Вики по perf(1) [Hunter 20].

13.14. ДОКУМЕНТАЦИЯ PERF Для каждой подкоманды есть своя страница в справочном руководстве man, ­начинающаяся с «perf-», например, perf-record(1) — для подкоманды record. Они находятся в дереве исходных текстов Linux в каталоге tools/perf/Documen­ tation. На wiki.kernel.org есть руководство по perf(1) [Perf 15]. Винс Уивер (Vince Weaver) ведет свою неофициальную страницу perf(1) [Weaver 11]. Еще одна неофициальная страница с примерами использования perf(1) поддерживается мной [Gregg 20f]. Моя страница содержит исчерпывающий список однострочных сценариев для perf(1), а также множество других примеров. Поскольку perf(1) продолжает развиваться, обязательно проверьте наличие обновлений в более поздних версиях ядра. Хорошим источником послужит раздел perf в журнале изменений для каждой версии ядра, опубликованном на KernelNewbies [KernelNewbies 20].

13.15. ССЫЛКИ [Weaver 11] Weaver, V., «The Unofficial Linux Perf Events Web-Page», http://web.eece.maine. edu/~vweaver/projects/perf_events, 2011. [Matz 13] Matz, M., Hubička, J., Jaeger, A., and Mitchell, M., «System V Application Binary Interface, AMD64 Architecture Processor Supplement, Draft Version 0.99.6», http://x86-64. org/documentation/abi.pdf, 2013. [Perf 15] «Tutorial: Linux kernel profiling with perf», perf wiki, https://perf.wiki.kernel.org/index. php/Tutorial, последнее обновление 2015. [Intel 16] Intel 64 and IA-32 Architectures Software Developer’s Manual Volume 3B: System Programming Guide, Part 2, September 2016, https://www.intel.com/content/www/us/en/ architecture-and-technology/64-ia-32-architectures-software-developer-vol-3b-part-2-manual.html, 2016. [AMD 18] Open-Source Register Reference for AMD Family 17h Processors Models 00h-2Fh, https://developer.amd.com/resources/developer-guides-manuals, 2018.

856  Глава 13. perf [ARM 19] Arm® Architecture Reference Manual Armv8, for Armv8-A architecture pro­ fi­l e, https://developer.arm.com/architectures/cpu-architecture/a-profile/docs?_ga= 2.78191124.1893781712.1575908489-930650904.1559325573, 2019. [Intelxed 19] «Intel XED», https://intelxed.github.io, 2019. [Gregg 20h] Gregg, B., «One-Liners», http://www.brendangregg.com/perf.html#OneLiners, последнее обновление 2020. [Gregg 20f] Gregg, B., «perf Examples», http://www.brendangregg.com/perf.html, последнее обновление 2020. [Hunter 20] Hunter, A., «Perf tools support for Intel® Processor Trace», https://perf.wiki.kernel.org/ index.php/Perf_tools_support_for_Intel%C2%AE_Processor_Trace, последнее обновление 2020. [Intel 20c] «/perfmon/», https://download.01.org/perfmon, по состоянию на 2020. [KernelNewbies 20] «KernelNewbies: LinuxVersions», https://kernelnewbies.org/LinuxVersions, по состоянию на 2020.

Глава 14

FTRACE Ftrace — официальный трассировщик Linux, многофункциональный инструмент, включающий разнообразные утилиты трассировки. Ftrace был создан Стивеном Ростедтом (Steven Rostedt) и включен в Linux 2.6.27 (2008). Инструмент можно использовать без любых дополнительных интерфейсов уровня пользователя, что делает его особенно привлекательным для встраиваемых сред Linux, где пространство для хранения ограниченно. Он пригодится и для анализа серверных сред. Как и главы 13 «perf» и 15 «BPF», эта глава необязательна для чтения и предназначена для желающих изучить один или несколько системных трассировщиков более подробно. С помощью Ftrace можно ответить на вопросы:

yy yy yy yy

Как часто вызываются те или иные функции ядра? Какие пути в коде приводят к вызову этой функции? Какие другие функции вызывает эта функция ядра? Какова самая высокая задержка, вызванная вытеснением выполняемого кода?

Следующие разделы представляют Ftrace более подробно, показывают некоторые из его профилировщиков и трассировщиков и перечисляют внешние интерфейсы, которые их используют:

yy 14.1: Обзор возможностей yy 14.2: tracefs (/sys) yy Профилировщики: • 14.3: Профилировщик функций • 14.10: Триггеры hist

yy Трассировщики: • 14.4: Трассировщик function • 14.5: Точки трассировки • 14.6: Зонды kprobes • 14.7: Зонды uprobes

858  Глава 14. Ftrace • 14.8: Трассировщик function_graph • 14.9: Трассировщик hwlat

yy Внешние интерфейсы: • 14.11: trace-cmd • 14.12: perf ftrace • 14.13: perf-tools

yy 14.14: Документация Ftrace yy 14.15: Ссылки Триггеры hist — сложная тема, которая требует предварительного знакомства с профилировщиками и трассировщиками, поэтому в этой главе она рассматривается почти в самом конце. Разделы, посвященные зондам kprobes и uprobes, включают описание базовых возможностей профилирования. На рис. 14.1 представлена структура Ftrace и внешних интерфейсов, где стрелками показаны пути от событий к выходным результатам в разных форматах. Ядро

Внешние интерфейсы cat/echo

События Профилировщики Ftrace Трассировщики Ftrace Выходной буфер

trace_stat tracefs (/sys) trace/ trace_pipe

счетчики/ образцы (текст)

perf-tools

счетчики/ образцы (текст) события (пользовательский интерфейс)

funccount

счетчики (текст)

trace-cmd KernelShark

execsnoop opensnoop

образцы (текст)

kprobe

Рис. 14.1. Профилировщики, трассировщики и внешние интерфейсы в Ftrace Обо всем этом подробно рассказано в следующих разделах.

14.1. ОБЗОР ВОЗМОЖНОСТЕЙ В отличие от профилировщика perf(1), в котором для решения разных задач используются подкоманды, в Ftrace есть отдельные профилировщики и трассировщики.

14.1. Обзор возможностей  859 Профилировщики выводят сводные статистики, например счетчики и гистограммы, а трассировщики — подробные сведения об отдельных событиях. В качестве примера использования Ftrace ниже показан инструмент funcgraph(8), использующий трассировщик Ftrace для функций, которые вызывает функция ядра vfs_read(): # funcgraph vfs_read Tracing "vfs_read"... Ctrl-C to end. 1) | vfs_read() { 1) | rw_verify_area() { 1) | security_file_permission() { 1) | apparmor_file_permission() { 1) | common_file_perm() { 1) 0.763 us | aa_file_perm(); 1) 2.209 us | } 1) 3.329 us | } 1) 0.571 us | __fsnotify_parent(); 1) 0.612 us | fsnotify(); 1) 7.019 us | } 1) 8.416 us | } 1) | __vfs_read() { 1) | new_sync_read() { 1) | ext4_file_read_iter() { [...]

Как показывает вывод, vfs_read() вызывает rw_verify_area(), которая, в свою очередь, вызывает security_file_permission(), и т. д. Во втором столбце отображается продолжительность выполнения каждой функции («us» — это микросекунды), что позволяет проанализировать их производительности и выявить дочерние функции, из-за которых родительская функция работает медленно. Эта конкретная возможность в Ftrace называется трассировкой графа функций (и рассматривается в разделе 14.8 «Трассировщик function_graph»). В табл. 14.1 и 14.2 перечислены профилировщики и трассировщики Ftrace из последней версии Linux (5.2) вместе с трассировщиками событий Linux: точки трассировки, а также зонды kprobes и uprobes. Трассировщики событий похожи на Ftrace, у них аналогичные настройки и интерфейсы вывода. Названия трассировщиков, выделенные моноширинным шрифтом в табл. 14.2, — это не только имена трассировщиков Ftrace, но и ключевые слова командной строки, используемые для их настройки. Таблица 14.1. Профилировщики Ftrace Профилировщик

Описание

Раздел

Профилировщик функций

Статистики функций ядра

14.3

Профилировщик зондов kprobes

Включает подсчет срабатываний зондов kprobes

14.6.5

Профилировщик зондов uprobes

Включает подсчет срабатываний зондов гprobes

14.7.4

Триггеры hist

Создает настраиваемые гистограммы распределения событий

14.10

860  Глава 14. Ftrace Таблица 14.2. Ftrace и трассировщики событий Трассировщик

Описание

Раздел

function

Трассировщик вызовов функций ядра

14.4

Точки трассировки

Статическая инструментация ядра (трассировщик событий)

14.5

Зонды kprobes

Динамическая инструментация ядра (трассировщик событий)

14.6

Зонды uprobes

Динамическая инструментация программного кода в пространстве пользователя (трассировщик событий)

14.7

function_graph

Трассировщик вызовов функций ядра с выводом графа вызовов дочерних функций

14.8

wakeup

Измеряет максимальную задержку планировщика процессорного времени



wakeup_rt

Измеряет максимальную задержку планировщика процессорного времени для задач реального времени (real-time, rt)



irqsoff

Трассирует события отключения прерываний, указывает их местоположение в коде и продолжительность (продолжительность выполнения с отключенными прерываниями)1



preemptoff

Трассирует события отключения вытеснения, указывает их местоположение в коде и продолжительность



preemptirqsoff

Трассировщик, объединяющий возможности irqsoff и preemptoff



blk

Трассировщик блочного ввода/вывода (используется командой blktrace(8))



hwlat

Трассировщик задержек оборудования: может обнаруживать внешние возмущения, вызывающие задержки

14.9

mmiotrace

Трассирует все обращения модуля к оборудованию



nop

Специальный трассировщик, отключающий все другие трассировщики



Получить список трассировщиков Ftrace, доступных в вашей версии ядра, можно командой: # cat /sys/kernel/debug/tracing/available_tracers hwlat blk mmiotrace function_graph wakeup_dl wakeup_rt wakeup function nop

Она использует интерфейс файловой системы tracefs, смонтированной в /sys, о котором идет речь в следующем разделе. В последующих разделах говорится о профилировщиках, трассировщиках и инструментах, которые их используют. Если хотите побыстрее познакомиться с инструментами на основе Ftrace, сразу переходите к разделу 14.13 «perf-tools», где также описывается инструмент funcgraph (8), показанный выше. Этот трассировщик (а также трассировщики preemptoff и preemptirqsoff) требует, чтобы ядро было скомпилировано с параметром CONFIG_PREEMPTIRQ_EVENTS.

1

14.2. tracefs (/sys)  861 В будущих версиях ядра в Ftrace могут добавиться дополнительные профилировщики и трассировщики: загляните в документацию с описанием Ftrace в исходном коде Linux в файле Documentation/trace/ftrace.rst [Rostedt 08].

14.2. TRACEFS (/SYS) Интерфейсом для доступа к возможностям Ftrace служит файловая система tracefs. Она должна быть смонтирована в каталог /sys/kernel/tracing: mount -t tracefs tracefs /sys/kernel/tracing

Ftrace изначально является частью файловой системы debugfs, но впоследствии был выделен в отдельные файлы в tracefs. Файловая система debugfs по-прежнему сохраняет исходную структуру каталогов, монтируя tracefs как подкаталог tracing. Узнать точки монтирования debugfs и tracefs можно командой: # mount -t debugfs,tracefs debugfs on /sys/kernel/debug type debugfs (rw,relatime) tracefs on /sys/kernel/debug/tracing type tracefs (rw,relatime)

Этот вывод получен в Ubuntu 19.10 и показывает, что tracefs смонтирована в /sys/ kernel/debug/tracing. Примеры в следующих разделах используют именно эту точку монтирования, поскольку она все еще распространена, но в будущем каталог должен измениться на /sys/kernel/tracing. Обратите внимание: если у вас не получается смонтировать tracefs, то, возможно, ваше ядро скомпилировано с выключенными параметрами поддержки Ftrace (CONFIG_FTRACE и т. д.).

14.2.1. Содержимое tracefs В смонтированной файловой системе tracefs — в каталоге tracing — должны быть доступны управляющие и выходные файлы: # ls -F /sys/kernel/debug/tracing available_events max_graph_depth available_filter_functions options/ available_tracers per_cpu/ buffer_percent printk_formats buffer_size_kb README buffer_total_size_kb saved_cmdlines current_tracer saved_cmdlines_size dynamic_events saved_tgids dyn_ftrace_total_info set_event enabled_functions set_event_pid error_log set_ftrace_filter events/ set_ftrace_notrace free_buffer set_ftrace_pid function_profile_enabled set_graph_function

stack_trace_filter synthetic_events timestamp_mode trace trace_clock trace_marker trace_marker_raw trace_options trace_pipe trace_stat/ tracing_cpumask tracing_max_latency tracing_on tracing_thresh

862  Глава 14. Ftrace hwlat_detector/ instances/ kprobe_events kprobe_profile

set_graph_notrace snapshot stack_max_size stack_trace

uprobe_events uprobe_profile

Назначение многих из них интуитивно понятно. Некоторые ключевые файлы и каталоги перечислены в табл. 14.3. Таблица 14.3. Ключевые файлы tracefs Файл

Доступен для

Описание

available_tracers

Чтения

Список доступных трассировщиков (см. табл. 14.2)

current_tracer

Чтения/записи

Показывает текущий активный трассировщик

function_profile_enabled

Чтения/записи

Активирует профилировщик function

available_filter_functions

Чтения

Список функций, доступных для трассировки

set_ftrace_filter

Чтения/записи

Выбирает функции для трассировки

tracing_on

Чтения/записи

Включает/выключает вывод содержимого кольцевого буфера

trace

Чтения/записи

Вывод трассировщиков (содержимое кольцевого буфера)

trace_pipe

Чтения

Вывод трассировщиков: эта версия является потребителем информации, передаваемой трассировщиками, и служит для потоковой трассировки

trace_options

Чтения/записи

Параметры настройки трассировщика и выходного буфера

trace_stat (каталог)

Чтения/записи

Вывод профилировщика function

kprobe_events

Чтения/записи

Активирует конфигурацию kprobe

uprobe_events

Чтения/записи

Активирует конфигурацию uprobe

events (каталог)

Чтения/записи

Файлы для управления трассировщиками: точками трассировки, зондами kprobes и uprobes

instances (каталог)

Чтения/записи

Экземпляры Ftrace для разных пользователей, работающих одновременно

Этот интерфейс /sys задокументирован в файле Documentation/trace/ftrace.rst [Rostedt 08]. Его можно использовать непосредственно из командной оболочки или с помощью внешних интерфейсов и библиотек. Например, чтобы увидеть, используются ли какие-либо трассировщики Ftrace в текущий момент, можно прочитать файл current_tracer командой cat(1): # cat /sys/kernel/debug/tracing/current_tracer nop

14.3. Профилировщик функций  863 В этом примере команда вывела nop (no operation — нет операции), то есть никакие трассировщики не используются. Чтобы включить трассировщик, нужно записать его имя в этот файл. Вот пример включения трассировщика blk: # echo blk > /sys/kernel/debug/tracing/current_tracer

Другие управляющие и выходные файлы Ftrace тоже можно использовать в командах echo(1) и cat(1). Это означает, что Ftrace практически не имеет зависимостей (достаточно обычной командной оболочки1). Стивен Ростедт создал Ftrace для собственного использования, когда разрабатывал свой набор исправлений поддержки режима реального времени в ядре; изначально не предполагалась одновременная работа с Ftrace нескольких пользователей. Например, в файл current_tracer можно было записать имя только одного трассировщика. Поддержка одновременной работы нескольких пользователей была добавлена позже в виде экземпляров, которые можно создавать в каталоге «instances». У каждого экземпляра есть свой файл current_tracer и выходные файлы, поэтому он выполняет трассировку независимо. В следующих разделах (14.3–14.10) показаны дополнительные примеры использования интерфейса /sys, а в последующих разделах (14.11–14.13) будут представлены внешние интерфейсы, основанные на интерфейсе /sys: trace-cmd, подкоманда ftrace профилировщика perf(1) и perf-tools.

14.3. ПРОФИЛИРОВЩИК ФУНКЦИЙ Профилировщик функций предоставляет статистики, характеризующие функции ядра, и помогает получить представление об их использовании и определить наиболее медленные из них. Я часто использую профилировщик функций в качестве отправной точки, чтобы понять, как работает код ядра с данной рабочей нагрузкой, — отчасти потому, что он очень эффективен и имеет относительно небольшой оверхед. С его помощью я могу выбрать функции для анализа с применением более дорогостоящей трассировки событий. Для работы этого профилировщика ядро должно быть скомпилировано с параметром CONFIG_FUNCTION_PROFILER = y. Принцип действия профилировщика функций основан на использовании встроенных вызовов механизма профилирования в начале каждой функции ядра. Примерно так работают профилировщики компилятора, такие как параметр -pg в gcc(1), вставляющий вызовы mcount() для использования с gprof(1). Начиная с версии gcc(1) 4.6, функция mcount() называется __fentry__(). На первый взгляд кажется, что добавление вызовов в каждую функцию ядра должно повлечь значительный echo(1) — это встроенная команда оболочки, команду cat(1) можно сымитировать в виде функции: function shellcat { (while read line; do echo "$line"; done) < $1; }. Также можно использовать busybox, чтобы включить командную оболочку, cat(1) и другие базовые команды.

1

864  Глава 14. Ftrace оверхед, особенно если учесть, что подобная возможность используется нечасто, однако проблема оверхеда была решена: когда вызовы __fentry__() не используются, они заменяются быстрыми инструкциями nop и включаются только при необходимости [Gregg 19f]. Ниже показано, как профилировщик функций использует интерфейс tracefs в /sys. Для справки ниже показано исходное неактивное состояние профилировщика функций: # cd /sys/kernel/debug/tracing # cat set_ftrace_filter #### all functions enabled #### # cat function_profile_enabled 0

Следующие команды (выполняемые в том же каталоге) используют профилировщик функций для подсчета вызовов всех функций ядра, имена которых начинаются с «tcp», в течение примерно 10 с: # # # # #

echo 'tcp*' > set_ftrace_filter echo 1 > function_profile_enabled sleep 10 echo 0 > function_profile_enabled echo > set_ftrace_filter

Команда sleep(1) использовалась для установки (приблизительной) продолжительности профилирования. Команды, следующие за ней, остановили профилирование функций и сбросили фильтр. Совет: обязательно используйте «0 >», а не «0>» — это далеко не одно и то же, последнее — это перенаправление файлового дескриптора 0. Избегайте и комбинации «1>», поскольку она определяет перенаправление файлового дескриптора 1. Статистики профиля теперь можно прочитать из каталога trace_stat, который хранит их в файлах «function» отдельно для каждого процессора. Этот пример был получен в системе с двумя процессорами. Следующий пример использует команду head(1), только чтобы показать первые десять строк из каждого файла: # head trace_stat/function* ==> trace_stat/function0 trace_stat/function1 (reset)

Выходной буфер

trace trace_pipe

Вывод трассировщика

Рис. 14.2. Файлы в tracefs, участвующие в трассировке функций

14.4.1. Использование файла trace В примере — получение результатов трассировки функций из выходного файла trace. Здесь для справки показано исходное неактивное состояние трассировщика function: # cd /sys/kernel/debug/tracing # cat set_ftrace_filter #### all functions enabled #### # cat current_tracer nop

То есть в настоящий момент никакие другие трассировщики не используются. В этом примере трассируются все функции ядра, имена которых оканчиваются на «sleep», и события в итоге сохраняются в файл /tmp/out.trace01.txt. Фиктивная команда sleep(1) используется для сбора данных трассировки не менее чем 10 с. Последовательность команд завершается выключением трассировщика function и возвратом системы в нормальное состояние: # # # # # # # #

cd /sys/kernel/debug/tracing echo 1 > tracing_on echo '*sleep' > set_ftrace_filter echo function > current_tracer sleep 10 cat trace > /tmp/out.trace01.txt echo nop > current_tracer echo > set_ftrace_filter

14.4. Трассировщик function  867 Запись числа 1 в tracing_on требуется не всегда (например, в моей системе Ubuntu этот файл содержит 1 по умолчанию). Я включил эту команду на случай, если в вашей системе этот файл содержит 0. Фиктивная команда sleep(1) попадет в вывод трассировщика при трассировке вызовов функций «sleep»: # more /tmp/out.trace01.txt # tracer: function # # entries-in-buffer/entries-written: 57/57 #P:2 # # _-----=> irqs-off # / _----=> need-resched # | / _---=> hardirq/softirq # || / _--=> preempt-depth # ||| / delay # TASK-PID CPU# |||| TIMESTAMP FUNCTION # | | | |||| | | multipathd-348 [001] .... 332762.532877: __x64_sys_nanosleep > kprobe_events

Зонд kprobe создается и удаляется путем записи специальных команд в файл kprobe_events. После создания файл зонда появляется в events вместе с точками трассировки и может использоваться аналогичным образом. Синтаксис зондов kprobe полностью объясняется в исходном коде ядра в файле Documentation/trace/kprobetrace.rst [Hiramatsu 20]. Механизм kprobes выполняет трассировку точек входа и выхода, а также инструкции с некоторым смещением от начала функции. Вот краткое описание: p[:[GRP/]EVENT] [MOD:]SYM[+offs]|MEMADDR [FETCHARGS] : Set a probe r[MAXACTIVE][:[GRP/]EVENT] [MOD:]SYM[+0] [FETCHARGS] : Set a return probe -:[GRP/]EVENT : Clear a probe

В моем примере строка «p:brendan do_nanosleep» создает зонд (p:) с именем «brendan» для символа ядра do_nanosleep(). Строка «-:brendan» удаляет зонд «brendan». Возможность выбора произвольных имен оказалась весьма полезной в случаях одновременного использования этого механизма несколькими пользователями. Трассировщик BCC (описанный в главе 15 «BPF», в разделе 15.1 «BCC») использует имена, включающие имя трассируемой функции, строку «bcc» и BCC PID. Например: # cat /sys/kernel/debug/tracing/kprobe_events p:kprobes/p_blk_account_io_start_bcc_19454 blk_account_io_start p:kprobes/p_blk_mq_start_request_bcc_19454 blk_mq_start_request

Обратите внимание, что в новых ядрах трассировщик BCC перешел на интерфейс, основанный на perf_event_open(2), и использует kprobes вместо файла kprobe_events (и события, активированные с помощью perf_event_open(2), не отображаются в kprobe_events).

14.6.2. Аргументы В отличие от трассировщика function (раздел 14.4 «Трассировщик function»), зонды kprobes позволяют исследовать аргументы и возвращаемые значения функций. Для примера взгляните на объявление функции do_nanosleep() из файла kernel/time/ hrtimer c выделенными типами переменных-аргументов: static int __sched do_nanosleep(struct hrtimer_sleeper *t, enum hrtimer_mode mode) { [...]

14.6. Зонды kprobes  873 А вот как выглядит извлечение двух первых аргументов в системе Intel x86_64 и их вывод в шестнадцатеричном формате (по умолчанию): # echo 'p:brendan do_nanosleep hrtimer_sleeper=$arg1 hrtimer_mode=$arg2' >> kprobe_events # echo 1 > events/kprobes/brendan/enable # cat trace_pipe multipathd-348 [001] .... 349138.128610: brendan: (do_nanosleep+0x0/0x170) hrtimer_sleeper=0xffffaa6a4030be80 hrtimer_mode=0x1 multipathd-348 [001] .... 349139.128695: brendan: (do_nanosleep+0x0/0x170) hrtimer_sleeper=0xffffaa6a4030be80 hrtimer_mode=0x1 multipathd-348 [001] .... 349140.128785: brendan: (do_nanosleep+0x0/0x170) hrtimer_sleeper=0xffffaa6a4030be80 hrtimer_mode=0x1 ^C # echo 0 > events/kprobes/brendan/enable # echo '-:brendan' >> kprobe_events

В описания событий в первой строке добавлены дополнительные элементы: например, строка «hrtimer_sleeper=$arg1» извлекает первый аргумент функции и связывает его с выбранным именем «hrtimer_sleeper». Имена и значения аргументов выделены в выводе. Возможность доступа к аргументам функций как $arg1, $arg2 и т. д. была добавлена в Linux 4.20. В предыдущих версиях требовалось использовать имена регистров1. Вот эквивалентное определение зонда kprobe с использованием имен регистров: # echo 'p:brendan do_nanosleep hrtimer_sleeper=%di hrtimer_mode=%si' >> kprobe_events

Чтобы использовать имена регистров, нужно знать тип процессора и используемое соглашение о вызове функций. В архитектуре используется AMD64 ABI [Matz 13], поэтому первые два аргумента доступны в регистрах rdi и rsi2. Этот же синтаксис использует perf(1). Я уже приводил в главе 13 «perf» в разделе 13.7.2 «uprobes» довольно сложный пример разыменования указателя на строку.

14.6.3. Возвращаемые значения Возвращаемое значение доступно под специальным псевдонимом $retval при использовании kretprobes. Вот как можно получить значение, возвращаемое функцией do_nanosleep(): # echo 'r:brendan do_nanosleep ret=$retval' >> kprobe_events # echo 1 > events/kprobes/brendan/enable # cat trace_pipe multipathd-348 [001] d... 349782.180370: brendan: 1

Это также может потребоваться в аппаратных архитектурах, где псевдонимы еще не были добавлены.

2

Обобщенные соглашения о вызовах для различных процессоров можно найти на странице справочного руководства man для syscall(2). Выдержка из этой страницы приводится в разделе 14.13.4 «Однострочные сценарии для perf-tools».

874  Глава 14. Ftrace (hrtimer_nanosleep+0xce/0x1e0 kprobe_events

ret=0x0 brendan: ret=0x0 brendan: ret=0x0

Как показывает этот вывод, в период, когда выполнялась трассировка, do_nanosleep() всегда возвращала 0 (успех).

14.6.4. Фильтры и триггеры Фильтры и триггеры доступны в каталоге events/kprobes/..., по аналогии с точками трассировки (см. раздел 14.5 «Точки трассировки»). Вот файл формата для зонда kprobe в do_nanosleep() с аргументами (из раздела 14.6.2 «Аргументы»): # cat events/kprobes/brendan/format name: brendan ID: 2024 format: field:unsigned short common_type; offset:0; size:2; signed:0; field:unsigned char common_flags; offset:2; size:1; signed:0; field:unsigned char common_preempt_count; offset:3; size:1; signed:0; field:int common_pid; offset:4; size:4; signed:1; field:unsigned long __probe_ip; offset:8; size:8; signed:0; field:u64 hrtimer_sleeper; offset:16; size:8; signed:0; field:u64 hrtimer_mode; offset:24; size:8; signed:0; print fmt: "(%lx) hrtimer_sleeper=0x%Lx hrtimer_mode=0x%Lx", REC->__probe_ip, REC>hrtimer_sleeper, REC->hrtimer_mode

Обратите внимание, что выбранные мной имена переменных hrtimer_sleeper и hrtimer_mode отображаются как поля, которые можно использовать в фильтрах. Например: # echo 'hrtimer_mode != 1' > events/kprobes/brendan/filter

Эта команда будет трассировать только вызовы do_nanosleep(), где в аргументе hrtimer_mode передается значение, отличное от 1.

14.6.5. Профилировщик kprobe После установки зондов kprobes Ftrace начинает подсчитывать соответствующие им события. Эти счетчики доступны для чтения в файле kprobe_profile. Например: # cat /sys/kernel/debug/tracing/kprobe_profile p_blk_account_io_start_bcc_19454 p_blk_mq_start_request_bcc_19454 p_blk_account_io_completion_bcc_19454 p_kbd_event_1_bcc_1119

1808 677 521 632

0 0 11 0

14.7. Зонды uprobes  875 В первом столбце выводится имя зонда (его определение можно увидеть в файле kprobe_events), во втором — счетчик попаданий и в третьем — счетчик промахов (когда зонд сработал, но затем возникла ошибка и событие не было записано, то есть было упущено). Получить количество вызовов функций можно с помощью профилировщика function (раздел 14.3). Однако я обнаружил, что профилировщик kprobe намного удобнее для проверки постоянно включенных зондов kprobes, используемых ПО для мониторинга, в случаях, когда функции вызываются очень часто и их профилирование с помощью function нежелательно.

14.7. ЗОНДЫ UPROBES Uprobes — это механизм динамической инструментации кода в пространстве пользователя. Он был представлен в главе 4 «Инструменты наблюдения», в разделе 4.3.7 «uprobes». Зонды uprobes генерируют события для трассировщиков, которые совместно с Ftrace используют общие выходные и управляющие файлы tracefs. В этом разделе рассматриваются трассировка событий uprobe и профилировщик uprobe.

14.7.1. Трассировка событий Управление зондами uprobes производится с помощью файла uprobe_events, синтаксис которого описан в исходном коде Linux в файле Documentation/trace/ uprobetracer.rst [Dronamraju 20]. Вот краткое описание: p[:[GRP/]EVENT] PATH:OFFSET [FETCHARGS] : Set a uprobe r[:[GRP/]EVENT] PATH:OFFSET [FETCHARGS] : Set a return uprobe (uretprobe) -:[GRP/]EVENT : Clear uprobe or uretprobe event

Современный синтаксис требует указывать пути и смещения для зондов. Ядро не имеет информации о символах в ПО пользовательского уровня, поэтому смещение должно быть определено и передано ядру с помощью инструментов, действующих в пространстве пользователя. Вот пример использования uprobes для инструментации функции readline() из командной оболочки bash(1). Сначала определяется смещение символа: # readelf -s /bin/bash | grep -w readline 882: 00000000000b61e0 153 FUNC GLOBAL DEFAULT 14 readline # echo 'p:brendan /bin/bash:0xb61e0' >> uprobe_events # echo 1 > events/uprobes/brendan/enable # cat trace_pipe bash-3970 [000] d... 347549.225818: brendan: (0x55d0857b71e0) bash-4802 [000] d... 347552.666943: brendan: (0x560bcc1821e0) bash-4802 [000] d... 347552.799480: brendan: (0x560bcc1821e0) ^C # echo 0 > events/uprobes/brendan/enable # echo '-:brendan' >> uprobe_events

876  Глава 14. Ftrace ВНИМАНИЕ: если по ошибке указать смещение, попадающее в середину инструкции, вы повредите целевой процесс (а в случае разделяемой библиотеки — все процессы, которые ее используют). Показанный способ поиска смещения символа с помощью readelf(1) может не сработать, если целевой двоичный файл был скомпилирован как позиционно-независимый исполняемый файл (Position-Independent Executable, PIE) с рандомизацией размещения адресного пространства (Address Space Layout Randomization, ASLR). Рекомендую не использовать этот интерфейс: переключитесь на трассировщик более высокого уровня, который сам позаботится о поиске символов (например, BCC или bpftrace).

14.7.2. Аргументы и возвращаемые значения Доступ к аргументам и возвращаемым значениям выполняется так же, как при использовании механизма kprobes (см. раздел 14.6 «Зонды kprobes»). Аргументы и возвращаемые значения можно получить, указав их при создании зонда uprobe. Синтаксис описывается в файле uprobetracer.rst [Dronamraju 20].

14.7.3. Фильтры и триггеры Фильтры и триггеры доступны в каталоге events/uprobes/..., по аналогии с kprobes (см. раздел 14.6 «Зонды kprobes»).

14.7.4. Профилировщик uprobe После установки зондов uprobes Ftrace начинает подсчитывать соответствующие им события. Эти счетчики доступны для чтения в файле uprobe_profile. Например: # cat /sys/kernel/debug/tracing/uprobe_profile /bin/bash brendan

11

В первом столбце выводится путь, во втором — имя зонда (его определение можно увидеть в файле uprobe_events) и в третьем — счетчик попаданий.

14.8. ТРАССИРОВЩИК FUNCTION_GRAPH Трассировщик function_graph выводит граф вызовов функций, показывая поток выполнения. В начале этой главы был приведен пример использования funcgraph (8) из perf-tools. А в этом разделе рассмотрим интерфейс Ftrace tracefs. Вот как выглядит исходное неактивное состояние трассировщика function_graph: # cd /sys/kernel/debug/tracing # cat set_graph_function #### all functions enabled #### # cat current_tracer nop

В данный момент не используются никакие трассировщики.

14.8. Трассировщик function_graph  877

14.8.1. Трассировка графа В примере ниже использован трассировщик function_graph для функции do_ nanosleep(), чтобы показать вызовы ее дочерних функций: # echo do_nanosleep > set_graph_function # echo function_graph > current_tracer # cat trace_pipe 1) 2.731 us | get_xsave_addr(); 1) | do_nanosleep() { 1) | hrtimer_start_range_ns() { 1) | lock_hrtimer_base.isra.0() { 1) 0.297 us | _raw_spin_lock_irqsave(); 1) 0.843 us | } 1) 0.276 us | ktime_get(); 1) 0.340 us | get_nohz_timer_target(); 1) 0.474 us | enqueue_hrtimer(); 1) 0.339 us | _raw_spin_unlock_irqrestore(); 1) 4.438 us | } 1) | schedule() { 1) | rcu_note_context_switch() { [...] 5) $ 1000383 us | } /* do_nanosleep */ ^C # echo nop > current_tracer # echo > set_graph_function

Этот вывод показывает дочерние вызовы и  поток выполнения кода: do_ nanosleep() вызывает hrtimer_start_range_ns(), которая вызывает lock_hrtimer_ base.isra.0(), и т. д. В первом столбце выводится номер процессора (в этом выводе в основном отображается процессор 1) и продолжительность выполнения функций, чтобы можно было определить причины задержек. Высокие задержки помечаются дополнительным символом для вашего внимания — в этом выводе символом «$» отмечена задержка в 1 000 383 мкс (1 с). Вот значения этих символов [Rostedt 08]:

yy yy yy yy yy yy

$: более 1 с; @: более 100 мс; *: более 10 мс; #: более 1 мс; !: более 100 мкс; +: более 10 мкс.

В этом примере намеренно не использовалась фильтрация функций (set_ftrace_ filter), чтобы можно было видеть все дочерние вызовы. Но это влечет за собой некоторый оверхед, что приводит к завышению задержек. Но даже в этом случае такая трассировка помогает выявить источники высоких задержек, которые могут многократно превосходить добавленный оверхед. Если потребуется более точно определить время выполнения некоторой функции, настройте фильтр

878  Глава 14. Ftrace функций — это уменьшит количество трассируемых функций. Вот как можно выполнить трассировку одной только do_nanosleep(): # echo do_nanosleep > set_ftrace_filter # cat trace_pipe [...] 7) $ 1000130 us | } /* do_nanosleep */ ^C

Здесь я повторил трассировку той же рабочей нагрузки (sleep 1). После применения фильтра сообщаемая длительность выполнения do_nanosleep() уменьшилась с 1 000 383 мкс до 1 000 130 мкс (в этом конкретном случае), потому что исчез оверхед на трассировку всех вызываемых функций. В этих примерах также используется trace_pipe, чтобы просматривать вывод в реальном времени, но обычно вывод получается слишком объемным и намного практичнее перенаправить вывод трассировщика в файл, как было показано в разделе 14.4 «Трассировщик function».

14.8.2. Параметры Для изменения формата вывода можно использовать дополнительные параметры, доступные в каталоге options: # ls options/funcgraph-* options/funcgraph-abstime options/funcgraph-cpu options/funcgraph-duration

options/funcgraph-irqs options/funcgraph-overhead options/funcgraph-overrun

options/funcgraph-proc options/funcgraph-tail

Они управляют выводом и могут добавлять или исключать некоторые детали: идентификатор процессора (funcgraph-cpu), имя процесса (funcgraph-proc), продолжительность выполнения функции (funcgraph-duration) и маркеры задержки (funcgraph-overhead).

14.9. ТРАССИРОВЩИК HWLAT Детектор задержек, обусловленных работой оборудования (hwlat), — пример специализированного трассировщика. Он может обнаруживать ситуации, когда внешние аппаратные события ухудшают производительность процессора: события, которые иначе невидимы для ядра и других инструментов. Например, прерывания управления системой (system management interrupt, SMI) и возмущения со стороны гипервизора (в том числе вызванные шумными соседями). Для этого он выполняет цикл с отключенными прерываниями и измеряет время, затраченное на выполнение каждой итерации. Этот цикл поочередно выполняется на каждом процессоре, а затем выводится продолжительность выполнения самой медленной итерации для каждого процессора, если она превышает пороговое значение (по умолчанию 10 мкс, но порог можно настроить с помощью файла tracing_thresh).

14.10. Триггеры hist  879 Вот пример: # cd /sys/kernel/debug/tracing # echo hwlat > current_tracer # cat trace_pipe -5820 [001] d... ts:1578801212.559595228 -5820 [000] d... ts:1578801213.571460991 -5820 [001] dn.. ts:1578801214.595380588 -5820 [000] d... ts:1578801215.619463259 -5820 [001] d... ts:1578801216.643451721 -5820 [000] d... ts:1578801217.667385514 ^C # echo nop > current_tracer

354016.973699: #1

inner/outer(us): 2152/1933

354017.985568: #2

inner/outer(us): 19/26

354019.009489: #3

inner/outer(us): 1699/5894

354020.033575: #4

inner/outer(us): 43/49

354021.057566: #5

inner/outer(us): 18/45

354022.081503: #6

inner/outer(us): 18/38

Многие из этих полей были описаны в предыдущих разделах (см. раздел 14.4 «Трассировщик function»). Любопытно, что за отметкой времени следует порядковый номер (#1, ...), затем числа «inner/outer(us)» и окончательная отметка времени. Числа «inner/outer» показывают время в микросекундах выполнения одной итерации (внутреннее — inner) и логики перехода между итерациями (внешнее — outer). В первой строке видно, что самая медленная итерация выполнялась 2152 мкс (inner) и переход между итерациями длился 1933 мкс (outer). Эти задержки намного превышают порог в 10 мкс и явно обусловлены внешними возмущениями. hwlat имеет параметры настройки: цикл выполняется в течение периода времени, который называется шириной (width), и запускает эксперименты, продолжительность которых называется окном (window). В течение каждого периода width регистрируется самая медленная итерация с продолжительностью, превышающей пороговое значение (10 мкс). Эти параметры можно изменить с помощью файлов width и window в /sys/kernel/debug/tracing/hwlat_detector; время в этих файлах измеряется в микросекундах. ВНИМАНИЕ: я бы классифицировал hwlat как инструмент микробенчмаркинга, а не наблюдения, потому что он выполняет эксперименты, которые сами по себе влияют на производительность системы: поочередно нагружает каждый процессор на длительное время, отключая при этом прерывания.

14.10. ТРИГГЕРЫ HIST Триггеры hist были добавлены в Linux 4.7 Томом Занусси (Tom Zanussi). Они позволяют создавать собственные гистограммы распределения событий. Это еще одна форма статистического анализа, позволяющая разбить счетчики на составляющие.

880  Глава 14. Ftrace Вот обобщенный порядок получения гистограммы: 1. echo 'hist:выражение' > events/.../trigger: создать триггер hist. 2. sleep продолжительность: дать время для заполнения гистограммы. 3. cat events/.../hist: вывести гистограмму. 4. echo '!hist:выражение' > events/.../trigger: удалить триггер. Выражения для триггеров hist определяются в следующем формате: hist:keys=[:values=] [:sort=][:size=#entries][:pause][:continue] [:clear][:name=histname1][:.] [if ]

Исчерпывающее описание синтаксиса можно найти в исходном коде Linux в файле Documentation/trace/histogram.rst, а далее приводится несколько примеров [Zanussi 20].

14.10.1. Гистограмма с единственным ключом В примере ниже показано применение триггеров hist для подсчета системных вызовов с использованием точки трассировки raw_syscalls:sys_enter. Выводится гистограмма с разбивкой по идентификатору процесса: # cd /sys/kernel/debug/tracing # echo 'hist:key=common_pid' > events/raw_syscalls/sys_enter/trigger # sleep 10 # cat events/raw_syscalls/sys_enter/hist # event histogram # # trigger info: hist:keys=common_pid.execname:vals=hitcount:sort=hitcount:size=2048 [active] # { { { { { { { { { { { {

common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid: common_pid:

347 345 504 494 502 344 348 32399 32400 32379 32296 32396

} } } } } } } } } } } }

hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount: hitcount:

1 3 8 20 30 32 36 136 138 177 187 882604

Totals: Hits: 883372 Entries: 12 Dropped: 0 # echo '!hist:key=common_pid' > events/raw_syscalls/sys_enter/trigger

14.10. Триггеры hist  881 Мы видим, что в период трассировки процесс с идентификатором PID 32396 обратился к системным вызовам 882 604 раза; также перечислены счетчики обращений для других процессов. В нескольких последних строках выводятся: количество операций записи в хеш (Hits), количество элементов в хеше (Entries) и количество прерванных операций записи из-за превышения размера хеша (Dropped). Если появляются прерванные операции записи, то можно увеличить размер хеша при его объявлении. По умолчанию хеш может содержать до 2048 элементов.

14.10.2. Поля Поля хеша определяются файлом формата события. В этом примере использовалось поле common_pid: # cat events/raw_syscalls/sys_enter/format [...] field:int common_pid; offset:4;

size:4;

signed:1;

field:long id; offset:8; size:8; signed:1; field:unsigned long args[6]; offset:16; size:48;

signed:0;

Также можно использовать другие поля. Поле id для этого события содержит идентификатор системного вызова. Вот пример использования этого поля в роли ключа хеша: # echo 'hist:key=id' > events/raw_syscalls/sys_enter/trigger # cat events/raw_syscalls/sys_enter/hist [...] { id: 14 } hitcount: 48 { id: 1 } hitcount: 80362 { id: 0 } hitcount: 80396 [...]

Эта гистограмма показывает, что чаще других вызывались системные вызовы с идентификаторами 0 и 1. В моей системе идентификаторы системных вызовов определены в этом заголовочном файле: # more /usr/include/x86_64-linux-gnu/asm/unistd_64.h [...] #define __NR_read 0 #define __NR_write 1 [...]

То есть идентификаторы 0 и 1 соответствуют системным вызовам read(2) и write(2).

14.10.3. Модификаторы Разбивка по идентификаторам процессов и системных вызовов используется настолько часто, что в триггеры hist была добавлена поддержка модификаторов, аннотирующих вывод: .execname — для идентификаторов PID и  .syscall — для

882  Глава 14. Ftrace идентификаторов системных вызовов. Вот пример добавления модификатора .execname к предыдущему примеру: # echo 'hist:key=common_pid.execname' > events/raw_syscalls/sys_enter/trigger [...] { common_pid: bash [ 32379] } hitcount: 166 { common_pid: sshd [ 32296] } hitcount: 259 { common_pid: dd [ 32396] } hitcount: 869024 [...]

Этот модификатор добавляет в вывод имена процессов, за которыми следуют их идентификаторы PID в квадратных скобках.

14.10.4. Фильтры PID Основываясь на предыдущих результатах с разбивкой по идентификаторам процессов и системных вызовов, можно предположить, что они связаны, а к системным вызовам read(2) и write(2) действительно обращалась команда dd(1). Чтобы убедиться в этом, можно запросить создание гистограммы с разбивкой по идентификаторам системных вызовов и добавить фильтр, чтобы выполнить подсчет обращений только для одного конкретного процесса: # echo 'hist:key=id.syscall if common_pid==32396' > \ events/raw_syscalls/sys_enter/trigger # cat events/raw_syscalls/sys_enter/hist # event histogram # # trigger info: hist:keys=id.syscall:vals=hitcount:sort=hitcount:size=2048 if common_ pid==32396 [active] # { id: sys_write { id: sys_read

[ [

1] } hitcount: 0] } hitcount:

106425 106425

Totals: Hits: 212850 Entries: 2 Dropped: 0

Теперь гистограмма показывает количество обращений к системным вызовам только для процесса с указанным PID, а модификатор .syscall добавил имена системных вызовов. Это подтверждает, что dd(1) вызывает read(2) и write(2). Другое решение — использовать несколько ключей, как показано в следующем разделе.

14.10.5. Гистограмма с несколькими ключами В следующем примере используется второй ключ — идентификатор системного вызова:

14.10. Триггеры hist  883 # echo 'hist:key=common_pid.execname,id' > events/raw_syscalls/sys_enter/trigger # sleep 10 # cat events/raw_syscalls/sys_enter/hist # event histogram # # trigger info: hist:keys=common_pid.execname,id:vals=hitcount:sort=hitcount:si ze=2048 [active] # [...] { common_pid: sshd [ 14250], id: 23 } hitcount: 36 { common_pid: bash [ 14261], id: 13 } hitcount: 42 { common_pid: sshd [ 14250], id: 14 } hitcount: 72 { common_pid: dd [ 14325], id: 0 } hitcount: 9195176 { common_pid: dd [ 14325], id: 1 } hitcount: 9195176 Totals: Hits: 18391064 Entries: 75 Dropped: 0 Dropped: 0

Теперь вывод включает имя и идентификатор PID процесса, выполняется дополнительная разбивка по идентификатору системного вызова. Как показывает этот вывод, процесс dd с идентификатором PID 142325 обращался к двум системным вызовам с идентификаторами 0 и 1. Ко второму ключу можно добавить модификатор .syscall, чтобы включить в вывод имена системных вызовов.

14.10.6. Трассировки стека в роли ключей Часто бывает нужно узнать путь кода, который привел к событию, и я предложил Тому Занусси добавить в Ftrace возможность использовать в роли ключа трассировку стека ядра. Вот подсчет путей в коде, которые привели к точке трассировки block:block_rq_issue: # echo 'hist:key=stacktrace' > events/block/block_rq_issue/trigger # sleep 10 # cat events/block/block_rq_issue/hist [...] { stacktrace: nvme_queue_rq+0x16c/0x1d0 __blk_mq_try_issue_directly+0x116/0x1c0 blk_mq_request_issue_directly+0x4b/0xe0 blk_mq_try_issue_list_directly+0x46/0xb0 blk_mq_sched_insert_requests+0xae/0x100 blk_mq_flush_plug_list+0x1e8/0x290 blk_flush_plug_list+0xe3/0x110 blk_finish_plug+0x26/0x34 read_pages+0x86/0x1a0 __do_page_cache_readahead+0x180/0x1a0 ondemand_readahead+0x192/0x2d0 page_cache_sync_readahead+0x78/0xc0 generic_file_buffered_read+0x571/0xc00

884  Глава 14. Ftrace generic_file_read_iter+0xdc/0x140 ext4_file_read_iter+0x4f/0x100 new_sync_read+0x122/0x1b0 } hitcount: 266 Totals: Hits: 522 Entries: 10 Dropped: 0

Я сократил вывод и оставил только последнюю, наиболее частую трассировку стека. Она показывает, что дисковый ввод/вывод производился через вызов функции new_sync_read(), которая вызывала ext4_file_read_iter() и т. д.

14.10.7. Синтетические события Именно здесь все начинает становиться по-настоящему странным (если еще не стало). При желании можно создать синтетическое событие, которое запускается другими событиями и может комбинировать их аргументы настраиваемыми способами. Чтобы получить доступ к аргументам предыдущих событий, сохраните их в гистограмму и извлеките из более позднего синтетического события. Cинтетические события наиболее полезны для основного варианта использования: создания нестандартных гистограмм задержки. С помощью синтетического события можно сохранить отметку времени для одного события, а затем извлечь ее по другому событию и рассчитать разность времени. Вот пример использования синтетического события syscall_latency для вычисления задержки всех системных вызовов и представления ее распределения в виде гистограммы с разбивкой по идентификатору и имени системного вызова: # cd /sys/kernel/debug/tracing # echo 'syscall_latency u64 lat_us; long id' >> synthetic_events # echo 'hist:keys=common_pid:ts0=common_timestamp.usecs' >> \ events/raw_syscalls/sys_enter/trigger # echo 'hist:keys=common_pid:lat_us=common_timestamp.usecs-$ts0:'\ 'onmatch(raw_syscalls.sys_enter).trace(syscall_latency,$lat_us,id)' >>\ events/raw_syscalls/sys_exit/trigger # echo 'hist:keys=lat_us,id.syscall:sort=lat_us' >> \ events/synthetic/syscall_latency/trigger # sleep 10 # cat events/synthetic/syscall_latency/hist [...] { lat_us: 5779085, id: sys_epoll_wait [232] } hitcount: { lat_us: 6232897, id: sys_poll [ 7] } hitcount: { lat_us: 6233840, id: sys_poll [ 7] } hitcount: { lat_us: 6233884, id: sys_futex [202] } hitcount: { lat_us: 7028672, id: sys_epoll_wait [232] } hitcount: { lat_us: 9999049, id: sys_poll [ 7] } hitcount: { lat_us: 10000097, id: sys_nanosleep [ 35] } hitcount: { lat_us: 10001535, id: sys_wait4 [ 61] } hitcount: { lat_us: 10002176, id: sys_select [ 23] } hitcount: [...]

1 1 1 1 1 1 1 1 1

14.10. Триггеры hist  885 Я сократил вывод, чтобы показать только самые большие задержки. Гистограмма подсчитывает задержки (в микросекундах) по идентификаторам системных вызовов и, как следует из этого вывода, sys_nanosleep один раз показала задержку в 10 000 097 мкс. Скорее всего, она соответствует команде sleep 10, которую я запустил, чтобы установить продолжительности сбора данных. Вывод получился очень объемным, потому что записывает ключ для каждой комбинации микросекунд и идентификатора системного вызова, и в действительности я превысил размер гистограммы, по умолчанию равный 2048. При необходимости размер можно увеличить, добавив оператор: size = ... в объявление гистограммы. Также можно использовать модификатор .log2 для ­записи задержек с шагом, кратным log2. Это значительно сократит количество элементов в гистограмме и при этом обеспечит достаточное разрешение для анализа задержек. Чтобы отключить и очистить синтетическое событие, нужно выполнить все команды в обратном порядке, добавив префикс «!». В табл. 14.4 на примерах фрагментов кода я объясняю, как работает это синтетическое событие. Таблица 14.4. Объяснение примера с синтетическим событием Описание

Синтаксис

Создается синтетическое событие с именем syscall_ latency и двумя аргументами: lat_us и id

echo 'syscall_latency u64 lat_us; long id' >> synthetic_events

При появлении события sys_enter выполнить запись в гистограмму, используя в качестве ключа common_pid (идентификатор PID текущего процесса),

echo 'hist:keys=common_pid: ... >>

и сохранить текущее время в микросекундах в гистограмме — в переменной с именем ts0, связанной с ключом гистограммы (common_pid)

ts0=common_timestamp.usecs

При появлении события sys_exit использовать common_pid в качестве ключа гистограммы и

echo 'hist:keys=common_pid: ... >>

вычислить задержку как разность между текущим и начальным временем, сохраненным в ts0 предыдущим событием, и записать ее в гистограмму — в переменную с именем lat_us,

lat_us=common_timestamp.usecs$ts0

сравнить ключи в гистограмме для этого события и события sys_enter. Если они совпадают (одно и то же значение common_pid), тогда задержка в lat_us вычислена правильно (от sys_enter до sys_exit для одного и того же PID), поэтому,

onmatch(raw_syscalls.sys_enter)

events/raw_syscalls/sys_enter/ trigger

events/raw_syscalls/sys_exit/ trigger

886  Глава 14. Ftrace Таблица 14.4 (окончание) Описание

Синтаксис

наконец, запускаем наше синтетическое событие syscall_latency с аргументами lat_us и id

.trace(syscall_latency, $lat_us,id)

Вывести это синтетическое событие в виде гистограммы с полями lat_us и id

echo 'hist:keys=lat_us, id.syscall:sort=lat_us' >> events/synthetic/ syscall_latency/trigger

Гистограммы Ftrace реализованы на основе хеш-объектов (хранилищ пар ключ/ значение), и в более ранних примерах эти хеши использовались только для вывода: для подсчета количества системных вызовов по их идентификаторам и идентификаторам PID процессов. При использовании синтетических событий мы выполняем две дополнительные операции с этими хешами: а) сохраняем значения, которые не являются частью вывода (отметки времени) и б) в одном событии извлекаем пары ключ/значение, которые были установлены другим событием. Также мы выполняем арифметические операции, в частности вычитание. В каком-то смысле начинаем писать мини-программы. Дополнительную информацию о синтетических событиях ищите в документации [Zanussi 20]. Я не раз делился своим мнением и взглядами, прямо или косвенно, с инженерами Ftrace и BPF. Считаю, что в эволюции Ftrace есть смысл, потому что проблемы, которые я поднимал ранее, постепенно решаются. Я бы резюмировал развитие так: — Ftrace великолепен, но мне приходится использовать BPF для подсчета событий по PID и трассировкам стека. — Нет проблем, вот тебе триггеры hist. — Здорово! Но я все равно вынужден использовать BPF для вычисления нестандартных задержек. — Нет проблем, вот тебе синтетические события. — Отлично, я проверю эту возможность, когда закончу писать «BPF Performance Tools». — Ты серьезно? Да, теперь я хочу изучить возможность применения синтетических событий в определенных случаях. Это невероятно мощная возможность, встроенная в ядро, которую можно использовать с помощью обычных сценариев на языке командной оболочки. (И да, я закончил писать книгу о BPF, но потом занялся этой книгой.)

14.11. trace-cmd  887

14.11. TRACE-CMD trace-cmd — это интерфейс к Ftrace с открытым исходным кодом, разработанный Стивеном Ростедтом (Steven Rostedt) и другими [trace-cmd 20]. Он поддерживает подкоманды и параметры для настройки системы трассировки, двоичный формат вывода и другие функции. Для доступа к источникам событий он может использовать трассировщики function и function_graph в Ftrace, а также точки трассировки и уже настроенные зонды kprobes и uprobes. Вот пример использования trace-cmd для регистрации вызовов функции ядра do_nanosleep() с помощью трассировщика function в течение 10 с (выполнением фиктивной команды sleep(1)): # trace-cmd record -p function -l do_nanosleep sleep 10 plugin 'function' CPU0 data recorded at offset=0x4fe000 0 bytes in size CPU1 data recorded at offset=0x4fe000 4096 bytes in size # trace-cmd report CPU 0 is empty cpus=2 sleep-21145 [001] 573259.213076: function: multipathd-348 [001] 573259.523759: function: multipathd-348 [001] 573260.523923: function: multipathd-348 [001] 573261.524022: function: multipathd-348 [001] 573262.524119: function: [...]

do_nanosleep do_nanosleep do_nanosleep do_nanosleep do_nanosleep

Вывод начинается с команды sleep(1), которая была вызвана командой trace-cmd (сначала она настраивает трассировку, а затем запускает указанную команду), за которой следуют различные вызовы из процесса multipathd с PID 348. Этот пример также показывает, что trace-cmd более лаконичен, чем эквивалентные команды tracefs в /sys. Кроме того, этот интерфейс безопаснее: многие подкоманды сбрасывают состояние трассировки, когда это нужно. Интерфейс trace-cmd обычно распространяется в составе пакета «trace-cmd», но при желании его исходный код можно получить на сайте проекта [trace-cmd 20]. Далее в этом разделе я покажу некоторые подкоманды и возможности трассировки trace-cmd. Полную информацию обо всех возможностях и синтаксисе из примеров ниже ищите в документации по trace-cmd.

14.11.1. Обзор подкоманд Основные возможности trace-cmd доступны в виде подкоманд­ — trace-cmd record. Подкоманды, поддерживаемые последней версией trace-cmd (2.8.3), перечислены в табл. 14.5.

888  Глава 14. Ftrace Таблица 14.5. Некоторые подкоманды trace-cmd Команда

Описание

record

Выполняет трассировку и записывает результаты в файл trace.dat

report

Читает данные из файла trace.dat

stream

Выполняет трассировку и выводит результаты в стандартный вывод

list

Перечисляет доступные события трассировки

stat

Сообщает состояние подсистемы трассировки в ядре

profile

Трассирует и генерирует отчет со значениями времени и задержек в ядре

listen

Принимает сетевые запросы на выполнение трассировки

Также в число поддерживаемых входят подкоманды start, stop, restart и  clear, управляющие трассировкой аналогично подкоманде record. В будущих версиях в trace-cmd могут появиться дополнительные подкоманды. Чтобы не пропустить их, выполните команду trace-cmd без аргументов, которая выведет полный список подкоманд. Каждая подкоманда имеет множество параметров. Получить их список можно с помощью параметра -h. Вот список параметров для подкоманды record: # trace-cmd record -h trace-cmd version 2.8.3 usage: trace-cmd record [-v][-e event [-f filter]][-p plugin][-F][-d][-D][-o file] \ [-q][-s usecs][-O option ][-l func][-g func][-n func] \ [-P pid][-N host:port][-t][-r prio][-b size][-B buf][command ...] [-m max][-C clock] -e запустить команду, активировав трассировку указанного события -f фильтр по содержимому для события, указанного в параметре -e -R триггер для события, указанного в параметре -p запустить команду, активировав указанный плагин -F фильтровать события только для указанного процесса -P трассировать только процесс с указанным PID, действует так же, как параметр -F для команды -c трассировать также дочерние процессы при наличии параметра -F (или -P, если ядро поддерживает его) -C установить значение часов трассировки -T включить сохранение трассировки стека для каждого события -l трассировать только указанную функцию -g создать граф для указанной функции -n не трассировать указанную функцию1 [...]

Это перевод вывода команды. — Примеч. пер.

1

14.11. trace-cmd  889 Я обрезал список в этом выводе, оставив только первые 12 из 35 параметров, которые используются особенно часто. Обратите внимание, что под термином плагин (plugin; -p) здесь подразумеваются трассировщики Ftrace, в том числе function, function_graph и hwlat.

14.11.2. Однострочные сценарии для trace-cmd В этих примерах однострочных сценариев показаны различные возможности trace-cmd. Их синтаксис описан на соответствующих страницах справочного руководства man.

Список событий Вывести список всех источников событий и параметров: trace-cmd list

Вывести список трассировщиков, доступных в Ftrace: trace-cmd list -t

Вывести список источников событий (точек трассировки, зондов kprobe и uprobe): trace-cmd list -e

Вывести список точек трассировки системных вызовов: trace-cmd list -e syscalls:

Показать файл формата для указанной точки трассировки: trace-cmd list -e syscalls:sys_enter_nanosleep -F

Трассировка функций Трассировать функцию ядра в масштабе всей системы: trace-cmd record -p function -l имя_функции

Трассировать все функции ядра с именами, начинающимися с «tcp_», в масштабе всей системы до нажатия Ctrl-C: trace-cmd record -p function -l 'tcp_*'

Трассировать все функции ядра с именами, начинающимися с «tcp_», в масштабе всей системы в течение 10 с: trace-cmd record -p function -l 'tcp_*' sleep 10

Трассировать все функции ядра с именами, начинающимися с «vfs_», которые вызываются командой ls(1): trace-cmd record -p function -l 'vfs_*' -F ls

890  Глава 14. Ftrace Трассировать все функции ядра с именами, начинающимися с «vfs_», которые вызываются командной оболочкой bash(1) и всеми ее дочерними процессами: trace-cmd record -p function -l 'vfs_*' -F -c bash

Трассировать все функции ядра с именами, начинающимися с «vfs_», которые вызываются процессом с PID 21124: trace-cmd record -p function -l 'vfs_*' -P 21124

Трассировка графа функций Трассировать функцию ядра и вызываемые ею функции в масштабе всей системы: trace-cmd record -p function_graph -g имя_функции

Трассировать функцию ядра do_nanosleep() и вызываемые ею функции в масштабе всей системы в течение 10 с: trace-cmd record -p function_graph -g do_nanosleep sleep 10

Трассировка событий Трассировать событие запуска нового процесса с использованием точки трассировки sched:sched_process_exec до нажатия Ctrl-C: trace-cmd record -e sched:sched_process_exec

Трассировать событие запуска нового процесса с использованием точки трассировки sched:sched_process_exec (короткая версия): trace-cmd record -e sched_process_exec

Трассировать запросы блочного ввода/вывода с записью трассировок стека ядра: trace-cmd record -e block_rq_issue -T

Трассировать все точки трассировки из семейства block до нажатия Ctrl-C: trace-cmd record -e block

Трассировать предварительно созданный зонд kprobe с именем «brendan» в течение 10 с: trace-cmd record -e probe:brendan sleep 10

Трассировать все системные вызовы, к которым обращается команда ls(1): trace-cmd record -e syscalls -F ls

Отчеты Вывести содержимое файла trace.dat: trace-cmd report

14.11. trace-cmd  891 Вывести содержимое файла trace.dat, только относящееся к процессору CPU 0: trace-cmd report --cpu 0

Другие возможности Трассировать события из плагина sched_switch: trace-cmd record -p sched_switch

Принимать запросы на трассировку на порту TCP с номером 8081: trace-cmd listen -p 8081

Подключиться к удаленному хосту для выполнения подкоманды record: trace-cmd record ... -N addr:port

14.11.3. Сравнение trace-cmd и perf(1) Своим стилем использования подкоманд trace-cmd напоминает профилировщик perf(1), описанный в главе 13. Они действительно имеют схожие возможности. В табл. 14.6 проводится сравнение trace-cmd и perf(1). Таблица 14.6. Сравнение perf(1) и trace-cmd Атрибут

perf(1)

trace-cmd

Двоичный выходной файл

perf.data

trace.dat

Точки трассировки

Да

Да

Зонды kprobes

Да

Частично(1)

Зонды uprobes

Да

Частично(1)

Зонды USDT

Да

Частично(1)

Счетчики PMC

Да

Нет

Выборка по времени

Да

Нет

Трассировка функций

Частично(2)

Да

Трассировка графов функций

Частично(2)

Да

Трассировка по сети (клиент/сервер)

Нет

Да

Оверхед на вывод в файл

Низкий

Очень низкий

Внешние интерфейсы

Разнообразные

KernelShark

Исходный код

В исходном коде ядра Linux в tools/perf

git.kernel.org

yy Частично(1): trace-cmd поддерживает эти события, только если они уже были созданы другими способами и есть в /sys/kernel/debug/tracing/events.

892  Глава 14. Ftrace

yy Частично(2): perf(1) поддерживает их с помощью подкоманды ftrace, но интеграция с perf(1) неполная (например, не поддерживается файл perf.data).

Чтобы показать эту схожесть, ниже приводятся команды, запускающие трассировку системных вызовов через точку трассировки sys_enter_read в масштабе всей системы в течение 10 с, а затем выводящие результаты трассировки. С помощью perf(1): # perf record -e syscalls:sys_enter_nanosleep -a sleep 10 # perf script

...и trace-cmd: # trace-cmd record -e syscalls:sys_enter_nanosleep sleep 10 # trace-cmd report

Одно из преимуществ trace-cmd — лучшая поддержка трассировщиков function и function_graph.

14.11.4. trace-cmd function_graph В начале этого раздела я показал применение трассировщика function с использованием trace-cmd. Следующая команда демонстрирует применение function_graph для трассировки той же функции ядра do_nanosleep(): # trace-cmd record -p function_graph -g do_nanosleep sleep 10 plugin 'function_graph' CPU0 data recorded at offset=0x4fe000 12288 bytes in size CPU1 data recorded at offset=0x501000 45056 bytes in size # trace-cmd report | cut -c 66-

0.250 0.688 0.190 0.153 [...]

us us us us

| do_nanosleep() { | hrtimer_start_range_ns() { | lock_hrtimer_base.isra.0() { | _raw_spin_lock_irqsave(); | } | ktime_get(); | get_nohz_timer_target();

В этом примере я использовал команду cut(1), чтобы убрать из графа функции столбцы с временами. Из-за этого в выводе нет полей трассировки, показанных в предыдущем примере.

14.11.5. KernelShark KernelShark — это визуальный пользовательский интерфейс для изучения выходных файлов trace-cmd, созданный автором Ftrace Стивеном Ростедтом. Первоначально

14.11. trace-cmd  893 использовавший фреймворк GTK, KernelShark был переписан на Qt Йорданом Караджовым (Yordan Karadzhov) — нынешним мейнтейнером проекта. KernelShark можно установить в составе пакета kernelshark, если он доступен, или из исходного кода с сайта проекта [KernelShark 20]. Версия 1.0 — это версия на Qt, а версии 0.99 и ниже — это версии на GTK. Вот пример регистрации событий всех точек трассировки планировщика и последующая их визуализация с помощью KernelShark: # trace-cmd record -e 'sched:*' # kernelshark

По умолчанию KernelShark читает выходной файл trace.dat, созданный tracecmd (при необходимости с помощью параметра -i можно указать другой файл). На рис. 14.3 показано окно KernelShark с содержимым файла.

Рис. 14.3. KernelShark В верхней части окна отображается шкала времени для каждого процессора и задачи, окрашенные в разные цвета. В нижней части таблица событий. KernelShark поддерживает интерактивные возможности: щелчок и перетаскивание вправо увеличивают масштаб до выбранного диапазона времени, а щелчок и перетаскивание влево уменьшают масштаб. В ответ на щелчок правой кнопкой мыши на ­с обытии предлагаются дополнительные действия, например настройка фильтров. KernelShark можно использовать для выявления проблем с производительностью, вызванных взаимодействиями между различными потоками.

894  Глава 14. Ftrace

14.11.6. Документация для trace-cmd После установки пакета документация для trace-cmd должна быть доступна в виде страниц в справочном руководстве man (trace-cmd-record(1)), которые также можно найти в каталоге Documentation в дереве исходных текстов trace-cmd. Рекомендую посмотреть доклад «Understanding the Linux Kernel (via ftrace)» Стивена Ростедта, ведущего разработчика Ftrace и trace-cmd:

yy презентация: https://www.slideshare.net/ennael/kernel-recipes-2017-understanding-the-linuxkernel-via-ftrace-steven-rostedt;

yy видео: https://www.youtube.com/watch?v=2ff-7UTg5rE.

14.12. PERF FTRACE Утилита perf(1), о которой шла речь в главе 13, имеет подкоманду ftrace, благодаря которой можно использовать трассировщики function и function_graph. Вот пример использования трассировщика function для трассировки функции ядра do_nanosleep(): # perf ftrace -T do_nanosleep -a sleep 10 0) sleep-22821 | | do_nanosleep() { 1) multipa-348 | | do_nanosleep() { 1) multipa-348 | $ 1000068 us | } 1) multipa-348 | | do_nanosleep() { 1) multipa-348 | $ 1000068 us | } [...]

А это пример использования трассировщика function_graph: # perf ftrace -G do_nanosleep -a sleep 10 1) sleep-22828 | | do_nanosleep() { 1) sleep-22828 | ==========> | 1) sleep-22828 | | smp_irq_work_interrupt() { 1) sleep-22828 | | irq_enter() { 1) sleep-22828 | 0.258 us | rcu_irq_enter(); 1) sleep-22828 | 0.800 us | } 1) sleep-22828 | | __wake_up() { 1) sleep-22828 | | __wake_up_common_lock() { 1) sleep-22828 | 0.491 us | _raw_spin_lock_irqsave(); [...]

Подкоманда ftrace поддерживает несколько параметров, включая -p для фильтрации по значению PID. Это простая обертка, не интегрированная с другими возможностями perf(1): например, она выводит результаты трассировки на стандартный вывод и не использует файл perf.data.

14.13. perf-tools  895

14.13. PERF-TOOLS perf-tools — это набор инструментов с открытым исходным кодом, основанных на Ftrace и perf(1), для анализа производительности, который был разработан мной и по умолчанию устанавливается на серверы Netflix [Gregg 20i]. Я спроектировал эти инструменты так, чтобы их было легко установить (минимум зависимостей) и просто использовать: каждый инструмент должен делать что-то одно, и делать это хорошо. Сами инструменты perf-tools в большинстве своем реализованы в виде сценариев командной оболочки, автоматизирующих установку файлов tracefs /sys. Вот пример использования execsnoop(8) для трассировки событий запуска новых процессов: # execsnoop Tracing exec()s. Ctrl-C to end. PID PPID ARGS 6684 6682 cat -v trace_pipe 6683 6679 gawk -v o=1 -v opt_name=0 -v name= -v opt_duration=0 [...] 6685 20997 man ls 6695 6685 pager 6691 6685 preconv -e UTF-8 6692 6685 tbl 6693 6685 nroff -mandoc -rLL=148n -rLT=148n -Tutf8 6698 6693 locale charmap 6699 6693 groff -mtty-char -Tutf8 -mandoc -rLL=148n -rLT=148n 6700 6699 troff -mtty-char -mandoc -rLL=148n -rLT=148n -Tutf8 6701 6699 grotty [...]

Вывод начинается с отображения команд cat(1) и gawk(1), которые используются самой утилитой excesnoop(8). Далее следуют команды, которые запускает man ls. Этот подход можно использовать для отладки скоротечных процессов, которые могут оставаться незамеченными другими инструментами. execsnoop(8) поддерживает параметры, включая -t — для вывода отметок времени и -h — для вывода справки по использованию. У execsnoop(8) и других инструментов есть страница в справочном руководстве man и файлы примеров.

14.13.1. Покрытие инструментами На рис. 14.4 показаны различные инструменты perf и области системы, которые они покрывают. Многие из них являются узкоспециализированными инструментами и обозначены однонаправленной стрелкой, некоторые — многоцелевыми, они перечислены слева рядом с двунаправленными стрелками, показывающими их покрытие.

896  Глава 14. Ftrace

Приложения

Среда времени выполнения

Системные библиотеки Интерфейс системных вызовов Виртуальная файловая система Файловые системы

Сокеты TCP/UDP

Планировщик

Диспетчер томов

IP Сетевое Блочное устройство устройство Драйверы устройств

Виртуальная память

Рис. 14.4. perf-tools

14.13.2. Специализированные инструменты Специализированные инструменты отмечены на рис. 14.4 однонаправленной стрелкой. Некоторые из них были представлены в предыдущих главах. Специализированные инструменты, например execsnoop(8), делают что-то одно, и делают это хорошо (философия Unix). Согласно этой философии, выходные данные должны быть по умолчанию краткими и очевидными, что оказывает немалую помощь в их изучении. Вы можете «просто запустить execsnoop», не вдаваясь в изу­ чение параметров командной строки, и получить достаточный объем информации для решения проблемы. Параметры обычно предназначены для тонкой настройки. Специализированные инструменты перечислены в табл. 14.7. Таблица 14.7. Специализированные инструменты perf-tools Инструмент

Использует

Описание

bitesize(8)

perf

Отображает размеры дискового ввода/вывода в виде гистограммы

cachestat(8)

Ftrace

Выводит статистику попаданий и промахов кэша страниц

execsnoop(8)

Ftrace

Трассирует запуск новых процессов (через execve(2)) и их аргументы

iolatency(8)

Ftrace

Отображает задержки дискового ввода/вывода в виде гистограммы

iosnoop(8)

Ftrace

Трассирует дисковый ввод/вывод и отображает подробную информацию, включая задержки

killsnoop(8)

Ftrace

Трассирует отправку сигналов процессам с помощью kill(2) и отображает информацию о сигналах

14.13. perf-tools  897

Инструмент

Использует

opensnoop(8)

Ftrace

Трассирует семейство системных вызовов open(2) и отображает имена открывавшихся файлов

tcpretrans(8)

Ftrace

Трассирует повторные передачи TCP, выводит IP-адреса и состояние ядра

Описание

Инструмент execsnoop(8) был показан выше. Посмотрим также на инструмент iolatency(8), отображающий задержки дискового ввода/вывода в виде гистограммы: # iolatency Tracing block I/O. Output every 1 seconds. Ctrl-C to end. .. -> -> ->

=(ms) 0 1 2 4

.. -> -> -> ->

=(ms) 0 1 2

.. -> -> ->

=(ms) 0 1 2

Как показывает этот вывод, задержка ввода/вывода обычно оставалась довольно низкой, от 0 до 1 мс. Порядок реализации этого инструмента наглядно объясняет необходимость расширенного BPF. iolatency(8) трассирует события запуска и завершения блочного ввода/вывода, передает все события в пространство пользователя, анализирует их и преобразует в гистограммы с помощью awk(1). Поскольку на большинстве серверов дисковый ввод/вывод выполняется относительно редко, этот подход почти не влечет за собой оверхед. Но для более частых событий, таких как сетевой ввод/вывод или планирование процессов, оверхед может оказаться недопустимо высоким. В расширенном BPF эта проблема решается за счет формирования гистограммы в пространстве ядра; в пространство пользователя передается только окончательный результат, что значительно сокращает оверхед. Ftrace поддерживает некоторые похожие возможности в триггерах hist и синтетических событиях, описанных в разделе 14.10 «Триггеры hist» (я собираюсь обновить утилиту iolatency(8) и использовать в ней эти новые возможности). Я разработал решение для создания гистограмм еще до появления BPF и представил его как многоцелевой инструмент perf-stat-hist(8).

898  Глава 14. Ftrace

14.13.3. Многоцелевые инструменты Многоцелевые инструменты перечислены и описаны в табл. 14.8 . Каждый из них поддерживает несколько источников событий и может играть множество ролей, как perf(1) и trace-cmd, хотя это также усложняет их использование. Таблица 14.8. Многоцелевые инструменты perf-tools Инструмент

Использует

Описание

funccount(8)

Ftrace

Подсчитывает вызовы функций ядра

funcgraph(8)

Ftrace

Трассирует функции ядра и показывает путь выполнения через дочерние функции

functrace(8)

Ftrace

Трассирует функции ядра

funcslower(8)

Ftrace

Трассирует функции ядра, выполняющиеся дольше заданного порога

kprobe(8)

Ftrace

Динамическая трассировка функций ядра

perf-stat-hist(8)

perf(1)

Мощный инструмент для агрегирования аргументов точек трассировки

syscount(8)

perf(1)

Обобщает статистики о системных вызовах

tpoint(8)

Ftrace

Трассирует точки трассировки

uprobe(8)

Ftrace

Динамическая трассировка функций в пространстве пользователя

Чтобы облегчить использование этих инструментов, вы можете собрать свою коллекцию однострочных сценариев. Некоторые из таких сценариев, похожих на однострочные сценарии для perf(1) и trace-cmd, приводятся в разделе ниже.

14.13.4. Однострочные сценарии для perf-tools Следующие однострочные сценарии выполняют трассировку в масштабе всей системы, пока не будет нажато Ctrl-C, если явно не указано иное. Они объединены в группы по признаку использования профилировщиков Ftrace, трассировщиков Ftrace и трассировки событий (точки трассировки, kprobes, uprobes).

Профилировщики Ftrace Подсчитать вызовы функций ядра с именами, начинающимися на «tcp_»: funccount 'tcp_*'

Подсчитать вызовы функций VFS. Выводит 10 наиболее часто вызывавшихся функций каждую секунду: funccount -t 10 -i 1 'vfs*'

14.13. perf-tools  899

Трассировщики Ftrace Трассировать функцию ядра do_nanosleep() и показать вызываемые ею функции: funcgraph do_nanosleep

Трассировать функцию ядра do_nanosleep() и показать вызываемые ею функции вплоть до третьего уровня вложенности: funcgraph -m 3 do_nanosleep

Подсчитать вызовы функций ядра с именами, оканчивающимися на «sleep», которые вызываются процессом с PID 198: functrace -p 198 '*sleep'

Трассировать вызовы vfs_read(), продолжающиеся дольше 10 мс: funcslower vfs_read 10000

Трассировка событий Трассировать функцию ядра do_sys_open() с помощью kprobe: kprobe p:do_sys_open

Трассировать возврат из do_sys_open() с помощью kretprobe и вывести возвращаемое значение: kprobe 'r:do_sys_open $retval'

Трассировать аргумент с режимом открытия файла в вызове do_sys_open(): kprobe 'p:do_sys_open mode=$arg3:u16'

Трассировать аргумент с режимом открытия файла в вызове do_sys_open() (в архитектуре x86_64): kprobe 'p:do_sys_open mode=%dx:u16'

Трассировать аргумент с именем файла в вызове do_sys_open() и вывести его как строку: kprobe 'p:do_sys_open filename=+0($arg2):string'

Трассировать аргумент с именем файла в вызове do_sys_open() и вывести его как строку (в архитектуре x86_64): kprobe 'p:do_sys_open filename=+0(%si):string'

Трассировать do_sys_open(), когда имя файла совпадает с шаблоном «*stat»: kprobe 'p:do_sys_open file=+0($arg2):string' 'file ~ "*stat"'

900  Глава 14. Ftrace Трассировать tcp_retransmit_skb() и вывести трассировки стека ядра: kprobe -s p:tcp_retransmit_skb

Вывести список точек трассировки: tpoint -l

Трассировать дисковый ввод/вывод и вывести трассировки стека ядра: tpoint -s block:block_rq_issue

Трассировать вызовы функции readline() в пространстве пользователя во всех процессах «bash»: uprobe p:bash:readline

Трассировать возврат из readline() во всех процессах «bash» и вывести возвращаемое значение как строку: uprobe 'r:bash:readline +0($retval):string'

Трассировать точку входа в readline() в /bin/bash и вывести аргумент как строку (x86_64): uprobe 'p:/bin/bash:readline prompt=+0(%di):string'

Трассировать вызов gettimeofday() из библиотеки libc только для процесса с PID 1234: uprobe -p 1234 p:libc:gettimeofday

Трассировать возврат из fopen(), только когда возвращается NULL (и использовать псевдоним «file»): uprobe 'r:libc:fopen file=$retval' 'file == 0'

Регистры процессора Псевдонимы аргументов функций ($arg1, ..., $argN) — это новая возможность Ftrace, появившаяся в Linux 4.20. В системах с более старыми ядрами (или с аппаратными архитектурами без псевдонимов) вместо псевдонимов используют имена регистров процессоров, как описано в разделе 14.6.2 «Аргументы». Представленные там однострочные сценарии включали несколько примеров использования регистров x86_64 (%di, %si, %dx). Соглашения о вызовах задокументированы на странице справочного руководства man для syscall(2): $ man 2 syscall [...] Arch/ABI arg1 arg2 arg3 arg4 arg5 arg6 arg7 Notes ────────────────────────────────────────────────────────────── [...] sparc/32 o0 o1 o2 o3 o4 o5 -

14.13. perf-tools  901

[...]

sparc/64 tile x86-64 x32

o0 R00 rdi rdi

o1 R01 rsi rsi

o2 R02 rdx rdx

o3 R03 r10 r10

o4 R04 r8 r8

o5 R05 r9 r9

-

14.13.5. Пример Ниже показано применение funccount(8) для подсчета вызовов VFS (функция с именами, соответствующими шаблону «vfs_*»): # funccount 'vfs_*' Tracing "vfs_*"... Ctrl-C to end. ^C FUNC vfs_fsync_range vfs_statfs vfs_readlink vfs_statx vfs_write vfs_statx_fd vfs_open vfs_getattr vfs_getattr_nosec vfs_read

COUNT 10 10 35 673 782 922 1003 1390 1390 2604

Как показывает этот вывод, в течение трассировки функция vfs_read() была вызвана 2604 раза. Я регулярно использую funccount(8), чтобы определить, какие функции ядра вызываются и какие из них вызываются особенно часто. Этот инструмент имеет относительно низкий оверхед, поэтому его можно использовать для проверки частоты вызова функций перед более дорогостоящей трассировкой.

14.13.6. Сравнение perf-tools и BCC/BPF Изначально набор инструментов perf-tools разрабатывался для использования в облаке Netflix, когда оно работало под управлением Linux 3.2, где не было расширенного BPF. Сейчас в Netflix используются новые ядра, и я переписал многие из этих инструментов так, чтобы они использовали BPF. Например perf-tools и BCC имеют свои собственные версии funccount(8), execsnoop(8), opensnoop(8) и других инструментов. BPF обеспечивает возможность программирования и обладает более мощными возможностями (его интерфейсы BCC и bpftrace рассматриваются в главе 15). Однако и у perf-tools есть некоторые преимущества1:

yy funccount(8): версия в perf-tools использует более эффективные и менее ограниченные средства профилирования функций из Ftrace, чем имеющиеся в текущей версии BPF и основанные на kprobe.

1

Первоначально я полагал, что откажусь от perf-tools, когда закончу реализацию трассировки в BPF, но затем решил продолжить использовать их по этим причинам.

902  Глава 14. Ftrace

yy funcgraph(8): в BCC вообще нет такого инструмента, потому что он использует трассировщик function_graph из Ftrace.

yy Триггеры hist: в будущем этот механизм сможет обеспечить более эффективную работу инструментов perf-tools по сравнению с их версиями в BPF, основанными на kprobe.

yy Зависимости: инструменты perf-tools сохраняют свою актуальность для сред

с ограниченными ресурсами (например, для встраиваемых систем Linux), потому что им обычно достаточно лишь командной оболочки и awk(1).

Сам я иногда использую инструменты perf-tools для перекрестной проверки и отладки проблем с инструментами BPF1.

14.13.7. Документация Обычно инструменты выводят сообщение о порядке использования, описывающее их синтаксис. Например: # funccount -h ИСПОЛЬЗОВАНИЕ: funccount [-hT] [-i secs] [-d secs] [-t top] funcstring -d seconds # общая продолжительность трассировки -h # это сообщение о порядке использования -i seconds # сводная информация за интервал -t top # показать только top самых часто вызываемых функций -T # включить отметки времени (для -i) Примеры: funccount 'vfs*' # трассировать все функции, соответствующие "vfs*" funccount -d 5 'tcp*' # трассировать все функции "tcp*" в течение 5 с funccount -t 10 'ext3*' # показать 10 самых часто вызываемых функций "ext3*" funccount -i 1 'ext3*' # выводить сводку каждую секунду funccount -i 1 -d 5 'ext3*' # выводить сводку каждую секунду в течение 5 с

У каждого инструмента есть своя страница в справочном руководстве man и файл с примерами в репозитории perf-tools (funccount_example.txt).

14.14. ДОКУМЕНТАЦИЯ ДЛЯ FTRACE Ftrace (и события трассировки) подробно описываются в исходном коде Linux в каталоге Documentation/trace. Эта документация также доступна в интернете:

yy yy yy

https://www.kernel.org/doc/html/latest/trace/ftrace.html; https://www.kernel.org/doc/html/latest/trace/kprobetrace.html; https://www.kernel.org/doc/html/latest/trace/uprobetracer.html;

Я бы сказал так: «Человек с одним трассировщиком знает, какие события произошли. Человек с двумя трассировщиками знает, что один из них сломан, и обращается к lkml в надежде найти патч».

1

14.15. Ссылки  903

yy yy

https://www.kernel.org/doc/html/latest/trace/events.html; https://www.kernel.org/doc/html/latest/trace/histogram.html.

Ресурсы, где доступны внешние интерфейсы:

yy trace-cmd: https://trace-cmd.org; yy perf ftrace: исходный код Linux: tools/perf/Documentation/perf-ftrace.txt; yy perf-tools: https://github.com/brendangregg/perf-tools.

14.15. ССЫЛКИ [Rostedt 08] Rostedt, S., «ftrace — Function Tracer», Linux documentation, https://www.kernel. org/doc/html/latest/trace/ftrace.html, 2008+. [Matz 13] Matz, M., Hubička, J., Jaeger, A., and Mitchell, M., «System V Application Binary Interface, AMD64 Architecture Processor Supplement, Draft Version 0.99.6», http://x86-64. org/documentation/abi.pdf, 2013. [Gregg 19f] Gregg, B., «Two Kernel Mysteries and the Most Technical Talk I’ve Ever Seen», http:// www.brendangregg.com/blog/2019-10-15/kernelrecipes-kernel-ftrace-internals.html, 2019. [Dronamraju 20] Dronamraju, S., «Uprobe-tracer: Uprobe-based Event Tracing», Linux documentation, https://www.kernel.org/doc/html/latest/trace/uprobetracer.html, по состоянию на 2020. [Gregg 20i] Gregg, B., «Performance analysis tools based on Linux perf_events (aka perf) and ftrace», https://github.com/brendangregg/perf-tools, последнее обновление 2020. [Hiramatsu 20] Hiramatsu, M., «Kprobe-based Event Tracing», Linux documentation, https:// www.kernel.org/doc/html/latest/trace/kprobetrace.html, по состоянию на 2020. [KernelShark 20] «KernelShark», https://www.kernelshark.org, по состоянию на 2020. [trace-cmd 20] «TRACE-CMD», https://trace-cmd.org, по состоянию на 2020. [Ts’o 20] Ts’o, T., Zefan, L., and Zanussi, T., «Event Tracing», Linux documentation, https://www. kernel.org/doc/html/latest/trace/events.html, по состоянию на 2020. [Zanussi 20] Zanussi, T., «Event Histograms», Linux documentation, https://www.kernel.org/doc/ html/latest/trace/histogram.html, по состоянию на 2020.

Глава 15

BPF

В этой главе описываются BCC и bpftrace — внешние интерфейсы трассировки для расширенного BPF. Они включают набор инструментов для анализа производительности, которые мы уже рассматривали в предыдущих главах. Технология BPF была представлена в главе 3 «Операционная система», в разделе 3.4.4 «Расширенный BPF». Вкратце, расширенный BPF — это среда выполнения в пространстве ядра, предоставляющая трассировщикам возможности программирования. Как и главы 13 «perf» и 14 «Ftrace», эта глава необязательна для чтения и предназначена для желающих изучить один или несколько системных трассировщиков более подробно. С помощью инструментов расширенного BPF можно ответить на вопросы:

yy Как выглядит гистограмма распределения задержек дискового ввода/вывода? yy Не обусловлены ли проблемы слишком большой задержкой планировщика процессора?

yy Не страдают ли приложения из-за задержек в файловой системе? yy Какие сеансы TCP были и как долго существовали? yy Какие пути в коде блокируются и на какой срок? Главное отличие BPF от других трассировщиков — возможность программирования. Он позволяет запускать по событиям написанные пользователем программы, которые могут фильтровать, сохранять и извлекать информацию, вычислять задержки, производить агрегирование в ядре и формировать обобщенные сводки, а также многое другое. В отличие от других трассировщиков, которым может потребоваться копировать все события в пространство пользователя для их обработки, BPF позволяет выполнять такую обработку более эффективно в контексте ядра. Тем самым можно создавать инструменты анализа производительности, у которых иначе был бы слишком высокий оверхед в промышленной среде.

BPF  905 В этой главе каждому интерфейсу отводится отдельный большой раздел. Вот основ­ ные разделы с подразделами:

yy 15.1: BCC • 15.1.1: Установка • 15.1.2: Покрытие инструментами • 15.1.3: Специализированные инструменты • 15.1.4: Многоцелевые инструменты • 15.1.5: Однострочные сценарии

yy 15.2: bpftrace • 15.2.1: Установка • 15.2.2: Инструменты • 15.2.3: Однострочные сценарии • 15.2.4: Программирование • 15.2.5: Справочник Различия между BCC и bpftrace очевидно вытекают из примеров их использования в предыдущих главах: BCC лучше подходит для создания сложных инструментов, а bpftrace — для специализированных программ. Некоторые инструменты могут быть реализованы с использованием обоих интерфейсов, как показано на рис. 15.1. Ядро

Внешние интерфейсы

Events

BCC biolatency.py

BPF Карты BPF Выходной буфер

Счетчики/ сводная информация

biosnoop.py

Гистограмма (текст) Выборки (текст)

(еще > 80 инструментов)

Выборки по событиям biolatency.py biosnoop.py (еще > 150 инструментов)

Рис. 15.1. Внешние интерфейсы трассировки для BPF Конкретные различия между BCC и bpftrace перечислены в табл. 15.1.

Гистограмма (текст) Выборки (текст)

906  Глава 15. BPF Таблица 15.1. Сравнение BCC и bpftrace Характеристика

BCC

bpftrace

Количество инструментов в репозитории

> 80 (bcc)

> 30 (bpftrace) > 120 (в книге «BPF Performance Tools»)

Использование

Обычно сложное: с параметрами (-h, -P PID, и т. д.) и аргументами

Обычно простое: без параметров и иногда без аргументов

Документация

Страницы справочного руководства man, файлы с примерами

Страницы справочного руководства man, файлы с примерами

Язык программирования

В пространстве пользователя: Python, Lua, C или C++. В пространстве ядра: C

bpftrace

Сложность программирования

Сложно

Просто

Типы вывода событий

Любой

Текст, JSON

Типы сводок

Любой

Счетчики, минимальное, максимальное, сумма, среднее, гистограммы с масштабом log2, линейные гистограммы; по нулевому и большему количеству ключей

Поддержка библио­ тек

Да (например, библиотеки для Python)

Нет

Средний размер программы1 (без комментариев)

228 строк

28 строк

Оба интерфейса, BCC и bpftrace, используются во многих компаниях, включая Facebook и Netflix. В Netflix они устанавливаются по умолчанию во все облачные экземпляры и используются для глубокого анализа, если в ходе мониторинга облака выявляются проблемы [Gregg 18e]:

yy BCC: стандартные инструменты используются в командной строке для анализа

дискового и сетевого ввода/вывода и выполнения процессов, когда это необходимо. Некоторые инструменты BCC автоматически используются системой мониторинга производительности, чтобы сформировать тепловые карты задержек планировщика и дискового ввода/вывода, флейм-графики распределения процессорного времени и т. д. Кроме того, пользовательский инструмент BCC всегда работает как демон (на основе tcplife(8)), регистрируя сетевые события в облачном хранилище для анализа потока даных.

На основе инструментов в официальном репозитории и в моем репозитории для книги о BPF.

1

15.1. BCC  907

yy bpftrace: пользовательские инструменты bpftrace разрабатываются по мере необходимости для изучения патологий в ядре и приложениях.

В следующих разделах подробно описываются инструменты BCC и bpftrace, а также приемы программирования на bpftrace.

15.1. BCC Коллекция компиляторов BPF (BPF Compiler Collection, BCC) — это проект с открытым исходным кодом, содержащий большой набор инструментов расширенного анализа производительности, а также платформу для их создания. BCC был создан Бренденом Бланко (Brenden Blanco). Я помогал ему в разработке и сам создал множество инструментов для трассировки. Примером инструмента на основе BCC может служить biolatency(8), показывающий распределение задержки дискового ввода/вывода в виде гистограммы с масштабом, кратным степени двойки, и с возможностью разбивки по флагам ввода/вывода: # biolatency.py -mF Tracing block device I/O... Hit Ctrl-C to end. ^C flags = Priority-Metadata-Read msecs : count 0 -> 1 : 90 flags = Write msecs : count 0 -> 1 : 24 2 -> 3 : 0 4 -> 7 : 8 flags = ReadAhead-Read msecs : count 0 -> 1 : 3031 2 -> 3 : 10 4 -> 7 : 5 8 -> 15 : 3

distribution |****************************************| distribution |****************************************| | | |************* | distribution |****************************************| | | | | | |

Здесь видно бимодальное распределение задержек в операциях записи и множество операций ввода/вывода с флагами «ReadAhead-Read». Этот инструмент использует BPF для создания гистограмм в пространстве ядра, чтобы достичь большей эффективности. Поэтому компоненту, действующему в пространстве пользователя, остается только прочитать уже готовые гистограммы (счетчики) и вывести их. Инструменты на основе BCC обычно предусматривают вывод сообщения о порядке использования (-h), у них есть свои страницы в справочном руководстве man и файлы с примерами в репозитории BCC: https://github.com/iovisor/bcc

908  Глава 15. BPF В этом разделе кратко описаны BCC и его специализированные и многоцелевые инструменты анализа производительности.

15.1.1. Установка Пакеты BCC доступны для многих дистрибутивов Linux, включая Ubuntu, Debian, RHEL, Fedora и Amazon Linux, что весьма упрощает установку. Найдите пакет с названием «bcc-tools», «bpfcc-tools» или «bcc» (в разных дистрибутивах пакет называется по-разному). Можно скомпилировать BCC из исходного кода. Актуальные инструкции по установке и сборке ищите в файле INSTALL.md в репозитории BCC [Iovisor 20b]. В INSTALL.md перечислены требования к конфигурации ядра (включая CONFIG_ BPF=y, CONFIG_BPF_SYSCALL=y, CONFIG_BPF_EVENTS=y). Для некоторых инструментов BCC требуется версия Linux не ниже 4.4, но для большинства инструментов нужна версия 4.9 или выше.

15.1.2. Покрытие инструментами Инструменты трассировки BCC изображены на рис. 15.2 (некоторые сгруппированы с использованием подстановочных знаков: например, под java* подразумеваются все инструменты, названия которых начинаются с «java»). Многие из них — специализированные, они обозначены однонаправленной стрелкой, другие — многоцелевые, они перечислены слева рядом с двунаправленной стрелкой, показывающей покрытие.

Приложения

Среда времени выполнения

Системные библиотеки Интерфейс системных вызовов Виртуальная файловая система Файловые системы

Сокеты TCP/UDP

Диспетчер томов

IP

Блочное устройство

Сетевое устройство

Планировщик Виртуальная память

Драйверы устройств

Процессоры

Другие:

Рис. 15.2. Инструменты BCC

15.1. BCC  909

15.1.3. Специализированные инструменты Многие из этих инструментов я разрабатывал, следуя все той же философии «делать что-то одно, и делать это хорошо». Согласно этой философии, выходные данные должны быть по умолчанию краткими и очевидными. Вы можете «просто запустить biolatency», не вдаваясь в изучение параметров командной строки, и получить достаточно информации для решения проблемы. Параметры обычно предназначены для тонкой настройки. Например, параметр -F команды biolatency(8) служит для разбивки результатов по флагам ввода/вывода, как было показано выше. В табл. 15.2 перечислены некоторые специализированные инструменты, а также номера разделов этой книги, где они упоминаются. Полный список см. в репозитории BCC [Iovisor 20a]. Таблица 15.2. Некоторые специализированные инструменты BCC Инструмент

Описание

Раздел

biolatency(8)

Обобщает информацию о задержках блочного (дискового) ввода/вывода и представляет ее в виде гистограммы

9.6.6

biotop(8)

Обобщает информацию о блочном вводе/выводе по процессам

9.6.8

biosnoop(8)

Трассирует блочный ввод/вывод, сообщает информацию о задержках и другие детали

9.6.7

bitesize(8)

Обобщает информацию о размерах блочного ввода/вывода по процессам и представляет ее в виде гистограммы



btrfsdist(8)

Обобщает информацию о работе файловой системы btrfs и представляет ее в виде гистограммы

8.6.13

btrfsslower(8)

Трассирует медленные операции btrfs

8.6.14

cpudist(8)

Обобщает время, проведенное потоками выполнения на процессоре и вне процессора, и представляет ее в виде гистограммы

6.6.15, 16.1.7

cpuunclaimed(8)

Показывает процессоры, которые простаивают, несмотря на спрос



criticalstat(8)

Трассирует долго выполняющиеся атомарные критические секции в коде ядра



dbslower(8)

Трассирует долго обрабатывающиеся запросы к базе данных



dbstat(8)

Обобщает информацию о задержках в обработке запросов к базе данных и представляет ее в виде гистограммы



drsnoop(8)

Трассирует события прямого освобождения памяти и показывает затронутые процессы и задержки

7.5.11

execsnoop(8)

Трассирует запуск новых процессов через обращение к системному вызову execve(2)

1.7.3, 5.5.5

ext4dist(8)

Обобщает информацию о задержках в файловой системе ext4 и представляет ее в виде гистограммы

8.6.13

ext4slower(8)

Трассирует медленные операции ext4

8.6.14

910  Глава 15. BPF Таблица 15.2 (продолжение) Инструмент

Описание

Раздел

filelife(8)

Трассирует продолжительность жизни короткоживущих файлов



gethostlatency(8)

Трассирует задержки обращений к службе разрешения имен (DNS) через вызовы библиотечных функций



hardirqs(8)

Обобщает время, затраченное на обработку аппаратных прерываний

6.6.19

killsnoop(8)

Трассирует сигналы, посылаемые через системный вызов kill(2)



klockstat(8)

Обобщает статистики использования мьютексов ядра



llcstat(8)

Обобщает статистики обращений и промахов кэша процессора по процессам



memleak(8)

Показывает операции выделения памяти, не имеющие парных им операций освобождения памяти



mysqld_qslower(8)

Трассирует долго обрабатывающиеся запросы к базе данных MySQL



nfsdist(8)

Обобщает информацию о задержках в сетевой файловой системе NFS и представляет ее в виде гистограммы

8.6.13

nfsslower(8)

Трассирует медленные операции NFS

8.6.14

offcputime(8)

Обобщает время, проведенное вне процессора, по трассировкам стека

5.5.3

offwaketime(8)

Обобщает время ожидания на блокировках по трассировкам стека



oomkill(8)

Трассирует события механизма OOM Killer, срабатывающего в случае нехватки памяти



opensnoop(8)

Трассирует системные вызовы из семейства open(2)

8.6.10

profile(8)

Профилирует потребление процессора, отбирая образцы трассировок стека через регулярные интервалы времени

5.5.2

runqlat(8)

Обобщает информацию о задержках в очереди на выполнение (планировщика) и представляет ее в виде гистограммы

6.6.16

runqlen(8)

Обобщает информацию о размерах очередей на выполнение, выбирая их через регулярные интервалы времени, и представляет ее в виде гистограммы

6.6.17

runqslower(8)

Трассирует длительные задержки в очереди на выполнение



syncsnoop(8)

Трассирует системные вызовы из семейства sync(2)



syscount(8)

Обобщает информацию о количестве обращений к системным вызовам и задержках в них

5.5.6

tcplife(8)

Трассирует сеансы TCP и обобщает продолжительность их жизни

10.6.9

tcpretrans(8)

Трассирует повторные передачи TCP и выводит некоторые подробности, включая состояние ядра

10.6.11

tcptop(8)

Обобщает информацию об объеме приема/передачи по протоколу TCP с разбивкой по процессам и хостам

10.6.10

wakeuptime(8)

Обобщает время ожидания до возобновления по трассировкам стека



15.1. BCC  911

Инструмент

Описание

Раздел

xfsdist(8)

Обобщает информацию о задержках в файловой системе xfs и представляет ее в виде гистограммы

8.6.13

xfsslower(8)

Трассирует медленные операции xfs

8.6.14

zfsdist(8)

Обобщает информацию о задержках в файловой системе zfs и представляет ее в виде гистограммы

8.6.13

zfsslower(8)

Трассирует медленные операции zfs

8.6.14

Примеры использования этих инструментов ищите в предыдущих главах, а также в файлах *_example.txt в репозитории BCC (многие из которых написал я). Информацию об инструментах, не описанных в этой книге, ищите в [Gregg 19].

15.1.4. Многоцелевые инструменты Многоцелевые инструменты перечислены слева на рис. 15.2. Каждый из них поддерживает несколько источников событий и может играть множество ролей, как и perf(1), что помимо прочего усложняет их использование. Они описываются в табл. 15.3. Таблица 15.3. Многоцелевые инструменты BCC Инструмент

Описание

Раздел

argdist(8)

Выводит значения параметров функций в виде гистограммы или счетчиков

15.1.15

funccount(8)

Подсчитывает вызовы функций в пространстве ядра или пользователя

15.1.15

funcslower(8)

Трассирует медленные функции в пространстве ядра или пользователя



funclatency(8)

Обобщает информацию о задержках в функциях и представляет ее в виде гистограммы



stackcount(8)

Подсчитывает трассировки стека, приводящие к заданному событию

15.1.15

trace(8)

Трассирует произвольные функции с фильтрами

15.1.15

Чтобы облегчить использование этих инструментов, вы можете собрать свою коллекцию однострочных сценариев. Некоторые из таких сценариев, похожие на однострочные сценарии для perf(1) и trace-cmd, приводятся в следующем разделе.

15.1.5. Однострочные сценарии Однострочные сценарии ниже выполняют трассировку в масштабе всей системы, пока не будет нажато Ctrl-C, если явно не указано иное. Они объединены в группы по используемому инструменту.

912  Глава 15. BPF

funccount(8) Подсчитать вызовы функций VFS в ядре: funcgraph 'vfs_*'

Подсчитать вызовы функций TCP в ядре: funccount 'tcp_*'

Подсчитать количество операций отправки TCP в секунду: funccount -i 1 'tcp_send*'

Показать частоту следования событий блочного ввода/вывода в секунду: funccount -i 1 't:block:*'

Показать частоту вызовов getaddrinfo() (разрешение имен) из libc в секунду: funccount -i 1 c:getaddrinfo

stackcount(8) Подсчитать трассировки стека, ведущие к блочному вводу/выводу: stackcount t:block:block_rq_insert

Подсчитать трассировки стека, ведущие к отправке IP-пакетов, с разбивкой по процессам: stackcount -P ip_output

Подсчитать трассировки стека, ведущие к блокировке потока выполнения и оставлению им процессора: stackcount t:sched:sched_switch

trace(8) Трассировать функцию ядра do_sys_open() и вывести имя файла: trace 'do_sys_open "%s", arg2'

Трассировать значение, возвращаемое функцией ядра do_sys_open(), и вывести его: trace 'r::do_sys_open "ret: %d", retval'

Трассировать функцию ядра do_nanosleep() и вывести значение параметра режима и трассировки стека в пространстве пользователя: trace -U 'do_nanosleep "mode: %d", arg2'

Трассировать запросы на аутентификацию через библиотеку pam: trace 'pam:pam_start "%s: %s", arg1, arg2'

15.1. BCC  913

argdist(8) Обобщить операции чтения VFS по возвращаемому значению (размер или код ошибки): argdist -H 'r::vfs_read()'

Обобщить вызовы read() из libc по возвращаемому значению (размер или код ошибки) для процесса с PID 1005: argdist -p 1005 -H 'r:c:read()'

Подсчитать обращения к системным вызовам с разбивкой по их идентификаторам: argdist.py -C 't:raw_syscalls:sys_enter():int:args->id'

Обобщить вызовы функции ядра tcp_sendmsg() по аргументу size и представить результаты в виде счетчиков: argdist -C 'p::tcp_sendmsg(struct sock *sk, struct msghdr *msg, size_t size):u32:size'

Обобщить вызовы функции ядра tcp_sendmsg() по аргументу size и представить результаты в виде гистограммы с масштабом, кратным степени двойки: argdist -H 'p::tcp_sendmsg(struct sock *sk, struct msghdr *msg, size_t size):u32:size'

Подсчитать вызовы write() из libc, выполненные процессом с PID 181, с разбивкой по файловым дескрипторам: argdist -p 181 -C 'p:c:write(int fd):int:fd'

Обобщить операции чтения, длившиеся дольше 100 мкс, с разбивкой по процессам: argdist -C 'r::__vfs_read():u32:$PID:$latency > 100000

15.1.6. Пример многоцелевого инструмента В качестве примера использования многоцелевого инструмента показано применение trace(8), где с его помощью трассируется функция ядра do_sys_open() и выводится второй аргумент в виде строки: # trace PID 28887 28887 28887 28887 28887 28887 28887 28887 [...]

'do_sys_open "%s", arg2' TID COMM FUNC 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open 28887 ls do_sys_open

/etc/ld.so.cache /lib/x86_64-linux-gnu/libselinux.so.1 /lib/x86_64-linux-gnu/libc.so.6 /lib/x86_64-linux-gnu/libpcre2-8.so.0 /lib/x86_64-linux-gnu/libdl.so.2 /lib/x86_64-linux-gnu/libpthread.so.0 /proc/filesystems /usr/lib/locale/locale-archive

914  Глава 15. BPF Синтаксис trace(8) основан на printf(3) и поддерживает строку формата и аргументы. Здесь arg2, второй аргумент, выводится в виде строки, потому что содержит имя файла. Оба инструмента, trace(8) и argdist(8), поддерживают синтаксис, позволяющий создавать множество гибких однострочных сценариев. bpftrace, описанный в следующих разделах, идет еще дальше, предлагая полноценный язык программирования для создания однострочных и многострочных программ.

15.1.7. Сравнение BCC и bpftrace Различия между этими двумя интерфейсами кратко изложены в начале главы. BCC лучше подходит для создания гибких и сложных инструментов, поддерживающих множество аргументов или использующих множество библиотек. bpftrace лучше подходит для однострочных или коротких сценариев без параметров или с одним целочисленным параметром. BCC позволяет разрабатывать программы для BPF на языке C, обеспечивая полный контроль. Но за большие возможности приходится платить сложностью: на разработку инструмента BCC может потребоваться в десять раз больше времени, чем на разработку инструмента на bpftrace, а его исходный код может включать в десять раз больше строк. Обычно разработка инструмента выполняется в несколько итераций, и я обнаружил, что гораздо эффективнее разработать сначала инструмент на bpftrace, а затем, если понадобится, перенести в BCC. Разница между BCC и bpftrace — это как разница между программированием на C и языке командной оболочки, где BCC можно сравнить с программированием на C (некоторые инструменты фактически пишутся на C), а bpftrace — с языком командной оболочки. В обычной работе я использую множество готовых программ на C (top(1), vmstat(1) и т. д.) и пишу одноразовые сценарии командной оболочки. Точно так же я использую множество встроенных инструментов BCC и разрабатываю одноразовые инструменты на bpftrace. В этой книге я показал достаточно примеров такого подхода: во многих главах есть примеры использования инструментов BCC, а в следующих разделах этой главы вы увидите, как можно создавать собственные инструменты на bpftrace.

15.1.8. Документация Обычно инструменты выводят сообщение о порядке использования, описывающее их синтаксис. Например: # funccount -h использование: funccount [-h] [-p PID] [-i INTERVAL] [-d DURATION] [-T] [-r] [-D] pattern Подсчитывает вызовы функций, срабатывания точек трассировки и зондов USDT позиционные аргументы:

15.2. bpftrace  915 pattern

шаблон для поиска событий

необязательные аргументы: -h, --help вывести это справочное сообщение и выйти -p PID, --pid PID трассировать только процесс с этим PID -i INTERVAL, --interval INTERVAL выводить сводку через указанный интервал в секундах -d DURATION, --duration DURATION продолжительность трассировки в секундах -T, --timestamp включить в вывод отметки времени -r, --regexp использовать регулярные выражения. По умолчанию допускается использовать только подстановочный символ "*". -D, --debug вывести программу BPF перед запуском (для отладки) примеры: ./funccount 'vfs_*'

# # ./funccount -r '^vfs.*' # # ./funccount -Ti 5 'vfs_*' # # ./funccount -d 10 'vfs_*' # ./funccount -p 185 'vfs_*' # ./funccount t:sched:sched_fork # # ./funccount -p 185 u:node:gc* # # ./funccount c:malloc # ./funccount go:os.* # ./funccount -p 185 go:os.* # # ./funccount ./test:read* # #

подсчет вызовов функций ядра с именами, начинающимися с "vfs" то же, что и выше, но с использованием регулярного выражения выводить сводку с отметками времени каждые 5 с  трассировать 10 с подсчет вызовов vfs только для PID 181 подсчет срабатываний точки трассировки sched_fork подсчет срабатываний всех USDT-зондов GC на узле для PID 185 подсчет всех вызовов malloc() из libc подсчет всех вызовов "os.*" из libgo подсчет всех вызовов "os.*" из libgo для PID 185 подсчет вызовов "read*" в двоичном файле ./test

Для каждого инструмента есть своя страница в справочном руководстве (man/ man8/funccount.8) и файл с примерами (examples/funccount_example.txt) в репозитории bcc. Файлы с примерами содержат примеры вывода с комментариями. Я также добавил следующую документацию в репозиторий BCC [Iovisor 20b]:

yy туториал для конечных пользователей: docs/tutorial.md; yy туториал для разработчиков BCC: docs/tutorial_bcc_python_developer.md; yy справочник: docs/reference_guide.md. Глава 4 в моей книге «BPF Performance Tools» полностью посвящена BCC [Gregg 19].

15.2. BPFTRACE bpftrace — это трассировщик с открытым исходным кодом, основанный на BPF и BCC. Он включает не только набор инструментов для анализа производительности,

916  Глава 15. BPF но и язык высокого уровня для разработки новых инструментов. Язык проектировался с прицелом на простоту изучения. Это аналог awk(1) в мире трассировки, и он основан на awk(1). Используя awk(1), вы пишете короткую программу для обработки входной строки, а используя bpftrace — пишете короткую программу для обработки события. bpftrace был создан Аластером Робертсоном (Alastair Robertson), и я стал одним из основных участников этого проекта. Ниже приводится однострочный сценарий, отображающий распределение размеров принимаемых сообщений TCP с разбивкой по именам процессов: # bpftrace -e 'kr:tcp_recvmsg /retval >= 0/ { @recv_bytes[comm] = hist(retval); }' Attaching 1 probe... ^C @recv_bytes[sshd]: [32, 64) [64, 128) @recv_bytes[nodejs]: [0] [1] [2, 4) [4, 8) [8, 16) [16, 32) [32, 64) [64, 128) [128, 256)

7 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| 2 |@@@@@@@@@@@@@@ | 82 135 153 12 6 32 158 155 14

|@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@ | |@ | |@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@ |

Как показывает этот вывод, распределение размеров сообщений, принимаемых процессом nodejs, имеет бимодальный характер, одна мода находится в диапазоне от 0 до 4 байт, а другая в диапазоне от 32 до 128 байт. Этот компактный однострочный сценарий на bpftrace инициализирует kretprobe в функции tcp_recvmsg(), фильтрует отрицательные возвращаемые значения (чтобы избавиться от отрицательных кодов ошибок) и заполняет объект карты BPF с именем @recv_bytes гистограммами возвращаемых значений, сохраненными с использованием имени процесса (comm) в качестве ключа. После нажатия Ctrl-C bpftrace получает сигнал (SIGINT) и завершается, автоматически распечатывая карту BPF. Этот синтаксис более подробно объясняется в следующих разделах. Но bpftrace позволяет не только писать свои однострочные сценарии, но и включает множество готовых инструментов, которые можно найти в репозитории проекта: https://github.com/iovisor/bpftrace

В этом разделе кратко описаны инструменты и язык программирования bpftrace. За основу взята информация из моей книги [Gregg 19], где bpftrace рассматривается более подробно.

15.2. bpftrace  917

15.2.1. Установка Пакеты bpftrace доступны для многих дистрибутивов Linux, включая Ubuntu, что весьма упрощает установку. Найдите пакет с именем «bpftrace». Такие пакеты есть для Ubuntu, Fedora, Gentoo, Debian, OpenSUSE и CentOS. В RHEL 8.2 bpftrace включен как технология для предварительного ознакомления. Помимо пакетов, есть образы Docker с bpftrace, двоичные файлы bpftrace, не требующие никаких зависимостей, кроме glibc, и инструкции по сборке bpftrace из исходного кода. Описание этих вариантов установки можно найти в файле INSTALL.md в репозитории bpftrace [Iovisor 20a], где перечисляются требования к конфигурации ядра (включая CONFIG_BPF=y, CONFIG_BPF_SYSCALL=y, CONFIG_BPF_EVENTS=y). Для bpftrace требуется версия Linux 4.9 или выше.

15.2.2. Инструменты Инструменты трассировки bpftrace изображены на рис. 15.3. Инструменты, которые можно найти в репозитории bpftrace, показаны черным цветом. Для моей предыдущей книги я разработал еще множество инструментов на bpftrace и опубликовал их исходный код в репозитории bpf-perf-tools-book: они показаны серым цветом [Gregg 19g].

Приложения

Среда времени выполнения

Системные библиотеки Интерфейс системных вызовов Виртуальная файловая система Файловые системы

Сокеты TCP/UDP

Диспетчер томов

IP

Блочное устройство

Сетевое устройство

Планировщик Виртуальная память

Драйверы устройств

Другие:

Репозиторий:

Рис. 15.3. Инструменты bpftrace

Процессоры

918  Глава 15. BPF

15.2.3. Однострочные сценарии Однострочные сценарии ниже выполняют трассировку в масштабе всей системы, пока не будет нажато Ctrl-C, если явно не указано иное. Они не только несут практическую пользу, но также служат мини-примерами программирования на языке bpftrace. Сценарии сгруппированы по целям. Более длинные списки однострочных сценариев на bpftrace ищите в главах, посвященным конкретным ресурсам.

Процессоры Трассировать запуск новых процессов с выводом их аргументов: bpftrace -e 'tracepoint:syscalls:sys_enter_execve { join(args->argv); }'

Подсчитать обращения к системным вызовам с разбивкой по процессам: bpftrace -e 'tracepoint:raw_syscalls:sys_enter { @[pid, comm] = count(); }'

Выбирать трассировки стека в пространстве пользователя с частотой 49 Гц для PID 189: bpftrace -e 'profile:hz:49 /pid == 189/ { @[ustack] = count(); }'

Память Подсчитать события расширения кучи по процессам (brk()) по трассировкам стека: bpftrace -e tracepoint:syscalls:sys_enter_brk { @[ustack, comm] = count(); }

Подсчитать сбои страниц по трассировкам стека в пространстве пользователя: bpftrace -e 'tracepoint:exceptions:page_fault_user { @[ustack, comm] = count(); }'

Подсчитать операции в vmscan с использованием точек трассировки: bpftrace -e 'tracepoint:vmscan:* { @[probe]++; }'

Файловые системы Трассировать операции открытия файлов обращением к openat(2) с разбивкой по именам процессов: bpftrace -e 't:syscalls:sys_enter_openat { printf("%s %s\n", comm, str(args->filename)); }'

Показать распределение вызовов read() по размерам прочитанных блоков (и кодам ошибок): bpftrace -e 'tracepoint:syscalls:sys_exit_read { @ = hist(args->ret); }'

15.2. bpftrace  919 Подсчитать вызовы VFS: bpftrace -e 'kprobe:vfs_* { @[probe] = count(); }'

Подсчитать количество прохождений через точки трассировки в ext4: bpftrace -e 'tracepoint:ext4:* { @[probe] = count(); }'

Диск Показать гистограмму с распределением объемов данных, вовлеченных в операции блочного ввода/вывода: bpftrace -e 't:block:block_rq_issue { @bytes = hist(args->bytes); }'

Подсчитать количество запросов на выполнение блочного ввода/вывода по трассировкам стека в пространстве пользователя: bpftrace -e 't:block:block_rq_issue { @[ustack] = count(); }'

Подсчитать операции блочного ввода/вывода по флагам типа: bpftrace -e 't:block:block_rq_issue { @[args->rwbs] = count(); }'

Сеть Подсчитать вызовы accept(2) с разбивкой по идентификаторам (PID) и именам процессов: bpftrace -e 't:syscalls:sys_enter_accept* { @[pid, comm] = count(); }'

Подсчитать байты, вовлеченные в операции приема/передачи с сокетами с разбивкой по идентификаторам (PID) и именам процессов, выполняющихся на процессоре: bpftrace -e 'kr:sock_sendmsg,kr:sock_recvmsg /retval > 0/ { @[pid, comm] = sum(retval); }'

Показать гистограмму распределения размеров отправленных сообщений TCP в байтах: bpftrace -e 'k:tcp_sendmsg { @send_bytes = hist(arg2); }'

Показать гистограмму распределения размеров полученных сообщений TCP в байтах: bpftrace -e 'kr:tcp_recvmsg /retval >= 0/ { @recv_bytes = hist(retval); }'

Показать гистограмму с размерами отправленных пакетов UDP: bpftrace -e 'k:udp_sendmsg { @send_bytes = hist(arg2); }'

920  Глава 15. BPF

Приложения Суммировать объем памяти в байтах, выделяемой вызовом malloc(), с разбивкой по трассировкам стека в пространстве пользователя (имеет высокий оверхед): bpftrace -e 'u:/lib/x86_64-linux-gnu/libc-2.27.so:malloc { @[ustack(5)] = sum(arg0); }'

Трассировать вызовы kill() для отправки сигналов и показать имя процесса-отправителя, идентификатор (PID) получателя и номер сигнала: bpftrace -e 't:syscalls:sys_enter_kill { printf("%s -> PID %d SIG %d\n", comm, args->pid, args->sig); }'

Ядро Подсчитать обращения к системным вызовам по их именам: bpftrace -e 'tracepoint:raw_syscalls:sys_enter { @[ksym(*(kaddr("sys_call_table") + args->id * 8))] = count(); }'

Подсчитать вызовы функций ядра с именами, начинающимися с «attach»: bpftrace -e 'kprobe:attach* { @[probe] = count(); }'

Показать гистограмму с распределением значений в третьем аргументе vfs_write() (размер): bpftrace -e 'kprobe:vfs_write { @[arg2] = count(); }'

Измерить время выполнения функции ядра vfs_read() и вывести результаты в виде гистограммы: bpftrace -e 'k:vfs_read { @ts[tid] = nsecs; } kr:vfs_read /@ts[tid]/ { @ = hist(nsecs - @ts[tid]); delete(@ts[tid]); }'

Подсчитать количество переключений контекста по трассировкам стека: bpftrace -e 't:sched:sched_switch { @[kstack, ustack, comm] = count(); }'

Выбирать трассировки стека в пространстве ядра с частотой 99 Гц, исключая поток, обслуживающий бездействие системы: bpftrace -e 'profile:hz:99 /pid/ { @[kstack] = count(); }'

15.2.4. Программирование В этом разделе — краткое руководство по использованию bpftrace и программированию на языке bpftrace. Оформление раздела заимствовано из оригинальной статьи с описанием awk [Aho 78], [Aho 88], уместившейся на шести страницах. Сам язык bpftrace создавался по образу и подобию awk и C, а также трассировщиков — DTrace и SystemTap.

15.2. bpftrace  921 Ниже пример сценария на bpftrace, который измеряет время выполнения функции ядра vfs_read() и выводит результаты в виде гистограммы. #!/usr/local/bin/bpftrace // эта программа измеряет время выполнения vfs_read() kprobe:vfs_read { @start[tid] = nsecs; } kretprobe:vfs_read /@start[tid]/ { $duration_us = (nsecs - @start[tid]) / 1000; @us = hist($duration_us); delete(@start[tid]); }

В подразделах ниже подробно описаны компоненты инструмента bpftrace, рассматривайте это как туториал. В разделе 15.2.5 «Справочник» описаны типы зондов, проверки, операторы, переменные, функции и типы карт.

1. Использование Команда bpftrace -e program

выполнит программу program и инструментирует любые события, которые в ней определены. Программа будет выполняться до нажатия Ctrl-C или пока не будет вызвана функция exit(). Программа на bpftrace, которая передается как аргумент в параметре -e, называется однострочным сценарием. Однако программу можно сохранить в файл и запустить командой: bpftrace file.bt

Расширение .bt необязательное, но помогает идентифицировать программы. Добавив в начало файла строку, определяющую интерпретатор1, #!/usr/local/bin/bpftrace

файл с программой можно сделать исполняемым (chmod a+x file.bt) и запускать как любую другую программу: ./file.bt

Некоторые предпочитают использовать #!/usr/bin/env bpftrace, чтобы система могла найти bpftrace в любом каталоге, включенном в $PATH. Но у утилиты env(1) есть проблемы, и в некоторых проектах она больше не используется.

1

922  Глава 15. BPF Команда bpftrace должна запускаться пользователем root (суперпользователем)1. В некоторых средах для запуска программы можно использовать командную оболочку root непосредственно, тогда как бывает предпочтительнее запускать привилегированные команды через sudo(1): sudo ./file.bt

2. Структура программы Программа на bpftrace определяет собой серию зондов с соответствующими действиями: зонды { действия } зонды { действия } ...

При срабатывании зондов выполняются соответствующие действия. Перед действиями можно указать необязательное выражение фильтра: зонды /фильтр/ { действия }

В этом случае действия выполняются, только если выражение фильтра, стоящее перед действиями, вернет истинное значение. Своей структурой это напоминает программу на awk(1): /шаблон/ { действия }

Программирование на awk(1) напоминает программирование на bpftrace: можно определить несколько блоков действий, и они могут выполняться в любом порядке, когда обнаруживается соответствующий шаблон или срабатывает зонд + выражение фильтра оказывается истинным.

3. Комментарии В файлах программ на bpftrace можно вставлять однострочные комментарии, начинающиеся с «//»: // это комментарий

Комментарии не выполняются. Также можно использовать многострочные комментарии в стиле C: /* * Это многострочный * комментарий. */

Этот же синтаксис можно использовать, чтобы закомментировать часть строки (например, /* комментарий */). bpftrace проверяет UID 0, но возможно, появятся проверки конкретных привилегий.

1

15.2. bpftrace  923

4. Формат определения зондов Определение зонда начинается с имени типа зонда, за которым следует иерархия идентификаторов, разделенных двоеточиями: тип:идентификатор1[:идентификатор2[...]]

Иерархия зависит от типа зонда. Взгляните на два следующих примера: kprobe:vfs_read uprobe:/bin/bash:readline

Зонды типа kprobe инструментируют вызовы функций ядра и требуют указать только один идентификатор: имя функции. Зонды типа uprobe инструментируют вызовы функций в пространстве пользователя и требуют указать путь к двоичному файлу и имя функции. Для выполнения одних и тех же действий перечислите несколько зондов через запятую. Например: зонд1,зонд2,... { действия }

Есть два специальных типа зондов, для которых не нужно указывать дополнительные идентификаторы: BEGIN и END. Они срабатывают в начале и в конце программы bpftrace (как и в awk(1)). Вот пример вывода сообщения перед началом трассировки: BEGIN { printf("Tracing. Hit Ctrl-C to end.\n"); }

Дополнительную информацию о типах зондов и их использовании ищите в разделе 15.2.5 «Справочник», в подразделе 1 «Типы зондов».

5. Подстановочные знаки в определениях зондов Некоторые типы зондов допускают использование подстановочных знаков. Определение kprobe:vfs_*

инструментирует все зонды kprobes (функции ядра) с именами, начинающимися с «vfs_». Инструментация слишком большого количества зондов может привести к избыточному оверхеду. Чтобы избежать этого, в bpftrace есть параметр настройки, ограничивающий максимальное количество зондов, которые можно активировать. Этот параметр устанавливается с помощью переменной среды BPFTRACE_MAX_ PROBES (сейчас у него значение по умолчанию 5121). Большее количество зондов замедлит запуск и завершение программы на bpftrace, так как они обрабатываются последовательно. В будущем предполагается добавить в ядро пакетную обработку зондов, после чего этот предел можно значительно увеличить или вообще убрать.

1

924  Глава 15. BPF Вы можете протестировать определения с подстановочными знаками перед использованием, запустив программу командой bpftrace -l, чтобы вывести список зондов, совпавших с определениями: # bpftrace -l 'kprobe:vfs_*' kprobe:vfs_fallocate kprobe:vfs_truncate kprobe:vfs_open kprobe:vfs_setpos kprobe:vfs_llseek […] bpftrace -l 'kprobe:vfs_*' | wc -l 56

Этому определению соответствует 56 зондов. Имя указывается в кавычках, чтобы избежать нежелательной интерпретации подстановочных знаков командной оболочкой.

6. Фильтры Фильтры — это логические выражения, определяющие возможность выполнения действий. Фильтр /pid == 123/

разрешит выполнение действия, только если встроенная переменная pid (идентификатор процесса) содержит значение 123. Если проверка не указана /pid/

то фильтр проверяет значение на неравенство нулю (выражение /pid/ эквивалентно выражению /pid != 0/). Фильтры можно комбинировать с помощью логических операторов, таких как логическое И (&&). Например, фильтр: /pid > 100 && pid < 1000/

требует, чтобы оба выражения оценивались как «истинные».

7. Действия Действие может быть одной или несколькими инструкциями, разделенными точкой с запятой: { действие первое; действие второе; действие третье }

К последней инструкции можно добавить точку с запятой. Инструкции записываются на языке bpftrace, который похож на C, и могут управлять переменными и вызывать функции bpftrace. Следующее действие: { $x = 42; printf("$x is %d", $x); }

15.2. bpftrace  925 присваивает переменной $x значение 42, а затем выводит ее вызовом функции printf(). Список других доступных функций ищите в разделе 15.2.5 «Справочник», в подразделах 4 «Функции» и 5 «Функции карты».

8. Hello, World! После этого краткого введения вы легко поймете, как написать простую программу, которая выводит текст «Hello, World!»: # bpftrace -e 'BEGIN { printf("Hello, World!\n"); }' Attaching 1 probe... Hello, World! ^C

В файле эту программу можно структурировать так: #!/usr/local/bin/bpftrace BEGIN { printf("Hello, World!\n"); }

Оформление отступов и пустые строки — условие необязательное, но это повышает удобочитаемость кода.

9. Функции Помимо функции форматированного вывода printf(), в число встроенных функций входят также:

yy exit(): выход из программы на bpftrace; yy str(char *): возвращает строку, находящуюся по адресу в указателе; yy system(формат [, аргументы ...]): запускает команду в оболочке. Действие printf("got: %llx %s\n", $x, str($x)); exit();

выведет значение переменной $x в виде шестнадцатеричного целого числа, а затем интерпретирует ее как указатель на массив символов с завершающим символом NULL (char *), выведет его как строку и завершит выполнение программы.

10. Переменные Есть три типа переменных: встроенные, временные и карты. Встроенные переменные заранее определены, доступны в программах на bpftrace по умолчанию, как правило, только для чтения и обычно являются источниками информации. К ним относятся: pid — идентификатор процесса, comm — имя процесса,

926  Глава 15. BPF nsecs — отметка времени в наносекундах и curtask — адрес структуры task_struct

текущего потока.

Временные переменные (scratch variables) могут использоваться для хранения промежуточных результатов вычислений; их имена начинаются префиксом «$». Имена и тип определяются по первой инструкции присваивания. Инструкции: $x = 1; $y = "hello"; $z = (struct task_struct *)curtask;

объявляют целочисленную переменную $x, строковую переменную $y и указатель $z на структуру task_struct. Эти переменные можно использовать только внутри блока с действиями, в котором они были созданы. При попытке сослаться на переменную, которой не было присвоено значение, bpftrace сообщит об ошибке (это поможет отловить опечатки). Переменные-карты (map variables) хранятся в хранилище карт BPF; их имена должны начинаться с префикса «@». В этих переменных можно сохранять данные для передачи между действиями. Программа: probe1 { @a = 1; } probe2 { $x = @a; }

присвоит число 1 переменной @a, когда возникнет событие probe1, а затем, когда возникнет событие probe2, присвоит значение @a переменной $x. Если сначала возникнет событие probe1, а затем probe2, переменная $x получит значение 1, в противном случае — значение 0 (значение по умолчанию для неинициализированной переменной). После имени переменной-карты можно указать ключ и использовать такие переменные на манер хеш-таблицы (ассоциативного массива). Инструкция: @start[tid] = nsecs;

часто используется в реальности: здесь она сохранит значение встроенной переменной nsecs в карте с именем @start и с ключом tid (идентификатором текущего потока). Такой прием позволяет потокам хранить отметки времени, которые не будут затерты другими потоками. @path[pid, $fd] = str(arg0);

Это пример карты с несколькими ключами, здесь роль ключей играют значение встроенной переменной pid и значение временной переменной $fd.

11. Функции карт Картам можно присваивать специальные функции. Эти функции хранят и выводят данные. Например, инструкция @x = count();

15.2. bpftrace  927 подсчитает события и при выводе вернет значение счетчика. Она использует карту, хранящую данные для каждого процессора отдельно, соответственно, @x превращается в специальный объект с типом счетчика. Следующая инструкция тоже подсчитывает события: @x++;

Но здесь используется карта с одним общим целочисленным счетчиком, а не со счетчиками для каждого процессора в отдельности. Такие глобальные целочисленные переменные пригодятся в программах, где нужно целое число, а не счетчик. Но имейте в виду, что из-за одновременных изменений такой переменной в нескольких потоках выполнения может возникнуть некоторая погрешность подсчета. Инструкция: @y = sum($x);

суммирует значения переменной $x и при выводе вернет общий результат. Инструкция: @z = hist($x);

сохранит значение $x в гистограмму и при выводе вернет интервальные счетчики и гистограмму в формате ASCII. Некоторые функции оперируют картами как единым целым. Например: print(@x);

выведет карту @x. Этот прием можно использовать, например, для вывода содержимого карты через некоторые интервалы времени. Однако он используется нечасто, потому что после завершения bpftrace все карты выводятся автоматически1. Некоторые функции карт оперируют ключами в картах. Например, инструкция: delete(@start[tid]);

удалит из карты @start пару ключ/значение с ключом tid.

12. Определение продолжительности выполнения vfs_read() Теперь, после знакомства с новыми синтаксическими конструкциями, более сложный пример будет понять проще. Вот программа vfsread.bt, которая измеряет продолжительность выполнения функции ядра vfs_read и выводит гистограмму распределения продолжительностей в микросекундах: Кроме того, вывод карты по завершении программы имеет более низкий оверхед, потому что во время выполнения карты обновляются и их вывод может замедлить процедуру обхода карты.

1

928  Глава 15. BPF #!/usr/local/bin/bpftrace // эта программа измеряет продолжительность выполнения vfs_read() kprobe:vfs_read { @start[tid] = nsecs; } kretprobe:vfs_read /@start[tid]/ { $duration_us = (nsecs - @start[tid]) / 1000; @us = hist($duration_us); delete(@start[tid]); }

Чтобы измерить продолжительность выполнения функции ядра vfs_read(), по событию входа в функцию программа сохраняет отметку времени в карте @start с ключом, соответствующим идентификатору потока, а затем, по событию выхода, вычисляет разность между текущим временем и временем входа. Фильтр при событии выхода проверяет наличие отметки времени вызова функции, иначе вычисление разности возвращало бы ошибочный результат в случаях, когда трассировка была запущена в момент выполнения vfs_read(). Пример вывода: # bpftrace vfsread.bt Attaching 2 probes... ^C @us: [0] [1] [2, 4) [4, 8) [8, 16) [16, 32) [32, 64) [64, 128) [128, 256) [256, 512) [512, 1K)

23 138 538 744 641 122 13 17 2 0 1

|@ | |@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@| |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ | |@@@@@@@@ | | | |@ | | | | | | |

Программа продолжает выполняться до нажатия Ctrl-C, затем выводит эти строки и завершается. Этой карте с гистограммой было дано имя «us», чтобы в выводе напомнить единицы измерения, потому что имя карты выводится вместе с данными. Давая картам осмысленные имена, такие как «bytes» и «latency_ns», можно аннотировать вывод и сделать его более понятным. Эту программу можно немного модифицировать. Взгляните на изменения в строке, где присваивается функция hist(): @us[pid, comm] = hist($duration_us);

15.2. bpftrace  929 Теперь для каждой пары имя/идентификатор процесса будет создана своя гистограмма. Традиционные системные инструменты — iostat(1) и vmstat(1) — имеют фиксированный вывод, который очень непросто настроить. Но с помощью bpftrace наблюдаемые метрики можно разбивать на части и дополнять метриками из других событий, пока не будут найдены ответы на интересующие вас вопросы. См. также главу 8 «Файловые системы», раздел 8.6.15 «bpftrace», подраздел «Трассировка VFS», где приводится расширенный пример, разбивающий задержку в vfs_read() по типу: файловая система, сокет и т. д.

15.2.5. Справочник Ниже кратко описаны основные аспекты, связанные с программированием на bpftrace: типы зондов, управление потоком выполнения, переменные, функции и функции карт.

1. Типы зондов В табл. 15.4 перечислены доступные типы зондов. У многих из них короткие псевдонимы, упрощающие создание коротких однострочных сценариев. Таблица 15.4. Типы зондов в bpftrace Тип

Псевдоним

Описание

tracepoint

t

Точки статической инструментации в коде ядра

usdt

U

Статически определяемые точки трассировки на уровне пользователя

kprobe

k

Динамически определяемые точки инструментации функций ядра

kretprobe

kr

Динамически определяемые точки инструментации выхода из функций ядра

kfunc

f

Динамически определяемые точки инструментации функций ядра (на основе BPF)

kretfunc

fr

Динамически определяемые точки инструментации выхода из функций ядра (на основе BPF)

uprobe

u

Динамически определяемые точки инструментации функций на уровне пользователя

uretprobe

ur

Динамически определяемые точки инструментации выхода из функций на уровне пользователя

software

s

Программные события ядра

hardware

h

Аппаратные события ядра на основе счетчиков

watchpoint

w

Точки наблюдения для инструментации событий обращения к памяти

profile

p

Выборка трассировок стека по всем процессорам с некоторой частотой

interval

i

Вывод отчета по времени (с одного процессора)

BEGIN

Пуск программы на bpftrace

END

Окончание программы на bpftrace

930  Глава 15. BPF Большинство из этих типов зондов — интерфейсы к существующим механизмам ядра. В главе 4 я объяснил, как работают эти технологии: kprobes, uprobes, точки трассировки, USDT и PMC (счетчики производительности, используются зондами типа hardware). Тип зондов kfunc/kretfunc — это новый интерфейс с низким оверхедом, использующий трамплины eBPF и BTF. Некоторые зонды, например, инструментирующие события планировщика, распределения памяти и приема/передачи сетевых пакетов, могут срабатывать очень часто. Чтобы уменьшить оверхед, старайтесь решать проблемы, по возможности используя менее частые события. Если вы не знаете, с какой частотой срабатывает зонд, измерьте ее с помощью bpftrace, например, подсчитав через kprobe вызовы vfs_read() за одну секунду: # bpftrace -e 'k:vfs_read { @ = count(); } interval:s:1 { exit(); }'

Экспериментируйте в течение короткого времени, чтобы минимизировать оверхед, если он окажется значительным. То, что мы считаем высокой или низкой частотой событий, зависит от быстродействия процессоров, их количества и запаса мощности, а также стоимости инструментации зондов. Чтобы задать приблизительный ориентир: для современных компьютеров я бы посчитал низкой частоту менее 100 000 событий в секунду.

Аргументы зондов Зонды разных типов поддерживают разные типы аргументов для передачи контекста событий. Например, в точках трассировки доступны поля из файла формата, при этом имена полей используются в структуре данных args. Следующий сценарий инструментирует точку трассировки syscalls:sys_enter_read и использует аргумент args>count для записи гистограммы из аргумента count (запрошенный размер чтения): bpftrace -e 'tracepoint:syscalls:sys_enter_read { @req_bytes = hist(args->count); }'

Узнать, какие поля поддерживаются, можно в файле формата в каталоге /sys или выполнив bpftrace с параметрами -lv: # bpftrace -lv 'tracepoint:syscalls:sys_enter_read' tracepoint:syscalls:sys_enter_read int __syscall_nr; unsigned int fd; char * buf; size_t count;

В справочнике по bpftrace «bpftrace Reference Guide», доступном в интернете, вы найдете описание каждого типа зондов и их аргументов [Iovisor 20c].

2. Управление потоком выполнения В bpftrace есть три типа проверок условий: фильтры, тернарные операторы и операторы if. Все они изменяют ход выполнения программы в соответствии с логическими выражениями, поддерживающими операторы, которые перечислены в табл. 15.5.

15.2. bpftrace  931 Таблица 15.5. Логические операторы в bpftrace Оператор

Описание

==

Равно

!=

Не равно

>

Больше, чем


=

Больше или равно

= 0 ? $x : - $x;

Операторы if Операторы if имеют следующий синтаксис: if (проверка) { инструкции_для_пройденной_проверки } if (test) { инструкции_для_пройденной_проверки } else { инструкции_для_ непройденной_проверки }

Один из вариантов использования — выполнение разных действий для IPv4 и IPv6. Например (для простоты игнорируются семейства, отличные от IPv4 и IPv6):

932  Глава 15. BPF if ($inet_family == $AF_INET) { // IPv4 ... } else { // предполагается IPv6 ... }

Начиная с версии bpftrace v0.10.0, поддерживается также оператор else if1.

Циклы bpftrace поддерживает циклы, доступные для развертывания с помощью unroll(). В ядре Linux версии 5.3 и выше поддерживаются также циклы while()2: while (проверка) { инструкции }

Они основаны на поддержке циклов в BPF, добавленной в Linux 5.3.

Операторы В предыдущем разделе были перечислены логические операторы, которые можно использовать в условных выражениях (проверках). Однако bpftrace поддерживает еще ряд операторов, перечисленных в табл. 15.6. Таблица 15.6. Операторы в bpftrace Оператор

Описание

=

Присваивание

+, -, *, /

Сложение, вычитание, умножение, деление (только целочисленное)

++, --

Автоинкремент, автодекремент

&, |, ^

Битовые И, ИЛИ и ИСКЛЮЧАЮЩЕЕ ИЛИ

!

Логическое НЕ

Логический сдвиг влево, логический сдвиг вправо

+=, -=, *=, /=, %=, &=, ^=, =

Составные операторы присваивания

Эти операторы действуют так же, как и аналогичные операторы в C.

Спасибо Даниэлю Сю (Daniel Xu; PR#1211).

1

Спасибо Базу Смиту (Bas Smit) за добавление логики в bpftrace (PR#1066).

2

15.2. bpftrace  933

3. Переменные Встроенные переменные, поддерживаемые в bpftrace, обычно доступны только для чтения. В табл. 15.7 перечислены некоторые особенно важные встроенные переменные. Таблица 15.7. Некоторые встроенные переменные в bpftrace Встроенная переменная

Тип

Описание

pid

Целое число

Идентификатор процесса (группы потоков ядра)

tid

Целое число

Идентификатор потока выполнения (процесса ядра)

uid

Целое число

Идентификатор пользователя

username

Строка

Имя пользователя

nsecs

Целое число

Отметка времени в наносекундах

elapsed

Целое число

Время в наносекундах, прошедшее с момента инициализации bpftrace

cpu

Целое число

Идентификатор процессора

comm

Строка

Имя процесса

kstack

Строка

Трассировка стека ядра

ustack

Строка

Трассировка стека в пространстве пользователя

arg0, ..., argN

Целое число

Аргументы в зондах некоторых типов

args

Целое число

Аргументы в зондах некоторых типов

sarg0, ..., sargN

Целое число

Аргументы на основе стека в зондах некоторых типов

retval

Целое число

Возвращаемое значение в зондах некоторых типов

func

Строка

Имя трассируемой функции

probe

Строка

Полное имя текущего зонда

curstack

Структура/целое число

Структура task_struct в ядре (либо как task_struct, либо как 64-битное целое без знака, в зависимости от доступности информации о типе)

cgroup

Целое число

Идентификатор cgroup v2 по умолчанию текущего процесса (для сравнения с cgroupid())

$1, ..., $N

int, char *

Позиционные параметры программы на bpftrace

Сейчас все целочисленные переменные имеют тип uint64. Все перечисленные переменные относятся к текущему выполняющемуся потоку, зонду, функции и процессору.

934  Глава 15. BPF Выше в этой главе были показаны встроенные переменные: retval, comm, tid и nsecs. Полный и актуальный список встроенных переменных ищите в справочнике по bpftrace «bpftrace Reference Guide», доступном в интернете [Iovisor 20c].

4. Функции В табл. 15.8 перечислены некоторые встроенные функции для решения разных задач. Некоторые из них уже использовались в примерах, например, printf(). Таблица 15.8. Некоторые встроенные функции в bpftrace Функция

Описание

printf(char *fmt [, ...])

Форматированный вывод

time(char *fmt)

Форматированный вывод времени

join(char *arr[])

Вывод массива строк через пробел

str(char *s [, int len])

Возвращает строку, находящуюся по указателю s, позволяет ограничить длину строки

buf(void *d [, int length])

Возвращает блок данных, доступный по указателю d, в виде строки шестнадцатеричных чисел

strncmp(char *s1, char *s2, int length)

Сравнивает до length первых символов в строках

sizeof(expression)

Возвращает размер выражения expression или типа данных

kstack([int limit])

Возвращает стек ядра с глубиной до limit фреймов

ustack([int limit])

Возвращает стек в пространстве пользователя с глубиной до limit фреймов

ksym(void *p)

По заданному адресу определяет и возвращает символ ядра в виде строки

usym(void *p)

По заданному адресу определяет и возвращает символ в пространстве пользователя в виде строки

kaddr(char *name)

По заданному символу ядра определяет его адрес

uaddr(char *name)

По заданному символу в пространстве пользователя определяет его адрес

reg(char *name)

Возвращает значение, хранящееся в указанном регистре

ntop([int af,] int addr)

Возвращает строковое представление адреса IPv4/IPv6

cgroupid(char *path)

Возвращает идентификатор контрольной группы (cgroup) по заданному пути (/sys/fs/cgroup/...)

system(char *fmt [, ...])

Выполняет команду оболочки

cat(char *filename)

Выводит содержимое файла

signal(char[] sig | u32 sig)

Посылает сигнал текущей задаче (например, SIGTERM)

15.2. bpftrace  935

Функция

Описание

override(u64 rc)

Переопределяет возвращаемое значение в kprobe1

exit()

Завершает программу на bpftrace

Некоторые из этих функций действуют асинхронно: ядро ставит событие в очередь и через короткое время обрабатывает его в пространстве пользователя. К числу асинхронных относятся функции: printf(), time(), cat(), join() и system(). Функции kstack(), ustack(), ksym() и usym() действуют синхронно, но трансляцию символов выполняют асинхронно. Ниже показан пример применения функций printf() и  str() для вывода имени файла, переданного системному вызову openat(2): # bpftrace -e 't:syscalls:sys_enter_open { printf("%s %s\n", comm, str(args->filename)); }' Attaching 1 probe... top /etc/ld.so.cache top /lib/x86_64-linux-gnu/libprocps.so.7 top /lib/x86_64-linux-gnu/libtinfo.so.6 top /lib/x86_64-linux-gnu/libc.so.6 [...]

Полный и актуальный список функций ищите в справочнике по bpftrace «bpftrace Reference Guide», доступном в интернете [Iovisor 20c].

5. Функции карт Карты — это специальные объекты в BPF, предназначенные для хранения хештаблиц, которые можно использовать для разных целей, например для хранения пар ключ/значение или статистических сводок. Для работы с картами bpftrace предоставляет ряд встроенных функций. Наиболее важные из них перечислены в табл. 15.9. Таблица 15.9. Некоторые функции карт в bpftrace

1

Функция

Описание

count()

Подсчитывает вхождения

sum(int n)

Суммирует указанное значение

avg(int n)

Усредняет указанное значение

min(int n)

Сохраняет минимальное значение

max(int n)

Сохраняет максимальное значение

ВНИМАНИЕ: используйте эту функцию, только если точно знаете, что делаете: небольшая ошибка может вызвать крах ядра.

936  Глава 15. BPF Таблица 15.9 (окончание) Функция

Описание

stats(int n)

Возвращает количество, среднее и общую сумму

hist(int n)

Выводит гистограмму значений с шагом, равным степени двойки

lhist(int n, int min, int max, int step)

Выводит линейную гистограмму значений

delete(@m[key])

Удаляет пару ключ/значение из карты

print(@m [, top [, div]])

Выводит содержимое карты с необязательными ограничением количества выводимых значений и делителем

clear(@m)

Удаляет все пары ключ/значение из карты

zero(@m)

Обнуляет все значения в карте

Некоторые из этих функций действуют асинхронно: ядро ставит событие в очередь и через короткое время обрабатывает его в пространстве пользователя. К числу асинхронных относятся функции: print(), clear() и zero(). Не забывайте о свойственных им задержках, когда будете писать программы. Ниже показан пример применения функции lhist() для создания линейной гистограммы распределения размеров блоков, запрашиваемых у системного вызова read(2), с разбивкой по именам процессов и с размером шага, равным единице, чтобы каждый дескриптор файла можно было видеть отдельно: # bpftrace -e 'tracepoint:syscalls:sys_enter_read { @fd[comm] = lhist(args->fd, 0, 100, 1); }' Attaching 1 probe... ^C [...] @fd[sshd]: [4, 5) 22 | | [5, 6) 0 | | [6, 7) 0 | | [7, 8) 0 | | [8, 9) 0 | | [9, 10) 0 | | [10, 11) 0 | | [11, 12) 0 | | [12, 13) 7760 |@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@|

Как показывает вывод, в этой системе процессы sshd обычно читают данные из файлового дескриптора 12. В выходных данных используется обозначение интервалов, где «[» означает >=, а «)» означает < (такие интервалы также называют полуоткрытыми — закрытыми слева и открытыми справа). Полный и актуальный список функций карт ищите в справочнике по bpftrace «bpftrace Reference Guide», доступном в интернете [Iovisor 20c].

15.3. Ссылки  937

15.2.6. Документация В предыдущих главах этой книги можно найти дополнительную информацию о bpftrace:

yy yy yy yy yy yy

глава 5 «Приложения», раздел 5.5.7; глава 6 «Процессоры», раздел 6.6.20; глава 7 «Память», раздел 7.5.13; глава 8 «Файловые системы», раздел 8.6.15; глава 9 «Диски», раздел 9.6.11; глава 10 «Сеть», раздел 10.6.12.

Примеры использования bpftrace можно найти в главе 4 «Инструменты наблюдения» и в главе 11 «Облачные вычисления». В репозитории bpftrace доступна следующая документация:

yy справочник: docs/reference_guide.md [Iovisor 20c]; yy туториал: docs/tutorial_one_liners.md [Iovisor 20d]. Подробную информацию о bpftrace ищите в моей книге «BPF Performance Tools» [Gregg 19], где в главе 5 «bpftrace» на множестве примеров показаны возможности языка программирования bpftrace, а в последующих главах представлено еще больше программ на bpftrace для разных целей. Обратите внимание, что некоторые возможности bpftrace, описанные в [Gregg 19] как «запланированные», уже были добавлены в bpftrace и включены в эту главу. Это циклы while(), операторы else-if, signal(), override() и события точек наблюдения. В числе других дополнений, появившихся в bpftrace, тип зондов kfunc и функции buf() и sizeof(). Ознакомьтесь с примечаниями к релизу в репозитории bpftrace, где перечислены будущие дополнения, хотя их не так много: bpftrace уже обладает достаточно широкими возможностями и на нем написано более 120 инструментов.

15.3. ССЫЛКИ [Aho 78] Aho, A. V., Kernighan, B. W., and Weinberger, P. J., «Awk: A Pattern Scanning and Processing Language (Second Edition)», Unix 7th Edition man pages, 1978. Доступно в Интернете по адресу: http://plan9.bell-labs.com/7thEdMan/index.html. [Aho 88] Aho, A. V., Kernighan, B. W., and Weinberger, P. J., «The AWK Programming Language», Addison Wesley, 1988. [Gregg 18e] Gregg, B., «YOW! 2018 Cloud Performance Root Cause Analysis at Netflix», http:// www.brendangregg.com/blog/2019-04-26/yow2018-cloud-performance-netflix.html, 2018.

938  Глава 15. BPF [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»1, Addison-Wesley, 2019. [Gregg 19g] Gregg, B., «BPF Performance Tools (book): Tools», http://www.brendangregg.com/ bpf-performance-tools-book.html#tools, 2019. [Iovisor 20a] «bpftrace: High-level Tracing Language for Linux eBPF», https://github.com/iovisor/ bpftrace, последнее обновление 2020. [Iovisor 20b] «BCC - Tools for BPF-based Linux IO Analysis, Networking, Monitoring, and More», https://github.com/iovisor/bcc, последнее обновление 2020. [Iovisor 20c] «bpftrace Reference Guide», https://github.com/iovisor/bpftrace/blob/master/docs/ reference_guide.md, последнее обновление 2020. [Iovisor 20d] Gregg, B., et al., «The bpftrace One-Liner Tutorial», https://github.com/iovisor/bpftrace/ blob/master/docs/tutorial_one_liners.md, последнее обновление 2020.

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

1

Глава 16

ПРИМЕР ИЗ ПРАКТИКИ

В этой главе описан пример исследования производительности системы. Я расскажу о реальной проблеме — от первоначального отчета до окончательного решения. Эта проблема возникла в производственной облачной среде, и я выбрал ее как наглядный пример анализа производительности системы. Я не буду вводить здесь никаких новых технических понятий. Рассказанная история покажет вам, как можно применять инструменты и методологии на практике, в реальной рабочей среде. Это будет особенно полезно новичкам, которым по жизни еще предстоит столкнуться с проблемами производительности. Они смогут узнать, как такие проблемы решают эксперты, о чем эти эксперты думают во время анализа и почему. Моя история не документирование наилучшего подхода, а скорее объяснение, почему тот или иной подход был выбран.

16.1. НЕОБЪЯСНИМЫЙ ВЫИГРЫШ В Netflix протестировали микросервис на новой платформе на основе контейнеров и обнаружили, что задержка обработки запроса снизилась в 3–4 раза. У контейнерной платформы множество преимуществ, но такой большой выигрыш стал полной неожиданностью! Звучало слишком хорошо, чтобы быть правдой, и меня попросили изучить ситуацию и объяснить, как это произошло. Для анализа я использовал различные инструменты, в том числе основанные на счетчиках, статической конфигурации, PMC, программных событиях и трассировке. Все эти типы инструментов сыграли свою роль и дали подсказки, сложившиеся в общую непротиворечивую картину. Поскольку это исследование делалось для широкой аудитории, то стало вступительным материалом к моему докладу на USENIX LISA 2019 о производительности систем [Gregg 19h]. Я включил его в эту книгу для примера.

16.1.1. Постановка задачи Поговорив с командой обслуживания, я выяснил подробности о микросервисе: это было приложение на Java, которое работает на экземплярах виртуальных машин

940  Глава 16. Пример из практики в облаке AWS EC2 и определяет рекомендации для клиентов. Микросервис состоит из двух компонентов, и один из них тестировался на новой контейнерной платформе Netflix под названием Titus, также работающей на AWS EC2. Задержка запроса этого компонента составляла 3–4 с на экземплярах виртуальных машин, а на контейнерах — 1 с: в 3–4 раза быстрее! Мне нужно было объяснить эту разницу в производительности. Если такой прирост производительности был вызван простым перемещением микросервиса в контейнер, то можно было бы рассчитывать на выигрыш в 3–4 раза при перемещении других микросервисов. Если это связано с каким-то другим фактором, то стоило бы понять, что это за фактор и будет ли он проявляться постоянно. Возможно, его влияние может проявиться где-то еще и в большей степени. Первое, что пришло мне в голову: проявились преимущества изолированного запуска одного компонента рабочей нагрузки, и теперь микросервис может использовать все кэши процессора, не конкурируя с другими компонентами. Из-за этого увеличивается коэффициент попаданий в кэш и, следовательно, производительность. Другое предположение — bursting, то есть увеличение доступной вычислительной мощности. Это когда контейнер может использовать простаивающие вычислительные ресурсы из других контейнеров.

16.1.2. Стратегия анализа Трафик обрабатывается балансировщиком нагрузки (AWS ELB), поэтому можно разделить трафик между виртуальной машиной и контейнерами, чтобы проанализировать одновременно обе системы. Это идеальная ситуация для сравнительного анализа: можно одновременно запустить одну и ту же команду анализа в обеих системах (что гарантирует одинаковый трафик и нагрузку) и сразу сравнить результат. В этом случае у меня был доступ не только к контейнеру, но и к хосту контейнера. Это позволило использовать любые инструменты анализа и дать этим инструментам разрешение на доступ к любому системному вызову. Если бы мне был доступен только контейнер, то анализ стал бы более трудоемким из-за ограниченности источников наблюдаемости и привилегий и потребовал бы проанализировать гораздо больший объем косвенных метрик из-за недоступности прямых показателей. Некоторые проблемы производительности нецелесообразно анализировать только из контейнера (см. главу 11 «Облачные вычисления»). Что касается методологий, я планировал начать с 60-секундного чек-листа (глава 1 «Введение», раздел 1.10.1 «Анализ производительности Linux за 60 секунд») и метода USE (глава 2 «Методологии», раздел 2.5.9 «Метод USE») и уже по их результатам сделать детальный анализ (раздел 2.5.12 «Анализ с увеличением детализации») и использовать другие методологии. Я включил команды, которые выполнял, и их вывод в следующие разделы. В них я использовал запрос «serverA#» для обозначения экземпляра виртуальной машины и «serverB#» — для хоста контейнера.

16.1. Необъяснимый выигрыш  941

16.1.3. Статистики Для начала я запустил uptime(1), чтобы проверить среднюю величину нагрузки. В обеих системах serverA# uptime 22:07:23 up 15 days, 5:01, 1 user, load average: 85.09, 89.25, 91.26 serverB# uptime 22:06:24 up 91 days, 23:52, 1 user, load average: 17.94, 16.92, 16.62

нагрузка была примерно стабильной: она немного уменьшилась для экземпляра виртуальной машины (с 91,26 до 85,09) и немного увеличилась для контейнера (с 16,62 до 17,94). Я проверил тенденции, чтобы увидеть, увеличивается ли проблема, уменьшается или остается постоянной: это особенно важно в облачных средах, которые могут автоматически переносить нагрузку с неработоспособного экземпляра. Я неоднократно заходил в проблемный экземпляр и обнаруживал, что активность невысока и средняя нагрузка за минуту приближается к нулю. Средние значения также показали, что виртуальная машина нагружена гораздо больше, чем хост контейнера (85,09 против 17,94), но чтобы понять, что это означает, нужно воспользоваться другими инструментами. Средняя высокая нагрузка обычно указывает на нагрузку на процессор, но иногда может быть обусловлена интенсивным вводом/выводом (см. главу 6 «Процессоры», раздел 6.6.1 «uptime»). Чтобы изучить нагрузку на процессор, я обратился к mpstat(1), начав со средних значений для всей системы. На виртуальной машине: serverA# mpstat 10 Linux 4.4.0-130-generic (...) 07/18/2019 10:07:55 10:08:05 10:08:15 10:08:25 10:08:35 10:08:45 ^C Average:

PM PM PM PM PM PM

CPU all all all all all

%usr 89.72 88.60 89.71 89.55 89.87

%nice 0.00 0.00 0.00 0.00 0.00

all

89.49

0.00

_x86_64_ (48 CPU)

%sys %iowait 7.84 0.00 9.18 0.00 9.01 0.00 8.11 0.00 8.21 0.00

%irq %soft %steal %guest %gnice 0.00 0.04 0.00 0.00 0.00 0.00 0.05 0.00 0.00 0.00 0.00 0.05 0.00 0.00 0.00 0.00 0.06 0.00 0.00 0.00 0.00 0.05 0.00 0.00 0.00

8.47

0.00

0.00

0.05

0.00

0.00

%idle 2.40 2.17 1.23 2.28 1.86

0.00

1.99

serverB# mpstat 10 Linux 4.19.26 (...) 07/18/2019 _x86_64_ (64 CPU) 09:56:11 PM CPU %usr %nice %sys %iowait %irq %soft %steal %guest %gnice 09:56:21 PM all 23.21 0.01 0.32 0.00 0.00 0.10 0.00 0.00 0.00 09:56:31 PM all 20.21 0.00 0.38 0.00 0.00 0.08 0.00 0.00 0.00 09:56:41 PM all 21.58 0.00 0.39 0.00 0.00 0.10 0.00 0.00 0.00 09:56:51 PM all 21.57 0.01 0.39 0.02 0.00 0.09 0.00 0.00 0.00 09:57:01 PM all 20.93 0.00 0.35 0.00 0.00 0.09 0.00 0.00 0.00 ^C Average: all 21.50 0.00 0.36 0.00 0.00 0.09 0.00 0.00 0.00

%idle 76.37 79.33 77.92 77.93 78.63

И в контейнере:

78.04

942  Глава 16. Пример из практики mpstat(1) выводит количество процессоров в первой строке. Вывод показывает, что виртуальная машина имеет 48 процессоров, а хост контейнера — 64. Это помогло мне интерпретировать средние значения нагрузки: если бы они были обусловлены нехваткой вычислительных ресурсов, то это означало бы, что экземпляр VM работает в условиях насыщения процессоров, потому что средняя нагрузка примерно вдвое превышает количество процессоров, в то время как хост контейнера используется недостаточно интенсивно. Метрики, полученные с помощью mpstat(1), подтвердили эту гипотезу: время бездействия на виртуальной машине составило около 2 %, тогда как на хосте контейнера — около 78 %. Изучив другие статистики mpstat(1), я обнаружил новые зацепки:

yy Потребление процессора (%usr + %sys + ...) виртуальной машиной составило 98 % по сравнению с 22 % у контейнера. Процессоры имеют по два гиперпотока на каждое ядро, поэтому превышение уровня потребления 50 % обычно означает конкуренцию между гиперпотоками и снижение производительности. Виртуальная машина уже перешагнула этот рубеж, тогда как узел контейнера мог продолжать получать выгоду от нагрузки, с которой прекрасно справляется один гиперпоток в каждом ядре.

yy Время выполнения в пространстве системы (%sys) на виртуальной машине на-

много выше: около 8 % против 0,38 %. Если виртуальная машина выполняется на нагруженном процессоре, это дополнительное время %sys может включать время, расходуемое на переключение контекста ядра. Подтвердить эту догадку можно трассировкой или профилированием ядра.

Я прошелся по другим пунктам 60-секундного чек-листа. vmstat(8) показала длину очереди выполнения, аналогичную средней нагрузке, подтверждая, что средняя нагрузка зависит от CPU. iostat(1) показала небольшое количество операций дискового ввода/вывода, а sar(1) — небольшое количество операций сетевого ввода/ вывода. (Результаты этих измерений я здесь не привожу.) Это подтвердило, что виртуальная машина работает с насыщением процессора, в результате чего потоки выполнения вынуждены подолгу ждать своей очереди, в то время как хост контейнера — нет. top(1) на хосте контейнера также показала, что работает только один контейнер. Эти команды дали мне статистику для метода USE, который также выявил проблему высокой нагрузки на процессор. Решил ли я задачу? Я выяснил, что средняя нагрузка на виртуальную машину в системе с 48 CPU составляет около 85 и что эта средняя нагрузка обусловлена большой нагрузкой на процессор. Это означает, что примерно 77 % времени (85/48 – 1) потоки ожидают своей очереди на выполнение и избавление от этих простоев привело бы к ускорению примерно в 4 раза (1/(1 – 0,77)). Хотя эта величина соответствовала проблеме, я не мог объяснить, почему средняя нагрузка оказалась выше: требовался дополнительный анализ.

16.1. Необъяснимый выигрыш  943

16.1.4. Конфигурация Зная, что проблема в процессоре, я проверил конфигурацию и настроенные пределы (статическая настройка производительности: разделы 2.5.17 и 6.5.7). Для виртуальных машин и контейнеров использовались разные процессоры. Вот содержимое /proc/cpuinfo на хосте виртуальной машины: serverA# cat /proc/cpuinfo processor : 47 vendor_id : GenuineIntel cpu family : 6 model : 85 model name : Intel(R) Xeon(R) Platinum 8175M CPU @ 2.50GHz stepping : 4 microcode : 0x200005e cpu MHz : 2499.998 cache size : 33792 KB physical id : 0 siblings : 48 core id : 23 cpu cores : 24 apicid : 47 initial apicid : 47 fpu : yes fpu_exception : yes cpuid level : 13 wp : yes flags : fpu vme de pse tsc msr pae mce cx8 apic sep mtrr pge mca cmov pat pse36 clflush mmx fxsr sse sse2 ss ht syscall nx pdpe1gb rdtscp lm constant_tsc arch_perfmon rep_good nopl xtopology nonstop_tsc aperfmperf eagerfpu pni pclmulqdq monitor ssse3 fma cx16 pcid sse4_1 sse4_2 x2apic movbe popcnt tsc_deadline_timer aes xsave avx f16c rdrand hypervisor lahf_lm abm 3dnowprefetch invpcid_single kaiser fsgsbase tsc_adjust bmi1 hle avx2 smep bmi2 erms invpcid rtm mpx avx512f rdseed adx smap clflushopt clwb avx512cd xsaveopt xsavec xgetbv1 ida arat bugs : cpu_meltdown spectre_v1 spectre_v2 spec_store_bypass bogomips : 4999.99 clflush size : 64 cache_alignment : 64 address sizes : 46 bits physical, 48 bits virtual power management:

и на хосте контейнера: serverB# cat /proc/cpuinfo processor : 63 vendor_id : GenuineIntel cpu family : 6 model : 79 model name : Intel(R) Xeon(R) CPU E5-2686 v4 @ 2.30GHz stepping : 1 microcode : 0xb000033

944  Глава 16. Пример из практики cpu MHz : 1200.601 cache size : 46080 KB physical id : 1 siblings : 32 core id : 15 cpu cores : 16 apicid : 95 initial apicid : 95 fpu : yes fpu_exception : yes cpuid level : 13 wp : yes flags : fpu vme de pse tsc msr pae mce cx8 apic sep mtrr pge mca cmov pat pse36 clflush mmx fxsr sse sse2 ht syscall nx pdpe1gb rdtscp lm constant_tsc arch_ perfmon rep_good nopl xtopology nonstop_tsc cpuid aperfmperf pni pclmulqdq monitor est ssse3 fma cx16 pcid sse4_1 sse4_2 x2apic movbe popcnt tsc_deadline_timer aes xsave avx f16c rdrand hypervisor lahf_lm abm 3dnowprefetch cpuid_fault invpcid_single pti fsgsbase bmi1 hle avx2 smep bmi2 erms invpcid rtm rdseed adx xsaveopt ida bugs : cpu_meltdown spectre_v1 spectre_v2 spec_store_bypass l1tf bogomips : 4662.22 clflush size : 64 cache_alignment : 64 address sizes : 46 bits physical, 48 bits virtual power management:

Процессоры на хосте контейнера имели более низкую базовую тактовую частоту (2,30 против 2,50 ГГц), зато объем кэша последнего уровня у них был гораздо больше (45 против 33 Мбайт). В зависимости от рабочей нагрузки размер кэша может существенно сказываться на производительности. Дальше я решил исследовать счетчики PMC.

16.1.5. Счетчики PMC Счетчики мониторинга производительности (PMC) могут объяснить, куда расходуются такты процессора, и доступны в некоторых экземплярах AWS EC2. Я опубликовал набор инструментов для анализа PMC в облаке [Gregg 20e], куда входит pmcarch(8) (раздел 6.6.11 «pmcarch»). pmcarch(8) показывает «архитектурный набор» счетчиков PMC от Intel, который является базовым общедоступным набором. В виртуальной машине: serverA# ./pmcarch -p 4093 10 K_CYCLES K_INSTR IPC BR_RETIRED 982412660 575706336 0.59 126424862460 999621309 555043627 0.56 120449284756 991146940 558145849 0.56 126350181501 996314688 562276830 0.56 122215605985 979890037 560268707 0.57 125609807909 [...]

BR_MISPRED 2416880487 2317302514 2530383860 2348638980 2386085660

BMR% 1.91 1.92 2.00 1.92 1.90

LLCREF 15724006692 15378257714 15965082710 15558286345 15828820588

LLCMISS 10872315070 11121882510 11464682655 10835594199 11038597030

LLC% 30.86 27.68 28.19 30.35 30.26

16.1. Необъяснимый выигрыш  945 В экземпляре с контейнером: serverB# ./pmcarch -p 1928219 10 K_CYCLES K_INSTR IPC BR_RETIRED 147523816 222396364 1.51 46053921119 156634810 229801807 1.47 48236123575 152783226 237001219 1.55 49344315621 140787179 213570329 1.52 44518363978 136822760 219706637 1.61 45129020910 [...]

BR_MISPRED 641813770 653064504 692819230 631588112 651436401

BMR% 1.39 1.35 1.40 1.42 1.44

LLCREF 8880477235 9186609260 9314992450 8675999448 8689831639

LLCMISS 968809014 1183858023 879494418 712318917 617678747

LLC% 89.09 87.11 90.56 91.79 92.89

Судя по результатам, количество инструкций на такт (IPC) составило около 0,57 для виртуальной машины и 1,52 для контейнера: разница в 2,6 раза. Одна из причин более низкого значения IPC — возможная конкуренция между гиперпотоками, так как хост виртуальной машины работал с нагрузкой на процессор выше 50 %. В последнем столбце видна еще одна причина: коэффициент попаданий в кэш последнего уровня (LLC) составлял всего 30 % для виртуальной машины и примерно 90 % для контейнера. Из-за этого процессор на виртуальной машине может часто останавливаться, ожидая доступа к основной памяти, что снижает IPC и пропускную способность инструкций (производительность). Более низкий коэффициент попаданий в LLC на виртуальной машине может быть вызван как минимум тремя факторами:

yy меньший размер LLC (33 против 45 Мбайт); yy выполнение полной рабочей нагрузки вместо одного компонента (как указано

в формулировке проблемы); компонент, выполняющийся в одиночку, вероятно, будет иметь лучший коэффициент попаданий в кэш, так как в нем меньше инструкций и данных;

yy высокая нагрузка на процессор вызывает большее количество переключений контекста и переключений между путями в коде (включая пользователя и ядро), увеличивая нагрузку на кэш.

Последний фактор можно изучить с помощью инструментов трассировки.

16.1.6. Программные события Исследование переключений контекста я начал с команды perf(1), чтобы подсчитать частоту переключения контекста в системе в целом. Для этого я использовал программное событие, похожее на аппаратное (PMC), но реализованное программно (см. главу 4 «Инструменты наблюдения», рис. 4.5 и главу 13 «perf», раздел 13.5 «Программные события»). В виртуальной машине: serverA# perf stat -e cs -a -I 1000 # time counts unit events 1.000411740 2,063,105 cs 2.000977435 2,065,354 cs

946  Глава 16. Пример из практики

^C

3.001537756 4.002028407 5.002538455 6.003114251 7.003665091 8.004093520 9.004533912 10.005106500 10.513632795

1,527,297 515,509 2,447,126 2,021,182 2,329,157 1,740,898 1,235,641 2,340,443 1,496,555

cs cs cs cs cs cs cs cs

cs

Как показывает этот вывод, частота переключений контекста составила около двух миллионов в секунду. Затем я запустил ту же команду на хосте контейнера, добавив фильтрацию по PID приложения контейнера, чтобы исключить другие возможные контейнеры (такую же фильтрацию я пробовал включить на виртуальной машине, и это не сильно изменило предыдущие результаты1): serverB# perf stat -e cs -p 1928219 -I 1000 # time counts unit events 1.001931945 1,172 cs 2.002664012 1,370 cs 3.003441563 1,034 cs 4.004140394 1,207 cs 5.004947675 1,053 cs 6.005605844 955 cs 7.006311221 619 cs 8.007082057 1,050 cs 9.007716475 1,215 cs 10.008415042 1,373 cs ^C 10.584617028 894 cs

Этот вывод показал скорость всего около тысячи переключений контекста в секунду. Высокая частота переключений контекста может оказывать большое давление на кэши процессора, которые переключаются между различными путями в коде, включая код ядра, управляющий переключением контекста и, возможно, различными процессами2. Чтобы дальше исследовать переключения контекста, я использовал инструменты трассировки.

16.1.7. Трассировка Есть несколько инструментов трассировки на основе BPF, с помощью которых можно проанализировать потребление процессора и переключения контекста, в том числе из BCC: cpudist(8), cpuwalk(8), runqlen(8), runqlat(8), runqslower(8), cpuunclaimed(8) ) и т. д. (см. рис. 15.1).

Тогда почему я не привел вывод с результатами для виртуальной машины, полученными при включенной фильтрации по PID? У меня его нет.

1

2

Некоторые конфигурации процессора и ядра могут также предусматривать очистку кэша L1 при переключении контекста.

16.1. Необъяснимый выигрыш  947 cpudist(8) показывает продолжительность выполнения потоков на процессоре. В виртуальной машине: serverA# cpudist -p 4093 10 1 Tracing on-CPU time... Hit Ctrl-C to end. usecs : count distribution 0 -> 1 : 3618650 |****************************************| 2 -> 3 : 2704935 |***************************** | 4 -> 7 : 421179 |**** | 8 -> 15 : 99416 |* | 16 -> 31 : 16951 | | 32 -> 63 : 6355 | | 64 -> 127 : 3586 | | 128 -> 255 : 3400 | | 256 -> 511 : 4004 | | 512 -> 1023 : 4445 | | 1024 -> 2047 : 8173 | | 2048 -> 4095 : 9165 | | 4096 -> 8191 : 7194 | | 8192 -> 16383 : 11954 | | 16384 -> 32767 : 1426 | | 32768 -> 65535 : 967 | | 65536 -> 131071 : 338 | | 131072 -> 262143 : 93 | | 262144 -> 524287 : 28 | | 524288 -> 1048575 : 4 | |

Как показывает этот вывод, обычно приложение выполнялось на процессоре очень короткое время, часто менее 7 мкс. Другие инструменты (stackcount(8) для t:sched:sched_switch и /proc/PID/status) показали, что приложение обычно покидает CPU из-за принудительного1 переключения контекста. На хосте контейнера: serverB# cpudist -p 1928219 10 1 Tracing on-CPU time... Hit Ctrl-C to end. usecs 0 2 4 8 16 32 64 128 256 512 1024 2048 4096

-> -> -> -> -> -> -> -> -> -> -> -> ->

1 3 7 15 31 63 127 255 511 1023 2047 4095 8191

: : : : : : : : : : : : : :

count 0 16 6 7 8 10 18 40 44 156 238 4511 277

distribution | | | | | | | | | | | | | | | | | | |* | |** | |****************************************| |** |

В /proc/PID/status они называются nonvolption_ctxt_switches.

1

948  Глава 16. Пример из практики 8192 16384 32768 65536 131072 262144 524288

-> -> -> -> -> -> ->

16383 32767 65535 131071 262143 524287 1048575

: : : : : : :

286 77 63 44 9 14 5

|** | | | | | |

| | | | | | |

В этом случае приложение выполнялось на процессоре в среднем от 2 до 4 мс. Другие инструменты показали, что приложение не слишком часто прерывается принудительными переключениями контекста. Принудительные переключения контекста и высокая частота переключений на виртуальной машине вызвали проблемы с производительностью. Приложение вынуждено было оставлять процессор, проработав менее 10 мкс, что не давало времени кэшам процессора «прогреться» под особенности текущего исполняемого кода.

16.1.8. Заключение По результатам анализа я пришел к выводу, что увеличение производительности обусловлено следующими причинами:

yy Отсутствие соседних контейнеров: хост контейнера почти бездействовал, об-

служивая единственный контейнер. Это позволило контейнеру использовать все кэши процессора и не конкурировать с другими контейнерами за процессор. Это обусловило высокие показатели производительности во время тестирования, но едва ли эта тенденция сохранится в промышленной среде в долгосрочной перспективе, когда появятся соседние контейнеры. Падение производительности микросервиса в 3–4 раза может обнаружиться, когда появятся другие арендаторы.

yy Разница в размере LLC и рабочей нагрузке: величина IPC в виртуальной

машине оказалась в 2,6 раза ниже, что может объяснить замедление в 2,6 раза. Одной из причин, вероятно, была конкуренция между гиперпотоками, потому что хост виртуальной машины работал с нагрузкой выше 50 % (и имел по два гиперпотока на ядро). Но главная причина, скорее всего, заключается в низком коэффициенте попаданий в LLC: 30 % на виртуальной машине по сравнению с 90 % на контейнере. Такой низкий коэффициент попаданий объясняется тремя факторами. • Меньший размер LLC на виртуальной машине: 33 Мбайт против 45 Мбайт. • Более сложная рабочая нагрузка на виртуальной машине: полное приложение с большим объемом кода и данных по сравнению с компонентом, выполняемым в контейнере. • Высокая частота переключения контекста на виртуальной машине: около 2 миллионов в секунду. Это не позволяет потокам занимать процессор на долгое время и не дает прогреть кэш. Длительность выполнения потоков на процессоре в виртуальной машине обычно составляла менее 10 мкс, тогда как на хосте контейнера потоки выполнялись на процессоре в среднем 2–4 мс.

16.3. Ссылки  949

yy Разница в нагрузке на процессор: виртуальная машина испытывала более высо-

кую нагрузку, что привело к насыщению процессора: средний уровень нагрузки оказался равным 85 в системе с 48 CPU. Это привело к увеличению частоты переключений контекста до 2 миллионов в секунду и длительному ожиданию потоков в очереди на выполнение. Задержка в очереди на выполнение, подразу­ меваемая средними показателями нагрузки, показала, что виртуальная машина работает примерно в 4 раза медленнее.

Эти проблемы объясняют наблюдаемую разницу в производительности.

16.2. ДОПОЛНИТЕЛЬНАЯ ИНФОРМАЦИЯ Чтобы познакомиться с другими примерами анализа производительности, загляните в базу данных ошибок (или в систему отслеживания тикетов) вашей компании и поищите отчеты о решении предыдущих проблем с производительностью, а также публичные базы данных ошибок для приложений и ОС. Эти отчеты часто начинаются с описания проблемы и заканчиваются рецептом ее исправления. Многие базы данных ошибок также включают историю комментариев, которые можно изучить и увидеть ход анализа, включая исследованные гипотезы и принятые неверные решения. Поворот не туда и выявление нескольких факторов — это нормально. Время от времени в интернете публикуются тематические исследования производительности систем, например, в моем блоге [Gregg 20j]. Технические журналы с упором на практику, например «USENIX; login:» [USENIX 20] и «ACM Queue» [ACM 20], также часто используют тематические исследования в качестве контекста при описании новых решений проблем.

16.3. ССЫЛКИ [Gregg 19h] Gregg, B., «LISA2019 Linux Systems Performance», USENIX LISA, http://www. brendangregg.com/blog/2020-03-08/lisa2019-linux-systems-performance.html, 2019. [ACM 20] «acmqueue», http://queue.acm.org, по состоянию на 2020. [Gregg 20e] Gregg, B., «PMC (Performance Monitoring Counter) Tools for the Cloud», https:// github.com/brendangregg/pmc-cloud-tools, последнее обновление 2020. [Gregg 20j] «Brendan Gregg’s Blog», http://www.brendangregg.com/blog, последнее обновление 2020. [USENIX 20] «;login: The USENIX Magazine», https://www.usenix.org/publications/login, по состоянию на 2020.

Приложение A

МЕТОД USE: LINUX В этом приложении — чек-лист для Linux, созданный на основе метода USE [Gregg 13d]. Этот метод проверки работоспособности системы и выявления типичных узких мест и ошибок был описан в главе 2 «Методологии», в разделе 2.5.9 «Метод USE». В последующих главах (5, 6, 7, 9, 10) он описан в конкретном контексте вместе с инструментами для его поддержки. Инструменты производительности совершенствуются, постоянно разрабатываются новые, поэтому рассматривайте этот чек-лист как отправную точку, которую нужно корректировать и обновлять. Также могут появиться новые фреймворки и инструменты наблюдения, упрощающие выполнение метода USE.

A1. ФИЗИЧЕСКИЕ РЕСУРСЫ Компонент

Тип

Метрика

CPU

Потребление

С разбивкой по процессорам: mpstat -P ALL 1 столбцы, отражающие потребление процессоров (%usr, %nice, %sys, %irq, %soft, %guest, %gnice), и обратные им столбцы, характеризующие бездействие (%iowait, %steal, %idle); sar -P ALL столбцы, отражающие потребление процессоров (%user, %nice, %system) , и обратные им столбцы, характеризующие бездействие (%iowait, %steal, %idle) Для системы в целом: vmstat 1, столбцы us + sy; sar -u, столбцы %user + %nice + %system С разбивкой по процессам: top, столбец %CPU; htop, столбец CPU%; ps -o pcpu; pidstat 1, столбец %CPU С разбивкой по потокам ядра: top/htop (K для переключения), в строках с VIRT == 0 (эвристика)

A1. Физические ресурсы   951

Компонент

Тип

Метрика

CPU

Насыщение

Для системы в целом: vmstat 1, r > количества процессоров1; sar -q, runq-sz > количества процессоров; runqlat; runqlen

С разбивкой по процессам: /proc/PID/schedstat второе поле (sched_info.run_delay); getdelays.c, CPU2; perf sched latency (средняя и максимальная задержки планирования)3 CPU

Ошибки

Исключения проверки компьютера (Machine Check Exceptions, MCE) можно увидеть с помощью dmesg или rasdaemon и ras-mc-ctl –summary; события ошибок (PMC), характерные для процессоров, доступны в perf(1); например, для AMD64: «04Ah Single-bit ECC Errors Recorded by Scrubber»4 (которую также можно классифицировать как ошибку памяти); ipmtool sel list; ipmitool sdr list

Память

Потребление

Для системы в целом: free -m, поле Mem: (основная память), поле Swap: (виртуальная память); vmstat 1, поле free (основная память), поле swap (виртуальная память); sar -r, столбец %memused; slabtop -s c для определения потребления памяти ядра С разбивкой по процессам: top/htop, столбец RES (резидентная основная память), столбец VIRT (виртуальная память), поле Mem — для системы в целом

Память

Насыщение

Для системы в целом: vmstat 1, столбцы si/so (подкачка); sar -B, столбцы pgscank + pgscand (сканирование); sar -W

В столбце r указано количество потоков, ожидающих в очереди на выполнение плюс выполняющихся на процессоре. См. описание vmstat(1) в главе 6 «Процессоры».

1

Использует систему учета задержек, см. главу 4 «Инструменты наблюдения».

2

Также для perf(1) есть точка трассировки sched:sched_process_wait. Будьте осторожны с оверхедом при трассировке, так как события планировщика следуют друг за другом очень часто.

3

4

В последних руководствах по процессорам Intel и AMD перечислено не так много событий, связанных с ошибками.

952  Приложение A. Метод USE: Linux

Компонент

Тип

Метрика С разбивкой по процессам: getdelays.c, столбец SWAP2; 10-е поле (min_flt) в /proc/PID/stat для наблюдения за частотой отказов или для динамической инструментации5; dmesg | grep killed (события механизма OOM Killer)

Память

Ошибки

dmesg для поиска ошибок физической памяти или rasdaemon и ras-mcctl --summary или edac-util; dmidecode тоже можно использовать для выявления ошибок физической

памяти;

ipmtool sel list; ipmitool sdr list;

динамическая инструментация, например, зондов uretprobe для выявления отказов в malloc() (bpftrace) Сетевые интерфейсы

Потребление

ip -s link, прием/передача, пропускная способность и ширина полосы

Сетевые интерфейсы

Насыщение

nstat, поле TcpRetransSegs;

Сетевые интерфейсы

Ошибки

пропускания; sar -n DEV, скорость приема/передачи, ширина полосы пропускания; /proc/net/dev, пропускная способность приема/передачи

sar -n EDEV, столбцы *drop/s, *fifo/s6; /proc/net/dev, принято/получено/сброшено байтов; динамическая инструментация других механизмов очередей в стеке TCP/ IP (bpftrace) ip -s link, столбец errors; sar -n EDEV all;

/proc/net/dev, поля errs, drop6; дополнительные счетчики можно найти в /sys/class/net/*/statistics/*error*; динамическая инструментация возврата из функций драйвера Устройства хранения, ввод/вывод

Потребление

Для системы в целом: iostat -xz 1, столбец %util; sar -d, столбец %util; С разбивкой по процессам: iotop, biotop;

/proc/PID/sched поля se.statistics. iowait_sum

5

Используйте, чтобы выяснить, какие процессы потребляют память и вызывают насыщение, путем наблюдения за незначительными отказами. Должно быть доступно также в htop(1) в столбце MINFLT. Информация о количестве сброшенных пакетов может служить признаком насыщения и ошибок, так как пакеты могут сбрасываться в обоих случаях.

6

A1. Физические ресурсы   953

Компонент

Тип

Метрика

Устройства хранения, ввод/вывод

Насыщение

iostat -xnz 1, где avgqu-sz > 1 или высокое значение await; sar -d то же самое;

perf(1) точки трассировки из семейства block для определения длины очереди и задержки в ней; biolatency

Устройства хранения, ввод/вывод

Ошибки

/sys/devices/.../ioerr_cnt; smartctl; bioerr;

динамическая/статическая инструментация подсистемы ввода/вывода для получения кодов ответа7 Устройства хранения, емкость

Потребление

Подкачка: swapon -s; free;

/proc/meminfo поля SwapFree/SwapTotal; Файловые системы: df -h

7

Устройства хранения, емкость

Насыщение

Не уверен, что в этом есть смысл, — когда устройство хранения заполнено, возвращается признак ошибки ENOSPC (хотя когда устройство близко к заполнению, производительность может снижаться в зависимости от алгоритма поиска свободных блоков в файловой системе)

Устройства хранения, емкость

Ошибки файловых систем

strace для выявления ошибок ENOSPC;

Контроллер устройств хранения

Потребление

iostat -sxz 1, обобщенные статистики по устройствам и их сравнение

Контроллер устройств хранения

Насыщение

См. «Устройства хранения, ввод/вывод» — «Насыщение»

Контроллер устройств хранения

Ошибки

См. «Устройства хранения, ввод/вывод» — «Ошибки»

Сетевой контроллер

Потребление

Может косвенно определяться по выводу ip –s link (или sar, или /proc/ net/dev) с последующим сравнением с известными пределами пропускной способности контроллера

динамическая инструментация для выявления ошибок ENOSPC; /var/log/messages поле errs, в зависимости от файловой системы; журнал ошибок приложения с известными пределами пропускной способности контроллеров

В том числе трассировка функций из разных уровней подсистемы ввода/вывода: блочное устройство, SCSI, SATA, IDE... Также доступны некоторые статические зонды (точки трассировки scsi и block для perf(1)). Для прочих случаев используйте динамическую трассировку.

954  Приложение A. Метод USE: Linux

Компонент

Тип

Метрика

Сетевой контроллер

Насыщение

См. «Сетевые интерфейсы» — «Насыщение»

Сетевой контроллер

Ошибки

См. «Сетевые интерфейсы» — «Ошибки»

Внутренняя шина между процессорами

Потребление

perf stat со счетчиками PMC для портов внутренней шины между про-

Внутренняя шина между процессорами

Насыщение

perf stat со счетчиками PMC циклов простоя

Внутренняя шина между процессорами

Ошибки

perf stat со всеми доступными счетчиками PMC

Внутренняя шина памяти

Потребление

perf stat со счетчиками PMC для памяти шины памяти и сравнение

Внутренняя шина памяти

Насыщение

perf stat со счетчиками PMC циклов простоя

Внутренняя шина памяти

Ошибки

perf stat со всеми доступными счетчиками PMC;

Внутренняя шина ввода/вывода

Потребление

perf stat со счетчиками PMC для определения пропускной способности

Внутренняя шина ввода/вывода

Ошибки

perf stat со всеми доступными счетчиками PMC

цессорами, сравнение пропускной способности с шириной полосы пропускания

пропускной способности с шириной полосы пропускания; например, Intel uncore_imc/data_reads/, uncore_imc/data_writes/; или IPC меньше, чем, например, 0.2. Счетчики также могут быть локальными и удаленными

dmidecode тоже может дать полезную информацию

и полосы пропускания, если таковые доступны; косвенное определение пропускной способности по выводу iostat/ip/...

Общие замечания: метрика «load average», которую выводит команда uptime (или значение в /proc/loadavg) не учитывается в метриках процессора, потому что среднее значение нагрузки в Linux включает задачи, действующие в режиме непрерываемого ввода/вывода. perf(1): это мощный набор инструментов наблюдения, которые читают счетчики PMC, а также могут выполнять динамическую и статическую инструментацию. Интерфейсом к ним служит команда perf(1). См. главу 13 «perf». Счетчики PMC: счетчики мониторинга производительности. См. главу 6 «Процессоры» и примеры их использования с perf(1).

A2. Программные ресурсы

  955

Внутренние шины ввода/вывода: к ним относятся шины контроллера ввода/вывода, контроллер(-ы) ввода/вывода и шины устройств (например, PCIe). Динамическая инструментация: позволяет конструировать собственные метрики. См. главу 4 «Инструменты наблюдения» и примеры в последующих главах. К числу инструментов динамической трассировки для Linux входят: perf(1) (глава 13), Ftrace (глава 14), BCC и bpftrace (глава 15). В любой среде, где действуют средства управления ресурсами (например, в облачных средах), примените метод USE для каждого такого средства. Установленные ими ограничения могут быть существенно ниже фактических ограничений оборудования.

A2. ПРОГРАММНЫЕ РЕСУРСЫ Компонент

Тип

Метрика

Мьютексы ядра

Потребление

При условии CONFIG_LOCK_STATS = y, /proc/lock_stat поля holdtimetotal/acquisitions (см. также holdtime-min, holdtime-max)1; динамическая инструментация функций или инструкций блокировки (если возможно)

Мьютексы ядра

Насыщение

При условии CONFIG_LOCK_STATS = y, /proc/lock_stat поля waittimetotal/contentions (см. также waittime-min, waittime-max); динамическая инструментация функций блокировки, например, с помощью mlock.bt [Gregg 19]; спин-блокировки можно увидеть при профилировании командой

Мьютексы ядра

Ошибки

Динамическая инструментация (например, рекурсивные попытки приобретения мьютекса); другие ошибки, способные вызвать зависание или крах ядра, отлаживаются с помощью kdump/crash

Мьютексы в пространстве пользователя

Потребление

valgrind --tool=drd --exclusive-threshold=... (время

Мьютексы в пространстве пользователя

Насыщение

valgrind --tool=drd для выявления признаков конкуренции

perf record -a -g -F 99 ...

удержания); динамическая инструментация функций приобретения/освобождения блокировок2 по времени удержания; динамическая инструментация функций синхронизации для определения времени ожидания, например, с помощью pmlock.bt; профилирование по трассировкам стека в пространстве пользователя для выявления спин-блокировок (perf(1))

1

Раньше анализ блокировок ядра проводился с помощью инструмента lockmeter, доступного через интерфейс lockstat.

2

Эти функции могут вызываться очень часто, остерегайтесь оверхеда: трассировка каждого вызова может замедлить приложение в два и более раза.

956  Приложение A. Метод USE: Linux

Компонент

Тип

Метрика

Мьютексы в пространстве пользователя

Ошибки

valgrind --tool=drd различные ошибки;

Емкость пространства задач

Потребление

Емкость пространства задач

Насыщение

Блокировка потоков выполнения при выделении памяти; на этом этапе должен быть запущен сканер страниц (sar -B, pgscan*), в противном случае попробуйте выполнить динамическую трассировку

Емкость пространства задач

Ошибки

Ошибки "can’t fork()"; для потоков выполнения в пространстве пользователя: pthread_create() завершается с кодом ошибки EAGAIN, EINVAL, . . . ; для потоков ядра: выполните динамическую трассировку kernel_ thread() на предмет ошибки ENOMEM

Файловые дескрипторы

Потребление

Для системы в целом: sar -v, сравните file-nr с /proc/sys/fs/file-max; или просто проверьте /proc/sys/fs/file-nr

динамическая инструментация pthread_mutex_lock() для определения ошибок EAGAIN, EINVAL, EPERM, EDEADLK, ENOMEM, EOWNERDEAD, ... top/htop, поле Tasks (текущее значение); sysctl kernel.threads-max, /proc/sys/kernel/threads-max

(поле max)

С разбивкой по процессам: сравните echo /proc/PID/fd/* | wc -w с ulimit -n Файловые дескрипторы

Насыщение

Этот анализ может не иметь смысла

Файловые дескрипторы

Ошибки

strace возврат errno == EMFILE системными вызовами, возвращаю-

щими файловые дескрипторы (например, open(2), accept(2), ...); opensnoop -x

A3. ССЫЛКИ [Gregg 13d] Gregg, B., «USE Method: Linux Performance Checklist», http://www.brendangregg. com/USEmethod/use-linux.html, впервые опубликована в 2013 г.

Приложение B

КРАТКИЙ СПРАВОЧНИК ПО SAR В этом приложении — краткое описание параметров и метрик генератора отчетов по работе системы (system activity reporter) sar(1). Используйте это приложение как памятку о доступных метриках и параметрах. Полный список ищите на странице справочного руководства man. О sar(1) подробно рассказывалось в главе 4 «Инструменты наблюдения», в разделе 4.4, а некоторые параметры описывались в последующих главах (6, 7, 8, 9, 10). Параметр

Метрики

Описание

-u

%user %nice %system %iowait %steal %idle

Потребление процессоров по отдельности (-u — необязательный параметр)

-u

%user %nice %system %iowait %steal %idle

Потребление процессоров

-u ALL

... %irq %soft %guest %gnice

Расширенная информация о потреблении процессоров

-m CPU

МГц

Тактовая частота с разбивкой по процессорам

-q

runq-sz plist-sz ldavg-1 ldavg-5 ldavg-15 blocked

Размер очереди на выполнение

-w

proc/s cswch/s

События планировщика процессора

-B

pgpgin/s pgpgout/s fault/s majflt/s pgfree/s pgscank/s pgscand/s pgsteal/s %vmeff

Статистики подкачки страниц

-H

kbhugfree kbhugused %hugused

Огромные страницы

-r

kbmemfree kbavail kbmemused %memused

Потребление памяти

-P ALL

-P ALL

kbbuffers kbcached kbcommit %commit kbactive kbinact kbdirty -S

kbswpfree kbswpused %swpused kbswpcad %swpcad

Потребление виртуальной памяти

-W

pswpin/s pswpout/s

Статистики подкачки

958  Приложение B. Краткий справочник по sar

Параметр

Метрики

Описание

-v

dentunusd file-nr inode-nr pty-nr

Таблицы ядра

-d

tps rkB/s wkB/s areq-sz aqu-sz await svctm %util

Статистики дисков

-n DEV

rxpck/s txpck/s rxkB/s txkB/s rxcmp/s txcmp/s rxmcst/s %ifutil

Статистики сетевого интерфейса

-n EDEV

rxerr/s txerr/s coll/s rxdrop/s txdrop/s txcarr/s rxfram/s rxfifo/s txfifo/s

Ошибки сетевого интерфейса

-n IP

irec/s fwddgm/s idel/s orq/s asmrq/s asmok/s fragok/s fragcrt/s

Статистики протокола IP

-n EIP

ihdrerr/s iadrerr/s iukwnpr/s idisc/s odisc/s onort/s asmf/s fragf/s

Ошибки протокола IP

-n TCP

active/s passive/s iseg/s oseg/s

Статистики протокола ЕСP

-n ETCP

atmptf/s estres/s retrans/s isegerr/s orsts/s

Ошибки протокола ЕСP

-n SOCK

totsck tcpsck udpsck rawsck ip-frag tcptw

Статистики сокетов

Полужирным шрифтом я выделил ключевые метрики, на которые обращаю внимание. Некоторые параметры sar(1) могут требовать того, чтобы ядро было скомпилировано с определенными параметрами (например, поддержка огромных страниц). Некоторые метрики были добавлены в более поздних версиях sar(1) (здесь перечислены метрики и параметры для версии sar 12.0.6).

Приложение C

ОДНОСТРОЧНЫЕ СЦЕНАРИИ ДЛЯ BPFTRACE В этом приложении приведены удобные однострочные сценарии для bpftrace. Они полезны не только сами по себе: они помогут освоить программирование на языке bpftrace. Большинство из них уже приводилось в предыдущих главах. Многие могут не запускаться в некоторых системах, так как зависят от наличия определенных точек трассировки или функций либо от конкретной версии или конфигурации ядра. Для знакомства с bpftrace cм. главу 15, раздел 15.2.

ПРОЦЕССОРЫ Трассировка создания новых процессов с регистрацией аргументов: bpftrace -e 'tracepoint:syscalls:sys_enter_execve { join(args->argv); }'

Подсчет количества обращений к системным вызовам с разбивкой по процессам: bpftrace -e 'tracepoint:raw_syscalls:sys_enter { @[pid, comm] = count(); }'

Подсчет количества обращений к системным вызовам по их именам: bpftrace -e 'tracepoint:syscalls:sys_enter_* { @[probe] = count(); }'

Выборка имен работающих процессов с частотой 99 Гц: bpftrace -e 'profile:hz:99 { @[comm] = count(); }'

Выборка всех трассировок стека в системе в пространствах пользователя и ядра с частотой 49 Гц с именами процессов: bpftrace -e 'profile:hz:49 { @[kstack, ustack, comm] = count(); }'

Выборка трассировок стека в пространстве пользователя с частотой 49 Гц для PID 189: bpftrace -e 'profile:hz:49 /pid == 189/ { @[ustack] = count(); }'

960  Приложение C. Однострочные сценарии для bpftrace Выборка трассировок стека глубиной до пяти фреймов в пространстве пользователя с частотой 49 Гц для PID 189: bpftrace -e 'profile:hz:49 /pid == 189/ { @[ustack(5)] = count(); }'

Выборка трассировок стека в пространстве пользователя с частотой 49 Гц для процесса с именем «mysqld»: bpftrace -e 'profile:hz:49 /comm == "mysqld"/ { @[ustack] = count(); }'

Подсчет пересечений точек трассировки планировщика в ядре: bpftrace -e 'tracepoint:sched:* { @[probe] = count(); }'

Подсчет трассировок стека ядра вне процессора по событиям переключения контекста: bpftrace -e 'tracepoint:sched:sched_switch { @[kstack] = count(); }'

Подсчет количества вызовов функций ядра, имена которых начинаются с «vfs_»: bpftrace -e 'kprobe:vfs_* { @[func] = count(); }'

Трассировка создания новых потоков выполнения вызовом функции pthread_ create(): bpftrace -e 'u:/lib/x86_64-linux-gnu/libpthread-2.27.so:pthread_create { printf("%s by %s (%d)\n", probe, comm, pid); }'

ПАМЯТЬ Суммировать объем памяти в байтах, выделяемой вызовом функции malloc() из библиотеки libc с разбивкой по трассировкам стека в пространстве пользователя (имеет высокий оверхед): bpftrace -e 'u:/lib/x86_64-linux-gnu/libc.so.6:malloc { @[ustack, comm] = sum(arg0); }'

Суммировать объем памяти в байтах, выделяемой вызовом функции malloc() из библиотеки libc с разбивкой по трассировкам стека в пространстве пользователя для процесса с PID 181 (имеет высокий оверхед): bpftrace -e 'u:/lib/x86_64-linux-gnu/libc.so.6:malloc /pid == 181/ { @[ustack] = sum(arg0); }'

Показать гистограмму с шагом, равным степени 2, для распределения объемов выделяемой памяти через вызов функции malloc() из библиотеки libc с разбивкой по трассировкам стека в пространстве пользователя для процесса с PID 181 (имеет высокий оверхед): bpftrace -e 'u:/lib/x86_64-linux-gnu/libc.so.6:malloc /pid == 181/ { @[ustack] = hist(arg0); }'

Файловые системы  961 Подсчет количества байтов, выделенных в кэше ядра kmem, по трассировкам стека в пространстве ядра: bpftrace -e 't:kmem:kmem_cache_alloc { @bytes[kstack] = sum(args->bytes_alloc); }'

Подсчет событий увеличения кучи процесса (brk(2)) с разбивкой по трассировкам стека: bpftrace -e 'tracepoint:syscalls:sys_enter_brk { @[ustack, comm] = count(); }'

Подсчет сбоев страниц с разбивкой по процессам: bpftrace -e 'software:page-fault:1 { @[comm, pid] = count(); }'

Подсчет сбоев страниц в пространстве пользователя с разбивкой по трассировкам стека в пространстве пользователя: bpftrace -e 't:exceptions:page_fault_user { @[ustack, comm] = count(); }'

Подсчет операций в vmscan с использованием точек трассировки: bpftrace -e 'tracepoint:vmscan:* { @[probe]++; }'

Подсчет событий подкачки страниц с разбивкой по процессам: bpftrace -e 'kprobe:swap_readpage { @[comm, pid] = count(); }'

Подсчет миграций страниц: bpftrace -e 'tracepoint:migrate:mm_migrate_pages { @ = count(); }'

Трассировка событий уплотнения памяти: bpftrace -e 't:compaction:mm_compaction_begin { time(); }'

Список зондов USDT в libc: bpftrace -l 'usdt:/lib/x86_64-linux-gnu/libc.so.6:*'

Список точек трассировки kmem в ядре: bpftrace -l 't:kmem:*'

Список всех точек трассировки в подсистеме управления памятью (mm): bpftrace -l 't:*:mm_*'

ФАЙЛОВЫЕ СИСТЕМЫ Трассировка операций открытия файлов через вызов openat(2) с именами процессов: bpftrace -e 't:syscalls:sys_enter_openat { printf("%s %s\n", comm, str(args->filename)); }'

962  Приложение C. Однострочные сценарии для bpftrace Подсчет обращений к системным вызовам из семейства read: bpftrace -e 'tracepoint:syscalls:sys_enter_*read* { @[probe] = count(); }'

Подсчет обращений к системным вызовам из семейства write: bpftrace -e 'tracepoint:syscalls:sys_enter_*write* { @[probe] = count(); }'

Показать гистограмму с распределением вызовов read() по запрошенным размерам блоков: bpftrace -e 'tracepoint:syscalls:sys_enter_read { @ = hist(args->count); }'

Показать гистограмму с распределением вызовов read() по размерам прочитанных блоков (и кодам ошибок): bpftrace -e 'tracepoint:syscalls:sys_exit_read { @ = hist(args->ret); }'

Подсчет количества вызовов read(), завершившихся ошибкой, с разбивкой по кодам ошибок: bpftrace -e 't:syscalls:sys_exit_read /args->ret < 0/ { @[- args->ret] = count(); }'

Подсчет количества вызовов VFS: bpftrace -e 'kprobe:vfs_* { @[probe] = count(); }'

Подсчет количества вызовов VFS для процесса с PID 181: bpftrace -e 'kprobe:vfs_* /pid == 181/ { @[probe] = count(); }'

Подсчет количества пересечений точек трассировки в ext4: bpftrace -e 'tracepoint:ext4:* { @[probe] = count(); }'

Подсчет количества пересечений точек трассировки в xfs: bpftrace -e 'tracepoint:xfs:* { @[probe] = count(); }'

Подсчет количества операций чтения файлов в файловой системе ext4 с разбивкой по именам процессов и трассировкам стека в пространстве пользователя: bpftrace -e 'kprobe:ext4_file_read_iter { @[ustack, comm] = count(); }'

Определяет время выполнения spa_sync() в ZFS: bpftrace -e 'kprobe:spa_sync { time("%H:%M:%S ZFS spa_sync()\n"); }'

Подсчет количества обращений к кэшу каталогов с разбивкой по именам и идентификаторам (PID) процессов: bpftrace -e 'kprobe:lookup_fast { @[comm, pid] = count(); }'

Сети  963

ДИСКИ Подсчет событий блочного ввода/вывода по точкам трассировки: bpftrace -e 'tracepoint:block:* { @[probe] = count(); }'

Вывод размеров блочного ввода/вывода в виде гистограммы: bpftrace -e 't:block:block_rq_issue { @bytes = hist(args->bytes); }'

Подсчет операций ввода/вывода с разбивкой по трассировкам стека в пространстве пользователя: bpftrace -e 't:block:block_rq_issue { @[ustack] = count(); }'

Подсчет операций ввода/вывода с разбивкой по типам: bpftrace -e 't:block:block_rq_issue { @[args->rwbs] = count(); }'

Подсчет ошибок блочного ввода/вывода с разбивкой по типам операций и устройствам: bpftrace -e 't:block:block_rq_complete /args->error/ { printf("dev %d type %s error %d\n", args->dev, args->rwbs, args->error); }'

Подсчет операций SCSI с разбивкой по их кодам: bpftrace -e 't:scsi:scsi_dispatch_cmd_start { @opcode[args->opcode] = count(); }'

Подсчет операций SCSI с разбивкой по кодам завершения: bpftrace -e 't:scsi:scsi_dispatch_cmd_done { @result[args->result] = count(); }'

Подсчет количества вызовов функций в драйвере SCSI: bpftrace -e 'kprobe:scsi* { @[func] = count(); }'

СЕТИ Подсчет обращений к системному вызову accept(2) с разбивкой по PID и именам процессов: bpftrace -e 't:syscalls:sys_enter_accept* { @[pid, comm] = count(); }'

Подсчет обращений к системному вызову connect(2) с разбивкой по PID и именам процессов: bpftrace -e 't:syscalls:sys_enter_connect { @[pid, comm] = count(); }'

964  Приложение C. Однострочные сценарии для bpftrace Подсчет обращений к системному вызову connect(2) с разбивкой по трассировкам стека в пространстве пользователя: bpftrace -e 't:syscalls:sys_enter_connect { @[ustack, comm] = count(); }'

Подсчет операций приема/передачи с разбивкой по направлениям, а также PID и именам процессов на процессоре: bpftrace -e 'k:sock_sendmsg,k:sock_recvmsg { @[func, pid, comm] = count(); }'

Подсчет принятых/отправленных байтов с разбивкой по PID и именам процессов на процессоре: bpftrace -e 'kr:sock_sendmsg,kr:sock_recvmsg /(int32)retval > 0/ { @[pid, comm] = sum((int32)retval); }'

Подсчет соединений TCP с разбивкой по PID и именам процессов на процессоре: bpftrace -e 'k:tcp_v*_connect { @[pid, comm] = count(); }'

Подсчет принятых соединений TCP с разбивкой по PID и именам процессов на процессоре: bpftrace -e 'k:inet_csk_accept { @[pid, comm] = count(); }'

Подсчет операций приема/передачи с разбивкой по PID и именам процессов на процессоре: bpftrace -e 'k:tcp_sendmsg,k:tcp_recvmsg { @[func, pid, comm] = count(); }'

Гистограмма с распределением количества отправленных байтов по протоколу TCP: bpftrace -e 'k:tcp_sendmsg { @send_bytes = hist(arg2); }'

Гистограмма с распределением количества полученных байтов по протоколу TCP: bpftrace -e 'kr:tcp_recvmsg /retval >= 0/ { @recv_bytes = hist(retval); }'

Подсчет повторных передач TCP с разбивкой по типам и удаленным хостам (предполагается, что используется протокол IPv4): bpftrace -e 't:tcp:tcp_retransmit_* { @[probe, ntop(2, args->saddr)] = count(); }'

Подсчет вызовов всех функций TCP (добавляет значительный оверхед при большом количестве операций TCP): bpftrace -e 'k:tcp_* { @[func] = count(); }'

Подсчет операций приема/передачи по протоколу UDP с разбивкой по PID и именам процессов на процессоре: bpftrace -e 'k:udp*_sendmsg,k:udp*_recvmsg { @[func, pid, comm] = count(); }'

Сети  965 Гистограмма с распределением количества отправленных байтов по протоколу UDP: bpftrace -e 'k:udp_sendmsg { @send_bytes = hist(arg2); }'

Гистограмма с распределением количества принятых байтов по протоколу UDP: bpftrace -e 'kr:udp_recvmsg /retval >= 0/ { @recv_bytes = hist(retval); }'

Подсчет операций передачи с разбивкой по трассировкам стека в пространстве ядра: bpftrace -e 't:net:net_dev_xmit { @[kstack] = count(); }'

Показать гистограмму потребления процессорного времени на прием для каждого устройства: bpftrace -e 't:net:netif_receive_skb { @[str(args->name)] = lhist(cpu, 0, 128, 1); }'

Подсчет вызовов функций ieee80211 (добавляет значительный оверхед в операции с пакетами): bpftrace -e 'k:ieee80211_* { @[func] = count(); }'

Подсчет вызовов всех функций драйвера устройства ixgbevf (добавляет значительный оверхед в работу ixgbevf): bpftrace -e 'k:ixgbevf_* { @[func] = count(); }'

Подсчет пересечения всех точек трассировки драйвера устройства iwl (добавляет значительный оверхед в работу iwl): bpftrace -e 't:iwlwifi:*,t:iwlwifi_io:* { @[probe] = count(); }'

Приложение D

РЕШЕНИЯ НЕКОТОРЫХ УПРАЖНЕНИЙ

Ниже приводятся ответы на вопросы и варианты решения некоторых упражнений.

ГЛАВА 2. МЕТОДОЛОГИИ В. Что такое задержка? О. Мера времени, обычно время ожидания завершения чего-либо. В IT-индустрии термин используется по-разному в зависимости от контекста.

ГЛАВА 3. ОПЕРАЦИОННЫЕ СИСТЕМЫ В. Перечислите причины, почему поток выполнения может оставить процессор. О. В результате блокировки на операции ввода/вывода, приостановки на блокировке вызова yield, по истечении выделенного кванта времени, в результате вытеснения другим потоком, при получении прерывания от устройства, по завершении работы.

ГЛАВА 6. ПРОЦЕССОРЫ В. Вычислите среднюю нагрузку... О. 34.

ГЛАВА 7. ПАМЯТЬ В. В чем разница между подкачкой страниц (paging) и анонимной подкачкой (swapping) с точки зрения терминологии Linux?

Решения некоторых упражнений  967 О. Подкачка страниц (paging) — это передача страниц памяти между основной памятью и запоминающими устройствами. Подкачка — это анонимная передача страниц на устройство/файл подкачки и обратно. В. Опишите характеристики потребления и насыщения памяти. О. Потребление — это отношение объема памяти, используемой и недоступной для других нужд, к общему количеству доступной памяти. Потребление можно выразить в процентах, как емкость файловой системы. Насыщение — это мера потребности в доступной памяти, превышающей размер имеющейся памяти. Обычно когда наступает насыщение, запускается процедура ядра для освобождения памяти и удовлетворения этой потребности.

ГЛАВА 8. ФАЙЛОВЫЕ СИСТЕМЫ В. В чем разница между логическим и физическим вводом/выводом? О. Логический ввод/вывод — это ввод/вывод между приложением и файловой системой. Физический ввод/вывод — это ввод/вывод между файловой системой и устройствами хранения (дисками). В. Объясните, как использование алгоритма копирования при записи может повысить производительность файловой системы. О. Поскольку произвольная запись может быть записана в новое место, такие операции группируются (за счет увеличения объема ввода/вывода), и запись производится последовательно. Оба этих фактора обычно улучшают производительность в зависимости от типа устройства хранения.

ГЛАВА 9. ДИСКИ В. Опишите, что происходит, когда диски перегружены работой, включая влияние на производительность приложений. О. Такие диски работают с постоянно высоким уровнем потребления (до 100 %) и насыщения (запросы ставятся в очередь). Задержка ввода/вывода таких дисков увеличивается из-за вероятности постановки запроса в очередь (что можно смоделировать). Если приложение выполняет ввод/вывод, увеличенная задержка может снизить его производительность при условии, что ввод/вывод выполняется синхронно: чтение или синхронная запись. Это также всегда происходит при выполнении критических путей в коде приложения, например во время обслуживания запроса, но не во время выполнения асинхронной фоновой задачи (которая может лишь косвенно вызвать снижение производительности приложения). Обычно правильная реакция в ответ на увеличение задержки ввода/вывода позволяет держать частоту запросов ввода/вывода под контролем и не вызвать неограниченного увеличения задержки.

968  Приложение D. Решения некоторых упражнений

ГЛАВА 11. ОБЛАЧНЫЕ ВЫЧИСЛЕНИЯ В. Опишите наблюдаемость физической системы с точки зрения гостя в виртуализации операционной системы. О. В зависимости от реализации ядра хоста гость может видеть высокоуровневые метрики всех физических ресурсов, включая процессоры и диски, и замечать, когда они используются другими арендаторами. Метрики, которые могут привести к утечке пользовательских данных, должны блокироваться ядром. Например, ядро позволяет наблюдать уровень потребления процессора (скажем, 50 %), но не идентификаторы и имена процессов других арендаторов, влияющих на потребление.

Приложение E

ПРОИЗВОДИТЕЛЬНОСТЬ СИСТЕМ, КТО ЕСТЬ КТО Бывает полезно знать по именам создателей технологий, которыми мы пользуемся. Ниже — список «кто есть кто» в области производительности систем, основанный на технологиях, приведенных в этой книге. Я решил добавить его, вдохновившись списком «Кто есть кто в Unix», опубликованным в [Libes 89]. Заранее прошу прощения у всех, кого упустил из виду или включил в список по ошибке. Если вы хотите больше узнать о людях и истории, обратите внимание на разделы со ссылками в главах, имена, перечисленные в исходном коде Linux, а также на историю репозитория Linux и файл MAINTAINERS в исходном коде Linux. В разделе «Благодарности» моей книги о BPF [Gregg 19] также перечислены различные технологии, в частности расширенный BPF, BCC, bpftrace, kprobes и uprobes, и люди, стоящие за ними. Джон Олспоу (John Allspaw): планирование мощности [Allspaw 08]. Джин М. Амдал (Gene M. Amdahl): ранние работы над масштабируемостью компьютерных систем [Amdahl 67]. Йенс Аксбо (Jens Axboe): планировщик ввода/вывода CFQ, fio, blktrace, io_uring. Бренден Бланко (Brenden Blanco): BCC. Джефф Бонвик (Jeff Bonwick): автор механизма распределения памяти для объектов (slab allocation), соавтор механизма распределения памяти для объектов в пространстве пользователя, соавтор файловой системы ZFS, kstat, первый разработчик mpstat. Даниэль Боркманн (Daniel Borkmann): соавтор и мейнтейнер расширенного BPF. Рох Бурбонне (Roch Bourbonnais): эксперт в области анализа производительности систем Sun Microsystems. Тим Брей (Tim Bray): автор микробенчмарка Bonnie дискового ввода/вывода, известен как один из авторов спецификации XML. Брайан Кэнтрилл (Bryan Cantrill): соавтор DTrace; Oracle ZFS Storage Appliance Analytics.

970  Приложение E. Производительность систем, кто есть кто Реми Кард (Rémy Card): основной разработчик файловых систем ext2 и ext3. Надя Иветт Чамберс (Nadia Yvette Chambers): файловая система hugetlbfs для Linux. Гийом Шазарен (Guillaume Chazarain): iotop(1) для Linux. Адриан Кокрофт (Adrian Cockcroft): книги об анализе производительности [Cockcroft 95], [Cockcroft 98], Virtual Adrian (SE Toolkit). Тим Кук (Tim Cook): nicstat(1) для Linux и расширения. Алан Кокс (Alan Cox): анализ производительности сетевого стека Linux. Матье Дезнойерс (Mathieu Desnoyers): Linux Trace Toolkit (LTTng), точки трассировки ядра, ведущий автор RCU для пространства пользователя. Фрэнк Ч. Эйглер (Frank Ch. Eigler): ведущий разработчик SystemTap. Ричард Эллинг (Richard Elling): методология статической настройки производительности. Джулия Эванс (Julia Evans): документация и инструменты анализа производительности и отладки. Кевин Роберт Эльз (Kevin Robert Elz): DNLC. Роджер Фолкнер (Roger Faulkner): разработчик файловой системы /proc для UNIX System V, реализация потоков выполнения для Solaris и трассировщик системных вызовов truss(1). Томас Глейкснер (Thomas Gleixner): различные разработки для повышения производительности ядра Linux, включая таймеры высокого разрешения. Себастьян Годар (Sebastian Godard): пакет sysstat для Linux, содержащий множество инструментов анализа производительности, включая iostat(1), mpstat(1), pidstat(1), nfsiostat(1), cifsiostat(1), расширенную версию sar(1), sadc(8), sadf(1) (см. метрики в приложении B). Саша Гольдштейн (Sasha Goldshtein): инструменты BPF (argdist(8), trace(8) и др.), один из разработчиков BCC. Брендан Грегг (Brendan Gregg): nicstat(1), DTraceToolkit, ZFS L2ARC, инструменты BPF (execsnoop, biosnoop, ext4slower, tcptop и др.), один из разработчиков BCC/bpftrace, метод USE, тепловые карты (задержек, потребления, субсекундное смещение), флейм-графики, эта книга и предыдущая [Gregg 11a], [Gregg 19], другие работы, связанные с анализом производительности. Доктор Нил Гюнтер (Dr. Neil Gunther): универсальный закон масштабируемости, тройные графики потребления процессора, книги по производительности [Gunther 97]. Джеффри Холлингсворт (Jeffrey Hollingsworth): динамическая инструментация [Hollingsworth 94].

Производительность систем, кто есть кто  971 Ван Якобсон (Van Jacobson): traceroute(8), pathchar, анализ производительности TCP/IP. Радж Джайн (Raj Jain): теория производительности систем [Jain 91]. Джерри Елинек (Jerry Jelinek): зоны в Solaris. Билл Джой (Bill Joy): vmstat(1), участие в разработке виртуальной памяти BSD, анализ производительности TCP/IP, FFS. Энди Клин (Andi Kleen): анализ производительности Intel, участие в разработке ядра Linux. Кристоф Ламетер (Christoph Lameter): распределитель SLUB. Уильям Лефебвр (William LeFebvre): автор первой версии top(1), вдохновившей на создание многих других инструментов. Дэвид Левинталь (David Levinthal): эксперт по производительности процессоров Intel. Джон Левон (John Levon): OProfile. Майк Лукидес (Mike Loukides): первая книга об анализе производительности систем Unix [Loukides 90], заложившая и поддержавшая традицию анализа на основе ресурсов: процессор, память, диск, сеть. Роберт Лав (Robert Love): разработки, связанные с увеличением производительности ядра Linux, включая вытеснение. Мэри Марчини (Mary Marchini): libstapsdt: поддержка зондов USDT для разных языков. Джим Мауро (Jim Mauro): соавтор книг «Solaris Performance and Tools» [McDougall 06a] и «DTrace: Dynamic Tracing in Oracle Solaris, Mac OS X, and FreeBSD» [Gregg 11]. Ричард Макдугалл (Richard McDougall): система учета микросостояний в Solaris, соавтор «Solaris Performance and Tools» [McDougall 06a]. Маршалл Кирк Маккьюсик (Marshall Kirk McKusick): FFS, один из разработчиков BSD. Арнальдо Карвальо де Мело (Arnaldo Carvalho de Melo): мейнтейнер perf(1) в Linux. Бартон Миллер (Barton Miller): динамическая инструментация [Hollingsworth 94]. Дэвид С. Миллер (David S. Miller): мейнтейнер сетевого стека и архитектуры SPARC в Linux. Многочисленные улучшения производительности и поддержка расширенного BPF. Кэри Миллсап (Cary Millsap): метод R. Инго Молнар (Ingo Molnar): планировщик O(1), абсолютно справедливый планировщик, добровольное вытеснение ядра, ftrace, perf и работа над механизмом

972  Приложение E. Производительность систем, кто есть кто вытеснения в реальном времени, мьютексы, фьютексы, профилирование планировщика, очереди заданий. Ричард Дж. Мур (Richard J. Moore): DProbes, kprobes. Эндрю Мортон (Andrew Morton): fadvise, read-ahead. Джан-Паоло Д. Мусумеси (Gian-Paolo D. Musumeci): «System Performance Tuning, 2nd Ed.» [Musumeci 02]. Майк Муусс (Mike Muuss): ping(8). Шайлаб Нагар (Shailabh Nagar): учет задержек, taskstats. Рич Петтит (Rich Pettit): SE Toolkit. Ник Пиггин (Nick Piggin): планировщики Linux. Билл Пиевски (Bill Pijewski): Solaris vfsstat(1M), регулирование ввода/вывода в ZFS. Деннис Ричи (Dennis Ritchie): Unix и ее механизмы обеспечения производительности: приоритеты процессов, подкачка, кэш буферов и т. д. Аластер Робертсон (Alastair Robertson): создатель bpftrace. Стивен Ростедт (Steven Rostedt): Ftrace, KernelShark, поддержка реального времени в Linux, адаптивные мьютексы, поддержка трассировки в Linux. Расти Рассел (Rusty Russell): оригинальные фьютексы, участие в разработке ядра Linux. Майкл Шапиро (Michael Shapiro): соавтор DTrace. Алексей Шипилёв (Aleksey Shipilеv): эксперт по производительности Java. Бальбир Сингх (Balbir Singh): контроллер памяти в Linux, учет задержек, taskstats, cgroupstats, учет производительности процессоров. Юнхонг Сон (Yonghong Song): BTF и участие в разработке расширенного BPF и BCC. Алексей Старовойтов (Alexei Starovoitov): соавтор и мейнтейнер расширенного BPF. Кен Томпсон (Ken Thompson): Unix и ее механизмы обеспечения производительности: приоритеты процессов, подкачка, кэш буферов и т. д. Мартин Томпсон (Martin Thompson): Mechanical Sympathy. Линус Торвальдс (Linus Torvalds): ядро Linux и многочисленные базовые компоненты, необходимые для производительности системы, планировщик ввода/ вывода для Linux, Git. Арьян ван де Вен (Arjan van de Ven): latencytop, PowerTOP, irqbalance, профилирование планировщика Linux. Ницан Вакарт (Nitsan Wakart): эксперт по производительности Java.

E.1. Ссылки  973 Тобиас Вальдекранц (Tobias Waldekranz): ply (первый высокоуровневый трассировщик в BPF). Даг Вирс (Dag Wieers): dstat. Карим Ягмур (Karim Yaghmour): LTT, развитие трассировки в Linux. Джови Чжанвэй (Jovi Zhangwei): ktap. Том Занусси (Tom Zanussi): триггеры hist в Ftrace. Питер Зийлстра (Peter Zijlstra): реализация адаптивных мьютексов, фреймворк обратных вызовов для аппаратных прерываний, другие работы в сфере анализа производительности Linux.

E.1. ССЫЛКИ [Amdahl 67] Amdahl, G., «Validity of the Single Processor Approach to Achieving Large Scale Computing Capabilities», AFIPS, 1967. [Libes 89] Libes, D., and Ressler, S., «Life with UNIX: A Guide for Everyone», Prentice Hall, 1989. [Loukides 90] Loukides, M., «System Performance Tuning», O’Reilly, 1990. [Hollingsworth 94] Hollingsworth, J., Miller, B., and Cargille, J., «Dynamic Program Instrumentation for Scalable Performance Tools», Scalable High-Performance Computing Conference (SHPCC), May 1994. [Cockcroft 95] Cockcroft, A., «Sun Performance and Tuning», Prentice Hall, 1995. [Cockcroft 98] Cockcroft, A., and Pettit, R., «Sun Performance and Tuning: Java and the Internet», Prentice Hall, 1998. [Musumeci 02] Musumeci, G. D., and Loukidas, M., «System Performance Tuning, 2nd Edition», O’Reilly, 20021. [McDougall 06a] McDougall, R., Mauro, J., and Gregg, B., «Solaris Performance and Tools: DTrace and MDB Techniques for Solaris 10 and OpenSolaris», Prentice Hall, 2006. [Gunther 07] Gunther, N., «Guerrilla Capacity Planning», Springer, 2007. [Allspaw 08] Allspaw, J., «The Art of Capacity Planning», O’Reilly, 20082. [Gregg 11a] Gregg, B., and Mauro, J., «DTrace: Dynamic Tracing in Oracle Solaris, Mac OS X and FreeBSD», Prentice Hall, 2011. [Gregg 19] Gregg, B., «BPF Performance Tools: Linux System and Application Observability»3, Addison-Wesley, 2019. 1

Джан-Паоло Д. Мусумеси, Майк Лукидес. «Настройка производительности UNIX-систем, 2-е издание». Джон Оллспоу. «Искусство планирования мощностей». СПб., издательство «Питер».

2

Грегг Б. «BPF: Профессиональная оценка производительности». Выходит в издательстве «Питер» в 2023 году.

3

ГЛОССАРИЙ ABI

Application Binary Interface — прикладной двоичный интерфейс.

ACK

Пакет подтверждения в TCP.

AMD

Производитель процессоров.

API

Application Programming Interface — прикладной программный интерфейс.

ARM

Производитель процессоров.

ARP

Address Resolution Protocol — протокол разрешения адресов.

ASIC

Application-Specific Integrated Circuit — специализированная интегральная схема.

AT&T

Компания American Telephone and Telegraph Company, в состав которой входила Bell Laboratories, где была разработана ОС Unix.

BCC

BPF Compiler Collection — коллекция компиляторов BPF. BCC — это проект, включающий компилятор для фреймворка BPF, а также множество инструментов анализа производительности BPF. См. главу 15.

BIOS

Basic Input/Output System — базовая система ввода/вывода. Прошивка, выполняющая инициализацию компьютерного оборудования и управляющая процессом загрузки.

BPF

Berkeley Packet Filter — фильтр пакетов Беркли. Легковесная технология 1992 года, встроенная в ядро и созданная для увеличения производительности фильтрации пакетов. С 2014 года была расширена и превратилась в среду выполнения общего назначения (см. eBPF).

BSD

Berkeley Software Distribution, производная Unix.

C

Язык программирования C.

CDN

Content Delivery Network — сеть доставки содержимого.

CPI

Cycles Per Instruction — тактов на инструкцию. См. главу 6 «Процессоры».

CSV

Comma Separated Values — значения, разделенные запятыми. Формат файлов.

Глоссарий   975 CTSS

Compatible Time-Sharing System — совместимая система с разделением времени. Одна из первых систем с разделением времени.

DEC

Digital Equipment Corporation.

DNS

Domain Name Service — служба доменных имен.

DRAM

Dynamic Random-Access Memory — динамическая память с произвольным доступом. Тип энергозависимой памяти, обычно используемой в качестве основной.

eBPF

Extended BPF — расширенный BPF (см. BPF). Аббревиатура eBPF первоначально описывала расширенную версию BPF 2014 года с обновленными размером регистра и набором инструкций, с дополнительной поддержкой хранилища карт и ограниченной возможностью вызова функций ядра. В 2015 году eBPF решили называть просто BPF.

ECC

Error-Correcting Code — код исправления ошибки. Алгоритм обнаружения ошибок и исправления некоторых из них (обычно однобитовых).

ELF

Executable and Linkable Format — формат исполнимых и компонуемых модулей. Распространенный формат файлов для исполняемых программ.

errno

Переменная, содержащая последнюю ошибку в виде числа согласно стандарту (POSIX.1-2001).

Ethernet Набор стандартов для сетей физического уровня и уровня передачи данных. FPGA

Field-Programmable Gate Array — программируемая логическая матрица. Перепрограммируемая интегральная схема, используемая в вычислениях для ускорения определенной операции.

FreeBSD Unix-подобная операционная система с открытым исходным кодом. fsck

Команда проверки файловой системы. Используется для восстановления файловых систем после сбоя, например, из-за отключения питания или краха ядра. Этот процесс может занять несколько часов.

Gbps

Gigabits per second — гигабит в секунду.

GUI

Graphical User Interface — графический интерфейс пользователя.

HDD

Hard Disk Drive — привод жесткого диска. Устройство хранения с вращающимися магнитными дисками. См. главу 9 «Диски».

HTTP

Hyper Text Transfer Protocol — протокол передачи гипертекста.

ICMP

Internet Control Message Protocol — протокол управляющих сообщений интернета. Используется командой ping(1) (ICMP echo request/reply).

976  Глоссарий I/O

Input/Output — ввод/вывод.

IO Visor Проект Linux Foundation. Его репозитории bcc и bpftrace размещены в GitHub, что упрощает сотрудничество между разработчиками BPF в разных компаниях. IOPS

I/O Operations Per Second — операций ввода/вывода в секунду. Мера скорости ввода/вывода.

Intel

Производитель процессоров.

IP

Internet Protocol — протокол интернета. Основные версии — IPv4 и IPv6. См. главу 10 «Сети».

IPC

Может означать: либо количество инструкций на такт (Iinstructions Per Cycle) — низкоуровневая метрика производительности процессора, либо межпроцессные взаимодействия (Inter-Process Communication) — средства обмена данными между процессами. Один из примеров механизмов межпроцессных взаимодействий — сокеты.

IRIX

Производная Unix. Операционная система, разработанная в Silicon Graphics, Inc. (SGI).

IRQ

Interrupt Request — запрос на прерывание. Аппаратный сигнал процессору, запрашивающий выполнение некоторого задания. См. главу 3 «Операционные системы».

kprobes

Технология Linux для динамической инструментации ядра.

LRU

Least Recently Used — наиболее давно использовавшийся. См. главу 2 «Методологии», раздел 2.3.14 «Кэширование».

malloc

Выделение памяти — memory allocate. Под этим термином обычно подразу­мевают функцию, реализующую выделение памяти.

Mbps

Megabits per second — мегабит в секунду.

MMU

Memory Management Unit — блок управления памятью. Отвечает за представление памяти процессору и преобразование виртуальных адресов в физические.

mysqld

Демон базы данных MySQL.

NVMe

Non-Volatile Memory express — энергонезависимая память. Спецификация шины PCIe для устройств хранения.

PC

Program Counter — счетчик инструкций. Регистр процессора с адресом выполняемой в данный момент инструкции.

Глоссарий   977 PCID

Process-Context ID — идентификатор контекста процесса. Особенность процессора/MMU для маркировки виртуальных адресов идентификатором процесса. Это помогает избежать сброса кэша при переключении контекста.

PCIe

Peripheral Component Interconnect Express: стандарт шины, широко используемый для контроллеров устройств хранения и сетевых контроллеров.

PDP

Programmed Data Processor, серия мини-компьютеров, производившихся компанией Digital Equipment Corporation (DEC).

PEBS

Precise Event-Based Sampling, иногда Processor Event-Based Sampling — точная выборка инструкций на основе событий. Технология, реализованная в процессорах Intel для использования со счетчиками PMC. Она обеспечивает более точное определение состояния процессора во время событий.

PID

Process Identifier — идентификатор процесса. Уникальный числовой идентификатор процесса в операционной системе.

PMC

Performance Monitoring Counters — счетчики мониторинга производительности. Специальные аппаратные регистры процессора, которые можно запрограммировать для подсчета низкоуровневых событий: тактов, тактов простоя, инструкций, чтения/записи в память и т. д.

POSIX

Portable Operating System Interface — переносимый интерфейс операционных систем. Семейство стандартов, которыми управляет IEEE для определения Unix API. Сюда входит интерфейс файловой системы, используемый приложениями и предоставляемый через системные вызовы или системные библиотеки, использующие системные вызовы.

PSI

Pressure Stall Information. Набор метрик, используемых для выявления проблем с производительностью, обусловленных ресурсами.

RCU

Read-Copy-Update: механизм синхронизации в Linux.

RFC

Request For Comments — запрос на комментарии. Общедоступный документ, который публикует Рабочая группа по проектированию интернета (Internet Engineering Task Force, IETF) для обмена сетевыми стандартами и передовым опытом. Документы RFC используются для определения сетевых протоколов: RFC 793, например, определяет протокол TCP.

RSS

Resident Set Size — размер резидентного набора. Мера основной памяти.

ROI

Return On Investment — показатель возврата инвестиций, бизнес-­метрика.

RX

Прием (используется в сетевых технологиях).

978  Глоссарий SCSI

Small Computer System Interface — интерфейс малых вычислительных систем. Стандарт интерфейса для устройств хранения.

SLA

Service Level Agreement — соглашение об уровне обслуживания.

SMP

Symmetric Multiprocessing — симметричная многопроцессорность. Многопроцессорная архитектура, в которой несколько одинаковых процессоров совместно используют одну и ту же основную память.

SMT

Simultaneous Multithreading — одновременная многопоточность. Особенность процессора, позволяющая запускать несколько потоков на ядрах. См. гиперпоточность.

SNMP

Simple Network Management Protocol — простой протокол управления сетью.

Solaris

Операционная система Unix, первоначально разработанная в  Sun Microsystems. Славилась своей масштабируемостью и надежностью и была популярна в корпоративных средах. После покупки Sun корпорацией Oracle была переименована в Oracle Solaris.

SONET

Synchronous Optical Networking, физический протокол для оптоволоконных линий.

SPARC

Архитектура процессоров (от Scalable Processor Architecture) — масштабируемая архитектура процессоров.

SRE

Site Reliability Engineer: технический сотрудник, специализирующийся на инфраструктуре и надежности. SRE-инженеры анализируют производительность в ходе расследования инцидентов в короткие сроки.

SSD

Solid-State Drive — твердотельный накопитель. Устройство хранения, обычно на основе флеш-памяти. См. главу 9 «Диски».

SSH

Secure Shell. Безопасный протокол обмена с удаленной командной оболочкой.

SunOS

Операционная система Sun Microsystems Operating System. Позднее была переименована в Solaris.

SUT

System Under Test — тестируемая система.

SVG

Scalable Vector Graphs — масштабируемая векторная графика. Формат файлов.

SYN TCP

TCP-пакет синхронизации. Transmission Control Protocol — протокол управления передачей. Первоначально определен в RFC 793. См. главу 10 «Сети».

Глоссарий   979 TENEX

Операционная система TEN-Extended, основанная на TOPS-10 для PDP-10.

TLB

Translation Lookaside Buffer — буфер ассоциативной трансляции. Кэш для трансляции адресов в системах виртуальной памяти, который используется блоком управления памяти MMU (см. MMU).

TLS

Transport Layer Security — протокол безопасности транспортного уровня. Используется для шифрования данных, передаваемых по сети.

TPU

Tensor Processing Unit — тензорный процессор. Специализированная интегральная схема, ускоритель вычислений для задач искусственного интеллекта и машинного обучения, разработанный в Google и названный в честь TensorFlow (программная платформа для машинного обучения).

TX

Передача (используется в сетевых технологиях).

UDP

User Datagram Protocol — протокол пользовательских дейтаграмм. Первоначально определен в RFC 768. См. главу 10 «Сети».

uprobes

Технология ядра Linux для динамической инструментации программного обеспечения пользовательского уровня.

us

Микросекунды. Обычно используется аббревиатура μs, но в выводе инструментов оценки производительности вы часто будете видеть «us». (Обратите внимание, что вывод vmstat(8), примеры которого много раз приводились в этой книге, включает столбец us, сокращенно от «user time» — время в пространстве пользователя.)

μs

Микросекунды. См. us.

USDT

User-land Statically Defined Tracing — статически определяемые точки трассировки на стороне пользователя. Включает размещение программистом статических зондов трассировки в коде приложения в местах, где это может пригодиться.

vCPU

Виртуальный процессор. Современные процессоры могут предоставлять несколько виртуальных процессоров на каждое ядро (например, благодаря технологии гиперпоточности Intel).

VFS

Virtual File System — виртуальная файловая система. Абстракция, используемая ядром для единообразной поддержки файловых систем разных типов.

VMS

Virtual Memory System — система виртуальной памяти. Операционная система, выпущенная компанией DEC.

x86

Архитектура процессоров, основанная на Intel 8086.

ZFS

Комбинированная файловая система и диспетчер томов, созданные в Sun Microsystems.

980  Глоссарий Адаптивный мьютекс (adaptive mutex)

Мьютекс (mutual exclusion — взаимоисключающая блокировка) — тип блокировки, использующийся для синхронизации. См. главу 5 «Приложения», раздел 5.2.5 «Конкурентность и параллелизм».

Адрес (address)

Местоположение в памяти.

Адресное пространство (address space)

Контекст виртуальной памяти. См. главу 7 «Память».

Ассоциативный массив (associative array)

Тип данных в языках программирования, в котором доступ к элементам производится с помощью произвольного ключа или нескольких ключей.

Байт (byte)

Единица представления цифровых данных. Эта книга следует промышленному стандарту, согласно которому один байт равен восьми битам, а бит — это двоичная цифра, ноль или один.

Бенчмарк (benchmark)

В информационных технологиях бенчмарком называют инструмент, помогающий создать экспериментальную рабочую нагрузку и измерить производительность: результат бенчмаркинга. Обычно бенчмарки используются для оценки влияния различных параметров настройки на производительность.

Буфер (buffer)

Область памяти для временного хранения данных.

Вне процессора (off-CPU)

Этот термин применяется к потокам, которые в данный момент не выполняются на процессоре, то есть находятся «вне процессора», например, ожидая завершения ввода/ вывода, простаивая на блокировке, добровольно перей­дя в режим ожидания или вследствие некоторого другого события.

Внешний интерфейс, внешняя часть (front end)

Обозначает интерфейс конечного пользователя и программное обеспечение, представляющее информацию. Веб-приложение — это внешний интерфейс. См. внутренний интерфейс.

Внутренний интерфейс, внутренняя часть (back end)

Этот термин относится к хранилищам данных и компонентам инфраструктуры. Веб-сервер  — это внутренняя часть программного комплекса. См. внешний интерфейс.

Виртуальная память (virtual memory)

Абстракция основной памяти, поддерживающая многозадачность и возможность выделения большего объема памяти, чем есть.

Глоссарий   981 Выборка (sampling)

Метод наблюдения за целью путем проведения подмножества измерений: выборки.

Герц (Гц)

Такты в секунду.

Гиперпоточность (hyperthread)

Реализация SMT (Simultaneous Multithreading — одновременная многопоточность), выполненная компанией Intel. Технология масштабирования процессоров, позволяющая ОС создавать несколько виртуальных процессоров для одного ядра и планировать задания для них, которые процессор будет пытаться выполнять параллельно.

Графический процессор (Graphics Processing Unit, GPU)

Графические процессоры можно использовать для обработки некоторых рабочих нагрузок, например для машинного обучения.

Дейтаграмма (datagram)

См. сегмент.

Демон (daemon)

Системная программа, выполняющаяся постоянно для предоставления услуги.

Дескриптор файла (file descriptor)

Идентификатор, используемый программой для обращения к открытому файлу.

Динамическая трассировка (dynamic tracing)

Может означать ПО, реализующее динамическую инструментацию.

Динамическая инструментация (dynamic instrumentation)

Динамическая инструментация, или динамическая трассировка, — это технология, позволяющая инструментировать любое программное событие, включая вызовы функций и возврат из них, путем изменения команд в реальном времени и вставки временных команд трассировки.

Диск (disk)

Физическое устройство для хранения данных. См. также HDD и SSD.

Дуплекс (duplex)

Режим, когда обмен данными происходит одновременно в обоих направлениях.

Загрузка/выгрузка страниц (pagein/ pageout)

Действия, выполняемые ОС (ядром) для перемещения фрагментов памяти (страниц) с внешних устройств хранения и обратно.

Задача (task)

Запущенный выполняемый объект, который может быть процессом, потоком в многопоточном процессе или потоком ядра. См. главу 3 «Операционные системы».

982  Глоссарий Задержка (latency)

Время, потраченное на ожидание. В анализе производительности этот термин часто используется для описания времени ввода/вывода. Задержка играет важную роль, потому что часто она — наиболее эффективное средство оценки проблемы с производительностью. Где именно измеряется задержка, не всегда ясно без дополнительных уточнений. Например, «задержка диска» может означать время, потраченное на ожидание только в очереди драйвера диска, или все время ожидания завершения дискового ввода/вывода, включая и время ожидания в очереди, и время обслуживания. Задержка имеет нижнюю границу, полоса пропускания — верхнюю.

Значительный сбой (major fault)

Сбой доступа к памяти, требующий подкачки с запоминающих устройств (дисков). См. главу 3 «Операционные системы».

Карта расширения (expander card)

Физическое устройство (карта), подключенное к системе, обычно для подключения дополнительных контроллеров ввода/вывода.

Килобайт (Кбайт)

Международная система единиц (СИ) определяет килобайт как 1000 байт, но в информатике килобайт обычно составляет 1024 байта (в системе СИ обозначается как кибибайт). Инструменты, описываемые в книге, которые сообщают размеры данных в килобайтах, обычно используют определение 1024 (210) байт.

Клиент (client)

Потребитель сетевой службы. Обычно клиентом называют клиентский хост или клиентское приложение.

Кольцо процессора (processor ring)

Режим защиты процессора.

Команда (command)

Программа, выполняемая в командной оболочке.

Командная оболочка (shell)

Интерпретатор командной строки и язык сценариев.

Конкурентность (concurrency)

См. главу 5 «Приложения», раздел 5.2.5 «Конкурентность и параллелизм».

Конкуренция (contention)

Состязание за обладание ресурсом.

Глоссарий   983 Контроллер диска (disk controller)

Компонент, управляющий подключенными дисками и обеспечивающий их доступность для системы напрямую или через виртуальные диски. Контроллеры дисков могут встраиваться в системную материнскую плату, подключаться как карты расширения или встраиваться в массивы хранения. Они поддерживают один или несколько типов интерфейсов хранения (например, SCSI, SATA, SAS) и обычно называются адаптерами главной шины (Host Bus Adapter, HBA) с указанием типа интерфейса, например SAS HBA.

Коэффициент попаданий (hit ratio)

Часто используется для описания производительности кэша: отношение попаданий в кэш к общему числу обращений к кэшу (число попаданий плюс число промахов), выражается в процентах. Чем выше, тем лучше.

Логический процесДругое название виртуального процессора. См. главу 6 сор (logical processor) «Процессоры». Локальные диски (local disks)

Диски, подключенные непосредственно к серверу. К локальным относятся диски внутри корпуса сервера и диски, подключенные напрямую через транспортную шину.

Массив (array)

Набор значений. Тип данных в языках программирования. В низкоуровневых языках массивы обычно хранятся в непрерывных областях памяти, а индекс определяет смещение элемента массива от начала этой области.

Массив хранения (storage array)

Набор дисков в одном корпусе, который можно подключить к системе. Массивы хранения обычно предлагают различные дополнительные возможности для повышения надежности и производительности.

Материнская плата (main board)

Печатная плата, на которой размещены процессоры и системные соединения. Также называется системной платой.

Мегабайт (Мбайт)

Международная система единиц (СИ) определяет мегабайт как 1 000 000 байт, но в информатике мегабайт обычно составляет 1 048 576 байт (в системе СИ обозначается как мебибайт). Инструменты, описываемые в книге, которые сообщают размеры данных в мегабайтах, обычно используют определение 1 048 576 (220) байт.

984  Глоссарий Микробенчмарк (microbenchmark)

Бенчмарк для оценки одной или простой операции.

Мьютекс (mutex)

Взаимоисключающая блокировка. Мьютексы могут быть узким местом и часто исследуются на предмет проблем с производительностью. См. главу 5 «Приложения».

На процессоре (on-CPU)

Этот термин применяется к потокам, которые в текущий момент времени выполняются на процессоре.

Наблюдаемость (observability)

Возможность наблюдения за вычислительной системой. Включает методики и инструменты, используемые для анализа состояния вычислительных систем.

Настраиваемый параметр (tunable parameter)

Параметр для настройки различных аспектов работы программного обеспечения.

Незначительный сбой (minor fault)

Сбой доступа к памяти, не требующий подкачки с запоминающих устройств (дисков). См. главу 3 «Операционные системы».

ОС (OS)

Операционная система. Набор ПО, включая ядро, осуществляющий управление ресурсами и процессами пользовательского уровня.

Основная память (main memory)

Оперативная память системы, обычно реализованная на микросхемах DRAM.

Отказ страницы (page fault)

Системное прерывание; срабатывает, когда программа ссылается на область виртуальной памяти, которая еще не отображена в физическую память. Это нормальное следствие модели распределения памяти по требованию.

Очередь на выполнение (run queue)

Очередь планировщика процессора, в которой находятся задачи, ожидающие своей очереди выполнения на процессоре. В реальности очередь может быть реализована в виде древовидной структуры, но до сих пор широко используется термин «очередь на выполнение».

Пакет (packet)

Сетевое сообщение на сетевом уровне сетевой модели OSI (см. раздел 10.2.3).

Память (memory)

См. основная память.

Параллельное выполнение (concurrency)

См. главу 5 «Приложения», раздел 5.2.5 «Конкуренция и параллелизм».

Глоссарий   985 Перекрестный вызов (cross call)

См. перекрестный вызов процессоров.

Перекрестный вызов процессоров (CPU cross call)

Вызовы между процессорами в многопроцессорной системе для передачи заданий друг другу. Перекрестные вызовы могут выполняться для обработки общесистемных событий, например для согласования записей в кэше процессора. См. главу 6 «Процессоры». В Linux эти вызовы называются вызовами SMP.

Переменная (variable)

Именованный объект, использующийся в языках программирования для хранения данных.

Перформанс-инженер (performance engineer)

Технический сотрудник, занимающийся в основном вопросами производительности: планированием, оценкой, анализом и улучшениями. См. главу 1 «Введение», раздел 1.2 «Роли».

Полоса пропускания (bandwidth)

Частотный диапазон канала связи, измеряемый в герцах. В информатике под полосой пропускания под­ разумевается максимальная скорость передачи для канала связи. Этот термин также часто ошибочно используют для описания текущей пропускной способности (throughput).

Попадание в кэш (cache hit)

Запрос на получение данных, которые могут быть возвращены из кэша.

Поток/поток выполнения (thread)

Программная абстракция выполняемого экземпляра  программы, которую можно запланировать для ­в ыполнения на процессоре. Ядро имеет несколько потоков выполнения, а процесс содержит один или несколько таких потоков. См. главу 3 «Операционные системы».

Приложение (application)

Программа, обычно выполняющаяся в  пространстве пользователя.

Промах кэша (cache miss)

Запрос на получение данных, которые не могут быть возвращены из кэша.

Промышленная среда/продакшен (production)

Термин, использующийся для описания рабочей нагрузки, создаваемой реальными клиентами, и среды, которая их обрабатывает. У многих компаний есть «тестовая» среда, обслуживающая синтетические рабочие нагрузки и предназначенная для тестирования ПО перед развертыванием в промышленной среде.

986  Глоссарий Пропускная способность (throughput)

Для сетевых устройств под пропускной способностью обычно понимается скорость передачи данных в битах в секунду или байтах в секунду. В статическом анализе под пропускной способностью также может пониматься частота выполнения операций ввода/вывода в секунду (IOPS).

Пространство пользователя (user-space)

Адресное пространство процессов пользовательского уровня.

Пространство ядра (kernel-space)

Адресное пространство ядра.

Профилирование (profiling)

Метод сбора данных, характеризующих эффективность цели. Распространенный метод профилирования — выборка по времени (см. выборка).

Процесс (process)

Выполняющийся экземпляр программы. Многозадачные ОС могут выполнять несколько процессов одновременно и запускать несколько экземпляров одной и той же программы в разных процессах. Процесс содержит один или несколько потоков, выполняющихся на процессорах.

Процессор (Central Processing Unit, CPU)

Этим термином обозначается набор функциональных блоков, которые выполняют инструкции, включая регистры и арифметико-логический блок (Arithmetic Logic Unit, ALU). Используется для обозначения физических и виртуальных процессоров.

Прошивка (firmware)

Программное обеспечение, встроенное в устройство.

Рабочая нагрузка (workload)

Описывает запросы к системе или ресурсам.

Рабочая нагрузка реального времени (real-time workload)

Рабочая нагрузка, имеющая фиксированные требования к задержкам, обычно довольно низкие.

Регистры (registers)

Маленькие запоминающие устройства в процессоре, используемые машинными инструкциями при обработке данных.

Сбалансированная система (balanced system)

Система без узких мест.

Сегмент (segment)

Сообщение на транспортном уровне сетевой модели OSI (см. раздел 10.2.3 «Стек протоколов»).

Глоссарий   987 Сектор (sector)

Минимальная единица данных для устройств хранения, обычно 512 байт или 4 Кбайт. См. главу 9 «Диски».

Сервер (server)

В сети — сетевой хост, который предоставляет услуги сетевым клиентам, например HTTP-сервер или сервер базы данных. Термин «сервер» также может относиться к физической системе.

Системный вызов (system call/syscall)

Интерфейс для процессов, запрашивающих выполнение привилегированных действий у ядра. См. главу 3 «Операционные системы».

Сокет (socket)

Программная абстракция, представляющая конечную точку сетевого соединения.

Статическая инструментация/ трассировка (static instrumentation/ tracing)

Инструментация ПО с использованием предварительно скомпилированных точек зондирования. См. главу 4 «Инструменты наблюдения».

Стек (stack)

В контексте инструментов наблюдения под стеком обычно понимаются трассировки стека.

Сторона пользователя Относится к ПО и файлам пользовательского уров(user-land) ня, включая исполняемые программы в /usr/bin, /lib и т. д. Сторона ядра (kernel-land)

Программное обеспечение ядра.

Страница (page)

Фрагмент памяти, управляемой ядром и процессором. Вся память, использующаяся системой, разбита на страницы. Обычно используются страницы размерами 4 Кбайт и 2 Мбайт (в зависимости от процессора).

Структура (struct)

Экземпляр структуры, обычно в языке C.

Сценарий/скрипт (script)

В информатике — выполняемая программа, обычно короткая и написанная на языке высокого уровня.

Такт процессора (CPU cycle)

Единица времени, основанная на тактовой частоте процессора: при тактовой частоте 2 ГГц каждый такт длится 0,5 нс. Сам такт представляет собой электрический сигнал — перепад напряжения, запускающий цифровую логику.

Теплота кэша (cash warmth)

См. главу 2 «Методологии», раздел 2.3.14 «Кэширование», подраздел «Горячие, холодные и теплые кэши».

988  Глоссарий Точки трассировки (tracepoints)

Технология ядра Linux для поддержки статической инструментации.

Трассировка (tracing)

Наблюдение за событиями.

Трассировка стека (stack trace)

Стек вызовов, состоящий из нескольких фреймов стека, охватывающих путь выполнения кода. Трассировки стека часто исследуют в ходе анализа производительности, особенно при профилировании процессора.

Трассировщик (tracer)

Инструмент для трассировки. См. трассировка.

Удаленные диски (remote disks)

Диски (включая виртуальные), которые используются сервером, но подключены к удаленной системе.

Узкое место (bottleneck)

Что-то, ограничивающее производительность.

Уровень пользователя Режим привилегий процессора, используемый при выпол(user-level) нении на уровне пользователя. Это более низкий уровень привилегий, чем у ядра. Он запрещает прямой доступ к ресурсам, вынуждая ПО пользовательского уровня запрашивать доступ к ним через ядро. Уровень ядра (kernel-level)

Уровень привилегий процессора, устанавливаемый при выполнении кода ядра.

Файл с отладочной информацией (debuginfo file)

Файл с символами и отладочной информацией, которые используют отладчики и профилировщики.

Флейм-график/ график пламени (flame graph)

Метод визуализации набора трассировок стека. См. главу 2 «Методологии».

Фрейм (frame)

Сообщение на канальном уровне сетевой модели OSI (см. раздел 10.2.3 «Стек протоколов»).

Фрейм стека (stack frame)

Структура данных, содержащая информацию о состоянии функции, включая ее аргументы и адрес возврата.

Хост (host)

Система, подключенная к сети. Также называют сетевым узлом.

Частота операций (operation rate)

Количество операций, выполняемых в единицу времени (например, операций в секунду), могут подразумеваться любые операции, в том числе не связанные с вводом/ выводом.

Глоссарий   989 Экземпляр (instance)

Виртуальный сервер. Облачные среды предоставляют экземпляры серверов.

Ядро (core)

Конвейер выполнения в процессоре. ОС может видеть каждое ядро как отдельный процессор или, если поддерживается технология гиперпоточности, как несколько процессоров.

Ядро (kernel)

Базовая программа в системе, которая работает в привилегированном режиме и управляет ресурсами и процессами уровня пользователя.

Брендан Грегг Производительность систем Перевел с английского А. Киселев



Руководитель дивизиона Ю. Сергиенко Руководитель проекта А. Питиримов Ведущий редактор Е. Строганова Литературный редактор К. Тульцева Художественный редактор В. Мостипан Корректоры С. Беляева, Н. Викторова Верстка Л. Егорова

Изготовлено в России. Изготовитель: ООО «Прогресс книга». Место нахождения и фактический адрес: 194044, Россия, г. Санкт-Петербург, Б. Сампсониевский пр., д. 29А, пом. 52. Тел.: +78127037373. Дата изготовления: 05.2023. Наименование: книжная продукция. Срок годности: не ограничен. Налоговая льгота — общероссийский классификатор продукции ОК 034-2014, 58.11.12 — Книги печатные профессиональные, технические и научные. Импортер в Беларусь: ООО «ПИТЕР М», 220020, РБ, г. Минск, ул. Тимирязева, д. 121/3, к. 214, тел./факс: 208 80 01. Подписано в печать 06.03.23. Формат 70×100/16. Бумага офсетная. Усл. п. л. 79,980. Тираж 700. Заказ 0000.

Карл Олбинг, Джей Пи Фоссен

ИДИОМЫ BASH

Сценарии на языке командной оболочки получили самое широкое распространение, особенно написанные на языках, совместимых с bash. Но эти сценарии часто сложны и непонятны. Сложность — враг безопасности и причина неудобочитаемости кода. Эта книга на практических примерах покажет, как расшифровывать старые сценарии и писать новый код, максимально понятный и легко читаемый. Авторы Карл Олбинг (Carl Albing) и Джей Пи Фоссен (JP Vossen) покажут, как использовать мощь и гибкость командной оболочки. Даже если вы умеете писать сценарии на bash, эта книга поможет расширить ваши знания и навыки. Независимо от используемой ОС — Linux, Unix, Windows или Mac — к концу книги вы научитесь понимать и писать сценарии на экспертном уровне. Это вам обязательно пригодится. Вы познакомитесь с идиомами, которые следует использовать, и такими, которых следует избегать.

КУПИТЬ

Кристофер Негус

БИБЛИЯ LINUX 10-е издание

Полностью обновленное 10-е издание «Библии Linux» поможет как начинающим, так и опытным пользователям приобрести знания и навыки, которые выведут на новый уровень владения Linux. Известный эксперт и автор бестселлеров Кристофер Негус делает акцент на инструментах командной строки и новейших версиях Red Hat Enterprise Linux, Fedora и Ubuntu. Шаг за шагом на подробных примерах и упражнениях вы досконально поймете операционную систему Linux и пустите знания в дело. Кроме того, в 10-м издании содержатся материалы для подготовки к экзаменам на различные сертификаты по Linux.

КУПИТЬ