2010-07-23 16:16:14 +00:00
|
|
|
|
/* ldap-wrapper-ce.c - LDAP access via W32 threads
|
|
|
|
|
* Copyright (C) 2010 Free Software Foundation, Inc.
|
|
|
|
|
*
|
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/*
|
2010-07-23 16:16:14 +00:00
|
|
|
|
Alternative wrapper for use with WindowsCE. Under WindowsCE the
|
|
|
|
|
number of processes is strongly limited (32 processes including the
|
|
|
|
|
kernel processes) and thus we don't use the process approach but
|
|
|
|
|
implement a wrapper based on native threads.
|
|
|
|
|
|
|
|
|
|
See ldap-wrapper.c for the standard wrapper interface.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <pth.h>
|
2010-08-02 18:54:53 +00:00
|
|
|
|
#include <assert.h>
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
#include "dirmngr.h"
|
|
|
|
|
#include "misc.h"
|
|
|
|
|
#include "ldap-wrapper.h"
|
|
|
|
|
|
2010-07-26 14:01:32 +00:00
|
|
|
|
#ifdef USE_LDAPWRAPPER
|
|
|
|
|
# error This module is not expected to be build.
|
|
|
|
|
#endif
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
|
|
|
|
|
/* Read a fixed amount of data from READER into BUFFER. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_buffer (ksba_reader_t reader, unsigned char *buffer, size_t count)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
gpg_error_t err;
|
|
|
|
|
size_t nread;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
while (count)
|
|
|
|
|
{
|
|
|
|
|
err = ksba_reader_read (reader, buffer, count, &nread);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
buffer += nread;
|
|
|
|
|
count -= nread;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Start the reaper thread for this wrapper. */
|
|
|
|
|
void
|
|
|
|
|
ldap_wrapper_launch_thread (void)
|
|
|
|
|
{
|
|
|
|
|
/* Not required. */
|
|
|
|
|
}
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Wait until all ldap wrappers have terminated. We assume that the
|
|
|
|
|
kill has already been sent to all of them. */
|
2010-07-23 16:16:14 +00:00
|
|
|
|
void
|
2010-08-02 18:54:53 +00:00
|
|
|
|
ldap_wrapper_wait_connections ()
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Not required. */
|
|
|
|
|
}
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Cleanup all resources held by the connection associated with
|
|
|
|
|
CTRL. This is used after a cancel to kill running wrappers. */
|
|
|
|
|
void
|
|
|
|
|
ldap_wrapper_connection_cleanup (ctrl_t ctrl)
|
|
|
|
|
{
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
|
|
|
|
/* Not required. */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The cookie we use to implement the outstream of the wrapper thread. */
|
|
|
|
|
struct outstream_cookie_s
|
|
|
|
|
{
|
|
|
|
|
int refcount; /* Reference counter - possible values are 1 and 2. */
|
|
|
|
|
|
|
|
|
|
int eof_seen; /* EOF indicator. */
|
|
|
|
|
size_t buffer_len; /* The valid length of the BUFFER. */
|
|
|
|
|
size_t buffer_pos; /* The next read position of the BUFFER. */
|
|
|
|
|
char buffer[4000]; /* Data buffer. */
|
|
|
|
|
};
|
|
|
|
|
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* The writer function for the outstream. This is used to transfer
|
|
|
|
|
the output of the ldap wrapper thread to the ksba reader object. */
|
|
|
|
|
static ssize_t
|
|
|
|
|
outstream_cookie_writer (void *cookie_arg, const void *buffer, size_t size)
|
|
|
|
|
{
|
|
|
|
|
struct outstream_cookie_s *cookie = cookie_arg;
|
|
|
|
|
const char *src;
|
|
|
|
|
char *dst;
|
|
|
|
|
ssize_t nwritten = 0;
|
|
|
|
|
|
|
|
|
|
src = buffer;
|
|
|
|
|
do
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Wait for free space. */
|
|
|
|
|
while (cookie->buffer_len == DIM (cookie->buffer))
|
|
|
|
|
{
|
|
|
|
|
/* Buffer is full: Wait for space. */
|
|
|
|
|
pth_yield (NULL);
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Copy data. */
|
|
|
|
|
dst = cookie->buffer + cookie->buffer_len;
|
|
|
|
|
while (size && cookie->buffer_len < DIM (cookie->buffer))
|
|
|
|
|
{
|
|
|
|
|
*dst++ = *src++;
|
|
|
|
|
size--;
|
|
|
|
|
cookie->buffer_len++;
|
|
|
|
|
nwritten++;
|
|
|
|
|
}
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
2010-08-02 18:54:53 +00:00
|
|
|
|
while (size); /* Until done. */
|
|
|
|
|
|
|
|
|
|
if (nwritten)
|
|
|
|
|
{
|
|
|
|
|
/* Signal data is available - a pth_yield is sufficient because
|
|
|
|
|
the test is explicit. To increase performance we could do a
|
|
|
|
|
pth_yield to the other thread and only fall back to yielding
|
|
|
|
|
to any thread if that returns an error (i.e. the other thread
|
|
|
|
|
is not runnable). However our w32pth does not yet support
|
|
|
|
|
yielding to a specific thread, thus this won't help. */
|
|
|
|
|
pth_yield (NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nwritten;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
outstream_release_cookie (struct outstream_cookie_s *cookie)
|
|
|
|
|
{
|
|
|
|
|
cookie->refcount--;
|
|
|
|
|
if (!cookie->refcount)
|
|
|
|
|
xfree (cookie);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Closer function for the outstream. This deallocates the cookie if
|
|
|
|
|
it won't be used anymore. */
|
|
|
|
|
static int
|
|
|
|
|
outstream_cookie_closer (void *cookie_arg)
|
|
|
|
|
{
|
|
|
|
|
struct outstream_cookie_s *cookie = cookie_arg;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
if (!cookie)
|
|
|
|
|
return 0; /* Nothing to do. */
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
cookie->eof_seen = 1; /* (only useful if refcount > 1) */
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
assert (cookie->refcount > 0);
|
|
|
|
|
outstream_release_cookie (cookie);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The KSBA reader callback which takes the output of the ldap thread
|
|
|
|
|
form the outstream_cookie_writer and make it available to the ksba
|
|
|
|
|
reader. */
|
|
|
|
|
static int
|
|
|
|
|
outstream_reader_cb (void *cb_value, char *buffer, size_t count,
|
|
|
|
|
size_t *r_nread)
|
|
|
|
|
{
|
|
|
|
|
struct outstream_cookie_s *cookie = cb_value;
|
|
|
|
|
char *dst;
|
|
|
|
|
const char *src;
|
|
|
|
|
size_t nread = 0;
|
|
|
|
|
|
2010-08-06 13:52:01 +00:00
|
|
|
|
if (!buffer && !count && !r_nread)
|
2010-08-02 18:54:53 +00:00
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED); /* Rewind is not supported. */
|
|
|
|
|
|
|
|
|
|
*r_nread = 0;
|
|
|
|
|
dst = buffer;
|
|
|
|
|
|
|
|
|
|
while (cookie->buffer_pos == cookie->buffer_len)
|
|
|
|
|
{
|
|
|
|
|
if (cookie->eof_seen)
|
|
|
|
|
return gpg_error (GPG_ERR_EOF);
|
|
|
|
|
|
|
|
|
|
/* Wait for data to become available. */
|
|
|
|
|
pth_yield (NULL);
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
src = cookie->buffer + cookie->buffer_pos;
|
|
|
|
|
while (count && cookie->buffer_pos < cookie->buffer_len)
|
|
|
|
|
{
|
|
|
|
|
*dst++ = *src++;
|
|
|
|
|
count--;
|
|
|
|
|
cookie->buffer_pos++;
|
|
|
|
|
nread++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cookie->buffer_pos == cookie->buffer_len)
|
|
|
|
|
cookie->buffer_pos = cookie->buffer_len = 0;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/* Now there should be some space available. We do this even if
|
|
|
|
|
COUNT was zero so to give the writer end a chance to continue. */
|
|
|
|
|
pth_yield (NULL);
|
|
|
|
|
|
|
|
|
|
*r_nread = nread;
|
|
|
|
|
return 0; /* Success. */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* This function is called by ksba_reader_release. */
|
|
|
|
|
static void
|
|
|
|
|
outstream_reader_released (void *cb_value, ksba_reader_t r)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
struct outstream_cookie_s *cookie = cb_value;
|
|
|
|
|
|
|
|
|
|
(void)r;
|
|
|
|
|
|
|
|
|
|
assert (cookie->refcount > 0);
|
|
|
|
|
outstream_release_cookie (cookie);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
|
2010-07-23 16:16:14 +00:00
|
|
|
|
/* This function is to be used to release a context associated with the
|
2010-08-02 18:54:53 +00:00
|
|
|
|
given reader object. This does not release the reader object, though. */
|
2010-07-23 16:16:14 +00:00
|
|
|
|
void
|
|
|
|
|
ldap_wrapper_release_context (ksba_reader_t reader)
|
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
(void)reader;
|
|
|
|
|
/* Nothing to do. */
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Free a NULL terminated array of malloced strings and the array
|
|
|
|
|
itself. */
|
|
|
|
|
static void
|
|
|
|
|
free_arg_list (char **arg_list)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
int i;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
if (arg_list)
|
|
|
|
|
{
|
|
|
|
|
for (i=0; arg_list[i]; i++)
|
|
|
|
|
xfree (arg_list[i]);
|
|
|
|
|
xfree (arg_list);
|
|
|
|
|
}
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
|
|
|
|
|
/* Copy ARGV into a new array and prepend one element as name of the
|
|
|
|
|
program (which is more or less a stub). We need to allocate all
|
|
|
|
|
the strings to get ownership of them. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
create_arg_list (const char *argv[], char ***r_arg_list)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2010-08-02 18:54:53 +00:00
|
|
|
|
char **arg_list;
|
|
|
|
|
int i, j;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
|
|
|
|
for (i = 0; argv[i]; i++)
|
|
|
|
|
;
|
|
|
|
|
arg_list = xtrycalloc (i + 2, sizeof *arg_list);
|
|
|
|
|
if (!arg_list)
|
2010-08-02 18:54:53 +00:00
|
|
|
|
goto outofcore;
|
|
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
arg_list[i] = xtrystrdup ("<ldap-wrapper-thread>");
|
|
|
|
|
if (!arg_list[i])
|
|
|
|
|
goto outofcore;
|
|
|
|
|
i++;
|
|
|
|
|
for (j=0; argv[j]; j++)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
arg_list[i] = xtrystrdup (argv[j]);
|
|
|
|
|
if (!arg_list[i])
|
|
|
|
|
goto outofcore;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
arg_list[i] = NULL;
|
|
|
|
|
*r_arg_list = arg_list;
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
outofcore:
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error (_("error allocating memory: %s\n"), strerror (errno));
|
|
|
|
|
free_arg_list (arg_list);
|
|
|
|
|
*r_arg_list = NULL;
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Parameters passed to the wrapper thread. */
|
|
|
|
|
struct ldap_wrapper_thread_parms
|
|
|
|
|
{
|
|
|
|
|
char **arg_list;
|
|
|
|
|
estream_t outstream;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* The thread which runs the LDAP wrapper. */
|
|
|
|
|
static void *
|
|
|
|
|
ldap_wrapper_thread (void *opaque)
|
|
|
|
|
{
|
|
|
|
|
struct ldap_wrapper_thread_parms *parms = opaque;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
/*err =*/ ldap_wrapper_main (parms->arg_list, parms->outstream);
|
|
|
|
|
|
|
|
|
|
/* FIXME: Do we need to return ERR? */
|
|
|
|
|
|
|
|
|
|
free_arg_list (parms->arg_list);
|
|
|
|
|
es_fclose (parms->outstream);
|
|
|
|
|
xfree (parms);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Start a new LDAP thread and returns a new libksba reader
|
|
|
|
|
object at READER. ARGV is a NULL terminated list of arguments for
|
|
|
|
|
the wrapper. The function returns 0 on success or an error code. */
|
|
|
|
|
gpg_error_t
|
|
|
|
|
ldap_wrapper (ctrl_t ctrl, ksba_reader_t *r_reader, const char *argv[])
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
struct ldap_wrapper_thread_parms *parms;
|
|
|
|
|
pth_attr_t tattr;
|
|
|
|
|
es_cookie_io_functions_t outstream_func = { NULL };
|
|
|
|
|
struct outstream_cookie_s *outstream_cookie;
|
|
|
|
|
ksba_reader_t reader;
|
|
|
|
|
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
|
|
|
|
*r_reader = NULL;
|
|
|
|
|
|
|
|
|
|
parms = xtrycalloc (1, sizeof *parms);
|
|
|
|
|
if (!parms)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
|
|
|
|
|
err = create_arg_list (argv, &parms->arg_list);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
xfree (parms);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
outstream_cookie = xtrycalloc (1, sizeof *outstream_cookie);
|
|
|
|
|
if (!outstream_cookie)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
2010-08-02 18:54:53 +00:00
|
|
|
|
free_arg_list (parms->arg_list);
|
|
|
|
|
xfree (parms);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
2010-08-02 18:54:53 +00:00
|
|
|
|
outstream_cookie->refcount++;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
err = ksba_reader_new (&reader);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
if (!err)
|
2010-08-02 18:54:53 +00:00
|
|
|
|
err = ksba_reader_set_release_notify (reader,
|
|
|
|
|
outstream_reader_released,
|
|
|
|
|
outstream_cookie);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = ksba_reader_set_cb (reader,
|
|
|
|
|
outstream_reader_cb, outstream_cookie);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
log_error (_("error initializing reader object: %s\n"),
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
ksba_reader_release (reader);
|
|
|
|
|
outstream_release_cookie (outstream_cookie);
|
|
|
|
|
free_arg_list (parms->arg_list);
|
|
|
|
|
xfree (parms);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
|
|
|
|
|
outstream_func.func_write = outstream_cookie_writer;
|
|
|
|
|
outstream_func.func_close = outstream_cookie_closer;
|
|
|
|
|
parms->outstream = es_fopencookie (outstream_cookie, "wb", outstream_func);
|
|
|
|
|
if (!parms->outstream)
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
free_arg_list (parms->arg_list);
|
|
|
|
|
outstream_release_cookie (outstream_cookie);
|
|
|
|
|
xfree (parms);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
2010-08-02 18:54:53 +00:00
|
|
|
|
outstream_cookie->refcount++;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
tattr = pth_attr_new();
|
|
|
|
|
pth_attr_set (tattr, PTH_ATTR_JOINABLE, 0);
|
|
|
|
|
pth_attr_set (tattr, PTH_ATTR_STACK_SIZE, 128*1024);
|
|
|
|
|
pth_attr_set (tattr, PTH_ATTR_NAME, "ldap-wrapper");
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
if (pth_spawn (tattr, ldap_wrapper_thread, parms))
|
|
|
|
|
parms = NULL; /* Now owned by the thread. */
|
|
|
|
|
else
|
2010-07-23 16:16:14 +00:00
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
log_error ("error spawning ldap wrapper thread: %s\n",
|
|
|
|
|
strerror (errno) );
|
|
|
|
|
}
|
|
|
|
|
pth_attr_destroy (tattr);
|
|
|
|
|
if (parms)
|
|
|
|
|
{
|
|
|
|
|
free_arg_list (parms->arg_list);
|
|
|
|
|
es_fclose (parms->outstream);
|
|
|
|
|
xfree (parms);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
ksba_reader_release (reader);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Need to wait for the first byte so we are able to detect an empty
|
|
|
|
|
output and not let the consumer see an EOF without further error
|
|
|
|
|
indications. The CRL loading logic assumes that after return
|
|
|
|
|
from this function, a failed search (e.g. host not found ) is
|
|
|
|
|
indicated right away. */
|
|
|
|
|
{
|
|
|
|
|
unsigned char c;
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
err = read_buffer (reader, &c, 1);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2010-08-02 18:54:53 +00:00
|
|
|
|
ksba_reader_release (reader);
|
|
|
|
|
reader = NULL;
|
2010-07-23 16:16:14 +00:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
return gpg_error (GPG_ERR_NO_DATA);
|
|
|
|
|
else
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2010-08-02 18:54:53 +00:00
|
|
|
|
ksba_reader_unread (reader, &c, 1);
|
2010-07-23 16:16:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-08-02 18:54:53 +00:00
|
|
|
|
*r_reader = reader;
|
|
|
|
|
|
2010-07-23 16:16:14 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|