Linux ioctl系统调用实战
Linux ioctl系统调用实战ioctlinput/output control是Linux系统中一个强大的系统调用用于设备控制和配置。从网络接口配置到串口通信ioctl无处不在。本文将深入讲解ioctl的原理和实战应用。一、ioctl概述1.1 什么是ioctlioctl是一个通用的设备控制接口允许用户程序与设备驱动程序通信。它提供了一种执行设备特定操作的机制。#includesys/ioctl.hintioctl(intfd,unsignedlongrequest,...);参数fd文件描述符通常是设备文件request请求码表示要执行的操作…可选参数通常是数据指针返回值成功返回非负值取决于具体请求失败返回-1设置errno1.2 为什么需要ioctl设备操作种类繁多无法为每种操作都创建一个系统调用。ioctl提供了一种扩展机制统一接口一个系统调用处理多种操作灵活性驱动程序自定义请求码可扩展新增功能无需修改系统调用表1.3 请求码的构成Linux使用32位请求码分为4部分| 方向(2位) | 大小(14位) | 类型(8位) | 序号(8位) |方向数据传输方向_IOC_NONE无数据传输_IOC_READ从设备读取_IOC_WRITE向设备写入大小数据大小类型设备类型序号操作编号定义宏#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))二、网络接口配置2.1 修改IP地址#includestdio.h#includestring.h#includesys/socket.h#includesys/ioctl.h#includenetinet/in.h#includearpa/inet.h#includenet/if.hintset_ip_address(constchar*ifname,constchar*ipaddr){intsockfd;structifreqifr;structsockaddr_insin;// 创建socketsockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0){perror(socket);return-1;}// 初始化memset(ifr,0,sizeof(ifr));memset(sin,0,sizeof(sin));// 设置接口名strncpy(ifr.ifr_name,ifname,IFNAMSIZ-1);// 设置IP地址sin.sin_familyAF_INET;sin.sin_addr.s_addrinet_addr(ipaddr);memcpy(ifr.ifr_addr,sin,sizeof(sin));// 执行ioctl设置IPif(ioctl(sockfd,SIOCSIFADDR,ifr)0){perror(ioctl SIOCSIFADDR);close(sockfd);return-1;}// 启用接口ifr.ifr_flags|IFF_UP|IFF_RUNNING;if(ioctl(sockfd,SIOCSIFFLAGS,ifr)0){perror(ioctl SIOCSIFFLAGS);close(sockfd);return-1;}close(sockfd);return0;}intmain(){if(set_ip_address(eth0,192.168.1.100)0){printf(IP地址设置成功\n);}return0;}2.2 获取IP地址#includestdio.h#includestring.h#includesys/socket.h#includesys/ioctl.h#includenetinet/in.h#includearpa/inet.h#includenet/if.hintget_ip_address(constchar*ifname,char*ipaddr,intlen){intsockfd;structifreqifr;sockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0){perror(socket);return-1;}strncpy(ifr.ifr_name,ifname,IFNAMSIZ-1);if(ioctl(sockfd,SIOCGIFADDR,ifr)0){perror(ioctl SIOCGIFADDR);close(sockfd);return-1;}structsockaddr_in*sin(structsockaddr_in*)ifr.ifr_addr;snprintf(ipaddr,len,%s,inet_ntoa(sin-sin_addr));close(sockfd);return0;}intmain(){charipaddr[32];if(get_ip_address(eth0,ipaddr,sizeof(ipaddr))0){printf(当前IP: %s\n,ipaddr);}return0;}2.3 设置子网掩码intset_netmask(constchar*ifname,constchar*netmask){intsockfd;structifreqifr;structsockaddr_insin;sockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0)return-1;memset(ifr,0,sizeof(ifr));strncpy(ifr.ifr_name,ifname,IFNAMSIZ-1);sin.sin_familyAF_INET;sin.sin_addr.s_addrinet_addr(netmask);memcpy(ifr.ifr_netmask,sin,sizeof(sin));if(ioctl(sockfd,SIOCSIFNETMASK,ifr)0){perror(ioctl SIOCSIFNETMASK);close(sockfd);return-1;}close(sockfd);return0;}2.4 设置MAC地址#includenet/if_arp.hintset_mac_address(constchar*ifname,constchar*mac){intsockfd;structifreqifr;sockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0)return-1;memset(ifr,0,sizeof(ifr));strncpy(ifr.ifr_name,ifname,IFNAMSIZ-1);// 解析MAC地址 aa:bb:cc:dd:ee:ffsscanf(mac,%hhx:%hhx:%hhx:%hhx:%hhx:%hhx,ifr.ifr_hwaddr.sa_data[0],ifr.ifr_hwaddr.sa_data[1],ifr.ifr_hwaddr.sa_data[2],ifr.ifr_hwaddr.sa_data[3],ifr.ifr_hwaddr.sa_data[4],ifr.ifr_hwaddr.sa_data[5]);ifr.ifr_hwaddr.sa_familyARPHRD_ETHER;if(ioctl(sockfd,SIOCSIFHWADDR,ifr)0){perror(ioctl SIOCSIFHWADDR);close(sockfd);return-1;}close(sockfd);return0;}三、串口编程3.1 配置串口参数#includestdio.h#includefcntl.h#includeunistd.h#includetermios.h#includesys/ioctl.hintconfigure_serial(intfd,intbaudrate,intdatabits,intstopbits,charparity){structtermiosoptions;// 获取当前配置if(tcgetattr(fd,options)!0){perror(tcgetattr);return-1;}// 设置波特率speed_tspeed;switch(baudrate){case9600:speedB9600;break;case19200:speedB19200;break;case38400:speedB38400;break;case115200:speedB115200;break;default:speedB9600;}cfsetispeed(options,speed);cfsetospeed(options,speed);// 数据位options.c_cflag~CSIZE;switch(databits){case7:options.c_cflag|CS7;break;case8:options.c_cflag|CS8;break;}// 校验位switch(parity){caseN:// 无校验options.c_cflag~PARENB;break;caseO:// 奇校验options.c_cflag|PARENB|PARODD;break;caseE:// 偶校验options.c_cflag|PARENB;options.c_cflag~PARODD;break;}// 停止位switch(stopbits){case1:options.c_cflag~CSTOPB;break;case2:options.c_cflag|CSTOPB;break;}// 其他设置options.c_cflag|CLOCAL|CREAD;// 启用接收options.c_iflag~(IXON|IXOFF|IXANY);// 禁用流控// 应用配置if(tcsetattr(fd,TCSANOW,options)!0){perror(tcsetattr);return-1;}return0;}intmain(){intfdopen(/dev/ttyS0,O_RDWR|O_NOCTTY);if(fd0){perror(open);return1;}// 配置: 115200波特率, 8数据位, 无校验, 1停止位configure_serial(fd,115200,8,1,N);// 读写操作...close(fd);return0;}3.2 获取串口状态#includelinux/serial.hintget_serial_info(intfd){structserial_structserinfo;if(ioctl(fd,TIOCGSERIAL,serinfo)0){perror(ioctl TIOCGSERIAL);return-1;}printf(波特率: %d\n,serinfo.baud_base);printf(自定义波特率: %d\n,serinfo.custom_divisor);return0;}四、常用ioctl请求4.1 网络相关请求说明SIOCGIFADDR获取IP地址SIOCSIFADDR设置IP地址SIOCGIFNETMASK获取子网掩码SIOCSIFNETMASK设置子网掩码SIOCGIFHWADDR获取MAC地址SIOCSIFHWADDR设置MAC地址SIOCGIFFLAGS获取接口标志SIOCSIFFLAGS设置接口标志SIOCGIFMTU获取MTUSIOCSIFMTU设置MTU4.2 终端相关请求说明TCGETS获取终端参数TCSETS设置终端参数TIOCGWINSZ获取窗口大小TIOCSWINSZ设置窗口大小TIOCGPGRP获取前台进程组TIOCSPGRP设置前台进程组4.3 文件相关请求说明FIONREAD获取可读字节数FIONBIO设置非阻塞模式FIOASYNC设置异步I/O五、实战案例5.1 获取网口列表#includestdio.h#includestring.h#includesys/socket.h#includesys/ioctl.h#includenet/if.h#includeunistd.h#includelinux/sockios.hvoidlist_interfaces(){intsockfd;structifreqifr[16];structifconfifc;sockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0){perror(socket);return;}ifc.ifc_lensizeof(ifr);ifc.ifc_reqifr;if(ioctl(sockfd,SIOCGIFCONF,ifc)0){perror(ioctl SIOCGIFCONF);close(sockfd);return;}intcountifc.ifc_len/sizeof(structifreq);for(inti0;icount;i){printf(接口: %s\n,ifr[i].ifr_name);}close(sockfd);}intmain(){list_interfaces();return0;}5.2 设置接口状态#includestdio.h#includestring.h#includesys/socket.h#includesys/ioctl.h#includenet/if.h#includeunistd.hintset_interface_up(constchar*ifname,intup){intsockfd;structifreqifr;sockfdsocket(AF_INET,SOCK_DGRAM,0);if(sockfd0)return-1;strncpy(ifr.ifr_name,ifname,IFNAMSIZ-1);// 获取当前标志if(ioctl(sockfd,SIOCGIFFLAGS,ifr)0){perror(ioctl SIOCGIFFLAGS);close(sockfd);return-1;}// 修改标志if(up){ifr.ifr_flags|IFF_UP;}else{ifr.ifr_flags~IFF_UP;}// 设置新标志if(ioctl(sockfd,SIOCSIFFLAGS,ifr)0){perror(ioctl SIOCSIFFLAGS);close(sockfd);return-1;}close(sockfd);return0;}intmain(){// 启用eth0set_interface_up(eth0,1);// 禁用eth0set_interface_up(eth0,0);return0;}六、总结ioctl是Linux系统编程的重要工具统一接口一个系统调用处理多种操作网络配置IP、MAC、子网掩码等串口编程波特率、数据位、校验位设备控制各种硬件设备的配置使用要点正确处理返回值和错误了解具体设备的请求码注意参数的数据类型和对齐权限问题某些操作需要root
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2494703.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!