ip_addr.c
4e2fdd79
 /*
  * $Id$
  *
  *
  * ip address & address family related functions
7dd0b342
  *
53c7e0f1
  * Copyright (C) 2001-2003 FhG Fokus
7dd0b342
  *
  * This file is part of ser, a free SIP server.
  *
  * ser is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License as published by
  * the Free Software Foundation; either version 2 of the License, or
  * (at your option) any later version
  *
  * For a license to use the ser software under conditions
  * other than those described here, or to purchase support for this
  * software, please contact iptel.org by e-mail at the following addresses:
  *    info@iptel.org
  *
  * ser is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  * GNU General Public License for more details.
  *
  * You should have received a copy of the GNU General Public License 
  * along with this program; if not, write to the Free Software 
  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
4e2fdd79
  */
e3dccdc9
 /*
  * History:
  * --------
  *  2003-03-19  replaced all mallocs/frees w/ pkg_malloc/pkg_free
092bc818
  *  2004-10-01  mk_net fixes bad network addresses now (andrei)
e3dccdc9
  */
4e2fdd79
 
7dd0b342
 
381659ac
 #include <stdlib.h>
4e2fdd79
 #include <stdio.h>
 
 #include "ip_addr.h"
 #include "dprint.h"
e3dccdc9
 #include "mem/mem.h"
4e2fdd79
 
 
 struct net* mk_net(struct ip_addr* ip, struct ip_addr* mask)
 {
 	struct net* n;
092bc818
 	int warning;
 	int r;
4e2fdd79
 	
092bc818
 	warning=0;
4e2fdd79
 	if ((ip->af != mask->af) || (ip->len != mask->len)){
 		LOG(L_CRIT, "ERROR: mk_net: trying to use a different mask family"
 				" (eg. ipv4/ipv6mask or ipv6/ipv4mask)\n");
 		goto error;
 	}
e3dccdc9
 	n=(struct net*)pkg_malloc(sizeof(struct net));
4e2fdd79
 	if (n==0){ 
 		LOG(L_CRIT, "ERROR: mk_net: memory allocation failure\n");
 		goto error;
 	}
 	n->ip=*ip;
 	n->mask=*mask;
092bc818
 	for (r=0; r<n->ip.len/4; r++) { /*ipv4 & ipv6 addresses are multiple of 4*/
 		n->ip.u.addr32[r] &= n->mask.u.addr32[r];
 		if (n->ip.u.addr32[r]!=ip->u.addr32[r]) warning=1;
 	};
 	if (warning){
 		LOG(L_WARN, "WARNING: mk_net: invalid network address/netmask "
 					"combination fixed...\n");
 		print_ip("original network address:", ip, "/");
 		print_ip("", mask, "\n");
 		print_ip("fixed    network address:", &(n->ip), "/");
 		print_ip("", &(n->mask), "\n");
 	};
4e2fdd79
 	return n;
 error:
 	return 0;
 }
 
 
 
 struct net* mk_net_bitlen(struct ip_addr* ip, unsigned int bitlen)
 {
092bc818
 	struct ip_addr mask;
4e2fdd79
 	int r;
 	
 	if (bitlen>ip->len*8){
 		LOG(L_CRIT, "ERROR: mk_net_bitlen: bad bitlen number %d\n", bitlen);
 		goto error;
 	}
092bc818
 	memset(&mask,0, sizeof(mask));
 	for (r=0;r<bitlen/8;r++) mask.u.addr[r]=0xff;
 	if (bitlen%8) mask.u.addr[r]=  ~((1<<(8-(bitlen%8)))-1);
 	mask.af=ip->af;
 	mask.len=ip->len;
4e2fdd79
 	
092bc818
 	return mk_net(ip, &mask);
4e2fdd79
 error:
 	return 0;
 }
 
 
 
9c01c860
 void print_ip(char* p, struct ip_addr* ip, char *s)
4e2fdd79
 {
 	switch(ip->af){
 		case AF_INET:
9c01c860
 			DBG("%s%d.%d.%d.%d%s", (p)?p:"",
 								ip->u.addr[0],
4e2fdd79
 								ip->u.addr[1],
 								ip->u.addr[2],
9c01c860
 								ip->u.addr[3],
 								(s)?s:""
 								);
4e2fdd79
 			break;
145d6ad0
 #ifdef USE_IPV6
4e2fdd79
 		case AF_INET6:
9c01c860
 			DBG("%s%x:%x:%x:%x:%x:%x:%x:%x%s", (p)?p:"",
 											htons(ip->u.addr16[0]),
4e2fdd79
 											htons(ip->u.addr16[1]),
 											htons(ip->u.addr16[2]),
 											htons(ip->u.addr16[3]),
 											htons(ip->u.addr16[4]),
 											htons(ip->u.addr16[5]),
 											htons(ip->u.addr16[6]),
9c01c860
 											htons(ip->u.addr16[7]),
 											(s)?s:""
4e2fdd79
 				);
 			break;
145d6ad0
 #endif /* USE_IPV6 */
4e2fdd79
 		default:
53c7e0f1
 			DBG("print_ip: warning unknown address family %d\n", ip->af);
4e2fdd79
 	}
 }
 
 
 
 void stdout_print_ip(struct ip_addr* ip)
 {
 	switch(ip->af){
 		case AF_INET:
 			printf("%d.%d.%d.%d",	ip->u.addr[0],
 								ip->u.addr[1],
 								ip->u.addr[2],
 								ip->u.addr[3]);
 			break;
145d6ad0
 #ifdef USE_IPV6
4e2fdd79
 		case AF_INET6:
 			printf("%x:%x:%x:%x:%x:%x:%x:%x",	htons(ip->u.addr16[0]),
 											htons(ip->u.addr16[1]),
 											htons(ip->u.addr16[2]),
 											htons(ip->u.addr16[3]),
 											htons(ip->u.addr16[4]),
 											htons(ip->u.addr16[5]),
 											htons(ip->u.addr16[6]),
 											htons(ip->u.addr16[7])
 				);
 			break;
145d6ad0
 #endif /* USE_IPV6 */
4e2fdd79
 		default:
53c7e0f1
 			DBG("print_ip: warning unknown address family %d\n", ip->af);
4e2fdd79
 	}
 }
 
 
 
 void print_net(struct net* net)
 {
 	if (net==0){
 		LOG(L_WARN, "ERROR: print net: null pointer\n");
 		return;
 	}
9c01c860
 	print_ip("", &net->ip, "/"); print_ip("", &net->mask, "");
4e2fdd79
 }
2ba73117
 
 
 #ifdef USE_MCAST
 
 /* Returns 1 if the given address is a multicast address */
 int is_mcast(struct ip_addr* ip)
 {
 	if (!ip){
 		LOG(L_ERR, "ERROR: is_mcast: Invalid parameter value\n");
 		return -1;
 	}
 
 	if (ip->af==AF_INET){
 		return IN_MULTICAST(htonl(ip->u.addr32[0]));
 #ifdef USE_IPV6
 	} else if (ip->af==AF_INET6){
8a2ca526
 		return IN6_IS_ADDR_MULTICAST((struct in6_addr*)ip->u.addr32);
2ba73117
 #endif /* USE_IPV6 */
 	} else {
 		LOG(L_ERR, "ERROR: is_mcast: Unsupported protocol family\n");
 		return -1;
 	}
 }
 
 #endif /* USE_MCAST */