#include <sendmail.h>
SM_RCSID("@(#)$Id: util.c,v 8.416 2009/12/18 17:05:26 ca Exp $")
#include <sm/sendmail.h>
#include <sysexits.h>
#include <sm/xtrap.h>
char *
newstr(s)
const char *s;
{
size_t l;
char *n;
l = strlen(s);
SM_ASSERT(l + 1 > l);
n = xalloc(l + 1);
sm_strlcpy(n, s, l + 1);
return n;
}
char *
addquotes(s, rpool)
char *s;
SM_RPOOL_T *rpool;
{
int len = 0;
char c;
char *p = s, *q, *r;
if (s == NULL)
return NULL;
while ((c = *p++) != '\0')
{
len++;
if (c == '\\' || c == '"')
len++;
}
q = r = sm_rpool_malloc_x(rpool, len + 3);
p = s;
*q++ = '"';
while ((c = *p++) != '\0')
{
if (c == '\\' || c == '"')
*q++ = '\\';
*q++ = c;
}
*q++ = '"';
*q = '\0';
return r;
}
void
stripbackslash(s)
char *s;
{
char *p, *q, c;
if (s == NULL || *s == '\0')
return;
p = q = s;
while (*p == '\\' && (p[1] == '\\' || (isascii(p[1]) && isalnum(p[1]))))
p++;
do
{
c = *q++ = *p++;
} while (c != '\0');
}
bool
rfc822_string(s)
char *s;
{
bool quoted = false;
int commentlev = 0;
char *c = s;
if (s == NULL)
return false;
while (*c != '\0')
{
if (*c == '\\')
{
c++;
if (*c == '\0')
return false;
}
else if (commentlev == 0 && *c == '"')
quoted = !quoted;
else if (!quoted)
{
if (*c == ')')
{
if (commentlev == 0)
return false;
else
commentlev--;
}
else if (*c == '(')
commentlev++;
else if (commentlev == 0 &&
strchr(MustQuoteChars, *c) != NULL)
return false;
}
c++;
}
return !quoted && commentlev == 0;
}
bool
shorten_rfc822_string(string, length)
char *string;
size_t length;
{
bool backslash = false;
bool modified = false;
bool quoted = false;
size_t slen;
int parencount = 0;
char *ptr = string;
slen = strlen(string);
if (length == 0 || slen < length)
length = slen;
while (*ptr != '\0')
{
if (backslash)
{
backslash = false;
goto increment;
}
if (*ptr == '\\')
backslash = true;
else if (*ptr == '(')
{
if (!quoted)
parencount++;
}
else if (*ptr == ')')
{
if (--parencount < 0)
parencount = 0;
}
if (parencount <= 0 && *ptr == '"')
quoted = !quoted;
increment:
if (length - (ptr - string) <= (size_t) ((backslash ? 1 : 0) +
parencount +
(quoted ? 1 : 0)))
{
if (*ptr == '\\')
backslash = false;
else if (*ptr == '(' && !quoted)
parencount--;
else if (*ptr == '"' && parencount == 0)
quoted = false;
break;
}
ptr++;
}
while (parencount-- > 0)
{
if (*ptr != ')')
{
modified = true;
*ptr = ')';
}
ptr++;
}
if (quoted)
{
if (*ptr != '"')
{
modified = true;
*ptr = '"';
}
ptr++;
}
if (*ptr != '\0')
{
modified = true;
*ptr = '\0';
}
return modified;
}
char *
find_character(string, character)
char *string;
int character;
{
bool backslash = false;
bool quoted = false;
int parencount = 0;
while (string != NULL && *string != '\0')
{
if (backslash)
{
backslash = false;
if (!quoted && character == '\\' && *string == '\\')
break;
string++;
continue;
}
switch (*string)
{
case '\\':
backslash = true;
break;
case '(':
if (!quoted)
parencount++;
break;
case ')':
if (--parencount < 0)
parencount = 0;
break;
}
if (parencount > 0)
{
string++;
continue;
}
if (*string == '"')
quoted = !quoted;
else if (*string == character && !quoted)
break;
string++;
}
return string;
}
int
check_bodytype(bodytype)
char *bodytype;
{
if (bodytype == NULL)
return BODYTYPE_NONE;
if (sm_strcasecmp(bodytype, "7BIT") == 0)
return BODYTYPE_7BIT;
if (sm_strcasecmp(bodytype, "8BITMIME") == 0)
return BODYTYPE_8BITMIME;
return BODYTYPE_ILLEGAL;
}
void
truncate_at_delim(str, len, delim)
char *str;
size_t len;
int delim;
{
char *p;
if (str == NULL || len == 0 || strlen(str) < len)
return;
*(str + len - 1) = '\0';
while ((p = strrchr(str, delim)) != NULL)
{
*p = '\0';
if (p - str + 4 < len)
{
*p++ = (char) delim;
*p = '\0';
(void) sm_strlcat(str, "...", len);
return;
}
}
if (len > 3)
(void) sm_strlcpy(str, "...", len);
else
str[0] = '\0';
}
char *
#if SM_HEAP_CHECK
xalloc_tagged(sz, file, line)
register int sz;
char *file;
int line;
#else
xalloc(sz)
register int sz;
#endif
{
register char *p;
SM_REQUIRE(sz >= 0);
if (sz <= 0)
sz = 1;
sm_xtrap_raise_x(&SmHeapOutOfMemory);
p = sm_malloc_tagged((unsigned) sz, file, line, sm_heap_group());
if (p == NULL)
{
sm_exc_raise_x(&SmHeapOutOfMemory);
}
return p;
}
char **
copyplist(list, copycont, rpool)
char **list;
bool copycont;
SM_RPOOL_T *rpool;
{
register char **vp;
register char **newvp;
for (vp = list; *vp != NULL; vp++)
continue;
vp++;
newvp = (char **) sm_rpool_malloc_x(rpool, (vp - list) * sizeof(*vp));
memmove((char *) newvp, (char *) list, (int) (vp - list) * sizeof(*vp));
if (copycont)
{
for (vp = newvp; *vp != NULL; vp++)
*vp = sm_rpool_strdup_x(rpool, *vp);
}
return newvp;
}
ADDRESS *
copyqueue(addr, rpool)
ADDRESS *addr;
SM_RPOOL_T *rpool;
{
register ADDRESS *newaddr;
ADDRESS *ret;
register ADDRESS **tail = &ret;
while (addr != NULL)
{
if (!QS_IS_DEAD(addr->q_state))
{
newaddr = (ADDRESS *) sm_rpool_malloc_x(rpool,
sizeof(*newaddr));
STRUCTCOPY(*addr, *newaddr);
*tail = newaddr;
tail = &newaddr->q_next;
}
addr = addr->q_next;
}
*tail = NULL;
return ret;
}
static SM_FILE_T *Pidf = NULL;
void
log_sendmail_pid(e)
ENVELOPE *e;
{
long sff;
char pidpath[MAXPATHLEN];
extern char *CommandLineArgs;
sff = SFF_NOLINK|SFF_ROOTOK|SFF_REGONLY|SFF_CREAT|SFF_NBLOCK;
if (TrustedUid != 0 && RealUid == TrustedUid)
sff |= SFF_OPENASROOT;
expand(PidFile, pidpath, sizeof(pidpath), e);
Pidf = safefopen(pidpath, O_WRONLY|O_TRUNC, FileMode, sff);
if (Pidf == NULL)
{
if (errno == EWOULDBLOCK)
sm_syslog(LOG_ERR, NOQID,
"unable to write pid to %s: file in use by another process",
pidpath);
else
sm_syslog(LOG_ERR, NOQID,
"unable to write pid to %s: %s",
pidpath, sm_errstring(errno));
}
else
{
PidFilePid = getpid();
(void) sm_io_fprintf(Pidf, SM_TIME_DEFAULT, "%ld\n",
(long) PidFilePid);
(void) sm_io_fprintf(Pidf, SM_TIME_DEFAULT, "%s\n",
CommandLineArgs);
(void) sm_io_flush(Pidf, SM_TIME_DEFAULT);
}
if (LogLevel > 9)
sm_syslog(LOG_INFO, NOQID, "started as: %s", CommandLineArgs);
}
void
close_sendmail_pid()
{
if (Pidf == NULL)
return;
(void) sm_io_close(Pidf, SM_TIME_DEFAULT);
Pidf = NULL;
}
void
set_delivery_mode(mode, e)
int mode;
ENVELOPE *e;
{
char buf[2];
e->e_sendmode = (char) mode;
buf[0] = (char) mode;
buf[1] = '\0';
macdefine(&e->e_macro, A_TEMP, macid("{deliveryMode}"), buf);
}
void
set_op_mode(mode)
int mode;
{
char buf[2];
extern ENVELOPE BlankEnvelope;
OpMode = (char) mode;
buf[0] = (char) mode;
buf[1] = '\0';
macdefine(&BlankEnvelope.e_macro, A_TEMP, MID_OPMODE, buf);
}
void
printav(fp, av)
SM_FILE_T *fp;
char **av;
{
while (*av != NULL)
{
if (tTd(0, 44))
sm_dprintf("\n\t%08lx=", (unsigned long) *av);
else
(void) sm_io_putc(fp, SM_TIME_DEFAULT, ' ');
if (tTd(0, 99))
sm_dprintf("%s", str2prt(*av++));
else
xputs(fp, *av++);
}
(void) sm_io_putc(fp, SM_TIME_DEFAULT, '\n');
}
void
xputs(fp, s)
SM_FILE_T *fp;
const char *s;
{
int c;
struct metamac *mp;
bool shiftout = false;
extern struct metamac MetaMacros[];
static SM_DEBUG_T DebugANSI = SM_DEBUG_INITIALIZER("ANSI",
"@(#)$Debug: ANSI - enable reverse video in debug output $");
if (sm_debug_unknown(&DebugANSI))
{
if (sm_debug_active(&DebugANSI, 1))
{
TermEscape.te_rv_on = "\033[7m";
TermEscape.te_normal = "\033[0m";
}
else
{
TermEscape.te_rv_on = "";
TermEscape.te_normal = "";
}
}
if (s == NULL)
{
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, "%s<null>%s",
TermEscape.te_rv_on, TermEscape.te_normal);
return;
}
while ((c = (*s++ & 0377)) != '\0')
{
if (shiftout)
{
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, "%s",
TermEscape.te_normal);
shiftout = false;
}
if (!isascii(c) && !tTd(84, 1))
{
if (c == MATCHREPL)
{
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT,
"%s$",
TermEscape.te_rv_on);
shiftout = true;
if (*s == '\0')
continue;
c = *s++ & 0377;
goto printchar;
}
if (c == MACROEXPAND || c == MACRODEXPAND)
{
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT,
"%s$",
TermEscape.te_rv_on);
if (c == MACRODEXPAND)
(void) sm_io_putc(fp,
SM_TIME_DEFAULT, '&');
shiftout = true;
if (*s == '\0')
continue;
if (strchr("=~&?", *s) != NULL)
(void) sm_io_putc(fp,
SM_TIME_DEFAULT,
*s++);
if (bitset(0200, *s))
(void) sm_io_fprintf(fp,
SM_TIME_DEFAULT,
"{%s}",
macname(bitidx(*s++)));
else
(void) sm_io_fprintf(fp,
SM_TIME_DEFAULT,
"%c",
*s++);
continue;
}
for (mp = MetaMacros; mp->metaname != '\0'; mp++)
{
if (bitidx(mp->metaval) == c)
{
(void) sm_io_fprintf(fp,
SM_TIME_DEFAULT,
"%s$%c",
TermEscape.te_rv_on,
mp->metaname);
shiftout = true;
break;
}
}
if (c == MATCHCLASS || c == MATCHNCLASS)
{
if (bitset(0200, *s))
(void) sm_io_fprintf(fp,
SM_TIME_DEFAULT,
"{%s}",
macname(bitidx(*s++)));
else if (*s != '\0')
(void) sm_io_fprintf(fp,
SM_TIME_DEFAULT,
"%c",
*s++);
}
if (mp->metaname != '\0')
continue;
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, "%sM-",
TermEscape.te_rv_on);
shiftout = true;
c &= 0177;
}
printchar:
if (isascii(c) && isprint(c))
{
(void) sm_io_putc(fp, SM_TIME_DEFAULT, c);
continue;
}
switch (c)
{
case '\n':
c = 'n';
break;
case '\r':
c = 'r';
break;
case '\t':
c = 't';
break;
}
if (!shiftout)
{
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, "%s",
TermEscape.te_rv_on);
shiftout = true;
}
if (isascii(c) && isprint(c))
{
(void) sm_io_putc(fp, SM_TIME_DEFAULT, '\\');
(void) sm_io_putc(fp, SM_TIME_DEFAULT, c);
}
else if (tTd(84, 2))
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, " %o ", c);
else if (tTd(84, 1))
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, " %#x ", c);
else if (!isascii(c) && !tTd(84, 1))
{
(void) sm_io_putc(fp, SM_TIME_DEFAULT, '^');
(void) sm_io_putc(fp, SM_TIME_DEFAULT, c ^ 0100);
}
}
if (shiftout)
(void) sm_io_fprintf(fp, SM_TIME_DEFAULT, "%s",
TermEscape.te_normal);
(void) sm_io_flush(fp, SM_TIME_DEFAULT);
}
void
makelower(p)
register char *p;
{
register char c;
if (p == NULL)
return;
for (; (c = *p) != '\0'; p++)
if (isascii(c) && isupper(c))
*p = tolower(c);
}
void
fixcrlf(line, stripnl)
char *line;
bool stripnl;
{
register char *p;
p = strchr(line, '\n');
if (p == NULL)
return;
if (p > line && p[-1] == '\r')
p--;
if (!stripnl)
*p++ = '\n';
*p = '\0';
}
bool
putline(l, mci)
register char *l;
register MCI *mci;
{
return putxline(l, strlen(l), mci, PXLF_MAPFROM);
}
#define PUTX(limit) \
do \
{ \
quotenext = false; \
while (l < limit) \
{ \
unsigned char c = (unsigned char) *l++; \
\
if (bitset(PXLF_STRIPMQUOTE, pxflags) && \
!quotenext && c == METAQUOTE) \
{ \
quotenext = true; \
continue; \
} \
quotenext = false; \
if (strip8bit) \
c &= 0177; \
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT, \
c) == SM_IO_EOF) \
{ \
dead = true; \
break; \
} \
if (TrafficLogFile != NULL) \
(void) sm_io_putc(TrafficLogFile, \
SM_TIME_DEFAULT, \
c); \
} \
} while (0)
bool
putxline(l, len, mci, pxflags)
register char *l;
size_t len;
register MCI *mci;
int pxflags;
{
register char *p, *end;
int slop;
bool dead, quotenext, strip8bit;
strip8bit = bitset(MCIF_7BIT, mci->mci_flags) ||
bitset(PXLF_STRIP8BIT, pxflags);
dead = false;
slop = 0;
end = l + len;
do
{
bool noeol = false;
p = memchr(l, '\n', end - l);
if (p == NULL)
{
p = end;
noeol = true;
}
if (TrafficLogFile != NULL)
(void) sm_io_fprintf(TrafficLogFile, SM_TIME_DEFAULT,
"%05d >>> ", (int) CurrentPid);
while (mci->mci_mailer->m_linelimit > 0 &&
(p - l + slop) > mci->mci_mailer->m_linelimit)
{
register char *q = &l[mci->mci_mailer->m_linelimit - slop - 1];
if (l[0] == '.' && slop == 0 &&
bitnset(M_XDOT, mci->mci_mailer->m_flags))
{
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT,
'.') == SM_IO_EOF)
dead = true;
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile,
SM_TIME_DEFAULT, '.');
}
else if (l[0] == 'F' && slop == 0 &&
bitset(PXLF_MAPFROM, pxflags) &&
strncmp(l, "From ", 5) == 0 &&
bitnset(M_ESCFROM, mci->mci_mailer->m_flags))
{
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT,
'>') == SM_IO_EOF)
dead = true;
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile,
SM_TIME_DEFAULT,
'>');
}
if (dead)
break;
PUTX(q);
if (dead)
break;
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT,
'!') == SM_IO_EOF ||
sm_io_fputs(mci->mci_out, SM_TIME_DEFAULT,
mci->mci_mailer->m_eol) == SM_IO_EOF ||
sm_io_putc(mci->mci_out, SM_TIME_DEFAULT,
' ') == SM_IO_EOF)
{
dead = true;
break;
}
if (TrafficLogFile != NULL)
{
(void) sm_io_fprintf(TrafficLogFile,
SM_TIME_DEFAULT,
"!\n%05d >>> ",
(int) CurrentPid);
}
slop = 1;
}
if (dead)
break;
if (l[0] == '.' && slop == 0 &&
bitnset(M_XDOT, mci->mci_mailer->m_flags) &&
!bitset(MCIF_INLONGLINE, mci->mci_flags))
{
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT, '.') ==
SM_IO_EOF)
{
dead = true;
break;
}
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile,
SM_TIME_DEFAULT, '.');
}
else if (l[0] == 'F' && slop == 0 &&
bitset(PXLF_MAPFROM, pxflags) &&
strncmp(l, "From ", 5) == 0 &&
bitnset(M_ESCFROM, mci->mci_mailer->m_flags) &&
!bitset(MCIF_INLONGLINE, mci->mci_flags))
{
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT, '>') ==
SM_IO_EOF)
{
dead = true;
break;
}
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile,
SM_TIME_DEFAULT, '>');
}
PUTX(p);
if (dead)
break;
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile, SM_TIME_DEFAULT,
'\n');
if ((!bitset(PXLF_NOADDEOL, pxflags) || !noeol))
{
mci->mci_flags &= ~MCIF_INLONGLINE;
if (sm_io_fputs(mci->mci_out, SM_TIME_DEFAULT,
mci->mci_mailer->m_eol) == SM_IO_EOF)
{
dead = true;
break;
}
}
else
mci->mci_flags |= MCIF_INLONGLINE;
if (l < end && *l == '\n')
{
if (*++l != ' ' && *l != '\t' && *l != '\0' &&
bitset(PXLF_HEADER, pxflags))
{
if (sm_io_putc(mci->mci_out, SM_TIME_DEFAULT,
' ') == SM_IO_EOF)
{
dead = true;
break;
}
if (TrafficLogFile != NULL)
(void) sm_io_putc(TrafficLogFile,
SM_TIME_DEFAULT, ' ');
}
}
} while (l < end);
return !dead;
}
int
xunlink(f)
char *f;
{
register int i;
int save_errno;
if (LogLevel > 98)
sm_syslog(LOG_DEBUG, CurEnv->e_id, "unlink %s", f);
i = unlink(f);
save_errno = errno;
if (i < 0 && LogLevel > 97)
sm_syslog(LOG_DEBUG, CurEnv->e_id, "%s: unlink-fail %d",
f, errno);
if (i >= 0)
SYNC_DIR(f, false);
errno = save_errno;
return i;
}
char *
sfgets(buf, siz, fp, timeout, during)
char *buf;
int siz;
SM_FILE_T *fp;
time_t timeout;
char *during;
{
register char *p;
int save_errno;
int io_timeout;
SM_REQUIRE(siz > 0);
SM_REQUIRE(buf != NULL);
if (fp == NULL)
{
buf[0] = '\0';
errno = EBADF;
return NULL;
}
p = NULL;
errno = 0;
io_timeout = (timeout <= 0) ? SM_TIME_DEFAULT : timeout * 1000;
while (!sm_io_eof(fp) && !sm_io_error(fp))
{
errno = 0;
p = sm_io_fgets(fp, io_timeout, buf, siz);
if (p == NULL && errno == EAGAIN)
{
if (LogLevel > 1)
sm_syslog(LOG_NOTICE, CurEnv->e_id,
"timeout waiting for input from %.100s during %s",
CURHOSTNAME,
during);
buf[0] = '\0';
#if XDEBUG
checkfd012(during);
#endif
if (TrafficLogFile != NULL)
(void) sm_io_fprintf(TrafficLogFile,
SM_TIME_DEFAULT,
"%05d <<< [TIMEOUT]\n",
(int) CurrentPid);
errno = ETIMEDOUT;
return NULL;
}
if (p != NULL || errno != EINTR)
break;
(void) sm_io_clearerr(fp);
}
save_errno = errno;
LineNumber++;
if (p == NULL)
{
buf[0] = '\0';
if (TrafficLogFile != NULL)
(void) sm_io_fprintf(TrafficLogFile, SM_TIME_DEFAULT,
"%05d <<< [EOF]\n",
(int) CurrentPid);
errno = save_errno;
return NULL;
}
if (TrafficLogFile != NULL)
(void) sm_io_fprintf(TrafficLogFile, SM_TIME_DEFAULT,
"%05d <<< %s", (int) CurrentPid, buf);
if (SevenBitInput)
{
for (p = buf; *p != '\0'; p++)
*p &= ~0200;
}
else if (!HasEightBits)
{
for (p = buf; *p != '\0'; p++)
{
if (bitset(0200, *p))
{
HasEightBits = true;
break;
}
}
}
return buf;
}
char *
fgetfolded(buf, np, f)
char *buf;
int *np;
SM_FILE_T *f;
{
register char *p = buf;
char *bp = buf;
register int i;
int n;
SM_REQUIRE(np != NULL);
n = *np;
SM_REQUIRE(n > 0);
SM_REQUIRE(buf != NULL);
if (f == NULL)
{
buf[0] = '\0';
errno = EBADF;
return NULL;
}
n--;
while ((i = sm_io_getc(f, SM_TIME_DEFAULT)) != SM_IO_EOF)
{
if (i == '\r')
{
i = sm_io_getc(f, SM_TIME_DEFAULT);
if (i != '\n')
{
if (i != SM_IO_EOF)
(void) sm_io_ungetc(f, SM_TIME_DEFAULT,
i);
i = '\r';
}
}
if (--n <= 0)
{
char *nbp;
int nn;
nn = (p - bp);
if (nn < MEMCHUNKSIZE)
nn *= 2;
else
nn += MEMCHUNKSIZE;
nbp = sm_malloc_x(nn);
memmove(nbp, bp, p - bp);
p = &nbp[p - bp];
if (bp != buf)
sm_free(bp);
bp = nbp;
n = nn - (p - bp);
*np = nn;
}
*p++ = i;
if (i == '\n')
{
LineNumber++;
i = sm_io_getc(f, SM_TIME_DEFAULT);
if (i != SM_IO_EOF)
(void) sm_io_ungetc(f, SM_TIME_DEFAULT, i);
if (i != ' ' && i != '\t')
break;
}
}
if (p == bp)
return NULL;
if (p[-1] == '\n')
p--;
*p = '\0';
return bp;
}
time_t
curtime()
{
auto time_t t;
(void) time(&t);
return t;
}
bool
atobool(s)
register char *s;
{
if (s == NULL || *s == '\0' || strchr("tTyY", *s) != NULL)
return true;
return false;
}
int
atooct(s)
register char *s;
{
register int i = 0;
while (*s >= '0' && *s <= '7')
i = (i << 3) | (*s++ - '0');
return i;
}
bool
bitintersect(a, b)
BITMAP256 a;
BITMAP256 b;
{
int i;
for (i = BITMAPBYTES / sizeof(int); --i >= 0; )
{
if ((a[i] & b[i]) != 0)
return true;
}
return false;
}
bool
bitzerop(map)
BITMAP256 map;
{
int i;
for (i = BITMAPBYTES / sizeof(int); --i >= 0; )
{
if (map[i] != 0)
return false;
}
return true;
}
bool
strcontainedin(icase, a, b)
bool icase;
register char *a;
register char *b;
{
int la;
int lb;
int c;
la = strlen(a);
lb = strlen(b);
c = *a;
if (icase && isascii(c) && isupper(c))
c = tolower(c);
for (; lb-- >= la; b++)
{
if (icase)
{
if (*b != c &&
isascii(*b) && isupper(*b) && tolower(*b) != c)
continue;
if (sm_strncasecmp(a, b, la) == 0)
return true;
}
else
{
if (*b != c)
continue;
if (strncmp(a, b, la) == 0)
return true;
}
}
return false;
}
void
checkfd012(where)
char *where;
{
#if XDEBUG
register int i;
for (i = 0; i < 3; i++)
fill_fd(i, where);
#endif
}
void
checkfdopen(fd, where)
int fd;
char *where;
{
#if XDEBUG
struct stat st;
if (fstat(fd, &st) < 0 && errno == EBADF)
{
syserr("checkfdopen(%d): %s not open as expected!", fd, where);
printopenfds(true);
}
#endif
}
void
checkfds(where)
char *where;
{
int maxfd;
register int fd;
bool printhdr = true;
int save_errno = errno;
static BITMAP256 baseline;
extern int DtableSize;
if (DtableSize > BITMAPBITS)
maxfd = BITMAPBITS;
else
maxfd = DtableSize;
if (where == NULL)
clrbitmap(baseline);
for (fd = 0; fd < maxfd; fd++)
{
struct stat stbuf;
if (fstat(fd, &stbuf) < 0 && errno != EOPNOTSUPP)
{
if (!bitnset(fd, baseline))
continue;
clrbitn(fd, baseline);
}
else if (!bitnset(fd, baseline))
setbitn(fd, baseline);
else
continue;
if (where == NULL)
continue;
if (printhdr)
{
sm_syslog(LOG_DEBUG, CurEnv->e_id,
"%s: changed fds:",
where);
printhdr = false;
}
dumpfd(fd, true, true);
}
errno = save_errno;
}
#if NETINET || NETINET6
# include <arpa/inet.h>
#endif
void
printopenfds(logit)
bool logit;
{
register int fd;
extern int DtableSize;
for (fd = 0; fd < DtableSize; fd++)
dumpfd(fd, false, logit);
}
void
dumpfd(fd, printclosed, logit)
int fd;
bool printclosed;
bool logit;
{
register char *p;
char *hp;
#ifdef S_IFSOCK
SOCKADDR sa;
#endif
auto SOCKADDR_LEN_T slen;
int i;
#if STAT64 > 0
struct stat64 st;
#else
struct stat st;
#endif
char buf[200];
p = buf;
(void) sm_snprintf(p, SPACELEFT(buf, p), "%3d: ", fd);
p += strlen(p);
if (
#if STAT64 > 0
fstat64(fd, &st)
#else
fstat(fd, &st)
#endif
< 0)
{
if (errno != EBADF)
{
(void) sm_snprintf(p, SPACELEFT(buf, p),
"CANNOT STAT (%s)",
sm_errstring(errno));
goto printit;
}
else if (printclosed)
{
(void) sm_snprintf(p, SPACELEFT(buf, p), "CLOSED");
goto printit;
}
return;
}
i = fcntl(fd, F_GETFL, 0);
if (i != -1)
{
(void) sm_snprintf(p, SPACELEFT(buf, p), "fl=0x%x, ", i);
p += strlen(p);
}
(void) sm_snprintf(p, SPACELEFT(buf, p), "mode=%o: ",
(int) st.st_mode);
p += strlen(p);
switch (st.st_mode & S_IFMT)
{
#ifdef S_IFSOCK
case S_IFSOCK:
(void) sm_snprintf(p, SPACELEFT(buf, p), "SOCK ");
p += strlen(p);
memset(&sa, '\0', sizeof(sa));
slen = sizeof(sa);
if (getsockname(fd, &sa.sa, &slen) < 0)
(void) sm_snprintf(p, SPACELEFT(buf, p), "(%s)",
sm_errstring(errno));
else
{
hp = hostnamebyanyaddr(&sa);
if (hp == NULL)
{
}
# if NETINET
else if (sa.sa.sa_family == AF_INET)
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s/%d", hp, ntohs(sa.sin.sin_port));
# endif
# if NETINET6
else if (sa.sa.sa_family == AF_INET6)
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s/%d", hp, ntohs(sa.sin6.sin6_port));
# endif
else
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s", hp);
}
p += strlen(p);
(void) sm_snprintf(p, SPACELEFT(buf, p), "->");
p += strlen(p);
slen = sizeof(sa);
if (getpeername(fd, &sa.sa, &slen) < 0)
(void) sm_snprintf(p, SPACELEFT(buf, p), "(%s)",
sm_errstring(errno));
else
{
hp = hostnamebyanyaddr(&sa);
if (hp == NULL)
{
}
# if NETINET
else if (sa.sa.sa_family == AF_INET)
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s/%d", hp, ntohs(sa.sin.sin_port));
# endif
# if NETINET6
else if (sa.sa.sa_family == AF_INET6)
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s/%d", hp, ntohs(sa.sin6.sin6_port));
# endif
else
(void) sm_snprintf(p, SPACELEFT(buf, p),
"%s", hp);
}
break;
#endif
case S_IFCHR:
(void) sm_snprintf(p, SPACELEFT(buf, p), "CHR: ");
p += strlen(p);
goto defprint;
#ifdef S_IFBLK
case S_IFBLK:
(void) sm_snprintf(p, SPACELEFT(buf, p), "BLK: ");
p += strlen(p);
goto defprint;
#endif
#if defined(S_IFIFO) && (!defined(S_IFSOCK) || S_IFIFO != S_IFSOCK)
case S_IFIFO:
(void) sm_snprintf(p, SPACELEFT(buf, p), "FIFO: ");
p += strlen(p);
goto defprint;
#endif
#ifdef S_IFDIR
case S_IFDIR:
(void) sm_snprintf(p, SPACELEFT(buf, p), "DIR: ");
p += strlen(p);
goto defprint;
#endif
#ifdef S_IFLNK
case S_IFLNK:
(void) sm_snprintf(p, SPACELEFT(buf, p), "LNK: ");
p += strlen(p);
goto defprint;
#endif
default:
defprint:
(void) sm_snprintf(p, SPACELEFT(buf, p),
"dev=%d/%d, ino=%llu, nlink=%d, u/gid=%d/%d, ",
major(st.st_dev), minor(st.st_dev),
(ULONGLONG_T) st.st_ino,
(int) st.st_nlink, (int) st.st_uid,
(int) st.st_gid);
p += strlen(p);
(void) sm_snprintf(p, SPACELEFT(buf, p), "size=%llu",
(ULONGLONG_T) st.st_size);
break;
}
printit:
if (logit)
sm_syslog(LOG_DEBUG, CurEnv ? CurEnv->e_id : NULL,
"%.800s", buf);
else
sm_dprintf("%s\n", buf);
}
char *
shorten_hostname(host)
char host[];
{
register char *p;
char *mydom;
int i;
bool canon = false;
i = strlen(host);
p = &host[(i == 0) ? 0 : i - 1];
if (*p == '.')
{
*p = '\0';
canon = true;
}
p = strchr(host, '.');
if (p == NULL)
return NULL;
mydom = macvalue('m', CurEnv);
if (mydom == NULL)
mydom = "";
i = strlen(++p);
if ((canon ? sm_strcasecmp(p, mydom)
: sm_strncasecmp(p, mydom, i)) == 0 &&
(mydom[i] == '.' || mydom[i] == '\0'))
{
*--p = '\0';
return p;
}
return NULL;
}
pid_t
prog_open(argv, pfd, e)
char **argv;
int *pfd;
ENVELOPE *e;
{
pid_t pid;
int save_errno;
int sff;
int ret;
int fdv[2];
char *p, *q;
char buf[MAXPATHLEN];
extern int DtableSize;
if (pipe(fdv) < 0)
{
syserr("%s: cannot create pipe for stdout", argv[0]);
return -1;
}
pid = fork();
if (pid < 0)
{
syserr("%s: cannot fork", argv[0]);
(void) close(fdv[0]);
(void) close(fdv[1]);
return -1;
}
if (pid > 0)
{
(void) close(fdv[1]);
*pfd = fdv[0];
return pid;
}
RestartRequest = NULL;
RestartWorkGroup = false;
ShutdownRequest = NULL;
PendingSignal = 0;
CurrentPid = getpid();
sm_exc_newthread(fatal_error);
(void) close(0);
(void) close(fdv[0]);
if (dup2(fdv[1], 1) < 0)
{
syserr("%s: cannot dup2 for stdout", argv[0]);
_exit(EX_OSERR);
}
(void) close(fdv[1]);
if (e->e_xfp != NULL)
{
int xfd;
xfd = sm_io_getinfo(e->e_xfp, SM_IO_WHAT_FD, NULL);
if (xfd >= 0 && dup2(xfd, 2) < 0)
{
syserr("%s: cannot dup2 for stderr", argv[0]);
_exit(EX_OSERR);
}
}
if (e->e_lockfp != NULL)
{
int fd;
fd = sm_io_getinfo(e->e_lockfp, SM_IO_WHAT_FD, NULL);
if (fd >= 0)
(void) close(fd);
else
syserr("%s: lockfp does not have a fd", argv[0]);
}
if (ProgMailer != NULL && ProgMailer->m_rootdir != NULL)
{
expand(ProgMailer->m_rootdir, buf, sizeof(buf), e);
if (chroot(buf) < 0)
{
syserr("prog_open: cannot chroot(%s)", buf);
exit(EX_TEMPFAIL);
}
if (chdir("/") < 0)
{
syserr("prog_open: cannot chdir(/)");
exit(EX_TEMPFAIL);
}
}
endpwent();
sm_mbdb_terminate();
#if _FFR_MEMSTAT
(void) sm_memstat_close();
#endif
if (setgid(DefGid) < 0 && geteuid() == 0)
{
syserr("prog_open: setgid(%ld) failed", (long) DefGid);
exit(EX_TEMPFAIL);
}
if (setuid(DefUid) < 0 && geteuid() == 0)
{
syserr("prog_open: setuid(%ld) failed", (long) DefUid);
exit(EX_TEMPFAIL);
}
if (ProgMailer != NULL)
p = ProgMailer->m_execdir;
else
p = NULL;
for (; p != NULL; p = q)
{
q = strchr(p, ':');
if (q != NULL)
*q = '\0';
expand(p, buf, sizeof(buf), e);
if (q != NULL)
*q++ = ':';
if (buf[0] != '\0' && chdir(buf) >= 0)
break;
}
if (p == NULL)
{
if (chdir("/tmp") < 0)
(void) chdir("/");
}
sff = SFF_ROOTOK|SFF_EXECOK;
if (!bitnset(DBS_RUNWRITABLEPROGRAM, DontBlameSendmail))
sff |= SFF_NOGWFILES|SFF_NOWWFILES;
if (bitnset(DBS_RUNPROGRAMINUNSAFEDIRPATH, DontBlameSendmail))
sff |= SFF_NOPATHCHECK;
else
sff |= SFF_SAFEDIRPATH;
ret = safefile(argv[0], DefUid, DefGid, DefUser, sff, 0, NULL);
if (ret != 0)
sm_syslog(LOG_INFO, e->e_id,
"Warning: prog_open: program %s unsafe: %s",
argv[0], sm_errstring(ret));
sm_close_on_exec(STDERR_FILENO + 1, DtableSize);
(void) execve(argv[0], (ARGV_T) argv, (ARGV_T) UserEnviron);
save_errno = errno;
syserr("%s: cannot exec", argv[0]);
if (transienterror(save_errno))
_exit(EX_OSERR);
_exit(EX_CONFIG);
return -1;
}
char *
get_column(line, col, delim, buf, buflen)
char line[];
int col;
int delim;
char buf[];
int buflen;
{
char *p;
char *begin, *end;
int i;
char delimbuf[4];
if ((char) delim == '\0')
(void) sm_strlcpy(delimbuf, "\n\t ", sizeof(delimbuf));
else
{
delimbuf[0] = (char) delim;
delimbuf[1] = '\0';
}
p = line;
if (*p == '\0')
return NULL;
if (*p == (char) delim && col == 0)
return NULL;
begin = line;
if (col == 0 && (char) delim == '\0')
{
while (*begin != '\0' && isascii(*begin) && isspace(*begin))
begin++;
}
for (i = 0; i < col; i++)
{
if ((begin = strpbrk(begin, delimbuf)) == NULL)
return NULL;
begin++;
if ((char) delim == '\0')
{
while (*begin != '\0' && isascii(*begin) && isspace(*begin))
begin++;
}
}
end = strpbrk(begin, delimbuf);
if (end == NULL)
i = strlen(begin);
else
i = end - begin;
if (i >= buflen)
i = buflen - 1;
(void) sm_strlcpy(buf, begin, i + 1);
return buf;
}
void
cleanstrcpy(t, f, l)
register char *t;
register char *f;
int l;
{
(void) denlstring(f, true, true);
if (l <= 0)
syserr("!cleanstrcpy: length == 0");
l--;
while (l > 0 && *f != '\0')
{
if (isascii(*f) &&
(isalnum(*f) || strchr("!#$%&'*+-./^_`{|}~", *f) != NULL))
{
l--;
*t++ = *f;
}
f++;
}
*t = '\0';
}
char *
denlstring(s, strict, logattacks)
char *s;
bool strict;
bool logattacks;
{
register char *p;
int l;
static char *bp = NULL;
static int bl = 0;
p = s;
while ((p = strchr(p, '\n')) != NULL)
if (strict || (*++p != ' ' && *p != '\t'))
break;
if (p == NULL)
return s;
l = strlen(s) + 1;
if (bl < l)
{
char *nbp = sm_pmalloc_x(l);
if (bp != NULL)
sm_free(bp);
bp = nbp;
bl = l;
}
(void) sm_strlcpy(bp, s, l);
for (p = bp; (p = strchr(p, '\n')) != NULL; )
*p++ = ' ';
if (logattacks)
{
sm_syslog(LOG_NOTICE, CurEnv ? CurEnv->e_id : NULL,
"POSSIBLE ATTACK from %.100s: newline in string \"%s\"",
RealHostName == NULL ? "[UNKNOWN]" : RealHostName,
shortenstring(bp, MAXSHORTSTR));
}
return bp;
}
bool
strreplnonprt(s, c)
char *s;
int c;
{
bool ok;
ok = true;
if (s == NULL)
return ok;
while (*s != '\0')
{
if (!(isascii(*s) && isprint(*s)))
{
*s = c;
ok = false;
}
++s;
}
return ok;
}
bool
path_is_dir(pathname, createflag)
char *pathname;
bool createflag;
{
struct stat statbuf;
#if HASLSTAT
if (lstat(pathname, &statbuf) < 0)
#else
if (stat(pathname, &statbuf) < 0)
#endif
{
if (errno != ENOENT || !createflag)
return false;
if (mkdir(pathname, 0755) < 0)
return false;
return true;
}
if (!S_ISDIR(statbuf.st_mode))
{
errno = ENOTDIR;
return false;
}
if (bitset(S_IWGRP|S_IWOTH, statbuf.st_mode))
{
errno = EACCES;
return false;
}
return true;
}
typedef struct procs PROCS_T;
struct procs
{
pid_t proc_pid;
char *proc_task;
int proc_type;
int proc_count;
int proc_other;
SOCKADDR proc_hostaddr;
};
static PROCS_T *volatile ProcListVec = NULL;
static int ProcListSize = 0;
void
proc_list_add(pid, task, type, count, other, hostaddr)
pid_t pid;
char *task;
int type;
int count;
int other;
SOCKADDR *hostaddr;
{
int i;
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == NO_PID)
break;
}
if (i >= ProcListSize)
{
proc_list_probe();
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == NO_PID)
break;
}
}
if (i >= ProcListSize)
{
int chldwasblocked;
PROCS_T *npv;
SM_ASSERT(ProcListSize < INT_MAX - PROC_LIST_SEG);
npv = (PROCS_T *) sm_pmalloc_x((sizeof(*npv)) *
(ProcListSize + PROC_LIST_SEG));
chldwasblocked = sm_blocksignal(SIGCHLD);
if (ProcListSize > 0)
{
memmove(npv, ProcListVec,
ProcListSize * sizeof(PROCS_T));
sm_free(ProcListVec);
}
for (i = ProcListSize; i < ProcListSize + PROC_LIST_SEG; i++)
{
npv[i].proc_pid = NO_PID;
npv[i].proc_task = NULL;
npv[i].proc_type = PROC_NONE;
}
i = ProcListSize;
ProcListSize += PROC_LIST_SEG;
ProcListVec = npv;
if (chldwasblocked == 0)
(void) sm_releasesignal(SIGCHLD);
}
ProcListVec[i].proc_pid = pid;
PSTRSET(ProcListVec[i].proc_task, task);
ProcListVec[i].proc_type = type;
ProcListVec[i].proc_count = count;
ProcListVec[i].proc_other = other;
if (hostaddr != NULL)
ProcListVec[i].proc_hostaddr = *hostaddr;
else
memset(&ProcListVec[i].proc_hostaddr, 0,
sizeof(ProcListVec[i].proc_hostaddr));
if (pid != CurrentPid)
{
SM_ASSERT(CurChildren < INT_MAX);
CurChildren++;
}
}
void
proc_list_set(pid, task)
pid_t pid;
char *task;
{
int i;
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == pid)
{
PSTRSET(ProcListVec[i].proc_task, task);
break;
}
}
}
void
proc_list_drop(pid, st, other)
pid_t pid;
int st;
int *other;
{
int i;
int type = PROC_NONE;
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == pid)
{
ProcListVec[i].proc_pid = NO_PID;
type = ProcListVec[i].proc_type;
if (other != NULL)
*other = ProcListVec[i].proc_other;
if (CurChildren > 0)
CurChildren--;
break;
}
}
if (type == PROC_CONTROL && WIFEXITED(st))
{
if (WEXITSTATUS(st) == EX_RESTART)
RestartRequest = "control socket";
else if (WEXITSTATUS(st) == EX_SHUTDOWN)
ShutdownRequest = "control socket";
}
else if (type == PROC_QUEUE_CHILD && !WIFSTOPPED(st) &&
ProcListVec[i].proc_other > -1)
{
mark_work_group_restart(ProcListVec[i].proc_other, st);
}
else if (type == PROC_QUEUE)
CurRunners -= ProcListVec[i].proc_count;
}
void
proc_list_clear()
{
int i;
for (i = 1; i < ProcListSize; i++)
ProcListVec[i].proc_pid = NO_PID;
CurChildren = 0;
}
void
proc_list_probe()
{
int i, children;
int chldwasblocked;
pid_t pid;
children = 0;
chldwasblocked = sm_blocksignal(SIGCHLD);
for (i = 1; i < ProcListSize; i++)
{
pid = ProcListVec[i].proc_pid;
if (pid == NO_PID || pid == CurrentPid)
continue;
if (kill(pid, 0) < 0)
{
if (LogLevel > 3)
sm_syslog(LOG_DEBUG, CurEnv->e_id,
"proc_list_probe: lost pid %d",
(int) ProcListVec[i].proc_pid);
ProcListVec[i].proc_pid = NO_PID;
SM_FREE_CLR(ProcListVec[i].proc_task);
CurChildren--;
}
else
{
++children;
}
}
if (CurChildren < 0)
CurChildren = 0;
if (chldwasblocked == 0)
(void) sm_releasesignal(SIGCHLD);
if (LogLevel > 10 && children != CurChildren && CurrentPid == DaemonPid)
{
sm_syslog(LOG_ERR, NOQID,
"proc_list_probe: found %d children, expected %d",
children, CurChildren);
}
}
void
proc_list_display(out, prefix)
SM_FILE_T *out;
char *prefix;
{
int i;
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == NO_PID)
continue;
(void) sm_io_fprintf(out, SM_TIME_DEFAULT, "%s%d %s%s\n",
prefix,
(int) ProcListVec[i].proc_pid,
ProcListVec[i].proc_task != NULL ?
ProcListVec[i].proc_task : "(unknown)",
(OpMode == MD_SMTP ||
OpMode == MD_DAEMON ||
OpMode == MD_ARPAFTP) ? "\r" : "");
}
}
void
proc_list_signal(type, signal)
int type;
int signal;
{
int chldwasblocked;
int alrmwasblocked;
int i;
pid_t mypid = getpid();
chldwasblocked = sm_blocksignal(SIGCHLD);
alrmwasblocked = sm_blocksignal(SIGALRM);
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == NO_PID ||
ProcListVec[i].proc_pid == mypid)
continue;
if (ProcListVec[i].proc_type != type)
continue;
(void) kill(ProcListVec[i].proc_pid, signal);
}
if (alrmwasblocked == 0)
(void) sm_releasesignal(SIGALRM);
if (chldwasblocked == 0)
(void) sm_releasesignal(SIGCHLD);
}
int
count_open_connections(hostaddr)
SOCKADDR *hostaddr;
{
int i, n;
if (hostaddr == NULL)
return 0;
n = 1;
for (i = 0; i < ProcListSize; i++)
{
if (ProcListVec[i].proc_pid == NO_PID)
continue;
if (hostaddr->sa.sa_family !=
ProcListVec[i].proc_hostaddr.sa.sa_family)
continue;
#if NETINET
if (hostaddr->sa.sa_family == AF_INET &&
(hostaddr->sin.sin_addr.s_addr ==
ProcListVec[i].proc_hostaddr.sin.sin_addr.s_addr))
n++;
#endif
#if NETINET6
if (hostaddr->sa.sa_family == AF_INET6 &&
IN6_ARE_ADDR_EQUAL(&(hostaddr->sin6.sin6_addr),
&(ProcListVec[i].proc_hostaddr.sin6.sin6_addr)))
n++;
#endif
}
return n;
}