|           Line data    Source code 
       1             : /* gpgme.c - GnuPG Made Easy.
       2             :    Copyright (C) 2000 Werner Koch (dd9jn)
       3             :    Copyright (C) 2001, 2002, 2003, 2004, 2005, 2007, 2012,
       4             :                  2014, 2015 g10 Code GmbH
       5             : 
       6             :    This file is part of GPGME.
       7             : 
       8             :    GPGME is free software; you can redistribute it and/or modify it
       9             :    under the terms of the GNU Lesser General Public License as
      10             :    published by the Free Software Foundation; either version 2.1 of
      11             :    the License, or (at your option) any later version.
      12             : 
      13             :    GPGME is distributed in the hope that it will be useful, but
      14             :    WITHOUT ANY WARRANTY; without even the implied warranty of
      15             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      16             :    Lesser General Public License for more details.
      17             : 
      18             :    You should have received a copy of the GNU Lesser General Public
      19             :    License along with this program; if not, see <https://www.gnu.org/licenses/>.
      20             :  */
      21             : 
      22             : #if HAVE_CONFIG_H
      23             : #include <config.h>
      24             : #endif
      25             : #include <stdio.h>
      26             : #include <stdlib.h>
      27             : #include <string.h>
      28             : #include <assert.h>
      29             : #include <errno.h>
      30             : #ifdef HAVE_LOCALE_H
      31             : #include <locale.h>
      32             : #endif
      33             : 
      34             : #include "util.h"
      35             : #include "context.h"
      36             : #include "ops.h"
      37             : #include "wait.h"
      38             : #include "debug.h"
      39             : #include "priv-io.h"
      40             : #include "sys-util.h"
      41             : #include "mbox-util.h"
      42             : 
      43             : 
      44             : /* The default locale.  */
      45             : DEFINE_STATIC_LOCK (def_lc_lock);
      46             : static char *def_lc_ctype;
      47             : static char *def_lc_messages;
      48             : 
      49             : 
      50             : gpgme_error_t _gpgme_selftest = GPG_ERR_NOT_OPERATIONAL;
      51             : 
      52             : /* Protects all reference counters in result structures.  All other
      53             :    accesses to a result structure are read only.  */
      54             : DEFINE_STATIC_LOCK (result_ref_lock);
      55             : 
      56             : 
      57             : /* Set the global flag NAME to VALUE.  Return 0 on success.  Note that
      58             :    this function does not use gpgme_error and thus a non-zero return
      59             :    value merely means "error".  Certain flags may be set before
      60             :    gpgme_check_version is called.  See the manual for a description of
      61             :    supported flags.  The caller must assure that this function is
      62             :    called only by one thread at a time.  */
      63             : int
      64           0 : gpgme_set_global_flag (const char *name, const char *value)
      65             : {
      66           0 :   if (!name || !value)
      67           0 :     return -1;
      68           0 :   else if (!strcmp (name, "debug"))
      69           0 :     return _gpgme_debug_set_debug_envvar (value);
      70           0 :   else if (!strcmp (name, "disable-gpgconf"))
      71             :     {
      72           0 :       _gpgme_dirinfo_disable_gpgconf ();
      73           0 :       return 0;
      74             :     }
      75           0 :   else if (!strcmp (name, "require-gnupg"))
      76           0 :     return _gpgme_set_engine_minimal_version (value);
      77           0 :   else if (!strcmp (name, "gpgconf-name"))
      78           0 :     return _gpgme_set_default_gpgconf_name (value);
      79           0 :   else if (!strcmp (name, "gpg-name"))
      80           0 :     return _gpgme_set_default_gpg_name (value);
      81           0 :   else if (!strcmp (name, "w32-inst-dir"))
      82           0 :     return _gpgme_set_override_inst_dir (value);
      83             :   else
      84           0 :     return -1;
      85             : }
      86             : 
      87             : 
      88             : 
      89             : /* Create a new context as an environment for GPGME crypto
      90             :    operations.  */
      91             : gpgme_error_t
      92         716 : gpgme_new (gpgme_ctx_t *r_ctx)
      93             : {
      94             :   gpgme_error_t err;
      95             :   gpgme_ctx_t ctx;
      96         716 :   TRACE_BEG (DEBUG_CTX, "gpgme_new", r_ctx);
      97             : 
      98         715 :   if (_gpgme_selftest)
      99           0 :     return TRACE_ERR (_gpgme_selftest);
     100             : 
     101         715 :   if (!r_ctx)
     102           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     103             : 
     104         715 :   ctx = calloc (1, sizeof *ctx);
     105         715 :   if (!ctx)
     106           0 :     return TRACE_ERR (gpg_error_from_syserror ());
     107             : 
     108         715 :   INIT_LOCK (ctx->lock);
     109             : 
     110         715 :   err = _gpgme_engine_info_copy (&ctx->engine_info);
     111         715 :   if (!err && !ctx->engine_info)
     112           0 :     err = gpg_error (GPG_ERR_NO_ENGINE);
     113         715 :   if (err)
     114             :     {
     115           0 :       free (ctx);
     116           0 :       return TRACE_ERR (err);
     117             :     }
     118             : 
     119         715 :   ctx->keylist_mode = GPGME_KEYLIST_MODE_LOCAL;
     120         715 :   ctx->include_certs = GPGME_INCLUDE_CERTS_DEFAULT;
     121         715 :   ctx->protocol = GPGME_PROTOCOL_OpenPGP;
     122         715 :   ctx->sub_protocol = GPGME_PROTOCOL_DEFAULT;
     123         715 :   _gpgme_fd_table_init (&ctx->fdt);
     124             : 
     125         715 :   LOCK (def_lc_lock);
     126         715 :   if (def_lc_ctype)
     127             :     {
     128         138 :       ctx->lc_ctype = strdup (def_lc_ctype);
     129         138 :       if (!ctx->lc_ctype)
     130             :         {
     131           0 :           int saved_err = gpg_error_from_syserror ();
     132           0 :           UNLOCK (def_lc_lock);
     133           0 :           _gpgme_engine_info_release (ctx->engine_info);
     134           0 :           free (ctx);
     135           0 :           return TRACE_ERR (saved_err);
     136             :         }
     137             :     }
     138             :   else
     139         577 :     def_lc_ctype = NULL;
     140             : 
     141         715 :   if (def_lc_messages)
     142             :     {
     143         138 :       ctx->lc_messages = strdup (def_lc_messages);
     144         138 :       if (!ctx->lc_messages)
     145             :         {
     146           0 :           int saved_err = gpg_error_from_syserror ();
     147           0 :           UNLOCK (def_lc_lock);
     148           0 :           if (ctx->lc_ctype)
     149           0 :             free (ctx->lc_ctype);
     150           0 :           _gpgme_engine_info_release (ctx->engine_info);
     151           0 :           free (ctx);
     152           0 :           return TRACE_ERR (saved_err);
     153             :         }
     154             :     }
     155             :   else
     156         577 :     def_lc_messages = NULL;
     157         715 :   UNLOCK (def_lc_lock);
     158             : 
     159         715 :   *r_ctx = ctx;
     160             : 
     161         715 :   return TRACE_SUC1 ("ctx=%p", ctx);
     162             : }
     163             : 
     164             : 
     165             : gpgme_error_t
     166          32 : _gpgme_cancel_with_err (gpgme_ctx_t ctx, gpg_error_t ctx_err,
     167             :                         gpg_error_t op_err)
     168             : {
     169             :   gpgme_error_t err;
     170             :   struct gpgme_io_event_done_data data;
     171             : 
     172          32 :   TRACE_BEG2 (DEBUG_CTX, "_gpgme_cancel_with_err", ctx, "ctx_err=%i, op_err=%i",
     173             :               ctx_err, op_err);
     174             : 
     175          32 :   if (ctx_err)
     176             :     {
     177          30 :       err = _gpgme_engine_cancel (ctx->engine);
     178          30 :       if (err)
     179           0 :         return TRACE_ERR (err);
     180             :     }
     181             :   else
     182             :     {
     183           2 :       err = _gpgme_engine_cancel_op (ctx->engine);
     184           2 :       if (err)
     185           0 :         return TRACE_ERR (err);
     186             :     }
     187             : 
     188          32 :   data.err = ctx_err;
     189          32 :   data.op_err = op_err;
     190             : 
     191          32 :   _gpgme_engine_io_event (ctx->engine, GPGME_EVENT_DONE, &data);
     192             : 
     193          32 :   return TRACE_ERR (0);
     194             : }
     195             : 
     196             : 
     197             : /* Cancel a pending asynchronous operation.  */
     198             : gpgme_error_t
     199           0 : gpgme_cancel (gpgme_ctx_t ctx)
     200             : {
     201             :   gpg_error_t err;
     202             : 
     203           0 :   TRACE_BEG (DEBUG_CTX, "gpgme_cancel", ctx);
     204             : 
     205           0 :   if (!ctx)
     206           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     207             : 
     208           0 :   err = _gpgme_cancel_with_err (ctx, gpg_error (GPG_ERR_CANCELED), 0);
     209             : 
     210           0 :   return TRACE_ERR (err);
     211             : }
     212             : 
     213             : 
     214             : /* Cancel a pending operation asynchronously.  */
     215             : gpgme_error_t
     216          17 : gpgme_cancel_async (gpgme_ctx_t ctx)
     217             : {
     218          17 :   TRACE_BEG (DEBUG_CTX, "gpgme_cancel_async", ctx);
     219             : 
     220          17 :   if (!ctx)
     221           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     222             : 
     223          17 :   LOCK (ctx->lock);
     224          17 :   ctx->canceled = 1;
     225          17 :   UNLOCK (ctx->lock);
     226             : 
     227          17 :   return TRACE_ERR (0);
     228             : }
     229             : 
     230             : 
     231             : /* Release all resources associated with the given context.  */
     232             : void
     233         701 : gpgme_release (gpgme_ctx_t ctx)
     234             : {
     235         701 :   TRACE (DEBUG_CTX, "gpgme_release", ctx);
     236             : 
     237         701 :   if (!ctx)
     238           0 :     return;
     239             : 
     240         701 :   _gpgme_engine_release (ctx->engine);
     241         701 :   ctx->engine = NULL;
     242         701 :   _gpgme_fd_table_deinit (&ctx->fdt);
     243         701 :   _gpgme_release_result (ctx);
     244         701 :   _gpgme_signers_clear (ctx);
     245         701 :   _gpgme_sig_notation_clear (ctx);
     246         701 :   free (ctx->sender);
     247         701 :   free (ctx->signers);
     248         701 :   free (ctx->lc_ctype);
     249         701 :   free (ctx->lc_messages);
     250         701 :   free (ctx->override_session_key);
     251         701 :   free (ctx->request_origin);
     252         701 :   free (ctx->auto_key_locate);
     253         701 :   free (ctx->trust_model);
     254         701 :   _gpgme_engine_info_release (ctx->engine_info);
     255         701 :   ctx->engine_info = NULL;
     256         701 :   DESTROY_LOCK (ctx->lock);
     257         701 :   free (ctx);
     258             : }
     259             : 
     260             : 
     261             : void
     262           0 : gpgme_result_ref (void *result)
     263             : {
     264             :   struct ctx_op_data *data;
     265             : 
     266           0 :   if (! result)
     267           0 :     return;
     268             : 
     269           0 :   data = (void*)((char*)result - sizeof (struct ctx_op_data));
     270             : 
     271           0 :   assert (data->magic == CTX_OP_DATA_MAGIC);
     272             : 
     273           0 :   LOCK (result_ref_lock);
     274           0 :   data->references++;
     275           0 :   UNLOCK (result_ref_lock);
     276             : }
     277             : 
     278             : 
     279             : void
     280         919 : gpgme_result_unref (void *result)
     281             : {
     282             :   struct ctx_op_data *data;
     283             : 
     284         919 :   if (! result)
     285           0 :     return;
     286             : 
     287         919 :   data = (void*)((char*)result - sizeof (struct ctx_op_data));
     288             : 
     289         919 :   assert (data->magic == CTX_OP_DATA_MAGIC);
     290             : 
     291         919 :   LOCK (result_ref_lock);
     292         919 :   if (--data->references)
     293             :     {
     294           0 :       UNLOCK (result_ref_lock);
     295           0 :       return;
     296             :     }
     297         919 :   UNLOCK (result_ref_lock);
     298             : 
     299         919 :   if (data->cleanup)
     300         875 :     (*data->cleanup) (data->hook);
     301         919 :   free (data);
     302             : }
     303             : 
     304             : 
     305             : void
     306        1664 : _gpgme_release_result (gpgme_ctx_t ctx)
     307             : {
     308        1664 :   struct ctx_op_data *data = ctx->op_data;
     309             : 
     310        4247 :   while (data)
     311             :     {
     312         919 :       struct ctx_op_data *next_data = data->next;
     313         919 :       data->next = NULL;
     314         919 :       gpgme_result_unref (data->hook);
     315         919 :       data = next_data;
     316             :     }
     317        1664 :   ctx->op_data = NULL;
     318        1664 : }
     319             : 
     320             : 
     321             : gpgme_error_t
     322         493 : gpgme_set_protocol (gpgme_ctx_t ctx, gpgme_protocol_t protocol)
     323             : {
     324         493 :   TRACE_BEG2 (DEBUG_CTX, "gpgme_set_protocol", ctx, "protocol=%i (%s)",
     325             :               protocol, gpgme_get_protocol_name (protocol)
     326             :               ? gpgme_get_protocol_name (protocol) : "invalid");
     327             : 
     328         493 :   if (protocol != GPGME_PROTOCOL_OpenPGP
     329          22 :       && protocol != GPGME_PROTOCOL_CMS
     330          14 :       && protocol != GPGME_PROTOCOL_GPGCONF
     331          14 :       && protocol != GPGME_PROTOCOL_ASSUAN
     332           0 :       && protocol != GPGME_PROTOCOL_G13
     333           0 :       && protocol != GPGME_PROTOCOL_UISERVER
     334           0 :       && protocol != GPGME_PROTOCOL_SPAWN)
     335           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     336             : 
     337         493 :   if (!ctx)
     338           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     339             : 
     340         493 :   if (ctx->protocol != protocol)
     341             :     {
     342             :       /* Shut down the engine when switching protocols.  */
     343          22 :       if (ctx->engine)
     344             :         {
     345          12 :           TRACE_LOG1 ("releasing ctx->engine=%p", ctx->engine);
     346          12 :           _gpgme_engine_release (ctx->engine);
     347          12 :           ctx->engine = NULL;
     348             :         }
     349             : 
     350          22 :       ctx->protocol = protocol;
     351             :     }
     352         493 :   return TRACE_ERR (0);
     353             : }
     354             : 
     355             : 
     356             : gpgme_protocol_t
     357         538 : gpgme_get_protocol (gpgme_ctx_t ctx)
     358             : {
     359         538 :   TRACE2 (DEBUG_CTX, "gpgme_get_protocol", ctx,
     360             :           "ctx->protocol=%i (%s)", ctx->protocol,
     361             :           gpgme_get_protocol_name (ctx->protocol)
     362             :           ? gpgme_get_protocol_name (ctx->protocol) : "invalid");
     363             : 
     364         538 :   return ctx->protocol;
     365             : }
     366             : 
     367             : 
     368             : gpgme_error_t
     369           0 : gpgme_set_sub_protocol (gpgme_ctx_t ctx, gpgme_protocol_t protocol)
     370             : {
     371           0 :   TRACE2 (DEBUG_CTX, "gpgme_set_sub_protocol", ctx, "protocol=%i (%s)",
     372             :           protocol, gpgme_get_protocol_name (protocol)
     373             :           ? gpgme_get_protocol_name (protocol) : "invalid");
     374             : 
     375           0 :   if (!ctx)
     376           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     377             : 
     378           0 :   ctx->sub_protocol = protocol;
     379           0 :   return 0;
     380             : }
     381             : 
     382             : 
     383             : gpgme_protocol_t
     384           0 : gpgme_get_sub_protocol (gpgme_ctx_t ctx)
     385             : {
     386           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_sub_protocol", ctx,
     387             :           "ctx->sub_protocol=%i (%s)", ctx->sub_protocol,
     388             :           gpgme_get_protocol_name (ctx->sub_protocol)
     389             :           ? gpgme_get_protocol_name (ctx->sub_protocol) : "invalid");
     390             : 
     391           0 :   return ctx->sub_protocol;
     392             : }
     393             : 
     394             : 
     395             : const char *
     396        2916 : gpgme_get_protocol_name (gpgme_protocol_t protocol)
     397             : {
     398        2916 :   switch (protocol)
     399             :     {
     400             :     case GPGME_PROTOCOL_OpenPGP:
     401        2808 :       return "OpenPGP";
     402             : 
     403             :     case GPGME_PROTOCOL_CMS:
     404          20 :       return "CMS";
     405             : 
     406             :     case GPGME_PROTOCOL_GPGCONF:
     407           4 :       return "GPGCONF";
     408             : 
     409             :     case GPGME_PROTOCOL_ASSUAN:
     410          84 :       return "Assuan";
     411             : 
     412             :     case GPGME_PROTOCOL_G13:
     413           0 :       return "G13";
     414             : 
     415             :     case GPGME_PROTOCOL_UISERVER:
     416           0 :       return "UIServer";
     417             : 
     418             :     case GPGME_PROTOCOL_SPAWN:
     419           0 :       return "Spawn";
     420             : 
     421             :     case GPGME_PROTOCOL_DEFAULT:
     422           0 :       return "default";
     423             : 
     424             :     case GPGME_PROTOCOL_UNKNOWN:
     425           0 :       return "unknown";
     426             : 
     427             :     default:
     428           0 :       return NULL;
     429             :     }
     430             : }
     431             : 
     432             : 
     433             : /* Store the sender's address in the context.  ADDRESS is addr-spec of
     434             :  * mailbox but my also be a complete mailbox, in which case this
     435             :  * function extracts the addr-spec from it.  Returns 0 on success or
     436             :  * an error code if no valid addr-spec could be extracted from
     437             :  * ADDRESS.  */
     438             : gpgme_error_t
     439           5 : gpgme_set_sender (gpgme_ctx_t ctx, const char *address)
     440             : {
     441           5 :   char *p = NULL;
     442             : 
     443           5 :   TRACE_BEG1 (DEBUG_CTX, "gpgme_set_sender", ctx, "sender='%s'",
     444             :               address?address:"(null)");
     445             : 
     446           5 :   if (!ctx || (address && !(p = _gpgme_mailbox_from_userid (address))))
     447           1 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
     448             : 
     449           4 :   free (ctx->sender);
     450           4 :   ctx->sender = p;
     451           4 :   return TRACE_ERR (0);
     452             : }
     453             : 
     454             : 
     455             : /* Return the sender's address (addr-spec part) from the context or
     456             :  * NULL if none was set.  The returned value is valid as long as the
     457             :  * CTX is valid and gpgme_set_sender has not been used.  */
     458             : const char *
     459           4 : gpgme_get_sender (gpgme_ctx_t ctx)
     460             : {
     461           4 :   TRACE1 (DEBUG_CTX, "gpgme_get_sender", ctx, "sender='%s'",
     462             :           ctx?ctx->sender:"");
     463             : 
     464           4 :   return ctx->sender;
     465             : }
     466             : 
     467             : 
     468             : /* Enable or disable the use of an ascii armor for all output.  */
     469             : void
     470         238 : gpgme_set_armor (gpgme_ctx_t ctx, int use_armor)
     471             : {
     472         238 :   TRACE2 (DEBUG_CTX, "gpgme_set_armor", ctx, "use_armor=%i (%s)",
     473             :           use_armor, use_armor ? "yes" : "no");
     474             : 
     475         238 :   if (!ctx)
     476           0 :     return;
     477             : 
     478         238 :   ctx->use_armor = !!use_armor;
     479             : }
     480             : 
     481             : 
     482             : /* Return the state of the armor flag.  */
     483             : int
     484           0 : gpgme_get_armor (gpgme_ctx_t ctx)
     485             : {
     486           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_armor", ctx, "ctx->use_armor=%i (%s)",
     487             :           ctx->use_armor, ctx->use_armor ? "yes" : "no");
     488           0 :   return ctx->use_armor;
     489             : }
     490             : 
     491             : 
     492             : /* Set the flag NAME for CTX to VALUE.  Please consult the manual for
     493             :  * a description of the flags.
     494             :  */
     495             : gpgme_error_t
     496           4 : gpgme_set_ctx_flag (gpgme_ctx_t ctx, const char *name, const char *value)
     497             : {
     498           4 :   gpgme_error_t err = 0;
     499             :   int abool;
     500             : 
     501           4 :   TRACE2 (DEBUG_CTX, "gpgme_set_ctx_flag", ctx,
     502             :           "name='%s' value='%s'",
     503             :           name? name:"(null)", value?value:"(null)");
     504             : 
     505           4 :   abool = (value && *value)? !!atoi (value) : 0;
     506             : 
     507           4 :   if (!ctx || !name || !value)
     508           0 :     err = gpg_error (GPG_ERR_INV_VALUE);
     509           4 :   else if (!strcmp (name, "redraw"))
     510             :     {
     511           0 :       ctx->redraw_suggested = abool;
     512             :     }
     513           4 :   else if (!strcmp (name, "full-status"))
     514             :     {
     515           4 :       ctx->full_status = abool;
     516             :     }
     517           0 :   else if (!strcmp (name, "raw-description"))
     518             :     {
     519           0 :       ctx->raw_description = abool;
     520             :     }
     521           0 :   else if (!strcmp (name, "export-session-key"))
     522             :     {
     523           0 :       ctx->export_session_keys = abool;
     524             :     }
     525           0 :   else if (!strcmp (name, "override-session-key"))
     526             :     {
     527           0 :       free (ctx->override_session_key);
     528           0 :       ctx->override_session_key = strdup (value);
     529           0 :       if (!ctx->override_session_key)
     530           0 :         err = gpg_error_from_syserror ();
     531             :     }
     532           0 :   else if (!strcmp (name, "auto-key-retrieve"))
     533             :     {
     534           0 :       ctx->auto_key_retrieve = abool;
     535             :     }
     536           0 :   else if (!strcmp (name, "request-origin"))
     537             :     {
     538           0 :       free (ctx->request_origin);
     539           0 :       ctx->request_origin = strdup (value);
     540           0 :       if (!ctx->request_origin)
     541           0 :         err = gpg_error_from_syserror ();
     542             :     }
     543           0 :   else if (!strcmp (name, "no-symkey-cache"))
     544             :     {
     545           0 :       ctx->no_symkey_cache = abool;
     546             :     }
     547           0 :   else if (!strcmp (name, "ignore-mdc-error"))
     548             :     {
     549           0 :       ctx->ignore_mdc_error = abool;
     550             :     }
     551           0 :   else if (!strcmp (name, "auto-key-locate"))
     552             :     {
     553           0 :       free (ctx->auto_key_locate);
     554           0 :       ctx->auto_key_locate = strdup (value);
     555           0 :       if (!ctx->auto_key_locate)
     556           0 :         err = gpg_error_from_syserror ();
     557             :     }
     558           0 :   else if (!strcmp (name, "trust-model"))
     559             :     {
     560           0 :       free (ctx->trust_model);
     561           0 :       ctx->trust_model = strdup (value);
     562           0 :       if (!ctx->trust_model)
     563           0 :         err = gpg_error_from_syserror ();
     564             :     }
     565             :   else
     566           0 :     err = gpg_error (GPG_ERR_UNKNOWN_NAME);
     567             : 
     568           4 :   return err;
     569             : }
     570             : 
     571             : 
     572             : /* Get the context flag named NAME.  See gpgme_set_ctx_flag for a list
     573             :  * of valid names.  If the NAME is unknown NULL is returned.  For a
     574             :  * boolean flag an empty string is returned for False and the string
     575             :  * "1" for True; thus either atoi or a simple string test can be
     576             :  * used.  */
     577             : const char *
     578           0 : gpgme_get_ctx_flag (gpgme_ctx_t ctx, const char *name)
     579             : {
     580           0 :   if (!ctx || !name)
     581           0 :     return NULL;
     582           0 :   else if (!strcmp (name, "redraw"))
     583             :     {
     584           0 :       return ctx->redraw_suggested? "1":"";
     585             :     }
     586           0 :   else if (!strcmp (name, "full-status"))
     587             :     {
     588           0 :       return ctx->full_status? "1":"";
     589             :     }
     590           0 :   else if (!strcmp (name, "raw-description"))
     591             :     {
     592           0 :       return ctx->raw_description? "1":"";
     593             :     }
     594           0 :   else if (!strcmp (name, "export-session-key"))
     595             :     {
     596           0 :       return ctx->export_session_keys? "1":"";
     597             :     }
     598           0 :   else if (!strcmp (name, "override-session-key"))
     599             :     {
     600           0 :       return ctx->override_session_key? ctx->override_session_key : "";
     601             :     }
     602           0 :   else if (!strcmp (name, "auto-key-retrieve"))
     603             :     {
     604           0 :       return ctx->auto_key_retrieve? "1":"";
     605             :     }
     606           0 :   else if (!strcmp (name, "request-origin"))
     607             :     {
     608           0 :       return ctx->request_origin? ctx->request_origin : "";
     609             :     }
     610           0 :   else if (!strcmp (name, "no-symkey-cache"))
     611             :     {
     612           0 :       return ctx->no_symkey_cache? "1":"";
     613             :     }
     614           0 :   else if (!strcmp (name, "ignore-mdc-error"))
     615             :     {
     616           0 :       return ctx->ignore_mdc_error? "1":"";
     617             :     }
     618           0 :   else if (!strcmp (name, "auto-key-locate"))
     619             :     {
     620           0 :       return ctx->auto_key_locate? ctx->auto_key_locate : "";
     621             :     }
     622             :   else
     623           0 :     return NULL;
     624             : }
     625             : 
     626             : 
     627             : /* Enable or disable the use of the special textmode.  Textmode is for
     628             :   example used for the RFC2015 signatures; note that the updated RFC
     629             :   3156 mandates that the MUA does some preparations so that textmode
     630             :   is not needed anymore.  */
     631             : void
     632         199 : gpgme_set_textmode (gpgme_ctx_t ctx, int use_textmode)
     633             : {
     634         199 :   TRACE2 (DEBUG_CTX, "gpgme_set_textmode", ctx, "use_textmode=%i (%s)",
     635             :           use_textmode, use_textmode ? "yes" : "no");
     636             : 
     637         199 :   if (!ctx)
     638           0 :     return;
     639             : 
     640         199 :   ctx->use_textmode = !!use_textmode;
     641             : }
     642             : 
     643             : /* Return the state of the textmode flag.  */
     644             : int
     645           0 : gpgme_get_textmode (gpgme_ctx_t ctx)
     646             : {
     647           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_textmode", ctx, "ctx->use_textmode=%i (%s)",
     648             :           ctx->use_textmode, ctx->use_textmode ? "yes" : "no");
     649           0 :   return ctx->use_textmode;
     650             : }
     651             : 
     652             : 
     653             : /* Enable offline mode for this context. In offline mode dirmngr
     654             :   will be disabled. */
     655             : void
     656         167 : gpgme_set_offline (gpgme_ctx_t ctx, int offline)
     657             : {
     658         167 :   TRACE2 (DEBUG_CTX, "gpgme_set_offline", ctx, "offline=%i (%s)",
     659             :           offline, offline ? "yes" : "no");
     660             : 
     661         167 :   if (!ctx)
     662           0 :     return;
     663             : 
     664         167 :   ctx->offline = !!offline;
     665             : }
     666             : 
     667             : /* Return the state of the offline flag.  */
     668             : int
     669           0 : gpgme_get_offline (gpgme_ctx_t ctx)
     670             : {
     671           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_offline", ctx, "ctx->offline=%i (%s)",
     672             :           ctx->offline, ctx->offline ? "yes" : "no");
     673           0 :   return ctx->offline;
     674             : }
     675             : 
     676             : 
     677             : /* Set the number of certifications to include in an S/MIME message.
     678             :    The default is GPGME_INCLUDE_CERTS_DEFAULT.  -1 means all certs,
     679             :    and -2 means all certs except the root cert.  */
     680             : void
     681           0 : gpgme_set_include_certs (gpgme_ctx_t ctx, int nr_of_certs)
     682             : {
     683           0 :   if (!ctx)
     684           0 :     return;
     685             : 
     686           0 :   if (nr_of_certs == GPGME_INCLUDE_CERTS_DEFAULT)
     687           0 :     ctx->include_certs = GPGME_INCLUDE_CERTS_DEFAULT;
     688           0 :   else if (nr_of_certs < -2)
     689           0 :     ctx->include_certs = -2;
     690             :   else
     691           0 :     ctx->include_certs = nr_of_certs;
     692             : 
     693           0 :   TRACE2 (DEBUG_CTX, "gpgme_set_include_certs", ctx, "nr_of_certs=%i%s",
     694             :           nr_of_certs, nr_of_certs == ctx->include_certs ? "" : " (-2)");
     695             : }
     696             : 
     697             : 
     698             : /* Get the number of certifications to include in an S/MIME
     699             :    message.  */
     700             : int
     701           0 : gpgme_get_include_certs (gpgme_ctx_t ctx)
     702             : {
     703           0 :   TRACE1 (DEBUG_CTX, "gpgme_get_include_certs", ctx, "ctx->include_certs=%i",
     704             :           ctx->include_certs);
     705           0 :   return ctx->include_certs;
     706             : }
     707             : 
     708             : 
     709             : /* This function changes the default behaviour of the keylisting
     710             :    functions.  MODE is a bitwise-OR of the GPGME_KEYLIST_* flags.  The
     711             :    default mode is GPGME_KEYLIST_MODE_LOCAL.  */
     712             : gpgme_error_t
     713         308 : gpgme_set_keylist_mode (gpgme_ctx_t ctx, gpgme_keylist_mode_t mode)
     714             : {
     715         308 :   TRACE1 (DEBUG_CTX, "gpgme_set_keylist_mode", ctx, "keylist_mode=0x%x",
     716             :           mode);
     717             : 
     718         308 :   if (!ctx)
     719           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     720             : 
     721         308 :   ctx->keylist_mode = mode;
     722         308 :   return 0;
     723             : }
     724             : 
     725             : /* This function returns the default behaviour of the keylisting
     726             :    functions.  */
     727             : gpgme_keylist_mode_t
     728         265 : gpgme_get_keylist_mode (gpgme_ctx_t ctx)
     729             : {
     730         265 :   TRACE1 (DEBUG_CTX, "gpgme_get_keylist_mode", ctx,
     731             :           "ctx->keylist_mode=0x%x", ctx->keylist_mode);
     732         265 :   return ctx->keylist_mode;
     733             : }
     734             : 
     735             : 
     736             : /* Set the pinentry mode for CTX to MODE. */
     737             : gpgme_error_t
     738         223 : gpgme_set_pinentry_mode (gpgme_ctx_t ctx, gpgme_pinentry_mode_t mode)
     739             : {
     740         223 :   TRACE1 (DEBUG_CTX, "gpgme_set_pinentry_mode", ctx, "pinentry_mode=%u",
     741             :           (unsigned int)mode);
     742             : 
     743         223 :   if (!ctx)
     744           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     745             : 
     746         223 :   switch (mode)
     747             :     {
     748             :     case GPGME_PINENTRY_MODE_DEFAULT:
     749             :     case GPGME_PINENTRY_MODE_ASK:
     750             :     case GPGME_PINENTRY_MODE_CANCEL:
     751             :     case GPGME_PINENTRY_MODE_ERROR:
     752             :     case GPGME_PINENTRY_MODE_LOOPBACK:
     753         223 :       break;
     754             :     default:
     755           0 :       return gpg_error (GPG_ERR_INV_VALUE);
     756             :     }
     757             : 
     758         223 :   ctx->pinentry_mode = mode;
     759         223 :   return 0;
     760             : }
     761             : 
     762             : 
     763             : /* Get the pinentry mode of CTX.  */
     764             : gpgme_pinentry_mode_t
     765          12 : gpgme_get_pinentry_mode (gpgme_ctx_t ctx)
     766             : {
     767          12 :   TRACE1 (DEBUG_CTX, "gpgme_get_pinentry_mode", ctx,
     768             :           "ctx->pinentry_mode=%u", (unsigned int)ctx->pinentry_mode);
     769          12 :   return ctx->pinentry_mode;
     770             : }
     771             : 
     772             : 
     773             : /* This function sets a callback function to be used to pass a
     774             :    passphrase to gpg.  */
     775             : void
     776         250 : gpgme_set_passphrase_cb (gpgme_ctx_t ctx, gpgme_passphrase_cb_t cb,
     777             :                          void *cb_value)
     778             : {
     779         250 :   TRACE2 (DEBUG_CTX, "gpgme_set_passphrase_cb", ctx,
     780             :           "passphrase_cb=%p/%p", cb, cb_value);
     781             : 
     782         250 :   if (!ctx)
     783           0 :     return;
     784             : 
     785         250 :   ctx->passphrase_cb = cb;
     786         250 :   ctx->passphrase_cb_value = cb_value;
     787             : }
     788             : 
     789             : 
     790             : /* This function returns the callback function to be used to pass a
     791             :    passphrase to the crypto engine.  */
     792             : void
     793           0 : gpgme_get_passphrase_cb (gpgme_ctx_t ctx, gpgme_passphrase_cb_t *r_cb,
     794             :                          void **r_cb_value)
     795             : {
     796           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_passphrase_cb", ctx,
     797             :           "ctx->passphrase_cb=%p/%p",
     798             :           ctx->passphrase_cb, ctx->passphrase_cb_value);
     799           0 :   if (r_cb)
     800           0 :     *r_cb = ctx->passphrase_cb;
     801           0 :   if (r_cb_value)
     802           0 :     *r_cb_value = ctx->passphrase_cb_value;
     803           0 : }
     804             : 
     805             : 
     806             : /* This function sets a callback function to be used as a progress
     807             :    indicator.  */
     808             : void
     809         210 : gpgme_set_progress_cb (gpgme_ctx_t ctx, gpgme_progress_cb_t cb, void *cb_value)
     810             : {
     811         210 :   TRACE2 (DEBUG_CTX, "gpgme_set_progress_cb", ctx, "progress_cb=%p/%p",
     812             :           cb, cb_value);
     813             : 
     814         210 :   if (!ctx)
     815           0 :     return;
     816             : 
     817         210 :   ctx->progress_cb = cb;
     818         210 :   ctx->progress_cb_value = cb_value;
     819             : }
     820             : 
     821             : 
     822             : /* This function returns the callback function to be used as a
     823             :    progress indicator.  */
     824             : void
     825           0 : gpgme_get_progress_cb (gpgme_ctx_t ctx, gpgme_progress_cb_t *r_cb,
     826             :                        void **r_cb_value)
     827             : {
     828           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_progress_cb", ctx, "ctx->progress_cb=%p/%p",
     829             :           ctx->progress_cb, ctx->progress_cb_value);
     830           0 :   if (r_cb)
     831           0 :     *r_cb = ctx->progress_cb;
     832           0 :   if (r_cb_value)
     833           0 :     *r_cb_value = ctx->progress_cb_value;
     834           0 : }
     835             : 
     836             : 
     837             : /* This function sets a callback function to be used as a status
     838             :    message forwarder.  */
     839             : void
     840         163 : gpgme_set_status_cb (gpgme_ctx_t ctx, gpgme_status_cb_t cb, void *cb_value)
     841             : {
     842         163 :   TRACE2 (DEBUG_CTX, "gpgme_set_status_cb", ctx, "status_cb=%p/%p",
     843             :           cb, cb_value);
     844             : 
     845         163 :   if (!ctx)
     846           0 :     return;
     847             : 
     848         163 :   ctx->status_cb = cb;
     849         163 :   ctx->status_cb_value = cb_value;
     850             : }
     851             : 
     852             : 
     853             : /* This function returns the callback function to be used as a
     854             :    status message forwarder.  */
     855             : void
     856           0 : gpgme_get_status_cb (gpgme_ctx_t ctx, gpgme_status_cb_t *r_cb,
     857             :                        void **r_cb_value)
     858             : {
     859           0 :   TRACE2 (DEBUG_CTX, "gpgme_get_status_cb", ctx, "ctx->status_cb=%p/%p",
     860             :           ctx ? ctx->status_cb : NULL, ctx ? ctx->status_cb_value : NULL);
     861             : 
     862           0 :   if (r_cb)
     863           0 :     *r_cb = NULL;
     864             : 
     865           0 :   if (r_cb_value)
     866           0 :     *r_cb_value = NULL;
     867             : 
     868           0 :   if (!ctx || !ctx->status_cb)
     869           0 :     return;
     870             : 
     871           0 :   if (r_cb)
     872           0 :     *r_cb = ctx->status_cb;
     873           0 :   if (r_cb_value)
     874           0 :     *r_cb_value = ctx->status_cb_value;
     875             : }
     876             : 
     877             : 
     878             : /* Set the I/O callback functions for CTX to IO_CBS.  */
     879             : void
     880           1 : gpgme_set_io_cbs (gpgme_ctx_t ctx, gpgme_io_cbs_t io_cbs)
     881             : {
     882           1 :   if (!ctx)
     883           0 :     return;
     884             : 
     885           1 :   if (io_cbs)
     886             :     {
     887           1 :       TRACE6 (DEBUG_CTX, "gpgme_set_io_cbs", ctx,
     888             :               "io_cbs=%p (add=%p/%p, remove=%p, event=%p/%p",
     889             :               io_cbs, io_cbs->add, io_cbs->add_priv, io_cbs->remove,
     890             :               io_cbs->event, io_cbs->event_priv);
     891           1 :       ctx->io_cbs = *io_cbs;
     892             :     }
     893             :   else
     894             :     {
     895           0 :       TRACE1 (DEBUG_CTX, "gpgme_set_io_cbs", ctx,
     896             :               "io_cbs=%p (default)", io_cbs);
     897           0 :       ctx->io_cbs.add = NULL;
     898           0 :       ctx->io_cbs.add_priv = NULL;
     899           0 :       ctx->io_cbs.remove = NULL;
     900           0 :       ctx->io_cbs.event = NULL;
     901           0 :       ctx->io_cbs.event_priv = NULL;
     902             :     }
     903             : }
     904             : 
     905             : 
     906             : /* This function provides access to the internal read function; it is
     907             :    normally not used.  */
     908             : gpgme_ssize_t
     909           0 : gpgme_io_read (int fd, void *buffer, size_t count)
     910             : {
     911             :   int ret;
     912           0 :   TRACE_BEG2 (DEBUG_GLOBAL, "gpgme_io_read", fd,
     913             :               "buffer=%p, count=%u", buffer, count);
     914             : 
     915           0 :   ret = _gpgme_io_read (fd, buffer, count);
     916             : 
     917           0 :   return TRACE_SYSRES (ret);
     918             : }
     919             : 
     920             : 
     921             : /* This function provides access to the internal write function.  It
     922             :    is to be used by user callbacks to return data to gpgme.  See
     923             :    gpgme_passphrase_cb_t and gpgme_edit_cb_t.  */
     924             : gpgme_ssize_t
     925          26 : gpgme_io_write (int fd, const void *buffer, size_t count)
     926             : {
     927             :   int ret;
     928          26 :   TRACE_BEG2 (DEBUG_GLOBAL, "gpgme_io_write", fd,
     929             :               "buffer=%p, count=%u", buffer, count);
     930             : 
     931          26 :   ret = _gpgme_io_write (fd, buffer, count);
     932             : 
     933          26 :   return TRACE_SYSRES (ret);
     934             : }
     935             : 
     936             : /* This function provides access to the internal write function.  It
     937             :    is to be used by user callbacks to return data to gpgme.  See
     938             :    gpgme_passphrase_cb_t and gpgme_edit_cb_t.  Note that this is a
     939             :    variant of gpgme_io_write which guarantees that all COUNT bytes are
     940             :    written or an error is return.  Returns: 0 on success or -1 on
     941             :    error and the sets errno. */
     942             : int
     943          14 : gpgme_io_writen (int fd, const void *buffer_arg, size_t count)
     944             : {
     945          14 :   const char *buffer = buffer_arg;
     946          14 :   int ret = 0;
     947          14 :   TRACE_BEG2 (DEBUG_GLOBAL, "gpgme_io_writen", fd,
     948             :               "buffer=%p, count=%u", buffer, count);
     949          42 :   while (count)
     950             :     {
     951          14 :       ret = _gpgme_io_write (fd, buffer, count);
     952          14 :       if (ret < 0)
     953           0 :         break;
     954          14 :       buffer += ret;
     955          14 :       count -= ret;
     956          14 :       ret = 0;
     957             :     }
     958          14 :   return TRACE_SYSRES (ret);
     959             : }
     960             : 
     961             : 
     962             : /* This function returns the callback function for I/O.  */
     963             : void
     964           0 : gpgme_get_io_cbs (gpgme_ctx_t ctx, gpgme_io_cbs_t io_cbs)
     965             : {
     966           0 :   TRACE6 (DEBUG_CTX, "gpgme_get_io_cbs", ctx,
     967             :           "io_cbs=%p, ctx->io_cbs.add=%p/%p, .remove=%p, .event=%p/%p",
     968             :           io_cbs, io_cbs->add, io_cbs->add_priv, io_cbs->remove,
     969             :           io_cbs->event, io_cbs->event_priv);
     970             : 
     971           0 :   *io_cbs = ctx->io_cbs;
     972           0 : }
     973             : 
     974             : 
     975             : /* This function sets the locale for the context CTX, or the default
     976             :    locale if CTX is a null pointer.  */
     977             : gpgme_error_t
     978          98 : gpgme_set_locale (gpgme_ctx_t ctx, int category, const char *value)
     979             : {
     980          98 :   int failed = 0;
     981          98 :   char *new_lc_ctype = NULL;
     982          98 :   char *new_lc_messages = NULL;
     983             : 
     984          98 :   TRACE_BEG2 (DEBUG_CTX, "gpgme_set_locale", ctx,
     985             :                "category=%i, value=%s", category, value ? value : "(null)");
     986             : 
     987             : #define PREPARE_ONE_LOCALE(lcat, ucat)                          \
     988             :   if (!failed && value                                          \
     989             :       && (category == LC_ALL || category == LC_ ## ucat))       \
     990             :     {                                                           \
     991             :       new_lc_ ## lcat = strdup (value);                         \
     992             :       if (!new_lc_ ## lcat)                                     \
     993             :         failed = 1;                                             \
     994             :     }
     995             : 
     996             : #ifdef LC_CTYPE
     997          98 :   PREPARE_ONE_LOCALE (ctype, CTYPE);
     998             : #endif
     999             : #ifdef LC_MESSAGES
    1000          98 :   PREPARE_ONE_LOCALE (messages, MESSAGES);
    1001             : #endif
    1002             : 
    1003          98 :   if (failed)
    1004             :     {
    1005           0 :       int saved_err = gpg_error_from_syserror ();
    1006             : 
    1007           0 :       if (new_lc_ctype)
    1008           0 :         free (new_lc_ctype);
    1009           0 :       if (new_lc_messages)
    1010           0 :         free (new_lc_messages);
    1011             : 
    1012           0 :       return TRACE_ERR (saved_err);
    1013             :     }
    1014             : 
    1015             : #define SET_ONE_LOCALE(lcat, ucat)                      \
    1016             :   if (category == LC_ALL || category == LC_ ## ucat)    \
    1017             :     {                                                   \
    1018             :       if (ctx)                                          \
    1019             :         {                                               \
    1020             :           if (ctx->lc_ ## lcat)                              \
    1021             :             free (ctx->lc_ ## lcat);                 \
    1022             :           ctx->lc_ ## lcat = new_lc_ ## lcat;                \
    1023             :         }                                               \
    1024             :       else                                              \
    1025             :         {                                               \
    1026             :           if (def_lc_ ## lcat)                          \
    1027             :             free (def_lc_ ## lcat);                     \
    1028             :           def_lc_ ## lcat = new_lc_ ## lcat;            \
    1029             :         }                                               \
    1030             :     }
    1031             : 
    1032          98 :   if (!ctx)
    1033          98 :     LOCK (def_lc_lock);
    1034             : #ifdef LC_CTYPE
    1035          98 :   SET_ONE_LOCALE (ctype, CTYPE);
    1036             : #endif
    1037             : #ifdef LC_MESSAGES
    1038          98 :   SET_ONE_LOCALE (messages, MESSAGES);
    1039             : #endif
    1040          98 :   if (!ctx)
    1041          98 :     UNLOCK (def_lc_lock);
    1042             : 
    1043          98 :   return TRACE_ERR (0);
    1044             : }
    1045             : 
    1046             : 
    1047             : /* Get the information about the configured engines.  A pointer to the
    1048             :    first engine in the statically allocated linked list is returned.
    1049             :    The returned data is valid until the next gpgme_ctx_set_engine_info.  */
    1050             : gpgme_engine_info_t
    1051         318 : gpgme_ctx_get_engine_info (gpgme_ctx_t ctx)
    1052             : {
    1053         318 :   TRACE1 (DEBUG_CTX, "gpgme_ctx_get_engine_info", ctx,
    1054             :           "ctx->engine_info=%p", ctx->engine_info);
    1055         318 :   return ctx->engine_info;
    1056             : }
    1057             : 
    1058             : 
    1059             : /* Set the engine info for the context CTX, protocol PROTO, to the
    1060             :    file name FILE_NAME and the home directory HOME_DIR.  */
    1061             : gpgme_error_t
    1062         427 : gpgme_ctx_set_engine_info (gpgme_ctx_t ctx, gpgme_protocol_t proto,
    1063             :                            const char *file_name, const char *home_dir)
    1064             : {
    1065             :   gpgme_error_t err;
    1066         427 :   TRACE_BEG4 (DEBUG_CTX, "gpgme_ctx_set_engine_info", ctx,
    1067             :               "protocol=%i (%s), file_name=%s, home_dir=%s",
    1068             :               proto, gpgme_get_protocol_name (proto)
    1069             :               ? gpgme_get_protocol_name (proto) : "unknown",
    1070             :               file_name ? file_name : "(default)",
    1071             :               home_dir ? home_dir : "(default)");
    1072             : 
    1073         427 :   if (!ctx)
    1074           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
    1075             : 
    1076             :   /* Shut down the engine when changing engine info.  */
    1077         427 :   if (ctx->engine)
    1078             :     {
    1079           1 :       TRACE_LOG1 ("releasing ctx->engine=%p", ctx->engine);
    1080           1 :       _gpgme_engine_release (ctx->engine);
    1081           1 :       ctx->engine = NULL;
    1082             :     }
    1083         427 :   err = _gpgme_set_engine_info (ctx->engine_info, proto,
    1084             :                                 file_name, home_dir);
    1085         428 :   return TRACE_ERR (err);
    1086             : }
    1087             : 
    1088             : 
    1089             : /* Clear all notation data from the context.  */
    1090             : void
    1091         701 : _gpgme_sig_notation_clear (gpgme_ctx_t ctx)
    1092             : {
    1093             :   gpgme_sig_notation_t notation;
    1094             : 
    1095         701 :   if (!ctx)
    1096           0 :     return;
    1097             : 
    1098         701 :   notation = ctx->sig_notations;
    1099        1411 :   while (notation)
    1100             :     {
    1101           9 :       gpgme_sig_notation_t next_notation = notation->next;
    1102           9 :       _gpgme_sig_notation_free (notation);
    1103           9 :       notation = next_notation;
    1104             :     }
    1105         701 :   ctx->sig_notations = NULL;
    1106             : }
    1107             : 
    1108             : void
    1109           0 : gpgme_sig_notation_clear (gpgme_ctx_t ctx)
    1110             : {
    1111           0 :   TRACE (DEBUG_CTX, "gpgme_sig_notation_clear", ctx);
    1112             : 
    1113           0 :   if (!ctx)
    1114           0 :     return;
    1115             : 
    1116           0 :   _gpgme_sig_notation_clear (ctx);
    1117             : }
    1118             : 
    1119             : 
    1120             : /* Add the human-readable notation data with name NAME and value VALUE
    1121             :    to the context CTX, using the flags FLAGS.  If NAME is NULL, then
    1122             :    VALUE should be a policy URL.  The flag
    1123             :    GPGME_SIG_NOTATION_HUMAN_READABLE is forced to be true for notation
    1124             :    data, and false for policy URLs.  */
    1125             : gpgme_error_t
    1126           9 : gpgme_sig_notation_add (gpgme_ctx_t ctx, const char *name,
    1127             :                         const char *value, gpgme_sig_notation_flags_t flags)
    1128             : {
    1129             :   gpgme_error_t err;
    1130             :   gpgme_sig_notation_t notation;
    1131             :   gpgme_sig_notation_t *lastp;
    1132             : 
    1133           9 :   TRACE_BEG3 (DEBUG_CTX, "gpgme_sig_notation_add", ctx,
    1134             :               "name=%s, value=%s, flags=0x%x",
    1135             :               name ? name : "(null)", value ? value : "(null)",
    1136             :               flags);
    1137             : 
    1138           9 :   if (!ctx)
    1139           0 :     return TRACE_ERR (gpg_error (GPG_ERR_INV_VALUE));
    1140             : 
    1141           9 :   if (name)
    1142           6 :     flags |= GPGME_SIG_NOTATION_HUMAN_READABLE;
    1143             :   else
    1144           3 :     flags &= ~GPGME_SIG_NOTATION_HUMAN_READABLE;
    1145             : 
    1146          18 :   err = _gpgme_sig_notation_create (¬ation, name, name ? strlen (name) : 0,
    1147           9 :                                     value, value ? strlen (value) : 0, flags);
    1148           9 :   if (err)
    1149           0 :     return TRACE_ERR (err);
    1150             : 
    1151           9 :   lastp = &ctx->sig_notations;
    1152          27 :   while (*lastp)
    1153           9 :     lastp = &(*lastp)->next;
    1154             : 
    1155           9 :   *lastp = notation;
    1156           9 :   return TRACE_ERR (0);
    1157             : }
    1158             : 
    1159             : 
    1160             : /* Get the sig notations for this context.  */
    1161             : gpgme_sig_notation_t
    1162          63 : gpgme_sig_notation_get (gpgme_ctx_t ctx)
    1163             : {
    1164          63 :   if (!ctx)
    1165             :     {
    1166           0 :       TRACE (DEBUG_CTX, "gpgme_sig_notation_get", ctx);
    1167           0 :       return NULL;
    1168             :     }
    1169          63 :   TRACE1 (DEBUG_CTX, "gpgme_sig_notation_get", ctx,
    1170             :           "ctx->sig_notations=%p", ctx->sig_notations);
    1171             : 
    1172          63 :   return ctx->sig_notations;
    1173             : }
    1174             : 
    1175             : 
    1176             : 
    1177             : /* Return a public key algorithm string made of the algorithm and size
    1178             :    or the curve name.  May return NULL on error.  Caller must free the
    1179             :    result using gpgme_free.  */
    1180             : char *
    1181           0 : gpgme_pubkey_algo_string (gpgme_subkey_t subkey)
    1182             : {
    1183           0 :   const char *prefix = NULL;
    1184             :   char *result;
    1185             : 
    1186           0 :   if (!subkey)
    1187             :     {
    1188           0 :       gpg_err_set_errno (EINVAL);
    1189           0 :       return NULL;
    1190             :     }
    1191             : 
    1192           0 :   switch (subkey->pubkey_algo)
    1193             :     {
    1194             :     case GPGME_PK_RSA:
    1195             :     case GPGME_PK_RSA_E:
    1196           0 :     case GPGME_PK_RSA_S: prefix = "rsa"; break;
    1197           0 :     case GPGME_PK_ELG_E: prefix = "elg"; break;
    1198           0 :     case GPGME_PK_DSA:   prefix = "dsa"; break;
    1199           0 :     case GPGME_PK_ELG:   prefix = "xxx"; break;
    1200             :     case GPGME_PK_ECC:
    1201             :     case GPGME_PK_ECDH:
    1202             :     case GPGME_PK_ECDSA:
    1203           0 :     case GPGME_PK_EDDSA: prefix = "";    break;
    1204             :     }
    1205             : 
    1206           0 :   if (prefix && *prefix)
    1207           0 :     {
    1208             :       char buffer[40];
    1209           0 :       snprintf (buffer, sizeof buffer, "%s%u", prefix, subkey->length);
    1210           0 :       result = strdup (buffer);
    1211             :     }
    1212           0 :   else if (prefix && subkey->curve && *subkey->curve)
    1213           0 :     result = strdup (subkey->curve);
    1214           0 :   else if (prefix)
    1215           0 :     result =  strdup ("E_error");
    1216             :   else
    1217           0 :     result = strdup  ("unknown");
    1218             : 
    1219           0 :   return result;
    1220             : }
    1221             : 
    1222             : 
    1223             : const char *
    1224          70 : gpgme_pubkey_algo_name (gpgme_pubkey_algo_t algo)
    1225             : {
    1226          70 :   switch (algo)
    1227             :     {
    1228           6 :     case GPGME_PK_RSA:   return "RSA";
    1229           1 :     case GPGME_PK_RSA_E: return "RSA-E";
    1230           1 :     case GPGME_PK_RSA_S: return "RSA-S";
    1231           1 :     case GPGME_PK_ELG_E: return "ELG-E";
    1232          47 :     case GPGME_PK_DSA:   return "DSA";
    1233           1 :     case GPGME_PK_ECC:   return "ECC";
    1234           1 :     case GPGME_PK_ELG:   return "ELG";
    1235           1 :     case GPGME_PK_ECDSA: return "ECDSA";
    1236           1 :     case GPGME_PK_ECDH:  return "ECDH";
    1237           4 :     case GPGME_PK_EDDSA: return "EdDSA";
    1238           6 :     default:             return NULL;
    1239             :     }
    1240             : }
    1241             : 
    1242             : 
    1243             : const char *
    1244          60 : gpgme_hash_algo_name (gpgme_hash_algo_t algo)
    1245             : {
    1246          60 :   switch (algo)
    1247             :     {
    1248             :     case GPGME_MD_MD5:
    1249           0 :       return "MD5";
    1250             : 
    1251             :     case GPGME_MD_SHA1:
    1252          39 :       return "SHA1";
    1253             : 
    1254             :     case GPGME_MD_RMD160:
    1255           8 :       return "RIPEMD160";
    1256             : 
    1257             :     case GPGME_MD_MD2:
    1258           0 :       return "MD2";
    1259             : 
    1260             :     case GPGME_MD_TIGER:
    1261           0 :       return "TIGER192";
    1262             : 
    1263             :     case GPGME_MD_HAVAL:
    1264           0 :       return "HAVAL";
    1265             : 
    1266             :     case GPGME_MD_SHA256:
    1267           3 :       return "SHA256";
    1268             : 
    1269             :     case GPGME_MD_SHA384:
    1270           0 :       return "SHA384";
    1271             : 
    1272             :     case GPGME_MD_SHA512:
    1273           4 :       return "SHA512";
    1274             : 
    1275             :     case GPGME_MD_SHA224:
    1276           0 :       return "SHA224";
    1277             : 
    1278             :     case GPGME_MD_MD4:
    1279           0 :       return "MD4";
    1280             : 
    1281             :     case GPGME_MD_CRC32:
    1282           0 :       return "CRC32";
    1283             : 
    1284             :     case GPGME_MD_CRC32_RFC1510:
    1285           0 :       return "CRC32RFC1510";
    1286             : 
    1287             :     case GPGME_MD_CRC24_RFC2440:
    1288           0 :       return "CRC24RFC2440";
    1289             : 
    1290             :     default:
    1291           6 :       return NULL;
    1292             :     }
    1293             : }
 |