xref: /linux/tools/mm/show_page_info.py (revision 8d2b0853add1d7534dc0794e3c8e0b9e8c4ec640)
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