summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorache <ache@FreeBSD.org>2016-09-03 01:08:52 +0000
committerache <ache@FreeBSD.org>2016-09-03 01:08:52 +0000
commit487da86387ce37bb7038e5186a991f82d0c7b330 (patch)
tree7edeed17eef13e7b9b5035ac50b52f144323fe43
parentac7fe6d68fbb4c1cd358cfc5e23ed527de34a2f9 (diff)
downloadFreeBSD-src-487da86387ce37bb7038e5186a991f82d0c7b330.zip
FreeBSD-src-487da86387ce37bb7038e5186a991f82d0c7b330.tar.gz
MFC r305133
Apply the same qsort() usage fix as in r304911 getaddrinfo.c qsort() can't be stabilized with just return(-1) alone.
-rw-r--r--lib/libc/net/name6.c19
1 files changed, 19 insertions, 0 deletions
diff --git a/lib/libc/net/name6.c b/lib/libc/net/name6.c
index ebae9d3..b34580e 100644
--- a/lib/libc/net/name6.c
+++ b/lib/libc/net/name6.c
@@ -186,6 +186,7 @@ struct hp_order {
#define aio_sa aio_un.aiou_sa
int aio_matchlen;
char *aio_h_addr;
+ int aio_initial_sequence;
};
static struct hostent *_hpcopy(struct hostent *, int *);
@@ -711,6 +712,7 @@ _hpreorder(struct hostent *hp)
aio[i].aio_dstscope = gai_addr2scopetype(sa);
aio[i].aio_dstpolicy = match_addrselectpolicy(sa, &policyhead);
set_source(&aio[i], &policyhead);
+ aio[i].aio_initial_sequence = i;
}
/* perform sorting. */
@@ -1045,6 +1047,23 @@ comp_dst(const void *arg1, const void *arg2)
}
/* Rule 10: Otherwise, leave the order unchanged. */
+
+ /*
+ * Note that qsort is unstable; so, we can't return zero and
+ * expect the order to be unchanged.
+ * That also means we can't depend on the current position of
+ * dst2 being after dst1. We must enforce the initial order
+ * with an explicit compare on the original position.
+ * The qsort specification requires that "When the same objects
+ * (consisting of width bytes, irrespective of their current
+ * positions in the array) are passed more than once to the
+ * comparison function, the results shall be consistent with one
+ * another."
+ * In other words, If A < B, then we must also return B > A.
+ */
+ if (dst2->aio_initial_sequence < dst1->aio_initial_sequence)
+ return(1);
+
return(-1);
}
OpenPOWER on IntegriCloud