116fef6408
EFI is complicated enough that being able to view its pagetables is quite helpful. Rather than requiring users to fish it out of dmesg on an appropriately configured kernel, let users view it in debugfs as well. Signed-off-by: Andy Lutomirski <luto@kernel.org> Acked-by: Ard Biesheuvel <ard.biesheuvel@linaro.org> Cc: Borislav Petkov <bp@alien8.de> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: linux-efi@vger.kernel.org Cc: linux-mm@kvack.org Link: http://lkml.kernel.org/r/ba158a93f3250e6fca752cff2cfb1fcdd9f2b50c.1517414050.git.luto@kernel.org [ Fixed trivial whitespace damage and fixed missing export. ] Signed-off-by: Ingo Molnar <mingo@kernel.org>
147 lines
3.3 KiB
C
147 lines
3.3 KiB
C
#include <linux/debugfs.h>
|
|
#include <linux/module.h>
|
|
#include <linux/seq_file.h>
|
|
#include <asm/pgtable.h>
|
|
|
|
static int ptdump_show(struct seq_file *m, void *v)
|
|
{
|
|
ptdump_walk_pgd_level_debugfs(m, NULL, false);
|
|
return 0;
|
|
}
|
|
|
|
static int ptdump_open(struct inode *inode, struct file *filp)
|
|
{
|
|
return single_open(filp, ptdump_show, NULL);
|
|
}
|
|
|
|
static const struct file_operations ptdump_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = ptdump_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = single_release,
|
|
};
|
|
|
|
static int ptdump_show_curknl(struct seq_file *m, void *v)
|
|
{
|
|
if (current->mm->pgd) {
|
|
down_read(¤t->mm->mmap_sem);
|
|
ptdump_walk_pgd_level_debugfs(m, current->mm->pgd, false);
|
|
up_read(¤t->mm->mmap_sem);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int ptdump_open_curknl(struct inode *inode, struct file *filp)
|
|
{
|
|
return single_open(filp, ptdump_show_curknl, NULL);
|
|
}
|
|
|
|
static const struct file_operations ptdump_curknl_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = ptdump_open_curknl,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = single_release,
|
|
};
|
|
|
|
#ifdef CONFIG_PAGE_TABLE_ISOLATION
|
|
static struct dentry *pe_curusr;
|
|
|
|
static int ptdump_show_curusr(struct seq_file *m, void *v)
|
|
{
|
|
if (current->mm->pgd) {
|
|
down_read(¤t->mm->mmap_sem);
|
|
ptdump_walk_pgd_level_debugfs(m, current->mm->pgd, true);
|
|
up_read(¤t->mm->mmap_sem);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int ptdump_open_curusr(struct inode *inode, struct file *filp)
|
|
{
|
|
return single_open(filp, ptdump_show_curusr, NULL);
|
|
}
|
|
|
|
static const struct file_operations ptdump_curusr_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = ptdump_open_curusr,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = single_release,
|
|
};
|
|
#endif
|
|
|
|
#if defined(CONFIG_EFI) && defined(CONFIG_X86_64)
|
|
extern pgd_t *efi_pgd;
|
|
static struct dentry *pe_efi;
|
|
|
|
static int ptdump_show_efi(struct seq_file *m, void *v)
|
|
{
|
|
if (efi_pgd)
|
|
ptdump_walk_pgd_level_debugfs(m, efi_pgd, false);
|
|
return 0;
|
|
}
|
|
|
|
static int ptdump_open_efi(struct inode *inode, struct file *filp)
|
|
{
|
|
return single_open(filp, ptdump_show_efi, NULL);
|
|
}
|
|
|
|
static const struct file_operations ptdump_efi_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = ptdump_open_efi,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = single_release,
|
|
};
|
|
#endif
|
|
|
|
static struct dentry *dir, *pe_knl, *pe_curknl;
|
|
|
|
static int __init pt_dump_debug_init(void)
|
|
{
|
|
dir = debugfs_create_dir("page_tables", NULL);
|
|
if (!dir)
|
|
return -ENOMEM;
|
|
|
|
pe_knl = debugfs_create_file("kernel", 0400, dir, NULL,
|
|
&ptdump_fops);
|
|
if (!pe_knl)
|
|
goto err;
|
|
|
|
pe_curknl = debugfs_create_file("current_kernel", 0400,
|
|
dir, NULL, &ptdump_curknl_fops);
|
|
if (!pe_curknl)
|
|
goto err;
|
|
|
|
#ifdef CONFIG_PAGE_TABLE_ISOLATION
|
|
pe_curusr = debugfs_create_file("current_user", 0400,
|
|
dir, NULL, &ptdump_curusr_fops);
|
|
if (!pe_curusr)
|
|
goto err;
|
|
#endif
|
|
|
|
#if defined(CONFIG_EFI) && defined(CONFIG_X86_64)
|
|
pe_efi = debugfs_create_file("efi", 0400, dir, NULL, &ptdump_efi_fops);
|
|
if (!pe_efi)
|
|
goto err;
|
|
#endif
|
|
|
|
return 0;
|
|
err:
|
|
debugfs_remove_recursive(dir);
|
|
return -ENOMEM;
|
|
}
|
|
|
|
static void __exit pt_dump_debug_exit(void)
|
|
{
|
|
debugfs_remove_recursive(dir);
|
|
}
|
|
|
|
module_init(pt_dump_debug_init);
|
|
module_exit(pt_dump_debug_exit);
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_AUTHOR("Arjan van de Ven <arjan@linux.intel.com>");
|
|
MODULE_DESCRIPTION("Kernel debugging helper that dumps pagetables");
|