Blame src/testcurl/test_large_put.c

Packit 875988
/*
Packit 875988
     This file is part of libmicrohttpd
Packit 875988
     Copyright (C) 2007, 2008 Christian Grothoff
Packit 875988
Packit 875988
     libmicrohttpd is free software; you can redistribute it and/or modify
Packit 875988
     it under the terms of the GNU General Public License as published
Packit 875988
     by the Free Software Foundation; either version 2, or (at your
Packit 875988
     option) any later version.
Packit 875988
Packit 875988
     libmicrohttpd is distributed in the hope that it will be useful, but
Packit 875988
     WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 875988
     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 875988
     General Public License for more details.
Packit 875988
Packit 875988
     You should have received a copy of the GNU General Public License
Packit 875988
     along with libmicrohttpd; see the file COPYING.  If not, write to the
Packit 875988
     Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Packit 875988
     Boston, MA 02110-1301, USA.
Packit 875988
*/
Packit 875988
Packit 875988
/**
Packit 875988
 * @file test_large_put.c
Packit 875988
 * @brief  Testcase for libmicrohttpd PUT operations
Packit 875988
 * @author Christian Grothoff
Packit 875988
 */
Packit 875988
Packit 875988
#include "MHD_config.h"
Packit 875988
#include "platform.h"
Packit 875988
#include <curl/curl.h>
Packit 875988
#include <microhttpd.h>
Packit 875988
#include <stdlib.h>
Packit 875988
#include <string.h>
Packit 875988
#include <time.h>
Packit 875988
Packit 875988
#ifndef WINDOWS
Packit 875988
#include <unistd.h>
Packit 875988
#endif
Packit 875988
Packit 875988
#include "test_helpers.h"
Packit 875988
Packit 875988
#if defined(CPU_COUNT) && (CPU_COUNT+0) < 2
Packit 875988
#undef CPU_COUNT
Packit 875988
#endif
Packit 875988
#if !defined(CPU_COUNT)
Packit 875988
#define CPU_COUNT 2
Packit 875988
#endif
Packit 875988
Packit 875988
static int oneone;
Packit 875988
static int incr_read; /* Use incremental read */
Packit 875988
static int verbose; /* Be verbose */
Packit 875988
Packit 875988
#define PUT_SIZE (256 * 1024)
Packit 875988
Packit 875988
static char *put_buffer;
Packit 875988
Packit 875988
struct CBC
Packit 875988
{
Packit 875988
  char *buf;
Packit 875988
  size_t pos;
Packit 875988
  size_t size;
Packit 875988
};
Packit 875988
Packit 875988
char*
Packit 875988
alloc_init(size_t buf_size)
Packit 875988
{
Packit 875988
  static const char template[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ[]^_`abcdefghijklmnopqrstuvwxyz";
Packit 875988
  static const size_t templ_size = sizeof(template) / sizeof(char) - 1;
Packit 875988
  char *buf;
Packit 875988
  char *fill_ptr;
Packit 875988
  size_t to_fill;
Packit 875988
Packit 875988
  buf = malloc(buf_size);
Packit 875988
  if (NULL == buf)
Packit 875988
    return NULL;
Packit 875988
Packit 875988
  fill_ptr = buf;
Packit 875988
  to_fill = buf_size;
Packit 875988
  while (to_fill > 0)
Packit 875988
    {
Packit 875988
      const size_t to_copy = to_fill > templ_size ? templ_size : to_fill;
Packit 875988
      memcpy (fill_ptr, template, to_copy);
Packit 875988
      fill_ptr += to_copy;
Packit 875988
      to_fill -= to_copy;
Packit 875988
    }
Packit 875988
  return buf;
Packit 875988
}
Packit 875988
Packit 875988
static size_t
Packit 875988
putBuffer (void *stream, size_t size, size_t nmemb, void *ptr)
Packit 875988
{
Packit 875988
  size_t *pos = (size_t *)ptr;
Packit 875988
  size_t wrt;
Packit 875988
Packit 875988
  wrt = size * nmemb;
Packit 875988
  /* Check for overflow. */
Packit 875988
  if (wrt / size != nmemb)
Packit 875988
    return 0;
Packit 875988
  if (wrt > PUT_SIZE - (*pos))
Packit 875988
    wrt = PUT_SIZE - (*pos);
Packit 875988
  memcpy (stream, &put_buffer[*pos], wrt);
Packit 875988
  (*pos) += wrt;
Packit 875988
  return wrt;
Packit 875988
}
Packit 875988
Packit 875988
static size_t
Packit 875988
copyBuffer (void *ptr, size_t size, size_t nmemb, void *ctx)
Packit 875988
{
Packit 875988
  struct CBC *cbc = ctx;
Packit 875988
Packit 875988
  if (cbc->pos + size * nmemb > cbc->size)
Packit 875988
    return 0;                   /* overflow */
Packit 875988
  memcpy (&cbc->buf[cbc->pos], ptr, size * nmemb);
Packit 875988
  cbc->pos += size * nmemb;
Packit 875988
  return size * nmemb;
Packit 875988
}
Packit 875988
Packit 875988
static int
Packit 875988
ahc_echo (void *cls,
Packit 875988
          struct MHD_Connection *connection,
Packit 875988
          const char *url,
Packit 875988
          const char *method,
Packit 875988
          const char *version,
Packit 875988
          const char *upload_data, size_t *upload_data_size,
Packit 875988
          void **pparam)
Packit 875988
{
Packit 875988
  int *done = cls;
Packit 875988
  struct MHD_Response *response;
Packit 875988
  int ret;
Packit 875988
  static size_t processed;
Packit 875988
  (void)version;        /* Unused. Silent compiler warning. */
Packit 875988
Packit 875988
  if (0 != strcmp ("PUT", method))
Packit 875988
    return MHD_NO;              /* unexpected method */
Packit 875988
  if ((*done) == 0)
Packit 875988
    {
Packit 875988
      size_t *pproc;
Packit 875988
      if (NULL == *pparam)
Packit 875988
        {
Packit 875988
          processed = 0;
Packit 875988
          *pparam = &processed; /* Safe as long as only one parallel request served. */
Packit 875988
        }
Packit 875988
      pproc = (size_t*) *pparam;
Packit 875988
Packit 875988
      if (0 == *upload_data_size)
Packit 875988
        return MHD_YES; /* No data to process. */
Packit 875988
Packit 875988
      if (*pproc + *upload_data_size > PUT_SIZE)
Packit 875988
        {
Packit 875988
          fprintf (stderr, "Incoming data larger than expected.\n");
Packit 875988
          return MHD_NO;
Packit 875988
        }
Packit 875988
      if ( (!incr_read) && (*upload_data_size != PUT_SIZE) )
Packit 875988
        return MHD_YES; /* Wait until whole request is received. */
Packit 875988
Packit 875988
      if (0 != memcmp(upload_data, put_buffer + (*pproc), *upload_data_size))
Packit 875988
        {
Packit 875988
          fprintf (stderr, "Incoming data does not match sent data.\n");
Packit 875988
          return MHD_NO;
Packit 875988
        }
Packit 875988
      *pproc += *upload_data_size;
Packit 875988
      *upload_data_size = 0; /* Current block of data is fully processed. */
Packit 875988
Packit 875988
      if (PUT_SIZE == *pproc)
Packit 875988
        *done = 1; /* Whole request is processed. */
Packit 875988
      return MHD_YES;
Packit 875988
    }
Packit 875988
  response = MHD_create_response_from_buffer (strlen (url),
Packit 875988
					      (void *) url,
Packit 875988
					      MHD_RESPMEM_MUST_COPY);
Packit 875988
  ret = MHD_queue_response (connection, MHD_HTTP_OK, response);
Packit 875988
  MHD_destroy_response (response);
Packit 875988
  return ret;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
static int
Packit 875988
testPutInternalThread (unsigned int add_flag)
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  struct CBC cbc;
Packit 875988
  size_t pos = 0;
Packit 875988
  int done_flag = 0;
Packit 875988
  CURLcode errornum;
Packit 875988
  char buf[2048];
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1270;
Packit 875988
      if (oneone)
Packit 875988
        port += 10;
Packit 875988
      if (incr_read)
Packit 875988
        port += 20;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  d = MHD_start_daemon (MHD_USE_INTERNAL_POLLING_THREAD | MHD_USE_ERROR_LOG | add_flag,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
			MHD_OPTION_CONNECTION_MEMORY_LIMIT, (size_t)(incr_read ? 1024 : (PUT_SIZE * 4)),
Packit 875988
			MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 1;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &pos;;
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE, (long) PUT_SIZE);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
  if (CURLE_OK != (errornum = curl_easy_perform (c)))
Packit 875988
    {
Packit 875988
      fprintf (stderr,
Packit 875988
               "curl_easy_perform failed: `%s'\n",
Packit 875988
               curl_easy_strerror (errornum));
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 2;
Packit 875988
    }
Packit 875988
  curl_easy_cleanup (c);
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  if (cbc.pos != strlen ("/hello_world"))
Packit 875988
    return 4;
Packit 875988
  if (0 != strncmp ("/hello_world", cbc.buf, strlen ("/hello_world")))
Packit 875988
    return 8;
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
static int
Packit 875988
testPutThreadPerConn (unsigned int add_flag)
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  struct CBC cbc;
Packit 875988
  size_t pos = 0;
Packit 875988
  int done_flag = 0;
Packit 875988
  CURLcode errornum;
Packit 875988
  char buf[2048];
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1271;
Packit 875988
      if (oneone)
Packit 875988
        port += 10;
Packit 875988
      if (incr_read)
Packit 875988
        port += 20;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  d = MHD_start_daemon (MHD_USE_THREAD_PER_CONNECTION | MHD_USE_INTERNAL_POLLING_THREAD |
Packit 875988
                          MHD_USE_ERROR_LOG | add_flag,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
                        MHD_OPTION_CONNECTION_MEMORY_LIMIT, (size_t)(incr_read ? 1024 : (PUT_SIZE * 4)),
Packit 875988
			MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 16;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &pos;;
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE, (long) PUT_SIZE);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
  if (CURLE_OK != (errornum = curl_easy_perform (c)))
Packit 875988
    {
Packit 875988
      fprintf (stderr,
Packit 875988
               "curl_easy_perform failed: `%s'\n",
Packit 875988
               curl_easy_strerror (errornum));
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 32;
Packit 875988
    }
Packit 875988
  curl_easy_cleanup (c);
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  if (cbc.pos != strlen ("/hello_world"))
Packit 875988
    {
Packit 875988
      fprintf (stderr, "Got invalid response `%.*s'\n", (int)cbc.pos, cbc.buf);
Packit 875988
      return 64;
Packit 875988
    }
Packit 875988
  if (0 != strncmp ("/hello_world", cbc.buf, strlen ("/hello_world")))
Packit 875988
    return 128;
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
static int
Packit 875988
testPutThreadPool (unsigned int add_flag)
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  struct CBC cbc;
Packit 875988
  size_t pos = 0;
Packit 875988
  int done_flag = 0;
Packit 875988
  CURLcode errornum;
Packit 875988
  char buf[2048];
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1272;
Packit 875988
      if (oneone)
Packit 875988
        port += 10;
Packit 875988
      if (incr_read)
Packit 875988
        port += 20;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  d = MHD_start_daemon (MHD_USE_INTERNAL_POLLING_THREAD | MHD_USE_ERROR_LOG | add_flag,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
                        MHD_OPTION_THREAD_POOL_SIZE, CPU_COUNT,
Packit 875988
                        MHD_OPTION_CONNECTION_MEMORY_LIMIT, (size_t)(incr_read ? 1024 : (PUT_SIZE * 4)),
Packit 875988
			MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 16;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &pos;;
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE, (long) PUT_SIZE);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
  if (CURLE_OK != (errornum = curl_easy_perform (c)))
Packit 875988
    {
Packit 875988
      fprintf (stderr,
Packit 875988
               "curl_easy_perform failed: `%s'\n",
Packit 875988
               curl_easy_strerror (errornum));
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 32;
Packit 875988
    }
Packit 875988
  curl_easy_cleanup (c);
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  if (cbc.pos != strlen ("/hello_world"))
Packit 875988
    {
Packit 875988
      fprintf (stderr, "Got invalid response `%.*s'\n", (int)cbc.pos, cbc.buf);
Packit 875988
      return 64;
Packit 875988
    }
Packit 875988
  if (0 != strncmp ("/hello_world", cbc.buf, strlen ("/hello_world")))
Packit 875988
    return 128;
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
static int
Packit 875988
testPutExternal (void)
Packit 875988
{
Packit 875988
  struct MHD_Daemon *d;
Packit 875988
  CURL *c;
Packit 875988
  struct CBC cbc;
Packit 875988
  CURLM *multi;
Packit 875988
  CURLMcode mret;
Packit 875988
  fd_set rs;
Packit 875988
  fd_set ws;
Packit 875988
  fd_set es;
Packit 875988
  MHD_socket maxsock;
Packit 875988
#ifdef MHD_WINSOCK_SOCKETS
Packit 875988
  int maxposixs; /* Max socket number unused on W32 */
Packit 875988
#else  /* MHD_POSIX_SOCKETS */
Packit 875988
#define maxposixs maxsock
Packit 875988
#endif /* MHD_POSIX_SOCKETS */
Packit 875988
  int running;
Packit 875988
  struct CURLMsg *msg;
Packit 875988
  time_t start;
Packit 875988
  struct timeval tv;
Packit 875988
  size_t pos = 0;
Packit 875988
  int done_flag = 0;
Packit 875988
  char buf[2048];
Packit 875988
  int port;
Packit 875988
Packit 875988
  if (MHD_NO != MHD_is_feature_supported (MHD_FEATURE_AUTODETECT_BIND_PORT))
Packit 875988
    port = 0;
Packit 875988
  else
Packit 875988
    {
Packit 875988
      port = 1273;
Packit 875988
      if (oneone)
Packit 875988
        port += 10;
Packit 875988
      if (incr_read)
Packit 875988
        port += 20;
Packit 875988
    }
Packit 875988
Packit 875988
  cbc.buf = buf;
Packit 875988
  cbc.size = 2048;
Packit 875988
  cbc.pos = 0;
Packit 875988
  multi = NULL;
Packit 875988
  d = MHD_start_daemon (MHD_USE_ERROR_LOG,
Packit 875988
                        port,
Packit 875988
                        NULL, NULL, &ahc_echo, &done_flag,
Packit 875988
                        MHD_OPTION_CONNECTION_MEMORY_LIMIT, (size_t)(incr_read ? 1024 : (PUT_SIZE * 4)),
Packit 875988
                        MHD_OPTION_END);
Packit 875988
  if (d == NULL)
Packit 875988
    return 256;
Packit 875988
  if (0 == port)
Packit 875988
    {
Packit 875988
      const union MHD_DaemonInfo *dinfo;
Packit 875988
      dinfo = MHD_get_daemon_info (d, MHD_DAEMON_INFO_BIND_PORT);
Packit 875988
      if (NULL == dinfo || 0 == dinfo->port)
Packit 875988
        { MHD_stop_daemon (d); return 32; }
Packit 875988
      port = (int)dinfo->port;
Packit 875988
    }
Packit 875988
  c = curl_easy_init ();
Packit 875988
  curl_easy_setopt (c, CURLOPT_URL, "http://127.0.0.1/hello_world");
Packit 875988
  curl_easy_setopt (c, CURLOPT_PORT, (long)port);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEFUNCTION, &copyBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_WRITEDATA, &cbc);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READFUNCTION, &putBuffer);
Packit 875988
  curl_easy_setopt (c, CURLOPT_READDATA, &pos;;
Packit 875988
  curl_easy_setopt (c, CURLOPT_UPLOAD, 1L);
Packit 875988
  curl_easy_setopt (c, CURLOPT_INFILESIZE, (long) PUT_SIZE);
Packit 875988
  curl_easy_setopt (c, CURLOPT_FAILONERROR, 1);
Packit 875988
  curl_easy_setopt (c, CURLOPT_TIMEOUT, 150L);
Packit 875988
  if (oneone)
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
Packit 875988
  else
Packit 875988
    curl_easy_setopt (c, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_0);
Packit 875988
  curl_easy_setopt (c, CURLOPT_CONNECTTIMEOUT, 150L);
Packit 875988
  /* NOTE: use of CONNECTTIMEOUT without also
Packit 875988
   *   setting NOSIGNAL results in really weird
Packit 875988
   *   crashes on my system! */
Packit 875988
  curl_easy_setopt (c, CURLOPT_NOSIGNAL, 1);
Packit 875988
Packit 875988
Packit 875988
  multi = curl_multi_init ();
Packit 875988
  if (multi == NULL)
Packit 875988
    {
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 512;
Packit 875988
    }
Packit 875988
  mret = curl_multi_add_handle (multi, c);
Packit 875988
  if (mret != CURLM_OK)
Packit 875988
    {
Packit 875988
      curl_multi_cleanup (multi);
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      MHD_stop_daemon (d);
Packit 875988
      return 1024;
Packit 875988
    }
Packit 875988
  start = time (NULL);
Packit 875988
  while ((time (NULL) - start < 5) && (multi != NULL))
Packit 875988
    {
Packit 875988
      maxsock = MHD_INVALID_SOCKET;
Packit 875988
      maxposixs = -1;
Packit 875988
      FD_ZERO (&rs);
Packit 875988
      FD_ZERO (&ws);
Packit 875988
      FD_ZERO (&es);
Packit 875988
      curl_multi_perform (multi, &running);
Packit 875988
      mret = curl_multi_fdset (multi, &rs, &ws, &es, &maxposixs);
Packit 875988
      if (mret != CURLM_OK)
Packit 875988
        {
Packit 875988
          curl_multi_remove_handle (multi, c);
Packit 875988
          curl_multi_cleanup (multi);
Packit 875988
          curl_easy_cleanup (c);
Packit 875988
          MHD_stop_daemon (d);
Packit 875988
          return 2048;
Packit 875988
        }
Packit 875988
      if (MHD_YES != MHD_get_fdset (d, &rs, &ws, &es, &maxsock))
Packit 875988
        {
Packit 875988
          curl_multi_remove_handle (multi, c);
Packit 875988
          curl_multi_cleanup (multi);
Packit 875988
          curl_easy_cleanup (c);
Packit 875988
          MHD_stop_daemon (d);
Packit 875988
          return 4096;
Packit 875988
        }
Packit 875988
      tv.tv_sec = 0;
Packit 875988
      tv.tv_usec = 1000;
Packit 875988
      if (-1 == select (maxposixs + 1, &rs, &ws, &es, &tv))
Packit 875988
        {
Packit 875988
#ifdef MHD_POSIX_SOCKETS
Packit 875988
          if (EINTR != errno)
Packit 875988
            abort ();
Packit 875988
#else
Packit 875988
          if (WSAEINVAL != WSAGetLastError() || 0 != rs.fd_count || 0 != ws.fd_count || 0 != es.fd_count)
Packit 875988
            abort ();
Packit 875988
          Sleep (1000);
Packit 875988
#endif
Packit 875988
        }
Packit 875988
      curl_multi_perform (multi, &running);
Packit 875988
      if (running == 0)
Packit 875988
        {
Packit 875988
          msg = curl_multi_info_read (multi, &running);
Packit 875988
          if (msg == NULL)
Packit 875988
            break;
Packit 875988
          if (msg->msg == CURLMSG_DONE)
Packit 875988
            {
Packit 875988
              if (msg->data.result != CURLE_OK)
Packit 875988
                printf ("%s failed at %s:%d: `%s'\n",
Packit 875988
                        "curl_multi_perform",
Packit 875988
                        __FILE__,
Packit 875988
                        __LINE__, curl_easy_strerror (msg->data.result));
Packit 875988
              curl_multi_remove_handle (multi, c);
Packit 875988
              curl_multi_cleanup (multi);
Packit 875988
              curl_easy_cleanup (c);
Packit 875988
              c = NULL;
Packit 875988
              multi = NULL;
Packit 875988
            }
Packit 875988
        }
Packit 875988
      MHD_run (d);
Packit 875988
    }
Packit 875988
  if (multi != NULL)
Packit 875988
    {
Packit 875988
      curl_multi_remove_handle (multi, c);
Packit 875988
      curl_easy_cleanup (c);
Packit 875988
      curl_multi_cleanup (multi);
Packit 875988
    }
Packit 875988
  MHD_stop_daemon (d);
Packit 875988
  if (cbc.pos != strlen ("/hello_world"))
Packit 875988
    {
Packit 875988
      fprintf (stderr, "Got invalid response `%.*s'\n", (int)cbc.pos, cbc.buf);
Packit 875988
      return 8192;
Packit 875988
    }
Packit 875988
  if (0 != strncmp ("/hello_world", cbc.buf, strlen ("/hello_world")))
Packit 875988
    return 16384;
Packit 875988
  return 0;
Packit 875988
}
Packit 875988
Packit 875988
Packit 875988
Packit 875988
int
Packit 875988
main (int argc, char *const *argv)
Packit 875988
{
Packit 875988
  unsigned int errorCount = 0;
Packit 875988
  unsigned int lastErr;
Packit 875988
Packit 875988
  oneone = has_in_name(argv[0], "11");
Packit 875988
  incr_read = has_in_name(argv[0], "_inc");
Packit 875988
  verbose = has_param(argc, argv, "-v");
Packit 875988
  if (0 != curl_global_init (CURL_GLOBAL_WIN32))
Packit 875988
    return 99;
Packit 875988
  put_buffer = alloc_init (PUT_SIZE);
Packit 875988
  if (NULL == put_buffer)
Packit 875988
    return 99;
Packit 875988
  lastErr = testPutInternalThread (0);
Packit 875988
  if (verbose && 0 != lastErr)
Packit 875988
    fprintf (stderr, "Error during testing with internal thread with select().\n");
Packit 875988
  errorCount += lastErr;
Packit 875988
  lastErr = testPutThreadPerConn (0);
Packit 875988
  if (verbose && 0 != lastErr)
Packit 875988
    fprintf (stderr, "Error during testing with internal thread per connection with select().\n");
Packit 875988
  errorCount += lastErr;
Packit 875988
  lastErr = testPutThreadPool (0);
Packit 875988
  if (verbose && 0 != lastErr)
Packit 875988
    fprintf (stderr, "Error during testing with thread pool per connection with select().\n");
Packit 875988
  errorCount += lastErr;
Packit 875988
  lastErr = testPutExternal ();
Packit 875988
  if (verbose && 0 != lastErr)
Packit 875988
    fprintf (stderr, "Error during testing with external select().\n");
Packit 875988
  errorCount += lastErr;
Packit 875988
  if (MHD_is_feature_supported(MHD_FEATURE_POLL))
Packit 875988
    {
Packit 875988
      lastErr = testPutInternalThread (MHD_USE_POLL);
Packit 875988
      if (verbose && 0 != lastErr)
Packit 875988
        fprintf (stderr, "Error during testing with internal thread with poll().\n");
Packit 875988
      errorCount += lastErr;
Packit 875988
      lastErr = testPutThreadPerConn (MHD_USE_POLL);
Packit 875988
      if (verbose && 0 != lastErr)
Packit 875988
        fprintf (stderr, "Error during testing with internal thread per connection with poll().\n");
Packit 875988
      errorCount += lastErr;
Packit 875988
      lastErr = testPutThreadPool (MHD_USE_POLL);
Packit 875988
      if (verbose && 0 != lastErr)
Packit 875988
        fprintf (stderr, "Error during testing with thread pool per connection with poll().\n");
Packit 875988
      errorCount += lastErr;
Packit 875988
    }
Packit 875988
  if (MHD_is_feature_supported(MHD_FEATURE_EPOLL))
Packit 875988
    {
Packit 875988
      lastErr = testPutInternalThread (MHD_USE_EPOLL);
Packit 875988
      if (verbose && 0 != lastErr)
Packit 875988
        fprintf (stderr, "Error during testing with internal thread with epoll.\n");
Packit 875988
      errorCount += lastErr;
Packit 875988
      lastErr = testPutThreadPool (MHD_USE_EPOLL);
Packit 875988
      if (verbose && 0 != lastErr)
Packit 875988
        fprintf (stderr, "Error during testing with thread pool per connection with epoll.\n");
Packit 875988
      errorCount += lastErr;
Packit 875988
    }
Packit 875988
  free (put_buffer);
Packit 875988
  if (errorCount != 0)
Packit 875988
    fprintf (stderr, "Error (code: %u)\n", errorCount);
Packit 875988
  else if (verbose)
Packit 875988
    printf ("All checks passed successfully.\n");
Packit 875988
  curl_global_cleanup ();
Packit 875988
  return (errorCount == 0) ? 0 : 1;
Packit 875988
}