mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-08 12:44:23 +01:00
agent: Introduce management of timer to expire cache entries.
* agent/cache.c (struct timer_s): New. (struct cache_item_s): Add a member filed T for timer. (the_timer_list, the_timer_list_new): New. (insert_to_timer_list_new, insert_to_timer_list): New. (remove_from_timer_list, remove_from_timer_list_new): New. (housekeeping): Remove. (compute_expiration, update_expiration): New. (do_expire): New. (TIMERTICK_INTERVAL): Remove. (agent_cache_expiration): Use timer list to manage the expiration of cache entries. (agent_flush_cache): Call update_expiration when needed. (agent_put_cache): Don't call housekeeping any more, but update_expiration for an entry in question. (agent_get_cache): Likewise. -- GnuPG-bug-id: 6681 Signed-off-by: NIIBE Yutaka <gniibe@fsij.org>
This commit is contained in:
parent
76a2f18028
commit
92de0387f0
352
agent/cache.c
352
agent/cache.c
@ -53,8 +53,20 @@ struct secret_data_s {
|
|||||||
char data[1]; /* A string. */
|
char data[1]; /* A string. */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* The cache object. */
|
/* The type of cache object. */
|
||||||
typedef struct cache_item_s *ITEM;
|
typedef struct cache_item_s *ITEM;
|
||||||
|
|
||||||
|
/* The timer entry in a linked list. */
|
||||||
|
struct timer_s {
|
||||||
|
ITEM next;
|
||||||
|
int tv_sec;
|
||||||
|
int reason;
|
||||||
|
};
|
||||||
|
#define CACHE_EXPIRE_UNUSED 0
|
||||||
|
#define CACHE_EXPIRE_LAST_ACCESS 1
|
||||||
|
#define CACHE_EXPIRE_CREATION 2
|
||||||
|
|
||||||
|
/* The cache object. */
|
||||||
struct cache_item_s {
|
struct cache_item_s {
|
||||||
ITEM next;
|
ITEM next;
|
||||||
time_t created;
|
time_t created;
|
||||||
@ -63,12 +75,18 @@ struct cache_item_s {
|
|||||||
struct secret_data_s *pw;
|
struct secret_data_s *pw;
|
||||||
cache_mode_t cache_mode;
|
cache_mode_t cache_mode;
|
||||||
int restricted; /* The value of ctrl->restricted is part of the key. */
|
int restricted; /* The value of ctrl->restricted is part of the key. */
|
||||||
|
struct timer_s t;
|
||||||
char key[1];
|
char key[1];
|
||||||
};
|
};
|
||||||
|
|
||||||
/* The cache himself. */
|
/* The cache himself. */
|
||||||
static ITEM thecache;
|
static ITEM thecache;
|
||||||
|
|
||||||
|
/* The timer list of expiration, in active. */
|
||||||
|
static ITEM the_timer_list;
|
||||||
|
/* Newly created entries, to be inserted into the timer list. */
|
||||||
|
static ITEM the_timer_list_new;
|
||||||
|
|
||||||
/* NULL or the last cache key stored by agent_store_cache_hit. */
|
/* NULL or the last cache key stored by agent_store_cache_hit. */
|
||||||
static char *last_stored_cache_key;
|
static char *last_stored_cache_key;
|
||||||
|
|
||||||
@ -193,123 +211,301 @@ new_data (const char *string, struct secret_data_s **r_data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* Check whether there are items to expire. */
|
|
||||||
static void
|
static void
|
||||||
housekeeping (void)
|
insert_to_timer_list_new (ITEM entry)
|
||||||
{
|
{
|
||||||
ITEM r, rprev;
|
entry->t.next = the_timer_list_new;
|
||||||
time_t current = gnupg_get_time ();
|
the_timer_list_new = entry;
|
||||||
|
}
|
||||||
|
|
||||||
/* First expire the actual data */
|
/* Insert to the active timer list. */
|
||||||
for (r=thecache; r; r = r->next)
|
static void
|
||||||
|
insert_to_timer_list (struct timespec *ts, ITEM entry)
|
||||||
|
{
|
||||||
|
ITEM e, eprev;
|
||||||
|
|
||||||
|
if (!the_timer_list || ts->tv_sec >= entry->t.tv_sec)
|
||||||
{
|
{
|
||||||
if (r->cache_mode == CACHE_MODE_PIN)
|
if (the_timer_list && ts->tv_nsec)
|
||||||
; /* Don't let it expire - scdaemon explicitly flushes them. */
|
the_timer_list->t.tv_sec++;
|
||||||
else if (r->pw && r->ttl >= 0 && r->accessed + r->ttl < current)
|
|
||||||
{
|
ts->tv_sec = entry->t.tv_sec;
|
||||||
if (DBG_CACHE)
|
ts->tv_nsec = 0;
|
||||||
log_debug (" expired '%s'.%d (%ds after last access)\n",
|
|
||||||
r->key, r->restricted, r->ttl);
|
entry->t.tv_sec = 0;
|
||||||
release_data (r->pw);
|
entry->t.next = the_timer_list;
|
||||||
r->pw = NULL;
|
the_timer_list = entry;
|
||||||
r->accessed = current;
|
return;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Second, make sure that we also remove them based on the created
|
entry->t.tv_sec -= ts->tv_sec;
|
||||||
* stamp so that the user has to enter it from time to time. We
|
eprev = NULL;
|
||||||
* don't do this for data items which are used to storage secrets in
|
for (e = the_timer_list; e; e = e->t.next)
|
||||||
* meory and are not user entered passphrases etc. */
|
|
||||||
for (r=thecache; r; r = r->next)
|
|
||||||
{
|
{
|
||||||
|
if (e->t.tv_sec > entry->t.tv_sec)
|
||||||
|
break;
|
||||||
|
|
||||||
|
eprev = e;
|
||||||
|
entry->t.tv_sec -= e->t.tv_sec;
|
||||||
|
}
|
||||||
|
|
||||||
|
entry->t.next = e;
|
||||||
|
if (e)
|
||||||
|
e->t.tv_sec -= entry->t.tv_sec;
|
||||||
|
|
||||||
|
if (eprev)
|
||||||
|
eprev->t.next = entry;
|
||||||
|
else
|
||||||
|
the_timer_list = entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
remove_from_timer_list (ITEM entry)
|
||||||
|
{
|
||||||
|
ITEM e, eprev;
|
||||||
|
|
||||||
|
eprev = NULL;
|
||||||
|
for (e = the_timer_list; e; e = e->t.next)
|
||||||
|
if (e != entry)
|
||||||
|
eprev = e;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (e->t.next)
|
||||||
|
e->t.next->t.tv_sec += e->t.tv_sec;
|
||||||
|
|
||||||
|
if (eprev)
|
||||||
|
eprev->t.next = e->t.next;
|
||||||
|
else
|
||||||
|
the_timer_list = e->t.next;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
entry->t.next = NULL;
|
||||||
|
entry->t.tv_sec = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
remove_from_timer_list_new (ITEM entry)
|
||||||
|
{
|
||||||
|
ITEM e, eprev;
|
||||||
|
|
||||||
|
eprev = NULL;
|
||||||
|
for (e = the_timer_list_new; e; e = e->t.next)
|
||||||
|
if (e != entry)
|
||||||
|
eprev = e;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (e->t.next)
|
||||||
|
e->t.next->t.tv_sec += e->t.tv_sec;
|
||||||
|
|
||||||
|
if (eprev)
|
||||||
|
eprev->t.next = e->t.next;
|
||||||
|
else
|
||||||
|
the_timer_list_new = e->t.next;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
entry->t.next = NULL;
|
||||||
|
entry->t.tv_sec = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
compute_expiration (ITEM r)
|
||||||
|
{
|
||||||
unsigned long maxttl;
|
unsigned long maxttl;
|
||||||
|
time_t current = gnupg_get_time ();
|
||||||
|
time_t next;
|
||||||
|
|
||||||
|
if (r->cache_mode == CACHE_MODE_PIN)
|
||||||
|
return 0; /* Don't let it expire - scdaemon explicitly flushes them. */
|
||||||
|
|
||||||
|
if (!r->pw)
|
||||||
|
{
|
||||||
|
/* Expire an old and unused entry after 30 minutes. */
|
||||||
|
r->t.tv_sec = 60*30;
|
||||||
|
r->t.reason = CACHE_EXPIRE_UNUSED;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
switch (r->cache_mode)
|
switch (r->cache_mode)
|
||||||
{
|
{
|
||||||
case CACHE_MODE_DATA:
|
case CACHE_MODE_DATA:
|
||||||
case CACHE_MODE_PIN:
|
case CACHE_MODE_PIN:
|
||||||
continue; /* No MAX TTL here. */
|
maxttl = 0; /* No MAX TTL here. */
|
||||||
|
break;
|
||||||
case CACHE_MODE_SSH: maxttl = opt.max_cache_ttl_ssh; break;
|
case CACHE_MODE_SSH: maxttl = opt.max_cache_ttl_ssh; break;
|
||||||
default: maxttl = opt.max_cache_ttl; break;
|
default: maxttl = opt.max_cache_ttl; break;
|
||||||
}
|
}
|
||||||
if (r->pw && r->created + maxttl < current)
|
|
||||||
|
if (maxttl)
|
||||||
{
|
{
|
||||||
if (DBG_CACHE)
|
if (r->created + maxttl < current)
|
||||||
log_debug (" expired '%s'.%d (%lus after creation)\n",
|
{
|
||||||
r->key, r->restricted, opt.max_cache_ttl);
|
r->t.tv_sec = 0;
|
||||||
release_data (r->pw);
|
r->t.reason = CACHE_EXPIRE_CREATION;
|
||||||
r->pw = NULL;
|
return 1;
|
||||||
r->accessed = current;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Third, make sure that we don't have too many items in the list.
|
next = r->created + maxttl - current;
|
||||||
* Expire old and unused entries after 30 minutes. */
|
|
||||||
for (rprev=NULL, r=thecache; r; )
|
|
||||||
{
|
|
||||||
if (!r->pw && r->ttl >= 0 && r->accessed + 60*30 < current)
|
|
||||||
{
|
|
||||||
ITEM r2 = r->next;
|
|
||||||
if (DBG_CACHE)
|
|
||||||
log_debug (" removed '%s'.%d (mode %d) (slot not used for 30m)\n",
|
|
||||||
r->key, r->restricted, r->cache_mode);
|
|
||||||
xfree (r);
|
|
||||||
if (!rprev)
|
|
||||||
thecache = r2;
|
|
||||||
else
|
|
||||||
rprev->next = r2;
|
|
||||||
r = r2;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
next = 0;
|
||||||
|
|
||||||
|
if (r->ttl >= 0 && (next == 0 || r->ttl < next))
|
||||||
{
|
{
|
||||||
rprev = r;
|
r->t.tv_sec = r->ttl;
|
||||||
r = r->next;
|
r->t.reason = CACHE_EXPIRE_LAST_ACCESS;
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (next)
|
||||||
|
{
|
||||||
|
r->t.tv_sec = next;
|
||||||
|
r->t.reason = CACHE_EXPIRE_CREATION;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
update_expiration (ITEM entry, int is_new_entry)
|
||||||
|
{
|
||||||
|
if (!is_new_entry)
|
||||||
|
{
|
||||||
|
remove_from_timer_list (entry);
|
||||||
|
remove_from_timer_list_new (entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (compute_expiration (entry))
|
||||||
|
{
|
||||||
|
insert_to_timer_list_new (entry);
|
||||||
|
agent_kick_the_loop ();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#define TIMERTICK_INTERVAL (4)
|
/* Expire the cache entry. Returns 1 when the entry should be removed
|
||||||
|
* from the cache. */
|
||||||
|
static int
|
||||||
|
do_expire (ITEM e)
|
||||||
|
{
|
||||||
|
if (!e->pw)
|
||||||
|
/* Unused entry after 30 minutes. */
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
if (e->t.reason == CACHE_EXPIRE_LAST_ACCESS)
|
||||||
|
{
|
||||||
|
if (DBG_CACHE)
|
||||||
|
log_debug (" expired '%s'.%d (%ds after last access)\n",
|
||||||
|
e->key, e->restricted, e->ttl);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (DBG_CACHE)
|
||||||
|
log_debug (" expired '%s'.%d (%lus after creation)\n",
|
||||||
|
e->key, e->restricted, opt.max_cache_ttl);
|
||||||
|
}
|
||||||
|
|
||||||
|
release_data (e->pw);
|
||||||
|
e->pw = NULL;
|
||||||
|
e->accessed = 0;
|
||||||
|
|
||||||
|
if (compute_expiration (e))
|
||||||
|
insert_to_timer_list_new (e);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
struct timespec *
|
struct timespec *
|
||||||
agent_cache_expiration (void)
|
agent_cache_expiration (void)
|
||||||
{
|
{
|
||||||
static struct timespec abstime;
|
static struct timespec abstime;
|
||||||
static struct timespec timeout;
|
static struct timespec timeout;
|
||||||
static int initialized = 0;
|
struct timespec *tp;
|
||||||
struct timespec curtime;
|
struct timespec curtime;
|
||||||
int res;
|
int res;
|
||||||
|
int expired = 0;
|
||||||
if (!initialized)
|
ITEM e, enext;
|
||||||
{
|
|
||||||
initialized = 1;
|
|
||||||
npth_clock_gettime (&abstime);
|
|
||||||
abstime.tv_sec += TIMERTICK_INTERVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
npth_clock_gettime (&curtime);
|
|
||||||
if (!(npth_timercmp (&curtime, &abstime, <)))
|
|
||||||
{
|
|
||||||
/* Timeout. */
|
|
||||||
npth_clock_gettime (&abstime);
|
|
||||||
abstime.tv_sec += TIMERTICK_INTERVAL;
|
|
||||||
|
|
||||||
if (DBG_CACHE)
|
|
||||||
log_debug ("agent_cache_housekeeping\n");
|
|
||||||
|
|
||||||
res = npth_mutex_lock (&cache_lock);
|
res = npth_mutex_lock (&cache_lock);
|
||||||
if (res)
|
if (res)
|
||||||
log_fatal ("failed to acquire cache mutex: %s\n", strerror (res));
|
log_fatal ("failed to acquire cache mutex: %s\n", strerror (res));
|
||||||
|
|
||||||
housekeeping ();
|
npth_clock_gettime (&curtime);
|
||||||
|
if (the_timer_list)
|
||||||
|
{
|
||||||
|
if (npth_timercmp (&abstime, &curtime, <))
|
||||||
|
expired = 1;
|
||||||
|
else
|
||||||
|
npth_timersub (&abstime, &curtime, &timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (expired && (e = the_timer_list) && e->t.tv_sec == 0)
|
||||||
|
{
|
||||||
|
the_timer_list = e->t.next;
|
||||||
|
e->t.next = NULL;
|
||||||
|
|
||||||
|
if (do_expire (e))
|
||||||
|
{
|
||||||
|
ITEM r, rprev;
|
||||||
|
|
||||||
|
if (DBG_CACHE)
|
||||||
|
log_debug (" removed '%s'.%d (mode %d) (slot not used for 30m)\n",
|
||||||
|
e->key, e->restricted, e->cache_mode);
|
||||||
|
|
||||||
|
rprev = NULL;
|
||||||
|
for (r = thecache; r; r = r->next)
|
||||||
|
if (r == e)
|
||||||
|
{
|
||||||
|
if (!rprev)
|
||||||
|
thecache = r->next;
|
||||||
|
else
|
||||||
|
rprev->next = r->next;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
rprev = r;
|
||||||
|
|
||||||
|
remove_from_timer_list_new (e);
|
||||||
|
|
||||||
|
xfree (e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (expired || !the_timer_list)
|
||||||
|
timeout.tv_sec = timeout.tv_nsec = 0;
|
||||||
|
|
||||||
|
for (e = the_timer_list_new; e; e = enext)
|
||||||
|
{
|
||||||
|
enext = e->t.next;
|
||||||
|
e->t.next = NULL;
|
||||||
|
insert_to_timer_list (&timeout, e);
|
||||||
|
}
|
||||||
|
the_timer_list_new = NULL;
|
||||||
|
|
||||||
|
if (!the_timer_list)
|
||||||
|
tp = NULL;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (the_timer_list->t.tv_sec != 0)
|
||||||
|
{
|
||||||
|
timeout.tv_sec += the_timer_list->t.tv_sec;
|
||||||
|
the_timer_list->t.tv_sec = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
npth_timeradd (&timeout, &curtime, &abstime);
|
||||||
|
tp = &timeout;
|
||||||
|
}
|
||||||
|
|
||||||
res = npth_mutex_unlock (&cache_lock);
|
res = npth_mutex_unlock (&cache_lock);
|
||||||
if (res)
|
if (res)
|
||||||
log_fatal ("failed to release cache mutex: %s\n", strerror (res));
|
log_fatal ("failed to release cache mutex: %s\n", strerror (res));
|
||||||
}
|
|
||||||
|
|
||||||
npth_timersub (&abstime, &curtime, &timeout);
|
return tp;
|
||||||
return &timeout;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -337,6 +533,7 @@ agent_flush_cache (int pincache_only)
|
|||||||
release_data (r->pw);
|
release_data (r->pw);
|
||||||
r->pw = NULL;
|
r->pw = NULL;
|
||||||
r->accessed = 0;
|
r->accessed = 0;
|
||||||
|
update_expiration (r, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -381,7 +578,6 @@ agent_put_cache (ctrl_t ctrl, const char *key, cache_mode_t cache_mode,
|
|||||||
if (DBG_CACHE)
|
if (DBG_CACHE)
|
||||||
log_debug ("agent_put_cache '%s'.%d (mode %d) requested ttl=%d\n",
|
log_debug ("agent_put_cache '%s'.%d (mode %d) requested ttl=%d\n",
|
||||||
key, restricted, cache_mode, ttl);
|
key, restricted, cache_mode, ttl);
|
||||||
housekeeping ();
|
|
||||||
|
|
||||||
if (!ttl)
|
if (!ttl)
|
||||||
{
|
{
|
||||||
@ -433,6 +629,7 @@ agent_put_cache (ctrl_t ctrl, const char *key, cache_mode_t cache_mode,
|
|||||||
err = new_data (data, &r->pw);
|
err = new_data (data, &r->pw);
|
||||||
if (err)
|
if (err)
|
||||||
log_error ("error replacing cache item: %s\n", gpg_strerror (err));
|
log_error ("error replacing cache item: %s\n", gpg_strerror (err));
|
||||||
|
update_expiration (r, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (data) /* Insert. */
|
else if (data) /* Insert. */
|
||||||
@ -454,6 +651,7 @@ agent_put_cache (ctrl_t ctrl, const char *key, cache_mode_t cache_mode,
|
|||||||
{
|
{
|
||||||
r->next = thecache;
|
r->next = thecache;
|
||||||
thecache = r;
|
thecache = r;
|
||||||
|
update_expiration (r, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (err)
|
if (err)
|
||||||
@ -501,7 +699,6 @@ agent_get_cache (ctrl_t ctrl, const char *key, cache_mode_t cache_mode)
|
|||||||
log_debug ("agent_get_cache '%s'.%d (mode %d)%s ...\n",
|
log_debug ("agent_get_cache '%s'.%d (mode %d)%s ...\n",
|
||||||
key, restricted, cache_mode,
|
key, restricted, cache_mode,
|
||||||
last_stored? " (stored cache key)":"");
|
last_stored? " (stored cache key)":"");
|
||||||
housekeeping ();
|
|
||||||
|
|
||||||
for (r=thecache; r; r = r->next)
|
for (r=thecache; r; r = r->next)
|
||||||
{
|
{
|
||||||
@ -523,7 +720,10 @@ agent_get_cache (ctrl_t ctrl, const char *key, cache_mode_t cache_mode)
|
|||||||
* below. Note also that we don't update the accessed time
|
* below. Note also that we don't update the accessed time
|
||||||
* for data items. */
|
* for data items. */
|
||||||
if (r->cache_mode != CACHE_MODE_DATA)
|
if (r->cache_mode != CACHE_MODE_DATA)
|
||||||
|
{
|
||||||
r->accessed = gnupg_get_time ();
|
r->accessed = gnupg_get_time ();
|
||||||
|
update_expiration (r, 0);
|
||||||
|
}
|
||||||
if (DBG_CACHE)
|
if (DBG_CACHE)
|
||||||
log_debug ("... hit\n");
|
log_debug ("... hit\n");
|
||||||
if (r->pw->totallen < 32)
|
if (r->pw->totallen < 32)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user