16d7d7433SHuang, Ying /* 26d7d7433SHuang, Ying * Architecture specific debugfs files 36d7d7433SHuang, Ying * 46d7d7433SHuang, Ying * Copyright (C) 2007, Intel Corp. 56d7d7433SHuang, Ying * Huang Ying <ying.huang@intel.com> 66d7d7433SHuang, Ying * 76d7d7433SHuang, Ying * This file is released under the GPLv2. 86d7d7433SHuang, Ying */ 96d7d7433SHuang, Ying #include <linux/debugfs.h> 10c14b2adfSHuang, Ying #include <linux/uaccess.h> 11390cd85cSJaswinder Singh Rajput #include <linux/module.h> 125a0e3ad6STejun Heo #include <linux/slab.h> 136d7d7433SHuang, Ying #include <linux/init.h> 14390cd85cSJaswinder Singh Rajput #include <linux/stat.h> 15c14b2adfSHuang, Ying #include <linux/io.h> 16c14b2adfSHuang, Ying #include <linux/mm.h> 176d7d7433SHuang, Ying 186d7d7433SHuang, Ying #include <asm/setup.h> 196d7d7433SHuang, Ying 20ae79cdaaSvenkatesh.pallipadi@intel.com struct dentry *arch_debugfs_dir; 21ae79cdaaSvenkatesh.pallipadi@intel.com EXPORT_SYMBOL(arch_debugfs_dir); 22ae79cdaaSvenkatesh.pallipadi@intel.com 236d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 24c14b2adfSHuang, Ying struct setup_data_node { 25c14b2adfSHuang, Ying u64 paddr; 26c14b2adfSHuang, Ying u32 type; 27c14b2adfSHuang, Ying u32 len; 28c14b2adfSHuang, Ying }; 29c14b2adfSHuang, Ying 30390cd85cSJaswinder Singh Rajput static ssize_t setup_data_read(struct file *file, char __user *user_buf, 31390cd85cSJaswinder Singh Rajput size_t count, loff_t *ppos) 32c14b2adfSHuang, Ying { 33c14b2adfSHuang, Ying struct setup_data_node *node = file->private_data; 34c14b2adfSHuang, Ying unsigned long remain; 35c14b2adfSHuang, Ying loff_t pos = *ppos; 36c14b2adfSHuang, Ying struct page *pg; 37c14b2adfSHuang, Ying void *p; 38c14b2adfSHuang, Ying u64 pa; 39c14b2adfSHuang, Ying 40c14b2adfSHuang, Ying if (pos < 0) 41c14b2adfSHuang, Ying return -EINVAL; 42390cd85cSJaswinder Singh Rajput 43c14b2adfSHuang, Ying if (pos >= node->len) 44c14b2adfSHuang, Ying return 0; 45c14b2adfSHuang, Ying 46c14b2adfSHuang, Ying if (count > node->len - pos) 47c14b2adfSHuang, Ying count = node->len - pos; 48390cd85cSJaswinder Singh Rajput 49c14b2adfSHuang, Ying pa = node->paddr + sizeof(struct setup_data) + pos; 50c14b2adfSHuang, Ying pg = pfn_to_page((pa + count - 1) >> PAGE_SHIFT); 51c14b2adfSHuang, Ying if (PageHighMem(pg)) { 52c14b2adfSHuang, Ying p = ioremap_cache(pa, count); 53c14b2adfSHuang, Ying if (!p) 54c14b2adfSHuang, Ying return -ENXIO; 55390cd85cSJaswinder Singh Rajput } else 56c14b2adfSHuang, Ying p = __va(pa); 57c14b2adfSHuang, Ying 58c14b2adfSHuang, Ying remain = copy_to_user(user_buf, p, count); 59c14b2adfSHuang, Ying 60c14b2adfSHuang, Ying if (PageHighMem(pg)) 61c14b2adfSHuang, Ying iounmap(p); 62c14b2adfSHuang, Ying 63c14b2adfSHuang, Ying if (remain) 64c14b2adfSHuang, Ying return -EFAULT; 65c14b2adfSHuang, Ying 66c14b2adfSHuang, Ying *ppos = pos + count; 67c14b2adfSHuang, Ying 68c14b2adfSHuang, Ying return count; 69c14b2adfSHuang, Ying } 70c14b2adfSHuang, Ying 71c14b2adfSHuang, Ying static const struct file_operations fops_setup_data = { 72c14b2adfSHuang, Ying .read = setup_data_read, 73234e3405SStephen Boyd .open = simple_open, 746038f373SArnd Bergmann .llseek = default_llseek, 75c14b2adfSHuang, Ying }; 76c14b2adfSHuang, Ying 77c14b2adfSHuang, Ying static int __init 78c14b2adfSHuang, Ying create_setup_data_node(struct dentry *parent, int no, 79c14b2adfSHuang, Ying struct setup_data_node *node) 80c14b2adfSHuang, Ying { 81c14b2adfSHuang, Ying struct dentry *d, *type, *data; 82c14b2adfSHuang, Ying char buf[16]; 83c14b2adfSHuang, Ying 84c14b2adfSHuang, Ying sprintf(buf, "%d", no); 85c14b2adfSHuang, Ying d = debugfs_create_dir(buf, parent); 86390cd85cSJaswinder Singh Rajput if (!d) 87390cd85cSJaswinder Singh Rajput return -ENOMEM; 88390cd85cSJaswinder Singh Rajput 89c14b2adfSHuang, Ying type = debugfs_create_x32("type", S_IRUGO, d, &node->type); 90390cd85cSJaswinder Singh Rajput if (!type) 91c14b2adfSHuang, Ying goto err_dir; 92390cd85cSJaswinder Singh Rajput 93c14b2adfSHuang, Ying data = debugfs_create_file("data", S_IRUGO, d, node, &fops_setup_data); 94390cd85cSJaswinder Singh Rajput if (!data) 95c14b2adfSHuang, Ying goto err_type; 96390cd85cSJaswinder Singh Rajput 97c14b2adfSHuang, Ying return 0; 98c14b2adfSHuang, Ying 99c14b2adfSHuang, Ying err_type: 100c14b2adfSHuang, Ying debugfs_remove(type); 101c14b2adfSHuang, Ying err_dir: 102c14b2adfSHuang, Ying debugfs_remove(d); 103390cd85cSJaswinder Singh Rajput return -ENOMEM; 104c14b2adfSHuang, Ying } 105c14b2adfSHuang, Ying 106c14b2adfSHuang, Ying static int __init create_setup_data_nodes(struct dentry *parent) 107c14b2adfSHuang, Ying { 108c14b2adfSHuang, Ying struct setup_data_node *node; 109c14b2adfSHuang, Ying struct setup_data *data; 110*41fb433bSJulia Lawall int error; 111c14b2adfSHuang, Ying struct dentry *d; 112c14b2adfSHuang, Ying struct page *pg; 113c14b2adfSHuang, Ying u64 pa_data; 114390cd85cSJaswinder Singh Rajput int no = 0; 115c14b2adfSHuang, Ying 116c14b2adfSHuang, Ying d = debugfs_create_dir("setup_data", parent); 117390cd85cSJaswinder Singh Rajput if (!d) 118390cd85cSJaswinder Singh Rajput return -ENOMEM; 119c14b2adfSHuang, Ying 120c14b2adfSHuang, Ying pa_data = boot_params.hdr.setup_data; 121c14b2adfSHuang, Ying 122c14b2adfSHuang, Ying while (pa_data) { 123c14b2adfSHuang, Ying node = kmalloc(sizeof(*node), GFP_KERNEL); 124*41fb433bSJulia Lawall if (!node) { 125*41fb433bSJulia Lawall error = -ENOMEM; 126c14b2adfSHuang, Ying goto err_dir; 127*41fb433bSJulia Lawall } 128390cd85cSJaswinder Singh Rajput 129c14b2adfSHuang, Ying pg = pfn_to_page((pa_data+sizeof(*data)-1) >> PAGE_SHIFT); 130c14b2adfSHuang, Ying if (PageHighMem(pg)) { 131c14b2adfSHuang, Ying data = ioremap_cache(pa_data, sizeof(*data)); 132c14b2adfSHuang, Ying if (!data) { 133f461a1d8SJulia Lawall kfree(node); 134c14b2adfSHuang, Ying error = -ENXIO; 135c14b2adfSHuang, Ying goto err_dir; 136c14b2adfSHuang, Ying } 137390cd85cSJaswinder Singh Rajput } else 138c14b2adfSHuang, Ying data = __va(pa_data); 139c14b2adfSHuang, Ying 140c14b2adfSHuang, Ying node->paddr = pa_data; 141c14b2adfSHuang, Ying node->type = data->type; 142c14b2adfSHuang, Ying node->len = data->len; 143c14b2adfSHuang, Ying error = create_setup_data_node(d, no, node); 144c14b2adfSHuang, Ying pa_data = data->next; 145c14b2adfSHuang, Ying 146c14b2adfSHuang, Ying if (PageHighMem(pg)) 147c14b2adfSHuang, Ying iounmap(data); 148c14b2adfSHuang, Ying if (error) 149c14b2adfSHuang, Ying goto err_dir; 150c14b2adfSHuang, Ying no++; 151c14b2adfSHuang, Ying } 152390cd85cSJaswinder Singh Rajput 153c14b2adfSHuang, Ying return 0; 154c14b2adfSHuang, Ying 155c14b2adfSHuang, Ying err_dir: 156c14b2adfSHuang, Ying debugfs_remove(d); 157c14b2adfSHuang, Ying return error; 158c14b2adfSHuang, Ying } 159c14b2adfSHuang, Ying 1606d7d7433SHuang, Ying static struct debugfs_blob_wrapper boot_params_blob = { 1616d7d7433SHuang, Ying .data = &boot_params, 1626d7d7433SHuang, Ying .size = sizeof(boot_params), 1636d7d7433SHuang, Ying }; 1646d7d7433SHuang, Ying 1656d7d7433SHuang, Ying static int __init boot_params_kdebugfs_init(void) 1666d7d7433SHuang, Ying { 1676d7d7433SHuang, Ying struct dentry *dbp, *version, *data; 168390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 1696d7d7433SHuang, Ying 1706d7d7433SHuang, Ying dbp = debugfs_create_dir("boot_params", NULL); 171390cd85cSJaswinder Singh Rajput if (!dbp) 172390cd85cSJaswinder Singh Rajput return -ENOMEM; 173390cd85cSJaswinder Singh Rajput 1746d7d7433SHuang, Ying version = debugfs_create_x16("version", S_IRUGO, dbp, 1756d7d7433SHuang, Ying &boot_params.hdr.version); 176390cd85cSJaswinder Singh Rajput if (!version) 1776d7d7433SHuang, Ying goto err_dir; 178390cd85cSJaswinder Singh Rajput 1796d7d7433SHuang, Ying data = debugfs_create_blob("data", S_IRUGO, dbp, 1806d7d7433SHuang, Ying &boot_params_blob); 181390cd85cSJaswinder Singh Rajput if (!data) 1826d7d7433SHuang, Ying goto err_version; 183390cd85cSJaswinder Singh Rajput 184c14b2adfSHuang, Ying error = create_setup_data_nodes(dbp); 185c14b2adfSHuang, Ying if (error) 186c14b2adfSHuang, Ying goto err_data; 187390cd85cSJaswinder Singh Rajput 1886d7d7433SHuang, Ying return 0; 189c14b2adfSHuang, Ying 190c14b2adfSHuang, Ying err_data: 191c14b2adfSHuang, Ying debugfs_remove(data); 1926d7d7433SHuang, Ying err_version: 1936d7d7433SHuang, Ying debugfs_remove(version); 1946d7d7433SHuang, Ying err_dir: 1956d7d7433SHuang, Ying debugfs_remove(dbp); 1966d7d7433SHuang, Ying return error; 1976d7d7433SHuang, Ying } 198390cd85cSJaswinder Singh Rajput #endif /* CONFIG_DEBUG_BOOT_PARAMS */ 1996d7d7433SHuang, Ying 2006d7d7433SHuang, Ying static int __init arch_kdebugfs_init(void) 2016d7d7433SHuang, Ying { 2026d7d7433SHuang, Ying int error = 0; 2036d7d7433SHuang, Ying 204ae79cdaaSvenkatesh.pallipadi@intel.com arch_debugfs_dir = debugfs_create_dir("x86", NULL); 205ae79cdaaSvenkatesh.pallipadi@intel.com if (!arch_debugfs_dir) 206ae79cdaaSvenkatesh.pallipadi@intel.com return -ENOMEM; 207ae79cdaaSvenkatesh.pallipadi@intel.com 2086d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 2096d7d7433SHuang, Ying error = boot_params_kdebugfs_init(); 2106d7d7433SHuang, Ying #endif 2116d7d7433SHuang, Ying 2126d7d7433SHuang, Ying return error; 2136d7d7433SHuang, Ying } 2146d7d7433SHuang, Ying arch_initcall(arch_kdebugfs_init); 215