git: b6f31ce42c6e - main - mail/exim: fix queue runners that are not starting

From: Kurt Jaeger <pi_at_FreeBSD.org>
Date: Tue, 02 Jan 2024 12:50:26 UTC
The branch main has been updated by pi:

URL: https://cgit.FreeBSD.org/ports/commit/?id=b6f31ce42c6e27b91a666f0a7a550e945124f3b5

commit b6f31ce42c6e27b91a666f0a7a550e945124f3b5
Author:     Niels Kobschätzki <niels@kobschaetzki.net>
AuthorDate: 2024-01-02 11:49:10 +0000
Commit:     Kurt Jaeger <pi@FreeBSD.org>
CommitDate: 2024-01-02 11:49:10 +0000

    mail/exim: fix queue runners that are not starting
    
    upstream:
    - https://bugs.exim.org/show_bug.cgi?id=3046
    - https://github.com/Exim/exim/commit/ced0bbf43860a2e4ddc88e3ba20ae731324586c5
    
    PR:             275635
    Author:         Niels Kobschätzki <niels@kobschaetzki.net>
    Approved-by:    fluffy (implicit)
---
 mail/exim/Makefile                        |   2 +-
 mail/exim/files/patch-periodic-queue-runs | 212 ++++++++++++++++++++++++++++++
 2 files changed, 213 insertions(+), 1 deletion(-)

diff --git a/mail/exim/Makefile b/mail/exim/Makefile
index d23dec014c7e..c74df7d216b6 100644
--- a/mail/exim/Makefile
+++ b/mail/exim/Makefile
@@ -1,6 +1,6 @@
 PORTNAME=	exim
 PORTVERSION?=	${EXIM_VERSION}
-PORTREVISION?=	0
+PORTREVISION?=	1
 CATEGORIES=	mail
 MASTER_SITES=	EXIM:exim
 MASTER_SITE_SUBDIR=	/exim4/:exim \
diff --git a/mail/exim/files/patch-periodic-queue-runs b/mail/exim/files/patch-periodic-queue-runs
new file mode 100644
index 000000000000..b19336269151
--- /dev/null
+++ b/mail/exim/files/patch-periodic-queue-runs
@@ -0,0 +1,212 @@
+From ced0bbf43860a2e4ddc88e3ba20ae731324586c5 Mon Sep 17 00:00:00 2001
+From: Jeremy Harris <jgh146exb@wizmail.org>
+Date: Fri, 15 Dec 2023 21:05:32 +0000
+Subject: [PATCH] Fix periodic queue runs.  Bug 3046
+
+Broken-by: 7d5055276a22
+---
+ doc/ChangeLog |  5 ++++
+ src/daemon.c      | 58 ++++++++++++++++++++++++++++---------------
+ src/structs.h     |  2 +-
+ 3 files changed, 44 insertions(+), 21 deletions(-)
+
+diff --git a/doc/doc-txt/ChangeLog b/doc/doc-txt/ChangeLog
+index 85064cc8d..c46f3b8c0 100644
+--- doc/ChangeLog
++++ doc/ChangeLog
+@@ -53,6 +53,11 @@ JH/10 Bug 3058: Ensure that a failing expansion in a router "set" option defers
+       the routing operation.  Previously it would silently stop routing the
+       message.
+ 
++JH/11 Bug 3046: Fix queue-runs.  Previously, the arrivel of a notification or
++      info-request event close in time to a scheduled run timer could result in
++      the latter being missed, and no further queue scheduled runs being
++      initiated.  This ouwld be more likely on high-load systems.
++
+ 
+ 
+ Exim version 4.97
+diff --git a/src/src/daemon.c b/src/src/daemon.c
+index f2183c735..aff05120a 100644
+--- src/daemon.c
++++ src/daemon.c
+@@ -1258,10 +1258,9 @@ static const uschar * queuerun_msg_qname;
+ 
+ /* The notifier socket has something to read. Pull the message from it, decode
+ and do the action.
++*/
+ 
+-Return TRUE if a sigalrm should be emulated */
+-
+-static BOOL
++static void
+ daemon_notification(void)
+ {
+ uschar buf[256], cbuf[256];
+@@ -1277,8 +1276,8 @@ struct msghdr msg = { .msg_name = &sa_un,
+ ssize_t sz;
+ 
+ buf[sizeof(buf)-1] = 0;
+-if ((sz = recvmsg(daemon_notifier_fd, &msg, 0)) <= 0) return FALSE;
+-if (sz >= sizeof(buf)) return FALSE;
++if ((sz = recvmsg(daemon_notifier_fd, &msg, 0)) <= 0) return;
++if (sz >= sizeof(buf)) return;
+ 
+ #ifdef notdef
+ debug_printf("addrlen %d\n", msg.msg_namelen);
+@@ -1351,7 +1350,7 @@ switch (buf[0])
+ 	  : !buf[1+MESSAGE_ID_LENGTH+1]
+ 	 )
+ 	{ queuerun_msg_qname = q->name; break; }
+-    return TRUE;
++    return;
+ #endif
+ 
+   case NOTIFY_QUEUE_SIZE_REQ:
+@@ -1373,7 +1372,7 @@ switch (buf[0])
+     regex_at_daemon(buf);
+     break;
+   }
+-return FALSE;
++return;
+ }
+ 
+ 
+@@ -1432,7 +1431,7 @@ for (qrunner * q = qrunners, * next; q; q = next)
+   if (sorted)
+     {
+     qrunner ** p = &sorted;
+-    for (qrunner * qq; qq = *p; p = &(qq->next))
++    for (qrunner * qq; qq = *p; p = &qq->next)
+       if (  q->next_tick < qq->next_tick
+ 	 || q->next_tick == qq->next_tick && q->interval < qq->interval
+ 	 )
+@@ -1451,6 +1450,13 @@ qrunners = sorted;
+ return qrunners ? qrunners->next_tick - time(NULL) : 0;
+ }
+ 
++/* See if we can do a queue run.  If policy limit permit, kick one off.
++If both notification and timer events are present, handle the former
++and leave the timer outstanding.
++
++Return the number of seconds until the next due runner.
++*/
++
+ static int
+ daemon_qrun(int local_queue_run_max, struct pollfd * fd_polls, int listen_socket_count)
+ {
+@@ -1464,13 +1470,16 @@ DEBUG(D_any) debug_printf("%s received\n",
+ enough queue runners on the go. If we are not running as root, a re-exec is
+ required. In the calling process, restart the alamr timer for the next run.  */
+ 
+-if (is_multiple_qrun())
++if (is_multiple_qrun())				/* we are managing periodic runs */
+   if (local_queue_run_max <= 0 || queue_run_count < local_queue_run_max)
+     {
+     qrunner * q = NULL;
+ 
+ #ifndef DISABLE_QUEUE_RAMP
+-    if (*queuerun_msgid)	/* See if we can start another runner for this queue */
++    /* If this is a triggered run for a specific message, see if we can start
++    another runner for this queue. */
++
++    if (*queuerun_msgid)
+       {
+       for (qrunner * qq = qrunners; qq; qq = qq->next)
+ 	if (qq->name == queuerun_msg_qname)
+@@ -1481,13 +1490,13 @@ if (is_multiple_qrun())
+       }
+     else
+ #endif
+-      /* In order of run priority, find the first queue for which we can start
+-      a runner */
++      /* Normal periodic runL in order of run priority, find the first queue
++      for which we can start a runner */
+ 
+       for (q = qrunners; q; q = q->next)
+ 	if (q->run_count < q->run_max) break;
+ 
+-    if (q)
++    if (q)					/* found a queue to run */
+       {
+       pid_t pid;
+ 
+@@ -1619,19 +1628,23 @@ if (is_multiple_qrun())
+       }
+     }
+ 
+-sigalrm_seen = FALSE;
++/* The queue run has been initiated (unless we were already running enough) */
++
+ #ifndef DISABLE_QUEUE_RAMP
+-if (*queuerun_msgid)		/* it was a fast-ramp kick */
++if (*queuerun_msgid)		/* it was a fast-ramp kick; dealt with */
+   *queuerun_msgid = 0;
+ else				/* periodic or one-time queue run */
+ #endif
+-  {		/* Impose a minimum 1s tick, even when a run was outstanding */
++  /* Set up next timer callback. Impose a minimum 1s tick,
++  even when a run was outstanding */
++  {
+   int interval = next_qrunner_interval();
+   if (interval <= 0) interval = 1;
+ 
++  sigalrm_seen = FALSE;
+   if (qrunners)			/* there are still periodic qrunners */
+     {
+-    ALARM(interval);
++    ALARM(interval);		/* set up next qrun tick */
+     return interval;
+     }
+   }
+@@ -2612,7 +2625,7 @@ for (;;)
+ 
+   The other option is that we have an inetd wait timeout specified to -bw. */
+ 
+-  if (sigalrm_seen)
++  if (sigalrm_seen || *queuerun_msgid)
+     if (inetd_wait_timeout > 0)
+       daemon_inetd_wtimeout(last_connection_time);	/* Might not return */
+     else
+@@ -2641,7 +2654,9 @@ for (;;)
+     select() was interrupted so that we reap the child. This might still leave
+     a small window when a SIGCHLD could get lost. However, since we use SIGCHLD
+     only to do the reaping more quickly, it shouldn't result in anything other
+-    than a delay until something else causes a wake-up. */
++    than a delay until something else causes a wake-up.
++    For the normal case, wait for either a pollable fd (eg. new connection) or
++    or a SIGALRM (for a periodic queue run) */
+ 
+     if (sigchld_seen)
+       {
+@@ -2706,10 +2721,13 @@ for (;;)
+ 	  break;	/* to top of daemon loop */
+ 	  }
+ #endif
++	/* Handle the daemon-notifier socket.  If it was a fast-ramp
++	notification then queuerun_msgid will have a nonzerolength string. */
++
+ 	if (dnotify_poll && dnotify_poll->revents & POLLIN)
+ 	  {
+ 	  dnotify_poll->revents = 0;
+-	  sigalrm_seen = daemon_notification();
++	  daemon_notification();
+ 	  break;	/* to top of daemon loop */
+ 	  }
+ 	for (struct pollfd * p = fd_polls; p < fd_polls + listen_socket_count;
+diff --git a/src/src/structs.h b/src/src/structs.h
+index 209d657c6..256560ef8 100644
+--- src/structs.h
++++ src/structs.h
+@@ -964,7 +964,7 @@ typedef struct qrunner {
+   struct qrunner * next;	/* list sorted by next tick */
+ 
+   uschar *	name;		/* NULL for the default queue */
+-  unsigned	interval;	/* tick rate, seconds */
++  unsigned	interval;	/* tick rate, seconds. Zero for a one-time run */
+   time_t	next_tick;	/* next run should, or should have, start(ed) */
+   unsigned	run_max;	/* concurrent queue runner limit */
+   unsigned	run_count;	/* current runners */
+-- 
+2.30.2
+