Files
nohang/trash/thanatolog2
2019-07-08 00:04:37 +09:00

178 lines
4.0 KiB
Python
Executable File

#!/usr/bin/env python3
import os
from time import sleep, time
from signal import (signal,
SIGKILL, SIGTERM, SIGINT, SIGQUIT,
SIGCONT, SIGUSR1, SIGUSR2,
SIGHUP, SIGABRT, SIGSEGV, SIGBUS)
from sys import argv, exit
def mlockall():
"""Lock all memory to prevent swapping the process."""
from ctypes import CDLL
MCL_CURRENT = 1
MCL_FUTURE = 2
MCL_ONFAULT = 4
libc = CDLL('libc.so.6', use_errno=True)
result = libc.mlockall(
MCL_CURRENT | MCL_FUTURE | MCL_ONFAULT
)
def check_mem():
"""find mem_available"""
with open('/proc/meminfo') as f:
for n, line in enumerate(f):
if n is 2:
mem_available = int(line.split(':')[1][:-4])
return round(mem_available / 1024.0)
def pid_to_name(pid):
"""
"""
try:
with open('/proc/' + pid + '/comm', 'rb') as f:
return f.read().decode('utf-8', 'ignore')[:-1]
except FileNotFoundError:
return ''
except ProcessLookupError:
return ''
def pid_to_state(pid):
x = rline1('/proc/' + pid + '/stat')
if ')' in x:
return x.rpartition(')')[2][1]
else:
return ' '
def pid_to_rss(pid, SC_PAGESIZE):
try:
rss = rline1('/proc/{}/statm'.format(pid)).split(' ')[1]
except IndexError:
rss = '-0'
return round(int(rss) * SC_PAGESIZE / (1024.0 ** 2))
def pid_to_realpath(pid):
try:
return os.path.realpath('/proc/' + pid + '/exe')
except FileNotFoundError:
return ''
def rline1(path):
"""read 1st line from path."""
try:
with open(path) as f:
for line in f:
return line[:-1]
except UnicodeDecodeError:
with open(path, 'rb') as f:
return f.read(999).decode(
'utf-8', 'ignore').split('\n')[0] # use partition()!
except FileNotFoundError:
return 'FileNotFoundError'
except ProcessLookupError:
return 'ProcessLookupError'
###############################################################################
if len(argv) != 2:
print("""Usage:
thanatolog PID""")
exit()
mlockall()
SC_PAGESIZE = os.sysconf(os.sysconf_names['SC_PAGESIZE'])
pid = argv[1]
name = pid_to_name(pid)
rss0 = float(pid_to_rss(pid, SC_PAGESIZE))
ma = check_mem()
print('PID:', pid)
print('Name:', name)
print('RSS at startup: {} (100.0 %)'.format(int(rss0)))
print('MemAvail:', ma)
send_signal = SIGKILL
# os.kill(int(pid), SIGCONT)
t0 = time()
for i in range(10):
sleep(0.001)
rpe = os.path.exists('/proc/{}/exe'.format(pid))
rss = pid_to_rss(pid, SC_PAGESIZE)
pe = os.path.exists('/proc/{}'.format(pid))
t1 = time()
d = t1 - t0
state = pid_to_state(pid)
ma = check_mem()
print('RP: {} | RSS: {} ({} %) | {} | t: {:0<6} | MemAv'
'ail: {}'.format(rpe, rss, round(float(rss) / (
rss0 + 0.0001) * 100, 1), state, str(round(d, 4)), ma))
print()
print('Send SIGKILL')
os.kill(int(pid), send_signal)
t0 = time()
ma0 = ma
while True:
sleep(0.001)
rpe = os.path.exists('/proc/{}/exe'.format(pid))
rss = pid_to_rss(pid, SC_PAGESIZE)
pe = os.path.exists('/proc/{}'.format(pid))
t1 = time()
d = t1 - t0
state = pid_to_state(pid)
ma = check_mem()
print('RP: {} | RSS: {} ({} %) | State: {} | time: {} | MemAvail: {} | dMA {}'.format(
rpe, rss, round(float(rss) / (rss0 + 0.0001) * 100, 1), state, round(d, 3), ma, ma0 - ma))
if pe is False:
break
print('Process {} ({}) died in {} sec'.format(pid, name, round(d, 3)))
print()
for i in range(10):
sleep(0.001)
rpe = os.path.exists('/proc/{}/exe'.format(pid))
rss = pid_to_rss(pid, SC_PAGESIZE)
pe = os.path.exists('/proc/{}'.format(pid))
t1 = time()
d = t1 - t0
state = pid_to_state(pid)
ma = check_mem()
print('RP: {} | RSS: {} ({} %) | State: {} | time: {} | MemAvail: {} | dMA {}'.format(
rpe, rss, round(float(rss) / (rss0 + 0.0001) * 100, 1), state, round(d, 3), ma, ma0 - ma))