歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux基礎 >> Linux教程 >> Linux字符設備-內核態數據與用戶態數據互傳

Linux字符設備-內核態數據與用戶態數據互傳

日期:2017/2/28 13:54:20   编辑:Linux教程

Linux字符設備-內核態數據與用戶態數據互傳

_IO,_IOR,_IOW和_IORW的含義

對於系統支持設備的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

#include <linux/module.h>
#include <linux/init.h>
#include <linux/io.h>
#include <linux/fs.h>
#include <asm/device.h> //下面這三個頭文件是由於動態創建需要加的
#include <linux/device.h>
#include <linux/cdev.h>
#include "my_cdev.h"
#include <asm/uaccess.h>


struct cdev cdev;
dev_t devno;//這裡是動態分配設備號和動態創建設備結點需要用到的
struct class *cdev_class;
int param;
int my_cdev_open(struct inode *node,struct file *filp)
{
printk("my_cdev_open sucess!\n");
return 0;
}

long my_cdev_ioctl(struct file *filp ,unsigned int cmd ,unsigned long arg)
{
int rc = -1;

switch(cmd)
{
case LED_ON:
rc = copy_from_user(&param, (int __user*)arg, 4);
if (0 != rc)
{
printk("copy_from_user failed.\n");
break;
}
printk("Param is %d.\n",param);
printk("CMD test: LED_ON is set!\n");
return 0;
case LED_OFF:
printk("CMD test: LED_OFF is set!\n");
param = 1000;
rc = copy_to_user((int __user*)arg,&param,4);
if (0 != rc)
{
printk("copy_to_user failed.\n");
}
param = 0;
return 0;
default :
return -EINVAL;
}
}

struct file_operations my_cdev_fops=
{
.open = my_cdev_open,
.unlocked_ioctl = my_cdev_ioctl,

};

static int my_cdev_init(void)
{
int ret;
/**動態分配設備號*/
ret = alloc_chrdev_region(&devno,0,1,"my_chardev");
if(ret)
{
printk("alloc_chrdev_region fail!\n");
unregister_chrdev_region(devno,1);
return ret;
}
else
{
printk("alloc_chrdev_region sucess!\n");
}
/**描述結構初始化*/
cdev_init(&cdev,&my_cdev_fops);
/**描述結構注冊*/
ret = cdev_add(&cdev,devno,1);
if(ret)
{
printk("cdev add fail.\n");
unregister_chrdev_region(devno,1);
return ret;
}
else
{
printk("cdev add sucess!\n");
}

cdev_class = class_create(THIS_MODULE,"my_chardev");
if(IS_ERR(cdev_class))
{
printk("Create class fail!\n");
unregister_chrdev_region(devno,1);
return -1;
}
else
{
printk("Create class sucess!\n");
}

device_create(cdev_class,NULL,devno,0,"my_chardev");

return 0;
}
static void my_cdev_exit(void)
{
device_destroy(cdev_class,devno);
class_destroy(cdev_class);
cdev_del(&cdev);
unregister_chrdev_region(devno,1);
printk("my_cdev_exit sucess!\n");
}
module_init(my_cdev_init);
module_exit(my_cdev_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("YEFEI");
MODULE_DESCRIPTION("YEFEI Driver");

--------------------------------------------

#ifndef __MY_CDEV_H__
#define __MY_CDEV_H__

#define LED_MAGIC 'L'
#define LED_ON _IOW(LED_MAGIC,0,int)
#define LED_OFF _IOR(LED_MAGIC,1,int *)

#endif

--------------------------------------------

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

int main(int argc,char *argv[])
{
int fd;
int cmd;
long ret = 0;
unsigned long param = 0;
if(argc < 2)
{
printf("Please enter secend param!\n");
return 0;
}
cmd = atoi(argv[1]);
fd = open("/dev/my_chardev",O_RDWR);
if(fd < 0)
{
printf("Open dev/my_chardev fail!\n");
close(fd);
return 0;
}
switch(cmd)
{
case 1:
param = 500;
ret = ioctl(fd,LED_ON,&param);
break;
case 2:
ret = ioctl(fd,LED_OFF,&param);
printf("ret is %d. read data is %d.\n",ret,param);
break;
default:
break;
}
close(fd);
return 0;
}

--------------------------------------------

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

Copyright © Linux教程網 All Rights Reserved