Files
BUT2FI_2025_R3.05/tp/tp1/src/scripts/parse_smaps.py
2025-09-02 17:14:06 +02:00

124 lines
3.6 KiB
Python
Executable File

#!/usr/bin/env python
#
# Author: Craig Chi <craig10624@gmail.com>
#
import sys
import os
import getopt
from collections import defaultdict, OrderedDict
from subprocess import check_output
def usage():
print("""
usage: parse_smaps.py [-p process_name] [-t memory_type] [-h] [smaps_filename]
example: parse_smaps.py /proc/12424/smaps
parse_smaps.py -p smbd
parse_smaps.py -p smbd -t Pss
""")
def print_header(mem_idx):
print('=' * 70)
for title in zip(*map(lambda x: x.split('_'), mem_idx.keys()),
('', '= Total : library')):
print('{:>8} + {:>8} + {:>8} + {:>8} {}'.format(*title,))
print('=' * 70)
def main():
try:
opts, args = getopt.getopt(sys.argv[1:], 'p:t:ah',
['process-name=', 'memory-type=',
'all', 'help'])
except getopt.GetoptError as err:
print(err)
sys.exit(2)
ps_name = ''
mem_type = ''
mem_idx = OrderedDict([
('Private_Clean', 0),
('Private_Dirty', 1),
('Shared_Clean', 2),
('Shared_Dirty', 3)
])
for o, a in opts:
if o in ('-p', '--process-name'):
ps_name = a
elif o in ('-t', '--memory-type'):
mem_type = a
mem_idx = {a: 0}
else:
usage()
sys.exit(2)
if (len(args) == 0 and ps_name == '') or len(args) > 1:
usage()
sys.exit(2)
smaps_file = ''
if ps_name == '':
smaps_file = os.path.abspath(args[0])
else:
try:
pids = check_output(['pidof', ps_name]).decode().strip().split()
if len(pids) > 1:
print('There are multiple pids:')
for i, p in enumerate(pids):
cmdline_file = '/proc/' + p + '/cmdline'
with open(cmdline_file, 'r') as cmdline:
line = next(cmdline)
print('[{}] {:>8}: {}'.format(i, p, line))
num = input('Choose which one process you want (default=0): ')
num = int(num) if num != '' else 0
pid = pids[num]
else:
pid = pids[0]
except Exception as err:
print(err)
sys.exit(1)
smaps_file = '/proc/' + pid + '/smaps'
mapinfo = defaultdict(lambda: [0] * len(mem_idx))
total = [0] * len(mem_idx)
with open(smaps_file, 'r') as smap:
for line in smap:
line_arr = line.split()
if '-' in line_arr[0]:
if len(line_arr) < 6:
filename = '[anonymous]'
else:
filename = os.path.basename(line_arr[-1])
else:
line_arr[0] = line_arr[0].strip(':')
if line_arr[0] in mem_idx:
mapinfo[filename][mem_idx[line_arr[0]]] += int(line_arr[1])
total[mem_idx[line_arr[0]]] += int(line_arr[1])
if mem_type == '':
print_header(mem_idx)
for filename, mem in sorted(mapinfo.items(), key=lambda x: -sum(x[1])):
print('{:>5} kB + {:>5} kB + {:>5} kB + {:>5} kB'
' = {:>5} kB : {:<}'.format(*mem, sum(mem), filename))
print('=' * 70)
print('{:>5} kB + {:>5} kB + {:>5} kB + {:>5} kB'
' = {:>5} kB : Total'.format(*total, sum(total)))
else:
for filename, mem in sorted(mapinfo.items(), key=lambda x: -sum(x[1])):
print('{:>11} kB {:<}'.format(mem[0], filename))
print('=' * 30)
print('Total: {} kB'.format(total[0]))
if __name__ == '__main__':
main()