Skip to content

Commit c577d8c

Browse files
committed
smtp, tls upgrade fix
There were two places in the code that tried to connect the SSL filter, e.g. do the TLS handshake, but only one changed stmp state to EHLO afterwards. Depending on timing, the wrong path was taken and the connection was hanging, waiting for a server reply to a command not sent. Do the upgrade to tls in one place and update connection filter and smtps protocol handler at the same time. Always transition to EHLO on success. refs curl#16189
1 parent 553248f commit c577d8c

File tree

1 file changed

+20
-32
lines changed

1 file changed

+20
-32
lines changed

lib/smtp.c

+20-32
Original file line numberDiff line numberDiff line change
@@ -189,19 +189,6 @@ static const struct SASLproto saslsmtp = {
189189
SASL_FLAG_BASE64 /* Configuration flags */
190190
};
191191

192-
#ifdef USE_SSL
193-
static void smtp_to_smtps(struct connectdata *conn)
194-
{
195-
/* Change the connection handler */
196-
conn->handler = &Curl_handler_smtps;
197-
198-
/* Set the connection's upgraded to TLS flag */
199-
conn->bits.tls_upgraded = TRUE;
200-
}
201-
#else
202-
#define smtp_to_smtps(x) Curl_nop_stmt
203-
#endif
204-
205192
/***********************************************************************
206193
*
207194
* smtp_endofresp()
@@ -402,25 +389,31 @@ static CURLcode smtp_perform_upgrade_tls(struct Curl_easy *data)
402389
CURLcode result;
403390
bool ssldone = FALSE;
404391

392+
#ifdef USE_SSL
393+
DEBUGASSERT(smtpc->state == SMTP_UPGRADETLS);
405394
if(!Curl_conn_is_ssl(conn, FIRSTSOCKET)) {
406395
result = Curl_ssl_cfilter_add(data, conn, FIRSTSOCKET);
407396
if(result)
408397
goto out;
398+
/* Change the connection handler and SMTP state */
399+
conn->handler = &Curl_handler_smtps;
400+
conn->bits.tls_upgraded = TRUE;
409401
}
410402

403+
DEBUGASSERT(!smtpc->ssldone);
411404
result = Curl_conn_connect(data, FIRSTSOCKET, FALSE, &ssldone);
412-
if(!result) {
405+
DEBUGF(infof(data, "smtp_perform_upgrade_tls, connect -> %d, %d",
406+
result, ssldone));
407+
if(!result && ssldone) {
413408
smtpc->ssldone = ssldone;
414-
if(smtpc->state != SMTP_UPGRADETLS)
415-
smtp_state(data, SMTP_UPGRADETLS);
416-
417-
if(smtpc->ssldone) {
418-
smtp_to_smtps(conn);
419-
result = smtp_perform_ehlo(data);
420-
}
409+
/* perform EHLO now, changes smpt->state out of SMTP_UPGRADETLS */
410+
result = smtp_perform_ehlo(data);
421411
}
422412
out:
423413
return result;
414+
#else
415+
return CURLE_NOT_BUILT_IN;
416+
#endif
424417
}
425418

426419
/***********************************************************************
@@ -875,7 +868,7 @@ static CURLcode smtp_state_starttls_resp(struct Curl_easy *data,
875868
result = smtp_perform_authentication(data);
876869
}
877870
else
878-
result = smtp_perform_upgrade_tls(data);
871+
smtp_state(data, SMTP_UPGRADETLS);
879872

880873
return result;
881874
}
@@ -1204,8 +1197,11 @@ static CURLcode smtp_statemachine(struct Curl_easy *data,
12041197

12051198
/* Busy upgrading the connection; right now all I/O is SSL/TLS, not SMTP */
12061199
upgrade_tls:
1207-
if(smtpc->state == SMTP_UPGRADETLS)
1208-
return smtp_perform_upgrade_tls(data);
1200+
if(smtpc->state == SMTP_UPGRADETLS) {
1201+
result = smtp_perform_upgrade_tls(data);
1202+
if(result || (smtpc->state == SMTP_UPGRADETLS))
1203+
return result;
1204+
}
12091205

12101206
/* Flush any data that needs to be sent */
12111207
if(pp->sendleft)
@@ -1288,14 +1284,6 @@ static CURLcode smtp_multi_statemach(struct Curl_easy *data, bool *done)
12881284
struct connectdata *conn = data->conn;
12891285
struct smtp_conn *smtpc = &conn->proto.smtpc;
12901286

1291-
if(Curl_conn_is_ssl(conn, FIRSTSOCKET) && !smtpc->ssldone) {
1292-
bool ssldone = FALSE;
1293-
result = Curl_conn_connect(data, FIRSTSOCKET, FALSE, &ssldone);
1294-
smtpc->ssldone = ssldone;
1295-
if(result || !smtpc->ssldone)
1296-
return result;
1297-
}
1298-
12991287
result = Curl_pp_statemach(data, &smtpc->pp, FALSE, FALSE);
13001288
*done = (smtpc->state == SMTP_STOP);
13011289

0 commit comments

Comments
 (0)