/dev下添加设备节点的方法步骤(通过device_create)
将自己开发的内核代码加入到Linux内核中,需要3个步骤:
1、确定把自己开发代码放入到内核合适的位置
将demo_chardev.c文件拷贝到.../drivers/char/目录下。
demo_chardev.c
- #include <linux/init.h>
- #include <linux/module.h>
- #include <linux/kernel.h>
- /*结构体file_operations定义的头文件*/
- #include <linux/fs.h>
- /*声明copy_to/from_user函数的头文件*/
- #include <linux/uaccess.h>
- /*声明class_create 和device_create相关信息*/
- #include <linux/device.h>
- #define DEMO_DEBUG
- #ifdef DEMO_DEBUG
- #define dem_dbg(fmt, arg...) printk(KERN_WARNING fmt, ##arg)
- #else
- #define dem_dbg(fmt, arg...) printk(KERN_DEBUG fmt, ##arg)
- #endif
- #define DEVICE_COUNT 2
- /*记录当前驱动所占用的主设备号*/
- static int major = 0;
- static int demo_open (struct inode *pnode, struct file *filp)
- {
- dem_dbg("[kern func]: %s major: %d minor: %d\n",
- __FUNCTION__, imajor(pnode), iminor(pnode));
- return 0;
- }
- static ssize_t demo_read (struct file *filp, char __user *buf, size_t count, loff_t *offp)
- {
- unsigned char ary[100] = "you are reading successfully!";
- unsigned long len = min(count, sizeof(ary)); //min是个宏,用来获取两个数中较小的值
- int retval;
- dem_dbg("[kern func]: %s major: %d minor: %d\n",
- __FUNCTION__, imajor(filp->f_dentry->d_inode),
- iminor(filp->f_dentry->d_inode));
- //file结构体的f_flags成员可用来判断是否阻塞读取,然后进行相应处理
- if(copy_to_user(buf, ary, len) != 0){
- retval = -EFAULT;
- goto cp_err;
- }
- return len; //成功返回实际传输的字节数
- cp_err:
- return retval;
- }
- static ssize_t demo_write(struct file *filp, const char __user *buf, size_t count, loff_t *offp)
- {
- unsigned char ary[100] = "";
- unsigned long len = min(count, sizeof(ary)); //min是个宏,用来获取两个数中较小的值
- int retval;
- dem_dbg("[kern func]: %s major: %d minor: %d\n",
- __FUNCTION__, imajor(filp->f_dentry->d_inode),
- iminor(filp->f_dentry->d_inode));
- if(copy_from_user(ary, buf, len) != 0){
- retval = -EFAULT;
- goto cp_err;
- }
- printk("[msg]: writing context: %s\n",ary);
- return len; //成功返回实际传输的字节数
- cp_err:
- return retval;
- }
- static int demo_release (struct inode *pnode, struct file *filp)
- {
- dem_dbg("[kern func]: %s major: %d minor: %d\n",
- __FUNCTION__, imajor(pnode), iminor(pnode));
- return 0;
- }
- /*@定义file_operations结构体变量*/
- static struct file_operations fops = {
- .owner = THIS_MODULE,
- .read = demo_read,
- .write = demo_write,
- .open = demo_open,
- .release = demo_release,
- };
- static struct class *demo_class;
- static int __init drvdemo_init(void)
- {
- struct device *demo_device;
- int i;
- int retval;
- dem_dbg("[msg]:this is a driver demo, in module initial function\n");
- /*注册字符驱动函数,成功 返回动态分配好的主设备号,失败
- *返回错误码(负值)*/
- major = register_chrdev(0, "demo_chrdev", &fops);
- if(major < 0){
- retval = major;
- goto chrdev_err;
- }
- /*创建设备类*/
- demo_class = class_create(THIS_MODULE,"demo_class");
- if(IS_ERR(demo_class)){
- retval = PTR_ERR(demo_class);
- goto class_err;
- }
- /*创建设备文件,通知用户在“/dev/”目录下创件名字为demoX的设备文件*/
- for(i=0; i<DEVICE_COUNT; i++){ //最多可创建255个设备节点(register_chrdev函数会申请0-254范围的从设备号)
- demo_device = device_create(demo_class,NULL, MKDEV(major, i), NULL,"demo%d",i);
- if(IS_ERR(demo_device)){
- retval = PTR_ERR(demo_device);
- goto device_err;
- }
- }
- return 0;
- device_err:
- while(i--) //设备节点创建的回滚操作 device_destroy(demo_class,MKDEV(major, i));
- class_destroy(demo_class); //删除设备类
- class_err:
- unregister_chrdev(major, "demo_chrdev");
- chrdev_err:
- return retval;
- }
- static void __exit drvdemo_exit(void)
- {
- int i;
- dem_dbg("[msg]:in module exit function\n");
- /*注销字符驱动函数,无返回值,major为已分配的主设备号*/
- unregister_chrdev(major, "demo_chrdev");
- /*删除设备节点和设备类*/
- for(i=0; i<DEVICE_COUNT; i++)
- device_destroy(demo_class,MKDEV(major, i));
- class_destroy(demo_class);
- }
- module_init(drvdemo_init);
- module_exit(drvdemo_exit);
- MODULE_LICENSE("Dual BSD/GPL"); //BSD/GPL双重许可证
- MODULE_AUTHOR("hanbo"); //模块作者(可选)
- MODULE_DESCRIPTION("used for studing linux drivers"); //模块儿简介(可选)
2、把自己开发的功能增加到Linux内核的配置选项中,使用户能够选择此功能
vi drivers/char/Konfig 在文件结尾,endmenu的前面加入一个config选项
- config DEMO_CHARDEV
- bool "demo_chardev driver for hanbo chardev boards"
- default y
- help
- this is CHARDEV driver for hanbo chardev boards.
3、构建或修改Makefile,根据用户的选择,将相应的代码编译到最终生成的Linux内核中去
make menuconfig(添加配置选项)(如果提示找不到“ncurses”库则执行命令: sudo apt-get install libncurses5-dev )
Device driver -->
character devices ->
[*] demo_chardev driver for hanbo chardev boards
4、vi drivers/char/Makefile 添加内容如下:
..........
obj-$(CONFIG_DEMO_CHARDEV) +=demo_chardev.o (添加)
obj-$(CONFIG_JS_RTC) +=js-rtc.o(自带)
js-rtc-y = rtc.o (自带)
5、make (更新内核镜像到开发板)
6、交叉编译测试程序,放到开发板运行
arm-linux-gcc-gcc test.c -o demo
test.c
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <string.h>
- int main(int argc, char *argv[])
- {
- int fd1 = 0, fd2 = 0;
- unsigned char buf1[100] = "I am a test program!";
- unsigned char buf2[100] = {0};
- int retval;
- //以读写、不阻塞方式打开设备文件
- fd1 = open("/dev/demo0", O_RDWR | O_NONBLOCK);
- if(fd1 < 0){
- perror("open /dev/demo1");
- goto out;
- }
- //以只读、阻塞方式打开设备文件
- fd2 = open("/dev/demo1", O_RDONLY);
- if(fd2 < 0){
- perror("open /dev/demo2");
- goto out;
- }
- //成功返回实际写入字节数,失败返回负值
- retval = write(fd1, buf1, strlen(buf1)+1);
- if(retval < 0){
- perror("writing fd1 failed!");
- goto out;
- }
- printf("<user space>: write bytes: %d write content: %s\n", retval, buf1);
- //成功返回实际读取字节数,失败返回负值
- retval = read(fd2, buf2, sizeof(buf2));
- if(retval < 0){
- perror("reading fd2 failed!");
- goto out;
- }
- printf("<user space>: read bytes: %d read content: %s\n", retval, buf2);
- return 0;
- out:
- if(fd1 > 0)
- close(fd1);
- if(fd2 > 0)
- close(fd2);
- return -1;
- }
二、手动加载驱动 .ko文件
1、上面的demo_chardev.c文件放到内核下编译生成 .ko文件
Makefile
- #如果已定义KERNELRELEASE,说明是由内核构造系统调用的
- #可以利用内建语句
- ifneq ($(KERNELRELEASE),)
- obj-m +=demo_chrdev.o
- #此时由内核构造系统调用
- else
- #定义并记录内核源码路径
- KERNELDIR = /home/hanbo/linux-2.6.35.7(自己源码路径,2.6.35.7指当前内核版本)
- #记录当前工程目录
- PWD := $(shell pwd)
- default:
- $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
- @rm -rf *.o .t* .m* .*.cmd *.mod.c *.order *.symvers
- endif
- clean:
- rm -rf *.ko *.o .t* .m* .*.cmd *.mod.c *.order *.symvers
2、 然后用命令加载 .ko 驱动
lsmod 列举当前系统中的所有模块
lsmod 列举当前系统中的所有模块
rmmod xxx 卸载指定模块(不需要.ko后缀)
3、如果自己编译的代码中没有用
/*创建设备类*/
demo_class = class_create(THIS_MODULE,"demo_class");
/*创建设备文件,通知用户在“/dev/”目录下创件名字为demoX的设备文件*/
demo_device = device_create(demo_class,NULL, MKDEV(major, i), NULL,"demo%d",i);
则需要手动添加设备节点
mknod /dev/demo1 c 主设备号 0
mknod /dev/demo2 c 主设备号 1
注意:若卸载时出现提示 rmmod:chdir(2.6.35.7):No such file or directory
则在开发板根文件系统下创建目录:/lib/modules/2.6.35.7(跟当前内核版本同名)
/dev下添加设备节点的方法步骤(通过device_create)的更多相关文章
- Linux /dev 自动创建设备节点
#include <linux/module.h> #include <linux/module.h> #include <linux/kernel.h> #inc ...
- Android 6.0中在/dev下添加新设备驱动下Selinux相关设置【转】
本文转载自:https://blog.csdn.net/fantasy_wxe/article/details/52013922 错误1: 07-23 13:06:57.617 117 117 ...
- linux系统下添加新硬盘的方法详解
对于linux新手来说,在linux上添加新硬盘,是很有挑战性的一项工作. 在Linux服务器上把硬盘接好,启动linux,以root登陆. fdisk -l ## 这里是查看目前系统上有几块硬盘 D ...
- android源码framework下添加新资源的方法
编译带有资源的jar包,需要更改frameworks层,方法如下: 一.增加png类型的图片资源 1.将appupdate模块所有用到的png格式图片拷贝到framework/base/core/re ...
- WordPress 添加Meta Box的方法步骤
需要使用到add meta boxes Action,该Action允许我们为任何文章类型注册Meta Box,在该Action中,我们需要使用add_meta_box()方法来添加Meta Box的 ...
- LINUX下添加磁盘空间的方法详解
给Linux系统添加磁盘空间在工作会经常遇到. 在添加第二块磁盘一般系统默认为hdb(IDE硬盘)sdb(SCSI 硬盘),以hdb为例. linux-isep:~ # fdisk /dev/hdb ...
- eclipse下添加viplugin插件的方法
http://www.viplugin.com/ 在eclipse根目录下建立文件:viplugin2.lic,然后在里面添加以下字符串: nd4UFjUMBADcUSeSW8ocLKoGP3lpbW ...
- JS添加父节点的方法。
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 【总文档】rac增加新节点的方法步骤 How to Add Node/Instance or Remove Node/Instance in 10gR2, 11gR1, 11gR2 and 12c Oracle Clusterware and RAC
[总文档]How to Add Node/Instance or Remove Node/Instance in 10gR2, 11gR1, 11gR2 and 12c Oracle Clusterw ...
随机推荐
- php那些坑
1.创建数组不是new array(),是$aaa=array(),没有new,数组可以传入键值$aaa=array("key"=>"value"); 2 ...
- 在windows下使用Cygwin模拟unix环境,并安装apt-cyg,svn等工具
在windows下使用Cygwin模拟unix环境,并安装apt-cyg,svn等工具 一.Cygwin的安装 1. 下载Cygwin,这个可以到这里下载 ,至于使用32位的还是64位的版本可以根据自 ...
- excel打乱各行的顺序,实现无序随机排列
由于公司做活动,经常会发些激活码过来,为了让激活码能够充分使用,经常要打乱激活码的顺序,百度了下,看了下网上的介绍,还不错,挺实用,记录下来. 具体方法如下: 1.将文本里的内容复制到Excel里的任 ...
- 转 Linux命令-文件管理命令
http://jingyan.baidu.com/article/9113f81bc1c7a72b3214c7d3.html Linux命令-文件管理命令 浏览:4118 | 更新:2012-11-1 ...
- 接口自动化测试之HTTP协议详解
协议 简单理解,计算机与计算机之间的通讯语言就叫做协议,不同的计算机之间只有使用相同的协议才能通信.所以网络协议就是为计算机网络中进行数据交换而建立的规则,标准或约定的集合. OSI模型 1978年国 ...
- Struts2 与SpringMVC之比较
1.Struts2是类级别的拦截, 一个类对应一个request上下文,SpringMVC是方法级别的拦截,一个方法对应一个request上下文,而方法同时又跟一个url对应,所以说从架构本身上Spr ...
- java多线程异步执行
import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.ut ...
- Activiti Model Editor组件
通过Activiti Modeler架构图可知,Activiti Explorer采用的是Vaadin框架. Vaadin 是一种 Java Web 应用程序的开发框架, 其设计目标是便利地创建和维护 ...
- 揭秘jbpm流程引擎内核设计思想及构架
揭秘jbpm流程引擎内核设计思想及构架 作者 胡长城(银狐999) 1 前言 2 阅读本篇的基础准备 2.1 概念的基础 2.2 环境的基础 3 什么是 ...
- Go -- PipleLine
1.pipeline的产生 从一个现象说起,有一家咖啡吧生意特别好,每天来的客人络绎不绝,客人A来到柜台,客人B紧随其后,客人C排在客人B后面,客人D排在客人C后面,客人E排在客人D后面,一直排到店面 ...