curl-curl/tests/unit/unit3301.c
Stefan Eissing 6f9f4b3cb7
lib: add thread pool and queue
- new source files for thread pool and queue
- test cases 3217 and 3218 for them
- internal documentation

Closes #20916
2026-03-23 23:03:58 +01:00

145 lines
3.9 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 "unitcheck.h"
#include "curlx/wait.h"
#include "thrdqueue.h"
#include "curl_threads.h"
#ifdef USE_THREADS
struct unit3301_item {
int id;
BIT(processed);
};
struct unit3301_ctx {
volatile int event;
};
static struct unit3301_item *unit3301_item_create(int id)
{
struct unit3301_item *uitem;
uitem = curlx_calloc(1, sizeof(*uitem));
if(uitem) {
uitem->id = id;
curl_mfprintf(stderr, "created item %d\n", uitem->id);
}
return uitem;
}
static void unit3301_item_free(void *item)
{
struct unit3301_item *uitem = item;
curl_mfprintf(stderr, "free item %d\n", uitem->id);
curlx_free(uitem);
}
static void unit3301_event(const struct curl_thrdq *tqueue,
Curl_thrdq_event ev,
void *user_data)
{
struct unit3301_ctx *ctx = user_data;
(void)tqueue;
switch(ev) {
case CURL_THRDQ_EV_ITEM_DONE:
ctx->event = 1;
break;
default:
break;
}
}
static void unit3301_process(void *item)
{
struct unit3301_item *uitem = item;
curlx_wait_ms(1);
uitem->processed = TRUE;
}
static CURLcode test_unit3301(const char *arg)
{
UNITTEST_BEGIN_SIMPLE
struct curl_thrdq *tqueue;
struct unit3301_ctx ctx;
int i, count, nrecvd;
CURLcode r;
/* create and teardown queue */
memset(&ctx, 0, sizeof(ctx));
r = Curl_thrdq_create(&tqueue, "unit3301-a", 0, 0, 2, 1,
unit3301_item_free, unit3301_process, unit3301_event,
&ctx);
fail_unless(!r, "queue-a create");
Curl_thrdq_destroy(tqueue, TRUE);
tqueue = NULL;
fail_unless(!ctx.event, "queue-a unexpected done count");
/* create queue, have it process `count` items */
count = 10;
memset(&ctx, 0, sizeof(ctx));
r = Curl_thrdq_create(&tqueue, "unit3301-b", 0, 0, 2, 1,
unit3301_item_free, unit3301_process, unit3301_event,
&ctx);
fail_unless(!r, "queue-b create");
for(i = 0; i < count; ++i) {
struct unit3301_item *uitem = unit3301_item_create(i);
fail_unless(uitem, "queue-b item create");
r = Curl_thrdq_send(tqueue, uitem, NULL, 0);
fail_unless(!r, "queue-b send");
}
r = Curl_thrdq_await_done(tqueue, 0);
fail_unless(!r, "queue-b await done");
nrecvd = 0;
for(i = 0; i < count; ++i) {
void *item;
r = Curl_thrdq_recv(tqueue, &item);
fail_unless(!r, "queue-b recv");
if(item) {
struct unit3301_item *uitem = item;
curl_mfprintf(stderr, "received item %d\n", uitem->id);
++nrecvd;
fail_unless(uitem->processed, "queue-b recv unprocessed item");
unit3301_item_free(item);
}
}
Curl_thrdq_destroy(tqueue, TRUE);
tqueue = NULL;
fail_unless(nrecvd == count, "queue-b unexpected done count");
UNITTEST_END_SIMPLE
}
#else
static CURLcode test_unit3301(const char *arg)
{
UNITTEST_BEGIN_SIMPLE
(void)arg;
UNITTEST_END_SIMPLE
}
#endif /* USE_THREADS */