赛迪网 > IT技术 Linux > 源码开发
  IT资讯搜索
 
IT产品搜索
[程序开发][网管世界][网络安全][数据库技术]
[操作系统][嘉宾聊天·在线访谈][活动集锦]
[精彩专题][Symantec专区][订阅IT技术周刊]
[开发论坛][网管论坛][安全论坛][数据库论坛]
[操作系统论坛][Sybase专区][IBM dW技术专区]
[病毒求助][病毒与漏洞播报][文档·源码下载]

内核入门:较为基础的Linux系统设备驱动

发布时间:2007.12.12 06:32     来源:赛迪网    作者:sixth

驱动程序为:

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

static unsigned int major = 0;
static unsigned int minor = 0;
static unsigned int devno;
static char *filename = "mydevice";
static struct cdev *mycdev = NULL;
static int mycdevflg = 0;
static int devnoflg = 0;
static int adddevflg = 0;
MODULE_LICENSE("Dual BSD/GPL");
static int myopen(struct inode *inodep, struct file *flipl)
{ 
printk("my open is run\n");
return 0;
}
static ssize_t myread(struct file *flip, char __user *buf, size_t size, loff_t offset)
{
printk("myread is ok!\n");
static int i = 0;
///copy_to_user(buf,from,size);
*buf = i++;
return 0;
}
static int myrelease(struct inode *myindoe, struct file *flip)
{
printk("myrelease is run.\n");
return 0;
}
static struct file_operations myfops =
{
.owner = THIS_MODULE,
.open = myopen,
.read = myread,
.release= myrelease,
};
/* 初始化设备的过程主要是三步:1,生成设备号;2初始化设备;3,添加到内核。
*/
static int __init myinit(void)
{
int result = -1;
if(major)
{
devno = MKDEV(major,minor); //生成设备号
result = register_chrdev_region(devno,1,filename);//注册设备号devno,1为设备的个数
devnoflg = 1;
}
else
{
result = alloc_chrdev_region(&devno,minor,1,filename);//生成设备号兵注册,
mior为次设备号,这里注意的是指针devno?
major = MAJOR(devno);
devnoflg = 1;
}
if(result < 0)
{
printk("can't register the major num!\n");
devnoflg = 0;
return -1;
}
//mycdev = kmalloc(sizeof(struct cdev),GFP_KERNEL);
//如果采用cdev_init(struct cdev*,struct file *)方式的话,这个才需要。
mycdev = cdev_alloc();//申请cdev内存兵初始化设备cdev,不能在这之前申请内存,
否则要释放两次!
if(NULL == mycdev)
{
printk("can't request the memory!\n");
}
mycdevflg = 1;
//memset(mycdev,0,sizeof(mycdev));

mycdev->owner = THIS_MODULE;//如果采用cdev_init(struct cdev*,struct file *)方式的话,
这两项可以去掉
mycdev->ops = &myfops;//如果采用cdev_init(struct cdev*,struct file *)方式的话,
这两项可以去掉
result = cdev_add(mycdev,devno,1);//设备和设备号联系起来,
即通常说的添加设备到内核
if(result < 0)
{
printk("can't add cdev.2\n");
adddevflg = 0;
}
else
{
adddevflg = 1;
}
return 0;
}
/* 以和设备注册的次序"卸载",注意有的时候,当出错的时候,要注意是否用到这些操作,
所以要加上判断,否则比如对空指针free会导致系统崩溃!
*/
static void myexit(void)
{
printk("myexit begin.\n");
if(adddevflg) 
{
cdev_del(mycdev);
adddevflg = 0;
}
if(mycdev) 
{
kfree(mycdev);
mycdev = NULL;
}
if(devnoflg)
{
unregister_chrdev_region(devno,1);
devno = 0;
devnoflg = 0;
}
printk("exit over.\n");
}
module_init(myinit);
module_exit(myexit);

应用程序为:

#include 
#include 
#include 
#include 
#include 
int main() 
{ 
int fd; 
int i=0; 
fd = open("/dev/mydevice", O_RDONLY); 
if (fd < 0) 
{ 
printf("error1\n"); 
return -1; 
} 
int j = 0;
while(j++<10000)
{
if (read(fd, &i, sizeof(int)) < 0) 
{ 
printf("read error2\n"); 
return -1; 
} 
printf("i = %d\n",i);
}
close("/dev/mydevice");
return 0; 
}

一定要注意register_chrdev_region(),alloc_chrdev_region(),cdev_init(),cdev_alloc(),cdev_add()这些函数的参数类型,不要把指针当成非指针,把int类型当作指针(或者&)来使用!

(责任编辑:云子)


[ 发表评论 ] 字体[  ] [ 打印 ] [ 进入博客 ] [ 进入论坛 ]  [ 推荐给朋友 ]
  相关文章
· Ubuntu上安装Tor并配置Firefox匿名冲浪 (12-11) · 新手学堂:Windows与Linux系统下的库介绍 (12-11)
· 自由软件社区也打侵权官司 直指Verizon (12-11) · 内核研发者Ted Ts'o加盟Linux Foundation (12-11)
· 新手学堂:Linux操作系统修改时区的方法 (12-11) · 实用技巧:同步Gnome和Google日历的方法 (12-11)
· 不同版本的Linux操作系统下软件源的设置 (12-11) · Linux下由于软件问题导致死机的情况列举 (12-11)
· 安全威胁无孔不入:基于Linux系统的病毒 (12-11) · Linux电话终端标准论坛发布LiPS规范1.0版本 (12-10)
  客户需求反馈表
* 姓  名:
更多资料  了解方案  认识厂商
* 单位名称:
* 联系电话:
* 电子邮件:
  赛迪推荐  
  手机·资费 ·新品·导购·评测·手机资费·宽带
手机搜索  诺基亚 N73 MOTO Z6
  IT产品 ·笔记本·台式机·服务器·打印·投影
IT产品搜索 
  IT技术 ·开发·网管·安全·数据库·操作系统
  信息化 ·热点·专题·访谈·周刊·方案案例
· 工信部“三定”公布 总编制731名设24司局
· 北京发电子商务监管意见 营利性网店须办照
· 直播 08中国城市信息化高峰论坛 案例点评
· 烽火网络校园解决方案 移民安置信息管理系统
  IT博客 ·曾剑秋·项立刚·Java学习·网管
  IT技术论坛 ·开发·网管·安全·数据库·系统