Blame src/vfs-mount.c

Packit d7e8d0
/* vfs-mount.c - vfs mount support in GPGME
Packit d7e8d0
   Copyright (C) 2009 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 <stdlib.h>
Packit d7e8d0
Packit d7e8d0
#include "gpgme.h"
Packit d7e8d0
#include "debug.h"
Packit d7e8d0
#include "context.h"
Packit d7e8d0
#include "ops.h"
Packit d7e8d0
#include "util.h"
Packit d7e8d0
Packit d7e8d0
typedef struct
Packit d7e8d0
{
Packit d7e8d0
  struct _gpgme_op_vfs_mount_result result;
Packit d7e8d0
} *op_data_t;
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
gpgme_vfs_mount_result_t
Packit d7e8d0
gpgme_op_vfs_mount_result (gpgme_ctx_t ctx)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_VFS_MOUNT, &hook, -1, NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  /* Check in case this function is used without having run a command
Packit d7e8d0
     before.  */
Packit d7e8d0
  if (err || !opd)
Packit d7e8d0
    return NULL;
Packit d7e8d0
Packit d7e8d0
  return &opd->result;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
_gpgme_vfs_mount_status_handler (void *priv, const char *code, const char *args)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_ctx_t ctx = (gpgme_ctx_t) priv;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_VFS_MOUNT, &hook, -1, NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  if (! strcasecmp ("MOUNTPOINT", code))
Packit d7e8d0
    {
Packit d7e8d0
      if (opd->result.mount_dir)
Packit d7e8d0
	free (opd->result.mount_dir);
Packit d7e8d0
      opd->result.mount_dir = strdup (args);
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
vfs_start (gpgme_ctx_t ctx, int synchronous,
Packit d7e8d0
	   const char *command,
Packit d7e8d0
	   gpgme_assuan_data_cb_t data_cb,
Packit d7e8d0
	   void *data_cb_value,
Packit d7e8d0
	   gpgme_assuan_inquire_cb_t inq_cb,
Packit d7e8d0
	   void *inq_cb_value,
Packit d7e8d0
	   gpgme_assuan_status_cb_t status_cb,
Packit d7e8d0
	   void *status_cb_value)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
Packit d7e8d0
  if (!command || !*command)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
Packit d7e8d0
  /* The flag value 256 is used to suppress an engine reset.  This is
Packit d7e8d0
     required to keep the connection running.  */
Packit d7e8d0
  err = _gpgme_op_reset (ctx, ((synchronous & 255) | 256));
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_VFS_MOUNT, &hook, sizeof (*opd),
Packit d7e8d0
			       NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_assuan_transact (ctx->engine, command,
Packit d7e8d0
					   data_cb, data_cb_value,
Packit d7e8d0
					   inq_cb, inq_cb_value,
Packit d7e8d0
					   status_cb, status_cb_value);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
#if 0
Packit d7e8d0
/* XXXX.  This is the asynchronous variant. */
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
gpgme_op_vfs_transact_start (gpgme_ctx_t ctx,
Packit d7e8d0
			     const char *command,
Packit d7e8d0
			     gpgme_assuan_data_cb_t data_cb,
Packit d7e8d0
			     void *data_cb_value,
Packit d7e8d0
			     gpgme_assuan_inquire_cb_t inq_cb,
Packit d7e8d0
			     void *inq_cb_value,
Packit d7e8d0
			     gpgme_assuan_status_cb_t status_cb,
Packit d7e8d0
			     void *status_cb_value)
Packit d7e8d0
{
Packit d7e8d0
  return vfs_start (ctx, 0, command, data_cb, data_cb_value,
Packit d7e8d0
		    inq_cb, inq_cb_value, status_cb, status_cb_value);
Packit d7e8d0
}
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* XXXX.  This is the synchronous variant. */
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
gpgme_op_vfs_transact (gpgme_ctx_t ctx,
Packit d7e8d0
		       const char *command,
Packit d7e8d0
		       gpgme_assuan_data_cb_t data_cb,
Packit d7e8d0
		       void *data_cb_value,
Packit d7e8d0
		       gpgme_assuan_inquire_cb_t inq_cb,
Packit d7e8d0
		       void *inq_cb_value,
Packit d7e8d0
		       gpgme_assuan_status_cb_t status_cb,
Packit d7e8d0
		       void *status_cb_value,
Packit d7e8d0
		       gpgme_error_t *op_err)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
Packit d7e8d0
  err = vfs_start (ctx, 1, command, data_cb, data_cb_value,
Packit d7e8d0
		   inq_cb, inq_cb_value, status_cb, status_cb_value);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one_ext (ctx, op_err);
Packit d7e8d0
  return err;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
/* The actual exported interface follows.  */
Packit d7e8d0
Packit d7e8d0
/* The container is automatically unmounted when the context is reset
Packit d7e8d0
   or destroyed.  This is a synchronous convenience interface, which
Packit d7e8d0
   automatically returns an operation error if there is no
Packit d7e8d0
   transmission error.  */
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
_gpgme_op_vfs_mount (gpgme_ctx_t ctx, const char *container_file,
Packit d7e8d0
		     const char *mount_dir, int flags, gpgme_error_t *op_err)
Packit d7e8d0
{
Packit d7e8d0
  gpg_error_t err;
Packit d7e8d0
  char *cmd;
Packit d7e8d0
  char *container_file_esc = NULL;
Packit d7e8d0
Packit d7e8d0
  (void)flags;
Packit d7e8d0
Packit d7e8d0
  /* We want to encourage people to check error values, so not getting
Packit d7e8d0
     them is discouraged here.  Also makes our code easier.  */
Packit d7e8d0
  if (! op_err)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_encode_percent_string (container_file, &container_file_esc, 0);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  if (gpgrt_asprintf (&cmd, "OPEN -- %s", container_file_esc) < 0)
Packit d7e8d0
    {
Packit d7e8d0
      err = gpg_error_from_syserror ();
Packit d7e8d0
      free (container_file_esc);
Packit d7e8d0
      return err;
Packit d7e8d0
    }
Packit d7e8d0
  free (container_file_esc);
Packit d7e8d0
Packit d7e8d0
  err = gpgme_op_vfs_transact (ctx, cmd, NULL, NULL, NULL, NULL,
Packit d7e8d0
			       NULL, NULL, op_err);
Packit d7e8d0
  gpgrt_free (cmd);
Packit d7e8d0
  if (err || *op_err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  if (mount_dir)
Packit d7e8d0
    {
Packit d7e8d0
      char *mount_dir_esc = NULL;
Packit d7e8d0
Packit d7e8d0
      err = _gpgme_encode_percent_string (mount_dir, &mount_dir_esc, 0);
Packit d7e8d0
      if (err)
Packit d7e8d0
	return err;
Packit d7e8d0
Packit d7e8d0
      if (gpgrt_asprintf (&cmd, "MOUNT -- %s", mount_dir_esc) < 0)
Packit d7e8d0
	{
Packit d7e8d0
	  err = gpg_error_from_syserror ();
Packit d7e8d0
	  free (mount_dir_esc);
Packit d7e8d0
	  return err;
Packit d7e8d0
	}
Packit d7e8d0
      free (mount_dir_esc);
Packit d7e8d0
    }
Packit d7e8d0
  else
Packit d7e8d0
    {
Packit d7e8d0
      if (gpgrt_asprintf (&cmd, "MOUNT") < 0)
Packit d7e8d0
	return gpg_error_from_syserror ();
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  err = gpgme_op_vfs_transact (ctx, cmd, NULL, NULL, NULL, NULL,
Packit d7e8d0
			       _gpgme_vfs_mount_status_handler, ctx, op_err);
Packit d7e8d0
  gpgrt_free (cmd);
Packit d7e8d0
Packit d7e8d0
  return err;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_vfs_mount (gpgme_ctx_t ctx, const char *container_file,
Packit d7e8d0
		    const char *mount_dir, unsigned int flags,
Packit d7e8d0
		    gpgme_error_t *op_err)
Packit d7e8d0
{
Packit d7e8d0
  gpg_error_t err;
Packit d7e8d0
Packit d7e8d0
  TRACE_BEG4 (DEBUG_CTX, "gpgme_op_vfs_mount", ctx,
Packit d7e8d0
	      "container=%s, mount_dir=%s, flags=0x%x, op_err=%p",
Packit d7e8d0
	      container_file, mount_dir, flags, op_err);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_vfs_mount (ctx, container_file, mount_dir, flags, op_err);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0