/* $OpenBSD: ospfe.c,v 1.6 2005/02/10 14:05:48 claudio 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); 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); } } } 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_DD_END: nbr = nbr_find_peerid(imsg.hdr.peerid); if (nbr == NULL) fatalx("ospfe_dispatch_rde: " "neighbor not found"); nbr->dd_pending--; if (nbr->dd_pending == 0 && nbr->state & NBR_STA_LOAD) { if (ls_req_list_empty(nbr)) nbr_fsm(nbr, NBR_EVT_LOAD_DONE); else start_ls_req_tx_timer(nbr); } 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_rtr_lsa(struct area *area) { struct lsa_hdr lsa_hdr; struct lsa_rtr lsa_rtr; struct lsa_rtr_link rtr_link; struct iface *iface; struct buf *buf; struct nbr *nbr, *self = NULL; u_int16_t num_links = 0; u_int16_t chksum; log_debug("orig_rtr_lsa: area %s", inet_ntoa(area->id)); if ((buf = buf_dynamic(sizeof(lsa_hdr), READ_BUF_SIZE /* XXX */)) == NULL) fatal("orig_rtr_lsa"); /* reserve space for LSA header and LSA Router header */ if (buf_reserve(buf, sizeof(lsa_hdr)) == NULL) fatal("orig_rtr_lsa: buf_reserve failed"); if (buf_reserve(buf, sizeof(lsa_rtr)) == NULL) fatal("orig_rtr_lsa: buf_reserve failed"); /* links */ LIST_FOREACH(iface, &area->iface_list, entry) { log_debug("orig_rtr_lsa: interface %s", iface->name); if (self == NULL && iface->self != NULL) self = iface->self; if (iface->state & IF_STA_DOWN) continue; bzero(&rtr_link, sizeof(rtr_link)); if (iface->state & IF_STA_LOOPBACK) { rtr_link.id = iface->addr.s_addr; rtr_link.data = 0xffffffff; rtr_link.type = LINK_TYPE_STUB_NET; num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: buf_add failed"); continue; } switch (iface->type) { case IF_TYPE_POINTOPOINT: LIST_FOREACH(nbr, &iface->nbr_list, entry) if (nbr != iface->self && nbr->state & NBR_STA_FULL) break; if (nbr) { log_debug("orig_rtr_lsa: point-to-point, " "interface %s", iface->name); rtr_link.id = nbr->addr.s_addr; rtr_link.data = iface->addr.s_addr; rtr_link.type = LINK_TYPE_POINTTOPOINT; rtr_link.metric = htons(iface->metric); num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: buf_add failed"); } if (iface->state & IF_STA_POINTTOPOINT) { log_debug("orig_rtr_lsa: stub net, " "interface %s", iface->name); bzero(&rtr_link, sizeof(rtr_link)); if (nbr) { rtr_link.id = nbr->addr.s_addr; rtr_link.data = 0xffffffff; } else { rtr_link.id = iface->addr.s_addr; rtr_link.data = iface->mask.s_addr; } rtr_link.metric = htons(iface->metric); rtr_link.type = LINK_TYPE_STUB_NET; num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: buf_add failed"); } continue; case IF_TYPE_BROADCAST: case IF_TYPE_NBMA: if ((iface->state & IF_STA_MULTI)) { if (iface->dr == iface->self) { LIST_FOREACH(nbr, &iface->nbr_list, entry) if (nbr != iface->self && nbr->state & NBR_STA_FULL) break; } else nbr = iface->dr; if (nbr && nbr->state & NBR_STA_FULL) { 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; break; } } 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; break; case IF_TYPE_VIRTUALLINK: log_debug("orig_rtr_lsa: not supported, interface %s", iface->name); continue; case IF_TYPE_POINTOMULTIPOINT: log_debug("orig_rtr_lsa: stub net, " "interface %s", iface->name); rtr_link.id = iface->addr.s_addr; rtr_link.data = 0xffffffff; rtr_link.type = LINK_TYPE_STUB_NET; num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: buf_add failed"); LIST_FOREACH(nbr, &iface->nbr_list, entry) { if (nbr != iface->self && nbr->state & NBR_STA_FULL) { bzero(&rtr_link, sizeof(rtr_link)); log_debug("orig_rtr_lsa: " "point-to-point, interface %s", iface->name); rtr_link.id = nbr->addr.s_addr; rtr_link.data = iface->addr.s_addr; rtr_link.type = LINK_TYPE_POINTTOPOINT; rtr_link.metric = htons(iface->metric); num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: " "buf_add failed"); } } continue; default: fatalx("orig_rtr_lsa: unknown interface type"); } rtr_link.num_tos = 0; rtr_link.metric = htons(iface->metric); num_links++; if (buf_add(buf, &rtr_link, sizeof(rtr_link))) fatalx("orig_rtr_lsa: buf_add failed"); } /* LSA router header */ lsa_rtr.flags = 0; /* XXX */ lsa_rtr.dummy = 0; lsa_rtr.nlinks = htons(num_links); memcpy(buf_seek(buf, sizeof(lsa_hdr), sizeof(lsa_rtr)), &lsa_rtr, sizeof(lsa_rtr)); /* LSA header */ lsa_hdr.age = htons(DEFAULT_AGE); 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.len = htons(buf->wpos); lsa_hdr.ls_chksum = 0; /* updated later */ memcpy(buf_seek(buf, 0, sizeof(lsa_hdr)), &lsa_hdr, sizeof(lsa_hdr)); chksum = htons(iso_cksum(buf->buf, buf->wpos, LS_CKSUM_OFFSET)); memcpy(buf_seek(buf, LS_CKSUM_OFFSET, sizeof(chksum)), &chksum, sizeof(chksum)); if (self) imsg_compose(ibuf_rde, IMSG_LS_UPD, self->peerid, 0, -1, buf->buf, buf->wpos); else log_warnx("orig_rtr_lsa: empty area %s", inet_ntoa(area->id)); buf_free(buf); } void orig_net_lsa(struct iface *iface) { struct lsa_hdr lsa_hdr; struct nbr *nbr; struct buf *buf; int num_rtr = 0; u_int16_t chksum; log_debug("orig_net_lsa: iface %s", iface->name); if ((buf = buf_dynamic(sizeof(lsa_hdr), READ_BUF_SIZE /* XXX */)) == NULL) fatal("orig_net_lsa"); /* reserve space for LSA header and LSA Router header */ if (buf_reserve(buf, sizeof(lsa_hdr)) == NULL) fatal("orig_net_lsa: buf_reserve failed"); /* LSA net mask and then all fully adjacent routers */ if (buf_add(buf, &iface->mask, sizeof(iface->mask))) fatal("orig_net_lsa: buf_add failed"); /* fully adjacent neighbors + self */ LIST_FOREACH(nbr, &iface->nbr_list, entry) if (nbr->state & NBR_STA_FULL) { if (buf_add(buf, &nbr->id, sizeof(nbr->id))) fatal("orig_net_lsa: buf_add failed"); num_rtr++; } if (num_rtr == 1) { /* non transit net therefor no need to generate a net lsa */ buf_free(buf); return; } /* LSA header */ if (iface->state & IF_STA_DR) lsa_hdr.age = htons(DEFAULT_AGE); else lsa_hdr.age = htons(MAX_AGE); lsa_hdr.opts = oeconf->options; /* XXX */ lsa_hdr.type = LSA_TYPE_NETWORK; lsa_hdr.ls_id = iface->addr.s_addr; lsa_hdr.adv_rtr = oeconf->rtr_id.s_addr; lsa_hdr.seq_num = htonl(INIT_SEQ_NUM); lsa_hdr.len = htons(buf->wpos); lsa_hdr.ls_chksum = 0; /* updated later */ memcpy(buf_seek(buf, 0, sizeof(lsa_hdr)), &lsa_hdr, sizeof(lsa_hdr)); chksum = htons(iso_cksum(buf->buf, buf->wpos, LS_CKSUM_OFFSET)); memcpy(buf_seek(buf, LS_CKSUM_OFFSET, sizeof(chksum)), &chksum, sizeof(chksum)); imsg_compose(ibuf_rde, IMSG_LS_UPD, iface->self->peerid, 0, -1, buf->buf, buf->wpos); buf_free(buf); } 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); }