summaryrefslogtreecommitdiffstats
path: root/sendmail/src/util.c
diff options
context:
space:
mode:
Diffstat (limited to 'sendmail/src/util.c')
-rw-r--r--sendmail/src/util.c2853
1 files changed, 2853 insertions, 0 deletions
diff --git a/sendmail/src/util.c b/sendmail/src/util.c
new file mode 100644
index 0000000..95d2f9a
--- /dev/null
+++ b/sendmail/src/util.c
@@ -0,0 +1,2853 @@
+/*
+ * Copyright (c) 1998-2007 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.413 2007/09/26 23:29:11 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 (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 (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;
+
+ /*
+ ** Initialize to 1 instead of 0 because this code gets called
+ ** before proc_list_add() gets called, so we (the daemon child
+ ** for this connection) don't count ourselves.
+ */
+
+ 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;
+}
OpenPOWER on IntegriCloud