VTE (Virtual TErminal library) — это библиотека, лежащая в основе различных эмуляторов терминала GNOME. Она предоставляет GTK-виджет, отображающий элемент терминала, используемый в приложениях наподобие GNOME Terminal, Console, Black Box, Tilix, Terminator, Ptyxis и других. Также она используется во встроенных терминалах Builder и Workbench.

На протяжении цикла GNOME 46 в VTE было внесено множество улучшений производительности. Кристиан Хергерт говорил о некоторых из них в своих постах о VTE и о своей работе в GNOME 46. Но насколько же улучшилась производительность? Чего стоит ожидать вам, пользователю, от установки свежего обновления Fedora 40 и запуска любимого терминала?

Давайте измерим и разберёмся!

Что мы измеряем?


У понятия «производительность» есть множество определений, особенно когда дело касается эмуляторов терминала. Одна из самых осязаемых метрик — это задержка ввода. Грубо говоря, она показывает скорость реакции программы на ваши действия: время, проходящее между нажатием кнопки на клавиатуре и изменением цвета пикселей на мониторе.

Приложения с низкой задержкой ввода кажутся быстрыми, а с высокой — тормозными.

Если задержка ввода достаточно низка, то вы можете привыкнуть к ней и вам будет казаться, что это нормально. Однако сравнение высокой и низкой задержки ввода (например, при переключении между двумя приложениями и вводом в обоих) может сделать её достаточно заметной. Наверно, вы слышали, что некоторые люди не могут вернуться к мониторам с частотой обновления 60 Гц, попробовав частоту 144 Гц; здесь эффект примерно такой же (и за это частично ответственна задержка ввода).

Как её измерить?

▍ Измеряем задержку ввода


Существуют инструменты наподобие Typometer, измеряющие задержку в ПО, распознавая события нажатия клавиш и выполняя захват экрана для определения изменений цвета пикселя. Они работают достаточно хорошо, но требуют настройки системы, чтобы случайно не внести искажений. Например, API захвата экрана может возвращать цвета пикселей несколькими миллисекундами раньше или позже их реального отображения на экране; это необходимо иметь в виду, когда вы пытаетесь замерить что-то с точностью до миллисекунд.

У меня есть нечто более интересное: аппаратный тестер задержки ввода! Он состоит из оптического датчика, соединённого с платой Teensy, которая, в свою очередь, подключена к компьютеру по USB.


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

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

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

Ниже представлено краткое описание его работы.

Оптический датчик направлен на небольшую область монитора, на которую влияет нажатие на клавишу (в нашем случае это ячейка конкретного символа в терминале). Плата отправляет нажатие клавиши (например, пробела) по USB и начинает мониторить показания оптического датчика. Как только она распознаёт скачок в величине освещённости, она отпускает клавишу. Затем она нажимает вторую клавишу (например, Backspace) и ждёт, пока освещённость изменится обратно. Мы вернулись к тому, с чего начинали; прошивка ожидает случайный промежуток времени (чтобы предотвратить попадание в частоту обновления монитора) и повторяет эксперимент.

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

Затем я выполняю агрегацию измерений и строю их график при помощи seaborn. Вот пример результата:


▍ Графики задержки ввода


Давайте посмотрим, что мы можем извлечь из этого графика задержек.


Маленькие чёрные точки — это отдельные измерения. То есть каждая точка — это величина времени между одним нажатием на клавишу и соответствующим изменением освещённости в датчике. Таких точек 120, потому что я повторял каждый тест 120 раз.

Судя по точкам, данные выглядят логично. Мы ожидаем, что основная часть измерений равномерно распределится по интервалу, примерно равному размеру одного цикла перерисовки монитора. Это вызвано тем, что мониторы обычно выполняют перерисовку с постоянной частотой, а при нажатии на клавишу в случайный момент времени мы попадём в случайную точку цикла перерисовки. Мы получаем наименьшую задержку, если приложение отрисовывает новый кадр в ответ прямо в тот момент, когда монитор должен его показать. Наибольшая задержка получается, когда приложение завершает рендеринг нового кадра, чуть-чуть пропустив «дедлайн» монитора, и вынуждено ждать ещё один цикл перерисовки для изменения цвета пикселя.

В приведённом выше примере точки распределены в интервале 7-8 мс, что примерно равно циклу обновления (около 6,94 мс) моего монитора с частотой обновления 144 Гц.

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

Мы не ожидаем увидеть никаких пробелов между кластерами точек. Они обычно говорят об алиасинге с циклом перерисовки монитора или о каком-то баге планировщика кадров в композиторе.

[Несколько недель назад сделанные мной измерения показали подозрительный пробел в точках длиной в один кадр. Оказалось, что это баг планировщика кадров в моём композиторе, винить за который можно было только меня. К счастью, устранить его было несложно, и я легко удостоверился в его отсутствии, проведя тот же тест повторно.]

Ящик демонстрирует статистику для отдельных измерений:

  • медиану (измерение, находящееся «ровно посередине»; половина измерений ниже него, а вторая половина — выше),
  • самое нижнее и верхнее измерение,
  • 25-й и 75-й перцентили (25% и 75% измерений ниже соответствующей линии).

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

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

Тестовая система


Я провёл все тесты на следующей системе:

  • Lenovo Legion 7 Gen 7 AMD с CPU Ryzen 7 6800H и dGPU Radeon RX 6700M (использующим dGPU исключительно через переключатель MUX).
  • Монитор: Acer Nitro XV320QU, 2560×1440, 144 Гц, масштабирование 100%.
  • Хост: Fedora 40 Silverblue Beta, Mesa 24.0.4.
  • Композитор: raw Mutter 46.0.

Возможно, вы спросите, что такое raw Mutter? Mutter — это композитор, поверх которого собирается GNOME Shell. Оказалось, можно запустить самостоятельный Mutter без GNOME Shell, переключившись на другой VT и запустив команду вида mutter --display-server -- alacritty. Так вы получите практически «голое» окружение, предназначенное только для тестирования. Однако оно довольно полезно для бенчмаркинга, так как представляет собой нечто близкое к идеальному случаю GNOME Shell с нулевой дополнительной нагрузкой.

Я протестировал несколько приложений терминала. На графиках они встречаются в следующем порядке:

  • Alacritty: не на основе VTE; служит своего рода точкой начала отсчёта, потому что по всем моим предыдущим тестам стабильно оказывается одним из самых быстрых терминалов.
  • Console: GTK 4, терминал GNOME по умолчанию. [У вашего дистрибутива может быть другое представление о том, какой терминал должен применяться по умолчанию в GNOME. Например, в Fedora по-прежнему по умолчанию применяется GNOME Terminal.]
  • VTE Test App: GTK 4, тестовый терминал, находящийся в репозитории VTE.
  • GNOME Terminal: GTK 3, раньше использовался в GNOME по умолчанию и по-прежнему распространяется в стандартном наборе многих дистрибутивов. [Для GNOME 47 GNOME Terminal будет портирован на GTK 4, но GNOME 46 это всё ещё приложение GTK 3.]

Так как мы хотим сравнить GNOME 45 и GNOME 46, для установки и запуска всех представленных выше терминалов я воспользовался контейнерами toolbx с Fedora 39 и Fedora 40; никакие дополнительные настройки терминалов не выполнялись.

Я по порядку запускал терминалы и перемещал их окна в левый верхний угол монитора. Во всех тестах курсор мыши находился вне пределов окна. [Чтобы избежать искажения результатов из-за срабатывания логики ссылки под курсором.]

Тесты задержки ввода


Первый тест прост: я запускаю cat > /dev/null, чтобы получить поле ввода без readline и другой подобной обработки, а затем измеряю время, необходимое терминалу для перемещения курсора блока на одну ячейку вправо после нажатия на пробел.

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

Вот как выглядит процесс теста:

А вот результаты:


Как и ожидалось, показатели Alacritty никак не изменились при переходе с F39 на F40.

Но взгляните на огромное улучшение у всех терминалов VTE! Они совершили скачок с «плохо» до почти уровня Alacritty; даже GNOME Terminal на GTK 3 очень к нему близок.

Скорее всего, главное изменение, ставшее причиной этого — вот этот коммит Кристиана, благодаря которому произошёл переход от таймера перерисовки VTE на 40 Гц к отрисовке в каждом кадре, синхронизованной с монитором, как и должен вести себя любой уважающий себя виджет GTK.

У Console есть несколько выбросов, которые возможно вызваны её отслеживанием процессов, но в этом нет ничего нового (вероятно, этой проблемой займутся в GNOME 47).

Для следующего теста я создал более реалистичный сценарий. Я сделал снэпшот моей системы с neovim и открыл README из Ptyxis. Затем я заменил квадрат текста полноблочными символами Unicode, чтобы у оптического датчика было яркое пятно для распознавания.


Тест заключается во многократном нажатии на Ctrl+D и Ctrl+U для скроллинга вверх и вниз текстового буфера в neovim. Оптический датчик попеременно считывает пустую строку (тёмную) и полноблочную область (яркую). В neovim есть множество изысков, поэтому терминалу приходится отрисовывать различные подчёркивания прямыми и волнистыми линиями, значки gutter и statusline.

Вот как выглядит процесс тестирования:

А вот результаты:

image

В этом тесте тоже очевидны огромные улучшения: терминалы GNOME 46 по-прежнему имеют почти одинаковые показатели с Alacritty!

Теперь давайте внимательнее изучим все результаты Fedora 40 на одном графике:


На этом графике видно, какой задержкой приходится расплачиваться в тесте neovim по сравнению с простым cat, но рост задержки схож у всех терминалов.

vtebench


Я провёл бенчмаркинг vtebench Alacritty для того же набора приложений и конфигураций. Это полностью автоматизированный бенчмарк, измеряющий не задержку ввода, а нечто совершенно иное: производительность считывания и парсинга PTY.

Вот что написано в README vtebench:

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

Длительность перерисовки может влиять и влияет на результаты этого теста, особенно в случае терминалов, считывающих и парсящих PTY в том же потоке, в котором выполняют свою логику перерисовки, например, в случае VTE.

Вот как выглядит один из бенчмарков vtebench:


А вот результаты:


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

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

Стоит отметить, что я исключил из этих результатов два бенчмарка: dense_cells и unicode. Это два основных стресс-теста vtebench, очень сильно нагружающие терминал. К сожалению, VTE по-прежнему испытывает с ними проблемы и демонстрирует огромный разброс, снижающий остальные результаты и ухудшающий читаемость графика.

Если вам любопытно, вот полные результаты

Заключение


В GNOME 46 выполнена большая работа по улучшению производительности VTE, действительно ощущаемая при обычной работе с терминалом. Задержка ввода снизилась и опустилась почти до показателей самых быстрых терминалов, даже при нетривиальной конфигурации neovim со сложной структурой экрана.

Оставшаяся разница, по крайней мере, в этих тестовых случаях, практически несущественна. Частично её можно объяснить тем, что VTE выполняет намного больше дополнительной работы для обеспечения accessibility (включённой в GNOME Terminal и пока отключённой в терминалах GTK 4), вычисление полос прокрутки и другие функции.

Если вы избегали пользоваться терминалами на основе VTE из-за их тормознутости и задержки ввода, то настало время дать им ещё один шанс. Только учтите, что все эти плюсы появились в VTE 0.76.

Выражаю огромную благодарность мейнтейнерам и контрибьюторам VTE и поздравляю их с потрясающим релизом!

P.S. Если вам любопытен Ptyxis или сравнение поведения рендереров GTK NGL, NVK и GL, то могу сказать, что их производительность примерно схожа с показанными выше результатами F40 VTE Test App. Месяц назад я выполнил их более подробные бенчмарки, которые можно найти здесь.

Telegram-канал со скидками, розыгрышами призов и новостями IT ?

Комментарии (20)


  1. 6666fantom9999
    22.04.2024 15:33
    +2

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


    1. Gugic
      22.04.2024 15:33
      +1

      Я слышал жалобы на производительность терминала от людей которые используют Vim. Что, наверное имеет смысл.


      1. mpa4b
        22.04.2024 15:33

        Как-то не очень ясно зачем в DE использовать vim в терминале вместо нативного gvim. Получается лишний слой (собственно сам терминал).


    1. VADemon
      22.04.2024 15:33
      +3

      Я тащусь от скорости реагирования на ввод в текстовом режиме Линукса. Молниеносно по сравнению с графическим окружением / Windows. Удивительно, что даже в гостевой Linux VM под Виндой текстовый режим всё равно ощущается намного быстрее того, что рядом. Поэтому работу в этом направлении всячески приветствую.

      Я точно не отношусь к тем, что без этого жить не может. Но это то, к чему надо стремиться.


      1. Einherjar
        22.04.2024 15:33

        Молниеносно по сравнению с графическим окружением / Windows.

        Если отключить всякие анимации, то GUI тоже молниеносным станет


    1. domix32
      22.04.2024 15:33

      Вы просто не работали на маках вероятно. Когда простой ls в терминале рендерится пол секунды это довольно печально.


    1. Ksoo
      22.04.2024 15:33

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


  1. vtb_k
    22.04.2024 15:33

    Уже пару лет использую wezterm, очень быстрый, но мне важнее то, что вся конфигурация на луа, очень гибко можнш под себя настроить


  1. Kahelman
    22.04.2024 15:33

    Компьютер за секунду делает столько же операций сколько человек за сто лет… соответственно он за минуту наделает столько ошибок, что разбираться устанем :)


    1. Kahelman
      22.04.2024 15:33
      +1

      Кто-то здесь с законам Мёрфи не согласен?


  1. markmariner
    22.04.2024 15:33
    +1

    Я из-за этого перешёл на VS Code с WebStorm, потому, что в продуктах JetBrains прямо заметная задержка при вводе. Круто, что кто-то это тоже замечает в других продуктах.


  1. fiego
    22.04.2024 15:33
    +10

    То есть, разработчик сделал ошибку три года назад, ему прислали патч, который он гордо не взял, а теперь, когда пользователи три года отмучились, он герой? Нормально там у вас в гноме...


  1. Samhuawei
    22.04.2024 15:33

    Гм, а нажать заветные Ctrl+Alt+F1 и очутиться в родном линуксовом терминале пролетарское происхождение мешает?


    1. Bonus2k
      22.04.2024 15:33

      Рабочая среда это не только терминал


      1. Samhuawei
        22.04.2024 15:33

        Смотря у кого. На программистах свет клином не сошёлся, да и там есть люди которые вполне могут подправить кусок кода в православном vi или залить изменения в гит из командной строки. Таких конечно маловато, вымирают как динозавры. Но текстовые терминалы живут и будут жить. И пальцы помнят заветные ls -lart, cat | grep | less и прочие sedы.

        А зазиповать поток на лету, передать в другой сервер по ssh и распаковать в нужную папку это вообще стандартная процедура того кто умеет справляться безо всяких sftp.


        1. Ritan
          22.04.2024 15:33

          кусок кода в православном vi

          Неофиты. Почему не ed?

          Таких конечно маловато

          https://xkcd.com/378/


  1. domix32
    22.04.2024 15:33

    Интересно, а кто-нибудь сравнивал насколько производителен родной терминал Asahi Linux в сравнении с дефолтным терминалом MacOS?


  1. mpa4b
    22.04.2024 15:33
    +1

    Ещё интересно было бы увидеть сравнение по скорости с xterm.


    1. inv2004
      22.04.2024 15:33

      И kitty, так как по тестам он был быстрее всех

      Еще st


    1. zstas
      22.04.2024 15:33
      +1

      xterm побеждал в предыдущих бенчмарках [0], в т.ч. alacritty [1]. учитывая что гном вышел только на его уровень - думаю что все еще xterm лучший по отзывчивости.

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

      ну и throughput у него так себе, но я гигабайты данных и не гоняю в терминале.

      [0] - https://lwn.net/Articles/751763/

      [1] - https://www.reddit.com/r/linux/comments/jc9ipw/why_do_all_newer_terminal_emulators_have_such_bad/