/*************************************************************************** * _ _ ____ _ * Project ___| | | | _ \| | * / __| | | | |_) | | * | (__| |_| | _ <| |___ * \___|\___/|_| \_\_____| * * Copyright (C) Daniel Stenberg, , 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 "test.h" #include "testutil.h" #include "memdebug.h" #include #if defined(USE_THREADS_POSIX) || defined(USE_THREADS_WIN32) #if defined(USE_THREADS_POSIX) #include #endif #include "curl_threads.h" #endif #define CAINFO libtest_arg2 #define THREAD_SIZE 16 #define PER_THREAD_SIZE 8 struct Ctx { const char *URL; CURLSH *share; int result; int thread_id; struct curl_slist *contents; }; static size_t write_memory_callback(char *contents, size_t size, size_t nmemb, void *userp) { /* append the data to contents */ size_t realsize = size * nmemb; struct Ctx *mem = (struct Ctx *)userp; char *data = (char *)malloc(realsize + 1); struct curl_slist *item_append = NULL; if(!data) { printf("not enough memory (malloc returned NULL)\n"); return 0; } memcpy(data, contents, realsize); data[realsize] = '\0'; item_append = curl_slist_append(mem->contents, data); free(data); if(item_append) { mem->contents = item_append; } else { printf("not enough memory (curl_slist_append returned NULL)\n"); return 0; } return realsize; } static #if defined(USE_THREADS_POSIX) || defined(USE_THREADS_WIN32) #if defined(_WIN32_WCE) || defined(CURL_WINDOWS_UWP) DWORD #else unsigned int #endif CURL_STDCALL #else unsigned int #endif test_thread(void *ptr) { struct Ctx *ctx = (struct Ctx *)ptr; CURLcode res = CURLE_OK; int i; /* Loop the transfer and cleanup the handle properly every lap. This will still reuse ssl session since the pool is in the shared object! */ for(i = 0; i < PER_THREAD_SIZE; i++) { CURL *curl = curl_easy_init(); if(curl) { curl_easy_setopt(curl, CURLOPT_URL, (char *)ctx->URL); /* use the share object */ curl_easy_setopt(curl, CURLOPT_SHARE, ctx->share); curl_easy_setopt(curl, CURLOPT_CAINFO, CAINFO); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_memory_callback); curl_easy_setopt(curl, CURLOPT_WRITEDATA, ptr); curl_easy_setopt(curl, CURLOPT_VERBOSE, 1); /* Perform the request, res will get the return code */ res = curl_easy_perform(curl); /* always cleanup */ curl_easy_cleanup(curl); /* Check for errors */ if(res != CURLE_OK) { fprintf(stderr, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res)); goto test_cleanup; } } } test_cleanup: ctx->result = (int)res; return 0; } #if defined(USE_THREADS_POSIX) || defined(USE_THREADS_WIN32) static void test_lock(CURL *handle, curl_lock_data data, curl_lock_access laccess, void *useptr) { curl_mutex_t *mutexes = (curl_mutex_t*) useptr; (void)handle; (void)laccess; Curl_mutex_acquire(&mutexes[data]); } static void test_unlock(CURL *handle, curl_lock_data data, void *useptr) { curl_mutex_t *mutexes = (curl_mutex_t*) useptr; (void)handle; Curl_mutex_release(&mutexes[data]); } static void execute(CURLSH *share, struct Ctx *ctx) { int i; curl_mutex_t mutexes[CURL_LOCK_DATA_LAST - 1]; curl_thread_t thread[THREAD_SIZE]; for(i = 0; i < CURL_LOCK_DATA_LAST - 1; i++) { Curl_mutex_init(&mutexes[i]); } curl_share_setopt(share, CURLSHOPT_LOCKFUNC, test_lock); curl_share_setopt(share, CURLSHOPT_UNLOCKFUNC, test_unlock); curl_share_setopt(share, CURLSHOPT_USERDATA, (void *)mutexes); curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_SSL_SESSION); for(i = 0; i < THREAD_SIZE; i++) { thread[i] = Curl_thread_create(test_thread, (void *)&ctx[i]); } for(i = 0; i < THREAD_SIZE; i++) { if(thread[i]) { Curl_thread_join(&thread[i]); Curl_thread_destroy(thread[i]); } } curl_share_setopt(share, CURLSHOPT_LOCKFUNC, NULL); curl_share_setopt(share, CURLSHOPT_UNLOCKFUNC, NULL); for(i = 0; i < CURL_LOCK_DATA_LAST - 1; i++) { Curl_mutex_destroy(&mutexes[i]); } } #else /* without pthread, run serially */ static void execute(CURLSH *share, struct Ctx *ctx) { int i; (void) share; for(i = 0; i < THREAD_SIZE; i++) { test_thread((void *)&ctx[i]); } } #endif CURLcode test(char *URL) { int res = 0; int i; CURLSH* share; struct Ctx ctx[THREAD_SIZE]; curl_global_init(CURL_GLOBAL_ALL); share = curl_share_init(); if(!share) { fprintf(stderr, "curl_share_init() failed\n"); goto test_cleanup; } for(i = 0; i < THREAD_SIZE; i++) { ctx[i].share = share; ctx[i].URL = URL; ctx[i].thread_id = i; ctx[i].result = 0; ctx[i].contents = NULL; } execute(share, ctx); for(i = 0; i < THREAD_SIZE; i++) { if(ctx[i].result) { res = ctx[i].result; } else { struct curl_slist *item = ctx[i].contents; while(item) { printf("%s", item->data); item = item->next; } } curl_slist_free_all(ctx[i].contents); } test_cleanup: if(share) curl_share_cleanup(share); curl_global_cleanup(); return (CURLcode)res; }