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

📄 network.c

📁 关系型数据库 Postgresql 6.5.2
💻 C
字号:
/* *	PostgreSQL type definitions for the INET type.	This *	is for IP V4 CIDR notation, but prepared for V6: just *	add the necessary bits where the comments indicate. * *	$Id: network.c,v 1.10.2.1 1999/08/02 05:24:55 scrappy Exp $ *	Jon Postel RIP 16 Oct 1998 */#include <sys/types.h>#include <sys/socket.h>#include <errno.h>#include <netinet/in.h>#include <arpa/inet.h>#include "postgres.h"#include "utils/builtins.h"static int	v4bitncmp(unsigned int a1, unsigned int a2, int bits);/* *	Access macros.	Add IPV6 support. */#define ip_addrsize(inetptr) \	(((inet_struct *)VARDATA(inetptr))->family == AF_INET ? 4 : -1)#define ip_family(inetptr) \	(((inet_struct *)VARDATA(inetptr))->family)#define ip_bits(inetptr) \	(((inet_struct *)VARDATA(inetptr))->bits)#define ip_type(inetptr) \	(((inet_struct *)VARDATA(inetptr))->type)#define ip_v4addr(inetptr) \	(((inet_struct *)VARDATA(inetptr))->addr.ipv4_addr)/* Common input routine */static inet *network_in(char *src, int type){	int			bits;	inet	   *dst;	if (!src)		return NULL;	dst = palloc(VARHDRSZ + sizeof(inet_struct));	if (dst == NULL)		elog(ERROR, "unable to allocate memory in network_in()");	/* First, try for an IP V4 address: */	ip_family(dst) = AF_INET;	bits = inet_net_pton(ip_family(dst), src, &ip_v4addr(dst),						 type ? ip_addrsize(dst) : -1);	if ((bits < 0) || (bits > 32))		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "could not parse \"%s\"", src);	VARSIZE(dst) = VARHDRSZ		+ ((char *) &ip_v4addr(dst) - (char *) VARDATA(dst))		+ ip_addrsize(dst);	ip_bits(dst) = bits;	ip_type(dst) = type;	return dst;}/* INET address reader.  */inet *inet_in(char *src){	return network_in(src, 0);}/* CIDR address reader.  */inet *cidr_in(char *src){	return network_in(src, 1);}/* *	INET address output function. */char *inet_out(inet *src){	char	   *dst,				tmp[sizeof("255.255.255.255/32")];	if (ip_family(src) == AF_INET)	{		/* It's an IP V4 address: */		if (ip_type(src))			dst = inet_cidr_ntop(AF_INET, &ip_v4addr(src), ip_bits(src),								 tmp, sizeof(tmp));		else			dst = inet_net_ntop(AF_INET, &ip_v4addr(src), ip_bits(src),								tmp, sizeof(tmp));		if (dst == NULL)			elog(ERROR, "unable to print address (%s)", strerror(errno));	}	else		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "unknown address family (%d)", ip_family(src));	dst = palloc(strlen(tmp) + 1);	if (dst == NULL)		elog(ERROR, "unable to allocate memory in inet_out()");	strcpy(dst, tmp);	return dst;}/* just a stub */char *cidr_out(inet *src){	return inet_out(src);}/* *	Boolean tests for magnitude.  Add V4/V6 testing! */boolnetwork_lt(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		int			order = v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a2));		return ((order < 0) || ((order == 0) && (ip_bits(a1) < ip_bits(a2))));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_le(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	return (network_lt(a1, a2) || network_eq(a1, a2));}boolnetwork_eq(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		return ((ip_bits(a1) == ip_bits(a2))		 && (v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a1)) == 0));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_ge(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	return (network_gt(a1, a2) || network_eq(a1, a2));}boolnetwork_gt(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		int			order = v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a2));		return ((order > 0) || ((order == 0) && (ip_bits(a1) > ip_bits(a2))));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_ne(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	return (!network_eq(a1, a2));}boolnetwork_sub(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		return ((ip_bits(a1) > ip_bits(a2))		 && (v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a2)) == 0));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_subeq(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		return ((ip_bits(a1) >= ip_bits(a2))		 && (v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a2)) == 0));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_sup(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		return ((ip_bits(a1) < ip_bits(a2))		 && (v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a1)) == 0));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}boolnetwork_supeq(inet *a1, inet *a2){	if (!PointerIsValid(a1) || !PointerIsValid(a2))		return FALSE;	if ((ip_family(a1) == AF_INET) && (ip_family(a2) == AF_INET))	{		return ((ip_bits(a1) <= ip_bits(a2))		 && (v4bitncmp(ip_v4addr(a1), ip_v4addr(a2), ip_bits(a1)) == 0));	}	else	{		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "cannot compare address families %d and %d",			 ip_family(a1), ip_family(a2));		return FALSE;	}}/* *	Comparison function for sorting.  Add V4/V6 testing! */int4network_cmp(inet *a1, inet *a2){	if (ntohl(ip_v4addr(a1)) < ntohl(ip_v4addr(a2)))		return (-1);	if (ntohl(ip_v4addr(a1)) > ntohl(ip_v4addr(a2)))		return (1);	if (ip_bits(a1) < ip_bits(a2))		return (-1);	if (ip_bits(a1) > ip_bits(a2))		return (1);	return 0;}text *network_host(inet *ip){	text	   *ret;	int			len;	char	   *ptr,				tmp[sizeof("255.255.255.255/32")];	if (!PointerIsValid(ip))		return NULL;	if (ip_type(ip))		elog(ERROR, "CIDR type has no host part");	if (ip_family(ip) == AF_INET)	{		/* It's an IP V4 address: */		if (inet_net_ntop(AF_INET, &ip_v4addr(ip), 32, tmp, sizeof(tmp)) == NULL)			elog(ERROR, "unable to print host (%s)", strerror(errno));	}	else		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "unknown address family (%d)", ip_family(ip));	if ((ptr = strchr(tmp, '/')) != NULL)		*ptr = 0;	len = VARHDRSZ + strlen(tmp) + 1;	ret = palloc(len);	if (ret == NULL)		elog(ERROR, "unable to allocate memory in network_host()");	VARSIZE(ret) = len;	strcpy(VARDATA(ret), tmp);	return (ret);}int4network_masklen(inet *ip){	if (!PointerIsValid(ip))		return 0;	return ip_bits(ip);}text *network_broadcast(inet *ip){	text	   *ret;	int			len;	char	   *ptr,				tmp[sizeof("255.255.255.255/32")];	if (!PointerIsValid(ip))		return NULL;	if (ip_family(ip) == AF_INET)	{		/* It's an IP V4 address: */		int			addr;		unsigned long mask = 0xffffffff;		if (ip_bits(ip) < 32)			mask >>= ip_bits(ip);		addr = htonl(ntohl(ip_v4addr(ip)) | mask);		if (inet_net_ntop(AF_INET, &addr, 32, tmp, sizeof(tmp)) == NULL)			elog(ERROR, "unable to print address (%s)", strerror(errno));	}	else		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "unknown address family (%d)", ip_family(ip));	if ((ptr = strchr(tmp, '/')) != NULL)		*ptr = 0;	len = VARHDRSZ + strlen(tmp) + 1;	ret = palloc(len);	if (ret == NULL)		elog(ERROR, "unable to allocate memory in network_broadcast()");	VARSIZE(ret) = len;	strcpy(VARDATA(ret), tmp);	return (ret);}text *network_network(inet *ip){	text	   *ret;	int			len;	char		tmp[sizeof("255.255.255.255/32")];	if (!PointerIsValid(ip))		return NULL;	if (ip_family(ip) == AF_INET)	{		/* It's an IP V4 address: */		int			addr = htonl(ntohl(ip_v4addr(ip)) & (0xffffffff << (32 - ip_bits(ip))));		if (inet_cidr_ntop(AF_INET, &addr, ip_bits(ip), tmp, sizeof(tmp)) == NULL)			elog(ERROR, "unable to print network (%s)", strerror(errno));	}	else		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "unknown address family (%d)", ip_family(ip));	len = VARHDRSZ + strlen(tmp) + 1;	ret = palloc(len);	if (ret == NULL)		elog(ERROR, "unable to allocate memory in network_network()");	VARSIZE(ret) = len;	strcpy(VARDATA(ret), tmp);	return (ret);}text *network_netmask(inet *ip){	text	   *ret;	int			len;	char	   *ptr,				tmp[sizeof("255.255.255.255/32")];	if (!PointerIsValid(ip))		return NULL;	if (ip_family(ip) == AF_INET)	{		/* It's an IP V4 address: */		int			addr = htonl((-1 << (32 - ip_bits(ip))) & 0xffffffff);		if (inet_net_ntop(AF_INET, &addr, 32, tmp, sizeof(tmp)) == NULL)			elog(ERROR, "unable to print netmask (%s)", strerror(errno));	}	else		/* Go for an IPV6 address here, before faulting out: */		elog(ERROR, "unknown address family (%d)", ip_family(ip));	if ((ptr = strchr(tmp, '/')) != NULL)		*ptr = 0;	len = VARHDRSZ + strlen(tmp) + 1;	ret = palloc(len);	if (ret == NULL)		elog(ERROR, "unable to allocate memory in network_netmask()");	VARSIZE(ret) = len;	strcpy(VARDATA(ret), tmp);	return (ret);}/* *	Bitwise comparison for V4 addresses.  Add V6 implementation! */static intv4bitncmp(unsigned int a1, unsigned int a2, int bits){	unsigned long mask = 0;	int			i;	for (i = 0; i < bits; i++)		mask = (mask >> 1) | 0x80000000;	a1 = ntohl(a1);	a2 = ntohl(a2);	if ((a1 & mask) < (a2 & mask))		return (-1);	else if ((a1 & mask) > (a2 & mask))		return (1);	return (0);}

⌨️ 快捷键说明

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