forked from hexiaolong2008/sample-code
-
Notifications
You must be signed in to change notification settings - Fork 0
/
mmap2.c
61 lines (50 loc) · 1.33 KB
/
mmap2.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
/*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*/
#include <linux/module.h>
#include <linux/miscdevice.h>
#include <linux/mm.h>
#include <linux/uaccess.h>
#include <linux/fs.h>
#include <linux/slab.h>
static void *kaddr;
static int my_fault(struct vm_fault *vmf)
{
struct vm_area_struct *vma = vmf->vma;
int offset, ret;
offset = vmf->pgoff * PAGE_SIZE;
ret = vm_insert_page(vma, vmf->address, virt_to_page(kaddr + offset));
if (ret)
return VM_FAULT_SIGBUS;
return VM_FAULT_NOPAGE;
}
static const struct vm_operations_struct vm_ops = {
.fault = my_fault,
};
static int my_mmap(struct file *file, struct vm_area_struct *vma)
{
vma->vm_flags |= VM_MIXEDMAP;
vma->vm_ops = &vm_ops;
return 0;
}
static struct file_operations my_fops = {
.owner = THIS_MODULE,
.mmap = my_mmap,
};
static struct miscdevice mdev = {
.minor = MISC_DYNAMIC_MINOR,
.name = "my_dev",
.fops = &my_fops,
};
static int __init my_init(void)
{
kaddr = kzalloc(PAGE_SIZE * 3, GFP_KERNEL);
return misc_register(&mdev);
}
module_init(my_init);
MODULE_AUTHOR("Leon He <hexiaolong2008@gmail.com");
MODULE_DESCRIPTION("mmap simple demo");
MODULE_LICENSE("GPL v2");