当前位置: 首页 > news >正文

长沙零基础学快速建站南通网站建设系统方案

长沙零基础学快速建站,南通网站建设系统方案,公司建网站多少钱一年,龙岗网站开发公司register_chrdev 和 unregister_chrdev 这两个函数是老版本驱动使用的函数#xff0c;现在新的字符设备驱动已经不再使用这两个函数#xff0c;而是使用Linux内核推荐的新字符设备驱动API函数。新字符设别驱动API函数在驱动模块加载的时候自动创建设备节点文件。 分配和释放…register_chrdev 和 unregister_chrdev 这两个函数是老版本驱动使用的函数现在新的字符设备驱动已经不再使用这两个函数而是使用Linux内核推荐的新字符设备驱动API函数。新字符设别驱动API函数在驱动模块加载的时候自动创建设备节点文件。 分配和释放设备号 使用 register_chrdev 函数注册字符设备的时候只需要给定一个主设备号即可但是这样会带来两个问题 ①、需要我们事先确定好哪些主设备号没有使用。 ②、会将一个主设备号下的所有次设备号都使用掉 如果没有指定设备号的话就使用如下函数来申请设备号 int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)如果给定了设备的主设备号和次设备号就使用如下所示函数来注册设备号即可 int register_chrdev_region(dev_t from, unsigned count, const char *name)参数 from 是要申请的起始设备号也就是给定的设备号参数 count 是要申请的数量一般都是一个参数 name 是设备名字。 注 销 字 符 设 备 之 后 要 释 放 掉 设 备 号 不 管 是 通 过 alloc_chrdev_region 函 数 还 是register_chrdev_region 函数申请的设备号统一使用如下释放函数 void unregister_chrdev_region(dev_t from, unsigned count)新字符设备驱动下设备号分配示例代码如下 示例代码 42.1.1.1 新字符设备驱动下设备号分配 1 int major; /* 主设备号 */ 2 int minor; /* 次设备号 */ 3 dev_t devid; /* 设备号 */ 4 5 if (major) { /* 定义了主设备号 */ 6 devid MKDEV(major, 0); /* 大部分驱动次设备号都选择 0*/ 7 register_chrdev_region(devid, 1, test); 8 } else { /* 没有定义设备号 */ 9 alloc_chrdev_region(devid, 0, 1, test); /* 申请设备号 */ 10 major MAJOR(devid); /* 获取分配号的主设备号 */ 11 minor MINOR(devid); /* 获取分配号的次设备号 */ 12 }注销设备号的话使用如下代码即可 示例代码 42.1.1.2 cdev 结构体 1 unregister_chrdev_region(devid, 1); /* 注销设备号 */新的字符设备注册方法 1、字符设备结构 在 Linux 中使用 cdev 结构体表示一个字符设备 cdev 结构体在 include/linux/cdev.h 文件中的定义如下 示例代码 42.1.2.1 cdev 结构体 1 struct cdev { 2 struct kobject kobj; 3 struct module *owner; 4 const struct file_operations *ops; 5 struct list_head list; 6 dev_t dev; 7 unsigned int count; 8 };ops 和 dev这两个就是字符设备文件操作函数集合file_operations 以及设备号 dev_t。编写字符设备驱动之前需要定义一个 cdev 结构体变量这个变量就表示一个字符设备如下所示struct cdev test_cdev; 2、 cdev_init 函数 cdev 变量以后就要使用 cdev_init 函数对其进行初始化 cdev_init 函数原型如下 void cdev_init(struct cdev *cdev, const struct file_operations *fops)参数 cdev 就是要初始化的 cdev 结构体变量参数 fops 就是字符设备文件操作函数集合。使用 cdev_init 函数初始化 cdev 变量的示例代码如下 示例代码 42.1.2.2 cdev_init 函数使用示例代码 1 struct cdev testcdev; 2 3 /* 设备操作函数 */ 4 static struct file_operations test_fops { 5 .owner THIS_MODULE, 6 /* 其他具体的初始项 */ 7 }; 8 9 testcdev.owner THIS_MODULE; 10 cdev_init(testcdev, test_fops); /* 初始化 cdev 结构体变量 */3、 cdev_add 函数 cdev_add 函数用于向 Linux 系统添加字符设备(cdev 结构体变量)首先使用 cdev_init 函数完成对 cdev 结构体变量的初始化然后使用 cdev_add 函数向 Linux 系统添加这个字符设备。 cdev_add 函数原型如下 int cdev_add(struct cdev *p, dev_t dev, unsigned count)参数 p 指向要添加的字符设备(cdev 结构体变量)参数 dev 就是设备所使用的设备号参数count 是要添加的设备数量。完善示例代码 42.1.2.2加入 cdev_add 函数内容如下所示 示例代码 42.1.2.2 cdev_add 函数使用示例 struct cdev testcdev; /* 设备操作函数 */ static struct file_operations test_fops { .owner THIS_MODULE, /* 其他具体的初始项 */ }; testcdev.owner THIS_MODULE; cdev_init(testcdev, test_fops); /* 初始化 cdev 结构体变量 */ cdev_add(testcdev, devid, 1); /* 添加字符设备 */示例代码 42.1.2.2 就是新的注册字符设备代码段 Linux 内核中大量的字符设备驱动都是采用这种方法向 Linux 内核添加字符设备。如果在加上示例代码 42.1.1.1 中分配设备号的程序那么就它们一起实现的就是函数 register_chrdev 的功能。 4、 cdev_del 函数 卸载驱动的时候一定要使用 cdev_del 函数从 Linux 内核中删除相应的字符设备 cdev_del 函数原型如下 void cdev_del(struct cdev *p)参数 p 就是要删除的字符设备。如果要删除字符设备参考如下代码 示例代码 42.1.2.3 cdev_del 函数使用示例 1 cdev_del(testcdev); /* 删除 cdev */cdev_del 和 unregister_chrdev_region 这两个函数合起来的功能相当于 unregister_chrdev 函 数。 上面了解了新的字符设备注册方法。接下来要了解下自动创建设备节点。 自动创建设备节点 在前面的 Linux 驱动实验中当我们使用 modprobe 加载驱动程序以后还需要使用命令“mknod”手动创建设备节点。本节就来讲解一下如何实现自动创建设备节点在驱动中实现自动创建设备节点的功能以后使用 modprobe 加载驱动模块成功的话就会自动在/dev 目录下创建对应的设备文件。 mdev 机制 udev 是一个用户程序在 Linux 下通过 udev 来实现设备文件的创建与删除 udev 可以检测系统中硬件设备状态可以根据系统中硬件设备状态来创建或者删除设备文件。比如使用modprobe 命令成功加载驱动模块以后就自动在/dev 目录下创建对应的设备节点文件,使用 rmmod 命令卸载驱动模块以后就删除掉/dev 目录下的设备节点文件。 使用 busybox 构建根文件系统的时候 busybox 会创建一个 udev 的简化版本—mdev。 创建和删除类 自动创建设备节点的工作是在驱动程序的入口函数中完成的一般在 cdev_add 函数后面添加自动创建设备节点相关代码。首先要创建一个 class 类 class 是个结构体定义在文件include/linux/device.h 里面。 class_create 是类创建函数 class_create 是个宏定义内容如下 示例代码 42.2.1.1 class_create 函数 1 #define class_create(owner, name) \ 2 ({ \ 3 static struct lock_class_key __key; \ 4 __class_create(owner, name, __key); \ 5 }) 6 7 struct class *__class_create(struct module *owner, const char *name, 8 struct lock_class_key *key)根据上述代码将宏 class_create 展开以后内容如下 struct class *class_create (struct module *owner, const char *name)class_create 一共有两个参数参数 owner 一般为 THIS_MODULE参数 name 是类名字。返回值是个指向结构体 class 的指针也就是创建的类。 卸载驱动程序的时候需要删除掉类类删除函数为 class_destroy函数原型如下 void class_destroy(struct class *cls);参数 cls 就是要删除的类。 创建设备 建好类以后还不能实现自动创建设备节点我们还需要在这个类下创建一个设备。使用 device_create 函数在类下面创建设备 device_create 函数原型如下 struct device *device_create(struct class *class, struct device *parent, dev_t devt, void *drvdata, const char *fmt, ...)device_create 是个可变参数函数参数 class 就是设备要创建哪个类下面参数 parent 是父设备一般为 NULL也就是没有父设备参数 devt 是设备号参数 drvdata 是设备可能会使用的一些数据一般为NULL参数 fmt 是设备名字如果设置 fmtxxx 的话就会生成/dev/xxx这个设备文件。返回值就是创建好的设备。 卸载驱动的时候需要删除掉创建的设备设备删除函数为 device_destroy函数原型如下 void device_destroy(struct class *class, dev_t devt)参数 class 是要删除的设备所处的类参数 devt 是要删除的设备号。 设置文件私有数据 每个硬件设备都有一些属性比如主设备号(dev_t)类(class)、设备(device)、开关状态(state) 等等在编写驱动的时候你可以将这些属性全部写成变量的形式如下所示 示例代码 42.3.1 变量形式的设备属性 dev_t devid; /* 设备号 */ struct cdev cdev; /* cdev */ struct class *class; /* 类 */ struct device *device; /* 设备 */ int major; /* 主设备号 */ int minor; /* 次设备号 */编写驱动 open 函数的时候将设备结构体作为私有数据添加到设备文件中如下所示 示例代码 42.3.2 设备结构体作为私有数据 /* 设备结构体 */ 1 struct test_dev{ 2 dev_t devid; /* 设备号 */ 3 struct cdev cdev; /* cdev */ 4 struct class *class; /* 类 */ 5 struct device *device; /* 设备 */ 6 int major; /* 主设备号 */ 7 int minor; /* 次设备号 */ 8 }; 9 10 struct test_dev testdev; 11 12 /* open 函数 */ 13 static int test_open(struct inode *inode, struct file *filp) 14 { 15 filp-private_data testdev; /* 设置私有数据 */ 16 return 0; 17 }LED 灯驱动程序编写 这里先定好设备数量以及设别名字具体如下 #define NEWCHRLED_CNT 1 /*设备号个数*/ #define NEWCHRLED_NAME newchrled接下来定义该设备中的结构体并设置为私有数据其中含有devid设备号cdev字符设备注册结构体自动创建设备节点用到的类class以及创建类后还需要创建设备才能完成自动创建设备节点的功能。所以结构体要包含这些内容。 /* newchrled 设备结构体 */ struct newchrled_dev {dev_t devid;struct cdev cdev;struct class *class;struct device *device;int major;int minor; };总的程序具体如下 #include linux/types.h #include linux/kernel.h #include linux/delay.h #include linux/ide.h #include linux/init.h #include linux/module.h #include linux/errno.h #include linux/gpio.h #include linux/cdev.h #include linux/device.h #include asm/mach/map.h #include asm/uaccess.h #include asm/io.h #define NEWCHRLED_CNT 1 /*设备号个数*/ #define NEWCHRLED_NAME newchrled #define LEDOFF 0 #define LEDON 1 /* 寄存器物理地址 */ #define CCM_CCGR1_BASE (0X020C406C) #define SW_MUX_GPIO1_IO03_BASE (0X020E0068) #define SW_PAD_GPIO1_IO03_BASE (0X020E02F4) #define GPIO1_DR_BASE (0X0209C000) #define GPIO1_GDIR_BASE (0X0209C004)/* 映射后的寄存器虚拟地址指针 */ static void __iomem *IMX6U_CCM_CCGR1; static void __iomem *SW_MUX_GPIO1_IO03; static void __iomem *SW_PAD_GPIO1_IO03; static void __iomem *GPIO1_DR; static void __iomem *GPIO1_GDIR;/* newchrled 设备结构体 */ struct newchrled_dev {dev_t devid;struct cdev cdev;struct class *class;struct device *device;int major;int minor; }; struct newchrled_dev newchrled;void LED_Switches(u8 state) {u32 retval 0;if (state LEDON){retval readl(GPIO1_DR);retval ~(1 3);writel(retval, GPIO1_DR);}else if (state LEDOFF){retval readl(GPIO1_DR);retval | (1 3);writel(retval, GPIO1_DR);} } /** description : 打开设备* param – inode : 传递给驱动的 inode* param - filp : 设备文件 file 结构体有个叫做 private_data 的成员变量* 一般在 open 的时候将 private_data 指向设备结构体。* return : 0 成功;其他 失败*/ static int ledopen(struct inode *inode, struct file *filp) {filp-private_data newchrled;return 0; } /** description : 从设备读取数据* param - filp : 要打开的设备文件(文件描述符)* param - buf : 返回给用户空间的数据缓冲区* param - cnt : 要读取的数据长度* param - offt : 相对于文件首地址的偏移* return : 读取的字节数如果为负值表示读取失败*/ static int ledread(struct file *filp, char __user *buf, size_t cnt, loff_t *offt) {return 0; } /** description : 向设备写数据* param - filp : 设备文件表示打开的文件描述符* param - buf : 要写给设备写入的数据* param - cnt : 要写入的数据长度* param - offt : 相对于文件首地址的偏移* return : 写入的字节数如果为负值表示写入失败*/ static ssize_t ledwrite(struct file *filp, char __user *buf, size_t cnt, loff_t off_t) {int retvalue 0;unsigned char databuf[1];u8 ledstat;retvalue copy_from_user(databuf, buf, cnt);if (retvalue 0){printk(kernel write failed!\r\n);return -EFAULT;}ledstat databuf[0];if (ledstat LEDON){LED_Switches(LEDON);}else if (ledstat LEDOFF){LED_Switches(LEDOFF);}return 0; } /** description : 关闭/释放设备* param – filp : 要关闭的设备文件(文件描述符)* return : 0 成功;其他 失败*/ static int led_release(struct inode *inode, struct file *filp) {return 0; } static struct file_operations newchrled_fops {.owner THIS_MODULE,.open ledopen,.read ledread,.write ledwrite,.release led_release, }; static int __init led_init(void) {int retvalue 0;u32 val 0;/* 初始化 LED *//* 1、寄存器地址映射 */IMX6U_CCM_CCGR1 ioremap(CCM_CCGR1_BASE, 4);SW_MUX_GPIO1_IO03 ioremap(SW_MUX_GPIO1_IO03_BASE, 4);SW_PAD_GPIO1_IO03 ioremap(SW_PAD_GPIO1_IO03_BASE, 4);GPIO1_DR ioremap(GPIO1_DR_BASE, 4);GPIO1_GDIR ioremap(GPIO1_GDIR_BASE, 4);/* 2、使能 GPIO1 时钟 */val readl(IMX6U_CCM_CCGR1);val ~(3 26); /* 清除以前的设置 */val | (3 26); /* 设置新值 */writel(val, IMX6U_CCM_CCGR1);/* 3、设置 GPIO1_IO03 的复用功能将其复用为* GPIO1_IO03最后设置 IO 属性。*/writel(5, SW_MUX_GPIO1_IO03);/* 寄存器 SW_PAD_GPIO1_IO03 设置 IO 属性 */writel(0x10B0, SW_PAD_GPIO1_IO03);/* 4、设置 GPIO1_IO03 为输出功能 */val readl(GPIO1_GDIR);val ~(1 3); /* 清除以前的设置 */val | (1 3); /* 设置为输出 */writel(val, GPIO1_GDIR);/* 5、默认关闭 LED */val readl(GPIO1_DR);val | (1 3);writel(val, GPIO1_DR);/* 6、注册字符设备驱动 *//*1、创建设备号*/if (newchrled.major){newchrled.devid MKDEV(newchrled.major, 0);register_chrdev_region(newchrled.devid, NEWCHRLED_CNT, NEWCHRLED_NAME);}else{alloc_chrdev_region(newchrled.devid, 0, NEWCHRLED_CNT, NEWCHRLED_NAME);newchrled.major MAJOR(newchrled.devid);newchrled.minor MINOR(newchrled.devid);}printk(newcheled major: %d minor: %d, newchrled.major, newchrled.minor);newchrled.cdev.owner THIS_MODULE;cdev_init(newchrled.cdev, newchrled_fops);cdev_add(newchrled.cdev, newchrled.devid, NEWCHRLED_CNT);newchrled.class class_create(THIS_MODULE, NEWCHRLED_NAME);if (IS_ERR(newchrled.class)){return PTR_ERR(newchrled.class);}newchrled.device device_create(newchrled.class, NULL, newchrled.devid, NULL, NEWCHRLED_NAME);if (IS_ERR(newchrled.device)){return PTR_ERR(newchrled.device);}return 0; } static void __exit led_exit(void) {/* 取消映射 */iounmap(IMX6U_CCM_CCGR1);iounmap(SW_MUX_GPIO1_IO03);iounmap(SW_PAD_GPIO1_IO03);iounmap(GPIO1_DR);iounmap(GPIO1_GDIR);/* 注销字符设备驱动 */cdev_del(newchrled.cdev); /* 删除 cdev */unregister_chrdev_region(newchrled.devid, NEWCHRLED_CNT);device_destroy(newchrled.class, newchrled.devid);class_destroy(newchrled.class); } module_init(led_init); module_exit(led_exit); MODULE_LICENSE(GPL); MODULE_AUTHOR(wyw);编写测试APP上一节类似参考上一节测试也是一样的。只不过这里只需要输入命令进入到目录 lib/modules/4.1.15 中输入如下命令加载 newchrled.ko 驱动模块 depmod //第一次加载驱动的时候需要运行此命令modprobe newchrled.ko //加载驱动 /ledApp /dev/newchrled 1 //打开 LED 灯 输入上述命令以后观察 I.MX6U-ALPHA 开发板上的红色 LED 灯是否点亮如果点亮的话 说明驱动工作正常。在输入如下命令关闭 LED 灯 ./ledApp /dev/newchrled 0 //关闭 LED 灯
http://www.w-s-a.com/news/775308/

相关文章:

  • 网站建设合约品牌设计有哪些
  • 织梦企业门户网站宝塔搭建wordpress网站
  • 网站为什么没有排名了11月将现新冠感染高峰
  • 网站开发维护专员岗位职责辽阳企业网站建设
  • 做外国订单有什么网站网站设计论文提纲
  • 商城网站建设报价方案导师让做网站
  • 清远市企业网站seo联系方式动易官方网站
  • 手机上怎么做能打开的网站一级域名和二级域名跨域
  • 网站首页效果图wordpress 在线教育
  • 电商网站开发团队广西桂林商贸旅游技工学校
  • 网站模板文件怎么下载东莞常平镇邮政编码
  • 建网站需要什么wordpress误删的后果
  • wordpress无插件实现网站地图做阿里巴巴网站店铺装修费用
  • 英文互动网站建设南宁住房和城乡建设局网站
  • 威海微网站建设乐清建网站哪家强
  • 网站和app的开发成本saas系统开发教程
  • ps切片工具做网站大气简洁网站
  • 网至普的营销型网站建设wordpress邮箱验证插件下载
  • 找权重高的网站方法张家港早晨网站建设
  • WordPress数据库添加管理员关键词优化举例
  • 河南国基建设集团--官方网站wordpress qode
  • 做农村电子商务的网站有哪些内容静态网站模板古典
  • 导航网站设计方案个人网站推广方法
  • 网站排名易下拉教程防wordpress花园
  • 计算机网站建设 是什么意思现在网站建站的主流语言是什么
  • php网站跟随导航西安百姓网免费发布信息网
  • 濮阳做公司网站html5 特效网站
  • ppt设计器怎么打开深圳seo网络推广营销
  • 建设银行网站用360浏览器建设信用卡中心网站
  • 创建公司网站 教程广州建设局