第一步:my74hc595.c

#include <linux/module.h> //模块所需的大量符号和函数定义
#include <linux/init.h> //指定初始化和清除函数
#include <linux/fs.h> //文件系统相关的函数和头文件
#include <linux/cdev.h> //cdev结构的头文件
#include <asm/uaccess.h> //在内核和用户空间中移动数据的函数
#include <linux/slab.h>
#include <linux/device.h>

MODULE_LICENSE("GPL"); //指定代码使用的许可证

//文件操作函数的声明
int my74hc595_open(struct inode *, struct file *);
int my74hc595_release(struct inode *, struct file *);
ssize_t my74hc595_read(struct file *, char *, size_t, loff_t *);
ssize_t my74hc595_write(struct file *, const char *, size_t, loff_t *);

int dev_major = 1253; //指定主设备号
int dev_minor = 0; //指定次设备号

static struct class *firstdrv_class;
static struct device *firstdrv_class_dev;

struct cdev *my74hc595_cdev; //内核中表示字符设备的结构
int *gp_testdata;//测试用数据

struct file_operations my74hc595_fops= //将文件操作与分配的设备号相连
{
  owner: THIS_MODULE, //指向拥有该模块结构的指针
  open: my74hc595_open,
  release: my74hc595_release,
  read: my74hc595_read,
  write: my74hc595_write,
};

static void __exit my74hc595_exit(void) //退出模块时的操作
{
  dev_t devno=MKDEV(dev_major, dev_minor); //dev_t是用来表示设备编号的结构

  cdev_del(my74hc595_cdev); //从系统中移除一个字符设备
  kfree(my74hc595_cdev); //释放自定义的设备结构
  kfree(gp_testdata);
  unregister_chrdev_region(devno, 1); //注销已注册的驱动程序

  device_unregister(firstdrv_class_dev); //删除/dev下对应的字符设备节点
  class_destroy(firstdrv_class);

  printk("my74hc595 unregister success\n");
}

static int __init my74hc595_init(void) //初始化模块的操作
{
  int ret, err;
  dev_t devno;
#if 1
  //动态分配设备号,次设备号已经指定
  ret=alloc_chrdev_region(&devno, dev_minor, 1, "my74hc595");
  //保存动态分配的主设备号
  dev_major=MAJOR(devno);

#else
  //根据期望值分配设备号
  devno=MKDEV(dev_major, dev_minor);
  ret=register_chrdev_region(devno, 1, "my74hc595");
#endif

  if(ret<0)
  {
    printk("my74hc595 register failure\n");
    //my74hc595_exit(); //如果注册设备号失败就退出系统
    return ret;
  }
  else
  {
    printk("my74hc595 register success\n");
  }

  gp_testdata = kmalloc(sizeof(int), GFP_KERNEL);
#if 0//两种初始化字符设备信息的方法
  my74hc595_cdev = cdev_alloc();//调试时,此中方法在rmmod后会出现异常,原因未知
  my74hc595_cdev->ops = &my74hc595_fops;
#else
  my74hc595_cdev = kmalloc(sizeof(struct cdev), GFP_KERNEL);
  cdev_init(my74hc595_cdev, &my74hc595_fops);
#endif

  my74hc595_cdev->owner = THIS_MODULE; //初始化cdev中的所有者字段

  err=cdev_add(my74hc595_cdev, devno, 1); //向内核添加这个cdev结构的信息
  if(err<0)
    printk("add device failure\n"); //如果添加失败打印错误消息

  firstdrv_class = class_create(THIS_MODULE, "my74hc595");
  firstdrv_class_dev = device_create(firstdrv_class, NULL, MKDEV(dev_major, 0), NULL,"my74hc595-%d", 0);//在/dev下创建节点

  printk("register my74hc595 dev OK\n");

  return 0;
}
//打开设备文件系统调用对应的操作
int my74hc595_open(struct inode *inode, struct file *filp)
{
  //将file结构中的private_data字段指向已分配的设备结构
  filp->private_data = gp_testdata;
  printk("open my74hc595 dev OK\n");
  return 0;
}
//关闭设备文件系统调用对应的操作
int my74hc595_release(struct inode *inode, struct file *filp)
{
  printk("close my74hc595 dev OK\n");
  return 0;
}
//读设备文件系统调用对应的操作
ssize_t my74hc595_read(struct file *filp, char *buf, size_t len, loff_t *off)
{
  //获取指向已分配数据的指针
  unsigned int *p_testdata = filp->private_data;
  //将设备变量值复制到用户空间
  if(copy_to_user(buf, p_testdata, sizeof(int)))
  {
    return -EFAULT;
  }
  printk("read my74hc595 dev OK\n");
  return sizeof(int); //返回读取数据的大小
}
//写设备文件系统调用对应的操作
ssize_t my74hc595_write(struct file *filp, const char *buf, size_t len, loff_t *off)
{
  //获取指向已分配数据的指针
  unsigned int *p_testdata = filp->private_data;
  //从用户空间复制数据到内核中的设备变量
  if(copy_from_user(p_testdata, buf, sizeof(int)))
  {
    return -EFAULT;
  }
  printk("write my74hc595 dev OK\n");
  return sizeof(int); //返回写数据的大小
}

module_init(my74hc595_init); //模块被装载时调用my74hc595_init
module_exit(my74hc595_exit); //模块被卸载时调用my74hc595_exit

第二步:Makefile

按如下内容编写一个Makefile文件,然后输入make就可以开始自动编译了。编译之后得到了一个名为my74hc595.ko的模块文件,这就是我们需要的设备驱动文件。

#Makefile
CROSS_COMPILE=arm-linux-
ARCH:=arm
CC:=$(CROSS_COMPILE)gcc
LD:=$(CROSS_COMPILE)ld
obj-m = my74hc595.o
module-objs = my74hc595.o

KDIR = /home/zhang/at91/linux-at91
PWD = $(shell pwd)
default:
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules ARCH=arm CROSS_COMPILE=arm-linux-
clean:
rm -rf *.o *.ko *~

接下来运行如下代码,将驱动加入内核。

insmod my74hc595.ko

将自动在/dev目录下创建设备节点

rmmod my74hc595.ko

将自动删除节点

第三步:CharDevTest.c

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <fcntl.h>

main()
{
  int fd, num;
  // fd=open("/dev/my74hc595", O_RDWR, S_IRUSR|S_IWUSR); //可读写方式打开设备文件
  fd=open("/dev/my74hc595-0", O_RDWR); //可读写方式打开设备文件
  if(fd!=-1)
  {
    read(fd, &num, sizeof(int)); //读取设备变量
    printf("The my74hc595 is %d\n", num);

    printf("Please input the num written to my74hc595\n");
    scanf("%d", &num);
    write(fd, &num, sizeof(int)); //写设备变量

    read(fd, &num, sizeof(int)); //再次读取刚才写的值
    printf("The my74hc595 is %d\n", num);

    close(fd); //关闭设备文件
  }
  else
  {
    printf("Device open failure\n");
    perror("open my74hc595");
  }
}

转载于:https://www.cnblogs.com/Charles-Zhang-Blog/p/3454382.html

Linux_2.6字符设备驱动实例相关推荐

  1. Linux字符设备驱动实例

    globalmem 看 linux 设备驱动开发详解时,字符设备驱动一章,写的测试代码和应用程序,加上自己的操作,对初学者我觉得非常有帮助. 写这篇文章的原因是因为我看了我之前发表的文章,还没有写过字 ...

  2. 深入浅出:Linux设备驱动之字符设备驱动

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

  3. linux注册函数机制,Linux可信计算机制模块详细分析之函数实现机制(1)字符设备驱动...

    原标题:Linux可信计算机制模块详细分析之函数实现机制(1)字符设备驱动 2.3 函数实现机制 2.3.1 Linux 字符设备驱动 在linux 3.5.4中,用结构体cdev描述字符设备,cde ...

  4. 【Linux驱动】字符设备驱动

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

  5. Linux字符设备驱动框架

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

  6. 设备驱动,字符设备驱动、(总线)设备驱动模型、sysfs文件系统、平台设备驱动

    以下内容转载于微信公众号:嵌入式企鹅圈.如有侵权,请告知删除. 学习Linux设备驱动开发的过程中自然会遇到字符设备驱动.平台设备驱动.设备驱动模型和sysfs等相关概念和技术. 对于初学者来说会非常 ...

  7. 蜕变成蝶~Linux设备驱动之字符设备驱动

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

  8. linux PCI驱动调用字符设备驱动方式

    上一篇文章写了字符设备驱动的基本结构及访问方式,在实际应用时首先需要绑定自己的硬件设备.本篇主要描述字符设备驱动与PCI接口类型的设备访问方式(内核为2.6.24及以上的方法,测试内核为2.6.32) ...

  9. char添加一个字符_LINUX字符设备驱动模型分析(起始篇)

    在前面几个模块的介绍中,我们主要以vfs为起始,完成了sysfs.设备-总线-驱动模型.platform设备驱动模型.i2c设备驱动模型.spi设备驱动模型的分析.在对这些模块进行分析的时候,我们或多 ...

最新文章

  1. Android 依赖库发布(上传 Library 到 JCenter)gradle最高支持4.4
  2. java中的匿名内部类
  3. iOS的相对路径和绝对路径
  4. arcgis Server Object Extension (SOE)部署jar文件失败 原因探索
  5. [转]软件测试演义——中高级系列(序)
  6. 简易版Dubbo方法级性能监控(实现TP90、TP99)
  7. wgs84坐标系转换工具_ArcGIS中不同坐标系之间的转换
  8. 【kafka】google提供的一些好用的并发工具类
  9. vlc_for_android(基于git-3.0.0)快速集成并播放电视节目直播
  10. Go基础-go语言的编码规范
  11. mac airdrop 隔空投送 我可以发现别人,别人发现不了我。搜索不到。
  12. 1.2 XML 的基本语法
  13. 一筐鸡蛋 1个1个拿……弱智题
  14. 用迅雷下载百度网盘的文件
  15. PDPS汉化包安装与语言界面切换操作
  16. rancher2.6部署k8s集群示例
  17. 小米android通知栏提示怎么关闭,怎么取消小米音乐自动在通知栏中显示控制开?...
  18. EF框架-SQL语句 增加 删除
  19. QQ自定义音乐卡片代码
  20. 什么?到现在你还不知道什么是 访问修饰限定符吗?

热门文章

  1. JAVA基础----java中E,T,?的区别
  2. 为什么移动硬盘的实际容量比标注容量小
  3. MySQL主从数据库配置
  4. 为文档快速插入页眉和页脚
  5. centos配置ftp
  6. 工作失职的处理决定_工作失职的处理决定
  7. leetcode 1473. 粉刷房子 III(dp)
  8. github pages_如何使用GitHub Actions和Pages发布GitHub事件数据
  9. freecodecamp_freeCodeCamp的服务器到底发生了什么?
  10. ios 应用商店_如何在预算范围内制作值得应用商店使用的iOS应用预览