Commit 0393c1c8 authored by Oliver O'Halloran's avatar Oliver O'Halloran Committed by Greg Kroah-Hartman

powerpc/mm: Add physical address to Linux page table dump

commit aaa22952 upstream.

The current page table dumper scans the Linux page tables and coalesces mappings
with adjacent virtual addresses and similar PTE flags. This behaviour is
somewhat broken when you consider the IOREMAP space where entirely unrelated
mappings will appear to be virtually contiguous. This patch modifies the range
coalescing so that only ranges that are both physically and virtually contiguous
are combined. This patch also adds to the dump output the physical address at
the start of each range.

Fixes: 8eb07b18 ("powerpc/mm: Dump linux pagetables")
Signed-off-by: default avatarOliver O'Halloran <oohall@gmail.com>
[mpe: Print the physicall address with 0x like the other addresses]
Signed-off-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 35434c87
...@@ -57,6 +57,8 @@ struct pg_state { ...@@ -57,6 +57,8 @@ struct pg_state {
struct seq_file *seq; struct seq_file *seq;
const struct addr_marker *marker; const struct addr_marker *marker;
unsigned long start_address; unsigned long start_address;
unsigned long start_pa;
unsigned long last_pa;
unsigned int level; unsigned int level;
u64 current_flags; u64 current_flags;
}; };
...@@ -253,7 +255,9 @@ static void dump_addr(struct pg_state *st, unsigned long addr) ...@@ -253,7 +255,9 @@ static void dump_addr(struct pg_state *st, unsigned long addr)
const char *unit = units; const char *unit = units;
unsigned long delta; unsigned long delta;
seq_printf(st->seq, "0x%016lx-0x%016lx ", st->start_address, addr-1); seq_printf(st->seq, "0x%016lx-0x%016lx ", st->start_address, addr-1);
seq_printf(st->seq, "0x%016lx ", st->start_pa);
delta = (addr - st->start_address) >> 10; delta = (addr - st->start_address) >> 10;
/* Work out what appropriate unit to use */ /* Work out what appropriate unit to use */
while (!(delta & 1023) && unit[1]) { while (!(delta & 1023) && unit[1]) {
...@@ -268,11 +272,15 @@ static void note_page(struct pg_state *st, unsigned long addr, ...@@ -268,11 +272,15 @@ static void note_page(struct pg_state *st, unsigned long addr,
unsigned int level, u64 val) unsigned int level, u64 val)
{ {
u64 flag = val & pg_level[level].mask; u64 flag = val & pg_level[level].mask;
u64 pa = val & PTE_RPN_MASK;
/* At first no level is set */ /* At first no level is set */
if (!st->level) { if (!st->level) {
st->level = level; st->level = level;
st->current_flags = flag; st->current_flags = flag;
st->start_address = addr; st->start_address = addr;
st->start_pa = pa;
st->last_pa = pa;
seq_printf(st->seq, "---[ %s ]---\n", st->marker->name); seq_printf(st->seq, "---[ %s ]---\n", st->marker->name);
/* /*
* Dump the section of virtual memory when: * Dump the section of virtual memory when:
...@@ -280,9 +288,11 @@ static void note_page(struct pg_state *st, unsigned long addr, ...@@ -280,9 +288,11 @@ static void note_page(struct pg_state *st, unsigned long addr,
* - we change levels in the tree. * - we change levels in the tree.
* - the address is in a different section of memory and is thus * - the address is in a different section of memory and is thus
* used for a different purpose, regardless of the flags. * used for a different purpose, regardless of the flags.
* - the pa of this page is not adjacent to the last inspected page
*/ */
} else if (flag != st->current_flags || level != st->level || } else if (flag != st->current_flags || level != st->level ||
addr >= st->marker[1].start_address) { addr >= st->marker[1].start_address ||
pa != st->last_pa + PAGE_SIZE) {
/* Check the PTE flags */ /* Check the PTE flags */
if (st->current_flags) { if (st->current_flags) {
...@@ -306,8 +316,12 @@ static void note_page(struct pg_state *st, unsigned long addr, ...@@ -306,8 +316,12 @@ static void note_page(struct pg_state *st, unsigned long addr,
seq_printf(st->seq, "---[ %s ]---\n", st->marker->name); seq_printf(st->seq, "---[ %s ]---\n", st->marker->name);
} }
st->start_address = addr; st->start_address = addr;
st->start_pa = pa;
st->last_pa = pa;
st->current_flags = flag; st->current_flags = flag;
st->level = level; st->level = level;
} else {
st->last_pa = pa;
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment