代码示例
下面的表 1 介绍了一些接口检索命令。该信息来源于 IBM System p 和 AIX。
表 1. 接口检索命令
ioctl 命令 描述
SIOCGSIZIFCONF 获得获取 SIOCGIFCONF 返回的所有接口的配置信息所需的内存。
ioctl(fd, SIOCGSIZIFCONF, (caddr_t)&ifconfsize);
int ifconfsize;
SIOCGIFADDR
SIOCSIFADDR
SIOCGIFADDR 获取接口地址,而 SIOCSIFADDR 设置接口地址。ifr.ifr_addr 字段返回地址。
ioctl(fd, SIOCGIFADDR, (caddr_t)&ifr, sizeof(struct ifreq));
ioctl(fd, SIOCSIFADDR, (caddr_t)&ifr, sizeof(struct ifreq));
struct ifreq ifr;
SIOCGIFCONF 返回系统中配置的所有接口的配置信息。
ioctl(fd, SIOCGIFCONF, (caddr_t)&ifc);
struct ifconf ifconf;
SIOCGSIZIFCONF
下面的代码段可以获取用来填充所有配置接口的信息所需的缓冲区大小。
清单 3. 使用 SIOCGSIZIFCONF 获得所需的缓冲区大小
/*Function to get the size needed to allocate for buffers*/
int get_interface_size(int sd){
int ret,size;
ret=ioctl(sd,SIOCGSIZIFCONF,&size);
if(ret==-1){
perror("Error getting size of interface :");
return ret;
}
return size;
}
main
{
struct ifconf ifc;
int sd;
sd=socket(AF_INET6,SOCK_DGRAM,0);
printf("Size of memory needed = %d",get_interface_size(sd));
}
上面代码的输出是:
$> ./myprog
Size of memory needed = 628
SIOCGIFCONF 和 SIOCGIFADDR
SIOCGIFCONF 和 SIOCGIFADDR 命令可以用来检索接口的地址。SIOCGIFCONF 可以用来获取所有配置接口的信息,SIOCGIFADDR 可以用来检索特定接口的地址。
清单 4. 使用 SIOCGIFCONF 获取配置信息
/*This function uses loops to find out buffer size instead of SIOCGSIZIFCONF
allocates the buffer and gets the information list*/
int alloc_buffer_size(int sd,struct ifconf *ifc){
int ret=-1,bsz=sizeof(struct ifreq);
int prevsz=bsz;
ifc->ifc_req=NULL;
ifc->ifc_len=bsz;
do{
ifc->ifc_req=(struct ifreq *)realloc(ifc->ifc_req,bsz);
if(!ifc->ifc_req){
perror("Malloc failed :");
return ret;
}
ifc->ifc_len=bsz;
ret=ioctl(sd,SIOCGIFCONF,(caddr_t)ifc);
if(ret==-1){
perror("Error getting size of interface :");
return ret;
}
if(prevsz==ifc->ifc_len)
break;
else{
bsz*=2;
prevsz=(0==ifc->ifc_len ? bsz : ifc->ifc_len) ;
}
}while(1);
ifc->ifc_req=(struct ifreq *)realloc(ifc->ifc_req,prevsz);
return ifc->ifc_len;
}
在使用 ifreq 结构中的配置信息填充了列表之后,可以对其进行遍历以检索所需的数据。根据调用之后该结构中填充的套接字地址的长度,可以对这个指针进行移动。下面的清单 5 显示了上面代码的列表中获得的系统中可用的 IP。
清单 5. 从配置列表中检索信息
#define MAX(x,y) ((x) >(y) ? (x) : (y))
#define SIZE(p) MAX((p).sa_len, sizeof(p))
/* This function prints all the configured IPs on the
* system given the ifconf structure allocated previously*/
void print_ips(struct ifconf *ifc){
char *cp,*cplim,addr[INET6_ADDRSTRLEN];
struct ifreq *ifr=ifc->ifc_req;
cp=(char *)ifc->ifc_req;
cplim=cp+ifc->ifc_len;
for(;cpifr_name) + SIZE(ifr->ifr_addr))){
/* NOTE: You cannot just increment cp with sizeof(struct ifreq)
* as structures returned are of different length.
*/
ifr=(struct ifreq *)cp;
printf("%s :",ifr->ifr_name);
getip(ifr,addr);
printf("%s",addr);
}
return ;
}
main
{
struct ifconf ifc;
int sd;
sd=socket(AF_INET6,SOCK_DGRAM,0);
alloc_buffer_size(sd,&ifc);
print_ips(&ifc);
}
上面代码的输出是:
$> ./myprog
en0 :6.3.6.0
en0 :9.182.192.169
en0 :fe80::209:6bff:feeb:70b2
sit0 :1.4.4.0
sit0 :::9.182.192.169
lo0 :24.3.0.0
lo0 :127.0.0.1
lo0 :::1
清单 6. 使用 SIOCGIFADDR 获取接口地址
/* Given a ifreq structure this function returns its IP address */
void getip(struct ifreq *ifr,char *addr){
struct sockaddr *sa;
sa=(struct sockaddr *)&(ifr->ifr_addr);
switch(sa->sa_family){
case AF_INET6:
inet_ntop(AF_INET6,(struct in6_addr *)&(((struct sockaddr_in6 *)sa)->sin6_addr),
addr,INET6_ADDRSTRLEN);
break;
default : strcpy(addr,inet_ntoa(((struct sockaddr_in *)sa)->sin_addr));
}
return;
}
main
{
char netaddr[INET_ADDRSTRLEN];
int sd;
sd=socket(AF_INET,SOCK_DGRAM,0);
strcpy(ifr.ifr_name,"en0");
if((ioctl(sd,SIOCGIFADDR,(caddr_t)&ifr,sizeof(struct ifreq)))<0)
perror("Error : ");
getip(&ifr,netaddr);
printf("%s",netaddr);
}
上面代码的输出是:
$./myprog
9.182.192.35
下面的表 2 介绍了接口标志和一些属性检索命令。
表 2. 接口标志和属性检索命令
ioctl 命令 描述
SIOCGIFFLAGS
SIOCSIFFLAGS
SIOCGIFFLAGS 可以获取接口标志。
SIOCSIFFLAGS 可以设置接口标志。
ioctl(fd, cmd, (caddr_t)&ifr);
struct ifreq ifr;
SIOCGIFFLAGS
在使用 SIOCGIFCONF 获取了接口列表之后,您可以使用 SIOCGIFFLAGS-SIOCSIFFLAGS 对来获取和设置接口的属性。接口标志表示为 IFF_XXX。有关完整的列表,请参阅文件 /usr/include/net/if.h。
下面的示例使用一对接口标志来检索其当前状态,并检查环回。也可以使用其他选项。
清单 7. 验证接口的状态和类型
for(;cp
ifr=(struct ifreq *)cp;
cp+=(sizeof(ifr->ifr_name) + SIZE(ifr->ifr_addr));
printf("%-9s ",ifr->ifr_name);
ret=ioctl(sd,SIOCGIFFLAGS,(caddr_t)ifr);
if(ret==-1){
perror("Error getting flags for interface :");
return ;
}
if((ifr->ifr_flags)&IFF_UP)
printf("UP");
else printf("DOWN");
if((ifr->ifr_flags)&IFF_LOOPBACK)
printf(",Loopback");
printf("");
}
上面代码的输出是:
$> ./myprog
en0 UP
en0 UP
en0 UP
sit0 UP
sit0 UP
lo0 UP,Loopback
lo0 UP,Loopback
lo0 UP,Loopback
清单 8. 设置接口标志
Partial output of ifconfig before setting the flag
sit0: flags=4900041
inet6 ::9.182.192.169/96
Code to set the interface flag
int set_interface_flag(int sd,short flag,char *interface){
struct ifreq ifr;
int oldflags;
strcpy(ifr.ifr_name,interface);
if((ioctl(sd,SIOCGIFFLAGS,(caddr_t)&ifr))==-1){
perror("Error getting interface flags :");
return -1;
}
printf("Setting new interface flag 0X%x",flag);
ifr.ifr_flags|=flag;
if((ioctl(sd,SIOCSIFFLAGS,(caddr_t)&ifr))==-1){
perror("Error setting interface flags :");
return -1;
}
return 0;
}
main
{
int sd;
sd=socket(AF_INET6,SOCK_DGRAM,0);
set_interface_flag(sd,IFF_NOTRAILERS,"sit0");
}
上面代码的输出是:
You must have permission to change the flag
$./myprog
Setting new interface flag 0X20
$ifconfig -a
.............
.............
sit0: flags=4900061
inet6 ::9.182.192.169/96
.............
.............
表 3 介绍了一些用于网络优化的命令。
表 3. 用于网络优化的命令
ioctl 命令 描述
SIOCGIFMTU SIOCGIFMTU 可以获取接口的最大传输单元 (MTU)。
ioctl(fd, cmd, (caddr_t)&ifr);
struct ifreq ifr;
这个 MTU 值存储在 ifr.ifr_mtu 字段中。
SIOCGNETOPT
SIOCGNETOPT1
SIOCGNETOPT 可以获取一个网络选项的值。
ioctl(fd, cmd, (caddr_t)&oreq);
struct optreq oreq;
SIOCGNETOPT1 可以获取当前值、缺省值和网络选项的范围。
ioctl(fd, SIOCGNETOPT1, (caddr_t)&oreq);
struct optreq1 oreq;
SIOCGIFMTU
清单 9. 从 ifreq 结构获取 MTU
ret=ioctl(sd,SIOCGIFMTU,ifr);
if(ret==-1){
perror("Error getting mtu for interface :");
return ;
}
printf(" %d",ifr->ifr_mtu);
SIOCGNETOPT1
SIOCGNETOPT1 给出当前值、缺省值和网络选项的范围。
清单 10. 从 optreq1 结构获取网络选项
/*Function to get the network options*/
int print_network_options(int sd){
int ret;
struct optreq1 oreq;
oreq.getnext=1;
while((ioctl(sd,SIOCGNETOPT1,(caddr_t)&oreq))!=-1)
printf("%s = %s",oreq.name,oreq.data);
return 0;
}
上面代码的输出是:
$> ./myprog
arpqsize = 12
arpt_killc = 20
arptab_bsiz = 7
arptab_nb = 149
........
........
........
........
ifsize = 256
inet_stack_size = 16
ip6_defttl = 64
........
在 AIX 中,您还可以使用 no 命令来管理各种网络优化参数。有关更详细的信息,请参阅相应的 man 页面。
清单 11. AIX 计算机上“no”命令的输出
$> no -a|more
arpqsize = 12
arpt_killc = 20
arptab_bsiz = 7
arptab_nb = 149
bcastping = 0
clean_partial_conns = 0
delayack = 0
delayackports = {}
dgd_packets_lost = 3
dgd_ping_time = 5
dgd_retry_time = 5
directed_broadcast = 0
extendednetstats = 0
fasttimo = 200
........
........
总结
ioctl 命令的功能非常强大,您可以使用它们来访问或修改网络(或其他)设备的可配置参数。它们使用了各种各样的数据结构,应该使用正确的数据来填充这些数据结构,以便实现预期的结果。您将发现在使用 AF_INET 和 AF_INET6 系列时存在的区别。本文提供了一小部分常用命令子集的代码示例。要获得 AIX Version 5.3 中完整的命令清单,请转到参考资料部分。我们希望这些关于 ioctl 的信息对您有用。