| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2023-09-07 16:59:15 +08:00
										 |  |  |  * Copyright 2022-2023 The OpenSSL Project Authors. All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the Apache License 2.0 (the "License").  You may not use | 
					
						
							|  |  |  |  * this file except in compliance with the License.  You can obtain a copy | 
					
						
							|  |  |  |  * in the file LICENSE in the source distribution or at | 
					
						
							|  |  |  |  * https://www.openssl.org/source/license.html
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "internal/uint_set.h"
 | 
					
						
							|  |  |  | #include "internal/common.h"
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * uint64_t Integer Sets | 
					
						
							|  |  |  |  * ===================== | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This data structure supports the following operations: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Insert Range: Adds an inclusive range of integers [start, end] | 
					
						
							|  |  |  |  *                 to the set. Equivalent to Insert for each number | 
					
						
							|  |  |  |  *                 in the range. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Remove Range: Removes an inclusive range of integers [start, end] | 
					
						
							|  |  |  |  *                 from the set. Not all of the range need already be in | 
					
						
							|  |  |  |  *                 the set, but any part of the range in the set is removed. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *   Query:        Is an integer in the data structure? | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The data structure can be iterated. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For greater efficiency in tracking large numbers of contiguous integers, we | 
					
						
							|  |  |  |  * track integer ranges rather than individual integers. The data structure | 
					
						
							|  |  |  |  * manages a list of integer ranges [[start, end]...]. Internally this is | 
					
						
							|  |  |  |  * implemented as a doubly linked sorted list of range structures, which are | 
					
						
							|  |  |  |  * automatically split and merged as necessary. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This data structure requires O(n) traversal of the list for insertion, | 
					
						
							|  |  |  |  * removal and query when we are not adding/removing ranges which are near the | 
					
						
							|  |  |  |  * beginning or end of the set of ranges. For the applications for which this | 
					
						
							|  |  |  |  * data structure is used (e.g. QUIC PN tracking for ACK generation), it is | 
					
						
							|  |  |  |  * expected that the number of integer ranges needed at any given time will | 
					
						
							|  |  |  |  * generally be small and that most operations will be close to the beginning or | 
					
						
							|  |  |  |  * end of the range. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Invariant: The data structure is always sorted in ascending order by value. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Invariant: No two adjacent ranges ever 'border' one another (have no | 
					
						
							|  |  |  |  *            numerical gap between them) as the data structure always ensures | 
					
						
							|  |  |  |  *            such ranges are merged. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Invariant: No two ranges ever overlap. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Invariant: No range [a, b] ever has a > b. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Invariant: Since ranges are represented using inclusive bounds, no range | 
					
						
							|  |  |  |  *            item inside the data structure can represent a span of zero | 
					
						
							|  |  |  |  *            integers. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void ossl_uint_set_init(UINT_SET *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     ossl_list_uint_set_init(s); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ossl_uint_set_destroy(UINT_SET *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     UINT_SET_ITEM *x, *xnext; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     for (x = ossl_list_uint_set_head(s); x != NULL; x = xnext) { | 
					
						
							|  |  |  |         xnext = ossl_list_uint_set_next(x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         OPENSSL_free(x); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  | /* Possible merge of x, prev(x) */ | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | static void uint_set_merge_adjacent(UINT_SET *s, UINT_SET_ITEM *x) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     UINT_SET_ITEM *xprev = ossl_list_uint_set_prev(x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (xprev == NULL) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (x->range.start - 1 != xprev->range.end) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     x->range.start = xprev->range.start; | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     ossl_list_uint_set_remove(s, xprev); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |     OPENSSL_free(xprev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint64_t u64_min(uint64_t x, uint64_t y) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return x < y ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint64_t u64_max(uint64_t x, uint64_t y) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return x > y ? x : y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Returns 1 if there exists an integer x which falls within both ranges a and | 
					
						
							|  |  |  |  * b. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int uint_range_overlaps(const UINT_RANGE *a, | 
					
						
							|  |  |  |                                const UINT_RANGE *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return u64_min(a->end, b->end) | 
					
						
							|  |  |  |         >= u64_max(a->start, b->start); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  | static UINT_SET_ITEM *create_set_item(uint64_t start, uint64_t end) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-27 23:24:34 +08:00
										 |  |  |     UINT_SET_ITEM *x = OPENSSL_malloc(sizeof(UINT_SET_ITEM)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (x == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ossl_list_uint_set_init_elem(x); | 
					
						
							|  |  |  |     x->range.start = start; | 
					
						
							|  |  |  |     x->range.end   = end; | 
					
						
							|  |  |  |     return x; | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | int ossl_uint_set_insert(UINT_SET *s, const UINT_RANGE *range) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     UINT_SET_ITEM *x, *xnext, *z, *zprev, *f; | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |     uint64_t start = range->start, end = range->end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ossl_assert(start <= end)) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     if (ossl_list_uint_set_is_empty(s)) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         /* Nothing in the set yet, so just add this range. */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         x = create_set_item(start, end); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         if (x == NULL) | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         ossl_list_uint_set_insert_head(s, x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     z = ossl_list_uint_set_tail(s); | 
					
						
							|  |  |  |     if (start > z->range.end) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         /*
 | 
					
						
							|  |  |  |          * Range is after the latest range in the set, so append. | 
					
						
							|  |  |  |          * | 
					
						
							|  |  |  |          * Note: The case where the range is before the earliest range in the | 
					
						
							|  |  |  |          * set is handled as a degenerate case of the final case below. See | 
					
						
							|  |  |  |          * optimization note (*) below. | 
					
						
							|  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         if (z->range.end + 1 == start) { | 
					
						
							|  |  |  |             z->range.end = end; | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             return 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         x = create_set_item(start, end); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         if (x == NULL) | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         ossl_list_uint_set_insert_tail(s, x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     f = ossl_list_uint_set_head(s); | 
					
						
							|  |  |  |     if (start <= f->range.start && end >= z->range.end) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         /*
 | 
					
						
							|  |  |  |          * New range dwarfs all ranges in our set. | 
					
						
							|  |  |  |          * | 
					
						
							|  |  |  |          * Free everything except the first range in the set, which we scavenge | 
					
						
							|  |  |  |          * and reuse. | 
					
						
							|  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |         x = ossl_list_uint_set_head(s); | 
					
						
							|  |  |  |         x->range.start = start; | 
					
						
							|  |  |  |         x->range.end = end; | 
					
						
							|  |  |  |         for (x = ossl_list_uint_set_next(x); x != NULL; x = xnext) { | 
					
						
							|  |  |  |             xnext = ossl_list_uint_set_next(x); | 
					
						
							|  |  |  |             ossl_list_uint_set_remove(s, x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |      * Walk backwards since we will most often be inserting at the end. As an | 
					
						
							|  |  |  |      * optimization, test the head node first and skip iterating over the | 
					
						
							|  |  |  |      * entire list if we are inserting at the start. The assumption is that | 
					
						
							|  |  |  |      * insertion at the start and end of the space will be the most common | 
					
						
							|  |  |  |      * operations. (*) | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     z = end < f->range.start ? f : z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; z != NULL; z = zprev) { | 
					
						
							|  |  |  |         zprev = ossl_list_uint_set_prev(z); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         /* An existing range dwarfs our new range (optimisation). */ | 
					
						
							|  |  |  |         if (z->range.start <= start && z->range.end >= end) | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (uint_range_overlaps(&z->range, range)) { | 
					
						
							|  |  |  |             /*
 | 
					
						
							|  |  |  |              * Our new range overlaps an existing range, or possibly several | 
					
						
							|  |  |  |              * existing ranges. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             UINT_SET_ITEM *ovend = z; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             ovend->range.end = u64_max(end, z->range.end); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* Get earliest overlapping range. */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             while (zprev != NULL && uint_range_overlaps(&zprev->range, range)) { | 
					
						
							|  |  |  |                 z = zprev; | 
					
						
							|  |  |  |                 zprev = ossl_list_uint_set_prev(z); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             ovend->range.start = u64_min(start, z->range.start); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             /* Replace sequence of nodes z..ovend with updated ovend only. */ | 
					
						
							|  |  |  |             while (z != ovend) { | 
					
						
							|  |  |  |                 z = ossl_list_uint_set_next(x = z); | 
					
						
							|  |  |  |                 ossl_list_uint_set_remove(s, x); | 
					
						
							|  |  |  |                 OPENSSL_free(x); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } else if (end < z->range.start | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |                     && (zprev == NULL || start > zprev->range.end)) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             if (z->range.start == end + 1) { | 
					
						
							|  |  |  |                 /* We can extend the following range backwards. */ | 
					
						
							|  |  |  |                 z->range.start = start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 /*
 | 
					
						
							|  |  |  |                  * If this closes a gap we now need to merge | 
					
						
							|  |  |  |                  * consecutive nodes. | 
					
						
							|  |  |  |                  */ | 
					
						
							|  |  |  |                 uint_set_merge_adjacent(s, z); | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             } else if (zprev != NULL && zprev->range.end + 1 == start) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |                 /* We can extend the preceding range forwards. */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |                 zprev->range.end = end; | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 /*
 | 
					
						
							|  |  |  |                  * If this closes a gap we now need to merge | 
					
						
							|  |  |  |                  * consecutive nodes. | 
					
						
							|  |  |  |                  */ | 
					
						
							|  |  |  |                 uint_set_merge_adjacent(s, z); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 /*
 | 
					
						
							|  |  |  |                  * The new interval is between intervals without overlapping or | 
					
						
							|  |  |  |                  * touching them, so insert between, preserving sort. | 
					
						
							|  |  |  |                  */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |                 x = create_set_item(start, end); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |                 if (x == NULL) | 
					
						
							|  |  |  |                     return 0; | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |                 ossl_list_uint_set_insert_before(s, z, x); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ossl_uint_set_remove(UINT_SET *s, const UINT_RANGE *range) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     UINT_SET_ITEM *z, *zprev, *y; | 
					
						
							|  |  |  |     uint64_t start = range->start, end = range->end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ossl_assert(start <= end)) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Walk backwards since we will most often be removing at the end. */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     for (z = ossl_list_uint_set_tail(s); z != NULL; z = zprev) { | 
					
						
							|  |  |  |         zprev = ossl_list_uint_set_prev(z); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (start > z->range.end) | 
					
						
							|  |  |  |             /* No overlapping ranges can exist beyond this point, so stop. */ | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (start <= z->range.start && end >= z->range.end) { | 
					
						
							|  |  |  |             /*
 | 
					
						
							|  |  |  |              * The range being removed dwarfs this range, so it should be | 
					
						
							|  |  |  |              * removed. | 
					
						
							|  |  |  |              */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             ossl_list_uint_set_remove(s, z); | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             OPENSSL_free(z); | 
					
						
							| 
									
										
										
										
											2023-07-27 22:55:34 +08:00
										 |  |  |         } else if (start <= z->range.start && end >= z->range.start) { | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             /*
 | 
					
						
							|  |  |  |              * The range being removed includes start of this range, but does | 
					
						
							|  |  |  |              * not cover the entire range (as this would be caught by the case | 
					
						
							|  |  |  |              * above). Shorten the range. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             assert(end < z->range.end); | 
					
						
							|  |  |  |             z->range.start = end + 1; | 
					
						
							|  |  |  |         } else if (end >= z->range.end) { | 
					
						
							|  |  |  |             /*
 | 
					
						
							|  |  |  |              * The range being removed includes the end of this range, but does | 
					
						
							|  |  |  |              * not cover the entire range (as this would be caught by the case | 
					
						
							|  |  |  |              * above). Shorten the range. We can also stop iterating. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             assert(start > z->range.start); | 
					
						
							|  |  |  |             assert(start > 0); | 
					
						
							|  |  |  |             z->range.end = start - 1; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } else if (start > z->range.start && end < z->range.end) { | 
					
						
							|  |  |  |             /*
 | 
					
						
							|  |  |  |              * The range being removed falls entirely in this range, so cut it | 
					
						
							|  |  |  |              * into two. Cases where a zero-length range would be created are | 
					
						
							|  |  |  |              * handled by the above cases. | 
					
						
							|  |  |  |              */ | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |             y = create_set_item(end + 1, z->range.end); | 
					
						
							|  |  |  |             ossl_list_uint_set_insert_after(s, z, y); | 
					
						
							| 
									
										
										
										
											2023-07-27 22:55:34 +08:00
										 |  |  |             z->range.end = start - 1; | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             /* Assert no partial overlap; all cases should be covered above. */ | 
					
						
							|  |  |  |             assert(!uint_range_overlaps(&z->range, range)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-27 22:55:34 +08:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ossl_uint_set_query(const UINT_SET *s, uint64_t v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     UINT_SET_ITEM *x; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     if (ossl_list_uint_set_is_empty(s)) | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-11 15:41:04 +08:00
										 |  |  |     for (x = ossl_list_uint_set_tail(s); x != NULL; x = ossl_list_uint_set_prev(x)) | 
					
						
							| 
									
										
										
										
											2022-09-06 20:23:29 +08:00
										 |  |  |         if (x->range.start <= v && x->range.end >= v) | 
					
						
							|  |  |  |             return 1; | 
					
						
							|  |  |  |         else if (x->range.end < v) | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } |