Blame src/data-compat.c

Packit d7e8d0
/* data-compat.c - Compatibility interfaces for data objects.
Packit d7e8d0
   Copyright (C) 2002, 2003, 2004, 2007 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
#if HAVE_CONFIG_H
Packit d7e8d0
#include <config.h>
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
#include <errno.h>
Packit d7e8d0
#ifdef HAVE_SYS_TIME_H
Packit d7e8d0
# include <sys/time.h>
Packit d7e8d0
#endif
Packit d7e8d0
#ifdef HAVE_SYS_STAT_H
Packit d7e8d0
# include <sys/stat.h>
Packit d7e8d0
#endif
Packit d7e8d0
#include <stdlib.h>
Packit d7e8d0
Packit d7e8d0
#include "data.h"
Packit d7e8d0
#include "util.h"
Packit d7e8d0
#include "debug.h"
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
/* Create a new data buffer filled with LENGTH bytes starting from
Packit d7e8d0
   OFFSET within the file FNAME or stream STREAM (exactly one must be
Packit d7e8d0
   non-zero).  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_data_new_from_filepart (gpgme_data_t *r_dh, const char *fname,
Packit d7e8d0
			      FILE *stream, gpgme_off_t offset, size_t length)
Packit d7e8d0
{
Packit d7e8d0
#if defined (HAVE_W32CE_SYSTEM) && defined (_MSC_VER)
Packit d7e8d0
  return gpgme_error (GPG_ERR_NOT_IMPLEMENTED);
Packit d7e8d0
#else
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  char *buf = NULL;
Packit d7e8d0
  int res;
Packit d7e8d0
Packit d7e8d0
  TRACE_BEG4 (DEBUG_DATA, "gpgme_data_new_from_filepart", r_dh,
Packit d7e8d0
	      "file_name=%s, stream=%p, offset=%lli, length=%u",
Packit d7e8d0
	      fname, stream, offset, length);
Packit d7e8d0
Packit d7e8d0
  if (stream && fname)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
Packit d7e8d0
Packit d7e8d0
  if (fname)
Packit d7e8d0
    stream = fopen (fname, "rb");
Packit d7e8d0
  if (!stream)
Packit d7e8d0
    return TRACE_ERR (gpg_error_from_syserror ());
Packit d7e8d0
Packit d7e8d0
#ifdef HAVE_FSEEKO
Packit d7e8d0
  res = fseeko (stream, offset, SEEK_SET);
Packit d7e8d0
#else
Packit d7e8d0
  /* FIXME: Check for overflow, or at least bail at compilation.  */
Packit d7e8d0
  res = fseek (stream, offset, SEEK_SET);
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
  if (res)
Packit d7e8d0
    {
Packit d7e8d0
      int saved_err = gpg_error_from_syserror ();
Packit d7e8d0
      if (fname)
Packit d7e8d0
	fclose (stream);
Packit d7e8d0
      return TRACE_ERR (saved_err);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  buf = malloc (length);
Packit d7e8d0
  if (!buf)
Packit d7e8d0
    {
Packit d7e8d0
      int saved_err = gpg_error_from_syserror ();
Packit d7e8d0
      if (fname)
Packit d7e8d0
	fclose (stream);
Packit d7e8d0
      return TRACE_ERR (saved_err);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  while (fread (buf, length, 1, stream) < 1
Packit d7e8d0
	 && ferror (stream) && errno == EINTR);
Packit d7e8d0
  if (ferror (stream))
Packit d7e8d0
    {
Packit d7e8d0
      int saved_err = gpg_error_from_syserror ();
Packit d7e8d0
      if (buf)
Packit d7e8d0
	free (buf);
Packit d7e8d0
      if (fname)
Packit d7e8d0
	fclose (stream);
Packit d7e8d0
      return TRACE_ERR (saved_err);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  if (fname)
Packit d7e8d0
    fclose (stream);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_data_new (r_dh);
Packit d7e8d0
  if (err)
Packit d7e8d0
    {
Packit d7e8d0
      if (buf)
Packit d7e8d0
	free (buf);
Packit d7e8d0
      return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  (*r_dh)->data.mem.buffer = buf;
Packit d7e8d0
  (*r_dh)->data.mem.size = length;
Packit d7e8d0
  (*r_dh)->data.mem.length = length;
Packit d7e8d0
Packit d7e8d0
  return TRACE_SUC1 ("r_dh=%p", *r_dh);
Packit d7e8d0
#endif
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
/* Create a new data buffer filled with the content of file FNAME.
Packit d7e8d0
   COPY must be non-zero (delayed reads are not supported yet).  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_data_new_from_file (gpgme_data_t *r_dh, const char *fname, int copy)
Packit d7e8d0
{
Packit d7e8d0
#if defined (HAVE_W32CE_SYSTEM) && defined (_MSC_VER)
Packit d7e8d0
  return gpgme_error (GPG_ERR_NOT_IMPLEMENTED);
Packit d7e8d0
#else
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  struct stat statbuf;
Packit d7e8d0
  TRACE_BEG3 (DEBUG_DATA, "gpgme_data_new_from_file", r_dh,
Packit d7e8d0
	      "file_name=%s, copy=%i (%s)", fname, copy, copy ? "yes" : "no");
Packit d7e8d0
Packit d7e8d0
  if (!fname || !copy)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
Packit d7e8d0
Packit d7e8d0
  if (stat (fname, &statbuf) < 0)
Packit d7e8d0
    return TRACE_ERR (gpg_error_from_syserror ());
Packit d7e8d0
Packit d7e8d0
  err = gpgme_data_new_from_filepart (r_dh, fname, NULL, 0, statbuf.st_size);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
#endif
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
static int
Packit d7e8d0
gpgme_error_to_errno (gpgme_error_t err)
Packit d7e8d0
{
Packit d7e8d0
  int res = gpg_err_code_to_errno (gpg_err_code (err));
Packit d7e8d0
Packit d7e8d0
  if (!err)
Packit d7e8d0
    {
Packit d7e8d0
      switch (gpg_err_code (err))
Packit d7e8d0
	{
Packit d7e8d0
	case GPG_ERR_EOF:
Packit d7e8d0
	  res = 0;
Packit d7e8d0
	  break;
Packit d7e8d0
	case GPG_ERR_INV_VALUE:
Packit d7e8d0
	  res = EINVAL;
Packit d7e8d0
	  break;
Packit d7e8d0
	case GPG_ERR_NOT_SUPPORTED:
Packit d7e8d0
	  res = ENOSYS;
Packit d7e8d0
	  break;
Packit d7e8d0
	default:
Packit d7e8d0
	  /* FIXME: Yeah, well.  */
Packit d7e8d0
	  res = EINVAL;
Packit d7e8d0
	  break;
Packit d7e8d0
	}
Packit d7e8d0
    }
Packit d7e8d0
  TRACE3 (DEBUG_DATA, "gpgme:gpgme_error_to_errno", 0,
Packit d7e8d0
	  "mapping %s <%s> to: %s", gpgme_strerror (err),
Packit d7e8d0
	  gpgme_strsource (err), strerror (res));
Packit d7e8d0
  gpg_err_set_errno (res);
Packit d7e8d0
  return res ? -1 : 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_ssize_t
Packit d7e8d0
old_user_read (gpgme_data_t dh, void *buffer, size_t size)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  size_t amt;
Packit d7e8d0
  TRACE_BEG2 (DEBUG_DATA, "gpgme:old_user_read", dh,
Packit d7e8d0
	      "buffer=%p, size=%u", buffer, size);
Packit d7e8d0
Packit d7e8d0
  err = (*dh->data.old_user.cb) (dh->data.old_user.handle,
Packit d7e8d0
				 buffer, size, &amt;;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return TRACE_SYSRES (gpgme_error_to_errno (err));
Packit d7e8d0
  return TRACE_SYSRES ((gpgme_ssize_t)amt);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_off_t
Packit d7e8d0
old_user_seek (gpgme_data_t dh, gpgme_off_t offset, int whence)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  TRACE_BEG2 (DEBUG_DATA, "gpgme:old_user_seek", dh,
Packit d7e8d0
	      "offset=%llu, whence=%i", offset, whence);
Packit d7e8d0
Packit d7e8d0
  if (whence != SEEK_SET || offset)
Packit d7e8d0
    {
Packit d7e8d0
      gpg_err_set_errno (EINVAL);
Packit d7e8d0
      return TRACE_SYSRES (-1);
Packit d7e8d0
    }
Packit d7e8d0
  err = (*dh->data.old_user.cb) (dh->data.old_user.handle, NULL, 0, NULL);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return TRACE_SYSRES (gpgme_error_to_errno (err));
Packit d7e8d0
  return TRACE_SYSRES (0);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static struct _gpgme_data_cbs old_user_cbs =
Packit d7e8d0
  {
Packit d7e8d0
    old_user_read,
Packit d7e8d0
    NULL,
Packit d7e8d0
    old_user_seek,
Packit d7e8d0
    NULL
Packit d7e8d0
  };
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Create a new data buffer which retrieves the data from the callback
Packit d7e8d0
   function READ_CB.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_data_new_with_read_cb (gpgme_data_t *r_dh,
Packit d7e8d0
                             int (*read_cb) (void *, char *, size_t, size_t *),
Packit d7e8d0
                             void *read_cb_value)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  TRACE_BEG2 (DEBUG_DATA, "gpgme_data_new_with_read_cb", r_dh,
Packit d7e8d0
	      "read_cb=%p/%p", read_cb, read_cb_value);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_data_new (r_dh, &old_user_cbs);
Packit d7e8d0
Packit d7e8d0
  if (err)
Packit d7e8d0
    return TRACE_ERR (err);
Packit d7e8d0
Packit d7e8d0
  (*r_dh)->data.old_user.cb = read_cb;
Packit d7e8d0
  (*r_dh)->data.old_user.handle = read_cb_value;
Packit d7e8d0
  return TRACE_ERR (0);
Packit d7e8d0
}