Annotation of embedaddon/freevrrpd/vrrp_interface.c, revision 1.1
1.1 ! misho 1: /*
! 2: * Copyright (c) 2001,2002 Sebastien Petit <spe@bsdfr.org>
! 3: *
! 4: * Redistribution and use in source forms, with and without modification,
! 5: * are permitted provided that the following conditions are met:
! 6: * 1. Redistributions of source code must retain the above copyright notice,
! 7: * this list of conditions and the following disclaimer.
! 8: * 2. Redistributions in binary form must reproduce the above copyright notice,
! 9: * this list of conditions and the following disclaimer in the documentation
! 10: * and/or other materials provided with the distribution. Obviously, it
! 11: * would be nice if you gave credit where credit is due but requiring it
! 12: * would be too onerous.
! 13: * 3. All advertising materials mentioning features or use of this software
! 14: * must display the following acknowledgement:
! 15: * This product includes software developed by Sebastien Petit.
! 16: * 4. Neither the name of its contributors may be used to endorse or promote
! 17: * products derived from this software without specific prior written
! 18: * permission.
! 19: *
! 20: * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
! 21: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
! 22: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
! 23: * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
! 24: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
! 25: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
! 26: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
! 27: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
! 28: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
! 29: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
! 30: * SUCH DAMAGE.
! 31: *
! 32: * $Id: vrrp_interface.c,v 1.5 2004/04/01 13:14:54 spe Exp $
! 33: */
! 34:
! 35: #include "vrrp_interface.h"
! 36:
! 37: void
! 38: vrrp_interface_owner_verify(struct vrrp_vr * vr)
! 39: {
! 40: int cpt, cpt2;
! 41:
! 42: for (cpt = 0; cpt < vr->cnt_ip; cpt++)
! 43: for (cpt2 = 0; cpt2 < vr->vr_if->nb_ip; cpt2++)
! 44: if (vr->vr_ip[cpt].addr.s_addr == vr->vr_if->ip_addrs[cpt2].s_addr)
! 45: vr->vr_ip[cpt].owner = VRRP_INTERFACE_IPADDR_OWNER;
! 46:
! 47: return;
! 48: }
! 49:
! 50: char
! 51: vrrp_interface_mac_set(char *if_name, struct ether_addr * ethaddr)
! 52: {
! 53: int sd;
! 54: struct ifreq ifr;
! 55:
! 56: bzero(&ifr, sizeof(ifr));
! 57: sd = socket(AF_INET, SOCK_DGRAM, 0);
! 58: if (sd == -1) {
! 59: syslog(LOG_WARNING, "cannot open socket for deleting ip address of interface %s: %s", if_name, strerror(errno));
! 60: return -1;
! 61: }
! 62: strncpy(ifr.ifr_name, if_name, sizeof(ifr.ifr_name));
! 63: ifr.ifr_addr.sa_len = ETHER_ADDR_LEN;
! 64: ifr.ifr_addr.sa_family = AF_LINK;
! 65: bcopy(ethaddr, ifr.ifr_addr.sa_data, ETHER_ADDR_LEN);
! 66: if (ioctl(sd, SIOCSIFLLADDR, (caddr_t) & ifr) == -1) {
! 67: syslog(LOG_ERR, "cannot set mac address for interface %s (ioctl): %s", if_name, strerror(errno));
! 68: return -1;
! 69: }
! 70:
! 71: return 0;
! 72: }
! 73:
! 74: /* Set VMAC for parent interface + all child VLANs */
! 75: int
! 76: vrrp_interface_all_ethaddr_set(struct vrrp_vr *vr, struct ether_addr *ethaddr) {
! 77: struct vrrp_vlan_list *vl = vr->vr_if->vlanp->next;
! 78:
! 79: while (vl != vr->vr_if->vland) {
! 80: if (vrrp_interface_mac_set(vl->vlan_ifname, ethaddr) == -1)
! 81: return -1;
! 82: vl = vl->next;
! 83: }
! 84:
! 85: return 0;
! 86: }
! 87:
! 88: in_addr_t
! 89: vrrp_interface_compute_netmask(u_int nbbits)
! 90: {
! 91: int cpt = 0;
! 92: in_addr_t netmask = 0;
! 93:
! 94: if (nbbits > 32) {
! 95: syslog(LOG_ERR, "specified netmask is invalid: /%u", nbbits);
! 96: syslog(LOG_ERR, "netmask /32 or 255.255.255.255 is applied");
! 97: return 0xFFFFFFFF;
! 98: }
! 99: while (cpt < nbbits) {
! 100: netmask |= (in_addr_t) pow(2, 31 - cpt);
! 101: cpt++;
! 102: }
! 103:
! 104: return netmask;
! 105: }
! 106:
! 107: char
! 108: vrrp_interface_ipaddr_set(char *if_name, struct in_addr addr, in_addr_t netmask)
! 109: {
! 110: int sd;
! 111: struct ifaliasreq ifra;
! 112:
! 113: bzero(&ifra, sizeof(ifra));
! 114: sd = socket(AF_INET, SOCK_DGRAM, 0);
! 115: if (sd == -1) {
! 116: syslog(LOG_WARNING, "cannot open socket for adding ip address of interface %s: %s", if_name, strerror(errno));
! 117: return -1;
! 118: }
! 119: strncpy(ifra.ifra_name, if_name, sizeof(ifra.ifra_name));
! 120: ifra.ifra_addr.sa_len = sizeof(struct sockaddr_in);
! 121: ifra.ifra_addr.sa_family = AF_INET;
! 122: ((struct sockaddr_in *) & ifra.ifra_addr)->sin_addr = addr;
! 123: ifra.ifra_mask.sa_len = sizeof(struct sockaddr_in);
! 124: ifra.ifra_mask.sa_family = AF_INET;
! 125: ((struct sockaddr_in *) & ifra.ifra_mask)->sin_addr.s_addr = htonl(netmask);
! 126: if (ioctl(sd, SIOCAIFADDR, &ifra) == -1) {
! 127: syslog(LOG_ERR, "cannot set ip addr %s for interface %s (ioctl SIOCAIFADDR): %s", inet_ntoa(addr), if_name, strerror(errno));
! 128: close(sd);
! 129: return -1;
! 130: }
! 131: close(sd);
! 132:
! 133: return 0;
! 134: }
! 135:
! 136: char
! 137: vrrp_interface_ipaddr_delete(char *if_name, struct in_addr addr, int verbose)
! 138: {
! 139: int sd;
! 140: struct ifreq ifr;
! 141:
! 142: bzero(&ifr, sizeof(ifr));
! 143: sd = socket(AF_INET, SOCK_DGRAM, 0);
! 144: if (sd == -1) {
! 145: syslog(LOG_WARNING, "cannot open socket for deleting ip address of interface %s: %s", if_name, strerror(errno));
! 146: return -1;
! 147: }
! 148: strncpy(ifr.ifr_name, if_name, sizeof(ifr.ifr_name));
! 149: ifr.ifr_addr.sa_len = sizeof(struct sockaddr_in);
! 150: ifr.ifr_addr.sa_family = AF_INET;
! 151: ((struct sockaddr_in *) & ifr.ifr_addr)->sin_addr = addr;
! 152: if ((ioctl(sd, SIOCDIFADDR, (char *)&ifr) == -1) && verbose) {
! 153: syslog(LOG_ERR, "cannot delete ip addr %s for interface %s (ioctl SIOCDIFADDR): %s", inet_ntoa(addr), if_name, strerror(errno));
! 154: close(sd);
! 155: return -1;
! 156: }
! 157: close(sd);
! 158:
! 159: return 0;
! 160: }
! 161:
! 162: char
! 163: vrrp_interface_vripaddr_set(struct vrrp_vr * vr)
! 164: {
! 165: int cpt;
! 166:
! 167: for (cpt = 0; cpt < vr->cnt_ip; cpt++) {
! 168: if (!vr->vr_ip[cpt].owner) {
! 169: if (vrrp_interface_ipaddr_set(vr->viface_name, vr->vr_ip[cpt].addr, vrrp_interface_compute_netmask(vr->vr_netmask[cpt])) == -1) {
! 170: if (errno != EEXIST) {
! 171: syslog(LOG_ERR, "vrid [%d] Error occured during setting virtual router ip address %s", vr->vr_id, inet_ntoa(vr->vr_ip[cpt].addr));
! 172: return -1;
! 173: }
! 174: }
! 175: }
! 176: }
! 177:
! 178: return 0;
! 179: }
! 180:
! 181: char
! 182: vrrp_interface_vripaddr_delete(struct vrrp_vr * vr)
! 183: {
! 184: int cpt;
! 185:
! 186: for (cpt = 0; cpt < vr->cnt_ip; cpt++) {
! 187: if (vr->vr_ip[cpt].owner != VRRP_INTERFACE_IPADDR_OWNER) {
! 188: if (vrrp_interface_ipaddr_delete(vr->viface_name, vr->vr_ip[cpt].addr, 0) == -1) {
! 189: if (errno != EADDRNOTAVAIL) {
! 190: syslog(LOG_ERR, "vrid [%d] Error occured during deleting virtual router ip address %s", vr->vr_id, inet_ntoa(vr->vr_ip[cpt].addr));
! 191: return -1;
! 192: }
! 193: }
! 194: }
! 195: }
! 196:
! 197: return 0;
! 198: }
! 199:
! 200:
! 201:
! 202: char
! 203: vrrp_interface_set_flags(char *if_name, int value) {
! 204: int sd;
! 205: struct ifreq ifr;
! 206: int flags;
! 207:
! 208: bzero(&ifr, sizeof(ifr));
! 209: strncpy(ifr.ifr_name, if_name, sizeof(ifr.ifr_name));
! 210: sd = socket(AF_INET, SOCK_DGRAM, 0);
! 211: if (sd == -1)
! 212: return -1;
! 213: if (ioctl(sd, SIOCGIFFLAGS, (caddr_t)&ifr) < 0) {
! 214: close(sd);
! 215: return -1;
! 216: }
! 217: flags = (ifr.ifr_flags & 0xffff) | (ifr.ifr_flagshigh << 16);
! 218:
! 219: if (value < 0) {
! 220: value = -value;
! 221: flags &= ~value;
! 222: } else
! 223: flags |= value;
! 224: ifr.ifr_flags = flags & 0xffff;
! 225: ifr.ifr_flagshigh = flags >> 16;
! 226:
! 227: if (ioctl(sd, SIOCSIFFLAGS, (caddr_t)&ifr) < 0) {
! 228: close(sd);
! 229: return -1;
! 230: }
! 231:
! 232: close(sd);
! 233:
! 234: return 0;
! 235: }
! 236:
! 237: char
! 238: vrrp_interface_promiscuous(char *if_name) {
! 239: return vrrp_interface_set_flags(if_name, IFF_PPROMISC);
! 240: }
! 241:
! 242: char
! 243: vrrp_interface_nopromiscuous(char *if_name) {
! 244: return vrrp_interface_set_flags(if_name, -IFF_PPROMISC);
! 245: }
! 246:
! 247: char
! 248: vrrp_interface_up(char *if_name) {
! 249: return vrrp_interface_set_flags(if_name, IFF_UP);
! 250: }
! 251:
! 252: char
! 253: vrrp_interface_down(char *if_name) {
! 254: return vrrp_interface_set_flags(if_name, -IFF_UP);
! 255: }
! 256:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>