-
Notifications
You must be signed in to change notification settings - Fork 0
/
my-file.txt
142 lines (128 loc) · 4.75 KB
/
my-file.txt
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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
/*
* File: memdrv.c
* Date: 12/09/2022
* Author: Zhiyi Huang
* Version: 0.1
*/
/*
* 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/moduleparam.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/fs.h>
#include <linux/errno.h>
#include <linux/types.h>
#include <linux/proc_fs.h>
#include <linux/fcntl.h>
#include <linux/aio.h>
#include <linux/uaccess.h>
#include <linux/ioctl.h>
#include <linux/cdev.h>
#include <linux/device.h>
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Zhiyi Huang");
MODULE_DESCRIPTION("A simple memory device driver module");
/* The parameter used as a major number of the device. This variable is
* readable in /sys/module/memdrv/parameters.*/
int major = 0;
module_param(major, int, S_IRUGO);
MODULE_PARM_DESC(major, "device major number");
// 256 4KB-pages
#define MAX_DSIZE 0x1000 * 0x100
struct my_dev {
char data[MAX_DSIZE]; /* data storage */
size_t size; /* 32-bit will suffice */
struct semaphore sem; /* Mutual exclusion */
struct cdev cdev; /* kernel device structure */
struct class *class; /* create a class for the device*/
struct device *device; /* create the device in the system */
} *temp_dev;
/*
* This function will be called when the system call open() is called
* in the user space. It is expected to check if the process has the
* permission to access; suspend the calling process if the device is
* busy or used by another process; otherwise record the id of the
* calling process so that the device is owned by the process;
* initialise or reset the device. For such simple device, we have
* done neither of the above.
*
* Params: inode - the inode of the device file /dev/memdrv
* filp - the file descriptor created when /dev/memdrv is opened.
* Return: an integer indicating success (0) or failure (<0)
*/
int temp_open(struct inode *inode, struct file *filp) {
return 0;
}
/*
* This function will be called when the system call close() is called
* in the user space. It is expected to undo everything of
* temp_open(). For example, release the device by removing the id of
* the process; check if there is any waiting process to wake up; undo
* anything done at temp_open(), e.g. free memory.
*
* Params: inode - the inode of the device file /dev/memdrv
* filp - the file descriptor created when /dev/memdrv is opened.
* Return: an integer indicating success (0) or failure (<0)
*/
int temp_release(struct inode *inode, struct file *filp) {
return 0;
}
/* This function will be called when the system call read()
* is called in the user space. It is expected to find the
* location of data; ask the controller to copy data to
* the kernel kbuf; then copy data from kbuf to user space.
* The current function only find the data in kernel and copy
* to user space after sanity check.
*
* Params: filp - the file descriptor created when /dev/memdrv is opened.
* buf - pointer to user space buffer
* count - the number of bytes to be read
* f_pos - the current location of the device file
* Return: an integer indicating success (0) or failure (<0)
*/
ssize_t temp_read(struct file *filp, char __user *buf, size_t count,
loff_t *f_pos) {
int rv;
if (down_interruptible(&temp_dev->sem))
return -ERESTARTSYS;
if (*f_pos >= MAX_DSIZE) {
rv = 0;
goto wrap_up;
}
if (*f_pos + count > MAX_DSIZE)
count = MAX_DSIZE - *f_pos;
if (copy_to_user(buf, temp_dev->data + *f_pos, count)) {
rv = -EFAULT;
goto wrap_up;
}
up(&temp_dev->sem);
*f_pos += count;
return count;
wrap_up:
up(&temp_dev->sem);
return rv;
}
/*
* This function will be called when the system call write() is called
* in the user space. It is expected to copy data from user space buf
* to kernel buffer kbuf. Then find the location of data; ask the
* controller to copy data from the kernel kbuf to the device. The
* current function only copy the data in user space to the kernel
* buffer in the right location after sanity check.
*
* Params: filp - the file descriptor created when /dev/memdrv is opened.
* buf - pointer to user space buffer
* count - the number of bytes to be read
* f_pos - the current location of the device file
* Return: an integer indicating success (0) or failure (<0)
*/
ssize_t temp_write(struct file *filp, const char __user *buf, size_t count,
loff_t *f_pos) {
int rv;
if (down_interruptible(&temp_dev->sem))