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 2513498266Sopenharmony_ci/* <DESC> 2613498266Sopenharmony_ci * multi_socket API using libevent 2713498266Sopenharmony_ci * </DESC> 2813498266Sopenharmony_ci */ 2913498266Sopenharmony_ci 3013498266Sopenharmony_ci#include <stdio.h> 3113498266Sopenharmony_ci#include <stdlib.h> 3213498266Sopenharmony_ci#include <event2/event.h> 3313498266Sopenharmony_ci#include <curl/curl.h> 3413498266Sopenharmony_ci 3513498266Sopenharmony_cistruct event_base *base; 3613498266Sopenharmony_ciCURLM *curl_handle; 3713498266Sopenharmony_cistruct event *timeout; 3813498266Sopenharmony_ci 3913498266Sopenharmony_citypedef struct curl_context_s { 4013498266Sopenharmony_ci struct event *event; 4113498266Sopenharmony_ci curl_socket_t sockfd; 4213498266Sopenharmony_ci} curl_context_t; 4313498266Sopenharmony_ci 4413498266Sopenharmony_cistatic void curl_perform(int fd, short event, void *arg); 4513498266Sopenharmony_ci 4613498266Sopenharmony_cistatic curl_context_t *create_curl_context(curl_socket_t sockfd) 4713498266Sopenharmony_ci{ 4813498266Sopenharmony_ci curl_context_t *context; 4913498266Sopenharmony_ci 5013498266Sopenharmony_ci context = (curl_context_t *) malloc(sizeof(*context)); 5113498266Sopenharmony_ci 5213498266Sopenharmony_ci context->sockfd = sockfd; 5313498266Sopenharmony_ci 5413498266Sopenharmony_ci context->event = event_new(base, sockfd, 0, curl_perform, context); 5513498266Sopenharmony_ci 5613498266Sopenharmony_ci return context; 5713498266Sopenharmony_ci} 5813498266Sopenharmony_ci 5913498266Sopenharmony_cistatic void destroy_curl_context(curl_context_t *context) 6013498266Sopenharmony_ci{ 6113498266Sopenharmony_ci event_del(context->event); 6213498266Sopenharmony_ci event_free(context->event); 6313498266Sopenharmony_ci free(context); 6413498266Sopenharmony_ci} 6513498266Sopenharmony_ci 6613498266Sopenharmony_cistatic void add_download(const char *url, int num) 6713498266Sopenharmony_ci{ 6813498266Sopenharmony_ci char filename[50]; 6913498266Sopenharmony_ci FILE *file; 7013498266Sopenharmony_ci CURL *handle; 7113498266Sopenharmony_ci 7213498266Sopenharmony_ci snprintf(filename, 50, "%d.download", num); 7313498266Sopenharmony_ci 7413498266Sopenharmony_ci file = fopen(filename, "wb"); 7513498266Sopenharmony_ci if(!file) { 7613498266Sopenharmony_ci fprintf(stderr, "Error opening %s\n", filename); 7713498266Sopenharmony_ci return; 7813498266Sopenharmony_ci } 7913498266Sopenharmony_ci 8013498266Sopenharmony_ci handle = curl_easy_init(); 8113498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_WRITEDATA, file); 8213498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_PRIVATE, file); 8313498266Sopenharmony_ci curl_easy_setopt(handle, CURLOPT_URL, url); 8413498266Sopenharmony_ci curl_multi_add_handle(curl_handle, handle); 8513498266Sopenharmony_ci fprintf(stderr, "Added download %s -> %s\n", url, filename); 8613498266Sopenharmony_ci} 8713498266Sopenharmony_ci 8813498266Sopenharmony_cistatic void check_multi_info(void) 8913498266Sopenharmony_ci{ 9013498266Sopenharmony_ci char *done_url; 9113498266Sopenharmony_ci CURLMsg *message; 9213498266Sopenharmony_ci int pending; 9313498266Sopenharmony_ci CURL *easy_handle; 9413498266Sopenharmony_ci FILE *file; 9513498266Sopenharmony_ci 9613498266Sopenharmony_ci while((message = curl_multi_info_read(curl_handle, &pending))) { 9713498266Sopenharmony_ci switch(message->msg) { 9813498266Sopenharmony_ci case CURLMSG_DONE: 9913498266Sopenharmony_ci /* Do not use message data after calling curl_multi_remove_handle() and 10013498266Sopenharmony_ci curl_easy_cleanup(). As per curl_multi_info_read() docs: 10113498266Sopenharmony_ci "WARNING: The data the returned pointer points to will not survive 10213498266Sopenharmony_ci calling curl_multi_cleanup, curl_multi_remove_handle or 10313498266Sopenharmony_ci curl_easy_cleanup." */ 10413498266Sopenharmony_ci easy_handle = message->easy_handle; 10513498266Sopenharmony_ci 10613498266Sopenharmony_ci curl_easy_getinfo(easy_handle, CURLINFO_EFFECTIVE_URL, &done_url); 10713498266Sopenharmony_ci curl_easy_getinfo(easy_handle, CURLINFO_PRIVATE, &file); 10813498266Sopenharmony_ci printf("%s DONE\n", done_url); 10913498266Sopenharmony_ci 11013498266Sopenharmony_ci curl_multi_remove_handle(curl_handle, easy_handle); 11113498266Sopenharmony_ci curl_easy_cleanup(easy_handle); 11213498266Sopenharmony_ci if(file) { 11313498266Sopenharmony_ci fclose(file); 11413498266Sopenharmony_ci } 11513498266Sopenharmony_ci break; 11613498266Sopenharmony_ci 11713498266Sopenharmony_ci default: 11813498266Sopenharmony_ci fprintf(stderr, "CURLMSG default\n"); 11913498266Sopenharmony_ci break; 12013498266Sopenharmony_ci } 12113498266Sopenharmony_ci } 12213498266Sopenharmony_ci} 12313498266Sopenharmony_ci 12413498266Sopenharmony_cistatic void curl_perform(int fd, short event, void *arg) 12513498266Sopenharmony_ci{ 12613498266Sopenharmony_ci int running_handles; 12713498266Sopenharmony_ci int flags = 0; 12813498266Sopenharmony_ci curl_context_t *context; 12913498266Sopenharmony_ci 13013498266Sopenharmony_ci if(event & EV_READ) 13113498266Sopenharmony_ci flags |= CURL_CSELECT_IN; 13213498266Sopenharmony_ci if(event & EV_WRITE) 13313498266Sopenharmony_ci flags |= CURL_CSELECT_OUT; 13413498266Sopenharmony_ci 13513498266Sopenharmony_ci context = (curl_context_t *) arg; 13613498266Sopenharmony_ci 13713498266Sopenharmony_ci curl_multi_socket_action(curl_handle, context->sockfd, flags, 13813498266Sopenharmony_ci &running_handles); 13913498266Sopenharmony_ci 14013498266Sopenharmony_ci check_multi_info(); 14113498266Sopenharmony_ci} 14213498266Sopenharmony_ci 14313498266Sopenharmony_cistatic void on_timeout(evutil_socket_t fd, short events, void *arg) 14413498266Sopenharmony_ci{ 14513498266Sopenharmony_ci int running_handles; 14613498266Sopenharmony_ci curl_multi_socket_action(curl_handle, CURL_SOCKET_TIMEOUT, 0, 14713498266Sopenharmony_ci &running_handles); 14813498266Sopenharmony_ci check_multi_info(); 14913498266Sopenharmony_ci} 15013498266Sopenharmony_ci 15113498266Sopenharmony_cistatic int start_timeout(CURLM *multi, long timeout_ms, void *userp) 15213498266Sopenharmony_ci{ 15313498266Sopenharmony_ci if(timeout_ms < 0) { 15413498266Sopenharmony_ci evtimer_del(timeout); 15513498266Sopenharmony_ci } 15613498266Sopenharmony_ci else { 15713498266Sopenharmony_ci if(timeout_ms == 0) 15813498266Sopenharmony_ci timeout_ms = 1; /* 0 means directly call socket_action, but we will do it 15913498266Sopenharmony_ci in a bit */ 16013498266Sopenharmony_ci struct timeval tv; 16113498266Sopenharmony_ci tv.tv_sec = timeout_ms / 1000; 16213498266Sopenharmony_ci tv.tv_usec = (timeout_ms % 1000) * 1000; 16313498266Sopenharmony_ci evtimer_del(timeout); 16413498266Sopenharmony_ci evtimer_add(timeout, &tv); 16513498266Sopenharmony_ci } 16613498266Sopenharmony_ci return 0; 16713498266Sopenharmony_ci} 16813498266Sopenharmony_ci 16913498266Sopenharmony_cistatic int handle_socket(CURL *easy, curl_socket_t s, int action, void *userp, 17013498266Sopenharmony_ci void *socketp) 17113498266Sopenharmony_ci{ 17213498266Sopenharmony_ci curl_context_t *curl_context; 17313498266Sopenharmony_ci int events = 0; 17413498266Sopenharmony_ci 17513498266Sopenharmony_ci switch(action) { 17613498266Sopenharmony_ci case CURL_POLL_IN: 17713498266Sopenharmony_ci case CURL_POLL_OUT: 17813498266Sopenharmony_ci case CURL_POLL_INOUT: 17913498266Sopenharmony_ci curl_context = socketp ? 18013498266Sopenharmony_ci (curl_context_t *) socketp : create_curl_context(s); 18113498266Sopenharmony_ci 18213498266Sopenharmony_ci curl_multi_assign(curl_handle, s, (void *) curl_context); 18313498266Sopenharmony_ci 18413498266Sopenharmony_ci if(action != CURL_POLL_IN) 18513498266Sopenharmony_ci events |= EV_WRITE; 18613498266Sopenharmony_ci if(action != CURL_POLL_OUT) 18713498266Sopenharmony_ci events |= EV_READ; 18813498266Sopenharmony_ci 18913498266Sopenharmony_ci events |= EV_PERSIST; 19013498266Sopenharmony_ci 19113498266Sopenharmony_ci event_del(curl_context->event); 19213498266Sopenharmony_ci event_assign(curl_context->event, base, curl_context->sockfd, events, 19313498266Sopenharmony_ci curl_perform, curl_context); 19413498266Sopenharmony_ci event_add(curl_context->event, NULL); 19513498266Sopenharmony_ci 19613498266Sopenharmony_ci break; 19713498266Sopenharmony_ci case CURL_POLL_REMOVE: 19813498266Sopenharmony_ci if(socketp) { 19913498266Sopenharmony_ci event_del(((curl_context_t*) socketp)->event); 20013498266Sopenharmony_ci destroy_curl_context((curl_context_t*) socketp); 20113498266Sopenharmony_ci curl_multi_assign(curl_handle, s, NULL); 20213498266Sopenharmony_ci } 20313498266Sopenharmony_ci break; 20413498266Sopenharmony_ci default: 20513498266Sopenharmony_ci abort(); 20613498266Sopenharmony_ci } 20713498266Sopenharmony_ci 20813498266Sopenharmony_ci return 0; 20913498266Sopenharmony_ci} 21013498266Sopenharmony_ci 21113498266Sopenharmony_ciint main(int argc, char **argv) 21213498266Sopenharmony_ci{ 21313498266Sopenharmony_ci if(argc <= 1) 21413498266Sopenharmony_ci return 0; 21513498266Sopenharmony_ci 21613498266Sopenharmony_ci if(curl_global_init(CURL_GLOBAL_ALL)) { 21713498266Sopenharmony_ci fprintf(stderr, "Could not init curl\n"); 21813498266Sopenharmony_ci return 1; 21913498266Sopenharmony_ci } 22013498266Sopenharmony_ci 22113498266Sopenharmony_ci base = event_base_new(); 22213498266Sopenharmony_ci timeout = evtimer_new(base, on_timeout, NULL); 22313498266Sopenharmony_ci 22413498266Sopenharmony_ci curl_handle = curl_multi_init(); 22513498266Sopenharmony_ci curl_multi_setopt(curl_handle, CURLMOPT_SOCKETFUNCTION, handle_socket); 22613498266Sopenharmony_ci curl_multi_setopt(curl_handle, CURLMOPT_TIMERFUNCTION, start_timeout); 22713498266Sopenharmony_ci 22813498266Sopenharmony_ci while(argc-- > 1) { 22913498266Sopenharmony_ci add_download(argv[argc], argc); 23013498266Sopenharmony_ci } 23113498266Sopenharmony_ci 23213498266Sopenharmony_ci event_base_dispatch(base); 23313498266Sopenharmony_ci 23413498266Sopenharmony_ci curl_multi_cleanup(curl_handle); 23513498266Sopenharmony_ci event_free(timeout); 23613498266Sopenharmony_ci event_base_free(base); 23713498266Sopenharmony_ci 23813498266Sopenharmony_ci libevent_global_shutdown(); 23913498266Sopenharmony_ci curl_global_cleanup(); 24013498266Sopenharmony_ci 24113498266Sopenharmony_ci return 0; 24213498266Sopenharmony_ci} 243