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