showing state of memmap by /dev/memstate.
-- Kame
/dev/memstate shows status of memmap.
A user can read state of a page as a byte.
This feature is useful for Memory-hotplug and other stuffs,
where we have to investigate for what a page is.
Signed-off-by: KAMEZAWA Hiroyuki <[email protected]>
---
linux-2.6.12-rc2-kamezawa/drivers/char/mem.c | 63 +++++++++++++++++++++++++++
1 files changed, 63 insertions(+)
diff -puN drivers/char/mem.c~show_memstate drivers/char/mem.c
--- linux-2.6.12-rc2/drivers/char/mem.c~show_memstate 2005-04-20 10:39:40.000000000 +0900
+++ linux-2.6.12-rc2-kamezawa/drivers/char/mem.c 2005-04-20 16:51:45.000000000 +0900
@@ -715,6 +715,59 @@ static int open_port(struct inode * inod
return capable(CAP_SYS_RAWIO) ? 0 : -EPERM;
}
+static inline unsigned char get_page_type(struct page *page)
+{
+ if ( !PageReserved(page) )
+ return Page_Common;
+ return reserved_page_type(page);
+}
+
+static ssize_t read_memstate(struct file *file, char __user *buf,
+ size_t count, loff_t *ppos)
+{
+ unsigned long pfn = *ppos;
+ unsigned long left, written;
+ ssize_t ret;
+ int len, i;
+ struct page *page;
+ char *kbuf;
+
+ if (!count)
+ return 0;
+
+ if (!access_ok(VERIFY_WRITE, buf, count))
+ return -EFAULT;
+
+ left = count;
+ written = 0;
+ kbuf = (char *)__get_free_page(GFP_KERNEL);
+
+ if (!kbuf)
+ return -ENOMEM;
+ ret = -EFAULT;
+ /* copy data */
+ while (left) {
+ len = (left < PAGE_SIZE) ? left : PAGE_SIZE;
+ for (i = 0; i < len; i++, pfn++) {
+ if ( !pfn_valid(pfn) ) {
+ kbuf[i] = Page_Invalid;
+ continue;
+ }
+ page = pfn_to_page(pfn);
+ kbuf[i] = get_page_type(page);
+ }
+ if (copy_to_user(buf, kbuf, len))
+ goto err_out;
+ written += len;
+ left -= len;
+ }
+ *ppos = pfn;
+ ret = written;
+ err_out:
+ free_page((unsigned long)kbuf);
+ return ret;
+}
+
#define zero_lseek null_lseek
#define full_lseek null_lseek
#define write_zero write_null
@@ -770,6 +823,11 @@ static struct file_operations full_fops
.write = write_full,
};
+static struct file_operations memstate_fops = {
+ .llseek = memory_lseek,
+ .read = read_memstate,
+};
+
static ssize_t kmsg_write(struct file * file, const char __user * buf,
size_t count, loff_t *ppos)
{
@@ -825,6 +883,9 @@ static int memory_open(struct inode * in
case 11:
filp->f_op = &kmsg_fops;
break;
+ case 12:
+ filp->f_op = &memstate_fops;
+ break;
default:
return -ENXIO;
}
@@ -854,6 +915,7 @@ static const struct {
{8, "random", S_IRUGO | S_IWUSR, &random_fops},
{9, "urandom", S_IRUGO | S_IWUSR, &urandom_fops},
{11,"kmsg", S_IRUGO | S_IWUSR, &kmsg_fops},
+ {12,"memstate", S_IRUSR | S_IRGRP, &memstate_fops},
};
static struct class_simple *mem_class;
@@ -870,6 +932,7 @@ static int __init chr_dev_init(void)
class_simple_device_add(mem_class,
MKDEV(MEM_MAJOR, devlist[i].minor),
NULL, devlist[i].name);
+ printk("creating mem device %d %d\n",i,devlist[i].minor);
devfs_mk_cdev(MKDEV(MEM_MAJOR, devlist[i].minor),
S_IFCHR | devlist[i].mode, devlist[i].name);
}
_
[Index of Archives]
[Kernel Newbies]
[Netfilter]
[Bugtraq]
[Photo]
[Stuff]
[Gimp]
[Yosemite News]
[MIPS Linux]
[ARM Linux]
[Linux Security]
[Linux RAID]
[Video 4 Linux]
[Linux for the blind]
[Linux Resources]