Как использовать команду pmap в Linux
Опубликовано: 2022-06-25 Выяснить, сколько оперативной памяти использует процесс Linux, непросто, особенно когда необходимо учитывать общую память. К счастью, команда pmap
поможет вам разобраться во всем этом.
Отображение памяти
В современных операционных системах каждый процесс живет в своей собственной выделенной области памяти или аллокации . Границы выделенной области не сопоставляются напрямую с адресами физического оборудования. Операционная система создает пространство виртуальной памяти для каждого процесса и действует как уровень абстракции, отображающий виртуальную память в физическую память.
Ядро поддерживает таблицу трансляции для каждого процесса, и ЦП обращается к ней. Когда ядро изменяет процесс, работающий на определенном ядре ЦП, оно обновляет таблицу преобразования, которая связывает процессы и ядра ЦП вместе.
Преимущества абстракции
У этой схемы есть преимущества. Использование памяти несколько инкапсулировано и изолировано для каждого процесса в пространстве пользователя. Процесс «видит» память только с точки зрения адресов виртуальной памяти. Это означает, что он может работать только с той памятью, которую ему предоставила операционная система. Если у него нет доступа к некоторой общей памяти, он не знает и не имеет доступа к памяти, выделенной другим процессам.
Абстракция аппаратной физической памяти в адреса виртуальной памяти позволяет ядру изменить физический адрес, на который отображается некоторая виртуальная память. Он может поменять память на диск, изменив фактический адрес, на который указывает область виртуальной памяти. Он также может отложить предоставление физической памяти до тех пор, пока она действительно не потребуется.
Пока запросы на чтение или запись в память обслуживаются по мере их поступления, ядро может свободно манипулировать таблицей отображения по своему усмотрению.
ОЗУ по запросу
Таблица сопоставления и концепция «ОЗУ по запросу» открывают возможность использования разделяемой памяти . Ядро попытается избежать повторной загрузки одного и того же объекта в память. Например, он один раз загрузит разделяемую библиотеку в память и сопоставит ее с различными процессами, которые должны ее использовать. Каждый из процессов будет иметь свой собственный уникальный адрес общей библиотеки, но все они будут указывать на одно и то же фактическое местоположение.
Если разделяемая область памяти доступна для записи, ядро использует схему, называемую копированием при записи. Если один процесс записывает в разделяемую память, а другие процессы, совместно использующие эту память, не должны видеть изменения, копия разделяемой памяти создается в момент запроса на запись.
Ядро Linux 2.6.32, выпущенное в декабре 2009 года, предоставило Linux функцию под названием «Слияние с ядром SamePage». Это означает, что Linux может обнаруживать идентичные области данных в разных адресных пространствах. Представьте себе серию виртуальных машин, работающих на одном компьютере, и все виртуальные машины работают под управлением одной и той же операционной системы. Используя модель с общей памятью и копирование при записи, накладные расходы на хост-компьютер могут быть значительно снижены.
Все это делает работу с памятью в Linux сложной и максимально оптимальной. Но эта изощренность затрудняет просмотр процесса и определение фактического использования им памяти.
Утилита pmap
Ядро раскрывает многое из того, что оно делает с оперативной памятью, через два псевдофайла в псевдофайловой системе системной информации «/proc». На каждый процесс приходится два файла, названных по идентификатору процесса или PID каждого процесса: «/proc/maps» и «/proc//smaps».
Инструмент pmap
считывает информацию из этих файлов и отображает результаты в окне терминала. Будет очевидно, что нам нужно предоставить PID интересующего нас процесса всякий раз, когда мы используем pmap
.
Поиск идентификатора процесса
Есть несколько способов узнать PID процесса. Вот исходный код простой программы, которую мы будем использовать в наших примерах. Он написан на C. Все, что он делает, — это выводит сообщение в окно терминала и ждет, пока пользователь нажмет клавишу «Enter».
#include <stdio.h> int main(int argc, char *argv[]) { printf("Тестовая программа How-To Geek."); получить (стандартный ввод); } // конец основного
Программа была скомпилирована в исполняемый файл с именем pm
с помощью компилятора gcc
:
gcc -o pm pm.c
Поскольку программа будет ждать, пока пользователь нажмет «Enter», она будет работать столько, сколько нам нужно.
./вечера
Программа запускается, печатает сообщение и ждет нажатия клавиши. Теперь мы можем искать его PID. Команда ps
выводит список запущенных процессов. Опция -e
(показать все процессы) заставляет ps
отображать все процессы. Мы направим вывод через grep
и отфильтруем записи, в названии которых есть «pm».
пс-е | grep pm
В нем перечислены все записи, в именах которых есть «pm».
Мы можем быть более конкретными, используя команду pidof
. Мы даем pidof
имя интересующего нас процесса в командной строке, и он пытается найти совпадение. Если совпадение найдено, pidof
выводит PID соответствующего процесса.
pidof pm
Метод pidof
, когда вы знаете имя процесса, но метод ps
будет работать, даже если вы знаете только часть имени процесса.
Использование pmap
Когда наша тестовая программа запущена и мы определили ее PID, мы можем использовать pmap следующим образом:
рмап 40919
Отображения памяти для процесса перечислены для нас.
Вот полный вывод команды:
40919: ./pm 000056059f06c000 4K р---- чм 000056059f06d000 4K rx-- pm 000056059f06e000 4K р---- чм 000056059f06f000 4K р---- чм 000056059f070000 4K RW--- pm 000056059fc39000 132K rw--- [анонимно] 00007f97a3edb000 8K rw--- [ анонимно ] 00007f97a3edd000 160K r---- libc.so.6 00007f97a3f05000 1616K rx-- libc.so.6 00007f97a4099000 352K r---- libc.so.6 00007f97a40f1000 4K ----- libc.so.6 00007f97a40f2000 16K r---- libc.so.6 00007f97a40f6000 8K RW--- libc.so.6 00007f97a40f8000 60K RW --- [ анон ] 00007f97a4116000 4K r---- ld-linux-x86-64.so.2 00007f97a4117000 160K rx-- ld-linux-x86-64.so.2 00007f97a413f000 40K r---- ld-linux-x86-64.so.2 00007f97a4149000 8K r---- ld-linux-x86-64.so.2 00007f97a414b000 8K RW --- ld-linux-x86-64.so.2 00007ffca0e7e000 132K RW --- [стек] 00007ffca0fe1000 16K р---- [ анон ] 00007ffca0fe5000 8K rx-- [анонимно] ffffffffff600000 4K --x-- [ анон ] всего 2756K
Первая строка — это имя процесса и его PID. Каждая из других строк показывает сопоставленный адрес памяти и объем памяти по этому адресу, выраженный в килобайтах. Следующие пять символов каждой строки называются разрешениями на виртуальную память . Допустимые разрешения:
- r : Отображенная память может быть прочитана процессом.
- w : отображаемая память может быть записана процессом.
- x : процесс может выполнять любые инструкции, содержащиеся в отображаемой памяти.
- s : отображаемая память является общей, и изменения, внесенные в общую память, видны всем процессам, совместно использующим эту память.
- R : Для этой отображаемой памяти нет резервирования пространства подкачки.
Окончательной информацией о каждой строке является имя источника отображения. Это может быть имя процесса, имя библиотеки или имя системы, например стек или куча.
Расширенный дисплей
Параметр -x
(расширенный) предоставляет два дополнительных столбца.
pmap -x 40919
Столбцы имеют заголовки. Мы уже видели столбцы «Адрес», «Кбайт», «Режим» и «Отображение». Новые столбцы называются «RSS» и «Грязный».
Вот полный вывод:
40919: ./pm Адрес Кбайт RSS Сопоставление грязного режима 000056059f06c000 4 4 0 р---- чм 000056059f06d000 4 4 0 rx-- pm 000056059f06e000 4 4 0 r---- pm 000056059f06f000 4 4 4 р---- чм 000056059f070000 4 4 4 rw--- pm 000056059fc39000 132 4 4 rw--- [анонимно] 00007f97a3edb000 8 4 4 rw--- [анонимно] 00007f97a3edd000 160 160 0 r---- libc.so.6 00007f97a3f05000 1616 788 0 rx-- libc.so.6 00007f97a4099000 352 64 0 r---- libc.so.6 00007f97a40f1000 4 0 0 ----- libc.so.6 00007f97a40f2000 16 16 16 r---- libc.so.6 00007f97a40f6000 8 8 8 rw--- libc.so.6 00007f97a40f8000 60 28 28 rw--- [анонимно] 00007f97a4116000 4 4 0 r---- ld-linux-x86-64.so.2 00007f97a4117000 160 160 0 rx-- ld-linux-x86-64.so.2 00007f97a413f000 40 40 0 r---- ld-linux-x86-64.so.2 00007f97a4149000 8 8 8 р---- ld-linux-x86-64.so.2 00007f97a414b000 8 8 8 rw--- ld-linux-x86-64.so.2 00007ffca0e7e000 132 12 12 rw--- [стек] 00007ffca0fe1000 16 0 0 r---- [анонимно] 00007ffca0fe5000 8 4 0 rx-- [анонимно] ffffffffff600000 4 0 0 --x-- [ анон ] ---------------- ------- ------- ------- всего кБ 2756 1328 96
- RSS : размер резидентного набора . То есть объем памяти, которая в данный момент находится в оперативной памяти, а не выгружена.
- Dirty : «Грязная» память была изменена с момента запуска процесса и сопоставления.
Покажи мне все
-X
(даже более чем расширенный) добавляет к выводу дополнительные столбцы. Обратите внимание на заглавную букву «X». Другая опция, называемая -XX
(даже больше, чем -X
), показывает вам все, что pmap
может получить от ядра. Поскольку -X
является подмножеством -XX
, мы опишем вывод -XX
.
рмап-ХХ 40919
Вывод ужасно обтекается в окне терминала и практически не поддается расшифровке. Вот полный вывод:
40919: ./pm Адрес Perm Offset Device Размер Inode KernelPageSize MMUPageSize Rss Pss Shared_Clean Shared_Dirty Private_Clean Private_Dirty Referenced Anonymous LazyFree AnonHugePages ShmemPmdMapped FilePmdMapped Shared_Hugetlb Private_Hugetlb Swap SwapPss Locked THPeligible VmFlags Mapping 56059f06c000 r--p 00000000 08:03 393304 4 4 4 4 4 0 0 4 0 4 0 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw sd pm 56059f06d000 r-xp 00001000 08:03 393304 4 4 4 4 4 0 0 4 0 4 0 0 0 0 0 0 0 0 0 0 0 rd ex mr mw me dw sd pm 56059f06e000 r--p 00002000 08:03 393304 4 4 4 4 4 0 0 4 0 4 0 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw sd pm 56059f06f000 r--p 00002000 08:03 393304 4 4 4 4 4 0 0 0 4 4 4 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw ac sd pm 56059f070000 rw-p 00003000 08:03 393304 4 4 4 4 4 0 0 0 4 4 4 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me dw ac sd pm 56059fc39000 rw-p 00000000 00:00 0 132 4 4 4 4 0 0 0 4 4 4 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me ac sd [куча] 7f97a3edb000 rw-p 00000000 00:00 0 8 4 4 4 4 0 0 0 4 4 4 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me ac sd 7f97a3edd000 r--p 00000000 08:03 264328 160 4 4 160 4 160 0 0 0 160 0 0 0 0 0 0 0 0 0 0 rd mr mw me sd libc.so.6 7f97a3f05000 r-xp 00028000 08:03 264328 1616 4 4 788 32 788 0 0 0 788 0 0 0 0 0 0 0 0 0 0 0 rd ex mr mw me sd libc.so.6 7f97a4099000 r--p 001bc000 08:03 264328 352 4 4 64 1 64 0 0 0 64 0 0 0 0 0 0 0 0 0 0 rd mr mw me sd libc.so.6 7f97a40f1000 ---p 00214000 08:03 264328 4 4 4 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 mr mw me sd libc.so.6 7f97a40f2000 r--p 00214000 08:03 264328 16 4 4 16 16 0 0 0 16 16 16 0 0 0 0 0 0 0 0 0 rd mr mw me ac sd libc.so.6 7f97a40f6000 rw-p 00218000 08:03 264328 8 4 4 8 8 0 0 0 8 8 8 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me ac sd libc.so.6 7f97a40f8000 rw-p 00000000 00:00 0 60 4 4 28 28 0 0 0 28 28 28 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me ac sd 7f97a4116000 r--p 00000000 08:03 264305 4 4 4 4 0 4 0 0 0 4 0 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw sd ld-linux-x86-64.so.2 7f97a4117000 r-xp 00001000 08:03 264305 160 4 4 160 11 160 0 0 0 160 0 0 0 0 0 0 0 0 0 0 0 rd ex mr mw me dw sd ld-linux-x86-64.so.2 7f97a413f000 r--p 00029000 08:03 264305 40 4 4 40 1 40 0 0 0 40 0 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw sd ld-linux-x86-64.so.2 7f97a4149000 r--p 00032000 08:03 264305 8 4 4 8 8 0 0 0 8 8 8 0 0 0 0 0 0 0 0 0 0 rd mr mw me dw ac sd ld-linux-x86-64.so.2 7f97a414b000 rw-p 00034000 08:03 264305 8 4 4 8 8 0 0 0 8 8 8 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me dw ac sd ld-linux-x86-64.so.2 7ffca0e7e000 rw-p 00000000 00:00 0 132 4 4 12 12 0 0 0 12 12 12 0 0 0 0 0 0 0 0 0 0 rd wr mr mw me gd ac [стек] 7ffca0fe1000 r--p 00000000 00:00 0 16 4 4 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 rd mr pf io de dd sd [vvar] 7ffca0fe5000 r-xp 00000000 00:00 0 8 4 4 4 0 4 0 0 0 4 0 0 0 0 0 0 0 0 0 0 0 rd ex mr mw me de sd [vdso] ffffffffff600000 --xp 00000000 00:00 0 4 4 4 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 ex [vsyscall] ==== ============== =========== ==== === ============ == ========== ============= ============= ========== ==== ===== ======== ============= ============== ========== === ============== =============== ==== ======= ====== = ========== 2756 92 92 1328 157 1220 0 12 96 1328 96 0 0 0 0 0 0 0 0 0 0 КБ
Здесь много информации. Вот что содержат столбцы:
- Address : начальный адрес этого сопоставления. При этом используется адресация виртуальной памяти.
- Пермь : Разрешения памяти.
- Смещение : если память файловая, смещение этого сопоставления внутри файла.
- Устройство : номер устройства Linux, указанный в старшем и второстепенном числах. Вы можете увидеть номера устройств на своем компьютере, выполнив команду
lsblk
. - Inode : индекс файла, с которым связано сопоставление. Например, в нашем примере это может быть индексный дескриптор, содержащий информацию о программе pm.
- Size : размер отображаемой в памяти области.
- KernelPageSize : Размер страницы, используемый ядром.
- MMUPageSize : Размер страницы, используемый модулем управления памятью.
- Rss : Это размер резидентного набора . То есть объем памяти, которая в данный момент находится в оперативной памяти, а не выгружена.
- Pss : Это пропорциональный размер доли . Это частный общий размер, добавляемый к (общий размер, разделенный на количество общих ресурсов).
- Shared_Clean : объем памяти, совместно используемой с другими процессами, который не изменился с момента создания сопоставления. Обратите внимание, что даже если память является совместно используемой, если она на самом деле не является общей, она все равно считается частной памятью.
- Shared_Dirty : объем памяти, совместно используемой с другими процессами, которая была изменена с момента создания сопоставления.
- Private_Clean : объем частной памяти, не используемой совместно с другими процессами, которая не изменилась с момента создания сопоставления.
- Private_Dirty : объем частной памяти, который был изменен с момента создания сопоставления.
- Referenced : объем памяти, помеченный в настоящее время как указанный или доступный.
- Anonymous : Память, в которой нет устройства для подкачки. То есть, это не резервное копирование файлов.
- LazyFree : страницы, отмеченные как
MADV_FREE
. Эти страницы помечены как доступные для освобождения и восстановления, даже если на них могут быть неписаные изменения. Однако, если последующие изменения происходят после установкиMADV_FREE
в отображении памяти, флагMADV_FREE
снимается, и страницы не будут восстановлены до тех пор, пока изменения не будут записаны. - AnonHugePages : это «огромные» страницы памяти без файловой поддержки (больше 4 КБ).
- ShmemPmdMapped : общая память, связанная с огромными страницами. Они также могут использоваться файловыми системами, которые полностью находятся в памяти.
- FilePmdMapped : средний каталог страниц — это одна из схем разбиения по страницам, доступных ядру. Это количество страниц с файловой поддержкой, на которые указывают записи PMD.
- Shared_Hugetlb : Таблицы резервных копий перевода или TLB — это кэши памяти, используемые для оптимизации времени, необходимого для доступа к ячейкам памяти пользовательского пространства. Эта цифра представляет собой объем ОЗУ, используемый в TLB, которые связаны с общими огромными страницами памяти.
- Private_Hugetlb : этот показатель представляет собой объем оперативной памяти, используемый в TLB, которые связаны с частными огромными страницами памяти.
- Swap : количество используемого свопа.
- SwapPss : размер доли, пропорциональный свопу . Это объем подкачки, состоящий из замененных страниц частной памяти, добавленных к (общий размер, разделенный на количество общих ресурсов).
- Заблокировано : отображение памяти можно заблокировать, чтобы операционная система не выгружала память из кучи или вне кучи.
- THPeligible : это флаг, указывающий, подходит ли сопоставление для выделения прозрачных огромных страниц . 1 означает истину, 0 означает ложь. Прозрачные огромные страницы — это система управления памятью, которая снижает нагрузку на поиск страниц TLB на компьютерах с большим объемом оперативной памяти.
- VmFlags : см. список флагов ниже.
- Сопоставление : имя источника сопоставления. Это может быть имя процесса, имя библиотеки или системные имена, такие как стек или куча.
VmFlags — флаги виртуальной памяти — будут подмножеством следующего списка.
- рд : Читаемый.
- wr : Доступно для записи.
- пример : исполняемый файл.
- ш : Общий.
- мр : Может читать.
- мв : Может писать.
- я : Может казнить.
- мс : Может поделиться.
- gd : Сегмент стека растет вниз.
- pf : Чистый диапазон номеров фреймов страниц. Номера фреймов страниц представляют собой список страниц физической памяти.
- dw : отключена запись в сопоставленный файл.
- lo : Страницы заблокированы в памяти.
- io : область ввода-вывода с отображением памяти.
- sr : Совет по последовательному чтению предоставляется (
madvise()
.) - rr : Предусмотрен случайный совет по чтению.
- dc : не копировать эту область памяти, если процесс разветвлен.
- de : Не расширяйте эту область памяти при переназначении.
- ac : Область подотчетна.
- nr : пространство подкачки не зарезервировано для области.
- ht : Area использует огромные страницы TLB.
- sf : Ошибка синхронной страницы.
- ar : флаг, зависящий от архитектуры.
- wf : стереть эту область памяти, если процесс разветвлен.
- dd : не включать эту область памяти в дампы ядра.
- sd : Мягкий грязный флаг.
- mm : Смешанная область карты.
- hg : Огромный флаг рекомендации страницы.
- nh : Нет флага рекомендации огромной страницы.
- mg : Объединяемый флаг рекомендации.
- bt : страница с защитой от нестабильности температуры смещения ARM64.
- mt : Теги расширения маркировки памяти ARM64 включены.
- гм : отсутствует отслеживание Userfaultfd.
- uw : Userfaultfd отслеживание защиты от записи.
Управление памятью сложно
И работать в обратном направлении от таблиц данных, чтобы понять, что на самом деле происходит, сложно. Но, по крайней мере pmap
дает вам полную картину, поэтому у вас больше шансов выяснить, что вам нужно знать.
Интересно отметить, что наш пример программы скомпилирован в двоичный исполняемый файл размером 16 КБ, но при этом использует (или совместно использует) около 2756 КБ памяти, почти полностью из-за библиотек времени выполнения.
Еще один изящный трюк заключается в том, что вы можете использовать pmap
и pidof
вместе, объединяя действия по поиску PID процесса и передаче его в pmap
в одну команду:
pmap $(pidof pm)