diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2017-10-19 19:26:29 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2017-10-19 19:27:57 +0100 |
commit | 4d66a68e45cc353e4fa4d5f903fdb151091c5338 (patch) | |
tree | 89568b77728a4476adffdee5f2c700b0c5b042c1 /src | |
parent | 353909278aaee7985784d70c95be50ecc7b3bce4 (diff) | |
download | rspamd-4d66a68e45cc353e4fa4d5f903fdb151091c5338.tar.gz rspamd-4d66a68e45cc353e4fa4d5f903fdb151091c5338.zip |
[CritFix] Fix milter commands pipelining
Diffstat (limited to 'src')
-rw-r--r-- | src/libserver/milter.c | 44 | ||||
-rw-r--r-- | src/libserver/milter_internal.h | 3 |
2 files changed, 37 insertions, 10 deletions
diff --git a/src/libserver/milter.c b/src/libserver/milter.c index d7e7a6a3d..2cced96ec 100644 --- a/src/libserver/milter.c +++ b/src/libserver/milter.c @@ -644,13 +644,24 @@ rspamd_milter_process_command (struct rspamd_milter_session *session, version, actions, protocol); break; case RSPAMD_MILTER_CMD_QUIT: - msg_debug_milter ("quit command, refcount: %d", session->ref.refcount); - priv->state = RSPAMD_MILTER_WANNA_DIE; - REF_RETAIN (session); - priv->fin_cb (priv->fd, session, priv->ud); - REF_RELEASE (session); + if (priv->out_chain) { + msg_debug_milter ("quit command, refcount: %d, " + "some output buffers left - draining", + session->ref.refcount); + + priv->state = RSPAMD_MILTER_WRITE_AND_DIE; + } + else { + msg_debug_milter ("quit command, refcount: %d", + session->ref.refcount); - return FALSE; + priv->state = RSPAMD_MILTER_WANNA_DIE; + REF_RETAIN (session); + priv->fin_cb (priv->fd, session, priv->ud); + REF_RELEASE (session); + + return FALSE; + } break; case RSPAMD_MILTER_CMD_RCPT: msg_debug_milter ("rcpt command"); @@ -935,10 +946,25 @@ rspamd_milter_handle_session (struct rspamd_milter_session *session, return rspamd_milter_consume_input (session, priv); } case RSPAMD_MILTER_WRITE_REPLY: + case RSPAMD_MILTER_WRITE_AND_DIE: if (priv->out_chain == NULL) { - /* We have written everything, so we can read something */ - priv->state = RSPAMD_MILTER_READ_MORE; - rspamd_milter_plan_io (session, priv, EV_READ); + if (priv->state == RSPAMD_MILTER_WRITE_AND_DIE) { + /* Finished writing, let's die finally */ + msg_debug_milter ("output drained, terminating, refcount: %d", + session->ref.refcount); + + /* Session should be destroyed by fin_cb... */ + REF_RETAIN (session); + priv->fin_cb (priv->fd, session, priv->ud); + REF_RELEASE (session); + + return FALSE; + } + else { + /* We have written everything, so we can read something */ + priv->state = RSPAMD_MILTER_READ_MORE; + rspamd_milter_plan_io (session, priv, EV_READ); + } } else { DL_FOREACH_SAFE (priv->out_chain, obuf, obuf_tmp) { diff --git a/src/libserver/milter_internal.h b/src/libserver/milter_internal.h index b2a2dc712..9087bdbfb 100644 --- a/src/libserver/milter_internal.h +++ b/src/libserver/milter_internal.h @@ -48,7 +48,8 @@ struct rspamd_milter_outbuf { enum rspamd_milter_io_state { RSPAMD_MILTER_READ_MORE, RSPAMD_MILTER_WRITE_REPLY, - RSPAMD_MILTER_WANNA_DIE + RSPAMD_MILTER_WANNA_DIE, + RSPAMD_MILTER_WRITE_AND_DIE, }; struct rspamd_milter_private { |