mirror of
https://github.com/curl/curl.git
synced 2026-01-18 17:21:26 +01:00
- asyn-thrdd.c: scope an include. - apply more clang-format suggestions. - tidy-up PP guard comments. - delete empty line from the top of headers. - add empty line after `curl_setup.h` include where missing. - fix indent. - CODE_STYLE.md: add `strcpy`. Follow-up to8636ad55df#20088 - lib1901.c: drop unnecessary line. Follow-up to436e67f65b#20076 Closes #20070
201 lines
5.4 KiB
C
201 lines
5.4 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-table.h"
|
|
|
|
#ifdef DEBUGBUILD
|
|
#define CURL_UINT32_TBL_MAGIC 0x62757473
|
|
#endif
|
|
|
|
/* Clear the table, making it empty. */
|
|
UNITTEST void Curl_uint32_tbl_clear(struct uint32_tbl *tbl);
|
|
|
|
void Curl_uint32_tbl_init(struct uint32_tbl *tbl,
|
|
Curl_uint32_tbl_entry_dtor *entry_dtor)
|
|
{
|
|
memset(tbl, 0, sizeof(*tbl));
|
|
tbl->entry_dtor = entry_dtor;
|
|
tbl->last_key_added = UINT32_MAX;
|
|
#ifdef DEBUGBUILD
|
|
tbl->init = CURL_UINT32_TBL_MAGIC;
|
|
#endif
|
|
}
|
|
|
|
static void uint32_tbl_clear_rows(struct uint32_tbl *tbl,
|
|
uint32_t from,
|
|
uint32_t upto_excluding)
|
|
{
|
|
uint32_t i, end;
|
|
|
|
end = CURLMIN(upto_excluding, tbl->nrows);
|
|
for(i = from; i < end; ++i) {
|
|
if(tbl->rows[i]) {
|
|
if(tbl->entry_dtor)
|
|
tbl->entry_dtor(i, tbl->rows[i]);
|
|
tbl->rows[i] = NULL;
|
|
tbl->nentries--;
|
|
}
|
|
}
|
|
}
|
|
|
|
CURLcode Curl_uint32_tbl_resize(struct uint32_tbl *tbl, uint32_t nrows)
|
|
{
|
|
/* we use `tbl->nrows + 1` during iteration, want that to work */
|
|
DEBUGASSERT(tbl->init == CURL_UINT32_TBL_MAGIC);
|
|
if(!nrows)
|
|
return CURLE_BAD_FUNCTION_ARGUMENT;
|
|
if(nrows != tbl->nrows) {
|
|
void **rows = curlx_calloc(nrows, sizeof(void *));
|
|
if(!rows)
|
|
return CURLE_OUT_OF_MEMORY;
|
|
if(tbl->rows) {
|
|
memcpy(rows, tbl->rows, (CURLMIN(nrows, tbl->nrows) * sizeof(void *)));
|
|
if(nrows < tbl->nrows)
|
|
uint32_tbl_clear_rows(tbl, nrows, tbl->nrows);
|
|
curlx_free(tbl->rows);
|
|
}
|
|
tbl->rows = rows;
|
|
tbl->nrows = nrows;
|
|
}
|
|
return CURLE_OK;
|
|
}
|
|
|
|
void Curl_uint32_tbl_destroy(struct uint32_tbl *tbl)
|
|
{
|
|
DEBUGASSERT(tbl->init == CURL_UINT32_TBL_MAGIC);
|
|
Curl_uint32_tbl_clear(tbl);
|
|
curlx_free(tbl->rows);
|
|
memset(tbl, 0, sizeof(*tbl));
|
|
}
|
|
|
|
UNITTEST void Curl_uint32_tbl_clear(struct uint32_tbl *tbl)
|
|
{
|
|
DEBUGASSERT(tbl->init == CURL_UINT32_TBL_MAGIC);
|
|
uint32_tbl_clear_rows(tbl, 0, tbl->nrows);
|
|
DEBUGASSERT(!tbl->nentries);
|
|
tbl->last_key_added = UINT32_MAX;
|
|
}
|
|
|
|
uint32_t Curl_uint32_tbl_capacity(struct uint32_tbl *tbl)
|
|
{
|
|
return tbl->nrows;
|
|
}
|
|
|
|
uint32_t Curl_uint32_tbl_count(struct uint32_tbl *tbl)
|
|
{
|
|
return tbl->nentries;
|
|
}
|
|
|
|
void *Curl_uint32_tbl_get(struct uint32_tbl *tbl, uint32_t key)
|
|
{
|
|
return (key < tbl->nrows) ? tbl->rows[key] : NULL;
|
|
}
|
|
|
|
bool Curl_uint32_tbl_add(struct uint32_tbl *tbl, void *entry, uint32_t *pkey)
|
|
{
|
|
uint32_t key, start_pos;
|
|
|
|
DEBUGASSERT(tbl->init == CURL_UINT32_TBL_MAGIC);
|
|
if(!entry || !pkey)
|
|
return FALSE;
|
|
*pkey = UINT32_MAX;
|
|
if(tbl->nentries == tbl->nrows) /* full */
|
|
return FALSE;
|
|
|
|
start_pos = CURLMIN(tbl->last_key_added, tbl->nrows) + 1;
|
|
for(key = start_pos; key < tbl->nrows; ++key) {
|
|
if(!tbl->rows[key]) {
|
|
tbl->rows[key] = entry;
|
|
tbl->nentries++;
|
|
tbl->last_key_added = key;
|
|
*pkey = key;
|
|
return TRUE;
|
|
}
|
|
}
|
|
/* no free entry at or above tbl->maybe_next_key, wrap around */
|
|
for(key = 0; key < start_pos; ++key) {
|
|
if(!tbl->rows[key]) {
|
|
tbl->rows[key] = entry;
|
|
tbl->nentries++;
|
|
tbl->last_key_added = key;
|
|
*pkey = key;
|
|
return TRUE;
|
|
}
|
|
}
|
|
/* Did not find any free row? Should not happen */
|
|
DEBUGASSERT(0);
|
|
return FALSE;
|
|
}
|
|
|
|
void Curl_uint32_tbl_remove(struct uint32_tbl *tbl, uint32_t key)
|
|
{
|
|
uint32_tbl_clear_rows(tbl, key, key + 1);
|
|
}
|
|
|
|
bool Curl_uint32_tbl_contains(struct uint32_tbl *tbl, uint32_t key)
|
|
{
|
|
return (key < tbl->nrows) ? !!tbl->rows[key] : FALSE;
|
|
}
|
|
|
|
static bool uint32_tbl_next_at(struct uint32_tbl *tbl, uint32_t key,
|
|
uint32_t *pkey, void **pentry)
|
|
{
|
|
for(; key < tbl->nrows; ++key) {
|
|
if(tbl->rows[key]) {
|
|
*pkey = key;
|
|
*pentry = tbl->rows[key];
|
|
return TRUE;
|
|
}
|
|
}
|
|
*pkey = UINT32_MAX; /* always invalid */
|
|
*pentry = NULL;
|
|
return FALSE;
|
|
}
|
|
|
|
bool Curl_uint32_tbl_first(struct uint32_tbl *tbl,
|
|
uint32_t *pkey, void **pentry)
|
|
{
|
|
if(!pkey || !pentry)
|
|
return FALSE;
|
|
if(tbl->nentries && uint32_tbl_next_at(tbl, 0, pkey, pentry))
|
|
return TRUE;
|
|
DEBUGASSERT(!tbl->nentries);
|
|
*pkey = UINT32_MAX; /* always invalid */
|
|
*pentry = NULL;
|
|
return FALSE;
|
|
}
|
|
|
|
bool Curl_uint32_tbl_next(struct uint32_tbl *tbl, uint32_t last_key,
|
|
uint32_t *pkey, void **pentry)
|
|
{
|
|
if(!pkey || !pentry)
|
|
return FALSE;
|
|
if(uint32_tbl_next_at(tbl, last_key + 1, pkey, pentry))
|
|
return TRUE;
|
|
*pkey = UINT32_MAX; /* always invalid */
|
|
*pentry = NULL;
|
|
return FALSE;
|
|
}
|