xref: /aosp_15_r20/external/curl/tests/unit/unit1603.c (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker  *                                  _   _ ____  _
3*6236dae4SAndroid Build Coastguard Worker  *  Project                     ___| | | |  _ \| |
4*6236dae4SAndroid Build Coastguard Worker  *                             / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker  *                            | (__| |_| |  _ <| |___
6*6236dae4SAndroid Build Coastguard Worker  *                             \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker  *
8*6236dae4SAndroid Build Coastguard Worker  * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker  *
10*6236dae4SAndroid Build Coastguard Worker  * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker  * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker  * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker  *
14*6236dae4SAndroid Build Coastguard Worker  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker  * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker  * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker  *
18*6236dae4SAndroid Build Coastguard Worker  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker  * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker  *
21*6236dae4SAndroid Build Coastguard Worker  * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker  *
23*6236dae4SAndroid Build Coastguard Worker  ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker #include "curlcheck.h"
25*6236dae4SAndroid Build Coastguard Worker 
26*6236dae4SAndroid Build Coastguard Worker #include "curlx.h"
27*6236dae4SAndroid Build Coastguard Worker 
28*6236dae4SAndroid Build Coastguard Worker #include "hash.h"
29*6236dae4SAndroid Build Coastguard Worker 
30*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h" /* LAST include file */
31*6236dae4SAndroid Build Coastguard Worker 
32*6236dae4SAndroid Build Coastguard Worker static struct Curl_hash hash_static;
33*6236dae4SAndroid Build Coastguard Worker static const size_t slots = 3;
34*6236dae4SAndroid Build Coastguard Worker 
mydtor(void * p)35*6236dae4SAndroid Build Coastguard Worker static void mydtor(void *p)
36*6236dae4SAndroid Build Coastguard Worker {
37*6236dae4SAndroid Build Coastguard Worker   /* Data are statically allocated */
38*6236dae4SAndroid Build Coastguard Worker  (void)p; /* unused */
39*6236dae4SAndroid Build Coastguard Worker }
40*6236dae4SAndroid Build Coastguard Worker 
41*6236dae4SAndroid Build Coastguard Worker static size_t elem_dtor_calls;
42*6236dae4SAndroid Build Coastguard Worker 
my_elem_dtor(void * key,size_t key_len,void * p)43*6236dae4SAndroid Build Coastguard Worker static void my_elem_dtor(void *key, size_t key_len, void *p)
44*6236dae4SAndroid Build Coastguard Worker {
45*6236dae4SAndroid Build Coastguard Worker   (void)p; /* unused */
46*6236dae4SAndroid Build Coastguard Worker   (void)key; /* unused */
47*6236dae4SAndroid Build Coastguard Worker   (void)key_len; /* unused */
48*6236dae4SAndroid Build Coastguard Worker   ++elem_dtor_calls;
49*6236dae4SAndroid Build Coastguard Worker }
50*6236dae4SAndroid Build Coastguard Worker 
unit_setup(void)51*6236dae4SAndroid Build Coastguard Worker static CURLcode unit_setup(void)
52*6236dae4SAndroid Build Coastguard Worker {
53*6236dae4SAndroid Build Coastguard Worker   Curl_hash_init(&hash_static, slots, Curl_hash_str,
54*6236dae4SAndroid Build Coastguard Worker                  Curl_str_key_compare, mydtor);
55*6236dae4SAndroid Build Coastguard Worker   return CURLE_OK;
56*6236dae4SAndroid Build Coastguard Worker }
57*6236dae4SAndroid Build Coastguard Worker 
unit_stop(void)58*6236dae4SAndroid Build Coastguard Worker static void unit_stop(void)
59*6236dae4SAndroid Build Coastguard Worker {
60*6236dae4SAndroid Build Coastguard Worker   Curl_hash_destroy(&hash_static);
61*6236dae4SAndroid Build Coastguard Worker }
62*6236dae4SAndroid Build Coastguard Worker 
63*6236dae4SAndroid Build Coastguard Worker UNITTEST_START
64*6236dae4SAndroid Build Coastguard Worker   char key1[] = "key1";
65*6236dae4SAndroid Build Coastguard Worker   char key2[] = "key2b";
66*6236dae4SAndroid Build Coastguard Worker   char key3[] = "key3";
67*6236dae4SAndroid Build Coastguard Worker   char key4[] = "key4";
68*6236dae4SAndroid Build Coastguard Worker   char notakey[] = "notakey";
69*6236dae4SAndroid Build Coastguard Worker   char *nodep;
70*6236dae4SAndroid Build Coastguard Worker   int rc;
71*6236dae4SAndroid Build Coastguard Worker 
72*6236dae4SAndroid Build Coastguard Worker   /* Ensure the key hashes are as expected in order to test both hash
73*6236dae4SAndroid Build Coastguard Worker      collisions and a full table. Unfortunately, the hashes can vary
74*6236dae4SAndroid Build Coastguard Worker      between architectures. */
75*6236dae4SAndroid Build Coastguard Worker   if(Curl_hash_str(key1, strlen(key1), slots) != 1 ||
76*6236dae4SAndroid Build Coastguard Worker      Curl_hash_str(key2, strlen(key2), slots) != 0 ||
77*6236dae4SAndroid Build Coastguard Worker      Curl_hash_str(key3, strlen(key3), slots) != 2 ||
78*6236dae4SAndroid Build Coastguard Worker      Curl_hash_str(key4, strlen(key4), slots) != 1)
79*6236dae4SAndroid Build Coastguard Worker     fprintf(stderr, "Warning: hashes are not computed as expected on this "
80*6236dae4SAndroid Build Coastguard Worker             "architecture; test coverage will be less comprehensive\n");
81*6236dae4SAndroid Build Coastguard Worker 
82*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key1, strlen(key1), &key1);
83*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
84*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
85*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key1, "hash retrieval failed");
86*6236dae4SAndroid Build Coastguard Worker 
87*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key2, strlen(key2), &key2);
88*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
89*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key2, strlen(key2));
90*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key2, "hash retrieval failed");
91*6236dae4SAndroid Build Coastguard Worker 
92*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key3, strlen(key3), &key3);
93*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
94*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key3, strlen(key3));
95*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key3, "hash retrieval failed");
96*6236dae4SAndroid Build Coastguard Worker 
97*6236dae4SAndroid Build Coastguard Worker   /* The fourth element exceeds the number of slots & collides */
98*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key4, strlen(key4), &key4);
99*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
100*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
101*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key4, "hash retrieval failed");
102*6236dae4SAndroid Build Coastguard Worker 
103*6236dae4SAndroid Build Coastguard Worker   /* Make sure all elements are still accessible */
104*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
105*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key1, "hash retrieval failed");
106*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key2, strlen(key2));
107*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key2, "hash retrieval failed");
108*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key3, strlen(key3));
109*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key3, "hash retrieval failed");
110*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
111*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key4, "hash retrieval failed");
112*6236dae4SAndroid Build Coastguard Worker 
113*6236dae4SAndroid Build Coastguard Worker   /* Delete the second of two entries in a bucket */
114*6236dae4SAndroid Build Coastguard Worker   rc = Curl_hash_delete(&hash_static, &key4, strlen(key4));
115*6236dae4SAndroid Build Coastguard Worker   fail_unless(rc == 0, "hash delete failed");
116*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
117*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key1, "hash retrieval failed");
118*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
119*6236dae4SAndroid Build Coastguard Worker   fail_unless(!nodep, "hash retrieval should have failed");
120*6236dae4SAndroid Build Coastguard Worker 
121*6236dae4SAndroid Build Coastguard Worker   /* Insert that deleted node again */
122*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key4, strlen(key4), &key4);
123*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
124*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
125*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key4, "hash retrieval failed");
126*6236dae4SAndroid Build Coastguard Worker 
127*6236dae4SAndroid Build Coastguard Worker   /* Delete the first of two entries in a bucket */
128*6236dae4SAndroid Build Coastguard Worker   rc = Curl_hash_delete(&hash_static, &key1, strlen(key1));
129*6236dae4SAndroid Build Coastguard Worker   fail_unless(rc == 0, "hash delete failed");
130*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
131*6236dae4SAndroid Build Coastguard Worker   fail_unless(!nodep, "hash retrieval should have failed");
132*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
133*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key4, "hash retrieval failed");
134*6236dae4SAndroid Build Coastguard Worker 
135*6236dae4SAndroid Build Coastguard Worker   /* Delete the remaining one of two entries in a bucket */
136*6236dae4SAndroid Build Coastguard Worker   rc = Curl_hash_delete(&hash_static, &key4, strlen(key4));
137*6236dae4SAndroid Build Coastguard Worker   fail_unless(rc == 0, "hash delete failed");
138*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
139*6236dae4SAndroid Build Coastguard Worker   fail_unless(!nodep, "hash retrieval should have failed");
140*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key4, strlen(key4));
141*6236dae4SAndroid Build Coastguard Worker   fail_unless(!nodep, "hash retrieval should have failed");
142*6236dae4SAndroid Build Coastguard Worker 
143*6236dae4SAndroid Build Coastguard Worker   /* Delete an already deleted node */
144*6236dae4SAndroid Build Coastguard Worker   rc = Curl_hash_delete(&hash_static, &key4, strlen(key4));
145*6236dae4SAndroid Build Coastguard Worker   fail_unless(rc, "hash delete should have failed");
146*6236dae4SAndroid Build Coastguard Worker 
147*6236dae4SAndroid Build Coastguard Worker   /* Replace an existing node */
148*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add(&hash_static, &key1, strlen(key1), &notakey);
149*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "insertion into hash failed");
150*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key1, strlen(key1));
151*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == notakey, "hash retrieval failed");
152*6236dae4SAndroid Build Coastguard Worker 
153*6236dae4SAndroid Build Coastguard Worker   /* Make sure all remaining elements are still accessible */
154*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key2, strlen(key2));
155*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key2, "hash retrieval failed");
156*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_pick(&hash_static, &key3, strlen(key3));
157*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep == key3, "hash retrieval failed");
158*6236dae4SAndroid Build Coastguard Worker 
159*6236dae4SAndroid Build Coastguard Worker   /* Add element with own destructor */
160*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add2(&hash_static, &key1, strlen(key1), &key1,
161*6236dae4SAndroid Build Coastguard Worker                          my_elem_dtor);
162*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "add2 insertion into hash failed");
163*6236dae4SAndroid Build Coastguard Worker   fail_unless(elem_dtor_calls == 0, "element destructor count should be 0");
164*6236dae4SAndroid Build Coastguard Worker   /* Add it again, should invoke destructor on first */
165*6236dae4SAndroid Build Coastguard Worker   nodep = Curl_hash_add2(&hash_static, &key1, strlen(key1), &key1,
166*6236dae4SAndroid Build Coastguard Worker                          my_elem_dtor);
167*6236dae4SAndroid Build Coastguard Worker   fail_unless(nodep, "add2 again, insertion into hash failed");
168*6236dae4SAndroid Build Coastguard Worker   fail_unless(elem_dtor_calls == 1, "element destructor count should be 1");
169*6236dae4SAndroid Build Coastguard Worker   /* remove, should invoke destructor */
170*6236dae4SAndroid Build Coastguard Worker   rc = Curl_hash_delete(&hash_static, &key1, strlen(key1));
171*6236dae4SAndroid Build Coastguard Worker   fail_unless(rc == 0, "hash delete failed");
172*6236dae4SAndroid Build Coastguard Worker   fail_unless(elem_dtor_calls == 2, "element destructor count should be 1");
173*6236dae4SAndroid Build Coastguard Worker 
174*6236dae4SAndroid Build Coastguard Worker 
175*6236dae4SAndroid Build Coastguard Worker   /* Clean up */
176*6236dae4SAndroid Build Coastguard Worker   Curl_hash_clean(&hash_static);
177*6236dae4SAndroid Build Coastguard Worker 
178*6236dae4SAndroid Build Coastguard Worker UNITTEST_STOP
179