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 1506f25ae9SGregory Neil Shapiro static char id[] = "@(#)$Id: readcf.c,v 8.382.4.14 2000/07/12 00:00:27 geir 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 { 113706f25ae9SGregory Neil Shapiro #if _FFR_REMOVE_TCP_PATH 113806f25ae9SGregory Neil Shapiro syserr("M%s: P=[TCP] is deprecated, use P=[IPC] instead\n", 113906f25ae9SGregory Neil Shapiro m->m_name); 114006f25ae9SGregory Neil Shapiro #else /* _FFR_REMOVE_TCP_PATH */ 114106f25ae9SGregory Neil Shapiro printf("M%s: Warning: P=[TCP] is deprecated, use P=[IPC] instead\n", 114206f25ae9SGregory Neil Shapiro m->m_name); 114306f25ae9SGregory Neil Shapiro #endif /* _FFR_REMOVE_TCP_PATH */ 1144c2aa98e2SPeter Wemm } 1145c2aa98e2SPeter Wemm 114606f25ae9SGregory Neil Shapiro if (strcmp(m->m_mailer, "[IPC]") == 0 || 114706f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_TCP_MAILER_PATH 114806f25ae9SGregory 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 } 115906f25ae9SGregory Neil Shapiro if (strcmp(m->m_argv[0], "TCP") != 0 && 116006f25ae9SGregory Neil Shapiro #if NETUNIX 116106f25ae9SGregory Neil Shapiro strcmp(m->m_argv[0], "FILE") != 0 && 116206f25ae9SGregory Neil Shapiro #endif /* NETUNIX */ 116306f25ae9SGregory Neil Shapiro #if !_FFR_DEPRECATE_IPC_MAILER_ARG 116406f25ae9SGregory 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') 2193c2aa98e2SPeter Wemm QueueDir = "mqueue"; 2194c2aa98e2SPeter Wemm else 2195c2aa98e2SPeter Wemm QueueDir = newstr(val); 2196c2aa98e2SPeter Wemm if (RealUid != 0 && !safe) 2197c2aa98e2SPeter Wemm Warn_Q_option = TRUE; 2198c2aa98e2SPeter Wemm break; 2199c2aa98e2SPeter Wemm 2200c2aa98e2SPeter Wemm case 'R': /* don't prune routes */ 2201c2aa98e2SPeter Wemm DontPruneRoutes = atobool(val); 2202c2aa98e2SPeter Wemm break; 2203c2aa98e2SPeter Wemm 2204c2aa98e2SPeter Wemm case 'r': /* read timeout */ 2205c2aa98e2SPeter Wemm if (subopt == NULL) 220606f25ae9SGregory Neil Shapiro inittimeouts(val, sticky); 2207c2aa98e2SPeter Wemm else 220806f25ae9SGregory Neil Shapiro settimeout(subopt, val, sticky); 2209c2aa98e2SPeter Wemm break; 2210c2aa98e2SPeter Wemm 2211c2aa98e2SPeter Wemm case 'S': /* status file */ 2212c2aa98e2SPeter Wemm if (val[0] == '\0') 221306f25ae9SGregory Neil Shapiro StatFile = "statistics"; 2214c2aa98e2SPeter Wemm else 2215c2aa98e2SPeter Wemm StatFile = newstr(val); 2216c2aa98e2SPeter Wemm break; 2217c2aa98e2SPeter Wemm 2218c2aa98e2SPeter Wemm case 's': /* be super safe, even if expensive */ 2219c2aa98e2SPeter Wemm SuperSafe = atobool(val); 2220c2aa98e2SPeter Wemm break; 2221c2aa98e2SPeter Wemm 2222c2aa98e2SPeter Wemm case 'T': /* queue timeout */ 2223c2aa98e2SPeter Wemm p = strchr(val, '/'); 2224c2aa98e2SPeter Wemm if (p != NULL) 2225c2aa98e2SPeter Wemm { 2226c2aa98e2SPeter Wemm *p++ = '\0'; 222706f25ae9SGregory Neil Shapiro settimeout("queuewarn", p, sticky); 2228c2aa98e2SPeter Wemm } 222906f25ae9SGregory Neil Shapiro settimeout("queuereturn", val, sticky); 2230c2aa98e2SPeter Wemm break; 2231c2aa98e2SPeter Wemm 2232c2aa98e2SPeter Wemm case 't': /* time zone name */ 2233c2aa98e2SPeter Wemm TimeZoneSpec = newstr(val); 2234c2aa98e2SPeter Wemm break; 2235c2aa98e2SPeter Wemm 2236c2aa98e2SPeter Wemm case 'U': /* location of user database */ 2237c2aa98e2SPeter Wemm UdbSpec = newstr(val); 2238c2aa98e2SPeter Wemm break; 2239c2aa98e2SPeter Wemm 2240c2aa98e2SPeter Wemm case 'u': /* set default uid */ 2241c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2242c2aa98e2SPeter Wemm { 2243c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2244c2aa98e2SPeter Wemm { 2245c2aa98e2SPeter Wemm *p++ = '\0'; 2246c2aa98e2SPeter Wemm break; 2247c2aa98e2SPeter Wemm } 2248c2aa98e2SPeter Wemm } 2249c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2250c2aa98e2SPeter Wemm { 2251c2aa98e2SPeter Wemm DefUid = atoi(val); 2252c2aa98e2SPeter Wemm setdefuser(); 2253c2aa98e2SPeter Wemm } 2254c2aa98e2SPeter Wemm else 2255c2aa98e2SPeter Wemm { 2256c2aa98e2SPeter Wemm register struct passwd *pw; 2257c2aa98e2SPeter Wemm 2258c2aa98e2SPeter Wemm DefUid = -1; 2259c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2260c2aa98e2SPeter Wemm if (pw == NULL) 2261c2aa98e2SPeter Wemm syserr("readcf: option u: unknown user %s", val); 2262c2aa98e2SPeter Wemm else 2263c2aa98e2SPeter Wemm { 2264c2aa98e2SPeter Wemm DefUid = pw->pw_uid; 2265c2aa98e2SPeter Wemm DefGid = pw->pw_gid; 2266c2aa98e2SPeter Wemm DefUser = newstr(pw->pw_name); 2267c2aa98e2SPeter Wemm } 2268c2aa98e2SPeter Wemm } 2269c2aa98e2SPeter Wemm 2270c2aa98e2SPeter Wemm #ifdef UID_MAX 2271c2aa98e2SPeter Wemm if (DefUid > UID_MAX) 2272c2aa98e2SPeter Wemm { 2273c2aa98e2SPeter Wemm syserr("readcf: option u: uid value (%ld) > UID_MAX (%ld); ignored", 2274c2aa98e2SPeter Wemm DefUid, UID_MAX); 2275c2aa98e2SPeter Wemm } 227606f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2277c2aa98e2SPeter Wemm 2278c2aa98e2SPeter Wemm /* handle the group if it is there */ 2279c2aa98e2SPeter Wemm if (*p == '\0') 2280c2aa98e2SPeter Wemm break; 2281c2aa98e2SPeter Wemm val = p; 2282c2aa98e2SPeter Wemm goto g_opt; 2283c2aa98e2SPeter Wemm 2284c2aa98e2SPeter Wemm case 'V': /* fallback MX host */ 2285c2aa98e2SPeter Wemm if (val[0] != '\0') 2286c2aa98e2SPeter Wemm FallBackMX = newstr(val); 2287c2aa98e2SPeter Wemm break; 2288c2aa98e2SPeter Wemm 2289c2aa98e2SPeter Wemm case 'v': /* run in verbose mode */ 2290c2aa98e2SPeter Wemm Verbose = atobool(val) ? 1 : 0; 2291c2aa98e2SPeter Wemm break; 2292c2aa98e2SPeter Wemm 2293c2aa98e2SPeter Wemm case 'w': /* if we are best MX, try host directly */ 2294c2aa98e2SPeter Wemm TryNullMXList = atobool(val); 2295c2aa98e2SPeter Wemm break; 2296c2aa98e2SPeter Wemm 2297c2aa98e2SPeter Wemm /* 'W' available -- was wizard password */ 2298c2aa98e2SPeter Wemm 2299c2aa98e2SPeter Wemm case 'x': /* load avg at which to auto-queue msgs */ 2300c2aa98e2SPeter Wemm QueueLA = atoi(val); 2301c2aa98e2SPeter Wemm break; 2302c2aa98e2SPeter Wemm 2303c2aa98e2SPeter Wemm case 'X': /* load avg at which to auto-reject connections */ 2304c2aa98e2SPeter Wemm RefuseLA = atoi(val); 2305c2aa98e2SPeter Wemm break; 2306c2aa98e2SPeter Wemm 2307c2aa98e2SPeter Wemm case 'y': /* work recipient factor */ 2308c2aa98e2SPeter Wemm WkRecipFact = atoi(val); 2309c2aa98e2SPeter Wemm break; 2310c2aa98e2SPeter Wemm 2311c2aa98e2SPeter Wemm case 'Y': /* fork jobs during queue runs */ 2312c2aa98e2SPeter Wemm ForkQueueRuns = atobool(val); 2313c2aa98e2SPeter Wemm break; 2314c2aa98e2SPeter Wemm 2315c2aa98e2SPeter Wemm case 'z': /* work message class factor */ 2316c2aa98e2SPeter Wemm WkClassFact = atoi(val); 2317c2aa98e2SPeter Wemm break; 2318c2aa98e2SPeter Wemm 2319c2aa98e2SPeter Wemm case 'Z': /* work time factor */ 2320c2aa98e2SPeter Wemm WkTimeFact = atoi(val); 2321c2aa98e2SPeter Wemm break; 2322c2aa98e2SPeter Wemm 232306f25ae9SGregory Neil Shapiro 2324c2aa98e2SPeter Wemm case O_QUEUESORTORD: /* queue sorting order */ 2325c2aa98e2SPeter Wemm switch (*val) 2326c2aa98e2SPeter Wemm { 2327c2aa98e2SPeter Wemm case 'h': /* Host first */ 2328c2aa98e2SPeter Wemm case 'H': 232906f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYHOST; 2330c2aa98e2SPeter Wemm break; 2331c2aa98e2SPeter Wemm 2332c2aa98e2SPeter Wemm case 'p': /* Priority order */ 2333c2aa98e2SPeter Wemm case 'P': 233406f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYPRIORITY; 2335c2aa98e2SPeter Wemm break; 2336c2aa98e2SPeter Wemm 2337c2aa98e2SPeter Wemm case 't': /* Submission time */ 2338c2aa98e2SPeter Wemm case 'T': 233906f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYTIME; 234006f25ae9SGregory Neil Shapiro break; 234106f25ae9SGregory Neil Shapiro 234206f25ae9SGregory Neil Shapiro case 'f': /* File Name */ 234306f25ae9SGregory Neil Shapiro case 'F': 234406f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYFILENAME; 2345c2aa98e2SPeter Wemm break; 2346c2aa98e2SPeter Wemm 2347c2aa98e2SPeter Wemm default: 2348c2aa98e2SPeter Wemm syserr("Invalid queue sort order \"%s\"", val); 2349c2aa98e2SPeter Wemm } 2350c2aa98e2SPeter Wemm break; 2351c2aa98e2SPeter Wemm 235206f25ae9SGregory Neil Shapiro #if _FFR_QUEUEDELAY 235306f25ae9SGregory Neil Shapiro case O_QUEUEDELAY: /* queue delay algorithm */ 235406f25ae9SGregory Neil Shapiro switch (*val) 235506f25ae9SGregory Neil Shapiro { 235606f25ae9SGregory Neil Shapiro case 'e': /* exponential */ 235706f25ae9SGregory Neil Shapiro case 'E': 235806f25ae9SGregory Neil Shapiro QueueAlg = QD_EXP; 235906f25ae9SGregory Neil Shapiro QueueInitDelay = 10 MINUTES; 236006f25ae9SGregory Neil Shapiro QueueMaxDelay = 2 HOURS; 236106f25ae9SGregory Neil Shapiro p = strchr(val, '/'); 236206f25ae9SGregory Neil Shapiro if (p != NULL) 236306f25ae9SGregory Neil Shapiro { 236406f25ae9SGregory Neil Shapiro char *q; 236506f25ae9SGregory Neil Shapiro 236606f25ae9SGregory Neil Shapiro *p++ = '\0'; 236706f25ae9SGregory Neil Shapiro q = strchr(p, '/'); 236806f25ae9SGregory Neil Shapiro if (q != NULL) 236906f25ae9SGregory Neil Shapiro *q++ = '\0'; 237006f25ae9SGregory Neil Shapiro QueueInitDelay = convtime(p, 's'); 237106f25ae9SGregory Neil Shapiro if (q != NULL) 237206f25ae9SGregory Neil Shapiro { 237306f25ae9SGregory Neil Shapiro QueueMaxDelay = convtime(q, 's'); 237406f25ae9SGregory Neil Shapiro } 237506f25ae9SGregory Neil Shapiro } 237606f25ae9SGregory Neil Shapiro break; 237706f25ae9SGregory Neil Shapiro 237806f25ae9SGregory Neil Shapiro case 'l': /* linear */ 237906f25ae9SGregory Neil Shapiro case 'L': 238006f25ae9SGregory Neil Shapiro QueueAlg = QD_LINEAR; 238106f25ae9SGregory Neil Shapiro break; 238206f25ae9SGregory Neil Shapiro 238306f25ae9SGregory Neil Shapiro default: 238406f25ae9SGregory Neil Shapiro syserr("Invalid queue delay algorithm \"%s\"", val); 238506f25ae9SGregory Neil Shapiro } 238606f25ae9SGregory Neil Shapiro break; 238706f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUEDELAY */ 238806f25ae9SGregory Neil Shapiro 2389c2aa98e2SPeter Wemm case O_HOSTSFILE: /* pathname of /etc/hosts file */ 2390c2aa98e2SPeter Wemm HostsFile = newstr(val); 2391c2aa98e2SPeter Wemm break; 2392c2aa98e2SPeter Wemm 2393c2aa98e2SPeter Wemm case O_MQA: /* minimum queue age between deliveries */ 2394c2aa98e2SPeter Wemm MinQueueAge = convtime(val, 'm'); 2395c2aa98e2SPeter Wemm break; 2396c2aa98e2SPeter Wemm 2397c2aa98e2SPeter Wemm case O_DEFCHARSET: /* default character set for mimefying */ 2398c2aa98e2SPeter Wemm DefaultCharSet = newstr(denlstring(val, TRUE, TRUE)); 2399c2aa98e2SPeter Wemm break; 2400c2aa98e2SPeter Wemm 2401c2aa98e2SPeter Wemm case O_SSFILE: /* service switch file */ 2402c2aa98e2SPeter Wemm ServiceSwitchFile = newstr(val); 2403c2aa98e2SPeter Wemm break; 2404c2aa98e2SPeter Wemm 2405c2aa98e2SPeter Wemm case O_DIALDELAY: /* delay for dial-on-demand operation */ 2406c2aa98e2SPeter Wemm DialDelay = convtime(val, 's'); 2407c2aa98e2SPeter Wemm break; 2408c2aa98e2SPeter Wemm 2409c2aa98e2SPeter Wemm case O_NORCPTACTION: /* what to do if no recipient */ 2410c2aa98e2SPeter Wemm if (strcasecmp(val, "none") == 0) 2411c2aa98e2SPeter Wemm NoRecipientAction = NRA_NO_ACTION; 2412c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to") == 0) 2413c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO; 2414c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-apparently-to") == 0) 2415c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_APPARENTLY_TO; 2416c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-bcc") == 0) 2417c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_BCC; 2418c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to-undisclosed") == 0) 2419c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO_UNDISCLOSED; 2420c2aa98e2SPeter Wemm else 2421c2aa98e2SPeter Wemm syserr("Invalid NoRecipientAction: %s", val); 2422c2aa98e2SPeter Wemm break; 2423c2aa98e2SPeter Wemm 2424c2aa98e2SPeter Wemm case O_SAFEFILEENV: /* chroot() environ for writing to files */ 2425c2aa98e2SPeter Wemm SafeFileEnv = newstr(val); 2426c2aa98e2SPeter Wemm break; 2427c2aa98e2SPeter Wemm 2428c2aa98e2SPeter Wemm case O_MAXMSGSIZE: /* maximum message size */ 2429c2aa98e2SPeter Wemm MaxMessageSize = atol(val); 2430c2aa98e2SPeter Wemm break; 2431c2aa98e2SPeter Wemm 2432c2aa98e2SPeter Wemm case O_COLONOKINADDR: /* old style handling of colon addresses */ 2433c2aa98e2SPeter Wemm ColonOkInAddr = atobool(val); 2434c2aa98e2SPeter Wemm break; 2435c2aa98e2SPeter Wemm 2436c2aa98e2SPeter Wemm case O_MAXQUEUERUN: /* max # of jobs in a single queue run */ 2437c2aa98e2SPeter Wemm MaxQueueRun = atol(val); 2438c2aa98e2SPeter Wemm break; 2439c2aa98e2SPeter Wemm 2440c2aa98e2SPeter Wemm case O_MAXCHILDREN: /* max # of children of daemon */ 2441c2aa98e2SPeter Wemm MaxChildren = atoi(val); 2442c2aa98e2SPeter Wemm break; 2443c2aa98e2SPeter Wemm 244406f25ae9SGregory Neil Shapiro #if _FFR_MAX_FORWARD_ENTRIES 244506f25ae9SGregory Neil Shapiro case O_MAXFORWARD: /* max # of forward entries */ 244606f25ae9SGregory Neil Shapiro MaxForwardEntries = atoi(val); 244706f25ae9SGregory Neil Shapiro break; 244806f25ae9SGregory Neil Shapiro #endif /* _FFR_MAX_FORWARD_ENTRIES */ 244906f25ae9SGregory Neil Shapiro 2450c2aa98e2SPeter Wemm case O_KEEPCNAMES: /* don't expand CNAME records */ 2451c2aa98e2SPeter Wemm DontExpandCnames = atobool(val); 2452c2aa98e2SPeter Wemm break; 2453c2aa98e2SPeter Wemm 2454c2aa98e2SPeter Wemm case O_MUSTQUOTE: /* must quote these characters in phrases */ 245506f25ae9SGregory Neil Shapiro (void) strlcpy(buf, "@,;:\\()[]", sizeof buf); 2456c2aa98e2SPeter Wemm if (strlen(val) < (SIZE_T) sizeof buf - 10) 245706f25ae9SGregory Neil Shapiro (void) strlcat(buf, val, sizeof buf); 245806f25ae9SGregory Neil Shapiro else 245906f25ae9SGregory Neil Shapiro printf("Warning: MustQuoteChars too long, ignored.\n"); 2460c2aa98e2SPeter Wemm MustQuoteChars = newstr(buf); 2461c2aa98e2SPeter Wemm break; 2462c2aa98e2SPeter Wemm 2463c2aa98e2SPeter Wemm case O_SMTPGREETING: /* SMTP greeting message (old $e macro) */ 2464c2aa98e2SPeter Wemm SmtpGreeting = newstr(munchstring(val, NULL, '\0')); 2465c2aa98e2SPeter Wemm break; 2466c2aa98e2SPeter Wemm 2467c2aa98e2SPeter Wemm case O_UNIXFROM: /* UNIX From_ line (old $l macro) */ 2468c2aa98e2SPeter Wemm UnixFromLine = newstr(munchstring(val, NULL, '\0')); 2469c2aa98e2SPeter Wemm break; 2470c2aa98e2SPeter Wemm 2471c2aa98e2SPeter Wemm case O_OPCHARS: /* operator characters (old $o macro) */ 247206f25ae9SGregory Neil Shapiro if (OperatorChars != NULL) 247306f25ae9SGregory Neil Shapiro printf("Warning: OperatorChars is being redefined.\n It should only be set before ruleset definitions.\n"); 2474c2aa98e2SPeter Wemm OperatorChars = newstr(munchstring(val, NULL, '\0')); 2475c2aa98e2SPeter Wemm break; 2476c2aa98e2SPeter Wemm 2477c2aa98e2SPeter Wemm case O_DONTINITGRPS: /* don't call initgroups(3) */ 2478c2aa98e2SPeter Wemm DontInitGroups = atobool(val); 2479c2aa98e2SPeter Wemm break; 2480c2aa98e2SPeter Wemm 2481c2aa98e2SPeter Wemm case O_SLFH: /* make sure from fits on one line */ 2482c2aa98e2SPeter Wemm SingleLineFromHeader = atobool(val); 2483c2aa98e2SPeter Wemm break; 2484c2aa98e2SPeter Wemm 2485c2aa98e2SPeter Wemm case O_ABH: /* allow HELO commands with syntax errors */ 2486c2aa98e2SPeter Wemm AllowBogusHELO = atobool(val); 2487c2aa98e2SPeter Wemm break; 2488c2aa98e2SPeter Wemm 2489c2aa98e2SPeter Wemm case O_CONNTHROT: /* connection rate throttle */ 2490c2aa98e2SPeter Wemm ConnRateThrottle = atoi(val); 2491c2aa98e2SPeter Wemm break; 2492c2aa98e2SPeter Wemm 2493c2aa98e2SPeter Wemm case O_UGW: /* group writable files are unsafe */ 2494c2aa98e2SPeter Wemm if (!atobool(val)) 249506f25ae9SGregory Neil Shapiro { 249606f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEFORWARDFILESAFE, 249706f25ae9SGregory Neil Shapiro DontBlameSendmail); 249806f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEINCLUDEFILESAFE, 249906f25ae9SGregory Neil Shapiro DontBlameSendmail); 250006f25ae9SGregory Neil Shapiro } 2501c2aa98e2SPeter Wemm break; 2502c2aa98e2SPeter Wemm 2503c2aa98e2SPeter Wemm case O_DBLBOUNCE: /* address to which to send double bounces */ 2504c2aa98e2SPeter Wemm if (val[0] != '\0') 2505c2aa98e2SPeter Wemm DoubleBounceAddr = newstr(val); 2506c2aa98e2SPeter Wemm else 2507c2aa98e2SPeter Wemm syserr("readcf: option DoubleBounceAddress: value required"); 2508c2aa98e2SPeter Wemm break; 2509c2aa98e2SPeter Wemm 2510c2aa98e2SPeter Wemm case O_HSDIR: /* persistent host status directory */ 2511c2aa98e2SPeter Wemm if (val[0] != '\0') 2512c2aa98e2SPeter Wemm HostStatDir = newstr(val); 2513c2aa98e2SPeter Wemm break; 2514c2aa98e2SPeter Wemm 2515c2aa98e2SPeter Wemm case O_SINGTHREAD: /* single thread deliveries (requires hsdir) */ 2516c2aa98e2SPeter Wemm SingleThreadDelivery = atobool(val); 2517c2aa98e2SPeter Wemm break; 2518c2aa98e2SPeter Wemm 2519c2aa98e2SPeter Wemm case O_RUNASUSER: /* run bulk of code as this user */ 2520c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2521c2aa98e2SPeter Wemm { 2522c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2523c2aa98e2SPeter Wemm { 2524c2aa98e2SPeter Wemm *p++ = '\0'; 2525c2aa98e2SPeter Wemm break; 2526c2aa98e2SPeter Wemm } 2527c2aa98e2SPeter Wemm } 2528c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2529c2aa98e2SPeter Wemm { 2530c2aa98e2SPeter Wemm if (can_setuid) 2531c2aa98e2SPeter Wemm RunAsUid = atoi(val); 2532c2aa98e2SPeter Wemm } 2533c2aa98e2SPeter Wemm else 2534c2aa98e2SPeter Wemm { 2535c2aa98e2SPeter Wemm register struct passwd *pw; 2536c2aa98e2SPeter Wemm 2537c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2538c2aa98e2SPeter Wemm if (pw == NULL) 2539c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown user %s", val); 2540c2aa98e2SPeter Wemm else if (can_setuid) 2541c2aa98e2SPeter Wemm { 2542c2aa98e2SPeter Wemm if (*p == '\0') 2543c2aa98e2SPeter Wemm RunAsUserName = newstr(val); 2544c2aa98e2SPeter Wemm RunAsUid = pw->pw_uid; 2545c2aa98e2SPeter Wemm RunAsGid = pw->pw_gid; 2546c2aa98e2SPeter Wemm } 2547c2aa98e2SPeter Wemm } 2548c2aa98e2SPeter Wemm #ifdef UID_MAX 2549c2aa98e2SPeter Wemm if (RunAsUid > UID_MAX) 2550c2aa98e2SPeter Wemm { 2551c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: uid value (%ld) > UID_MAX (%ld); ignored", 2552c2aa98e2SPeter Wemm RunAsUid, UID_MAX); 2553c2aa98e2SPeter Wemm } 255406f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2555c2aa98e2SPeter Wemm if (*p != '\0') 2556c2aa98e2SPeter Wemm { 2557c2aa98e2SPeter Wemm if (isascii(*p) && isdigit(*p)) 2558c2aa98e2SPeter Wemm { 2559c2aa98e2SPeter Wemm if (can_setuid) 2560c2aa98e2SPeter Wemm RunAsGid = atoi(p); 2561c2aa98e2SPeter Wemm } 2562c2aa98e2SPeter Wemm else 2563c2aa98e2SPeter Wemm { 2564c2aa98e2SPeter Wemm register struct group *gr; 2565c2aa98e2SPeter Wemm 2566c2aa98e2SPeter Wemm gr = getgrnam(p); 2567c2aa98e2SPeter Wemm if (gr == NULL) 2568c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown group %s", 2569c2aa98e2SPeter Wemm p); 2570c2aa98e2SPeter Wemm else if (can_setuid) 2571c2aa98e2SPeter Wemm RunAsGid = gr->gr_gid; 2572c2aa98e2SPeter Wemm } 2573c2aa98e2SPeter Wemm } 2574c2aa98e2SPeter Wemm if (tTd(47, 5)) 257506f25ae9SGregory Neil Shapiro dprintf("readcf: RunAsUser = %d:%d\n", 257606f25ae9SGregory Neil Shapiro (int)RunAsUid, (int)RunAsGid); 2577c2aa98e2SPeter Wemm break; 2578c2aa98e2SPeter Wemm 2579c2aa98e2SPeter Wemm case O_DSN_RRT: 2580c2aa98e2SPeter Wemm RrtImpliesDsn = atobool(val); 2581c2aa98e2SPeter Wemm break; 2582c2aa98e2SPeter Wemm 2583c2aa98e2SPeter Wemm case O_PIDFILE: 258406f25ae9SGregory Neil Shapiro if (PidFile != NULL) 2585c2aa98e2SPeter Wemm free(PidFile); 2586c2aa98e2SPeter Wemm PidFile = newstr(val); 2587c2aa98e2SPeter Wemm break; 2588c2aa98e2SPeter Wemm 2589c2aa98e2SPeter Wemm case O_DONTBLAMESENDMAIL: 2590c2aa98e2SPeter Wemm p = val; 2591c2aa98e2SPeter Wemm for (;;) 2592c2aa98e2SPeter Wemm { 2593c2aa98e2SPeter Wemm register struct dbsval *dbs; 2594c2aa98e2SPeter Wemm extern struct dbsval DontBlameSendmailValues[]; 2595c2aa98e2SPeter Wemm 2596c2aa98e2SPeter Wemm while (isascii(*p) && (isspace(*p) || ispunct(*p))) 2597c2aa98e2SPeter Wemm p++; 2598c2aa98e2SPeter Wemm if (*p == '\0') 2599c2aa98e2SPeter Wemm break; 2600c2aa98e2SPeter Wemm val = p; 2601c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 2602c2aa98e2SPeter Wemm p++; 2603c2aa98e2SPeter Wemm if (*p != '\0') 2604c2aa98e2SPeter Wemm *p++ = '\0'; 2605c2aa98e2SPeter Wemm 2606c2aa98e2SPeter Wemm for (dbs = DontBlameSendmailValues; 2607c2aa98e2SPeter Wemm dbs->dbs_name != NULL; dbs++) 2608c2aa98e2SPeter Wemm { 2609c2aa98e2SPeter Wemm if (strcasecmp(val, dbs->dbs_name) == 0) 2610c2aa98e2SPeter Wemm break; 2611c2aa98e2SPeter Wemm } 2612c2aa98e2SPeter Wemm if (dbs->dbs_name == NULL) 2613c2aa98e2SPeter Wemm syserr("readcf: DontBlameSendmail option: %s unrecognized", val); 2614c2aa98e2SPeter Wemm else if (dbs->dbs_flag == DBS_SAFE) 261506f25ae9SGregory Neil Shapiro clrbitmap(DontBlameSendmail); 2616c2aa98e2SPeter Wemm else 261706f25ae9SGregory Neil Shapiro setbitn(dbs->dbs_flag, DontBlameSendmail); 2618c2aa98e2SPeter Wemm } 2619c2aa98e2SPeter Wemm sticky = FALSE; 2620c2aa98e2SPeter Wemm break; 2621c2aa98e2SPeter Wemm 2622c2aa98e2SPeter Wemm case O_DPI: 2623c2aa98e2SPeter Wemm DontProbeInterfaces = atobool(val); 2624c2aa98e2SPeter Wemm break; 2625c2aa98e2SPeter Wemm 2626c2aa98e2SPeter Wemm case O_MAXRCPT: 2627c2aa98e2SPeter Wemm MaxRcptPerMsg = atoi(val); 2628c2aa98e2SPeter Wemm break; 2629c2aa98e2SPeter Wemm 2630c2aa98e2SPeter Wemm case O_DEADLETTER: 2631c2aa98e2SPeter Wemm if (DeadLetterDrop != NULL) 2632c2aa98e2SPeter Wemm free(DeadLetterDrop); 2633c2aa98e2SPeter Wemm DeadLetterDrop = newstr(val); 2634c2aa98e2SPeter Wemm break; 2635c2aa98e2SPeter Wemm 2636c2aa98e2SPeter Wemm #if _FFR_DONTLOCKFILESFORREAD_OPTION 2637c2aa98e2SPeter Wemm case O_DONTLOCK: 2638c2aa98e2SPeter Wemm DontLockReadFiles = atobool(val); 2639c2aa98e2SPeter Wemm break; 264006f25ae9SGregory Neil Shapiro #endif /* _FFR_DONTLOCKFILESFORREAD_OPTION */ 2641c2aa98e2SPeter Wemm 2642c2aa98e2SPeter Wemm case O_MAXALIASRCSN: 2643c2aa98e2SPeter Wemm MaxAliasRecursion = atoi(val); 2644c2aa98e2SPeter Wemm break; 2645c2aa98e2SPeter Wemm 2646c2aa98e2SPeter Wemm case O_CNCTONLYTO: 2647c2aa98e2SPeter Wemm /* XXX should probably use gethostbyname */ 264806f25ae9SGregory Neil Shapiro #if NETINET || NETINET6 264906f25ae9SGregory Neil Shapiro # if NETINET6 265006f25ae9SGregory Neil Shapiro if (inet_addr(val) == INADDR_NONE) 265106f25ae9SGregory Neil Shapiro { 265206f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET6; 265306f25ae9SGregory Neil Shapiro if (inet_pton(AF_INET6, val, 265406f25ae9SGregory Neil Shapiro &ConnectOnlyTo.sin6.sin6_addr) != 1) 265506f25ae9SGregory Neil Shapiro syserr("readcf: option ConnectOnlyTo: invalid IP address %s", 265606f25ae9SGregory Neil Shapiro val); 265706f25ae9SGregory Neil Shapiro } 265806f25ae9SGregory Neil Shapiro else 265906f25ae9SGregory Neil Shapiro # endif /* NETINET6 */ 266006f25ae9SGregory Neil Shapiro { 266106f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET; 266206f25ae9SGregory Neil Shapiro ConnectOnlyTo.sin.sin_addr.s_addr = inet_addr(val); 266306f25ae9SGregory Neil Shapiro } 266406f25ae9SGregory Neil Shapiro #endif /* NETINET || NETINET6 */ 2665c2aa98e2SPeter Wemm break; 2666c2aa98e2SPeter Wemm 2667065a643dSPeter Wemm case O_TRUSTUSER: 266806f25ae9SGregory Neil Shapiro #if HASFCHOWN 2669c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2670065a643dSPeter Wemm TrustedUid = atoi(val); 2671c2aa98e2SPeter Wemm else 2672c2aa98e2SPeter Wemm { 2673c2aa98e2SPeter Wemm register struct passwd *pw; 2674c2aa98e2SPeter Wemm 2675065a643dSPeter Wemm TrustedUid = 0; 2676c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2677c2aa98e2SPeter Wemm if (pw == NULL) 2678065a643dSPeter Wemm syserr("readcf: option TrustedUser: unknown user %s", val); 2679c2aa98e2SPeter Wemm else 2680065a643dSPeter Wemm TrustedUid = pw->pw_uid; 2681c2aa98e2SPeter Wemm } 2682c2aa98e2SPeter Wemm 2683c2aa98e2SPeter Wemm # ifdef UID_MAX 2684065a643dSPeter Wemm if (TrustedUid > UID_MAX) 2685c2aa98e2SPeter Wemm { 2686065a643dSPeter Wemm syserr("readcf: option TrustedUser: uid value (%ld) > UID_MAX (%ld)", 2687065a643dSPeter Wemm TrustedUid, UID_MAX); 2688065a643dSPeter Wemm TrustedUid = 0; 2689c2aa98e2SPeter Wemm } 269006f25ae9SGregory Neil Shapiro # endif /* UID_MAX */ 269106f25ae9SGregory Neil Shapiro #else /* HASFCHOWN */ 269206f25ae9SGregory Neil Shapiro syserr("readcf: option TrustedUser: can not be used on systems which do not support fchown()"); 269306f25ae9SGregory Neil Shapiro #endif /* HASFCHOWN */ 2694c2aa98e2SPeter Wemm break; 2695c2aa98e2SPeter Wemm 2696065a643dSPeter Wemm case O_MAXMIMEHDRLEN: 2697065a643dSPeter Wemm p = strchr(val, '/'); 2698065a643dSPeter Wemm if (p != NULL) 2699065a643dSPeter Wemm *p++ = '\0'; 2700065a643dSPeter Wemm MaxMimeHeaderLength = atoi(val); 2701065a643dSPeter Wemm if (p != NULL && *p != '\0') 2702065a643dSPeter Wemm MaxMimeFieldLength = atoi(p); 2703065a643dSPeter Wemm else 2704065a643dSPeter Wemm MaxMimeFieldLength = MaxMimeHeaderLength / 2; 2705065a643dSPeter Wemm 2706065a643dSPeter Wemm if (MaxMimeHeaderLength < 0) 2707065a643dSPeter Wemm MaxMimeHeaderLength = 0; 2708065a643dSPeter Wemm else if (MaxMimeHeaderLength < 128) 2709065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: header length limit set lower than 128\n"); 2710065a643dSPeter Wemm 2711065a643dSPeter Wemm if (MaxMimeFieldLength < 0) 2712065a643dSPeter Wemm MaxMimeFieldLength = 0; 2713065a643dSPeter Wemm else if (MaxMimeFieldLength < 40) 2714065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: field length limit set lower than 40\n"); 2715065a643dSPeter Wemm break; 2716065a643dSPeter Wemm 2717065a643dSPeter Wemm case O_CONTROLSOCKET: 2718065a643dSPeter Wemm if (ControlSocketName != NULL) 2719065a643dSPeter Wemm free(ControlSocketName); 2720065a643dSPeter Wemm ControlSocketName = newstr(val); 2721065a643dSPeter Wemm break; 2722065a643dSPeter Wemm 27232e43090eSPeter Wemm case O_MAXHDRSLEN: 27242e43090eSPeter Wemm MaxHeadersLength = atoi(val); 272525bab6e9SPeter Wemm 27262e43090eSPeter Wemm if (MaxHeadersLength > 0 && 27272e43090eSPeter Wemm MaxHeadersLength < (MAXHDRSLEN / 2)) 272806f25ae9SGregory Neil Shapiro printf("Warning: MaxHeadersLength: headers length limit set lower than %d\n", (MAXHDRSLEN / 2)); 272925bab6e9SPeter Wemm break; 273006f25ae9SGregory Neil Shapiro 273106f25ae9SGregory Neil Shapiro case O_PROCTITLEPREFIX: 273206f25ae9SGregory Neil Shapiro if (ProcTitlePrefix != NULL) 273306f25ae9SGregory Neil Shapiro free(ProcTitlePrefix); 273406f25ae9SGregory Neil Shapiro ProcTitlePrefix = newstr(val); 273506f25ae9SGregory Neil Shapiro break; 273606f25ae9SGregory Neil Shapiro 273706f25ae9SGregory Neil Shapiro #if SASL 273806f25ae9SGregory Neil Shapiro case O_SASLINFO: 273906f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 274006f25ae9SGregory Neil Shapiro /* 274106f25ae9SGregory Neil Shapiro ** Allow users to select their own authinfo file. 274206f25ae9SGregory Neil Shapiro ** However, this is not a "perfect" solution. 274306f25ae9SGregory Neil Shapiro ** If mail is queued, the authentication info 274406f25ae9SGregory Neil Shapiro ** will not be used in subsequent delivery attempts. 274506f25ae9SGregory Neil Shapiro ** If we really want to support this, then it has 274606f25ae9SGregory Neil Shapiro ** to be stored in the queue file. 274706f25ae9SGregory Neil Shapiro */ 274806f25ae9SGregory Neil Shapiro if (!bitset(SUBMIT_MSA, SubmitMode) && RealUid != 0 && 274906f25ae9SGregory Neil Shapiro RunAsUid != RealUid) 275006f25ae9SGregory Neil Shapiro { 275106f25ae9SGregory Neil Shapiro errno = 0; 275206f25ae9SGregory Neil Shapiro syserr("Error: %s only allowed with -U\n", 275306f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 275406f25ae9SGregory Neil Shapiro ExitStat = EX_USAGE; 275506f25ae9SGregory Neil Shapiro break; 275606f25ae9SGregory Neil Shapiro } 275706f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 275806f25ae9SGregory Neil Shapiro if (SASLInfo != NULL) 275906f25ae9SGregory Neil Shapiro free(SASLInfo); 276006f25ae9SGregory Neil Shapiro SASLInfo = newstr(val); 276106f25ae9SGregory Neil Shapiro break; 276206f25ae9SGregory Neil Shapiro 276306f25ae9SGregory Neil Shapiro case O_SASLMECH: 276406f25ae9SGregory Neil Shapiro if (AuthMechanisms != NULL) 276506f25ae9SGregory Neil Shapiro free(AuthMechanisms); 276606f25ae9SGregory Neil Shapiro if (*val != '\0') 276706f25ae9SGregory Neil Shapiro AuthMechanisms = newstr(val); 276806f25ae9SGregory Neil Shapiro else 276906f25ae9SGregory Neil Shapiro AuthMechanisms = NULL; 277006f25ae9SGregory Neil Shapiro break; 277106f25ae9SGregory Neil Shapiro 277206f25ae9SGregory Neil Shapiro case O_SASLOPTS: 277306f25ae9SGregory Neil Shapiro while (val != NULL && *val != '\0') 277406f25ae9SGregory Neil Shapiro { 277506f25ae9SGregory Neil Shapiro switch(*val) 277606f25ae9SGregory Neil Shapiro { 277706f25ae9SGregory Neil Shapiro case 'A': 277806f25ae9SGregory Neil Shapiro SASLOpts |= SASL_AUTH_AUTH; 277906f25ae9SGregory Neil Shapiro break; 278006f25ae9SGregory Neil Shapiro # if _FFR_SASL_OPTS 278106f25ae9SGregory Neil Shapiro case 'a': 278206f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOACTIVE; 278306f25ae9SGregory Neil Shapiro break; 278406f25ae9SGregory Neil Shapiro case 'c': 278506f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_PASS_CREDENTIALS; 278606f25ae9SGregory Neil Shapiro break; 278706f25ae9SGregory Neil Shapiro case 'd': 278806f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NODICTIONARY; 278906f25ae9SGregory Neil Shapiro break; 279006f25ae9SGregory Neil Shapiro case 'f': 279106f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_FORWARD_SECRECY; 279206f25ae9SGregory Neil Shapiro break; 279306f25ae9SGregory Neil Shapiro case 'p': 279406f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOPLAINTEXT; 279506f25ae9SGregory Neil Shapiro break; 279606f25ae9SGregory Neil Shapiro case 'y': 279706f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOANONYMOUS; 279806f25ae9SGregory Neil Shapiro break; 279906f25ae9SGregory Neil Shapiro # endif /* _FFR_SASL_OPTS */ 280006f25ae9SGregory Neil Shapiro default: 280106f25ae9SGregory Neil Shapiro printf("Warning: Option: %s unknown parameter '%c'\n", 280206f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" 280306f25ae9SGregory Neil Shapiro : o->o_name, 280406f25ae9SGregory Neil Shapiro (isascii(*val) && isprint(*val)) ? *val 280506f25ae9SGregory Neil Shapiro : '?'); 280606f25ae9SGregory Neil Shapiro break; 280706f25ae9SGregory Neil Shapiro } 280806f25ae9SGregory Neil Shapiro ++val; 280906f25ae9SGregory Neil Shapiro val = strpbrk(val, ", \t"); 281006f25ae9SGregory Neil Shapiro if (val != NULL) 281106f25ae9SGregory Neil Shapiro ++val; 281206f25ae9SGregory Neil Shapiro } 281306f25ae9SGregory Neil Shapiro break; 281406f25ae9SGregory Neil Shapiro 281506f25ae9SGregory Neil Shapiro #else /* SASL */ 281606f25ae9SGregory Neil Shapiro case O_SASLINFO: 281706f25ae9SGregory Neil Shapiro case O_SASLMECH: 281806f25ae9SGregory Neil Shapiro case O_SASLOPTS: 281906f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires SASL support (-DSASL)\n", 282006f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 282106f25ae9SGregory Neil Shapiro break; 282206f25ae9SGregory Neil Shapiro #endif /* SASL */ 282306f25ae9SGregory Neil Shapiro 282406f25ae9SGregory Neil Shapiro #if STARTTLS 282506f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 282606f25ae9SGregory Neil Shapiro if (SrvCERTfile != NULL) 282706f25ae9SGregory Neil Shapiro free(SrvCERTfile); 282806f25ae9SGregory Neil Shapiro SrvCERTfile = newstr(val); 282906f25ae9SGregory Neil Shapiro break; 283006f25ae9SGregory Neil Shapiro 283106f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 283206f25ae9SGregory Neil Shapiro if (Srvkeyfile != NULL) 283306f25ae9SGregory Neil Shapiro free(Srvkeyfile); 283406f25ae9SGregory Neil Shapiro Srvkeyfile = newstr(val); 283506f25ae9SGregory Neil Shapiro break; 283606f25ae9SGregory Neil Shapiro 283706f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 283806f25ae9SGregory Neil Shapiro if (CltCERTfile != NULL) 283906f25ae9SGregory Neil Shapiro free(CltCERTfile); 284006f25ae9SGregory Neil Shapiro CltCERTfile = newstr(val); 284106f25ae9SGregory Neil Shapiro break; 284206f25ae9SGregory Neil Shapiro 284306f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 284406f25ae9SGregory Neil Shapiro if (Cltkeyfile != NULL) 284506f25ae9SGregory Neil Shapiro free(Cltkeyfile); 284606f25ae9SGregory Neil Shapiro Cltkeyfile = newstr(val); 284706f25ae9SGregory Neil Shapiro break; 284806f25ae9SGregory Neil Shapiro 284906f25ae9SGregory Neil Shapiro case O_CACERTFILE: 285006f25ae9SGregory Neil Shapiro if (CACERTfile != NULL) 285106f25ae9SGregory Neil Shapiro free(CACERTfile); 285206f25ae9SGregory Neil Shapiro CACERTfile = newstr(val); 285306f25ae9SGregory Neil Shapiro break; 285406f25ae9SGregory Neil Shapiro 285506f25ae9SGregory Neil Shapiro case O_CACERTPATH: 285606f25ae9SGregory Neil Shapiro if (CACERTpath != NULL) 285706f25ae9SGregory Neil Shapiro free(CACERTpath); 285806f25ae9SGregory Neil Shapiro CACERTpath = newstr(val); 285906f25ae9SGregory Neil Shapiro break; 286006f25ae9SGregory Neil Shapiro 286106f25ae9SGregory Neil Shapiro case O_DHPARAMS: 286206f25ae9SGregory Neil Shapiro if (DHParams != NULL) 286306f25ae9SGregory Neil Shapiro free(DHParams); 286406f25ae9SGregory Neil Shapiro DHParams = newstr(val); 286506f25ae9SGregory Neil Shapiro break; 286606f25ae9SGregory Neil Shapiro 286706f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 286806f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 286906f25ae9SGregory Neil Shapiro if (DHParams5 != NULL) 287006f25ae9SGregory Neil Shapiro free(DHParams5); 287106f25ae9SGregory Neil Shapiro DHParams5 = newstr(val); 287206f25ae9SGregory Neil Shapiro break; 287306f25ae9SGregory Neil Shapiro 287406f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 287506f25ae9SGregory Neil Shapiro if (CipherList != NULL) 287606f25ae9SGregory Neil Shapiro free(CipherList); 287706f25ae9SGregory Neil Shapiro CipherList = newstr(val); 287806f25ae9SGregory Neil Shapiro break; 287906f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 288006f25ae9SGregory Neil Shapiro 288106f25ae9SGregory Neil Shapiro case O_RANDFILE: 288206f25ae9SGregory Neil Shapiro if (RandFile != NULL) 288306f25ae9SGregory Neil Shapiro free(RandFile); 288406f25ae9SGregory Neil Shapiro RandFile= newstr(val); 288506f25ae9SGregory Neil Shapiro break; 288606f25ae9SGregory Neil Shapiro 288706f25ae9SGregory Neil Shapiro # else /* STARTTLS */ 288806f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 288906f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 289006f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 289106f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 289206f25ae9SGregory Neil Shapiro case O_CACERTFILE: 289306f25ae9SGregory Neil Shapiro case O_CACERTPATH: 289406f25ae9SGregory Neil Shapiro case O_DHPARAMS: 289506f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 289606f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 289706f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 289806f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 289906f25ae9SGregory Neil Shapiro case O_RANDFILE: 290006f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires TLS support\n", 290106f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 290206f25ae9SGregory Neil Shapiro break; 290306f25ae9SGregory Neil Shapiro 290406f25ae9SGregory Neil Shapiro # endif /* STARTTLS */ 290506f25ae9SGregory Neil Shapiro 290606f25ae9SGregory Neil Shapiro case O_CLIENTPORT: 290706f25ae9SGregory Neil Shapiro #if DAEMON 290806f25ae9SGregory Neil Shapiro setclientoptions(val); 290906f25ae9SGregory Neil Shapiro #else /* DAEMON */ 291006f25ae9SGregory Neil Shapiro syserr("ClientPortOptions (O option) set but DAEMON not compiled in"); 291106f25ae9SGregory Neil Shapiro #endif /* DAEMON */ 291206f25ae9SGregory Neil Shapiro break; 291306f25ae9SGregory Neil Shapiro 291406f25ae9SGregory Neil Shapiro case O_DF_BUFSIZE: 291506f25ae9SGregory Neil Shapiro DataFileBufferSize = atoi(val); 291606f25ae9SGregory Neil Shapiro break; 291706f25ae9SGregory Neil Shapiro 291806f25ae9SGregory Neil Shapiro case O_XF_BUFSIZE: 291906f25ae9SGregory Neil Shapiro XscriptFileBufferSize = atoi(val); 292006f25ae9SGregory Neil Shapiro break; 292106f25ae9SGregory Neil Shapiro 292206f25ae9SGregory Neil Shapiro case O_LDAPDEFAULTSPEC: 292306f25ae9SGregory Neil Shapiro #ifdef LDAPMAP 292406f25ae9SGregory Neil Shapiro ldapmap_set_defaults(val); 292506f25ae9SGregory Neil Shapiro #else /* LDAPMAP */ 292606f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires LDAP support (-DLDAPMAP)\n", 292706f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 292806f25ae9SGregory Neil Shapiro #endif /* LDAPMAP */ 292906f25ae9SGregory Neil Shapiro break; 293006f25ae9SGregory Neil Shapiro 293106f25ae9SGregory Neil Shapiro #if _FFR_MILTER 293206f25ae9SGregory Neil Shapiro case O_INPUTMILTER: 293306f25ae9SGregory Neil Shapiro InputFilterList = newstr(val); 293406f25ae9SGregory Neil Shapiro break; 293506f25ae9SGregory Neil Shapiro 293606f25ae9SGregory Neil Shapiro case O_MILTER: 293706f25ae9SGregory Neil Shapiro milter_set_option(subopt, val, sticky); 293806f25ae9SGregory Neil Shapiro break; 293906f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 294006f25ae9SGregory Neil Shapiro 294106f25ae9SGregory Neil Shapiro #if _FFR_QUEUE_FILE_MODE 294206f25ae9SGregory Neil Shapiro case O_QUEUE_FILE_MODE: /* queue file mode */ 294306f25ae9SGregory Neil Shapiro QueueFileMode = atooct(val) & 0777; 294406f25ae9SGregory Neil Shapiro break; 294506f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUE_FILE_MODE */ 294625bab6e9SPeter Wemm 2947c2aa98e2SPeter Wemm default: 2948c2aa98e2SPeter Wemm if (tTd(37, 1)) 2949c2aa98e2SPeter Wemm { 2950c2aa98e2SPeter Wemm if (isascii(opt) && isprint(opt)) 295106f25ae9SGregory Neil Shapiro dprintf("Warning: option %c unknown\n", opt); 2952c2aa98e2SPeter Wemm else 295306f25ae9SGregory Neil Shapiro dprintf("Warning: option 0x%x unknown\n", opt); 2954c2aa98e2SPeter Wemm } 2955c2aa98e2SPeter Wemm break; 2956c2aa98e2SPeter Wemm } 295706f25ae9SGregory Neil Shapiro 295806f25ae9SGregory Neil Shapiro /* 295906f25ae9SGregory Neil Shapiro ** Options with suboptions are responsible for taking care 296006f25ae9SGregory Neil Shapiro ** of sticky-ness (e.g., that a command line setting is kept 296106f25ae9SGregory Neil Shapiro ** when reading in the sendmail.cf file). This has to be done 296206f25ae9SGregory Neil Shapiro ** when the suboptions are parsed since each suboption must be 296306f25ae9SGregory Neil Shapiro ** sticky, not the root option. 296406f25ae9SGregory Neil Shapiro */ 296506f25ae9SGregory Neil Shapiro 296606f25ae9SGregory Neil Shapiro if (sticky && !bitset(OI_SUBOPT, o->o_flags)) 2967c2aa98e2SPeter Wemm setbitn(opt, StickyOpt); 2968c2aa98e2SPeter Wemm } 2969c2aa98e2SPeter Wemm /* 2970c2aa98e2SPeter Wemm ** SETCLASS -- set a string into a class 2971c2aa98e2SPeter Wemm ** 2972c2aa98e2SPeter Wemm ** Parameters: 2973c2aa98e2SPeter Wemm ** class -- the class to put the string in. 2974c2aa98e2SPeter Wemm ** str -- the string to enter 2975c2aa98e2SPeter Wemm ** 2976c2aa98e2SPeter Wemm ** Returns: 2977c2aa98e2SPeter Wemm ** none. 2978c2aa98e2SPeter Wemm ** 2979c2aa98e2SPeter Wemm ** Side Effects: 2980c2aa98e2SPeter Wemm ** puts the word into the symbol table. 2981c2aa98e2SPeter Wemm */ 2982c2aa98e2SPeter Wemm 2983c2aa98e2SPeter Wemm void 2984c2aa98e2SPeter Wemm setclass(class, str) 2985c2aa98e2SPeter Wemm int class; 2986c2aa98e2SPeter Wemm char *str; 2987c2aa98e2SPeter Wemm { 2988c2aa98e2SPeter Wemm register STAB *s; 2989c2aa98e2SPeter Wemm 299006f25ae9SGregory Neil Shapiro if ((*str & 0377) == MATCHCLASS) 299106f25ae9SGregory Neil Shapiro { 299206f25ae9SGregory Neil Shapiro int mid; 299306f25ae9SGregory Neil Shapiro 299406f25ae9SGregory Neil Shapiro str++; 299506f25ae9SGregory Neil Shapiro mid = macid(str, NULL); 299606f25ae9SGregory Neil Shapiro if (mid == '\0') 299706f25ae9SGregory Neil Shapiro return; 299806f25ae9SGregory Neil Shapiro 2999c2aa98e2SPeter Wemm if (tTd(37, 8)) 300006f25ae9SGregory Neil Shapiro dprintf("setclass(%s, $=%s)\n", 300106f25ae9SGregory Neil Shapiro macname(class), macname(mid)); 300206f25ae9SGregory Neil Shapiro copy_class(mid, class); 300306f25ae9SGregory Neil Shapiro } 300406f25ae9SGregory Neil Shapiro else 300506f25ae9SGregory Neil Shapiro { 300606f25ae9SGregory Neil Shapiro if (tTd(37, 8)) 300706f25ae9SGregory Neil Shapiro dprintf("setclass(%s, %s)\n", macname(class), str); 300806f25ae9SGregory Neil Shapiro 3009c2aa98e2SPeter Wemm s = stab(str, ST_CLASS, ST_ENTER); 3010c2aa98e2SPeter Wemm setbitn(class, s->s_class); 3011c2aa98e2SPeter Wemm } 301206f25ae9SGregory Neil Shapiro } 3013c2aa98e2SPeter Wemm /* 3014c2aa98e2SPeter Wemm ** MAKEMAPENTRY -- create a map entry 3015c2aa98e2SPeter Wemm ** 3016c2aa98e2SPeter Wemm ** Parameters: 3017c2aa98e2SPeter Wemm ** line -- the config file line 3018c2aa98e2SPeter Wemm ** 3019c2aa98e2SPeter Wemm ** Returns: 3020c2aa98e2SPeter Wemm ** A pointer to the map that has been created. 3021c2aa98e2SPeter Wemm ** NULL if there was a syntax error. 3022c2aa98e2SPeter Wemm ** 3023c2aa98e2SPeter Wemm ** Side Effects: 3024c2aa98e2SPeter Wemm ** Enters the map into the dictionary. 3025c2aa98e2SPeter Wemm */ 3026c2aa98e2SPeter Wemm 3027c2aa98e2SPeter Wemm MAP * 3028c2aa98e2SPeter Wemm makemapentry(line) 3029c2aa98e2SPeter Wemm char *line; 3030c2aa98e2SPeter Wemm { 3031c2aa98e2SPeter Wemm register char *p; 3032c2aa98e2SPeter Wemm char *mapname; 3033c2aa98e2SPeter Wemm char *classname; 3034c2aa98e2SPeter Wemm register STAB *s; 3035c2aa98e2SPeter Wemm STAB *class; 3036c2aa98e2SPeter Wemm 3037c2aa98e2SPeter Wemm for (p = line; isascii(*p) && isspace(*p); p++) 3038c2aa98e2SPeter Wemm continue; 3039c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3040c2aa98e2SPeter Wemm { 3041c2aa98e2SPeter Wemm syserr("readcf: config K line: no map name"); 3042c2aa98e2SPeter Wemm return NULL; 3043c2aa98e2SPeter Wemm } 3044c2aa98e2SPeter Wemm 3045c2aa98e2SPeter Wemm mapname = p; 3046c2aa98e2SPeter Wemm while ((isascii(*++p) && isalnum(*p)) || *p == '_' || *p == '.') 3047c2aa98e2SPeter Wemm continue; 3048c2aa98e2SPeter Wemm if (*p != '\0') 3049c2aa98e2SPeter Wemm *p++ = '\0'; 3050c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3051c2aa98e2SPeter Wemm p++; 3052c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3053c2aa98e2SPeter Wemm { 3054c2aa98e2SPeter Wemm syserr("readcf: config K line, map %s: no map class", mapname); 3055c2aa98e2SPeter Wemm return NULL; 3056c2aa98e2SPeter Wemm } 3057c2aa98e2SPeter Wemm classname = p; 3058c2aa98e2SPeter Wemm while (isascii(*++p) && isalnum(*p)) 3059c2aa98e2SPeter Wemm continue; 3060c2aa98e2SPeter Wemm if (*p != '\0') 3061c2aa98e2SPeter Wemm *p++ = '\0'; 3062c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3063c2aa98e2SPeter Wemm p++; 3064c2aa98e2SPeter Wemm 3065c2aa98e2SPeter Wemm /* look up the class */ 3066c2aa98e2SPeter Wemm class = stab(classname, ST_MAPCLASS, ST_FIND); 3067c2aa98e2SPeter Wemm if (class == NULL) 3068c2aa98e2SPeter Wemm { 3069c2aa98e2SPeter Wemm syserr("readcf: map %s: class %s not available", mapname, classname); 3070c2aa98e2SPeter Wemm return NULL; 3071c2aa98e2SPeter Wemm } 3072c2aa98e2SPeter Wemm 3073c2aa98e2SPeter Wemm /* enter the map */ 3074c2aa98e2SPeter Wemm s = stab(mapname, ST_MAP, ST_ENTER); 3075c2aa98e2SPeter Wemm s->s_map.map_class = &class->s_mapclass; 3076c2aa98e2SPeter Wemm s->s_map.map_mname = newstr(mapname); 3077c2aa98e2SPeter Wemm 3078c2aa98e2SPeter Wemm if (class->s_mapclass.map_parse(&s->s_map, p)) 3079c2aa98e2SPeter Wemm s->s_map.map_mflags |= MF_VALID; 3080c2aa98e2SPeter Wemm 3081c2aa98e2SPeter Wemm if (tTd(37, 5)) 3082c2aa98e2SPeter Wemm { 308306f25ae9SGregory Neil Shapiro dprintf("map %s, class %s, flags %lx, file %s,\n", 3084c2aa98e2SPeter Wemm s->s_map.map_mname, s->s_map.map_class->map_cname, 3085c2aa98e2SPeter Wemm s->s_map.map_mflags, 3086c2aa98e2SPeter Wemm s->s_map.map_file == NULL ? "(null)" : s->s_map.map_file); 308706f25ae9SGregory Neil Shapiro dprintf("\tapp %s, domain %s, rebuild %s\n", 3088c2aa98e2SPeter Wemm s->s_map.map_app == NULL ? "(null)" : s->s_map.map_app, 3089c2aa98e2SPeter Wemm s->s_map.map_domain == NULL ? "(null)" : s->s_map.map_domain, 3090c2aa98e2SPeter Wemm s->s_map.map_rebuild == NULL ? "(null)" : s->s_map.map_rebuild); 3091c2aa98e2SPeter Wemm } 3092c2aa98e2SPeter Wemm 3093c2aa98e2SPeter Wemm return &s->s_map; 3094c2aa98e2SPeter Wemm } 3095c2aa98e2SPeter Wemm /* 3096c2aa98e2SPeter Wemm ** STRTORWSET -- convert string to rewriting set number 3097c2aa98e2SPeter Wemm ** 3098c2aa98e2SPeter Wemm ** Parameters: 3099c2aa98e2SPeter Wemm ** p -- the pointer to the string to decode. 3100c2aa98e2SPeter Wemm ** endp -- if set, store the trailing delimiter here. 3101c2aa98e2SPeter Wemm ** stabmode -- ST_ENTER to create this entry, ST_FIND if 3102c2aa98e2SPeter Wemm ** it must already exist. 3103c2aa98e2SPeter Wemm ** 3104c2aa98e2SPeter Wemm ** Returns: 3105c2aa98e2SPeter Wemm ** The appropriate ruleset number. 3106c2aa98e2SPeter Wemm ** -1 if it is not valid (error already printed) 3107c2aa98e2SPeter Wemm */ 3108c2aa98e2SPeter Wemm 3109c2aa98e2SPeter Wemm int 3110c2aa98e2SPeter Wemm strtorwset(p, endp, stabmode) 3111c2aa98e2SPeter Wemm char *p; 3112c2aa98e2SPeter Wemm char **endp; 3113c2aa98e2SPeter Wemm int stabmode; 3114c2aa98e2SPeter Wemm { 3115c2aa98e2SPeter Wemm int ruleset; 3116c2aa98e2SPeter Wemm static int nextruleset = MAXRWSETS; 3117c2aa98e2SPeter Wemm 3118c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3119c2aa98e2SPeter Wemm p++; 3120c2aa98e2SPeter Wemm if (!isascii(*p)) 3121c2aa98e2SPeter Wemm { 3122c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", p); 3123c2aa98e2SPeter Wemm return -1; 3124c2aa98e2SPeter Wemm } 3125c2aa98e2SPeter Wemm if (isdigit(*p)) 3126c2aa98e2SPeter Wemm { 3127c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3128c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3129c2aa98e2SPeter Wemm { 3130c2aa98e2SPeter Wemm syserr("bad ruleset %d (%d max)", 3131c2aa98e2SPeter Wemm ruleset, MAXRWSETS / 2); 3132c2aa98e2SPeter Wemm ruleset = -1; 3133c2aa98e2SPeter Wemm } 3134c2aa98e2SPeter Wemm } 3135c2aa98e2SPeter Wemm else 3136c2aa98e2SPeter Wemm { 3137c2aa98e2SPeter Wemm STAB *s; 3138c2aa98e2SPeter Wemm char delim; 313906f25ae9SGregory Neil Shapiro char *q = NULL; 3140c2aa98e2SPeter Wemm 3141c2aa98e2SPeter Wemm q = p; 3142c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && 3143c2aa98e2SPeter Wemm (isalnum(*p) || *p == '_')) 3144c2aa98e2SPeter Wemm p++; 3145c2aa98e2SPeter Wemm if (q == p || !(isascii(*q) && isalpha(*q))) 3146c2aa98e2SPeter Wemm { 3147c2aa98e2SPeter Wemm /* no valid characters */ 3148c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", q); 3149c2aa98e2SPeter Wemm return -1; 3150c2aa98e2SPeter Wemm } 3151c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3152c2aa98e2SPeter Wemm *p++ = '\0'; 3153c2aa98e2SPeter Wemm delim = *p; 3154c2aa98e2SPeter Wemm if (delim != '\0') 3155c2aa98e2SPeter Wemm *p = '\0'; 3156c2aa98e2SPeter Wemm s = stab(q, ST_RULESET, stabmode); 3157c2aa98e2SPeter Wemm if (delim != '\0') 3158c2aa98e2SPeter Wemm *p = delim; 3159c2aa98e2SPeter Wemm 3160c2aa98e2SPeter Wemm if (s == NULL) 3161c2aa98e2SPeter Wemm return -1; 3162c2aa98e2SPeter Wemm 3163c2aa98e2SPeter Wemm if (stabmode == ST_ENTER && delim == '=') 3164c2aa98e2SPeter Wemm { 3165c2aa98e2SPeter Wemm while (isascii(*++p) && isspace(*p)) 3166c2aa98e2SPeter Wemm continue; 3167c2aa98e2SPeter Wemm if (!(isascii(*p) && isdigit(*p))) 3168c2aa98e2SPeter Wemm { 3169c2aa98e2SPeter Wemm syserr("bad ruleset definition \"%s\" (number required after `=')", q); 3170c2aa98e2SPeter Wemm ruleset = -1; 3171c2aa98e2SPeter Wemm } 3172c2aa98e2SPeter Wemm else 3173c2aa98e2SPeter Wemm { 3174c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3175c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3176c2aa98e2SPeter Wemm { 3177c2aa98e2SPeter Wemm syserr("bad ruleset number %d in \"%s\" (%d max)", 3178c2aa98e2SPeter Wemm ruleset, q, MAXRWSETS / 2); 3179c2aa98e2SPeter Wemm ruleset = -1; 3180c2aa98e2SPeter Wemm } 3181c2aa98e2SPeter Wemm } 3182c2aa98e2SPeter Wemm } 3183c2aa98e2SPeter Wemm else 3184c2aa98e2SPeter Wemm { 3185c2aa98e2SPeter Wemm if (endp != NULL) 3186c2aa98e2SPeter Wemm *endp = p; 318706f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0) 3188c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3189c2aa98e2SPeter Wemm else if ((ruleset = --nextruleset) < MAXRWSETS / 2) 3190c2aa98e2SPeter Wemm { 3191c2aa98e2SPeter Wemm syserr("%s: too many named rulesets (%d max)", 3192c2aa98e2SPeter Wemm q, MAXRWSETS / 2); 3193c2aa98e2SPeter Wemm ruleset = -1; 3194c2aa98e2SPeter Wemm } 3195c2aa98e2SPeter Wemm } 319606f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0 && 319706f25ae9SGregory Neil Shapiro ruleset >= 0 && 319806f25ae9SGregory Neil Shapiro ruleset != s->s_ruleset) 3199c2aa98e2SPeter Wemm { 3200c2aa98e2SPeter Wemm syserr("%s: ruleset changed value (old %d, new %d)", 3201c2aa98e2SPeter Wemm q, s->s_ruleset, ruleset); 3202c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3203c2aa98e2SPeter Wemm } 320406f25ae9SGregory Neil Shapiro else if (ruleset >= 0) 3205c2aa98e2SPeter Wemm { 3206c2aa98e2SPeter Wemm s->s_ruleset = ruleset; 3207c2aa98e2SPeter Wemm } 320806f25ae9SGregory Neil Shapiro if (stabmode == ST_ENTER) 320906f25ae9SGregory Neil Shapiro { 321006f25ae9SGregory Neil Shapiro char *h = NULL; 321106f25ae9SGregory Neil Shapiro 321206f25ae9SGregory Neil Shapiro if (RuleSetNames[ruleset] != NULL) 321306f25ae9SGregory Neil Shapiro free(RuleSetNames[ruleset]); 321406f25ae9SGregory Neil Shapiro if (delim != '\0' && (h = strchr(q, delim)) != NULL) 321506f25ae9SGregory Neil Shapiro *h = '\0'; 321606f25ae9SGregory Neil Shapiro RuleSetNames[ruleset] = newstr(q); 321706f25ae9SGregory Neil Shapiro if (delim == '/' && h != NULL) 321806f25ae9SGregory Neil Shapiro *h = delim; /* put back delim */ 321906f25ae9SGregory Neil Shapiro } 3220c2aa98e2SPeter Wemm } 3221c2aa98e2SPeter Wemm return ruleset; 3222c2aa98e2SPeter Wemm } 3223c2aa98e2SPeter Wemm /* 322406f25ae9SGregory Neil Shapiro ** SETTIMEOUT -- set an individual timeout 322506f25ae9SGregory Neil Shapiro ** 322606f25ae9SGregory Neil Shapiro ** Parameters: 322706f25ae9SGregory Neil Shapiro ** name -- the name of the timeout. 322806f25ae9SGregory Neil Shapiro ** val -- the value of the timeout. 322906f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 323006f25ae9SGregory Neil Shapiro ** this value. 323106f25ae9SGregory Neil Shapiro ** 323206f25ae9SGregory Neil Shapiro ** Returns: 323306f25ae9SGregory Neil Shapiro ** none. 323406f25ae9SGregory Neil Shapiro */ 323506f25ae9SGregory Neil Shapiro 323606f25ae9SGregory Neil Shapiro /* set if Timeout sub-option is stuck */ 323706f25ae9SGregory Neil Shapiro static BITMAP256 StickyTimeoutOpt; 323806f25ae9SGregory Neil Shapiro 323906f25ae9SGregory Neil Shapiro static struct timeoutinfo 324006f25ae9SGregory Neil Shapiro { 324106f25ae9SGregory Neil Shapiro char *to_name; /* long name of timeout */ 324206f25ae9SGregory Neil Shapiro u_char to_code; /* code for option */ 324306f25ae9SGregory Neil Shapiro } TimeOutTab[] = 324406f25ae9SGregory Neil Shapiro { 324506f25ae9SGregory Neil Shapiro #define TO_INITIAL 0x01 324606f25ae9SGregory Neil Shapiro { "initial", TO_INITIAL }, 324706f25ae9SGregory Neil Shapiro #define TO_MAIL 0x02 324806f25ae9SGregory Neil Shapiro { "mail", TO_MAIL }, 324906f25ae9SGregory Neil Shapiro #define TO_RCPT 0x03 325006f25ae9SGregory Neil Shapiro { "rcpt", TO_RCPT }, 325106f25ae9SGregory Neil Shapiro #define TO_DATAINIT 0x04 325206f25ae9SGregory Neil Shapiro { "datainit", TO_DATAINIT }, 325306f25ae9SGregory Neil Shapiro #define TO_DATABLOCK 0x05 325406f25ae9SGregory Neil Shapiro { "datablock", TO_DATABLOCK }, 325506f25ae9SGregory Neil Shapiro #define TO_DATAFINAL 0x06 325606f25ae9SGregory Neil Shapiro { "datafinal", TO_DATAFINAL }, 325706f25ae9SGregory Neil Shapiro #define TO_COMMAND 0x07 325806f25ae9SGregory Neil Shapiro { "command", TO_COMMAND }, 325906f25ae9SGregory Neil Shapiro #define TO_RSET 0x08 326006f25ae9SGregory Neil Shapiro { "rset", TO_RSET }, 326106f25ae9SGregory Neil Shapiro #define TO_HELO 0x09 326206f25ae9SGregory Neil Shapiro { "helo", TO_HELO }, 326306f25ae9SGregory Neil Shapiro #define TO_QUIT 0x0A 326406f25ae9SGregory Neil Shapiro { "quit", TO_QUIT }, 326506f25ae9SGregory Neil Shapiro #define TO_MISC 0x0B 326606f25ae9SGregory Neil Shapiro { "misc", TO_MISC }, 326706f25ae9SGregory Neil Shapiro #define TO_IDENT 0x0C 326806f25ae9SGregory Neil Shapiro { "ident", TO_IDENT }, 326906f25ae9SGregory Neil Shapiro #define TO_FILEOPEN 0x0D 327006f25ae9SGregory Neil Shapiro { "fileopen", TO_FILEOPEN }, 327106f25ae9SGregory Neil Shapiro #define TO_CONNECT 0x0E 327206f25ae9SGregory Neil Shapiro { "connect", TO_CONNECT }, 327306f25ae9SGregory Neil Shapiro #define TO_ICONNECT 0x0F 327406f25ae9SGregory Neil Shapiro { "iconnect", TO_ICONNECT }, 327506f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN 0x10 327606f25ae9SGregory Neil Shapiro { "queuewarn", TO_QUEUEWARN }, 327706f25ae9SGregory Neil Shapiro { "queuewarn.*", TO_QUEUEWARN }, 327806f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NORMAL 0x11 327906f25ae9SGregory Neil Shapiro { "queuewarn.normal", TO_QUEUEWARN_NORMAL }, 328006f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_URGENT 0x12 328106f25ae9SGregory Neil Shapiro { "queuewarn.urgent", TO_QUEUEWARN_URGENT }, 328206f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NON_URGENT 0x13 328306f25ae9SGregory Neil Shapiro { "queuewarn.non-urgent", TO_QUEUEWARN_NON_URGENT }, 328406f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN 0x14 328506f25ae9SGregory Neil Shapiro { "queuereturn", TO_QUEUERETURN }, 328606f25ae9SGregory Neil Shapiro { "queuereturn.*", TO_QUEUERETURN }, 328706f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NORMAL 0x15 328806f25ae9SGregory Neil Shapiro { "queuereturn.normal", TO_QUEUERETURN_NORMAL }, 328906f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_URGENT 0x16 329006f25ae9SGregory Neil Shapiro { "queuereturn.urgent", TO_QUEUERETURN_URGENT }, 329106f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NON_URGENT 0x17 329206f25ae9SGregory Neil Shapiro { "queuereturn.non-urgent", TO_QUEUERETURN_NON_URGENT }, 329306f25ae9SGregory Neil Shapiro #define TO_HOSTSTATUS 0x18 329406f25ae9SGregory Neil Shapiro { "hoststatus", TO_HOSTSTATUS }, 329506f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS 0x19 329606f25ae9SGregory Neil Shapiro { "resolver.retrans", TO_RESOLVER_RETRANS }, 329706f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_NORMAL 0x1A 329806f25ae9SGregory Neil Shapiro { "resolver.retrans.normal", TO_RESOLVER_RETRANS_NORMAL }, 329906f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_FIRST 0x1B 330006f25ae9SGregory Neil Shapiro { "resolver.retrans.first", TO_RESOLVER_RETRANS_FIRST }, 330106f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY 0x1C 330206f25ae9SGregory Neil Shapiro { "resolver.retry", TO_RESOLVER_RETRY }, 330306f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_NORMAL 0x1D 330406f25ae9SGregory Neil Shapiro { "resolver.retry.normal", TO_RESOLVER_RETRY_NORMAL }, 330506f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_FIRST 0x1E 330606f25ae9SGregory Neil Shapiro { "resolver.retry.first", TO_RESOLVER_RETRY_FIRST }, 330706f25ae9SGregory Neil Shapiro #define TO_CONTROL 0x1F 330806f25ae9SGregory Neil Shapiro { "control", TO_CONTROL }, 330906f25ae9SGregory Neil Shapiro { NULL, 0 }, 331006f25ae9SGregory Neil Shapiro }; 331106f25ae9SGregory Neil Shapiro 331206f25ae9SGregory Neil Shapiro 331306f25ae9SGregory Neil Shapiro static void 331406f25ae9SGregory Neil Shapiro settimeout(name, val, sticky) 331506f25ae9SGregory Neil Shapiro char *name; 331606f25ae9SGregory Neil Shapiro char *val; 331706f25ae9SGregory Neil Shapiro bool sticky; 331806f25ae9SGregory Neil Shapiro { 331906f25ae9SGregory Neil Shapiro register struct timeoutinfo *to; 332006f25ae9SGregory Neil Shapiro int i; 332106f25ae9SGregory Neil Shapiro time_t toval; 332206f25ae9SGregory Neil Shapiro 332306f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 332406f25ae9SGregory Neil Shapiro dprintf("settimeout(%s = %s)", name, val); 332506f25ae9SGregory Neil Shapiro 332606f25ae9SGregory Neil Shapiro for (to = TimeOutTab; to->to_name != NULL; to++) 332706f25ae9SGregory Neil Shapiro { 332806f25ae9SGregory Neil Shapiro if (strcasecmp(to->to_name, name) == 0) 332906f25ae9SGregory Neil Shapiro break; 333006f25ae9SGregory Neil Shapiro } 333106f25ae9SGregory Neil Shapiro 333206f25ae9SGregory Neil Shapiro if (to->to_name == NULL) 333306f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 333406f25ae9SGregory Neil Shapiro 333506f25ae9SGregory Neil Shapiro /* 333606f25ae9SGregory Neil Shapiro ** See if this option is preset for us. 333706f25ae9SGregory Neil Shapiro */ 333806f25ae9SGregory Neil Shapiro 333906f25ae9SGregory Neil Shapiro if (!sticky && bitnset(to->to_code, StickyTimeoutOpt)) 334006f25ae9SGregory Neil Shapiro { 334106f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 334206f25ae9SGregory Neil Shapiro dprintf(" (ignored)\n"); 334306f25ae9SGregory Neil Shapiro return; 334406f25ae9SGregory Neil Shapiro } 334506f25ae9SGregory Neil Shapiro 334606f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 334706f25ae9SGregory Neil Shapiro dprintf("\n"); 334806f25ae9SGregory Neil Shapiro 334906f25ae9SGregory Neil Shapiro toval = convtime(val, 'm'); 335006f25ae9SGregory Neil Shapiro 335106f25ae9SGregory Neil Shapiro switch (to->to_code) 335206f25ae9SGregory Neil Shapiro { 335306f25ae9SGregory Neil Shapiro case TO_INITIAL: 335406f25ae9SGregory Neil Shapiro TimeOuts.to_initial = toval; 335506f25ae9SGregory Neil Shapiro break; 335606f25ae9SGregory Neil Shapiro 335706f25ae9SGregory Neil Shapiro case TO_MAIL: 335806f25ae9SGregory Neil Shapiro TimeOuts.to_mail = toval; 335906f25ae9SGregory Neil Shapiro break; 336006f25ae9SGregory Neil Shapiro 336106f25ae9SGregory Neil Shapiro case TO_RCPT: 336206f25ae9SGregory Neil Shapiro TimeOuts.to_rcpt = toval; 336306f25ae9SGregory Neil Shapiro break; 336406f25ae9SGregory Neil Shapiro 336506f25ae9SGregory Neil Shapiro case TO_DATAINIT: 336606f25ae9SGregory Neil Shapiro TimeOuts.to_datainit = toval; 336706f25ae9SGregory Neil Shapiro break; 336806f25ae9SGregory Neil Shapiro 336906f25ae9SGregory Neil Shapiro case TO_DATABLOCK: 337006f25ae9SGregory Neil Shapiro TimeOuts.to_datablock = toval; 337106f25ae9SGregory Neil Shapiro break; 337206f25ae9SGregory Neil Shapiro 337306f25ae9SGregory Neil Shapiro case TO_DATAFINAL: 337406f25ae9SGregory Neil Shapiro TimeOuts.to_datafinal = toval; 337506f25ae9SGregory Neil Shapiro break; 337606f25ae9SGregory Neil Shapiro 337706f25ae9SGregory Neil Shapiro case TO_COMMAND: 337806f25ae9SGregory Neil Shapiro TimeOuts.to_nextcommand = toval; 337906f25ae9SGregory Neil Shapiro break; 338006f25ae9SGregory Neil Shapiro 338106f25ae9SGregory Neil Shapiro case TO_RSET: 338206f25ae9SGregory Neil Shapiro TimeOuts.to_rset = toval; 338306f25ae9SGregory Neil Shapiro break; 338406f25ae9SGregory Neil Shapiro 338506f25ae9SGregory Neil Shapiro case TO_HELO: 338606f25ae9SGregory Neil Shapiro TimeOuts.to_helo = toval; 338706f25ae9SGregory Neil Shapiro break; 338806f25ae9SGregory Neil Shapiro 338906f25ae9SGregory Neil Shapiro case TO_QUIT: 339006f25ae9SGregory Neil Shapiro TimeOuts.to_quit = toval; 339106f25ae9SGregory Neil Shapiro break; 339206f25ae9SGregory Neil Shapiro 339306f25ae9SGregory Neil Shapiro case TO_MISC: 339406f25ae9SGregory Neil Shapiro TimeOuts.to_miscshort = toval; 339506f25ae9SGregory Neil Shapiro break; 339606f25ae9SGregory Neil Shapiro 339706f25ae9SGregory Neil Shapiro case TO_IDENT: 339806f25ae9SGregory Neil Shapiro TimeOuts.to_ident = toval; 339906f25ae9SGregory Neil Shapiro break; 340006f25ae9SGregory Neil Shapiro 340106f25ae9SGregory Neil Shapiro case TO_FILEOPEN: 340206f25ae9SGregory Neil Shapiro TimeOuts.to_fileopen = toval; 340306f25ae9SGregory Neil Shapiro break; 340406f25ae9SGregory Neil Shapiro 340506f25ae9SGregory Neil Shapiro case TO_CONNECT: 340606f25ae9SGregory Neil Shapiro TimeOuts.to_connect = toval; 340706f25ae9SGregory Neil Shapiro break; 340806f25ae9SGregory Neil Shapiro 340906f25ae9SGregory Neil Shapiro case TO_ICONNECT: 341006f25ae9SGregory Neil Shapiro TimeOuts.to_iconnect = toval; 341106f25ae9SGregory Neil Shapiro break; 341206f25ae9SGregory Neil Shapiro 341306f25ae9SGregory Neil Shapiro case TO_QUEUEWARN: 341406f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 341506f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 341606f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 341706f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 341806f25ae9SGregory Neil Shapiro break; 341906f25ae9SGregory Neil Shapiro 342006f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NORMAL: 342106f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 342206f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 342306f25ae9SGregory Neil Shapiro break; 342406f25ae9SGregory Neil Shapiro 342506f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_URGENT: 342606f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 342706f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 342806f25ae9SGregory Neil Shapiro break; 342906f25ae9SGregory Neil Shapiro 343006f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NON_URGENT: 343106f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 343206f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 343306f25ae9SGregory Neil Shapiro break; 343406f25ae9SGregory Neil Shapiro 343506f25ae9SGregory Neil Shapiro case TO_QUEUERETURN: 343606f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 343706f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 343806f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 343906f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 344006f25ae9SGregory Neil Shapiro break; 344106f25ae9SGregory Neil Shapiro 344206f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NORMAL: 344306f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 344406f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 344506f25ae9SGregory Neil Shapiro break; 344606f25ae9SGregory Neil Shapiro 344706f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_URGENT: 344806f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 344906f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 345006f25ae9SGregory Neil Shapiro break; 345106f25ae9SGregory Neil Shapiro 345206f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NON_URGENT: 345306f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 345406f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 345506f25ae9SGregory Neil Shapiro break; 345606f25ae9SGregory Neil Shapiro 345706f25ae9SGregory Neil Shapiro 345806f25ae9SGregory Neil Shapiro case TO_HOSTSTATUS: 345906f25ae9SGregory Neil Shapiro MciInfoTimeout = toval; 346006f25ae9SGregory Neil Shapiro break; 346106f25ae9SGregory Neil Shapiro 346206f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS: 346306f25ae9SGregory Neil Shapiro toval = convtime(val, 's'); 346406f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_DEFAULT] = toval; 346506f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = toval; 346606f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = toval; 346706f25ae9SGregory Neil Shapiro break; 346806f25ae9SGregory Neil Shapiro 346906f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY: 347006f25ae9SGregory Neil Shapiro i = atoi(val); 347106f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_DEFAULT] = i; 347206f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = i; 347306f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = i; 347406f25ae9SGregory Neil Shapiro break; 347506f25ae9SGregory Neil Shapiro 347606f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_NORMAL: 347706f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = convtime(val, 's'); 347806f25ae9SGregory Neil Shapiro break; 347906f25ae9SGregory Neil Shapiro 348006f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_NORMAL: 348106f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = atoi(val); 348206f25ae9SGregory Neil Shapiro break; 348306f25ae9SGregory Neil Shapiro 348406f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_FIRST: 348506f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = convtime(val, 's'); 348606f25ae9SGregory Neil Shapiro break; 348706f25ae9SGregory Neil Shapiro 348806f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_FIRST: 348906f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = atoi(val); 349006f25ae9SGregory Neil Shapiro break; 349106f25ae9SGregory Neil Shapiro 349206f25ae9SGregory Neil Shapiro case TO_CONTROL: 349306f25ae9SGregory Neil Shapiro TimeOuts.to_control = toval; 349406f25ae9SGregory Neil Shapiro break; 349506f25ae9SGregory Neil Shapiro 349606f25ae9SGregory Neil Shapiro default: 349706f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 349806f25ae9SGregory Neil Shapiro break; 349906f25ae9SGregory Neil Shapiro } 350006f25ae9SGregory Neil Shapiro 350106f25ae9SGregory Neil Shapiro if (sticky) 350206f25ae9SGregory Neil Shapiro setbitn(to->to_code, StickyTimeoutOpt); 350306f25ae9SGregory Neil Shapiro } 350406f25ae9SGregory Neil Shapiro /* 3505c2aa98e2SPeter Wemm ** INITTIMEOUTS -- parse and set timeout values 3506c2aa98e2SPeter Wemm ** 3507c2aa98e2SPeter Wemm ** Parameters: 3508c2aa98e2SPeter Wemm ** val -- a pointer to the values. If NULL, do initial 3509c2aa98e2SPeter Wemm ** settings. 351006f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 351106f25ae9SGregory Neil Shapiro ** this suboption value. 3512c2aa98e2SPeter Wemm ** 3513c2aa98e2SPeter Wemm ** Returns: 3514c2aa98e2SPeter Wemm ** none. 3515c2aa98e2SPeter Wemm ** 3516c2aa98e2SPeter Wemm ** Side Effects: 3517c2aa98e2SPeter Wemm ** Initializes the TimeOuts structure 3518c2aa98e2SPeter Wemm */ 3519c2aa98e2SPeter Wemm 3520c2aa98e2SPeter Wemm void 352106f25ae9SGregory Neil Shapiro inittimeouts(val, sticky) 3522c2aa98e2SPeter Wemm register char *val; 352306f25ae9SGregory Neil Shapiro bool sticky; 3524c2aa98e2SPeter Wemm { 3525c2aa98e2SPeter Wemm register char *p; 3526c2aa98e2SPeter Wemm 3527c2aa98e2SPeter Wemm if (tTd(37, 2)) 352806f25ae9SGregory Neil Shapiro dprintf("inittimeouts(%s)\n", val == NULL ? "<NULL>" : val); 3529c2aa98e2SPeter Wemm if (val == NULL) 3530c2aa98e2SPeter Wemm { 3531c2aa98e2SPeter Wemm TimeOuts.to_connect = (time_t) 0 SECONDS; 3532c2aa98e2SPeter Wemm TimeOuts.to_initial = (time_t) 5 MINUTES; 3533c2aa98e2SPeter Wemm TimeOuts.to_helo = (time_t) 5 MINUTES; 3534c2aa98e2SPeter Wemm TimeOuts.to_mail = (time_t) 10 MINUTES; 3535c2aa98e2SPeter Wemm TimeOuts.to_rcpt = (time_t) 1 HOUR; 3536c2aa98e2SPeter Wemm TimeOuts.to_datainit = (time_t) 5 MINUTES; 3537c2aa98e2SPeter Wemm TimeOuts.to_datablock = (time_t) 1 HOUR; 3538c2aa98e2SPeter Wemm TimeOuts.to_datafinal = (time_t) 1 HOUR; 3539c2aa98e2SPeter Wemm TimeOuts.to_rset = (time_t) 5 MINUTES; 3540c2aa98e2SPeter Wemm TimeOuts.to_quit = (time_t) 2 MINUTES; 3541c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = (time_t) 1 HOUR; 3542c2aa98e2SPeter Wemm TimeOuts.to_miscshort = (time_t) 2 MINUTES; 3543c2aa98e2SPeter Wemm #if IDENTPROTO 354406f25ae9SGregory Neil Shapiro TimeOuts.to_ident = (time_t) 5 SECONDS; 354506f25ae9SGregory Neil Shapiro #else /* IDENTPROTO */ 3546c2aa98e2SPeter Wemm TimeOuts.to_ident = (time_t) 0 SECONDS; 354706f25ae9SGregory Neil Shapiro #endif /* IDENTPROTO */ 3548c2aa98e2SPeter Wemm TimeOuts.to_fileopen = (time_t) 60 SECONDS; 354906f25ae9SGregory Neil Shapiro TimeOuts.to_control = (time_t) 2 MINUTES; 3550c2aa98e2SPeter Wemm if (tTd(37, 5)) 3551c2aa98e2SPeter Wemm { 355206f25ae9SGregory Neil Shapiro dprintf("Timeouts:\n"); 355306f25ae9SGregory Neil Shapiro dprintf(" connect = %ld\n", (long)TimeOuts.to_connect); 355406f25ae9SGregory Neil Shapiro dprintf(" initial = %ld\n", (long)TimeOuts.to_initial); 355506f25ae9SGregory Neil Shapiro dprintf(" helo = %ld\n", (long)TimeOuts.to_helo); 355606f25ae9SGregory Neil Shapiro dprintf(" mail = %ld\n", (long)TimeOuts.to_mail); 355706f25ae9SGregory Neil Shapiro dprintf(" rcpt = %ld\n", (long)TimeOuts.to_rcpt); 355806f25ae9SGregory Neil Shapiro dprintf(" datainit = %ld\n", (long)TimeOuts.to_datainit); 355906f25ae9SGregory Neil Shapiro dprintf(" datablock = %ld\n", (long)TimeOuts.to_datablock); 356006f25ae9SGregory Neil Shapiro dprintf(" datafinal = %ld\n", (long)TimeOuts.to_datafinal); 356106f25ae9SGregory Neil Shapiro dprintf(" rset = %ld\n", (long)TimeOuts.to_rset); 356206f25ae9SGregory Neil Shapiro dprintf(" quit = %ld\n", (long)TimeOuts.to_quit); 356306f25ae9SGregory Neil Shapiro dprintf(" nextcommand = %ld\n", (long)TimeOuts.to_nextcommand); 356406f25ae9SGregory Neil Shapiro dprintf(" miscshort = %ld\n", (long)TimeOuts.to_miscshort); 356506f25ae9SGregory Neil Shapiro dprintf(" ident = %ld\n", (long)TimeOuts.to_ident); 356606f25ae9SGregory Neil Shapiro dprintf(" fileopen = %ld\n", (long)TimeOuts.to_fileopen); 356706f25ae9SGregory Neil Shapiro dprintf(" control = %ld\n", (long)TimeOuts.to_control); 3568c2aa98e2SPeter Wemm } 3569c2aa98e2SPeter Wemm return; 3570c2aa98e2SPeter Wemm } 3571c2aa98e2SPeter Wemm 3572c2aa98e2SPeter Wemm for (;; val = p) 3573c2aa98e2SPeter Wemm { 3574c2aa98e2SPeter Wemm while (isascii(*val) && isspace(*val)) 3575c2aa98e2SPeter Wemm val++; 3576c2aa98e2SPeter Wemm if (*val == '\0') 3577c2aa98e2SPeter Wemm break; 3578c2aa98e2SPeter Wemm for (p = val; *p != '\0' && *p != ','; p++) 3579c2aa98e2SPeter Wemm continue; 3580c2aa98e2SPeter Wemm if (*p != '\0') 3581c2aa98e2SPeter Wemm *p++ = '\0'; 3582c2aa98e2SPeter Wemm 3583c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 3584c2aa98e2SPeter Wemm { 3585c2aa98e2SPeter Wemm /* old syntax -- set everything */ 3586c2aa98e2SPeter Wemm TimeOuts.to_mail = convtime(val, 'm'); 3587c2aa98e2SPeter Wemm TimeOuts.to_rcpt = TimeOuts.to_mail; 3588c2aa98e2SPeter Wemm TimeOuts.to_datainit = TimeOuts.to_mail; 3589c2aa98e2SPeter Wemm TimeOuts.to_datablock = TimeOuts.to_mail; 3590c2aa98e2SPeter Wemm TimeOuts.to_datafinal = TimeOuts.to_mail; 3591c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = TimeOuts.to_mail; 359206f25ae9SGregory Neil Shapiro if (sticky) 359306f25ae9SGregory Neil Shapiro { 359406f25ae9SGregory Neil Shapiro setbitn(TO_MAIL, StickyTimeoutOpt); 359506f25ae9SGregory Neil Shapiro setbitn(TO_RCPT, StickyTimeoutOpt); 359606f25ae9SGregory Neil Shapiro setbitn(TO_DATAINIT, StickyTimeoutOpt); 359706f25ae9SGregory Neil Shapiro setbitn(TO_DATABLOCK, StickyTimeoutOpt); 359806f25ae9SGregory Neil Shapiro setbitn(TO_DATAFINAL, StickyTimeoutOpt); 359906f25ae9SGregory Neil Shapiro setbitn(TO_COMMAND, StickyTimeoutOpt); 360006f25ae9SGregory Neil Shapiro } 3601c2aa98e2SPeter Wemm continue; 3602c2aa98e2SPeter Wemm } 3603c2aa98e2SPeter Wemm else 3604c2aa98e2SPeter Wemm { 3605c2aa98e2SPeter Wemm register char *q = strchr(val, ':'); 3606c2aa98e2SPeter Wemm 3607c2aa98e2SPeter Wemm if (q == NULL && (q = strchr(val, '=')) == NULL) 3608c2aa98e2SPeter Wemm { 3609c2aa98e2SPeter Wemm /* syntax error */ 3610c2aa98e2SPeter Wemm continue; 3611c2aa98e2SPeter Wemm } 3612c2aa98e2SPeter Wemm *q++ = '\0'; 361306f25ae9SGregory Neil Shapiro settimeout(val, q, sticky); 3614c2aa98e2SPeter Wemm } 3615c2aa98e2SPeter Wemm } 3616c2aa98e2SPeter Wemm } 3617