Home | History | Annotate | Download | only in examples
      1 /***************************************************************************
      2  *                                  _   _ ____  _
      3  *  Project                     ___| | | |  _ \| |
      4  *                             / __| | | | |_) | |
      5  *                            | (__| |_| |  _ <| |___
      6  *                             \___|\___/|_| \_\_____|
      7  *
      8  * Copyright (C) 1998 - 2011, Daniel Stenberg, <daniel (at) haxx.se>, et al.
      9  *
     10  * This software is licensed as described in the file COPYING, which
     11  * you should have received as part of this distribution. The terms
     12  * are also available at http://curl.haxx.se/docs/copyright.html.
     13  *
     14  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
     15  * copies of the Software, and permit persons to whom the Software is
     16  * furnished to do so, under the terms of the COPYING file.
     17  *
     18  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
     19  * KIND, either express or implied.
     20  *
     21  ***************************************************************************/
     22 /* A multi-threaded example that uses pthreads and fetches 4 remote files at
     23  * once over HTTPS. The lock callbacks and stuff assume OpenSSL or GnuTLS
     24  * (libgcrypt) so far.
     25  *
     26  * OpenSSL docs for this:
     27  *   http://www.openssl.org/docs/crypto/threads.html
     28  * gcrypt docs for this:
     29  *   http://gnupg.org/documentation/manuals/gcrypt/Multi_002dThreading.html
     30  */
     31 
     32 #define USE_OPENSSL /* or USE_GNUTLS accordingly */
     33 
     34 #include <stdio.h>
     35 #include <pthread.h>
     36 #include <curl/curl.h>
     37 
     38 #define NUMT 4
     39 
     40 /* we have this global to let the callback get easy access to it */
     41 static pthread_mutex_t *lockarray;
     42 
     43 #ifdef USE_OPENSSL
     44 #include <openssl/crypto.h>
     45 static void lock_callback(int mode, int type, char *file, int line)
     46 {
     47   (void)file;
     48   (void)line;
     49   if (mode & CRYPTO_LOCK) {
     50     pthread_mutex_lock(&(lockarray[type]));
     51   }
     52   else {
     53     pthread_mutex_unlock(&(lockarray[type]));
     54   }
     55 }
     56 
     57 static unsigned long thread_id(void)
     58 {
     59   unsigned long ret;
     60 
     61   ret=(unsigned long)pthread_self();
     62   return(ret);
     63 }
     64 
     65 static void init_locks(void)
     66 {
     67   int i;
     68 
     69   lockarray=(pthread_mutex_t *)OPENSSL_malloc(CRYPTO_num_locks() *
     70                                             sizeof(pthread_mutex_t));
     71   for (i=0; i<CRYPTO_num_locks(); i++) {
     72     pthread_mutex_init(&(lockarray[i]),NULL);
     73   }
     74 
     75   CRYPTO_set_id_callback((unsigned long (*)())thread_id);
     76   CRYPTO_set_locking_callback((void (*)())lock_callback);
     77 }
     78 
     79 static void kill_locks(void)
     80 {
     81   int i;
     82 
     83   CRYPTO_set_locking_callback(NULL);
     84   for (i=0; i<CRYPTO_num_locks(); i++)
     85     pthread_mutex_destroy(&(lockarray[i]));
     86 
     87   OPENSSL_free(lockarray);
     88 }
     89 #endif
     90 
     91 #ifdef USE_GNUTLS
     92 #include <gcrypt.h>
     93 #include <errno.h>
     94 
     95 GCRY_THREAD_OPTION_PTHREAD_IMPL;
     96 
     97 void init_locks(void)
     98 {
     99   gcry_control(GCRYCTL_SET_THREAD_CBS);
    100 }
    101 
    102 #define kill_locks()
    103 #endif
    104 
    105 /* List of URLs to fetch.*/
    106 const char * const urls[]= {
    107   "https://www.example.com/",
    108   "https://www2.example.com/",
    109   "https://www3.example.com/",
    110   "https://www4.example.com/",
    111 };
    112 
    113 static void *pull_one_url(void *url)
    114 {
    115   CURL *curl;
    116 
    117   curl = curl_easy_init();
    118   curl_easy_setopt(curl, CURLOPT_URL, url);
    119   /* this example doesn't verify the server's certificate, which means we
    120      might be downloading stuff from an impostor */
    121   curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
    122   curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
    123   curl_easy_perform(curl); /* ignores error */
    124   curl_easy_cleanup(curl);
    125 
    126   return NULL;
    127 }
    128 
    129 int main(int argc, char **argv)
    130 {
    131   pthread_t tid[NUMT];
    132   int i;
    133   int error;
    134   (void)argc; /* we don't use any arguments in this example */
    135   (void)argv;
    136 
    137   /* Must initialize libcurl before any threads are started */
    138   curl_global_init(CURL_GLOBAL_ALL);
    139 
    140   init_locks();
    141 
    142   for(i=0; i< NUMT; i++) {
    143     error = pthread_create(&tid[i],
    144                            NULL, /* default attributes please */
    145                            pull_one_url,
    146                            (void *)urls[i]);
    147     if(0 != error)
    148       fprintf(stderr, "Couldn't run thread number %d, errno %d\n", i, error);
    149     else
    150       fprintf(stderr, "Thread %d, gets %s\n", i, urls[i]);
    151   }
    152 
    153   /* now wait for all threads to terminate */
    154   for(i=0; i< NUMT; i++) {
    155     error = pthread_join(tid[i], NULL);
    156     fprintf(stderr, "Thread %d terminated\n", i);
    157   }
    158 
    159   kill_locks();
    160 
    161   return 0;
    162 }
    163