diff options
-rw-r--r-- | usr.sbin/smtpd/client.c | 68 | ||||
-rw-r--r-- | usr.sbin/smtpd/client.h | 13 |
2 files changed, 40 insertions, 41 deletions
diff --git a/usr.sbin/smtpd/client.c b/usr.sbin/smtpd/client.c index a1589f97705..0ea810abd34 100644 --- a/usr.sbin/smtpd/client.c +++ b/usr.sbin/smtpd/client.c @@ -1,4 +1,4 @@ -/* $OpenBSD: client.c,v 1.25 2010/01/02 13:42:42 jacekm Exp $ */ +/* $OpenBSD: client.c,v 1.26 2010/01/02 16:41:19 jacekm Exp $ */ /* * Copyright (c) 2009 Jacek Masiulaniec <jacekm@dobremiasto.net> @@ -77,13 +77,13 @@ client_init(int fd, int body, char *ehlo, int verbose) { struct smtp_client *sp = NULL; struct client_cmd *c; + socklen_t len; if ((sp = calloc(1, sizeof(*sp))) == NULL) fatal(NULL); if (ehlo == NULL || *ehlo == '\0') { char buf[NI_MAXHOST]; struct sockaddr_storage sa; - socklen_t len; len = sizeof(sa); if (getsockname(fd, (struct sockaddr *)&sa, &len)) @@ -104,7 +104,10 @@ client_init(int fd, int body, char *ehlo, int verbose) sp->timeout.tv_sec = 300; msgbuf_init(&sp->w); sp->w.fd = fd; - sp->sndlowat = -1; + sp->flags = CLIENT_FLAG_FIRSTTIME; + len = sizeof(sp->sndlowat); + if (getsockopt(fd, SOL_SOCKET, SO_SNDLOWAT, &sp->sndlowat, &len) == -1) + fatal("client_init: getsockopt"); sp->exts[CLIENT_EXT_STARTTLS].want = 1; sp->exts[CLIENT_EXT_STARTTLS].must = 1; @@ -168,7 +171,7 @@ cmd_free(struct client_cmd *cmd) void client_ssl_smtps(struct smtp_client *sp) { - sp->ssl_handshake = 1; + sp->flags |= CLIENT_FLAG_HANDSHAKING; sp->exts[CLIENT_EXT_STARTTLS].want = 0; sp->exts[CLIENT_EXT_STARTTLS].must = 0; } @@ -297,14 +300,10 @@ int client_talk(struct smtp_client *sp, int writable) { struct client_cmd *c; - socklen_t len; /* first call -> complete the initialisation */ - if (sp->sndlowat == -1) { - len = sizeof(sp->sndlowat); - if (getsockopt(sp->w.fd, SOL_SOCKET, SO_SNDLOWAT, - &sp->sndlowat, &len) == -1) - fatal("client_talk: getsockopt"); + if (sp->flags & CLIENT_FLAG_FIRSTTIME) { + sp->flags &= ~CLIENT_FLAG_FIRSTTIME; c = cmd_new(CLIENT_DATA, "DATA"); TAILQ_INSERT_TAIL(&sp->cmdsendq, c, entry); @@ -320,8 +319,7 @@ client_talk(struct smtp_client *sp, int writable) } #ifndef CLIENT_NO_SSL - /* transition to ssl requested? */ - if (sp->ssl_handshake) { + if (sp->flags & CLIENT_FLAG_HANDSHAKING) { if (sp->ssl == NULL) { log_debug("client: ssl handshake started"); sp->ssl = ssl_client_init(sp->w.fd, @@ -364,7 +362,7 @@ client_read(struct smtp_client *sp) sp->cmdi--; /* if dying, ignore all replies as we wait for an EOF. */ - if (!sp->dying) + if (!(sp->flags & CLIENT_FLAG_DYING)) client_get_reply(sp, cmd, &ret); cmd_free(cmd); @@ -416,7 +414,7 @@ client_get_reply(struct smtp_client *sp, struct client_cmd *cmd, int *ret) if (client_use_extensions(sp) < 0) client_quit(sp); } else - sp->ssl_handshake = 1; + sp->flags |= CLIENT_FLAG_HANDSHAKING; return; case CLIENT_AUTH: @@ -431,7 +429,7 @@ client_get_reply(struct smtp_client *sp, struct client_cmd *cmd, int *ret) case CLIENT_RCPTTO: if (*sp->reply == '2') - sp->rcptokay++; + sp->flags |= CLIENT_FLAG_RCPTOKAY; else { sp->rcptfail = cmd->data; *ret = CLIENT_RCPT_FAIL; @@ -442,7 +440,7 @@ client_get_reply(struct smtp_client *sp, struct client_cmd *cmd, int *ret) if (*sp->reply != '3') { client_status(sp, "%s", sp->reply); client_quit(sp); - } else if (sp->rcptokay > 0) { + } else if (sp->flags & CLIENT_FLAG_RCPTOKAY) { sp->content = sp->head; sp->head = NULL; if (sp->content == NULL) @@ -538,35 +536,33 @@ client_ssl_connect(struct smtp_client *sp) return (CLIENT_WANT_WRITE); case SSL_ERROR_NONE: - sp->ssl_handshake = 0; - break; + log_debug("client: ssl handshake completed"); + sp->flags &= ~CLIENT_FLAG_HANDSHAKING; + + if (sp->exts[CLIENT_EXT_STARTTLS].want) { + c = cmd_new(CLIENT_EHLO, "EHLO %s", sp->ehlo); + TAILQ_INSERT_HEAD(&sp->cmdsendq, c, entry); + } + sp->exts[CLIENT_EXT_STARTTLS].done = 1; + return client_poll(sp); default: log_debug("client: ssl handshake failed"); - sp->ssl_handshake = 0; + sp->flags &= ~CLIENT_FLAG_HANDSHAKING; if (sp->exts[CLIENT_EXT_STARTTLS].want) { sp->exts[CLIENT_EXT_STARTTLS].fail = 1; SSL_free(sp->ssl); sp->ssl = NULL; - if (client_use_extensions(sp) == 0) + if (client_use_extensions(sp) < 0) + return (CLIENT_DONE); + else return client_poll(sp); - } else + } else { client_status(sp, "130 SSL_connect error"); - - return (CLIENT_DONE); - } - - log_debug("client: ssl handshake completed"); - - if (sp->exts[CLIENT_EXT_STARTTLS].want) { - c = cmd_new(CLIENT_EHLO, "EHLO %s", sp->ehlo); - TAILQ_INSERT_HEAD(&sp->cmdsendq, c, entry); + return (CLIENT_DONE); + } } - - sp->exts[CLIENT_EXT_STARTTLS].done = 1; - - return client_poll(sp); } #endif @@ -655,7 +651,7 @@ client_status(struct smtp_client *sp, char *fmt, ...) { va_list ap; - if (sp->dying) + if (sp->flags & CLIENT_FLAG_DYING) return; va_start(ap, fmt); @@ -833,7 +829,7 @@ client_quit(struct smtp_client *sp) TAILQ_REMOVE(&sp->cmdsendq, cmd, entry); cmd_free(cmd); } - sp->dying = 1; + sp->flags |= CLIENT_FLAG_DYING; } /* diff --git a/usr.sbin/smtpd/client.h b/usr.sbin/smtpd/client.h index e10458b3afe..34fa90bed34 100644 --- a/usr.sbin/smtpd/client.h +++ b/usr.sbin/smtpd/client.h @@ -1,4 +1,4 @@ -/* $OpenBSD: client.h,v 1.10 2010/01/02 13:42:42 jacekm Exp $ */ +/* $OpenBSD: client.h,v 1.11 2010/01/02 16:41:19 jacekm Exp $ */ /* * Copyright (c) 2009 Jacek Masiulaniec <jacekm@dobremiasto.net> @@ -70,20 +70,24 @@ struct client_auth { size_t keysz; }; +/* session flags */ +#define CLIENT_FLAG_FIRSTTIME 0x1 +#define CLIENT_FLAG_HANDSHAKING 0x2 +#define CLIENT_FLAG_RCPTOKAY 0x4 +#define CLIENT_FLAG_DYING 0x8 + struct smtp_client { size_t cmdi; /* iterator */ size_t cmdw; /* window */ struct cmdqueue cmdsendq; /* cmds to send */ struct cmdqueue cmdrecvq; /* replies waited for */ + int flags; void *rcptfail; - size_t rcptokay; - char *ehlo; char reply[1024]; struct buf_read r; struct msgbuf w; - short ssl_handshake; void *ssl; int sndlowat; struct timeval timeout; @@ -97,7 +101,6 @@ struct smtp_client { struct client_auth auth; char status[1024]; - short dying; }; struct smtp_client *client_init(int, int, char *, int); |