memcg_shrinker.py 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  1. #!/usr/bin/env python3
  2. #
  3. # Copyright (C) 2022 Roman Gushchin <roman.gushchin@linux.dev>
  4. # Copyright (C) 2022 Meta
  5. import os
  6. import argparse
  7. def scan_cgroups(cgroup_root):
  8. cgroups = {}
  9. for root, subdirs, _ in os.walk(cgroup_root):
  10. for cgroup in subdirs:
  11. path = os.path.join(root, cgroup)
  12. ino = os.stat(path).st_ino
  13. cgroups[ino] = path
  14. # (memcg ino, path)
  15. return cgroups
  16. def scan_shrinkers(shrinker_debugfs):
  17. shrinkers = []
  18. for root, subdirs, _ in os.walk(shrinker_debugfs):
  19. for shrinker in subdirs:
  20. count_path = os.path.join(root, shrinker, "count")
  21. with open(count_path) as f:
  22. for line in f.readlines():
  23. items = line.split(' ')
  24. ino = int(items[0])
  25. # (count, shrinker, memcg ino)
  26. shrinkers.append((int(items[1]), shrinker, ino))
  27. return shrinkers
  28. def main():
  29. parser = argparse.ArgumentParser(description='Display biggest shrinkers')
  30. parser.add_argument('-n', '--lines', type=int, help='Number of lines to print')
  31. args = parser.parse_args()
  32. cgroups = scan_cgroups("/sys/fs/cgroup/")
  33. shrinkers = scan_shrinkers("/sys/kernel/debug/shrinker/")
  34. shrinkers.sort(reverse = True, key = lambda x: x[0])
  35. n = 0
  36. for s in shrinkers:
  37. count, name, ino = (s[0], s[1], s[2])
  38. if count == 0:
  39. break
  40. if ino == 0 or ino == 1:
  41. cg = "/"
  42. else:
  43. try:
  44. cg = cgroups[ino]
  45. except KeyError:
  46. cg = "unknown (%d)" % ino
  47. print("%-8s %-20s %s" % (count, name, cg))
  48. n += 1
  49. if args.lines and n >= args.lines:
  50. break
  51. if __name__ == '__main__':
  52. main()