DN/External/curl-8.17.0/lib/uint-spbset.c

276 lines
7.3 KiB
C

/***************************************************************************
* _ _ ____ _
* Project ___| | | | _ \| |
* / __| | | | |_) | |
* | (__| |_| | _ <| |___
* \___|\___/|_| \_\_____|
*
* Copyright (C) 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
* are also available at https://curl.se/docs/copyright.html.
*
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
* copies of the Software, and permit persons to whom the Software is
* furnished to do so, under the terms of the COPYING file.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
* SPDX-License-Identifier: curl
*
***************************************************************************/
#include "curl_setup.h"
#include "uint-bset.h"
#include "uint-spbset.h"
/* The last 2 #include files should be in this order */
#include "curl_memory.h"
#include "memdebug.h"
#ifdef DEBUGBUILD
#define CURL_UINT_SPBSET_MAGIC 0x70737362
#endif
/* Clear the bitset, making it empty. */
UNITTEST void Curl_uint_spbset_clear(struct uint_spbset *bset);
void Curl_uint_spbset_init(struct uint_spbset *bset)
{
memset(bset, 0, sizeof(*bset));
#ifdef DEBUGBUILD
bset->init = CURL_UINT_SPBSET_MAGIC;
#endif
}
void Curl_uint_spbset_destroy(struct uint_spbset *bset)
{
DEBUGASSERT(bset->init == CURL_UINT_SPBSET_MAGIC);
Curl_uint_spbset_clear(bset);
}
unsigned int Curl_uint_spbset_count(struct uint_spbset *bset)
{
struct uint_spbset_chunk *chunk;
unsigned int i, n = 0;
for(chunk = &bset->head; chunk; chunk = chunk->next) {
for(i = 0; i < CURL_UINT_SPBSET_CH_SLOTS; ++i) {
if(chunk->slots[i])
n += CURL_POPCOUNT64(chunk->slots[i]);
}
}
return n;
}
bool Curl_uint_spbset_empty(struct uint_spbset *bset)
{
struct uint_spbset_chunk *chunk;
unsigned int i;
for(chunk = &bset->head; chunk; chunk = chunk->next) {
for(i = 0; i < CURL_UINT_SPBSET_CH_SLOTS; ++i) {
if(chunk->slots[i])
return FALSE;
}
}
return TRUE;
}
UNITTEST void Curl_uint_spbset_clear(struct uint_spbset *bset)
{
struct uint_spbset_chunk *next, *chunk;
for(chunk = bset->head.next; chunk; chunk = next) {
next = chunk->next;
free(chunk);
}
memset(&bset->head, 0, sizeof(bset->head));
}
static struct uint_spbset_chunk *
uint_spbset_get_chunk(struct uint_spbset *bset, unsigned int i, bool grow)
{
struct uint_spbset_chunk *chunk, **panchor = NULL;
unsigned int i_offset = (i & ~CURL_UINT_SPBSET_CH_MASK);
if(!bset)
return NULL;
for(chunk = &bset->head; chunk;
panchor = &chunk->next, chunk = chunk->next) {
if(chunk->offset == i_offset) {
return chunk;
}
else if(chunk->offset > i_offset) {
/* need new chunk here */
chunk = NULL;
break;
}
}
if(!grow)
return NULL;
/* need a new one */
chunk = calloc(1, sizeof(*chunk));
if(!chunk)
return NULL;
if(panchor) { /* insert between panchor and *panchor */
chunk->next = *panchor;
*panchor = chunk;
}
else { /* prepend to head, switching places */
memcpy(chunk, &bset->head, sizeof(*chunk));
memset(&bset->head, 0, sizeof(bset->head));
bset->head.next = chunk;
}
chunk->offset = i_offset;
return chunk;
}
bool Curl_uint_spbset_add(struct uint_spbset *bset, unsigned int i)
{
struct uint_spbset_chunk *chunk;
unsigned int i_chunk;
chunk = uint_spbset_get_chunk(bset, i, TRUE);
if(!chunk)
return FALSE;
DEBUGASSERT(i >= chunk->offset);
i_chunk = (i - chunk->offset);
DEBUGASSERT((i_chunk / 64) < CURL_UINT_SPBSET_CH_SLOTS);
chunk->slots[(i_chunk / 64)] |= ((curl_uint64_t)1 << (i_chunk % 64));
return TRUE;
}
void Curl_uint_spbset_remove(struct uint_spbset *bset, unsigned int i)
{
struct uint_spbset_chunk *chunk;
unsigned int i_chunk;
chunk = uint_spbset_get_chunk(bset, i, FALSE);
if(chunk) {
DEBUGASSERT(i >= chunk->offset);
i_chunk = (i - chunk->offset);
DEBUGASSERT((i_chunk / 64) < CURL_UINT_SPBSET_CH_SLOTS);
chunk->slots[(i_chunk / 64)] &= ~((curl_uint64_t)1 << (i_chunk % 64));
}
}
bool Curl_uint_spbset_contains(struct uint_spbset *bset, unsigned int i)
{
struct uint_spbset_chunk *chunk;
unsigned int i_chunk;
chunk = uint_spbset_get_chunk(bset, i, FALSE);
if(chunk) {
DEBUGASSERT(i >= chunk->offset);
i_chunk = (i - chunk->offset);
DEBUGASSERT((i_chunk / 64) < CURL_UINT_SPBSET_CH_SLOTS);
return (chunk->slots[i_chunk / 64] &
((curl_uint64_t)1 << (i_chunk % 64))) != 0;
}
return FALSE;
}
bool Curl_uint_spbset_first(struct uint_spbset *bset, unsigned int *pfirst)
{
struct uint_spbset_chunk *chunk;
unsigned int i;
for(chunk = &bset->head; chunk; chunk = chunk->next) {
for(i = 0; i < CURL_UINT_SPBSET_CH_SLOTS; ++i) {
if(chunk->slots[i]) {
*pfirst = chunk->offset + ((i * 64) + CURL_CTZ64(chunk->slots[i]));
return TRUE;
}
}
}
*pfirst = 0; /* give it a defined value even if it should not be used */
return FALSE;
}
static bool uint_spbset_chunk_first(struct uint_spbset_chunk *chunk,
unsigned int *pfirst)
{
unsigned int i;
for(i = 0; i < CURL_UINT_SPBSET_CH_SLOTS; ++i) {
if(chunk->slots[i]) {
*pfirst = chunk->offset + ((i * 64) + CURL_CTZ64(chunk->slots[i]));
return TRUE;
}
}
*pfirst = UINT_MAX; /* a value we cannot store */
return FALSE;
}
static bool uint_spbset_chunk_next(struct uint_spbset_chunk *chunk,
unsigned int last,
unsigned int *pnext)
{
if(chunk->offset <= last) {
curl_uint64_t x;
unsigned int i = ((last - chunk->offset) / 64);
if(i < CURL_UINT_SPBSET_CH_SLOTS) {
x = (chunk->slots[i] >> (last % 64));
if(x) {
/* more bits set, next is `last` + trailing0s of the shifted slot */
*pnext = last + CURL_CTZ64(x);
return TRUE;
}
/* no more bits set in the last slot, scan forward */
for(i = i + 1; i < CURL_UINT_SPBSET_CH_SLOTS; ++i) {
if(chunk->slots[i]) {
*pnext = chunk->offset + ((i * 64) + CURL_CTZ64(chunk->slots[i]));
return TRUE;
}
}
}
}
*pnext = UINT_MAX;
return FALSE;
}
bool Curl_uint_spbset_next(struct uint_spbset *bset, unsigned int last,
unsigned int *pnext)
{
struct uint_spbset_chunk *chunk;
unsigned int last_offset;
++last; /* look for the next higher number */
last_offset = (last & ~CURL_UINT_SPBSET_CH_MASK);
for(chunk = &bset->head; chunk; chunk = chunk->next) {
if(chunk->offset >= last_offset) {
break;
}
}
if(chunk && (chunk->offset == last_offset)) {
/* is there a number higher than last in this chunk? */
if(uint_spbset_chunk_next(chunk, last, pnext))
return TRUE;
/* not in this chunk */
chunk = chunk->next;
}
/* look for the first in the "higher" chunks, if there are any. */
while(chunk) {
if(uint_spbset_chunk_first(chunk, pnext))
return TRUE;
chunk = chunk->next;
}
*pnext = UINT_MAX;
return FALSE;
}