1、要求:实现简单的字符设备驱动程序

2、源码清单

#include <linux/module.h>

#include <linux/types.h>

#include <linux/fs.h>

#include <linux/errno.h>

#include <linux/mm.h>

#include <linux/sched.h>

#include <linux/init.h>

#include <linux/cdev.h>

#include <asm/io.h>

#include <asm/system.h>

#include <asm/uaccess.h>

#include <linux/slab.h>

#define GLOBALMEM_SIZE 0x1000 /*全局内存最大4K字节*/

#define MEM_CLEAR 0x1 /*清0全局内存*/

#define GLOBALMEM_MAJOR 254 /*预设的globalmem的主设备号*/

static globalmem_major = GLOBALMEM_MAJOR;

/*globalmem设备结构体*/

struct globalmem_dev

{

struct cdev cdev; /*cdev结构体*/

unsigned char mem[GLOBALMEM_SIZE]; /*全局内存*/

struct semaphore sem; /*并发控制用的信号量*/

};

struct globalmem_dev *globalmem_devp; /*设备结构体指针*/

/*文件打开函数*/

int globalmem_open(struct inode *inode, struct file *filp)

{

/*将设备结构体指针赋值给文件私有数据指针*/

filp->private_data = globalmem_devp;

return 0;

}

/*文件释放函数*/

int globalmem_release(struct inode *inode, struct file *filp)

{

return 0;

}

/* ioctl设备控制函数 */

static int globalmem_ioctl(struct inode *inodep, struct file *filp, unsigned int cmd, unsigned long arg)

{

struct globalmem_dev *dev = filp->private_data; /*获得设备结构体指针*/

switch (cmd)

{

case MEM_CLEAR:

if (down_interruptible(&dev->sem))

{

return - ERESTARTSYS;

}

memset(dev->mem, 0, GLOBALMEM_SIZE);

up(&dev->sem); //释放信号量

printk(KERN_INFO "globalmem is set to zero\n");

break;

default:

return - EINVAL;

}

return 0;

}

/*读函数*/

static ssize_t globalmem_read(struct file *filp, char __user *buf, size_t size, loff_t *ppos)

{

unsigned long p = *ppos;

unsigned int count = size;

int ret = 0;

struct globalmem_dev *dev = filp->private_data; /*获得设备结构体指针*/

/*分析和获取有效的写长度*/

if (p >= GLOBALMEM_SIZE)

return count ? - ENXIO: 0;

if (count > GLOBALMEM_SIZE - p)

count = GLOBALMEM_SIZE - p;

if (down_interruptible(&dev->sem))

{

return - ERESTARTSYS;

}

/*内核空间->用户空间*/

if (copy_to_user(buf, (void*)(dev->mem + p), count))

{

ret = - EFAULT;

}

else

{

*ppos += count;

ret = count;

printk(KERN_INFO "read %d bytes(s) from %d\n", count, p);

}

up(&dev->sem); //释放信号量

return ret;

}

/*写函数*/

static ssize_t globalmem_write(struct file *filp, const char __user *buf,

size_t size, loff_t *ppos)

{

unsigned long p = *ppos;

unsigned int count = size;

int ret = 0;

struct globalmem_dev *dev = filp->private_data; /*获得设备结构体指针*/

/*分析和获取有效的写长度*/

if (p >= GLOBALMEM_SIZE)

return count ? - ENXIO: 0;

if (count > GLOBALMEM_SIZE - p)

count = GLOBALMEM_SIZE - p;

if (down_interruptible(&dev->sem))//获得信号量

{

return - ERESTARTSYS;

}

/*用户空间->内核空间*/

if (copy_from_user(dev->mem + p, buf, count))

ret = - EFAULT;

else

{

*ppos += count;

ret = count;

printk(KERN_INFO "written %d bytes(s) from %d\n", count, p);

}

up(&dev->sem); //释放信号量

return ret;

}

/* seek文件定位函数 */

static loff_t globalmem_llseek(struct file *filp, loff_t offset, int orig)

{

loff_t ret = 0;

switch (orig)

{

case 0: /*相对文件开始位置偏移*/

if (offset < 0)

{

ret = - EINVAL;

break;

}

if ((unsigned int)offset > GLOBALMEM_SIZE)

{

ret = - EINVAL;

break;

}

filp->f_pos = (unsigned int)offset;

ret = filp->f_pos;

break;

case 1: /*相对文件当前位置偏移*/

if ((filp->f_pos + offset) > GLOBALMEM_SIZE)

{

ret = - EINVAL;

break;

}

if ((filp->f_pos + offset) < 0)

{

ret = - EINVAL;

break;

}

filp->f_pos += offset;

ret = filp->f_pos;

break;

default:

ret = - EINVAL;

break;

}

return ret;

}

/* http://blog.csdn.net/zhou1232006/article/details/6867584*/

/*文件操作结构体*/

static const struct file_operations globalmem_fops =

{

.owner = THIS_MODULE,

.llseek = globalmem_llseek,

.read = globalmem_read,

.write = globalmem_write,

//.compat_ioctl = globalmem_ioctl,

.unlocked_ioctl = globalmem_ioctl,

.open = globalmem_open,

.release = globalmem_release,

};

/*初始化并注册cdev*/

static void globalmem_setup_cdev(struct globalmem_dev *dev, int index)

{

int err, devno = MKDEV(globalmem_major, index);

cdev_init(&dev->cdev, &globalmem_fops);

dev->cdev.owner = THIS_MODULE;

dev->cdev.ops = &globalmem_fops;

err = cdev_add(&dev->cdev, devno, 1);

if (err)

printk(KERN_NOTICE "Error %d adding LED%d", err, index);

}

/*设备驱动模块加载函数*/

int globalmem_init(void)

{

int result;

dev_t devno = MKDEV(globalmem_major, 0);

/* 申请设备号*/

/*if (globalmem_major)

result = register_chrdev_region(devno, 1, "globalmem");

else /* 动态申请设备号 */

{*/

result = alloc_chrdev_region(&devno, 0, 1, "globalmem");

globalmem_major = MAJOR(devno);

//}

if (result < 0)

return result;

/* 动态申请设备结构体的内存*/

globalmem_devp = kmalloc(sizeof(struct globalmem_dev), GFP_KERNEL);

if (!globalmem_devp) /*申请失败*/

{

result = - ENOMEM;

goto fail_malloc;

}

memset(globalmem_devp, 0, sizeof(struct globalmem_dev));

globalmem_setup_cdev(globalmem_devp, 0);

sema_init(&globalmem_devp->sem,1); /*初始化信号量*/

return 0;

fail_malloc: unregister_chrdev_region(devno, 1);

return result;

}

/*模块卸载函数*/

void globalmem_exit(void)

{

cdev_del(&globalmem_devp->cdev); /*注销cdev*/

kfree(globalmem_devp); /*释放设备结构体内存*/

unregister_chrdev_region(MKDEV(globalmem_major, 0), 1); /*释放设备号*/

}

MODULE_AUTHOR("Zhao guohui");

MODULE_LICENSE("Dual BSD/GPL");

module_param(globalmem_major, int, S_IRUGO);

module_init(globalmem_init);

module_exit(globalmem_exit);

3、编译用Makefile

obj-m := globalmem.o

KERNELBUILD :=/usr/src/linux-headers-3.2.0-23-generic-pae

default:

make -C $(KERNELBUILD) M=$(shell pwd) modules

clean:

rm -rf *.o *.ko *.mod.c

1)error: unknown field ‘ioctl’ specified in initializer

这是由于跨版本移植的时候由于内核提供的函数变化而引起的错误:

解决:http://blog.csdn.net/zhou1232006/article/details/6867584

2)error:implicit declaration of function ‘kmalloc’ [-Werror=implicit-function-declaration]

解决:http://blog.csdn.net/liukun321/article/details/6785608

3)error: ‘globalfifo_devp’ undeclared (first use in this function)

参考:http://www.tuicool.com/articles/6r2M3a

error: implicit declaration of function ‘init_MUTEX’ [-Werror=implicit-function-declaration]

4、验证

首先加载内核模块insmod gllobalmem.ko

设备忙的解决方法

让程序一直寻找可以使用的设备号。

终端输入 cat /proc/devices

找到globalmem的设备号

建立设备文件:mknod /dev/globalmem c 你找到的设备号 0

[root@localhost driver_study]# echo 'hello world' > /dev/globalmem

[root@localhost driver_study]# cat /dev/globalmem

hello world

NeuSoft(4)编写字符设备驱动的更多相关文章

  1. Linux字符设备驱动框架

    字符设备是Linux三大设备之一(另外两种是块设备,网络设备),字符设备就是字节流形式通讯的I/O设备,绝大部分设备都是字符设备,常见的字符设备包括鼠标.键盘.显示器.串口等等,当我们执行ls -l ...

  2. LDD3 字符设备驱动简单分析

    最近在看LDD3,理解了一下,为了加深自己的印象,自己梳理一下.我用的CentOS release 6.6 (Final)系统. 一.编写编译内核模块的Makefile 以下是我用的Makefile ...

  3. 字符设备驱动之Led驱动学习记录

    一.概述 Linux内核就是由各种驱动组成的,内核源码中大约有85%的各种渠道程序的代码.一般来说,编写Linux设备驱动大致流程如下: 1.查看原理图,数据手册,了解设备的操作方法. 2.在内核中找 ...

  4. Linux应用程序访问字符设备驱动详细过程【转】

    本文转载自:http://blog.csdn.net/coding__madman/article/details/51346532 下面先通过一个编写好的内核驱动模块来体验以下字符设备驱动 可以暂时 ...

  5. 深入理解Linux字符设备驱动

    文章从上层应用访问字符设备驱动开始,一步步地深入分析Linux字符设备的软件层次.组成框架和交互.如何编写驱动.设备文件的创建和mdev原理,对Linux字符设备驱动有全面的讲解.本文整合之前发表的& ...

  6. Linux字符设备驱动结构(一)--cdev结构体、设备号相关知识机械【转】

    本文转载自:http://blog.csdn.net/zqixiao_09/article/details/50839042 一.字符设备基础知识 1.设备驱动分类 linux系统将设备分为3类:字符 ...

  7. Linux 字符设备驱动模型

    一.使用字符设备驱动程序 1. 编译/安装驱动 在Linux系统中,驱动程序通常采用内核模块的程序结构来进行编码.因此,编译/安装一个驱动程序,其实质就是编译/安装一个内核模块 2. 创建设备文件 通 ...

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

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

  9. 【驱动】linux设备驱动·字符设备驱动开发

    Preface 前面对linux设备驱动的相应知识点进行了总结,现在进入实践阶段! <linux设备驱动入门篇>:http://infohacker.blog.51cto.com/6751 ...

随机推荐

  1. hdu3448 01背包+dfs

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=3448 Description 0/1 bag problem should sound f ...

  2. http://www.cnblogs.com/younggun/archive/2013/07/16/3193800.html

    http://www.cnblogs.com/younggun/archive/2013/07/16/3193800.html

  3. 餐厅系统app7

    团队贡献分 杨子健:23 郭志豪:24 谭宇森:22 刘森松:31

  4. iOS 注意事项

    1.记得在项目中设置项目文件命名的prefix,避免命名冲突. 2.在适当的位置对属性和方法进行注释,建议利用插件(如VVDocument)提供效率.在给企业看文档时,可以利用(如Doxygen)这种 ...

  5. ccc progressbar

    cc.Class({ extends: cc.Component, properties: { progressBar: { default:null, type:cc.ProgressBar }, ...

  6. 转:Web页面通过URL地址传递参数常见问题及检测方法

    Web页面即我们在浏览器中所看到的网页,在Web应用程序中,其页面往往需要进行动态切换和数据交互,页面间的数据常规传递方法有多种,本文主要介绍Web页面处理程序中常见的URL地址参数传递方法,包括概述 ...

  7. 自己收集原生js-2014-2-15

    function testforbtn(event){ alert(window.EventUtil.getEventTarget(window.EventUtil.getEvent( event)) ...

  8. 【BZOJ】2924: [Poi1998]Flat broken lines

    题意 平面上有\(n\)个点,如果两个点的线段与\(x\)轴的角在\([-45^{\circ}, 45^{\circ}]\),则两个点可以连线.求最少的折线(折线由线段首尾相连)使得覆盖所有点. 分析 ...

  9. UVA 11609 - Teams(二项式系数)

    题目链接 想了一会,应该是跟二项式系数有关系,无奈自己推的式子,构不成二项式的系数. 选1个人Cn1*1,选2个人Cn2*2....这样一搞,以为还要消项什么的... 搜了一下题解,先选队长Cn1,选 ...

  10. 通过/etc/rc.local实现开机自动拉起服务

    添加服务到/etc/rc.local 如自动拉起apache服务: /etc/rc.local: #!/bin/sh # # This script will be executed *after* ...