/* $OpenBSD: labelmapping.c,v 1.17 2010/11/04 09:52:16 claudio Exp $ */ /* * Copyright (c) 2009 Michele Marchetto * * 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 "ldpd.h" #include "ldp.h" #include "log.h" #include "ldpe.h" void gen_label_tlv(struct ibuf *, u_int32_t); void gen_reqid_tlv(struct ibuf *, u_int32_t); void gen_fec_tlv(struct ibuf *, struct in_addr, u_int8_t); int tlv_decode_label(char *, u_int16_t, u_int32_t *); int tlv_decode_reqid(char *, u_int16_t, u_int32_t *); int tlv_decode_fec_elm(char *, u_int16_t, u_int8_t *, u_int32_t *, u_int8_t *); /* Label Mapping Message */ void send_labelmapping(struct nbr *nbr) { struct ibuf *buf; struct mapping_entry *me; struct ldp_hdr *ldp_hdr; u_int16_t tlv_size, size; if (nbr->iface->passive) return; if ((buf = ibuf_open(LDP_MAX_LEN)) == NULL) fatal("send_labelmapping"); /* real size will be set up later */ gen_ldp_hdr(buf, nbr->iface, 0); size = LDP_HDR_SIZE - TLV_HDR_LEN; TAILQ_FOREACH(me, &nbr->mapping_list, entry) { tlv_size = BASIC_LABEL_MAP_LEN + PREFIX_SIZE(me->map.prefixlen); if (me->map.flags & F_MAP_REQ_ID) tlv_size += REQID_TLV_LEN; size += tlv_size; gen_msg_tlv(buf, MSG_TYPE_LABELMAPPING, tlv_size); gen_fec_tlv(buf, me->map.prefix, me->map.prefixlen); gen_label_tlv(buf, me->map.label); if (me->map.flags & F_MAP_REQ_ID) gen_reqid_tlv(buf, me->map.requestid); } /* XXX: should we remove them first? */ nbr_mapping_list_clr(nbr, &nbr->mapping_list); ldp_hdr = ibuf_seek(buf, 0, sizeof(struct ldp_hdr)); ldp_hdr->length = htons(size); evbuf_enqueue(&nbr->wbuf, buf); } int recv_labelmapping(struct nbr *nbr, char *buf, u_int16_t len) { struct ldp_msg lm; struct fec_tlv ft; struct map map; u_int32_t label; int feclen, lbllen, tlen; u_int8_t addr_type; if (nbr->state != NBR_STA_OPER) { log_debug("recv_labelmapping: neighbor ID %s not operational", inet_ntoa(nbr->id)); return (-1); } bcopy(buf, &lm, sizeof(lm)); buf += sizeof(struct ldp_msg); len -= sizeof(struct ldp_msg); if (len < sizeof(ft)) { session_shutdown(nbr, S_BAD_TLV_LEN, lm.msgid, lm.type); return (-1); } bcopy(buf, &ft, sizeof(ft)); feclen = ntohs(ft.length); if (feclen > len - TLV_HDR_LEN) { session_shutdown(nbr, S_BAD_TLV_LEN, lm.msgid, lm.type); return (-1); } buf += TLV_HDR_LEN; /* just advance to the end of the fec header */ len -= TLV_HDR_LEN; lbllen = tlv_decode_label(buf + feclen, len - feclen, &label); if (lbllen == -1) { session_shutdown(nbr, S_BAD_TLV_LEN, lm.msgid, lm.type); return (-1); } if (label == NO_LABEL) { session_shutdown(nbr, S_BAD_TLV_VAL, lm.msgid, lm.type); return (-1); } /* TODO opt label request msg id, hop cnt and path vektor TLV */ bzero(&map, sizeof(map)); map.messageid = lm.msgid; map.label = label; do { if ((tlen = tlv_decode_fec_elm(buf, feclen, &addr_type, &map.prefix.s_addr, &map.prefixlen)) == -1 || addr_type == FEC_WILDCARD) { session_shutdown(nbr, S_BAD_TLV_VAL, lm.msgid, lm.type); return (-1); } ldpe_imsg_compose_lde(IMSG_LABEL_MAPPING, nbr->peerid, 0, &map, sizeof(map)); buf += tlen; feclen -= tlen; } while (feclen > 0); nbr_fsm(nbr, NBR_EVT_PDU_RCVD); return (ntohs(lm.length)); } /* Label Request Message */ void send_labelrequest(struct nbr *nbr) { struct ibuf *buf; struct mapping_entry *me; struct ldp_hdr *ldp_hdr; u_int16_t tlv_size, size; if (nbr->iface->passive) return; if ((buf = ibuf_open(LDP_MAX_LEN)) == NULL) fatal("send_labelrequest"); /* real size will be set up later */ gen_ldp_hdr(buf, nbr->iface, 0); size = LDP_HDR_SIZE - TLV_HDR_LEN; TAILQ_FOREACH(me, &nbr->request_list, entry) { tlv_size = PREFIX_SIZE(me->map.prefixlen); size += tlv_size; gen_msg_tlv(buf, MSG_TYPE_LABELREQUEST, tlv_size); gen_fec_tlv(buf, me->map.prefix, me->map.prefixlen); } /* XXX: should we remove them first? */ nbr_mapping_list_clr(nbr, &nbr->request_list); ldp_hdr = ibuf_seek(buf, 0, sizeof(struct ldp_hdr)); ldp_hdr->length = htons(size); evbuf_enqueue(&nbr->wbuf, buf); } int recv_labelrequest(struct nbr *nbr, char *buf, u_int16_t len) { struct ldp_msg lr; struct fec_tlv ft; struct map map; int feclen, tlen; u_int8_t addr_type; if (nbr->state != NBR_STA_OPER) { log_debug("recv_labelrequest: neighbor ID %s not operational", inet_ntoa(nbr->id)); return (-1); } bcopy(buf, &lr, sizeof(lr)); buf += sizeof(struct ldp_msg); len -= sizeof(struct ldp_msg); if (len < sizeof(ft)) { session_shutdown(nbr, S_BAD_MSG_LEN, lr.msgid, lr.type); return (-1); } bcopy(buf, &ft, sizeof(ft)); feclen = ntohs(ft.length); if (feclen > len - TLV_HDR_LEN) { session_shutdown(nbr, S_BAD_TLV_LEN, lr.msgid, lr.type); return (-1); } buf += TLV_HDR_LEN; /* just advance to the end of the fec header */ len -= TLV_HDR_LEN; /* TODO opt hop cnt and path vektor TLV */ bzero(&map, sizeof(map)); map.messageid = lr.msgid; do { if ((tlen = tlv_decode_fec_elm(buf, feclen, &addr_type, &map.prefix.s_addr, &map.prefixlen)) == -1 || addr_type == FEC_WILDCARD) { session_shutdown(nbr, S_BAD_TLV_VAL, lr.msgid, lr.type); return (-1); } ldpe_imsg_compose_lde(IMSG_LABEL_REQUEST, nbr->peerid, 0, &map, sizeof(map)); buf += tlen; feclen -= tlen; } while (feclen > 0); nbr_fsm(nbr, NBR_EVT_PDU_RCVD); return (ntohs(lr.length)); } /* Label Withdraw Message */ void send_labelwithdraw(struct nbr *nbr) { struct ibuf *buf; struct mapping_entry *me; struct ldp_hdr *ldp_hdr; u_int16_t tlv_size, size; if (nbr->iface->passive) return; if ((buf = ibuf_open(LDP_MAX_LEN)) == NULL) fatal("send_labelwithdraw"); /* real size will be set up later */ gen_ldp_hdr(buf, nbr->iface, 0); size = LDP_HDR_SIZE - TLV_HDR_LEN; TAILQ_FOREACH(me, &nbr->withdraw_list, entry) { if (me->map.label == NO_LABEL) tlv_size = PREFIX_SIZE(me->map.prefixlen); else tlv_size = BASIC_LABEL_MAP_LEN + PREFIX_SIZE(me->map.prefixlen); size += tlv_size; gen_msg_tlv(buf, MSG_TYPE_LABELWITHDRAW, tlv_size); gen_fec_tlv(buf, me->map.prefix, me->map.prefixlen); if (me->map.label != NO_LABEL) gen_label_tlv(buf, me->map.label); } /* XXX: should we remove them first? */ nbr_mapping_list_clr(nbr, &nbr->withdraw_list); ldp_hdr = ibuf_seek(buf, 0, sizeof(struct ldp_hdr)); ldp_hdr->length = htons(size); evbuf_enqueue(&nbr->wbuf, buf); } int recv_labelwithdraw(struct nbr *nbr, char *buf, u_int16_t len) { struct map map; struct ldp_msg lw; struct fec_tlv ft; u_int32_t label = NO_LABEL; int feclen, tlen, numfec = 0; u_int8_t addr_type; if (nbr->state != NBR_STA_OPER) { log_debug("recv_labelwithdraw: neighbor ID %s not operational", inet_ntoa(nbr->id)); return (-1); } bcopy(buf, &lw, sizeof(lw)); buf += sizeof(struct ldp_msg); len -= sizeof(struct ldp_msg); if (len < sizeof(ft)) { session_shutdown(nbr, S_BAD_MSG_LEN, lw.msgid, lw.type); return (-1); } bcopy(buf, &ft, sizeof(ft)); feclen = ntohs(ft.length); if (feclen > len - TLV_HDR_LEN) { session_shutdown(nbr, S_BAD_TLV_LEN, lw.msgid, lw.type); return (-1); } buf += TLV_HDR_LEN; /* just advance to the end of the fec header */ len -= TLV_HDR_LEN; /* withdraw may include optional label */ if (len > feclen) { int r; r = tlv_decode_label(buf + feclen, len - feclen, &label); if (r == -1 || len != feclen + r) { session_shutdown(nbr, S_BAD_TLV_VAL, lw.msgid, lw.type); return (-1); } } bzero(&map, sizeof(map)); map.messageid = lw.msgid; if (label != NO_LABEL) { map.label = label; map.flags = F_MAP_OPTLABEL; } do { if ((tlen = tlv_decode_fec_elm(buf, feclen, &addr_type, &map.prefix.s_addr, &map.prefixlen)) == -1) { session_shutdown(nbr, S_BAD_TLV_VAL, lw.msgid, lw.type); return (-1); } if (addr_type == FEC_WILDCARD) { /* Wildcard FEC must be the only FEC element */ if (numfec != 0) { session_shutdown(nbr, S_BAD_TLV_VAL, lw.msgid, lw.type); return (-1); } map.prefix.s_addr = 0; map.prefixlen = 0; map.flags |= F_MAP_WILDCARD; numfec = -1; } else { /* Wildcard FEC must be the only FEC element */ if (numfec == -1) { session_shutdown(nbr, S_BAD_TLV_VAL, lw.msgid, lw.type); return (-1); } numfec++; map.flags &= ~F_MAP_WILDCARD; } ldpe_imsg_compose_lde(IMSG_LABEL_WITHDRAW, nbr->peerid, 0, &map, sizeof(map)); buf += tlen; feclen -= tlen; } while (feclen > 0); nbr_fsm(nbr, NBR_EVT_PDU_RCVD); return (ntohs(lw.length)); } /* Label Release Message */ void send_labelrelease(struct nbr *nbr) { struct ibuf *buf; struct mapping_entry *me; struct ldp_hdr *ldp_hdr; u_int16_t tlv_size, size; if (nbr->iface->passive) return; if ((buf = ibuf_open(LDP_MAX_LEN)) == NULL) fatal("send_labelrelease"); /* real size will be set up later */ gen_ldp_hdr(buf, nbr->iface, 0); size = LDP_HDR_SIZE - TLV_HDR_LEN; TAILQ_FOREACH(me, &nbr->release_list, entry) { if (me->map.label == NO_LABEL) tlv_size = PREFIX_SIZE(me->map.prefixlen); else tlv_size = BASIC_LABEL_MAP_LEN + PREFIX_SIZE(me->map.prefixlen); size += tlv_size; gen_msg_tlv(buf, MSG_TYPE_LABELRELEASE, tlv_size); gen_fec_tlv(buf, me->map.prefix, me->map.prefixlen); if (me->map.label != NO_LABEL) gen_label_tlv(buf, me->map.label); } /* XXX: should we remove them first? */ nbr_mapping_list_clr(nbr, &nbr->release_list); ldp_hdr = ibuf_seek(buf, 0, sizeof(struct ldp_hdr)); ldp_hdr->length = htons(size); evbuf_enqueue(&nbr->wbuf, buf); } int recv_labelrelease(struct nbr *nbr, char *buf, u_int16_t len) { struct map map; struct ldp_msg lr; struct fec_tlv ft; u_int32_t label = NO_LABEL; int feclen, tlen, numfec = 0; u_int8_t addr_type; if (nbr->state != NBR_STA_OPER) { log_debug("recv_labelrelease: neighbor ID %s not operational", inet_ntoa(nbr->id)); return (-1); } bcopy(buf, &lr, sizeof(lr)); buf += sizeof(struct ldp_msg); len -= sizeof(struct ldp_msg); if (len < sizeof(ft)) { session_shutdown(nbr, S_BAD_MSG_LEN, lr.msgid, lr.type); return (-1); } bcopy(buf, &ft, sizeof(ft)); feclen = ntohs(ft.length); if (feclen > len - TLV_HDR_LEN) { session_shutdown(nbr, S_BAD_TLV_LEN, lr.msgid, lr.type); return (-1); } buf += TLV_HDR_LEN; /* just advance to the end of the fec header */ len -= TLV_HDR_LEN; /* release may include optional label */ if (len > feclen) { int r; r = tlv_decode_label(buf + feclen, len - feclen, &label); if (r == -1 || len != feclen + r) { session_shutdown(nbr, S_BAD_TLV_VAL, lr.msgid, lr.type); return (-1); } } bzero(&map, sizeof(map)); map.messageid = lr.msgid; if (label != NO_LABEL) { map.label = label; map.flags = F_MAP_OPTLABEL; } do { if ((tlen = tlv_decode_fec_elm(buf, feclen, &addr_type, &map.prefix.s_addr, &map.prefixlen)) == -1) { session_shutdown(nbr, S_BAD_TLV_VAL, lr.msgid, lr.type); return (-1); } if (addr_type == FEC_WILDCARD) { /* Wildcard FEC must be the only FEC element */ if (numfec != 0) { session_shutdown(nbr, S_BAD_TLV_VAL, lr.msgid, lr.type); return (-1); } map.prefix.s_addr = 0; map.prefixlen = 0; map.flags |= F_MAP_WILDCARD; } else { /* Wildcard FEC must be the only FEC element */ if (numfec == -1) { session_shutdown(nbr, S_BAD_TLV_VAL, lr.msgid, lr.type); return (-1); } map.flags &= ~F_MAP_WILDCARD; } ldpe_imsg_compose_lde(IMSG_LABEL_RELEASE, nbr->peerid, 0, &map, sizeof(map)); buf += tlen; feclen -= tlen; } while (feclen > 0); nbr_fsm(nbr, NBR_EVT_PDU_RCVD); return (ntohs(lr.length)); } /* Label Abort Req Message */ void send_labelabortreq(struct nbr *nbr) { struct ibuf *buf; u_int16_t size; if (nbr->iface->passive) return; if ((buf = ibuf_open(LDP_MAX_LEN)) == NULL) fatal("send_labelabortreq"); size = LDP_HDR_SIZE + sizeof(struct ldp_msg); gen_ldp_hdr(buf, nbr->iface, size); size -= LDP_HDR_SIZE; gen_msg_tlv(buf, MSG_TYPE_LABELABORTREQ, size); evbuf_enqueue(&nbr->wbuf, buf); } int recv_labelabortreq(struct nbr *nbr, char *buf, u_int16_t len) { struct map map; struct ldp_msg la; struct fec_tlv ft; int feclen, tlen; u_int8_t addr_type; if (nbr->state != NBR_STA_OPER) { log_debug("recv_labelabortreq: neighbor ID %s not operational", inet_ntoa(nbr->id)); return (-1); } log_debug("recv_labelabortreq: neighbor ID %s", inet_ntoa(nbr->id)); bcopy(buf, &la, sizeof(la)); buf += sizeof(struct ldp_msg); len -= sizeof(struct ldp_msg); if (len < sizeof(ft)) { session_shutdown(nbr, S_BAD_MSG_LEN, la.msgid, la.type); return (-1); } bcopy(buf, &ft, sizeof(ft)); feclen = ntohs(ft.length); if (feclen > len - TLV_HDR_LEN) { session_shutdown(nbr, S_BAD_TLV_LEN, la.msgid, la.type); return (-1); } buf += TLV_HDR_LEN; /* just advance to the end of the fec header */ len -= TLV_HDR_LEN; bzero(&map, sizeof(map)); map.messageid = la.msgid; /* abort request may include optional request msg id */ if (len > feclen) { int r; r = tlv_decode_reqid(buf + feclen, len - feclen, &map.requestid); if (r == -1 || len != feclen + r) { session_shutdown(nbr, S_BAD_TLV_VAL, la.msgid, la.type); return (-1); } map.flags = F_MAP_REQ_ID; } do { if ((tlen = tlv_decode_fec_elm(buf, feclen, &addr_type, &map.prefix.s_addr, &map.prefixlen)) == -1 || addr_type == FEC_WILDCARD) { session_shutdown(nbr, S_BAD_TLV_VAL, la.msgid, la.type); return (-1); } ldpe_imsg_compose_lde(IMSG_LABEL_ABORT, nbr->peerid, 0, &map, sizeof(map)); buf += tlen; feclen -= tlen; } while (feclen > 0); nbr_fsm(nbr, NBR_EVT_PDU_RCVD); return (ntohs(la.length)); } /* Other TLV related functions */ void gen_label_tlv(struct ibuf *buf, u_int32_t label) { struct label_tlv lt; lt.type = htons(TLV_TYPE_GENERICLABEL); lt.length = htons(sizeof(label)); lt.label = htonl(label); ibuf_add(buf, <, sizeof(lt)); } int tlv_decode_label(char *buf, u_int16_t len, u_int32_t *label) { struct label_tlv lt; if (len < sizeof(lt)) return (-1); bcopy(buf, <, sizeof(lt)); if (ntohs(lt.length) != sizeof(lt) - TLV_HDR_LEN) return (-1); if (lt.type != htons(TLV_TYPE_GENERICLABEL)) return (-1); *label = ntohl(lt.label); return (sizeof(lt)); } void gen_reqid_tlv(struct ibuf *buf, u_int32_t reqid) { struct reqid_tlv rt; rt.type = htons(TLV_TYPE_LABELREQUEST); rt.length = htons(sizeof(reqid)); rt.reqid = htonl(reqid); ibuf_add(buf, &rt, sizeof(rt)); } int tlv_decode_reqid(char *buf, u_int16_t len, u_int32_t *reqid) { struct reqid_tlv rt; if (len < sizeof(rt)) return (-1); bcopy(buf, &rt, sizeof(rt)); if (ntohs(rt.length) != sizeof(rt) - TLV_HDR_LEN) return (-1); if (rt.type != htons(TLV_TYPE_LABELREQUEST)) return (-1); *reqid = ntohl(rt.reqid); return (sizeof(rt)); } void gen_fec_tlv(struct ibuf *buf, struct in_addr prefix, u_int8_t prefixlen) { struct fec_tlv ft; u_int8_t type; u_int16_t family; u_int8_t len; len = PREFIX_SIZE(prefixlen); ft.type = htons(TLV_TYPE_FEC); ft.length = htons(sizeof(type) + sizeof(family) + sizeof(prefixlen) + len); ibuf_add(buf, &ft, sizeof(ft)); type = FEC_PREFIX; family = htons(FEC_IPV4); ibuf_add(buf, &type, sizeof(type)); ibuf_add(buf, &family, sizeof(family)); ibuf_add(buf, &prefixlen, sizeof(prefixlen)); if (len) ibuf_add(buf, &prefix, len); } int tlv_decode_fec_elm(char *buf, u_int16_t len, u_int8_t *type, u_int32_t *prefix, u_int8_t *prefixlen) { u_int16_t family, off = 0; *type = *buf; off += sizeof(u_int8_t); if (*type == FEC_WILDCARD) { if (len == 0) return (off); else return (-1); /* XXX Malformed TLV Value */ } if (*type != FEC_PREFIX) return (-1); /* XXX "Unknown FEC" Notification */ if (len < FEC_ELM_MIN_LEN) return (-1); /* XXX Bad TLV Length */ bcopy(buf + off, &family, sizeof(family)); off += sizeof(family); if (family != htons(FEC_IPV4)) return (-1); /* XXX "Unsupported Address Family" */ *prefixlen = buf[off]; off += sizeof(u_int8_t); if (len < off + PREFIX_SIZE(*prefixlen)) return (-1); /* XXX Bad TLV Length */ *prefix = 0; bcopy(buf + off, prefix, PREFIX_SIZE(*prefixlen)); return (off + PREFIX_SIZE(*prefixlen)); }