summaryrefslogtreecommitdiff
path: root/usr.sbin/ldpd/adjacency.c
diff options
context:
space:
mode:
authorRenato Westphal <renato@cvs.openbsd.org>2016-06-09 17:26:33 +0000
committerRenato Westphal <renato@cvs.openbsd.org>2016-06-09 17:26:33 +0000
commitf429e0627900f54d72548e50b7af0b713296264f (patch)
treed6fb243a4b73a9a1af440f8eb23623c40016d6b3 /usr.sbin/ldpd/adjacency.c
parent5e5312976e5a2bcd103b6fb160a065a2496dc956 (diff)
Send a fatal notification when the last hello adjacency is deleted.
RFC 5036 says: "When the last Hello adjacency for an LDP session is deleted, the LSR terminates the LDP session by sending a Notification message and closing the transport connection". Send a "Hold Timer Expired" notification when the triggering event is a hello hold time timeout. In the other cases, like disabling LDP on an interface, send a "Shutdown" notification instead. Before this patch we were just closing the neighbor's transport connection. Fixes the following ANVL LDP tests: 7.17 and 23.3.
Diffstat (limited to 'usr.sbin/ldpd/adjacency.c')
-rw-r--r--usr.sbin/ldpd/adjacency.c13
1 files changed, 8 insertions, 5 deletions
diff --git a/usr.sbin/ldpd/adjacency.c b/usr.sbin/ldpd/adjacency.c
index bf86dc931b5..6cc73af45ef 100644
--- a/usr.sbin/ldpd/adjacency.c
+++ b/usr.sbin/ldpd/adjacency.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: adjacency.c,v 1.22 2016/05/23 19:14:03 renato Exp $ */
+/* $OpenBSD: adjacency.c,v 1.23 2016/06/09 17:26:32 renato Exp $ */
/*
* Copyright (c) 2013, 2015 Renato Westphal <renato@openbsd.org>
@@ -68,7 +68,7 @@ adj_new(struct in_addr lsr_id, struct hello_source *source,
}
void
-adj_del(struct adj *adj)
+adj_del(struct adj *adj, int send_notif, uint32_t notif_status)
{
log_debug("%s: lsr-id %s, %s", __func__, inet_ntoa(adj->lsr_id),
log_hello_src(&adj->source));
@@ -82,8 +82,11 @@ adj_del(struct adj *adj)
LIST_REMOVE(adj, ia_entry);
/* last adjacency deleted */
- if (adj->nbr && LIST_EMPTY(&adj->nbr->adj_list))
+ if (adj->nbr && LIST_EMPTY(&adj->nbr->adj_list)) {
+ if (send_notif)
+ session_shutdown(adj->nbr, notif_status, 0, 0);
nbr_del(adj->nbr);
+ }
free(adj);
}
@@ -147,7 +150,7 @@ adj_itimer(int fd, short event, void *arg)
adj->source.target->adj = NULL;
}
- adj_del(adj);
+ adj_del(adj, 1, S_HOLDTIME_EXP);
}
void
@@ -193,7 +196,7 @@ tnbr_del(struct tnbr *tnbr)
{
tnbr_stop_hello_timer(tnbr);
if (tnbr->adj)
- adj_del(tnbr->adj);
+ adj_del(tnbr->adj, 1, S_SHUTDOWN);
LIST_REMOVE(tnbr, entry);
free(tnbr);
}