/* $OpenBSD: ospfe.c,v 1.4 2005/02/07 05:51:00 david Exp $ */ /* * Copyright (c) 2005 Claudio Jeker * Copyright (c) 2004 Esben Norby * Copyright (c) 2003, 2004 Henning Brauer * * 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 #include #include #include #include #include #include #include "ospf.h" #include "ospfd.h" #include "ospfe.h" #include "control.h" #include "log.h" void ospfe_sig_handler(int, short, void *); void ospfe_shutdown(void); void orig_lsa(void); void orig_rtr_lsa(struct area *); void orig_net_lsa(struct iface *); volatile sig_atomic_t ospfe_quit = 0; struct ospfd_conf *oeconf = NULL; struct imsgbuf *ibuf_main; struct imsgbuf *ibuf_rde; struct ctl_conn *ctl_conn; void ospfe_sig_handler(int sig, short event, void *bula) { switch (sig) { case SIGINT: case SIGTERM: ospfe_shutdown(); /* NOTREACHED */ default: fatalx("unexpected signal"); /* NOTREACHED */ } } /* ospf engine */ pid_t ospfe(struct ospfd_conf *xconf, int pipe_parent2ospfe[2], int pipe_ospfe2rde[2], int pipe_parent2rde[2]) { struct area *area = NULL; struct iface *iface = NULL; struct passwd *pw; struct event ev_sigint, ev_sigterm; pid_t pid; switch (pid = fork()) { case -1: fatal("cannot fork"); case 0: break; default: return (pid); } /* create ospfd control socket outside chroot */ if (control_init() == -1) fatalx("control socket setup failed"); oeconf = xconf; if ((pw = getpwnam(OSPFD_USER)) == NULL) fatal("getpwnam"); if (chroot(pw->pw_dir) == -1) fatal("chroot"); if (chdir("/") == -1) fatal("chdir(\"/\")"); setproctitle("ospf engine"); ospfd_process = PROC_OSPF_ENGINE; if (setgroups(1, &pw->pw_gid) || setegid(pw->pw_gid) || setgid(pw->pw_gid) || seteuid(pw->pw_uid) || setuid(pw->pw_uid)) { fatal("can't drop privileges"); } endpwent(); event_init(); nbr_init(NBR_HASHSIZE); lsa_cache_init(LSA_HASHSIZE); /* setup signal handler */ signal_set(&ev_sigint, SIGINT, ospfe_sig_handler, NULL); signal_set(&ev_sigterm, SIGTERM, ospfe_sig_handler, NULL); signal_add(&ev_sigint, NULL); signal_add(&ev_sigterm, NULL); signal(SIGPIPE, SIG_IGN); /* setup pipes */ close(pipe_parent2ospfe[0]); close(pipe_ospfe2rde[1]); close(pipe_parent2rde[0]); close(pipe_parent2rde[1]); if ((ibuf_rde = malloc(sizeof(struct imsgbuf))) == NULL || (ibuf_main = malloc(sizeof(struct imsgbuf))) == NULL) fatal(NULL); imsg_init(ibuf_rde, pipe_ospfe2rde[0], ospfe_dispatch_rde); imsg_init(ibuf_main, pipe_parent2ospfe[1], ospfe_dispatch_main); /* setup event handler */ ibuf_rde->events = EV_READ; event_set(&ibuf_rde->ev, ibuf_rde->fd, ibuf_rde->events, ibuf_rde->handler, ibuf_rde); event_add(&ibuf_rde->ev, NULL); ibuf_main->events = EV_READ; event_set(&ibuf_main->ev, ibuf_main->fd, ibuf_main->events, ibuf_main->handler, ibuf_main); event_add(&ibuf_main->ev, NULL); event_set(&oeconf->ev, oeconf->ospf_socket, EV_READ|EV_PERSIST, recv_packet, oeconf); event_add(&oeconf->ev, NULL); /* listen on ospfd control socket */ TAILQ_INIT(&ctl_conns); control_listen(); /* start interfaces */ LIST_FOREACH(area, &oeconf->area_list, entry) { LIST_FOREACH(iface, &area->iface_list, entry) { if (if_fsm(iface, IF_EVT_UP)) { log_debug("error starting interface %s", iface->name); } } } orig_lsa(); event_dispatch(); ospfe_shutdown(); /* NOTREACHED */ return (0); } void ospfe_shutdown(void) { struct area *area; struct iface *iface; /* stop all interfaces and remove all areas */ LIST_FOREACH(area, &oeconf->area_list, entry) { LIST_FOREACH(iface, &area->iface_list, entry) { if (if_fsm(iface, IF_EVT_DOWN)) { log_debug("error stopping interface %s", iface->name); } } area_del(area); } /* clean up */ msgbuf_write(&ibuf_rde->w); msgbuf_clear(&ibuf_rde->w); free(ibuf_rde); msgbuf_write(&ibuf_main->w); msgbuf_clear(&ibuf_main->w); free(ibuf_main); log_info("ospf engine exiting"); _exit(0); } /* imesg */ int ospfe_imsg_compose_parent(int type, pid_t pid, void *data, u_int16_t datalen) { return (imsg_compose(ibuf_main, type, 0, pid, -1, data, datalen)); } int ospfe_imsg_compose_rde(int type, u_int32_t peerid, pid_t pid, void *data, u_int16_t datalen) { return (imsg_compose(ibuf_rde, type, peerid, pid, -1, data, datalen)); } void ospfe_dispatch_main(int fd, short event, void *bula) { struct imsg imsg; struct imsgbuf *ibuf = bula; struct area *area = NULL; struct iface *iface = NULL; struct kif *kif; int n; switch (event) { case EV_READ: if ((n = imsg_read(ibuf)) == -1) fatal("imsg_read error"); if (n == 0) /* connection closed */ fatalx("pipe closed"); break; case EV_WRITE: if (msgbuf_write(&ibuf->w) == -1) fatal("msgbuf_write"); imsg_event_add(ibuf); return; default: fatalx("unknown event"); } for (;;) { if ((n = imsg_get(ibuf, &imsg)) == -1) fatal("ospfe_dispatch_main: imsg_read error"); if (n == 0) break; switch (imsg.hdr.type) { case IMSG_IFINFO: if (imsg.hdr.len != IMSG_HEADER_SIZE + sizeof(struct kif)) fatalx("IFINFO imsg with wrong len"); kif = imsg.data; log_debug("ospfe_dispatch_main: ifindex %d changed", kif->ifindex); LIST_FOREACH(area, &oeconf->area_list, entry) { LIST_FOREACH(iface, &area->iface_list, entry) { if (kif->ifindex == iface->ifindex) { if (kif->flags & IFF_UP) { if_fsm(iface, IF_EVT_UP); } else { if_fsm(iface, IF_EVT_DOWN); } } } } break; case IMSG_CTL_END: log_debug("ospfe_dispatch_main: IMSG_CTL_END"); control_imsg_relay(&imsg); break; default: log_debug("ospfe_dispatch_main: error handling imsg %d", imsg.hdr.type); break; } imsg_free(&imsg); } imsg_event_add(ibuf); } void ospfe_dispatch_rde(int fd, short event, void *bula) { struct lsa_hdr lsa_hdr; struct imsgbuf *ibuf = bula; struct nbr *nbr; struct lsa_hdr *lhp; struct lsa_ref *ref; struct area *area; struct iface *iface; struct lsa_entry *le; struct imsg imsg; int n, noack = 0; u_int16_t l, age; switch (event) { case EV_READ: if ((n = imsg_read(ibuf)) == -1) fatal("imsg_read error"); if (n == 0) /* connection closed */ fatalx("pipe closed"); break; case EV_WRITE: if (msgbuf_write(&ibuf->w) == -1) fatal("msgbuf_write"); imsg_event_add(ibuf); return; default: fatalx("unknown event"); } for (;;) { if ((n = imsg_get(ibuf, &imsg)) == -1) fatal("ospfe_dispatch_rde: imsg_read error"); if (n == 0) break; switch (imsg.hdr.type) { case IMSG_DD: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_DD, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); /* put these on my ls_req_list for retrieval */ lhp = lsa_hdr_new(); memcpy(lhp, imsg.data, sizeof(*lhp)); ls_req_list_add(nbr, lhp); break; case IMSG_DB_SNAPSHOT: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_DB_SNAPSHOT, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); /* add LSA header to the neighbor db_sum_list */ lhp = lsa_hdr_new(); memcpy(lhp, imsg.data, sizeof(*lhp)); db_sum_list_add(nbr, lhp); break; case IMSG_DB_END: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_DB_END, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); /* snapshot done, start tx of dd packets */ nbr_fsm(nbr, NBR_EVT_SNAP_DONE); break; case IMSG_LS_FLOOD: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_LS_FLOOD, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); l = imsg.hdr.len - IMSG_HEADER_SIZE; if (l < sizeof(lsa_hdr)) fatalx("ospfe_dispatch_rde: " "bad imsg size"); memcpy(&lsa_hdr, imsg.data, sizeof(lsa_hdr)); ref = lsa_cache_add(imsg.data, l); if (lsa_hdr.type == LSA_TYPE_EXTERNAL) { /* * flood on all areas but stub areas and * virtual links */ LIST_FOREACH(area, &oeconf->area_list, entry) { if (area->stub) continue; LIST_FOREACH(iface, &area->iface_list, entry) { noack += lsa_flood(iface, nbr, &lsa_hdr, imsg.data, l); } } } else { /* * flood on all area interfaces on * area 0.0.0.0 include also virtual links. */ area = nbr->iface->area; LIST_FOREACH(iface, &area->iface_list, entry) { noack += lsa_flood(iface, nbr, &lsa_hdr, imsg.data, l); } /* XXX virtual links */ } /* remove from ls_req_list */ le = ls_req_list_get(nbr, &lsa_hdr); if (!(nbr->state & NBR_STA_FULL) && le != NULL) { ls_req_list_free(nbr, le); /* XXX no need to ack requested lsa */ noack = 1; } if (!noack && nbr->iface->self != nbr) { if (!(nbr->iface->state & IF_STA_BACKUP) || nbr->iface->dr == nbr) { /* delayed ack */ lhp = lsa_hdr_new(); memcpy(lhp, &lsa_hdr, sizeof(*lhp)); ls_ack_list_add(nbr->iface, lhp); } } lsa_cache_put(ref, nbr); break; case IMSG_LS_UPD: /* * IMSG_LS_UPD is used in three cases: * 1. as response to ls requests * 2. as response to ls updates where the DB * is newer then the sent LSA * 3. in EXSTART when the LSA has age MaxAge */ l = imsg.hdr.len - IMSG_HEADER_SIZE; if (l < sizeof(lsa_hdr)) fatalx("ospfe_dispatch_rde: " "bad imsg size"); nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_LS_UPD, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); if (nbr->iface->self == nbr) break; memcpy(&age, imsg.data, sizeof(age)); if (ntohs(age) >= MAX_AGE) { /* add to retransmit list */ ref = lsa_cache_add(imsg.data, l); ls_retrans_list_add(nbr, imsg.data); /* XXX */ lsa_cache_put(ref, nbr); } /* send direct don't add to retransmit list */ send_ls_update(nbr->iface, nbr->addr, imsg.data, l); break; case IMSG_LS_ACK: /* * IMSG_LS_ACK is used in two cases: * 1. LSA was a duplicate * 2. LSA's age is MaxAge and there is no current * instance in the DB plus no neighbor is state * Exchange or Loading */ nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_LS_ACK, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); if (nbr->iface->self == nbr) break; /* TODO for case one check for implied acks */ /* send a direct acknowledgement */ send_ls_ack(nbr->iface, nbr->addr, imsg.data, imsg.hdr.len - IMSG_HEADER_SIZE); break; case IMSG_LS_BADREQ: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); log_debug("ospfe_dispatch_rde: IMSG_LS_BADREQ, " "neighbor id %s, len %d", inet_ntoa(nbr->id), imsg.hdr.len - IMSG_HEADER_SIZE); if (nbr->iface->self == nbr) fatalx("ospfe_dispatch_rde: " "dummy neighbor got BADREQ"); nbr_fsm(nbr, NBR_EVT_BAD_LS_REQ); break; case IMSG_CTL_SHOW_DATABASE: case IMSG_CTL_AREA: case IMSG_CTL_END: control_imsg_relay(&imsg); break; default: log_debug("ospfe_dispatch_rde: error handling imsg %d", imsg.hdr.type); break; } imsg_free(&imsg); } imsg_event_add(ibuf); } void orig_lsa(void) { struct area *area; struct iface *iface; LIST_FOREACH(area, &oeconf->area_list, entry) { orig_rtr_lsa(area); LIST_FOREACH(iface, &area->iface_list, entry) orig_net_lsa(iface); } } void orig_rtr_lsa(struct area *area) { struct lsa *lsa; struct iface *iface; char *buf; struct lsa_rtr_link *rtr_link; int num_links = 0; u_int16_t len; log_debug("orig_rtr_lsa: area %s", inet_ntoa(area->id)); if ((buf = calloc(1, READ_BUF_SIZE)) == NULL) fatal("orig_rtr_lsa"); /* LSA header */ lsa = (struct lsa *)buf; lsa->hdr.age = htons(DEFAULT_AGE); /* XXX */ lsa->hdr.opts = oeconf->options; /* XXX */ lsa->hdr.type = LSA_TYPE_ROUTER; lsa->hdr.ls_id = oeconf->rtr_id.s_addr; lsa->hdr.adv_rtr = oeconf->rtr_id.s_addr; lsa->hdr.seq_num = htonl(INIT_SEQ_NUM); lsa->hdr.ls_chksum = 0; /* updated later */ lsa->hdr.len = 0; /* updated later */ len = sizeof(struct lsa_hdr) + sizeof(struct lsa_rtr); /* links */ LIST_FOREACH(iface, &area->iface_list, entry) { log_debug("orig_rtr_lsa: interface %s", iface->name); rtr_link = (struct lsa_rtr_link *)(buf + len); switch (iface->type) { case IF_TYPE_POINTOPOINT: case IF_TYPE_VIRTUALLINK: log_debug("orig_rtr_lsa: not supported, interface %s", iface->name); break; case IF_TYPE_BROADCAST: if (iface->state == IF_STA_DOWN) break; if ((iface->state == IF_STA_WAITING) || iface->dr == NULL) { /* XXX */ log_debug("orig_rtr_lsa: stub net, " "interface %s", iface->name); rtr_link->id = iface->addr.s_addr & iface->mask.s_addr; rtr_link->data = iface->mask.s_addr; rtr_link->type = LINK_TYPE_STUB_NET; } else { log_debug("orig_rtr_lsa: transit net, " "interface %s", iface->name); rtr_link->id = iface->dr->addr.s_addr; rtr_link->data = iface->addr.s_addr; rtr_link->type = LINK_TYPE_TRANSIT_NET; } rtr_link->num_tos = 0; rtr_link->metric = htons(iface->metric); num_links++; len += sizeof(struct lsa_rtr_link); break; case IF_TYPE_NBMA: case IF_TYPE_POINTOMULTIPOINT: log_debug("orig_rtr_lsa: not supported, interface %s", iface->name); break; default: fatalx("orig_rtr_lsa: unknown interface type"); } } lsa->data.rtr.flags = 0; /* XXX */ lsa->data.rtr.dummy = 0; lsa->data.rtr.nlinks = htons(num_links); lsa->hdr.len = htons(len); lsa->hdr.ls_chksum = htons(iso_cksum(lsa, len, LS_CKSUM_OFFSET)); imsg_compose(ibuf_rde, IMSG_LS_UPD, LIST_FIRST(&area->iface_list)->self->peerid, 0, -1, buf, ntohs(lsa->hdr.len)); } void orig_net_lsa(struct iface *iface) { struct lsa *lsa; struct nbr *nbr; char *buf; int num_rtr = 0; u_int16_t len; log_debug("orig_net_lsa: iface %s", iface->name); /* XXX if not dr quit */ /* ... */ if ((buf = calloc(1, READ_BUF_SIZE)) == NULL) fatal("orig_net_lsa"); /* LSA header */ lsa = (struct lsa *)buf; lsa->hdr.age = htons(DEFAULT_AGE); /* XXX */ lsa->hdr.opts = oeconf->options; /* XXX */ lsa->hdr.type = LSA_TYPE_NETWORK; lsa->hdr.ls_id = oeconf->rtr_id.s_addr; lsa->hdr.adv_rtr = oeconf->rtr_id.s_addr; lsa->hdr.seq_num = htonl(INIT_SEQ_NUM); lsa->hdr.ls_chksum = 0; /* updated later */ lsa->hdr.len = 0; /* updated later */ len = sizeof(lsa->hdr); lsa->data.net.mask = iface->mask.s_addr; len += sizeof(lsa->data.net.mask); /* attached routers */ /* fully adjacent neighbors + self */ LIST_FOREACH(nbr, &iface->nbr_list, entry) { if (nbr->state != NBR_STA_FULL) { /* XXX */ lsa->data.net.att_rtr[num_rtr] = nbr->id.s_addr; num_rtr++; len += sizeof(nbr->id); } } lsa->hdr.len = htons(len); lsa->hdr.ls_chksum = htons(iso_cksum(lsa, len, LS_CKSUM_OFFSET)); /* imsg_compose(ibuf_rde, IMSG_LS_UPD, iface->self->peerid, 0, -1, buf, ntohs(lsa->hdr.len)); */ } u_int32_t ospfe_router_id(void) { return (oeconf->rtr_id.s_addr); } void ospfe_iface_ctl(struct ctl_conn *c, unsigned int idx) { struct area *area; struct iface *iface; struct ctl_iface *ictl; LIST_FOREACH(area, &oeconf->area_list, entry) LIST_FOREACH(iface, &area->iface_list, entry) if (idx == 0 || idx == iface->ifindex) { ictl = if_to_ctl(iface); imsg_compose(&c->ibuf, IMSG_CTL_SHOW_INTERFACE, 0, 0, -1, ictl, sizeof(struct ctl_iface)); } } void ospfe_nbr_ctl(struct ctl_conn *c) { struct area *area; struct iface *iface; struct nbr *nbr; struct ctl_nbr *nctl; LIST_FOREACH(area, &oeconf->area_list, entry) LIST_FOREACH(iface, &area->iface_list, entry) LIST_FOREACH(nbr, &iface->nbr_list, entry) { if (iface->self != nbr) { nctl = nbr_to_ctl(nbr); imsg_compose(&c->ibuf, IMSG_CTL_SHOW_NBR, 0, 0, -1, nctl, sizeof(struct ctl_nbr)); } } imsg_compose(&c->ibuf, IMSG_CTL_END, 0, 0, -1, NULL, 0); }