summaryrefslogtreecommitdiff
path: root/lib/cookie.c
diff options
context:
space:
mode:
authorMarc Hoersken <info@marc-hoersken.de>2013-04-07 10:34:32 +0200
committerDaniel Stenberg <daniel@haxx.se>2013-04-07 18:38:49 +0200
commit762961fe352dbb8bc08f58b26ca8a18e7dd1999d (patch)
tree14b7cbe6542b85d1b4898c9a28425c0183910c67 /lib/cookie.c
parent4b643f1ca44e91683be8697709b786efd43c64ef (diff)
downloadgnurl-762961fe352dbb8bc08f58b26ca8a18e7dd1999d.tar.gz
gnurl-762961fe352dbb8bc08f58b26ca8a18e7dd1999d.tar.bz2
gnurl-762961fe352dbb8bc08f58b26ca8a18e7dd1999d.zip
cookie.c: Made cookie sort function more deterministic
Since qsort implementations vary with regards to handling the order of similiar elements, this change makes the internal sort function more deterministic by comparing path length first, then domain length and finally the cookie name. Spotted with testcase 62 on Windows.
Diffstat (limited to 'lib/cookie.c')
-rw-r--r--lib/cookie.c25
1 files changed, 21 insertions, 4 deletions
diff --git a/lib/cookie.c b/lib/cookie.c
index c11197679..4b9ec0bdd 100644
--- a/lib/cookie.c
+++ b/lib/cookie.c
@@ -5,7 +5,7 @@
* | (__| |_| | _ <| |___
* \___|\___/|_| \_\_____|
*
- * Copyright (C) 1998 - 2012, Daniel Stenberg, <daniel@haxx.se>, et al.
+ * Copyright (C) 1998 - 2013, Daniel Stenberg, <daniel@haxx.se>, et al.
*
* This software is licensed as described in the file COPYING, which
* you should have received as part of this distribution. The terms
@@ -777,11 +777,28 @@ static int cookie_sort(const void *p1, const void *p2)
{
struct Cookie *c1 = *(struct Cookie **)p1;
struct Cookie *c2 = *(struct Cookie **)p2;
+ size_t l1, l2;
- size_t l1 = c1->path?strlen(c1->path):0;
- size_t l2 = c2->path?strlen(c2->path):0;
+ /* 1 - compare cookie path lengths */
+ l1 = c1->path ? strlen(c1->path) : 0;
+ l2 = c2->path ? strlen(c2->path) : 0;
- return (l2 > l1) ? 1 : (l2 < l1) ? -1 : 0 ;
+ if(l1 != l2)
+ return (l2 > l1) ? 1 : -1 ; /* avoid size_t <=> int conversions */
+
+ /* 2 - compare cookie domain lengths */
+ l1 = c1->domain ? strlen(c1->domain) : 0;
+ l2 = c2->domain ? strlen(c2->domain) : 0;
+
+ if(l1 != l2)
+ return (l2 > l1) ? 1 : -1 ; /* avoid size_t <=> int conversions */
+
+ /* 3 - compare cookie names */
+ if(c1->name && c2->name)
+ return strcmp(c1->name, c2->name);
+
+ /* sorry, can't be more deterministic */
+ return 0;
}
/*****************************************************************************