- 论坛徽章:
- 0
|
先上程序
/*----------------------part 1 start ----------------------------*/
/* Necessary includes for device drivers */
#include linux/init.h>
//#include
#include linux/module.h>
#include linux/kernel.h> /* printk() */
#include linux/slab.h> /* kmalloc() */
#include linux/fs.h> /* everything... */
#include linux/errno.h> /* error codes */
#include linux/types.h> /* size_t */
#include linux/proc_fs.h>
#include linux/fcntl.h> /* O_ACCMODE */
#include asm/system.h> /* cli(), *_flags */
#include asm/uaccess.h> /* copy_from/to_user */
MODULE_LICENSE("Dual BSD/GPL");
/* Declaration of memory.c functions */
int memory_open(struct inode *inode, struct file *filp);
int memory_release(struct inode *inode, struct file *filp);
ssize_t memory_read(struct file *filp, char *buf, size_t count, loff_t *f_pos);
ssize_t memory_write(struct file *filp, char *buf,size_t count, loff_t *f_pos);
void memory_exit(void);
int memory_init(void);
/* Structure that declares the usual file */
/* access functions */
struct file_operations memory_fops = {
read: memory_read,
write: memory_write,
open: memory_open,
release: memory_release
};
/* Declaration of the init and exit functions */
module_init(memory_init);
module_exit(memory_exit);
/* Global variables of the driver */
/* Major number */
int memory_major = 60;
/* Buffer to store data */
char *memory_buffer;
/*----------------------part 1 end -----------------------------*/
/*----------------------part 2 start ---------------------------*/
int memory_init(void) {
int result;
/* Registering device */
result = register_chrdev(memory_major, "memory",&memory_fops);
if (result 0) {
printk("memory: cannot obtain major number %d\n",memory_major);
return result;
}
/* Allocating memory for the buffer */
memory_buffer = kmalloc(1, GFP_KERNEL);
if (!memory_buffer) {
result = -ENOMEM;
goto fail;
}
memset(memory_buffer, 0, 1);
printk("Inserting memory module\n");
return 0;
fail:
memory_exit();
return result;
}
/*----------------------part 2 end -----------------------------*/
/*----------------------part 3 start ---------------------------*/
void memory_exit(void) {
/* Freeing the major number */
unregister_chrdev(memory_major, "memory");
/* Freeing buffer memory */
if (memory_buffer) {
kfree(memory_buffer);
}
printk("Removing memory module\n");
}
/*----------------------part 3 end -----------------------------*/
/*----------------------part 4 start ---------------------------*/
int memory_open(struct inode *inode, struct file *filp)
{
/* Success */
return 0;
}
int memory_release(struct inode *inode, struct file*filp)
{
/* Success */
return 0;
}
/*----------------------part 4 end -----------------------------*/
/*----------------------part 5 start ---------------------------*/
ssize_t memory_read(struct file *filp, char *buf,size_t count, loff_t *f_pos) {
/* Transfering data to user space */
copy_to_user(buf,memory_buffer,1);
/* Changing reading position as best suits */
if (*f_pos == 0)
{
*f_pos+=1;
return 1;
}
else {
return 0;
}
}
ssize_t memory_write( struct file *filp, char *buf,size_t count, loff_t *f_pos)
{
char *tmp;
tmp=buf+count-1;
copy_from_user(memory_buffer,tmp,1);
return 1;
}
/*----------------------part 5 end -----------------------------*/
说明:此程序使用电脑内存虚拟了一个硬件设备,他没有什么特别的硬件特性要初始化,只能实现简单的文字写入与读取
Part 1 :
在设备驱动中常用的 #include 声明;定义两了模块的初始化及卸载函数(对模块加载和卸载的时候被执行);定义了这个模块使用的主设备号为 60;定义一个字符型指针用来保存该驱动的数据(的起始地址)。
Part 2:
register_chrdev函数用于在内核空间,把驱动和/dev下设备文件链接在一起。它又三个参数:主设备号,模块名称和一个file_operations结构的指针。用于实现模块的安装代码。另外,代码使用了kmalloc函数。这个函数工作在内核空间,用于为该驱动程序的缓冲区分配内存。它和我们熟悉的malloc函数很相似。最后,如果注册主设备号或者分配内存失败,模块将退出。
Part 3:
通过memory_exit函数卸载模块,需要定义unregsiter_chrdev函数。这将释放驱动之前向内核申请的主设备号。同时缓冲区也需要通过该函数进行释放。
Part 4:
linux,所有设备都视为文件!例如网络、设备等。当设备文件被打开后,通常就需要初始化驱动的各个变量,对设备进行复位;当设备文件关闭后,通常需要释放该设备使用的内存,释放各种操作该设备相关的变量。但在本例中,这些操作都没进行。
Part 5:
和用户空间函数fread类似,内核空间里,读取设备文件使用read函数:read是file_operations的成员,用于调用register_chrdev。本例中,是memory_read函数。它的参数有:一个file结构;一个缓冲区(buf),用户空间的fread函数将从该缓冲区读数据;一个记录要传输的字节数量的计数器(count),它和用户空间的fread使用的计数器值相同;最后一个参数(f_pos)指示从哪里开始读取该设备文件。本例中,memory_read函数通过copy_to_user函数从驱动的缓冲区(memory_buffer)向用户空间传送一个简单的字节。设备文件的读取位置(f_pos)也改变了。如果起始点是文件的开头,那么f_pos的值将增加1,如果要读取的字节读取正常,则返回值为1。如果读取位置不是文件开头,则是文件的末尾,返回值将是0(因为文件只存储了 1 个字节)。
和用户空间里写文件的fwrite对应,内核空间里是write:write是file_operations的成员,用于调用register_chrdev。本例中是memory_write函数,它有如下几个参数:一个file结构;buf,一个缓冲区,用户空间函数fwrite将向该该缓冲区写数据;count,统计将传送的字节数的计数器,和用户空间函数fwrite的计数器有相同的数值;最后是f_pos,指示从哪里开始写文件。
运行结果
[root@HUNK 2]# ls
Makefile memory.c~ memory.mod.c memory.o
memory.c memory.ko memory.mod.o Module.symvers
[root@HUNK 2]# insmod memory.ko
[root@HUNK 2]# echo "write data into memory-dev">/dev/memory
[root@HUNK 2]# cat /dev/memory
write data into memory-dev
[root@HUNK 2]#
本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u2/65955/showart_693239.html |
|