File: vtop.py

package info (click to toggle)
libkdumpfile 0.5.4-3
  • links: PTS, VCS
  • area: main
  • in suites: sid, trixie
  • size: 3,800 kB
  • sloc: ansic: 34,639; sh: 3,853; python: 1,490; makefile: 755
file content (90 lines) | stat: -rw-r--r-- 2,807 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
#!/usr/bin/env python
# vim:sw=4 ts=4 et

from __future__ import print_function
import kdumpfile
import addrxlat
from sys import argv

tbl_names = ( 'PAGE', 'PTE', 'PMD', 'PUD', 'PGD' )

class kphysnote(object):
    def __init__(self, ctx, sys):
        self.ctx = ctx
        self.sys = sys

        map = sys.get_map(addrxlat.SYS_MAP_MACHPHYS_KPHYS)
        self.ident = True
        for range in map:
            if range.meth == addrxlat.SYS_METH_NONE:
                continue
            meth = sys.get_meth(range.meth)
            if meth.kind != addrxlat.LINEAR or meth.off != 0:
                self.ident = False
                break

    def note(self, fulladdr, fmt='{}'):
        if self.ident or fulladdr.addrspace == addrxlat.KPHYSADDR:
            return ''
        tmp = fulladdr.copy()
        try:
            tmp.conv(addrxlat.KPHYSADDR, self.ctx, self.sys)
            return fmt.format('{:x}'.format(tmp.addr))
        except (addrxlat.NotPresentError, addrxlat.NoDataError):
            return fmt.format('N/A')

def vtop(addr, ctx, sys):
    fulladdr = addrxlat.FullAddress(addrxlat.KVADDR, addr)
    print('{:16}  {:16}'.format('VIRTUAL', 'PHYSICAL'))
    try:
        fulladdr.conv(addrxlat.KPHYSADDR, ctx, sys)
        print('{:<16x}  {:<16x}\n'.format(addr, fulladdr.addr))
    except addrxlat.BaseException:
        print('{:<16x}  {:<16}\n'.format(addr, '---'))

    step = addrxlat.Step(ctx, sys)
    meth = sys.get_map(addrxlat.SYS_MAP_HW).search(addr)
    if meth == addrxlat.SYS_METH_NONE:
        meth = sys.get_map(addrxlat.SYS_MAP_KV_PHYS).search(addr)
    if meth == addrxlat.SYS_METH_NONE:
        print('NO METHOD')
        return

    step.meth = sys.get_meth(meth)
    step.launch(addr)

    note = kphysnote(ctx, sys)
    while step.remain:
        tbl = tbl_names[step.remain - 1]
        if step.remain > 1:
            addr = step.base.copy()
            addr.addr += step.idx[step.remain - 1] * step.elemsz
        else:
            addr = step.base
        remark = note.note(addr, ' ({})')
        print('{:>4}: {:16x}{}'.format(tbl, addr.addr, remark), end='')

        try:
            step.step()
            if step.remain and step.raw is not None:
                print(' => {:x}'.format(step.raw))
        except addrxlat.NotPresentError:
            print(' => {:x}  NOT PRESENT'.format(step.raw))
            return
    print()

if len(argv) != 3:
    print('Usage: {} <dumpfile> <virtual address>'.format(argv[0]))
    exit(1)

kdf = kdumpfile.kdumpfile(argv[1])
kdf.addrxlat_convert = addrxlat.convert
kdf.attr['addrxlat.ostype'] = 'linux'
sys = kdf.get_addrxlat_sys()
ctx = kdf.get_addrxlat_ctx()

addr = int(argv[2], base=0)
try:
    vtop(addr, ctx, sys)
except addrxlat.BaseException as e:
    print('Translation failed: {}'.format(e.message))