来源: http://hi.baidu.com/deep_pro/item/dcbfb108f3c1ce69d45a114e
今天值得纪念,作为技术支持,总算解决了第一个客户提出的实际问题。
实在是诡异的问题,为了重现错误尝试了多次,居然无意中发现跟延时有关。
最后解决问题还白白花了2个多小时,linux shell脚本真的很缺乏交互功能
神清气爽啊,回来都很晚了
这时有同学求助,工作上一堆问题,女生当然得帮了,时间不够先挑个软柿子,就是c语言取得网卡工作模式,半工还是全双工。
这个我很快就想到,要是/proc 、/sys 等下面没有的话,只能是ioctl 接口了
但是具体是那个ioctl命令就纠结了,lxr.linux.no上不了,手头没有源码查看环境,这个只能baidu了
很冷门啊,找到了SIOCETHTOOL
也可以strace 一下两个命令
strace ethtool eth0
strace mii-tool -v
但是一些ioctl命令不是所有网卡驱动都支持的,所以往往没跟到SIOCETHTOOL,这些程序就自己退出了,
很是靠不住
网络设备没有设备节点,fd只要socket就可以获得了
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <fcntl.h>
#include <getopt.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <stdlib.h>
#include <unistd.h>
typedef unsigned short u16;
typedef unsigned int u32;
typedef unsigned char u8;
#include <linux/ethtool.h>
#include <linux/sockios.h>
int get_settings(const char* devname)
{
struct ifreq ifr;
int fd;
struct ethtool_cmd ecmd;
int allfail = 1;
/* Setup our control structures. */
memset(&ifr, 0, sizeof(ifr));
strcpy(ifr.ifr_name, devname);
fd = socket(AF_INET, SOCK_DGRAM, 0);
if (fd < 0)
{
return -1;
}
ecmd.cmd = 1;
ifr.ifr_data = (caddr_t)&ecmd;
if(ioctl(fd, SIOCETHTOOL, &ifr))
{
close(fd);
perror("SIOCETHTOOL");
return -1;
}
//duplex 0 half 1 full
fprintf(stdout, "Settings for %s: speed:%d, duplex:%s\n", devname, ecmd.speed,
ecmd.duplex ? "full" : "half");
close(fd);
return 1;
}
int main(int argc, char *argv[])
{
int ret =get_settings(argv[1]);
printf("%d\n",ret);
return EXIT_SUCCESS;
}