Blame tests/gpg/t-cancel.c

Packit d7e8d0
/* t-thread-cancel.c - Regression test.
Packit d7e8d0
   Copyright (C) 2000 Werner Koch (dd9jn)
Packit d7e8d0
   Copyright (C) 2001, 2003, 2004 g10 Code GmbH
Packit d7e8d0
Packit d7e8d0
   This file is part of GPGME.
Packit d7e8d0
Packit d7e8d0
   GPGME is free software; you can redistribute it and/or modify it
Packit d7e8d0
   under the terms of the GNU Lesser General Public License as
Packit d7e8d0
   published by the Free Software Foundation; either version 2.1 of
Packit d7e8d0
   the License, or (at your option) any later version.
Packit d7e8d0
Packit d7e8d0
   GPGME is distributed in the hope that it will be useful, but
Packit d7e8d0
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit d7e8d0
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit d7e8d0
   Lesser General Public License for more details.
Packit d7e8d0
Packit d7e8d0
   You should have received a copy of the GNU Lesser General Public
Packit d7e8d0
   License along with this program; if not, write to the Free Software
Packit d7e8d0
   Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
Packit d7e8d0
   02111-1307, USA.  */
Packit d7e8d0
Packit d7e8d0
/* We need to include config.h so that we know whether we are building
Packit d7e8d0
   with large file system (LFS) support. */
Packit d7e8d0
#ifdef HAVE_CONFIG_H
Packit d7e8d0
#include <config.h>
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
#include <stdlib.h>
Packit d7e8d0
#include <stdio.h>
Packit d7e8d0
#include <string.h>
Packit d7e8d0
#include <assert.h>
Packit d7e8d0
#include <limits.h>
Packit d7e8d0
#include <ctype.h>
Packit d7e8d0
#include <errno.h>
Packit d7e8d0
#ifdef HAVE_SYS_TIME_H
Packit d7e8d0
# include <sys/time.h>
Packit d7e8d0
#endif
Packit d7e8d0
#include <sys/types.h>
Packit d7e8d0
#include <unistd.h>
Packit d7e8d0
#include <pthread.h>
Packit d7e8d0
#ifdef HAVE_SYS_SELECT_H
Packit d7e8d0
# include <sys/select.h>
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
#include <gpgme.h>
Packit d7e8d0
Packit d7e8d0
#include "t-support.h"
Packit d7e8d0
Packit d7e8d0
struct op_result
Packit d7e8d0
{
Packit d7e8d0
  int done;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
};
Packit d7e8d0
Packit d7e8d0
static struct op_result op_result;
Packit d7e8d0
Packit d7e8d0
struct one_fd
Packit d7e8d0
{
Packit d7e8d0
  int fd;
Packit d7e8d0
  int dir;
Packit d7e8d0
  gpgme_io_cb_t fnc;
Packit d7e8d0
  void *fnc_data;
Packit d7e8d0
};
Packit d7e8d0
Packit d7e8d0
#define FDLIST_MAX 32
Packit d7e8d0
static struct one_fd fdlist[FDLIST_MAX];
Packit d7e8d0
Packit d7e8d0
static pthread_mutex_t lock;
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
add_io_cb (void *data, int fd, int dir, gpgme_io_cb_t fnc, void *fnc_data,
Packit d7e8d0
           void **r_tag)
Packit d7e8d0
{
Packit d7e8d0
  struct one_fd *fds = data;
Packit d7e8d0
  int i;
Packit d7e8d0
Packit d7e8d0
  pthread_mutex_lock (&lock);
Packit d7e8d0
  for (i = 0; i < FDLIST_MAX; i++)
Packit d7e8d0
    {
Packit d7e8d0
      if (fds[i].fd == -1)
Packit d7e8d0
        {
Packit d7e8d0
          fds[i].fd = fd;
Packit d7e8d0
          fds[i].dir = dir;
Packit d7e8d0
          fds[i].fnc = fnc;
Packit d7e8d0
          fds[i].fnc_data = fnc_data;
Packit d7e8d0
          break;
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
  pthread_mutex_unlock (&lock);
Packit d7e8d0
  if (i == FDLIST_MAX)
Packit d7e8d0
    return gpgme_err_make (GPG_ERR_SOURCE_USER_1, GPG_ERR_GENERAL);
Packit d7e8d0
  *r_tag = &fds[i];
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
remove_io_cb (void *tag)
Packit d7e8d0
{
Packit d7e8d0
  struct one_fd *fd = tag;
Packit d7e8d0
Packit d7e8d0
  pthread_mutex_lock (&lock);
Packit d7e8d0
  fd->fd = -1;
Packit d7e8d0
  pthread_mutex_unlock (&lock);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
io_event (void *data, gpgme_event_io_t type, void *type_data)
Packit d7e8d0
{
Packit d7e8d0
  struct op_result *result = data;
Packit d7e8d0
Packit d7e8d0
  if (type == GPGME_EVENT_DONE)
Packit d7e8d0
    {
Packit d7e8d0
      result->done = 1;
Packit d7e8d0
      result->err = * (gpgme_error_t *) type_data;
Packit d7e8d0
    }
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static int
Packit d7e8d0
do_select (void)
Packit d7e8d0
{
Packit d7e8d0
  fd_set rfds;
Packit d7e8d0
  fd_set wfds;
Packit d7e8d0
  int i, n;
Packit d7e8d0
  int any = 0;
Packit d7e8d0
  struct timeval tv;
Packit d7e8d0
Packit d7e8d0
  pthread_mutex_lock (&lock);
Packit d7e8d0
  FD_ZERO (&rfds);
Packit d7e8d0
  FD_ZERO (&wfds);
Packit d7e8d0
  for (i = 0; i < FDLIST_MAX; i++)
Packit d7e8d0
    if (fdlist[i].fd != -1)
Packit d7e8d0
      FD_SET (fdlist[i].fd, fdlist[i].dir ? &rfds : &wfds);
Packit d7e8d0
  pthread_mutex_unlock (&lock);
Packit d7e8d0
Packit d7e8d0
  tv.tv_sec = 0;
Packit d7e8d0
  tv.tv_usec = 1000;
Packit d7e8d0
Packit d7e8d0
  do
Packit d7e8d0
    {
Packit d7e8d0
      n = select (FD_SETSIZE, &rfds, &wfds, NULL, &tv;;
Packit d7e8d0
    }
Packit d7e8d0
  while (n < 0 && errno == EINTR);
Packit d7e8d0
Packit d7e8d0
  if (n < 0)
Packit d7e8d0
    return n;   /* Error or timeout.  */
Packit d7e8d0
Packit d7e8d0
  pthread_mutex_lock (&lock);
Packit d7e8d0
  for (i = 0; i < FDLIST_MAX && n; i++)
Packit d7e8d0
    {
Packit d7e8d0
      if (fdlist[i].fd != -1)
Packit d7e8d0
        {
Packit d7e8d0
          if (FD_ISSET (fdlist[i].fd, fdlist[i].dir ? &rfds : &wfds))
Packit d7e8d0
            {
Packit d7e8d0
              assert (n);
Packit d7e8d0
              n--;
Packit d7e8d0
              any = 1;
Packit d7e8d0
              (*fdlist[i].fnc) (fdlist[i].fnc_data, fdlist[i].fd);
Packit d7e8d0
            }
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
  pthread_mutex_unlock (&lock);
Packit d7e8d0
  return any;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
static int
Packit d7e8d0
my_wait (void)
Packit d7e8d0
{
Packit d7e8d0
  int n;
Packit d7e8d0
Packit d7e8d0
  do
Packit d7e8d0
    {
Packit d7e8d0
      n = do_select ();
Packit d7e8d0
    }
Packit d7e8d0
  while (n >= 0 && !op_result.done);
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static struct gpgme_io_cbs io_cbs =
Packit d7e8d0
  {
Packit d7e8d0
    add_io_cb,
Packit d7e8d0
    fdlist,
Packit d7e8d0
    remove_io_cb,
Packit d7e8d0
    io_event,
Packit d7e8d0
    &op_result
Packit d7e8d0
  };
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void *
Packit d7e8d0
thread_cancel (void *data)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_ctx_t ctx = data;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit d7e8d0
  usleep (100000);
Packit d7e8d0
  err = gpgme_cancel (ctx);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
Packit d7e8d0
  return NULL;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
int
Packit d7e8d0
main (void)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_ctx_t ctx;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  gpgme_engine_info_t info;
Packit d7e8d0
  int i;
Packit d7e8d0
  pthread_mutexattr_t attr;
Packit d7e8d0
  pthread_t tcancel;
Packit d7e8d0
  const char *parms = "<GnupgKeyParms format=\"internal\">\n"
Packit d7e8d0
    "Key-Type: RSA\n"
Packit d7e8d0
    "Key-Length: 2048\n"
Packit d7e8d0
    "Subkey-Type: RSA\n"
Packit d7e8d0
    "Subkey-Length: 2048\n"
Packit d7e8d0
    "Name-Real: Joe Tester\n"
Packit d7e8d0
    "Name-Comment: (pp=abc)\n"
Packit d7e8d0
    "Name-Email: joe@foo.bar\n"
Packit d7e8d0
    "Expire-Date: 0\n"
Packit d7e8d0
    "Passphrase: abc\n"
Packit d7e8d0
    "</GnupgKeyParms>\n";
Packit d7e8d0
Packit d7e8d0
  init_gpgme (GPGME_PROTOCOL_OpenPGP);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_get_engine_info (&info;;
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
Packit d7e8d0
  /* The mutex must be recursive, since remove_io_cb (which acquires a
Packit d7e8d0
     lock) can be called while holding a lock acquired in do_select.  */
Packit d7e8d0
  pthread_mutexattr_init (&attr);
Packit d7e8d0
  pthread_mutexattr_settype (&attr, PTHREAD_MUTEX_RECURSIVE);
Packit d7e8d0
  pthread_mutex_init (&lock, &attr);
Packit d7e8d0
  pthread_mutexattr_destroy (&attr);
Packit d7e8d0
Packit d7e8d0
  for (i = 0; i < FDLIST_MAX; i++)
Packit d7e8d0
    fdlist[i].fd = -1;
Packit d7e8d0
Packit d7e8d0
  err = gpgme_new (&ctx;;
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
  gpgme_set_armor (ctx, 1);
Packit d7e8d0
  gpgme_set_io_cbs (ctx, &io_cbs);
Packit d7e8d0
  op_result.done = 0;
Packit d7e8d0
Packit d7e8d0
  pthread_create (&tcancel, NULL, thread_cancel, ctx);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_op_genkey_start (ctx, parms, NULL, NULL);
Packit d7e8d0
  fail_if_err (err);
Packit d7e8d0
Packit d7e8d0
  my_wait ();
Packit d7e8d0
Packit d7e8d0
  pthread_join (tcancel, NULL);
Packit d7e8d0
Packit d7e8d0
  if (op_result.err)
Packit d7e8d0
    {
Packit d7e8d0
      if (gpgme_err_code (op_result.err) == GPG_ERR_CANCELED)
Packit d7e8d0
	fputs ("Successfully cancelled\n", stdout);
Packit d7e8d0
      else
Packit d7e8d0
	{
Packit d7e8d0
	  fprintf (stderr,
Packit d7e8d0
		   "%s:%i: Operation finished with unexpected error: %s\n",
Packit d7e8d0
		   __FILE__, __LINE__, gpgme_strerror (op_result.err));
Packit d7e8d0
	  exit (1);
Packit d7e8d0
	}
Packit d7e8d0
    }
Packit d7e8d0
  else
Packit d7e8d0
    fputs ("Successfully finished before cancellation\n", stdout);
Packit d7e8d0
Packit d7e8d0
  gpgme_release (ctx);
Packit d7e8d0
Packit d7e8d0
  return 0;
Packit d7e8d0
}