/* $OpenBSD: hello.c,v 1.11 2008/12/28 20:08:31 claudio Exp $ */ /* * Copyright (c) 2005 Claudio Jeker * Copyright (c) 2004, 2005 Esben Norby * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include "ospf6d.h" #include "ospf6.h" #include "log.h" #include "ospfe.h" extern struct ospfd_conf *oeconf; /* hello packet handling */ int send_hello(struct iface *iface) { struct in6_addr dst; struct hello_hdr hello; struct nbr *nbr; struct buf *buf; int ret; u_int32_t opts; switch (iface->type) { case IF_TYPE_POINTOPOINT: case IF_TYPE_BROADCAST: inet_pton(AF_INET6, AllSPFRouters, &dst); break; case IF_TYPE_NBMA: case IF_TYPE_POINTOMULTIPOINT: log_debug("send_hello: type %s not supported, interface %s", if_type_name(iface->type), iface->name); return (-1); case IF_TYPE_VIRTUALLINK: dst = iface->dst; break; default: fatalx("send_hello: unknown interface type"); } /* XXX READ_BUF_SIZE */ if ((buf = buf_dynamic(PKG_DEF_SIZE, READ_BUF_SIZE)) == NULL) fatal("send_hello"); /* OSPF header */ if (gen_ospf_hdr(buf, iface, PACKET_TYPE_HELLO)) goto fail; /* hello header */ hello.iface_id = htonl(iface->ifindex); LSA_24_SETHI(hello.opts, iface->priority); opts = area_ospf_options(area_find(oeconf, iface->area_id)); LSA_24_SETLO(hello.opts, opts); hello.opts = htonl(hello.opts); hello.hello_interval = htons(iface->hello_interval); hello.rtr_dead_interval = htons(iface->dead_interval); if (iface->dr) { hello.d_rtr = iface->dr->id.s_addr; iface->self->dr.s_addr = iface->dr->id.s_addr; } else hello.d_rtr = 0; if (iface->bdr) { hello.bd_rtr = iface->bdr->id.s_addr; iface->self->bdr.s_addr = iface->bdr->id.s_addr; } else hello.bd_rtr = 0; if (buf_add(buf, &hello, sizeof(hello))) goto fail; /* active neighbor(s) */ LIST_FOREACH(nbr, &iface->nbr_list, entry) { if ((nbr->state >= NBR_STA_INIT) && (nbr != iface->self)) if (buf_add(buf, &nbr->id, sizeof(nbr->id))) goto fail; } /* calculate checksum */ if (upd_ospf_hdr(buf, iface)) goto fail; ret = send_packet(iface, buf->buf, buf->wpos, &dst); buf_free(buf); return (ret); fail: log_warn("send_hello"); buf_free(buf); return (-1); } void recv_hello(struct iface *iface, struct in6_addr *src, u_int32_t rtr_id, char *buf, u_int16_t len) { struct hello_hdr hello; struct nbr *nbr = NULL, *dr; struct area *area; u_int32_t nbr_id, opts; int nbr_change = 0; if (len < sizeof(hello) && (len & 0x03)) { log_warnx("recv_hello: bad packet size, interface %s", iface->name); return; } memcpy(&hello, buf, sizeof(hello)); buf += sizeof(hello); len -= sizeof(hello); if (ntohs(hello.hello_interval) != iface->hello_interval) { log_warnx("recv_hello: invalid hello-interval %d, " "interface %s", ntohs(hello.hello_interval), iface->name); return; } if (ntohs(hello.rtr_dead_interval) != iface->dead_interval) { log_warnx("recv_hello: invalid router-dead-interval %d, " "interface %s", ntohl(hello.rtr_dead_interval), iface->name); return; } if ((area = area_find(oeconf, iface->area_id)) == NULL) fatalx("interface lost area"); opts = LSA_24_GETLO(ntohl(hello.opts)); if ((opts & OSPF_OPTION_E && area->stub) || ((opts & OSPF_OPTION_E) == 0 && !area->stub)) { log_warnx("recv_hello: ExternalRoutingCapability mismatch, " "interface %s", iface->name); return; } switch (iface->type) { case IF_TYPE_POINTOPOINT: case IF_TYPE_VIRTUALLINK: /* match router-id */ LIST_FOREACH(nbr, &iface->nbr_list, entry) { if (nbr == iface->self) continue; if (nbr->id.s_addr == rtr_id) break; } break; case IF_TYPE_BROADCAST: case IF_TYPE_NBMA: case IF_TYPE_POINTOMULTIPOINT: /* match src IP */ LIST_FOREACH(nbr, &iface->nbr_list, entry) { if (nbr == iface->self) continue; if (IN6_ARE_ADDR_EQUAL(&nbr->addr, src)) break; } break; default: fatalx("recv_hello: unknown interface type"); } if (!nbr) { nbr = nbr_new(rtr_id, iface, 0); /* set neighbor parameters */ nbr->dr.s_addr = hello.d_rtr; nbr->bdr.s_addr = hello.bd_rtr; nbr->priority = LSA_24_GETHI(ntohl(hello.opts)); nbr_change = 1; } /* actually the neighbor address shouldn't be stored on virtual links */ nbr->addr = *src; nbr->options = opts; nbr_fsm(nbr, NBR_EVT_HELLO_RCVD); while (len >= sizeof(nbr_id)) { memcpy(&nbr_id, buf, sizeof(nbr_id)); if (nbr_id == ospfe_router_id()) { /* seen myself */ if (nbr->state & NBR_STA_PRELIM) nbr_fsm(nbr, NBR_EVT_2_WAY_RCVD); break; } buf += sizeof(nbr_id); len -= sizeof(nbr_id); } if (len == 0) { nbr_fsm(nbr, NBR_EVT_1_WAY_RCVD); /* set neighbor parameters */ nbr->dr.s_addr = hello.d_rtr; nbr->bdr.s_addr = hello.bd_rtr; nbr->priority = LSA_24_GETHI(ntohl(hello.opts)); nbr->iface_id = ntohl(hello.iface_id); return; } if (nbr->priority != LSA_24_GETHI(ntohl(hello.opts))) { nbr->priority = LSA_24_GETHI(ntohl(hello.opts)); nbr_change = 1; } if (iface->state & IF_STA_WAITING && hello.d_rtr == nbr->id.s_addr && hello.bd_rtr == 0) if_fsm(iface, IF_EVT_BACKUP_SEEN); if (iface->state & IF_STA_WAITING && hello.bd_rtr == nbr->id.s_addr) { /* * In case we see the BDR make sure that the DR is around * with a bidirectional (2_WAY or better) connection */ LIST_FOREACH(dr, &iface->nbr_list, entry) if (hello.d_rtr == dr->id.s_addr && dr->state & NBR_STA_BIDIR) if_fsm(iface, IF_EVT_BACKUP_SEEN); } if ((nbr->id.s_addr == nbr->dr.s_addr && nbr->id.s_addr != hello.d_rtr) || (nbr->id.s_addr != nbr->dr.s_addr && nbr->id.s_addr == hello.d_rtr)) /* neighbor changed from or to DR */ nbr_change = 1; if ((nbr->id.s_addr == nbr->bdr.s_addr && nbr->id.s_addr != hello.bd_rtr) || (nbr->id.s_addr != nbr->bdr.s_addr && nbr->id.s_addr == hello.bd_rtr)) /* neighbor changed from or to BDR */ nbr_change = 1; nbr->dr.s_addr = hello.d_rtr; nbr->bdr.s_addr = hello.bd_rtr; if (nbr_change) if_fsm(iface, IF_EVT_NBR_CHNG); /* TODO NBMA needs some special handling */ }