summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/netinet/ip_carp.c50
1 files changed, 37 insertions, 13 deletions
diff --git a/sys/netinet/ip_carp.c b/sys/netinet/ip_carp.c
index a446bb67a43..eea46e842dc 100644
--- a/sys/netinet/ip_carp.c
+++ b/sys/netinet/ip_carp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ip_carp.c,v 1.113 2005/11/04 08:11:54 mcbride Exp $ */
+/* $OpenBSD: ip_carp.c,v 1.114 2005/11/27 10:48:59 mcbride Exp $ */
/*
* Copyright (c) 2002 Michael Shalayeff. All rights reserved.
@@ -224,9 +224,10 @@ carp_hmac_prepare(struct carp_softc *sc)
SHA1_CTX sha1ctx;
u_int32_t kmd[5];
struct ifaddr *ifa;
- int i;
+ int i, found;
+ struct in_addr last, cur, in;
#ifdef INET6
- struct in6_addr in6;
+ struct in6_addr last6, cur6, in6;
#endif /* INET6 */
/* compute ipad from key */
@@ -249,23 +250,46 @@ carp_hmac_prepare(struct carp_softc *sc)
/* the rest of the precomputation */
SHA1Update(&sc->sc_sha1, (void *)&vhid, sizeof(vhid));
+
+ /* Hash the addresses from smallest to largest, not interface order */
#ifdef INET
- TAILQ_FOREACH(ifa, &sc->sc_if.if_addrlist, ifa_list) {
- if (ifa->ifa_addr->sa_family == AF_INET)
- SHA1Update(&sc->sc_sha1,
- (void *)&ifatoia(ifa)->ia_addr.sin_addr.s_addr,
- sizeof(struct in_addr));
- }
+ cur.s_addr = 0;
+ do {
+ found = 0;
+ last = cur;
+ cur.s_addr = 0xffffff;
+ TAILQ_FOREACH(ifa, &sc->sc_if.if_addrlist, ifa_list) {
+ in.s_addr = ifatoia(ifa)->ia_addr.sin_addr.s_addr;
+ if (ifa->ifa_addr->sa_family == AF_INET &&
+ in.s_addr > last.s_addr && in.s_addr < cur.s_addr) {
+ cur.s_addr = in.s_addr;
+ found++;
+ }
+ }
+ if (found)
+ SHA1Update(&sc->sc_sha1, (void *)&cur, sizeof(cur));
+ } while (found);
#endif /* INET */
#ifdef INET6
- TAILQ_FOREACH(ifa, &sc->sc_if.if_addrlist, ifa_list) {
- if (ifa->ifa_addr->sa_family == AF_INET6) {
+ memset(&cur, 0x00, sizeof(cur));
+ do {
+ found = 0;
+ last6 = cur6;
+ memset(&cur, 0xff, sizeof(cur));
+ TAILQ_FOREACH(ifa, &sc->sc_if.if_addrlist, ifa_list) {
in6 = ifatoia6(ifa)->ia_addr.sin6_addr;
if (IN6_IS_ADDR_LINKLOCAL(&in6))
in6.s6_addr16[1] = 0;
- SHA1Update(&sc->sc_sha1, (void *)&in6, sizeof(in6));
+ if (ifa->ifa_addr->sa_family == AF_INET6 &&
+ memcmp(&in6, &last6, sizeof(in6)) > 0 &&
+ memcmp(&in6, &cur6, sizeof(in6)) < 0) {
+ cur6 = in6;
+ found++;
+ }
}
- }
+ if (found)
+ SHA1Update(&sc->sc_sha1, (void *)&cur6, sizeof(in6));
+ } while (found);
#endif /* INET6 */
/* convert ipad to opad */