| # mem-phys-addr.py: Resolve physical address samples |
| # SPDX-License-Identifier: GPL-2.0 |
| # |
| # Copyright (c) 2018, Intel Corporation. |
| |
| import os |
| import sys |
| import re |
| import bisect |
| import collections |
| from dataclasses import dataclass |
| from typing import (Dict, Optional) |
| |
| sys.path.append(os.environ['PERF_EXEC_PATH'] + \ |
| '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') |
| |
| @dataclass(frozen=True) |
| class IomemEntry: |
| """Read from a line in /proc/iomem""" |
| begin: int |
| end: int |
| indent: int |
| label: str |
| |
| # Physical memory layout from /proc/iomem. Key is the indent and then |
| # a list of ranges. |
| iomem: Dict[int, list[IomemEntry]] = collections.defaultdict(list) |
| # Child nodes from the iomem parent. |
| children: Dict[IomemEntry, set[IomemEntry]] = collections.defaultdict(set) |
| # Maximum indent seen before an entry in the iomem file. |
| max_indent: int = 0 |
| # Count for each range of memory. |
| load_mem_type_cnt: Dict[IomemEntry, int] = collections.Counter() |
| # Perf event name set from the first sample in the data. |
| event_name: Optional[str] = None |
| |
| def parse_iomem(): |
| """Populate iomem from /proc/iomem file""" |
| global iomem |
| global max_indent |
| global children |
| with open('/proc/iomem', 'r', encoding='ascii') as f: |
| for line in f: |
| indent = 0 |
| while line[indent] == ' ': |
| indent += 1 |
| if indent > max_indent: |
| max_indent = indent |
| m = re.split('-|:', line, 2) |
| begin = int(m[0], 16) |
| end = int(m[1], 16) |
| label = m[2].strip() |
| entry = IomemEntry(begin, end, indent, label) |
| # Before adding entry, search for a parent node using its begin. |
| if indent > 0: |
| parent = find_memory_type(begin) |
| assert parent, f"Given indent expected a parent for {label}" |
| children[parent].add(entry) |
| iomem[indent].append(entry) |
| |
| def find_memory_type(phys_addr) -> Optional[IomemEntry]: |
| """Search iomem for the range containing phys_addr with the maximum indent""" |
| for i in range(max_indent, -1, -1): |
| if i not in iomem: |
| continue |
| position = bisect.bisect_right(iomem[i], phys_addr, |
| key=lambda entry: entry.begin) |
| if position is None: |
| continue |
| iomem_entry = iomem[i][position-1] |
| if iomem_entry.begin <= phys_addr <= iomem_entry.end: |
| return iomem_entry |
| print(f"Didn't find {phys_addr}") |
| return None |
| |
| def print_memory_type(): |
| print(f"Event: {event_name}") |
| print(f"{'Memory type':<40} {'count':>10} {'percentage':>10}") |
| print(f"{'-' * 40:<40} {'-' * 10:>10} {'-' * 10:>10}") |
| total = sum(load_mem_type_cnt.values()) |
| # Add count from children into the parent. |
| for i in range(max_indent, -1, -1): |
| if i not in iomem: |
| continue |
| for entry in iomem[i]: |
| global children |
| for child in children[entry]: |
| if load_mem_type_cnt[child] > 0: |
| load_mem_type_cnt[entry] += load_mem_type_cnt[child] |
| |
| def print_entries(entries): |
| """Print counts from parents down to their children""" |
| global children |
| for entry in sorted(entries, |
| key = lambda entry: load_mem_type_cnt[entry], |
| reverse = True): |
| count = load_mem_type_cnt[entry] |
| if count > 0: |
| mem_type = ' ' * entry.indent + f"{entry.begin:x}-{entry.end:x} : {entry.label}" |
| percent = 100 * count / total |
| print(f"{mem_type:<40} {count:>10} {percent:>10.1f}") |
| print_entries(children[entry]) |
| |
| print_entries(iomem[0]) |
| |
| def trace_begin(): |
| parse_iomem() |
| |
| def trace_end(): |
| print_memory_type() |
| |
| def process_event(param_dict): |
| if "sample" not in param_dict: |
| return |
| |
| sample = param_dict["sample"] |
| if "phys_addr" not in sample: |
| return |
| |
| phys_addr = sample["phys_addr"] |
| entry = find_memory_type(phys_addr) |
| if entry: |
| load_mem_type_cnt[entry] += 1 |
| |
| global event_name |
| if event_name is None: |
| event_name = param_dict["ev_name"] |