diff options
author | ache <ache@FreeBSD.org> | 2016-09-03 01:08:52 +0000 |
---|---|---|
committer | ache <ache@FreeBSD.org> | 2016-09-03 01:08:52 +0000 |
commit | 487da86387ce37bb7038e5186a991f82d0c7b330 (patch) | |
tree | 7edeed17eef13e7b9b5035ac50b52f144323fe43 | |
parent | ac7fe6d68fbb4c1cd358cfc5e23ed527de34a2f9 (diff) | |
download | FreeBSD-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.c | 19 |
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); } |