原文转自(http://www.embedhq.org/html/jsbw/2011/0214/1066.html)
//=======================字符设备驱动模板开始 ===========================//
#define CHAR_DEV_DEVICE_NAME "char_dev" // 设备名
struct class *char_dev _class; // class结构用于自动创建设备结点
static int major = 0;/* 0表示动态分配主设备号,可以设置成未被系统分配的具体的数字。*/
static struct cdev char_dev_devs;// 定义一个cdev结构
// 设备建立子函数,被char_dev_init函数调用
static void char_dev_setup_cdev(struct cdev *dev, int minor, struct file_operations *fops)
{
int err, devno = MKDEV(major, minor);
cdev_init(dev, fops);
dev->owner = THIS_MODULE;
dev->ops = fops;
err = cdev_add(dev, devno, 1);
if( err )
{
printk(KERN_NOTICE "Error %d adding char_dev %d\n", err, minor);
}
}
// file_operations 结构体设置,该设备的所有对外接口在这里明确,此处只写出了几常用的
static struct file_operations char_dev_fops =
{
.owner = THIS_MODULE,
.open = char_dev_open, // 打开设备
.release = char_dev_release, // 关闭设备
.read = char_dev_read, // 实现设备读功能
.write = char_dev_write, // 实现设备写功能
.ioctl = char_dev_ioctl, //实现设备控制功能
};
// 进行初始化设置,打开设备,对应应用空间的open 系统调用
int char_dev_open (struct inode *inode, struct file *filp)
{
... // 这里可以进行一些初始化
return 0;
}
// 释放设备,关闭设备,对应应用空间的close 系统调用
static int char_dev_release (struct inode *node, struct file *file)
{
... // 这里可以进行一些资源的释放
return 0;
}
// 实现读功能,读设备,对应应用空间的read 系统调用
ssize_t char_dev_read (struct file *file,char __user *buff,size_t count,loff_t *offp)
{
...
return 0;
}
// 实现写功能,写设备,对应应用空间的write 系统调用
ssize_t char_dev_write(struct file *file,const char __user *buff,size_t count,loff_t *offp)
{
...
return 0;
}
// 实现主要控制功能,控制设备,对应应用空间的ioctl 系统调用
static int char_dev _ioctl(struct inode *inode,struct file *file,unsigned int cmd,unsigned long arg)
{
...
return 0;
}
// 设备初始化
static int char_dev_init(void)
{
int result;
dev_t dev = MKDEV(major, 0);
if( major )
{
// 给定设备号注册
result = register_chrdev_region(dev, 1, CHAR_DEV_DEVICE_NAME);
}
else
{
// 动态分配设备号
result = alloc_chrdev_region(&dev, 0, 1, CHAR_DEV_DEVICE_NAME);
major = MAJOR(dev);
}
char_dev_setup_cdev(&char_dev_devs, 0, &char_dev_fops);
printk("The major of the char_dev device is %d\n", major);
//==== 有中断的可以在此注册中断:request_irq,并要实现中断服务程序 ===//
// 创建设备结点
char_dev _class = class_create(THIS_MODULE,"ad_class");
if (IS_ERR(char_dev _class))
{
printk("Err: failed in creating class.\n");
return 0;
}
device_create(char_dev_class, NULL, dev, NULL, "char_dev");
return 0;
}
// 设备注销
static void char_dev_cleanup(void)
{
device_destroy(adc_class,dev);
class_destroy(adc_class);
cdev_del(&char_dev_devs);//字符设备的注销*/
unregister_chrdev_region(MKDEV(major, 0), 1);//设备号的注销
//======== 有中断的可以在此注销中断:free_irq ======//
printk("char_dev device uninstalled\n");
}
module_init(char_dev_init);//模块初始化接口
module_exit(char_dev_cleanup);//模块注销接口
// 以下两句不能省略,否则编译不通过
MODULE_AUTHOR("www.embedhq.org");
MODULE_LICENSE("GPL");
//==================== 字符设备驱动模板结束 ========================//
用Makefile模板编译,Makefile如下:
//======================= Makefile开始 ===========================//
ifeq ($(KERNELRELEASE),)
#KERNELDIR ?= /your/target/source/directory/
KERNELDIR ?= /opt/kernal/linux-2.6.32.10/
PWD := $(shell pwd)
modules:
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
modules_install:
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules_install
clean:
rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions
.PHONY: modules modules_install clean
//========================= Makefile结束 =============================//
make编译后生成char_dev.ko,控制台输入加载和卸载命令,还可以使用lsmod查看已经加载的模块信息。
insmod char_dev.ko #加载驱动,会执行module_init中的语句
rmmod char_dev #卸载驱动,会执行module_exit中的语句
分享到:
相关推荐
这个是linux2.6.32内核源码,没有做任何修改。可以放心下载。
Linux 2.6.32 内核在 mini2440 上的移植主要讲述 YAFFS2 文件系统移植,YAFFS2 是专门针对嵌入式设备,特别是使用 NAND Flash 作为存储器的嵌入式设备而创建的一种文件系统。下面是 YAFFS2 文件系统移植的详细步骤。...
该程序是CY68013基于Linux 2.6.32内核的驱动源码,程序基于源码中usb-skeleton.c框架进行修改,能够实现33MB/S的传输速度
总的来说,移植Linux内核到Mini2440涉及硬件适配、驱动编写和加载,以及系统配置等多个环节。这些文档应该提供了一个全面的指南,帮助开发者完成这一过程。对于初学者,理解每个步骤的原理和实践操作至关重要,而...
Linux 2.6.32是Linux内核的一个版本,它主要包含了系统的核心功能与驱动程序。该版本的内核目录结构清晰地组织了各种组件,方便开发者进行维护和扩展。 #### 二、目录结构详解 ##### 1. `arch` 目录 `arch` 目录...
《Linux设备驱动程序》第三版(通常简称ldd3)是Linux系统开发领域的经典之作,由Robert Love编写,深入浅出地介绍了Linux内核设备驱动的编写与理解。该书的源码包“ldd3_examples”包含了书中所涉及的所有示例代码...
Linux 2.6.32 是一个历史悠久的Linux内核版本,发布于2009年11月,由Linux之父Linus Torvalds主导。这个版本在Linux内核的发展历程中扮演了重要的角色,引入了许多新特性、优化和改进,提升了系统的稳定性和性能。...
Linux内核是操作系统的核心部分,负责管理系统的进程、内存、设备驱动、文件系统以及网络等。在2.6.32版本中,内核调试工具KDB(Kernel Debugger)是用于对内核进行动态调试的重要组件。KDB允许开发者在内核运行时...
标题提到的"linux2.6.32下rootkit,仿照all_root"是指针对Linux内核版本2.6.32的一个Rootkit实现,这个实现是基于旧版的all_root工具进行修改,使其适应了新的内核版本。 all_root原本是为Linux 2.4内核设计的,而...
linux-2.6.32内核源代码之Part02
构建Linux 2.6内核树是开发和定制Linux内核的一个关键步骤,尤其是在编写或编译设备驱动程序时。这一过程涉及到从源代码构建一个新的内核,以便于支持特定的硬件设备或进行性能优化。以下是对构建Linux 2.6内核树的...
在配置过程中,我们还需要关注设备驱动部分。例如,要支持LCD显示,需要在`Device Drivers`菜单中选择`LCD support`,并配置对应的LCD尺寸和背光控制。对于Mini2440,可能选择3.5英寸240x320分辨率的统宝LCD。同样,...
【定制Linux内核(linux2.6.32)】是指在Linux操作系统中,针对特定硬件或应用场景,对内核进行个性化配置的过程。这个过程通常包括选择必要的内核模块、裁剪不必要的功能,以达到优化性能、减小内核体积、提高系统...
通过对Mini2440嵌入式Linux内核定制的剖析,我们可以了解到内核定制不仅仅是针对特定硬件平台进行必要的驱动支持配置,还需要考虑到各种外设的支持,如LCD、触摸屏和USB设备等。通过这种方式,可以更好地满足不同...
本文以Linux内核版本2.6.32为例,介绍如何在Ubuntu 9.04操作系统上,使用arm-linux-gcc-4.4.3交叉编译器进行内核配置和编译,以适应基于S3C2440处理器的FriendlyARM Mini2440开发板。 首先,你需要确保已安装了必要...
Linux2.6最新最稳定的内核版本源码,Linux-2.6.38.4,加入了新的功能,下载体验吧
《Linux内核详解:聚焦2.6.32版本》 在开源软件的世界里,Linux内核无疑是最为耀眼的一颗星。它以其强大的稳定性和灵活性,被广泛应用于各种操作系统和设备之中。当我们谈论"linux-2.6.32.tar.gz"时,我们实际上是...
此外,Linux 2.6内核对I/O子系统进行了改进,增加了对更多多媒体应用的支持,这在多媒体设备和通信设备中非常关键。内核还扩展了对不同处理器架构的支持,包括对没有硬件内存管理单元(MMU-less)系统的支持,这意味...