root/usr/src/cmd/sendmail/src/util.c
/*
 * Copyright (c) 1998-2007, 2009 Sendmail, Inc. and its suppliers.
 *      All rights reserved.
 * Copyright (c) 1983, 1995-1997 Eric P. Allman.  All rights reserved.
 * Copyright (c) 1988, 1993
 *      The Regents of the University of California.  All rights reserved.
 *
 * By using this file, you agree to the terms and conditions set
 * forth in the LICENSE file which can be found at the top level of
 * the sendmail distribution.
 *
 */

#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>

/*
**  NEWSTR -- Create a copy of a C string
**
**      Parameters:
**              s -- the string to copy.
**
**      Returns:
**              pointer to newly allocated string.
*/

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;
}

/*
**  ADDQUOTES -- Adds quotes & quote bits to a string.
**
**      Runs through a string and adds backslashes and quote bits.
**
**      Parameters:
**              s -- the string to modify.
**              rpool -- resource pool from which to allocate result
**
**      Returns:
**              pointer to quoted string.
*/

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;

        /* Find length of quoted string */
        while ((c = *p++) != '\0')
        {
                len++;
                if (c == '\\' || c == '"')
                        len++;
        }

        q = r = sm_rpool_malloc_x(rpool, len + 3);
        p = s;

        /* add leading quote */
        *q++ = '"';
        while ((c = *p++) != '\0')
        {
                /* quote \ or " */
                if (c == '\\' || c == '"')
                        *q++ = '\\';
                *q++ = c;
        }
        *q++ = '"';
        *q = '\0';
        return r;
}

/*
**  STRIPBACKSLASH -- Strip all leading backslashes from a string, provided
**      the following character is alpha-numerical.
**
**      This is done in place.
**
**      Parameters:
**              s -- the string to strip.
**
**      Returns:
**              none.
*/

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');
}

/*
**  RFC822_STRING -- Checks string for proper RFC822 string quoting.
**
**      Runs through a string and verifies RFC822 special characters
**      are only found inside comments, quoted strings, or backslash
**      escaped.  Also verified balanced quotes and parenthesis.
**
**      Parameters:
**              s -- the string to modify.
**
**      Returns:
**              true iff the string is RFC822 compliant, false otherwise.
*/

bool
rfc822_string(s)
        char *s;
{
        bool quoted = false;
        int commentlev = 0;
        char *c = s;

        if (s == NULL)
                return false;

        while (*c != '\0')
        {
                /* escaped character */
                if (*c == '\\')
                {
                        c++;
                        if (*c == '\0')
                                return false;
                }
                else if (commentlev == 0 && *c == '"')
                        quoted = !quoted;
                else if (!quoted)
                {
                        if (*c == ')')
                        {
                                /* unbalanced ')' */
                                if (commentlev == 0)
                                        return false;
                                else
                                        commentlev--;
                        }
                        else if (*c == '(')
                                commentlev++;
                        else if (commentlev == 0 &&
                                 strchr(MustQuoteChars, *c) != NULL)
                                return false;
                }
                c++;
        }

        /* unbalanced '"' or '(' */
        return !quoted && commentlev == 0;
}

/*
**  SHORTEN_RFC822_STRING -- Truncate and rebalance an RFC822 string
**
**      Arbitrarily shorten (in place) an RFC822 string and rebalance
**      comments and quotes.
**
**      Parameters:
**              string -- the string to shorten
**              length -- the maximum size, 0 if no maximum
**
**      Returns:
**              true if string is changed, false otherwise
**
**      Side Effects:
**              Changes string in place, possibly resulting
**              in a shorter string.
*/

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;

        /*
        **  If have to rebalance an already short enough string,
        **  need to do it within allocated space.
        */

        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;
                }

                /* Inside a comment, quotes don't matter */
                if (parencount <= 0 && *ptr == '"')
                        quoted = !quoted;

increment:
                /* Check for sufficient space for next character */
                if (length - (ptr - string) <= (size_t) ((backslash ? 1 : 0) +
                                                parencount +
                                                (quoted ? 1 : 0)))
                {
                        /* Not enough, backtrack */
                        if (*ptr == '\\')
                                backslash = false;
                        else if (*ptr == '(' && !quoted)
                                parencount--;
                        else if (*ptr == '"' && parencount == 0)
                                quoted = false;
                        break;
                }
                ptr++;
        }

        /* Rebalance */
        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;
}

/*
**  FIND_CHARACTER -- find an unquoted character in an RFC822 string
**
**      Find an unquoted, non-commented character in an RFC822
**      string and return a pointer to its location in the
**      string.
**
**      Parameters:
**              string -- the string to search
**              character -- the character to find
**
**      Returns:
**              pointer to the character, or
**              a pointer to the end of the line if character is not found
*/

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;
                }

                /* Inside a comment, nothing matters */
                if (parencount > 0)
                {
                        string++;
                        continue;
                }

                if (*string == '"')
                        quoted = !quoted;
                else if (*string == character && !quoted)
                        break;
                string++;
        }

        /* Return pointer to the character */
        return string;
}

/*
**  CHECK_BODYTYPE -- check bodytype parameter
**
**      Parameters:
**              bodytype -- bodytype parameter
**
**      Returns:
**              BODYTYPE_* according to parameter
**
*/

int
check_bodytype(bodytype)
        char *bodytype;
{
        /* check body type for legality */
        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;
}

/*
**  TRUNCATE_AT_DELIM -- truncate string at a delimiter and append "..."
**
**      Parameters:
**              str -- string to truncate
**              len -- maximum length (including '\0') (0 for unlimited)
**              delim -- delimiter character
**
**      Returns:
**              None.
*/

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;
                }
        }

        /* Couldn't find a place to append "..." */
        if (len > 3)
                (void) sm_strlcpy(str, "...", len);
        else
                str[0] = '\0';
}

/*
**  XALLOC -- Allocate memory, raise an exception on error
**
**      Parameters:
**              sz -- size of area to allocate.
**
**      Returns:
**              pointer to data region.
**
**      Exceptions:
**              SmHeapOutOfMemory (F:sm.heap) -- cannot allocate memory
**
**      Side Effects:
**              Memory is allocated.
*/

char *
#if SM_HEAP_CHECK
xalloc_tagged(sz, file, line)
        register int sz;
        char *file;
        int line;
#else /* SM_HEAP_CHECK */
xalloc(sz)
        register int sz;
#endif /* SM_HEAP_CHECK */
{
        register char *p;

        SM_REQUIRE(sz >= 0);

        /* some systems can't handle size zero mallocs */
        if (sz <= 0)
                sz = 1;

        /* scaffolding for testing error handling code */
        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;
}

/*
**  COPYPLIST -- copy list of pointers.
**
**      This routine is the equivalent of strdup for lists of
**      pointers.
**
**      Parameters:
**              list -- list of pointers to copy.
**                      Must be NULL terminated.
**              copycont -- if true, copy the contents of the vector
**                      (which must be a string) also.
**              rpool -- resource pool from which to allocate storage,
**                      or NULL
**
**      Returns:
**              a copy of 'list'.
*/

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;
}

/*
**  COPYQUEUE -- copy address queue.
**
**      This routine is the equivalent of strdup for address queues;
**      addresses marked as QS_IS_DEAD() aren't copied
**
**      Parameters:
**              addr -- list of address structures to copy.
**              rpool -- resource pool from which to allocate storage
**
**      Returns:
**              a copy of 'addr'.
*/

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;
}

/*
**  LOG_SENDMAIL_PID -- record sendmail pid and command line.
**
**      Parameters:
**              e -- the current envelope.
**
**      Returns:
**              none.
**
**      Side Effects:
**              writes pidfile, logs command line.
**              keeps file open and locked to prevent overwrite of active file
*/

static SM_FILE_T        *Pidf = NULL;

void
log_sendmail_pid(e)
        ENVELOPE *e;
{
        long sff;
        char pidpath[MAXPATHLEN];
        extern char *CommandLineArgs;

        /* write the pid to the log file for posterity */
        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();

                /* write the process id on line 1 */
                (void) sm_io_fprintf(Pidf, SM_TIME_DEFAULT, "%ld\n",
                                     (long) PidFilePid);

                /* line 2 contains all command line flags */
                (void) sm_io_fprintf(Pidf, SM_TIME_DEFAULT, "%s\n",
                                     CommandLineArgs);

                /* flush */
                (void) sm_io_flush(Pidf, SM_TIME_DEFAULT);

                /*
                **  Leave pid file open until process ends
                **  so it's not overwritten by another
                **  process.
                */
        }
        if (LogLevel > 9)
                sm_syslog(LOG_INFO, NOQID, "started as: %s", CommandLineArgs);
}

/*
**  CLOSE_SENDMAIL_PID -- close sendmail pid file
**
**      Parameters:
**              none.
**
**      Returns:
**              none.
*/

void
close_sendmail_pid()
{
        if (Pidf == NULL)
                return;

        (void) sm_io_close(Pidf, SM_TIME_DEFAULT);
        Pidf = NULL;
}

/*
**  SET_DELIVERY_MODE -- set and record the delivery mode
**
**      Parameters:
**              mode -- delivery mode
**              e -- the current envelope.
**
**      Returns:
**              none.
**
**      Side Effects:
**              sets {deliveryMode} macro
*/

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);
}

/*
**  SET_OP_MODE -- set and record the op mode
**
**      Parameters:
**              mode -- op mode
**              e -- the current envelope.
**
**      Returns:
**              none.
**
**      Side Effects:
**              sets {opMode} macro
*/

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);
}

/*
**  PRINTAV -- print argument vector.
**
**      Parameters:
**              fp -- output file pointer.
**              av -- argument vector.
**
**      Returns:
**              none.
**
**      Side Effects:
**              prints av.
*/

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');
}

/*
**  XPUTS -- put string doing control escapes.
**
**      Parameters:
**              fp -- output file pointer.
**              s -- string to put.
**
**      Returns:
**              none.
**
**      Side Effects:
**              output to stdout
*/

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 $");

        /*
        **  TermEscape is set here, rather than in main(),
        **  because ANSI mode can be turned on or off at any time
        **  if we are in -bt rule testing mode.
        */

        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;

                        /* unrecognized meta character */
                        (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;
                }

                /* wasn't a meta-macro -- find another way to print it */
                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);
}

/*
**  MAKELOWER -- Translate a line into lower case
**
**      Parameters:
**              p -- the string to translate.  If NULL, return is
**                      immediate.
**
**      Returns:
**              none.
**
**      Side Effects:
**              String pointed to by p is translated to lower case.
*/

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);
}

/*
**  FIXCRLF -- fix <CR><LF> in line.
**
**      Looks for the <CR><LF> combination and turns it into the
**      UNIX canonical <NL> character.  It only takes one line,
**      i.e., it is assumed that the first <NL> found is the end
**      of the line.
**
**      Parameters:
**              line -- the line to fix.
**              stripnl -- if true, strip the newline also.
**
**      Returns:
**              none.
**
**      Side Effects:
**              line is changed in place.
*/

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';
}

/*
**  PUTLINE -- put a line like fputs obeying SMTP conventions
**
**      This routine always guarantees outputing a newline (or CRLF,
**      as appropriate) at the end of the string.
**
**      Parameters:
**              l -- line to put.
**              mci -- the mailer connection information.
**
**      Returns:
**              true iff line was written successfully
**
**      Side Effects:
**              output of l to mci->mci_out.
*/

bool
putline(l, mci)
        register char *l;
        register MCI *mci;
{
        return putxline(l, strlen(l), mci, PXLF_MAPFROM);
}

/*
**  PUTXLINE -- putline with flags bits.
**
**      This routine always guarantees outputing a newline (or CRLF,
**      as appropriate) at the end of the string.
**
**      Parameters:
**              l -- line to put.
**              len -- the length of the line.
**              mci -- the mailer connection information.
**              pxflags -- flag bits:
**                  PXLF_MAPFROM -- map From_ to >From_.
**                  PXLF_STRIP8BIT -- strip 8th bit.
**                  PXLF_HEADER -- map bare newline in header to newline space.
**                  PXLF_NOADDEOL -- don't add an EOL if one wasn't present.
**                  PXLF_STRIPMQUOTE -- strip METAQUOTE bytes.
**
**      Returns:
**              true iff line was written successfully
**
**      Side Effects:
**              output of l to mci->mci_out.
*/


#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;

        /* strip out 0200 bits -- these can look like TELNET protocol */
        strip8bit = bitset(MCIF_7BIT, mci->mci_flags) ||
                    bitset(PXLF_STRIP8BIT, pxflags);
        dead = false;
        slop = 0;

        end = l + len;
        do
        {
                bool noeol = false;

                /* find the end of the line */
                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);

                /* check for line overflow */
                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;

                /* output last part */
                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;
}

/*
**  XUNLINK -- unlink a file, doing logging as appropriate.
**
**      Parameters:
**              f -- name of file to unlink.
**
**      Returns:
**              return value of unlink()
**
**      Side Effects:
**              f is unlinked.
*/

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;
}

/*
**  SFGETS -- "safe" fgets -- times out and ignores random interrupts.
**
**      Parameters:
**              buf -- place to put the input line.
**              siz -- size of buf.
**              fp -- file to read from.
**              timeout -- the timeout before error occurs.
**              during -- what we are trying to read (for error messages).
**
**      Returns:
**              NULL on error (including timeout).  This may also leave
**                      buf containing a null string.
**              buf otherwise.
*/


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;
        }

        /* try to read */
        p = NULL;
        errno = 0;

        /* convert the timeout to sm_io notation */
        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)
                {
                        /* The sm_io_fgets() call timedout */
                        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 /* XDEBUG */
                        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;

        /* clean up the books and exit */
        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;
}

/*
**  FGETFOLDED -- like fgets, but knows about folded lines.
**
**      Parameters:
**              buf -- place to put result.
**              np -- pointer to bytes available; will be updated with
**                      the actual buffer size (not number of bytes filled)
**                      on return.
**              f -- file to read from.
**
**      Returns:
**              input line(s) on success, NULL on error or SM_IO_EOF.
**              This will normally be buf -- unless the line is too
**                      long, when it will be sm_malloc_x()ed.
**
**      Side Effects:
**              buf gets lines from f, with continuation lines (lines
**              with leading white space) appended.  CRLF's are mapped
**              into single newlines.  Any trailing NL is stripped.
*/

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)
                {
                        /* allocate new space */
                        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;
}

/*
**  CURTIME -- return current time.
**
**      Parameters:
**              none.
**
**      Returns:
**              the current time.
*/

time_t
curtime()
{
        auto time_t t;

        (void) time(&t);
        return t;
}

/*
**  ATOBOOL -- convert a string representation to boolean.
**
**      Defaults to false
**
**      Parameters:
**              s -- string to convert.  Takes "tTyY", empty, and NULL as true,
**                      others as false.
**
**      Returns:
**              A boolean representation of the string.
*/

bool
atobool(s)
        register char *s;
{
        if (s == NULL || *s == '\0' || strchr("tTyY", *s) != NULL)
                return true;
        return false;
}

/*
**  ATOOCT -- convert a string representation to octal.
**
**      Parameters:
**              s -- string to convert.
**
**      Returns:
**              An integer representing the string interpreted as an
**              octal number.
*/

int
atooct(s)
        register char *s;
{
        register int i = 0;

        while (*s >= '0' && *s <= '7')
                i = (i << 3) | (*s++ - '0');
        return i;
}

/*
**  BITINTERSECT -- tell if two bitmaps intersect
**
**      Parameters:
**              a, b -- the bitmaps in question
**
**      Returns:
**              true if they have a non-null intersection
**              false otherwise
*/

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;
}

/*
**  BITZEROP -- tell if a bitmap is all zero
**
**      Parameters:
**              map -- the bit map to check
**
**      Returns:
**              true if map is all zero.
**              false if there are any bits set in map.
*/

bool
bitzerop(map)
        BITMAP256 map;
{
        int i;

        for (i = BITMAPBYTES / sizeof(int); --i >= 0; )
        {
                if (map[i] != 0)
                        return false;
        }
        return true;
}

/*
**  STRCONTAINEDIN -- tell if one string is contained in another
**
**      Parameters:
**              icase -- ignore case?
**              a -- possible substring.
**              b -- possible superstring.
**
**      Returns:
**              true if a is contained in b (case insensitive).
**              false otherwise.
*/

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;
}

/*
**  CHECKFD012 -- check low numbered file descriptors
**
**      File descriptors 0, 1, and 2 should be open at all times.
**      This routine verifies that, and fixes it if not true.
**
**      Parameters:
**              where -- a tag printed if the assertion failed
**
**      Returns:
**              none
*/

void
checkfd012(where)
        char *where;
{
#if XDEBUG
        register int i;

        for (i = 0; i < 3; i++)
                fill_fd(i, where);
#endif /* XDEBUG */
}

/*
**  CHECKFDOPEN -- make sure file descriptor is open -- for extended debugging
**
**      Parameters:
**              fd -- file descriptor to check.
**              where -- tag to print on failure.
**
**      Returns:
**              none.
*/

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 /* XDEBUG */
}

/*
**  CHECKFDS -- check for new or missing file descriptors
**
**      Parameters:
**              where -- tag for printing.  If null, take a base line.
**
**      Returns:
**              none
**
**      Side Effects:
**              If where is set, shows changes since the last call.
*/

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;

                /* file state has changed */
                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;
}

/*
**  PRINTOPENFDS -- print the open file descriptors (for debugging)
**
**      Parameters:
**              logit -- if set, send output to syslog; otherwise
**                      print for debugging.
**
**      Returns:
**              none.
*/

#if NETINET || NETINET6
# include <arpa/inet.h>
#endif /* NETINET || NETINET6 */

void
printopenfds(logit)
        bool logit;
{
        register int fd;
        extern int DtableSize;

        for (fd = 0; fd < DtableSize; fd++)
                dumpfd(fd, false, logit);
}

/*
**  DUMPFD -- dump a file descriptor
**
**      Parameters:
**              fd -- the file descriptor to dump.
**              printclosed -- if set, print a notification even if
**                      it is closed; otherwise print nothing.
**              logit -- if set, use sm_syslog instead of sm_dprintf()
**
**      Returns:
**              none.
*/

void
dumpfd(fd, printclosed, logit)
        int fd;
        bool printclosed;
        bool logit;
{
        register char *p;
        char *hp;
#ifdef S_IFSOCK
        SOCKADDR sa;
#endif /* S_IFSOCK */
        auto SOCKADDR_LEN_T slen;
        int i;
#if STAT64 > 0
        struct stat64 st;
#else /* STAT64 > 0 */
        struct stat st;
#endif /* STAT64 > 0 */
        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 /* STAT64 > 0 */
            fstat(fd, &st)
#endif /* STAT64 > 0 */
            < 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)
                        {
                                /* EMPTY */
                                /* do nothing */
                        }
# 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 /* NETINET */
# 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 /* NETINET6 */
                        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)
                        {
                                /* EMPTY */
                                /* do nothing */
                        }
# 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 /* NETINET */
# 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 /* NETINET6 */
                        else
                                (void) sm_snprintf(p, SPACELEFT(buf, p),
                                        "%s", hp);
                }
                break;
#endif /* S_IFSOCK */

          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 /* S_IFBLK */

#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 /* defined(S_IFIFO) && (!defined(S_IFSOCK) || S_IFIFO != S_IFSOCK) */

#ifdef S_IFDIR
          case S_IFDIR:
                (void) sm_snprintf(p, SPACELEFT(buf, p), "DIR: ");
                p += strlen(p);
                goto defprint;
#endif /* S_IFDIR */

#ifdef S_IFLNK
          case S_IFLNK:
                (void) sm_snprintf(p, SPACELEFT(buf, p), "LNK: ");
                p += strlen(p);
                goto defprint;
#endif /* S_IFLNK */

          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);
}

/*
**  SHORTEN_HOSTNAME -- strip local domain information off of hostname.
**
**      Parameters:
**              host -- the host to shorten (stripped in place).
**
**      Returns:
**              place where string was truncated, NULL if not truncated.
*/

char *
shorten_hostname(host)
        char host[];
{
        register char *p;
        char *mydom;
        int i;
        bool canon = false;

        /* strip off final dot */
        i = strlen(host);
        p = &host[(i == 0) ? 0 : i - 1];
        if (*p == '.')
        {
                *p = '\0';
                canon = true;
        }

        /* see if there is any domain at all -- if not, we are done */
        p = strchr(host, '.');
        if (p == NULL)
                return NULL;

        /* yes, we have a domain -- see if it looks like us */
        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;
}

/*
**  PROG_OPEN -- open a program for reading
**
**      Parameters:
**              argv -- the argument list.
**              pfd -- pointer to a place to store the file descriptor.
**              e -- the current envelope.
**
**      Returns:
**              pid of the process -- -1 if it failed.
*/

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)
        {
                /* parent */
                (void) close(fdv[1]);
                *pfd = fdv[0];
                return pid;
        }

        /* Reset global flags */
        RestartRequest = NULL;
        RestartWorkGroup = false;
        ShutdownRequest = NULL;
        PendingSignal = 0;
        CurrentPid = getpid();

        /*
        **  Initialize exception stack and default exception
        **  handler for child process.
        */

        sm_exc_newthread(fatal_error);

        /* child -- close stdin */
        (void) close(0);

        /* stdout goes back to parent */
        (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]);

        /* stderr goes to transcript if available */
        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);
                }
        }

        /* this process has no right to the queue file */
        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]);
        }

        /* chroot to the program mailer directory, if defined */
        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);
                }
        }

        /* run as default user */
        endpwent();
        sm_mbdb_terminate();
#if _FFR_MEMSTAT
        (void) sm_memstat_close();
#endif /* _FFR_MEMSTAT */
        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);
        }

        /* run in some directory */
        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)
        {
                /* backup directories */
                if (chdir("/tmp") < 0)
                        (void) chdir("/");
        }

        /* Check safety of program to be run */
        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));

        /* arrange for all the files to be closed */
        sm_close_on_exec(STDERR_FILENO + 1, DtableSize);

        /* now exec the process */
        (void) execve(argv[0], (ARGV_T) argv, (ARGV_T) UserEnviron);

        /* woops!  failed */
        save_errno = errno;
        syserr("%s: cannot exec", argv[0]);
        if (transienterror(save_errno))
                _exit(EX_OSERR);
        _exit(EX_CONFIG);
        return -1;      /* avoid compiler warning on IRIX */
}

/*
**  GET_COLUMN -- look up a Column in a line buffer
**
**      Parameters:
**              line -- the raw text line to search.
**              col -- the column number to fetch.
**              delim -- the delimiter between columns.  If null,
**                      use white space.
**              buf -- the output buffer.
**              buflen -- the length of buf.
**
**      Returns:
**              buf if successful.
**              NULL otherwise.
*/

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;                    /* line empty */
        if (*p == (char) delim && col == 0)
                return NULL;                    /* first column empty */

        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;            /* no such column */
                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;
}

/*
**  CLEANSTRCPY -- copy string keeping out bogus characters
**
**      Parameters:
**              t -- "to" string.
**              f -- "from" string.
**              l -- length of space available in "to" string.
**
**      Returns:
**              none.
*/

void
cleanstrcpy(t, f, l)
        register char *t;
        register char *f;
        int l;
{
        /* check for newlines and log if necessary */
        (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';
}

/*
**  DENLSTRING -- convert newlines in a string to spaces
**
**      Parameters:
**              s -- the input string
**              strict -- if set, don't permit continuation lines.
**              logattacks -- if set, log attempted attacks.
**
**      Returns:
**              A pointer to a version of the string with newlines
**              mapped to spaces.  This should be copied.
*/

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)
        {
                /* allocate more space */
                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;
}

/*
**  STRREPLNONPRT -- replace "unprintable" characters in a string with subst
**
**      Parameters:
**              s -- string to manipulate (in place)
**              subst -- character to use as replacement
**
**      Returns:
**              true iff string did not contain "unprintable" characters
*/

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;
}

/*
**  PATH_IS_DIR -- check to see if file exists and is a directory.
**
**      There are some additional checks for security violations in
**      here.  This routine is intended to be used for the host status
**      support.
**
**      Parameters:
**              pathname -- pathname to check for directory-ness.
**              createflag -- if set, create directory if needed.
**
**      Returns:
**              true -- if the indicated pathname is a directory
**              false -- otherwise
*/

bool
path_is_dir(pathname, createflag)
        char *pathname;
        bool createflag;
{
        struct stat statbuf;

#if HASLSTAT
        if (lstat(pathname, &statbuf) < 0)
#else /* HASLSTAT */
        if (stat(pathname, &statbuf) < 0)
#endif /* HASLSTAT */
        {
                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;
        }

        /* security: don't allow writable directories */
        if (bitset(S_IWGRP|S_IWOTH, statbuf.st_mode))
        {
                errno = EACCES;
                return false;
        }
        return true;
}

/*
**  PROC_LIST_ADD -- add process id to list of our children
**
**      Parameters:
**              pid -- pid to add to list.
**              task -- task of pid.
**              type -- type of process.
**              count -- number of processes.
**              other -- other information for this type.
**
**      Returns:
**              none
**
**      Side Effects:
**              May increase CurChildren. May grow ProcList.
*/

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)
        {
                /* probe the existing vector to avoid growing infinitely */
                proc_list_probe();

                /* now scan again */
                for (i = 0; i < ProcListSize; i++)
                {
                        if (ProcListVec[i].proc_pid == NO_PID)
                                break;
                }
        }
        if (i >= ProcListSize)
        {
                /* grow process list */
                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));

                /* Block SIGCHLD so reapchild() doesn't mess with us */
                chldwasblocked = sm_blocksignal(SIGCHLD);
                if (ProcListSize > 0)
                {
                        memmove(npv, ProcListVec,
                                ProcListSize * sizeof(PROCS_T));
                        sm_free(ProcListVec);
                }

                /* XXX just use memset() to initialize this part? */
                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 process adding itself, it's not a child */
        if (pid != CurrentPid)
        {
                SM_ASSERT(CurChildren < INT_MAX);
                CurChildren++;
        }
}

/*
**  PROC_LIST_SET -- set pid task in process list
**
**      Parameters:
**              pid -- pid to set
**              task -- task of pid
**
**      Returns:
**              none.
*/

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;
                }
        }
}

/*
**  PROC_LIST_DROP -- drop pid from process list
**
**      Parameters:
**              pid -- pid to drop
**              st -- process status
**              other -- storage for proc_other (return).
**
**      Returns:
**              none.
**
**      Side Effects:
**              May decrease CurChildren, CurRunners, or
**              set RestartRequest or ShutdownRequest.
**
**      NOTE:   THIS CAN BE CALLED FROM A SIGNAL HANDLER.  DO NOT ADD
**              ANYTHING TO THIS ROUTINE UNLESS YOU KNOW WHAT YOU ARE
**              DOING.
*/

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 so, see if we need to restart or shutdown */
                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)
        {
                /* restart this persistent runner */
                mark_work_group_restart(ProcListVec[i].proc_other, st);
        }
        else if (type == PROC_QUEUE)
                CurRunners -= ProcListVec[i].proc_count;
}

/*
**  PROC_LIST_CLEAR -- clear the process list
**
**      Parameters:
**              none.
**
**      Returns:
**              none.
**
**      Side Effects:
**              Sets CurChildren to zero.
*/

void
proc_list_clear()
{
        int i;

        /* start from 1 since 0 is the daemon itself */
        for (i = 1; i < ProcListSize; i++)
                ProcListVec[i].proc_pid = NO_PID;
        CurChildren = 0;
}

/*
**  PROC_LIST_PROBE -- probe processes in the list to see if they still exist
**
**      Parameters:
**              none
**
**      Returns:
**              none
**
**      Side Effects:
**              May decrease CurChildren.
*/

void
proc_list_probe()
{
        int i, children;
        int chldwasblocked;
        pid_t pid;

        children = 0;
        chldwasblocked = sm_blocksignal(SIGCHLD);

        /* start from 1 since 0 is the daemon itself */
        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);
        }
}

/*
**  PROC_LIST_DISPLAY -- display the process list
**
**      Parameters:
**              out -- output file pointer
**              prefix -- string to output in front of each line.
**
**      Returns:
**              none.
*/

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" : "");
        }
}

/*
**  PROC_LIST_SIGNAL -- send a signal to a type of process in the list
**
**      Parameters:
**              type -- type of process to signal
**              signal -- the type of signal to send
**
**      Results:
**              none.
**
**      NOTE:   THIS CAN BE CALLED FROM A SIGNAL HANDLER.  DO NOT ADD
**              ANYTHING TO THIS ROUTINE UNLESS YOU KNOW WHAT YOU ARE
**              DOING.
*/

void
proc_list_signal(type, signal)
        int type;
        int signal;
{
        int chldwasblocked;
        int alrmwasblocked;
        int i;
        pid_t mypid = getpid();

        /* block these signals so that we may signal cleanly */
        chldwasblocked = sm_blocksignal(SIGCHLD);
        alrmwasblocked = sm_blocksignal(SIGALRM);

        /* Find all processes of type and send signal */
        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);
        }

        /* restore the signals */
        if (alrmwasblocked == 0)
                (void) sm_releasesignal(SIGALRM);
        if (chldwasblocked == 0)
                (void) sm_releasesignal(SIGCHLD);
}

/*
**  COUNT_OPEN_CONNECTIONS
**
**      Parameters:
**              hostaddr - ClientAddress
**
**      Returns:
**              the number of open connections for this client
**
*/

int
count_open_connections(hostaddr)
        SOCKADDR *hostaddr;
{
        int i, n;

        if (hostaddr == NULL)
                return 0;

        /*
        **  This code gets called before proc_list_add() gets called,
        **  so we (the daemon child for this connection) have not yet
        **  counted ourselves.  Hence initialize the counter to 1
        **  instead of 0 to compensate.
        */

        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 /* NETINET */
#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 /* NETINET6 */
        }
        return n;
}