show_page_info.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. #!/usr/bin/env drgn
  2. # SPDX-License-Identifier: GPL-2.0-only
  3. # Copyright (C) 2025 Ye Liu <liuye@kylinos.cn>
  4. import argparse
  5. import sys
  6. from drgn import Object, FaultError, PlatformFlags, cast
  7. from drgn.helpers.linux import find_task, follow_page, page_size
  8. from drgn.helpers.linux.mm import (
  9. decode_page_flags, page_to_pfn, page_to_phys, page_to_virt, vma_find,
  10. PageSlab, PageCompound, PageHead, PageTail, compound_head, compound_order, compound_nr
  11. )
  12. from drgn.helpers.linux.cgroup import cgroup_name, cgroup_path
  13. DESC = """
  14. This is a drgn script to show the page state.
  15. For more info on drgn, visit https://github.com/osandov/drgn.
  16. """
  17. def format_page_data(page):
  18. """
  19. Format raw page data into a readable hex dump with "RAW:" prefix.
  20. :param page: drgn.Object instance representing the page.
  21. :return: Formatted string of memory contents.
  22. """
  23. try:
  24. address = page.value_()
  25. size = prog.type("struct page").size
  26. if prog.platform.flags & PlatformFlags.IS_64_BIT:
  27. word_size = 8
  28. else:
  29. word_size = 4
  30. num_words = size // word_size
  31. values = []
  32. for i in range(num_words):
  33. word_address = address + i * word_size
  34. word = prog.read_word(word_address)
  35. values.append(f"{word:0{word_size * 2}x}")
  36. lines = [f"RAW: {' '.join(values[i:i + 4])}" for i in range(0, len(values), 4)]
  37. return "\n".join(lines)
  38. except FaultError as e:
  39. return f"Error reading memory: {e}"
  40. except Exception as e:
  41. return f"Unexpected error: {e}"
  42. def get_memcg_info(page):
  43. """Retrieve memory cgroup information for a page."""
  44. try:
  45. MEMCG_DATA_OBJEXTS = prog.constant("MEMCG_DATA_OBJEXTS").value_()
  46. MEMCG_DATA_KMEM = prog.constant("MEMCG_DATA_KMEM").value_()
  47. mask = prog.constant('__NR_MEMCG_DATA_FLAGS').value_() - 1
  48. memcg_data = page.memcg_data.read_()
  49. if memcg_data & MEMCG_DATA_OBJEXTS:
  50. slabobj_ext = cast("struct slabobj_ext *", memcg_data & ~mask)
  51. memcg = slabobj_ext.objcg.memcg.value_()
  52. elif memcg_data & MEMCG_DATA_KMEM:
  53. objcg = cast("struct obj_cgroup *", memcg_data & ~mask)
  54. memcg = objcg.memcg.value_()
  55. else:
  56. memcg = cast("struct mem_cgroup *", memcg_data & ~mask)
  57. if memcg.value_() == 0:
  58. return "none", "/sys/fs/cgroup/memory/"
  59. cgrp = memcg.css.cgroup
  60. return cgroup_name(cgrp).decode(), f"/sys/fs/cgroup/memory{cgroup_path(cgrp).decode()}"
  61. except FaultError as e:
  62. return "unknown", f"Error retrieving memcg info: {e}"
  63. except Exception as e:
  64. return "unknown", f"Unexpected error: {e}"
  65. def show_page_state(page, addr, mm, pid, task):
  66. """Display detailed information about a page."""
  67. try:
  68. print(f'PID: {pid} Comm: {task.comm.string_().decode()} mm: {hex(mm)}')
  69. try:
  70. print(format_page_data(page))
  71. except FaultError as e:
  72. print(f"Error reading page data: {e}")
  73. fields = {
  74. "Page Address": hex(page.value_()),
  75. "Page Flags": decode_page_flags(page),
  76. "Page Size": prog["PAGE_SIZE"].value_(),
  77. "Page PFN": hex(page_to_pfn(page).value_()),
  78. "Page Physical": hex(page_to_phys(page).value_()),
  79. "Page Virtual": hex(page_to_virt(page).value_()),
  80. "Page Refcount": page._refcount.counter.value_(),
  81. "Page Mapcount": page._mapcount.counter.value_(),
  82. "Page Index": hex(page.__folio_index.value_()),
  83. "Page Memcg Data": hex(page.memcg_data.value_()),
  84. }
  85. memcg_name, memcg_path = get_memcg_info(page)
  86. fields["Memcg Name"] = memcg_name
  87. fields["Memcg Path"] = memcg_path
  88. fields["Page Mapping"] = hex(page.mapping.value_())
  89. fields["Page Anon/File"] = "Anon" if page.mapping.value_() & 0x1 else "File"
  90. try:
  91. vma = vma_find(mm, addr)
  92. fields["Page VMA"] = hex(vma.value_())
  93. fields["VMA Start"] = hex(vma.vm_start.value_())
  94. fields["VMA End"] = hex(vma.vm_end.value_())
  95. except FaultError as e:
  96. fields["Page VMA"] = "Unavailable"
  97. fields["VMA Start"] = "Unavailable"
  98. fields["VMA End"] = "Unavailable"
  99. print(f"Error retrieving VMA information: {e}")
  100. # Calculate the maximum field name length for alignment
  101. max_field_len = max(len(field) for field in fields)
  102. # Print aligned fields
  103. for field, value in fields.items():
  104. print(f"{field}:".ljust(max_field_len + 2) + f"{value}")
  105. # Additional information about the page
  106. if PageSlab(page):
  107. print("This page belongs to the slab allocator.")
  108. if PageCompound(page):
  109. print("This page is part of a compound page.")
  110. if PageHead(page):
  111. print("This page is the head page of a compound page.")
  112. if PageTail(page):
  113. print("This page is the tail page of a compound page.")
  114. print(f"{'Head Page:'.ljust(max_field_len + 2)}{hex(compound_head(page).value_())}")
  115. print(f"{'Compound Order:'.ljust(max_field_len + 2)}{compound_order(page).value_()}")
  116. print(f"{'Number of Pages:'.ljust(max_field_len + 2)}{compound_nr(page).value_()}")
  117. else:
  118. print("This page is not part of a compound page.")
  119. except FaultError as e:
  120. print(f"Error accessing page state: {e}")
  121. except Exception as e:
  122. print(f"Unexpected error: {e}")
  123. def main():
  124. """Main function to parse arguments and display page state."""
  125. parser = argparse.ArgumentParser(description=DESC, formatter_class=argparse.RawTextHelpFormatter)
  126. parser.add_argument('pid', metavar='PID', type=int, help='Target process ID (PID)')
  127. parser.add_argument('vaddr', metavar='VADDR', type=str, help='Target virtual address in hexadecimal format (e.g., 0x7fff1234abcd)')
  128. args = parser.parse_args()
  129. try:
  130. vaddr = int(args.vaddr, 16)
  131. except ValueError:
  132. sys.exit(f"Error: Invalid virtual address format: {args.vaddr}")
  133. try:
  134. task = find_task(args.pid)
  135. mm = task.mm
  136. page = follow_page(mm, vaddr)
  137. if page:
  138. show_page_state(page, vaddr, mm, args.pid, task)
  139. else:
  140. sys.exit(f"Address {hex(vaddr)} is not mapped.")
  141. except FaultError as e:
  142. sys.exit(f"Error accessing task or memory: {e}")
  143. except Exception as e:
  144. sys.exit(f"Unexpected error: {e}")
  145. if __name__ == "__main__":
  146. main()