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 int setup_data_open(struct inode *inode, struct file *file) 72c14b2adfSHuang, Ying { 73c14b2adfSHuang, Ying file->private_data = inode->i_private; 74390cd85cSJaswinder Singh Rajput 75c14b2adfSHuang, Ying return 0; 76c14b2adfSHuang, Ying } 77c14b2adfSHuang, Ying 78c14b2adfSHuang, Ying static const struct file_operations fops_setup_data = { 79c14b2adfSHuang, Ying .read = setup_data_read, 80c14b2adfSHuang, Ying .open = setup_data_open, 81*6038f373SArnd Bergmann .llseek = default_llseek, 82c14b2adfSHuang, Ying }; 83c14b2adfSHuang, Ying 84c14b2adfSHuang, Ying static int __init 85c14b2adfSHuang, Ying create_setup_data_node(struct dentry *parent, int no, 86c14b2adfSHuang, Ying struct setup_data_node *node) 87c14b2adfSHuang, Ying { 88c14b2adfSHuang, Ying struct dentry *d, *type, *data; 89c14b2adfSHuang, Ying char buf[16]; 90c14b2adfSHuang, Ying 91c14b2adfSHuang, Ying sprintf(buf, "%d", no); 92c14b2adfSHuang, Ying d = debugfs_create_dir(buf, parent); 93390cd85cSJaswinder Singh Rajput if (!d) 94390cd85cSJaswinder Singh Rajput return -ENOMEM; 95390cd85cSJaswinder Singh Rajput 96c14b2adfSHuang, Ying type = debugfs_create_x32("type", S_IRUGO, d, &node->type); 97390cd85cSJaswinder Singh Rajput if (!type) 98c14b2adfSHuang, Ying goto err_dir; 99390cd85cSJaswinder Singh Rajput 100c14b2adfSHuang, Ying data = debugfs_create_file("data", S_IRUGO, d, node, &fops_setup_data); 101390cd85cSJaswinder Singh Rajput if (!data) 102c14b2adfSHuang, Ying goto err_type; 103390cd85cSJaswinder Singh Rajput 104c14b2adfSHuang, Ying return 0; 105c14b2adfSHuang, Ying 106c14b2adfSHuang, Ying err_type: 107c14b2adfSHuang, Ying debugfs_remove(type); 108c14b2adfSHuang, Ying err_dir: 109c14b2adfSHuang, Ying debugfs_remove(d); 110390cd85cSJaswinder Singh Rajput return -ENOMEM; 111c14b2adfSHuang, Ying } 112c14b2adfSHuang, Ying 113c14b2adfSHuang, Ying static int __init create_setup_data_nodes(struct dentry *parent) 114c14b2adfSHuang, Ying { 115c14b2adfSHuang, Ying struct setup_data_node *node; 116c14b2adfSHuang, Ying struct setup_data *data; 117390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 118c14b2adfSHuang, Ying struct dentry *d; 119c14b2adfSHuang, Ying struct page *pg; 120c14b2adfSHuang, Ying u64 pa_data; 121390cd85cSJaswinder Singh Rajput int no = 0; 122c14b2adfSHuang, Ying 123c14b2adfSHuang, Ying d = debugfs_create_dir("setup_data", parent); 124390cd85cSJaswinder Singh Rajput if (!d) 125390cd85cSJaswinder Singh Rajput return -ENOMEM; 126c14b2adfSHuang, Ying 127c14b2adfSHuang, Ying pa_data = boot_params.hdr.setup_data; 128c14b2adfSHuang, Ying 129c14b2adfSHuang, Ying while (pa_data) { 130c14b2adfSHuang, Ying node = kmalloc(sizeof(*node), GFP_KERNEL); 131390cd85cSJaswinder Singh Rajput if (!node) 132c14b2adfSHuang, Ying goto err_dir; 133390cd85cSJaswinder Singh Rajput 134c14b2adfSHuang, Ying pg = pfn_to_page((pa_data+sizeof(*data)-1) >> PAGE_SHIFT); 135c14b2adfSHuang, Ying if (PageHighMem(pg)) { 136c14b2adfSHuang, Ying data = ioremap_cache(pa_data, sizeof(*data)); 137c14b2adfSHuang, Ying if (!data) { 138f461a1d8SJulia Lawall kfree(node); 139c14b2adfSHuang, Ying error = -ENXIO; 140c14b2adfSHuang, Ying goto err_dir; 141c14b2adfSHuang, Ying } 142390cd85cSJaswinder Singh Rajput } else 143c14b2adfSHuang, Ying data = __va(pa_data); 144c14b2adfSHuang, Ying 145c14b2adfSHuang, Ying node->paddr = pa_data; 146c14b2adfSHuang, Ying node->type = data->type; 147c14b2adfSHuang, Ying node->len = data->len; 148c14b2adfSHuang, Ying error = create_setup_data_node(d, no, node); 149c14b2adfSHuang, Ying pa_data = data->next; 150c14b2adfSHuang, Ying 151c14b2adfSHuang, Ying if (PageHighMem(pg)) 152c14b2adfSHuang, Ying iounmap(data); 153c14b2adfSHuang, Ying if (error) 154c14b2adfSHuang, Ying goto err_dir; 155c14b2adfSHuang, Ying no++; 156c14b2adfSHuang, Ying } 157390cd85cSJaswinder Singh Rajput 158c14b2adfSHuang, Ying return 0; 159c14b2adfSHuang, Ying 160c14b2adfSHuang, Ying err_dir: 161c14b2adfSHuang, Ying debugfs_remove(d); 162c14b2adfSHuang, Ying return error; 163c14b2adfSHuang, Ying } 164c14b2adfSHuang, Ying 1656d7d7433SHuang, Ying static struct debugfs_blob_wrapper boot_params_blob = { 1666d7d7433SHuang, Ying .data = &boot_params, 1676d7d7433SHuang, Ying .size = sizeof(boot_params), 1686d7d7433SHuang, Ying }; 1696d7d7433SHuang, Ying 1706d7d7433SHuang, Ying static int __init boot_params_kdebugfs_init(void) 1716d7d7433SHuang, Ying { 1726d7d7433SHuang, Ying struct dentry *dbp, *version, *data; 173390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 1746d7d7433SHuang, Ying 1756d7d7433SHuang, Ying dbp = debugfs_create_dir("boot_params", NULL); 176390cd85cSJaswinder Singh Rajput if (!dbp) 177390cd85cSJaswinder Singh Rajput return -ENOMEM; 178390cd85cSJaswinder Singh Rajput 1796d7d7433SHuang, Ying version = debugfs_create_x16("version", S_IRUGO, dbp, 1806d7d7433SHuang, Ying &boot_params.hdr.version); 181390cd85cSJaswinder Singh Rajput if (!version) 1826d7d7433SHuang, Ying goto err_dir; 183390cd85cSJaswinder Singh Rajput 1846d7d7433SHuang, Ying data = debugfs_create_blob("data", S_IRUGO, dbp, 1856d7d7433SHuang, Ying &boot_params_blob); 186390cd85cSJaswinder Singh Rajput if (!data) 1876d7d7433SHuang, Ying goto err_version; 188390cd85cSJaswinder Singh Rajput 189c14b2adfSHuang, Ying error = create_setup_data_nodes(dbp); 190c14b2adfSHuang, Ying if (error) 191c14b2adfSHuang, Ying goto err_data; 192390cd85cSJaswinder Singh Rajput 1936d7d7433SHuang, Ying return 0; 194c14b2adfSHuang, Ying 195c14b2adfSHuang, Ying err_data: 196c14b2adfSHuang, Ying debugfs_remove(data); 1976d7d7433SHuang, Ying err_version: 1986d7d7433SHuang, Ying debugfs_remove(version); 1996d7d7433SHuang, Ying err_dir: 2006d7d7433SHuang, Ying debugfs_remove(dbp); 2016d7d7433SHuang, Ying return error; 2026d7d7433SHuang, Ying } 203390cd85cSJaswinder Singh Rajput #endif /* CONFIG_DEBUG_BOOT_PARAMS */ 2046d7d7433SHuang, Ying 2056d7d7433SHuang, Ying static int __init arch_kdebugfs_init(void) 2066d7d7433SHuang, Ying { 2076d7d7433SHuang, Ying int error = 0; 2086d7d7433SHuang, Ying 209ae79cdaaSvenkatesh.pallipadi@intel.com arch_debugfs_dir = debugfs_create_dir("x86", NULL); 210ae79cdaaSvenkatesh.pallipadi@intel.com if (!arch_debugfs_dir) 211ae79cdaaSvenkatesh.pallipadi@intel.com return -ENOMEM; 212ae79cdaaSvenkatesh.pallipadi@intel.com 2136d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 2146d7d7433SHuang, Ying error = boot_params_kdebugfs_init(); 2156d7d7433SHuang, Ying #endif 2166d7d7433SHuang, Ying 2176d7d7433SHuang, Ying return error; 2186d7d7433SHuang, Ying } 2196d7d7433SHuang, Ying arch_initcall(arch_kdebugfs_init); 220