linux命令码(_IO宏)

Posted 嵌入式Linux中文站

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了linux命令码(_IO宏)相关的知识,希望对你有一定的参考价值。

嵌入式linux QQ交流群:175159209,欢迎爱好者加入交流技术问题!


在ioctl.h头文件中定义了命令码

命令码用一个32位的整型数表达

bit29~31表示命令传输的方向,bit16~29记录要传输的数据的大小,bit8~15表示设备类型(一般用一个ASCII表示),bit0~7表示命令编号

其中数据大小可以和方向的bit29重叠

linux命令码(_IO宏)


#ifndef _SPARC_IOCTL_H
#define _SPARC_IOCTL_H
#define _IOC_NRBITS 8 //命令
#define _IOC_TYPEBITS 8 //设备类型
#define _IOC_SIZEBITS 13 /* Actually 14, see below. */ //数据大小
#define _IOC_DIRBITS 3 //方向
#define _IOC_NRMASK ((1 << _IOC_NRBITS)-1) //命令掩码    0x000000FF
#define _IOC_TYPEMASK ((1 << _IOC_TYPEBITS)-1) //设备类型掩码  0x000000FF
#define _IOC_SIZEMASK ((1 << _IOC_SIZEBITS)-1) //数据大小掩码  0x00000FFF
#define _IOC_XSIZEMASK ((1 << (_IOC_SIZEBITS+1))-1) //数据大小掩码  0x00001FFF
#define _IOC_DIRMASK ((1 << _IOC_DIRBITS)-1) //方向掩码    0x00000003
#define _IOC_NRSHIFT 0
#define _IOC_TYPESHIFT (_IOC_NRSHIFT + _IOC_NRBITS) //8
#define _IOC_SIZESHIFT (_IOC_TYPESHIFT + _IOC_TYPEBITS) //21
#define _IOC_DIRSHIFT (_IOC_SIZESHIFT + _IOC_SIZEBITS) //29
#define _IOC_NONE 1U //0x00000001无方向
#define _IOC_READ 2U //0x00000010读
#define _IOC_WRITE 4U //0x00000100写
#define _IOC(dir,type,nr,size) \//命令宏      (((dir)  << _IOC_DIRSHIFT) | \//方向 ((type) << _IOC_TYPESHIFT) | \//类型 ((nr) << _IOC_NRSHIFT) | \//命令 ((size) << _IOC_SIZESHIFT))//数据大小

#define _IO(type,nr) _IOC(_IOC_NONE,(type),(nr),0)//无方向命令
#define _IOR(type,nr,size) _IOC(_IOC_READ,(type),(nr),sizeof(size))//读命令
#define _IOW(type,nr,size) _IOC(_IOC_WRITE,(type),(nr),sizeof(size))//写命令#define _IOWR(type,nr,size) _IOC(_IOC_READ|_IOC_WRITE,(type),(nr),sizeof(size))//读写命令/* Used to decode ioctl numbers in drivers despite the leading underscore... */

#define _IOC_DIR(nr) \//获取方向 ( (((((nr) >> _IOC_DIRSHIFT) & _IOC_DIRMASK) & (_IOC_WRITE|_IOC_READ)) != 0)? \ (((nr) >> _IOC_DIRSHIFT) & (_IOC_WRITE|_IOC_READ)):  \ (((nr) >> _IOC_DIRSHIFT) & _IOC_DIRMASK) )

#define _IOC_TYPE(nr) (((nr) >> _IOC_TYPESHIFT) & _IOC_TYPEMASK)//获取类型#define _IOC_NR(nr) (((nr) >> _IOC_NRSHIFT) & _IOC_NRMASK)//获取命令

#define _IOC_SIZE(nr) \//获取数据大小((((((nr) >> _IOC_DIRSHIFT) & _IOC_DIRMASK) & (_IOC_WRITE|_IOC_READ)) == 0)?    \ 0: (((nr) >> _IOC_SIZESHIFT) & _IOC_XSIZEMASK))
/* ...and for the PCMCIA and sound. */

#define IOC_IN (_IOC_WRITE << _IOC_DIRSHIFT)
#define IOC_OUT (_IOC_READ << _IOC_DIRSHIFT)
#define IOC_INOUT ((_IOC_WRITE|_IOC_READ) << _IOC_DIRSHIFT)
#define IOCSIZE_MASK (_IOC_XSIZEMASK << _IOC_SIZESHIFT)
#define IOCSIZE_SHIFT (_IOC_SIZESHIFT)
#endif /* !(_SPARC_IOCTL_H) */


用法:

dir    =_IOC_DIR(nr)    //根据命令获取传输方向
type =_IOC_TYPE(nr) //根据命令获取类型
nr =_IOC_NR(nr) //根据命令获取类型命令
size =_IOC_SIZE(nr) //根据命令获取传输数据大小//////////////////////////////////////////////////////////////////////////////

cmd=_IOC(dir,type,nr,size) //根据传输方向,类型,命令,传输数据大小构成命令码cmdcmd=_IOW(type,nr,size) //根据类型,命令,传输数据类型生成读方向命令码cmd=_IOR(type,nr,size) //根据类型,命令,传输数据类型生成写方向命令码cmd=_IOWR(type,nr,size) //根据类型,命令,传输数据类型生成读写方向命令码cmd=_IOW_BAD(type,nr,size) //根据类型,命令,传输数据生成读方向命令码cmd=_IOR_BAD(type,nr,size) //根据类型,命令,传输数据生成写方向命令码cmd=_IOWR_BAD(type,nr,size) //根据类型,命令,传输数据生成读写方向命令码cmd=_IO(type,nr) //根据类型,命令生成没数据传输的无方向的命令码


实例:

#define BINDER_WRITE_READ  _IOWR('b', 1, struct binder_write_read)    //读写命令#define BINDER_SET_IDLE_TIMEOUT    _IOW('b', 3, int64_t)            //未实现
#define BINDER_SET_MAX_THREADS _IOW('b', 5, size_t) //设置最大线程数#define BINDER_SET_IDLE_PRIORITY_IOW('b', 6, int) //未实现
#define BINDER_SET_CONTEXT_MGR _IOW('b', 7, int) //设置binder上下文管理者
#define BINDER_THREAD_EXIT _IOW('b', 8, int) //删除线程信息
#define BINDER_VERSION _IOWR('b', 9, struct binder_version) //返回版本



嵌入式Linux中文站

最专业的中文嵌入式Linux网站,8年磨剑,注册用户数万人

分享 嵌入式 & Linux 技术干货、教程、资讯、高薪职位

订阅点击标题下方“嵌入式Linux中文站”

分享点击右上角分享按钮

投稿admin@embeddedlinux.org.cn

       点击下方“ 阅读原文 ”查看更多

以上是关于linux命令码(_IO宏)的主要内容,如果未能解决你的问题,请参考以下文章

12_关于flask中的宏

《linux 内核全然剖析》 include/asm/io.h

__FUNCTION__, __LINE__ 有助于debug的宏定义

inline内联函数

Linux bash基础特性二

008_Linux驱动之_IO口的配置