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> 11*390cd85cSJaswinder Singh Rajput #include <linux/module.h> 126d7d7433SHuang, Ying #include <linux/init.h> 13*390cd85cSJaswinder Singh Rajput #include <linux/stat.h> 14c14b2adfSHuang, Ying #include <linux/io.h> 15c14b2adfSHuang, Ying #include <linux/mm.h> 166d7d7433SHuang, Ying 176d7d7433SHuang, Ying #include <asm/setup.h> 186d7d7433SHuang, Ying 19ae79cdaaSvenkatesh.pallipadi@intel.com struct dentry *arch_debugfs_dir; 20ae79cdaaSvenkatesh.pallipadi@intel.com EXPORT_SYMBOL(arch_debugfs_dir); 21ae79cdaaSvenkatesh.pallipadi@intel.com 226d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 23c14b2adfSHuang, Ying struct setup_data_node { 24c14b2adfSHuang, Ying u64 paddr; 25c14b2adfSHuang, Ying u32 type; 26c14b2adfSHuang, Ying u32 len; 27c14b2adfSHuang, Ying }; 28c14b2adfSHuang, Ying 29*390cd85cSJaswinder Singh Rajput static ssize_t setup_data_read(struct file *file, char __user *user_buf, 30*390cd85cSJaswinder Singh Rajput size_t count, loff_t *ppos) 31c14b2adfSHuang, Ying { 32c14b2adfSHuang, Ying struct setup_data_node *node = file->private_data; 33c14b2adfSHuang, Ying unsigned long remain; 34c14b2adfSHuang, Ying loff_t pos = *ppos; 35c14b2adfSHuang, Ying struct page *pg; 36c14b2adfSHuang, Ying void *p; 37c14b2adfSHuang, Ying u64 pa; 38c14b2adfSHuang, Ying 39c14b2adfSHuang, Ying if (pos < 0) 40c14b2adfSHuang, Ying return -EINVAL; 41*390cd85cSJaswinder Singh Rajput 42c14b2adfSHuang, Ying if (pos >= node->len) 43c14b2adfSHuang, Ying return 0; 44c14b2adfSHuang, Ying 45c14b2adfSHuang, Ying if (count > node->len - pos) 46c14b2adfSHuang, Ying count = node->len - pos; 47*390cd85cSJaswinder Singh Rajput 48c14b2adfSHuang, Ying pa = node->paddr + sizeof(struct setup_data) + pos; 49c14b2adfSHuang, Ying pg = pfn_to_page((pa + count - 1) >> PAGE_SHIFT); 50c14b2adfSHuang, Ying if (PageHighMem(pg)) { 51c14b2adfSHuang, Ying p = ioremap_cache(pa, count); 52c14b2adfSHuang, Ying if (!p) 53c14b2adfSHuang, Ying return -ENXIO; 54*390cd85cSJaswinder Singh Rajput } else 55c14b2adfSHuang, Ying p = __va(pa); 56c14b2adfSHuang, Ying 57c14b2adfSHuang, Ying remain = copy_to_user(user_buf, p, count); 58c14b2adfSHuang, Ying 59c14b2adfSHuang, Ying if (PageHighMem(pg)) 60c14b2adfSHuang, Ying iounmap(p); 61c14b2adfSHuang, Ying 62c14b2adfSHuang, Ying if (remain) 63c14b2adfSHuang, Ying return -EFAULT; 64c14b2adfSHuang, Ying 65c14b2adfSHuang, Ying *ppos = pos + count; 66c14b2adfSHuang, Ying 67c14b2adfSHuang, Ying return count; 68c14b2adfSHuang, Ying } 69c14b2adfSHuang, Ying 70c14b2adfSHuang, Ying static int setup_data_open(struct inode *inode, struct file *file) 71c14b2adfSHuang, Ying { 72c14b2adfSHuang, Ying file->private_data = inode->i_private; 73*390cd85cSJaswinder Singh Rajput 74c14b2adfSHuang, Ying return 0; 75c14b2adfSHuang, Ying } 76c14b2adfSHuang, Ying 77c14b2adfSHuang, Ying static const struct file_operations fops_setup_data = { 78c14b2adfSHuang, Ying .read = setup_data_read, 79c14b2adfSHuang, Ying .open = setup_data_open, 80c14b2adfSHuang, Ying }; 81c14b2adfSHuang, Ying 82c14b2adfSHuang, Ying static int __init 83c14b2adfSHuang, Ying create_setup_data_node(struct dentry *parent, int no, 84c14b2adfSHuang, Ying struct setup_data_node *node) 85c14b2adfSHuang, Ying { 86c14b2adfSHuang, Ying struct dentry *d, *type, *data; 87c14b2adfSHuang, Ying char buf[16]; 88c14b2adfSHuang, Ying 89c14b2adfSHuang, Ying sprintf(buf, "%d", no); 90c14b2adfSHuang, Ying d = debugfs_create_dir(buf, parent); 91*390cd85cSJaswinder Singh Rajput if (!d) 92*390cd85cSJaswinder Singh Rajput return -ENOMEM; 93*390cd85cSJaswinder Singh Rajput 94c14b2adfSHuang, Ying type = debugfs_create_x32("type", S_IRUGO, d, &node->type); 95*390cd85cSJaswinder Singh Rajput if (!type) 96c14b2adfSHuang, Ying goto err_dir; 97*390cd85cSJaswinder Singh Rajput 98c14b2adfSHuang, Ying data = debugfs_create_file("data", S_IRUGO, d, node, &fops_setup_data); 99*390cd85cSJaswinder Singh Rajput if (!data) 100c14b2adfSHuang, Ying goto err_type; 101*390cd85cSJaswinder Singh Rajput 102c14b2adfSHuang, Ying return 0; 103c14b2adfSHuang, Ying 104c14b2adfSHuang, Ying err_type: 105c14b2adfSHuang, Ying debugfs_remove(type); 106c14b2adfSHuang, Ying err_dir: 107c14b2adfSHuang, Ying debugfs_remove(d); 108*390cd85cSJaswinder Singh Rajput return -ENOMEM; 109c14b2adfSHuang, Ying } 110c14b2adfSHuang, Ying 111c14b2adfSHuang, Ying static int __init create_setup_data_nodes(struct dentry *parent) 112c14b2adfSHuang, Ying { 113c14b2adfSHuang, Ying struct setup_data_node *node; 114c14b2adfSHuang, Ying struct setup_data *data; 115*390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 116c14b2adfSHuang, Ying struct dentry *d; 117c14b2adfSHuang, Ying struct page *pg; 118c14b2adfSHuang, Ying u64 pa_data; 119*390cd85cSJaswinder Singh Rajput int no = 0; 120c14b2adfSHuang, Ying 121c14b2adfSHuang, Ying d = debugfs_create_dir("setup_data", parent); 122*390cd85cSJaswinder Singh Rajput if (!d) 123*390cd85cSJaswinder Singh Rajput return -ENOMEM; 124c14b2adfSHuang, Ying 125c14b2adfSHuang, Ying pa_data = boot_params.hdr.setup_data; 126c14b2adfSHuang, Ying 127c14b2adfSHuang, Ying while (pa_data) { 128c14b2adfSHuang, Ying node = kmalloc(sizeof(*node), GFP_KERNEL); 129*390cd85cSJaswinder Singh Rajput if (!node) 130c14b2adfSHuang, Ying goto err_dir; 131*390cd85cSJaswinder Singh Rajput 132c14b2adfSHuang, Ying pg = pfn_to_page((pa_data+sizeof(*data)-1) >> PAGE_SHIFT); 133c14b2adfSHuang, Ying if (PageHighMem(pg)) { 134c14b2adfSHuang, Ying data = ioremap_cache(pa_data, sizeof(*data)); 135c14b2adfSHuang, Ying if (!data) { 136f461a1d8SJulia Lawall kfree(node); 137c14b2adfSHuang, Ying error = -ENXIO; 138c14b2adfSHuang, Ying goto err_dir; 139c14b2adfSHuang, Ying } 140*390cd85cSJaswinder Singh Rajput } else 141c14b2adfSHuang, Ying data = __va(pa_data); 142c14b2adfSHuang, Ying 143c14b2adfSHuang, Ying node->paddr = pa_data; 144c14b2adfSHuang, Ying node->type = data->type; 145c14b2adfSHuang, Ying node->len = data->len; 146c14b2adfSHuang, Ying error = create_setup_data_node(d, no, node); 147c14b2adfSHuang, Ying pa_data = data->next; 148c14b2adfSHuang, Ying 149c14b2adfSHuang, Ying if (PageHighMem(pg)) 150c14b2adfSHuang, Ying iounmap(data); 151c14b2adfSHuang, Ying if (error) 152c14b2adfSHuang, Ying goto err_dir; 153c14b2adfSHuang, Ying no++; 154c14b2adfSHuang, Ying } 155*390cd85cSJaswinder Singh Rajput 156c14b2adfSHuang, Ying return 0; 157c14b2adfSHuang, Ying 158c14b2adfSHuang, Ying err_dir: 159c14b2adfSHuang, Ying debugfs_remove(d); 160c14b2adfSHuang, Ying return error; 161c14b2adfSHuang, Ying } 162c14b2adfSHuang, Ying 1636d7d7433SHuang, Ying static struct debugfs_blob_wrapper boot_params_blob = { 1646d7d7433SHuang, Ying .data = &boot_params, 1656d7d7433SHuang, Ying .size = sizeof(boot_params), 1666d7d7433SHuang, Ying }; 1676d7d7433SHuang, Ying 1686d7d7433SHuang, Ying static int __init boot_params_kdebugfs_init(void) 1696d7d7433SHuang, Ying { 1706d7d7433SHuang, Ying struct dentry *dbp, *version, *data; 171*390cd85cSJaswinder Singh Rajput int error = -ENOMEM; 1726d7d7433SHuang, Ying 1736d7d7433SHuang, Ying dbp = debugfs_create_dir("boot_params", NULL); 174*390cd85cSJaswinder Singh Rajput if (!dbp) 175*390cd85cSJaswinder Singh Rajput return -ENOMEM; 176*390cd85cSJaswinder Singh Rajput 1776d7d7433SHuang, Ying version = debugfs_create_x16("version", S_IRUGO, dbp, 1786d7d7433SHuang, Ying &boot_params.hdr.version); 179*390cd85cSJaswinder Singh Rajput if (!version) 1806d7d7433SHuang, Ying goto err_dir; 181*390cd85cSJaswinder Singh Rajput 1826d7d7433SHuang, Ying data = debugfs_create_blob("data", S_IRUGO, dbp, 1836d7d7433SHuang, Ying &boot_params_blob); 184*390cd85cSJaswinder Singh Rajput if (!data) 1856d7d7433SHuang, Ying goto err_version; 186*390cd85cSJaswinder Singh Rajput 187c14b2adfSHuang, Ying error = create_setup_data_nodes(dbp); 188c14b2adfSHuang, Ying if (error) 189c14b2adfSHuang, Ying goto err_data; 190*390cd85cSJaswinder Singh Rajput 1916d7d7433SHuang, Ying return 0; 192c14b2adfSHuang, Ying 193c14b2adfSHuang, Ying err_data: 194c14b2adfSHuang, Ying debugfs_remove(data); 1956d7d7433SHuang, Ying err_version: 1966d7d7433SHuang, Ying debugfs_remove(version); 1976d7d7433SHuang, Ying err_dir: 1986d7d7433SHuang, Ying debugfs_remove(dbp); 1996d7d7433SHuang, Ying return error; 2006d7d7433SHuang, Ying } 201*390cd85cSJaswinder Singh Rajput #endif /* CONFIG_DEBUG_BOOT_PARAMS */ 2026d7d7433SHuang, Ying 2036d7d7433SHuang, Ying static int __init arch_kdebugfs_init(void) 2046d7d7433SHuang, Ying { 2056d7d7433SHuang, Ying int error = 0; 2066d7d7433SHuang, Ying 207ae79cdaaSvenkatesh.pallipadi@intel.com arch_debugfs_dir = debugfs_create_dir("x86", NULL); 208ae79cdaaSvenkatesh.pallipadi@intel.com if (!arch_debugfs_dir) 209ae79cdaaSvenkatesh.pallipadi@intel.com return -ENOMEM; 210ae79cdaaSvenkatesh.pallipadi@intel.com 2116d7d7433SHuang, Ying #ifdef CONFIG_DEBUG_BOOT_PARAMS 2126d7d7433SHuang, Ying error = boot_params_kdebugfs_init(); 2136d7d7433SHuang, Ying #endif 2146d7d7433SHuang, Ying 2156d7d7433SHuang, Ying return error; 2166d7d7433SHuang, Ying } 2176d7d7433SHuang, Ying arch_initcall(arch_kdebugfs_init); 218