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> 12*5a0e3ad6STejun 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, 81c14b2adfSHuang, Ying }; 82c14b2adfSHuang, Ying 83c14b2adfSHuang, Ying static int __init 84c14b2adfSHuang, Ying create_setup_data_node(struct dentry *parent, int no, 85c14b2adfSHuang, Ying struct setup_data_node *node) 86c14b2adfSHuang, Ying { 87c14b2adfSHuang, Ying struct dentry *d, *type, *data; 88c14b2adfSHuang, Ying char buf[16]; 89c14b2adfSHuang, Ying 90c14b2adfSHuang, Ying sprintf(buf, "%d", no); 91c14b2adfSHuang, Ying d = debugfs_create_dir(buf, parent); 92390cd85cSJaswinder Singh Rajput if (!d) 93390cd85cSJaswinder Singh Rajput return -ENOMEM; 94390cd85cSJaswinder Singh Rajput 95c14b2adfSHuang, Ying type = debugfs_create_x32("type", S_IRUGO, d, &node->type); 96390cd85cSJaswinder Singh Rajput if (!type) 97c14b2adfSHuang, Ying goto err_dir; 98390cd85cSJaswinder Singh Rajput 99c14b2adfSHuang, Ying data = debugfs_create_file("data", S_IRUGO, d, node, &fops_setup_data); 100390cd85cSJaswinder Singh Rajput if (!data) 101c14b2adfSHuang, Ying goto err_type; 102390cd85cSJaswinder Singh Rajput 103c14b2adfSHuang, Ying return 0; 104c14b2adfSHuang, Ying 105c14b2adfSHuang, Ying err_type: 106c14b2adfSHuang, Ying debugfs_remove(type); 107c14b2adfSHuang, Ying err_dir: 108c14b2adfSHuang, Ying debugfs_remove(d); 109390cd85cSJaswinder Singh Rajput return -ENOMEM; 110c14b2adfSHuang, Ying } 111c14b2adfSHuang, Ying 112c14b2adfSHuang, Ying static int __init create_setup_data_nodes(struct dentry *parent) 113c14b2adfSHuang, Ying { 114c14b2adfSHuang, Ying struct setup_data_node *node; 115c14b2adfSHuang, Ying struct setup_data *data; 116390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 117c14b2adfSHuang, Ying struct dentry *d; 118c14b2adfSHuang, Ying struct page *pg; 119c14b2adfSHuang, Ying u64 pa_data; 120390cd85cSJaswinder Singh Rajput int no = 0; 121c14b2adfSHuang, Ying 122c14b2adfSHuang, Ying d = debugfs_create_dir("setup_data", parent); 123390cd85cSJaswinder Singh Rajput if (!d) 124390cd85cSJaswinder Singh Rajput return -ENOMEM; 125c14b2adfSHuang, Ying 126c14b2adfSHuang, Ying pa_data = boot_params.hdr.setup_data; 127c14b2adfSHuang, Ying 128c14b2adfSHuang, Ying while (pa_data) { 129c14b2adfSHuang, Ying node = kmalloc(sizeof(*node), GFP_KERNEL); 130390cd85cSJaswinder Singh Rajput if (!node) 131c14b2adfSHuang, Ying goto err_dir; 132390cd85cSJaswinder Singh Rajput 133c14b2adfSHuang, Ying pg = pfn_to_page((pa_data+sizeof(*data)-1) >> PAGE_SHIFT); 134c14b2adfSHuang, Ying if (PageHighMem(pg)) { 135c14b2adfSHuang, Ying data = ioremap_cache(pa_data, sizeof(*data)); 136c14b2adfSHuang, Ying if (!data) { 137f461a1d8SJulia Lawall kfree(node); 138c14b2adfSHuang, Ying error = -ENXIO; 139c14b2adfSHuang, Ying goto err_dir; 140c14b2adfSHuang, Ying } 141390cd85cSJaswinder Singh Rajput } else 142c14b2adfSHuang, Ying data = __va(pa_data); 143c14b2adfSHuang, Ying 144c14b2adfSHuang, Ying node->paddr = pa_data; 145c14b2adfSHuang, Ying node->type = data->type; 146c14b2adfSHuang, Ying node->len = data->len; 147c14b2adfSHuang, Ying error = create_setup_data_node(d, no, node); 148c14b2adfSHuang, Ying pa_data = data->next; 149c14b2adfSHuang, Ying 150c14b2adfSHuang, Ying if (PageHighMem(pg)) 151c14b2adfSHuang, Ying iounmap(data); 152c14b2adfSHuang, Ying if (error) 153c14b2adfSHuang, Ying goto err_dir; 154c14b2adfSHuang, Ying no++; 155c14b2adfSHuang, Ying } 156390cd85cSJaswinder Singh Rajput 157c14b2adfSHuang, Ying return 0; 158c14b2adfSHuang, Ying 159c14b2adfSHuang, Ying err_dir: 160c14b2adfSHuang, Ying debugfs_remove(d); 161c14b2adfSHuang, Ying return error; 162c14b2adfSHuang, Ying } 163c14b2adfSHuang, Ying 1646d7d7433SHuang, Ying static struct debugfs_blob_wrapper boot_params_blob = { 1656d7d7433SHuang, Ying .data = &boot_params, 1666d7d7433SHuang, Ying .size = sizeof(boot_params), 1676d7d7433SHuang, Ying }; 1686d7d7433SHuang, Ying 1696d7d7433SHuang, Ying static int __init boot_params_kdebugfs_init(void) 1706d7d7433SHuang, Ying { 1716d7d7433SHuang, Ying struct dentry *dbp, *version, *data; 172390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 1736d7d7433SHuang, Ying 1746d7d7433SHuang, Ying dbp = debugfs_create_dir("boot_params", NULL); 175390cd85cSJaswinder Singh Rajput if (!dbp) 176390cd85cSJaswinder Singh Rajput return -ENOMEM; 177390cd85cSJaswinder Singh Rajput 1786d7d7433SHuang, Ying version = debugfs_create_x16("version", S_IRUGO, dbp, 1796d7d7433SHuang, Ying &boot_params.hdr.version); 180390cd85cSJaswinder Singh Rajput if (!version) 1816d7d7433SHuang, Ying goto err_dir; 182390cd85cSJaswinder Singh Rajput 1836d7d7433SHuang, Ying data = debugfs_create_blob("data", S_IRUGO, dbp, 1846d7d7433SHuang, Ying &boot_params_blob); 185390cd85cSJaswinder Singh Rajput if (!data) 1866d7d7433SHuang, Ying goto err_version; 187390cd85cSJaswinder Singh Rajput 188c14b2adfSHuang, Ying error = create_setup_data_nodes(dbp); 189c14b2adfSHuang, Ying if (error) 190c14b2adfSHuang, Ying goto err_data; 191390cd85cSJaswinder Singh Rajput 1926d7d7433SHuang, Ying return 0; 193c14b2adfSHuang, Ying 194c14b2adfSHuang, Ying err_data: 195c14b2adfSHuang, Ying debugfs_remove(data); 1966d7d7433SHuang, Ying err_version: 1976d7d7433SHuang, Ying debugfs_remove(version); 1986d7d7433SHuang, Ying err_dir: 1996d7d7433SHuang, Ying debugfs_remove(dbp); 2006d7d7433SHuang, Ying return error; 2016d7d7433SHuang, Ying } 202390cd85cSJaswinder Singh Rajput #endif /* CONFIG_DEBUG_BOOT_PARAMS */ 2036d7d7433SHuang, Ying 2046d7d7433SHuang, Ying static int __init arch_kdebugfs_init(void) 2056d7d7433SHuang, Ying { 2066d7d7433SHuang, Ying int error = 0; 2076d7d7433SHuang, Ying 208ae79cdaaSvenkatesh.pallipadi@intel.com arch_debugfs_dir = debugfs_create_dir("x86", NULL); 209ae79cdaaSvenkatesh.pallipadi@intel.com if (!arch_debugfs_dir) 210ae79cdaaSvenkatesh.pallipadi@intel.com return -ENOMEM; 211ae79cdaaSvenkatesh.pallipadi@intel.com 2126d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 2136d7d7433SHuang, Ying error = boot_params_kdebugfs_init(); 2146d7d7433SHuang, Ying #endif 2156d7d7433SHuang, Ying 2166d7d7433SHuang, Ying return error; 2176d7d7433SHuang, Ying } 2186d7d7433SHuang, Ying arch_initcall(arch_kdebugfs_init); 219