1c2aa98e2SPeter Wemm /* 206f25ae9SGregory Neil Shapiro * Copyright (c) 1998-2000 Sendmail, Inc. and its suppliers. 306f25ae9SGregory Neil Shapiro * All rights reserved. 4c2aa98e2SPeter Wemm * Copyright (c) 1983, 1995-1997 Eric P. Allman. All rights reserved. 5c2aa98e2SPeter Wemm * Copyright (c) 1988, 1993 6c2aa98e2SPeter Wemm * The Regents of the University of California. All rights reserved. 7c2aa98e2SPeter Wemm * 8c2aa98e2SPeter Wemm * By using this file, you agree to the terms and conditions set 9c2aa98e2SPeter Wemm * forth in the LICENSE file which can be found at the top level of 10c2aa98e2SPeter Wemm * the sendmail distribution. 11c2aa98e2SPeter Wemm * 12c2aa98e2SPeter Wemm */ 13c2aa98e2SPeter Wemm 14c2aa98e2SPeter Wemm #ifndef lint 1542e5d165SGregory Neil Shapiro static char id[] = "@(#)$Id: readcf.c,v 8.382.4.27 2000/09/28 01:31:16 gshapiro Exp $"; 1606f25ae9SGregory Neil Shapiro #endif /* ! lint */ 17c2aa98e2SPeter Wemm 1806f25ae9SGregory Neil Shapiro #include <sendmail.h> 1906f25ae9SGregory Neil Shapiro 2006f25ae9SGregory Neil Shapiro 2106f25ae9SGregory Neil Shapiro #if NETINET || NETINET6 2206f25ae9SGregory Neil Shapiro # include <arpa/inet.h> 2306f25ae9SGregory Neil Shapiro #endif /* NETINET || NETINET6 */ 2406f25ae9SGregory Neil Shapiro 2506f25ae9SGregory Neil Shapiro #define SECONDS 2606f25ae9SGregory Neil Shapiro #define MINUTES * 60 2706f25ae9SGregory Neil Shapiro #define HOUR * 3600 2806f25ae9SGregory Neil Shapiro #define HOURS HOUR 2906f25ae9SGregory Neil Shapiro 3006f25ae9SGregory Neil Shapiro static void fileclass __P((int, char *, char *, bool, bool)); 3106f25ae9SGregory Neil Shapiro static char **makeargv __P((char *)); 3206f25ae9SGregory Neil Shapiro static void settimeout __P((char *, char *, bool)); 3306f25ae9SGregory Neil Shapiro static void toomany __P((int, int)); 34c2aa98e2SPeter Wemm 35c2aa98e2SPeter Wemm /* 36c2aa98e2SPeter Wemm ** READCF -- read configuration file. 37c2aa98e2SPeter Wemm ** 38c2aa98e2SPeter Wemm ** This routine reads the configuration file and builds the internal 39c2aa98e2SPeter Wemm ** form. 40c2aa98e2SPeter Wemm ** 41c2aa98e2SPeter Wemm ** The file is formatted as a sequence of lines, each taken 42c2aa98e2SPeter Wemm ** atomically. The first character of each line describes how 43c2aa98e2SPeter Wemm ** the line is to be interpreted. The lines are: 44c2aa98e2SPeter Wemm ** Dxval Define macro x to have value val. 45c2aa98e2SPeter Wemm ** Cxword Put word into class x. 46c2aa98e2SPeter Wemm ** Fxfile [fmt] Read file for lines to put into 47c2aa98e2SPeter Wemm ** class x. Use scanf string 'fmt' 48c2aa98e2SPeter Wemm ** or "%s" if not present. Fmt should 49c2aa98e2SPeter Wemm ** only produce one string-valued result. 50c2aa98e2SPeter Wemm ** Hname: value Define header with field-name 'name' 51c2aa98e2SPeter Wemm ** and value as specified; this will be 52c2aa98e2SPeter Wemm ** macro expanded immediately before 53c2aa98e2SPeter Wemm ** use. 54c2aa98e2SPeter Wemm ** Sn Use rewriting set n. 55c2aa98e2SPeter Wemm ** Rlhs rhs Rewrite addresses that match lhs to 56c2aa98e2SPeter Wemm ** be rhs. 57c2aa98e2SPeter Wemm ** Mn arg=val... Define mailer. n is the internal name. 58c2aa98e2SPeter Wemm ** Args specify mailer parameters. 59c2aa98e2SPeter Wemm ** Oxvalue Set option x to value. 60c2aa98e2SPeter Wemm ** Pname=value Set precedence name to value. 61c2aa98e2SPeter Wemm ** Vversioncode[/vendorcode] 62c2aa98e2SPeter Wemm ** Version level/vendor name of 63c2aa98e2SPeter Wemm ** configuration syntax. 64c2aa98e2SPeter Wemm ** Kmapname mapclass arguments.... 65c2aa98e2SPeter Wemm ** Define keyed lookup of a given class. 66c2aa98e2SPeter Wemm ** Arguments are class dependent. 67c2aa98e2SPeter Wemm ** Eenvar=value Set the environment value to the given value. 68c2aa98e2SPeter Wemm ** 69c2aa98e2SPeter Wemm ** Parameters: 70c2aa98e2SPeter Wemm ** cfname -- configuration file name. 71c2aa98e2SPeter Wemm ** safe -- TRUE if this is the system config file; 72c2aa98e2SPeter Wemm ** FALSE otherwise. 73c2aa98e2SPeter Wemm ** e -- the main envelope. 74c2aa98e2SPeter Wemm ** 75c2aa98e2SPeter Wemm ** Returns: 76c2aa98e2SPeter Wemm ** none. 77c2aa98e2SPeter Wemm ** 78c2aa98e2SPeter Wemm ** Side Effects: 79c2aa98e2SPeter Wemm ** Builds several internal tables. 80c2aa98e2SPeter Wemm */ 81c2aa98e2SPeter Wemm 82c2aa98e2SPeter Wemm void 83c2aa98e2SPeter Wemm readcf(cfname, safe, e) 84c2aa98e2SPeter Wemm char *cfname; 85c2aa98e2SPeter Wemm bool safe; 86c2aa98e2SPeter Wemm register ENVELOPE *e; 87c2aa98e2SPeter Wemm { 88c2aa98e2SPeter Wemm FILE *cf; 8906f25ae9SGregory Neil Shapiro int ruleset = -1; 90c2aa98e2SPeter Wemm char *q; 91c2aa98e2SPeter Wemm struct rewrite *rwp = NULL; 92c2aa98e2SPeter Wemm char *bp; 93c2aa98e2SPeter Wemm auto char *ep; 94c2aa98e2SPeter Wemm int nfuzzy; 95c2aa98e2SPeter Wemm char *file; 96c2aa98e2SPeter Wemm bool optional; 97c2aa98e2SPeter Wemm int mid; 98c2aa98e2SPeter Wemm register char *p; 9906f25ae9SGregory Neil Shapiro long sff = SFF_OPENASROOT; 100c2aa98e2SPeter Wemm struct stat statb; 101c2aa98e2SPeter Wemm char buf[MAXLINE]; 102c2aa98e2SPeter Wemm char exbuf[MAXLINE]; 103c2aa98e2SPeter Wemm char pvpbuf[MAXLINE + MAXATOM]; 104c2aa98e2SPeter Wemm static char *null_list[1] = { NULL }; 10506f25ae9SGregory Neil Shapiro extern u_char TokTypeNoC[]; 106c2aa98e2SPeter Wemm 107c2aa98e2SPeter Wemm FileName = cfname; 108c2aa98e2SPeter Wemm LineNumber = 0; 109c2aa98e2SPeter Wemm 110c2aa98e2SPeter Wemm if (DontLockReadFiles) 111c2aa98e2SPeter Wemm sff |= SFF_NOLOCK; 112c2aa98e2SPeter Wemm cf = safefopen(cfname, O_RDONLY, 0444, sff); 113c2aa98e2SPeter Wemm if (cf == NULL) 114c2aa98e2SPeter Wemm { 115c2aa98e2SPeter Wemm syserr("cannot open"); 116065a643dSPeter Wemm finis(FALSE, EX_OSFILE); 117c2aa98e2SPeter Wemm } 118c2aa98e2SPeter Wemm 119c2aa98e2SPeter Wemm if (fstat(fileno(cf), &statb) < 0) 120c2aa98e2SPeter Wemm { 121c2aa98e2SPeter Wemm syserr("cannot fstat"); 122065a643dSPeter Wemm finis(FALSE, EX_OSFILE); 123c2aa98e2SPeter Wemm } 124c2aa98e2SPeter Wemm 125c2aa98e2SPeter Wemm if (!S_ISREG(statb.st_mode)) 126c2aa98e2SPeter Wemm { 127c2aa98e2SPeter Wemm syserr("not a plain file"); 128065a643dSPeter Wemm finis(FALSE, EX_OSFILE); 129c2aa98e2SPeter Wemm } 130c2aa98e2SPeter Wemm 131c2aa98e2SPeter Wemm if (OpMode != MD_TEST && bitset(S_IWGRP|S_IWOTH, statb.st_mode)) 132c2aa98e2SPeter Wemm { 133c2aa98e2SPeter Wemm if (OpMode == MD_DAEMON || OpMode == MD_INITALIAS) 134c2aa98e2SPeter Wemm fprintf(stderr, "%s: WARNING: dangerous write permissions\n", 135c2aa98e2SPeter Wemm FileName); 136c2aa98e2SPeter Wemm if (LogLevel > 0) 137c2aa98e2SPeter Wemm sm_syslog(LOG_CRIT, NOQID, 138c2aa98e2SPeter Wemm "%s: WARNING: dangerous write permissions", 139c2aa98e2SPeter Wemm FileName); 140c2aa98e2SPeter Wemm } 141c2aa98e2SPeter Wemm 142c2aa98e2SPeter Wemm #ifdef XLA 143c2aa98e2SPeter Wemm xla_zero(); 14406f25ae9SGregory Neil Shapiro #endif /* XLA */ 145c2aa98e2SPeter Wemm 146c2aa98e2SPeter Wemm while ((bp = fgetfolded(buf, sizeof buf, cf)) != NULL) 147c2aa98e2SPeter Wemm { 148c2aa98e2SPeter Wemm if (bp[0] == '#') 149c2aa98e2SPeter Wemm { 150c2aa98e2SPeter Wemm if (bp != buf) 151c2aa98e2SPeter Wemm free(bp); 152c2aa98e2SPeter Wemm continue; 153c2aa98e2SPeter Wemm } 154c2aa98e2SPeter Wemm 155c2aa98e2SPeter Wemm /* do macro expansion mappings */ 156c2aa98e2SPeter Wemm translate_dollars(bp); 157c2aa98e2SPeter Wemm 158c2aa98e2SPeter Wemm /* interpret this line */ 159c2aa98e2SPeter Wemm errno = 0; 160c2aa98e2SPeter Wemm switch (bp[0]) 161c2aa98e2SPeter Wemm { 162c2aa98e2SPeter Wemm case '\0': 163c2aa98e2SPeter Wemm case '#': /* comment */ 164c2aa98e2SPeter Wemm break; 165c2aa98e2SPeter Wemm 166c2aa98e2SPeter Wemm case 'R': /* rewriting rule */ 16706f25ae9SGregory Neil Shapiro if (ruleset < 0) 16806f25ae9SGregory Neil Shapiro { 16906f25ae9SGregory Neil Shapiro syserr("missing valid ruleset for \"%s\"", bp); 17006f25ae9SGregory Neil Shapiro break; 17106f25ae9SGregory Neil Shapiro } 172c2aa98e2SPeter Wemm for (p = &bp[1]; *p != '\0' && *p != '\t'; p++) 173c2aa98e2SPeter Wemm continue; 174c2aa98e2SPeter Wemm 175c2aa98e2SPeter Wemm if (*p == '\0') 176c2aa98e2SPeter Wemm { 177c2aa98e2SPeter Wemm syserr("invalid rewrite line \"%s\" (tab expected)", bp); 178c2aa98e2SPeter Wemm break; 179c2aa98e2SPeter Wemm } 180c2aa98e2SPeter Wemm 181c2aa98e2SPeter Wemm /* allocate space for the rule header */ 182c2aa98e2SPeter Wemm if (rwp == NULL) 183c2aa98e2SPeter Wemm { 184c2aa98e2SPeter Wemm RewriteRules[ruleset] = rwp = 185c2aa98e2SPeter Wemm (struct rewrite *) xalloc(sizeof *rwp); 186c2aa98e2SPeter Wemm } 187c2aa98e2SPeter Wemm else 188c2aa98e2SPeter Wemm { 189c2aa98e2SPeter Wemm rwp->r_next = (struct rewrite *) xalloc(sizeof *rwp); 190c2aa98e2SPeter Wemm rwp = rwp->r_next; 191c2aa98e2SPeter Wemm } 192c2aa98e2SPeter Wemm rwp->r_next = NULL; 193c2aa98e2SPeter Wemm 194c2aa98e2SPeter Wemm /* expand and save the LHS */ 195c2aa98e2SPeter Wemm *p = '\0'; 196c2aa98e2SPeter Wemm expand(&bp[1], exbuf, sizeof exbuf, e); 197c2aa98e2SPeter Wemm rwp->r_lhs = prescan(exbuf, '\t', pvpbuf, 19806f25ae9SGregory Neil Shapiro sizeof pvpbuf, NULL, 19906f25ae9SGregory Neil Shapiro ConfigLevel >= 9 ? TokTypeNoC : NULL); 200c2aa98e2SPeter Wemm nfuzzy = 0; 201c2aa98e2SPeter Wemm if (rwp->r_lhs != NULL) 202c2aa98e2SPeter Wemm { 203c2aa98e2SPeter Wemm register char **ap; 204c2aa98e2SPeter Wemm 205c2aa98e2SPeter Wemm rwp->r_lhs = copyplist(rwp->r_lhs, TRUE); 206c2aa98e2SPeter Wemm 207c2aa98e2SPeter Wemm /* count the number of fuzzy matches in LHS */ 208c2aa98e2SPeter Wemm for (ap = rwp->r_lhs; *ap != NULL; ap++) 209c2aa98e2SPeter Wemm { 210c2aa98e2SPeter Wemm char *botch; 211c2aa98e2SPeter Wemm 212c2aa98e2SPeter Wemm botch = NULL; 213c2aa98e2SPeter Wemm switch (**ap & 0377) 214c2aa98e2SPeter Wemm { 215c2aa98e2SPeter Wemm case MATCHZANY: 216c2aa98e2SPeter Wemm case MATCHANY: 217c2aa98e2SPeter Wemm case MATCHONE: 218c2aa98e2SPeter Wemm case MATCHCLASS: 219c2aa98e2SPeter Wemm case MATCHNCLASS: 220c2aa98e2SPeter Wemm nfuzzy++; 221c2aa98e2SPeter Wemm break; 222c2aa98e2SPeter Wemm 223c2aa98e2SPeter Wemm case MATCHREPL: 224c2aa98e2SPeter Wemm botch = "$0-$9"; 225c2aa98e2SPeter Wemm break; 226c2aa98e2SPeter Wemm 227c2aa98e2SPeter Wemm case CANONUSER: 228c2aa98e2SPeter Wemm botch = "$:"; 229c2aa98e2SPeter Wemm break; 230c2aa98e2SPeter Wemm 231c2aa98e2SPeter Wemm case CALLSUBR: 232c2aa98e2SPeter Wemm botch = "$>"; 233c2aa98e2SPeter Wemm break; 234c2aa98e2SPeter Wemm 235c2aa98e2SPeter Wemm case CONDIF: 236c2aa98e2SPeter Wemm botch = "$?"; 237c2aa98e2SPeter Wemm break; 238c2aa98e2SPeter Wemm 239c2aa98e2SPeter Wemm case CONDFI: 240c2aa98e2SPeter Wemm botch = "$."; 241c2aa98e2SPeter Wemm break; 242c2aa98e2SPeter Wemm 243c2aa98e2SPeter Wemm case HOSTBEGIN: 244c2aa98e2SPeter Wemm botch = "$["; 245c2aa98e2SPeter Wemm break; 246c2aa98e2SPeter Wemm 247c2aa98e2SPeter Wemm case HOSTEND: 248c2aa98e2SPeter Wemm botch = "$]"; 249c2aa98e2SPeter Wemm break; 250c2aa98e2SPeter Wemm 251c2aa98e2SPeter Wemm case LOOKUPBEGIN: 252c2aa98e2SPeter Wemm botch = "$("; 253c2aa98e2SPeter Wemm break; 254c2aa98e2SPeter Wemm 255c2aa98e2SPeter Wemm case LOOKUPEND: 256c2aa98e2SPeter Wemm botch = "$)"; 257c2aa98e2SPeter Wemm break; 258c2aa98e2SPeter Wemm } 259c2aa98e2SPeter Wemm if (botch != NULL) 260c2aa98e2SPeter Wemm syserr("Inappropriate use of %s on LHS", 261c2aa98e2SPeter Wemm botch); 262c2aa98e2SPeter Wemm } 26306f25ae9SGregory Neil Shapiro rwp->r_line = LineNumber; 264c2aa98e2SPeter Wemm } 265c2aa98e2SPeter Wemm else 266c2aa98e2SPeter Wemm { 267c2aa98e2SPeter Wemm syserr("R line: null LHS"); 268c2aa98e2SPeter Wemm rwp->r_lhs = null_list; 269c2aa98e2SPeter Wemm } 270c2aa98e2SPeter Wemm 271c2aa98e2SPeter Wemm /* expand and save the RHS */ 272c2aa98e2SPeter Wemm while (*++p == '\t') 273c2aa98e2SPeter Wemm continue; 274c2aa98e2SPeter Wemm q = p; 275c2aa98e2SPeter Wemm while (*p != '\0' && *p != '\t') 276c2aa98e2SPeter Wemm p++; 277c2aa98e2SPeter Wemm *p = '\0'; 278c2aa98e2SPeter Wemm expand(q, exbuf, sizeof exbuf, e); 279c2aa98e2SPeter Wemm rwp->r_rhs = prescan(exbuf, '\t', pvpbuf, 28006f25ae9SGregory Neil Shapiro sizeof pvpbuf, NULL, 28106f25ae9SGregory Neil Shapiro ConfigLevel >= 9 ? TokTypeNoC : NULL); 282c2aa98e2SPeter Wemm if (rwp->r_rhs != NULL) 283c2aa98e2SPeter Wemm { 284c2aa98e2SPeter Wemm register char **ap; 285c2aa98e2SPeter Wemm 286c2aa98e2SPeter Wemm rwp->r_rhs = copyplist(rwp->r_rhs, TRUE); 287c2aa98e2SPeter Wemm 288c2aa98e2SPeter Wemm /* check no out-of-bounds replacements */ 289c2aa98e2SPeter Wemm nfuzzy += '0'; 290c2aa98e2SPeter Wemm for (ap = rwp->r_rhs; *ap != NULL; ap++) 291c2aa98e2SPeter Wemm { 292c2aa98e2SPeter Wemm char *botch; 293c2aa98e2SPeter Wemm 294c2aa98e2SPeter Wemm botch = NULL; 295c2aa98e2SPeter Wemm switch (**ap & 0377) 296c2aa98e2SPeter Wemm { 297c2aa98e2SPeter Wemm case MATCHREPL: 298c2aa98e2SPeter Wemm if ((*ap)[1] <= '0' || (*ap)[1] > nfuzzy) 299c2aa98e2SPeter Wemm { 300c2aa98e2SPeter Wemm syserr("replacement $%c out of bounds", 301c2aa98e2SPeter Wemm (*ap)[1]); 302c2aa98e2SPeter Wemm } 303c2aa98e2SPeter Wemm break; 304c2aa98e2SPeter Wemm 305c2aa98e2SPeter Wemm case MATCHZANY: 306c2aa98e2SPeter Wemm botch = "$*"; 307c2aa98e2SPeter Wemm break; 308c2aa98e2SPeter Wemm 309c2aa98e2SPeter Wemm case MATCHANY: 310c2aa98e2SPeter Wemm botch = "$+"; 311c2aa98e2SPeter Wemm break; 312c2aa98e2SPeter Wemm 313c2aa98e2SPeter Wemm case MATCHONE: 314c2aa98e2SPeter Wemm botch = "$-"; 315c2aa98e2SPeter Wemm break; 316c2aa98e2SPeter Wemm 317c2aa98e2SPeter Wemm case MATCHCLASS: 318c2aa98e2SPeter Wemm botch = "$="; 319c2aa98e2SPeter Wemm break; 320c2aa98e2SPeter Wemm 321c2aa98e2SPeter Wemm case MATCHNCLASS: 322c2aa98e2SPeter Wemm botch = "$~"; 323c2aa98e2SPeter Wemm break; 324c2aa98e2SPeter Wemm } 325c2aa98e2SPeter Wemm if (botch != NULL) 326c2aa98e2SPeter Wemm syserr("Inappropriate use of %s on RHS", 327c2aa98e2SPeter Wemm botch); 328c2aa98e2SPeter Wemm } 329c2aa98e2SPeter Wemm } 330c2aa98e2SPeter Wemm else 331c2aa98e2SPeter Wemm { 332c2aa98e2SPeter Wemm syserr("R line: null RHS"); 333c2aa98e2SPeter Wemm rwp->r_rhs = null_list; 334c2aa98e2SPeter Wemm } 335c2aa98e2SPeter Wemm break; 336c2aa98e2SPeter Wemm 337c2aa98e2SPeter Wemm case 'S': /* select rewriting set */ 338c2aa98e2SPeter Wemm expand(&bp[1], exbuf, sizeof exbuf, e); 339c2aa98e2SPeter Wemm ruleset = strtorwset(exbuf, NULL, ST_ENTER); 340c2aa98e2SPeter Wemm if (ruleset < 0) 341c2aa98e2SPeter Wemm break; 34206f25ae9SGregory Neil Shapiro 343c2aa98e2SPeter Wemm rwp = RewriteRules[ruleset]; 344c2aa98e2SPeter Wemm if (rwp != NULL) 345c2aa98e2SPeter Wemm { 34606f25ae9SGregory Neil Shapiro if (OpMode == MD_TEST) 347c2aa98e2SPeter Wemm printf("WARNING: Ruleset %s has multiple definitions\n", 348c2aa98e2SPeter Wemm &bp[1]); 34906f25ae9SGregory Neil Shapiro if (tTd(37, 1)) 35006f25ae9SGregory Neil Shapiro dprintf("WARNING: Ruleset %s has multiple definitions\n", 35106f25ae9SGregory Neil Shapiro &bp[1]); 352c2aa98e2SPeter Wemm while (rwp->r_next != NULL) 353c2aa98e2SPeter Wemm rwp = rwp->r_next; 354c2aa98e2SPeter Wemm } 355c2aa98e2SPeter Wemm break; 356c2aa98e2SPeter Wemm 357c2aa98e2SPeter Wemm case 'D': /* macro definition */ 358c2aa98e2SPeter Wemm mid = macid(&bp[1], &ep); 359c2aa98e2SPeter Wemm p = munchstring(ep, NULL, '\0'); 360c2aa98e2SPeter Wemm define(mid, newstr(p), e); 361c2aa98e2SPeter Wemm break; 362c2aa98e2SPeter Wemm 363c2aa98e2SPeter Wemm case 'H': /* required header line */ 36406f25ae9SGregory Neil Shapiro (void) chompheader(&bp[1], CHHDR_DEF, NULL, e); 365c2aa98e2SPeter Wemm break; 366c2aa98e2SPeter Wemm 367c2aa98e2SPeter Wemm case 'C': /* word class */ 368c2aa98e2SPeter Wemm case 'T': /* trusted user (set class `t') */ 369c2aa98e2SPeter Wemm if (bp[0] == 'C') 370c2aa98e2SPeter Wemm { 371c2aa98e2SPeter Wemm mid = macid(&bp[1], &ep); 372c2aa98e2SPeter Wemm expand(ep, exbuf, sizeof exbuf, e); 373c2aa98e2SPeter Wemm p = exbuf; 374c2aa98e2SPeter Wemm } 375c2aa98e2SPeter Wemm else 376c2aa98e2SPeter Wemm { 377c2aa98e2SPeter Wemm mid = 't'; 378c2aa98e2SPeter Wemm p = &bp[1]; 379c2aa98e2SPeter Wemm } 380c2aa98e2SPeter Wemm while (*p != '\0') 381c2aa98e2SPeter Wemm { 382c2aa98e2SPeter Wemm register char *wd; 383c2aa98e2SPeter Wemm char delim; 384c2aa98e2SPeter Wemm 385c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && isspace(*p)) 386c2aa98e2SPeter Wemm p++; 387c2aa98e2SPeter Wemm wd = p; 388c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 389c2aa98e2SPeter Wemm p++; 390c2aa98e2SPeter Wemm delim = *p; 391c2aa98e2SPeter Wemm *p = '\0'; 392c2aa98e2SPeter Wemm if (wd[0] != '\0') 393c2aa98e2SPeter Wemm setclass(mid, wd); 394c2aa98e2SPeter Wemm *p = delim; 395c2aa98e2SPeter Wemm } 396c2aa98e2SPeter Wemm break; 397c2aa98e2SPeter Wemm 398c2aa98e2SPeter Wemm case 'F': /* word class from file */ 399c2aa98e2SPeter Wemm mid = macid(&bp[1], &ep); 400c2aa98e2SPeter Wemm for (p = ep; isascii(*p) && isspace(*p); ) 401c2aa98e2SPeter Wemm p++; 402c2aa98e2SPeter Wemm if (p[0] == '-' && p[1] == 'o') 403c2aa98e2SPeter Wemm { 404c2aa98e2SPeter Wemm optional = TRUE; 405c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 406c2aa98e2SPeter Wemm p++; 407c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 408c2aa98e2SPeter Wemm p++; 409c2aa98e2SPeter Wemm } 410c2aa98e2SPeter Wemm else 411c2aa98e2SPeter Wemm optional = FALSE; 41206f25ae9SGregory Neil Shapiro 413c2aa98e2SPeter Wemm file = p; 41406f25ae9SGregory Neil Shapiro q = p; 41506f25ae9SGregory Neil Shapiro while (*q != '\0' && !(isascii(*q) && isspace(*q))) 41606f25ae9SGregory Neil Shapiro q++; 417c2aa98e2SPeter Wemm if (*file == '|') 418c2aa98e2SPeter Wemm p = "%s"; 419c2aa98e2SPeter Wemm else 420c2aa98e2SPeter Wemm { 42106f25ae9SGregory Neil Shapiro p = q; 422c2aa98e2SPeter Wemm if (*p == '\0') 423c2aa98e2SPeter Wemm p = "%s"; 424c2aa98e2SPeter Wemm else 425c2aa98e2SPeter Wemm { 426c2aa98e2SPeter Wemm *p = '\0'; 427c2aa98e2SPeter Wemm while (isascii(*++p) && isspace(*p)) 428c2aa98e2SPeter Wemm continue; 429c2aa98e2SPeter Wemm } 430c2aa98e2SPeter Wemm } 431c2aa98e2SPeter Wemm fileclass(mid, file, p, safe, optional); 432c2aa98e2SPeter Wemm break; 433c2aa98e2SPeter Wemm 434c2aa98e2SPeter Wemm #ifdef XLA 435c2aa98e2SPeter Wemm case 'L': /* extended load average description */ 436c2aa98e2SPeter Wemm xla_init(&bp[1]); 437c2aa98e2SPeter Wemm break; 43806f25ae9SGregory Neil Shapiro #endif /* XLA */ 439c2aa98e2SPeter Wemm 440c2aa98e2SPeter Wemm #if defined(SUN_EXTENSIONS) && defined(SUN_LOOKUP_MACRO) 441c2aa98e2SPeter Wemm case 'L': /* lookup macro */ 442c2aa98e2SPeter Wemm case 'G': /* lookup class */ 443c2aa98e2SPeter Wemm /* reserved for Sun -- NIS+ database lookup */ 444c2aa98e2SPeter Wemm if (VendorCode != VENDOR_SUN) 445c2aa98e2SPeter Wemm goto badline; 446c2aa98e2SPeter Wemm sun_lg_config_line(bp, e); 447c2aa98e2SPeter Wemm break; 44806f25ae9SGregory Neil Shapiro #endif /* defined(SUN_EXTENSIONS) && defined(SUN_LOOKUP_MACRO) */ 449c2aa98e2SPeter Wemm 450c2aa98e2SPeter Wemm case 'M': /* define mailer */ 451c2aa98e2SPeter Wemm makemailer(&bp[1]); 452c2aa98e2SPeter Wemm break; 453c2aa98e2SPeter Wemm 454c2aa98e2SPeter Wemm case 'O': /* set option */ 455c2aa98e2SPeter Wemm setoption(bp[1], &bp[2], safe, FALSE, e); 456c2aa98e2SPeter Wemm break; 457c2aa98e2SPeter Wemm 458c2aa98e2SPeter Wemm case 'P': /* set precedence */ 459c2aa98e2SPeter Wemm if (NumPriorities >= MAXPRIORITIES) 460c2aa98e2SPeter Wemm { 461c2aa98e2SPeter Wemm toomany('P', MAXPRIORITIES); 462c2aa98e2SPeter Wemm break; 463c2aa98e2SPeter Wemm } 464c2aa98e2SPeter Wemm for (p = &bp[1]; *p != '\0' && *p != '='; p++) 465c2aa98e2SPeter Wemm continue; 466c2aa98e2SPeter Wemm if (*p == '\0') 467c2aa98e2SPeter Wemm goto badline; 468c2aa98e2SPeter Wemm *p = '\0'; 469c2aa98e2SPeter Wemm Priorities[NumPriorities].pri_name = newstr(&bp[1]); 470c2aa98e2SPeter Wemm Priorities[NumPriorities].pri_val = atoi(++p); 471c2aa98e2SPeter Wemm NumPriorities++; 472c2aa98e2SPeter Wemm break; 473c2aa98e2SPeter Wemm 474c2aa98e2SPeter Wemm case 'V': /* configuration syntax version */ 475c2aa98e2SPeter Wemm for (p = &bp[1]; isascii(*p) && isspace(*p); p++) 476c2aa98e2SPeter Wemm continue; 477c2aa98e2SPeter Wemm if (!isascii(*p) || !isdigit(*p)) 478c2aa98e2SPeter Wemm { 479c2aa98e2SPeter Wemm syserr("invalid argument to V line: \"%.20s\"", 480c2aa98e2SPeter Wemm &bp[1]); 481c2aa98e2SPeter Wemm break; 482c2aa98e2SPeter Wemm } 483c2aa98e2SPeter Wemm ConfigLevel = strtol(p, &ep, 10); 484c2aa98e2SPeter Wemm 485c2aa98e2SPeter Wemm /* 486c2aa98e2SPeter Wemm ** Do heuristic tweaking for back compatibility. 487c2aa98e2SPeter Wemm */ 488c2aa98e2SPeter Wemm 489c2aa98e2SPeter Wemm if (ConfigLevel >= 5) 490c2aa98e2SPeter Wemm { 491c2aa98e2SPeter Wemm /* level 5 configs have short name in $w */ 492c2aa98e2SPeter Wemm p = macvalue('w', e); 493c2aa98e2SPeter Wemm if (p != NULL && (p = strchr(p, '.')) != NULL) 494c2aa98e2SPeter Wemm *p = '\0'; 495c2aa98e2SPeter Wemm define('w', macvalue('w', e), e); 496c2aa98e2SPeter Wemm } 497c2aa98e2SPeter Wemm if (ConfigLevel >= 6) 498c2aa98e2SPeter Wemm { 499c2aa98e2SPeter Wemm ColonOkInAddr = FALSE; 500c2aa98e2SPeter Wemm } 501c2aa98e2SPeter Wemm 502c2aa98e2SPeter Wemm /* 503c2aa98e2SPeter Wemm ** Look for vendor code. 504c2aa98e2SPeter Wemm */ 505c2aa98e2SPeter Wemm 506c2aa98e2SPeter Wemm if (*ep++ == '/') 507c2aa98e2SPeter Wemm { 508c2aa98e2SPeter Wemm /* extract vendor code */ 509c2aa98e2SPeter Wemm for (p = ep; isascii(*p) && isalpha(*p); ) 510c2aa98e2SPeter Wemm p++; 511c2aa98e2SPeter Wemm *p = '\0'; 512c2aa98e2SPeter Wemm 513c2aa98e2SPeter Wemm if (!setvendor(ep)) 514c2aa98e2SPeter Wemm syserr("invalid V line vendor code: \"%s\"", 515c2aa98e2SPeter Wemm ep); 516c2aa98e2SPeter Wemm } 517c2aa98e2SPeter Wemm break; 518c2aa98e2SPeter Wemm 519c2aa98e2SPeter Wemm case 'K': 520c2aa98e2SPeter Wemm expand(&bp[1], exbuf, sizeof exbuf, e); 521c2aa98e2SPeter Wemm (void) makemapentry(exbuf); 522c2aa98e2SPeter Wemm break; 523c2aa98e2SPeter Wemm 524c2aa98e2SPeter Wemm case 'E': 525c2aa98e2SPeter Wemm p = strchr(bp, '='); 526c2aa98e2SPeter Wemm if (p != NULL) 527c2aa98e2SPeter Wemm *p++ = '\0'; 528c2aa98e2SPeter Wemm setuserenv(&bp[1], p); 529c2aa98e2SPeter Wemm break; 530c2aa98e2SPeter Wemm 53106f25ae9SGregory Neil Shapiro #if _FFR_MILTER 53206f25ae9SGregory Neil Shapiro case 'X': /* mail filter */ 53306f25ae9SGregory Neil Shapiro milter_setup(&bp[1]); 53406f25ae9SGregory Neil Shapiro break; 53506f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 53606f25ae9SGregory Neil Shapiro 537c2aa98e2SPeter Wemm default: 538c2aa98e2SPeter Wemm badline: 539c2aa98e2SPeter Wemm syserr("unknown configuration line \"%s\"", bp); 540c2aa98e2SPeter Wemm } 541c2aa98e2SPeter Wemm if (bp != buf) 542c2aa98e2SPeter Wemm free(bp); 543c2aa98e2SPeter Wemm } 544c2aa98e2SPeter Wemm if (ferror(cf)) 545c2aa98e2SPeter Wemm { 546c2aa98e2SPeter Wemm syserr("I/O read error"); 547065a643dSPeter Wemm finis(FALSE, EX_OSFILE); 548c2aa98e2SPeter Wemm } 54906f25ae9SGregory Neil Shapiro (void) fclose(cf); 550c2aa98e2SPeter Wemm FileName = NULL; 551c2aa98e2SPeter Wemm 552c2aa98e2SPeter Wemm /* initialize host maps from local service tables */ 553c2aa98e2SPeter Wemm inithostmaps(); 554c2aa98e2SPeter Wemm 55506f25ae9SGregory Neil Shapiro /* initialize daemon (if not defined yet) */ 55606f25ae9SGregory Neil Shapiro initdaemon(); 55706f25ae9SGregory Neil Shapiro 558c2aa98e2SPeter Wemm /* determine if we need to do special name-server frotz */ 559c2aa98e2SPeter Wemm { 560c2aa98e2SPeter Wemm int nmaps; 561c2aa98e2SPeter Wemm char *maptype[MAXMAPSTACK]; 562c2aa98e2SPeter Wemm short mapreturn[MAXMAPACTIONS]; 563c2aa98e2SPeter Wemm 564c2aa98e2SPeter Wemm nmaps = switch_map_find("hosts", maptype, mapreturn); 565c2aa98e2SPeter Wemm UseNameServer = FALSE; 566c2aa98e2SPeter Wemm if (nmaps > 0 && nmaps <= MAXMAPSTACK) 567c2aa98e2SPeter Wemm { 568c2aa98e2SPeter Wemm register int mapno; 569c2aa98e2SPeter Wemm 570c2aa98e2SPeter Wemm for (mapno = 0; mapno < nmaps && !UseNameServer; mapno++) 571c2aa98e2SPeter Wemm { 572c2aa98e2SPeter Wemm if (strcmp(maptype[mapno], "dns") == 0) 573c2aa98e2SPeter Wemm UseNameServer = TRUE; 574c2aa98e2SPeter Wemm } 575c2aa98e2SPeter Wemm } 576c2aa98e2SPeter Wemm 577c2aa98e2SPeter Wemm #ifdef HESIOD 578c2aa98e2SPeter Wemm nmaps = switch_map_find("passwd", maptype, mapreturn); 579c2aa98e2SPeter Wemm UseHesiod = FALSE; 580c2aa98e2SPeter Wemm if (nmaps > 0 && nmaps <= MAXMAPSTACK) 581c2aa98e2SPeter Wemm { 582c2aa98e2SPeter Wemm register int mapno; 583c2aa98e2SPeter Wemm 584c2aa98e2SPeter Wemm for (mapno = 0; mapno < nmaps && !UseHesiod; mapno++) 585c2aa98e2SPeter Wemm { 586c2aa98e2SPeter Wemm if (strcmp(maptype[mapno], "hesiod") == 0) 587c2aa98e2SPeter Wemm UseHesiod = TRUE; 588c2aa98e2SPeter Wemm } 589c2aa98e2SPeter Wemm } 59006f25ae9SGregory Neil Shapiro #endif /* HESIOD */ 591c2aa98e2SPeter Wemm } 592c2aa98e2SPeter Wemm } 593c2aa98e2SPeter Wemm /* 594c2aa98e2SPeter Wemm ** TRANSLATE_DOLLARS -- convert $x into internal form 595c2aa98e2SPeter Wemm ** 596c2aa98e2SPeter Wemm ** Actually does all appropriate pre-processing of a config line 597c2aa98e2SPeter Wemm ** to turn it into internal form. 598c2aa98e2SPeter Wemm ** 599c2aa98e2SPeter Wemm ** Parameters: 600c2aa98e2SPeter Wemm ** bp -- the buffer to translate. 601c2aa98e2SPeter Wemm ** 602c2aa98e2SPeter Wemm ** Returns: 603c2aa98e2SPeter Wemm ** None. The buffer is translated in place. Since the 604c2aa98e2SPeter Wemm ** translations always make the buffer shorter, this is 605c2aa98e2SPeter Wemm ** safe without a size parameter. 606c2aa98e2SPeter Wemm */ 607c2aa98e2SPeter Wemm 608c2aa98e2SPeter Wemm void 609c2aa98e2SPeter Wemm translate_dollars(bp) 610c2aa98e2SPeter Wemm char *bp; 611c2aa98e2SPeter Wemm { 612c2aa98e2SPeter Wemm register char *p; 613c2aa98e2SPeter Wemm auto char *ep; 614c2aa98e2SPeter Wemm 615c2aa98e2SPeter Wemm for (p = bp; *p != '\0'; p++) 616c2aa98e2SPeter Wemm { 617c2aa98e2SPeter Wemm if (*p == '#' && p > bp && ConfigLevel >= 3) 618c2aa98e2SPeter Wemm { 619c2aa98e2SPeter Wemm /* this is an on-line comment */ 620c2aa98e2SPeter Wemm register char *e; 621c2aa98e2SPeter Wemm 622c2aa98e2SPeter Wemm switch (*--p & 0377) 623c2aa98e2SPeter Wemm { 624c2aa98e2SPeter Wemm case MACROEXPAND: 625c2aa98e2SPeter Wemm /* it's from $# -- let it go through */ 626c2aa98e2SPeter Wemm p++; 627c2aa98e2SPeter Wemm break; 628c2aa98e2SPeter Wemm 629c2aa98e2SPeter Wemm case '\\': 630c2aa98e2SPeter Wemm /* it's backslash escaped */ 63106f25ae9SGregory Neil Shapiro (void) strlcpy(p, p + 1, strlen(p)); 632c2aa98e2SPeter Wemm break; 633c2aa98e2SPeter Wemm 634c2aa98e2SPeter Wemm default: 63506f25ae9SGregory Neil Shapiro /* delete leading white space */ 636c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p) && 637c2aa98e2SPeter Wemm *p != '\n' && p > bp) 638c2aa98e2SPeter Wemm p--; 639c2aa98e2SPeter Wemm if ((e = strchr(++p, '\n')) != NULL) 64006f25ae9SGregory Neil Shapiro (void) strlcpy(p, e, strlen(p)); 641c2aa98e2SPeter Wemm else 642c2aa98e2SPeter Wemm *p-- = '\0'; 643c2aa98e2SPeter Wemm break; 644c2aa98e2SPeter Wemm } 645c2aa98e2SPeter Wemm continue; 646c2aa98e2SPeter Wemm } 647c2aa98e2SPeter Wemm 648c2aa98e2SPeter Wemm if (*p != '$' || p[1] == '\0') 649c2aa98e2SPeter Wemm continue; 650c2aa98e2SPeter Wemm 651c2aa98e2SPeter Wemm if (p[1] == '$') 652c2aa98e2SPeter Wemm { 653c2aa98e2SPeter Wemm /* actual dollar sign.... */ 65406f25ae9SGregory Neil Shapiro (void) strlcpy(p, p + 1, strlen(p)); 655c2aa98e2SPeter Wemm continue; 656c2aa98e2SPeter Wemm } 657c2aa98e2SPeter Wemm 658c2aa98e2SPeter Wemm /* convert to macro expansion character */ 659c2aa98e2SPeter Wemm *p++ = MACROEXPAND; 660c2aa98e2SPeter Wemm 661c2aa98e2SPeter Wemm /* special handling for $=, $~, $&, and $? */ 662c2aa98e2SPeter Wemm if (*p == '=' || *p == '~' || *p == '&' || *p == '?') 663c2aa98e2SPeter Wemm p++; 664c2aa98e2SPeter Wemm 665c2aa98e2SPeter Wemm /* convert macro name to code */ 666c2aa98e2SPeter Wemm *p = macid(p, &ep); 66706f25ae9SGregory Neil Shapiro if (ep != p + 1) 66806f25ae9SGregory Neil Shapiro (void) strlcpy(p + 1, ep, strlen(p + 1)); 669c2aa98e2SPeter Wemm } 670c2aa98e2SPeter Wemm 671c2aa98e2SPeter Wemm /* strip trailing white space from the line */ 672c2aa98e2SPeter Wemm while (--p > bp && isascii(*p) && isspace(*p)) 673c2aa98e2SPeter Wemm *p = '\0'; 674c2aa98e2SPeter Wemm } 675c2aa98e2SPeter Wemm /* 676c2aa98e2SPeter Wemm ** TOOMANY -- signal too many of some option 677c2aa98e2SPeter Wemm ** 678c2aa98e2SPeter Wemm ** Parameters: 679c2aa98e2SPeter Wemm ** id -- the id of the error line 680c2aa98e2SPeter Wemm ** maxcnt -- the maximum possible values 681c2aa98e2SPeter Wemm ** 682c2aa98e2SPeter Wemm ** Returns: 683c2aa98e2SPeter Wemm ** none. 684c2aa98e2SPeter Wemm ** 685c2aa98e2SPeter Wemm ** Side Effects: 686c2aa98e2SPeter Wemm ** gives a syserr. 687c2aa98e2SPeter Wemm */ 688c2aa98e2SPeter Wemm 68906f25ae9SGregory Neil Shapiro static void 690c2aa98e2SPeter Wemm toomany(id, maxcnt) 691c2aa98e2SPeter Wemm int id; 692c2aa98e2SPeter Wemm int maxcnt; 693c2aa98e2SPeter Wemm { 694c2aa98e2SPeter Wemm syserr("too many %c lines, %d max", id, maxcnt); 695c2aa98e2SPeter Wemm } 696c2aa98e2SPeter Wemm /* 697c2aa98e2SPeter Wemm ** FILECLASS -- read members of a class from a file 698c2aa98e2SPeter Wemm ** 699c2aa98e2SPeter Wemm ** Parameters: 700c2aa98e2SPeter Wemm ** class -- class to define. 701c2aa98e2SPeter Wemm ** filename -- name of file to read. 702c2aa98e2SPeter Wemm ** fmt -- scanf string to use for match. 703c2aa98e2SPeter Wemm ** safe -- if set, this is a safe read. 704c2aa98e2SPeter Wemm ** optional -- if set, it is not an error for the file to 705c2aa98e2SPeter Wemm ** not exist. 706c2aa98e2SPeter Wemm ** 707c2aa98e2SPeter Wemm ** Returns: 708c2aa98e2SPeter Wemm ** none 709c2aa98e2SPeter Wemm ** 710c2aa98e2SPeter Wemm ** Side Effects: 711c2aa98e2SPeter Wemm ** 712c2aa98e2SPeter Wemm ** puts all lines in filename that match a scanf into 713c2aa98e2SPeter Wemm ** the named class. 714c2aa98e2SPeter Wemm */ 715c2aa98e2SPeter Wemm 71606f25ae9SGregory Neil Shapiro static void 717c2aa98e2SPeter Wemm fileclass(class, filename, fmt, safe, optional) 718c2aa98e2SPeter Wemm int class; 719c2aa98e2SPeter Wemm char *filename; 720c2aa98e2SPeter Wemm char *fmt; 721c2aa98e2SPeter Wemm bool safe; 722c2aa98e2SPeter Wemm bool optional; 723c2aa98e2SPeter Wemm { 724c2aa98e2SPeter Wemm FILE *f; 72506f25ae9SGregory Neil Shapiro long sff; 726c2aa98e2SPeter Wemm pid_t pid; 727c2aa98e2SPeter Wemm register char *p; 728c2aa98e2SPeter Wemm char buf[MAXLINE]; 729c2aa98e2SPeter Wemm 730c2aa98e2SPeter Wemm if (tTd(37, 2)) 73106f25ae9SGregory Neil Shapiro dprintf("fileclass(%s, fmt=%s)\n", filename, fmt); 732c2aa98e2SPeter Wemm 733c2aa98e2SPeter Wemm if (filename[0] == '|') 734c2aa98e2SPeter Wemm { 735c2aa98e2SPeter Wemm auto int fd; 736c2aa98e2SPeter Wemm int i; 737c2aa98e2SPeter Wemm char *argv[MAXPV + 1]; 738c2aa98e2SPeter Wemm 739c2aa98e2SPeter Wemm i = 0; 740c2aa98e2SPeter Wemm for (p = strtok(&filename[1], " \t"); p != NULL; p = strtok(NULL, " \t")) 741c2aa98e2SPeter Wemm { 742c2aa98e2SPeter Wemm if (i >= MAXPV) 743c2aa98e2SPeter Wemm break; 744c2aa98e2SPeter Wemm argv[i++] = p; 745c2aa98e2SPeter Wemm } 746c2aa98e2SPeter Wemm argv[i] = NULL; 747c2aa98e2SPeter Wemm pid = prog_open(argv, &fd, CurEnv); 748c2aa98e2SPeter Wemm if (pid < 0) 749c2aa98e2SPeter Wemm f = NULL; 750c2aa98e2SPeter Wemm else 751c2aa98e2SPeter Wemm f = fdopen(fd, "r"); 752c2aa98e2SPeter Wemm } 753c2aa98e2SPeter Wemm else 754c2aa98e2SPeter Wemm { 755c2aa98e2SPeter Wemm pid = -1; 756c2aa98e2SPeter Wemm sff = SFF_REGONLY; 75706f25ae9SGregory Neil Shapiro if (!bitnset(DBS_CLASSFILEINUNSAFEDIRPATH, DontBlameSendmail)) 758c2aa98e2SPeter Wemm sff |= SFF_SAFEDIRPATH; 75906f25ae9SGregory Neil Shapiro if (!bitnset(DBS_LINKEDCLASSFILEINWRITABLEDIR, 76006f25ae9SGregory Neil Shapiro DontBlameSendmail)) 761c2aa98e2SPeter Wemm sff |= SFF_NOWLINK; 762c2aa98e2SPeter Wemm if (safe) 763c2aa98e2SPeter Wemm sff |= SFF_OPENASROOT; 764c2aa98e2SPeter Wemm if (DontLockReadFiles) 765c2aa98e2SPeter Wemm sff |= SFF_NOLOCK; 766c2aa98e2SPeter Wemm f = safefopen(filename, O_RDONLY, 0, sff); 767c2aa98e2SPeter Wemm } 768c2aa98e2SPeter Wemm if (f == NULL) 769c2aa98e2SPeter Wemm { 770c2aa98e2SPeter Wemm if (!optional) 77106f25ae9SGregory Neil Shapiro syserr("fileclass: cannot open '%s'", filename); 772c2aa98e2SPeter Wemm return; 773c2aa98e2SPeter Wemm } 774c2aa98e2SPeter Wemm 775c2aa98e2SPeter Wemm while (fgets(buf, sizeof buf, f) != NULL) 776c2aa98e2SPeter Wemm { 777c2aa98e2SPeter Wemm #if SCANF 778c2aa98e2SPeter Wemm char wordbuf[MAXLINE + 1]; 77906f25ae9SGregory Neil Shapiro #endif /* SCANF */ 780c2aa98e2SPeter Wemm 781c2aa98e2SPeter Wemm if (buf[0] == '#') 782c2aa98e2SPeter Wemm continue; 783c2aa98e2SPeter Wemm #if SCANF 784c2aa98e2SPeter Wemm if (sscanf(buf, fmt, wordbuf) != 1) 785c2aa98e2SPeter Wemm continue; 786c2aa98e2SPeter Wemm p = wordbuf; 787c2aa98e2SPeter Wemm #else /* SCANF */ 788c2aa98e2SPeter Wemm p = buf; 789c2aa98e2SPeter Wemm #endif /* SCANF */ 790c2aa98e2SPeter Wemm 791c2aa98e2SPeter Wemm /* 792c2aa98e2SPeter Wemm ** Break up the match into words. 793c2aa98e2SPeter Wemm */ 794c2aa98e2SPeter Wemm 795c2aa98e2SPeter Wemm while (*p != '\0') 796c2aa98e2SPeter Wemm { 797c2aa98e2SPeter Wemm register char *q; 798c2aa98e2SPeter Wemm 799c2aa98e2SPeter Wemm /* strip leading spaces */ 800c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 801c2aa98e2SPeter Wemm p++; 802c2aa98e2SPeter Wemm if (*p == '\0') 803c2aa98e2SPeter Wemm break; 804c2aa98e2SPeter Wemm 805c2aa98e2SPeter Wemm /* find the end of the word */ 806c2aa98e2SPeter Wemm q = p; 807c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 808c2aa98e2SPeter Wemm p++; 809c2aa98e2SPeter Wemm if (*p != '\0') 810c2aa98e2SPeter Wemm *p++ = '\0'; 811c2aa98e2SPeter Wemm 812c2aa98e2SPeter Wemm /* enter the word in the symbol table */ 813c2aa98e2SPeter Wemm setclass(class, q); 814c2aa98e2SPeter Wemm } 815c2aa98e2SPeter Wemm } 816c2aa98e2SPeter Wemm 817c2aa98e2SPeter Wemm (void) fclose(f); 818c2aa98e2SPeter Wemm if (pid > 0) 819c2aa98e2SPeter Wemm (void) waitfor(pid); 820c2aa98e2SPeter Wemm } 821c2aa98e2SPeter Wemm /* 822c2aa98e2SPeter Wemm ** MAKEMAILER -- define a new mailer. 823c2aa98e2SPeter Wemm ** 824c2aa98e2SPeter Wemm ** Parameters: 825c2aa98e2SPeter Wemm ** line -- description of mailer. This is in labeled 826c2aa98e2SPeter Wemm ** fields. The fields are: 827c2aa98e2SPeter Wemm ** A -- the argv for this mailer 828c2aa98e2SPeter Wemm ** C -- the character set for MIME conversions 829c2aa98e2SPeter Wemm ** D -- the directory to run in 830c2aa98e2SPeter Wemm ** E -- the eol string 831c2aa98e2SPeter Wemm ** F -- the flags associated with the mailer 832c2aa98e2SPeter Wemm ** L -- the maximum line length 833c2aa98e2SPeter Wemm ** M -- the maximum message size 834c2aa98e2SPeter Wemm ** N -- the niceness at which to run 835c2aa98e2SPeter Wemm ** P -- the path to the mailer 836c2aa98e2SPeter Wemm ** R -- the recipient rewriting set 837c2aa98e2SPeter Wemm ** S -- the sender rewriting set 838c2aa98e2SPeter Wemm ** T -- the mailer type (for DSNs) 839c2aa98e2SPeter Wemm ** U -- the uid to run as 84006f25ae9SGregory Neil Shapiro ** W -- the time to wait at the end 841c2aa98e2SPeter Wemm ** The first word is the canonical name of the mailer. 842c2aa98e2SPeter Wemm ** 843c2aa98e2SPeter Wemm ** Returns: 844c2aa98e2SPeter Wemm ** none. 845c2aa98e2SPeter Wemm ** 846c2aa98e2SPeter Wemm ** Side Effects: 847c2aa98e2SPeter Wemm ** enters the mailer into the mailer table. 848c2aa98e2SPeter Wemm */ 849c2aa98e2SPeter Wemm 850c2aa98e2SPeter Wemm void 851c2aa98e2SPeter Wemm makemailer(line) 852c2aa98e2SPeter Wemm char *line; 853c2aa98e2SPeter Wemm { 854c2aa98e2SPeter Wemm register char *p; 855c2aa98e2SPeter Wemm register struct mailer *m; 856c2aa98e2SPeter Wemm register STAB *s; 857c2aa98e2SPeter Wemm int i; 858c2aa98e2SPeter Wemm char fcode; 859c2aa98e2SPeter Wemm auto char *endp; 860c2aa98e2SPeter Wemm extern int NextMailer; 861c2aa98e2SPeter Wemm 862c2aa98e2SPeter Wemm /* allocate a mailer and set up defaults */ 863c2aa98e2SPeter Wemm m = (struct mailer *) xalloc(sizeof *m); 86406f25ae9SGregory Neil Shapiro memset((char *) m, '\0', sizeof *m); 865c2aa98e2SPeter Wemm 866c2aa98e2SPeter Wemm /* collect the mailer name */ 867c2aa98e2SPeter Wemm for (p = line; *p != '\0' && *p != ',' && !(isascii(*p) && isspace(*p)); p++) 868c2aa98e2SPeter Wemm continue; 869c2aa98e2SPeter Wemm if (*p != '\0') 870c2aa98e2SPeter Wemm *p++ = '\0'; 871c2aa98e2SPeter Wemm if (line[0] == '\0') 872c2aa98e2SPeter Wemm syserr("name required for mailer"); 873c2aa98e2SPeter Wemm m->m_name = newstr(line); 874c2aa98e2SPeter Wemm 875c2aa98e2SPeter Wemm /* now scan through and assign info from the fields */ 876c2aa98e2SPeter Wemm while (*p != '\0') 877c2aa98e2SPeter Wemm { 878c2aa98e2SPeter Wemm auto char *delimptr; 879c2aa98e2SPeter Wemm 880c2aa98e2SPeter Wemm while (*p != '\0' && (*p == ',' || (isascii(*p) && isspace(*p)))) 881c2aa98e2SPeter Wemm p++; 882c2aa98e2SPeter Wemm 883c2aa98e2SPeter Wemm /* p now points to field code */ 884c2aa98e2SPeter Wemm fcode = *p; 885c2aa98e2SPeter Wemm while (*p != '\0' && *p != '=' && *p != ',') 886c2aa98e2SPeter Wemm p++; 887c2aa98e2SPeter Wemm if (*p++ != '=') 888c2aa98e2SPeter Wemm { 889c2aa98e2SPeter Wemm syserr("mailer %s: `=' expected", m->m_name); 890c2aa98e2SPeter Wemm return; 891c2aa98e2SPeter Wemm } 892c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 893c2aa98e2SPeter Wemm p++; 894c2aa98e2SPeter Wemm 895c2aa98e2SPeter Wemm /* p now points to the field body */ 896c2aa98e2SPeter Wemm p = munchstring(p, &delimptr, ','); 897c2aa98e2SPeter Wemm 898c2aa98e2SPeter Wemm /* install the field into the mailer struct */ 899c2aa98e2SPeter Wemm switch (fcode) 900c2aa98e2SPeter Wemm { 901c2aa98e2SPeter Wemm case 'P': /* pathname */ 902c2aa98e2SPeter Wemm if (*p == '\0') 903c2aa98e2SPeter Wemm syserr("mailer %s: empty path name", m->m_name); 904c2aa98e2SPeter Wemm m->m_mailer = newstr(p); 905c2aa98e2SPeter Wemm break; 906c2aa98e2SPeter Wemm 907c2aa98e2SPeter Wemm case 'F': /* flags */ 908c2aa98e2SPeter Wemm for (; *p != '\0'; p++) 909c2aa98e2SPeter Wemm if (!(isascii(*p) && isspace(*p))) 910c2aa98e2SPeter Wemm setbitn(*p, m->m_flags); 911c2aa98e2SPeter Wemm break; 912c2aa98e2SPeter Wemm 913c2aa98e2SPeter Wemm case 'S': /* sender rewriting ruleset */ 914c2aa98e2SPeter Wemm case 'R': /* recipient rewriting ruleset */ 915c2aa98e2SPeter Wemm i = strtorwset(p, &endp, ST_ENTER); 916c2aa98e2SPeter Wemm if (i < 0) 917c2aa98e2SPeter Wemm return; 918c2aa98e2SPeter Wemm if (fcode == 'S') 919c2aa98e2SPeter Wemm m->m_sh_rwset = m->m_se_rwset = i; 920c2aa98e2SPeter Wemm else 921c2aa98e2SPeter Wemm m->m_rh_rwset = m->m_re_rwset = i; 922c2aa98e2SPeter Wemm 923c2aa98e2SPeter Wemm p = endp; 924c2aa98e2SPeter Wemm if (*p++ == '/') 925c2aa98e2SPeter Wemm { 926c2aa98e2SPeter Wemm i = strtorwset(p, NULL, ST_ENTER); 927c2aa98e2SPeter Wemm if (i < 0) 928c2aa98e2SPeter Wemm return; 929c2aa98e2SPeter Wemm if (fcode == 'S') 930c2aa98e2SPeter Wemm m->m_sh_rwset = i; 931c2aa98e2SPeter Wemm else 932c2aa98e2SPeter Wemm m->m_rh_rwset = i; 933c2aa98e2SPeter Wemm } 934c2aa98e2SPeter Wemm break; 935c2aa98e2SPeter Wemm 936c2aa98e2SPeter Wemm case 'E': /* end of line string */ 937c2aa98e2SPeter Wemm if (*p == '\0') 938c2aa98e2SPeter Wemm syserr("mailer %s: null end-of-line string", 939c2aa98e2SPeter Wemm m->m_name); 940c2aa98e2SPeter Wemm m->m_eol = newstr(p); 941c2aa98e2SPeter Wemm break; 942c2aa98e2SPeter Wemm 943c2aa98e2SPeter Wemm case 'A': /* argument vector */ 944c2aa98e2SPeter Wemm if (*p == '\0') 945c2aa98e2SPeter Wemm syserr("mailer %s: null argument vector", 946c2aa98e2SPeter Wemm m->m_name); 947c2aa98e2SPeter Wemm m->m_argv = makeargv(p); 948c2aa98e2SPeter Wemm break; 949c2aa98e2SPeter Wemm 950c2aa98e2SPeter Wemm case 'M': /* maximum message size */ 951c2aa98e2SPeter Wemm m->m_maxsize = atol(p); 952c2aa98e2SPeter Wemm break; 953c2aa98e2SPeter Wemm 95406f25ae9SGregory Neil Shapiro case 'm': /* maximum messages per connection */ 95506f25ae9SGregory Neil Shapiro m->m_maxdeliveries = atoi(p); 95606f25ae9SGregory Neil Shapiro break; 95706f25ae9SGregory Neil Shapiro 95806f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 95906f25ae9SGregory Neil Shapiro case 'r': /* max recipient per envelope */ 96006f25ae9SGregory Neil Shapiro m->m_maxrcpt = atoi(p); 96106f25ae9SGregory Neil Shapiro break; 96206f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 96306f25ae9SGregory Neil Shapiro 964c2aa98e2SPeter Wemm case 'L': /* maximum line length */ 965c2aa98e2SPeter Wemm m->m_linelimit = atoi(p); 966c2aa98e2SPeter Wemm if (m->m_linelimit < 0) 967c2aa98e2SPeter Wemm m->m_linelimit = 0; 968c2aa98e2SPeter Wemm break; 969c2aa98e2SPeter Wemm 970c2aa98e2SPeter Wemm case 'N': /* run niceness */ 971c2aa98e2SPeter Wemm m->m_nice = atoi(p); 972c2aa98e2SPeter Wemm break; 973c2aa98e2SPeter Wemm 974c2aa98e2SPeter Wemm case 'D': /* working directory */ 975c2aa98e2SPeter Wemm if (*p == '\0') 976c2aa98e2SPeter Wemm syserr("mailer %s: null working directory", 977c2aa98e2SPeter Wemm m->m_name); 978c2aa98e2SPeter Wemm m->m_execdir = newstr(p); 979c2aa98e2SPeter Wemm break; 980c2aa98e2SPeter Wemm 981c2aa98e2SPeter Wemm case 'C': /* default charset */ 982c2aa98e2SPeter Wemm if (*p == '\0') 983c2aa98e2SPeter Wemm syserr("mailer %s: null charset", m->m_name); 984c2aa98e2SPeter Wemm m->m_defcharset = newstr(p); 985c2aa98e2SPeter Wemm break; 986c2aa98e2SPeter Wemm 987c2aa98e2SPeter Wemm case 'T': /* MTA-Name/Address/Diagnostic types */ 988c2aa98e2SPeter Wemm /* extract MTA name type; default to "dns" */ 989c2aa98e2SPeter Wemm m->m_mtatype = newstr(p); 990c2aa98e2SPeter Wemm p = strchr(m->m_mtatype, '/'); 991c2aa98e2SPeter Wemm if (p != NULL) 992c2aa98e2SPeter Wemm { 993c2aa98e2SPeter Wemm *p++ = '\0'; 994c2aa98e2SPeter Wemm if (*p == '\0') 995c2aa98e2SPeter Wemm p = NULL; 996c2aa98e2SPeter Wemm } 997c2aa98e2SPeter Wemm if (*m->m_mtatype == '\0') 998c2aa98e2SPeter Wemm m->m_mtatype = "dns"; 999c2aa98e2SPeter Wemm 1000c2aa98e2SPeter Wemm /* extract address type; default to "rfc822" */ 1001c2aa98e2SPeter Wemm m->m_addrtype = p; 1002c2aa98e2SPeter Wemm if (p != NULL) 1003c2aa98e2SPeter Wemm p = strchr(p, '/'); 1004c2aa98e2SPeter Wemm if (p != NULL) 1005c2aa98e2SPeter Wemm { 1006c2aa98e2SPeter Wemm *p++ = '\0'; 1007c2aa98e2SPeter Wemm if (*p == '\0') 1008c2aa98e2SPeter Wemm p = NULL; 1009c2aa98e2SPeter Wemm } 1010c2aa98e2SPeter Wemm if (m->m_addrtype == NULL || *m->m_addrtype == '\0') 1011c2aa98e2SPeter Wemm m->m_addrtype = "rfc822"; 1012c2aa98e2SPeter Wemm 1013c2aa98e2SPeter Wemm /* extract diagnostic type; default to "smtp" */ 1014c2aa98e2SPeter Wemm m->m_diagtype = p; 1015c2aa98e2SPeter Wemm if (m->m_diagtype == NULL || *m->m_diagtype == '\0') 1016c2aa98e2SPeter Wemm m->m_diagtype = "smtp"; 1017c2aa98e2SPeter Wemm break; 1018c2aa98e2SPeter Wemm 1019c2aa98e2SPeter Wemm case 'U': /* user id */ 1020c2aa98e2SPeter Wemm if (isascii(*p) && !isdigit(*p)) 1021c2aa98e2SPeter Wemm { 1022c2aa98e2SPeter Wemm char *q = p; 1023c2aa98e2SPeter Wemm struct passwd *pw; 1024c2aa98e2SPeter Wemm 1025c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && 1026c2aa98e2SPeter Wemm (isalnum(*p) || strchr("-_", *p) != NULL)) 1027c2aa98e2SPeter Wemm p++; 1028c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1029c2aa98e2SPeter Wemm *p++ = '\0'; 1030c2aa98e2SPeter Wemm if (*p != '\0') 1031c2aa98e2SPeter Wemm *p++ = '\0'; 1032c2aa98e2SPeter Wemm if (*q == '\0') 1033c2aa98e2SPeter Wemm syserr("mailer %s: null user name", 1034c2aa98e2SPeter Wemm m->m_name); 1035c2aa98e2SPeter Wemm pw = sm_getpwnam(q); 1036c2aa98e2SPeter Wemm if (pw == NULL) 1037c2aa98e2SPeter Wemm syserr("readcf: mailer U= flag: unknown user %s", q); 1038c2aa98e2SPeter Wemm else 1039c2aa98e2SPeter Wemm { 1040c2aa98e2SPeter Wemm m->m_uid = pw->pw_uid; 1041c2aa98e2SPeter Wemm m->m_gid = pw->pw_gid; 1042c2aa98e2SPeter Wemm } 1043c2aa98e2SPeter Wemm } 1044c2aa98e2SPeter Wemm else 1045c2aa98e2SPeter Wemm { 1046c2aa98e2SPeter Wemm auto char *q; 1047c2aa98e2SPeter Wemm 1048c2aa98e2SPeter Wemm m->m_uid = strtol(p, &q, 0); 1049c2aa98e2SPeter Wemm p = q; 1050c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1051c2aa98e2SPeter Wemm p++; 1052c2aa98e2SPeter Wemm if (*p != '\0') 1053c2aa98e2SPeter Wemm p++; 1054c2aa98e2SPeter Wemm } 1055c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1056c2aa98e2SPeter Wemm p++; 1057c2aa98e2SPeter Wemm if (*p == '\0') 1058c2aa98e2SPeter Wemm break; 1059c2aa98e2SPeter Wemm if (isascii(*p) && !isdigit(*p)) 1060c2aa98e2SPeter Wemm { 1061c2aa98e2SPeter Wemm char *q = p; 1062c2aa98e2SPeter Wemm struct group *gr; 1063c2aa98e2SPeter Wemm 1064c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 1065c2aa98e2SPeter Wemm p++; 1066c2aa98e2SPeter Wemm *p++ = '\0'; 1067c2aa98e2SPeter Wemm if (*q == '\0') 1068c2aa98e2SPeter Wemm syserr("mailer %s: null group name", 1069c2aa98e2SPeter Wemm m->m_name); 1070c2aa98e2SPeter Wemm gr = getgrnam(q); 1071c2aa98e2SPeter Wemm if (gr == NULL) 1072c2aa98e2SPeter Wemm syserr("readcf: mailer U= flag: unknown group %s", q); 1073c2aa98e2SPeter Wemm else 1074c2aa98e2SPeter Wemm m->m_gid = gr->gr_gid; 1075c2aa98e2SPeter Wemm } 1076c2aa98e2SPeter Wemm else 1077c2aa98e2SPeter Wemm { 1078c2aa98e2SPeter Wemm m->m_gid = strtol(p, NULL, 0); 1079c2aa98e2SPeter Wemm } 1080c2aa98e2SPeter Wemm break; 108106f25ae9SGregory Neil Shapiro 108206f25ae9SGregory Neil Shapiro case 'W': /* wait timeout */ 108306f25ae9SGregory Neil Shapiro m->m_wait = convtime(p, 's'); 108406f25ae9SGregory Neil Shapiro break; 108506f25ae9SGregory Neil Shapiro 108606f25ae9SGregory Neil Shapiro case '/': /* new root directory */ 108706f25ae9SGregory Neil Shapiro if (*p == '\0') 108806f25ae9SGregory Neil Shapiro syserr("mailer %s: null root directory", 108906f25ae9SGregory Neil Shapiro m->m_name); 109006f25ae9SGregory Neil Shapiro else 109106f25ae9SGregory Neil Shapiro m->m_rootdir = newstr(p); 109206f25ae9SGregory Neil Shapiro break; 109306f25ae9SGregory Neil Shapiro 109406f25ae9SGregory Neil Shapiro default: 109506f25ae9SGregory Neil Shapiro syserr("M%s: unknown mailer equate %c=", 109606f25ae9SGregory Neil Shapiro m->m_name, fcode); 109706f25ae9SGregory Neil Shapiro break; 1098c2aa98e2SPeter Wemm } 1099c2aa98e2SPeter Wemm 1100c2aa98e2SPeter Wemm p = delimptr; 1101c2aa98e2SPeter Wemm } 1102c2aa98e2SPeter Wemm 1103c2aa98e2SPeter Wemm /* do some rationality checking */ 1104c2aa98e2SPeter Wemm if (m->m_argv == NULL) 1105c2aa98e2SPeter Wemm { 1106c2aa98e2SPeter Wemm syserr("M%s: A= argument required", m->m_name); 1107c2aa98e2SPeter Wemm return; 1108c2aa98e2SPeter Wemm } 1109c2aa98e2SPeter Wemm if (m->m_mailer == NULL) 1110c2aa98e2SPeter Wemm { 1111c2aa98e2SPeter Wemm syserr("M%s: P= argument required", m->m_name); 1112c2aa98e2SPeter Wemm return; 1113c2aa98e2SPeter Wemm } 1114c2aa98e2SPeter Wemm 1115c2aa98e2SPeter Wemm if (NextMailer >= MAXMAILERS) 1116c2aa98e2SPeter Wemm { 1117c2aa98e2SPeter Wemm syserr("too many mailers defined (%d max)", MAXMAILERS); 1118c2aa98e2SPeter Wemm return; 1119c2aa98e2SPeter Wemm } 1120c2aa98e2SPeter Wemm 112106f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 112206f25ae9SGregory Neil Shapiro if (m->m_maxrcpt <= 0) 112306f25ae9SGregory Neil Shapiro m->m_maxrcpt = DEFAULT_MAX_RCPT; 112406f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 112506f25ae9SGregory Neil Shapiro 1126c2aa98e2SPeter Wemm /* do some heuristic cleanup for back compatibility */ 1127c2aa98e2SPeter Wemm if (bitnset(M_LIMITS, m->m_flags)) 1128c2aa98e2SPeter Wemm { 1129c2aa98e2SPeter Wemm if (m->m_linelimit == 0) 1130c2aa98e2SPeter Wemm m->m_linelimit = SMTPLINELIM; 1131c2aa98e2SPeter Wemm if (ConfigLevel < 2) 1132c2aa98e2SPeter Wemm setbitn(M_7BITS, m->m_flags); 1133c2aa98e2SPeter Wemm } 1134c2aa98e2SPeter Wemm 113506f25ae9SGregory Neil Shapiro if (strcmp(m->m_mailer, "[TCP]") == 0) 1136c2aa98e2SPeter Wemm { 113742e5d165SGregory Neil Shapiro #if _FFR_REMOVE_TCP_MAILER_PATH 113806f25ae9SGregory Neil Shapiro syserr("M%s: P=[TCP] is deprecated, use P=[IPC] instead\n", 113906f25ae9SGregory Neil Shapiro m->m_name); 114042e5d165SGregory Neil Shapiro #else /* _FFR_REMOVE_TCP_MAILER_PATH */ 114106f25ae9SGregory Neil Shapiro printf("M%s: Warning: P=[TCP] is deprecated, use P=[IPC] instead\n", 114206f25ae9SGregory Neil Shapiro m->m_name); 114342e5d165SGregory Neil Shapiro #endif /* _FFR_REMOVE_TCP_MAILER_PATH */ 1144c2aa98e2SPeter Wemm } 1145c2aa98e2SPeter Wemm 114642e5d165SGregory Neil Shapiro if (strcmp(m->m_mailer, "[IPC]") == 0 114706f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_TCP_MAILER_PATH 114842e5d165SGregory Neil Shapiro || strcmp(m->m_mailer, "[TCP]") == 0 114906f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_TCP_MAILER_PATH */ 115006f25ae9SGregory Neil Shapiro ) 115106f25ae9SGregory Neil Shapiro { 115206f25ae9SGregory Neil Shapiro /* Use the second argument for host or path to socket */ 115306f25ae9SGregory Neil Shapiro if (m->m_argv[0] == NULL || m->m_argv[1] == NULL || 115406f25ae9SGregory Neil Shapiro m->m_argv[1][0] == '\0') 115506f25ae9SGregory Neil Shapiro { 115606f25ae9SGregory Neil Shapiro syserr("M%s: too few parameters for %s mailer", 115706f25ae9SGregory Neil Shapiro m->m_name, m->m_mailer); 115806f25ae9SGregory Neil Shapiro } 115942e5d165SGregory Neil Shapiro if (strcmp(m->m_argv[0], "TCP") != 0 116006f25ae9SGregory Neil Shapiro #if NETUNIX 116142e5d165SGregory Neil Shapiro && strcmp(m->m_argv[0], "FILE") != 0 116206f25ae9SGregory Neil Shapiro #endif /* NETUNIX */ 116306f25ae9SGregory Neil Shapiro #if !_FFR_DEPRECATE_IPC_MAILER_ARG 116442e5d165SGregory Neil Shapiro && strcmp(m->m_argv[0], "IPC") != 0 116506f25ae9SGregory Neil Shapiro #endif /* !_FFR_DEPRECATE_IPC_MAILER_ARG */ 116606f25ae9SGregory Neil Shapiro ) 116706f25ae9SGregory Neil Shapiro { 116806f25ae9SGregory Neil Shapiro printf("M%s: Warning: first argument in %s mailer must be %s\n", 116906f25ae9SGregory Neil Shapiro m->m_name, m->m_mailer, 117006f25ae9SGregory Neil Shapiro #if NETUNIX 117106f25ae9SGregory Neil Shapiro "TCP or FILE" 117206f25ae9SGregory Neil Shapiro #else /* NETUNIX */ 117306f25ae9SGregory Neil Shapiro "TCP" 117406f25ae9SGregory Neil Shapiro #endif /* NETUNIX */ 117506f25ae9SGregory Neil Shapiro ); 117606f25ae9SGregory Neil Shapiro } 117706f25ae9SGregory Neil Shapiro 117806f25ae9SGregory Neil Shapiro } 117906f25ae9SGregory Neil Shapiro else if (strcmp(m->m_mailer, "[FILE]") == 0) 1180c2aa98e2SPeter Wemm { 1181c2aa98e2SPeter Wemm /* Use the second argument for filename */ 1182c2aa98e2SPeter Wemm if (m->m_argv[0] == NULL || m->m_argv[1] == NULL || 1183c2aa98e2SPeter Wemm m->m_argv[2] != NULL) 1184c2aa98e2SPeter Wemm { 1185c2aa98e2SPeter Wemm syserr("M%s: too %s parameters for [FILE] mailer", 1186c2aa98e2SPeter Wemm m->m_name, 1187c2aa98e2SPeter Wemm (m->m_argv[0] == NULL || 1188c2aa98e2SPeter Wemm m->m_argv[1] == NULL) ? "few" : "many"); 1189c2aa98e2SPeter Wemm } 1190c2aa98e2SPeter Wemm else if (strcmp(m->m_argv[0], "FILE") != 0) 1191c2aa98e2SPeter Wemm { 1192c2aa98e2SPeter Wemm syserr("M%s: first argument in [FILE] mailer must be FILE", 1193c2aa98e2SPeter Wemm m->m_name); 1194c2aa98e2SPeter Wemm } 1195c2aa98e2SPeter Wemm } 1196c2aa98e2SPeter Wemm 119706f25ae9SGregory Neil Shapiro if (strcmp(m->m_mailer, "[IPC]") == 0 || 119806f25ae9SGregory Neil Shapiro strcmp(m->m_mailer, "[TCP]") == 0) 119906f25ae9SGregory Neil Shapiro { 120006f25ae9SGregory Neil Shapiro if (m->m_mtatype == NULL) 120106f25ae9SGregory Neil Shapiro m->m_mtatype = "dns"; 120206f25ae9SGregory Neil Shapiro if (m->m_addrtype == NULL) 120306f25ae9SGregory Neil Shapiro m->m_addrtype = "rfc822"; 120406f25ae9SGregory Neil Shapiro if (m->m_diagtype == NULL) 120506f25ae9SGregory Neil Shapiro { 120606f25ae9SGregory Neil Shapiro if (m->m_argv[0] != NULL && 120706f25ae9SGregory Neil Shapiro strcmp(m->m_argv[0], "FILE") == 0) 120806f25ae9SGregory Neil Shapiro m->m_diagtype = "x-unix"; 120906f25ae9SGregory Neil Shapiro else 121006f25ae9SGregory Neil Shapiro m->m_diagtype = "smtp"; 121106f25ae9SGregory Neil Shapiro } 121206f25ae9SGregory Neil Shapiro } 121306f25ae9SGregory Neil Shapiro 1214c2aa98e2SPeter Wemm if (m->m_eol == NULL) 1215c2aa98e2SPeter Wemm { 1216c2aa98e2SPeter Wemm char **pp; 1217c2aa98e2SPeter Wemm 1218c2aa98e2SPeter Wemm /* default for SMTP is \r\n; use \n for local delivery */ 1219c2aa98e2SPeter Wemm for (pp = m->m_argv; *pp != NULL; pp++) 1220c2aa98e2SPeter Wemm { 1221c2aa98e2SPeter Wemm for (p = *pp; *p != '\0'; ) 1222c2aa98e2SPeter Wemm { 1223c2aa98e2SPeter Wemm if ((*p++ & 0377) == MACROEXPAND && *p == 'u') 1224c2aa98e2SPeter Wemm break; 1225c2aa98e2SPeter Wemm } 1226c2aa98e2SPeter Wemm if (*p != '\0') 1227c2aa98e2SPeter Wemm break; 1228c2aa98e2SPeter Wemm } 1229c2aa98e2SPeter Wemm if (*pp == NULL) 1230c2aa98e2SPeter Wemm m->m_eol = "\r\n"; 1231c2aa98e2SPeter Wemm else 1232c2aa98e2SPeter Wemm m->m_eol = "\n"; 1233c2aa98e2SPeter Wemm } 1234c2aa98e2SPeter Wemm 1235c2aa98e2SPeter Wemm /* enter the mailer into the symbol table */ 1236c2aa98e2SPeter Wemm s = stab(m->m_name, ST_MAILER, ST_ENTER); 1237c2aa98e2SPeter Wemm if (s->s_mailer != NULL) 1238c2aa98e2SPeter Wemm { 1239c2aa98e2SPeter Wemm i = s->s_mailer->m_mno; 1240c2aa98e2SPeter Wemm free(s->s_mailer); 1241c2aa98e2SPeter Wemm } 1242c2aa98e2SPeter Wemm else 1243c2aa98e2SPeter Wemm { 1244c2aa98e2SPeter Wemm i = NextMailer++; 1245c2aa98e2SPeter Wemm } 1246c2aa98e2SPeter Wemm Mailer[i] = s->s_mailer = m; 1247c2aa98e2SPeter Wemm m->m_mno = i; 1248c2aa98e2SPeter Wemm } 1249c2aa98e2SPeter Wemm /* 1250c2aa98e2SPeter Wemm ** MUNCHSTRING -- translate a string into internal form. 1251c2aa98e2SPeter Wemm ** 1252c2aa98e2SPeter Wemm ** Parameters: 1253c2aa98e2SPeter Wemm ** p -- the string to munch. 1254c2aa98e2SPeter Wemm ** delimptr -- if non-NULL, set to the pointer of the 1255c2aa98e2SPeter Wemm ** field delimiter character. 1256c2aa98e2SPeter Wemm ** delim -- the delimiter for the field. 1257c2aa98e2SPeter Wemm ** 1258c2aa98e2SPeter Wemm ** Returns: 1259c2aa98e2SPeter Wemm ** the munched string. 126006f25ae9SGregory Neil Shapiro ** 126106f25ae9SGregory Neil Shapiro ** Side Effects: 126206f25ae9SGregory Neil Shapiro ** the munched string is a local static buffer. 126306f25ae9SGregory Neil Shapiro ** it must be copied before the function is called again. 1264c2aa98e2SPeter Wemm */ 1265c2aa98e2SPeter Wemm 1266c2aa98e2SPeter Wemm char * 1267c2aa98e2SPeter Wemm munchstring(p, delimptr, delim) 1268c2aa98e2SPeter Wemm register char *p; 1269c2aa98e2SPeter Wemm char **delimptr; 1270c2aa98e2SPeter Wemm int delim; 1271c2aa98e2SPeter Wemm { 1272c2aa98e2SPeter Wemm register char *q; 1273c2aa98e2SPeter Wemm bool backslash = FALSE; 1274c2aa98e2SPeter Wemm bool quotemode = FALSE; 1275c2aa98e2SPeter Wemm static char buf[MAXLINE]; 1276c2aa98e2SPeter Wemm 1277c2aa98e2SPeter Wemm for (q = buf; *p != '\0' && q < &buf[sizeof buf - 1]; p++) 1278c2aa98e2SPeter Wemm { 1279c2aa98e2SPeter Wemm if (backslash) 1280c2aa98e2SPeter Wemm { 1281c2aa98e2SPeter Wemm /* everything is roughly literal */ 1282c2aa98e2SPeter Wemm backslash = FALSE; 1283c2aa98e2SPeter Wemm switch (*p) 1284c2aa98e2SPeter Wemm { 1285c2aa98e2SPeter Wemm case 'r': /* carriage return */ 1286c2aa98e2SPeter Wemm *q++ = '\r'; 1287c2aa98e2SPeter Wemm continue; 1288c2aa98e2SPeter Wemm 1289c2aa98e2SPeter Wemm case 'n': /* newline */ 1290c2aa98e2SPeter Wemm *q++ = '\n'; 1291c2aa98e2SPeter Wemm continue; 1292c2aa98e2SPeter Wemm 1293c2aa98e2SPeter Wemm case 'f': /* form feed */ 1294c2aa98e2SPeter Wemm *q++ = '\f'; 1295c2aa98e2SPeter Wemm continue; 1296c2aa98e2SPeter Wemm 1297c2aa98e2SPeter Wemm case 'b': /* backspace */ 1298c2aa98e2SPeter Wemm *q++ = '\b'; 1299c2aa98e2SPeter Wemm continue; 1300c2aa98e2SPeter Wemm } 1301c2aa98e2SPeter Wemm *q++ = *p; 1302c2aa98e2SPeter Wemm } 1303c2aa98e2SPeter Wemm else 1304c2aa98e2SPeter Wemm { 1305c2aa98e2SPeter Wemm if (*p == '\\') 1306c2aa98e2SPeter Wemm backslash = TRUE; 1307c2aa98e2SPeter Wemm else if (*p == '"') 1308c2aa98e2SPeter Wemm quotemode = !quotemode; 1309c2aa98e2SPeter Wemm else if (quotemode || *p != delim) 1310c2aa98e2SPeter Wemm *q++ = *p; 1311c2aa98e2SPeter Wemm else 1312c2aa98e2SPeter Wemm break; 1313c2aa98e2SPeter Wemm } 1314c2aa98e2SPeter Wemm } 1315c2aa98e2SPeter Wemm 1316c2aa98e2SPeter Wemm if (delimptr != NULL) 1317c2aa98e2SPeter Wemm *delimptr = p; 1318c2aa98e2SPeter Wemm *q++ = '\0'; 131906f25ae9SGregory Neil Shapiro return buf; 1320c2aa98e2SPeter Wemm } 1321c2aa98e2SPeter Wemm /* 1322c2aa98e2SPeter Wemm ** MAKEARGV -- break up a string into words 1323c2aa98e2SPeter Wemm ** 1324c2aa98e2SPeter Wemm ** Parameters: 1325c2aa98e2SPeter Wemm ** p -- the string to break up. 1326c2aa98e2SPeter Wemm ** 1327c2aa98e2SPeter Wemm ** Returns: 1328c2aa98e2SPeter Wemm ** a char **argv (dynamically allocated) 1329c2aa98e2SPeter Wemm ** 1330c2aa98e2SPeter Wemm ** Side Effects: 1331c2aa98e2SPeter Wemm ** munges p. 1332c2aa98e2SPeter Wemm */ 1333c2aa98e2SPeter Wemm 133406f25ae9SGregory Neil Shapiro static char ** 1335c2aa98e2SPeter Wemm makeargv(p) 1336c2aa98e2SPeter Wemm register char *p; 1337c2aa98e2SPeter Wemm { 1338c2aa98e2SPeter Wemm char *q; 1339c2aa98e2SPeter Wemm int i; 1340c2aa98e2SPeter Wemm char **avp; 1341c2aa98e2SPeter Wemm char *argv[MAXPV + 1]; 1342c2aa98e2SPeter Wemm 1343c2aa98e2SPeter Wemm /* take apart the words */ 1344c2aa98e2SPeter Wemm i = 0; 1345c2aa98e2SPeter Wemm while (*p != '\0' && i < MAXPV) 1346c2aa98e2SPeter Wemm { 1347c2aa98e2SPeter Wemm q = p; 1348c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 1349c2aa98e2SPeter Wemm p++; 1350c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1351c2aa98e2SPeter Wemm *p++ = '\0'; 1352c2aa98e2SPeter Wemm argv[i++] = newstr(q); 1353c2aa98e2SPeter Wemm } 1354c2aa98e2SPeter Wemm argv[i++] = NULL; 1355c2aa98e2SPeter Wemm 1356c2aa98e2SPeter Wemm /* now make a copy of the argv */ 1357c2aa98e2SPeter Wemm avp = (char **) xalloc(sizeof *avp * i); 135806f25ae9SGregory Neil Shapiro memmove((char *) avp, (char *) argv, sizeof *avp * i); 1359c2aa98e2SPeter Wemm 136006f25ae9SGregory Neil Shapiro return avp; 1361c2aa98e2SPeter Wemm } 1362c2aa98e2SPeter Wemm /* 1363c2aa98e2SPeter Wemm ** PRINTRULES -- print rewrite rules (for debugging) 1364c2aa98e2SPeter Wemm ** 1365c2aa98e2SPeter Wemm ** Parameters: 1366c2aa98e2SPeter Wemm ** none. 1367c2aa98e2SPeter Wemm ** 1368c2aa98e2SPeter Wemm ** Returns: 1369c2aa98e2SPeter Wemm ** none. 1370c2aa98e2SPeter Wemm ** 1371c2aa98e2SPeter Wemm ** Side Effects: 1372c2aa98e2SPeter Wemm ** prints rewrite rules. 1373c2aa98e2SPeter Wemm */ 1374c2aa98e2SPeter Wemm 1375c2aa98e2SPeter Wemm void 1376c2aa98e2SPeter Wemm printrules() 1377c2aa98e2SPeter Wemm { 1378c2aa98e2SPeter Wemm register struct rewrite *rwp; 1379c2aa98e2SPeter Wemm register int ruleset; 1380c2aa98e2SPeter Wemm 1381c2aa98e2SPeter Wemm for (ruleset = 0; ruleset < 10; ruleset++) 1382c2aa98e2SPeter Wemm { 1383c2aa98e2SPeter Wemm if (RewriteRules[ruleset] == NULL) 1384c2aa98e2SPeter Wemm continue; 1385c2aa98e2SPeter Wemm printf("\n----Rule Set %d:", ruleset); 1386c2aa98e2SPeter Wemm 1387c2aa98e2SPeter Wemm for (rwp = RewriteRules[ruleset]; rwp != NULL; rwp = rwp->r_next) 1388c2aa98e2SPeter Wemm { 1389c2aa98e2SPeter Wemm printf("\nLHS:"); 1390c2aa98e2SPeter Wemm printav(rwp->r_lhs); 1391c2aa98e2SPeter Wemm printf("RHS:"); 1392c2aa98e2SPeter Wemm printav(rwp->r_rhs); 1393c2aa98e2SPeter Wemm } 1394c2aa98e2SPeter Wemm } 1395c2aa98e2SPeter Wemm } 1396c2aa98e2SPeter Wemm /* 1397c2aa98e2SPeter Wemm ** PRINTMAILER -- print mailer structure (for debugging) 1398c2aa98e2SPeter Wemm ** 1399c2aa98e2SPeter Wemm ** Parameters: 1400c2aa98e2SPeter Wemm ** m -- the mailer to print 1401c2aa98e2SPeter Wemm ** 1402c2aa98e2SPeter Wemm ** Returns: 1403c2aa98e2SPeter Wemm ** none. 1404c2aa98e2SPeter Wemm */ 1405c2aa98e2SPeter Wemm 1406c2aa98e2SPeter Wemm void 1407c2aa98e2SPeter Wemm printmailer(m) 1408c2aa98e2SPeter Wemm register MAILER *m; 1409c2aa98e2SPeter Wemm { 1410c2aa98e2SPeter Wemm int j; 1411c2aa98e2SPeter Wemm 141206f25ae9SGregory Neil Shapiro printf("mailer %d (%s): P=%s S=", m->m_mno, m->m_name, m->m_mailer); 141306f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_se_rwset] == NULL) 141406f25ae9SGregory Neil Shapiro printf("%d/", m->m_se_rwset); 141506f25ae9SGregory Neil Shapiro else 141606f25ae9SGregory Neil Shapiro printf("%s/", RuleSetNames[m->m_se_rwset]); 141706f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_sh_rwset] == NULL) 141806f25ae9SGregory Neil Shapiro printf("%d R=", m->m_sh_rwset); 141906f25ae9SGregory Neil Shapiro else 142006f25ae9SGregory Neil Shapiro printf("%s R=", RuleSetNames[m->m_sh_rwset]); 142106f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_re_rwset] == NULL) 142206f25ae9SGregory Neil Shapiro printf("%d/", m->m_re_rwset); 142306f25ae9SGregory Neil Shapiro else 142406f25ae9SGregory Neil Shapiro printf("%s/", RuleSetNames[m->m_re_rwset]); 142506f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_rh_rwset] == NULL) 142606f25ae9SGregory Neil Shapiro printf("%d ", m->m_rh_rwset); 142706f25ae9SGregory Neil Shapiro else 142806f25ae9SGregory Neil Shapiro printf("%s ", RuleSetNames[m->m_rh_rwset]); 142906f25ae9SGregory Neil Shapiro printf("M=%ld U=%d:%d F=", m->m_maxsize, 1430c2aa98e2SPeter Wemm (int) m->m_uid, (int) m->m_gid); 1431c2aa98e2SPeter Wemm for (j = '\0'; j <= '\177'; j++) 1432c2aa98e2SPeter Wemm if (bitnset(j, m->m_flags)) 1433c2aa98e2SPeter Wemm (void) putchar(j); 1434c2aa98e2SPeter Wemm printf(" L=%d E=", m->m_linelimit); 1435c2aa98e2SPeter Wemm xputs(m->m_eol); 1436c2aa98e2SPeter Wemm if (m->m_defcharset != NULL) 1437c2aa98e2SPeter Wemm printf(" C=%s", m->m_defcharset); 1438c2aa98e2SPeter Wemm printf(" T=%s/%s/%s", 1439c2aa98e2SPeter Wemm m->m_mtatype == NULL ? "<undefined>" : m->m_mtatype, 1440c2aa98e2SPeter Wemm m->m_addrtype == NULL ? "<undefined>" : m->m_addrtype, 1441c2aa98e2SPeter Wemm m->m_diagtype == NULL ? "<undefined>" : m->m_diagtype); 144206f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 144306f25ae9SGregory Neil Shapiro printf(" r=%d", m->m_maxrcpt); 144406f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 1445c2aa98e2SPeter Wemm if (m->m_argv != NULL) 1446c2aa98e2SPeter Wemm { 1447c2aa98e2SPeter Wemm char **a = m->m_argv; 1448c2aa98e2SPeter Wemm 1449c2aa98e2SPeter Wemm printf(" A="); 1450c2aa98e2SPeter Wemm while (*a != NULL) 1451c2aa98e2SPeter Wemm { 1452c2aa98e2SPeter Wemm if (a != m->m_argv) 1453c2aa98e2SPeter Wemm printf(" "); 1454c2aa98e2SPeter Wemm xputs(*a++); 1455c2aa98e2SPeter Wemm } 1456c2aa98e2SPeter Wemm } 1457c2aa98e2SPeter Wemm printf("\n"); 1458c2aa98e2SPeter Wemm } 1459c2aa98e2SPeter Wemm /* 1460c2aa98e2SPeter Wemm ** SETOPTION -- set global processing option 1461c2aa98e2SPeter Wemm ** 1462c2aa98e2SPeter Wemm ** Parameters: 1463c2aa98e2SPeter Wemm ** opt -- option name. 1464c2aa98e2SPeter Wemm ** val -- option value (as a text string). 1465c2aa98e2SPeter Wemm ** safe -- set if this came from a configuration file. 1466c2aa98e2SPeter Wemm ** Some options (if set from the command line) will 1467c2aa98e2SPeter Wemm ** reset the user id to avoid security problems. 1468c2aa98e2SPeter Wemm ** sticky -- if set, don't let other setoptions override 1469c2aa98e2SPeter Wemm ** this value. 1470c2aa98e2SPeter Wemm ** e -- the main envelope. 1471c2aa98e2SPeter Wemm ** 1472c2aa98e2SPeter Wemm ** Returns: 1473c2aa98e2SPeter Wemm ** none. 1474c2aa98e2SPeter Wemm ** 1475c2aa98e2SPeter Wemm ** Side Effects: 1476c2aa98e2SPeter Wemm ** Sets options as implied by the arguments. 1477c2aa98e2SPeter Wemm */ 1478c2aa98e2SPeter Wemm 147906f25ae9SGregory Neil Shapiro static BITMAP256 StickyOpt; /* set if option is stuck */ 1480c2aa98e2SPeter Wemm 1481c2aa98e2SPeter Wemm #if NAMED_BIND 1482c2aa98e2SPeter Wemm 148306f25ae9SGregory Neil Shapiro static struct resolverflags 1484c2aa98e2SPeter Wemm { 1485c2aa98e2SPeter Wemm char *rf_name; /* name of the flag */ 1486c2aa98e2SPeter Wemm long rf_bits; /* bits to set/clear */ 1487c2aa98e2SPeter Wemm } ResolverFlags[] = 1488c2aa98e2SPeter Wemm { 1489c2aa98e2SPeter Wemm { "debug", RES_DEBUG }, 1490c2aa98e2SPeter Wemm { "aaonly", RES_AAONLY }, 1491c2aa98e2SPeter Wemm { "usevc", RES_USEVC }, 1492c2aa98e2SPeter Wemm { "primary", RES_PRIMARY }, 1493c2aa98e2SPeter Wemm { "igntc", RES_IGNTC }, 1494c2aa98e2SPeter Wemm { "recurse", RES_RECURSE }, 1495c2aa98e2SPeter Wemm { "defnames", RES_DEFNAMES }, 1496c2aa98e2SPeter Wemm { "stayopen", RES_STAYOPEN }, 1497c2aa98e2SPeter Wemm { "dnsrch", RES_DNSRCH }, 1498c2aa98e2SPeter Wemm { "true", 0 }, /* avoid error on old syntax */ 1499c2aa98e2SPeter Wemm { NULL, 0 } 1500c2aa98e2SPeter Wemm }; 1501c2aa98e2SPeter Wemm 150206f25ae9SGregory Neil Shapiro #endif /* NAMED_BIND */ 1503c2aa98e2SPeter Wemm 150406f25ae9SGregory Neil Shapiro #define OI_NONE 0 /* no special treatment */ 150506f25ae9SGregory Neil Shapiro #define OI_SAFE 0x0001 /* safe for random people to use */ 150606f25ae9SGregory Neil Shapiro #define OI_SUBOPT 0x0002 /* option has suboptions */ 150706f25ae9SGregory Neil Shapiro 150806f25ae9SGregory Neil Shapiro static struct optioninfo 1509c2aa98e2SPeter Wemm { 1510c2aa98e2SPeter Wemm char *o_name; /* long name of option */ 1511c2aa98e2SPeter Wemm u_char o_code; /* short name of option */ 151206f25ae9SGregory Neil Shapiro u_short o_flags; /* option flags */ 1513c2aa98e2SPeter Wemm } OptionTab[] = 1514c2aa98e2SPeter Wemm { 151506f25ae9SGregory Neil Shapiro #if defined(SUN_EXTENSIONS) && defined(REMOTE_MODE) 151606f25ae9SGregory Neil Shapiro { "RemoteMode", '>', OI_NONE }, 151706f25ae9SGregory Neil Shapiro #endif /* defined(SUN_EXTENSIONS) && defined(REMOTE_MODE) */ 151806f25ae9SGregory Neil Shapiro { "SevenBitInput", '7', OI_SAFE }, 1519c2aa98e2SPeter Wemm #if MIME8TO7 152006f25ae9SGregory Neil Shapiro { "EightBitMode", '8', OI_SAFE }, 152106f25ae9SGregory Neil Shapiro #endif /* MIME8TO7 */ 152206f25ae9SGregory Neil Shapiro { "AliasFile", 'A', OI_NONE }, 152306f25ae9SGregory Neil Shapiro { "AliasWait", 'a', OI_NONE }, 152406f25ae9SGregory Neil Shapiro { "BlankSub", 'B', OI_NONE }, 152506f25ae9SGregory Neil Shapiro { "MinFreeBlocks", 'b', OI_SAFE }, 152606f25ae9SGregory Neil Shapiro { "CheckpointInterval", 'C', OI_SAFE }, 152706f25ae9SGregory Neil Shapiro { "HoldExpensive", 'c', OI_NONE }, 152806f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_AUTOREBUILD 152906f25ae9SGregory Neil Shapiro { "AutoRebuildAliases", 'D', OI_NONE }, 153006f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_AUTOREBUILD */ 153106f25ae9SGregory Neil Shapiro { "DeliveryMode", 'd', OI_SAFE }, 153206f25ae9SGregory Neil Shapiro { "ErrorHeader", 'E', OI_NONE }, 153306f25ae9SGregory Neil Shapiro { "ErrorMode", 'e', OI_SAFE }, 153406f25ae9SGregory Neil Shapiro { "TempFileMode", 'F', OI_NONE }, 153506f25ae9SGregory Neil Shapiro { "SaveFromLine", 'f', OI_NONE }, 153606f25ae9SGregory Neil Shapiro { "MatchGECOS", 'G', OI_NONE }, 153706f25ae9SGregory Neil Shapiro { "HelpFile", 'H', OI_NONE }, 153806f25ae9SGregory Neil Shapiro { "MaxHopCount", 'h', OI_NONE }, 153906f25ae9SGregory Neil Shapiro { "ResolverOptions", 'I', OI_NONE }, 154006f25ae9SGregory Neil Shapiro { "IgnoreDots", 'i', OI_SAFE }, 154106f25ae9SGregory Neil Shapiro { "ForwardPath", 'J', OI_NONE }, 154206f25ae9SGregory Neil Shapiro { "SendMimeErrors", 'j', OI_SAFE }, 154306f25ae9SGregory Neil Shapiro { "ConnectionCacheSize", 'k', OI_NONE }, 154406f25ae9SGregory Neil Shapiro { "ConnectionCacheTimeout", 'K', OI_NONE }, 154506f25ae9SGregory Neil Shapiro { "UseErrorsTo", 'l', OI_NONE }, 154606f25ae9SGregory Neil Shapiro { "LogLevel", 'L', OI_SAFE }, 154706f25ae9SGregory Neil Shapiro { "MeToo", 'm', OI_SAFE }, 154806f25ae9SGregory Neil Shapiro { "CheckAliases", 'n', OI_NONE }, 154906f25ae9SGregory Neil Shapiro { "OldStyleHeaders", 'o', OI_SAFE }, 155006f25ae9SGregory Neil Shapiro { "DaemonPortOptions", 'O', OI_NONE }, 155106f25ae9SGregory Neil Shapiro { "PrivacyOptions", 'p', OI_SAFE }, 155206f25ae9SGregory Neil Shapiro { "PostmasterCopy", 'P', OI_NONE }, 155306f25ae9SGregory Neil Shapiro { "QueueFactor", 'q', OI_NONE }, 155406f25ae9SGregory Neil Shapiro { "QueueDirectory", 'Q', OI_NONE }, 155506f25ae9SGregory Neil Shapiro { "DontPruneRoutes", 'R', OI_NONE }, 155606f25ae9SGregory Neil Shapiro { "Timeout", 'r', OI_SUBOPT }, 155706f25ae9SGregory Neil Shapiro { "StatusFile", 'S', OI_NONE }, 155806f25ae9SGregory Neil Shapiro { "SuperSafe", 's', OI_SAFE }, 155906f25ae9SGregory Neil Shapiro { "QueueTimeout", 'T', OI_NONE }, 156006f25ae9SGregory Neil Shapiro { "TimeZoneSpec", 't', OI_NONE }, 156106f25ae9SGregory Neil Shapiro { "UserDatabaseSpec", 'U', OI_NONE }, 156206f25ae9SGregory Neil Shapiro { "DefaultUser", 'u', OI_NONE }, 156306f25ae9SGregory Neil Shapiro { "FallbackMXhost", 'V', OI_NONE }, 156406f25ae9SGregory Neil Shapiro { "Verbose", 'v', OI_SAFE }, 156506f25ae9SGregory Neil Shapiro { "TryNullMXList", 'w', OI_NONE }, 156606f25ae9SGregory Neil Shapiro { "QueueLA", 'x', OI_NONE }, 156706f25ae9SGregory Neil Shapiro { "RefuseLA", 'X', OI_NONE }, 156806f25ae9SGregory Neil Shapiro { "RecipientFactor", 'y', OI_NONE }, 156906f25ae9SGregory Neil Shapiro { "ForkEachJob", 'Y', OI_NONE }, 157006f25ae9SGregory Neil Shapiro { "ClassFactor", 'z', OI_NONE }, 157106f25ae9SGregory Neil Shapiro { "RetryFactor", 'Z', OI_NONE }, 1572c2aa98e2SPeter Wemm #define O_QUEUESORTORD 0x81 157306f25ae9SGregory Neil Shapiro { "QueueSortOrder", O_QUEUESORTORD, OI_SAFE }, 1574c2aa98e2SPeter Wemm #define O_HOSTSFILE 0x82 157506f25ae9SGregory Neil Shapiro { "HostsFile", O_HOSTSFILE, OI_NONE }, 1576c2aa98e2SPeter Wemm #define O_MQA 0x83 157706f25ae9SGregory Neil Shapiro { "MinQueueAge", O_MQA, OI_SAFE }, 1578c2aa98e2SPeter Wemm #define O_DEFCHARSET 0x85 157906f25ae9SGregory Neil Shapiro { "DefaultCharSet", O_DEFCHARSET, OI_SAFE }, 1580c2aa98e2SPeter Wemm #define O_SSFILE 0x86 158106f25ae9SGregory Neil Shapiro { "ServiceSwitchFile", O_SSFILE, OI_NONE }, 1582c2aa98e2SPeter Wemm #define O_DIALDELAY 0x87 158306f25ae9SGregory Neil Shapiro { "DialDelay", O_DIALDELAY, OI_SAFE }, 1584c2aa98e2SPeter Wemm #define O_NORCPTACTION 0x88 158506f25ae9SGregory Neil Shapiro { "NoRecipientAction", O_NORCPTACTION, OI_SAFE }, 1586c2aa98e2SPeter Wemm #define O_SAFEFILEENV 0x89 158706f25ae9SGregory Neil Shapiro { "SafeFileEnvironment", O_SAFEFILEENV, OI_NONE }, 1588c2aa98e2SPeter Wemm #define O_MAXMSGSIZE 0x8a 158906f25ae9SGregory Neil Shapiro { "MaxMessageSize", O_MAXMSGSIZE, OI_NONE }, 1590c2aa98e2SPeter Wemm #define O_COLONOKINADDR 0x8b 159106f25ae9SGregory Neil Shapiro { "ColonOkInAddr", O_COLONOKINADDR, OI_SAFE }, 1592c2aa98e2SPeter Wemm #define O_MAXQUEUERUN 0x8c 159306f25ae9SGregory Neil Shapiro { "MaxQueueRunSize", O_MAXQUEUERUN, OI_SAFE }, 1594c2aa98e2SPeter Wemm #define O_MAXCHILDREN 0x8d 159506f25ae9SGregory Neil Shapiro { "MaxDaemonChildren", O_MAXCHILDREN, OI_NONE }, 1596c2aa98e2SPeter Wemm #define O_KEEPCNAMES 0x8e 159706f25ae9SGregory Neil Shapiro { "DontExpandCnames", O_KEEPCNAMES, OI_NONE }, 1598c2aa98e2SPeter Wemm #define O_MUSTQUOTE 0x8f 159906f25ae9SGregory Neil Shapiro { "MustQuoteChars", O_MUSTQUOTE, OI_NONE }, 1600c2aa98e2SPeter Wemm #define O_SMTPGREETING 0x90 160106f25ae9SGregory Neil Shapiro { "SmtpGreetingMessage", O_SMTPGREETING, OI_NONE }, 1602c2aa98e2SPeter Wemm #define O_UNIXFROM 0x91 160306f25ae9SGregory Neil Shapiro { "UnixFromLine", O_UNIXFROM, OI_NONE }, 1604c2aa98e2SPeter Wemm #define O_OPCHARS 0x92 160506f25ae9SGregory Neil Shapiro { "OperatorChars", O_OPCHARS, OI_NONE }, 1606c2aa98e2SPeter Wemm #define O_DONTINITGRPS 0x93 160706f25ae9SGregory Neil Shapiro { "DontInitGroups", O_DONTINITGRPS, OI_NONE }, 1608c2aa98e2SPeter Wemm #define O_SLFH 0x94 160906f25ae9SGregory Neil Shapiro { "SingleLineFromHeader", O_SLFH, OI_SAFE }, 1610c2aa98e2SPeter Wemm #define O_ABH 0x95 161106f25ae9SGregory Neil Shapiro { "AllowBogusHELO", O_ABH, OI_SAFE }, 1612c2aa98e2SPeter Wemm #define O_CONNTHROT 0x97 161306f25ae9SGregory Neil Shapiro { "ConnectionRateThrottle", O_CONNTHROT, OI_NONE }, 1614c2aa98e2SPeter Wemm #define O_UGW 0x99 161506f25ae9SGregory Neil Shapiro { "UnsafeGroupWrites", O_UGW, OI_NONE }, 1616c2aa98e2SPeter Wemm #define O_DBLBOUNCE 0x9a 161706f25ae9SGregory Neil Shapiro { "DoubleBounceAddress", O_DBLBOUNCE, OI_NONE }, 1618c2aa98e2SPeter Wemm #define O_HSDIR 0x9b 161906f25ae9SGregory Neil Shapiro { "HostStatusDirectory", O_HSDIR, OI_NONE }, 1620c2aa98e2SPeter Wemm #define O_SINGTHREAD 0x9c 162106f25ae9SGregory Neil Shapiro { "SingleThreadDelivery", O_SINGTHREAD, OI_NONE }, 1622c2aa98e2SPeter Wemm #define O_RUNASUSER 0x9d 162306f25ae9SGregory Neil Shapiro { "RunAsUser", O_RUNASUSER, OI_NONE }, 1624c2aa98e2SPeter Wemm #define O_DSN_RRT 0x9e 162506f25ae9SGregory Neil Shapiro { "RrtImpliesDsn", O_DSN_RRT, OI_NONE }, 1626c2aa98e2SPeter Wemm #define O_PIDFILE 0x9f 162706f25ae9SGregory Neil Shapiro { "PidFile", O_PIDFILE, OI_NONE }, 1628c2aa98e2SPeter Wemm #define O_DONTBLAMESENDMAIL 0xa0 162906f25ae9SGregory Neil Shapiro { "DontBlameSendmail", O_DONTBLAMESENDMAIL, OI_NONE }, 1630c2aa98e2SPeter Wemm #define O_DPI 0xa1 163106f25ae9SGregory Neil Shapiro { "DontProbeInterfaces", O_DPI, OI_NONE }, 1632c2aa98e2SPeter Wemm #define O_MAXRCPT 0xa2 163306f25ae9SGregory Neil Shapiro { "MaxRecipientsPerMessage", O_MAXRCPT, OI_SAFE }, 1634c2aa98e2SPeter Wemm #define O_DEADLETTER 0xa3 163506f25ae9SGregory Neil Shapiro { "DeadLetterDrop", O_DEADLETTER, OI_NONE }, 1636c2aa98e2SPeter Wemm #if _FFR_DONTLOCKFILESFORREAD_OPTION 1637c2aa98e2SPeter Wemm # define O_DONTLOCK 0xa4 163806f25ae9SGregory Neil Shapiro { "DontLockFilesForRead", O_DONTLOCK, OI_NONE }, 163906f25ae9SGregory Neil Shapiro #endif /* _FFR_DONTLOCKFILESFORREAD_OPTION */ 1640c2aa98e2SPeter Wemm #define O_MAXALIASRCSN 0xa5 164106f25ae9SGregory Neil Shapiro { "MaxAliasRecursion", O_MAXALIASRCSN, OI_NONE }, 1642c2aa98e2SPeter Wemm #define O_CNCTONLYTO 0xa6 164306f25ae9SGregory Neil Shapiro { "ConnectOnlyTo", O_CNCTONLYTO, OI_NONE }, 1644065a643dSPeter Wemm #define O_TRUSTUSER 0xa7 164506f25ae9SGregory Neil Shapiro { "TrustedUser", O_TRUSTUSER, OI_NONE }, 1646065a643dSPeter Wemm #define O_MAXMIMEHDRLEN 0xa8 164706f25ae9SGregory Neil Shapiro { "MaxMimeHeaderLength", O_MAXMIMEHDRLEN, OI_NONE }, 1648065a643dSPeter Wemm #define O_CONTROLSOCKET 0xa9 164906f25ae9SGregory Neil Shapiro { "ControlSocketName", O_CONTROLSOCKET, OI_NONE }, 16502e43090eSPeter Wemm #define O_MAXHDRSLEN 0xaa 165106f25ae9SGregory Neil Shapiro { "MaxHeadersLength", O_MAXHDRSLEN, OI_NONE }, 165206f25ae9SGregory Neil Shapiro #if _FFR_MAX_FORWARD_ENTRIES 165306f25ae9SGregory Neil Shapiro # define O_MAXFORWARD 0xab 165406f25ae9SGregory Neil Shapiro { "MaxForwardEntries", O_MAXFORWARD, OI_NONE }, 165506f25ae9SGregory Neil Shapiro #endif /* _FFR_MAX_FORWARD_ENTRIES */ 165606f25ae9SGregory Neil Shapiro #define O_PROCTITLEPREFIX 0xac 165706f25ae9SGregory Neil Shapiro { "ProcessTitlePrefix", O_PROCTITLEPREFIX, OI_NONE }, 165806f25ae9SGregory Neil Shapiro #define O_SASLINFO 0xad 165906f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 166006f25ae9SGregory Neil Shapiro { "DefaultAuthInfo", O_SASLINFO, OI_SAFE }, 166106f25ae9SGregory Neil Shapiro #else /* _FFR_ALLOW_SASLINFO */ 166206f25ae9SGregory Neil Shapiro { "DefaultAuthInfo", O_SASLINFO, OI_NONE }, 166306f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 166406f25ae9SGregory Neil Shapiro #define O_SASLMECH 0xae 166506f25ae9SGregory Neil Shapiro { "AuthMechanisms", O_SASLMECH, OI_NONE }, 166606f25ae9SGregory Neil Shapiro #define O_CLIENTPORT 0xaf 166706f25ae9SGregory Neil Shapiro { "ClientPortOptions", O_CLIENTPORT, OI_NONE }, 166806f25ae9SGregory Neil Shapiro #define O_DF_BUFSIZE 0xb0 166906f25ae9SGregory Neil Shapiro { "DataFileBufferSize", O_DF_BUFSIZE, OI_NONE }, 167006f25ae9SGregory Neil Shapiro #define O_XF_BUFSIZE 0xb1 167106f25ae9SGregory Neil Shapiro { "XscriptFileBufferSize", O_XF_BUFSIZE, OI_NONE }, 167206f25ae9SGregory Neil Shapiro # define O_LDAPDEFAULTSPEC 0xb2 167306f25ae9SGregory Neil Shapiro { "LDAPDefaultSpec", O_LDAPDEFAULTSPEC, OI_NONE }, 167406f25ae9SGregory Neil Shapiro #if _FFR_QUEUEDELAY 167506f25ae9SGregory Neil Shapiro #define O_QUEUEDELAY 0xb3 167606f25ae9SGregory Neil Shapiro { "QueueDelay", O_QUEUEDELAY, OI_NONE }, 167706f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUEDELAY */ 167806f25ae9SGregory Neil Shapiro # define O_SRVCERTFILE 0xb4 167906f25ae9SGregory Neil Shapiro { "ServerCertFile", O_SRVCERTFILE, OI_NONE }, 168006f25ae9SGregory Neil Shapiro # define O_SRVKEYFILE 0xb5 168106f25ae9SGregory Neil Shapiro { "Serverkeyfile", O_SRVKEYFILE, OI_NONE }, 168206f25ae9SGregory Neil Shapiro # define O_CLTCERTFILE 0xb6 168306f25ae9SGregory Neil Shapiro { "ClientCertFile", O_CLTCERTFILE, OI_NONE }, 168406f25ae9SGregory Neil Shapiro # define O_CLTKEYFILE 0xb7 168506f25ae9SGregory Neil Shapiro { "Clientkeyfile", O_CLTKEYFILE, OI_NONE }, 168606f25ae9SGregory Neil Shapiro # define O_CACERTFILE 0xb8 168706f25ae9SGregory Neil Shapiro { "CACERTFile", O_CACERTFILE, OI_NONE }, 168806f25ae9SGregory Neil Shapiro # define O_CACERTPATH 0xb9 168906f25ae9SGregory Neil Shapiro { "CACERTPath", O_CACERTPATH, OI_NONE }, 169006f25ae9SGregory Neil Shapiro # define O_DHPARAMS 0xba 169106f25ae9SGregory Neil Shapiro { "DHParameters", O_DHPARAMS, OI_NONE }, 169206f25ae9SGregory Neil Shapiro #if _FFR_MILTER 169306f25ae9SGregory Neil Shapiro #define O_INPUTMILTER 0xbb 169406f25ae9SGregory Neil Shapiro { "InputMailFilters", O_INPUTMILTER, OI_NONE }, 169506f25ae9SGregory Neil Shapiro #define O_MILTER 0xbc 169606f25ae9SGregory Neil Shapiro { "Milter", O_MILTER, OI_SUBOPT }, 169706f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 169806f25ae9SGregory Neil Shapiro #define O_SASLOPTS 0xbd 169906f25ae9SGregory Neil Shapiro { "AuthOptions", O_SASLOPTS, OI_NONE }, 170006f25ae9SGregory Neil Shapiro #if _FFR_QUEUE_FILE_MODE 170106f25ae9SGregory Neil Shapiro #define O_QUEUE_FILE_MODE 0xbe 170206f25ae9SGregory Neil Shapiro { "QueueFileMode", O_QUEUE_FILE_MODE, OI_NONE }, 170306f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUE_FILE_MODE */ 170406f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 170506f25ae9SGregory Neil Shapiro # define O_DHPARAMS5 0xbf 170606f25ae9SGregory Neil Shapiro { "DHParameters512", O_DHPARAMS5, OI_NONE }, 170706f25ae9SGregory Neil Shapiro # define O_CIPHERLIST 0xc0 170806f25ae9SGregory Neil Shapiro { "CipherList", O_CIPHERLIST, OI_NONE }, 170906f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 171006f25ae9SGregory Neil Shapiro # define O_RANDFILE 0xc1 171106f25ae9SGregory Neil Shapiro { "RandFile", O_RANDFILE, OI_NONE }, 171206f25ae9SGregory Neil Shapiro { NULL, '\0', OI_NONE } 1713c2aa98e2SPeter Wemm }; 1714c2aa98e2SPeter Wemm 1715c2aa98e2SPeter Wemm void 1716c2aa98e2SPeter Wemm setoption(opt, val, safe, sticky, e) 1717c2aa98e2SPeter Wemm int opt; 1718c2aa98e2SPeter Wemm char *val; 1719c2aa98e2SPeter Wemm bool safe; 1720c2aa98e2SPeter Wemm bool sticky; 1721c2aa98e2SPeter Wemm register ENVELOPE *e; 1722c2aa98e2SPeter Wemm { 1723c2aa98e2SPeter Wemm register char *p; 1724c2aa98e2SPeter Wemm register struct optioninfo *o; 1725c2aa98e2SPeter Wemm char *subopt; 1726c2aa98e2SPeter Wemm int mid; 1727c2aa98e2SPeter Wemm bool can_setuid = RunAsUid == 0; 1728c2aa98e2SPeter Wemm auto char *ep; 1729c2aa98e2SPeter Wemm char buf[50]; 1730c2aa98e2SPeter Wemm extern bool Warn_Q_option; 173106f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 173206f25ae9SGregory Neil Shapiro extern int SubmitMode; 173306f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 1734c2aa98e2SPeter Wemm 1735c2aa98e2SPeter Wemm errno = 0; 1736c2aa98e2SPeter Wemm if (opt == ' ') 1737c2aa98e2SPeter Wemm { 1738c2aa98e2SPeter Wemm /* full word options */ 1739c2aa98e2SPeter Wemm struct optioninfo *sel; 1740c2aa98e2SPeter Wemm 1741c2aa98e2SPeter Wemm p = strchr(val, '='); 1742c2aa98e2SPeter Wemm if (p == NULL) 1743c2aa98e2SPeter Wemm p = &val[strlen(val)]; 1744c2aa98e2SPeter Wemm while (*--p == ' ') 1745c2aa98e2SPeter Wemm continue; 1746c2aa98e2SPeter Wemm while (*++p == ' ') 1747c2aa98e2SPeter Wemm *p = '\0'; 1748c2aa98e2SPeter Wemm if (p == val) 1749c2aa98e2SPeter Wemm { 1750c2aa98e2SPeter Wemm syserr("readcf: null option name"); 1751c2aa98e2SPeter Wemm return; 1752c2aa98e2SPeter Wemm } 1753c2aa98e2SPeter Wemm if (*p == '=') 1754c2aa98e2SPeter Wemm *p++ = '\0'; 1755c2aa98e2SPeter Wemm while (*p == ' ') 1756c2aa98e2SPeter Wemm p++; 1757c2aa98e2SPeter Wemm subopt = strchr(val, '.'); 1758c2aa98e2SPeter Wemm if (subopt != NULL) 1759c2aa98e2SPeter Wemm *subopt++ = '\0'; 1760c2aa98e2SPeter Wemm sel = NULL; 1761c2aa98e2SPeter Wemm for (o = OptionTab; o->o_name != NULL; o++) 1762c2aa98e2SPeter Wemm { 1763c2aa98e2SPeter Wemm if (strncasecmp(o->o_name, val, strlen(val)) != 0) 1764c2aa98e2SPeter Wemm continue; 1765c2aa98e2SPeter Wemm if (strlen(o->o_name) == strlen(val)) 1766c2aa98e2SPeter Wemm { 1767c2aa98e2SPeter Wemm /* completely specified -- this must be it */ 1768c2aa98e2SPeter Wemm sel = NULL; 1769c2aa98e2SPeter Wemm break; 1770c2aa98e2SPeter Wemm } 1771c2aa98e2SPeter Wemm if (sel != NULL) 1772c2aa98e2SPeter Wemm break; 1773c2aa98e2SPeter Wemm sel = o; 1774c2aa98e2SPeter Wemm } 1775c2aa98e2SPeter Wemm if (sel != NULL && o->o_name == NULL) 1776c2aa98e2SPeter Wemm o = sel; 1777c2aa98e2SPeter Wemm else if (o->o_name == NULL) 1778c2aa98e2SPeter Wemm { 1779c2aa98e2SPeter Wemm syserr("readcf: unknown option name %s", val); 1780c2aa98e2SPeter Wemm return; 1781c2aa98e2SPeter Wemm } 1782c2aa98e2SPeter Wemm else if (sel != NULL) 1783c2aa98e2SPeter Wemm { 1784c2aa98e2SPeter Wemm syserr("readcf: ambiguous option name %s (matches %s and %s)", 1785c2aa98e2SPeter Wemm val, sel->o_name, o->o_name); 1786c2aa98e2SPeter Wemm return; 1787c2aa98e2SPeter Wemm } 1788c2aa98e2SPeter Wemm if (strlen(val) != strlen(o->o_name)) 1789c2aa98e2SPeter Wemm { 1790c2aa98e2SPeter Wemm int oldVerbose = Verbose; 1791c2aa98e2SPeter Wemm 1792c2aa98e2SPeter Wemm Verbose = 1; 1793c2aa98e2SPeter Wemm message("Option %s used as abbreviation for %s", 1794c2aa98e2SPeter Wemm val, o->o_name); 1795c2aa98e2SPeter Wemm Verbose = oldVerbose; 1796c2aa98e2SPeter Wemm } 1797c2aa98e2SPeter Wemm opt = o->o_code; 1798c2aa98e2SPeter Wemm val = p; 1799c2aa98e2SPeter Wemm } 1800c2aa98e2SPeter Wemm else 1801c2aa98e2SPeter Wemm { 1802c2aa98e2SPeter Wemm for (o = OptionTab; o->o_name != NULL; o++) 1803c2aa98e2SPeter Wemm { 1804c2aa98e2SPeter Wemm if (o->o_code == opt) 1805c2aa98e2SPeter Wemm break; 1806c2aa98e2SPeter Wemm } 1807c2aa98e2SPeter Wemm subopt = NULL; 1808c2aa98e2SPeter Wemm } 1809c2aa98e2SPeter Wemm 181006f25ae9SGregory Neil Shapiro if (subopt != NULL && !bitset(OI_SUBOPT, o->o_flags)) 181106f25ae9SGregory Neil Shapiro { 181206f25ae9SGregory Neil Shapiro if (tTd(37, 1)) 181306f25ae9SGregory Neil Shapiro dprintf("setoption: %s does not support suboptions, ignoring .%s\n", 181406f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name, 181506f25ae9SGregory Neil Shapiro subopt); 181606f25ae9SGregory Neil Shapiro subopt = NULL; 181706f25ae9SGregory Neil Shapiro } 181806f25ae9SGregory Neil Shapiro 1819c2aa98e2SPeter Wemm if (tTd(37, 1)) 1820c2aa98e2SPeter Wemm { 182106f25ae9SGregory Neil Shapiro dprintf(isascii(opt) && isprint(opt) ? 182206f25ae9SGregory Neil Shapiro "setoption %s (%c)%s%s=" : 182306f25ae9SGregory Neil Shapiro "setoption %s (0x%x)%s%s=", 1824c2aa98e2SPeter Wemm o->o_name == NULL ? "<unknown>" : o->o_name, 1825c2aa98e2SPeter Wemm opt, 182606f25ae9SGregory Neil Shapiro subopt == NULL ? "" : ".", 1827c2aa98e2SPeter Wemm subopt == NULL ? "" : subopt); 1828c2aa98e2SPeter Wemm xputs(val); 1829c2aa98e2SPeter Wemm } 1830c2aa98e2SPeter Wemm 1831c2aa98e2SPeter Wemm /* 1832c2aa98e2SPeter Wemm ** See if this option is preset for us. 1833c2aa98e2SPeter Wemm */ 1834c2aa98e2SPeter Wemm 1835c2aa98e2SPeter Wemm if (!sticky && bitnset(opt, StickyOpt)) 1836c2aa98e2SPeter Wemm { 1837c2aa98e2SPeter Wemm if (tTd(37, 1)) 183806f25ae9SGregory Neil Shapiro dprintf(" (ignored)\n"); 1839c2aa98e2SPeter Wemm return; 1840c2aa98e2SPeter Wemm } 1841c2aa98e2SPeter Wemm 1842c2aa98e2SPeter Wemm /* 1843c2aa98e2SPeter Wemm ** Check to see if this option can be specified by this user. 1844c2aa98e2SPeter Wemm */ 1845c2aa98e2SPeter Wemm 1846c2aa98e2SPeter Wemm if (!safe && RealUid == 0) 1847c2aa98e2SPeter Wemm safe = TRUE; 184806f25ae9SGregory Neil Shapiro if (!safe && !bitset(OI_SAFE, o->o_flags)) 1849c2aa98e2SPeter Wemm { 1850c2aa98e2SPeter Wemm if (opt != 'M' || (val[0] != 'r' && val[0] != 's')) 1851c2aa98e2SPeter Wemm { 185206f25ae9SGregory Neil Shapiro int dp; 185306f25ae9SGregory Neil Shapiro 1854c2aa98e2SPeter Wemm if (tTd(37, 1)) 185506f25ae9SGregory Neil Shapiro dprintf(" (unsafe)"); 185606f25ae9SGregory Neil Shapiro dp = drop_privileges(TRUE); 185706f25ae9SGregory Neil Shapiro setstat(dp); 1858c2aa98e2SPeter Wemm } 1859c2aa98e2SPeter Wemm } 1860c2aa98e2SPeter Wemm if (tTd(37, 1)) 186106f25ae9SGregory Neil Shapiro dprintf("\n"); 1862c2aa98e2SPeter Wemm 1863c2aa98e2SPeter Wemm switch (opt & 0xff) 1864c2aa98e2SPeter Wemm { 1865c2aa98e2SPeter Wemm case '7': /* force seven-bit input */ 1866c2aa98e2SPeter Wemm SevenBitInput = atobool(val); 1867c2aa98e2SPeter Wemm break; 1868c2aa98e2SPeter Wemm 1869c2aa98e2SPeter Wemm #if MIME8TO7 1870c2aa98e2SPeter Wemm case '8': /* handling of 8-bit input */ 1871c2aa98e2SPeter Wemm switch (*val) 1872c2aa98e2SPeter Wemm { 1873c2aa98e2SPeter Wemm case 'm': /* convert 8-bit, convert MIME */ 1874c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME|MM_MIME8BIT; 1875c2aa98e2SPeter Wemm break; 1876c2aa98e2SPeter Wemm 1877c2aa98e2SPeter Wemm case 'p': /* pass 8 bit, convert MIME */ 1878c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME|MM_PASS8BIT; 1879c2aa98e2SPeter Wemm break; 1880c2aa98e2SPeter Wemm 1881c2aa98e2SPeter Wemm case 's': /* strict adherence */ 1882c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME; 1883c2aa98e2SPeter Wemm break; 1884c2aa98e2SPeter Wemm 1885c2aa98e2SPeter Wemm # if 0 1886c2aa98e2SPeter Wemm case 'r': /* reject 8-bit, don't convert MIME */ 1887c2aa98e2SPeter Wemm MimeMode = 0; 1888c2aa98e2SPeter Wemm break; 1889c2aa98e2SPeter Wemm 1890c2aa98e2SPeter Wemm case 'j': /* "just send 8" */ 1891c2aa98e2SPeter Wemm MimeMode = MM_PASS8BIT; 1892c2aa98e2SPeter Wemm break; 1893c2aa98e2SPeter Wemm 1894c2aa98e2SPeter Wemm case 'a': /* encode 8 bit if available */ 1895c2aa98e2SPeter Wemm MimeMode = MM_MIME8BIT|MM_PASS8BIT|MM_CVTMIME; 1896c2aa98e2SPeter Wemm break; 1897c2aa98e2SPeter Wemm 1898c2aa98e2SPeter Wemm case 'c': /* convert 8 bit to MIME, never 7 bit */ 1899c2aa98e2SPeter Wemm MimeMode = MM_MIME8BIT; 1900c2aa98e2SPeter Wemm break; 190106f25ae9SGregory Neil Shapiro # endif /* 0 */ 1902c2aa98e2SPeter Wemm 1903c2aa98e2SPeter Wemm default: 1904c2aa98e2SPeter Wemm syserr("Unknown 8-bit mode %c", *val); 1905065a643dSPeter Wemm finis(FALSE, EX_USAGE); 1906c2aa98e2SPeter Wemm } 1907c2aa98e2SPeter Wemm break; 190806f25ae9SGregory Neil Shapiro #endif /* MIME8TO7 */ 1909c2aa98e2SPeter Wemm 1910c2aa98e2SPeter Wemm case 'A': /* set default alias file */ 1911c2aa98e2SPeter Wemm if (val[0] == '\0') 1912c2aa98e2SPeter Wemm setalias("aliases"); 1913c2aa98e2SPeter Wemm else 1914c2aa98e2SPeter Wemm setalias(val); 1915c2aa98e2SPeter Wemm break; 1916c2aa98e2SPeter Wemm 1917c2aa98e2SPeter Wemm case 'a': /* look N minutes for "@:@" in alias file */ 1918c2aa98e2SPeter Wemm if (val[0] == '\0') 1919c2aa98e2SPeter Wemm SafeAlias = 5 * 60; /* five minutes */ 1920c2aa98e2SPeter Wemm else 1921c2aa98e2SPeter Wemm SafeAlias = convtime(val, 'm'); 1922c2aa98e2SPeter Wemm break; 1923c2aa98e2SPeter Wemm 1924c2aa98e2SPeter Wemm case 'B': /* substitution for blank character */ 1925c2aa98e2SPeter Wemm SpaceSub = val[0]; 1926c2aa98e2SPeter Wemm if (SpaceSub == '\0') 1927c2aa98e2SPeter Wemm SpaceSub = ' '; 1928c2aa98e2SPeter Wemm break; 1929c2aa98e2SPeter Wemm 1930c2aa98e2SPeter Wemm case 'b': /* min blocks free on queue fs/max msg size */ 1931c2aa98e2SPeter Wemm p = strchr(val, '/'); 1932c2aa98e2SPeter Wemm if (p != NULL) 1933c2aa98e2SPeter Wemm { 1934c2aa98e2SPeter Wemm *p++ = '\0'; 1935c2aa98e2SPeter Wemm MaxMessageSize = atol(p); 1936c2aa98e2SPeter Wemm } 1937c2aa98e2SPeter Wemm MinBlocksFree = atol(val); 1938c2aa98e2SPeter Wemm break; 1939c2aa98e2SPeter Wemm 1940c2aa98e2SPeter Wemm case 'c': /* don't connect to "expensive" mailers */ 1941c2aa98e2SPeter Wemm NoConnect = atobool(val); 1942c2aa98e2SPeter Wemm break; 1943c2aa98e2SPeter Wemm 1944c2aa98e2SPeter Wemm case 'C': /* checkpoint every N addresses */ 1945c2aa98e2SPeter Wemm CheckpointInterval = atoi(val); 1946c2aa98e2SPeter Wemm break; 1947c2aa98e2SPeter Wemm 1948c2aa98e2SPeter Wemm case 'd': /* delivery mode */ 1949c2aa98e2SPeter Wemm switch (*val) 1950c2aa98e2SPeter Wemm { 1951c2aa98e2SPeter Wemm case '\0': 195206f25ae9SGregory Neil Shapiro set_delivery_mode(SM_DELIVER, e); 1953c2aa98e2SPeter Wemm break; 1954c2aa98e2SPeter Wemm 1955c2aa98e2SPeter Wemm case SM_QUEUE: /* queue only */ 1956c2aa98e2SPeter Wemm case SM_DEFER: /* queue only and defer map lookups */ 1957c2aa98e2SPeter Wemm #if !QUEUE 1958c2aa98e2SPeter Wemm syserr("need QUEUE to set -odqueue or -oddefer"); 195906f25ae9SGregory Neil Shapiro #endif /* !QUEUE */ 196006f25ae9SGregory Neil Shapiro /* FALLTHROUGH */ 1961c2aa98e2SPeter Wemm 1962c2aa98e2SPeter Wemm case SM_DELIVER: /* do everything */ 1963c2aa98e2SPeter Wemm case SM_FORK: /* fork after verification */ 196406f25ae9SGregory Neil Shapiro set_delivery_mode(*val, e); 1965c2aa98e2SPeter Wemm break; 1966c2aa98e2SPeter Wemm 1967c2aa98e2SPeter Wemm default: 1968c2aa98e2SPeter Wemm syserr("Unknown delivery mode %c", *val); 1969065a643dSPeter Wemm finis(FALSE, EX_USAGE); 1970c2aa98e2SPeter Wemm } 1971c2aa98e2SPeter Wemm break; 1972c2aa98e2SPeter Wemm 197306f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_AUTOREBUILD 1974c2aa98e2SPeter Wemm case 'D': /* rebuild alias database as needed */ 1975c2aa98e2SPeter Wemm AutoRebuild = atobool(val); 1976c2aa98e2SPeter Wemm break; 197706f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_AUTOREBUILD */ 1978c2aa98e2SPeter Wemm 1979c2aa98e2SPeter Wemm case 'E': /* error message header/header file */ 1980c2aa98e2SPeter Wemm if (*val != '\0') 1981c2aa98e2SPeter Wemm ErrMsgFile = newstr(val); 1982c2aa98e2SPeter Wemm break; 1983c2aa98e2SPeter Wemm 1984c2aa98e2SPeter Wemm case 'e': /* set error processing mode */ 1985c2aa98e2SPeter Wemm switch (*val) 1986c2aa98e2SPeter Wemm { 1987c2aa98e2SPeter Wemm case EM_QUIET: /* be silent about it */ 1988c2aa98e2SPeter Wemm case EM_MAIL: /* mail back */ 1989c2aa98e2SPeter Wemm case EM_BERKNET: /* do berknet error processing */ 1990c2aa98e2SPeter Wemm case EM_WRITE: /* write back (or mail) */ 1991c2aa98e2SPeter Wemm case EM_PRINT: /* print errors normally (default) */ 1992c2aa98e2SPeter Wemm e->e_errormode = *val; 1993c2aa98e2SPeter Wemm break; 1994c2aa98e2SPeter Wemm } 1995c2aa98e2SPeter Wemm break; 1996c2aa98e2SPeter Wemm 1997c2aa98e2SPeter Wemm case 'F': /* file mode */ 1998c2aa98e2SPeter Wemm FileMode = atooct(val) & 0777; 1999c2aa98e2SPeter Wemm break; 2000c2aa98e2SPeter Wemm 2001c2aa98e2SPeter Wemm case 'f': /* save Unix-style From lines on front */ 2002c2aa98e2SPeter Wemm SaveFrom = atobool(val); 2003c2aa98e2SPeter Wemm break; 2004c2aa98e2SPeter Wemm 2005c2aa98e2SPeter Wemm case 'G': /* match recipients against GECOS field */ 2006c2aa98e2SPeter Wemm MatchGecos = atobool(val); 2007c2aa98e2SPeter Wemm break; 2008c2aa98e2SPeter Wemm 2009c2aa98e2SPeter Wemm case 'g': /* default gid */ 2010c2aa98e2SPeter Wemm g_opt: 2011c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2012c2aa98e2SPeter Wemm DefGid = atoi(val); 2013c2aa98e2SPeter Wemm else 2014c2aa98e2SPeter Wemm { 2015c2aa98e2SPeter Wemm register struct group *gr; 2016c2aa98e2SPeter Wemm 2017c2aa98e2SPeter Wemm DefGid = -1; 2018c2aa98e2SPeter Wemm gr = getgrnam(val); 2019c2aa98e2SPeter Wemm if (gr == NULL) 2020c2aa98e2SPeter Wemm syserr("readcf: option %c: unknown group %s", 2021c2aa98e2SPeter Wemm opt, val); 2022c2aa98e2SPeter Wemm else 2023c2aa98e2SPeter Wemm DefGid = gr->gr_gid; 2024c2aa98e2SPeter Wemm } 2025c2aa98e2SPeter Wemm break; 2026c2aa98e2SPeter Wemm 2027c2aa98e2SPeter Wemm case 'H': /* help file */ 2028c2aa98e2SPeter Wemm if (val[0] == '\0') 202906f25ae9SGregory Neil Shapiro HelpFile = "helpfile"; 2030c2aa98e2SPeter Wemm else 2031c2aa98e2SPeter Wemm HelpFile = newstr(val); 2032c2aa98e2SPeter Wemm break; 2033c2aa98e2SPeter Wemm 2034c2aa98e2SPeter Wemm case 'h': /* maximum hop count */ 2035c2aa98e2SPeter Wemm MaxHopCount = atoi(val); 2036c2aa98e2SPeter Wemm break; 2037c2aa98e2SPeter Wemm 2038c2aa98e2SPeter Wemm case 'I': /* use internet domain name server */ 2039c2aa98e2SPeter Wemm #if NAMED_BIND 2040c2aa98e2SPeter Wemm for (p = val; *p != 0; ) 2041c2aa98e2SPeter Wemm { 2042c2aa98e2SPeter Wemm bool clearmode; 2043c2aa98e2SPeter Wemm char *q; 2044c2aa98e2SPeter Wemm struct resolverflags *rfp; 2045c2aa98e2SPeter Wemm 2046c2aa98e2SPeter Wemm while (*p == ' ') 2047c2aa98e2SPeter Wemm p++; 2048c2aa98e2SPeter Wemm if (*p == '\0') 2049c2aa98e2SPeter Wemm break; 2050c2aa98e2SPeter Wemm clearmode = FALSE; 2051c2aa98e2SPeter Wemm if (*p == '-') 2052c2aa98e2SPeter Wemm clearmode = TRUE; 2053c2aa98e2SPeter Wemm else if (*p != '+') 2054c2aa98e2SPeter Wemm p--; 2055c2aa98e2SPeter Wemm p++; 2056c2aa98e2SPeter Wemm q = p; 2057c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 2058c2aa98e2SPeter Wemm p++; 2059c2aa98e2SPeter Wemm if (*p != '\0') 2060c2aa98e2SPeter Wemm *p++ = '\0'; 2061c2aa98e2SPeter Wemm if (strcasecmp(q, "HasWildcardMX") == 0) 2062c2aa98e2SPeter Wemm { 2063c2aa98e2SPeter Wemm HasWildcardMX = !clearmode; 2064c2aa98e2SPeter Wemm continue; 2065c2aa98e2SPeter Wemm } 2066c2aa98e2SPeter Wemm for (rfp = ResolverFlags; rfp->rf_name != NULL; rfp++) 2067c2aa98e2SPeter Wemm { 2068c2aa98e2SPeter Wemm if (strcasecmp(q, rfp->rf_name) == 0) 2069c2aa98e2SPeter Wemm break; 2070c2aa98e2SPeter Wemm } 2071c2aa98e2SPeter Wemm if (rfp->rf_name == NULL) 2072c2aa98e2SPeter Wemm syserr("readcf: I option value %s unrecognized", q); 2073c2aa98e2SPeter Wemm else if (clearmode) 2074c2aa98e2SPeter Wemm _res.options &= ~rfp->rf_bits; 2075c2aa98e2SPeter Wemm else 2076c2aa98e2SPeter Wemm _res.options |= rfp->rf_bits; 2077c2aa98e2SPeter Wemm } 2078c2aa98e2SPeter Wemm if (tTd(8, 2)) 207906f25ae9SGregory Neil Shapiro dprintf("_res.options = %x, HasWildcardMX = %d\n", 2080c2aa98e2SPeter Wemm (u_int) _res.options, HasWildcardMX); 208106f25ae9SGregory Neil Shapiro #else /* NAMED_BIND */ 2082c2aa98e2SPeter Wemm usrerr("name server (I option) specified but BIND not compiled in"); 208306f25ae9SGregory Neil Shapiro #endif /* NAMED_BIND */ 2084c2aa98e2SPeter Wemm break; 2085c2aa98e2SPeter Wemm 2086c2aa98e2SPeter Wemm case 'i': /* ignore dot lines in message */ 2087c2aa98e2SPeter Wemm IgnrDot = atobool(val); 2088c2aa98e2SPeter Wemm break; 2089c2aa98e2SPeter Wemm 2090c2aa98e2SPeter Wemm case 'j': /* send errors in MIME (RFC 1341) format */ 2091c2aa98e2SPeter Wemm SendMIMEErrors = atobool(val); 2092c2aa98e2SPeter Wemm break; 2093c2aa98e2SPeter Wemm 2094c2aa98e2SPeter Wemm case 'J': /* .forward search path */ 2095c2aa98e2SPeter Wemm ForwardPath = newstr(val); 2096c2aa98e2SPeter Wemm break; 2097c2aa98e2SPeter Wemm 2098c2aa98e2SPeter Wemm case 'k': /* connection cache size */ 2099c2aa98e2SPeter Wemm MaxMciCache = atoi(val); 2100c2aa98e2SPeter Wemm if (MaxMciCache < 0) 2101c2aa98e2SPeter Wemm MaxMciCache = 0; 2102c2aa98e2SPeter Wemm break; 2103c2aa98e2SPeter Wemm 2104c2aa98e2SPeter Wemm case 'K': /* connection cache timeout */ 2105c2aa98e2SPeter Wemm MciCacheTimeout = convtime(val, 'm'); 2106c2aa98e2SPeter Wemm break; 2107c2aa98e2SPeter Wemm 2108c2aa98e2SPeter Wemm case 'l': /* use Errors-To: header */ 2109c2aa98e2SPeter Wemm UseErrorsTo = atobool(val); 2110c2aa98e2SPeter Wemm break; 2111c2aa98e2SPeter Wemm 2112c2aa98e2SPeter Wemm case 'L': /* log level */ 2113c2aa98e2SPeter Wemm if (safe || LogLevel < atoi(val)) 2114c2aa98e2SPeter Wemm LogLevel = atoi(val); 2115c2aa98e2SPeter Wemm break; 2116c2aa98e2SPeter Wemm 2117c2aa98e2SPeter Wemm case 'M': /* define macro */ 2118c2aa98e2SPeter Wemm mid = macid(val, &ep); 2119c2aa98e2SPeter Wemm p = newstr(ep); 2120c2aa98e2SPeter Wemm if (!safe) 2121c2aa98e2SPeter Wemm cleanstrcpy(p, p, MAXNAME); 2122c2aa98e2SPeter Wemm define(mid, p, CurEnv); 2123c2aa98e2SPeter Wemm sticky = FALSE; 2124c2aa98e2SPeter Wemm break; 2125c2aa98e2SPeter Wemm 2126c2aa98e2SPeter Wemm case 'm': /* send to me too */ 2127c2aa98e2SPeter Wemm MeToo = atobool(val); 2128c2aa98e2SPeter Wemm break; 2129c2aa98e2SPeter Wemm 2130c2aa98e2SPeter Wemm case 'n': /* validate RHS in newaliases */ 2131c2aa98e2SPeter Wemm CheckAliases = atobool(val); 2132c2aa98e2SPeter Wemm break; 2133c2aa98e2SPeter Wemm 2134c2aa98e2SPeter Wemm /* 'N' available -- was "net name" */ 2135c2aa98e2SPeter Wemm 2136c2aa98e2SPeter Wemm case 'O': /* daemon options */ 2137c2aa98e2SPeter Wemm #if DAEMON 213806f25ae9SGregory Neil Shapiro if (!setdaemonoptions(val)) 213906f25ae9SGregory Neil Shapiro { 214006f25ae9SGregory Neil Shapiro syserr("too many daemons defined (%d max)", MAXDAEMONS); 214106f25ae9SGregory Neil Shapiro } 214206f25ae9SGregory Neil Shapiro #else /* DAEMON */ 2143c2aa98e2SPeter Wemm syserr("DaemonPortOptions (O option) set but DAEMON not compiled in"); 214406f25ae9SGregory Neil Shapiro #endif /* DAEMON */ 2145c2aa98e2SPeter Wemm break; 2146c2aa98e2SPeter Wemm 2147c2aa98e2SPeter Wemm case 'o': /* assume old style headers */ 2148c2aa98e2SPeter Wemm if (atobool(val)) 2149c2aa98e2SPeter Wemm CurEnv->e_flags |= EF_OLDSTYLE; 2150c2aa98e2SPeter Wemm else 2151c2aa98e2SPeter Wemm CurEnv->e_flags &= ~EF_OLDSTYLE; 2152c2aa98e2SPeter Wemm break; 2153c2aa98e2SPeter Wemm 2154c2aa98e2SPeter Wemm case 'p': /* select privacy level */ 2155c2aa98e2SPeter Wemm p = val; 2156c2aa98e2SPeter Wemm for (;;) 2157c2aa98e2SPeter Wemm { 2158c2aa98e2SPeter Wemm register struct prival *pv; 2159c2aa98e2SPeter Wemm extern struct prival PrivacyValues[]; 2160c2aa98e2SPeter Wemm 2161c2aa98e2SPeter Wemm while (isascii(*p) && (isspace(*p) || ispunct(*p))) 2162c2aa98e2SPeter Wemm p++; 2163c2aa98e2SPeter Wemm if (*p == '\0') 2164c2aa98e2SPeter Wemm break; 2165c2aa98e2SPeter Wemm val = p; 2166c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 2167c2aa98e2SPeter Wemm p++; 2168c2aa98e2SPeter Wemm if (*p != '\0') 2169c2aa98e2SPeter Wemm *p++ = '\0'; 2170c2aa98e2SPeter Wemm 2171c2aa98e2SPeter Wemm for (pv = PrivacyValues; pv->pv_name != NULL; pv++) 2172c2aa98e2SPeter Wemm { 2173c2aa98e2SPeter Wemm if (strcasecmp(val, pv->pv_name) == 0) 2174c2aa98e2SPeter Wemm break; 2175c2aa98e2SPeter Wemm } 2176c2aa98e2SPeter Wemm if (pv->pv_name == NULL) 2177c2aa98e2SPeter Wemm syserr("readcf: Op line: %s unrecognized", val); 2178c2aa98e2SPeter Wemm PrivacyFlags |= pv->pv_flag; 2179c2aa98e2SPeter Wemm } 2180c2aa98e2SPeter Wemm sticky = FALSE; 2181c2aa98e2SPeter Wemm break; 2182c2aa98e2SPeter Wemm 2183c2aa98e2SPeter Wemm case 'P': /* postmaster copy address for returned mail */ 2184c2aa98e2SPeter Wemm PostMasterCopy = newstr(val); 2185c2aa98e2SPeter Wemm break; 2186c2aa98e2SPeter Wemm 2187c2aa98e2SPeter Wemm case 'q': /* slope of queue only function */ 2188c2aa98e2SPeter Wemm QueueFactor = atoi(val); 2189c2aa98e2SPeter Wemm break; 2190c2aa98e2SPeter Wemm 2191c2aa98e2SPeter Wemm case 'Q': /* queue directory */ 2192c2aa98e2SPeter Wemm if (val[0] == '\0') 219342e5d165SGregory Neil Shapiro { 2194c2aa98e2SPeter Wemm QueueDir = "mqueue"; 219542e5d165SGregory Neil Shapiro } 2196c2aa98e2SPeter Wemm else 219742e5d165SGregory Neil Shapiro { 2198c2aa98e2SPeter Wemm QueueDir = newstr(val); 219942e5d165SGregory Neil Shapiro } 2200c2aa98e2SPeter Wemm if (RealUid != 0 && !safe) 2201c2aa98e2SPeter Wemm Warn_Q_option = TRUE; 2202c2aa98e2SPeter Wemm break; 2203c2aa98e2SPeter Wemm 2204c2aa98e2SPeter Wemm case 'R': /* don't prune routes */ 2205c2aa98e2SPeter Wemm DontPruneRoutes = atobool(val); 2206c2aa98e2SPeter Wemm break; 2207c2aa98e2SPeter Wemm 2208c2aa98e2SPeter Wemm case 'r': /* read timeout */ 2209c2aa98e2SPeter Wemm if (subopt == NULL) 221006f25ae9SGregory Neil Shapiro inittimeouts(val, sticky); 2211c2aa98e2SPeter Wemm else 221206f25ae9SGregory Neil Shapiro settimeout(subopt, val, sticky); 2213c2aa98e2SPeter Wemm break; 2214c2aa98e2SPeter Wemm 2215c2aa98e2SPeter Wemm case 'S': /* status file */ 2216c2aa98e2SPeter Wemm if (val[0] == '\0') 221706f25ae9SGregory Neil Shapiro StatFile = "statistics"; 2218c2aa98e2SPeter Wemm else 2219c2aa98e2SPeter Wemm StatFile = newstr(val); 2220c2aa98e2SPeter Wemm break; 2221c2aa98e2SPeter Wemm 2222c2aa98e2SPeter Wemm case 's': /* be super safe, even if expensive */ 2223c2aa98e2SPeter Wemm SuperSafe = atobool(val); 2224c2aa98e2SPeter Wemm break; 2225c2aa98e2SPeter Wemm 2226c2aa98e2SPeter Wemm case 'T': /* queue timeout */ 2227c2aa98e2SPeter Wemm p = strchr(val, '/'); 2228c2aa98e2SPeter Wemm if (p != NULL) 2229c2aa98e2SPeter Wemm { 2230c2aa98e2SPeter Wemm *p++ = '\0'; 223106f25ae9SGregory Neil Shapiro settimeout("queuewarn", p, sticky); 2232c2aa98e2SPeter Wemm } 223306f25ae9SGregory Neil Shapiro settimeout("queuereturn", val, sticky); 2234c2aa98e2SPeter Wemm break; 2235c2aa98e2SPeter Wemm 2236c2aa98e2SPeter Wemm case 't': /* time zone name */ 2237c2aa98e2SPeter Wemm TimeZoneSpec = newstr(val); 2238c2aa98e2SPeter Wemm break; 2239c2aa98e2SPeter Wemm 2240c2aa98e2SPeter Wemm case 'U': /* location of user database */ 2241c2aa98e2SPeter Wemm UdbSpec = newstr(val); 2242c2aa98e2SPeter Wemm break; 2243c2aa98e2SPeter Wemm 2244c2aa98e2SPeter Wemm case 'u': /* set default uid */ 2245c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2246c2aa98e2SPeter Wemm { 2247c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2248c2aa98e2SPeter Wemm { 2249c2aa98e2SPeter Wemm *p++ = '\0'; 2250c2aa98e2SPeter Wemm break; 2251c2aa98e2SPeter Wemm } 2252c2aa98e2SPeter Wemm } 2253c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2254c2aa98e2SPeter Wemm { 2255c2aa98e2SPeter Wemm DefUid = atoi(val); 2256c2aa98e2SPeter Wemm setdefuser(); 2257c2aa98e2SPeter Wemm } 2258c2aa98e2SPeter Wemm else 2259c2aa98e2SPeter Wemm { 2260c2aa98e2SPeter Wemm register struct passwd *pw; 2261c2aa98e2SPeter Wemm 2262c2aa98e2SPeter Wemm DefUid = -1; 2263c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2264c2aa98e2SPeter Wemm if (pw == NULL) 2265c2aa98e2SPeter Wemm syserr("readcf: option u: unknown user %s", val); 2266c2aa98e2SPeter Wemm else 2267c2aa98e2SPeter Wemm { 2268c2aa98e2SPeter Wemm DefUid = pw->pw_uid; 2269c2aa98e2SPeter Wemm DefGid = pw->pw_gid; 2270c2aa98e2SPeter Wemm DefUser = newstr(pw->pw_name); 2271c2aa98e2SPeter Wemm } 2272c2aa98e2SPeter Wemm } 2273c2aa98e2SPeter Wemm 2274c2aa98e2SPeter Wemm #ifdef UID_MAX 2275c2aa98e2SPeter Wemm if (DefUid > UID_MAX) 2276c2aa98e2SPeter Wemm { 2277c2aa98e2SPeter Wemm syserr("readcf: option u: uid value (%ld) > UID_MAX (%ld); ignored", 2278c2aa98e2SPeter Wemm DefUid, UID_MAX); 2279c2aa98e2SPeter Wemm } 228006f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2281c2aa98e2SPeter Wemm 2282c2aa98e2SPeter Wemm /* handle the group if it is there */ 2283c2aa98e2SPeter Wemm if (*p == '\0') 2284c2aa98e2SPeter Wemm break; 2285c2aa98e2SPeter Wemm val = p; 2286c2aa98e2SPeter Wemm goto g_opt; 2287c2aa98e2SPeter Wemm 2288c2aa98e2SPeter Wemm case 'V': /* fallback MX host */ 2289c2aa98e2SPeter Wemm if (val[0] != '\0') 2290c2aa98e2SPeter Wemm FallBackMX = newstr(val); 2291c2aa98e2SPeter Wemm break; 2292c2aa98e2SPeter Wemm 2293c2aa98e2SPeter Wemm case 'v': /* run in verbose mode */ 2294c2aa98e2SPeter Wemm Verbose = atobool(val) ? 1 : 0; 2295c2aa98e2SPeter Wemm break; 2296c2aa98e2SPeter Wemm 2297c2aa98e2SPeter Wemm case 'w': /* if we are best MX, try host directly */ 2298c2aa98e2SPeter Wemm TryNullMXList = atobool(val); 2299c2aa98e2SPeter Wemm break; 2300c2aa98e2SPeter Wemm 2301c2aa98e2SPeter Wemm /* 'W' available -- was wizard password */ 2302c2aa98e2SPeter Wemm 2303c2aa98e2SPeter Wemm case 'x': /* load avg at which to auto-queue msgs */ 2304c2aa98e2SPeter Wemm QueueLA = atoi(val); 2305c2aa98e2SPeter Wemm break; 2306c2aa98e2SPeter Wemm 2307c2aa98e2SPeter Wemm case 'X': /* load avg at which to auto-reject connections */ 2308c2aa98e2SPeter Wemm RefuseLA = atoi(val); 2309c2aa98e2SPeter Wemm break; 2310c2aa98e2SPeter Wemm 2311c2aa98e2SPeter Wemm case 'y': /* work recipient factor */ 2312c2aa98e2SPeter Wemm WkRecipFact = atoi(val); 2313c2aa98e2SPeter Wemm break; 2314c2aa98e2SPeter Wemm 2315c2aa98e2SPeter Wemm case 'Y': /* fork jobs during queue runs */ 2316c2aa98e2SPeter Wemm ForkQueueRuns = atobool(val); 2317c2aa98e2SPeter Wemm break; 2318c2aa98e2SPeter Wemm 2319c2aa98e2SPeter Wemm case 'z': /* work message class factor */ 2320c2aa98e2SPeter Wemm WkClassFact = atoi(val); 2321c2aa98e2SPeter Wemm break; 2322c2aa98e2SPeter Wemm 2323c2aa98e2SPeter Wemm case 'Z': /* work time factor */ 2324c2aa98e2SPeter Wemm WkTimeFact = atoi(val); 2325c2aa98e2SPeter Wemm break; 2326c2aa98e2SPeter Wemm 232706f25ae9SGregory Neil Shapiro 2328c2aa98e2SPeter Wemm case O_QUEUESORTORD: /* queue sorting order */ 2329c2aa98e2SPeter Wemm switch (*val) 2330c2aa98e2SPeter Wemm { 2331c2aa98e2SPeter Wemm case 'h': /* Host first */ 2332c2aa98e2SPeter Wemm case 'H': 233306f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYHOST; 2334c2aa98e2SPeter Wemm break; 2335c2aa98e2SPeter Wemm 2336c2aa98e2SPeter Wemm case 'p': /* Priority order */ 2337c2aa98e2SPeter Wemm case 'P': 233806f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYPRIORITY; 2339c2aa98e2SPeter Wemm break; 2340c2aa98e2SPeter Wemm 2341c2aa98e2SPeter Wemm case 't': /* Submission time */ 2342c2aa98e2SPeter Wemm case 'T': 234306f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYTIME; 234406f25ae9SGregory Neil Shapiro break; 234506f25ae9SGregory Neil Shapiro 234606f25ae9SGregory Neil Shapiro case 'f': /* File Name */ 234706f25ae9SGregory Neil Shapiro case 'F': 234806f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYFILENAME; 2349c2aa98e2SPeter Wemm break; 2350c2aa98e2SPeter Wemm 2351c2aa98e2SPeter Wemm default: 2352c2aa98e2SPeter Wemm syserr("Invalid queue sort order \"%s\"", val); 2353c2aa98e2SPeter Wemm } 2354c2aa98e2SPeter Wemm break; 2355c2aa98e2SPeter Wemm 235606f25ae9SGregory Neil Shapiro #if _FFR_QUEUEDELAY 235706f25ae9SGregory Neil Shapiro case O_QUEUEDELAY: /* queue delay algorithm */ 235806f25ae9SGregory Neil Shapiro switch (*val) 235906f25ae9SGregory Neil Shapiro { 236006f25ae9SGregory Neil Shapiro case 'e': /* exponential */ 236106f25ae9SGregory Neil Shapiro case 'E': 236206f25ae9SGregory Neil Shapiro QueueAlg = QD_EXP; 236306f25ae9SGregory Neil Shapiro QueueInitDelay = 10 MINUTES; 236406f25ae9SGregory Neil Shapiro QueueMaxDelay = 2 HOURS; 236506f25ae9SGregory Neil Shapiro p = strchr(val, '/'); 236606f25ae9SGregory Neil Shapiro if (p != NULL) 236706f25ae9SGregory Neil Shapiro { 236806f25ae9SGregory Neil Shapiro char *q; 236906f25ae9SGregory Neil Shapiro 237006f25ae9SGregory Neil Shapiro *p++ = '\0'; 237106f25ae9SGregory Neil Shapiro q = strchr(p, '/'); 237206f25ae9SGregory Neil Shapiro if (q != NULL) 237306f25ae9SGregory Neil Shapiro *q++ = '\0'; 237406f25ae9SGregory Neil Shapiro QueueInitDelay = convtime(p, 's'); 237506f25ae9SGregory Neil Shapiro if (q != NULL) 237606f25ae9SGregory Neil Shapiro { 237706f25ae9SGregory Neil Shapiro QueueMaxDelay = convtime(q, 's'); 237806f25ae9SGregory Neil Shapiro } 237906f25ae9SGregory Neil Shapiro } 238006f25ae9SGregory Neil Shapiro break; 238106f25ae9SGregory Neil Shapiro 238206f25ae9SGregory Neil Shapiro case 'l': /* linear */ 238306f25ae9SGregory Neil Shapiro case 'L': 238406f25ae9SGregory Neil Shapiro QueueAlg = QD_LINEAR; 238506f25ae9SGregory Neil Shapiro break; 238606f25ae9SGregory Neil Shapiro 238706f25ae9SGregory Neil Shapiro default: 238806f25ae9SGregory Neil Shapiro syserr("Invalid queue delay algorithm \"%s\"", val); 238906f25ae9SGregory Neil Shapiro } 239006f25ae9SGregory Neil Shapiro break; 239106f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUEDELAY */ 239206f25ae9SGregory Neil Shapiro 2393c2aa98e2SPeter Wemm case O_HOSTSFILE: /* pathname of /etc/hosts file */ 2394c2aa98e2SPeter Wemm HostsFile = newstr(val); 2395c2aa98e2SPeter Wemm break; 2396c2aa98e2SPeter Wemm 2397c2aa98e2SPeter Wemm case O_MQA: /* minimum queue age between deliveries */ 2398c2aa98e2SPeter Wemm MinQueueAge = convtime(val, 'm'); 2399c2aa98e2SPeter Wemm break; 2400c2aa98e2SPeter Wemm 2401c2aa98e2SPeter Wemm case O_DEFCHARSET: /* default character set for mimefying */ 2402c2aa98e2SPeter Wemm DefaultCharSet = newstr(denlstring(val, TRUE, TRUE)); 2403c2aa98e2SPeter Wemm break; 2404c2aa98e2SPeter Wemm 2405c2aa98e2SPeter Wemm case O_SSFILE: /* service switch file */ 2406c2aa98e2SPeter Wemm ServiceSwitchFile = newstr(val); 2407c2aa98e2SPeter Wemm break; 2408c2aa98e2SPeter Wemm 2409c2aa98e2SPeter Wemm case O_DIALDELAY: /* delay for dial-on-demand operation */ 2410c2aa98e2SPeter Wemm DialDelay = convtime(val, 's'); 2411c2aa98e2SPeter Wemm break; 2412c2aa98e2SPeter Wemm 2413c2aa98e2SPeter Wemm case O_NORCPTACTION: /* what to do if no recipient */ 2414c2aa98e2SPeter Wemm if (strcasecmp(val, "none") == 0) 2415c2aa98e2SPeter Wemm NoRecipientAction = NRA_NO_ACTION; 2416c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to") == 0) 2417c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO; 2418c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-apparently-to") == 0) 2419c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_APPARENTLY_TO; 2420c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-bcc") == 0) 2421c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_BCC; 2422c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to-undisclosed") == 0) 2423c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO_UNDISCLOSED; 2424c2aa98e2SPeter Wemm else 2425c2aa98e2SPeter Wemm syserr("Invalid NoRecipientAction: %s", val); 2426c2aa98e2SPeter Wemm break; 2427c2aa98e2SPeter Wemm 2428c2aa98e2SPeter Wemm case O_SAFEFILEENV: /* chroot() environ for writing to files */ 2429c2aa98e2SPeter Wemm SafeFileEnv = newstr(val); 2430c2aa98e2SPeter Wemm break; 2431c2aa98e2SPeter Wemm 2432c2aa98e2SPeter Wemm case O_MAXMSGSIZE: /* maximum message size */ 2433c2aa98e2SPeter Wemm MaxMessageSize = atol(val); 2434c2aa98e2SPeter Wemm break; 2435c2aa98e2SPeter Wemm 2436c2aa98e2SPeter Wemm case O_COLONOKINADDR: /* old style handling of colon addresses */ 2437c2aa98e2SPeter Wemm ColonOkInAddr = atobool(val); 2438c2aa98e2SPeter Wemm break; 2439c2aa98e2SPeter Wemm 2440c2aa98e2SPeter Wemm case O_MAXQUEUERUN: /* max # of jobs in a single queue run */ 2441c2aa98e2SPeter Wemm MaxQueueRun = atol(val); 2442c2aa98e2SPeter Wemm break; 2443c2aa98e2SPeter Wemm 2444c2aa98e2SPeter Wemm case O_MAXCHILDREN: /* max # of children of daemon */ 2445c2aa98e2SPeter Wemm MaxChildren = atoi(val); 2446c2aa98e2SPeter Wemm break; 2447c2aa98e2SPeter Wemm 244806f25ae9SGregory Neil Shapiro #if _FFR_MAX_FORWARD_ENTRIES 244906f25ae9SGregory Neil Shapiro case O_MAXFORWARD: /* max # of forward entries */ 245006f25ae9SGregory Neil Shapiro MaxForwardEntries = atoi(val); 245106f25ae9SGregory Neil Shapiro break; 245206f25ae9SGregory Neil Shapiro #endif /* _FFR_MAX_FORWARD_ENTRIES */ 245306f25ae9SGregory Neil Shapiro 2454c2aa98e2SPeter Wemm case O_KEEPCNAMES: /* don't expand CNAME records */ 2455c2aa98e2SPeter Wemm DontExpandCnames = atobool(val); 2456c2aa98e2SPeter Wemm break; 2457c2aa98e2SPeter Wemm 2458c2aa98e2SPeter Wemm case O_MUSTQUOTE: /* must quote these characters in phrases */ 245906f25ae9SGregory Neil Shapiro (void) strlcpy(buf, "@,;:\\()[]", sizeof buf); 2460c2aa98e2SPeter Wemm if (strlen(val) < (SIZE_T) sizeof buf - 10) 246106f25ae9SGregory Neil Shapiro (void) strlcat(buf, val, sizeof buf); 246206f25ae9SGregory Neil Shapiro else 246306f25ae9SGregory Neil Shapiro printf("Warning: MustQuoteChars too long, ignored.\n"); 2464c2aa98e2SPeter Wemm MustQuoteChars = newstr(buf); 2465c2aa98e2SPeter Wemm break; 2466c2aa98e2SPeter Wemm 2467c2aa98e2SPeter Wemm case O_SMTPGREETING: /* SMTP greeting message (old $e macro) */ 2468c2aa98e2SPeter Wemm SmtpGreeting = newstr(munchstring(val, NULL, '\0')); 2469c2aa98e2SPeter Wemm break; 2470c2aa98e2SPeter Wemm 2471c2aa98e2SPeter Wemm case O_UNIXFROM: /* UNIX From_ line (old $l macro) */ 2472c2aa98e2SPeter Wemm UnixFromLine = newstr(munchstring(val, NULL, '\0')); 2473c2aa98e2SPeter Wemm break; 2474c2aa98e2SPeter Wemm 2475c2aa98e2SPeter Wemm case O_OPCHARS: /* operator characters (old $o macro) */ 247606f25ae9SGregory Neil Shapiro if (OperatorChars != NULL) 247706f25ae9SGregory Neil Shapiro printf("Warning: OperatorChars is being redefined.\n It should only be set before ruleset definitions.\n"); 2478c2aa98e2SPeter Wemm OperatorChars = newstr(munchstring(val, NULL, '\0')); 2479c2aa98e2SPeter Wemm break; 2480c2aa98e2SPeter Wemm 2481c2aa98e2SPeter Wemm case O_DONTINITGRPS: /* don't call initgroups(3) */ 2482c2aa98e2SPeter Wemm DontInitGroups = atobool(val); 2483c2aa98e2SPeter Wemm break; 2484c2aa98e2SPeter Wemm 2485c2aa98e2SPeter Wemm case O_SLFH: /* make sure from fits on one line */ 2486c2aa98e2SPeter Wemm SingleLineFromHeader = atobool(val); 2487c2aa98e2SPeter Wemm break; 2488c2aa98e2SPeter Wemm 2489c2aa98e2SPeter Wemm case O_ABH: /* allow HELO commands with syntax errors */ 2490c2aa98e2SPeter Wemm AllowBogusHELO = atobool(val); 2491c2aa98e2SPeter Wemm break; 2492c2aa98e2SPeter Wemm 2493c2aa98e2SPeter Wemm case O_CONNTHROT: /* connection rate throttle */ 2494c2aa98e2SPeter Wemm ConnRateThrottle = atoi(val); 2495c2aa98e2SPeter Wemm break; 2496c2aa98e2SPeter Wemm 2497c2aa98e2SPeter Wemm case O_UGW: /* group writable files are unsafe */ 2498c2aa98e2SPeter Wemm if (!atobool(val)) 249906f25ae9SGregory Neil Shapiro { 250006f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEFORWARDFILESAFE, 250106f25ae9SGregory Neil Shapiro DontBlameSendmail); 250206f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEINCLUDEFILESAFE, 250306f25ae9SGregory Neil Shapiro DontBlameSendmail); 250406f25ae9SGregory Neil Shapiro } 2505c2aa98e2SPeter Wemm break; 2506c2aa98e2SPeter Wemm 2507c2aa98e2SPeter Wemm case O_DBLBOUNCE: /* address to which to send double bounces */ 2508c2aa98e2SPeter Wemm if (val[0] != '\0') 2509c2aa98e2SPeter Wemm DoubleBounceAddr = newstr(val); 2510c2aa98e2SPeter Wemm else 2511c2aa98e2SPeter Wemm syserr("readcf: option DoubleBounceAddress: value required"); 2512c2aa98e2SPeter Wemm break; 2513c2aa98e2SPeter Wemm 2514c2aa98e2SPeter Wemm case O_HSDIR: /* persistent host status directory */ 2515c2aa98e2SPeter Wemm if (val[0] != '\0') 2516c2aa98e2SPeter Wemm HostStatDir = newstr(val); 2517c2aa98e2SPeter Wemm break; 2518c2aa98e2SPeter Wemm 2519c2aa98e2SPeter Wemm case O_SINGTHREAD: /* single thread deliveries (requires hsdir) */ 2520c2aa98e2SPeter Wemm SingleThreadDelivery = atobool(val); 2521c2aa98e2SPeter Wemm break; 2522c2aa98e2SPeter Wemm 2523c2aa98e2SPeter Wemm case O_RUNASUSER: /* run bulk of code as this user */ 2524c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2525c2aa98e2SPeter Wemm { 2526c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2527c2aa98e2SPeter Wemm { 2528c2aa98e2SPeter Wemm *p++ = '\0'; 2529c2aa98e2SPeter Wemm break; 2530c2aa98e2SPeter Wemm } 2531c2aa98e2SPeter Wemm } 2532c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2533c2aa98e2SPeter Wemm { 2534c2aa98e2SPeter Wemm if (can_setuid) 2535c2aa98e2SPeter Wemm RunAsUid = atoi(val); 2536c2aa98e2SPeter Wemm } 2537c2aa98e2SPeter Wemm else 2538c2aa98e2SPeter Wemm { 2539c2aa98e2SPeter Wemm register struct passwd *pw; 2540c2aa98e2SPeter Wemm 2541c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2542c2aa98e2SPeter Wemm if (pw == NULL) 2543c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown user %s", val); 2544c2aa98e2SPeter Wemm else if (can_setuid) 2545c2aa98e2SPeter Wemm { 2546c2aa98e2SPeter Wemm if (*p == '\0') 2547c2aa98e2SPeter Wemm RunAsUserName = newstr(val); 2548c2aa98e2SPeter Wemm RunAsUid = pw->pw_uid; 2549c2aa98e2SPeter Wemm RunAsGid = pw->pw_gid; 2550c2aa98e2SPeter Wemm } 2551c2aa98e2SPeter Wemm } 2552c2aa98e2SPeter Wemm #ifdef UID_MAX 2553c2aa98e2SPeter Wemm if (RunAsUid > UID_MAX) 2554c2aa98e2SPeter Wemm { 2555c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: uid value (%ld) > UID_MAX (%ld); ignored", 2556c2aa98e2SPeter Wemm RunAsUid, UID_MAX); 2557c2aa98e2SPeter Wemm } 255806f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2559c2aa98e2SPeter Wemm if (*p != '\0') 2560c2aa98e2SPeter Wemm { 2561c2aa98e2SPeter Wemm if (isascii(*p) && isdigit(*p)) 2562c2aa98e2SPeter Wemm { 2563c2aa98e2SPeter Wemm if (can_setuid) 2564c2aa98e2SPeter Wemm RunAsGid = atoi(p); 2565c2aa98e2SPeter Wemm } 2566c2aa98e2SPeter Wemm else 2567c2aa98e2SPeter Wemm { 2568c2aa98e2SPeter Wemm register struct group *gr; 2569c2aa98e2SPeter Wemm 2570c2aa98e2SPeter Wemm gr = getgrnam(p); 2571c2aa98e2SPeter Wemm if (gr == NULL) 2572c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown group %s", 2573c2aa98e2SPeter Wemm p); 2574c2aa98e2SPeter Wemm else if (can_setuid) 2575c2aa98e2SPeter Wemm RunAsGid = gr->gr_gid; 2576c2aa98e2SPeter Wemm } 2577c2aa98e2SPeter Wemm } 2578c2aa98e2SPeter Wemm if (tTd(47, 5)) 257906f25ae9SGregory Neil Shapiro dprintf("readcf: RunAsUser = %d:%d\n", 258006f25ae9SGregory Neil Shapiro (int)RunAsUid, (int)RunAsGid); 2581c2aa98e2SPeter Wemm break; 2582c2aa98e2SPeter Wemm 2583c2aa98e2SPeter Wemm case O_DSN_RRT: 2584c2aa98e2SPeter Wemm RrtImpliesDsn = atobool(val); 2585c2aa98e2SPeter Wemm break; 2586c2aa98e2SPeter Wemm 2587c2aa98e2SPeter Wemm case O_PIDFILE: 258806f25ae9SGregory Neil Shapiro if (PidFile != NULL) 2589c2aa98e2SPeter Wemm free(PidFile); 2590c2aa98e2SPeter Wemm PidFile = newstr(val); 2591c2aa98e2SPeter Wemm break; 2592c2aa98e2SPeter Wemm 2593c2aa98e2SPeter Wemm case O_DONTBLAMESENDMAIL: 2594c2aa98e2SPeter Wemm p = val; 2595c2aa98e2SPeter Wemm for (;;) 2596c2aa98e2SPeter Wemm { 2597c2aa98e2SPeter Wemm register struct dbsval *dbs; 2598c2aa98e2SPeter Wemm extern struct dbsval DontBlameSendmailValues[]; 2599c2aa98e2SPeter Wemm 2600c2aa98e2SPeter Wemm while (isascii(*p) && (isspace(*p) || ispunct(*p))) 2601c2aa98e2SPeter Wemm p++; 2602c2aa98e2SPeter Wemm if (*p == '\0') 2603c2aa98e2SPeter Wemm break; 2604c2aa98e2SPeter Wemm val = p; 2605c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 2606c2aa98e2SPeter Wemm p++; 2607c2aa98e2SPeter Wemm if (*p != '\0') 2608c2aa98e2SPeter Wemm *p++ = '\0'; 2609c2aa98e2SPeter Wemm 2610c2aa98e2SPeter Wemm for (dbs = DontBlameSendmailValues; 2611c2aa98e2SPeter Wemm dbs->dbs_name != NULL; dbs++) 2612c2aa98e2SPeter Wemm { 2613c2aa98e2SPeter Wemm if (strcasecmp(val, dbs->dbs_name) == 0) 2614c2aa98e2SPeter Wemm break; 2615c2aa98e2SPeter Wemm } 2616c2aa98e2SPeter Wemm if (dbs->dbs_name == NULL) 2617c2aa98e2SPeter Wemm syserr("readcf: DontBlameSendmail option: %s unrecognized", val); 2618c2aa98e2SPeter Wemm else if (dbs->dbs_flag == DBS_SAFE) 261906f25ae9SGregory Neil Shapiro clrbitmap(DontBlameSendmail); 2620c2aa98e2SPeter Wemm else 262106f25ae9SGregory Neil Shapiro setbitn(dbs->dbs_flag, DontBlameSendmail); 2622c2aa98e2SPeter Wemm } 2623c2aa98e2SPeter Wemm sticky = FALSE; 2624c2aa98e2SPeter Wemm break; 2625c2aa98e2SPeter Wemm 2626c2aa98e2SPeter Wemm case O_DPI: 2627c2aa98e2SPeter Wemm DontProbeInterfaces = atobool(val); 2628c2aa98e2SPeter Wemm break; 2629c2aa98e2SPeter Wemm 2630c2aa98e2SPeter Wemm case O_MAXRCPT: 2631c2aa98e2SPeter Wemm MaxRcptPerMsg = atoi(val); 2632c2aa98e2SPeter Wemm break; 2633c2aa98e2SPeter Wemm 2634c2aa98e2SPeter Wemm case O_DEADLETTER: 2635c2aa98e2SPeter Wemm if (DeadLetterDrop != NULL) 2636c2aa98e2SPeter Wemm free(DeadLetterDrop); 2637c2aa98e2SPeter Wemm DeadLetterDrop = newstr(val); 2638c2aa98e2SPeter Wemm break; 2639c2aa98e2SPeter Wemm 2640c2aa98e2SPeter Wemm #if _FFR_DONTLOCKFILESFORREAD_OPTION 2641c2aa98e2SPeter Wemm case O_DONTLOCK: 2642c2aa98e2SPeter Wemm DontLockReadFiles = atobool(val); 2643c2aa98e2SPeter Wemm break; 264406f25ae9SGregory Neil Shapiro #endif /* _FFR_DONTLOCKFILESFORREAD_OPTION */ 2645c2aa98e2SPeter Wemm 2646c2aa98e2SPeter Wemm case O_MAXALIASRCSN: 2647c2aa98e2SPeter Wemm MaxAliasRecursion = atoi(val); 2648c2aa98e2SPeter Wemm break; 2649c2aa98e2SPeter Wemm 2650c2aa98e2SPeter Wemm case O_CNCTONLYTO: 2651c2aa98e2SPeter Wemm /* XXX should probably use gethostbyname */ 265206f25ae9SGregory Neil Shapiro #if NETINET || NETINET6 265306f25ae9SGregory Neil Shapiro # if NETINET6 265406f25ae9SGregory Neil Shapiro if (inet_addr(val) == INADDR_NONE) 265506f25ae9SGregory Neil Shapiro { 265606f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET6; 265706f25ae9SGregory Neil Shapiro if (inet_pton(AF_INET6, val, 265806f25ae9SGregory Neil Shapiro &ConnectOnlyTo.sin6.sin6_addr) != 1) 265906f25ae9SGregory Neil Shapiro syserr("readcf: option ConnectOnlyTo: invalid IP address %s", 266006f25ae9SGregory Neil Shapiro val); 266106f25ae9SGregory Neil Shapiro } 266206f25ae9SGregory Neil Shapiro else 266306f25ae9SGregory Neil Shapiro # endif /* NETINET6 */ 266406f25ae9SGregory Neil Shapiro { 266506f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET; 266606f25ae9SGregory Neil Shapiro ConnectOnlyTo.sin.sin_addr.s_addr = inet_addr(val); 266706f25ae9SGregory Neil Shapiro } 266806f25ae9SGregory Neil Shapiro #endif /* NETINET || NETINET6 */ 2669c2aa98e2SPeter Wemm break; 2670c2aa98e2SPeter Wemm 2671065a643dSPeter Wemm case O_TRUSTUSER: 267206f25ae9SGregory Neil Shapiro #if HASFCHOWN 2673c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2674065a643dSPeter Wemm TrustedUid = atoi(val); 2675c2aa98e2SPeter Wemm else 2676c2aa98e2SPeter Wemm { 2677c2aa98e2SPeter Wemm register struct passwd *pw; 2678c2aa98e2SPeter Wemm 2679065a643dSPeter Wemm TrustedUid = 0; 2680c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2681c2aa98e2SPeter Wemm if (pw == NULL) 2682065a643dSPeter Wemm syserr("readcf: option TrustedUser: unknown user %s", val); 2683c2aa98e2SPeter Wemm else 2684065a643dSPeter Wemm TrustedUid = pw->pw_uid; 2685c2aa98e2SPeter Wemm } 2686c2aa98e2SPeter Wemm 2687c2aa98e2SPeter Wemm # ifdef UID_MAX 2688065a643dSPeter Wemm if (TrustedUid > UID_MAX) 2689c2aa98e2SPeter Wemm { 2690065a643dSPeter Wemm syserr("readcf: option TrustedUser: uid value (%ld) > UID_MAX (%ld)", 2691065a643dSPeter Wemm TrustedUid, UID_MAX); 2692065a643dSPeter Wemm TrustedUid = 0; 2693c2aa98e2SPeter Wemm } 269406f25ae9SGregory Neil Shapiro # endif /* UID_MAX */ 269506f25ae9SGregory Neil Shapiro #else /* HASFCHOWN */ 269606f25ae9SGregory Neil Shapiro syserr("readcf: option TrustedUser: can not be used on systems which do not support fchown()"); 269706f25ae9SGregory Neil Shapiro #endif /* HASFCHOWN */ 2698c2aa98e2SPeter Wemm break; 2699c2aa98e2SPeter Wemm 2700065a643dSPeter Wemm case O_MAXMIMEHDRLEN: 2701065a643dSPeter Wemm p = strchr(val, '/'); 2702065a643dSPeter Wemm if (p != NULL) 2703065a643dSPeter Wemm *p++ = '\0'; 2704065a643dSPeter Wemm MaxMimeHeaderLength = atoi(val); 2705065a643dSPeter Wemm if (p != NULL && *p != '\0') 2706065a643dSPeter Wemm MaxMimeFieldLength = atoi(p); 2707065a643dSPeter Wemm else 2708065a643dSPeter Wemm MaxMimeFieldLength = MaxMimeHeaderLength / 2; 2709065a643dSPeter Wemm 2710065a643dSPeter Wemm if (MaxMimeHeaderLength < 0) 2711065a643dSPeter Wemm MaxMimeHeaderLength = 0; 2712065a643dSPeter Wemm else if (MaxMimeHeaderLength < 128) 2713065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: header length limit set lower than 128\n"); 2714065a643dSPeter Wemm 2715065a643dSPeter Wemm if (MaxMimeFieldLength < 0) 2716065a643dSPeter Wemm MaxMimeFieldLength = 0; 2717065a643dSPeter Wemm else if (MaxMimeFieldLength < 40) 2718065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: field length limit set lower than 40\n"); 2719065a643dSPeter Wemm break; 2720065a643dSPeter Wemm 2721065a643dSPeter Wemm case O_CONTROLSOCKET: 2722065a643dSPeter Wemm if (ControlSocketName != NULL) 2723065a643dSPeter Wemm free(ControlSocketName); 2724065a643dSPeter Wemm ControlSocketName = newstr(val); 2725065a643dSPeter Wemm break; 2726065a643dSPeter Wemm 27272e43090eSPeter Wemm case O_MAXHDRSLEN: 27282e43090eSPeter Wemm MaxHeadersLength = atoi(val); 272925bab6e9SPeter Wemm 27302e43090eSPeter Wemm if (MaxHeadersLength > 0 && 27312e43090eSPeter Wemm MaxHeadersLength < (MAXHDRSLEN / 2)) 273206f25ae9SGregory Neil Shapiro printf("Warning: MaxHeadersLength: headers length limit set lower than %d\n", (MAXHDRSLEN / 2)); 273325bab6e9SPeter Wemm break; 273406f25ae9SGregory Neil Shapiro 273506f25ae9SGregory Neil Shapiro case O_PROCTITLEPREFIX: 273606f25ae9SGregory Neil Shapiro if (ProcTitlePrefix != NULL) 273706f25ae9SGregory Neil Shapiro free(ProcTitlePrefix); 273806f25ae9SGregory Neil Shapiro ProcTitlePrefix = newstr(val); 273906f25ae9SGregory Neil Shapiro break; 274006f25ae9SGregory Neil Shapiro 274106f25ae9SGregory Neil Shapiro #if SASL 274206f25ae9SGregory Neil Shapiro case O_SASLINFO: 274306f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 274406f25ae9SGregory Neil Shapiro /* 274506f25ae9SGregory Neil Shapiro ** Allow users to select their own authinfo file. 274606f25ae9SGregory Neil Shapiro ** However, this is not a "perfect" solution. 274706f25ae9SGregory Neil Shapiro ** If mail is queued, the authentication info 274806f25ae9SGregory Neil Shapiro ** will not be used in subsequent delivery attempts. 274906f25ae9SGregory Neil Shapiro ** If we really want to support this, then it has 275006f25ae9SGregory Neil Shapiro ** to be stored in the queue file. 275106f25ae9SGregory Neil Shapiro */ 275206f25ae9SGregory Neil Shapiro if (!bitset(SUBMIT_MSA, SubmitMode) && RealUid != 0 && 275306f25ae9SGregory Neil Shapiro RunAsUid != RealUid) 275406f25ae9SGregory Neil Shapiro { 275506f25ae9SGregory Neil Shapiro errno = 0; 275606f25ae9SGregory Neil Shapiro syserr("Error: %s only allowed with -U\n", 275706f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 275806f25ae9SGregory Neil Shapiro ExitStat = EX_USAGE; 275906f25ae9SGregory Neil Shapiro break; 276006f25ae9SGregory Neil Shapiro } 276106f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 276206f25ae9SGregory Neil Shapiro if (SASLInfo != NULL) 276306f25ae9SGregory Neil Shapiro free(SASLInfo); 276406f25ae9SGregory Neil Shapiro SASLInfo = newstr(val); 276506f25ae9SGregory Neil Shapiro break; 276606f25ae9SGregory Neil Shapiro 276706f25ae9SGregory Neil Shapiro case O_SASLMECH: 276806f25ae9SGregory Neil Shapiro if (AuthMechanisms != NULL) 276906f25ae9SGregory Neil Shapiro free(AuthMechanisms); 277006f25ae9SGregory Neil Shapiro if (*val != '\0') 277106f25ae9SGregory Neil Shapiro AuthMechanisms = newstr(val); 277206f25ae9SGregory Neil Shapiro else 277306f25ae9SGregory Neil Shapiro AuthMechanisms = NULL; 277406f25ae9SGregory Neil Shapiro break; 277506f25ae9SGregory Neil Shapiro 277606f25ae9SGregory Neil Shapiro case O_SASLOPTS: 277706f25ae9SGregory Neil Shapiro while (val != NULL && *val != '\0') 277806f25ae9SGregory Neil Shapiro { 277906f25ae9SGregory Neil Shapiro switch(*val) 278006f25ae9SGregory Neil Shapiro { 278106f25ae9SGregory Neil Shapiro case 'A': 278206f25ae9SGregory Neil Shapiro SASLOpts |= SASL_AUTH_AUTH; 278306f25ae9SGregory Neil Shapiro break; 278406f25ae9SGregory Neil Shapiro # if _FFR_SASL_OPTS 278506f25ae9SGregory Neil Shapiro case 'a': 278606f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOACTIVE; 278706f25ae9SGregory Neil Shapiro break; 278806f25ae9SGregory Neil Shapiro case 'c': 278906f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_PASS_CREDENTIALS; 279006f25ae9SGregory Neil Shapiro break; 279106f25ae9SGregory Neil Shapiro case 'd': 279206f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NODICTIONARY; 279306f25ae9SGregory Neil Shapiro break; 279406f25ae9SGregory Neil Shapiro case 'f': 279506f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_FORWARD_SECRECY; 279606f25ae9SGregory Neil Shapiro break; 279706f25ae9SGregory Neil Shapiro case 'p': 279806f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOPLAINTEXT; 279906f25ae9SGregory Neil Shapiro break; 280006f25ae9SGregory Neil Shapiro case 'y': 280106f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOANONYMOUS; 280206f25ae9SGregory Neil Shapiro break; 280306f25ae9SGregory Neil Shapiro # endif /* _FFR_SASL_OPTS */ 280406f25ae9SGregory Neil Shapiro default: 280506f25ae9SGregory Neil Shapiro printf("Warning: Option: %s unknown parameter '%c'\n", 280606f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" 280706f25ae9SGregory Neil Shapiro : o->o_name, 280806f25ae9SGregory Neil Shapiro (isascii(*val) && isprint(*val)) ? *val 280906f25ae9SGregory Neil Shapiro : '?'); 281006f25ae9SGregory Neil Shapiro break; 281106f25ae9SGregory Neil Shapiro } 281206f25ae9SGregory Neil Shapiro ++val; 281306f25ae9SGregory Neil Shapiro val = strpbrk(val, ", \t"); 281406f25ae9SGregory Neil Shapiro if (val != NULL) 281506f25ae9SGregory Neil Shapiro ++val; 281606f25ae9SGregory Neil Shapiro } 281706f25ae9SGregory Neil Shapiro break; 281806f25ae9SGregory Neil Shapiro 281906f25ae9SGregory Neil Shapiro #else /* SASL */ 282006f25ae9SGregory Neil Shapiro case O_SASLINFO: 282106f25ae9SGregory Neil Shapiro case O_SASLMECH: 282206f25ae9SGregory Neil Shapiro case O_SASLOPTS: 282306f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires SASL support (-DSASL)\n", 282406f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 282506f25ae9SGregory Neil Shapiro break; 282606f25ae9SGregory Neil Shapiro #endif /* SASL */ 282706f25ae9SGregory Neil Shapiro 282806f25ae9SGregory Neil Shapiro #if STARTTLS 282906f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 283006f25ae9SGregory Neil Shapiro if (SrvCERTfile != NULL) 283106f25ae9SGregory Neil Shapiro free(SrvCERTfile); 283206f25ae9SGregory Neil Shapiro SrvCERTfile = newstr(val); 283306f25ae9SGregory Neil Shapiro break; 283406f25ae9SGregory Neil Shapiro 283506f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 283606f25ae9SGregory Neil Shapiro if (Srvkeyfile != NULL) 283706f25ae9SGregory Neil Shapiro free(Srvkeyfile); 283806f25ae9SGregory Neil Shapiro Srvkeyfile = newstr(val); 283906f25ae9SGregory Neil Shapiro break; 284006f25ae9SGregory Neil Shapiro 284106f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 284206f25ae9SGregory Neil Shapiro if (CltCERTfile != NULL) 284306f25ae9SGregory Neil Shapiro free(CltCERTfile); 284406f25ae9SGregory Neil Shapiro CltCERTfile = newstr(val); 284506f25ae9SGregory Neil Shapiro break; 284606f25ae9SGregory Neil Shapiro 284706f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 284806f25ae9SGregory Neil Shapiro if (Cltkeyfile != NULL) 284906f25ae9SGregory Neil Shapiro free(Cltkeyfile); 285006f25ae9SGregory Neil Shapiro Cltkeyfile = newstr(val); 285106f25ae9SGregory Neil Shapiro break; 285206f25ae9SGregory Neil Shapiro 285306f25ae9SGregory Neil Shapiro case O_CACERTFILE: 285406f25ae9SGregory Neil Shapiro if (CACERTfile != NULL) 285506f25ae9SGregory Neil Shapiro free(CACERTfile); 285606f25ae9SGregory Neil Shapiro CACERTfile = newstr(val); 285706f25ae9SGregory Neil Shapiro break; 285806f25ae9SGregory Neil Shapiro 285906f25ae9SGregory Neil Shapiro case O_CACERTPATH: 286006f25ae9SGregory Neil Shapiro if (CACERTpath != NULL) 286106f25ae9SGregory Neil Shapiro free(CACERTpath); 286206f25ae9SGregory Neil Shapiro CACERTpath = newstr(val); 286306f25ae9SGregory Neil Shapiro break; 286406f25ae9SGregory Neil Shapiro 286506f25ae9SGregory Neil Shapiro case O_DHPARAMS: 286606f25ae9SGregory Neil Shapiro if (DHParams != NULL) 286706f25ae9SGregory Neil Shapiro free(DHParams); 286806f25ae9SGregory Neil Shapiro DHParams = newstr(val); 286906f25ae9SGregory Neil Shapiro break; 287006f25ae9SGregory Neil Shapiro 287106f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 287206f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 287306f25ae9SGregory Neil Shapiro if (DHParams5 != NULL) 287406f25ae9SGregory Neil Shapiro free(DHParams5); 287506f25ae9SGregory Neil Shapiro DHParams5 = newstr(val); 287606f25ae9SGregory Neil Shapiro break; 287706f25ae9SGregory Neil Shapiro 287806f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 287906f25ae9SGregory Neil Shapiro if (CipherList != NULL) 288006f25ae9SGregory Neil Shapiro free(CipherList); 288106f25ae9SGregory Neil Shapiro CipherList = newstr(val); 288206f25ae9SGregory Neil Shapiro break; 288306f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 288406f25ae9SGregory Neil Shapiro 288506f25ae9SGregory Neil Shapiro case O_RANDFILE: 288606f25ae9SGregory Neil Shapiro if (RandFile != NULL) 288706f25ae9SGregory Neil Shapiro free(RandFile); 288806f25ae9SGregory Neil Shapiro RandFile= newstr(val); 288906f25ae9SGregory Neil Shapiro break; 289006f25ae9SGregory Neil Shapiro 289106f25ae9SGregory Neil Shapiro # else /* STARTTLS */ 289206f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 289306f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 289406f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 289506f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 289606f25ae9SGregory Neil Shapiro case O_CACERTFILE: 289706f25ae9SGregory Neil Shapiro case O_CACERTPATH: 289806f25ae9SGregory Neil Shapiro case O_DHPARAMS: 289906f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 290006f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 290106f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 290206f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 290306f25ae9SGregory Neil Shapiro case O_RANDFILE: 290406f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires TLS support\n", 290506f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 290606f25ae9SGregory Neil Shapiro break; 290706f25ae9SGregory Neil Shapiro 290806f25ae9SGregory Neil Shapiro # endif /* STARTTLS */ 290906f25ae9SGregory Neil Shapiro 291006f25ae9SGregory Neil Shapiro case O_CLIENTPORT: 291106f25ae9SGregory Neil Shapiro #if DAEMON 291206f25ae9SGregory Neil Shapiro setclientoptions(val); 291306f25ae9SGregory Neil Shapiro #else /* DAEMON */ 291406f25ae9SGregory Neil Shapiro syserr("ClientPortOptions (O option) set but DAEMON not compiled in"); 291506f25ae9SGregory Neil Shapiro #endif /* DAEMON */ 291606f25ae9SGregory Neil Shapiro break; 291706f25ae9SGregory Neil Shapiro 291806f25ae9SGregory Neil Shapiro case O_DF_BUFSIZE: 291906f25ae9SGregory Neil Shapiro DataFileBufferSize = atoi(val); 292006f25ae9SGregory Neil Shapiro break; 292106f25ae9SGregory Neil Shapiro 292206f25ae9SGregory Neil Shapiro case O_XF_BUFSIZE: 292306f25ae9SGregory Neil Shapiro XscriptFileBufferSize = atoi(val); 292406f25ae9SGregory Neil Shapiro break; 292506f25ae9SGregory Neil Shapiro 292606f25ae9SGregory Neil Shapiro case O_LDAPDEFAULTSPEC: 292706f25ae9SGregory Neil Shapiro #ifdef LDAPMAP 292806f25ae9SGregory Neil Shapiro ldapmap_set_defaults(val); 292906f25ae9SGregory Neil Shapiro #else /* LDAPMAP */ 293006f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires LDAP support (-DLDAPMAP)\n", 293106f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 293206f25ae9SGregory Neil Shapiro #endif /* LDAPMAP */ 293306f25ae9SGregory Neil Shapiro break; 293406f25ae9SGregory Neil Shapiro 293506f25ae9SGregory Neil Shapiro #if _FFR_MILTER 293606f25ae9SGregory Neil Shapiro case O_INPUTMILTER: 293706f25ae9SGregory Neil Shapiro InputFilterList = newstr(val); 293806f25ae9SGregory Neil Shapiro break; 293906f25ae9SGregory Neil Shapiro 294006f25ae9SGregory Neil Shapiro case O_MILTER: 294106f25ae9SGregory Neil Shapiro milter_set_option(subopt, val, sticky); 294206f25ae9SGregory Neil Shapiro break; 294306f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 294406f25ae9SGregory Neil Shapiro 294506f25ae9SGregory Neil Shapiro #if _FFR_QUEUE_FILE_MODE 294606f25ae9SGregory Neil Shapiro case O_QUEUE_FILE_MODE: /* queue file mode */ 294706f25ae9SGregory Neil Shapiro QueueFileMode = atooct(val) & 0777; 294806f25ae9SGregory Neil Shapiro break; 294906f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUE_FILE_MODE */ 295025bab6e9SPeter Wemm 2951c2aa98e2SPeter Wemm default: 2952c2aa98e2SPeter Wemm if (tTd(37, 1)) 2953c2aa98e2SPeter Wemm { 2954c2aa98e2SPeter Wemm if (isascii(opt) && isprint(opt)) 295506f25ae9SGregory Neil Shapiro dprintf("Warning: option %c unknown\n", opt); 2956c2aa98e2SPeter Wemm else 295706f25ae9SGregory Neil Shapiro dprintf("Warning: option 0x%x unknown\n", opt); 2958c2aa98e2SPeter Wemm } 2959c2aa98e2SPeter Wemm break; 2960c2aa98e2SPeter Wemm } 296106f25ae9SGregory Neil Shapiro 296206f25ae9SGregory Neil Shapiro /* 296306f25ae9SGregory Neil Shapiro ** Options with suboptions are responsible for taking care 296406f25ae9SGregory Neil Shapiro ** of sticky-ness (e.g., that a command line setting is kept 296506f25ae9SGregory Neil Shapiro ** when reading in the sendmail.cf file). This has to be done 296606f25ae9SGregory Neil Shapiro ** when the suboptions are parsed since each suboption must be 296706f25ae9SGregory Neil Shapiro ** sticky, not the root option. 296806f25ae9SGregory Neil Shapiro */ 296906f25ae9SGregory Neil Shapiro 297006f25ae9SGregory Neil Shapiro if (sticky && !bitset(OI_SUBOPT, o->o_flags)) 2971c2aa98e2SPeter Wemm setbitn(opt, StickyOpt); 2972c2aa98e2SPeter Wemm } 2973c2aa98e2SPeter Wemm /* 2974c2aa98e2SPeter Wemm ** SETCLASS -- set a string into a class 2975c2aa98e2SPeter Wemm ** 2976c2aa98e2SPeter Wemm ** Parameters: 2977c2aa98e2SPeter Wemm ** class -- the class to put the string in. 2978c2aa98e2SPeter Wemm ** str -- the string to enter 2979c2aa98e2SPeter Wemm ** 2980c2aa98e2SPeter Wemm ** Returns: 2981c2aa98e2SPeter Wemm ** none. 2982c2aa98e2SPeter Wemm ** 2983c2aa98e2SPeter Wemm ** Side Effects: 2984c2aa98e2SPeter Wemm ** puts the word into the symbol table. 2985c2aa98e2SPeter Wemm */ 2986c2aa98e2SPeter Wemm 2987c2aa98e2SPeter Wemm void 2988c2aa98e2SPeter Wemm setclass(class, str) 2989c2aa98e2SPeter Wemm int class; 2990c2aa98e2SPeter Wemm char *str; 2991c2aa98e2SPeter Wemm { 2992c2aa98e2SPeter Wemm register STAB *s; 2993c2aa98e2SPeter Wemm 299406f25ae9SGregory Neil Shapiro if ((*str & 0377) == MATCHCLASS) 299506f25ae9SGregory Neil Shapiro { 299606f25ae9SGregory Neil Shapiro int mid; 299706f25ae9SGregory Neil Shapiro 299806f25ae9SGregory Neil Shapiro str++; 299906f25ae9SGregory Neil Shapiro mid = macid(str, NULL); 300006f25ae9SGregory Neil Shapiro if (mid == '\0') 300106f25ae9SGregory Neil Shapiro return; 300206f25ae9SGregory Neil Shapiro 3003c2aa98e2SPeter Wemm if (tTd(37, 8)) 300406f25ae9SGregory Neil Shapiro dprintf("setclass(%s, $=%s)\n", 300506f25ae9SGregory Neil Shapiro macname(class), macname(mid)); 300606f25ae9SGregory Neil Shapiro copy_class(mid, class); 300706f25ae9SGregory Neil Shapiro } 300806f25ae9SGregory Neil Shapiro else 300906f25ae9SGregory Neil Shapiro { 301006f25ae9SGregory Neil Shapiro if (tTd(37, 8)) 301106f25ae9SGregory Neil Shapiro dprintf("setclass(%s, %s)\n", macname(class), str); 301206f25ae9SGregory Neil Shapiro 3013c2aa98e2SPeter Wemm s = stab(str, ST_CLASS, ST_ENTER); 3014c2aa98e2SPeter Wemm setbitn(class, s->s_class); 3015c2aa98e2SPeter Wemm } 301606f25ae9SGregory Neil Shapiro } 3017c2aa98e2SPeter Wemm /* 3018c2aa98e2SPeter Wemm ** MAKEMAPENTRY -- create a map entry 3019c2aa98e2SPeter Wemm ** 3020c2aa98e2SPeter Wemm ** Parameters: 3021c2aa98e2SPeter Wemm ** line -- the config file line 3022c2aa98e2SPeter Wemm ** 3023c2aa98e2SPeter Wemm ** Returns: 3024c2aa98e2SPeter Wemm ** A pointer to the map that has been created. 3025c2aa98e2SPeter Wemm ** NULL if there was a syntax error. 3026c2aa98e2SPeter Wemm ** 3027c2aa98e2SPeter Wemm ** Side Effects: 3028c2aa98e2SPeter Wemm ** Enters the map into the dictionary. 3029c2aa98e2SPeter Wemm */ 3030c2aa98e2SPeter Wemm 3031c2aa98e2SPeter Wemm MAP * 3032c2aa98e2SPeter Wemm makemapentry(line) 3033c2aa98e2SPeter Wemm char *line; 3034c2aa98e2SPeter Wemm { 3035c2aa98e2SPeter Wemm register char *p; 3036c2aa98e2SPeter Wemm char *mapname; 3037c2aa98e2SPeter Wemm char *classname; 3038c2aa98e2SPeter Wemm register STAB *s; 3039c2aa98e2SPeter Wemm STAB *class; 3040c2aa98e2SPeter Wemm 3041c2aa98e2SPeter Wemm for (p = line; isascii(*p) && isspace(*p); p++) 3042c2aa98e2SPeter Wemm continue; 3043c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3044c2aa98e2SPeter Wemm { 3045c2aa98e2SPeter Wemm syserr("readcf: config K line: no map name"); 3046c2aa98e2SPeter Wemm return NULL; 3047c2aa98e2SPeter Wemm } 3048c2aa98e2SPeter Wemm 3049c2aa98e2SPeter Wemm mapname = p; 3050c2aa98e2SPeter Wemm while ((isascii(*++p) && isalnum(*p)) || *p == '_' || *p == '.') 3051c2aa98e2SPeter Wemm continue; 3052c2aa98e2SPeter Wemm if (*p != '\0') 3053c2aa98e2SPeter Wemm *p++ = '\0'; 3054c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3055c2aa98e2SPeter Wemm p++; 3056c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3057c2aa98e2SPeter Wemm { 3058c2aa98e2SPeter Wemm syserr("readcf: config K line, map %s: no map class", mapname); 3059c2aa98e2SPeter Wemm return NULL; 3060c2aa98e2SPeter Wemm } 3061c2aa98e2SPeter Wemm classname = p; 3062c2aa98e2SPeter Wemm while (isascii(*++p) && isalnum(*p)) 3063c2aa98e2SPeter Wemm continue; 3064c2aa98e2SPeter Wemm if (*p != '\0') 3065c2aa98e2SPeter Wemm *p++ = '\0'; 3066c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3067c2aa98e2SPeter Wemm p++; 3068c2aa98e2SPeter Wemm 3069c2aa98e2SPeter Wemm /* look up the class */ 3070c2aa98e2SPeter Wemm class = stab(classname, ST_MAPCLASS, ST_FIND); 3071c2aa98e2SPeter Wemm if (class == NULL) 3072c2aa98e2SPeter Wemm { 3073c2aa98e2SPeter Wemm syserr("readcf: map %s: class %s not available", mapname, classname); 3074c2aa98e2SPeter Wemm return NULL; 3075c2aa98e2SPeter Wemm } 3076c2aa98e2SPeter Wemm 3077c2aa98e2SPeter Wemm /* enter the map */ 3078c2aa98e2SPeter Wemm s = stab(mapname, ST_MAP, ST_ENTER); 3079c2aa98e2SPeter Wemm s->s_map.map_class = &class->s_mapclass; 3080c2aa98e2SPeter Wemm s->s_map.map_mname = newstr(mapname); 3081c2aa98e2SPeter Wemm 3082c2aa98e2SPeter Wemm if (class->s_mapclass.map_parse(&s->s_map, p)) 3083c2aa98e2SPeter Wemm s->s_map.map_mflags |= MF_VALID; 3084c2aa98e2SPeter Wemm 3085c2aa98e2SPeter Wemm if (tTd(37, 5)) 3086c2aa98e2SPeter Wemm { 308706f25ae9SGregory Neil Shapiro dprintf("map %s, class %s, flags %lx, file %s,\n", 3088c2aa98e2SPeter Wemm s->s_map.map_mname, s->s_map.map_class->map_cname, 3089c2aa98e2SPeter Wemm s->s_map.map_mflags, 3090c2aa98e2SPeter Wemm s->s_map.map_file == NULL ? "(null)" : s->s_map.map_file); 309106f25ae9SGregory Neil Shapiro dprintf("\tapp %s, domain %s, rebuild %s\n", 3092c2aa98e2SPeter Wemm s->s_map.map_app == NULL ? "(null)" : s->s_map.map_app, 3093c2aa98e2SPeter Wemm s->s_map.map_domain == NULL ? "(null)" : s->s_map.map_domain, 3094c2aa98e2SPeter Wemm s->s_map.map_rebuild == NULL ? "(null)" : s->s_map.map_rebuild); 3095c2aa98e2SPeter Wemm } 3096c2aa98e2SPeter Wemm 3097c2aa98e2SPeter Wemm return &s->s_map; 3098c2aa98e2SPeter Wemm } 3099c2aa98e2SPeter Wemm /* 3100c2aa98e2SPeter Wemm ** STRTORWSET -- convert string to rewriting set number 3101c2aa98e2SPeter Wemm ** 3102c2aa98e2SPeter Wemm ** Parameters: 3103c2aa98e2SPeter Wemm ** p -- the pointer to the string to decode. 3104c2aa98e2SPeter Wemm ** endp -- if set, store the trailing delimiter here. 3105c2aa98e2SPeter Wemm ** stabmode -- ST_ENTER to create this entry, ST_FIND if 3106c2aa98e2SPeter Wemm ** it must already exist. 3107c2aa98e2SPeter Wemm ** 3108c2aa98e2SPeter Wemm ** Returns: 3109c2aa98e2SPeter Wemm ** The appropriate ruleset number. 3110c2aa98e2SPeter Wemm ** -1 if it is not valid (error already printed) 3111c2aa98e2SPeter Wemm */ 3112c2aa98e2SPeter Wemm 3113c2aa98e2SPeter Wemm int 3114c2aa98e2SPeter Wemm strtorwset(p, endp, stabmode) 3115c2aa98e2SPeter Wemm char *p; 3116c2aa98e2SPeter Wemm char **endp; 3117c2aa98e2SPeter Wemm int stabmode; 3118c2aa98e2SPeter Wemm { 3119c2aa98e2SPeter Wemm int ruleset; 3120c2aa98e2SPeter Wemm static int nextruleset = MAXRWSETS; 3121c2aa98e2SPeter Wemm 3122c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3123c2aa98e2SPeter Wemm p++; 3124c2aa98e2SPeter Wemm if (!isascii(*p)) 3125c2aa98e2SPeter Wemm { 3126c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", p); 3127c2aa98e2SPeter Wemm return -1; 3128c2aa98e2SPeter Wemm } 3129c2aa98e2SPeter Wemm if (isdigit(*p)) 3130c2aa98e2SPeter Wemm { 3131c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3132c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3133c2aa98e2SPeter Wemm { 3134c2aa98e2SPeter Wemm syserr("bad ruleset %d (%d max)", 3135c2aa98e2SPeter Wemm ruleset, MAXRWSETS / 2); 3136c2aa98e2SPeter Wemm ruleset = -1; 3137c2aa98e2SPeter Wemm } 3138c2aa98e2SPeter Wemm } 3139c2aa98e2SPeter Wemm else 3140c2aa98e2SPeter Wemm { 3141c2aa98e2SPeter Wemm STAB *s; 3142c2aa98e2SPeter Wemm char delim; 314306f25ae9SGregory Neil Shapiro char *q = NULL; 3144c2aa98e2SPeter Wemm 3145c2aa98e2SPeter Wemm q = p; 3146c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && 3147c2aa98e2SPeter Wemm (isalnum(*p) || *p == '_')) 3148c2aa98e2SPeter Wemm p++; 3149c2aa98e2SPeter Wemm if (q == p || !(isascii(*q) && isalpha(*q))) 3150c2aa98e2SPeter Wemm { 3151c2aa98e2SPeter Wemm /* no valid characters */ 3152c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", q); 3153c2aa98e2SPeter Wemm return -1; 3154c2aa98e2SPeter Wemm } 3155c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3156c2aa98e2SPeter Wemm *p++ = '\0'; 3157c2aa98e2SPeter Wemm delim = *p; 3158c2aa98e2SPeter Wemm if (delim != '\0') 3159c2aa98e2SPeter Wemm *p = '\0'; 3160c2aa98e2SPeter Wemm s = stab(q, ST_RULESET, stabmode); 3161c2aa98e2SPeter Wemm if (delim != '\0') 3162c2aa98e2SPeter Wemm *p = delim; 3163c2aa98e2SPeter Wemm 3164c2aa98e2SPeter Wemm if (s == NULL) 3165c2aa98e2SPeter Wemm return -1; 3166c2aa98e2SPeter Wemm 3167c2aa98e2SPeter Wemm if (stabmode == ST_ENTER && delim == '=') 3168c2aa98e2SPeter Wemm { 3169c2aa98e2SPeter Wemm while (isascii(*++p) && isspace(*p)) 3170c2aa98e2SPeter Wemm continue; 3171c2aa98e2SPeter Wemm if (!(isascii(*p) && isdigit(*p))) 3172c2aa98e2SPeter Wemm { 3173c2aa98e2SPeter Wemm syserr("bad ruleset definition \"%s\" (number required after `=')", q); 3174c2aa98e2SPeter Wemm ruleset = -1; 3175c2aa98e2SPeter Wemm } 3176c2aa98e2SPeter Wemm else 3177c2aa98e2SPeter Wemm { 3178c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3179c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3180c2aa98e2SPeter Wemm { 3181c2aa98e2SPeter Wemm syserr("bad ruleset number %d in \"%s\" (%d max)", 3182c2aa98e2SPeter Wemm ruleset, q, MAXRWSETS / 2); 3183c2aa98e2SPeter Wemm ruleset = -1; 3184c2aa98e2SPeter Wemm } 3185c2aa98e2SPeter Wemm } 3186c2aa98e2SPeter Wemm } 3187c2aa98e2SPeter Wemm else 3188c2aa98e2SPeter Wemm { 3189c2aa98e2SPeter Wemm if (endp != NULL) 3190c2aa98e2SPeter Wemm *endp = p; 319106f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0) 3192c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3193c2aa98e2SPeter Wemm else if ((ruleset = --nextruleset) < MAXRWSETS / 2) 3194c2aa98e2SPeter Wemm { 3195c2aa98e2SPeter Wemm syserr("%s: too many named rulesets (%d max)", 3196c2aa98e2SPeter Wemm q, MAXRWSETS / 2); 3197c2aa98e2SPeter Wemm ruleset = -1; 3198c2aa98e2SPeter Wemm } 3199c2aa98e2SPeter Wemm } 320006f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0 && 320106f25ae9SGregory Neil Shapiro ruleset >= 0 && 320206f25ae9SGregory Neil Shapiro ruleset != s->s_ruleset) 3203c2aa98e2SPeter Wemm { 3204c2aa98e2SPeter Wemm syserr("%s: ruleset changed value (old %d, new %d)", 3205c2aa98e2SPeter Wemm q, s->s_ruleset, ruleset); 3206c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3207c2aa98e2SPeter Wemm } 320806f25ae9SGregory Neil Shapiro else if (ruleset >= 0) 3209c2aa98e2SPeter Wemm { 3210c2aa98e2SPeter Wemm s->s_ruleset = ruleset; 3211c2aa98e2SPeter Wemm } 321206f25ae9SGregory Neil Shapiro if (stabmode == ST_ENTER) 321306f25ae9SGregory Neil Shapiro { 321406f25ae9SGregory Neil Shapiro char *h = NULL; 321506f25ae9SGregory Neil Shapiro 321606f25ae9SGregory Neil Shapiro if (RuleSetNames[ruleset] != NULL) 321706f25ae9SGregory Neil Shapiro free(RuleSetNames[ruleset]); 321806f25ae9SGregory Neil Shapiro if (delim != '\0' && (h = strchr(q, delim)) != NULL) 321906f25ae9SGregory Neil Shapiro *h = '\0'; 322006f25ae9SGregory Neil Shapiro RuleSetNames[ruleset] = newstr(q); 322106f25ae9SGregory Neil Shapiro if (delim == '/' && h != NULL) 322206f25ae9SGregory Neil Shapiro *h = delim; /* put back delim */ 322306f25ae9SGregory Neil Shapiro } 3224c2aa98e2SPeter Wemm } 3225c2aa98e2SPeter Wemm return ruleset; 3226c2aa98e2SPeter Wemm } 3227c2aa98e2SPeter Wemm /* 322806f25ae9SGregory Neil Shapiro ** SETTIMEOUT -- set an individual timeout 322906f25ae9SGregory Neil Shapiro ** 323006f25ae9SGregory Neil Shapiro ** Parameters: 323106f25ae9SGregory Neil Shapiro ** name -- the name of the timeout. 323206f25ae9SGregory Neil Shapiro ** val -- the value of the timeout. 323306f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 323406f25ae9SGregory Neil Shapiro ** this value. 323506f25ae9SGregory Neil Shapiro ** 323606f25ae9SGregory Neil Shapiro ** Returns: 323706f25ae9SGregory Neil Shapiro ** none. 323806f25ae9SGregory Neil Shapiro */ 323906f25ae9SGregory Neil Shapiro 324006f25ae9SGregory Neil Shapiro /* set if Timeout sub-option is stuck */ 324106f25ae9SGregory Neil Shapiro static BITMAP256 StickyTimeoutOpt; 324206f25ae9SGregory Neil Shapiro 324306f25ae9SGregory Neil Shapiro static struct timeoutinfo 324406f25ae9SGregory Neil Shapiro { 324506f25ae9SGregory Neil Shapiro char *to_name; /* long name of timeout */ 324606f25ae9SGregory Neil Shapiro u_char to_code; /* code for option */ 324706f25ae9SGregory Neil Shapiro } TimeOutTab[] = 324806f25ae9SGregory Neil Shapiro { 324906f25ae9SGregory Neil Shapiro #define TO_INITIAL 0x01 325006f25ae9SGregory Neil Shapiro { "initial", TO_INITIAL }, 325106f25ae9SGregory Neil Shapiro #define TO_MAIL 0x02 325206f25ae9SGregory Neil Shapiro { "mail", TO_MAIL }, 325306f25ae9SGregory Neil Shapiro #define TO_RCPT 0x03 325406f25ae9SGregory Neil Shapiro { "rcpt", TO_RCPT }, 325506f25ae9SGregory Neil Shapiro #define TO_DATAINIT 0x04 325606f25ae9SGregory Neil Shapiro { "datainit", TO_DATAINIT }, 325706f25ae9SGregory Neil Shapiro #define TO_DATABLOCK 0x05 325806f25ae9SGregory Neil Shapiro { "datablock", TO_DATABLOCK }, 325906f25ae9SGregory Neil Shapiro #define TO_DATAFINAL 0x06 326006f25ae9SGregory Neil Shapiro { "datafinal", TO_DATAFINAL }, 326106f25ae9SGregory Neil Shapiro #define TO_COMMAND 0x07 326206f25ae9SGregory Neil Shapiro { "command", TO_COMMAND }, 326306f25ae9SGregory Neil Shapiro #define TO_RSET 0x08 326406f25ae9SGregory Neil Shapiro { "rset", TO_RSET }, 326506f25ae9SGregory Neil Shapiro #define TO_HELO 0x09 326606f25ae9SGregory Neil Shapiro { "helo", TO_HELO }, 326706f25ae9SGregory Neil Shapiro #define TO_QUIT 0x0A 326806f25ae9SGregory Neil Shapiro { "quit", TO_QUIT }, 326906f25ae9SGregory Neil Shapiro #define TO_MISC 0x0B 327006f25ae9SGregory Neil Shapiro { "misc", TO_MISC }, 327106f25ae9SGregory Neil Shapiro #define TO_IDENT 0x0C 327206f25ae9SGregory Neil Shapiro { "ident", TO_IDENT }, 327306f25ae9SGregory Neil Shapiro #define TO_FILEOPEN 0x0D 327406f25ae9SGregory Neil Shapiro { "fileopen", TO_FILEOPEN }, 327506f25ae9SGregory Neil Shapiro #define TO_CONNECT 0x0E 327606f25ae9SGregory Neil Shapiro { "connect", TO_CONNECT }, 327706f25ae9SGregory Neil Shapiro #define TO_ICONNECT 0x0F 327806f25ae9SGregory Neil Shapiro { "iconnect", TO_ICONNECT }, 327906f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN 0x10 328006f25ae9SGregory Neil Shapiro { "queuewarn", TO_QUEUEWARN }, 328106f25ae9SGregory Neil Shapiro { "queuewarn.*", TO_QUEUEWARN }, 328206f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NORMAL 0x11 328306f25ae9SGregory Neil Shapiro { "queuewarn.normal", TO_QUEUEWARN_NORMAL }, 328406f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_URGENT 0x12 328506f25ae9SGregory Neil Shapiro { "queuewarn.urgent", TO_QUEUEWARN_URGENT }, 328606f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NON_URGENT 0x13 328706f25ae9SGregory Neil Shapiro { "queuewarn.non-urgent", TO_QUEUEWARN_NON_URGENT }, 328806f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN 0x14 328906f25ae9SGregory Neil Shapiro { "queuereturn", TO_QUEUERETURN }, 329006f25ae9SGregory Neil Shapiro { "queuereturn.*", TO_QUEUERETURN }, 329106f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NORMAL 0x15 329206f25ae9SGregory Neil Shapiro { "queuereturn.normal", TO_QUEUERETURN_NORMAL }, 329306f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_URGENT 0x16 329406f25ae9SGregory Neil Shapiro { "queuereturn.urgent", TO_QUEUERETURN_URGENT }, 329506f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NON_URGENT 0x17 329606f25ae9SGregory Neil Shapiro { "queuereturn.non-urgent", TO_QUEUERETURN_NON_URGENT }, 329706f25ae9SGregory Neil Shapiro #define TO_HOSTSTATUS 0x18 329806f25ae9SGregory Neil Shapiro { "hoststatus", TO_HOSTSTATUS }, 329906f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS 0x19 330006f25ae9SGregory Neil Shapiro { "resolver.retrans", TO_RESOLVER_RETRANS }, 330106f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_NORMAL 0x1A 330206f25ae9SGregory Neil Shapiro { "resolver.retrans.normal", TO_RESOLVER_RETRANS_NORMAL }, 330306f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_FIRST 0x1B 330406f25ae9SGregory Neil Shapiro { "resolver.retrans.first", TO_RESOLVER_RETRANS_FIRST }, 330506f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY 0x1C 330606f25ae9SGregory Neil Shapiro { "resolver.retry", TO_RESOLVER_RETRY }, 330706f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_NORMAL 0x1D 330806f25ae9SGregory Neil Shapiro { "resolver.retry.normal", TO_RESOLVER_RETRY_NORMAL }, 330906f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_FIRST 0x1E 331006f25ae9SGregory Neil Shapiro { "resolver.retry.first", TO_RESOLVER_RETRY_FIRST }, 331106f25ae9SGregory Neil Shapiro #define TO_CONTROL 0x1F 331206f25ae9SGregory Neil Shapiro { "control", TO_CONTROL }, 331306f25ae9SGregory Neil Shapiro { NULL, 0 }, 331406f25ae9SGregory Neil Shapiro }; 331506f25ae9SGregory Neil Shapiro 331606f25ae9SGregory Neil Shapiro 331706f25ae9SGregory Neil Shapiro static void 331806f25ae9SGregory Neil Shapiro settimeout(name, val, sticky) 331906f25ae9SGregory Neil Shapiro char *name; 332006f25ae9SGregory Neil Shapiro char *val; 332106f25ae9SGregory Neil Shapiro bool sticky; 332206f25ae9SGregory Neil Shapiro { 332306f25ae9SGregory Neil Shapiro register struct timeoutinfo *to; 332406f25ae9SGregory Neil Shapiro int i; 332506f25ae9SGregory Neil Shapiro time_t toval; 332606f25ae9SGregory Neil Shapiro 332706f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 332806f25ae9SGregory Neil Shapiro dprintf("settimeout(%s = %s)", name, val); 332906f25ae9SGregory Neil Shapiro 333006f25ae9SGregory Neil Shapiro for (to = TimeOutTab; to->to_name != NULL; to++) 333106f25ae9SGregory Neil Shapiro { 333206f25ae9SGregory Neil Shapiro if (strcasecmp(to->to_name, name) == 0) 333306f25ae9SGregory Neil Shapiro break; 333406f25ae9SGregory Neil Shapiro } 333506f25ae9SGregory Neil Shapiro 333606f25ae9SGregory Neil Shapiro if (to->to_name == NULL) 333706f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 333806f25ae9SGregory Neil Shapiro 333906f25ae9SGregory Neil Shapiro /* 334006f25ae9SGregory Neil Shapiro ** See if this option is preset for us. 334106f25ae9SGregory Neil Shapiro */ 334206f25ae9SGregory Neil Shapiro 334306f25ae9SGregory Neil Shapiro if (!sticky && bitnset(to->to_code, StickyTimeoutOpt)) 334406f25ae9SGregory Neil Shapiro { 334506f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 334606f25ae9SGregory Neil Shapiro dprintf(" (ignored)\n"); 334706f25ae9SGregory Neil Shapiro return; 334806f25ae9SGregory Neil Shapiro } 334906f25ae9SGregory Neil Shapiro 335006f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 335106f25ae9SGregory Neil Shapiro dprintf("\n"); 335206f25ae9SGregory Neil Shapiro 335306f25ae9SGregory Neil Shapiro toval = convtime(val, 'm'); 335406f25ae9SGregory Neil Shapiro 335506f25ae9SGregory Neil Shapiro switch (to->to_code) 335606f25ae9SGregory Neil Shapiro { 335706f25ae9SGregory Neil Shapiro case TO_INITIAL: 335806f25ae9SGregory Neil Shapiro TimeOuts.to_initial = toval; 335906f25ae9SGregory Neil Shapiro break; 336006f25ae9SGregory Neil Shapiro 336106f25ae9SGregory Neil Shapiro case TO_MAIL: 336206f25ae9SGregory Neil Shapiro TimeOuts.to_mail = toval; 336306f25ae9SGregory Neil Shapiro break; 336406f25ae9SGregory Neil Shapiro 336506f25ae9SGregory Neil Shapiro case TO_RCPT: 336606f25ae9SGregory Neil Shapiro TimeOuts.to_rcpt = toval; 336706f25ae9SGregory Neil Shapiro break; 336806f25ae9SGregory Neil Shapiro 336906f25ae9SGregory Neil Shapiro case TO_DATAINIT: 337006f25ae9SGregory Neil Shapiro TimeOuts.to_datainit = toval; 337106f25ae9SGregory Neil Shapiro break; 337206f25ae9SGregory Neil Shapiro 337306f25ae9SGregory Neil Shapiro case TO_DATABLOCK: 337406f25ae9SGregory Neil Shapiro TimeOuts.to_datablock = toval; 337506f25ae9SGregory Neil Shapiro break; 337606f25ae9SGregory Neil Shapiro 337706f25ae9SGregory Neil Shapiro case TO_DATAFINAL: 337806f25ae9SGregory Neil Shapiro TimeOuts.to_datafinal = toval; 337906f25ae9SGregory Neil Shapiro break; 338006f25ae9SGregory Neil Shapiro 338106f25ae9SGregory Neil Shapiro case TO_COMMAND: 338206f25ae9SGregory Neil Shapiro TimeOuts.to_nextcommand = toval; 338306f25ae9SGregory Neil Shapiro break; 338406f25ae9SGregory Neil Shapiro 338506f25ae9SGregory Neil Shapiro case TO_RSET: 338606f25ae9SGregory Neil Shapiro TimeOuts.to_rset = toval; 338706f25ae9SGregory Neil Shapiro break; 338806f25ae9SGregory Neil Shapiro 338906f25ae9SGregory Neil Shapiro case TO_HELO: 339006f25ae9SGregory Neil Shapiro TimeOuts.to_helo = toval; 339106f25ae9SGregory Neil Shapiro break; 339206f25ae9SGregory Neil Shapiro 339306f25ae9SGregory Neil Shapiro case TO_QUIT: 339406f25ae9SGregory Neil Shapiro TimeOuts.to_quit = toval; 339506f25ae9SGregory Neil Shapiro break; 339606f25ae9SGregory Neil Shapiro 339706f25ae9SGregory Neil Shapiro case TO_MISC: 339806f25ae9SGregory Neil Shapiro TimeOuts.to_miscshort = toval; 339906f25ae9SGregory Neil Shapiro break; 340006f25ae9SGregory Neil Shapiro 340106f25ae9SGregory Neil Shapiro case TO_IDENT: 340206f25ae9SGregory Neil Shapiro TimeOuts.to_ident = toval; 340306f25ae9SGregory Neil Shapiro break; 340406f25ae9SGregory Neil Shapiro 340506f25ae9SGregory Neil Shapiro case TO_FILEOPEN: 340606f25ae9SGregory Neil Shapiro TimeOuts.to_fileopen = toval; 340706f25ae9SGregory Neil Shapiro break; 340806f25ae9SGregory Neil Shapiro 340906f25ae9SGregory Neil Shapiro case TO_CONNECT: 341006f25ae9SGregory Neil Shapiro TimeOuts.to_connect = toval; 341106f25ae9SGregory Neil Shapiro break; 341206f25ae9SGregory Neil Shapiro 341306f25ae9SGregory Neil Shapiro case TO_ICONNECT: 341406f25ae9SGregory Neil Shapiro TimeOuts.to_iconnect = toval; 341506f25ae9SGregory Neil Shapiro break; 341606f25ae9SGregory Neil Shapiro 341706f25ae9SGregory Neil Shapiro case TO_QUEUEWARN: 341806f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 341906f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 342006f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 342106f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 342206f25ae9SGregory Neil Shapiro break; 342306f25ae9SGregory Neil Shapiro 342406f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NORMAL: 342506f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 342606f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 342706f25ae9SGregory Neil Shapiro break; 342806f25ae9SGregory Neil Shapiro 342906f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_URGENT: 343006f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 343106f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 343206f25ae9SGregory Neil Shapiro break; 343306f25ae9SGregory Neil Shapiro 343406f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NON_URGENT: 343506f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 343606f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 343706f25ae9SGregory Neil Shapiro break; 343806f25ae9SGregory Neil Shapiro 343906f25ae9SGregory Neil Shapiro case TO_QUEUERETURN: 344006f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 344106f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 344206f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 344306f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 344406f25ae9SGregory Neil Shapiro break; 344506f25ae9SGregory Neil Shapiro 344606f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NORMAL: 344706f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 344806f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 344906f25ae9SGregory Neil Shapiro break; 345006f25ae9SGregory Neil Shapiro 345106f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_URGENT: 345206f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 345306f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 345406f25ae9SGregory Neil Shapiro break; 345506f25ae9SGregory Neil Shapiro 345606f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NON_URGENT: 345706f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 345806f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 345906f25ae9SGregory Neil Shapiro break; 346006f25ae9SGregory Neil Shapiro 346106f25ae9SGregory Neil Shapiro 346206f25ae9SGregory Neil Shapiro case TO_HOSTSTATUS: 346306f25ae9SGregory Neil Shapiro MciInfoTimeout = toval; 346406f25ae9SGregory Neil Shapiro break; 346506f25ae9SGregory Neil Shapiro 346606f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS: 346706f25ae9SGregory Neil Shapiro toval = convtime(val, 's'); 346806f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_DEFAULT] = toval; 346906f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = toval; 347006f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = toval; 347106f25ae9SGregory Neil Shapiro break; 347206f25ae9SGregory Neil Shapiro 347306f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY: 347406f25ae9SGregory Neil Shapiro i = atoi(val); 347506f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_DEFAULT] = i; 347606f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = i; 347706f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = i; 347806f25ae9SGregory Neil Shapiro break; 347906f25ae9SGregory Neil Shapiro 348006f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_NORMAL: 348106f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = convtime(val, 's'); 348206f25ae9SGregory Neil Shapiro break; 348306f25ae9SGregory Neil Shapiro 348406f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_NORMAL: 348506f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = atoi(val); 348606f25ae9SGregory Neil Shapiro break; 348706f25ae9SGregory Neil Shapiro 348806f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_FIRST: 348906f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = convtime(val, 's'); 349006f25ae9SGregory Neil Shapiro break; 349106f25ae9SGregory Neil Shapiro 349206f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_FIRST: 349306f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = atoi(val); 349406f25ae9SGregory Neil Shapiro break; 349506f25ae9SGregory Neil Shapiro 349606f25ae9SGregory Neil Shapiro case TO_CONTROL: 349706f25ae9SGregory Neil Shapiro TimeOuts.to_control = toval; 349806f25ae9SGregory Neil Shapiro break; 349906f25ae9SGregory Neil Shapiro 350006f25ae9SGregory Neil Shapiro default: 350106f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 350206f25ae9SGregory Neil Shapiro break; 350306f25ae9SGregory Neil Shapiro } 350406f25ae9SGregory Neil Shapiro 350506f25ae9SGregory Neil Shapiro if (sticky) 350606f25ae9SGregory Neil Shapiro setbitn(to->to_code, StickyTimeoutOpt); 350706f25ae9SGregory Neil Shapiro } 350806f25ae9SGregory Neil Shapiro /* 3509c2aa98e2SPeter Wemm ** INITTIMEOUTS -- parse and set timeout values 3510c2aa98e2SPeter Wemm ** 3511c2aa98e2SPeter Wemm ** Parameters: 3512c2aa98e2SPeter Wemm ** val -- a pointer to the values. If NULL, do initial 3513c2aa98e2SPeter Wemm ** settings. 351406f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 351506f25ae9SGregory Neil Shapiro ** this suboption value. 3516c2aa98e2SPeter Wemm ** 3517c2aa98e2SPeter Wemm ** Returns: 3518c2aa98e2SPeter Wemm ** none. 3519c2aa98e2SPeter Wemm ** 3520c2aa98e2SPeter Wemm ** Side Effects: 3521c2aa98e2SPeter Wemm ** Initializes the TimeOuts structure 3522c2aa98e2SPeter Wemm */ 3523c2aa98e2SPeter Wemm 3524c2aa98e2SPeter Wemm void 352506f25ae9SGregory Neil Shapiro inittimeouts(val, sticky) 3526c2aa98e2SPeter Wemm register char *val; 352706f25ae9SGregory Neil Shapiro bool sticky; 3528c2aa98e2SPeter Wemm { 3529c2aa98e2SPeter Wemm register char *p; 3530c2aa98e2SPeter Wemm 3531c2aa98e2SPeter Wemm if (tTd(37, 2)) 353206f25ae9SGregory Neil Shapiro dprintf("inittimeouts(%s)\n", val == NULL ? "<NULL>" : val); 3533c2aa98e2SPeter Wemm if (val == NULL) 3534c2aa98e2SPeter Wemm { 3535c2aa98e2SPeter Wemm TimeOuts.to_connect = (time_t) 0 SECONDS; 3536c2aa98e2SPeter Wemm TimeOuts.to_initial = (time_t) 5 MINUTES; 3537c2aa98e2SPeter Wemm TimeOuts.to_helo = (time_t) 5 MINUTES; 3538c2aa98e2SPeter Wemm TimeOuts.to_mail = (time_t) 10 MINUTES; 3539c2aa98e2SPeter Wemm TimeOuts.to_rcpt = (time_t) 1 HOUR; 3540c2aa98e2SPeter Wemm TimeOuts.to_datainit = (time_t) 5 MINUTES; 3541c2aa98e2SPeter Wemm TimeOuts.to_datablock = (time_t) 1 HOUR; 3542c2aa98e2SPeter Wemm TimeOuts.to_datafinal = (time_t) 1 HOUR; 3543c2aa98e2SPeter Wemm TimeOuts.to_rset = (time_t) 5 MINUTES; 3544c2aa98e2SPeter Wemm TimeOuts.to_quit = (time_t) 2 MINUTES; 3545c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = (time_t) 1 HOUR; 3546c2aa98e2SPeter Wemm TimeOuts.to_miscshort = (time_t) 2 MINUTES; 3547c2aa98e2SPeter Wemm #if IDENTPROTO 354806f25ae9SGregory Neil Shapiro TimeOuts.to_ident = (time_t) 5 SECONDS; 354906f25ae9SGregory Neil Shapiro #else /* IDENTPROTO */ 3550c2aa98e2SPeter Wemm TimeOuts.to_ident = (time_t) 0 SECONDS; 355106f25ae9SGregory Neil Shapiro #endif /* IDENTPROTO */ 3552c2aa98e2SPeter Wemm TimeOuts.to_fileopen = (time_t) 60 SECONDS; 355306f25ae9SGregory Neil Shapiro TimeOuts.to_control = (time_t) 2 MINUTES; 3554c2aa98e2SPeter Wemm if (tTd(37, 5)) 3555c2aa98e2SPeter Wemm { 355606f25ae9SGregory Neil Shapiro dprintf("Timeouts:\n"); 355706f25ae9SGregory Neil Shapiro dprintf(" connect = %ld\n", (long)TimeOuts.to_connect); 355806f25ae9SGregory Neil Shapiro dprintf(" initial = %ld\n", (long)TimeOuts.to_initial); 355906f25ae9SGregory Neil Shapiro dprintf(" helo = %ld\n", (long)TimeOuts.to_helo); 356006f25ae9SGregory Neil Shapiro dprintf(" mail = %ld\n", (long)TimeOuts.to_mail); 356106f25ae9SGregory Neil Shapiro dprintf(" rcpt = %ld\n", (long)TimeOuts.to_rcpt); 356206f25ae9SGregory Neil Shapiro dprintf(" datainit = %ld\n", (long)TimeOuts.to_datainit); 356306f25ae9SGregory Neil Shapiro dprintf(" datablock = %ld\n", (long)TimeOuts.to_datablock); 356406f25ae9SGregory Neil Shapiro dprintf(" datafinal = %ld\n", (long)TimeOuts.to_datafinal); 356506f25ae9SGregory Neil Shapiro dprintf(" rset = %ld\n", (long)TimeOuts.to_rset); 356606f25ae9SGregory Neil Shapiro dprintf(" quit = %ld\n", (long)TimeOuts.to_quit); 356706f25ae9SGregory Neil Shapiro dprintf(" nextcommand = %ld\n", (long)TimeOuts.to_nextcommand); 356806f25ae9SGregory Neil Shapiro dprintf(" miscshort = %ld\n", (long)TimeOuts.to_miscshort); 356906f25ae9SGregory Neil Shapiro dprintf(" ident = %ld\n", (long)TimeOuts.to_ident); 357006f25ae9SGregory Neil Shapiro dprintf(" fileopen = %ld\n", (long)TimeOuts.to_fileopen); 357106f25ae9SGregory Neil Shapiro dprintf(" control = %ld\n", (long)TimeOuts.to_control); 3572c2aa98e2SPeter Wemm } 3573c2aa98e2SPeter Wemm return; 3574c2aa98e2SPeter Wemm } 3575c2aa98e2SPeter Wemm 3576c2aa98e2SPeter Wemm for (;; val = p) 3577c2aa98e2SPeter Wemm { 3578c2aa98e2SPeter Wemm while (isascii(*val) && isspace(*val)) 3579c2aa98e2SPeter Wemm val++; 3580c2aa98e2SPeter Wemm if (*val == '\0') 3581c2aa98e2SPeter Wemm break; 3582c2aa98e2SPeter Wemm for (p = val; *p != '\0' && *p != ','; p++) 3583c2aa98e2SPeter Wemm continue; 3584c2aa98e2SPeter Wemm if (*p != '\0') 3585c2aa98e2SPeter Wemm *p++ = '\0'; 3586c2aa98e2SPeter Wemm 3587c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 3588c2aa98e2SPeter Wemm { 3589c2aa98e2SPeter Wemm /* old syntax -- set everything */ 3590c2aa98e2SPeter Wemm TimeOuts.to_mail = convtime(val, 'm'); 3591c2aa98e2SPeter Wemm TimeOuts.to_rcpt = TimeOuts.to_mail; 3592c2aa98e2SPeter Wemm TimeOuts.to_datainit = TimeOuts.to_mail; 3593c2aa98e2SPeter Wemm TimeOuts.to_datablock = TimeOuts.to_mail; 3594c2aa98e2SPeter Wemm TimeOuts.to_datafinal = TimeOuts.to_mail; 3595c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = TimeOuts.to_mail; 359606f25ae9SGregory Neil Shapiro if (sticky) 359706f25ae9SGregory Neil Shapiro { 359806f25ae9SGregory Neil Shapiro setbitn(TO_MAIL, StickyTimeoutOpt); 359906f25ae9SGregory Neil Shapiro setbitn(TO_RCPT, StickyTimeoutOpt); 360006f25ae9SGregory Neil Shapiro setbitn(TO_DATAINIT, StickyTimeoutOpt); 360106f25ae9SGregory Neil Shapiro setbitn(TO_DATABLOCK, StickyTimeoutOpt); 360206f25ae9SGregory Neil Shapiro setbitn(TO_DATAFINAL, StickyTimeoutOpt); 360306f25ae9SGregory Neil Shapiro setbitn(TO_COMMAND, StickyTimeoutOpt); 360406f25ae9SGregory Neil Shapiro } 3605c2aa98e2SPeter Wemm continue; 3606c2aa98e2SPeter Wemm } 3607c2aa98e2SPeter Wemm else 3608c2aa98e2SPeter Wemm { 3609c2aa98e2SPeter Wemm register char *q = strchr(val, ':'); 3610c2aa98e2SPeter Wemm 3611c2aa98e2SPeter Wemm if (q == NULL && (q = strchr(val, '=')) == NULL) 3612c2aa98e2SPeter Wemm { 3613c2aa98e2SPeter Wemm /* syntax error */ 3614c2aa98e2SPeter Wemm continue; 3615c2aa98e2SPeter Wemm } 3616c2aa98e2SPeter Wemm *q++ = '\0'; 361706f25ae9SGregory Neil Shapiro settimeout(val, q, sticky); 3618c2aa98e2SPeter Wemm } 3619c2aa98e2SPeter Wemm } 3620c2aa98e2SPeter Wemm } 3621