kernel-parameters.rst 150 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832
  1. .. ARU (c) 2023 - 2024, Vasiliy Stelmachenok and contributors
  2. ARU is licensed under a
  3. Creative Commons Attribution-ShareAlike 4.0 International License.
  4. You should have received a copy of the license along with this
  5. work. If not, see <https://creativecommons.org/licenses/by-sa/4.0/>.
  6. .. _kernel-parameters:
  7. **************************
  8. Настройка параметров ядра
  9. **************************
  10. .. warning:: Данный раздел находится в разработке и предназначен для
  11. опытных пользователей Linux, которые хотят выполнить более тонкую
  12. настройку системы. Тем не менее, в нем автор немного упростил
  13. внутреннюю картину работы ядра, чтобы не делать текст чрезмерно
  14. объемным и сложным для понимания простому пользователю. Так,
  15. например, здесь вы увидите несколько упрощенную модель устройства
  16. виртуальной памяти ядра Linux, поэтому если вы эксперт в вопросе,
  17. то не судите строго.
  18. =========
  19. Введение
  20. =========
  21. Как и любая другая программа, ядро имеет свои собственные настройки и
  22. параметры, которые контролируют поведение определенных его частей. И
  23. хотя настройки ядра Linux являются менее очевидными для понимания и
  24. часто "скрыты" с глаз обычных пользователей, в данном разделе мы
  25. рассмотрим различные настройки ядра с целью улучшения
  26. производительности и отзывчивости системы для домашнего ПК или
  27. ноутбука, подобрав наиболее оптимальные значения в зависимости от
  28. вашей конфигурации.
  29. Как мы поймем далее, несмотря на то, что ядро Linux принято считать
  30. монолитным, все настройки ядра относятся к определенным его
  31. подсистемам, поэтому раздел будет разбит на систематические блоки,
  32. каждый из которых будет выполнять настройку конкретной подсистемы,
  33. будь то подсистема ввод/вывода или сети.
  34. Стоит также отметить, что так как ядро является общей составляющей
  35. всех дистрибутивов Linux, то почти вся информация которая будет
  36. представлена в этом разделе применима не только к Arch Linux, но и к
  37. другим дистрибутивам. Учтите, что некоторые параметры зависят от вашей
  38. версии ядра, которая, по понятным причинам, от одного дистрибутива к
  39. другому может отличаться, поэтому обращайте внимание на примечания, где
  40. указано с какой версии появился тот или иной параметр. Узнать
  41. версию используемого вами ядра можно через команду ``uname -r``.
  42. -------
  43. Зачем?
  44. -------
  45. Часто люди задаются вопросом, зачем пытаться лезть под капот, когда
  46. "очевидно", что все уже настроено и отполировано за тебя. Отчасти это
  47. правда, и ядро Linux с каждой версией улучшается и "вылизывается"
  48. тысячами разработчиками по всему миру, и, наверное, параметры о
  49. которых пойдет речь далее уже имеют наиболее оптимальные значения. Но
  50. к сожалению, это не совсем так. Или скорее совсем не так. Во-первых,
  51. разработчики ядра часто не могут иметь представления о том, на каком
  52. конкретном железе будет работать ядро и для каких целей оно будет
  53. использоваться, в следствии чего главным приоритетом при разработке
  54. является совместимость и адаптивность ядра к как можно большему числу
  55. возможных задач и конфигураций. Такой компромиссный подход к
  56. разработке не всегда дает наилучшие результаты в чем-то конкретном, но
  57. зато позволяет ядру Linux одинаково подходить как для работы на
  58. серверах, роутерах, телефонах, микроконтроллерах, так и простых ПК.
  59. Грубо говоря, ядро Linux представляет собой швейцарский нож от мира
  60. IT, которым хоть и можно порезать хлеб, но удобнее это сделать обычным
  61. ножом. Наша задача в данном разделе это как раз заточить ядро под
  62. конкретную задачу, в нашем случае это интерактивное использование на
  63. домашнем компьютере.
  64. Если вы переживаете за стабильность вашей системы, то предварительно
  65. сделайте резервную копию, хотя на самом деле все параметры, о которых
  66. пойдет речь далее, могут быть отключены в любой момент простым
  67. удалением файла настройки, поэтому даже при возникновении проблем со
  68. стабильностью или регрессиями у вас не должно возникнуть проблем с
  69. откатом к стоковым значениям.
  70. --------------
  71. Виды настроек
  72. --------------
  73. В ядре Linux все параметры можно поделить на типы в зависимости от
  74. способа установки их значения [#]_. Часть из них может быть
  75. установлена только на этапе загрузки ядра, то есть в качестве опций
  76. командой строки [#]_. Это то, что мы обычно пониманием под просто
  77. "параметрами ядра". Они указываются в настройках вашего загрузчика,
  78. будь то GRUB, refind или systemd-boot. К этой же категории можно
  79. отнести параметры модулей ядра, значения к которым передаются во время
  80. их загрузки. При этом значения параметров могут быть переданы как
  81. часть общих параметров загрузки ядра в конфиге вашего загрузчика при
  82. их указании в следующем формате: ``module.parameter=value`` (например
  83. ``nvidia-drm.modeset=1``). То есть сначала указывается имя модуля
  84. (драйвера), затем имя параметра и через знак равно передается
  85. значение. Другой способ передачи параметров для модулей - это
  86. использование конфигурационных файлов ``modprobe``. В этом случае не
  87. имеет значения какой у вас загрузчик, достаточно создать файл в
  88. директории ``/etc/mopdrobe.d`` с любым названием, и прописать
  89. параметры в следующем формате::
  90. options module parameter1=value1 parameter2=value2
  91. Именно вторым способом передачи параметров по возможности мы будем
  92. пользоваться далее в разделе, когда речь будет заходить о настройке
  93. различных модулей.
  94. ~~~~~~~~~
  95. sysctl
  96. ~~~~~~~~~
  97. Другой тип, это параметры, значение которых можно изменить прямо во
  98. время работы системы, что называется "на лету". Такие настройки
  99. представлены в виде файлов на псевдофайловой системе procfs в
  100. директории ``/proc/sys`` [#]_. procfs называется псевдофайловой
  101. потому, что физически она не расположена на диске, а все файлы и
  102. директории создаются самим ядром при запуске системы в оперативной
  103. памяти. По этой причине у них отсутствует размер, и все они имеют
  104. чисто служебный характер. В директории ``/proc/sys/`` каждая настройка
  105. это отдельный файл, куда мы должны просто передать (записать) значение
  106. в виде числа (часто 1 и 0 означают включить/выключить, но некоторые
  107. параметры также могут принимать значение, которое имеет строго определенный
  108. смысл, и только из определенного диапазона). Все настройки объедены в
  109. директории, которые характеризуют их отношение к чему-то общему.
  110. Например, все файлы в подкаталоге ``vm`` - это настройки для механизма
  111. виртуальной памяти ядра [#]_ , включая настройки для подкачки,
  112. кэширования, и т. д. В ``kernel`` - общие настройки ядра [#]_, а в
  113. ``net`` [#]_ - настройки сетевой подсистемы, протоколов TCP и IP.
  114. Именно эти три категории мы и будем рассматривать далее.
  115. Конечно, процесс поиска всех файлов-настроек и записи значений
  116. средствами командой строки каждый раз весьма утомителен. Поэтому
  117. разработчики создали специальную утилиту под названием ``sysctl``,
  118. которая значительно упрощает данный процесс. Теперь нам не нужно
  119. лазить каждый раз в ``/proc/sys/``, чтобы изменить значение
  120. параметров. Вместо этого достаточно прописать в терминале::
  121. sysctl -w kernel.sysrq=1
  122. Это то же самое, что и данная команда, которая прописывает значение
  123. напрямую в файл из директории ``/proc/sys/``::
  124. echo "1" > /proc/sys/kernel/sysrq
  125. Обратите внимание, что для изменения настроек всегда нужны права root,
  126. поэтому перед каждой такой командой мы должны добавить ``sudo``::
  127. sudo sysctl -w kernel.sysrq=1
  128. Другим преимуществом sysctl является то, что мы можем делать такие
  129. изменения постоянными, просто прописав соответствующую строку в файл,
  130. который находится в директории ``/etc/sysctl.d/``, например в
  131. ``/etc/sysctl.d/99-sysctl.conf``::
  132. kernel.sysrq = 1
  133. Собственно именно добавлением таких строк мы и будем применять
  134. соответствующие настройки.
  135. .. warning:: Настройки прописываемые в файле ``/etc/sysctl.conf`` не
  136. применяются начиная с версии 21x в systemd, поэтому
  137. прописывайте их только в файлах, которые расположены в подкаталоге
  138. ``/etc/sysctl.d``. Имя файла не имеет значения.
  139. ~~~~~~~~~~~
  140. tmpfiles.d
  141. ~~~~~~~~~~~
  142. К сожалению, далеко не все настройки ядра можно изменить при помощи
  143. sysctl или псевдофайловой ФС ``/proc/sys``. Часть из них является
  144. отладочными, поэтому они расположены в виде файлов на другой
  145. псевдофайловой системе - sysfs, которая в основном отвечает за
  146. представление информации об устройствах, которыми управляет ядро. В
  147. директории в ``/sys/kernel`` представлены ряд других полезных
  148. параметров, которые мы рассмотрим в рамках общей темы. Чтобы выполнить
  149. установку значения в файлах, которые находятся в ``/sys/kernel/``, мы
  150. будем использовать такой инструмент как systemd-tmpfiles.d [#]_. Он
  151. есть только в дистрибутивах, использующих systemd в качестве системы
  152. инициализации, то есть в большей части дистрибутивов Linux включая
  153. Arch. Суть этой службы состоит в управлении, создании и удалении
  154. временных файлов или редактировании уже существующих. В нашем случае
  155. мы будем его использовать для записи значений в файлы настроек
  156. расположенных в ``/sys/kernel/``. Для этого, по аналогии с sysctl,
  157. нужно создать файл в директории ``/etc/tmpfiles.d``, например
  158. ``/etc/tmpfiles.d/99-settings.conf``. Формат записи каждой строки в
  159. файле будет следующим::
  160. w! /sys/kernel/mm/lru_gen/min_ttl_ms - - - - 2000
  161. Первый символ - это тип действия, который ``systemd-tmpfiles`` будет
  162. выполнять с указанным по пути файлом. В нашем случае мы будем
  163. использовать только запись *w* некоторого значения в уже существующие
  164. файлы, а не создавать новые. Восклицательный знак ``!`` указывает, что
  165. значение будет прописываться только один раз при загрузке системы.
  166. После пути до файла идут четыре прочерка, в них должны быть указаны
  167. права доступа, которые мы хотим изменить, но так как мы имеем
  168. дело со служебными файлами, то пишем везде прочерки, чтобы ничего не
  169. менять. В конце указываем значение, которое будет прописано в файл, то
  170. есть значение параметра.
  171. Другими словами, везде, куда не дотянется sysctl, мы будем
  172. использовать ``tmpfiles``.
  173. ~~~~~~
  174. udev
  175. ~~~~~~
  176. По сути первых двух инструментов уже достаточно, чтобы выполнить
  177. полную настройку ядра, но мы используем ещё одну вещь - правила udev.
  178. Udev [#]_ - менеджер для управления вашими устройствами, который
  179. отслеживает их подключение/выключение, и предоставляет возможность
  180. создавать так называемые "правила", которые вызываются каждый раз,
  181. когда происходит определенное действие с тем или иным устройством.
  182. Внутри этого правила можно указать, при каких событиях и для какого
  183. конкретно устройства (условие для срабатывания) мы будем выполнять
  184. определенную команду или устанавливать некоторое значение. Это очень
  185. полезный инструмент, который позволит нам применять целый ряд настроек
  186. в зависимости от некоторых условий и подстраиваясь под железо, которое
  187. у вас есть в системе. Приведу пример, чтобы стало понятнее. Для разных
  188. типов носителей подходит разный планировщик ввода/вывода. Для обычных
  189. SSD - ``mq-deadline``, для HDD - ``bfq``. Правила udev позволят нам
  190. при подключении определенного типа устройства сразу выбирать нужный
  191. планировщик и дополнительные параметры для него, даже если у вас в
  192. системе есть и SSD, и HDD одновременно. Подробнее планировщики
  193. ввода/вывода будут рассмотрены далее вместе с синтаксисом самих
  194. правил.
  195. .. [#] https://medium.rip/@justaboutcloud/a-dive-deep-into-kernel-parameters-part-1-kernel-boot-parameters-139905e3432
  196. .. [#] https://www.kernel.org/doc/html/latest/admin-guide/kernel-parameters.html
  197. .. [#] https://www.kernel.org/doc/html/latest/admin-guide/sysctl/index.html
  198. .. [#] https://www.kernel.org/doc/html/latest/admin-guide/sysctl/vm.html
  199. .. [#] https://www.kernel.org/doc/html/latest/admin-guide/sysctl/kernel.html
  200. .. [#] https://www.kernel.org/doc/html/latest/admin-guide/sysctl/net.html
  201. .. [#] https://www.freedesktop.org/software/systemd/man/systemd-tmpfiles.html
  202. .. [#] https://www.freedesktop.org/software/systemd/man/udev.html
  203. .. _io_optimization:
  204. =========================
  205. Оптимизация ввода/вывода
  206. =========================
  207. Фууух, что же, надеюсь вы не устали от всего этого скучного вступления
  208. выше и мы можем наконец-то переходить к сути. Начнем с оптимизации
  209. ввода/вывода, то бишь к настройке подкачки (она же *своп*, от англ.
  210. *swap*), различных кэшей и планировщиков.
  211. .. _virtual_memory:
  212. ---------------
  213. Общие сведения
  214. ---------------
  215. Прежде чем перейти непосредственно к настройке необходимо понять
  216. принцип работы механизма виртуальной памяти и подкачки в Linux. Это
  217. важно, так как в этой теме ходит целая куча различных мифов, которые
  218. мы сейчас разберем.
  219. Итак, для начала чрезвычайно важно понять, что ядро Linux разбивает
  220. всю вашу память на маленькие "гранулы" - страницы памяти, как правило
  221. по 4 КБ (для x86 архитектуры), не больше и не меньше. Это может
  222. показаться странным, но если не вдаваться в технические подробности,
  223. то такой подход позволяет ядру Linux проявлять достаточно большую
  224. гибкость, так как данные страницы могут быть одинаково обработаны
  225. ядром вне зависимости от того, что в них записано, предотвращая
  226. обильную фрагментацию. Тем не менее, все страницы памяти можно разбить
  227. на несколько типов. Сейчас мы не будем рассматривать их все, но
  228. остановимся на самых главных:
  229. - Файловая "подложка" или *файловые страницы* - это страницы в которых
  230. ядро "отображает", то есть представляет данные файла, считываемые с
  231. диска в виде страниц в памяти. С этими страницами тесно связано
  232. понятие страничного кэша (page cache) [#]_. Если некоторый процесс
  233. открывает какой-то новый файл и читает из него информацию, то в
  234. первый раз ядро считывает эти данные с диска и сохраняет их в
  235. страничном кэше, а все последующие операции ввода и вывода к этим же
  236. данным будут осуществляться уже при использовании кэша, что
  237. значительно ускоряет все базовые операции чтения и записи,
  238. предотвращая повторные обращения к диску. При этом память для таких
  239. страниц выделяется по требованию, поэтому если процесс открыл файл,
  240. но ничего из него не читает, то никакой реальной памяти для таких
  241. страниц выделено не будет. Собственно, то, что вы видите в графе
  242. "Кэш" в любой программе аналоге системного монитора в Linux - и есть
  243. страничный кэш. Обратите внимание, что исполняемые файлы (программы)
  244. тоже загружаются в память как файловые страницы.
  245. .. image:: https://biriukov.dev/docs/page-cache/images/page-cache.png
  246. :align: right
  247. (Licensed under the CC BY-NC 4.0. © Vladislav Biriukov, All rights reserved)
  248. - Очевидно, что далеко не все данные, которыми оперирует программа,
  249. могут быть представлены в виде реальных файлов на диске, поэтому
  250. были созданы *анонимные страницы*, которые, как следует из названия,
  251. не ассоциированы с файлами [#]_. Программы запрашивают их у ядра во время
  252. своей работы для динамических данных. Если вы разработчик, то вы
  253. наверняка сталкивались с такими понятиями как "Куча" (Heap) и "Стек"
  254. (Stack). Так вот, ядро хранит данные из кучи и стека именно в
  255. анонимных страницах памяти.
  256. - Грязные страницы (dirty pages) - по сути это подвид файловых
  257. страниц, ключевое отличие которых состоит в том, что программы в них
  258. пишут какие-то изменения, а так как ядро кэширует все данные
  259. считываемые из файлов во избежание излишней нагрузки на диск, то
  260. изменения, которые программа делает с файлом, на самом деле
  261. происходят сначала в кэше, и только потом синхронизируются с
  262. реальным файлом на диске. Более подробно об этом виде страниц и
  263. процессе их синхронизации с диском мы поговорим в следующем разделе.
  264. Вернемся к подкачке. Один из самых больших мифов, связанных с
  265. подкачкой, состоит в том, что пользователи рассматривают её как некую
  266. "дополнительную память", которую свободно можно использовать в случае
  267. нехватки реальной, то есть физической памяти. Это конечно же не так,
  268. хотя бы потому, что процессор имеет доступ к оперированию только
  269. данными, которые находятся внутри ОЗУ. В случае нехватки памяти у ядра
  270. есть по сути всего один вариант - это освобождать уже имеющуюся память
  271. от тех страниц, которые не используются в данный момент, выгружая их в
  272. область на диске которую мы и называем подкачкой. Да, память не
  273. берется из воздуха, и подкачка - это просто "чердак", куда ядро
  274. скидывает все неиспользуемые вещи, чтобы освободить место для новых
  275. или более часто используемых страниц. При этом для процессов не
  276. меняется ровным счетом ничего, ибо они как и раньше могут обратиться к
  277. данным в памяти, которые были расположены на странице, которая была
  278. вытеснена ядром в подкачку, но когда процесс это сделает, ядро найдет
  279. эту страницу, считает её из подкачки и обратно загрузит в оперативную
  280. память. Это ещё одно преимущество механизма виртуальной памяти,
  281. повсеместно используемого ядром Linux.
  282. Вопрос лишь в том, какие именно страницы нужно "вытеснить" из памяти.
  283. На самом деле, это достаточно сложный вопрос. Прежде всего, конечно же
  284. это будут именно анонимные страницы, так как файловые страницы и так
  285. по сути ассоциированы с данными на диске, следовательно в случае чего
  286. их точно так же можно повторно считать, и выгружать их в подкачку
  287. просто не имеет никакого смысла, что и происходит на практике. Но что
  288. если анонимных страниц много, а часть из них реально используется
  289. программами в данный момент? Какие из них тогда должны первым делом
  290. попасть в подкачку? На данный и многие другие вопросы отвечает
  291. специальный алгоритм в ядре Linux, называемый :abbr:`LRU (Least
  292. recently used)` (а поныне и MGLRU). Если очень упрощенно, то данный
  293. алгоритм ведет учет использования каждой страницы, то есть количество
  294. обращений к ней, и на основе данной статистики предполагает, какие из
  295. них реже всего используются процессами, и следовательно какие из них
  296. можно без проблем выгрузить в подкачку.
  297. Рядовые пользователи часто не до конца понимают, какие именно данные
  298. расположены у них в подкачке. Теперь мы можем дать чёткий ответ: в
  299. подкачке хранятся только неиспользуемые анонимные страницы памяти.
  300. .. [#] https://docs.kernel.org/admin-guide/mm/concepts.html#anonymous-memory
  301. .. [#] https://biriukov.dev/docs/page-cache/2-essential-page-cache-theory/
  302. .. _setup_swappiness:
  303. -------------------
  304. Настройка подкачки
  305. -------------------
  306. Мы разобрались с основополагающими понятиями, и наконец-то можем
  307. переходить к настройке. Для настройки поведения подкачки используется
  308. параметр sysctl ``vm.swappiness`` (значение по умолчанию 60) [4]_. Вокруг
  309. него так же ходит целый ряд заблуждений, что приводит к неправильным
  310. умозаключениям. Итак, во-первых, ``vm.swappiness`` напрямую никак не
  311. влияет на то, когда у вас начнет использоваться подкачка, то есть его
  312. значение - это вовсе не процент занятой памяти, при достижении которого
  313. начинает использоваться подкачка. Ядро всегда начинает использовать
  314. подкачку только в ситуациях нехватки памяти (это, как правило, когда
  315. занято 85-90% ОЗУ).
  316. Во-вторых, параметр ``vm.swappiness`` влияет только на предпочтение
  317. ядра к вытеснению определенного типа страниц в случае этой самой
  318. нехватки. Он принимает значения от ``0`` до ``200`` (начиная с версии
  319. ядра 5.8 и выше, до этого максимальным значением было 100). Для более
  320. наглядного понимания, параметр ``vm.swappiness`` можно представить в
  321. виде весов, где более низкие (ниже 100) значения приводят к склонности
  322. ядра сначала вытеснять все страницы из файлового кэша, а более высокие
  323. (больше 100) - освобождение анонимных страниц из памяти в подкачку [#]_.
  324. Значение ``100`` - это своего рода баланс, при котором ядро будет в
  325. одинаковой степени стараться вытеснять как файловые, так и анонимные
  326. страницы.
  327. Другим крайне распространенным заблуждением является то, что более
  328. низкие значения ``vm.swappiness`` уменьшают использование подкачки -
  329. следовательно уменьшается нагрузка на диск, и что это якобы
  330. увеличивает отзывчивость системы. На деле это лишь на половину правда,
  331. так как, да, ядро при низких значениях старается откладывать
  332. использование подкачки, хотя это и не значит, что она вообще не будет
  333. использоваться, но важно понять, что это происходит за счёт более
  334. агрессивного вытеснения файловых страниц из страничного кэша - что
  335. точно так же приводит к нагрузке на ввод/вывод. Почему? Потому что
  336. каждый раз, когда ядро вытесняет страницу из страничного кэша, это
  337. приводит к тому, что все ранее хранящиеся в ней данные снова придется
  338. считывать с диска по новой.
  339. Во-вторых, нагрузка на ввод/вывод, которую создаёт подкачка
  340. оказывается слишком переоценена. Для современных SSD накопителей
  341. переварить такую нагрузку без замедления работы системы не составит
  342. труда. Тем не менее, если страница была вытеснена в подкачку, то любая
  343. операция обращения к ней будет в разы медленнее, чем если бы она
  344. находилась в ОЗУ, даже если ваш носитель это NVMe накопитель, то
  345. операция записи страницы в файл/раздел подкачки и последующая операция
  346. чтения из него будет в любом случае затратна. Но даже если у вас HDD,
  347. то вам на помощь спешит Zswap - ещё один встроенный механизм ядра
  348. Linux, позволяющий значительно снизить нагрузку на диск и ускорить
  349. процесс вытеснения. Он представляет собой буфер в памяти, в который
  350. попадают анонимные страницы, которые на самом деле должны были попасть
  351. в подкачку на диске, и сжимаются внутри него, экономя тем
  352. самым драгоценную память насколько это возможно. Если пул страниц
  353. Zswap заполнится (по умолчанию он равен 20%), то ядро выполнит
  354. выгрузку страниц из Zswap в подкачку [#]_.
  355. На сегодняшний день механизм Zswap используется во многих
  356. дистрибутивах Linux *по умолчанию*, в том числе в Arch, просто вы об
  357. этом могли не знать, и потому могли думать, что ядро "насилует" ваш
  358. диск при малейшем использовании подкачки. Никакой дополнительной
  359. настройки для его работы как правило не требуется.
  360. Учитывая всё вышеперечисленное, автор рекомендует устанавливать
  361. значение ``vm.swappiness`` в ``100``. Это позволит ядру равномерно
  362. вытеснять в подкачку оба типа страниц. В современных реалиях
  363. выкручивание параметра в низкие значения не приводит к желаемому
  364. эффекту. Конечно, всё индивидуально, и имеет смысл поиграться на своем
  365. железе, чтобы понять что лучше подходит лично вам имея прописанный
  366. багаж знаний по теме. Зафиксировать это значение можно через конфиг
  367. sysctl:
  368. .. code-block:: shell
  369. :caption: ``sudo nano /etc/sysctl.d/90-sysctl.conf``
  370. vm.swappiness = 100
  371. .. warning:: Автор настоятельно не рекомендует устанавливать значение
  372. параметра в 0 или отключать подкачку вовсе. Подробнее о том, почему
  373. это вредно читайте в данной статье -
  374. https://habr.com/ru/company/flant/blog/348324/. Если вы хотите
  375. минимизировать использование подкачки чтобы минимизировать нагрузку
  376. на ввод/вывод, то используйте ZRAM, о котором пойдет речь далее.
  377. .. [#] https://www.howtogeek.com/449691/what-is-swapiness-on-linux-and-how-to-change-it/
  378. .. [#] https://docs.kernel.org/admin-guide/mm/zswap.html
  379. .. _zram:
  380. ~~~~~~~
  381. ZRAM
  382. ~~~~~~~
  383. Но что делать, если у вас и правда очень медленный носитель или вы
  384. хотите минимизировать нагрузку на ввод/вывод и износ диска? В этом
  385. случае лучшим решением является использование ZRAM - вида подкачки,
  386. при котором все неиспользуемые анонимные страницы не выгружаются на
  387. диск, а сжимаются прямо внутри памяти при помощи алгоритмов сжатия без
  388. потерь [#]_. Точно так же как вы сжимаете простые файлы через
  389. архиватор, то же самое делает ядро со страницами памяти. Понятно, что
  390. уже сжатые страницы использовать нельзя, поэтому если они снова
  391. понадобятся процессу, то ядру придется их расжать перед
  392. использованием. Конечно, стоит учитывать, что сжатие и расжатие
  393. страниц происходит ресурсами процессора, и это имеет определенные
  394. накладные расходы, но они довольно несущественны для современных
  395. многоядерных процессоров, чтобы ими можно было пренебречь. Тем не
  396. менее, всегда можно выбрать более "легковесный" алгоритм сжатия.
  397. .. note:: Некоторые пользователи задаются вопросом: В чем разница
  398. между zswap и ZRAM? На самом деле хотя они и занимаются по сути
  399. одной и той же работой, разница здесь в том, что Zswap является
  400. сжатым *буфером* в памяти, то есть промежуточным звеном между памятью
  401. и подкачкой, которое призвано помочь минимизировать нагрузку на
  402. ввод/вывод, а не заменить обычную подкачку на диске целиком как это
  403. делает ZRAM. Вытеснная страница при включенном Zswap имеет
  404. следующий цикл жизни: RAM -> Zswap -> Подкачка. Если процесс
  405. обратиться к странице, которая была вытеснена в Zswap, но которая
  406. так и не попала в подкачку на диске, то тогда ядро просто распакует
  407. её внутри памяти готовой для использования. В случае если она всё
  408. таки была вытеснена на диск, ядро считает её с диска и загрузит в
  409. память, как это обычно и происходит без zswap.
  410. Об установке ZRAM было уже коротко рассказано в разделе
  411. :ref:`services`. Однако не во всех дистрибутивах Linux есть служба
  412. ``zram-generator``, поэтому покажем универсальный способ его
  413. настройки, основанный на обычных правилах udev.
  414. Прежде чем мы перейдем к настройке ZRAM надо уточнить, что
  415. одновременное использование ZRAM и zswap имеет неопределенный эффект.
  416. С одной стороны, это вполне возможно, и в этом случае Zswap становится
  417. промежуточным буфером уже для ZRAM, но это не имеет особого смысла,
  418. так как они оба занимаются одним и тем же - сжатием данных внутри ОЗУ.
  419. ZRAM также ведет свою статистику о том, какие страницы и в каком
  420. количестве были сжаты, и которая может быть искажена, в силу того что
  421. помимо него в системе может работать Zswap, поэтому настоятельно
  422. рекомендуется его отключить перед использованием ZRAM. Для этого
  423. достаточно указать параметр ядра ``zswap.enabled=0`` в конфиге вашего
  424. загрузчика, либо деактивировать прямо во время работы системы::
  425. echo 0 > /sys/module/zswap/parameters/enabled
  426. Если у вас затруднения с настройкой вашего загрузчика (а такое вполне
  427. может быть на атомарных системах), то вы можете настроить его
  428. перманентное отключение через создание файла в директории
  429. ``/etc/tmpfiles.d`` со следующим содержимым:
  430. .. code-block:: shell
  431. :caption: ``sudo nano /etc/tmpfiles.d/90-disable-zswap.conf``
  432. w! /sys/module/zswap/parameters/enabled - - - - 0
  433. .. note:: Важно отметить, что для использования ZRAM вам вовсе не
  434. обязательно отключать обычную подкачку, если она у вас до этого
  435. была настроена. В этом случае ядро по умолчанию будет использовать
  436. в качестве основной подкачки тот раздел или файл, примонтированный
  437. в служебную точку монтирования ``[swap]``, который имеет приоритет
  438. выше, чем другой. Поэтому если вы установите для ZRAM приоритет
  439. ``100``, как мы это сделаем ниже в файле ``/etc/fstab``, то обычная
  440. подкачка на диске станет использоваться ядром только как запасная в
  441. случае если ZRAM переполнится, либо при использовании функции
  442. гибернации, которая может работать только с подкачкой на диске.
  443. Перейдем к настройке ZRAM. Обратите внимание, что среди "мейнстримных"
  444. дистрибутивов Linux (как например Fedora) ZRAM начинают поставлять по
  445. умолчанию вместо обычной подкачки на диске. Поэтому сначала проверьте,
  446. не задействован ли уже ZRAM в вашей системе. Сделать это можно очень
  447. просто через команду ``zramctl``, либо проверив по наличию файла
  448. ``/dev/zram0``, который представляет собой блочное устройство куда
  449. будут попадать все вытесняемые ядром страницы (этакий виртуальный
  450. раздел подкачки).
  451. Если же нет, то продолжаем. Для начала нам нужно форсировать загрузку
  452. модуля ZRAM, для этого нужно создать файл в директории
  453. ``/etc/modules-load.d/30-zram.conf`` и прописать в него всего одну
  454. строчку:
  455. .. code-block:: shell
  456. :caption: ``sudo nano /etc/modules-load.d/zram.conf``
  457. zram
  458. Теперь используя правила udev, мы будем создавать наше блочное
  459. устройство ``/dev/zram0`` и делать из него раздел подкачки. Для этого
  460. создадим файл в директории ``/etc/udev/rules.d/30-zram.rules``:
  461. .. code-block:: shell
  462. :caption: ``sudo nano /etc/udev/rules.d/30-zram.rules``
  463. ACTION=="add", KERNEL=="zram0", ATTR{comp_algorithm}="zstd", \
  464. ATTR{disksize}="8G", \
  465. RUN="/usr/bin/mkswap -U clear /dev/%k", TAG+="systemd"
  466. Теперь подробно о том, что из себя представляет само udev правило. В
  467. начале мы указываем при каком действии мы хотим, чтобы оно
  468. срабатывало. В нашем случае это ``ACTION=="add"``, то есть появление
  469. нового блочного устройства под названием ``KERNEL=="zram0"``. Это
  470. блочное устройство создается ядром автоматически при загрузке модуля
  471. ZRAM, форсированную загрузку которого мы уже прописали выше. Здесь
  472. можно заметить, что все проверки в правилах udev осуществляются через
  473. ``==``.
  474. А дальше мы говорим, что в этом случае нужно делать. Во-первых, мы
  475. меняем значение атрибута (в udev правилах все они пишутся как
  476. ``ATTR{name}``, где *name* - имя атрибута) ``comp_algorithm`` нашего
  477. блочного устройства, который указывает на используемый алгоритм
  478. сжатия. Для ZRAM в ядре предложены три алгоритма сжатия: ``lzo``,
  479. ``lz4``, ``zstd``. В подавляющем большинстве случаев вы должны
  480. использовать только ``zstd``, так как это наиболее оптимальный
  481. алгоритм по соотношению скорости/эффективности сжатия. LZ4 может быть
  482. быстрее при расжатии, но в остальном он не имеет больших преимуществ.
  483. LZO следует использовать только на очень слабых процессорах, которые
  484. просто не тянут сжатие большого объема страниц через Zstd.
  485. Следующим атрибутом мы меняем ``disksize`` - это размер блочного
  486. устройства. Теперь очень важно: размер блочного устройства - это тот
  487. объем **несжатых страниц**, который может попасть внутрь ZRAM, и он
  488. может быть равен объему ОЗУ или даже быть в два раза больше него. Как
  489. это возможно? Представим, что у вас 4 Гб ОЗУ. Вы устанавливаете объем
  490. ZRAM тоже в 4 Гб. Вы полностью забиваете всю свою память, открывая 300
  491. вкладок в Chromium, и любой системный монитор или аналог ``htop``
  492. покажет вам, что подкачка тоже полностью забита, но проблема в том,
  493. что это тот размер страниц, которые попали в ZRAM до сжатия. То есть на
  494. деле у вас в ОЗУ вытесненные страницы занимают в разы меньший объем
  495. *из-за сжатия*. Увидеть это можно через команду ``zramctl``, вывод
  496. которой может быть следующим::
  497. NAME ALGORITHM DISKSIZE DATA COMPR TOTAL STREAMS MOUNTPOINT
  498. /dev/zram0 zstd 15G 1G 232M 243.3M 16 [SWAP]
  499. Здесь колонка ``DATA`` показывает какой объем страниц попал в
  500. ``/dev/zram0``. Если вы опять откроете ``htop`` или другой аналог
  501. системного монитора, то вы увидите точно такой же объем того сколько у
  502. вас "занято" подкачки, но вот колонка ``COMPR`` показывает уже
  503. реальный размер вытесненных внутрь ZRAM страниц *после сжатия*,
  504. который очевидно будет меньше в 2-3 раза. Именно поэтому я рекомендую
  505. вам установить объем блочного устройства ZRAM, который в два раза
  506. больше, чем объем всей вашей памяти (Значение ``8Gb`` - **это лишь
  507. пример**, замените его на то, какой объем вашей памяти и умножьте
  508. это на **два**). Конечно, здесь нужно оговориться, что не все страницы
  509. бывают так уж хорошо сжимаемыми, но в большинстве случаев они будут
  510. помещаться без каких-либо проблем.
  511. Надеюсь это добавило понимание того, почему не всегда нужно верить
  512. цифрам, которые вам говорит, например, команда ``free``. Завершает наше
  513. udev правило действие, которое мы хотим сделать с нашим блочным
  514. устройством - запустить команду ``mkswap``, чтобы сделать из нашего
  515. ``/dev/zram0`` раздел подкачки.
  516. Всё, что нам осталось теперь - это добавить запись в ``/etc/fstab``,
  517. что ``/dev/zram0`` это вообще-то наша подкачка и установить ей
  518. приоритет ``100``.
  519. .. code-block:: shell
  520. :caption: ``sudo nano /etc/fstab``
  521. /dev/zram0 none swap defaults,pri=100 0 0
  522. На этом все, теперь можно перезагружаться и проверять работу через
  523. ``zramctl``. Если такой способ для вас показался слишком сложным, то
  524. обратитесь к использованию ``zram-generator`` как уже было показано
  525. ранее.
  526. Значение же ``vm.swappiness`` при использовании ZRAM рекомендуется
  527. установить в ``150``, так как более низкие значения приведут к
  528. излишнему вытеснению из файлового кэша, а анонимные страницы, которые
  529. потенциально могут быть легко сжаты, будут вытесняться в последний
  530. момент, что нежелательно. А вот при значении ``150``, файловый кэш
  531. будет дольше оставаться нетронутым, благодаря чему обращения к ранее
  532. открытым файлам останутся быстрыми, но при этом анонимные страницы
  533. просто сожмутся внутри памяти. Такой подход минимизирует нагрузку на
  534. ввод/вывод.
  535. .. [#] https://docs.kernel.org/admin-guide/blockdev/zram.html
  536. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  537. Отключение упреждающего чтения
  538. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  539. Из-за того, что процесс чтения вытесненной в подкачку страницы с диска
  540. и её записи обратно в оперативную память является довольно
  541. дорогостоящей операцией, ядро использует некоторые трюки, для того
  542. чтобы делать их как можно реже. Один из таких трюков это "упреждающее
  543. чтение" (*readahead*), когда при обращении процесса к вытесненной
  544. странице, ядро считывает не только запрошенную страницу, но и ещё
  545. некоторое количество страниц последовательно следующих за ней внутри
  546. подкачки.
  547. Смысл здесь в том, что страница на практике это очень маленький
  548. фрагмент данных, которыми оперирует процесс, поэтому с большой долей
  549. вероятности обратившись к одной 4 Кб странице, процесс сделает ещё два
  550. и более запросов к тем страницам, которые тоже могли быть вытеснены в
  551. подкачку и быть записанными в него после той, которую процесс
  552. запрашивает в данный момент, и чтобы их потом тоже не искать и не
  553. читать ядро делает это сразу вместе с той вытесненной страницей,
  554. которую запросил процесс сейчас, так скажем, двух зайцев одним
  555. выстрелом.
  556. Количество таких последовательно считываемых страниц за раз
  557. контролируется значением параметра ``vm.page-cluster``. Это значение
  558. является степенью двойки, возведя в которую и можно получить
  559. количество страниц. Например, если установлено значение ``1``, то
  560. количество страниц, которые ядро считает заранее, будет равно 2^1, то
  561. есть просто два. Если значение параметра равно ``2``, то количество
  562. страниц уже будет равно в 2^2, то есть ``4`` и так далее. При значении
  563. ``0`` количество страниц будет 2^0, то есть 1 - это значение отключает
  564. упреждающее чтение страниц из подкачки.
  565. На первый взгляд всё звучит здорово, и надо бы выкрутить значение
  566. побольше, чтобы ядро читало больше страниц за раз, но есть одна
  567. маленькая проблема, из-за которой я настоятельно рекомендую отключать
  568. этот параметр. Дело в том, что ядро считывает из подкачки страницы,
  569. которые были записаны по порядку за той страницей, которая в данный
  570. момент запрошена для загрузки обратно в память. Мы подразумевали, что
  571. это будут страницы того же процесса, который запросил данную страницу,
  572. но на деле это может вообще не так. Ядро записывает страницы из памяти
  573. в подкачку в том порядке, в котором они были вытеснены, и они вообще
  574. не обязательно могут относится к одному и тому же процессу, а даже
  575. если к одному, то могут быть совсем не теми, которые процесс запросит
  576. в будущем. Короче говоря, с упреждающим чтением мы играем в своего
  577. рода рулетку, повезет или нет. Но в подавляющем большинстве случаев
  578. ядро просто вернет в память обратно ещё 8 страниц (согласно значению
  579. по умолчанию), которые могут никогда не пригодиться в будущем, а если
  580. они не пригодятся, то их придется опять вытеснять в подкачку.
  581. Таким образом, упреждающее чтение не только не решает заявленную
  582. проблему, но и наоборот её усугубляет. Для ZRAM это, конечно, может и
  583. не так критично, так как это вызовет лишь дополнительные циклы
  584. сжатия/расжатия страниц, но это в любом случае холостая работа. По
  585. этой причине разработчики ChromeOS и Android отключают данный параметр
  586. в своих системах по умолчанию [#]_ [#]_, что советую сделать и вам. Для этого
  587. как обычно достаточно просто прописать значение в конфиге sysctl:
  588. .. code-block:: shell
  589. :caption: ``sudo nano /etc/sysctl.d/99-sysctl.conf``
  590. vm.page-cluster = 0
  591. .. [#] https://issues.chromium.org/issues/41028506
  592. .. [#] https://chromium.googlesource.com/chromiumos/overlays/chromiumos-overlay/+/HEAD/chromeos-base/chromeos-base/files/00-sysctl.conf#116
  593. .. _mglru:
  594. ---------------
  595. Алгоритм MGLRU
  596. ---------------
  597. Мы уже говорили, что LRU - это алгоритм используемый ядром Linux для
  598. ведения учёта количества обращений ко всем страницам внутри памяти,
  599. позволяющий составлять выборку тех страниц, которые реже всего
  600. используются процессами и соответственно могут быть спокойно вытеснены
  601. в подкачку. Но начиная с версии 6.1 в ядре появилась альтернативная
  602. реализация этого алгоритма, называемая *MGLRU* (Multi-Generational
  603. LRU) [#]_. Принципиальное отличие MGLRU от простого LRU алгоритма
  604. состоит в том, что выборка страниц, которые должны быть вытеснены,
  605. формируется не на основе только лишь одного признака (количества
  606. обращений к странице), а на основе целых двух признаков - количества
  607. обращений и времени последнего обращения. По этой причине новый
  608. алгоритм объединяет все страницы в так называемые "поколения" на
  609. основе времени обращения к ним, собственно именно поэтому его название
  610. и можно дословно перевести как "Многопоколенный LRU". Такой подход
  611. позволяет добиться большей точности в выборе из имеющихся страниц тех,
  612. которые по настоящему используются реже других, что в свою очередь
  613. позволяет уменьшать количество операций возврата страниц из подкачки,
  614. ибо чем точнее работает алгоритм выборки, тем больше вероятность, что
  615. вытесненная страница действительно никогда больше не понадобится и её
  616. не надо будет считывать с диска и загружать обратно в память.
  617. Для того чтобы проверить собрана ли ваша версия ядра с поддержкой
  618. MGLRU достаточно прописать одну команду::
  619. zgrep "CONFIG_LRU_GEN_ENABLED" /proc/config.gz
  620. Если вывод команды не пустой, значит ваша текущая версия ядра собрана
  621. с поддержкой данного алгоритма, но это вовсе не значит, что он
  622. используется по умолчанию. Алгоритм MGLRU можно бесприпятственно
  623. включить или выключить прямо во время работы системы. Проверить статус
  624. работы алгоритма можно через файл ``/sys/kernel/mm/lru_gen/enabled``::
  625. cat /sys/kernel/mm/lru_gen/enabled
  626. Если вывод команды равен ``0x0000``, значит MGLRU выключен, и его
  627. нужно самостоятельно включить следующей командой::
  628. echo "y" | sudo tee /sys/kernel/mm/lru_gen/enabled
  629. Обратите внимание, что в большинстве дистрибутивов Linux версии ядра с
  630. поддержкой MGLRU поставляются по умолчанию, поэтому никаких
  631. дополнительных действий для его включения делать как правило не нужно.
  632. .. [#] https://docs.kernel.org/admin-guide/mm/multigen_lru.html
  633. .. _page_trashing_prevention:
  634. ~~~~~~~~~~~~~~~~~~~~~~~~
  635. Защита от Page Trashing
  636. ~~~~~~~~~~~~~~~~~~~~~~~~
  637. Одним из преимуществ алгоритма MGLRU над своим предшественником
  638. является предоставление дополнительной защиты от ситуаций Page
  639. Trashing.
  640. Page Trashing - это ситуация острой нехватки памяти, при которой
  641. памяти становится настолько мало, что ядро начинает вытеснять в
  642. подкачку даже те страницы, которые активно используются процессами во
  643. время своей работы, так как все остальные малоиспользуемые страницы
  644. уже были вытеснены. Это приводит к тому, что количество операций
  645. возврата страниц из подкачки многократно увеличивается, так как к
  646. данным часто используемым страницам все время обращаются процессы,
  647. из-за чего ядру приходится читать их из подкачки с диска или
  648. распаковывать их из памяти, если речь идёт про ZRAM, и заново
  649. загружать память, после чего снова их вытеснять, так как других
  650. кандидатов для этого больше не осталось. Такой цикл становится очень
  651. заметным для пользователя, так как он порождает кратковременные
  652. зависания системы, ибо процессу каждый раз приходится ожидать, пока
  653. ядро достанет страницы из подкачки и загрузит их обратно в память.
  654. Конечно, если потребление памяти в этом случае продолжит расти, то мы
  655. столкнемся с ситуацией Out Of Memory (OOM), после чего либо
  656. специальный демон по наводке ядра убьёт самый прожорливый процесс,
  657. чтобы освободить память, либо система полностью зависнет. Если
  658. потребление останется тем же, то мы продолжим испытывать постоянные
  659. микрозависания, что не очень приятно.
  660. Здесь на сцену выходит алгоритм MGLRU, который хоть и не позволяет на
  661. 100% защититься от таких ситуаций, но позволяет убрать те самые
  662. кратковременные зависания, сделав систему более стрессоустойчивой и
  663. отзывчивой в условиях нехватки ОЗУ. Суть защиты состоит в том, что
  664. MGLRU предотвращает вытеснение "рабочего набора" страниц процесса (то
  665. есть таких страниц, которые действительно активно используются) в
  666. течении ``N`` миллисекунд, оставляя их не тронутыми в памяти на
  667. протяжении по крайне мере этого гарантированного времени. В этом
  668. случае процессам не придется каждый раз ожидать долгого восстановления
  669. страниц из подкачки и они сохранят свою скорость работы, но с другой
  670. стороны это увеличивает шанс возникновения ситуаций OOM, так как чем
  671. больше разрастается такой "рабочий набор" страниц, тем больше
  672. потребление памяти. По этой причине данный механизм защиты выключен по
  673. умолчанию, так как возникновение OOM ситуаций часто нежелательно на
  674. серверах и системах с большой нагрузкой, не предназначенных для
  675. интерактивного использования, где такие небольшие зависания были бы
  676. заметны глазу.
  677. Для того чтобы включить данный механизм при использовании MGLRU нам
  678. нужно изменить значение параметра ``min_ttl_ms`` (по умолчанию 0),
  679. который как раз таки и устанавливает то время в миллисекундах, в
  680. течении которого рабочий набор страниц не будет вытесняться. Автор
  681. рекомендует брать значение от ``1000`` (это одна секунда), но не
  682. большее ``5000``, ибо это приведет к более частому возникновению OOM.
  683. Оптимальное значение для большинства - ``2000`` (2 секунды). В этом
  684. случае система достаточно сохранит свою интерактивность под нагрузкой.
  685. Указать значение можно как всегда через псевдофайловую систему sysfs,
  686. для автоматизации процесса воспользуемся файлом конфигурации
  687. ``systemd-tmpfiles``:
  688. .. code-block:: shell
  689. :caption: ``sudo nano /etc/tmpfiles.d/90-page-trashing.conf``
  690. w! /sys/kernel/mm/lru_gen/min_ttl_ms - - - - 2000
  691. .. _dirty_pages:
  692. ---------------------------
  693. Настройка грязных страниц
  694. ---------------------------
  695. В теоретическом разделе про работу памяти в Linux мы уже говорили, что
  696. ядро отображает всю информацию об обычных файлах в виде кусочков -
  697. файловых страниц, при этом реальную память данная страница получает
  698. только непосредственно когда какая-то программа, то есть процесс
  699. начинает что-то читать или писать в файл, и если точнее, в определенное
  700. место внутри файла ассоциированное с данной страницей. Со чтением все
  701. понятно, мы просто сохраняем считанный набор байт с диска в память и
  702. многократно переиспользуем результат. Но что происходит в случае с
  703. записью?
  704. Когда какой-то процесс начинает писать изменения в файл, то эти
  705. изменения сначала попадают в файловые страницы, но так как
  706. подразумевается, что проделанные изменения происходят с реальными
  707. файлами на диске, то перед ядром возникает задача синхронизации
  708. изменений между страничным кэшом и диском. С этой целью все измененные
  709. файловые страницы помечаются как "грязные" (*dirty pages*). Ядро ведет
  710. учёт таких страниц и в фоновом режиме, при определенных условиях, о
  711. которых пойдет речь далее, начинает "сбрасывать" такие страницы на
  712. диск, то есть записывать изменения над файлами уже по настоящему.
  713. Смысл от такого буферизированного подхода состоит в том, чтобы
  714. минимизировать количество реальных операций записи, ибо приложения как
  715. правило большую часть времени не добавляют новые данные внутрь
  716. файла, а изменяют уже существующие и могут делать это много раз подряд
  717. в течение времени своей работы. Если приложение X изменяет 10 раз один
  718. и тот же файл в одном месте с малыми интервалами между такими
  719. операциями записи, то нет никакого смысла делать запись сразу же, ведь
  720. чем дольше ядро удерживает изменения внутри страничного кэша, тем
  721. больше уменьшает количество конечных, настоящих записей на диск, и
  722. вместо 10 операцией записи на диск мы можем получить одну запись уже
  723. итогового варианта изменений. Однако такой подход порождает и
  724. определенные риски, так как избыточное кэширование изменений внутри
  725. ОЗУ может привести к потери данных в случае отключения питания или
  726. непредвиденного зависания системы.
  727. Стоит также отметить, что у приложений остается возможность выполнять
  728. прямую запись в файл минуя страничный кэш. Первый способ это
  729. использование *Direct I/O* (буквально: прямой ввод/вывод). Для его
  730. применения приложению нужно установить специальный флаг при открытии
  731. файла - ``O_DIRECT``, после чего все операции над этим файлом будут
  732. производиться в обход страничного кэша. Второй способ заключается в
  733. том, чтобы использовать страничный кэш большую часть времени работы
  734. программы, но форсировать его "промывку" (термин "промывка" (flush)
  735. является антонимом к слову "грязный") в определенные моменты времени,
  736. например при окончании работы с файлом или его сохранении в текстовом
  737. редакторе. В этом случае приложение выполняет системные вызовы
  738. ``sync()`` или ``fsync()``, которые сигнализируют ядру о том, что
  739. нужно в принудительном порядке записать все проделанные им изменения
  740. из страничного кэша на диск.
  741. Но вернемся к тому, как именно ядро сбрасывает грязные страницы на
  742. диск. За это отвечают так называемые специальные ядерные потоки
  743. ``pdflush``, которые производят "промывку" грязных страниц в фоновом
  744. режиме при соблюдении некоторых условий. Во-первых, данные потоки
  745. начинают работать только тогда, когда набирается необходимый общий
  746. объем грязных страниц, который устанавливается параметрами
  747. ``vm.dirty_background_ratio`` или ``vm.dirty_background_bytes``. До
  748. тех пор пока указанная нижняя граница не будет достигнута, изменения
  749. внутри грязных страниц так и будут оставаться в ОЗУ, за тем
  750. исключением, если, как и было указано выше, процесс явно не попросит
  751. записать на диск изменения через вызовы ``sync()`` или ``fsync()``.
  752. При этом важно отметить, что если страница была изменена процессом, то
  753. при штатной работе потоков ``pdflush`` без принудительной промывки со
  754. стороны самого приложения, страница становится готовой к записи не
  755. сразу же, а только по истечению времени указанного в качестве значения
  756. параметра ``vm.dirty_expire_centisecs``, которое представлено в виде
  757. сантисекунд (одна сотая от секунды) и по умолчанию равно ``3000`` [4]_
  758. (30 секунд).
  759. После запуска потоков ``pdflush`` их работа происходит не непрерывно
  760. как можно было бы подумать, а с интервалами между которыми они
  761. просыпаются и выполняют часть работы. Время этих промежутков
  762. определяется значением параметра ``vm.dirty_writeback_centisecs``, так
  763. же принимающего значение в виде сантисекунд и равного по умолчанию
  764. ``500`` [4]_, то есть 5 секунд, что весьма много, но это гарантирует,
  765. что потоки ``pdflush`` не будут создавать чрезмерной нагрузки.
  766. Наконец, существует также верхняя граница, которая определяет
  767. максимально возможный объем грязных страниц. Она определяется
  768. значением параметра ``vm.dirty_ratio``, либо ``vm.dirty_bytes``. Если
  769. к тому времени, когда потоки ``pdflush`` начали свою работу, объем
  770. грязных страниц продолжал увеличиваться с такой скоростью, что ядро
  771. просто не успевало записать все поступающие изменения на диск, то
  772. возникает так называемый "троттлинг" ввода/вывода.
  773. В старых версиях ядра "троттлинг" ввода/вывода проявлялся только
  774. непосредственно по достижению верхней границы количества грязных
  775. страниц, и приводил к полной блокировке всех операций ввода/вывода до
  776. тех пор пока потоки ``pdflush`` полностью не запишут уже накопленные
  777. ранее изменения на диск. Это приводило к очень печальным последствиям,
  778. в том числе известный баг в ядре `12309
  779. <https://bugzilla.kernel.org/show_bug.cgi?id=12309>`_ был связан
  780. именно с тем, что интенсивная запись каким-либо процессом на носитель
  781. с очень низкой скоростью (вроде простой USB флешки) приводила к ярко
  782. выраженным зависаниям всей системы, так как операции I/O
  783. блокировались, а фоновые потоки ``pdflush`` не могли быстро записать
  784. изменения в силу аппаратных ограничений самого носителя.
  785. В новых версиях ядра были предприняты большие усилия к исправлению
  786. данной проблемы [#]_, и в конце концов было принято решение, которое можно
  787. охарактеризовать как "размывание" процесса троттлинга во времени. То
  788. есть, когда текущий объем грязных страниц начинает быть равным
  789. примерно 1/2 между значениями ``vm.dirty_background_bytes`` (или
  790. ``vm.dirty_background_ratio``) и ``vm.dirty_bytes`` (или
  791. ``vm.dirty_ratio``), то есть между нижней и верхней границей
  792. соответственно, то тогда ядро начинает постепенно создать
  793. кратковременные паузы (блокировки) в работе ввода/вывода для процесса,
  794. в результате работы которого появляется большое количество грязных
  795. страниц, так чтобы потоки ``pdflush`` успевали обработать уже
  796. накопленные грязные страницы. Такие палки в колеса активно пишущему
  797. процессу закономерно приводят к падению пропускной способности записи,
  798. но позволяют избавиться от эффекта "голодания", когда один процесс
  799. полностью оккупирует всю квоту на грязные страницы, не позволяя ничего
  800. писать другим процессам, а также от полных блокировок ввода/вывода,
  801. так как в случае достижения верхней границы ядро просто тормозит
  802. работу ввода/вывода для процесса таким образом, чтобы потоки
  803. ``pdflush`` гарантированно могли записать все полученные грязные
  804. страницы до снятия блокировки, как правило тем самым уравнивая
  805. скорость записи грязных страниц приложением со скоростью записи
  806. потоков ``pdflush`` [#]_, [#]_.
  807. Учитывая количество параметров, контролирующих поведение грязных
  808. страниц и факторов, оказывающих влияние на их работу, возникает вполне
  809. закономерный вопрос о том, как это настроить оптимальным образом для
  810. своей конфигурации и задач? Для начала, как вы уже могли заметить,
  811. существует некоторая двойственность в вопросе указания нижней и
  812. верхней границы работы потоков ``pdflush``, так для их настройки
  813. существует две пары настроек ``vm.dirty_background_bytes`` и
  814. ``vm.dirty_bytes`` или ``vm.dirty_background_ratio`` и
  815. ``vm.dirty_background_ratio``. Несмотря на то, что обе пары
  816. контроллируют по сути одно и то же, они конфликтуют друг с другом, то
  817. есть указать можно только один из пары, так как указание одного
  818. отменяет значение другого. Кроме того существует некоторая разница в
  819. их семантике. Все параметры с окончанием ``ratio`` указывают процент
  820. от *свободной в данный момент памяти*, который могут занимать грязные
  821. страницы вообще (в случае с ``vm.dirty_ratio``) или же пороговое
  822. значение для начала работы потоков ``pdflush``
  823. (``vm.dirty_background_ratio``). Частое заблуждение относительно этой
  824. пары параметров состоит в том, что процент берется от общего
  825. количества памяти в целом, а не от свободной, что приводит к
  826. неправильным умозаключения о выборе значения в зависимости от объема
  827. памяти.
  828. В целом, по мнению автора, использование параметров ``vm.dirty_ratio``
  829. и ``vm.dirty_background_ratio`` нежелательно, так как их поведение не
  830. является строго фиксированным и объем грязных страниц таким образом
  831. находится в обратной пропорциональной зависимости по отношению к
  832. текущему уровню потребления памяти, который склонен к тенденции
  833. увеличения в процессе работы системы больше, чем к уменьшению. Скажем,
  834. мы можем взять 2% от 32 Гб в качестве значения к параметру
  835. ``vm.dirty_ratio``. Если в моменте вся память свободна (что, конечно,
  836. в действительности невозможно), мы получаем максимальный объем грязных
  837. страниц равный примерно 678 Мб, что на первый взгляд много, но
  838. среднестатический пользователь гораздо чаще открывает новые вкладки в
  839. браузере или открывает новые приложения, чем их закрывает, поэтому
  840. легко представить ситуацию, когда даже с 32 Гб ОЗУ вы достигаете
  841. уровня потребления 28 Гб ОЗУ, к примеру, компилируя что-то внутри
  842. tmpfs, и в этом случае объем грязных уже будет составлять всего 85 Мб
  843. и дальше ещё больше уменьшаться. То есть, по существу использование
  844. параметров с окончанием ``ratio`` приводит к тому, что большую часть
  845. времени работы системы объем грязных страниц представляет собой
  846. убывающую геометрическую прогрессию. В то же время другая пара
  847. параметров, ``vm.dirty_bytes`` и ``vm.dirty_background_bytes``, не
  848. имеет такой зависимости [#]_ и позволяет однозначно определить порог
  849. грязных страниц для начала работы потоков ``pdflush`` и установить
  850. максимально возможный объем грязных страниц вне зависимости от
  851. текущего уровня потребления памяти.
  852. Сами же значения к ``vm.dirty_bytes`` и ``vm.dirty_background_bytes``
  853. следует выбирать в зависимости от ваших целей и задач, но для
  854. домашнего использования в качестве ``vm.dirty_bytes`` разумно брать
  855. тот объем данных, который ваш основной носитель может обработать за
  856. единицу времени, то есть его пропускную способность, так как тогда
  857. даже в худшем случае указанный объем грязных страниц может быть
  858. записан достаточно быстро. Значение же ``vm.dirty_background_bytes``
  859. как правило лучше делать равным 1/2 или даже 1/4 от значения
  860. ``vm.dirty_bytes``, так как чем больше "расстояние" между порогом к
  861. запуску потоков ``pdflush`` и максимальным объемом грязных страниц,
  862. тем меньше вероятность столкнутся с эффектом троттлинга и падением
  863. пропускной способности записи. Так же слишком завышенное значение
  864. ``vm.dirty_background_bytes`` черевато "застоем" данных внутри ОЗУ,
  865. что сулит риски их потери при отключении питания или зависаниях
  866. системы. Нужно понимать, что сверхвысокие значения просто не имеют
  867. смысла при простом домашнем использовании, так как рядовой
  868. пользователь не имеет приложений, которые могли бы иметь большую
  869. интенсивность записи данных на диск, как например СУБД. Как правило
  870. самыми интенсивными приложениями с точки зрения записи остаются
  871. торрент клиенты, Steam, и другие программы для загрузки контента,
  872. однако объем данных, который они записывают на диск ограничен
  873. пропускной способностью вашего сетевого канала, который у большинства
  874. людей хоть и чисто номинально составляет 100 Мб/c, однако в ряде
  875. случаев оказывается куда ниже, так что сверх большие объемы грязных
  876. страниц указывать просто нет смысла. В качестве начальных значений, на
  877. которые можно было бы опереться, автор рекомендует взять 32 или 64 Мб в
  878. качестве ``dirty_background_bytes`` и 256 Мб в качестве
  879. ``dirty_bytes``:
  880. .. code-block:: shell
  881. :caption: ``sudo nano /etc/sysctl.d/30-dirty-pages.conf``
  882. vm.dirty_background_bytes=67108864
  883. vm.dirty_bytes=268435456
  884. Вы вправе кратно уменьшить значение параметра ``vm.dirty_bytes``,
  885. если у вас медленный HDD, или же наоборот увеличить вплоть до 1-2 Гб,
  886. если имеете сверхбыстрый носитель и высокую скорость передачи данных
  887. по сети.
  888. Что касается значений параметров ``vm.dirty_expire_centisecs`` и
  889. ``vm.dirty_writeback_centisecs``, которые управляют частотой работы
  890. ``pdflush`` потоков, то вы могли заметить, что значения по умолчанию
  891. сильно завышены. Ожидать 30 секунд, как предписывает значение по
  892. умолчанию параметра ``vm.dirty_expire_centisecs``, перед тем чтобы
  893. позволить записывать ``pdflush`` новую грязную страницу кажется
  894. чрезмерным, поэтому разумно уменьшить значение данного параметра вдвое,
  895. то есть сократить период ожидания до 15 секунд, либо же ещё
  896. меньше, но устанавливать сверх низкие значения вроде 1-3 секунд также
  897. не рекомендуется, так как это может свести на нет все преимущества
  898. кэширования при записи. Оптимальным, по мнению автора, является
  899. значение в 15 секунд, то есть значение ``1500`` при переводе в
  900. сантисекунды:
  901. .. code-block:: shell
  902. :caption: ``sudo nano /etc/sysctl.d/30-dirty-pages-expire.conf``
  903. vm.dirty_expire_centisecs=1500
  904. Интервал времени между периодами работы потоков ``pdflush``
  905. определяемый параметром ``vm.dirty_writeback_centisecs`` так же можно
  906. уменьшить, так как современные SSD носители достаточно хорошо
  907. справляются с интенсивной нагрузкой, поэтому можно увеличить частоту
  908. работы ``pdflush`` потоков и таким образом ещё больше уменьшить шансы
  909. на столкновение с эффектом троттлинга при записи:
  910. .. code-block:: shell
  911. :caption: ``sudo nano /etc/sysctl.d/30-dirty-pages-writeback.conf``
  912. vm.dirty_writeback_centisecs=100
  913. .. [#] https://unix.stackexchange.com/questions/480399/why-were-usb-stick-stall-problems-reported-in-2013-why-wasnt-this-problem-so/480400#480400
  914. .. [#] https://github.com/torvalds/linux/blob/fb527fc1f36e252cd1f62a26be4906949e7708ff/mm/page-writeback.c#L410-L411
  915. .. [#] https://stackoverflow.com/a/73808616
  916. .. [#] https://lwn.net/Articles/456904/
  917. .. _vfs_cache_pressure:
  918. -------------------
  919. Настройка кэша VFS
  920. -------------------
  921. В страничный кэш попадают не только файловые страницы, в которых
  922. хранятся непосредственно данные считываемые с диска, но и метаданные к
  923. файлам и директориям. Доступ к ним осуществляется через так называемые
  924. индексные дескрипторы (*inode*) - специальные структуры, которые
  925. используются вашей файловой системой для хранения атрибутов, прав
  926. доступа и прочей служебной информации, а также они содержат номера
  927. секторов диска, которые указывают, где хранятся данные самого файла на
  928. носителе.
  929. Перед открытием любого файла или дириктории сначала нужно выполнить
  930. его поиск на файловой системе, и это не самая быстрая операция как
  931. может показаться, даже несмотря на различные оптимизации,
  932. предоставляемые современными файловыми системами такими как
  933. использование B-деревьев для быстрого прохода по ним. В результате
  934. этой операции ядро как раз таки находит нужный индексный дескриптор,
  935. имея который можно обратиться к данным файла. Поэтому ядро кэширует
  936. все используемые во время работы системы дескрипторы и информацию о
  937. директориях внутри VFS [#]_ кэша, для того чтобы сделать все
  938. последующие обращения к файлам быстрыми, потому что ядро уже будет
  939. знать про них всё, что нужно.
  940. Но все эти метаданные так или иначе занимают место внутри памяти,
  941. поэтому когда ядро начинает "промывку" (flush) страничного кэша, то
  942. оно вытесняет из него как данные самих файлов, так и метаданные для
  943. них. В ядре также есть специальный параметр sysctl
  944. ``vm.vfs_cache_pressure``, который как раз таки регулирует, что будет
  945. вытесняться в первую очередь - сами данные или метаданные из кэша VFS.
  946. Здесь всё по аналогии с параметром ``vm.swappiness``. При значении
  947. равном ``100`` (значение по умолчанию) ядро будет пытаться равномерно
  948. выгружать из памяти как кусочки содержимого самих файлов, так и
  949. индексные дескрипторы из кэша VFS. При значениях меньше ``100`` ядро
  950. будет больше отдавать предпочтение хранению метаданных в памяти, при
  951. значениях больше ``100`` - наоборот, больше избавляться от них в
  952. пользу обычных данных считываемых с диска.
  953. Для наилучшего быстродействия системы рекомендуется устанавливать
  954. значение равным ``50`` [#]_, при котором вытеснение страниц,
  955. относящихся к VFS кэшу, происходит реже, чем для обычных файловых
  956. страниц, так как метаданные имеют большую ценность по сравнению с
  957. данными самих файлов, которые можно достаточно быстро повторно считать
  958. в страничный кэш на большинстве SSD накопителей при наличии индексного
  959. дескриптора файла, который как раз таки хранится внутри VFS кэша. Для
  960. сохранения значения как и всегда пропишем его в конфигурационный файл
  961. sysctl:
  962. .. code-block:: shell
  963. :caption: ``sudo nano /etc/sysctl.d/90-vfs-cache.conf``
  964. vm.vfs_cache_pressure = 50
  965. Конечно, лучший способ увеличения быстродействия ввод/вывода это
  966. кэшировать как можно больше данных в памяти, так как это самое быстрое
  967. устройство хранения в вашем компьютере (без учета кэша процессора),
  968. поэтому лучше всего как можно больше минимизировать вытеснение страниц
  969. из страничного кэша, но мы это уже сделали в разделе про настройку
  970. подкачки, установив большое значение параметра ``vm.swappiness`` и
  971. используя ZRAM для сжатия анонимных страниц прямо внутри памяти.
  972. .. [#] VFS (Virtual File System) - виртуальная файловая система, на
  973. деле является программным интерфейсом, который абстрагирует всё
  974. взаимодействие между конкретной файловой системой (Btrfs/ext4/xfs и
  975. т.д.) и программами, позволяя тем осуществлять запись и чтение
  976. ничего не зная о том, какая именно файловая система используется в
  977. данный момент.
  978. .. [#] https://github.com/xanmod/linux/commit/530ab0753af93a405ce429088fe1c04602e5c646
  979. .. _io_schedulers:
  980. --------------------------------------
  981. Настройка планировщиков ввода/вывода
  982. --------------------------------------
  983. Планировщики ввода/вывода - это специальные модули ядра, которые
  984. регулируют порядок выполнения операций ввода/вывода во времени на
  985. уровне обращения к блочным устройстам (HDD дискам или
  986. SSD/NVMe/microSD/SD накопителям). Если вам казалось, что все запросы
  987. на чтение или запись происходят сразу же, то это не так.
  988. Все запросы к носителю сначала попадают в очередь, которой и управляет
  989. планировщик ввода/вывода. В зависимости от используемого алгоритма он
  990. "ранжирует" все поступающие запросы таким образом, чтобы запросы
  991. которые осуществляются к соседним блокам на диске шли как бы друг за
  992. другом, а не в том порядке в котором они поступили в очередь. К
  993. примеру, если к планировщику поступили запросы на чтение ``9``, ``3``
  994. и ``5`` блоков (условная запись), то он попытается разместить их в
  995. очереди как ``3``, ``5`` и ``9``. Зачем это делается? В силу
  996. исторических причин, все планировщики изначально разрабатывались с
  997. целью нивелировать недостатки механических дисков (и HDD в том числе),
  998. которые в силу своей специфики работы были чувствительны к порядку
  999. осуществления любых операций чтения или записи, так как чтобы
  1000. выполнить любую операцию головке жесткого диска нужно было сначала
  1001. найти нужный блок, а когда головка сначала выполняет чтение блока
  1002. ``9``, а потом чтение "назад" блока ``3``, чтобы потом опять
  1003. переместить головку вперед на блок ``5``, то очевидно что это
  1004. несколько уменьшает пропускную способность диска.
  1005. Поэтому все планировщики и работают по принципу "лифта" (*elevator*):
  1006. когда планировщик добавляет все запросы в очередь, но при этом
  1007. планирует их выполнение уже в порядке возрастания по номерам блоков, к
  1008. которым они обращаются. Кроме того, планировщик всегда будет отдавать
  1009. предпочтение запросам на чтение запросам на запись, в силу того, что
  1010. выполнение запросов на запись может быть неявно отложено ядром, либо
  1011. происходит куда быстрее в силу того, что запись сначала осуществляется
  1012. в страничный кэш (то есть в ОЗУ), а только потом на диск. В случае с
  1013. операциями чтения их выполнение не может быть отложено, банально в
  1014. силу того, что все программы, которые читают файлы, явно ожидают
  1015. получения какого-то результата.
  1016. Конечно, на деле алгоритм планирования запросов ввода/вывода куда
  1017. сложнее, но общий принцип остается тем же. На текущий момент в ядре
  1018. существует три "реальных" планировщика ввода/вывода: ``BFQ``,
  1019. ``mq-deadline``, ``kyber``. Существует также четвертый вариант
  1020. ``none``, который устанавливает простую FIFO очередь для всех
  1021. запросов. Это значит, что они будут обрабатываться ровно в том
  1022. порядке, в котором поступили без какого-либо планирования.
  1023. Хотя выбор не велик, выбор планировщика может сильно зависеть от типа
  1024. используемого носителя. Общие рекомендации к выбору планировщика под
  1025. определенный тип носителя состоят в следующем:
  1026. - Для NVMe и SATA SSD накопителей используйте ``none``. Дело в том,
  1027. что вся вышеописанная логика нахождения нужных блоков с
  1028. использованием головки совершенно не актуальна для твердотельных
  1029. накопителей с быстрым произвольным доступом [#]_, где любое
  1030. обращение к блокам осуществляется за фиксированное время, поэтому
  1031. порядок выполнения запросов для них не имеет такого же значения как
  1032. для жёстких дисков. В то же время, накладные расходы при
  1033. планировании прямо пропорциональны количеству запросов в очереди,
  1034. которые планировщику нужно обработать ресурсами CPU, но в NVMe и
  1035. простых SSD носителях планированием поступающих запросов на
  1036. аппаратном уровне уже занимается встроенный контроллер, поэтому
  1037. планировщик в ядре Linux по сути работает в холостую [#]_, нагружая
  1038. при этом процессор, что в свою очередь может вызывать
  1039. кратковременные зависания системы при большой нагрузке на
  1040. ввод/вывод.
  1041. - Однако для SATA SSD с плохим контроллером или устаревшим интерфейсом
  1042. подключения (SATA 2) имеет смысл использовать планировщик
  1043. ``mq-deadline``. Для SD/microSD карт так же имеет смысл использовать
  1044. только mq-deadline.
  1045. - Для HDD следует использовать BFQ, но в целом любой планировщик
  1046. должен быть лучше, чем его отсутствие как уже объяснено выше.
  1047. Как вы видите, здесь мы проигнорировали планировщик Kyber по той
  1048. причине, что он практически не развивается за последние 3 года (то
  1049. есть не получает новых значимых улучшений/оптимизаций) и рассчитан на
  1050. работу со сверх быстрыми накопителями, которые чувствительны к
  1051. задержкам, что не совсем актуально для домашней системы.
  1052. Итак, теория это хорошо, но как их все таки включить? Самый
  1053. универсальный способ это написать собственные правила Udev, которые
  1054. могли бы автоматически выбирать нужный планировщик в зависимости от
  1055. типа носителя. Чтобы создать новые правила просто создадим
  1056. новый файл в ``/etc/udev/rules.d/90-io-schedulers.rules``:
  1057. .. code-block:: shell
  1058. :caption: ``sudo nano /etc/udev/rules.d/90-io-schedulers.rules``
  1059. # HDD
  1060. ACTION=="add|change", KERNEL=="sd[a-z]*", ATTR{queue/rotational}=="1", ATTR{queue/scheduler}="bfq"
  1061. # eMMC/SD/microSD cards
  1062. ACTION=="add|change", KERNEL=="mmcblk[0-9]*", ATTR{queue/rotational}=="0", ATTR{queue/scheduler}="mq-deadline"
  1063. # SSD
  1064. ACTION=="add|change", KERNEL=="sd[a-z]*", ATTR{queue/rotational}=="0", ATTR{queue/scheduler}="none"
  1065. # NVMe SSD
  1066. ACTION=="add|change", KERNEL=="nvme[0-9]*", ATTR{queue/rotational}=="0", ATTR{queue/scheduler}="none"
  1067. (Чтобы использовать планировщик ``mq-deadline`` для SATA SSD просто
  1068. поменяйте значение внутри кавычек в третьей строке с ``none`` на
  1069. ``mq-deadline``).
  1070. Помните, что универсального рецепта не существует, и всегда следует
  1071. выполнить собственные тесты и бенчмарки (например при помощи программы
  1072. KDiskMark), чтобы понять какой из планировщиков вам подходит лучше.
  1073. .. [#] https://www.hotstorage.org/2023/papers/hotstorage23-final1.pdf
  1074. .. [#] https://www.phoronix.com/review/linux-56-nvme
  1075. .. _vm_max_map_count:
  1076. -----------------------------------------
  1077. Увеличение размера карты памяти процесса
  1078. -----------------------------------------
  1079. Так как виртуальные страницы процесса представляют собой кучу
  1080. маленьких фрагментов его данных, то для удобства вся его виртуальная
  1081. память разграничивается ядром на *зоны*. Например, в одной зоне памяти
  1082. процесса может быть загружена библиотека ``libc.so.6``, а в других
  1083. зонах - бинарный код другой библиотеки или данные самой программы,
  1084. память под которые она запросила у ядра через функцию ``mmap``. Зон
  1085. может быть несколько, так как они различаются по своим правам доступа
  1086. (Да, права есть не только у файлов, но и у виртуальных страниц).
  1087. Информация об этих зонах памяти процесса ядро хранит в так называемой
  1088. *виртуальной карте памяти* (*memory map*). Здесь речь идёт вовсе не о
  1089. носителе данных, а о той карте, которая используются ядром для того
  1090. чтобы понимать, начиная с какого адреса в памяти процесса расположена
  1091. та или иная зона. Вы можете просмотреть эту карту прочитав файл
  1092. ``maps`` на псевдофайловой системе procfs в директории, которая
  1093. предоставляет информацию о процессе с соответствующим ID.
  1094. Размер таких карт у каждого процесса ограничен значением параметра
  1095. ``vm.max_map_count`` , которое указывает на максимальное количество
  1096. записей, которое может хранится в карте у процесса. По умолчанию это
  1097. значение равно ``65530`` [4]_. К сожалению, современные программы, в
  1098. особенности игры запускаемые через Wine/Proton, с их потреблением
  1099. более 8 Гб на процесс, могут запрашивать чрезмерно много виртуальных
  1100. страниц у ядра, из-за чего количество зон для их процессов начинает
  1101. превышать установленный лимит по умолчанию. Это приводит к тому, что
  1102. ядро просто не даёт выделить ещё одну зону в памяти у процесса, что в
  1103. свою очередь приводит к проблемам со стабильностью (приложение может
  1104. просто аварийно завершится) и производительностью в таких прожорливых
  1105. программах. Поэтому если вы столкнулись с неполадками во время игры,
  1106. такими как частые вылеты или микрофризы, не спишите сразу писать
  1107. гневные письма разработчикам Wine, а попробуйте увеличить значение
  1108. данного параметра.
  1109. Чтобы избежать всех этих потенциальных проблем, рекомендуется
  1110. увеличить допустимый размер карты памяти процесса, то есть значение
  1111. параметра sysctl ``vm_max_map_count`` до ``1048576``. Тогда памяти
  1112. хватит точно всем :)
  1113. .. code-block:: shell
  1114. :caption: ``sudo nano /etc/sysctl.d/99-sysctl.conf``
  1115. vm.max_map_count = 1048576
  1116. .. note:: Во многих дистрибутивах., например таких как Fedora и Arch
  1117. Linux, данное значение уже установлено по умолчанию [#]_ [#]_
  1118. поэтому пользователям данных дистрибутивов не нужно делать никаких
  1119. дополнительных действий.
  1120. .. [#] https://pagure.io/fesco/issue/2993
  1121. .. [#] https://archlinux.org/news/increasing-the-default-vmmax_map_count-value/
  1122. .. _libahci_disable_sss:
  1123. ---------------------------------------------
  1124. Отключение многоступенчатого включения дисков
  1125. ---------------------------------------------
  1126. Для классических жестких дисков и иных носителей, подключаемых через
  1127. интерфейс SATA, существует механизм по многоступенчатому включению
  1128. питания, называемый *Staggered spin-up* (сокращенно - *SSU*), суть
  1129. которого состоит в том, что питание всех носителей у вас в системе
  1130. включается последовательно. То есть сначала начинает свою работу диск
  1131. №1, потом диск №2 и так далее. SSU был разработан и включен в
  1132. спецификацию SATA 2.5 достаточно давно - в 2005 году, когда компьютеры
  1133. были слишком слабыми и не выдерживали одновременного включения всех
  1134. дисков при загрузке системы.
  1135. В современных реалиях никакой необходимости в использовании SSU больше
  1136. нет, так как современные блоки питания спокойно выдерживают пиковое
  1137. энергопотребление, вызванное включением 3-4х и более носителей
  1138. одновременно, каждый из которых как правило имеет пиковую нагрузку
  1139. всего в 5W-12W [#]_, в то же время использование SSU по умолчанию
  1140. приводит к снижению скорости загрузки системы, поэтому он однозначно
  1141. рекомендуется к отключению на всех современных системах, если вы
  1142. имеете блок питания, который покрывает потребление вашего железа не
  1143. "впритык", а хотя бы с небольшим запасом.
  1144. В Linux это можно сделать через использование параметра
  1145. ``ignore_sss=1`` для модуля ядра ``libahci``, отвечающего за
  1146. реализацию общих функций по управлению всеми ATA устройствами. Для
  1147. этого следует создать новый файл в директории ``/etc/modprobe.d``:
  1148. .. code-block:: shell
  1149. :caption: ``sudo nano /etc/modprobe.d/30-ahci-disable-sss.conf``
  1150. options libahci ignore_sss=1
  1151. После этого на системах под управлением дистрибутива Arch Linux или
  1152. другого, основанного на нем, требуется также выполнить обновление
  1153. образов initramfs, чтобы данная конфигурация для ``modprobe`` стала их
  1154. частью и была применена на этапе ранней загрузки системы::
  1155. sudo mkinitcpio -P
  1156. .. note:: Вы можете пропустить шаги из данного раздел, если имеете
  1157. всего один жесткий носитель или не имеете их вообще, так как данный
  1158. параметр не оказывает влияния на устройства, подключенные не через
  1159. SATA.
  1160. .. [#] https://superuser.com/questions/565653/how-much-power-does-a-hard-drive-use
  1161. .. _cpu_optimization:
  1162. =========================
  1163. Оптимизация работы CPU
  1164. =========================
  1165. Нельзя прямо или косвенно улучшить характеристики вашего процессора,
  1166. однако можно использовать более эффективное и отвечающее вашим задачам
  1167. планирование работы процессов, а также отключить некоторые лишние
  1168. возможности ядра, которые могут приносить больше вреда, чем пользы для
  1169. простого пользователя. В этом разделе пойдет речь о том, как добиться
  1170. более эффективного использования ресурсов вашего "камня" с целью
  1171. получить более хорошую производительность.
  1172. .. index:: cpu, governor, cpupower, scaling
  1173. .. _cpu_governor:
  1174. ----------------------------
  1175. Масштабирование частоты CPU
  1176. ----------------------------
  1177. В Linux существует специальные модули ядра, так называемые драйверы
  1178. масштабирования CPU. Они корректируют частоту вашего процессора в
  1179. режиме реального времени в зависимости от используемой политики
  1180. (``governor``) масштабирования частот процессора. Для x86 архитектуры
  1181. стандартным таким драйвером в ядре Linux принято считать
  1182. ``acpi-cpufreq``, который, как понятно из названия, осуществляет
  1183. регулировку путем специальных вызовов APCI, которые устанавливают так
  1184. называемый P-state (Peformance state) уровень для вашего процессора.
  1185. Уровни P-State вашего процессора можно сравнить с коробкой передач у
  1186. автомобиля: он может лететь на всей скорости, ехать с обычной
  1187. автомобиля: он может лететь на всей скорости, ехать с обычной
  1188. скоростью, и стоять на месте в ожидании того, когда надо будет
  1189. сдвинуться с места. Собственно, по этой аналогии ``acpi-cpufreq`` и
  1190. переключается между 3-х стандартных P-State уровней в зависимости от
  1191. используемой политики масштабирования.
  1192. .. _governors:
  1193. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1194. Политики масштабирования частоты
  1195. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1196. Вернемся к политикам масштабирования частоты. Их можно сравнить с
  1197. планами электропитания в настройках Windows, только в отличии от них
  1198. политик масштабирования в Linux довольно много: ``performance``,
  1199. ``powersave``, ``userspace``, ``ondemand``,
  1200. ``conservative``, ``schedutil``.
  1201. Рассмотрим каждую из них подробнее:
  1202. ``performance`` - как понятно из названия, данная политика
  1203. используется для достижения максимальной производительности, так как
  1204. она сильно снижает порог нагрузки, переходя через который процессор
  1205. начинает работать на полную мощность. Хорошо подходит для настольных
  1206. ПК, но не слишком желательно для использования на ноутбуках, где важна
  1207. автономность. Обратите внимание, что поведение данной политики зависит
  1208. от используемого драйвера масштабирования, об этом подробнее читайте
  1209. далее в разделе :ref:`pstate-drivers`.
  1210. ``powersave`` - полная противоположность ``performance``, минимизирует
  1211. потребление энергии через занижение частот процессора до минимума.
  1212. Может быть очень полезно для ноутбуков при работе от батареи. Обратите
  1213. внимание, что поведение данной политики зависит от используемого
  1214. драйвера масштабирования, об этом подробнее читайте далее в разделе
  1215. :ref:`pstate-drivers`.
  1216. ``ondemand`` - политика, которая регулирует частоту процессора на
  1217. основе общей нагрузки на процессор, то есть частота прямо
  1218. пропорциональна нагрузке: Чем выше нагрузка, тем больше частота, и
  1219. наоборот, чем ниже, тем ниже и частота. Конечно, скорость возрастания
  1220. частоты и нагрузки не всегда коррелируют между собой, ибо для принятия
  1221. решения о том, когда нужно повышать частоту до максимальных значений,
  1222. драйвер руководствуется значением параметра ``up_threshold`` (по
  1223. умолчанию 80%), которое устанавливает порог максимальной нагрузки в
  1224. процентах, которое должно достичь хотя бы одно из ядер вашего
  1225. процессора. К примеру, если у вас есть два ядра, и в текущий момент
  1226. времени драйвер масштабирования зафиксировал на одном ядре процессора
  1227. нагрузку в 70%, а на другом в 81%, то он станет повышать их частоты в
  1228. соответствии со значением ``up_threshold`` по умолчанию. Обратное
  1229. решение - понижение частоты CPU, драйвер принимает в соответствии со
  1230. значением другого параметра - ``down_threshold``, которое также
  1231. устанавливает порог ниже которого должно пройти хотя бы одно ядро
  1232. процессора, тогда частота будет понижена. Во всех остальных случаях
  1233. частота будет регулироваться как обычно, то есть пропорционально общей
  1234. нагрузке.
  1235. Довольно хороший выбор для большинства конфигураций и задач.
  1236. Рекомендуется к использованию.
  1237. ``conservative`` - должно быть вы замечали, как запуская игру или
  1238. "тяжёлое" приложение ваш компьютер или ноутбук начинает гудеть как
  1239. самолет, так как происходит резкое повышение частот процессора и
  1240. следовательно растет его температура. ``conservative`` очень похож на
  1241. политику ``ondemand``, но он делает процесс увеличения частоты CPU
  1242. более "гладким" и поступательным даже при значительном повышении
  1243. нагрузки. Это может быть очень полезным, когда вы не хотите чтобы ваш
  1244. ноутбук резко повышал свою температуру или уходил в так называемый
  1245. "турбобуст".
  1246. ``schedutil`` - регулирует частоту процессора на основе метрик
  1247. планировщика CPU, например таких как количество активных процессов на
  1248. ядро процессора. Не слишком рекомендуется в силу того, что
  1249. использование данной политики часто приводит к резким скачкам частоты
  1250. без необходимости. Несмотря на это, данная политика используется по
  1251. умолчанию в стандартном ядре Arch Linux.
  1252. ``userspace`` - данная политика является заглушкой, которая позволяет
  1253. передать полномочия управления частотой с драйвера масштабирования на
  1254. программу, запущенную с правами root, в пространстве пользователя,
  1255. которая и будет осуществлять процесс регулировки частоты в
  1256. соответствии с собственной логикой. В большинстве случаев вам никогда
  1257. не понадобиться эта политика.
  1258. .. index:: amd-pstate, intel-pstate, cpufreq
  1259. .. _pstate-drivers:
  1260. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1261. Альтернативные драйверы масштабирования
  1262. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1263. Современные процессоры Intel и AMD могут самостоятельно осуществлять
  1264. масштабирование своей частоты на основе информации получаемой через
  1265. механизм *CPPC* (Collaborative Processor Performance Control), с
  1266. которым процессор получает от драйвера масштабирования "подсказку", о
  1267. том какой уровень производительности следует выдавать в данный момент.
  1268. Специально для работы с этим механизмом были созданы драйверы
  1269. amd-pstate (поддерживается процессорами Zen 2 и выше) и intel-pstate
  1270. (поддерживается Sandy Bridge и выше) для процессоров AMD и Intel
  1271. соответственно.
  1272. .. note:: Для процессоров Intel термин CPPC обычно не используется,
  1273. так как их технология для автономного управления частотой
  1274. процессора называется HWP (Hardware P-states), но суть остается той
  1275. же, что и в случае с CPPC.
  1276. У драйверов P-state существует несколько режимов работы. Как для
  1277. intel-pstate, так и для amd-pstate есть ``active`` и ``passive``
  1278. режим, но для amd-pstate есть также ``guided`` режим.
  1279. В режиме ``active``, который используется по умолчанию во всех P-state
  1280. драйверах, управление частотой выполняется полностью автономно самим
  1281. процессором, но он получает от драйвера масштабирования "подсказку" -
  1282. так называемый уровень ``energy_performance_preference`` (далее EPP),
  1283. на основе которого процессор понимает с каким уклоном ему регулировать
  1284. собственную частоту. Таких уровней всего пять: ``power``,
  1285. ``balance_power``, ``default``, ``balance_performance``,
  1286. ``performance``. Как понятно из названия, эти уровни указывают
  1287. процессору предпочтение к тому, чтобы он работал на максимальную
  1288. мощность (при использовании уровней ``balance_performance`` и
  1289. ``performance``) или наоборот экономил энергию и чаще принимал решение
  1290. о понижении своей частоты или уходе в состояние сна. По умолчанию
  1291. используется ``default``, что представляет собой баланс между
  1292. максимальной производительностью и энергосбережением.
  1293. Важно отметить, что классические политики для управления частоты,
  1294. которые мы описывали ранее, отходят на второй план, и более того, в
  1295. режиме ``active`` вы сможете выбрать всего две "фиктивные" политики
  1296. масштабирования, это ``powersave`` и ``performance``. Обе из них не
  1297. оказывают того влияния на частоту процессора, которое мы приписывали
  1298. им ранее, так как в режиме ``active`` драйвер не может самостоятельно
  1299. устанавливать частоту процессора и теперь это зависит только от
  1300. используемого значения EPP. Поэтому при выборе ``performance``
  1301. политики вы на самом деле просто измените текущий уровень EPP на
  1302. ``performance``, значение которого P-state драйвер передаст процессору
  1303. через специальный регистр. Но при переключении политики на
  1304. ``powersave`` уровень EPP не измениться и вы должны будете установить
  1305. его самостоятельно (об этом читайте далее).
  1306. При использовании режима ``passive`` P-State драйвер может напрямую
  1307. устанавливать желаемый уровень производительности, в связи с чем в нем
  1308. доступен полный набор политик масштабирования, о которых мы говорили
  1309. ранее. При этом установить уровень EPP становится невозможно, так как
  1310. процессор уже не управляет частотой полностью самостоятельно, а
  1311. ожидает переключения уровня P-State со стороны драйвера
  1312. масштабирования. Данный режим отличается от использования
  1313. классического драйвера ``acpi-cpufreq`` тем, что драйвер переключается
  1314. не между 3-мя уровнями P-State, которые определены стандартом ACPI, а
  1315. между сразу всеми доступными диапазонами частоты для вашего
  1316. процессора, что гораздо эффективнее.
  1317. Для драйвера ``amd-pstate`` существует также третий режим работы -
  1318. ``guided``. Он работает аналогично режиму ``active``, позволяя
  1319. процессору самому управлять частотой, но при этом драйвер может
  1320. устанавливать процессору пороги минимальной и максимальной частоты,
  1321. что позволяет использовать классические политики масштабирования как в
  1322. случае с ``passive`` режимом.
  1323. Переключение между всеми тремя режимами может быть осуществлено как
  1324. при помощи соответствующих параметров ядра ``amd_pstate`` (например,
  1325. ``amd_pstate=guided``) или ``intel_pstate`` в зависимости от
  1326. используемого драйвера масштабирования, так и прямо во время работы
  1327. системы при помощи файла ``status`` на псевдофайловой системе sysfs:
  1328. .. tab-set::
  1329. .. tab-item:: AMD
  1330. ::
  1331. echo "passive" | sudo tee /sys/devices/system/cpu/amd_pstate/status
  1332. .. tab-item:: Intel
  1333. ::
  1334. echo "passive" | sudo tee /sys/devices/system/cpu/intel_pstate/status
  1335. .. index:: cpupower, cpufreq
  1336. .. _cpufreq_tuning:
  1337. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1338. Настройка параметров масштабирования
  1339. ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  1340. Перейдем от теории к практике. Чтобы изменить текущую политику
  1341. масштабирования частоты можно воспользоваться множеством различных
  1342. способов, начиная от способа "руками" при помощи sysfs, заканчивая
  1343. специализированными утилитами как ``cpupower`` и
  1344. ``power-profiles-daemon``, которые мы и будем использовать для
  1345. удобства. Для начала установим программу ``cpupower``::
  1346. sudo pacman -S cpupower
  1347. С её помощью мы можем быстро увидеть информацию о текущей политике
  1348. масштабирования, используемом драйвере, а также текущую частоту::
  1349. cpupower frequency-info
  1350. Установить желаемую политику масштабирования можно через команду
  1351. ``frequency-set``. К примеру, установим политику ``performance``::
  1352. sudo cpupower frequency-set -g performance
  1353. .. note:: Если команда ``cpupower frequency-info`` указывает на то,
  1354. что используется P-State драйвер в автономном режиме, то не следует
  1355. пытаться применять классические политики масштабирования при помощи
  1356. ``cpupower``, вместо этого нужно указывать значение параметра
  1357. ``energy_performance_preference`` (EPP) при помощи sysfs,
  1358. например::
  1359. echo "balance_performance" | sudo tee /sys/devices/system/cpu/cpu*/cpufreq/energy_performance_preference
  1360. Узнать доступные значения параметра EPP можно через::
  1361. cat /sys/devices/system/cpu/cpu0/cpufreq/energy_performance_available_preferences
  1362. Это значение будет той самой подсказкой для процессора, о которой
  1363. мы говорили выше, в соответствии с которой он будет осуществлять
  1364. самостоятельный контроль своей частоты.
  1365. Если вы хотите ограничить максимальную частоту процессора, то вы
  1366. можете использовать ключ ``-u``::
  1367. # Ограничит максимальную частоту в 3 ГГц
  1368. sudo cpupower frequency-set -u 3.0Ghz
  1369. Все проделанные изменения выше работают только до перезагрузки
  1370. системы, чтобы их сохранить нам понадобиться одноименный демон
  1371. ``cpupower.service``::
  1372. sudo systemctl enable --now cpupower
  1373. А также изменить конфигурацию ``/etc/default/cpupower``, которая
  1374. содержит все применяемые при запуске системы настройки. К примеру,
  1375. если вам нужно изменить политику масштабирования на постоянной основе,
  1376. то нужно указать значение параметра ``governor`` внутри
  1377. ``/etc/default/cpupower``:
  1378. .. code-block:: shell
  1379. :caption: ``sudo nano /etc/default/cpupower``
  1380. governor='conservative'
  1381. .. note:: По умолчанию все строки в файле закоментированы. Чтобы
  1382. раскоментировать нужные параметры уберите знак ``#`` в начале
  1383. строки.
  1384. .. _no_watchdog_timers:
  1385. ------------------------------
  1386. Отключение сторожевых таймеров
  1387. ------------------------------
  1388. Сторожевые таймеры (*watchdog timer*) - это аппаратные или программные
  1389. средства, которые отслеживают зависания системы во время её работы.
  1390. Все такие таймеры работают по принципу "пуллинга", при котором система
  1391. должна постоянно выполнять прерывания таймера. Если прерывание не было
  1392. сделано вовремя, то подразумевается, что система зависла и таймер по
  1393. истечению времени ожидания должен сигнализировать другое устройство
  1394. или программу (если речь о программном сторожевом таймере) о
  1395. зависании, так чтобы были предприняты действия по автоматическому
  1396. восстановлению работы системы, что может включать в себя её перезапуск
  1397. или другие процедуры в зависимости от конкретной системы.
  1398. На первый взгляд это звучит очень здорово, однако стоит понимать, что
  1399. сторожевые таймеры задумывались в основном для систем, которые должны
  1400. работать полностью автономно без прямого участия со стороны
  1401. пользователя. К ним относятся например сервера, SoC платы для
  1402. встраивания в различные малые или крупные аппаратные комплексы (вплоть
  1403. до таких систем как космические зонды) и т. д. На простых домашних ПК
  1404. или ноутбуках у пользователя всегда есть возможность вручную выполнить
  1405. перезапуск системы, не говоря о том, что как правило зависания
  1406. достаточно мощных систем происходит редко и в основном из-за ошибок
  1407. внутри самого ядра, а на маломощных в основном из-за ситуаций OOM
  1408. (нехватки памяти).
  1409. Так как на домашних системах сторожевые таймеры большую часть времени
  1410. выполняют только холостую работу, их отключение позволяет снизить
  1411. общее энергопотребление [#]_ и увеличить производительность системы
  1412. [#]_. Сделать это можно как и всегда через соответствующий параметр
  1413. sysctl:
  1414. .. code-block:: shell
  1415. :caption: ``sudo nano /etc/sysctl.d/30-no-watchdog-timers.conf``
  1416. kernel.watchdog = 0
  1417. К сожалению, на современных процессорах от AMD и Intel использование
  1418. данного параметра ядра недостаточно для полного отключения аппаратных
  1419. сторожевых таймеров [#]_, которые являются частью чипсета процессора.
  1420. Поэтому дополнительно требуется также запретить загрузку модулей ядра,
  1421. которые занимаются управлением этих таймеров. Для процессоров от Intel
  1422. это ``iTCO_wdt``, а для AMD ``sp5100-tco``. Чтобы запретить их
  1423. загрузку нужно создать файл в директории ``/etc/modprobe.d`` и внести
  1424. модули в чёрный список:
  1425. .. code-block:: shell
  1426. :caption: ``sudo nano /etc/modprobe.d/30-blacklist-watchdog-timers.conf``
  1427. blacklist sp5100-tco
  1428. blacklist iTCO_wdt
  1429. На системах под управлением дистрибутива Arch Linux или другого,
  1430. основанного на нем требуется, также выполнить обновление образов
  1431. initramfs, чтобы данная конфигурация для ``modprobe`` стала их частью
  1432. и была применена на этапе ранней загрузки системы::
  1433. sudo mkinitcpio -P
  1434. После перезагрузки системы можно удостовериться, что сторожевые
  1435. таймеры были успешно отключены через команду ``wdctl``, она должна
  1436. вывести ошибку об отсутствии соответствующих устройств.
  1437. .. [#] https://wiki.archlinux.org/title/Power_management#Disabling_NMI_watchdog
  1438. .. [#] https://bbs.archlinux.org/viewtopic.php?id=163768
  1439. .. [#] https://bbs.archlinux.org/viewtopic.php?id=165834
  1440. .. _mitigations_off:
  1441. ---------------------------------------------
  1442. Отключение защиты от уязвимостей (по желанию)
  1443. ---------------------------------------------
  1444. .. caution:: Данный раздел носит сугубо информационный характер и ни к
  1445. чему не призывает. Безопасность системы является таким же важным её
  1446. аспектом как и производительность, однако для тех, кто все же хочет
  1447. "выжать максимум" несмотря на связанные с этим риски, и представлен
  1448. материал ниже.
  1449. К сожалению современные процессоры (если говорить об архитектуре x86)
  1450. обзавелись достаточно большим багажом аппаратных уязвимостей, которые
  1451. приходится различными способами исправлять разработчикам ядра. И как
  1452. правило все такие исправления влекут за собой большие потери
  1453. производительности. Иногда такие исправления приводят к потерям
  1454. порядка ~27-28% производительности [#]_ [#]_, а иногда и все 50% [#]_, как в
  1455. случае с нашумевшей уязвимостью Spectre, что часто приводит к
  1456. недоумению относительно целесообразности таких исправлений не только
  1457. со стороны простых пользователей, но и со стороны других разработчиков
  1458. ядра [#]_.
  1459. Здесь стоит отметить, что те же уязвимости Spectre и Meltdown сами по
  1460. себе не являются "прямым ключом" к вашей системе, так как даже при их
  1461. использовании скорость утечки в продемонстрированных ранее примерах
  1462. эксплуатации составляет всего 1 Кб/с. Кроме того, большинство
  1463. современных браузеров имеет свою встроенную систему защиты для запуска
  1464. всех критически важных процессов по обмену данными в изолированном
  1465. окружении, которое предотвращает эксплуатацию подобных уязвимостей,
  1466. хотя некоторые браузеры, например такие как Firefox, в последних
  1467. версиях отказываются от включения по умолчанию частей такой
  1468. дополнительной защиты в пользу лучшей производительности [#]_, в
  1469. частности на практике использование уязвимостей класса Spectre v2
  1470. имеет слабый практический характер, а также такая защита может быть не
  1471. нужна из-за включенной по умолчанию в ядре общей защиты против
  1472. уязвимостей класса Spectre. Если вы хотите форсировать включение этой
  1473. дополнительной защиты в Firefox вдобавок к уже существующим мерам
  1474. защиты в ядре, то сделать это можно через переменную окружения
  1475. ``MOZ_SANDBOX_NO_SPEC_ALLOW=1``.
  1476. Если же вы все же решили выполнить полное отключение всей защиты в
  1477. ядре в пользу большей производительности, то сделать это можно только
  1478. при указании параметра загрузки ядра ``mitigations=off``, который
  1479. следует прописать в конфигурации вашего загрузчика:
  1480. .. warning:: Отключение защиты с целью повышения производительности
  1481. имеет смысл только на старых поколениях процессоров Intel и AMD. На
  1482. самых последних линейках это может иметь даже негативные
  1483. последствия [#]_.
  1484. .. tab-set::
  1485. .. tab-item:: GRUB 2
  1486. Чтобы изменить параметры по умолчанию к загрузке любого ядра при
  1487. использовании загрузчика GRUB 2 нужно открыть файл
  1488. ``/etc/default/grub``, найти строку
  1489. ``GRUB_CMDLINE_LINUX_DEFAULT`` и внутрь двойных кавычек дописать
  1490. к уже имеющимся параметрам ``mitigations=off`` через пробел.
  1491. Пример такой строки конфигурации:
  1492. .. code-block:: shell
  1493. :caption: ``sudo nano /etc/default/grub``
  1494. ...
  1495. GRUB_CMDLINE_LINUX_DEFAULT="quiet loglevel=3 mitigations=off"
  1496. ...
  1497. После чего необходимо обновить конфигурацию загрузчика::
  1498. sudo grub-mkconfig -o /boot/grub/grub.cfg
  1499. .. tab-item:: rEFInd
  1500. Если вы используете загрузчик rEFInd вместе со всеми его
  1501. возможностями по автоматическому определению всех установленных
  1502. версией ядра у вас в системе, то чтобы добавить параметр
  1503. ``mitigations=off`` нужно отредактировать файл
  1504. ``/boot/refind_linux.conf`` и в первую строку содержащую
  1505. дописать данный параметр внутрь вторых двойных кавычек. Пример
  1506. такого файла конфигурации (не копировать полностью!):
  1507. .. code-block:: shell
  1508. :caption: ``sudo nano /boot/refind_linux.conf``
  1509. "Boot to default params" "root=PARTUUID=XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX rw mitigations=off"
  1510. "Boot to single-user mode" "root=PARTUUID=XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX rw single"
  1511. .. tab-item:: systemd-boot
  1512. К сожалению при использовании загрузчика systemd-boot нельзя
  1513. указать общие для всех установленных ядер в системе параметры
  1514. загрузки, поэтому приходится указывать их в отдельности для
  1515. каждой версии ядра. Например, чтобы добавить параметр
  1516. ``mitigations=off`` для обычного ядра используемого в Arch
  1517. Linux, вам нужно найти файл внутри директории
  1518. ``/boot/efi/entries``, который содержит строку ``linux
  1519. /vmlinuz-linux`` и ниже внутри этого файла дописать через пробел
  1520. параметр в строке с ``options``. Пример конфигурации:
  1521. .. code-block:: shell
  1522. :caption: ``sudo nano /boot/efi/loader/entries/arch.conf``
  1523. title Arch Linux
  1524. linux /vmlinuz-linux
  1525. initrd /initramfs-linux.img
  1526. options root=UUID=xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx rw mitigations=off
  1527. .. warning:: Путь до файла конфигурации может отличаться в
  1528. зависимости от того в какую директорию вы смонтировали
  1529. загрузочный раздел, некоторые пользователи вместо ``/boot/efi``
  1530. используют точку монтирования ``/efi``. Узнать используемую
  1531. точку монтирования можно через команду ``bootctl
  1532. --print-esp-path``.
  1533. .. [#] https://linuxreviews.org/HOWTO_make_Linux_run_blazing_fast_(again)_on_Intel_CPUs
  1534. .. [#] https://talawah.io/blog/extreme-http-performance-tuning-one-point-two-million/#_2-speculative-execution-mitigations
  1535. .. [#] https://lkml.org/lkml/2018/11/19/37
  1536. .. [#] https://lkml.org/lkml/2018/11/19/69
  1537. .. [#] https://hg.mozilla.org/mozilla-central/rev/ebdd80f675b6
  1538. .. [#] https://www.tomshardware.com/news/ryzen-7000-security-mitigations-faster-on-linux
  1539. .. vim:set textwidth=70: