⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ipaddress.c

📁 busybox最新版的源码:学习和应用的好东东,多的不说了,大家看后再说吧
💻 C
📖 第 1 页 / 共 2 页
字号:
/* vi: set sw=4 ts=4: *//* * ipaddress.c		"ip address". * * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. * * Authors:	Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru> * * Changes: *	Laszlo Valko <valko@linux.karinthy.hu> 990223: address label must be zero terminated *///#include <sys/socket.h>//#include <sys/ioctl.h>#include <fnmatch.h>#include <net/if.h>#include <net/if_arp.h>#include "ip_common.h"  /* #include "libbb.h" is inside */#include "rt_names.h"#include "utils.h"typedef struct filter_t {	int ifindex;	int family;	int oneline;	int showqueue;	inet_prefix pfx;	int scope, scopemask;	int flags, flagmask;	int up;	char *label;	int flushed;	char *flushb;	int flushp;	int flushe;	struct rtnl_handle *rth;} filter_t;#define filter (*(filter_t*)&bb_common_bufsiz1)static void print_link_flags(FILE *fp, unsigned flags, unsigned mdown){	fprintf(fp, "<");	flags &= ~IFF_RUNNING;#define _PF(f) if (flags&IFF_##f) { \		  flags &= ~IFF_##f; \		  fprintf(fp, #f "%s", flags ? "," : ""); }	_PF(LOOPBACK);	_PF(BROADCAST);	_PF(POINTOPOINT);	_PF(MULTICAST);	_PF(NOARP);#if 0	_PF(ALLMULTI);	_PF(PROMISC);	_PF(MASTER);	_PF(SLAVE);	_PF(DEBUG);	_PF(DYNAMIC);	_PF(AUTOMEDIA);	_PF(PORTSEL);	_PF(NOTRAILERS);#endif	_PF(UP);#undef _PF	if (flags)		fprintf(fp, "%x", flags);	if (mdown)		fprintf(fp, ",M-DOWN");	fprintf(fp, "> ");}static void print_queuelen(char *name){	struct ifreq ifr;	int s;	s = socket(AF_INET, SOCK_STREAM, 0);	if (s < 0)		return;	memset(&ifr, 0, sizeof(ifr));	strncpy(ifr.ifr_name, name, sizeof(ifr.ifr_name));	if (ioctl_or_warn(s, SIOCGIFTXQLEN, &ifr) < 0) {		close(s);		return;	}	close(s);	if (ifr.ifr_qlen)		printf("qlen %d", ifr.ifr_qlen);}static int print_linkinfo(struct sockaddr_nl ATTRIBUTE_UNUSED *who,		const struct nlmsghdr *n, void ATTRIBUTE_UNUSED *arg){	FILE *fp = (FILE*)arg;	struct ifinfomsg *ifi = NLMSG_DATA(n);	struct rtattr * tb[IFLA_MAX+1];	int len = n->nlmsg_len;	unsigned m_flag = 0;	if (n->nlmsg_type != RTM_NEWLINK && n->nlmsg_type != RTM_DELLINK)		return 0;	len -= NLMSG_LENGTH(sizeof(*ifi));	if (len < 0)		return -1;	if (filter.ifindex && ifi->ifi_index != filter.ifindex)		return 0;	if (filter.up && !(ifi->ifi_flags&IFF_UP))		return 0;	memset(tb, 0, sizeof(tb));	parse_rtattr(tb, IFLA_MAX, IFLA_RTA(ifi), len);	if (tb[IFLA_IFNAME] == NULL) {		bb_error_msg("nil ifname");		return -1;	}	if (filter.label	 && (!filter.family || filter.family == AF_PACKET)	 && fnmatch(filter.label, RTA_DATA(tb[IFLA_IFNAME]), 0)	) {		return 0;	}	if (n->nlmsg_type == RTM_DELLINK)		fprintf(fp, "Deleted ");	fprintf(fp, "%d: %s", ifi->ifi_index,		tb[IFLA_IFNAME] ? (char*)RTA_DATA(tb[IFLA_IFNAME]) : "<nil>");	if (tb[IFLA_LINK]) {		SPRINT_BUF(b1);		int iflink = *(int*)RTA_DATA(tb[IFLA_LINK]);		if (iflink == 0)			fprintf(fp, "@NONE: ");		else {			fprintf(fp, "@%s: ", ll_idx_n2a(iflink, b1));			m_flag = ll_index_to_flags(iflink);			m_flag = !(m_flag & IFF_UP);		}	} else {		fprintf(fp, ": ");	}	print_link_flags(fp, ifi->ifi_flags, m_flag);	if (tb[IFLA_MTU])		fprintf(fp, "mtu %u ", *(int*)RTA_DATA(tb[IFLA_MTU]));	if (tb[IFLA_QDISC])		fprintf(fp, "qdisc %s ", (char*)RTA_DATA(tb[IFLA_QDISC]));#ifdef IFLA_MASTER	if (tb[IFLA_MASTER]) {		SPRINT_BUF(b1);		fprintf(fp, "master %s ", ll_idx_n2a(*(int*)RTA_DATA(tb[IFLA_MASTER]), b1));	}#endif	if (filter.showqueue)		print_queuelen((char*)RTA_DATA(tb[IFLA_IFNAME]));	if (!filter.family || filter.family == AF_PACKET) {		SPRINT_BUF(b1);		fprintf(fp, "%c    link/%s ", _SL_, ll_type_n2a(ifi->ifi_type, b1, sizeof(b1)));		if (tb[IFLA_ADDRESS]) {			fprintf(fp, "%s", ll_addr_n2a(RTA_DATA(tb[IFLA_ADDRESS]),						      RTA_PAYLOAD(tb[IFLA_ADDRESS]),						      ifi->ifi_type,						      b1, sizeof(b1)));		}		if (tb[IFLA_BROADCAST]) {			if (ifi->ifi_flags&IFF_POINTOPOINT)				fprintf(fp, " peer ");			else				fprintf(fp, " brd ");			fprintf(fp, "%s", ll_addr_n2a(RTA_DATA(tb[IFLA_BROADCAST]),						      RTA_PAYLOAD(tb[IFLA_BROADCAST]),						      ifi->ifi_type,						      b1, sizeof(b1)));		}	}	fputc('\n', fp);	fflush(fp);	return 0;}static int flush_update(void){	if (rtnl_send(filter.rth, filter.flushb, filter.flushp) < 0) {		bb_perror_msg("failed to send flush request");		return -1;	}	filter.flushp = 0;	return 0;}static int print_addrinfo(struct sockaddr_nl ATTRIBUTE_UNUSED *who,		struct nlmsghdr *n, void ATTRIBUTE_UNUSED *arg){	FILE *fp = (FILE*)arg;	struct ifaddrmsg *ifa = NLMSG_DATA(n);	int len = n->nlmsg_len;	struct rtattr * rta_tb[IFA_MAX+1];	char abuf[256];	SPRINT_BUF(b1);	if (n->nlmsg_type != RTM_NEWADDR && n->nlmsg_type != RTM_DELADDR)		return 0;	len -= NLMSG_LENGTH(sizeof(*ifa));	if (len < 0) {		bb_error_msg("wrong nlmsg len %d", len);		return -1;	}	if (filter.flushb && n->nlmsg_type != RTM_NEWADDR)		return 0;	memset(rta_tb, 0, sizeof(rta_tb));	parse_rtattr(rta_tb, IFA_MAX, IFA_RTA(ifa), n->nlmsg_len - NLMSG_LENGTH(sizeof(*ifa)));	if (!rta_tb[IFA_LOCAL])		rta_tb[IFA_LOCAL] = rta_tb[IFA_ADDRESS];	if (!rta_tb[IFA_ADDRESS])		rta_tb[IFA_ADDRESS] = rta_tb[IFA_LOCAL];	if (filter.ifindex && filter.ifindex != ifa->ifa_index)		return 0;	if ((filter.scope^ifa->ifa_scope)&filter.scopemask)		return 0;	if ((filter.flags^ifa->ifa_flags)&filter.flagmask)		return 0;	if (filter.label) {		const char *label;		if (rta_tb[IFA_LABEL])			label = RTA_DATA(rta_tb[IFA_LABEL]);		else			label = ll_idx_n2a(ifa->ifa_index, b1);		if (fnmatch(filter.label, label, 0) != 0)			return 0;	}	if (filter.pfx.family) {		if (rta_tb[IFA_LOCAL]) {			inet_prefix dst;			memset(&dst, 0, sizeof(dst));			dst.family = ifa->ifa_family;			memcpy(&dst.data, RTA_DATA(rta_tb[IFA_LOCAL]), RTA_PAYLOAD(rta_tb[IFA_LOCAL]));			if (inet_addr_match(&dst, &filter.pfx, filter.pfx.bitlen))				return 0;		}	}	if (filter.flushb) {		struct nlmsghdr *fn;		if (NLMSG_ALIGN(filter.flushp) + n->nlmsg_len > filter.flushe) {			if (flush_update())				return -1;		}		fn = (struct nlmsghdr*)(filter.flushb + NLMSG_ALIGN(filter.flushp));		memcpy(fn, n, n->nlmsg_len);		fn->nlmsg_type = RTM_DELADDR;		fn->nlmsg_flags = NLM_F_REQUEST;		fn->nlmsg_seq = ++filter.rth->seq;		filter.flushp = (((char*)fn) + n->nlmsg_len) - filter.flushb;		filter.flushed++;		return 0;	}	if (n->nlmsg_type == RTM_DELADDR)		fprintf(fp, "Deleted ");	if (filter.oneline)		fprintf(fp, "%u: %s", ifa->ifa_index, ll_index_to_name(ifa->ifa_index));	if (ifa->ifa_family == AF_INET)		fprintf(fp, "    inet ");	else if (ifa->ifa_family == AF_INET6)		fprintf(fp, "    inet6 ");	else		fprintf(fp, "    family %d ", ifa->ifa_family);	if (rta_tb[IFA_LOCAL]) {		fprintf(fp, "%s", rt_addr_n2a(ifa->ifa_family,					      RTA_PAYLOAD(rta_tb[IFA_LOCAL]),					      RTA_DATA(rta_tb[IFA_LOCAL]),					      abuf, sizeof(abuf)));		if (rta_tb[IFA_ADDRESS] == NULL ||		    memcmp(RTA_DATA(rta_tb[IFA_ADDRESS]), RTA_DATA(rta_tb[IFA_LOCAL]), 4) == 0) {			fprintf(fp, "/%d ", ifa->ifa_prefixlen);		} else {			fprintf(fp, " peer %s/%d ",				rt_addr_n2a(ifa->ifa_family,					    RTA_PAYLOAD(rta_tb[IFA_ADDRESS]),					    RTA_DATA(rta_tb[IFA_ADDRESS]),					    abuf, sizeof(abuf)),				ifa->ifa_prefixlen);		}	}	if (rta_tb[IFA_BROADCAST]) {		fprintf(fp, "brd %s ",			rt_addr_n2a(ifa->ifa_family,				    RTA_PAYLOAD(rta_tb[IFA_BROADCAST]),				    RTA_DATA(rta_tb[IFA_BROADCAST]),				    abuf, sizeof(abuf)));	}	if (rta_tb[IFA_ANYCAST]) {		fprintf(fp, "any %s ",			rt_addr_n2a(ifa->ifa_family,				    RTA_PAYLOAD(rta_tb[IFA_ANYCAST]),				    RTA_DATA(rta_tb[IFA_ANYCAST]),				    abuf, sizeof(abuf)));	}	fprintf(fp, "scope %s ", rtnl_rtscope_n2a(ifa->ifa_scope, b1, sizeof(b1)));	if (ifa->ifa_flags&IFA_F_SECONDARY) {		ifa->ifa_flags &= ~IFA_F_SECONDARY;		fprintf(fp, "secondary ");	}	if (ifa->ifa_flags&IFA_F_TENTATIVE) {		ifa->ifa_flags &= ~IFA_F_TENTATIVE;		fprintf(fp, "tentative ");	}	if (ifa->ifa_flags&IFA_F_DEPRECATED) {		ifa->ifa_flags &= ~IFA_F_DEPRECATED;		fprintf(fp, "deprecated ");	}	if (!(ifa->ifa_flags&IFA_F_PERMANENT)) {		fprintf(fp, "dynamic ");	} else		ifa->ifa_flags &= ~IFA_F_PERMANENT;	if (ifa->ifa_flags)		fprintf(fp, "flags %02x ", ifa->ifa_flags);	if (rta_tb[IFA_LABEL])		fprintf(fp, "%s", (char*)RTA_DATA(rta_tb[IFA_LABEL]));	if (rta_tb[IFA_CACHEINFO]) {		struct ifa_cacheinfo *ci = RTA_DATA(rta_tb[IFA_CACHEINFO]);		char buf[128];		fputc(_SL_, fp);		if (ci->ifa_valid == 0xFFFFFFFFU)			sprintf(buf, "valid_lft forever");		else			sprintf(buf, "valid_lft %dsec", ci->ifa_valid);		if (ci->ifa_prefered == 0xFFFFFFFFU)			sprintf(buf+strlen(buf), " preferred_lft forever");		else			sprintf(buf+strlen(buf), " preferred_lft %dsec", ci->ifa_prefered);		fprintf(fp, "       %s", buf);	}	fputc('\n', fp);	fflush(fp);	return 0;}struct nlmsg_list{	struct nlmsg_list *next;	struct nlmsghdr	  h;};static int print_selected_addrinfo(int ifindex, struct nlmsg_list *ainfo, FILE *fp){	for (; ainfo; ainfo = ainfo->next) {		struct nlmsghdr *n = &ainfo->h;		struct ifaddrmsg *ifa = NLMSG_DATA(n);		if (n->nlmsg_type != RTM_NEWADDR)			continue;		if (n->nlmsg_len < NLMSG_LENGTH(sizeof(ifa)))			return -1;		if (ifa->ifa_index != ifindex ||		    (filter.family && filter.family != ifa->ifa_family))			continue;		print_addrinfo(NULL, n, fp);	}	return 0;}static int store_nlmsg(struct sockaddr_nl *who, struct nlmsghdr *n, void *arg){	struct nlmsg_list **linfo = (struct nlmsg_list**)arg;	struct nlmsg_list *h;	struct nlmsg_list **lp;	h = malloc(n->nlmsg_len+sizeof(void*));	if (h == NULL)		return -1;	memcpy(&h->h, n, n->nlmsg_len);

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -