gpg: Avoid multiple open calls to the keybox file.

* g10/keydb.h (KEYDB_HANDLE): Move typedef to ...
* g10/gpg.h: here.
(struct server_control_s): Add field 'cached_getkey_kdb'.
* g10/gpg.c (gpg_deinit_default_ctrl): Release that keydb handle.
* g10/getkey.c (getkey_end): Cache keydb handle.
(get_pubkey): Use cached keydb handle.
* kbx/keybox-search.c (keybox_search_reset): Use lseek instead of
closing the file.
--

Before this patch a "gpg --check-sigs" opened and closed the keybox
file for almost every signature check.  By caching the keydb handle
and using lseek(2) this can be limited to just 2 times.  This might
speed up things on Windows.

Signed-off-by: Werner Koch <wk@gnupg.org>
This commit is contained in:
Werner Koch 2017-03-31 20:44:05 +02:00
parent aca5f494a8
commit 5556eca5ac
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B
5 changed files with 33 additions and 11 deletions

View File

@ -736,11 +736,21 @@ get_pubkey (ctrl_t ctrl, PKT_public_key * pk, u32 * keyid)
memset (&ctx, 0, sizeof ctx); memset (&ctx, 0, sizeof ctx);
ctx.exact = 1; /* Use the key ID exactly as given. */ ctx.exact = 1; /* Use the key ID exactly as given. */
ctx.not_allocated = 1; ctx.not_allocated = 1;
ctx.kr_handle = keydb_new ();
if (!ctx.kr_handle) if (ctrl && ctrl->cached_getkey_kdb)
{ {
rc = gpg_error_from_syserror (); ctx.kr_handle = ctrl->cached_getkey_kdb;
goto leave; ctrl->cached_getkey_kdb = NULL;
keydb_search_reset (ctx.kr_handle);
}
else
{
ctx.kr_handle = keydb_new ();
if (!ctx.kr_handle)
{
rc = gpg_error_from_syserror ();
goto leave;
}
} }
ctx.nitems = 1; ctx.nitems = 1;
ctx.items[0].mode = KEYDB_SEARCH_MODE_LONG_KID; ctx.items[0].mode = KEYDB_SEARCH_MODE_LONG_KID;
@ -2208,7 +2218,10 @@ getkey_end (ctrl_t ctrl, getkey_ctx_t ctx)
{ {
if (ctx) if (ctx)
{ {
keydb_release (ctx->kr_handle); if (ctrl && !ctrl->cached_getkey_kdb)
ctrl->cached_getkey_kdb = ctx->kr_handle;
else
keydb_release (ctx->kr_handle);
free_strlist (ctx->extra_list); free_strlist (ctx->extra_list);
if (!ctx->not_allocated) if (!ctx->not_allocated)
xfree (ctx); xfree (ctx);

View File

@ -2195,6 +2195,8 @@ gpg_deinit_default_ctrl (ctrl_t ctrl)
tofu_closedbs (ctrl); tofu_closedbs (ctrl);
#endif #endif
gpg_dirmngr_deinit_session_data (ctrl); gpg_dirmngr_deinit_session_data (ctrl);
keydb_release (ctrl->cached_getkey_kdb);
} }

View File

@ -59,10 +59,13 @@ struct server_local_s;
struct dirmngr_local_s; struct dirmngr_local_s;
typedef struct dirmngr_local_s *dirmngr_local_t; typedef struct dirmngr_local_s *dirmngr_local_t;
/* Object used to describe a keyblok node. */ /* Object used to describe a keyblock node. */
typedef struct kbnode_struct *KBNODE; /* Deprecated use kbnode_t. */ typedef struct kbnode_struct *KBNODE; /* Deprecated use kbnode_t. */
typedef struct kbnode_struct *kbnode_t; typedef struct kbnode_struct *kbnode_t;
/* The handle for keydb operations. */
typedef struct keydb_handle *KEYDB_HANDLE;
/* TOFU database meta object. */ /* TOFU database meta object. */
struct tofu_dbs_s; struct tofu_dbs_s;
typedef struct tofu_dbs_s *tofu_dbs_t; typedef struct tofu_dbs_s *tofu_dbs_t;
@ -94,6 +97,8 @@ struct server_control_s
int batch_updated_wanted; int batch_updated_wanted;
} tofu; } tofu;
/* This is used to cache a key data base handle. */
KEYDB_HANDLE cached_getkey_kdb;
}; };

View File

@ -109,9 +109,6 @@ struct pubkey_find_info {
}; };
typedef struct keydb_handle *KEYDB_HANDLE;
/* Helper type for preference fucntions. */ /* Helper type for preference fucntions. */
union pref_hint union pref_hint
{ {

View File

@ -746,8 +746,13 @@ keybox_search_reset (KEYBOX_HANDLE hd)
if (hd->fp) if (hd->fp)
{ {
fclose (hd->fp); if (fseeko (hd->fp, 0, SEEK_SET))
hd->fp = NULL; {
/* Ooops. Seek did not work. Close so that the search will
* open the file again. */
fclose (hd->fp);
hd->fp = NULL;
}
} }
hd->error = 0; hd->error = 0;
hd->eof = 0; hd->eof = 0;