1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-24 15:17:02 +01:00

common: Improve lock strategy for dotlock.

* common/dotlock.c (next_wait_interval): New.
(dotlock_take_unix): Use new function.
(dotlock_take_w32): Ditto.
--

In particular when using a dotlock file for protecting the spawning
and several processes try to spawn the agent or another component, we
often run into long delays.  The solution is to is to exponential
backoff and also to reduce the initial delay from 50ms to 4ms.  We
further limit the maximum wait period to about 2 seconds and then
repeat at intervals of 512, 1024 and 2048ms.  In the wait-forever case
we add a small random value to have different intervals per process.

GnuPG-bug-id: 3380

For testing this code snippet in the spawning function might be
useful:

          const char *s;
          if ((s=getenv("hold_gpg_file")))
            while (!gnupg_access (s, F_OK))
              gnupg_sleep (1);
This commit is contained in:
Werner Koch 2023-10-02 12:53:41 +02:00
parent d546fdd531
commit 45a1ab5017
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B

View File

@ -1024,6 +1024,40 @@ dotlock_is_locked (dotlock_t h)
} }
/* Return the next interval to wait. WTIME and TIMEOUT are pointers
* to the current state and are updated by this function. The
* returned value might be different from the value of WTIME. */
static int
next_wait_interval (int *wtime, long *timeout)
{
int result;
/* Wait until lock has been released. We use retry intervals of 4,
* 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 512, 1024, 2048ms, and
* so on. If wait-forever was requested we add a small random value
* to have different timeouts per process. */
if (!*wtime)
*wtime = 4;
else if (*wtime < 2048)
*wtime *= 2;
else
*wtime = 512;
result = *wtime;
if (*wtime > 8 && *timeout < 0)
result += ((unsigned int)getpid() % 37);
if (*timeout > 0)
{
if (result > *timeout)
result = *timeout;
*timeout -= result;
}
return result;
}
#ifdef HAVE_POSIX_SYSTEM #ifdef HAVE_POSIX_SYSTEM
/* Unix specific code of make_dotlock. Returns 0 on success and -1 on /* Unix specific code of make_dotlock. Returns 0 on success and -1 on
@ -1183,27 +1217,14 @@ dotlock_take_unix (dotlock_t h, long timeout)
if (timeout) if (timeout)
{ {
struct timeval tv; struct timeval tv;
int wtimereal;
/* Wait until lock has been released. We use increasing retry if (ownerchanged)
intervals of 50ms, 100ms, 200ms, 400ms, 800ms, 2s, 4s and 8s wtime = 0; /* Reset because owner chnaged. */
but reset it if the lock owner meanwhile changed. */
if (!wtime || ownerchanged)
wtime = 50;
else if (wtime < 800)
wtime *= 2;
else if (wtime == 800)
wtime = 2000;
else if (wtime < 8000)
wtime *= 2;
if (timeout > 0) wtimereal = next_wait_interval (&wtime, &timeout);
{
if (wtime > timeout)
wtime = timeout;
timeout -= wtime;
}
sumtime += wtime; sumtime += wtimereal;
if (sumtime >= 1500) if (sumtime >= 1500)
{ {
sumtime = 0; sumtime = 0;
@ -1211,9 +1232,8 @@ dotlock_take_unix (dotlock_t h, long timeout)
pid, maybe_dead, maybe_deadlock(h)? _("(deadlock?) "):""); pid, maybe_dead, maybe_deadlock(h)? _("(deadlock?) "):"");
} }
tv.tv_sec = wtimereal / 1000;
tv.tv_sec = wtime / 1000; tv.tv_usec = (wtimereal % 1000) * 1000;
tv.tv_usec = (wtime % 1000) * 1000;
select (0, NULL, NULL, NULL, &tv); select (0, NULL, NULL, NULL, &tv);
goto again; goto again;
} }
@ -1255,28 +1275,14 @@ dotlock_take_w32 (dotlock_t h, long timeout)
if (timeout) if (timeout)
{ {
/* Wait until lock has been released. We use retry intervals of int wtimereal;
50ms, 100ms, 200ms, 400ms, 800ms, 2s, 4s and 8s. */
if (!wtime)
wtime = 50;
else if (wtime < 800)
wtime *= 2;
else if (wtime == 800)
wtime = 2000;
else if (wtime < 8000)
wtime *= 2;
if (timeout > 0) wtimereal = next_wait_interval (&wtime, &timeout);
{
if (wtime > timeout)
wtime = timeout;
timeout -= wtime;
}
if (wtime >= 800) if (wtime >= 800)
my_info_1 (_("waiting for lock %s...\n"), h->lockname); my_info_1 (_("waiting for lock %s...\n"), h->lockname);
Sleep (wtime); Sleep (wtimereal);
goto again; goto again;
} }