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