113498266Sopenharmony_ci/*************************************************************************** 213498266Sopenharmony_ci * _ _ ____ _ 313498266Sopenharmony_ci * Project ___| | | | _ \| | 413498266Sopenharmony_ci * / __| | | | |_) | | 513498266Sopenharmony_ci * | (__| |_| | _ <| |___ 613498266Sopenharmony_ci * \___|\___/|_| \_\_____| 713498266Sopenharmony_ci * 813498266Sopenharmony_ci * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al. 913498266Sopenharmony_ci * 1013498266Sopenharmony_ci * This software is licensed as described in the file COPYING, which 1113498266Sopenharmony_ci * you should have received as part of this distribution. The terms 1213498266Sopenharmony_ci * are also available at https://curl.se/docs/copyright.html. 1313498266Sopenharmony_ci * 1413498266Sopenharmony_ci * You may opt to use, copy, modify, merge, publish, distribute and/or sell 1513498266Sopenharmony_ci * copies of the Software, and permit persons to whom the Software is 1613498266Sopenharmony_ci * furnished to do so, under the terms of the COPYING file. 1713498266Sopenharmony_ci * 1813498266Sopenharmony_ci * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY 1913498266Sopenharmony_ci * KIND, either express or implied. 2013498266Sopenharmony_ci * 2113498266Sopenharmony_ci * SPDX-License-Identifier: curl 2213498266Sopenharmony_ci * 2313498266Sopenharmony_ci ***************************************************************************/ 2413498266Sopenharmony_ci#include "test.h" 2513498266Sopenharmony_ci 2613498266Sopenharmony_ci#include "testutil.h" 2713498266Sopenharmony_ci#include "warnless.h" 2813498266Sopenharmony_ci#include "memdebug.h" 2913498266Sopenharmony_ci 3013498266Sopenharmony_ci#ifdef HAVE_PTHREAD_H 3113498266Sopenharmony_ci#include <pthread.h> 3213498266Sopenharmony_ci#include <unistd.h> 3313498266Sopenharmony_ci 3413498266Sopenharmony_ci#define TEST_HANG_TIMEOUT 60 * 1000 3513498266Sopenharmony_ci#define CONN_NUM 3 3613498266Sopenharmony_ci#define TIME_BETWEEN_START_SECS 2 3713498266Sopenharmony_ci 3813498266Sopenharmony_cistatic pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; 3913498266Sopenharmony_cistatic CURL *pending_handles[CONN_NUM]; 4013498266Sopenharmony_cistatic int pending_num = 0; 4113498266Sopenharmony_cistatic int test_failure = 0; 4213498266Sopenharmony_ci 4313498266Sopenharmony_cistatic CURLM *multi = NULL; 4413498266Sopenharmony_cistatic const char *url; 4513498266Sopenharmony_ci 4613498266Sopenharmony_cistatic void *run_thread(void *ptr) 4713498266Sopenharmony_ci{ 4813498266Sopenharmony_ci CURL *easy = NULL; 4913498266Sopenharmony_ci int res = 0; 5013498266Sopenharmony_ci int i; 5113498266Sopenharmony_ci 5213498266Sopenharmony_ci (void)ptr; 5313498266Sopenharmony_ci 5413498266Sopenharmony_ci for(i = 0; i < CONN_NUM; i++) { 5513498266Sopenharmony_ci wait_ms(TIME_BETWEEN_START_SECS * 1000); 5613498266Sopenharmony_ci 5713498266Sopenharmony_ci easy_init(easy); 5813498266Sopenharmony_ci 5913498266Sopenharmony_ci easy_setopt(easy, CURLOPT_URL, url); 6013498266Sopenharmony_ci easy_setopt(easy, CURLOPT_VERBOSE, 0L); 6113498266Sopenharmony_ci 6213498266Sopenharmony_ci pthread_mutex_lock(&lock); 6313498266Sopenharmony_ci 6413498266Sopenharmony_ci if(test_failure) { 6513498266Sopenharmony_ci pthread_mutex_unlock(&lock); 6613498266Sopenharmony_ci goto test_cleanup; 6713498266Sopenharmony_ci } 6813498266Sopenharmony_ci 6913498266Sopenharmony_ci pending_handles[pending_num] = easy; 7013498266Sopenharmony_ci pending_num++; 7113498266Sopenharmony_ci easy = NULL; 7213498266Sopenharmony_ci 7313498266Sopenharmony_ci pthread_mutex_unlock(&lock); 7413498266Sopenharmony_ci 7513498266Sopenharmony_ci res_multi_wakeup(multi); 7613498266Sopenharmony_ci } 7713498266Sopenharmony_ci 7813498266Sopenharmony_citest_cleanup: 7913498266Sopenharmony_ci 8013498266Sopenharmony_ci curl_easy_cleanup(easy); 8113498266Sopenharmony_ci 8213498266Sopenharmony_ci pthread_mutex_lock(&lock); 8313498266Sopenharmony_ci 8413498266Sopenharmony_ci if(!test_failure) 8513498266Sopenharmony_ci test_failure = res; 8613498266Sopenharmony_ci 8713498266Sopenharmony_ci pthread_mutex_unlock(&lock); 8813498266Sopenharmony_ci 8913498266Sopenharmony_ci return NULL; 9013498266Sopenharmony_ci} 9113498266Sopenharmony_ci 9213498266Sopenharmony_ciint test(char *URL) 9313498266Sopenharmony_ci{ 9413498266Sopenharmony_ci int still_running; 9513498266Sopenharmony_ci int num; 9613498266Sopenharmony_ci int i; 9713498266Sopenharmony_ci int res = 0; 9813498266Sopenharmony_ci CURL *started_handles[CONN_NUM]; 9913498266Sopenharmony_ci int started_num = 0; 10013498266Sopenharmony_ci int finished_num = 0; 10113498266Sopenharmony_ci pthread_t tid; 10213498266Sopenharmony_ci bool tid_valid = false; 10313498266Sopenharmony_ci struct CURLMsg *message; 10413498266Sopenharmony_ci 10513498266Sopenharmony_ci start_test_timing(); 10613498266Sopenharmony_ci 10713498266Sopenharmony_ci global_init(CURL_GLOBAL_ALL); 10813498266Sopenharmony_ci 10913498266Sopenharmony_ci multi_init(multi); 11013498266Sopenharmony_ci 11113498266Sopenharmony_ci url = URL; 11213498266Sopenharmony_ci 11313498266Sopenharmony_ci res = pthread_create(&tid, NULL, run_thread, NULL); 11413498266Sopenharmony_ci if(!res) 11513498266Sopenharmony_ci tid_valid = true; 11613498266Sopenharmony_ci else { 11713498266Sopenharmony_ci fprintf(stderr, "%s:%d Couldn't create thread, errno %d\n", 11813498266Sopenharmony_ci __FILE__, __LINE__, res); 11913498266Sopenharmony_ci goto test_cleanup; 12013498266Sopenharmony_ci } 12113498266Sopenharmony_ci 12213498266Sopenharmony_ci while(1) { 12313498266Sopenharmony_ci multi_perform(multi, &still_running); 12413498266Sopenharmony_ci 12513498266Sopenharmony_ci abort_on_test_timeout(); 12613498266Sopenharmony_ci 12713498266Sopenharmony_ci while((message = curl_multi_info_read(multi, &num))) { 12813498266Sopenharmony_ci if(message->msg == CURLMSG_DONE) { 12913498266Sopenharmony_ci res = message->data.result; 13013498266Sopenharmony_ci if(res) 13113498266Sopenharmony_ci goto test_cleanup; 13213498266Sopenharmony_ci multi_remove_handle(multi, message->easy_handle); 13313498266Sopenharmony_ci finished_num++; 13413498266Sopenharmony_ci } 13513498266Sopenharmony_ci else { 13613498266Sopenharmony_ci fprintf(stderr, "%s:%d Got an unexpected message from curl: %i\n", 13713498266Sopenharmony_ci __FILE__, __LINE__, (int)message->msg); 13813498266Sopenharmony_ci res = TEST_ERR_MAJOR_BAD; 13913498266Sopenharmony_ci goto test_cleanup; 14013498266Sopenharmony_ci } 14113498266Sopenharmony_ci 14213498266Sopenharmony_ci abort_on_test_timeout(); 14313498266Sopenharmony_ci } 14413498266Sopenharmony_ci 14513498266Sopenharmony_ci if(CONN_NUM == finished_num) 14613498266Sopenharmony_ci break; 14713498266Sopenharmony_ci 14813498266Sopenharmony_ci multi_poll(multi, NULL, 0, TEST_HANG_TIMEOUT, &num); 14913498266Sopenharmony_ci 15013498266Sopenharmony_ci abort_on_test_timeout(); 15113498266Sopenharmony_ci 15213498266Sopenharmony_ci pthread_mutex_lock(&lock); 15313498266Sopenharmony_ci 15413498266Sopenharmony_ci while(pending_num > 0) { 15513498266Sopenharmony_ci res_multi_add_handle(multi, pending_handles[pending_num - 1]); 15613498266Sopenharmony_ci if(res) { 15713498266Sopenharmony_ci pthread_mutex_unlock(&lock); 15813498266Sopenharmony_ci goto test_cleanup; 15913498266Sopenharmony_ci } 16013498266Sopenharmony_ci 16113498266Sopenharmony_ci started_handles[started_num] = pending_handles[pending_num - 1]; 16213498266Sopenharmony_ci started_num++; 16313498266Sopenharmony_ci pending_num--; 16413498266Sopenharmony_ci } 16513498266Sopenharmony_ci 16613498266Sopenharmony_ci pthread_mutex_unlock(&lock); 16713498266Sopenharmony_ci 16813498266Sopenharmony_ci abort_on_test_timeout(); 16913498266Sopenharmony_ci } 17013498266Sopenharmony_ci 17113498266Sopenharmony_ci if(CONN_NUM != started_num) { 17213498266Sopenharmony_ci fprintf(stderr, "%s:%d Not all connections started: %d of %d\n", 17313498266Sopenharmony_ci __FILE__, __LINE__, started_num, CONN_NUM); 17413498266Sopenharmony_ci goto test_cleanup; 17513498266Sopenharmony_ci } 17613498266Sopenharmony_ci 17713498266Sopenharmony_ci if(CONN_NUM != finished_num) { 17813498266Sopenharmony_ci fprintf(stderr, "%s:%d Not all connections finished: %d of %d\n", 17913498266Sopenharmony_ci __FILE__, __LINE__, started_num, CONN_NUM); 18013498266Sopenharmony_ci goto test_cleanup; 18113498266Sopenharmony_ci } 18213498266Sopenharmony_ci 18313498266Sopenharmony_citest_cleanup: 18413498266Sopenharmony_ci 18513498266Sopenharmony_ci pthread_mutex_lock(&lock); 18613498266Sopenharmony_ci if(!test_failure) 18713498266Sopenharmony_ci test_failure = res; 18813498266Sopenharmony_ci pthread_mutex_unlock(&lock); 18913498266Sopenharmony_ci 19013498266Sopenharmony_ci if(tid_valid) 19113498266Sopenharmony_ci pthread_join(tid, NULL); 19213498266Sopenharmony_ci 19313498266Sopenharmony_ci curl_multi_cleanup(multi); 19413498266Sopenharmony_ci for(i = 0; i < pending_num; i++) 19513498266Sopenharmony_ci curl_easy_cleanup(pending_handles[i]); 19613498266Sopenharmony_ci for(i = 0; i < started_num; i++) 19713498266Sopenharmony_ci curl_easy_cleanup(started_handles[i]); 19813498266Sopenharmony_ci curl_global_cleanup(); 19913498266Sopenharmony_ci 20013498266Sopenharmony_ci return test_failure; 20113498266Sopenharmony_ci} 20213498266Sopenharmony_ci 20313498266Sopenharmony_ci#else /* without pthread, this test doesn't work */ 20413498266Sopenharmony_ciint test(char *URL) 20513498266Sopenharmony_ci{ 20613498266Sopenharmony_ci (void)URL; 20713498266Sopenharmony_ci return 0; 20813498266Sopenharmony_ci} 20913498266Sopenharmony_ci#endif 210