最近在看LDD3,理解了一下,为了加深自己的印象,自己梳理一下。我用的CentOS release 6.6 (Final)系统。

一、编写编译内核模块的Makefile

以下是我用的Makefile

ifneq ($(KERNELRELEASE),)
# 第一次被调用时,KERNELRELEASE为空,所以不会被执行
obj-m := scull.o
else
# 将内核编译用的Makefile路径赋值给KERNELRELEASE
KERNELRELEASE ?= /lib/modules/`uname -r`/build
# 当前工作目录
PWD := `pwd`
default:
# 通过内核编译树编译一个模块
$(MAKE) -C $(KERNELRELEASE) M=$(PWD) modules
endif clean:
rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions

Makefile要注意tab键和空格不能互相替换使用。这个Makfile其实会被调用2次,第一次这个Makefile会调用  /lib/modules/`uname -r`/build 目录下的Makefile,也就是linux编译内核用的Makefile并且传入后面参数。linux的内核Makefile会再次调用这个Makefile编译内核模块。

二、编写内核模块

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/cdev.h> MODULE_LICENSE("GPL"); static int scull_major = ; struct scull_dev {
struct cdev cdev;
}; struct scull_dev scull_device; static ssize_t
scull_read(struct file * filep, char __user * buf, size_t count, loff_t *f_pos)
{
printk(KERN_INFO "scull_read\n");
return ;
} static ssize_t
scull_write(struct file *filep, const char __user *buf, size_t count, loff_t *fpos)
{
printk(KERN_INFO "scull_write\n");
return count;
} static int
scull_open(struct inode *inode, struct file *filep)
{
printk(KERN_INFO "scull_open\n");
return ;
} static int
scull_release(struct inode *inode, struct file *filep)
{
printk(KERN_INFO "scull_release\n");
return ;
} struct file_operations scull_fops = { .owner = THIS_MODULE,
.read = scull_read,
.write = scull_write,
.open = scull_open,
.release = scull_release
}; static void scull_setup_cdev(struct scull_dev *dev, int index)
{
cdev_init(&(dev->cdev), &scull_fops);
dev->cdev.owner = THIS_MODULE;
dev->cdev.ops = &scull_fops; if (cdev_add(&dev->cdev, MKDEV(scull_major, ), )) {
printk(KERN_NOTICE "error adding scull\n");
}
} static int scull_init(void)
{
if (printk_ratelimit())
printk(KERN_INFO "scull loaded.\n"); if (register_chrdev_region(MKDEV(scull_major, ), , "scullc") != ) {
printk(KERN_NOTICE "register chrdev failed.\n");
} scull_setup_cdev(&scull_device, ); return ;
} static void scull_cleanup(void)
{
if (printk_ratelimit())
printk(KERN_INFO "scull unloaded.\n"); unregister_chrdev_region(MKDEV(scull_major, ), );
} module_init(scull_init);
module_exit(scull_cleanup);

scull.c

在模块的初始化函数scull_init里面注册,字符设备。我这边直接是固定的主设备为66,次设备号为0。注册字符设备函数为register_chrdev_region。另外,在scull_cleanup也就是模块卸载函数里面调用unregister_chrdev_region释放申请主设备号。大家自己要注意主设备号不要和自己机器上已有的设备号重复,否则会注册不成功。注册成功的话,我们可以通过/proc/devices这个文件来看。

从图片中,可以看出来,我们已经注册成功了。注册成功之后,自然是要初始化这个设备,通过使用cdev_init和cdev_add来向内核注册字符设备。同时会将struct file_operations scull_fops赋值给这个设备。也就是说当用到这个驱动的设备打开时,scull_open会被调用,scull_read会在读取设备被调用,scull_write会在写入时被调用。这个后面我有演示。

三、创建设备文件

在第二步中,我们已经编写好模块,并且也向内核注册了,申请了主设备号和次设备号,我们现在只要通过下面命令来创建一个设备文件,对这个设备文件的操作,就会使用我们刚刚写的那个简单字符设备驱动。

mknod /dev/scull0 c  

创建完后,通过命令可以看到

图片中就是我们注册的主设备号为66,次设备号为0.

四、测试字符驱动

insmod scull.ko 加载驱动模块

echo 1 > /dev/scull0 向设备文件写入 1

cat /dev/scull0 输出设备文件内容

rmmod scull.ko卸载驱动

以上四个命令分别对应下图的输出

从图中可以看出来 echo的时候会先打开设备,再写入,最后关闭。cat命令会先打开设备,再读取,最后关闭。后面,我会再模仿LDD3中,把这个驱动继续扩展。

LDD3 字符设备驱动简单分析的更多相关文章

  1. LCD驱动分析(一)字符设备驱动框架分析

    参考:S3C2440 LCD驱动(FrameBuffer)实例开发<一>   S3C2440 LCD驱动(FrameBuffer)实例开发<二> LCD驱动也是字符设备驱动,也 ...

  2. Linux 字符设备驱动简单总结(转)

    http://my.oschina.net/u/1169027/blog/191538

  3. 【转】linux设备驱动程序之简单字符设备驱动

    原文网址:http://www.cnblogs.com/geneil/archive/2011/12/03/2272869.html 一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用 ...

  4. Linux内核分析(五)----字符设备驱动实现

    原文:Linux内核分析(五)----字符设备驱动实现 Linux内核分析(五) 昨天我们对linux内核的子系统进行简单的认识,今天我们正式进入驱动的开发,我们今后的学习为了避免大家没有硬件的缺陷, ...

  5. arm-linux字符设备驱动开发之---简单字符设备驱动

    一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 1.字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面 ...

  6. linux设备驱动第三篇:如何实现一个简单的字符设备驱动

    在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存 ...

  7. linux设备驱动第三篇:如何写一个简单的字符设备驱动?

    在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存 ...

  8. 【Linux-驱动】简单字符设备驱动结构和初始化

    (1)在编写简单字符设备驱动的时候,首先要申请一个设备结构struct cdev: struct cdev { struct kobject kobj; struct module *owner; / ...

  9. linux设备驱动第三篇:写一个简单的字符设备驱动

          在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分 ...

随机推荐

  1. JVM内存区域异常分析

    在Java虚拟机规范描述中,除程序计数器外,其他几个运行时区域都有可能发生OutOfMemoryError异常.接下来将对各区域分别进行分析介绍,内容包括触发各区域OutOfMemoryError异常 ...

  2. Android端接收和发送cookie

    流程: 首先android端使用HttpClient的方式发送HTTP请求,此时服务器创立cookie,并发送cookie给android端,android端再将cookie保存起来,在需要发送coo ...

  3. java面向对象_抽象类和接口

    一.抽象类 1.抽象方法:由abstract修饰.只有定义没有方法体.用一个分号结尾. 2.抽象类: 1)包含抽象方法的类必须是抽象类 2)由abstract修饰 3)不能被实例化 4)抽象类如果不被 ...

  4. 使用XStream解析MXL文件用到的jar包---xpp3_min-1.1.3.4.O.jar和xstream-1.3.1.jar

    使用XStream解析MXL文件用到的jar包---xpp3_min-1.1.3.4.O.jar和xstream-1.3.1.jar

  5. Linux下搭建Windows KMS服务器

    这几天微软发布了Windows 10 RedStone 1 Build 14390, 于是我第一时间下载进行了试用.和之前那种不激活也没有任何异样不同,现在的版本如果不激活有些功能就受限了,比如你无法 ...

  6. PostGr-SQL database创建表

    postgres=# create database vertigo_sandbox; postgres=# \connect vertigo_sandbox vertigo_sandbox=# CR ...

  7. eclipse护眼颜色和字体大小设置

    ♣eclipse护眼颜色和关键字颜色设置 ♣eclipse字体大小设置(包括jsp , .xml ,.java) 1.Eclipse字体大小调整: 窗口(Window)-首选项(Preferences ...

  8. PHPExcel 使用笔记

    获取sheet 有效效值行列数 $maxCell = $objWorksheet->getHighestRowAndColumn(); $data = $objWorksheet->ran ...

  9. Flink – window operator

      参考, http://wuchong.me/blog/2016/05/25/flink-internals-window-mechanism/ http://wuchong.me/blog/201 ...

  10. UML序列图总结(Loop、Opt、Par和Alt)

    序列图主要用于展示对象之间交互的顺序. 序列图将交互关系表示为一个二维图.纵向是时间轴,时间沿竖线向下延伸.横向轴代表了在协作中各独立对象的类元角色.类元角色用生命线表示.当对象存在时,角色用一条虚线 ...