Linux字符设备-内核态数据与用户态数据互传
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux字符设备-内核态数据与用户态数据互传相关的知识,希望对你有一定的参考价值。
Linux字符设备-内核态数据与用户态数据互传
对于系统支持设备的ioctl号,你可以在/usr/include下面的头文件中找到,对于你自己的设备,如果需要使用ioctl接口,则需要定义自己 的ioctl号。以前的2.4中有个问题是,大家都随便定义自己的ioctl号,造成很大可能性的重复性。一个坏处是难以管理,另外一个是容易造成错误, 例如如果用户本来希望打开一个串口设备,结果通过open打开了网口,如果串口的某个ioctl号正好是网口的关闭操作,这样就会造成错误。在2.6里 面,你定义自己的ioctl号最好使用_IO, _IOR, _IOW和_IORW来定义,这些宏考虑了第三个参数的长度,设备的magic number,以及操作的方向等,避免了2.4中的问题
: _IO(type,nr)(给没有参数的命令),
_IOR(type, nre, datatype)(给从驱动中读数据的),
_IOW(type,nr,datatype)(给写数据),
_IOWR(type,nr,datatype)(给双向传送).
type 和 number 成员作为参数被传递,
并且 size 成员通过应用 sizeof 到 datatype 参数而得到
int ioctl( int fd, int request, .../* void *arg */ ) 详解
第三个参数总是一个指针,但指针的类型依赖于request 参数。我们可以把和网络相关的请求划分为6 类:
套接口操作
文件操作
接口操作
ARP 高速缓存操作
路由表操作
流系统
先写一个内核态数据与用户态数据互传的例子及APP
手动安装步骤:
Insmod my_char_dev.ko
不需要再安装设备节点
然后是测试app
./my_char_dev_app 1
1 #include <linux/module.h> 2 #include <linux/init.h> 3 #include <linux/io.h> 4 #include <linux/fs.h> 5 #include <asm/device.h> //下面这三个头文件是由于动态创建需要加的 6 #include <linux/device.h> 7 #include <linux/cdev.h> 8 #include "my_cdev.h" 9 #include <asm/uaccess.h> 10 11 12 struct cdev cdev; 13 dev_t devno;//这里是动态分配设备号和动态创建设备结点需要用到的 14 struct class *cdev_class; 15 int param; 16 int my_cdev_open(struct inode *node,struct file *filp) 17 { 18 printk("my_cdev_open sucess!\n"); 19 return 0; 20 } 21 22 long my_cdev_ioctl(struct file *filp ,unsigned int cmd ,unsigned long arg) 23 { 24 int rc = -1; 25 26 switch(cmd) 27 { 28 case LED_ON: 29 rc = copy_from_user(¶m, (int __user*)arg, 4); 30 if (0 != rc) 31 { 32 printk("copy_from_user failed.\n"); 33 break; 34 } 35 printk("Param is %d.\n",param); 36 printk("CMD test: LED_ON is set!\n"); 37 return 0; 38 case LED_OFF: 39 printk("CMD test: LED_OFF is set!\n"); 40 param = 1000; 41 rc = copy_to_user((int __user*)arg,¶m,4); 42 if (0 != rc) 43 { 44 printk("copy_to_user failed.\n"); 45 } 46 param = 0; 47 return 0; 48 default : 49 return -EINVAL; 50 } 51 } 52 53 struct file_operations my_cdev_fops= 54 { 55 .open = my_cdev_open, 56 .unlocked_ioctl = my_cdev_ioctl, 57 58 }; 59 60 static int my_cdev_init(void) 61 { 62 int ret; 63 /**动态分配设备号*/ 64 ret = alloc_chrdev_region(&devno,0,1,"my_chardev"); 65 if(ret) 66 { 67 printk("alloc_chrdev_region fail!\n"); 68 unregister_chrdev_region(devno,1); 69 return ret; 70 } 71 else 72 { 73 printk("alloc_chrdev_region sucess!\n"); 74 } 75 /**描述结构初始化*/ 76 cdev_init(&cdev,&my_cdev_fops); 77 /**描述结构注册*/ 78 ret = cdev_add(&cdev,devno,1); 79 if(ret) 80 { 81 printk("cdev add fail.\n"); 82 unregister_chrdev_region(devno,1); 83 return ret; 84 } 85 else 86 { 87 printk("cdev add sucess!\n"); 88 } 89 90 cdev_class = class_create(THIS_MODULE,"my_chardev"); 91 if(IS_ERR(cdev_class)) 92 { 93 printk("Create class fail!\n"); 94 unregister_chrdev_region(devno,1); 95 return -1; 96 } 97 else 98 { 99 printk("Create class sucess!\n"); 100 } 101 102 device_create(cdev_class,NULL,devno,0,"my_chardev"); 103 104 return 0; 105 } 106 static void my_cdev_exit(void) 107 { 108 device_destroy(cdev_class,devno); 109 class_destroy(cdev_class); 110 cdev_del(&cdev); 111 unregister_chrdev_region(devno,1); 112 printk("my_cdev_exit sucess!\n"); 113 } 114 module_init(my_cdev_init); 115 module_exit(my_cdev_exit); 116 MODULE_LICENSE("GPL"); 117 MODULE_AUTHOR("YEFEI"); 118 MODULE_DESCRIPTION("YEFEI Driver");
1 #ifndef __MY_CDEV_H__ 2 #define __MY_CDEV_H__ 3 4 #define LED_MAGIC ‘L‘ 5 #define LED_ON _IOW(LED_MAGIC,0,int) 6 #define LED_OFF _IOR(LED_MAGIC,1,int *) 7 8 #endif
1 #include <sys/stat.h> 2 #include <sys/types.h> 3 #include <sys/ioctl.h> 4 #include <fcntl.h> 5 #include <stdio.h> 6 #include "my_cdev.h" 7 8 int main(int argc,char *argv[]) 9 { 10 int fd; 11 int cmd; 12 long ret = 0; 13 unsigned long param = 0; 14 if(argc < 2) 15 { 16 printf("Please enter secend param!\n"); 17 return 0; 18 } 19 cmd = atoi(argv[1]); 20 fd = open("/dev/my_chardev",O_RDWR); 21 if(fd < 0) 22 { 23 printf("Open dev/my_chardev fail!\n"); 24 close(fd); 25 return 0; 26 } 27 switch(cmd) 28 { 29 case 1: 30 param = 500; 31 ret = ioctl(fd,LED_ON,¶m); 32 break; 33 case 2: 34 ret = ioctl(fd,LED_OFF,¶m); 35 printf("ret is %d. read data is %d.\n",ret,param); 36 break; 37 default: 38 break; 39 } 40 close(fd); 41 return 0; 42 }
1 obj-m := my_char_dev.o 2 KDIR := /home/win/dn377org/trunk/bcm7252/linux/ 3 all: 4 make -C $(KDIR) M=$(PWD) modules CROSS_COMPILE=arm-linux- ARCH=arm 5 clean: 6 rm -f *.ko *.o *.mod.o *.mod.c *.symvers *.bak *.order
以上是关于Linux字符设备-内核态数据与用户态数据互传的主要内容,如果未能解决你的问题,请参考以下文章
Linux 内核Linux 内核体系架构 ( 硬件层面 | 内核空间 | 用户空间 | 内核态与用户态切换 | 系统调用 | 体系结构抽象层 )