Blame src/genkey.c

Packit d7e8d0
/* genkey.c - Key generation.
Packit d7e8d0
 * Copyright (C) 2000 Werner Koch (dd9jn)
Packit d7e8d0
 * Copyright (C) 2001, 2002, 2003, 2004, 2016 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 Service 30b792
 * License along with this program; if not, see <https://gnu.org/licenses/>.
Packit Service 30b792
 * SPDX-License-Identifier: LGPL-2.1-or-later
Packit d7e8d0
 */
Packit d7e8d0
Packit d7e8d0
#if HAVE_CONFIG_H
Packit d7e8d0
#include <config.h>
Packit d7e8d0
#endif
Packit d7e8d0
#include <stdlib.h>
Packit d7e8d0
#include <string.h>
Packit d7e8d0
#include <errno.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

Packit d7e8d0
typedef struct
Packit d7e8d0
{
Packit d7e8d0
  struct _gpgme_op_genkey_result result;
Packit d7e8d0
Packit d7e8d0
  /* The error code from a FAILURE status line or 0.  */
Packit d7e8d0
  gpg_error_t failure_code;
Packit d7e8d0
Packit d7e8d0
  /* The error code from certain ERROR status lines or 0.  */
Packit d7e8d0
  gpg_error_t error_code;
Packit d7e8d0
Packit d7e8d0
  /* Flag to indicate that a UID is to be added.  */
Packit d7e8d0
  gpg_error_t uidmode;
Packit d7e8d0
Packit d7e8d0
  /* The key parameters passed to the crypto engine.  */
Packit d7e8d0
  gpgme_data_t key_parameter;
Packit d7e8d0
} *op_data_t;
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static void
Packit d7e8d0
release_op_data (void *hook)
Packit d7e8d0
{
Packit d7e8d0
  op_data_t opd = (op_data_t) hook;
Packit d7e8d0
Packit d7e8d0
  if (opd->result.fpr)
Packit d7e8d0
    free (opd->result.fpr);
Packit d7e8d0
  if (opd->key_parameter)
Packit d7e8d0
    gpgme_data_release (opd->key_parameter);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_genkey_result_t
Packit d7e8d0
gpgme_op_genkey_result (gpgme_ctx_t ctx)
Packit d7e8d0
{
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG (DEBUG_CTX, "gpgme_op_genkey_result", ctx, "");
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook, -1, NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err || !opd)
Packit d7e8d0
    {
Packit Service 30b792
      TRACE_SUC ("result=(null)");
Packit d7e8d0
      return NULL;
Packit d7e8d0
    }
Packit d7e8d0
Packit Service 30b792
  TRACE_LOG  ("fpr = %s, %s, %s", opd->result.fpr,
Packit d7e8d0
	      opd->result.primary ? "primary" : "no primary",
Packit d7e8d0
	      opd->result.sub ? "sub" : "no sub");
Packit d7e8d0
Packit Service 30b792
  TRACE_SUC ("result=%p", &opd->result);
Packit d7e8d0
  return &opd->result;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
/* Parse an error status line.  Return the error location and the
Packit d7e8d0
   error code.  The function may modify ARGS. */
Packit d7e8d0
static char *
Packit d7e8d0
parse_error (char *args, gpg_error_t *r_err)
Packit d7e8d0
{
Packit d7e8d0
  char *where = strchr (args, ' ');
Packit d7e8d0
  char *which;
Packit d7e8d0
Packit d7e8d0
  if (where)
Packit d7e8d0
    {
Packit d7e8d0
      *where = '\0';
Packit d7e8d0
      which = where + 1;
Packit d7e8d0
Packit d7e8d0
      where = strchr (which, ' ');
Packit d7e8d0
      if (where)
Packit d7e8d0
	*where = '\0';
Packit d7e8d0
Packit d7e8d0
      where = args;
Packit d7e8d0
    }
Packit d7e8d0
  else
Packit d7e8d0
    {
Packit d7e8d0
      *r_err = trace_gpg_error (GPG_ERR_INV_ENGINE);
Packit d7e8d0
      return NULL;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  *r_err = atoi (which);
Packit d7e8d0
Packit d7e8d0
  return where;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
genkey_status_handler (void *priv, gpgme_status_code_t code, 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
  char *loc;
Packit d7e8d0
Packit d7e8d0
  /* Pipe the status code through the progress status handler.  */
Packit d7e8d0
  err = _gpgme_progress_status_handler (ctx, code, args);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook, -1, NULL);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  switch (code)
Packit d7e8d0
    {
Packit d7e8d0
    case GPGME_STATUS_KEY_CREATED:
Packit d7e8d0
      if (args && *args)
Packit d7e8d0
	{
Packit d7e8d0
	  if (*args == 'B' || *args == 'P')
Packit d7e8d0
            {
Packit d7e8d0
              opd->result.primary = 1;
Packit d7e8d0
              opd->result.uid = 1;
Packit d7e8d0
            }
Packit d7e8d0
	  if (*args == 'B' || *args == 'S')
Packit d7e8d0
	    opd->result.sub = 1;
Packit d7e8d0
	  if (args[1] == ' ')
Packit d7e8d0
	    {
Packit d7e8d0
	      if (opd->result.fpr)
Packit d7e8d0
		free (opd->result.fpr);
Packit d7e8d0
	      opd->result.fpr = strdup (&args[2]);
Packit d7e8d0
	      if (!opd->result.fpr)
Packit d7e8d0
		return gpg_error_from_syserror ();
Packit d7e8d0
	    }
Packit d7e8d0
	}
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    case GPGME_STATUS_ERROR:
Packit d7e8d0
      loc = parse_error (args, &err;;
Packit d7e8d0
      if (!loc)
Packit d7e8d0
        return err;
Packit d7e8d0
      if (!opd->error_code)
Packit d7e8d0
        opd->error_code = err;
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    case GPGME_STATUS_FAILURE:
Packit d7e8d0
      opd->failure_code = _gpgme_parse_failure (args);
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    case GPGME_STATUS_EOF:
Packit d7e8d0
      if (opd->error_code)
Packit d7e8d0
        return opd->error_code;
Packit d7e8d0
      else if (!opd->uidmode && !opd->result.primary && !opd->result.sub)
Packit d7e8d0
	return gpg_error (GPG_ERR_GENERAL);
Packit d7e8d0
      else if (opd->failure_code)
Packit d7e8d0
        return opd->failure_code;
Packit d7e8d0
      else if (opd->uidmode == 1)
Packit d7e8d0
        opd->result.uid = 1;  /* We have no status line, thus this hack.  */
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    case GPGME_STATUS_INQUIRE_MAXLEN:
Packit d7e8d0
      if (ctx->status_cb && !ctx->full_status)
Packit d7e8d0
        {
Packit d7e8d0
          err = ctx->status_cb (ctx->status_cb_value, "INQUIRE_MAXLEN", args);
Packit d7e8d0
          if (err)
Packit d7e8d0
            return err;
Packit d7e8d0
        }
Packit d7e8d0
      break;
Packit d7e8d0
Packit d7e8d0
    default:
Packit d7e8d0
      break;
Packit d7e8d0
    }
Packit d7e8d0
  return 0;
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
get_key_parameter (const char *parms, gpgme_data_t *key_parameter)
Packit d7e8d0
{
Packit d7e8d0
  const char *content;
Packit d7e8d0
  const char *attrib;
Packit d7e8d0
  const char *endtag;
Packit d7e8d0
Packit d7e8d0
  /* Extract the key parameter from the XML structure.  */
Packit d7e8d0
  parms = strstr (parms, "
Packit d7e8d0
  if (!parms)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
Packit d7e8d0
  content = strchr (parms, '>');
Packit d7e8d0
  if (!content)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
  content++;
Packit d7e8d0
Packit d7e8d0
  attrib = strstr (parms, "format=\"internal\"");
Packit d7e8d0
  if (!attrib || attrib >= content)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_VALUE);
Packit d7e8d0
Packit d7e8d0
  endtag = strstr (content, "</GnupgKeyParms>");
Packit Service 30b792
  if (!endtag)
Packit Service 30b792
    endtag = content + strlen (content);
Packit Service 30b792
Packit d7e8d0
  /* FIXME: Check that there are no control statements inside.  */
Packit Service 30b792
  while (content < endtag
Packit Service 30b792
         && (content[0] == '\n'
Packit Service 30b792
             || (content[0] == '\r' && content[1] == '\n')))
Packit d7e8d0
    content++;
Packit d7e8d0
Packit d7e8d0
  return gpgme_data_new_from_mem (key_parameter, content,
Packit d7e8d0
				  endtag - content, 1);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
genkey_start (gpgme_ctx_t ctx, int synchronous, const char *parms,
Packit d7e8d0
	      gpgme_data_t pubkey, gpgme_data_t seckey)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
  err = _gpgme_op_reset (ctx, synchronous);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
Packit d7e8d0
			       sizeof (*opd), release_op_data);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  err = get_key_parameter (parms, &opd->key_parameter);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  _gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
Packit d7e8d0
Packit d7e8d0
  if (ctx->passphrase_cb)
Packit d7e8d0
    {
Packit d7e8d0
      err = _gpgme_engine_set_command_handler
Packit Service 30b792
        (ctx->engine, _gpgme_passphrase_command_handler, ctx);
Packit d7e8d0
      if (err)
Packit d7e8d0
        return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_genkey (ctx->engine,
Packit d7e8d0
                                  NULL, NULL, 0, 0, NULL, 0,
Packit d7e8d0
                                  opd->key_parameter,
Packit d7e8d0
				  ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
Packit d7e8d0
                                  pubkey, seckey);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Generate a new keypair and add it to the keyring.  PUBKEY and
Packit d7e8d0
   SECKEY should be null for now.  PARMS specifies what keys should be
Packit d7e8d0
   generated.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_genkey_start (gpgme_ctx_t ctx, const char *parms,
Packit d7e8d0
		       gpgme_data_t pubkey, gpgme_data_t seckey)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_genkey_start", ctx,
Packit d7e8d0
	      "pubkey=%p, seckey=%p", pubkey, seckey);
Packit Service 30b792
  TRACE_LOGBUF (parms, parms? strlen (parms):0);
Packit d7e8d0
Packit Service 30b792
  if (!ctx || !parms)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = genkey_start (ctx, 0, parms, pubkey, seckey);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Generate a new keypair and add it to the keyring.  PUBKEY and
Packit d7e8d0
   SECKEY should be null for now.  PARMS specifies what keys should be
Packit d7e8d0
   generated.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_genkey (gpgme_ctx_t ctx, const char *parms, gpgme_data_t pubkey,
Packit d7e8d0
		 gpgme_data_t seckey)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_genkey", ctx,
Packit d7e8d0
	      "pubkey=%p, seckey=%p", pubkey, seckey);
Packit Service 30b792
  TRACE_LOGBUF (parms, parms? strlen (parms):0);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = genkey_start (ctx, 1, parms, pubkey, seckey);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
createkey_start (gpgme_ctx_t ctx, int synchronous,
Packit d7e8d0
                 const char *userid, const char *algo,
Packit d7e8d0
                 unsigned long reserved, unsigned long expires,
Packit d7e8d0
                 gpgme_key_t anchorkey, unsigned int flags)
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_reset (ctx, synchronous);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  if (reserved || anchorkey || !userid)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_ARG);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
Packit d7e8d0
			       sizeof (*opd), release_op_data);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  _gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
Packit d7e8d0
Packit d7e8d0
  if (ctx->passphrase_cb)
Packit d7e8d0
    {
Packit d7e8d0
      err = _gpgme_engine_set_command_handler
Packit Service 30b792
        (ctx->engine, _gpgme_passphrase_command_handler, ctx);
Packit d7e8d0
      if (err)
Packit d7e8d0
        return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_genkey (ctx->engine,
Packit d7e8d0
                                  userid, algo, reserved, expires,
Packit d7e8d0
                                  anchorkey, flags,
Packit d7e8d0
                                  NULL,
Packit d7e8d0
				  ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
Packit d7e8d0
                                  NULL, NULL);
Packit d7e8d0
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_createkey_start (gpgme_ctx_t ctx, const char *userid, const char *algo,
Packit d7e8d0
                          unsigned long reserved, unsigned long expires,
Packit d7e8d0
                          gpgme_key_t anchorkey, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_createkey_start", ctx,
Packit d7e8d0
	      "userid='%s', algo='%s' flags=0x%x", userid, algo, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = createkey_start (ctx, 0,
Packit d7e8d0
                         userid, algo, reserved, expires, anchorkey, flags);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_createkey (gpgme_ctx_t ctx, const char *userid, const char *algo,
Packit d7e8d0
                    unsigned long reserved, unsigned long expires,
Packit d7e8d0
                    gpgme_key_t anchorkey, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_createkey", ctx,
Packit d7e8d0
	      "userid='%s', algo='%s' flags=0x%x", userid, algo, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = createkey_start (ctx, 1,
Packit d7e8d0
                         userid, algo, reserved, expires, anchorkey, flags);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
createsubkey_start (gpgme_ctx_t ctx, int synchronous,
Packit d7e8d0
                    gpgme_key_t key,
Packit d7e8d0
                    const char *algo,
Packit d7e8d0
                    unsigned long reserved, unsigned long expires,
Packit d7e8d0
                    unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
Packit d7e8d0
  if (ctx->protocol != GPGME_PROTOCOL_OPENPGP)
Packit d7e8d0
    return gpgme_error (GPG_ERR_UNSUPPORTED_PROTOCOL);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_reset (ctx, synchronous);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  if (reserved || !key)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_ARG);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
Packit d7e8d0
			       sizeof (*opd), release_op_data);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  _gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
Packit d7e8d0
Packit d7e8d0
  if (ctx->passphrase_cb)
Packit d7e8d0
    {
Packit d7e8d0
      err = _gpgme_engine_set_command_handler
Packit Service 30b792
        (ctx->engine, _gpgme_passphrase_command_handler, ctx);
Packit d7e8d0
      if (err)
Packit d7e8d0
        return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_genkey (ctx->engine,
Packit d7e8d0
                                  NULL, algo, reserved, expires,
Packit d7e8d0
                                  key, flags,
Packit d7e8d0
                                  NULL,
Packit d7e8d0
				  ctx->use_armor? GENKEY_EXTRAFLAG_ARMOR:0,
Packit d7e8d0
                                  NULL, NULL);
Packit d7e8d0
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Add a subkey to an existing KEY.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_createsubkey_start (gpgme_ctx_t ctx, gpgme_key_t key, const char *algo,
Packit d7e8d0
                             unsigned long reserved, unsigned long expires,
Packit d7e8d0
                             unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_createsubkey_start", ctx,
Packit d7e8d0
	      "key=%p, algo='%s' flags=0x%x", key, algo, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = createsubkey_start (ctx, 0, key, algo, reserved, expires, flags);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_createsubkey (gpgme_ctx_t ctx, gpgme_key_t key, const char *algo,
Packit d7e8d0
                       unsigned long reserved, unsigned long expires,
Packit d7e8d0
                       unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_createsubkey", ctx,
Packit d7e8d0
	      "key=%p, algo='%s' flags=0x%x", key, algo, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = createsubkey_start (ctx, 1, key, algo, reserved, expires, flags);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0

Packit d7e8d0
static gpgme_error_t
Packit d7e8d0
addrevuid_start (gpgme_ctx_t ctx, int synchronous, int extraflags,
Packit d7e8d0
                 gpgme_key_t key, const char *userid, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
  void *hook;
Packit d7e8d0
  op_data_t opd;
Packit d7e8d0
Packit d7e8d0
  if (ctx->protocol != GPGME_PROTOCOL_OPENPGP)
Packit d7e8d0
    return gpgme_error (GPG_ERR_UNSUPPORTED_PROTOCOL);
Packit d7e8d0
Packit d7e8d0
  if (!key || !userid)
Packit d7e8d0
    return gpg_error (GPG_ERR_INV_ARG);
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_reset (ctx, synchronous);
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  err = _gpgme_op_data_lookup (ctx, OPDATA_GENKEY, &hook,
Packit d7e8d0
			       sizeof (*opd), release_op_data);
Packit d7e8d0
  opd = hook;
Packit d7e8d0
  if (err)
Packit d7e8d0
    return err;
Packit d7e8d0
Packit d7e8d0
  opd->uidmode = extraflags? 2 : 1;
Packit d7e8d0
Packit d7e8d0
  _gpgme_engine_set_status_handler (ctx->engine, genkey_status_handler, ctx);
Packit d7e8d0
Packit d7e8d0
  if (ctx->passphrase_cb)
Packit d7e8d0
    {
Packit d7e8d0
      err = _gpgme_engine_set_command_handler
Packit Service 30b792
        (ctx->engine, _gpgme_passphrase_command_handler, ctx);
Packit d7e8d0
      if (err)
Packit d7e8d0
        return err;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
  return _gpgme_engine_op_genkey (ctx->engine,
Packit d7e8d0
                                  userid, NULL, 0, 0,
Packit d7e8d0
                                  key, flags,
Packit d7e8d0
                                  NULL,
Packit d7e8d0
				  extraflags,
Packit d7e8d0
                                  NULL, NULL);
Packit d7e8d0
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Add USERID to an existing KEY.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_adduid_start (gpgme_ctx_t ctx,
Packit d7e8d0
                       gpgme_key_t key, const char *userid, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_adduid_start", ctx,
Packit d7e8d0
	      "uid='%s' flags=0x%x", userid, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = addrevuid_start (ctx, 0, 0, key, userid, flags);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_adduid (gpgme_ctx_t ctx,
Packit d7e8d0
                 gpgme_key_t key, const char *userid, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_adduid", ctx,
Packit d7e8d0
	      "uid='%s' flags=0x%x", userid, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = addrevuid_start (ctx, 1, 0, key, userid, flags);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Revoke USERID from KEY.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_revuid_start (gpgme_ctx_t ctx,
Packit d7e8d0
                       gpgme_key_t key, const char *userid, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_revuid_start", ctx,
Packit d7e8d0
	      "uid='%s' flags=0x%x", userid, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = addrevuid_start (ctx, 0, GENKEY_EXTRAFLAG_REVOKE, key, userid, flags);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_revuid (gpgme_ctx_t ctx,
Packit d7e8d0
                 gpgme_key_t key, const char *userid, unsigned int flags)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_revuid", ctx,
Packit d7e8d0
	      "uid='%s' flags=0x%x", userid, flags);
Packit d7e8d0
Packit d7e8d0
  if (!ctx)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  err = addrevuid_start (ctx, 1, GENKEY_EXTRAFLAG_REVOKE, key, userid, flags);
Packit d7e8d0
  if (!err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* Set a flag on the USERID of KEY.  The only supported flag right now
Packit d7e8d0
 * is "primary" to mark the primary key.  */
Packit d7e8d0
static gpg_error_t
Packit d7e8d0
set_uid_flag (gpgme_ctx_t ctx, int synchronous,
Packit d7e8d0
              gpgme_key_t key, const char *userid,
Packit d7e8d0
              const char *name, const char *value)
Packit d7e8d0
{
Packit d7e8d0
  gpgme_error_t err;
Packit d7e8d0
Packit Service 30b792
  TRACE_BEG  (DEBUG_CTX, "gpgme_op_set_uid_flag", ctx,
Packit d7e8d0
	      "%d uid='%s' '%s'='%s'", synchronous, userid, name, value);
Packit d7e8d0
Packit d7e8d0
  if (!ctx || !name || !key || !userid)
Packit d7e8d0
    return TRACE_ERR (gpg_error (GPG_ERR_INV_ARG));
Packit d7e8d0
Packit d7e8d0
  if (!strcmp (name, "primary"))
Packit d7e8d0
    {
Packit d7e8d0
      if (value)
Packit d7e8d0
        err = gpg_error (GPG_ERR_INV_ARG);
Packit d7e8d0
      else
Packit d7e8d0
        err = addrevuid_start (ctx, synchronous,
Packit d7e8d0
                               GENKEY_EXTRAFLAG_SETPRIMARY, key, userid, 0);
Packit d7e8d0
    }
Packit d7e8d0
  else
Packit d7e8d0
    return err = gpg_error (GPG_ERR_UNKNOWN_NAME);
Packit d7e8d0
Packit d7e8d0
  if (synchronous && !err)
Packit d7e8d0
    err = _gpgme_wait_one (ctx);
Packit d7e8d0
  return TRACE_ERR (err);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* See set_uid_flag. */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_set_uid_flag_start (gpgme_ctx_t ctx,
Packit d7e8d0
                             gpgme_key_t key, const char *userid,
Packit d7e8d0
                             const char *name, const char *value)
Packit d7e8d0
{
Packit d7e8d0
  return set_uid_flag (ctx, 0, key, userid, name, value);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
Packit d7e8d0
/* See set_uid_flag.  This is the synchronous variant.  */
Packit d7e8d0
gpgme_error_t
Packit d7e8d0
gpgme_op_set_uid_flag (gpgme_ctx_t ctx,
Packit d7e8d0
                       gpgme_key_t key, const char *userid,
Packit d7e8d0
                       const char *name, const char *value)
Packit d7e8d0
{
Packit d7e8d0
  return set_uid_flag (ctx, 1, key, userid, name, value);
Packit d7e8d0
}