mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-25 15:27:03 +01:00
made keylisting faster
This commit is contained in:
parent
1f4bdf4d26
commit
9a5c0fd75a
@ -1,5 +1,15 @@
|
|||||||
2001-09-25 Werner Koch <wk@gnupg.org>
|
2001-09-25 Werner Koch <wk@gnupg.org>
|
||||||
|
|
||||||
|
* keyring.c (new_offset_item,release_offset_items)
|
||||||
|
(new_offset_hash_table, lookup_offset_hash_table)
|
||||||
|
(update_offset_hash_table, update_offset_hash_table_from_kb): New.
|
||||||
|
(keyring_search): Use a offset table to optimize search for
|
||||||
|
unknown keys.
|
||||||
|
(keyring_update_keyblock, keyring_insert_keyblock): Insert new
|
||||||
|
offsets.
|
||||||
|
* getkey.c (MAX_UNK_CACHE_ENTRIES): Removed the unknown keys
|
||||||
|
caching code.
|
||||||
|
|
||||||
* g10.c, options.h, import.c: Removed the entire
|
* g10.c, options.h, import.c: Removed the entire
|
||||||
allow-secret-key-import stuff because the validity is now
|
allow-secret-key-import stuff because the validity is now
|
||||||
controlled by other means.
|
controlled by other means.
|
||||||
|
47
g10/getkey.c
47
g10/getkey.c
@ -34,8 +34,6 @@
|
|||||||
#include "trustdb.h"
|
#include "trustdb.h"
|
||||||
#include "i18n.h"
|
#include "i18n.h"
|
||||||
|
|
||||||
#define MAX_UNK_CACHE_ENTRIES 1000 /* we use a linked list - so I guess
|
|
||||||
* this is a reasonable limit */
|
|
||||||
#define MAX_PK_CACHE_ENTRIES 200
|
#define MAX_PK_CACHE_ENTRIES 200
|
||||||
#define MAX_UID_CACHE_ENTRIES 200
|
#define MAX_UID_CACHE_ENTRIES 200
|
||||||
|
|
||||||
@ -73,12 +71,6 @@ typedef struct keyid_list {
|
|||||||
} *keyid_list_t;
|
} *keyid_list_t;
|
||||||
|
|
||||||
|
|
||||||
#if MAX_UNK_CACHE_ENTRIES
|
|
||||||
static keyid_list_t unknown_keyids;
|
|
||||||
static int unk_cache_entries; /* number of entries in unknown keys cache */
|
|
||||||
static int unk_cache_disabled;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if MAX_PK_CACHE_ENTRIES
|
#if MAX_PK_CACHE_ENTRIES
|
||||||
typedef struct pk_cache_entry {
|
typedef struct pk_cache_entry {
|
||||||
struct pk_cache_entry *next;
|
struct pk_cache_entry *next;
|
||||||
@ -270,17 +262,6 @@ cache_user_id( KBNODE keyblock )
|
|||||||
void
|
void
|
||||||
getkey_disable_caches()
|
getkey_disable_caches()
|
||||||
{
|
{
|
||||||
#if MAX_UNK_CACHE_ENTRIES
|
|
||||||
{
|
|
||||||
keyid_list_t kl, kl2;
|
|
||||||
for( kl = unknown_keyids; kl; kl = kl2 ) {
|
|
||||||
kl2 = kl->next;
|
|
||||||
m_free(kl);
|
|
||||||
}
|
|
||||||
unknown_keyids = NULL;
|
|
||||||
unk_cache_disabled = 1;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if MAX_PK_CACHE_ENTRIES
|
#if MAX_PK_CACHE_ENTRIES
|
||||||
{
|
{
|
||||||
pk_cache_entry_t ce, ce2;
|
pk_cache_entry_t ce, ce2;
|
||||||
@ -334,15 +315,6 @@ get_pubkey( PKT_public_key *pk, u32 *keyid )
|
|||||||
int internal = 0;
|
int internal = 0;
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
|
|
||||||
#if MAX_UNK_CACHE_ENTRIES
|
|
||||||
{ /* let's see whether we checked the keyid already */
|
|
||||||
keyid_list_t kl;
|
|
||||||
for( kl = unknown_keyids; kl; kl = kl->next )
|
|
||||||
if( kl->keyid[0] == keyid[0] && kl->keyid[1] == keyid[1] )
|
|
||||||
return G10ERR_NO_PUBKEY; /* already checked and not found */
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if MAX_PK_CACHE_ENTRIES
|
#if MAX_PK_CACHE_ENTRIES
|
||||||
{ /* Try to get it from the cache */
|
{ /* Try to get it from the cache */
|
||||||
pk_cache_entry_t ce;
|
pk_cache_entry_t ce;
|
||||||
@ -385,25 +357,6 @@ get_pubkey( PKT_public_key *pk, u32 *keyid )
|
|||||||
if( !rc )
|
if( !rc )
|
||||||
goto leave;
|
goto leave;
|
||||||
|
|
||||||
#if MAX_UNK_CACHE_ENTRIES
|
|
||||||
/* not found: store it for future reference */
|
|
||||||
if( unk_cache_disabled )
|
|
||||||
;
|
|
||||||
else if( ++unk_cache_entries > MAX_UNK_CACHE_ENTRIES ) {
|
|
||||||
unk_cache_disabled = 1;
|
|
||||||
if( opt.verbose > 1 )
|
|
||||||
log_info(_("too many entries in unk cache - disabled\n"));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
keyid_list_t kl;
|
|
||||||
|
|
||||||
kl = m_alloc( sizeof *kl );
|
|
||||||
kl->keyid[0] = keyid[0];
|
|
||||||
kl->keyid[1] = keyid[1];
|
|
||||||
kl->next = unknown_keyids;
|
|
||||||
unknown_keyids = kl;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
rc = G10ERR_NO_PUBKEY;
|
rc = G10ERR_NO_PUBKEY;
|
||||||
|
|
||||||
leave:
|
leave:
|
||||||
|
213
g10/keyring.c
213
g10/keyring.c
@ -35,45 +35,153 @@
|
|||||||
#include "main.h" /*for check_key_signature()*/
|
#include "main.h" /*for check_key_signature()*/
|
||||||
#include "i18n.h"
|
#include "i18n.h"
|
||||||
|
|
||||||
|
struct off_item {
|
||||||
|
struct off_item *next;
|
||||||
|
u32 kid[2];
|
||||||
|
off_t off;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct off_item **OffsetHashTable;
|
||||||
|
|
||||||
|
|
||||||
typedef struct keyring_name *KR_NAME;
|
typedef struct keyring_name *KR_NAME;
|
||||||
struct keyring_name {
|
struct keyring_name {
|
||||||
struct keyring_name *next;
|
struct keyring_name *next;
|
||||||
int secret;
|
int secret;
|
||||||
DOTLOCK lockhd;
|
OffsetHashTable offtbl;
|
||||||
int is_locked;
|
int offtbl_ready;
|
||||||
char fname[1];
|
DOTLOCK lockhd;
|
||||||
|
int is_locked;
|
||||||
|
char fname[1];
|
||||||
};
|
};
|
||||||
typedef struct keyring_name const * CONST_KR_NAME;
|
typedef struct keyring_name const * CONST_KR_NAME;
|
||||||
|
|
||||||
static KR_NAME kr_names;
|
static KR_NAME kr_names;
|
||||||
static int active_handles;
|
static int active_handles;
|
||||||
|
|
||||||
|
|
||||||
struct keyring_handle {
|
struct keyring_handle {
|
||||||
int secret; /* this is for a secret keyring */
|
int secret; /* this is for a secret keyring */
|
||||||
struct {
|
struct {
|
||||||
CONST_KR_NAME kr;
|
CONST_KR_NAME kr;
|
||||||
IOBUF iobuf;
|
IOBUF iobuf;
|
||||||
int eof;
|
int eof;
|
||||||
int error;
|
int error;
|
||||||
} current;
|
} current;
|
||||||
struct {
|
struct {
|
||||||
CONST_KR_NAME kr;
|
CONST_KR_NAME kr;
|
||||||
off_t offset;
|
off_t offset;
|
||||||
size_t pk_no;
|
size_t pk_no;
|
||||||
size_t uid_no;
|
size_t uid_no;
|
||||||
unsigned int n_packets; /*used for delete and update*/
|
unsigned int n_packets; /*used for delete and update*/
|
||||||
} found;
|
} found;
|
||||||
struct {
|
struct {
|
||||||
char *name;
|
char *name;
|
||||||
char *pattern;
|
char *pattern;
|
||||||
} word_match;
|
} word_match;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static int do_copy (int mode, const char *fname, KBNODE root, int secret,
|
static int do_copy (int mode, const char *fname, KBNODE root, int secret,
|
||||||
off_t start_offset, unsigned int n_packets );
|
off_t start_offset, unsigned int n_packets );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static struct off_item *
|
||||||
|
new_offset_item (void)
|
||||||
|
{
|
||||||
|
struct off_item *k;
|
||||||
|
|
||||||
|
k = m_alloc_clear (sizeof *k);
|
||||||
|
return k;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
release_offset_items (struct off_item *k)
|
||||||
|
{
|
||||||
|
struct off_item *k2;
|
||||||
|
|
||||||
|
for (; k; k = k2)
|
||||||
|
{
|
||||||
|
k2 = k->next;
|
||||||
|
m_free (k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static OffsetHashTable
|
||||||
|
new_offset_hash_table (void)
|
||||||
|
{
|
||||||
|
struct off_item **tbl;
|
||||||
|
|
||||||
|
tbl = m_alloc_clear (2048 * sizeof *tbl);
|
||||||
|
return tbl;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
release_offset_hash_table (OffsetHashTable tbl)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!tbl)
|
||||||
|
return;
|
||||||
|
for (i=0; i < 2048; i++)
|
||||||
|
release_offset_items (tbl[i]);
|
||||||
|
m_free (tbl);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct off_item *
|
||||||
|
lookup_offset_hash_table (OffsetHashTable tbl, u32 *kid)
|
||||||
|
{
|
||||||
|
struct off_item *k;
|
||||||
|
|
||||||
|
for (k = tbl[(kid[1] & 0x07ff)]; k; k = k->next)
|
||||||
|
if (k->kid[0] == kid[0] && k->kid[1] == kid[1])
|
||||||
|
return k;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
update_offset_hash_table (OffsetHashTable tbl, u32 *kid, off_t off)
|
||||||
|
{
|
||||||
|
struct off_item *k;
|
||||||
|
|
||||||
|
for (k = tbl[(kid[1] & 0x07ff)]; k; k = k->next)
|
||||||
|
{
|
||||||
|
if (k->kid[0] == kid[0] && k->kid[1] == kid[1])
|
||||||
|
{
|
||||||
|
k->off = off;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
k = new_offset_item ();
|
||||||
|
k->kid[0] = kid[0];
|
||||||
|
k->kid[1] = kid[1];
|
||||||
|
k->off = off;
|
||||||
|
k->next = tbl[(kid[1] & 0x07ff)];
|
||||||
|
tbl[(kid[1] & 0x07ff)] = k;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
update_offset_hash_table_from_kb (OffsetHashTable tbl, KBNODE node, off_t off)
|
||||||
|
{
|
||||||
|
for (; node; node = node->next)
|
||||||
|
{
|
||||||
|
if (node->pkt->pkttype == PKT_PUBLIC_KEY
|
||||||
|
|| node->pkt->pkttype == PKT_PUBLIC_SUBKEY)
|
||||||
|
{
|
||||||
|
u32 aki[2];
|
||||||
|
keyid_from_pk (node->pkt->pkt.public_key, aki);
|
||||||
|
update_offset_hash_table (tbl, aki, off);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Register a filename for plain keyring files
|
* Register a filename for plain keyring files
|
||||||
*/
|
*/
|
||||||
@ -93,6 +201,8 @@ keyring_register_filename (const char *fname, int secret)
|
|||||||
kr = m_alloc (sizeof *kr + strlen (fname));
|
kr = m_alloc (sizeof *kr + strlen (fname));
|
||||||
strcpy (kr->fname, fname);
|
strcpy (kr->fname, fname);
|
||||||
kr->secret = !!secret;
|
kr->secret = !!secret;
|
||||||
|
kr->offtbl = new_offset_hash_table ();
|
||||||
|
kr->offtbl_ready = 0;
|
||||||
kr->lockhd = NULL;
|
kr->lockhd = NULL;
|
||||||
kr->is_locked = 0;
|
kr->is_locked = 0;
|
||||||
kr->next = kr_names;
|
kr->next = kr_names;
|
||||||
@ -369,9 +479,15 @@ keyring_update_keyblock (KEYRING_HANDLE hd, KBNODE kb)
|
|||||||
rc = do_copy (3, hd->found.kr->fname, kb, hd->secret,
|
rc = do_copy (3, hd->found.kr->fname, kb, hd->secret,
|
||||||
hd->found.offset, hd->found.n_packets );
|
hd->found.offset, hd->found.n_packets );
|
||||||
if (!rc) {
|
if (!rc) {
|
||||||
/* better reset the found info */
|
if (hd->current.kr->offtbl)
|
||||||
hd->found.kr = NULL;
|
{
|
||||||
hd->found.offset = 0;
|
/* we do not have the offset but as it is not use it does not
|
||||||
|
* matter*/
|
||||||
|
update_offset_hash_table_from_kb (hd->current.kr->offtbl, kb, 0);
|
||||||
|
}
|
||||||
|
/* better reset the found info */
|
||||||
|
hd->found.kr = NULL;
|
||||||
|
hd->found.offset = 0;
|
||||||
}
|
}
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
@ -405,6 +521,12 @@ keyring_insert_keyblock (KEYRING_HANDLE hd, KBNODE kb)
|
|||||||
|
|
||||||
/* do the insert */
|
/* do the insert */
|
||||||
rc = do_copy (1, fname, kb, hd->secret, 0, 0 );
|
rc = do_copy (1, fname, kb, hd->secret, 0, 0 );
|
||||||
|
if (!rc && hd->current.kr->offtbl)
|
||||||
|
{
|
||||||
|
/* we do not have the offset but as it is not use it does not matter*/
|
||||||
|
update_offset_hash_table_from_kb (hd->current.kr->offtbl, kb, 0);
|
||||||
|
}
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -441,6 +563,8 @@ keyring_delete_keyblock (KEYRING_HANDLE hd)
|
|||||||
/* better reset the found info */
|
/* better reset the found info */
|
||||||
hd->found.kr = NULL;
|
hd->found.kr = NULL;
|
||||||
hd->found.offset = 0;
|
hd->found.offset = 0;
|
||||||
|
/* Delete is a rare operations, so we don't remove the keys
|
||||||
|
* from the offset table */
|
||||||
}
|
}
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
@ -714,6 +838,8 @@ keyring_search (KEYRING_HANDLE hd, KEYDB_SEARCH_DESC *desc, size_t ndesc)
|
|||||||
PKT_user_id *uid = NULL;
|
PKT_user_id *uid = NULL;
|
||||||
PKT_public_key *pk = NULL;
|
PKT_public_key *pk = NULL;
|
||||||
PKT_secret_key *sk = NULL;
|
PKT_secret_key *sk = NULL;
|
||||||
|
OffsetHashTable offtbl;
|
||||||
|
int offtbl_ready;
|
||||||
|
|
||||||
/* figure out what information we need */
|
/* figure out what information we need */
|
||||||
need_uid = need_words = need_keyid = need_fpr = any_skip = 0;
|
need_uid = need_words = need_keyid = need_fpr = any_skip = 0;
|
||||||
@ -755,6 +881,30 @@ keyring_search (KEYRING_HANDLE hd, KEYDB_SEARCH_DESC *desc, size_t ndesc)
|
|||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
|
offtbl = hd->current.kr->offtbl;
|
||||||
|
offtbl_ready = hd->current.kr->offtbl_ready;
|
||||||
|
if (!offtbl)
|
||||||
|
;
|
||||||
|
else if (!offtbl_ready)
|
||||||
|
need_keyid = 1;
|
||||||
|
else if (ndesc == 1 && desc[0].mode == KEYDB_SEARCH_MODE_LONG_KID)
|
||||||
|
{
|
||||||
|
struct off_item *oi;
|
||||||
|
|
||||||
|
oi = lookup_offset_hash_table (offtbl, desc[0].u.kid);
|
||||||
|
if (!oi)
|
||||||
|
{ /* We know that we don't have this key */
|
||||||
|
hd->found.kr = NULL;
|
||||||
|
hd->current.eof = 1;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/* We could now create a positive search status and return.
|
||||||
|
* However the problem is that another instance of gpg may
|
||||||
|
* have changed the keyring so that the offsets are not valid
|
||||||
|
* anymore - therefore we don't do it
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
if (need_words) {
|
if (need_words) {
|
||||||
BUG();
|
BUG();
|
||||||
@ -807,6 +957,9 @@ keyring_search (KEYRING_HANDLE hd, KEYDB_SEARCH_DESC *desc, size_t ndesc)
|
|||||||
}
|
}
|
||||||
if (need_keyid)
|
if (need_keyid)
|
||||||
keyid_from_pk (pk, aki);
|
keyid_from_pk (pk, aki);
|
||||||
|
|
||||||
|
if (offtbl && !offtbl_ready)
|
||||||
|
update_offset_hash_table (offtbl, aki, main_offset);
|
||||||
}
|
}
|
||||||
else if (pkt.pkttype == PKT_USER_ID) {
|
else if (pkt.pkttype == PKT_USER_ID) {
|
||||||
uid = pkt.pkt.user_id;
|
uid = pkt.pkt.user_id;
|
||||||
@ -824,6 +977,7 @@ keyring_search (KEYRING_HANDLE hd, KEYDB_SEARCH_DESC *desc, size_t ndesc)
|
|||||||
}
|
}
|
||||||
if (need_keyid)
|
if (need_keyid)
|
||||||
keyid_from_sk (sk, aki);
|
keyid_from_sk (sk, aki);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (n=0; n < ndesc; n++) {
|
for (n=0; n < ndesc; n++) {
|
||||||
@ -894,10 +1048,17 @@ keyring_search (KEYRING_HANDLE hd, KEYDB_SEARCH_DESC *desc, size_t ndesc)
|
|||||||
hd->found.uid_no = uid? uid_no : 0;
|
hd->found.uid_no = uid? uid_no : 0;
|
||||||
}
|
}
|
||||||
else if (rc == -1)
|
else if (rc == -1)
|
||||||
|
{
|
||||||
hd->current.eof = 1;
|
hd->current.eof = 1;
|
||||||
|
/* if we scanned the entire keyring, we are sure that
|
||||||
|
* all known key IDs are in our offtbl, mark that. */
|
||||||
|
hd->current.kr->offtbl_ready = 1;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
hd->current.error = rc;
|
hd->current.error = rc;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
free_packet(&pkt);
|
free_packet(&pkt);
|
||||||
set_packet_list_mode(save_mode);
|
set_packet_list_mode(save_mode);
|
||||||
return rc;
|
return rc;
|
||||||
|
BIN
g10/keyring.o
BIN
g10/keyring.o
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user