Annotation of embedaddon/freevrrpd/vrrp_interface.c, revision 1.1.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>