1c2aa98e2SPeter Wemm /* 2602a2b1bSGregory Neil Shapiro * Copyright (c) 1998-2001 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 15602a2b1bSGregory Neil Shapiro static char id[] = "@(#)$Id: readcf.c,v 8.382.4.38 2001/02/17 00:05:12 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); 359193538b7SGregory Neil Shapiro if (mid == 0) 360193538b7SGregory Neil Shapiro break; 361c2aa98e2SPeter Wemm p = munchstring(ep, NULL, '\0'); 362c2aa98e2SPeter Wemm define(mid, newstr(p), e); 363c2aa98e2SPeter Wemm break; 364c2aa98e2SPeter Wemm 365c2aa98e2SPeter Wemm case 'H': /* required header line */ 36606f25ae9SGregory Neil Shapiro (void) chompheader(&bp[1], CHHDR_DEF, NULL, e); 367c2aa98e2SPeter Wemm break; 368c2aa98e2SPeter Wemm 369c2aa98e2SPeter Wemm case 'C': /* word class */ 370c2aa98e2SPeter Wemm case 'T': /* trusted user (set class `t') */ 371c2aa98e2SPeter Wemm if (bp[0] == 'C') 372c2aa98e2SPeter Wemm { 373c2aa98e2SPeter Wemm mid = macid(&bp[1], &ep); 374193538b7SGregory Neil Shapiro if (mid == 0) 375193538b7SGregory Neil Shapiro break; 376c2aa98e2SPeter Wemm expand(ep, exbuf, sizeof exbuf, e); 377c2aa98e2SPeter Wemm p = exbuf; 378c2aa98e2SPeter Wemm } 379c2aa98e2SPeter Wemm else 380c2aa98e2SPeter Wemm { 381c2aa98e2SPeter Wemm mid = 't'; 382c2aa98e2SPeter Wemm p = &bp[1]; 383c2aa98e2SPeter Wemm } 384c2aa98e2SPeter Wemm while (*p != '\0') 385c2aa98e2SPeter Wemm { 386c2aa98e2SPeter Wemm register char *wd; 387c2aa98e2SPeter Wemm char delim; 388c2aa98e2SPeter Wemm 389c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && isspace(*p)) 390c2aa98e2SPeter Wemm p++; 391c2aa98e2SPeter Wemm wd = p; 392c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 393c2aa98e2SPeter Wemm p++; 394c2aa98e2SPeter Wemm delim = *p; 395c2aa98e2SPeter Wemm *p = '\0'; 396c2aa98e2SPeter Wemm if (wd[0] != '\0') 397c2aa98e2SPeter Wemm setclass(mid, wd); 398c2aa98e2SPeter Wemm *p = delim; 399c2aa98e2SPeter Wemm } 400c2aa98e2SPeter Wemm break; 401c2aa98e2SPeter Wemm 402c2aa98e2SPeter Wemm case 'F': /* word class from file */ 403c2aa98e2SPeter Wemm mid = macid(&bp[1], &ep); 404193538b7SGregory Neil Shapiro if (mid == 0) 405193538b7SGregory Neil Shapiro break; 406c2aa98e2SPeter Wemm for (p = ep; isascii(*p) && isspace(*p); ) 407c2aa98e2SPeter Wemm p++; 408c2aa98e2SPeter Wemm if (p[0] == '-' && p[1] == 'o') 409c2aa98e2SPeter Wemm { 410c2aa98e2SPeter Wemm optional = TRUE; 411c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 412c2aa98e2SPeter Wemm p++; 413c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 414c2aa98e2SPeter Wemm p++; 415c2aa98e2SPeter Wemm } 416c2aa98e2SPeter Wemm else 417c2aa98e2SPeter Wemm optional = FALSE; 41806f25ae9SGregory Neil Shapiro 419c2aa98e2SPeter Wemm file = p; 42006f25ae9SGregory Neil Shapiro q = p; 42106f25ae9SGregory Neil Shapiro while (*q != '\0' && !(isascii(*q) && isspace(*q))) 42206f25ae9SGregory Neil Shapiro q++; 423c2aa98e2SPeter Wemm if (*file == '|') 424c2aa98e2SPeter Wemm p = "%s"; 425c2aa98e2SPeter Wemm else 426c2aa98e2SPeter Wemm { 42706f25ae9SGregory Neil Shapiro p = q; 428c2aa98e2SPeter Wemm if (*p == '\0') 429c2aa98e2SPeter Wemm p = "%s"; 430c2aa98e2SPeter Wemm else 431c2aa98e2SPeter Wemm { 432c2aa98e2SPeter Wemm *p = '\0'; 433c2aa98e2SPeter Wemm while (isascii(*++p) && isspace(*p)) 434c2aa98e2SPeter Wemm continue; 435c2aa98e2SPeter Wemm } 436c2aa98e2SPeter Wemm } 437c2aa98e2SPeter Wemm fileclass(mid, file, p, safe, optional); 438c2aa98e2SPeter Wemm break; 439c2aa98e2SPeter Wemm 440c2aa98e2SPeter Wemm #ifdef XLA 441c2aa98e2SPeter Wemm case 'L': /* extended load average description */ 442c2aa98e2SPeter Wemm xla_init(&bp[1]); 443c2aa98e2SPeter Wemm break; 44406f25ae9SGregory Neil Shapiro #endif /* XLA */ 445c2aa98e2SPeter Wemm 446c2aa98e2SPeter Wemm #if defined(SUN_EXTENSIONS) && defined(SUN_LOOKUP_MACRO) 447c2aa98e2SPeter Wemm case 'L': /* lookup macro */ 448c2aa98e2SPeter Wemm case 'G': /* lookup class */ 449c2aa98e2SPeter Wemm /* reserved for Sun -- NIS+ database lookup */ 450c2aa98e2SPeter Wemm if (VendorCode != VENDOR_SUN) 451c2aa98e2SPeter Wemm goto badline; 452c2aa98e2SPeter Wemm sun_lg_config_line(bp, e); 453c2aa98e2SPeter Wemm break; 45406f25ae9SGregory Neil Shapiro #endif /* defined(SUN_EXTENSIONS) && defined(SUN_LOOKUP_MACRO) */ 455c2aa98e2SPeter Wemm 456c2aa98e2SPeter Wemm case 'M': /* define mailer */ 457c2aa98e2SPeter Wemm makemailer(&bp[1]); 458c2aa98e2SPeter Wemm break; 459c2aa98e2SPeter Wemm 460c2aa98e2SPeter Wemm case 'O': /* set option */ 461c2aa98e2SPeter Wemm setoption(bp[1], &bp[2], safe, FALSE, e); 462c2aa98e2SPeter Wemm break; 463c2aa98e2SPeter Wemm 464c2aa98e2SPeter Wemm case 'P': /* set precedence */ 465c2aa98e2SPeter Wemm if (NumPriorities >= MAXPRIORITIES) 466c2aa98e2SPeter Wemm { 467c2aa98e2SPeter Wemm toomany('P', MAXPRIORITIES); 468c2aa98e2SPeter Wemm break; 469c2aa98e2SPeter Wemm } 470c2aa98e2SPeter Wemm for (p = &bp[1]; *p != '\0' && *p != '='; p++) 471c2aa98e2SPeter Wemm continue; 472c2aa98e2SPeter Wemm if (*p == '\0') 473c2aa98e2SPeter Wemm goto badline; 474c2aa98e2SPeter Wemm *p = '\0'; 475c2aa98e2SPeter Wemm Priorities[NumPriorities].pri_name = newstr(&bp[1]); 476c2aa98e2SPeter Wemm Priorities[NumPriorities].pri_val = atoi(++p); 477c2aa98e2SPeter Wemm NumPriorities++; 478c2aa98e2SPeter Wemm break; 479c2aa98e2SPeter Wemm 480c2aa98e2SPeter Wemm case 'V': /* configuration syntax version */ 481c2aa98e2SPeter Wemm for (p = &bp[1]; isascii(*p) && isspace(*p); p++) 482c2aa98e2SPeter Wemm continue; 483c2aa98e2SPeter Wemm if (!isascii(*p) || !isdigit(*p)) 484c2aa98e2SPeter Wemm { 485c2aa98e2SPeter Wemm syserr("invalid argument to V line: \"%.20s\"", 486c2aa98e2SPeter Wemm &bp[1]); 487c2aa98e2SPeter Wemm break; 488c2aa98e2SPeter Wemm } 489c2aa98e2SPeter Wemm ConfigLevel = strtol(p, &ep, 10); 490c2aa98e2SPeter Wemm 491c2aa98e2SPeter Wemm /* 492c2aa98e2SPeter Wemm ** Do heuristic tweaking for back compatibility. 493c2aa98e2SPeter Wemm */ 494c2aa98e2SPeter Wemm 495c2aa98e2SPeter Wemm if (ConfigLevel >= 5) 496c2aa98e2SPeter Wemm { 497c2aa98e2SPeter Wemm /* level 5 configs have short name in $w */ 498c2aa98e2SPeter Wemm p = macvalue('w', e); 499c2aa98e2SPeter Wemm if (p != NULL && (p = strchr(p, '.')) != NULL) 500c2aa98e2SPeter Wemm *p = '\0'; 501c2aa98e2SPeter Wemm define('w', macvalue('w', e), e); 502c2aa98e2SPeter Wemm } 503c2aa98e2SPeter Wemm if (ConfigLevel >= 6) 504c2aa98e2SPeter Wemm { 505c2aa98e2SPeter Wemm ColonOkInAddr = FALSE; 506c2aa98e2SPeter Wemm } 507c2aa98e2SPeter Wemm 508c2aa98e2SPeter Wemm /* 509c2aa98e2SPeter Wemm ** Look for vendor code. 510c2aa98e2SPeter Wemm */ 511c2aa98e2SPeter Wemm 512c2aa98e2SPeter Wemm if (*ep++ == '/') 513c2aa98e2SPeter Wemm { 514c2aa98e2SPeter Wemm /* extract vendor code */ 515c2aa98e2SPeter Wemm for (p = ep; isascii(*p) && isalpha(*p); ) 516c2aa98e2SPeter Wemm p++; 517c2aa98e2SPeter Wemm *p = '\0'; 518c2aa98e2SPeter Wemm 519c2aa98e2SPeter Wemm if (!setvendor(ep)) 520c2aa98e2SPeter Wemm syserr("invalid V line vendor code: \"%s\"", 521c2aa98e2SPeter Wemm ep); 522c2aa98e2SPeter Wemm } 523c2aa98e2SPeter Wemm break; 524c2aa98e2SPeter Wemm 525c2aa98e2SPeter Wemm case 'K': 526c2aa98e2SPeter Wemm expand(&bp[1], exbuf, sizeof exbuf, e); 527c2aa98e2SPeter Wemm (void) makemapentry(exbuf); 528c2aa98e2SPeter Wemm break; 529c2aa98e2SPeter Wemm 530c2aa98e2SPeter Wemm case 'E': 531c2aa98e2SPeter Wemm p = strchr(bp, '='); 532c2aa98e2SPeter Wemm if (p != NULL) 533c2aa98e2SPeter Wemm *p++ = '\0'; 534c2aa98e2SPeter Wemm setuserenv(&bp[1], p); 535c2aa98e2SPeter Wemm break; 536c2aa98e2SPeter Wemm 53706f25ae9SGregory Neil Shapiro #if _FFR_MILTER 53806f25ae9SGregory Neil Shapiro case 'X': /* mail filter */ 53906f25ae9SGregory Neil Shapiro milter_setup(&bp[1]); 54006f25ae9SGregory Neil Shapiro break; 54106f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 54206f25ae9SGregory Neil Shapiro 543c2aa98e2SPeter Wemm default: 544c2aa98e2SPeter Wemm badline: 545c2aa98e2SPeter Wemm syserr("unknown configuration line \"%s\"", bp); 546c2aa98e2SPeter Wemm } 547c2aa98e2SPeter Wemm if (bp != buf) 548c2aa98e2SPeter Wemm free(bp); 549c2aa98e2SPeter Wemm } 550c2aa98e2SPeter Wemm if (ferror(cf)) 551c2aa98e2SPeter Wemm { 552c2aa98e2SPeter Wemm syserr("I/O read error"); 553065a643dSPeter Wemm finis(FALSE, EX_OSFILE); 554c2aa98e2SPeter Wemm } 55506f25ae9SGregory Neil Shapiro (void) fclose(cf); 556c2aa98e2SPeter Wemm FileName = NULL; 557c2aa98e2SPeter Wemm 558c2aa98e2SPeter Wemm /* initialize host maps from local service tables */ 559c2aa98e2SPeter Wemm inithostmaps(); 560c2aa98e2SPeter Wemm 56106f25ae9SGregory Neil Shapiro /* initialize daemon (if not defined yet) */ 56206f25ae9SGregory Neil Shapiro initdaemon(); 56306f25ae9SGregory Neil Shapiro 564c2aa98e2SPeter Wemm /* determine if we need to do special name-server frotz */ 565c2aa98e2SPeter Wemm { 566c2aa98e2SPeter Wemm int nmaps; 567c2aa98e2SPeter Wemm char *maptype[MAXMAPSTACK]; 568c2aa98e2SPeter Wemm short mapreturn[MAXMAPACTIONS]; 569c2aa98e2SPeter Wemm 570c2aa98e2SPeter Wemm nmaps = switch_map_find("hosts", maptype, mapreturn); 571c2aa98e2SPeter Wemm UseNameServer = FALSE; 572c2aa98e2SPeter Wemm if (nmaps > 0 && nmaps <= MAXMAPSTACK) 573c2aa98e2SPeter Wemm { 574c2aa98e2SPeter Wemm register int mapno; 575c2aa98e2SPeter Wemm 576c2aa98e2SPeter Wemm for (mapno = 0; mapno < nmaps && !UseNameServer; mapno++) 577c2aa98e2SPeter Wemm { 578c2aa98e2SPeter Wemm if (strcmp(maptype[mapno], "dns") == 0) 579c2aa98e2SPeter Wemm UseNameServer = TRUE; 580c2aa98e2SPeter Wemm } 581c2aa98e2SPeter Wemm } 582c2aa98e2SPeter Wemm 583c2aa98e2SPeter Wemm #ifdef HESIOD 584c2aa98e2SPeter Wemm nmaps = switch_map_find("passwd", maptype, mapreturn); 585c2aa98e2SPeter Wemm UseHesiod = FALSE; 586c2aa98e2SPeter Wemm if (nmaps > 0 && nmaps <= MAXMAPSTACK) 587c2aa98e2SPeter Wemm { 588c2aa98e2SPeter Wemm register int mapno; 589c2aa98e2SPeter Wemm 590c2aa98e2SPeter Wemm for (mapno = 0; mapno < nmaps && !UseHesiod; mapno++) 591c2aa98e2SPeter Wemm { 592c2aa98e2SPeter Wemm if (strcmp(maptype[mapno], "hesiod") == 0) 593c2aa98e2SPeter Wemm UseHesiod = TRUE; 594c2aa98e2SPeter Wemm } 595c2aa98e2SPeter Wemm } 59606f25ae9SGregory Neil Shapiro #endif /* HESIOD */ 597c2aa98e2SPeter Wemm } 598c2aa98e2SPeter Wemm } 599c2aa98e2SPeter Wemm /* 600c2aa98e2SPeter Wemm ** TRANSLATE_DOLLARS -- convert $x into internal form 601c2aa98e2SPeter Wemm ** 602c2aa98e2SPeter Wemm ** Actually does all appropriate pre-processing of a config line 603c2aa98e2SPeter Wemm ** to turn it into internal form. 604c2aa98e2SPeter Wemm ** 605c2aa98e2SPeter Wemm ** Parameters: 606c2aa98e2SPeter Wemm ** bp -- the buffer to translate. 607c2aa98e2SPeter Wemm ** 608c2aa98e2SPeter Wemm ** Returns: 609c2aa98e2SPeter Wemm ** None. The buffer is translated in place. Since the 610c2aa98e2SPeter Wemm ** translations always make the buffer shorter, this is 611c2aa98e2SPeter Wemm ** safe without a size parameter. 612c2aa98e2SPeter Wemm */ 613c2aa98e2SPeter Wemm 614c2aa98e2SPeter Wemm void 615c2aa98e2SPeter Wemm translate_dollars(bp) 616c2aa98e2SPeter Wemm char *bp; 617c2aa98e2SPeter Wemm { 618c2aa98e2SPeter Wemm register char *p; 619c2aa98e2SPeter Wemm auto char *ep; 620c2aa98e2SPeter Wemm 621c2aa98e2SPeter Wemm for (p = bp; *p != '\0'; p++) 622c2aa98e2SPeter Wemm { 623c2aa98e2SPeter Wemm if (*p == '#' && p > bp && ConfigLevel >= 3) 624c2aa98e2SPeter Wemm { 625c2aa98e2SPeter Wemm /* this is an on-line comment */ 626c2aa98e2SPeter Wemm register char *e; 627c2aa98e2SPeter Wemm 628c2aa98e2SPeter Wemm switch (*--p & 0377) 629c2aa98e2SPeter Wemm { 630c2aa98e2SPeter Wemm case MACROEXPAND: 631c2aa98e2SPeter Wemm /* it's from $# -- let it go through */ 632c2aa98e2SPeter Wemm p++; 633c2aa98e2SPeter Wemm break; 634c2aa98e2SPeter Wemm 635c2aa98e2SPeter Wemm case '\\': 636c2aa98e2SPeter Wemm /* it's backslash escaped */ 63706f25ae9SGregory Neil Shapiro (void) strlcpy(p, p + 1, strlen(p)); 638c2aa98e2SPeter Wemm break; 639c2aa98e2SPeter Wemm 640c2aa98e2SPeter Wemm default: 64106f25ae9SGregory Neil Shapiro /* delete leading white space */ 642c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p) && 643c2aa98e2SPeter Wemm *p != '\n' && p > bp) 644c2aa98e2SPeter Wemm p--; 645c2aa98e2SPeter Wemm if ((e = strchr(++p, '\n')) != NULL) 64606f25ae9SGregory Neil Shapiro (void) strlcpy(p, e, strlen(p)); 647c2aa98e2SPeter Wemm else 648c2aa98e2SPeter Wemm *p-- = '\0'; 649c2aa98e2SPeter Wemm break; 650c2aa98e2SPeter Wemm } 651c2aa98e2SPeter Wemm continue; 652c2aa98e2SPeter Wemm } 653c2aa98e2SPeter Wemm 654c2aa98e2SPeter Wemm if (*p != '$' || p[1] == '\0') 655c2aa98e2SPeter Wemm continue; 656c2aa98e2SPeter Wemm 657c2aa98e2SPeter Wemm if (p[1] == '$') 658c2aa98e2SPeter Wemm { 659c2aa98e2SPeter Wemm /* actual dollar sign.... */ 66006f25ae9SGregory Neil Shapiro (void) strlcpy(p, p + 1, strlen(p)); 661c2aa98e2SPeter Wemm continue; 662c2aa98e2SPeter Wemm } 663c2aa98e2SPeter Wemm 664c2aa98e2SPeter Wemm /* convert to macro expansion character */ 665c2aa98e2SPeter Wemm *p++ = MACROEXPAND; 666c2aa98e2SPeter Wemm 667c2aa98e2SPeter Wemm /* special handling for $=, $~, $&, and $? */ 668c2aa98e2SPeter Wemm if (*p == '=' || *p == '~' || *p == '&' || *p == '?') 669c2aa98e2SPeter Wemm p++; 670c2aa98e2SPeter Wemm 671c2aa98e2SPeter Wemm /* convert macro name to code */ 672c2aa98e2SPeter Wemm *p = macid(p, &ep); 67306f25ae9SGregory Neil Shapiro if (ep != p + 1) 67406f25ae9SGregory Neil Shapiro (void) strlcpy(p + 1, ep, strlen(p + 1)); 675c2aa98e2SPeter Wemm } 676c2aa98e2SPeter Wemm 677c2aa98e2SPeter Wemm /* strip trailing white space from the line */ 678c2aa98e2SPeter Wemm while (--p > bp && isascii(*p) && isspace(*p)) 679c2aa98e2SPeter Wemm *p = '\0'; 680c2aa98e2SPeter Wemm } 681c2aa98e2SPeter Wemm /* 682c2aa98e2SPeter Wemm ** TOOMANY -- signal too many of some option 683c2aa98e2SPeter Wemm ** 684c2aa98e2SPeter Wemm ** Parameters: 685c2aa98e2SPeter Wemm ** id -- the id of the error line 686c2aa98e2SPeter Wemm ** maxcnt -- the maximum possible values 687c2aa98e2SPeter Wemm ** 688c2aa98e2SPeter Wemm ** Returns: 689c2aa98e2SPeter Wemm ** none. 690c2aa98e2SPeter Wemm ** 691c2aa98e2SPeter Wemm ** Side Effects: 692c2aa98e2SPeter Wemm ** gives a syserr. 693c2aa98e2SPeter Wemm */ 694c2aa98e2SPeter Wemm 69506f25ae9SGregory Neil Shapiro static void 696c2aa98e2SPeter Wemm toomany(id, maxcnt) 697c2aa98e2SPeter Wemm int id; 698c2aa98e2SPeter Wemm int maxcnt; 699c2aa98e2SPeter Wemm { 700c2aa98e2SPeter Wemm syserr("too many %c lines, %d max", id, maxcnt); 701c2aa98e2SPeter Wemm } 702c2aa98e2SPeter Wemm /* 703c2aa98e2SPeter Wemm ** FILECLASS -- read members of a class from a file 704c2aa98e2SPeter Wemm ** 705c2aa98e2SPeter Wemm ** Parameters: 706c2aa98e2SPeter Wemm ** class -- class to define. 707c2aa98e2SPeter Wemm ** filename -- name of file to read. 708c2aa98e2SPeter Wemm ** fmt -- scanf string to use for match. 709c2aa98e2SPeter Wemm ** safe -- if set, this is a safe read. 710c2aa98e2SPeter Wemm ** optional -- if set, it is not an error for the file to 711c2aa98e2SPeter Wemm ** not exist. 712c2aa98e2SPeter Wemm ** 713c2aa98e2SPeter Wemm ** Returns: 714c2aa98e2SPeter Wemm ** none 715c2aa98e2SPeter Wemm ** 716c2aa98e2SPeter Wemm ** Side Effects: 717c2aa98e2SPeter Wemm ** 718c2aa98e2SPeter Wemm ** puts all lines in filename that match a scanf into 719c2aa98e2SPeter Wemm ** the named class. 720c2aa98e2SPeter Wemm */ 721c2aa98e2SPeter Wemm 72206f25ae9SGregory Neil Shapiro static void 723c2aa98e2SPeter Wemm fileclass(class, filename, fmt, safe, optional) 724c2aa98e2SPeter Wemm int class; 725c2aa98e2SPeter Wemm char *filename; 726c2aa98e2SPeter Wemm char *fmt; 727c2aa98e2SPeter Wemm bool safe; 728c2aa98e2SPeter Wemm bool optional; 729c2aa98e2SPeter Wemm { 730c2aa98e2SPeter Wemm FILE *f; 73106f25ae9SGregory Neil Shapiro long sff; 732c2aa98e2SPeter Wemm pid_t pid; 733c2aa98e2SPeter Wemm register char *p; 734c2aa98e2SPeter Wemm char buf[MAXLINE]; 735c2aa98e2SPeter Wemm 736c2aa98e2SPeter Wemm if (tTd(37, 2)) 73706f25ae9SGregory Neil Shapiro dprintf("fileclass(%s, fmt=%s)\n", filename, fmt); 738c2aa98e2SPeter Wemm 739c2aa98e2SPeter Wemm if (filename[0] == '|') 740c2aa98e2SPeter Wemm { 741c2aa98e2SPeter Wemm auto int fd; 742c2aa98e2SPeter Wemm int i; 743c2aa98e2SPeter Wemm char *argv[MAXPV + 1]; 744c2aa98e2SPeter Wemm 745c2aa98e2SPeter Wemm i = 0; 746c2aa98e2SPeter Wemm for (p = strtok(&filename[1], " \t"); p != NULL; p = strtok(NULL, " \t")) 747c2aa98e2SPeter Wemm { 748c2aa98e2SPeter Wemm if (i >= MAXPV) 749c2aa98e2SPeter Wemm break; 750c2aa98e2SPeter Wemm argv[i++] = p; 751c2aa98e2SPeter Wemm } 752c2aa98e2SPeter Wemm argv[i] = NULL; 753c2aa98e2SPeter Wemm pid = prog_open(argv, &fd, CurEnv); 754c2aa98e2SPeter Wemm if (pid < 0) 755c2aa98e2SPeter Wemm f = NULL; 756c2aa98e2SPeter Wemm else 757c2aa98e2SPeter Wemm f = fdopen(fd, "r"); 758c2aa98e2SPeter Wemm } 759c2aa98e2SPeter Wemm else 760c2aa98e2SPeter Wemm { 761c2aa98e2SPeter Wemm pid = -1; 762c2aa98e2SPeter Wemm sff = SFF_REGONLY; 76306f25ae9SGregory Neil Shapiro if (!bitnset(DBS_CLASSFILEINUNSAFEDIRPATH, DontBlameSendmail)) 764c2aa98e2SPeter Wemm sff |= SFF_SAFEDIRPATH; 76506f25ae9SGregory Neil Shapiro if (!bitnset(DBS_LINKEDCLASSFILEINWRITABLEDIR, 76606f25ae9SGregory Neil Shapiro DontBlameSendmail)) 767c2aa98e2SPeter Wemm sff |= SFF_NOWLINK; 768c2aa98e2SPeter Wemm if (safe) 769c2aa98e2SPeter Wemm sff |= SFF_OPENASROOT; 770c2aa98e2SPeter Wemm if (DontLockReadFiles) 771c2aa98e2SPeter Wemm sff |= SFF_NOLOCK; 772c2aa98e2SPeter Wemm f = safefopen(filename, O_RDONLY, 0, sff); 773c2aa98e2SPeter Wemm } 774c2aa98e2SPeter Wemm if (f == NULL) 775c2aa98e2SPeter Wemm { 776c2aa98e2SPeter Wemm if (!optional) 77706f25ae9SGregory Neil Shapiro syserr("fileclass: cannot open '%s'", filename); 778c2aa98e2SPeter Wemm return; 779c2aa98e2SPeter Wemm } 780c2aa98e2SPeter Wemm 781c2aa98e2SPeter Wemm while (fgets(buf, sizeof buf, f) != NULL) 782c2aa98e2SPeter Wemm { 783c2aa98e2SPeter Wemm #if SCANF 784c2aa98e2SPeter Wemm char wordbuf[MAXLINE + 1]; 78506f25ae9SGregory Neil Shapiro #endif /* SCANF */ 786c2aa98e2SPeter Wemm 787c2aa98e2SPeter Wemm if (buf[0] == '#') 788c2aa98e2SPeter Wemm continue; 789c2aa98e2SPeter Wemm #if SCANF 790c2aa98e2SPeter Wemm if (sscanf(buf, fmt, wordbuf) != 1) 791c2aa98e2SPeter Wemm continue; 792c2aa98e2SPeter Wemm p = wordbuf; 793c2aa98e2SPeter Wemm #else /* SCANF */ 794c2aa98e2SPeter Wemm p = buf; 795c2aa98e2SPeter Wemm #endif /* SCANF */ 796c2aa98e2SPeter Wemm 797c2aa98e2SPeter Wemm /* 798c2aa98e2SPeter Wemm ** Break up the match into words. 799c2aa98e2SPeter Wemm */ 800c2aa98e2SPeter Wemm 801c2aa98e2SPeter Wemm while (*p != '\0') 802c2aa98e2SPeter Wemm { 803c2aa98e2SPeter Wemm register char *q; 804c2aa98e2SPeter Wemm 805c2aa98e2SPeter Wemm /* strip leading spaces */ 806c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 807c2aa98e2SPeter Wemm p++; 808c2aa98e2SPeter Wemm if (*p == '\0') 809c2aa98e2SPeter Wemm break; 810c2aa98e2SPeter Wemm 811c2aa98e2SPeter Wemm /* find the end of the word */ 812c2aa98e2SPeter Wemm q = p; 813c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 814c2aa98e2SPeter Wemm p++; 815c2aa98e2SPeter Wemm if (*p != '\0') 816c2aa98e2SPeter Wemm *p++ = '\0'; 817c2aa98e2SPeter Wemm 818c2aa98e2SPeter Wemm /* enter the word in the symbol table */ 819c2aa98e2SPeter Wemm setclass(class, q); 820c2aa98e2SPeter Wemm } 821c2aa98e2SPeter Wemm } 822c2aa98e2SPeter Wemm 823c2aa98e2SPeter Wemm (void) fclose(f); 824c2aa98e2SPeter Wemm if (pid > 0) 825c2aa98e2SPeter Wemm (void) waitfor(pid); 826c2aa98e2SPeter Wemm } 827c2aa98e2SPeter Wemm /* 828c2aa98e2SPeter Wemm ** MAKEMAILER -- define a new mailer. 829c2aa98e2SPeter Wemm ** 830c2aa98e2SPeter Wemm ** Parameters: 831c2aa98e2SPeter Wemm ** line -- description of mailer. This is in labeled 832c2aa98e2SPeter Wemm ** fields. The fields are: 833c2aa98e2SPeter Wemm ** A -- the argv for this mailer 834c2aa98e2SPeter Wemm ** C -- the character set for MIME conversions 835c2aa98e2SPeter Wemm ** D -- the directory to run in 836c2aa98e2SPeter Wemm ** E -- the eol string 837c2aa98e2SPeter Wemm ** F -- the flags associated with the mailer 838c2aa98e2SPeter Wemm ** L -- the maximum line length 839c2aa98e2SPeter Wemm ** M -- the maximum message size 840c2aa98e2SPeter Wemm ** N -- the niceness at which to run 841c2aa98e2SPeter Wemm ** P -- the path to the mailer 842c2aa98e2SPeter Wemm ** R -- the recipient rewriting set 843c2aa98e2SPeter Wemm ** S -- the sender rewriting set 844c2aa98e2SPeter Wemm ** T -- the mailer type (for DSNs) 845c2aa98e2SPeter Wemm ** U -- the uid to run as 84606f25ae9SGregory Neil Shapiro ** W -- the time to wait at the end 847602a2b1bSGregory Neil Shapiro ** m -- maximum messages per connection 848602a2b1bSGregory Neil Shapiro ** / -- new root directory 849c2aa98e2SPeter Wemm ** The first word is the canonical name of the mailer. 850c2aa98e2SPeter Wemm ** 851c2aa98e2SPeter Wemm ** Returns: 852c2aa98e2SPeter Wemm ** none. 853c2aa98e2SPeter Wemm ** 854c2aa98e2SPeter Wemm ** Side Effects: 855c2aa98e2SPeter Wemm ** enters the mailer into the mailer table. 856c2aa98e2SPeter Wemm */ 857c2aa98e2SPeter Wemm 858c2aa98e2SPeter Wemm void 859c2aa98e2SPeter Wemm makemailer(line) 860c2aa98e2SPeter Wemm char *line; 861c2aa98e2SPeter Wemm { 862c2aa98e2SPeter Wemm register char *p; 863c2aa98e2SPeter Wemm register struct mailer *m; 864c2aa98e2SPeter Wemm register STAB *s; 865c2aa98e2SPeter Wemm int i; 866c2aa98e2SPeter Wemm char fcode; 867c2aa98e2SPeter Wemm auto char *endp; 868c2aa98e2SPeter Wemm extern int NextMailer; 869c2aa98e2SPeter Wemm 870c2aa98e2SPeter Wemm /* allocate a mailer and set up defaults */ 871c2aa98e2SPeter Wemm m = (struct mailer *) xalloc(sizeof *m); 87206f25ae9SGregory Neil Shapiro memset((char *) m, '\0', sizeof *m); 873c2aa98e2SPeter Wemm 874c2aa98e2SPeter Wemm /* collect the mailer name */ 875c2aa98e2SPeter Wemm for (p = line; *p != '\0' && *p != ',' && !(isascii(*p) && isspace(*p)); p++) 876c2aa98e2SPeter Wemm continue; 877c2aa98e2SPeter Wemm if (*p != '\0') 878c2aa98e2SPeter Wemm *p++ = '\0'; 879c2aa98e2SPeter Wemm if (line[0] == '\0') 880193538b7SGregory Neil Shapiro { 881c2aa98e2SPeter Wemm syserr("name required for mailer"); 882193538b7SGregory Neil Shapiro return; 883193538b7SGregory Neil Shapiro } 884c2aa98e2SPeter Wemm m->m_name = newstr(line); 885c2aa98e2SPeter Wemm 886c2aa98e2SPeter Wemm /* now scan through and assign info from the fields */ 887c2aa98e2SPeter Wemm while (*p != '\0') 888c2aa98e2SPeter Wemm { 889c2aa98e2SPeter Wemm auto char *delimptr; 890c2aa98e2SPeter Wemm 891c2aa98e2SPeter Wemm while (*p != '\0' && (*p == ',' || (isascii(*p) && isspace(*p)))) 892c2aa98e2SPeter Wemm p++; 893c2aa98e2SPeter Wemm 894c2aa98e2SPeter Wemm /* p now points to field code */ 895c2aa98e2SPeter Wemm fcode = *p; 896c2aa98e2SPeter Wemm while (*p != '\0' && *p != '=' && *p != ',') 897c2aa98e2SPeter Wemm p++; 898c2aa98e2SPeter Wemm if (*p++ != '=') 899c2aa98e2SPeter Wemm { 900c2aa98e2SPeter Wemm syserr("mailer %s: `=' expected", m->m_name); 901c2aa98e2SPeter Wemm return; 902c2aa98e2SPeter Wemm } 903c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 904c2aa98e2SPeter Wemm p++; 905c2aa98e2SPeter Wemm 906c2aa98e2SPeter Wemm /* p now points to the field body */ 907c2aa98e2SPeter Wemm p = munchstring(p, &delimptr, ','); 908c2aa98e2SPeter Wemm 909c2aa98e2SPeter Wemm /* install the field into the mailer struct */ 910c2aa98e2SPeter Wemm switch (fcode) 911c2aa98e2SPeter Wemm { 912c2aa98e2SPeter Wemm case 'P': /* pathname */ 913c2aa98e2SPeter Wemm if (*p == '\0') 914c2aa98e2SPeter Wemm syserr("mailer %s: empty path name", m->m_name); 915193538b7SGregory Neil Shapiro else 916c2aa98e2SPeter Wemm m->m_mailer = newstr(p); 917c2aa98e2SPeter Wemm break; 918c2aa98e2SPeter Wemm 919c2aa98e2SPeter Wemm case 'F': /* flags */ 920c2aa98e2SPeter Wemm for (; *p != '\0'; p++) 921c2aa98e2SPeter Wemm if (!(isascii(*p) && isspace(*p))) 922193538b7SGregory Neil Shapiro setbitn(bitidx(*p), m->m_flags); 923c2aa98e2SPeter Wemm break; 924c2aa98e2SPeter Wemm 925c2aa98e2SPeter Wemm case 'S': /* sender rewriting ruleset */ 926c2aa98e2SPeter Wemm case 'R': /* recipient rewriting ruleset */ 927c2aa98e2SPeter Wemm i = strtorwset(p, &endp, ST_ENTER); 928c2aa98e2SPeter Wemm if (i < 0) 929c2aa98e2SPeter Wemm return; 930c2aa98e2SPeter Wemm if (fcode == 'S') 931c2aa98e2SPeter Wemm m->m_sh_rwset = m->m_se_rwset = i; 932c2aa98e2SPeter Wemm else 933c2aa98e2SPeter Wemm m->m_rh_rwset = m->m_re_rwset = i; 934c2aa98e2SPeter Wemm 935c2aa98e2SPeter Wemm p = endp; 936c2aa98e2SPeter Wemm if (*p++ == '/') 937c2aa98e2SPeter Wemm { 938c2aa98e2SPeter Wemm i = strtorwset(p, NULL, ST_ENTER); 939c2aa98e2SPeter Wemm if (i < 0) 940c2aa98e2SPeter Wemm return; 941c2aa98e2SPeter Wemm if (fcode == 'S') 942c2aa98e2SPeter Wemm m->m_sh_rwset = i; 943c2aa98e2SPeter Wemm else 944c2aa98e2SPeter Wemm m->m_rh_rwset = i; 945c2aa98e2SPeter Wemm } 946c2aa98e2SPeter Wemm break; 947c2aa98e2SPeter Wemm 948c2aa98e2SPeter Wemm case 'E': /* end of line string */ 949c2aa98e2SPeter Wemm if (*p == '\0') 950c2aa98e2SPeter Wemm syserr("mailer %s: null end-of-line string", 951c2aa98e2SPeter Wemm m->m_name); 952193538b7SGregory Neil Shapiro else 953c2aa98e2SPeter Wemm m->m_eol = newstr(p); 954c2aa98e2SPeter Wemm break; 955c2aa98e2SPeter Wemm 956c2aa98e2SPeter Wemm case 'A': /* argument vector */ 957c2aa98e2SPeter Wemm if (*p == '\0') 958c2aa98e2SPeter Wemm syserr("mailer %s: null argument vector", 959c2aa98e2SPeter Wemm m->m_name); 960193538b7SGregory Neil Shapiro else 961c2aa98e2SPeter Wemm m->m_argv = makeargv(p); 962c2aa98e2SPeter Wemm break; 963c2aa98e2SPeter Wemm 964c2aa98e2SPeter Wemm case 'M': /* maximum message size */ 965c2aa98e2SPeter Wemm m->m_maxsize = atol(p); 966c2aa98e2SPeter Wemm break; 967c2aa98e2SPeter Wemm 96806f25ae9SGregory Neil Shapiro case 'm': /* maximum messages per connection */ 96906f25ae9SGregory Neil Shapiro m->m_maxdeliveries = atoi(p); 97006f25ae9SGregory Neil Shapiro break; 97106f25ae9SGregory Neil Shapiro 97206f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 97306f25ae9SGregory Neil Shapiro case 'r': /* max recipient per envelope */ 97406f25ae9SGregory Neil Shapiro m->m_maxrcpt = atoi(p); 97506f25ae9SGregory Neil Shapiro break; 97606f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 97706f25ae9SGregory Neil Shapiro 978c2aa98e2SPeter Wemm case 'L': /* maximum line length */ 979c2aa98e2SPeter Wemm m->m_linelimit = atoi(p); 980c2aa98e2SPeter Wemm if (m->m_linelimit < 0) 981c2aa98e2SPeter Wemm m->m_linelimit = 0; 982c2aa98e2SPeter Wemm break; 983c2aa98e2SPeter Wemm 984c2aa98e2SPeter Wemm case 'N': /* run niceness */ 985c2aa98e2SPeter Wemm m->m_nice = atoi(p); 986c2aa98e2SPeter Wemm break; 987c2aa98e2SPeter Wemm 988c2aa98e2SPeter Wemm case 'D': /* working directory */ 989c2aa98e2SPeter Wemm if (*p == '\0') 990c2aa98e2SPeter Wemm syserr("mailer %s: null working directory", 991c2aa98e2SPeter Wemm m->m_name); 992193538b7SGregory Neil Shapiro else 993c2aa98e2SPeter Wemm m->m_execdir = newstr(p); 994c2aa98e2SPeter Wemm break; 995c2aa98e2SPeter Wemm 996c2aa98e2SPeter Wemm case 'C': /* default charset */ 997c2aa98e2SPeter Wemm if (*p == '\0') 998c2aa98e2SPeter Wemm syserr("mailer %s: null charset", m->m_name); 999193538b7SGregory Neil Shapiro else 1000c2aa98e2SPeter Wemm m->m_defcharset = newstr(p); 1001c2aa98e2SPeter Wemm break; 1002c2aa98e2SPeter Wemm 1003c2aa98e2SPeter Wemm case 'T': /* MTA-Name/Address/Diagnostic types */ 1004c2aa98e2SPeter Wemm /* extract MTA name type; default to "dns" */ 1005c2aa98e2SPeter Wemm m->m_mtatype = newstr(p); 1006c2aa98e2SPeter Wemm p = strchr(m->m_mtatype, '/'); 1007c2aa98e2SPeter Wemm if (p != NULL) 1008c2aa98e2SPeter Wemm { 1009c2aa98e2SPeter Wemm *p++ = '\0'; 1010c2aa98e2SPeter Wemm if (*p == '\0') 1011c2aa98e2SPeter Wemm p = NULL; 1012c2aa98e2SPeter Wemm } 1013c2aa98e2SPeter Wemm if (*m->m_mtatype == '\0') 1014c2aa98e2SPeter Wemm m->m_mtatype = "dns"; 1015c2aa98e2SPeter Wemm 1016c2aa98e2SPeter Wemm /* extract address type; default to "rfc822" */ 1017c2aa98e2SPeter Wemm m->m_addrtype = p; 1018c2aa98e2SPeter Wemm if (p != NULL) 1019c2aa98e2SPeter Wemm p = strchr(p, '/'); 1020c2aa98e2SPeter Wemm if (p != NULL) 1021c2aa98e2SPeter Wemm { 1022c2aa98e2SPeter Wemm *p++ = '\0'; 1023c2aa98e2SPeter Wemm if (*p == '\0') 1024c2aa98e2SPeter Wemm p = NULL; 1025c2aa98e2SPeter Wemm } 1026c2aa98e2SPeter Wemm if (m->m_addrtype == NULL || *m->m_addrtype == '\0') 1027c2aa98e2SPeter Wemm m->m_addrtype = "rfc822"; 1028c2aa98e2SPeter Wemm 1029c2aa98e2SPeter Wemm /* extract diagnostic type; default to "smtp" */ 1030c2aa98e2SPeter Wemm m->m_diagtype = p; 1031c2aa98e2SPeter Wemm if (m->m_diagtype == NULL || *m->m_diagtype == '\0') 1032c2aa98e2SPeter Wemm m->m_diagtype = "smtp"; 1033c2aa98e2SPeter Wemm break; 1034c2aa98e2SPeter Wemm 1035c2aa98e2SPeter Wemm case 'U': /* user id */ 1036c2aa98e2SPeter Wemm if (isascii(*p) && !isdigit(*p)) 1037c2aa98e2SPeter Wemm { 1038c2aa98e2SPeter Wemm char *q = p; 1039c2aa98e2SPeter Wemm struct passwd *pw; 1040c2aa98e2SPeter Wemm 1041c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && 1042c2aa98e2SPeter Wemm (isalnum(*p) || strchr("-_", *p) != NULL)) 1043c2aa98e2SPeter Wemm p++; 1044c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1045c2aa98e2SPeter Wemm *p++ = '\0'; 1046c2aa98e2SPeter Wemm if (*p != '\0') 1047c2aa98e2SPeter Wemm *p++ = '\0'; 1048c2aa98e2SPeter Wemm if (*q == '\0') 1049193538b7SGregory Neil Shapiro { 1050c2aa98e2SPeter Wemm syserr("mailer %s: null user name", 1051c2aa98e2SPeter Wemm m->m_name); 1052193538b7SGregory Neil Shapiro break; 1053193538b7SGregory Neil Shapiro } 1054c2aa98e2SPeter Wemm pw = sm_getpwnam(q); 1055c2aa98e2SPeter Wemm if (pw == NULL) 1056193538b7SGregory Neil Shapiro { 1057c2aa98e2SPeter Wemm syserr("readcf: mailer U= flag: unknown user %s", q); 1058193538b7SGregory Neil Shapiro break; 1059193538b7SGregory Neil Shapiro } 1060c2aa98e2SPeter Wemm else 1061c2aa98e2SPeter Wemm { 1062c2aa98e2SPeter Wemm m->m_uid = pw->pw_uid; 1063c2aa98e2SPeter Wemm m->m_gid = pw->pw_gid; 1064c2aa98e2SPeter Wemm } 1065c2aa98e2SPeter Wemm } 1066c2aa98e2SPeter Wemm else 1067c2aa98e2SPeter Wemm { 1068c2aa98e2SPeter Wemm auto char *q; 1069c2aa98e2SPeter Wemm 1070c2aa98e2SPeter Wemm m->m_uid = strtol(p, &q, 0); 1071c2aa98e2SPeter Wemm p = q; 1072c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1073c2aa98e2SPeter Wemm p++; 1074c2aa98e2SPeter Wemm if (*p != '\0') 1075c2aa98e2SPeter Wemm p++; 1076c2aa98e2SPeter Wemm } 1077c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1078c2aa98e2SPeter Wemm p++; 1079c2aa98e2SPeter Wemm if (*p == '\0') 1080c2aa98e2SPeter Wemm break; 1081c2aa98e2SPeter Wemm if (isascii(*p) && !isdigit(*p)) 1082c2aa98e2SPeter Wemm { 1083c2aa98e2SPeter Wemm char *q = p; 1084c2aa98e2SPeter Wemm struct group *gr; 1085c2aa98e2SPeter Wemm 1086c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 1087c2aa98e2SPeter Wemm p++; 1088c2aa98e2SPeter Wemm *p++ = '\0'; 1089c2aa98e2SPeter Wemm if (*q == '\0') 1090193538b7SGregory Neil Shapiro { 1091c2aa98e2SPeter Wemm syserr("mailer %s: null group name", 1092c2aa98e2SPeter Wemm m->m_name); 1093193538b7SGregory Neil Shapiro break; 1094193538b7SGregory Neil Shapiro } 1095c2aa98e2SPeter Wemm gr = getgrnam(q); 1096c2aa98e2SPeter Wemm if (gr == NULL) 1097193538b7SGregory Neil Shapiro { 1098c2aa98e2SPeter Wemm syserr("readcf: mailer U= flag: unknown group %s", q); 1099193538b7SGregory Neil Shapiro break; 1100193538b7SGregory Neil Shapiro } 1101c2aa98e2SPeter Wemm else 1102c2aa98e2SPeter Wemm m->m_gid = gr->gr_gid; 1103c2aa98e2SPeter Wemm } 1104c2aa98e2SPeter Wemm else 1105c2aa98e2SPeter Wemm { 1106c2aa98e2SPeter Wemm m->m_gid = strtol(p, NULL, 0); 1107c2aa98e2SPeter Wemm } 1108c2aa98e2SPeter Wemm break; 110906f25ae9SGregory Neil Shapiro 111006f25ae9SGregory Neil Shapiro case 'W': /* wait timeout */ 111106f25ae9SGregory Neil Shapiro m->m_wait = convtime(p, 's'); 111206f25ae9SGregory Neil Shapiro break; 111306f25ae9SGregory Neil Shapiro 111406f25ae9SGregory Neil Shapiro case '/': /* new root directory */ 111506f25ae9SGregory Neil Shapiro if (*p == '\0') 111606f25ae9SGregory Neil Shapiro syserr("mailer %s: null root directory", 111706f25ae9SGregory Neil Shapiro m->m_name); 111806f25ae9SGregory Neil Shapiro else 111906f25ae9SGregory Neil Shapiro m->m_rootdir = newstr(p); 112006f25ae9SGregory Neil Shapiro break; 112106f25ae9SGregory Neil Shapiro 112206f25ae9SGregory Neil Shapiro default: 112306f25ae9SGregory Neil Shapiro syserr("M%s: unknown mailer equate %c=", 112406f25ae9SGregory Neil Shapiro m->m_name, fcode); 112506f25ae9SGregory Neil Shapiro break; 1126c2aa98e2SPeter Wemm } 1127c2aa98e2SPeter Wemm 1128c2aa98e2SPeter Wemm p = delimptr; 1129c2aa98e2SPeter Wemm } 1130c2aa98e2SPeter Wemm 1131c2aa98e2SPeter Wemm /* do some rationality checking */ 1132c2aa98e2SPeter Wemm if (m->m_argv == NULL) 1133c2aa98e2SPeter Wemm { 1134c2aa98e2SPeter Wemm syserr("M%s: A= argument required", m->m_name); 1135c2aa98e2SPeter Wemm return; 1136c2aa98e2SPeter Wemm } 1137c2aa98e2SPeter Wemm if (m->m_mailer == NULL) 1138c2aa98e2SPeter Wemm { 1139c2aa98e2SPeter Wemm syserr("M%s: P= argument required", m->m_name); 1140c2aa98e2SPeter Wemm return; 1141c2aa98e2SPeter Wemm } 1142c2aa98e2SPeter Wemm 1143c2aa98e2SPeter Wemm if (NextMailer >= MAXMAILERS) 1144c2aa98e2SPeter Wemm { 1145c2aa98e2SPeter Wemm syserr("too many mailers defined (%d max)", MAXMAILERS); 1146c2aa98e2SPeter Wemm return; 1147c2aa98e2SPeter Wemm } 1148c2aa98e2SPeter Wemm 114906f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 115006f25ae9SGregory Neil Shapiro if (m->m_maxrcpt <= 0) 115106f25ae9SGregory Neil Shapiro m->m_maxrcpt = DEFAULT_MAX_RCPT; 115206f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 115306f25ae9SGregory Neil Shapiro 1154c2aa98e2SPeter Wemm /* do some heuristic cleanup for back compatibility */ 1155c2aa98e2SPeter Wemm if (bitnset(M_LIMITS, m->m_flags)) 1156c2aa98e2SPeter Wemm { 1157c2aa98e2SPeter Wemm if (m->m_linelimit == 0) 1158c2aa98e2SPeter Wemm m->m_linelimit = SMTPLINELIM; 1159c2aa98e2SPeter Wemm if (ConfigLevel < 2) 1160c2aa98e2SPeter Wemm setbitn(M_7BITS, m->m_flags); 1161c2aa98e2SPeter Wemm } 1162c2aa98e2SPeter Wemm 116306f25ae9SGregory Neil Shapiro if (strcmp(m->m_mailer, "[TCP]") == 0) 1164c2aa98e2SPeter Wemm { 116542e5d165SGregory Neil Shapiro #if _FFR_REMOVE_TCP_MAILER_PATH 116606f25ae9SGregory Neil Shapiro syserr("M%s: P=[TCP] is deprecated, use P=[IPC] instead\n", 116706f25ae9SGregory Neil Shapiro m->m_name); 1168193538b7SGregory Neil Shapiro return; 116942e5d165SGregory Neil Shapiro #else /* _FFR_REMOVE_TCP_MAILER_PATH */ 117006f25ae9SGregory Neil Shapiro printf("M%s: Warning: P=[TCP] is deprecated, use P=[IPC] instead\n", 117106f25ae9SGregory Neil Shapiro m->m_name); 117242e5d165SGregory Neil Shapiro #endif /* _FFR_REMOVE_TCP_MAILER_PATH */ 1173c2aa98e2SPeter Wemm } 1174c2aa98e2SPeter Wemm 117542e5d165SGregory Neil Shapiro if (strcmp(m->m_mailer, "[IPC]") == 0 117606f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_TCP_MAILER_PATH 117742e5d165SGregory Neil Shapiro || strcmp(m->m_mailer, "[TCP]") == 0 117806f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_TCP_MAILER_PATH */ 117906f25ae9SGregory Neil Shapiro ) 118006f25ae9SGregory Neil Shapiro { 118106f25ae9SGregory Neil Shapiro /* Use the second argument for host or path to socket */ 118206f25ae9SGregory Neil Shapiro if (m->m_argv[0] == NULL || m->m_argv[1] == NULL || 118306f25ae9SGregory Neil Shapiro m->m_argv[1][0] == '\0') 118406f25ae9SGregory Neil Shapiro { 118506f25ae9SGregory Neil Shapiro syserr("M%s: too few parameters for %s mailer", 118606f25ae9SGregory Neil Shapiro m->m_name, m->m_mailer); 1187193538b7SGregory Neil Shapiro return; 118806f25ae9SGregory Neil Shapiro } 118942e5d165SGregory Neil Shapiro if (strcmp(m->m_argv[0], "TCP") != 0 119006f25ae9SGregory Neil Shapiro #if NETUNIX 119142e5d165SGregory Neil Shapiro && strcmp(m->m_argv[0], "FILE") != 0 119206f25ae9SGregory Neil Shapiro #endif /* NETUNIX */ 119306f25ae9SGregory Neil Shapiro #if !_FFR_DEPRECATE_IPC_MAILER_ARG 119442e5d165SGregory Neil Shapiro && strcmp(m->m_argv[0], "IPC") != 0 119506f25ae9SGregory Neil Shapiro #endif /* !_FFR_DEPRECATE_IPC_MAILER_ARG */ 119606f25ae9SGregory Neil Shapiro ) 119706f25ae9SGregory Neil Shapiro { 119806f25ae9SGregory Neil Shapiro printf("M%s: Warning: first argument in %s mailer must be %s\n", 119906f25ae9SGregory Neil Shapiro m->m_name, m->m_mailer, 120006f25ae9SGregory Neil Shapiro #if NETUNIX 120106f25ae9SGregory Neil Shapiro "TCP or FILE" 120206f25ae9SGregory Neil Shapiro #else /* NETUNIX */ 120306f25ae9SGregory Neil Shapiro "TCP" 120406f25ae9SGregory Neil Shapiro #endif /* NETUNIX */ 120506f25ae9SGregory Neil Shapiro ); 120606f25ae9SGregory Neil Shapiro } 120706f25ae9SGregory Neil Shapiro 120806f25ae9SGregory Neil Shapiro } 120906f25ae9SGregory Neil Shapiro else if (strcmp(m->m_mailer, "[FILE]") == 0) 1210c2aa98e2SPeter Wemm { 1211c2aa98e2SPeter Wemm /* Use the second argument for filename */ 1212c2aa98e2SPeter Wemm if (m->m_argv[0] == NULL || m->m_argv[1] == NULL || 1213c2aa98e2SPeter Wemm m->m_argv[2] != NULL) 1214c2aa98e2SPeter Wemm { 1215c2aa98e2SPeter Wemm syserr("M%s: too %s parameters for [FILE] mailer", 1216c2aa98e2SPeter Wemm m->m_name, 1217c2aa98e2SPeter Wemm (m->m_argv[0] == NULL || 1218c2aa98e2SPeter Wemm m->m_argv[1] == NULL) ? "few" : "many"); 1219193538b7SGregory Neil Shapiro return; 1220c2aa98e2SPeter Wemm } 1221c2aa98e2SPeter Wemm else if (strcmp(m->m_argv[0], "FILE") != 0) 1222c2aa98e2SPeter Wemm { 1223c2aa98e2SPeter Wemm syserr("M%s: first argument in [FILE] mailer must be FILE", 1224c2aa98e2SPeter Wemm m->m_name); 1225193538b7SGregory Neil Shapiro return; 1226c2aa98e2SPeter Wemm } 1227c2aa98e2SPeter Wemm } 1228c2aa98e2SPeter Wemm 122906f25ae9SGregory Neil Shapiro if (strcmp(m->m_mailer, "[IPC]") == 0 || 123006f25ae9SGregory Neil Shapiro strcmp(m->m_mailer, "[TCP]") == 0) 123106f25ae9SGregory Neil Shapiro { 123206f25ae9SGregory Neil Shapiro if (m->m_mtatype == NULL) 123306f25ae9SGregory Neil Shapiro m->m_mtatype = "dns"; 123406f25ae9SGregory Neil Shapiro if (m->m_addrtype == NULL) 123506f25ae9SGregory Neil Shapiro m->m_addrtype = "rfc822"; 123606f25ae9SGregory Neil Shapiro if (m->m_diagtype == NULL) 123706f25ae9SGregory Neil Shapiro { 123806f25ae9SGregory Neil Shapiro if (m->m_argv[0] != NULL && 123906f25ae9SGregory Neil Shapiro strcmp(m->m_argv[0], "FILE") == 0) 124006f25ae9SGregory Neil Shapiro m->m_diagtype = "x-unix"; 124106f25ae9SGregory Neil Shapiro else 124206f25ae9SGregory Neil Shapiro m->m_diagtype = "smtp"; 124306f25ae9SGregory Neil Shapiro } 124406f25ae9SGregory Neil Shapiro } 124506f25ae9SGregory Neil Shapiro 1246c2aa98e2SPeter Wemm if (m->m_eol == NULL) 1247c2aa98e2SPeter Wemm { 1248c2aa98e2SPeter Wemm char **pp; 1249c2aa98e2SPeter Wemm 1250c2aa98e2SPeter Wemm /* default for SMTP is \r\n; use \n for local delivery */ 1251c2aa98e2SPeter Wemm for (pp = m->m_argv; *pp != NULL; pp++) 1252c2aa98e2SPeter Wemm { 1253c2aa98e2SPeter Wemm for (p = *pp; *p != '\0'; ) 1254c2aa98e2SPeter Wemm { 1255c2aa98e2SPeter Wemm if ((*p++ & 0377) == MACROEXPAND && *p == 'u') 1256c2aa98e2SPeter Wemm break; 1257c2aa98e2SPeter Wemm } 1258c2aa98e2SPeter Wemm if (*p != '\0') 1259c2aa98e2SPeter Wemm break; 1260c2aa98e2SPeter Wemm } 1261c2aa98e2SPeter Wemm if (*pp == NULL) 1262c2aa98e2SPeter Wemm m->m_eol = "\r\n"; 1263c2aa98e2SPeter Wemm else 1264c2aa98e2SPeter Wemm m->m_eol = "\n"; 1265c2aa98e2SPeter Wemm } 1266c2aa98e2SPeter Wemm 1267c2aa98e2SPeter Wemm /* enter the mailer into the symbol table */ 1268c2aa98e2SPeter Wemm s = stab(m->m_name, ST_MAILER, ST_ENTER); 1269c2aa98e2SPeter Wemm if (s->s_mailer != NULL) 1270c2aa98e2SPeter Wemm { 1271c2aa98e2SPeter Wemm i = s->s_mailer->m_mno; 1272c2aa98e2SPeter Wemm free(s->s_mailer); 1273c2aa98e2SPeter Wemm } 1274c2aa98e2SPeter Wemm else 1275c2aa98e2SPeter Wemm { 1276c2aa98e2SPeter Wemm i = NextMailer++; 1277c2aa98e2SPeter Wemm } 1278c2aa98e2SPeter Wemm Mailer[i] = s->s_mailer = m; 1279c2aa98e2SPeter Wemm m->m_mno = i; 1280c2aa98e2SPeter Wemm } 1281c2aa98e2SPeter Wemm /* 1282c2aa98e2SPeter Wemm ** MUNCHSTRING -- translate a string into internal form. 1283c2aa98e2SPeter Wemm ** 1284c2aa98e2SPeter Wemm ** Parameters: 1285c2aa98e2SPeter Wemm ** p -- the string to munch. 1286c2aa98e2SPeter Wemm ** delimptr -- if non-NULL, set to the pointer of the 1287c2aa98e2SPeter Wemm ** field delimiter character. 1288c2aa98e2SPeter Wemm ** delim -- the delimiter for the field. 1289c2aa98e2SPeter Wemm ** 1290c2aa98e2SPeter Wemm ** Returns: 1291c2aa98e2SPeter Wemm ** the munched string. 129206f25ae9SGregory Neil Shapiro ** 129306f25ae9SGregory Neil Shapiro ** Side Effects: 129406f25ae9SGregory Neil Shapiro ** the munched string is a local static buffer. 129506f25ae9SGregory Neil Shapiro ** it must be copied before the function is called again. 1296c2aa98e2SPeter Wemm */ 1297c2aa98e2SPeter Wemm 1298c2aa98e2SPeter Wemm char * 1299c2aa98e2SPeter Wemm munchstring(p, delimptr, delim) 1300c2aa98e2SPeter Wemm register char *p; 1301c2aa98e2SPeter Wemm char **delimptr; 1302c2aa98e2SPeter Wemm int delim; 1303c2aa98e2SPeter Wemm { 1304c2aa98e2SPeter Wemm register char *q; 1305c2aa98e2SPeter Wemm bool backslash = FALSE; 1306c2aa98e2SPeter Wemm bool quotemode = FALSE; 1307c2aa98e2SPeter Wemm static char buf[MAXLINE]; 1308c2aa98e2SPeter Wemm 1309c2aa98e2SPeter Wemm for (q = buf; *p != '\0' && q < &buf[sizeof buf - 1]; p++) 1310c2aa98e2SPeter Wemm { 1311c2aa98e2SPeter Wemm if (backslash) 1312c2aa98e2SPeter Wemm { 1313c2aa98e2SPeter Wemm /* everything is roughly literal */ 1314c2aa98e2SPeter Wemm backslash = FALSE; 1315c2aa98e2SPeter Wemm switch (*p) 1316c2aa98e2SPeter Wemm { 1317c2aa98e2SPeter Wemm case 'r': /* carriage return */ 1318c2aa98e2SPeter Wemm *q++ = '\r'; 1319c2aa98e2SPeter Wemm continue; 1320c2aa98e2SPeter Wemm 1321c2aa98e2SPeter Wemm case 'n': /* newline */ 1322c2aa98e2SPeter Wemm *q++ = '\n'; 1323c2aa98e2SPeter Wemm continue; 1324c2aa98e2SPeter Wemm 1325c2aa98e2SPeter Wemm case 'f': /* form feed */ 1326c2aa98e2SPeter Wemm *q++ = '\f'; 1327c2aa98e2SPeter Wemm continue; 1328c2aa98e2SPeter Wemm 1329c2aa98e2SPeter Wemm case 'b': /* backspace */ 1330c2aa98e2SPeter Wemm *q++ = '\b'; 1331c2aa98e2SPeter Wemm continue; 1332c2aa98e2SPeter Wemm } 1333c2aa98e2SPeter Wemm *q++ = *p; 1334c2aa98e2SPeter Wemm } 1335c2aa98e2SPeter Wemm else 1336c2aa98e2SPeter Wemm { 1337c2aa98e2SPeter Wemm if (*p == '\\') 1338c2aa98e2SPeter Wemm backslash = TRUE; 1339c2aa98e2SPeter Wemm else if (*p == '"') 1340c2aa98e2SPeter Wemm quotemode = !quotemode; 1341c2aa98e2SPeter Wemm else if (quotemode || *p != delim) 1342c2aa98e2SPeter Wemm *q++ = *p; 1343c2aa98e2SPeter Wemm else 1344c2aa98e2SPeter Wemm break; 1345c2aa98e2SPeter Wemm } 1346c2aa98e2SPeter Wemm } 1347c2aa98e2SPeter Wemm 1348c2aa98e2SPeter Wemm if (delimptr != NULL) 1349c2aa98e2SPeter Wemm *delimptr = p; 1350c2aa98e2SPeter Wemm *q++ = '\0'; 135106f25ae9SGregory Neil Shapiro return buf; 1352c2aa98e2SPeter Wemm } 1353c2aa98e2SPeter Wemm /* 1354c2aa98e2SPeter Wemm ** MAKEARGV -- break up a string into words 1355c2aa98e2SPeter Wemm ** 1356c2aa98e2SPeter Wemm ** Parameters: 1357c2aa98e2SPeter Wemm ** p -- the string to break up. 1358c2aa98e2SPeter Wemm ** 1359c2aa98e2SPeter Wemm ** Returns: 1360c2aa98e2SPeter Wemm ** a char **argv (dynamically allocated) 1361c2aa98e2SPeter Wemm ** 1362c2aa98e2SPeter Wemm ** Side Effects: 1363c2aa98e2SPeter Wemm ** munges p. 1364c2aa98e2SPeter Wemm */ 1365c2aa98e2SPeter Wemm 136606f25ae9SGregory Neil Shapiro static char ** 1367c2aa98e2SPeter Wemm makeargv(p) 1368c2aa98e2SPeter Wemm register char *p; 1369c2aa98e2SPeter Wemm { 1370c2aa98e2SPeter Wemm char *q; 1371c2aa98e2SPeter Wemm int i; 1372c2aa98e2SPeter Wemm char **avp; 1373c2aa98e2SPeter Wemm char *argv[MAXPV + 1]; 1374c2aa98e2SPeter Wemm 1375c2aa98e2SPeter Wemm /* take apart the words */ 1376c2aa98e2SPeter Wemm i = 0; 1377c2aa98e2SPeter Wemm while (*p != '\0' && i < MAXPV) 1378c2aa98e2SPeter Wemm { 1379c2aa98e2SPeter Wemm q = p; 1380c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 1381c2aa98e2SPeter Wemm p++; 1382c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 1383c2aa98e2SPeter Wemm *p++ = '\0'; 1384c2aa98e2SPeter Wemm argv[i++] = newstr(q); 1385c2aa98e2SPeter Wemm } 1386c2aa98e2SPeter Wemm argv[i++] = NULL; 1387c2aa98e2SPeter Wemm 1388c2aa98e2SPeter Wemm /* now make a copy of the argv */ 1389c2aa98e2SPeter Wemm avp = (char **) xalloc(sizeof *avp * i); 139006f25ae9SGregory Neil Shapiro memmove((char *) avp, (char *) argv, sizeof *avp * i); 1391c2aa98e2SPeter Wemm 139206f25ae9SGregory Neil Shapiro return avp; 1393c2aa98e2SPeter Wemm } 1394c2aa98e2SPeter Wemm /* 1395c2aa98e2SPeter Wemm ** PRINTRULES -- print rewrite rules (for debugging) 1396c2aa98e2SPeter Wemm ** 1397c2aa98e2SPeter Wemm ** Parameters: 1398c2aa98e2SPeter Wemm ** none. 1399c2aa98e2SPeter Wemm ** 1400c2aa98e2SPeter Wemm ** Returns: 1401c2aa98e2SPeter Wemm ** none. 1402c2aa98e2SPeter Wemm ** 1403c2aa98e2SPeter Wemm ** Side Effects: 1404c2aa98e2SPeter Wemm ** prints rewrite rules. 1405c2aa98e2SPeter Wemm */ 1406c2aa98e2SPeter Wemm 1407c2aa98e2SPeter Wemm void 1408c2aa98e2SPeter Wemm printrules() 1409c2aa98e2SPeter Wemm { 1410c2aa98e2SPeter Wemm register struct rewrite *rwp; 1411c2aa98e2SPeter Wemm register int ruleset; 1412c2aa98e2SPeter Wemm 1413c2aa98e2SPeter Wemm for (ruleset = 0; ruleset < 10; ruleset++) 1414c2aa98e2SPeter Wemm { 1415c2aa98e2SPeter Wemm if (RewriteRules[ruleset] == NULL) 1416c2aa98e2SPeter Wemm continue; 1417c2aa98e2SPeter Wemm printf("\n----Rule Set %d:", ruleset); 1418c2aa98e2SPeter Wemm 1419c2aa98e2SPeter Wemm for (rwp = RewriteRules[ruleset]; rwp != NULL; rwp = rwp->r_next) 1420c2aa98e2SPeter Wemm { 1421c2aa98e2SPeter Wemm printf("\nLHS:"); 1422c2aa98e2SPeter Wemm printav(rwp->r_lhs); 1423c2aa98e2SPeter Wemm printf("RHS:"); 1424c2aa98e2SPeter Wemm printav(rwp->r_rhs); 1425c2aa98e2SPeter Wemm } 1426c2aa98e2SPeter Wemm } 1427c2aa98e2SPeter Wemm } 1428c2aa98e2SPeter Wemm /* 1429c2aa98e2SPeter Wemm ** PRINTMAILER -- print mailer structure (for debugging) 1430c2aa98e2SPeter Wemm ** 1431c2aa98e2SPeter Wemm ** Parameters: 1432c2aa98e2SPeter Wemm ** m -- the mailer to print 1433c2aa98e2SPeter Wemm ** 1434c2aa98e2SPeter Wemm ** Returns: 1435c2aa98e2SPeter Wemm ** none. 1436c2aa98e2SPeter Wemm */ 1437c2aa98e2SPeter Wemm 1438c2aa98e2SPeter Wemm void 1439c2aa98e2SPeter Wemm printmailer(m) 1440c2aa98e2SPeter Wemm register MAILER *m; 1441c2aa98e2SPeter Wemm { 1442c2aa98e2SPeter Wemm int j; 1443c2aa98e2SPeter Wemm 144406f25ae9SGregory Neil Shapiro printf("mailer %d (%s): P=%s S=", m->m_mno, m->m_name, m->m_mailer); 144506f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_se_rwset] == NULL) 144606f25ae9SGregory Neil Shapiro printf("%d/", m->m_se_rwset); 144706f25ae9SGregory Neil Shapiro else 144806f25ae9SGregory Neil Shapiro printf("%s/", RuleSetNames[m->m_se_rwset]); 144906f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_sh_rwset] == NULL) 145006f25ae9SGregory Neil Shapiro printf("%d R=", m->m_sh_rwset); 145106f25ae9SGregory Neil Shapiro else 145206f25ae9SGregory Neil Shapiro printf("%s R=", RuleSetNames[m->m_sh_rwset]); 145306f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_re_rwset] == NULL) 145406f25ae9SGregory Neil Shapiro printf("%d/", m->m_re_rwset); 145506f25ae9SGregory Neil Shapiro else 145606f25ae9SGregory Neil Shapiro printf("%s/", RuleSetNames[m->m_re_rwset]); 145706f25ae9SGregory Neil Shapiro if (RuleSetNames[m->m_rh_rwset] == NULL) 145806f25ae9SGregory Neil Shapiro printf("%d ", m->m_rh_rwset); 145906f25ae9SGregory Neil Shapiro else 146006f25ae9SGregory Neil Shapiro printf("%s ", RuleSetNames[m->m_rh_rwset]); 146106f25ae9SGregory Neil Shapiro printf("M=%ld U=%d:%d F=", m->m_maxsize, 1462c2aa98e2SPeter Wemm (int) m->m_uid, (int) m->m_gid); 1463c2aa98e2SPeter Wemm for (j = '\0'; j <= '\177'; j++) 1464c2aa98e2SPeter Wemm if (bitnset(j, m->m_flags)) 1465c2aa98e2SPeter Wemm (void) putchar(j); 1466c2aa98e2SPeter Wemm printf(" L=%d E=", m->m_linelimit); 1467c2aa98e2SPeter Wemm xputs(m->m_eol); 1468c2aa98e2SPeter Wemm if (m->m_defcharset != NULL) 1469c2aa98e2SPeter Wemm printf(" C=%s", m->m_defcharset); 1470c2aa98e2SPeter Wemm printf(" T=%s/%s/%s", 1471c2aa98e2SPeter Wemm m->m_mtatype == NULL ? "<undefined>" : m->m_mtatype, 1472c2aa98e2SPeter Wemm m->m_addrtype == NULL ? "<undefined>" : m->m_addrtype, 1473c2aa98e2SPeter Wemm m->m_diagtype == NULL ? "<undefined>" : m->m_diagtype); 147406f25ae9SGregory Neil Shapiro #if _FFR_DYNAMIC_TOBUF 147506f25ae9SGregory Neil Shapiro printf(" r=%d", m->m_maxrcpt); 147606f25ae9SGregory Neil Shapiro #endif /* _FFR_DYNAMIC_TOBUF */ 1477c2aa98e2SPeter Wemm if (m->m_argv != NULL) 1478c2aa98e2SPeter Wemm { 1479c2aa98e2SPeter Wemm char **a = m->m_argv; 1480c2aa98e2SPeter Wemm 1481c2aa98e2SPeter Wemm printf(" A="); 1482c2aa98e2SPeter Wemm while (*a != NULL) 1483c2aa98e2SPeter Wemm { 1484c2aa98e2SPeter Wemm if (a != m->m_argv) 1485c2aa98e2SPeter Wemm printf(" "); 1486c2aa98e2SPeter Wemm xputs(*a++); 1487c2aa98e2SPeter Wemm } 1488c2aa98e2SPeter Wemm } 1489c2aa98e2SPeter Wemm printf("\n"); 1490c2aa98e2SPeter Wemm } 1491c2aa98e2SPeter Wemm /* 1492c2aa98e2SPeter Wemm ** SETOPTION -- set global processing option 1493c2aa98e2SPeter Wemm ** 1494c2aa98e2SPeter Wemm ** Parameters: 1495c2aa98e2SPeter Wemm ** opt -- option name. 1496c2aa98e2SPeter Wemm ** val -- option value (as a text string). 1497c2aa98e2SPeter Wemm ** safe -- set if this came from a configuration file. 1498c2aa98e2SPeter Wemm ** Some options (if set from the command line) will 1499c2aa98e2SPeter Wemm ** reset the user id to avoid security problems. 1500c2aa98e2SPeter Wemm ** sticky -- if set, don't let other setoptions override 1501c2aa98e2SPeter Wemm ** this value. 1502c2aa98e2SPeter Wemm ** e -- the main envelope. 1503c2aa98e2SPeter Wemm ** 1504c2aa98e2SPeter Wemm ** Returns: 1505c2aa98e2SPeter Wemm ** none. 1506c2aa98e2SPeter Wemm ** 1507c2aa98e2SPeter Wemm ** Side Effects: 1508c2aa98e2SPeter Wemm ** Sets options as implied by the arguments. 1509c2aa98e2SPeter Wemm */ 1510c2aa98e2SPeter Wemm 151106f25ae9SGregory Neil Shapiro static BITMAP256 StickyOpt; /* set if option is stuck */ 1512c2aa98e2SPeter Wemm 1513c2aa98e2SPeter Wemm #if NAMED_BIND 1514c2aa98e2SPeter Wemm 151506f25ae9SGregory Neil Shapiro static struct resolverflags 1516c2aa98e2SPeter Wemm { 1517c2aa98e2SPeter Wemm char *rf_name; /* name of the flag */ 1518c2aa98e2SPeter Wemm long rf_bits; /* bits to set/clear */ 1519c2aa98e2SPeter Wemm } ResolverFlags[] = 1520c2aa98e2SPeter Wemm { 1521c2aa98e2SPeter Wemm { "debug", RES_DEBUG }, 1522c2aa98e2SPeter Wemm { "aaonly", RES_AAONLY }, 1523c2aa98e2SPeter Wemm { "usevc", RES_USEVC }, 1524c2aa98e2SPeter Wemm { "primary", RES_PRIMARY }, 1525c2aa98e2SPeter Wemm { "igntc", RES_IGNTC }, 1526c2aa98e2SPeter Wemm { "recurse", RES_RECURSE }, 1527c2aa98e2SPeter Wemm { "defnames", RES_DEFNAMES }, 1528c2aa98e2SPeter Wemm { "stayopen", RES_STAYOPEN }, 1529c2aa98e2SPeter Wemm { "dnsrch", RES_DNSRCH }, 1530c2aa98e2SPeter Wemm { "true", 0 }, /* avoid error on old syntax */ 1531c2aa98e2SPeter Wemm { NULL, 0 } 1532c2aa98e2SPeter Wemm }; 1533c2aa98e2SPeter Wemm 153406f25ae9SGregory Neil Shapiro #endif /* NAMED_BIND */ 1535c2aa98e2SPeter Wemm 153606f25ae9SGregory Neil Shapiro #define OI_NONE 0 /* no special treatment */ 153706f25ae9SGregory Neil Shapiro #define OI_SAFE 0x0001 /* safe for random people to use */ 153806f25ae9SGregory Neil Shapiro #define OI_SUBOPT 0x0002 /* option has suboptions */ 153906f25ae9SGregory Neil Shapiro 154006f25ae9SGregory Neil Shapiro static struct optioninfo 1541c2aa98e2SPeter Wemm { 1542c2aa98e2SPeter Wemm char *o_name; /* long name of option */ 1543c2aa98e2SPeter Wemm u_char o_code; /* short name of option */ 154406f25ae9SGregory Neil Shapiro u_short o_flags; /* option flags */ 1545c2aa98e2SPeter Wemm } OptionTab[] = 1546c2aa98e2SPeter Wemm { 154706f25ae9SGregory Neil Shapiro #if defined(SUN_EXTENSIONS) && defined(REMOTE_MODE) 154806f25ae9SGregory Neil Shapiro { "RemoteMode", '>', OI_NONE }, 154906f25ae9SGregory Neil Shapiro #endif /* defined(SUN_EXTENSIONS) && defined(REMOTE_MODE) */ 155006f25ae9SGregory Neil Shapiro { "SevenBitInput", '7', OI_SAFE }, 1551c2aa98e2SPeter Wemm #if MIME8TO7 155206f25ae9SGregory Neil Shapiro { "EightBitMode", '8', OI_SAFE }, 155306f25ae9SGregory Neil Shapiro #endif /* MIME8TO7 */ 155406f25ae9SGregory Neil Shapiro { "AliasFile", 'A', OI_NONE }, 155506f25ae9SGregory Neil Shapiro { "AliasWait", 'a', OI_NONE }, 155606f25ae9SGregory Neil Shapiro { "BlankSub", 'B', OI_NONE }, 155706f25ae9SGregory Neil Shapiro { "MinFreeBlocks", 'b', OI_SAFE }, 155806f25ae9SGregory Neil Shapiro { "CheckpointInterval", 'C', OI_SAFE }, 155906f25ae9SGregory Neil Shapiro { "HoldExpensive", 'c', OI_NONE }, 156006f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_AUTOREBUILD 156106f25ae9SGregory Neil Shapiro { "AutoRebuildAliases", 'D', OI_NONE }, 156206f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_AUTOREBUILD */ 156306f25ae9SGregory Neil Shapiro { "DeliveryMode", 'd', OI_SAFE }, 156406f25ae9SGregory Neil Shapiro { "ErrorHeader", 'E', OI_NONE }, 156506f25ae9SGregory Neil Shapiro { "ErrorMode", 'e', OI_SAFE }, 156606f25ae9SGregory Neil Shapiro { "TempFileMode", 'F', OI_NONE }, 156706f25ae9SGregory Neil Shapiro { "SaveFromLine", 'f', OI_NONE }, 156806f25ae9SGregory Neil Shapiro { "MatchGECOS", 'G', OI_NONE }, 156906f25ae9SGregory Neil Shapiro { "HelpFile", 'H', OI_NONE }, 157006f25ae9SGregory Neil Shapiro { "MaxHopCount", 'h', OI_NONE }, 157106f25ae9SGregory Neil Shapiro { "ResolverOptions", 'I', OI_NONE }, 157206f25ae9SGregory Neil Shapiro { "IgnoreDots", 'i', OI_SAFE }, 157306f25ae9SGregory Neil Shapiro { "ForwardPath", 'J', OI_NONE }, 157406f25ae9SGregory Neil Shapiro { "SendMimeErrors", 'j', OI_SAFE }, 157506f25ae9SGregory Neil Shapiro { "ConnectionCacheSize", 'k', OI_NONE }, 157606f25ae9SGregory Neil Shapiro { "ConnectionCacheTimeout", 'K', OI_NONE }, 157706f25ae9SGregory Neil Shapiro { "UseErrorsTo", 'l', OI_NONE }, 157806f25ae9SGregory Neil Shapiro { "LogLevel", 'L', OI_SAFE }, 157906f25ae9SGregory Neil Shapiro { "MeToo", 'm', OI_SAFE }, 158006f25ae9SGregory Neil Shapiro { "CheckAliases", 'n', OI_NONE }, 158106f25ae9SGregory Neil Shapiro { "OldStyleHeaders", 'o', OI_SAFE }, 158206f25ae9SGregory Neil Shapiro { "DaemonPortOptions", 'O', OI_NONE }, 158306f25ae9SGregory Neil Shapiro { "PrivacyOptions", 'p', OI_SAFE }, 158406f25ae9SGregory Neil Shapiro { "PostmasterCopy", 'P', OI_NONE }, 158506f25ae9SGregory Neil Shapiro { "QueueFactor", 'q', OI_NONE }, 158606f25ae9SGregory Neil Shapiro { "QueueDirectory", 'Q', OI_NONE }, 158706f25ae9SGregory Neil Shapiro { "DontPruneRoutes", 'R', OI_NONE }, 158806f25ae9SGregory Neil Shapiro { "Timeout", 'r', OI_SUBOPT }, 158906f25ae9SGregory Neil Shapiro { "StatusFile", 'S', OI_NONE }, 159006f25ae9SGregory Neil Shapiro { "SuperSafe", 's', OI_SAFE }, 159106f25ae9SGregory Neil Shapiro { "QueueTimeout", 'T', OI_NONE }, 159206f25ae9SGregory Neil Shapiro { "TimeZoneSpec", 't', OI_NONE }, 159306f25ae9SGregory Neil Shapiro { "UserDatabaseSpec", 'U', OI_NONE }, 159406f25ae9SGregory Neil Shapiro { "DefaultUser", 'u', OI_NONE }, 159506f25ae9SGregory Neil Shapiro { "FallbackMXhost", 'V', OI_NONE }, 159606f25ae9SGregory Neil Shapiro { "Verbose", 'v', OI_SAFE }, 159706f25ae9SGregory Neil Shapiro { "TryNullMXList", 'w', OI_NONE }, 159806f25ae9SGregory Neil Shapiro { "QueueLA", 'x', OI_NONE }, 159906f25ae9SGregory Neil Shapiro { "RefuseLA", 'X', OI_NONE }, 160006f25ae9SGregory Neil Shapiro { "RecipientFactor", 'y', OI_NONE }, 160106f25ae9SGregory Neil Shapiro { "ForkEachJob", 'Y', OI_NONE }, 160206f25ae9SGregory Neil Shapiro { "ClassFactor", 'z', OI_NONE }, 160306f25ae9SGregory Neil Shapiro { "RetryFactor", 'Z', OI_NONE }, 1604c2aa98e2SPeter Wemm #define O_QUEUESORTORD 0x81 160506f25ae9SGregory Neil Shapiro { "QueueSortOrder", O_QUEUESORTORD, OI_SAFE }, 1606c2aa98e2SPeter Wemm #define O_HOSTSFILE 0x82 160706f25ae9SGregory Neil Shapiro { "HostsFile", O_HOSTSFILE, OI_NONE }, 1608c2aa98e2SPeter Wemm #define O_MQA 0x83 160906f25ae9SGregory Neil Shapiro { "MinQueueAge", O_MQA, OI_SAFE }, 1610c2aa98e2SPeter Wemm #define O_DEFCHARSET 0x85 161106f25ae9SGregory Neil Shapiro { "DefaultCharSet", O_DEFCHARSET, OI_SAFE }, 1612c2aa98e2SPeter Wemm #define O_SSFILE 0x86 161306f25ae9SGregory Neil Shapiro { "ServiceSwitchFile", O_SSFILE, OI_NONE }, 1614c2aa98e2SPeter Wemm #define O_DIALDELAY 0x87 161506f25ae9SGregory Neil Shapiro { "DialDelay", O_DIALDELAY, OI_SAFE }, 1616c2aa98e2SPeter Wemm #define O_NORCPTACTION 0x88 161706f25ae9SGregory Neil Shapiro { "NoRecipientAction", O_NORCPTACTION, OI_SAFE }, 1618c2aa98e2SPeter Wemm #define O_SAFEFILEENV 0x89 161906f25ae9SGregory Neil Shapiro { "SafeFileEnvironment", O_SAFEFILEENV, OI_NONE }, 1620c2aa98e2SPeter Wemm #define O_MAXMSGSIZE 0x8a 162106f25ae9SGregory Neil Shapiro { "MaxMessageSize", O_MAXMSGSIZE, OI_NONE }, 1622c2aa98e2SPeter Wemm #define O_COLONOKINADDR 0x8b 162306f25ae9SGregory Neil Shapiro { "ColonOkInAddr", O_COLONOKINADDR, OI_SAFE }, 1624c2aa98e2SPeter Wemm #define O_MAXQUEUERUN 0x8c 162506f25ae9SGregory Neil Shapiro { "MaxQueueRunSize", O_MAXQUEUERUN, OI_SAFE }, 1626c2aa98e2SPeter Wemm #define O_MAXCHILDREN 0x8d 162706f25ae9SGregory Neil Shapiro { "MaxDaemonChildren", O_MAXCHILDREN, OI_NONE }, 1628c2aa98e2SPeter Wemm #define O_KEEPCNAMES 0x8e 162906f25ae9SGregory Neil Shapiro { "DontExpandCnames", O_KEEPCNAMES, OI_NONE }, 1630c2aa98e2SPeter Wemm #define O_MUSTQUOTE 0x8f 163106f25ae9SGregory Neil Shapiro { "MustQuoteChars", O_MUSTQUOTE, OI_NONE }, 1632c2aa98e2SPeter Wemm #define O_SMTPGREETING 0x90 163306f25ae9SGregory Neil Shapiro { "SmtpGreetingMessage", O_SMTPGREETING, OI_NONE }, 1634c2aa98e2SPeter Wemm #define O_UNIXFROM 0x91 163506f25ae9SGregory Neil Shapiro { "UnixFromLine", O_UNIXFROM, OI_NONE }, 1636c2aa98e2SPeter Wemm #define O_OPCHARS 0x92 163706f25ae9SGregory Neil Shapiro { "OperatorChars", O_OPCHARS, OI_NONE }, 1638c2aa98e2SPeter Wemm #define O_DONTINITGRPS 0x93 163906f25ae9SGregory Neil Shapiro { "DontInitGroups", O_DONTINITGRPS, OI_NONE }, 1640c2aa98e2SPeter Wemm #define O_SLFH 0x94 164106f25ae9SGregory Neil Shapiro { "SingleLineFromHeader", O_SLFH, OI_SAFE }, 1642c2aa98e2SPeter Wemm #define O_ABH 0x95 164306f25ae9SGregory Neil Shapiro { "AllowBogusHELO", O_ABH, OI_SAFE }, 1644c2aa98e2SPeter Wemm #define O_CONNTHROT 0x97 164506f25ae9SGregory Neil Shapiro { "ConnectionRateThrottle", O_CONNTHROT, OI_NONE }, 1646c2aa98e2SPeter Wemm #define O_UGW 0x99 164706f25ae9SGregory Neil Shapiro { "UnsafeGroupWrites", O_UGW, OI_NONE }, 1648c2aa98e2SPeter Wemm #define O_DBLBOUNCE 0x9a 164906f25ae9SGregory Neil Shapiro { "DoubleBounceAddress", O_DBLBOUNCE, OI_NONE }, 1650c2aa98e2SPeter Wemm #define O_HSDIR 0x9b 165106f25ae9SGregory Neil Shapiro { "HostStatusDirectory", O_HSDIR, OI_NONE }, 1652c2aa98e2SPeter Wemm #define O_SINGTHREAD 0x9c 165306f25ae9SGregory Neil Shapiro { "SingleThreadDelivery", O_SINGTHREAD, OI_NONE }, 1654c2aa98e2SPeter Wemm #define O_RUNASUSER 0x9d 165506f25ae9SGregory Neil Shapiro { "RunAsUser", O_RUNASUSER, OI_NONE }, 1656c2aa98e2SPeter Wemm #define O_DSN_RRT 0x9e 165706f25ae9SGregory Neil Shapiro { "RrtImpliesDsn", O_DSN_RRT, OI_NONE }, 1658c2aa98e2SPeter Wemm #define O_PIDFILE 0x9f 165906f25ae9SGregory Neil Shapiro { "PidFile", O_PIDFILE, OI_NONE }, 1660c2aa98e2SPeter Wemm #define O_DONTBLAMESENDMAIL 0xa0 166106f25ae9SGregory Neil Shapiro { "DontBlameSendmail", O_DONTBLAMESENDMAIL, OI_NONE }, 1662c2aa98e2SPeter Wemm #define O_DPI 0xa1 166306f25ae9SGregory Neil Shapiro { "DontProbeInterfaces", O_DPI, OI_NONE }, 1664c2aa98e2SPeter Wemm #define O_MAXRCPT 0xa2 166506f25ae9SGregory Neil Shapiro { "MaxRecipientsPerMessage", O_MAXRCPT, OI_SAFE }, 1666c2aa98e2SPeter Wemm #define O_DEADLETTER 0xa3 166706f25ae9SGregory Neil Shapiro { "DeadLetterDrop", O_DEADLETTER, OI_NONE }, 1668c2aa98e2SPeter Wemm #if _FFR_DONTLOCKFILESFORREAD_OPTION 1669c2aa98e2SPeter Wemm # define O_DONTLOCK 0xa4 167006f25ae9SGregory Neil Shapiro { "DontLockFilesForRead", O_DONTLOCK, OI_NONE }, 167106f25ae9SGregory Neil Shapiro #endif /* _FFR_DONTLOCKFILESFORREAD_OPTION */ 1672c2aa98e2SPeter Wemm #define O_MAXALIASRCSN 0xa5 167306f25ae9SGregory Neil Shapiro { "MaxAliasRecursion", O_MAXALIASRCSN, OI_NONE }, 1674c2aa98e2SPeter Wemm #define O_CNCTONLYTO 0xa6 167506f25ae9SGregory Neil Shapiro { "ConnectOnlyTo", O_CNCTONLYTO, OI_NONE }, 1676065a643dSPeter Wemm #define O_TRUSTUSER 0xa7 167706f25ae9SGregory Neil Shapiro { "TrustedUser", O_TRUSTUSER, OI_NONE }, 1678065a643dSPeter Wemm #define O_MAXMIMEHDRLEN 0xa8 167906f25ae9SGregory Neil Shapiro { "MaxMimeHeaderLength", O_MAXMIMEHDRLEN, OI_NONE }, 1680065a643dSPeter Wemm #define O_CONTROLSOCKET 0xa9 168106f25ae9SGregory Neil Shapiro { "ControlSocketName", O_CONTROLSOCKET, OI_NONE }, 16822e43090eSPeter Wemm #define O_MAXHDRSLEN 0xaa 168306f25ae9SGregory Neil Shapiro { "MaxHeadersLength", O_MAXHDRSLEN, OI_NONE }, 168406f25ae9SGregory Neil Shapiro #if _FFR_MAX_FORWARD_ENTRIES 168506f25ae9SGregory Neil Shapiro # define O_MAXFORWARD 0xab 168606f25ae9SGregory Neil Shapiro { "MaxForwardEntries", O_MAXFORWARD, OI_NONE }, 168706f25ae9SGregory Neil Shapiro #endif /* _FFR_MAX_FORWARD_ENTRIES */ 168806f25ae9SGregory Neil Shapiro #define O_PROCTITLEPREFIX 0xac 168906f25ae9SGregory Neil Shapiro { "ProcessTitlePrefix", O_PROCTITLEPREFIX, OI_NONE }, 169006f25ae9SGregory Neil Shapiro #define O_SASLINFO 0xad 169106f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 169206f25ae9SGregory Neil Shapiro { "DefaultAuthInfo", O_SASLINFO, OI_SAFE }, 169306f25ae9SGregory Neil Shapiro #else /* _FFR_ALLOW_SASLINFO */ 169406f25ae9SGregory Neil Shapiro { "DefaultAuthInfo", O_SASLINFO, OI_NONE }, 169506f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 169606f25ae9SGregory Neil Shapiro #define O_SASLMECH 0xae 169706f25ae9SGregory Neil Shapiro { "AuthMechanisms", O_SASLMECH, OI_NONE }, 169806f25ae9SGregory Neil Shapiro #define O_CLIENTPORT 0xaf 169906f25ae9SGregory Neil Shapiro { "ClientPortOptions", O_CLIENTPORT, OI_NONE }, 170006f25ae9SGregory Neil Shapiro #define O_DF_BUFSIZE 0xb0 170106f25ae9SGregory Neil Shapiro { "DataFileBufferSize", O_DF_BUFSIZE, OI_NONE }, 170206f25ae9SGregory Neil Shapiro #define O_XF_BUFSIZE 0xb1 170306f25ae9SGregory Neil Shapiro { "XscriptFileBufferSize", O_XF_BUFSIZE, OI_NONE }, 170406f25ae9SGregory Neil Shapiro # define O_LDAPDEFAULTSPEC 0xb2 170506f25ae9SGregory Neil Shapiro { "LDAPDefaultSpec", O_LDAPDEFAULTSPEC, OI_NONE }, 170606f25ae9SGregory Neil Shapiro #if _FFR_QUEUEDELAY 170706f25ae9SGregory Neil Shapiro #define O_QUEUEDELAY 0xb3 170806f25ae9SGregory Neil Shapiro { "QueueDelay", O_QUEUEDELAY, OI_NONE }, 170906f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUEDELAY */ 171006f25ae9SGregory Neil Shapiro # define O_SRVCERTFILE 0xb4 171106f25ae9SGregory Neil Shapiro { "ServerCertFile", O_SRVCERTFILE, OI_NONE }, 171206f25ae9SGregory Neil Shapiro # define O_SRVKEYFILE 0xb5 171306f25ae9SGregory Neil Shapiro { "Serverkeyfile", O_SRVKEYFILE, OI_NONE }, 171406f25ae9SGregory Neil Shapiro # define O_CLTCERTFILE 0xb6 171506f25ae9SGregory Neil Shapiro { "ClientCertFile", O_CLTCERTFILE, OI_NONE }, 171606f25ae9SGregory Neil Shapiro # define O_CLTKEYFILE 0xb7 171706f25ae9SGregory Neil Shapiro { "Clientkeyfile", O_CLTKEYFILE, OI_NONE }, 171806f25ae9SGregory Neil Shapiro # define O_CACERTFILE 0xb8 171906f25ae9SGregory Neil Shapiro { "CACERTFile", O_CACERTFILE, OI_NONE }, 172006f25ae9SGregory Neil Shapiro # define O_CACERTPATH 0xb9 172106f25ae9SGregory Neil Shapiro { "CACERTPath", O_CACERTPATH, OI_NONE }, 172206f25ae9SGregory Neil Shapiro # define O_DHPARAMS 0xba 172306f25ae9SGregory Neil Shapiro { "DHParameters", O_DHPARAMS, OI_NONE }, 172406f25ae9SGregory Neil Shapiro #if _FFR_MILTER 172506f25ae9SGregory Neil Shapiro #define O_INPUTMILTER 0xbb 172606f25ae9SGregory Neil Shapiro { "InputMailFilters", O_INPUTMILTER, OI_NONE }, 172706f25ae9SGregory Neil Shapiro #define O_MILTER 0xbc 172806f25ae9SGregory Neil Shapiro { "Milter", O_MILTER, OI_SUBOPT }, 172906f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 173006f25ae9SGregory Neil Shapiro #define O_SASLOPTS 0xbd 173106f25ae9SGregory Neil Shapiro { "AuthOptions", O_SASLOPTS, OI_NONE }, 173206f25ae9SGregory Neil Shapiro #if _FFR_QUEUE_FILE_MODE 173306f25ae9SGregory Neil Shapiro #define O_QUEUE_FILE_MODE 0xbe 173406f25ae9SGregory Neil Shapiro { "QueueFileMode", O_QUEUE_FILE_MODE, OI_NONE }, 173506f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUE_FILE_MODE */ 173606f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 173706f25ae9SGregory Neil Shapiro # define O_DHPARAMS5 0xbf 173806f25ae9SGregory Neil Shapiro { "DHParameters512", O_DHPARAMS5, OI_NONE }, 173906f25ae9SGregory Neil Shapiro # define O_CIPHERLIST 0xc0 174006f25ae9SGregory Neil Shapiro { "CipherList", O_CIPHERLIST, OI_NONE }, 174106f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 174206f25ae9SGregory Neil Shapiro # define O_RANDFILE 0xc1 174306f25ae9SGregory Neil Shapiro { "RandFile", O_RANDFILE, OI_NONE }, 174406f25ae9SGregory Neil Shapiro { NULL, '\0', OI_NONE } 1745c2aa98e2SPeter Wemm }; 1746c2aa98e2SPeter Wemm 1747c2aa98e2SPeter Wemm void 1748c2aa98e2SPeter Wemm setoption(opt, val, safe, sticky, e) 1749c2aa98e2SPeter Wemm int opt; 1750c2aa98e2SPeter Wemm char *val; 1751c2aa98e2SPeter Wemm bool safe; 1752c2aa98e2SPeter Wemm bool sticky; 1753c2aa98e2SPeter Wemm register ENVELOPE *e; 1754c2aa98e2SPeter Wemm { 1755c2aa98e2SPeter Wemm register char *p; 1756c2aa98e2SPeter Wemm register struct optioninfo *o; 1757c2aa98e2SPeter Wemm char *subopt; 1758c2aa98e2SPeter Wemm int mid; 1759c2aa98e2SPeter Wemm bool can_setuid = RunAsUid == 0; 1760c2aa98e2SPeter Wemm auto char *ep; 1761c2aa98e2SPeter Wemm char buf[50]; 1762c2aa98e2SPeter Wemm extern bool Warn_Q_option; 176306f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 176406f25ae9SGregory Neil Shapiro extern int SubmitMode; 176506f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 1766c2aa98e2SPeter Wemm 1767c2aa98e2SPeter Wemm errno = 0; 1768c2aa98e2SPeter Wemm if (opt == ' ') 1769c2aa98e2SPeter Wemm { 1770c2aa98e2SPeter Wemm /* full word options */ 1771c2aa98e2SPeter Wemm struct optioninfo *sel; 1772c2aa98e2SPeter Wemm 1773c2aa98e2SPeter Wemm p = strchr(val, '='); 1774c2aa98e2SPeter Wemm if (p == NULL) 1775c2aa98e2SPeter Wemm p = &val[strlen(val)]; 1776c2aa98e2SPeter Wemm while (*--p == ' ') 1777c2aa98e2SPeter Wemm continue; 1778c2aa98e2SPeter Wemm while (*++p == ' ') 1779c2aa98e2SPeter Wemm *p = '\0'; 1780c2aa98e2SPeter Wemm if (p == val) 1781c2aa98e2SPeter Wemm { 1782c2aa98e2SPeter Wemm syserr("readcf: null option name"); 1783c2aa98e2SPeter Wemm return; 1784c2aa98e2SPeter Wemm } 1785c2aa98e2SPeter Wemm if (*p == '=') 1786c2aa98e2SPeter Wemm *p++ = '\0'; 1787c2aa98e2SPeter Wemm while (*p == ' ') 1788c2aa98e2SPeter Wemm p++; 1789c2aa98e2SPeter Wemm subopt = strchr(val, '.'); 1790c2aa98e2SPeter Wemm if (subopt != NULL) 1791c2aa98e2SPeter Wemm *subopt++ = '\0'; 1792c2aa98e2SPeter Wemm sel = NULL; 1793c2aa98e2SPeter Wemm for (o = OptionTab; o->o_name != NULL; o++) 1794c2aa98e2SPeter Wemm { 1795c2aa98e2SPeter Wemm if (strncasecmp(o->o_name, val, strlen(val)) != 0) 1796c2aa98e2SPeter Wemm continue; 1797c2aa98e2SPeter Wemm if (strlen(o->o_name) == strlen(val)) 1798c2aa98e2SPeter Wemm { 1799c2aa98e2SPeter Wemm /* completely specified -- this must be it */ 1800c2aa98e2SPeter Wemm sel = NULL; 1801c2aa98e2SPeter Wemm break; 1802c2aa98e2SPeter Wemm } 1803c2aa98e2SPeter Wemm if (sel != NULL) 1804c2aa98e2SPeter Wemm break; 1805c2aa98e2SPeter Wemm sel = o; 1806c2aa98e2SPeter Wemm } 1807c2aa98e2SPeter Wemm if (sel != NULL && o->o_name == NULL) 1808c2aa98e2SPeter Wemm o = sel; 1809c2aa98e2SPeter Wemm else if (o->o_name == NULL) 1810c2aa98e2SPeter Wemm { 1811c2aa98e2SPeter Wemm syserr("readcf: unknown option name %s", val); 1812c2aa98e2SPeter Wemm return; 1813c2aa98e2SPeter Wemm } 1814c2aa98e2SPeter Wemm else if (sel != NULL) 1815c2aa98e2SPeter Wemm { 1816c2aa98e2SPeter Wemm syserr("readcf: ambiguous option name %s (matches %s and %s)", 1817c2aa98e2SPeter Wemm val, sel->o_name, o->o_name); 1818c2aa98e2SPeter Wemm return; 1819c2aa98e2SPeter Wemm } 1820c2aa98e2SPeter Wemm if (strlen(val) != strlen(o->o_name)) 1821c2aa98e2SPeter Wemm { 1822c2aa98e2SPeter Wemm int oldVerbose = Verbose; 1823c2aa98e2SPeter Wemm 1824c2aa98e2SPeter Wemm Verbose = 1; 1825c2aa98e2SPeter Wemm message("Option %s used as abbreviation for %s", 1826c2aa98e2SPeter Wemm val, o->o_name); 1827c2aa98e2SPeter Wemm Verbose = oldVerbose; 1828c2aa98e2SPeter Wemm } 1829c2aa98e2SPeter Wemm opt = o->o_code; 1830c2aa98e2SPeter Wemm val = p; 1831c2aa98e2SPeter Wemm } 1832c2aa98e2SPeter Wemm else 1833c2aa98e2SPeter Wemm { 1834c2aa98e2SPeter Wemm for (o = OptionTab; o->o_name != NULL; o++) 1835c2aa98e2SPeter Wemm { 1836c2aa98e2SPeter Wemm if (o->o_code == opt) 1837c2aa98e2SPeter Wemm break; 1838c2aa98e2SPeter Wemm } 1839c2aa98e2SPeter Wemm subopt = NULL; 1840c2aa98e2SPeter Wemm } 1841c2aa98e2SPeter Wemm 184206f25ae9SGregory Neil Shapiro if (subopt != NULL && !bitset(OI_SUBOPT, o->o_flags)) 184306f25ae9SGregory Neil Shapiro { 184406f25ae9SGregory Neil Shapiro if (tTd(37, 1)) 184506f25ae9SGregory Neil Shapiro dprintf("setoption: %s does not support suboptions, ignoring .%s\n", 184606f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name, 184706f25ae9SGregory Neil Shapiro subopt); 184806f25ae9SGregory Neil Shapiro subopt = NULL; 184906f25ae9SGregory Neil Shapiro } 185006f25ae9SGregory Neil Shapiro 1851c2aa98e2SPeter Wemm if (tTd(37, 1)) 1852c2aa98e2SPeter Wemm { 185306f25ae9SGregory Neil Shapiro dprintf(isascii(opt) && isprint(opt) ? 185406f25ae9SGregory Neil Shapiro "setoption %s (%c)%s%s=" : 185506f25ae9SGregory Neil Shapiro "setoption %s (0x%x)%s%s=", 1856c2aa98e2SPeter Wemm o->o_name == NULL ? "<unknown>" : o->o_name, 1857c2aa98e2SPeter Wemm opt, 185806f25ae9SGregory Neil Shapiro subopt == NULL ? "" : ".", 1859c2aa98e2SPeter Wemm subopt == NULL ? "" : subopt); 1860c2aa98e2SPeter Wemm xputs(val); 1861c2aa98e2SPeter Wemm } 1862c2aa98e2SPeter Wemm 1863c2aa98e2SPeter Wemm /* 1864c2aa98e2SPeter Wemm ** See if this option is preset for us. 1865c2aa98e2SPeter Wemm */ 1866c2aa98e2SPeter Wemm 1867c2aa98e2SPeter Wemm if (!sticky && bitnset(opt, StickyOpt)) 1868c2aa98e2SPeter Wemm { 1869c2aa98e2SPeter Wemm if (tTd(37, 1)) 187006f25ae9SGregory Neil Shapiro dprintf(" (ignored)\n"); 1871c2aa98e2SPeter Wemm return; 1872c2aa98e2SPeter Wemm } 1873c2aa98e2SPeter Wemm 1874c2aa98e2SPeter Wemm /* 1875c2aa98e2SPeter Wemm ** Check to see if this option can be specified by this user. 1876c2aa98e2SPeter Wemm */ 1877c2aa98e2SPeter Wemm 1878c2aa98e2SPeter Wemm if (!safe && RealUid == 0) 1879c2aa98e2SPeter Wemm safe = TRUE; 188006f25ae9SGregory Neil Shapiro if (!safe && !bitset(OI_SAFE, o->o_flags)) 1881c2aa98e2SPeter Wemm { 1882c2aa98e2SPeter Wemm if (opt != 'M' || (val[0] != 'r' && val[0] != 's')) 1883c2aa98e2SPeter Wemm { 188406f25ae9SGregory Neil Shapiro int dp; 188506f25ae9SGregory Neil Shapiro 1886c2aa98e2SPeter Wemm if (tTd(37, 1)) 188706f25ae9SGregory Neil Shapiro dprintf(" (unsafe)"); 188806f25ae9SGregory Neil Shapiro dp = drop_privileges(TRUE); 188906f25ae9SGregory Neil Shapiro setstat(dp); 1890c2aa98e2SPeter Wemm } 1891c2aa98e2SPeter Wemm } 1892c2aa98e2SPeter Wemm if (tTd(37, 1)) 189306f25ae9SGregory Neil Shapiro dprintf("\n"); 1894c2aa98e2SPeter Wemm 1895c2aa98e2SPeter Wemm switch (opt & 0xff) 1896c2aa98e2SPeter Wemm { 1897c2aa98e2SPeter Wemm case '7': /* force seven-bit input */ 1898c2aa98e2SPeter Wemm SevenBitInput = atobool(val); 1899c2aa98e2SPeter Wemm break; 1900c2aa98e2SPeter Wemm 1901c2aa98e2SPeter Wemm #if MIME8TO7 1902c2aa98e2SPeter Wemm case '8': /* handling of 8-bit input */ 1903c2aa98e2SPeter Wemm switch (*val) 1904c2aa98e2SPeter Wemm { 1905c2aa98e2SPeter Wemm case 'm': /* convert 8-bit, convert MIME */ 1906c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME|MM_MIME8BIT; 1907c2aa98e2SPeter Wemm break; 1908c2aa98e2SPeter Wemm 1909c2aa98e2SPeter Wemm case 'p': /* pass 8 bit, convert MIME */ 1910c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME|MM_PASS8BIT; 1911c2aa98e2SPeter Wemm break; 1912c2aa98e2SPeter Wemm 1913c2aa98e2SPeter Wemm case 's': /* strict adherence */ 1914c2aa98e2SPeter Wemm MimeMode = MM_CVTMIME; 1915c2aa98e2SPeter Wemm break; 1916c2aa98e2SPeter Wemm 1917c2aa98e2SPeter Wemm # if 0 1918c2aa98e2SPeter Wemm case 'r': /* reject 8-bit, don't convert MIME */ 1919c2aa98e2SPeter Wemm MimeMode = 0; 1920c2aa98e2SPeter Wemm break; 1921c2aa98e2SPeter Wemm 1922c2aa98e2SPeter Wemm case 'j': /* "just send 8" */ 1923c2aa98e2SPeter Wemm MimeMode = MM_PASS8BIT; 1924c2aa98e2SPeter Wemm break; 1925c2aa98e2SPeter Wemm 1926c2aa98e2SPeter Wemm case 'a': /* encode 8 bit if available */ 1927c2aa98e2SPeter Wemm MimeMode = MM_MIME8BIT|MM_PASS8BIT|MM_CVTMIME; 1928c2aa98e2SPeter Wemm break; 1929c2aa98e2SPeter Wemm 1930c2aa98e2SPeter Wemm case 'c': /* convert 8 bit to MIME, never 7 bit */ 1931c2aa98e2SPeter Wemm MimeMode = MM_MIME8BIT; 1932c2aa98e2SPeter Wemm break; 193306f25ae9SGregory Neil Shapiro # endif /* 0 */ 1934c2aa98e2SPeter Wemm 1935c2aa98e2SPeter Wemm default: 1936c2aa98e2SPeter Wemm syserr("Unknown 8-bit mode %c", *val); 1937065a643dSPeter Wemm finis(FALSE, EX_USAGE); 1938c2aa98e2SPeter Wemm } 1939c2aa98e2SPeter Wemm break; 194006f25ae9SGregory Neil Shapiro #endif /* MIME8TO7 */ 1941c2aa98e2SPeter Wemm 1942c2aa98e2SPeter Wemm case 'A': /* set default alias file */ 1943c2aa98e2SPeter Wemm if (val[0] == '\0') 1944c2aa98e2SPeter Wemm setalias("aliases"); 1945c2aa98e2SPeter Wemm else 1946c2aa98e2SPeter Wemm setalias(val); 1947c2aa98e2SPeter Wemm break; 1948c2aa98e2SPeter Wemm 1949c2aa98e2SPeter Wemm case 'a': /* look N minutes for "@:@" in alias file */ 1950c2aa98e2SPeter Wemm if (val[0] == '\0') 1951c2aa98e2SPeter Wemm SafeAlias = 5 * 60; /* five minutes */ 1952c2aa98e2SPeter Wemm else 1953c2aa98e2SPeter Wemm SafeAlias = convtime(val, 'm'); 1954c2aa98e2SPeter Wemm break; 1955c2aa98e2SPeter Wemm 1956c2aa98e2SPeter Wemm case 'B': /* substitution for blank character */ 1957c2aa98e2SPeter Wemm SpaceSub = val[0]; 1958c2aa98e2SPeter Wemm if (SpaceSub == '\0') 1959c2aa98e2SPeter Wemm SpaceSub = ' '; 1960c2aa98e2SPeter Wemm break; 1961c2aa98e2SPeter Wemm 1962c2aa98e2SPeter Wemm case 'b': /* min blocks free on queue fs/max msg size */ 1963c2aa98e2SPeter Wemm p = strchr(val, '/'); 1964c2aa98e2SPeter Wemm if (p != NULL) 1965c2aa98e2SPeter Wemm { 1966c2aa98e2SPeter Wemm *p++ = '\0'; 1967c2aa98e2SPeter Wemm MaxMessageSize = atol(p); 1968c2aa98e2SPeter Wemm } 1969c2aa98e2SPeter Wemm MinBlocksFree = atol(val); 1970c2aa98e2SPeter Wemm break; 1971c2aa98e2SPeter Wemm 1972c2aa98e2SPeter Wemm case 'c': /* don't connect to "expensive" mailers */ 1973c2aa98e2SPeter Wemm NoConnect = atobool(val); 1974c2aa98e2SPeter Wemm break; 1975c2aa98e2SPeter Wemm 1976c2aa98e2SPeter Wemm case 'C': /* checkpoint every N addresses */ 1977c2aa98e2SPeter Wemm CheckpointInterval = atoi(val); 1978c2aa98e2SPeter Wemm break; 1979c2aa98e2SPeter Wemm 1980c2aa98e2SPeter Wemm case 'd': /* delivery mode */ 1981c2aa98e2SPeter Wemm switch (*val) 1982c2aa98e2SPeter Wemm { 1983c2aa98e2SPeter Wemm case '\0': 198406f25ae9SGregory Neil Shapiro set_delivery_mode(SM_DELIVER, e); 1985c2aa98e2SPeter Wemm break; 1986c2aa98e2SPeter Wemm 1987c2aa98e2SPeter Wemm case SM_QUEUE: /* queue only */ 1988c2aa98e2SPeter Wemm case SM_DEFER: /* queue only and defer map lookups */ 1989c2aa98e2SPeter Wemm #if !QUEUE 1990c2aa98e2SPeter Wemm syserr("need QUEUE to set -odqueue or -oddefer"); 1991193538b7SGregory Neil Shapiro break; 199206f25ae9SGregory Neil Shapiro #endif /* !QUEUE */ 199306f25ae9SGregory Neil Shapiro /* FALLTHROUGH */ 1994c2aa98e2SPeter Wemm 1995c2aa98e2SPeter Wemm case SM_DELIVER: /* do everything */ 1996c2aa98e2SPeter Wemm case SM_FORK: /* fork after verification */ 199706f25ae9SGregory Neil Shapiro set_delivery_mode(*val, e); 1998c2aa98e2SPeter Wemm break; 1999c2aa98e2SPeter Wemm 2000c2aa98e2SPeter Wemm default: 2001c2aa98e2SPeter Wemm syserr("Unknown delivery mode %c", *val); 2002065a643dSPeter Wemm finis(FALSE, EX_USAGE); 2003c2aa98e2SPeter Wemm } 2004c2aa98e2SPeter Wemm break; 2005c2aa98e2SPeter Wemm 200606f25ae9SGregory Neil Shapiro #if !_FFR_REMOVE_AUTOREBUILD 2007c2aa98e2SPeter Wemm case 'D': /* rebuild alias database as needed */ 2008c2aa98e2SPeter Wemm AutoRebuild = atobool(val); 2009c2aa98e2SPeter Wemm break; 201006f25ae9SGregory Neil Shapiro #endif /* !_FFR_REMOVE_AUTOREBUILD */ 2011c2aa98e2SPeter Wemm 2012c2aa98e2SPeter Wemm case 'E': /* error message header/header file */ 2013c2aa98e2SPeter Wemm if (*val != '\0') 2014c2aa98e2SPeter Wemm ErrMsgFile = newstr(val); 2015c2aa98e2SPeter Wemm break; 2016c2aa98e2SPeter Wemm 2017c2aa98e2SPeter Wemm case 'e': /* set error processing mode */ 2018c2aa98e2SPeter Wemm switch (*val) 2019c2aa98e2SPeter Wemm { 2020c2aa98e2SPeter Wemm case EM_QUIET: /* be silent about it */ 2021c2aa98e2SPeter Wemm case EM_MAIL: /* mail back */ 2022c2aa98e2SPeter Wemm case EM_BERKNET: /* do berknet error processing */ 2023c2aa98e2SPeter Wemm case EM_WRITE: /* write back (or mail) */ 2024c2aa98e2SPeter Wemm case EM_PRINT: /* print errors normally (default) */ 2025c2aa98e2SPeter Wemm e->e_errormode = *val; 2026c2aa98e2SPeter Wemm break; 2027c2aa98e2SPeter Wemm } 2028c2aa98e2SPeter Wemm break; 2029c2aa98e2SPeter Wemm 2030c2aa98e2SPeter Wemm case 'F': /* file mode */ 2031c2aa98e2SPeter Wemm FileMode = atooct(val) & 0777; 2032c2aa98e2SPeter Wemm break; 2033c2aa98e2SPeter Wemm 2034c2aa98e2SPeter Wemm case 'f': /* save Unix-style From lines on front */ 2035c2aa98e2SPeter Wemm SaveFrom = atobool(val); 2036c2aa98e2SPeter Wemm break; 2037c2aa98e2SPeter Wemm 2038c2aa98e2SPeter Wemm case 'G': /* match recipients against GECOS field */ 2039c2aa98e2SPeter Wemm MatchGecos = atobool(val); 2040c2aa98e2SPeter Wemm break; 2041c2aa98e2SPeter Wemm 2042c2aa98e2SPeter Wemm case 'g': /* default gid */ 2043c2aa98e2SPeter Wemm g_opt: 2044c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2045c2aa98e2SPeter Wemm DefGid = atoi(val); 2046c2aa98e2SPeter Wemm else 2047c2aa98e2SPeter Wemm { 2048c2aa98e2SPeter Wemm register struct group *gr; 2049c2aa98e2SPeter Wemm 2050c2aa98e2SPeter Wemm DefGid = -1; 2051c2aa98e2SPeter Wemm gr = getgrnam(val); 2052c2aa98e2SPeter Wemm if (gr == NULL) 2053c2aa98e2SPeter Wemm syserr("readcf: option %c: unknown group %s", 2054c2aa98e2SPeter Wemm opt, val); 2055c2aa98e2SPeter Wemm else 2056c2aa98e2SPeter Wemm DefGid = gr->gr_gid; 2057c2aa98e2SPeter Wemm } 2058c2aa98e2SPeter Wemm break; 2059c2aa98e2SPeter Wemm 2060c2aa98e2SPeter Wemm case 'H': /* help file */ 2061c2aa98e2SPeter Wemm if (val[0] == '\0') 206206f25ae9SGregory Neil Shapiro HelpFile = "helpfile"; 2063c2aa98e2SPeter Wemm else 2064602a2b1bSGregory Neil Shapiro { 2065c2aa98e2SPeter Wemm HelpFile = newstr(val); 2066602a2b1bSGregory Neil Shapiro } 2067c2aa98e2SPeter Wemm break; 2068c2aa98e2SPeter Wemm 2069c2aa98e2SPeter Wemm case 'h': /* maximum hop count */ 2070c2aa98e2SPeter Wemm MaxHopCount = atoi(val); 2071c2aa98e2SPeter Wemm break; 2072c2aa98e2SPeter Wemm 2073c2aa98e2SPeter Wemm case 'I': /* use internet domain name server */ 2074c2aa98e2SPeter Wemm #if NAMED_BIND 2075c2aa98e2SPeter Wemm for (p = val; *p != 0; ) 2076c2aa98e2SPeter Wemm { 2077c2aa98e2SPeter Wemm bool clearmode; 2078c2aa98e2SPeter Wemm char *q; 2079c2aa98e2SPeter Wemm struct resolverflags *rfp; 2080c2aa98e2SPeter Wemm 2081c2aa98e2SPeter Wemm while (*p == ' ') 2082c2aa98e2SPeter Wemm p++; 2083c2aa98e2SPeter Wemm if (*p == '\0') 2084c2aa98e2SPeter Wemm break; 2085c2aa98e2SPeter Wemm clearmode = FALSE; 2086c2aa98e2SPeter Wemm if (*p == '-') 2087c2aa98e2SPeter Wemm clearmode = TRUE; 2088c2aa98e2SPeter Wemm else if (*p != '+') 2089c2aa98e2SPeter Wemm p--; 2090c2aa98e2SPeter Wemm p++; 2091c2aa98e2SPeter Wemm q = p; 2092c2aa98e2SPeter Wemm while (*p != '\0' && !(isascii(*p) && isspace(*p))) 2093c2aa98e2SPeter Wemm p++; 2094c2aa98e2SPeter Wemm if (*p != '\0') 2095c2aa98e2SPeter Wemm *p++ = '\0'; 2096c2aa98e2SPeter Wemm if (strcasecmp(q, "HasWildcardMX") == 0) 2097c2aa98e2SPeter Wemm { 2098c2aa98e2SPeter Wemm HasWildcardMX = !clearmode; 2099c2aa98e2SPeter Wemm continue; 2100c2aa98e2SPeter Wemm } 2101602a2b1bSGregory Neil Shapiro #if _FFR_WORKAROUND_BROKEN_NAMESERVERS 2102602a2b1bSGregory Neil Shapiro if (sm_strcasecmp(q, "WorkAroundBrokenAAAA") == 0) 2103602a2b1bSGregory Neil Shapiro { 2104602a2b1bSGregory Neil Shapiro WorkAroundBrokenAAAA = !clearmode; 2105602a2b1bSGregory Neil Shapiro continue; 2106602a2b1bSGregory Neil Shapiro } 2107602a2b1bSGregory Neil Shapiro #endif /* _FFR_WORKAROUND_BROKEN_NAMESERVERS */ 2108c2aa98e2SPeter Wemm for (rfp = ResolverFlags; rfp->rf_name != NULL; rfp++) 2109c2aa98e2SPeter Wemm { 2110c2aa98e2SPeter Wemm if (strcasecmp(q, rfp->rf_name) == 0) 2111c2aa98e2SPeter Wemm break; 2112c2aa98e2SPeter Wemm } 2113c2aa98e2SPeter Wemm if (rfp->rf_name == NULL) 2114c2aa98e2SPeter Wemm syserr("readcf: I option value %s unrecognized", q); 2115c2aa98e2SPeter Wemm else if (clearmode) 2116c2aa98e2SPeter Wemm _res.options &= ~rfp->rf_bits; 2117c2aa98e2SPeter Wemm else 2118c2aa98e2SPeter Wemm _res.options |= rfp->rf_bits; 2119c2aa98e2SPeter Wemm } 2120c2aa98e2SPeter Wemm if (tTd(8, 2)) 212106f25ae9SGregory Neil Shapiro dprintf("_res.options = %x, HasWildcardMX = %d\n", 2122c2aa98e2SPeter Wemm (u_int) _res.options, HasWildcardMX); 212306f25ae9SGregory Neil Shapiro #else /* NAMED_BIND */ 2124c2aa98e2SPeter Wemm usrerr("name server (I option) specified but BIND not compiled in"); 212506f25ae9SGregory Neil Shapiro #endif /* NAMED_BIND */ 2126c2aa98e2SPeter Wemm break; 2127c2aa98e2SPeter Wemm 2128c2aa98e2SPeter Wemm case 'i': /* ignore dot lines in message */ 2129c2aa98e2SPeter Wemm IgnrDot = atobool(val); 2130c2aa98e2SPeter Wemm break; 2131c2aa98e2SPeter Wemm 2132c2aa98e2SPeter Wemm case 'j': /* send errors in MIME (RFC 1341) format */ 2133c2aa98e2SPeter Wemm SendMIMEErrors = atobool(val); 2134c2aa98e2SPeter Wemm break; 2135c2aa98e2SPeter Wemm 2136c2aa98e2SPeter Wemm case 'J': /* .forward search path */ 2137c2aa98e2SPeter Wemm ForwardPath = newstr(val); 2138c2aa98e2SPeter Wemm break; 2139c2aa98e2SPeter Wemm 2140c2aa98e2SPeter Wemm case 'k': /* connection cache size */ 2141c2aa98e2SPeter Wemm MaxMciCache = atoi(val); 2142c2aa98e2SPeter Wemm if (MaxMciCache < 0) 2143c2aa98e2SPeter Wemm MaxMciCache = 0; 2144c2aa98e2SPeter Wemm break; 2145c2aa98e2SPeter Wemm 2146c2aa98e2SPeter Wemm case 'K': /* connection cache timeout */ 2147c2aa98e2SPeter Wemm MciCacheTimeout = convtime(val, 'm'); 2148c2aa98e2SPeter Wemm break; 2149c2aa98e2SPeter Wemm 2150c2aa98e2SPeter Wemm case 'l': /* use Errors-To: header */ 2151c2aa98e2SPeter Wemm UseErrorsTo = atobool(val); 2152c2aa98e2SPeter Wemm break; 2153c2aa98e2SPeter Wemm 2154c2aa98e2SPeter Wemm case 'L': /* log level */ 2155c2aa98e2SPeter Wemm if (safe || LogLevel < atoi(val)) 2156c2aa98e2SPeter Wemm LogLevel = atoi(val); 2157c2aa98e2SPeter Wemm break; 2158c2aa98e2SPeter Wemm 2159c2aa98e2SPeter Wemm case 'M': /* define macro */ 2160193538b7SGregory Neil Shapiro sticky = FALSE; 2161c2aa98e2SPeter Wemm mid = macid(val, &ep); 2162193538b7SGregory Neil Shapiro if (mid == 0) 2163193538b7SGregory Neil Shapiro break; 2164c2aa98e2SPeter Wemm p = newstr(ep); 2165c2aa98e2SPeter Wemm if (!safe) 2166c2aa98e2SPeter Wemm cleanstrcpy(p, p, MAXNAME); 2167c2aa98e2SPeter Wemm define(mid, p, CurEnv); 2168c2aa98e2SPeter Wemm break; 2169c2aa98e2SPeter Wemm 2170c2aa98e2SPeter Wemm case 'm': /* send to me too */ 2171c2aa98e2SPeter Wemm MeToo = atobool(val); 2172c2aa98e2SPeter Wemm break; 2173c2aa98e2SPeter Wemm 2174c2aa98e2SPeter Wemm case 'n': /* validate RHS in newaliases */ 2175c2aa98e2SPeter Wemm CheckAliases = atobool(val); 2176c2aa98e2SPeter Wemm break; 2177c2aa98e2SPeter Wemm 2178c2aa98e2SPeter Wemm /* 'N' available -- was "net name" */ 2179c2aa98e2SPeter Wemm 2180c2aa98e2SPeter Wemm case 'O': /* daemon options */ 2181c2aa98e2SPeter Wemm #if DAEMON 218206f25ae9SGregory Neil Shapiro if (!setdaemonoptions(val)) 218306f25ae9SGregory Neil Shapiro syserr("too many daemons defined (%d max)", MAXDAEMONS); 218406f25ae9SGregory Neil Shapiro #else /* DAEMON */ 2185c2aa98e2SPeter Wemm syserr("DaemonPortOptions (O option) set but DAEMON not compiled in"); 218606f25ae9SGregory Neil Shapiro #endif /* DAEMON */ 2187c2aa98e2SPeter Wemm break; 2188c2aa98e2SPeter Wemm 2189c2aa98e2SPeter Wemm case 'o': /* assume old style headers */ 2190c2aa98e2SPeter Wemm if (atobool(val)) 2191c2aa98e2SPeter Wemm CurEnv->e_flags |= EF_OLDSTYLE; 2192c2aa98e2SPeter Wemm else 2193c2aa98e2SPeter Wemm CurEnv->e_flags &= ~EF_OLDSTYLE; 2194c2aa98e2SPeter Wemm break; 2195c2aa98e2SPeter Wemm 2196c2aa98e2SPeter Wemm case 'p': /* select privacy level */ 2197c2aa98e2SPeter Wemm p = val; 2198c2aa98e2SPeter Wemm for (;;) 2199c2aa98e2SPeter Wemm { 2200c2aa98e2SPeter Wemm register struct prival *pv; 2201c2aa98e2SPeter Wemm extern struct prival PrivacyValues[]; 2202c2aa98e2SPeter Wemm 2203c2aa98e2SPeter Wemm while (isascii(*p) && (isspace(*p) || ispunct(*p))) 2204c2aa98e2SPeter Wemm p++; 2205c2aa98e2SPeter Wemm if (*p == '\0') 2206c2aa98e2SPeter Wemm break; 2207c2aa98e2SPeter Wemm val = p; 2208c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 2209c2aa98e2SPeter Wemm p++; 2210c2aa98e2SPeter Wemm if (*p != '\0') 2211c2aa98e2SPeter Wemm *p++ = '\0'; 2212c2aa98e2SPeter Wemm 2213c2aa98e2SPeter Wemm for (pv = PrivacyValues; pv->pv_name != NULL; pv++) 2214c2aa98e2SPeter Wemm { 2215c2aa98e2SPeter Wemm if (strcasecmp(val, pv->pv_name) == 0) 2216c2aa98e2SPeter Wemm break; 2217c2aa98e2SPeter Wemm } 2218c2aa98e2SPeter Wemm if (pv->pv_name == NULL) 2219c2aa98e2SPeter Wemm syserr("readcf: Op line: %s unrecognized", val); 2220193538b7SGregory Neil Shapiro else 2221c2aa98e2SPeter Wemm PrivacyFlags |= pv->pv_flag; 2222c2aa98e2SPeter Wemm } 2223c2aa98e2SPeter Wemm sticky = FALSE; 2224c2aa98e2SPeter Wemm break; 2225c2aa98e2SPeter Wemm 2226c2aa98e2SPeter Wemm case 'P': /* postmaster copy address for returned mail */ 2227c2aa98e2SPeter Wemm PostMasterCopy = newstr(val); 2228c2aa98e2SPeter Wemm break; 2229c2aa98e2SPeter Wemm 2230c2aa98e2SPeter Wemm case 'q': /* slope of queue only function */ 2231c2aa98e2SPeter Wemm QueueFactor = atoi(val); 2232c2aa98e2SPeter Wemm break; 2233c2aa98e2SPeter Wemm 2234c2aa98e2SPeter Wemm case 'Q': /* queue directory */ 2235c2aa98e2SPeter Wemm if (val[0] == '\0') 223642e5d165SGregory Neil Shapiro { 2237c2aa98e2SPeter Wemm QueueDir = "mqueue"; 223842e5d165SGregory Neil Shapiro } 2239c2aa98e2SPeter Wemm else 224042e5d165SGregory Neil Shapiro { 2241c2aa98e2SPeter Wemm QueueDir = newstr(val); 224242e5d165SGregory Neil Shapiro } 2243c2aa98e2SPeter Wemm if (RealUid != 0 && !safe) 2244c2aa98e2SPeter Wemm Warn_Q_option = TRUE; 2245c2aa98e2SPeter Wemm break; 2246c2aa98e2SPeter Wemm 2247c2aa98e2SPeter Wemm case 'R': /* don't prune routes */ 2248c2aa98e2SPeter Wemm DontPruneRoutes = atobool(val); 2249c2aa98e2SPeter Wemm break; 2250c2aa98e2SPeter Wemm 2251c2aa98e2SPeter Wemm case 'r': /* read timeout */ 2252c2aa98e2SPeter Wemm if (subopt == NULL) 225306f25ae9SGregory Neil Shapiro inittimeouts(val, sticky); 2254c2aa98e2SPeter Wemm else 225506f25ae9SGregory Neil Shapiro settimeout(subopt, val, sticky); 2256c2aa98e2SPeter Wemm break; 2257c2aa98e2SPeter Wemm 2258c2aa98e2SPeter Wemm case 'S': /* status file */ 2259c2aa98e2SPeter Wemm if (val[0] == '\0') 226006f25ae9SGregory Neil Shapiro StatFile = "statistics"; 2261c2aa98e2SPeter Wemm else 2262602a2b1bSGregory Neil Shapiro { 2263c2aa98e2SPeter Wemm StatFile = newstr(val); 2264602a2b1bSGregory Neil Shapiro } 2265c2aa98e2SPeter Wemm break; 2266c2aa98e2SPeter Wemm 2267c2aa98e2SPeter Wemm case 's': /* be super safe, even if expensive */ 2268c2aa98e2SPeter Wemm SuperSafe = atobool(val); 2269c2aa98e2SPeter Wemm break; 2270c2aa98e2SPeter Wemm 2271c2aa98e2SPeter Wemm case 'T': /* queue timeout */ 2272c2aa98e2SPeter Wemm p = strchr(val, '/'); 2273c2aa98e2SPeter Wemm if (p != NULL) 2274c2aa98e2SPeter Wemm { 2275c2aa98e2SPeter Wemm *p++ = '\0'; 227606f25ae9SGregory Neil Shapiro settimeout("queuewarn", p, sticky); 2277c2aa98e2SPeter Wemm } 227806f25ae9SGregory Neil Shapiro settimeout("queuereturn", val, sticky); 2279c2aa98e2SPeter Wemm break; 2280c2aa98e2SPeter Wemm 2281c2aa98e2SPeter Wemm case 't': /* time zone name */ 2282c2aa98e2SPeter Wemm TimeZoneSpec = newstr(val); 2283c2aa98e2SPeter Wemm break; 2284c2aa98e2SPeter Wemm 2285c2aa98e2SPeter Wemm case 'U': /* location of user database */ 2286c2aa98e2SPeter Wemm UdbSpec = newstr(val); 2287c2aa98e2SPeter Wemm break; 2288c2aa98e2SPeter Wemm 2289c2aa98e2SPeter Wemm case 'u': /* set default uid */ 2290c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2291c2aa98e2SPeter Wemm { 2292c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2293c2aa98e2SPeter Wemm { 2294c2aa98e2SPeter Wemm *p++ = '\0'; 2295c2aa98e2SPeter Wemm break; 2296c2aa98e2SPeter Wemm } 2297c2aa98e2SPeter Wemm } 2298c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2299c2aa98e2SPeter Wemm { 2300c2aa98e2SPeter Wemm DefUid = atoi(val); 2301c2aa98e2SPeter Wemm setdefuser(); 2302c2aa98e2SPeter Wemm } 2303c2aa98e2SPeter Wemm else 2304c2aa98e2SPeter Wemm { 2305c2aa98e2SPeter Wemm register struct passwd *pw; 2306c2aa98e2SPeter Wemm 2307c2aa98e2SPeter Wemm DefUid = -1; 2308c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2309c2aa98e2SPeter Wemm if (pw == NULL) 2310193538b7SGregory Neil Shapiro { 2311c2aa98e2SPeter Wemm syserr("readcf: option u: unknown user %s", val); 2312193538b7SGregory Neil Shapiro break; 2313193538b7SGregory Neil Shapiro } 2314c2aa98e2SPeter Wemm else 2315c2aa98e2SPeter Wemm { 2316c2aa98e2SPeter Wemm DefUid = pw->pw_uid; 2317c2aa98e2SPeter Wemm DefGid = pw->pw_gid; 2318c2aa98e2SPeter Wemm DefUser = newstr(pw->pw_name); 2319c2aa98e2SPeter Wemm } 2320c2aa98e2SPeter Wemm } 2321c2aa98e2SPeter Wemm 2322c2aa98e2SPeter Wemm #ifdef UID_MAX 2323c2aa98e2SPeter Wemm if (DefUid > UID_MAX) 2324c2aa98e2SPeter Wemm { 2325c2aa98e2SPeter Wemm syserr("readcf: option u: uid value (%ld) > UID_MAX (%ld); ignored", 2326193538b7SGregory Neil Shapiro (long) DefUid, (long) UID_MAX); 2327193538b7SGregory Neil Shapiro break; 2328c2aa98e2SPeter Wemm } 232906f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2330c2aa98e2SPeter Wemm 2331c2aa98e2SPeter Wemm /* handle the group if it is there */ 2332c2aa98e2SPeter Wemm if (*p == '\0') 2333c2aa98e2SPeter Wemm break; 2334c2aa98e2SPeter Wemm val = p; 2335c2aa98e2SPeter Wemm goto g_opt; 2336c2aa98e2SPeter Wemm 2337c2aa98e2SPeter Wemm case 'V': /* fallback MX host */ 2338c2aa98e2SPeter Wemm if (val[0] != '\0') 2339c2aa98e2SPeter Wemm FallBackMX = newstr(val); 2340c2aa98e2SPeter Wemm break; 2341c2aa98e2SPeter Wemm 2342c2aa98e2SPeter Wemm case 'v': /* run in verbose mode */ 2343c2aa98e2SPeter Wemm Verbose = atobool(val) ? 1 : 0; 2344c2aa98e2SPeter Wemm break; 2345c2aa98e2SPeter Wemm 2346c2aa98e2SPeter Wemm case 'w': /* if we are best MX, try host directly */ 2347c2aa98e2SPeter Wemm TryNullMXList = atobool(val); 2348c2aa98e2SPeter Wemm break; 2349c2aa98e2SPeter Wemm 2350c2aa98e2SPeter Wemm /* 'W' available -- was wizard password */ 2351c2aa98e2SPeter Wemm 2352c2aa98e2SPeter Wemm case 'x': /* load avg at which to auto-queue msgs */ 2353c2aa98e2SPeter Wemm QueueLA = atoi(val); 2354c2aa98e2SPeter Wemm break; 2355c2aa98e2SPeter Wemm 2356c2aa98e2SPeter Wemm case 'X': /* load avg at which to auto-reject connections */ 2357c2aa98e2SPeter Wemm RefuseLA = atoi(val); 2358c2aa98e2SPeter Wemm break; 2359c2aa98e2SPeter Wemm 2360c2aa98e2SPeter Wemm case 'y': /* work recipient factor */ 2361c2aa98e2SPeter Wemm WkRecipFact = atoi(val); 2362c2aa98e2SPeter Wemm break; 2363c2aa98e2SPeter Wemm 2364c2aa98e2SPeter Wemm case 'Y': /* fork jobs during queue runs */ 2365c2aa98e2SPeter Wemm ForkQueueRuns = atobool(val); 2366c2aa98e2SPeter Wemm break; 2367c2aa98e2SPeter Wemm 2368c2aa98e2SPeter Wemm case 'z': /* work message class factor */ 2369c2aa98e2SPeter Wemm WkClassFact = atoi(val); 2370c2aa98e2SPeter Wemm break; 2371c2aa98e2SPeter Wemm 2372c2aa98e2SPeter Wemm case 'Z': /* work time factor */ 2373c2aa98e2SPeter Wemm WkTimeFact = atoi(val); 2374c2aa98e2SPeter Wemm break; 2375c2aa98e2SPeter Wemm 237606f25ae9SGregory Neil Shapiro 2377c2aa98e2SPeter Wemm case O_QUEUESORTORD: /* queue sorting order */ 2378c2aa98e2SPeter Wemm switch (*val) 2379c2aa98e2SPeter Wemm { 2380c2aa98e2SPeter Wemm case 'h': /* Host first */ 2381c2aa98e2SPeter Wemm case 'H': 238206f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYHOST; 2383c2aa98e2SPeter Wemm break; 2384c2aa98e2SPeter Wemm 2385c2aa98e2SPeter Wemm case 'p': /* Priority order */ 2386c2aa98e2SPeter Wemm case 'P': 238706f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYPRIORITY; 2388c2aa98e2SPeter Wemm break; 2389c2aa98e2SPeter Wemm 2390c2aa98e2SPeter Wemm case 't': /* Submission time */ 2391c2aa98e2SPeter Wemm case 'T': 239206f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYTIME; 239306f25ae9SGregory Neil Shapiro break; 239406f25ae9SGregory Neil Shapiro 239506f25ae9SGregory Neil Shapiro case 'f': /* File Name */ 239606f25ae9SGregory Neil Shapiro case 'F': 239706f25ae9SGregory Neil Shapiro QueueSortOrder = QSO_BYFILENAME; 2398c2aa98e2SPeter Wemm break; 2399c2aa98e2SPeter Wemm 2400c2aa98e2SPeter Wemm default: 2401c2aa98e2SPeter Wemm syserr("Invalid queue sort order \"%s\"", val); 2402c2aa98e2SPeter Wemm } 2403c2aa98e2SPeter Wemm break; 2404c2aa98e2SPeter Wemm 240506f25ae9SGregory Neil Shapiro #if _FFR_QUEUEDELAY 240606f25ae9SGregory Neil Shapiro case O_QUEUEDELAY: /* queue delay algorithm */ 240706f25ae9SGregory Neil Shapiro switch (*val) 240806f25ae9SGregory Neil Shapiro { 240906f25ae9SGregory Neil Shapiro case 'e': /* exponential */ 241006f25ae9SGregory Neil Shapiro case 'E': 241106f25ae9SGregory Neil Shapiro QueueAlg = QD_EXP; 241206f25ae9SGregory Neil Shapiro QueueInitDelay = 10 MINUTES; 241306f25ae9SGregory Neil Shapiro QueueMaxDelay = 2 HOURS; 241406f25ae9SGregory Neil Shapiro p = strchr(val, '/'); 241506f25ae9SGregory Neil Shapiro if (p != NULL) 241606f25ae9SGregory Neil Shapiro { 241706f25ae9SGregory Neil Shapiro char *q; 241806f25ae9SGregory Neil Shapiro 241906f25ae9SGregory Neil Shapiro *p++ = '\0'; 242006f25ae9SGregory Neil Shapiro q = strchr(p, '/'); 242106f25ae9SGregory Neil Shapiro if (q != NULL) 242206f25ae9SGregory Neil Shapiro *q++ = '\0'; 242306f25ae9SGregory Neil Shapiro QueueInitDelay = convtime(p, 's'); 242406f25ae9SGregory Neil Shapiro if (q != NULL) 242506f25ae9SGregory Neil Shapiro { 242606f25ae9SGregory Neil Shapiro QueueMaxDelay = convtime(q, 's'); 242706f25ae9SGregory Neil Shapiro } 242806f25ae9SGregory Neil Shapiro } 242906f25ae9SGregory Neil Shapiro break; 243006f25ae9SGregory Neil Shapiro 243106f25ae9SGregory Neil Shapiro case 'l': /* linear */ 243206f25ae9SGregory Neil Shapiro case 'L': 243306f25ae9SGregory Neil Shapiro QueueAlg = QD_LINEAR; 243406f25ae9SGregory Neil Shapiro break; 243506f25ae9SGregory Neil Shapiro 243606f25ae9SGregory Neil Shapiro default: 243706f25ae9SGregory Neil Shapiro syserr("Invalid queue delay algorithm \"%s\"", val); 243806f25ae9SGregory Neil Shapiro } 243906f25ae9SGregory Neil Shapiro break; 244006f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUEDELAY */ 244106f25ae9SGregory Neil Shapiro 2442c2aa98e2SPeter Wemm case O_HOSTSFILE: /* pathname of /etc/hosts file */ 2443c2aa98e2SPeter Wemm HostsFile = newstr(val); 2444c2aa98e2SPeter Wemm break; 2445c2aa98e2SPeter Wemm 2446c2aa98e2SPeter Wemm case O_MQA: /* minimum queue age between deliveries */ 2447c2aa98e2SPeter Wemm MinQueueAge = convtime(val, 'm'); 2448c2aa98e2SPeter Wemm break; 2449c2aa98e2SPeter Wemm 2450c2aa98e2SPeter Wemm case O_DEFCHARSET: /* default character set for mimefying */ 2451c2aa98e2SPeter Wemm DefaultCharSet = newstr(denlstring(val, TRUE, TRUE)); 2452c2aa98e2SPeter Wemm break; 2453c2aa98e2SPeter Wemm 2454c2aa98e2SPeter Wemm case O_SSFILE: /* service switch file */ 2455c2aa98e2SPeter Wemm ServiceSwitchFile = newstr(val); 2456c2aa98e2SPeter Wemm break; 2457c2aa98e2SPeter Wemm 2458c2aa98e2SPeter Wemm case O_DIALDELAY: /* delay for dial-on-demand operation */ 2459c2aa98e2SPeter Wemm DialDelay = convtime(val, 's'); 2460c2aa98e2SPeter Wemm break; 2461c2aa98e2SPeter Wemm 2462c2aa98e2SPeter Wemm case O_NORCPTACTION: /* what to do if no recipient */ 2463c2aa98e2SPeter Wemm if (strcasecmp(val, "none") == 0) 2464c2aa98e2SPeter Wemm NoRecipientAction = NRA_NO_ACTION; 2465c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to") == 0) 2466c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO; 2467c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-apparently-to") == 0) 2468c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_APPARENTLY_TO; 2469c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-bcc") == 0) 2470c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_BCC; 2471c2aa98e2SPeter Wemm else if (strcasecmp(val, "add-to-undisclosed") == 0) 2472c2aa98e2SPeter Wemm NoRecipientAction = NRA_ADD_TO_UNDISCLOSED; 2473c2aa98e2SPeter Wemm else 2474c2aa98e2SPeter Wemm syserr("Invalid NoRecipientAction: %s", val); 2475c2aa98e2SPeter Wemm break; 2476c2aa98e2SPeter Wemm 2477c2aa98e2SPeter Wemm case O_SAFEFILEENV: /* chroot() environ for writing to files */ 2478c2aa98e2SPeter Wemm SafeFileEnv = newstr(val); 2479c2aa98e2SPeter Wemm break; 2480c2aa98e2SPeter Wemm 2481c2aa98e2SPeter Wemm case O_MAXMSGSIZE: /* maximum message size */ 2482c2aa98e2SPeter Wemm MaxMessageSize = atol(val); 2483c2aa98e2SPeter Wemm break; 2484c2aa98e2SPeter Wemm 2485c2aa98e2SPeter Wemm case O_COLONOKINADDR: /* old style handling of colon addresses */ 2486c2aa98e2SPeter Wemm ColonOkInAddr = atobool(val); 2487c2aa98e2SPeter Wemm break; 2488c2aa98e2SPeter Wemm 2489c2aa98e2SPeter Wemm case O_MAXQUEUERUN: /* max # of jobs in a single queue run */ 2490c2aa98e2SPeter Wemm MaxQueueRun = atol(val); 2491c2aa98e2SPeter Wemm break; 2492c2aa98e2SPeter Wemm 2493c2aa98e2SPeter Wemm case O_MAXCHILDREN: /* max # of children of daemon */ 2494c2aa98e2SPeter Wemm MaxChildren = atoi(val); 2495c2aa98e2SPeter Wemm break; 2496c2aa98e2SPeter Wemm 249706f25ae9SGregory Neil Shapiro #if _FFR_MAX_FORWARD_ENTRIES 249806f25ae9SGregory Neil Shapiro case O_MAXFORWARD: /* max # of forward entries */ 249906f25ae9SGregory Neil Shapiro MaxForwardEntries = atoi(val); 250006f25ae9SGregory Neil Shapiro break; 250106f25ae9SGregory Neil Shapiro #endif /* _FFR_MAX_FORWARD_ENTRIES */ 250206f25ae9SGregory Neil Shapiro 2503c2aa98e2SPeter Wemm case O_KEEPCNAMES: /* don't expand CNAME records */ 2504c2aa98e2SPeter Wemm DontExpandCnames = atobool(val); 2505c2aa98e2SPeter Wemm break; 2506c2aa98e2SPeter Wemm 2507c2aa98e2SPeter Wemm case O_MUSTQUOTE: /* must quote these characters in phrases */ 250806f25ae9SGregory Neil Shapiro (void) strlcpy(buf, "@,;:\\()[]", sizeof buf); 2509c2aa98e2SPeter Wemm if (strlen(val) < (SIZE_T) sizeof buf - 10) 251006f25ae9SGregory Neil Shapiro (void) strlcat(buf, val, sizeof buf); 251106f25ae9SGregory Neil Shapiro else 251206f25ae9SGregory Neil Shapiro printf("Warning: MustQuoteChars too long, ignored.\n"); 2513c2aa98e2SPeter Wemm MustQuoteChars = newstr(buf); 2514c2aa98e2SPeter Wemm break; 2515c2aa98e2SPeter Wemm 2516c2aa98e2SPeter Wemm case O_SMTPGREETING: /* SMTP greeting message (old $e macro) */ 2517c2aa98e2SPeter Wemm SmtpGreeting = newstr(munchstring(val, NULL, '\0')); 2518c2aa98e2SPeter Wemm break; 2519c2aa98e2SPeter Wemm 2520c2aa98e2SPeter Wemm case O_UNIXFROM: /* UNIX From_ line (old $l macro) */ 2521c2aa98e2SPeter Wemm UnixFromLine = newstr(munchstring(val, NULL, '\0')); 2522c2aa98e2SPeter Wemm break; 2523c2aa98e2SPeter Wemm 2524c2aa98e2SPeter Wemm case O_OPCHARS: /* operator characters (old $o macro) */ 252506f25ae9SGregory Neil Shapiro if (OperatorChars != NULL) 252606f25ae9SGregory Neil Shapiro printf("Warning: OperatorChars is being redefined.\n It should only be set before ruleset definitions.\n"); 2527c2aa98e2SPeter Wemm OperatorChars = newstr(munchstring(val, NULL, '\0')); 2528c2aa98e2SPeter Wemm break; 2529c2aa98e2SPeter Wemm 2530c2aa98e2SPeter Wemm case O_DONTINITGRPS: /* don't call initgroups(3) */ 2531c2aa98e2SPeter Wemm DontInitGroups = atobool(val); 2532c2aa98e2SPeter Wemm break; 2533c2aa98e2SPeter Wemm 2534c2aa98e2SPeter Wemm case O_SLFH: /* make sure from fits on one line */ 2535c2aa98e2SPeter Wemm SingleLineFromHeader = atobool(val); 2536c2aa98e2SPeter Wemm break; 2537c2aa98e2SPeter Wemm 2538c2aa98e2SPeter Wemm case O_ABH: /* allow HELO commands with syntax errors */ 2539c2aa98e2SPeter Wemm AllowBogusHELO = atobool(val); 2540c2aa98e2SPeter Wemm break; 2541c2aa98e2SPeter Wemm 2542c2aa98e2SPeter Wemm case O_CONNTHROT: /* connection rate throttle */ 2543c2aa98e2SPeter Wemm ConnRateThrottle = atoi(val); 2544c2aa98e2SPeter Wemm break; 2545c2aa98e2SPeter Wemm 2546c2aa98e2SPeter Wemm case O_UGW: /* group writable files are unsafe */ 2547c2aa98e2SPeter Wemm if (!atobool(val)) 254806f25ae9SGregory Neil Shapiro { 254906f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEFORWARDFILESAFE, 255006f25ae9SGregory Neil Shapiro DontBlameSendmail); 255106f25ae9SGregory Neil Shapiro setbitn(DBS_GROUPWRITABLEINCLUDEFILESAFE, 255206f25ae9SGregory Neil Shapiro DontBlameSendmail); 255306f25ae9SGregory Neil Shapiro } 2554c2aa98e2SPeter Wemm break; 2555c2aa98e2SPeter Wemm 2556c2aa98e2SPeter Wemm case O_DBLBOUNCE: /* address to which to send double bounces */ 2557c2aa98e2SPeter Wemm if (val[0] != '\0') 2558c2aa98e2SPeter Wemm DoubleBounceAddr = newstr(val); 2559c2aa98e2SPeter Wemm else 2560c2aa98e2SPeter Wemm syserr("readcf: option DoubleBounceAddress: value required"); 2561c2aa98e2SPeter Wemm break; 2562c2aa98e2SPeter Wemm 2563c2aa98e2SPeter Wemm case O_HSDIR: /* persistent host status directory */ 2564c2aa98e2SPeter Wemm if (val[0] != '\0') 2565602a2b1bSGregory Neil Shapiro { 2566c2aa98e2SPeter Wemm HostStatDir = newstr(val); 2567602a2b1bSGregory Neil Shapiro } 2568c2aa98e2SPeter Wemm break; 2569c2aa98e2SPeter Wemm 2570c2aa98e2SPeter Wemm case O_SINGTHREAD: /* single thread deliveries (requires hsdir) */ 2571c2aa98e2SPeter Wemm SingleThreadDelivery = atobool(val); 2572c2aa98e2SPeter Wemm break; 2573c2aa98e2SPeter Wemm 2574c2aa98e2SPeter Wemm case O_RUNASUSER: /* run bulk of code as this user */ 2575c2aa98e2SPeter Wemm for (p = val; *p != '\0'; p++) 2576c2aa98e2SPeter Wemm { 2577c2aa98e2SPeter Wemm if (*p == '.' || *p == '/' || *p == ':') 2578c2aa98e2SPeter Wemm { 2579c2aa98e2SPeter Wemm *p++ = '\0'; 2580c2aa98e2SPeter Wemm break; 2581c2aa98e2SPeter Wemm } 2582c2aa98e2SPeter Wemm } 2583c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2584c2aa98e2SPeter Wemm { 2585c2aa98e2SPeter Wemm if (can_setuid) 2586c2aa98e2SPeter Wemm RunAsUid = atoi(val); 2587c2aa98e2SPeter Wemm } 2588c2aa98e2SPeter Wemm else 2589c2aa98e2SPeter Wemm { 2590c2aa98e2SPeter Wemm register struct passwd *pw; 2591c2aa98e2SPeter Wemm 2592c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2593c2aa98e2SPeter Wemm if (pw == NULL) 2594193538b7SGregory Neil Shapiro { 2595c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown user %s", val); 2596193538b7SGregory Neil Shapiro break; 2597193538b7SGregory Neil Shapiro } 2598c2aa98e2SPeter Wemm else if (can_setuid) 2599c2aa98e2SPeter Wemm { 2600c2aa98e2SPeter Wemm if (*p == '\0') 2601c2aa98e2SPeter Wemm RunAsUserName = newstr(val); 2602c2aa98e2SPeter Wemm RunAsUid = pw->pw_uid; 2603c2aa98e2SPeter Wemm RunAsGid = pw->pw_gid; 2604c2aa98e2SPeter Wemm } 2605c2aa98e2SPeter Wemm } 2606c2aa98e2SPeter Wemm #ifdef UID_MAX 2607c2aa98e2SPeter Wemm if (RunAsUid > UID_MAX) 2608c2aa98e2SPeter Wemm { 2609c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: uid value (%ld) > UID_MAX (%ld); ignored", 2610193538b7SGregory Neil Shapiro (long) RunAsUid, (long) UID_MAX); 2611193538b7SGregory Neil Shapiro break; 2612c2aa98e2SPeter Wemm } 261306f25ae9SGregory Neil Shapiro #endif /* UID_MAX */ 2614c2aa98e2SPeter Wemm if (*p != '\0') 2615c2aa98e2SPeter Wemm { 2616c2aa98e2SPeter Wemm if (isascii(*p) && isdigit(*p)) 2617c2aa98e2SPeter Wemm { 2618c2aa98e2SPeter Wemm if (can_setuid) 2619c2aa98e2SPeter Wemm RunAsGid = atoi(p); 2620c2aa98e2SPeter Wemm } 2621c2aa98e2SPeter Wemm else 2622c2aa98e2SPeter Wemm { 2623c2aa98e2SPeter Wemm register struct group *gr; 2624c2aa98e2SPeter Wemm 2625c2aa98e2SPeter Wemm gr = getgrnam(p); 2626c2aa98e2SPeter Wemm if (gr == NULL) 2627c2aa98e2SPeter Wemm syserr("readcf: option RunAsUser: unknown group %s", 2628c2aa98e2SPeter Wemm p); 2629c2aa98e2SPeter Wemm else if (can_setuid) 2630c2aa98e2SPeter Wemm RunAsGid = gr->gr_gid; 2631c2aa98e2SPeter Wemm } 2632c2aa98e2SPeter Wemm } 2633c2aa98e2SPeter Wemm if (tTd(47, 5)) 263406f25ae9SGregory Neil Shapiro dprintf("readcf: RunAsUser = %d:%d\n", 263506f25ae9SGregory Neil Shapiro (int)RunAsUid, (int)RunAsGid); 2636c2aa98e2SPeter Wemm break; 2637c2aa98e2SPeter Wemm 2638c2aa98e2SPeter Wemm case O_DSN_RRT: 2639c2aa98e2SPeter Wemm RrtImpliesDsn = atobool(val); 2640c2aa98e2SPeter Wemm break; 2641c2aa98e2SPeter Wemm 2642c2aa98e2SPeter Wemm case O_PIDFILE: 264306f25ae9SGregory Neil Shapiro if (PidFile != NULL) 2644c2aa98e2SPeter Wemm free(PidFile); 2645c2aa98e2SPeter Wemm PidFile = newstr(val); 2646c2aa98e2SPeter Wemm break; 2647c2aa98e2SPeter Wemm 2648c2aa98e2SPeter Wemm case O_DONTBLAMESENDMAIL: 2649c2aa98e2SPeter Wemm p = val; 2650c2aa98e2SPeter Wemm for (;;) 2651c2aa98e2SPeter Wemm { 2652c2aa98e2SPeter Wemm register struct dbsval *dbs; 2653c2aa98e2SPeter Wemm extern struct dbsval DontBlameSendmailValues[]; 2654c2aa98e2SPeter Wemm 2655c2aa98e2SPeter Wemm while (isascii(*p) && (isspace(*p) || ispunct(*p))) 2656c2aa98e2SPeter Wemm p++; 2657c2aa98e2SPeter Wemm if (*p == '\0') 2658c2aa98e2SPeter Wemm break; 2659c2aa98e2SPeter Wemm val = p; 2660c2aa98e2SPeter Wemm while (isascii(*p) && isalnum(*p)) 2661c2aa98e2SPeter Wemm p++; 2662c2aa98e2SPeter Wemm if (*p != '\0') 2663c2aa98e2SPeter Wemm *p++ = '\0'; 2664c2aa98e2SPeter Wemm 2665c2aa98e2SPeter Wemm for (dbs = DontBlameSendmailValues; 2666c2aa98e2SPeter Wemm dbs->dbs_name != NULL; dbs++) 2667c2aa98e2SPeter Wemm { 2668c2aa98e2SPeter Wemm if (strcasecmp(val, dbs->dbs_name) == 0) 2669c2aa98e2SPeter Wemm break; 2670c2aa98e2SPeter Wemm } 2671c2aa98e2SPeter Wemm if (dbs->dbs_name == NULL) 2672c2aa98e2SPeter Wemm syserr("readcf: DontBlameSendmail option: %s unrecognized", val); 2673c2aa98e2SPeter Wemm else if (dbs->dbs_flag == DBS_SAFE) 267406f25ae9SGregory Neil Shapiro clrbitmap(DontBlameSendmail); 2675c2aa98e2SPeter Wemm else 267606f25ae9SGregory Neil Shapiro setbitn(dbs->dbs_flag, DontBlameSendmail); 2677c2aa98e2SPeter Wemm } 2678c2aa98e2SPeter Wemm sticky = FALSE; 2679c2aa98e2SPeter Wemm break; 2680c2aa98e2SPeter Wemm 2681c2aa98e2SPeter Wemm case O_DPI: 2682c2aa98e2SPeter Wemm DontProbeInterfaces = atobool(val); 2683c2aa98e2SPeter Wemm break; 2684c2aa98e2SPeter Wemm 2685c2aa98e2SPeter Wemm case O_MAXRCPT: 2686c2aa98e2SPeter Wemm MaxRcptPerMsg = atoi(val); 2687c2aa98e2SPeter Wemm break; 2688c2aa98e2SPeter Wemm 2689c2aa98e2SPeter Wemm case O_DEADLETTER: 2690c2aa98e2SPeter Wemm if (DeadLetterDrop != NULL) 2691c2aa98e2SPeter Wemm free(DeadLetterDrop); 2692c2aa98e2SPeter Wemm DeadLetterDrop = newstr(val); 2693c2aa98e2SPeter Wemm break; 2694c2aa98e2SPeter Wemm 2695c2aa98e2SPeter Wemm #if _FFR_DONTLOCKFILESFORREAD_OPTION 2696c2aa98e2SPeter Wemm case O_DONTLOCK: 2697c2aa98e2SPeter Wemm DontLockReadFiles = atobool(val); 2698c2aa98e2SPeter Wemm break; 269906f25ae9SGregory Neil Shapiro #endif /* _FFR_DONTLOCKFILESFORREAD_OPTION */ 2700c2aa98e2SPeter Wemm 2701c2aa98e2SPeter Wemm case O_MAXALIASRCSN: 2702c2aa98e2SPeter Wemm MaxAliasRecursion = atoi(val); 2703c2aa98e2SPeter Wemm break; 2704c2aa98e2SPeter Wemm 2705c2aa98e2SPeter Wemm case O_CNCTONLYTO: 2706c2aa98e2SPeter Wemm /* XXX should probably use gethostbyname */ 270706f25ae9SGregory Neil Shapiro #if NETINET || NETINET6 270806f25ae9SGregory Neil Shapiro # if NETINET6 270906f25ae9SGregory Neil Shapiro if (inet_addr(val) == INADDR_NONE) 271006f25ae9SGregory Neil Shapiro { 271106f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET6; 271206f25ae9SGregory Neil Shapiro if (inet_pton(AF_INET6, val, 271306f25ae9SGregory Neil Shapiro &ConnectOnlyTo.sin6.sin6_addr) != 1) 271406f25ae9SGregory Neil Shapiro syserr("readcf: option ConnectOnlyTo: invalid IP address %s", 271506f25ae9SGregory Neil Shapiro val); 271606f25ae9SGregory Neil Shapiro } 271706f25ae9SGregory Neil Shapiro else 271806f25ae9SGregory Neil Shapiro # endif /* NETINET6 */ 271906f25ae9SGregory Neil Shapiro { 272006f25ae9SGregory Neil Shapiro ConnectOnlyTo.sa.sa_family = AF_INET; 272106f25ae9SGregory Neil Shapiro ConnectOnlyTo.sin.sin_addr.s_addr = inet_addr(val); 272206f25ae9SGregory Neil Shapiro } 272306f25ae9SGregory Neil Shapiro #endif /* NETINET || NETINET6 */ 2724c2aa98e2SPeter Wemm break; 2725c2aa98e2SPeter Wemm 2726065a643dSPeter Wemm case O_TRUSTUSER: 272706f25ae9SGregory Neil Shapiro #if HASFCHOWN 2728c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 2729065a643dSPeter Wemm TrustedUid = atoi(val); 2730c2aa98e2SPeter Wemm else 2731c2aa98e2SPeter Wemm { 2732c2aa98e2SPeter Wemm register struct passwd *pw; 2733c2aa98e2SPeter Wemm 2734065a643dSPeter Wemm TrustedUid = 0; 2735c2aa98e2SPeter Wemm pw = sm_getpwnam(val); 2736c2aa98e2SPeter Wemm if (pw == NULL) 2737193538b7SGregory Neil Shapiro { 2738065a643dSPeter Wemm syserr("readcf: option TrustedUser: unknown user %s", val); 2739193538b7SGregory Neil Shapiro break; 2740193538b7SGregory Neil Shapiro } 2741c2aa98e2SPeter Wemm else 2742065a643dSPeter Wemm TrustedUid = pw->pw_uid; 2743c2aa98e2SPeter Wemm } 2744c2aa98e2SPeter Wemm 2745c2aa98e2SPeter Wemm # ifdef UID_MAX 2746065a643dSPeter Wemm if (TrustedUid > UID_MAX) 2747c2aa98e2SPeter Wemm { 2748065a643dSPeter Wemm syserr("readcf: option TrustedUser: uid value (%ld) > UID_MAX (%ld)", 2749193538b7SGregory Neil Shapiro (long) TrustedUid, (long) UID_MAX); 2750065a643dSPeter Wemm TrustedUid = 0; 2751c2aa98e2SPeter Wemm } 275206f25ae9SGregory Neil Shapiro # endif /* UID_MAX */ 275306f25ae9SGregory Neil Shapiro #else /* HASFCHOWN */ 275406f25ae9SGregory Neil Shapiro syserr("readcf: option TrustedUser: can not be used on systems which do not support fchown()"); 275506f25ae9SGregory Neil Shapiro #endif /* HASFCHOWN */ 2756c2aa98e2SPeter Wemm break; 2757c2aa98e2SPeter Wemm 2758065a643dSPeter Wemm case O_MAXMIMEHDRLEN: 2759065a643dSPeter Wemm p = strchr(val, '/'); 2760065a643dSPeter Wemm if (p != NULL) 2761065a643dSPeter Wemm *p++ = '\0'; 2762065a643dSPeter Wemm MaxMimeHeaderLength = atoi(val); 2763065a643dSPeter Wemm if (p != NULL && *p != '\0') 2764065a643dSPeter Wemm MaxMimeFieldLength = atoi(p); 2765065a643dSPeter Wemm else 2766065a643dSPeter Wemm MaxMimeFieldLength = MaxMimeHeaderLength / 2; 2767065a643dSPeter Wemm 2768065a643dSPeter Wemm if (MaxMimeHeaderLength < 0) 2769065a643dSPeter Wemm MaxMimeHeaderLength = 0; 2770065a643dSPeter Wemm else if (MaxMimeHeaderLength < 128) 2771065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: header length limit set lower than 128\n"); 2772065a643dSPeter Wemm 2773065a643dSPeter Wemm if (MaxMimeFieldLength < 0) 2774065a643dSPeter Wemm MaxMimeFieldLength = 0; 2775065a643dSPeter Wemm else if (MaxMimeFieldLength < 40) 2776065a643dSPeter Wemm printf("Warning: MaxMimeHeaderLength: field length limit set lower than 40\n"); 2777065a643dSPeter Wemm break; 2778065a643dSPeter Wemm 2779065a643dSPeter Wemm case O_CONTROLSOCKET: 2780065a643dSPeter Wemm if (ControlSocketName != NULL) 2781065a643dSPeter Wemm free(ControlSocketName); 2782065a643dSPeter Wemm ControlSocketName = newstr(val); 2783065a643dSPeter Wemm break; 2784065a643dSPeter Wemm 27852e43090eSPeter Wemm case O_MAXHDRSLEN: 27862e43090eSPeter Wemm MaxHeadersLength = atoi(val); 278725bab6e9SPeter Wemm 27882e43090eSPeter Wemm if (MaxHeadersLength > 0 && 27892e43090eSPeter Wemm MaxHeadersLength < (MAXHDRSLEN / 2)) 279006f25ae9SGregory Neil Shapiro printf("Warning: MaxHeadersLength: headers length limit set lower than %d\n", (MAXHDRSLEN / 2)); 279125bab6e9SPeter Wemm break; 279206f25ae9SGregory Neil Shapiro 279306f25ae9SGregory Neil Shapiro case O_PROCTITLEPREFIX: 279406f25ae9SGregory Neil Shapiro if (ProcTitlePrefix != NULL) 279506f25ae9SGregory Neil Shapiro free(ProcTitlePrefix); 279606f25ae9SGregory Neil Shapiro ProcTitlePrefix = newstr(val); 279706f25ae9SGregory Neil Shapiro break; 279806f25ae9SGregory Neil Shapiro 279906f25ae9SGregory Neil Shapiro #if SASL 280006f25ae9SGregory Neil Shapiro case O_SASLINFO: 280106f25ae9SGregory Neil Shapiro #if _FFR_ALLOW_SASLINFO 280206f25ae9SGregory Neil Shapiro /* 280306f25ae9SGregory Neil Shapiro ** Allow users to select their own authinfo file. 280406f25ae9SGregory Neil Shapiro ** However, this is not a "perfect" solution. 280506f25ae9SGregory Neil Shapiro ** If mail is queued, the authentication info 280606f25ae9SGregory Neil Shapiro ** will not be used in subsequent delivery attempts. 280706f25ae9SGregory Neil Shapiro ** If we really want to support this, then it has 280806f25ae9SGregory Neil Shapiro ** to be stored in the queue file. 280906f25ae9SGregory Neil Shapiro */ 281006f25ae9SGregory Neil Shapiro if (!bitset(SUBMIT_MSA, SubmitMode) && RealUid != 0 && 281106f25ae9SGregory Neil Shapiro RunAsUid != RealUid) 281206f25ae9SGregory Neil Shapiro { 281306f25ae9SGregory Neil Shapiro errno = 0; 281406f25ae9SGregory Neil Shapiro syserr("Error: %s only allowed with -U\n", 281506f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 281606f25ae9SGregory Neil Shapiro ExitStat = EX_USAGE; 281706f25ae9SGregory Neil Shapiro break; 281806f25ae9SGregory Neil Shapiro } 281906f25ae9SGregory Neil Shapiro #endif /* _FFR_ALLOW_SASLINFO */ 282006f25ae9SGregory Neil Shapiro if (SASLInfo != NULL) 282106f25ae9SGregory Neil Shapiro free(SASLInfo); 282206f25ae9SGregory Neil Shapiro SASLInfo = newstr(val); 282306f25ae9SGregory Neil Shapiro break; 282406f25ae9SGregory Neil Shapiro 282506f25ae9SGregory Neil Shapiro case O_SASLMECH: 282606f25ae9SGregory Neil Shapiro if (AuthMechanisms != NULL) 282706f25ae9SGregory Neil Shapiro free(AuthMechanisms); 282806f25ae9SGregory Neil Shapiro if (*val != '\0') 282906f25ae9SGregory Neil Shapiro AuthMechanisms = newstr(val); 283006f25ae9SGregory Neil Shapiro else 283106f25ae9SGregory Neil Shapiro AuthMechanisms = NULL; 283206f25ae9SGregory Neil Shapiro break; 283306f25ae9SGregory Neil Shapiro 283406f25ae9SGregory Neil Shapiro case O_SASLOPTS: 283506f25ae9SGregory Neil Shapiro while (val != NULL && *val != '\0') 283606f25ae9SGregory Neil Shapiro { 283706f25ae9SGregory Neil Shapiro switch(*val) 283806f25ae9SGregory Neil Shapiro { 283906f25ae9SGregory Neil Shapiro case 'A': 284006f25ae9SGregory Neil Shapiro SASLOpts |= SASL_AUTH_AUTH; 284106f25ae9SGregory Neil Shapiro break; 284206f25ae9SGregory Neil Shapiro # if _FFR_SASL_OPTS 284306f25ae9SGregory Neil Shapiro case 'a': 284406f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOACTIVE; 284506f25ae9SGregory Neil Shapiro break; 284606f25ae9SGregory Neil Shapiro case 'c': 284706f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_PASS_CREDENTIALS; 284806f25ae9SGregory Neil Shapiro break; 284906f25ae9SGregory Neil Shapiro case 'd': 285006f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NODICTIONARY; 285106f25ae9SGregory Neil Shapiro break; 285206f25ae9SGregory Neil Shapiro case 'f': 285306f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_FORWARD_SECRECY; 285406f25ae9SGregory Neil Shapiro break; 285506f25ae9SGregory Neil Shapiro case 'p': 285606f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOPLAINTEXT; 285706f25ae9SGregory Neil Shapiro break; 285806f25ae9SGregory Neil Shapiro case 'y': 285906f25ae9SGregory Neil Shapiro SASLOpts |= SASL_SEC_NOANONYMOUS; 286006f25ae9SGregory Neil Shapiro break; 286106f25ae9SGregory Neil Shapiro # endif /* _FFR_SASL_OPTS */ 286206f25ae9SGregory Neil Shapiro default: 286306f25ae9SGregory Neil Shapiro printf("Warning: Option: %s unknown parameter '%c'\n", 286406f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" 286506f25ae9SGregory Neil Shapiro : o->o_name, 286606f25ae9SGregory Neil Shapiro (isascii(*val) && isprint(*val)) ? *val 286706f25ae9SGregory Neil Shapiro : '?'); 286806f25ae9SGregory Neil Shapiro break; 286906f25ae9SGregory Neil Shapiro } 287006f25ae9SGregory Neil Shapiro ++val; 287106f25ae9SGregory Neil Shapiro val = strpbrk(val, ", \t"); 287206f25ae9SGregory Neil Shapiro if (val != NULL) 287306f25ae9SGregory Neil Shapiro ++val; 287406f25ae9SGregory Neil Shapiro } 287506f25ae9SGregory Neil Shapiro break; 287606f25ae9SGregory Neil Shapiro 287706f25ae9SGregory Neil Shapiro #else /* SASL */ 287806f25ae9SGregory Neil Shapiro case O_SASLINFO: 287906f25ae9SGregory Neil Shapiro case O_SASLMECH: 288006f25ae9SGregory Neil Shapiro case O_SASLOPTS: 288106f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires SASL support (-DSASL)\n", 288206f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 288306f25ae9SGregory Neil Shapiro break; 288406f25ae9SGregory Neil Shapiro #endif /* SASL */ 288506f25ae9SGregory Neil Shapiro 288606f25ae9SGregory Neil Shapiro #if STARTTLS 288706f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 288806f25ae9SGregory Neil Shapiro if (SrvCERTfile != NULL) 288906f25ae9SGregory Neil Shapiro free(SrvCERTfile); 289006f25ae9SGregory Neil Shapiro SrvCERTfile = newstr(val); 289106f25ae9SGregory Neil Shapiro break; 289206f25ae9SGregory Neil Shapiro 289306f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 289406f25ae9SGregory Neil Shapiro if (Srvkeyfile != NULL) 289506f25ae9SGregory Neil Shapiro free(Srvkeyfile); 289606f25ae9SGregory Neil Shapiro Srvkeyfile = newstr(val); 289706f25ae9SGregory Neil Shapiro break; 289806f25ae9SGregory Neil Shapiro 289906f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 290006f25ae9SGregory Neil Shapiro if (CltCERTfile != NULL) 290106f25ae9SGregory Neil Shapiro free(CltCERTfile); 290206f25ae9SGregory Neil Shapiro CltCERTfile = newstr(val); 290306f25ae9SGregory Neil Shapiro break; 290406f25ae9SGregory Neil Shapiro 290506f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 290606f25ae9SGregory Neil Shapiro if (Cltkeyfile != NULL) 290706f25ae9SGregory Neil Shapiro free(Cltkeyfile); 290806f25ae9SGregory Neil Shapiro Cltkeyfile = newstr(val); 290906f25ae9SGregory Neil Shapiro break; 291006f25ae9SGregory Neil Shapiro 291106f25ae9SGregory Neil Shapiro case O_CACERTFILE: 291206f25ae9SGregory Neil Shapiro if (CACERTfile != NULL) 291306f25ae9SGregory Neil Shapiro free(CACERTfile); 291406f25ae9SGregory Neil Shapiro CACERTfile = newstr(val); 291506f25ae9SGregory Neil Shapiro break; 291606f25ae9SGregory Neil Shapiro 291706f25ae9SGregory Neil Shapiro case O_CACERTPATH: 291806f25ae9SGregory Neil Shapiro if (CACERTpath != NULL) 291906f25ae9SGregory Neil Shapiro free(CACERTpath); 292006f25ae9SGregory Neil Shapiro CACERTpath = newstr(val); 292106f25ae9SGregory Neil Shapiro break; 292206f25ae9SGregory Neil Shapiro 292306f25ae9SGregory Neil Shapiro case O_DHPARAMS: 292406f25ae9SGregory Neil Shapiro if (DHParams != NULL) 292506f25ae9SGregory Neil Shapiro free(DHParams); 292606f25ae9SGregory Neil Shapiro DHParams = newstr(val); 292706f25ae9SGregory Neil Shapiro break; 292806f25ae9SGregory Neil Shapiro 292906f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 293006f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 293106f25ae9SGregory Neil Shapiro if (DHParams5 != NULL) 293206f25ae9SGregory Neil Shapiro free(DHParams5); 293306f25ae9SGregory Neil Shapiro DHParams5 = newstr(val); 293406f25ae9SGregory Neil Shapiro break; 293506f25ae9SGregory Neil Shapiro 293606f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 293706f25ae9SGregory Neil Shapiro if (CipherList != NULL) 293806f25ae9SGregory Neil Shapiro free(CipherList); 293906f25ae9SGregory Neil Shapiro CipherList = newstr(val); 294006f25ae9SGregory Neil Shapiro break; 294106f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 294206f25ae9SGregory Neil Shapiro 294306f25ae9SGregory Neil Shapiro case O_RANDFILE: 294406f25ae9SGregory Neil Shapiro if (RandFile != NULL) 294506f25ae9SGregory Neil Shapiro free(RandFile); 294606f25ae9SGregory Neil Shapiro RandFile= newstr(val); 294706f25ae9SGregory Neil Shapiro break; 294806f25ae9SGregory Neil Shapiro 294906f25ae9SGregory Neil Shapiro # else /* STARTTLS */ 295006f25ae9SGregory Neil Shapiro case O_SRVCERTFILE: 295106f25ae9SGregory Neil Shapiro case O_SRVKEYFILE: 295206f25ae9SGregory Neil Shapiro case O_CLTCERTFILE: 295306f25ae9SGregory Neil Shapiro case O_CLTKEYFILE: 295406f25ae9SGregory Neil Shapiro case O_CACERTFILE: 295506f25ae9SGregory Neil Shapiro case O_CACERTPATH: 295606f25ae9SGregory Neil Shapiro case O_DHPARAMS: 295706f25ae9SGregory Neil Shapiro # if _FFR_TLS_1 295806f25ae9SGregory Neil Shapiro case O_DHPARAMS5: 295906f25ae9SGregory Neil Shapiro case O_CIPHERLIST: 296006f25ae9SGregory Neil Shapiro # endif /* _FFR_TLS_1 */ 296106f25ae9SGregory Neil Shapiro case O_RANDFILE: 296206f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires TLS support\n", 296306f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 296406f25ae9SGregory Neil Shapiro break; 296506f25ae9SGregory Neil Shapiro 296606f25ae9SGregory Neil Shapiro # endif /* STARTTLS */ 296706f25ae9SGregory Neil Shapiro 296806f25ae9SGregory Neil Shapiro case O_CLIENTPORT: 296906f25ae9SGregory Neil Shapiro #if DAEMON 297006f25ae9SGregory Neil Shapiro setclientoptions(val); 297106f25ae9SGregory Neil Shapiro #else /* DAEMON */ 297206f25ae9SGregory Neil Shapiro syserr("ClientPortOptions (O option) set but DAEMON not compiled in"); 297306f25ae9SGregory Neil Shapiro #endif /* DAEMON */ 297406f25ae9SGregory Neil Shapiro break; 297506f25ae9SGregory Neil Shapiro 297606f25ae9SGregory Neil Shapiro case O_DF_BUFSIZE: 297706f25ae9SGregory Neil Shapiro DataFileBufferSize = atoi(val); 297806f25ae9SGregory Neil Shapiro break; 297906f25ae9SGregory Neil Shapiro 298006f25ae9SGregory Neil Shapiro case O_XF_BUFSIZE: 298106f25ae9SGregory Neil Shapiro XscriptFileBufferSize = atoi(val); 298206f25ae9SGregory Neil Shapiro break; 298306f25ae9SGregory Neil Shapiro 298406f25ae9SGregory Neil Shapiro case O_LDAPDEFAULTSPEC: 298506f25ae9SGregory Neil Shapiro #ifdef LDAPMAP 298606f25ae9SGregory Neil Shapiro ldapmap_set_defaults(val); 298706f25ae9SGregory Neil Shapiro #else /* LDAPMAP */ 298806f25ae9SGregory Neil Shapiro printf("Warning: Option: %s requires LDAP support (-DLDAPMAP)\n", 298906f25ae9SGregory Neil Shapiro o->o_name == NULL ? "<unknown>" : o->o_name); 299006f25ae9SGregory Neil Shapiro #endif /* LDAPMAP */ 299106f25ae9SGregory Neil Shapiro break; 299206f25ae9SGregory Neil Shapiro 299306f25ae9SGregory Neil Shapiro #if _FFR_MILTER 299406f25ae9SGregory Neil Shapiro case O_INPUTMILTER: 299506f25ae9SGregory Neil Shapiro InputFilterList = newstr(val); 299606f25ae9SGregory Neil Shapiro break; 299706f25ae9SGregory Neil Shapiro 299806f25ae9SGregory Neil Shapiro case O_MILTER: 299906f25ae9SGregory Neil Shapiro milter_set_option(subopt, val, sticky); 300006f25ae9SGregory Neil Shapiro break; 300106f25ae9SGregory Neil Shapiro #endif /* _FFR_MILTER */ 300206f25ae9SGregory Neil Shapiro 300306f25ae9SGregory Neil Shapiro #if _FFR_QUEUE_FILE_MODE 300406f25ae9SGregory Neil Shapiro case O_QUEUE_FILE_MODE: /* queue file mode */ 300506f25ae9SGregory Neil Shapiro QueueFileMode = atooct(val) & 0777; 300606f25ae9SGregory Neil Shapiro break; 300706f25ae9SGregory Neil Shapiro #endif /* _FFR_QUEUE_FILE_MODE */ 300825bab6e9SPeter Wemm 3009c2aa98e2SPeter Wemm default: 3010c2aa98e2SPeter Wemm if (tTd(37, 1)) 3011c2aa98e2SPeter Wemm { 3012c2aa98e2SPeter Wemm if (isascii(opt) && isprint(opt)) 301306f25ae9SGregory Neil Shapiro dprintf("Warning: option %c unknown\n", opt); 3014c2aa98e2SPeter Wemm else 301506f25ae9SGregory Neil Shapiro dprintf("Warning: option 0x%x unknown\n", opt); 3016c2aa98e2SPeter Wemm } 3017c2aa98e2SPeter Wemm break; 3018c2aa98e2SPeter Wemm } 301906f25ae9SGregory Neil Shapiro 302006f25ae9SGregory Neil Shapiro /* 302106f25ae9SGregory Neil Shapiro ** Options with suboptions are responsible for taking care 302206f25ae9SGregory Neil Shapiro ** of sticky-ness (e.g., that a command line setting is kept 302306f25ae9SGregory Neil Shapiro ** when reading in the sendmail.cf file). This has to be done 302406f25ae9SGregory Neil Shapiro ** when the suboptions are parsed since each suboption must be 302506f25ae9SGregory Neil Shapiro ** sticky, not the root option. 302606f25ae9SGregory Neil Shapiro */ 302706f25ae9SGregory Neil Shapiro 302806f25ae9SGregory Neil Shapiro if (sticky && !bitset(OI_SUBOPT, o->o_flags)) 3029c2aa98e2SPeter Wemm setbitn(opt, StickyOpt); 3030c2aa98e2SPeter Wemm } 3031c2aa98e2SPeter Wemm /* 3032c2aa98e2SPeter Wemm ** SETCLASS -- set a string into a class 3033c2aa98e2SPeter Wemm ** 3034c2aa98e2SPeter Wemm ** Parameters: 3035c2aa98e2SPeter Wemm ** class -- the class to put the string in. 3036c2aa98e2SPeter Wemm ** str -- the string to enter 3037c2aa98e2SPeter Wemm ** 3038c2aa98e2SPeter Wemm ** Returns: 3039c2aa98e2SPeter Wemm ** none. 3040c2aa98e2SPeter Wemm ** 3041c2aa98e2SPeter Wemm ** Side Effects: 3042c2aa98e2SPeter Wemm ** puts the word into the symbol table. 3043c2aa98e2SPeter Wemm */ 3044c2aa98e2SPeter Wemm 3045c2aa98e2SPeter Wemm void 3046c2aa98e2SPeter Wemm setclass(class, str) 3047c2aa98e2SPeter Wemm int class; 3048c2aa98e2SPeter Wemm char *str; 3049c2aa98e2SPeter Wemm { 3050c2aa98e2SPeter Wemm register STAB *s; 3051c2aa98e2SPeter Wemm 305206f25ae9SGregory Neil Shapiro if ((*str & 0377) == MATCHCLASS) 305306f25ae9SGregory Neil Shapiro { 305406f25ae9SGregory Neil Shapiro int mid; 305506f25ae9SGregory Neil Shapiro 305606f25ae9SGregory Neil Shapiro str++; 305706f25ae9SGregory Neil Shapiro mid = macid(str, NULL); 3058193538b7SGregory Neil Shapiro if (mid == 0) 305906f25ae9SGregory Neil Shapiro return; 306006f25ae9SGregory Neil Shapiro 3061c2aa98e2SPeter Wemm if (tTd(37, 8)) 306206f25ae9SGregory Neil Shapiro dprintf("setclass(%s, $=%s)\n", 306306f25ae9SGregory Neil Shapiro macname(class), macname(mid)); 306406f25ae9SGregory Neil Shapiro copy_class(mid, class); 306506f25ae9SGregory Neil Shapiro } 306606f25ae9SGregory Neil Shapiro else 306706f25ae9SGregory Neil Shapiro { 306806f25ae9SGregory Neil Shapiro if (tTd(37, 8)) 306906f25ae9SGregory Neil Shapiro dprintf("setclass(%s, %s)\n", macname(class), str); 307006f25ae9SGregory Neil Shapiro 3071c2aa98e2SPeter Wemm s = stab(str, ST_CLASS, ST_ENTER); 3072193538b7SGregory Neil Shapiro setbitn(bitidx(class), s->s_class); 3073c2aa98e2SPeter Wemm } 307406f25ae9SGregory Neil Shapiro } 3075c2aa98e2SPeter Wemm /* 3076c2aa98e2SPeter Wemm ** MAKEMAPENTRY -- create a map entry 3077c2aa98e2SPeter Wemm ** 3078c2aa98e2SPeter Wemm ** Parameters: 3079c2aa98e2SPeter Wemm ** line -- the config file line 3080c2aa98e2SPeter Wemm ** 3081c2aa98e2SPeter Wemm ** Returns: 3082c2aa98e2SPeter Wemm ** A pointer to the map that has been created. 3083c2aa98e2SPeter Wemm ** NULL if there was a syntax error. 3084c2aa98e2SPeter Wemm ** 3085c2aa98e2SPeter Wemm ** Side Effects: 3086c2aa98e2SPeter Wemm ** Enters the map into the dictionary. 3087c2aa98e2SPeter Wemm */ 3088c2aa98e2SPeter Wemm 3089c2aa98e2SPeter Wemm MAP * 3090c2aa98e2SPeter Wemm makemapentry(line) 3091c2aa98e2SPeter Wemm char *line; 3092c2aa98e2SPeter Wemm { 3093c2aa98e2SPeter Wemm register char *p; 3094c2aa98e2SPeter Wemm char *mapname; 3095c2aa98e2SPeter Wemm char *classname; 3096c2aa98e2SPeter Wemm register STAB *s; 3097c2aa98e2SPeter Wemm STAB *class; 3098c2aa98e2SPeter Wemm 3099c2aa98e2SPeter Wemm for (p = line; isascii(*p) && isspace(*p); p++) 3100c2aa98e2SPeter Wemm continue; 3101c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3102c2aa98e2SPeter Wemm { 3103c2aa98e2SPeter Wemm syserr("readcf: config K line: no map name"); 3104c2aa98e2SPeter Wemm return NULL; 3105c2aa98e2SPeter Wemm } 3106c2aa98e2SPeter Wemm 3107c2aa98e2SPeter Wemm mapname = p; 3108c2aa98e2SPeter Wemm while ((isascii(*++p) && isalnum(*p)) || *p == '_' || *p == '.') 3109c2aa98e2SPeter Wemm continue; 3110c2aa98e2SPeter Wemm if (*p != '\0') 3111c2aa98e2SPeter Wemm *p++ = '\0'; 3112c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3113c2aa98e2SPeter Wemm p++; 3114c2aa98e2SPeter Wemm if (!(isascii(*p) && isalnum(*p))) 3115c2aa98e2SPeter Wemm { 3116c2aa98e2SPeter Wemm syserr("readcf: config K line, map %s: no map class", mapname); 3117c2aa98e2SPeter Wemm return NULL; 3118c2aa98e2SPeter Wemm } 3119c2aa98e2SPeter Wemm classname = p; 3120c2aa98e2SPeter Wemm while (isascii(*++p) && isalnum(*p)) 3121c2aa98e2SPeter Wemm continue; 3122c2aa98e2SPeter Wemm if (*p != '\0') 3123c2aa98e2SPeter Wemm *p++ = '\0'; 3124c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3125c2aa98e2SPeter Wemm p++; 3126c2aa98e2SPeter Wemm 3127c2aa98e2SPeter Wemm /* look up the class */ 3128c2aa98e2SPeter Wemm class = stab(classname, ST_MAPCLASS, ST_FIND); 3129c2aa98e2SPeter Wemm if (class == NULL) 3130c2aa98e2SPeter Wemm { 3131c2aa98e2SPeter Wemm syserr("readcf: map %s: class %s not available", mapname, classname); 3132c2aa98e2SPeter Wemm return NULL; 3133c2aa98e2SPeter Wemm } 3134c2aa98e2SPeter Wemm 3135c2aa98e2SPeter Wemm /* enter the map */ 3136c2aa98e2SPeter Wemm s = stab(mapname, ST_MAP, ST_ENTER); 3137c2aa98e2SPeter Wemm s->s_map.map_class = &class->s_mapclass; 3138c2aa98e2SPeter Wemm s->s_map.map_mname = newstr(mapname); 3139c2aa98e2SPeter Wemm 3140c2aa98e2SPeter Wemm if (class->s_mapclass.map_parse(&s->s_map, p)) 3141c2aa98e2SPeter Wemm s->s_map.map_mflags |= MF_VALID; 3142c2aa98e2SPeter Wemm 3143c2aa98e2SPeter Wemm if (tTd(37, 5)) 3144c2aa98e2SPeter Wemm { 314506f25ae9SGregory Neil Shapiro dprintf("map %s, class %s, flags %lx, file %s,\n", 3146c2aa98e2SPeter Wemm s->s_map.map_mname, s->s_map.map_class->map_cname, 3147c2aa98e2SPeter Wemm s->s_map.map_mflags, 3148c2aa98e2SPeter Wemm s->s_map.map_file == NULL ? "(null)" : s->s_map.map_file); 314906f25ae9SGregory Neil Shapiro dprintf("\tapp %s, domain %s, rebuild %s\n", 3150c2aa98e2SPeter Wemm s->s_map.map_app == NULL ? "(null)" : s->s_map.map_app, 3151c2aa98e2SPeter Wemm s->s_map.map_domain == NULL ? "(null)" : s->s_map.map_domain, 3152c2aa98e2SPeter Wemm s->s_map.map_rebuild == NULL ? "(null)" : s->s_map.map_rebuild); 3153c2aa98e2SPeter Wemm } 3154c2aa98e2SPeter Wemm 3155c2aa98e2SPeter Wemm return &s->s_map; 3156c2aa98e2SPeter Wemm } 3157c2aa98e2SPeter Wemm /* 3158c2aa98e2SPeter Wemm ** STRTORWSET -- convert string to rewriting set number 3159c2aa98e2SPeter Wemm ** 3160c2aa98e2SPeter Wemm ** Parameters: 3161c2aa98e2SPeter Wemm ** p -- the pointer to the string to decode. 3162c2aa98e2SPeter Wemm ** endp -- if set, store the trailing delimiter here. 3163c2aa98e2SPeter Wemm ** stabmode -- ST_ENTER to create this entry, ST_FIND if 3164c2aa98e2SPeter Wemm ** it must already exist. 3165c2aa98e2SPeter Wemm ** 3166c2aa98e2SPeter Wemm ** Returns: 3167c2aa98e2SPeter Wemm ** The appropriate ruleset number. 3168c2aa98e2SPeter Wemm ** -1 if it is not valid (error already printed) 3169c2aa98e2SPeter Wemm */ 3170c2aa98e2SPeter Wemm 3171c2aa98e2SPeter Wemm int 3172c2aa98e2SPeter Wemm strtorwset(p, endp, stabmode) 3173c2aa98e2SPeter Wemm char *p; 3174c2aa98e2SPeter Wemm char **endp; 3175c2aa98e2SPeter Wemm int stabmode; 3176c2aa98e2SPeter Wemm { 3177c2aa98e2SPeter Wemm int ruleset; 3178c2aa98e2SPeter Wemm static int nextruleset = MAXRWSETS; 3179c2aa98e2SPeter Wemm 3180c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3181c2aa98e2SPeter Wemm p++; 3182c2aa98e2SPeter Wemm if (!isascii(*p)) 3183c2aa98e2SPeter Wemm { 3184c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", p); 3185c2aa98e2SPeter Wemm return -1; 3186c2aa98e2SPeter Wemm } 3187c2aa98e2SPeter Wemm if (isdigit(*p)) 3188c2aa98e2SPeter Wemm { 3189c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3190c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3191c2aa98e2SPeter Wemm { 3192c2aa98e2SPeter Wemm syserr("bad ruleset %d (%d max)", 3193c2aa98e2SPeter Wemm ruleset, MAXRWSETS / 2); 3194c2aa98e2SPeter Wemm ruleset = -1; 3195c2aa98e2SPeter Wemm } 3196c2aa98e2SPeter Wemm } 3197c2aa98e2SPeter Wemm else 3198c2aa98e2SPeter Wemm { 3199c2aa98e2SPeter Wemm STAB *s; 3200c2aa98e2SPeter Wemm char delim; 320106f25ae9SGregory Neil Shapiro char *q = NULL; 3202c2aa98e2SPeter Wemm 3203c2aa98e2SPeter Wemm q = p; 3204c2aa98e2SPeter Wemm while (*p != '\0' && isascii(*p) && 3205c2aa98e2SPeter Wemm (isalnum(*p) || *p == '_')) 3206c2aa98e2SPeter Wemm p++; 3207c2aa98e2SPeter Wemm if (q == p || !(isascii(*q) && isalpha(*q))) 3208c2aa98e2SPeter Wemm { 3209c2aa98e2SPeter Wemm /* no valid characters */ 3210c2aa98e2SPeter Wemm syserr("invalid ruleset name: \"%.20s\"", q); 3211c2aa98e2SPeter Wemm return -1; 3212c2aa98e2SPeter Wemm } 3213c2aa98e2SPeter Wemm while (isascii(*p) && isspace(*p)) 3214c2aa98e2SPeter Wemm *p++ = '\0'; 3215c2aa98e2SPeter Wemm delim = *p; 3216c2aa98e2SPeter Wemm if (delim != '\0') 3217c2aa98e2SPeter Wemm *p = '\0'; 3218c2aa98e2SPeter Wemm s = stab(q, ST_RULESET, stabmode); 3219c2aa98e2SPeter Wemm if (delim != '\0') 3220c2aa98e2SPeter Wemm *p = delim; 3221c2aa98e2SPeter Wemm 3222c2aa98e2SPeter Wemm if (s == NULL) 3223c2aa98e2SPeter Wemm return -1; 3224c2aa98e2SPeter Wemm 3225c2aa98e2SPeter Wemm if (stabmode == ST_ENTER && delim == '=') 3226c2aa98e2SPeter Wemm { 3227c2aa98e2SPeter Wemm while (isascii(*++p) && isspace(*p)) 3228c2aa98e2SPeter Wemm continue; 3229c2aa98e2SPeter Wemm if (!(isascii(*p) && isdigit(*p))) 3230c2aa98e2SPeter Wemm { 3231c2aa98e2SPeter Wemm syserr("bad ruleset definition \"%s\" (number required after `=')", q); 3232c2aa98e2SPeter Wemm ruleset = -1; 3233c2aa98e2SPeter Wemm } 3234c2aa98e2SPeter Wemm else 3235c2aa98e2SPeter Wemm { 3236c2aa98e2SPeter Wemm ruleset = strtol(p, endp, 10); 3237c2aa98e2SPeter Wemm if (ruleset >= MAXRWSETS / 2 || ruleset < 0) 3238c2aa98e2SPeter Wemm { 3239c2aa98e2SPeter Wemm syserr("bad ruleset number %d in \"%s\" (%d max)", 3240c2aa98e2SPeter Wemm ruleset, q, MAXRWSETS / 2); 3241c2aa98e2SPeter Wemm ruleset = -1; 3242c2aa98e2SPeter Wemm } 3243c2aa98e2SPeter Wemm } 3244c2aa98e2SPeter Wemm } 3245c2aa98e2SPeter Wemm else 3246c2aa98e2SPeter Wemm { 3247c2aa98e2SPeter Wemm if (endp != NULL) 3248c2aa98e2SPeter Wemm *endp = p; 324906f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0) 3250c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3251c2aa98e2SPeter Wemm else if ((ruleset = --nextruleset) < MAXRWSETS / 2) 3252c2aa98e2SPeter Wemm { 3253c2aa98e2SPeter Wemm syserr("%s: too many named rulesets (%d max)", 3254c2aa98e2SPeter Wemm q, MAXRWSETS / 2); 3255c2aa98e2SPeter Wemm ruleset = -1; 3256c2aa98e2SPeter Wemm } 3257c2aa98e2SPeter Wemm } 325806f25ae9SGregory Neil Shapiro if (s->s_ruleset >= 0 && 325906f25ae9SGregory Neil Shapiro ruleset >= 0 && 326006f25ae9SGregory Neil Shapiro ruleset != s->s_ruleset) 3261c2aa98e2SPeter Wemm { 3262c2aa98e2SPeter Wemm syserr("%s: ruleset changed value (old %d, new %d)", 3263c2aa98e2SPeter Wemm q, s->s_ruleset, ruleset); 3264c2aa98e2SPeter Wemm ruleset = s->s_ruleset; 3265c2aa98e2SPeter Wemm } 326606f25ae9SGregory Neil Shapiro else if (ruleset >= 0) 3267c2aa98e2SPeter Wemm { 3268c2aa98e2SPeter Wemm s->s_ruleset = ruleset; 3269c2aa98e2SPeter Wemm } 3270193538b7SGregory Neil Shapiro if (stabmode == ST_ENTER && ruleset >= 0) 327106f25ae9SGregory Neil Shapiro { 327206f25ae9SGregory Neil Shapiro char *h = NULL; 327306f25ae9SGregory Neil Shapiro 327406f25ae9SGregory Neil Shapiro if (RuleSetNames[ruleset] != NULL) 327506f25ae9SGregory Neil Shapiro free(RuleSetNames[ruleset]); 327606f25ae9SGregory Neil Shapiro if (delim != '\0' && (h = strchr(q, delim)) != NULL) 327706f25ae9SGregory Neil Shapiro *h = '\0'; 327806f25ae9SGregory Neil Shapiro RuleSetNames[ruleset] = newstr(q); 327906f25ae9SGregory Neil Shapiro if (delim == '/' && h != NULL) 328006f25ae9SGregory Neil Shapiro *h = delim; /* put back delim */ 328106f25ae9SGregory Neil Shapiro } 3282c2aa98e2SPeter Wemm } 3283c2aa98e2SPeter Wemm return ruleset; 3284c2aa98e2SPeter Wemm } 3285c2aa98e2SPeter Wemm /* 328606f25ae9SGregory Neil Shapiro ** SETTIMEOUT -- set an individual timeout 328706f25ae9SGregory Neil Shapiro ** 328806f25ae9SGregory Neil Shapiro ** Parameters: 328906f25ae9SGregory Neil Shapiro ** name -- the name of the timeout. 329006f25ae9SGregory Neil Shapiro ** val -- the value of the timeout. 329106f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 329206f25ae9SGregory Neil Shapiro ** this value. 329306f25ae9SGregory Neil Shapiro ** 329406f25ae9SGregory Neil Shapiro ** Returns: 329506f25ae9SGregory Neil Shapiro ** none. 329606f25ae9SGregory Neil Shapiro */ 329706f25ae9SGregory Neil Shapiro 329806f25ae9SGregory Neil Shapiro /* set if Timeout sub-option is stuck */ 329906f25ae9SGregory Neil Shapiro static BITMAP256 StickyTimeoutOpt; 330006f25ae9SGregory Neil Shapiro 330106f25ae9SGregory Neil Shapiro static struct timeoutinfo 330206f25ae9SGregory Neil Shapiro { 330306f25ae9SGregory Neil Shapiro char *to_name; /* long name of timeout */ 330406f25ae9SGregory Neil Shapiro u_char to_code; /* code for option */ 330506f25ae9SGregory Neil Shapiro } TimeOutTab[] = 330606f25ae9SGregory Neil Shapiro { 330706f25ae9SGregory Neil Shapiro #define TO_INITIAL 0x01 330806f25ae9SGregory Neil Shapiro { "initial", TO_INITIAL }, 330906f25ae9SGregory Neil Shapiro #define TO_MAIL 0x02 331006f25ae9SGregory Neil Shapiro { "mail", TO_MAIL }, 331106f25ae9SGregory Neil Shapiro #define TO_RCPT 0x03 331206f25ae9SGregory Neil Shapiro { "rcpt", TO_RCPT }, 331306f25ae9SGregory Neil Shapiro #define TO_DATAINIT 0x04 331406f25ae9SGregory Neil Shapiro { "datainit", TO_DATAINIT }, 331506f25ae9SGregory Neil Shapiro #define TO_DATABLOCK 0x05 331606f25ae9SGregory Neil Shapiro { "datablock", TO_DATABLOCK }, 331706f25ae9SGregory Neil Shapiro #define TO_DATAFINAL 0x06 331806f25ae9SGregory Neil Shapiro { "datafinal", TO_DATAFINAL }, 331906f25ae9SGregory Neil Shapiro #define TO_COMMAND 0x07 332006f25ae9SGregory Neil Shapiro { "command", TO_COMMAND }, 332106f25ae9SGregory Neil Shapiro #define TO_RSET 0x08 332206f25ae9SGregory Neil Shapiro { "rset", TO_RSET }, 332306f25ae9SGregory Neil Shapiro #define TO_HELO 0x09 332406f25ae9SGregory Neil Shapiro { "helo", TO_HELO }, 332506f25ae9SGregory Neil Shapiro #define TO_QUIT 0x0A 332606f25ae9SGregory Neil Shapiro { "quit", TO_QUIT }, 332706f25ae9SGregory Neil Shapiro #define TO_MISC 0x0B 332806f25ae9SGregory Neil Shapiro { "misc", TO_MISC }, 332906f25ae9SGregory Neil Shapiro #define TO_IDENT 0x0C 333006f25ae9SGregory Neil Shapiro { "ident", TO_IDENT }, 333106f25ae9SGregory Neil Shapiro #define TO_FILEOPEN 0x0D 333206f25ae9SGregory Neil Shapiro { "fileopen", TO_FILEOPEN }, 333306f25ae9SGregory Neil Shapiro #define TO_CONNECT 0x0E 333406f25ae9SGregory Neil Shapiro { "connect", TO_CONNECT }, 333506f25ae9SGregory Neil Shapiro #define TO_ICONNECT 0x0F 333606f25ae9SGregory Neil Shapiro { "iconnect", TO_ICONNECT }, 333706f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN 0x10 333806f25ae9SGregory Neil Shapiro { "queuewarn", TO_QUEUEWARN }, 333906f25ae9SGregory Neil Shapiro { "queuewarn.*", TO_QUEUEWARN }, 334006f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NORMAL 0x11 334106f25ae9SGregory Neil Shapiro { "queuewarn.normal", TO_QUEUEWARN_NORMAL }, 334206f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_URGENT 0x12 334306f25ae9SGregory Neil Shapiro { "queuewarn.urgent", TO_QUEUEWARN_URGENT }, 334406f25ae9SGregory Neil Shapiro #define TO_QUEUEWARN_NON_URGENT 0x13 334506f25ae9SGregory Neil Shapiro { "queuewarn.non-urgent", TO_QUEUEWARN_NON_URGENT }, 334606f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN 0x14 334706f25ae9SGregory Neil Shapiro { "queuereturn", TO_QUEUERETURN }, 334806f25ae9SGregory Neil Shapiro { "queuereturn.*", TO_QUEUERETURN }, 334906f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NORMAL 0x15 335006f25ae9SGregory Neil Shapiro { "queuereturn.normal", TO_QUEUERETURN_NORMAL }, 335106f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_URGENT 0x16 335206f25ae9SGregory Neil Shapiro { "queuereturn.urgent", TO_QUEUERETURN_URGENT }, 335306f25ae9SGregory Neil Shapiro #define TO_QUEUERETURN_NON_URGENT 0x17 335406f25ae9SGregory Neil Shapiro { "queuereturn.non-urgent", TO_QUEUERETURN_NON_URGENT }, 335506f25ae9SGregory Neil Shapiro #define TO_HOSTSTATUS 0x18 335606f25ae9SGregory Neil Shapiro { "hoststatus", TO_HOSTSTATUS }, 335706f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS 0x19 335806f25ae9SGregory Neil Shapiro { "resolver.retrans", TO_RESOLVER_RETRANS }, 335906f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_NORMAL 0x1A 336006f25ae9SGregory Neil Shapiro { "resolver.retrans.normal", TO_RESOLVER_RETRANS_NORMAL }, 336106f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRANS_FIRST 0x1B 336206f25ae9SGregory Neil Shapiro { "resolver.retrans.first", TO_RESOLVER_RETRANS_FIRST }, 336306f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY 0x1C 336406f25ae9SGregory Neil Shapiro { "resolver.retry", TO_RESOLVER_RETRY }, 336506f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_NORMAL 0x1D 336606f25ae9SGregory Neil Shapiro { "resolver.retry.normal", TO_RESOLVER_RETRY_NORMAL }, 336706f25ae9SGregory Neil Shapiro #define TO_RESOLVER_RETRY_FIRST 0x1E 336806f25ae9SGregory Neil Shapiro { "resolver.retry.first", TO_RESOLVER_RETRY_FIRST }, 336906f25ae9SGregory Neil Shapiro #define TO_CONTROL 0x1F 337006f25ae9SGregory Neil Shapiro { "control", TO_CONTROL }, 337106f25ae9SGregory Neil Shapiro { NULL, 0 }, 337206f25ae9SGregory Neil Shapiro }; 337306f25ae9SGregory Neil Shapiro 337406f25ae9SGregory Neil Shapiro 337506f25ae9SGregory Neil Shapiro static void 337606f25ae9SGregory Neil Shapiro settimeout(name, val, sticky) 337706f25ae9SGregory Neil Shapiro char *name; 337806f25ae9SGregory Neil Shapiro char *val; 337906f25ae9SGregory Neil Shapiro bool sticky; 338006f25ae9SGregory Neil Shapiro { 338106f25ae9SGregory Neil Shapiro register struct timeoutinfo *to; 338206f25ae9SGregory Neil Shapiro int i; 338306f25ae9SGregory Neil Shapiro time_t toval; 338406f25ae9SGregory Neil Shapiro 338506f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 338606f25ae9SGregory Neil Shapiro dprintf("settimeout(%s = %s)", name, val); 338706f25ae9SGregory Neil Shapiro 338806f25ae9SGregory Neil Shapiro for (to = TimeOutTab; to->to_name != NULL; to++) 338906f25ae9SGregory Neil Shapiro { 339006f25ae9SGregory Neil Shapiro if (strcasecmp(to->to_name, name) == 0) 339106f25ae9SGregory Neil Shapiro break; 339206f25ae9SGregory Neil Shapiro } 339306f25ae9SGregory Neil Shapiro 339406f25ae9SGregory Neil Shapiro if (to->to_name == NULL) 3395193538b7SGregory Neil Shapiro { 3396193538b7SGregory Neil Shapiro errno = 0; /* avoid bogus error text */ 339706f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 3398193538b7SGregory Neil Shapiro return; 3399193538b7SGregory Neil Shapiro } 340006f25ae9SGregory Neil Shapiro 340106f25ae9SGregory Neil Shapiro /* 340206f25ae9SGregory Neil Shapiro ** See if this option is preset for us. 340306f25ae9SGregory Neil Shapiro */ 340406f25ae9SGregory Neil Shapiro 340506f25ae9SGregory Neil Shapiro if (!sticky && bitnset(to->to_code, StickyTimeoutOpt)) 340606f25ae9SGregory Neil Shapiro { 340706f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 340806f25ae9SGregory Neil Shapiro dprintf(" (ignored)\n"); 340906f25ae9SGregory Neil Shapiro return; 341006f25ae9SGregory Neil Shapiro } 341106f25ae9SGregory Neil Shapiro 341206f25ae9SGregory Neil Shapiro if (tTd(37, 2)) 341306f25ae9SGregory Neil Shapiro dprintf("\n"); 341406f25ae9SGregory Neil Shapiro 341506f25ae9SGregory Neil Shapiro toval = convtime(val, 'm'); 341606f25ae9SGregory Neil Shapiro 341706f25ae9SGregory Neil Shapiro switch (to->to_code) 341806f25ae9SGregory Neil Shapiro { 341906f25ae9SGregory Neil Shapiro case TO_INITIAL: 342006f25ae9SGregory Neil Shapiro TimeOuts.to_initial = toval; 342106f25ae9SGregory Neil Shapiro break; 342206f25ae9SGregory Neil Shapiro 342306f25ae9SGregory Neil Shapiro case TO_MAIL: 342406f25ae9SGregory Neil Shapiro TimeOuts.to_mail = toval; 342506f25ae9SGregory Neil Shapiro break; 342606f25ae9SGregory Neil Shapiro 342706f25ae9SGregory Neil Shapiro case TO_RCPT: 342806f25ae9SGregory Neil Shapiro TimeOuts.to_rcpt = toval; 342906f25ae9SGregory Neil Shapiro break; 343006f25ae9SGregory Neil Shapiro 343106f25ae9SGregory Neil Shapiro case TO_DATAINIT: 343206f25ae9SGregory Neil Shapiro TimeOuts.to_datainit = toval; 343306f25ae9SGregory Neil Shapiro break; 343406f25ae9SGregory Neil Shapiro 343506f25ae9SGregory Neil Shapiro case TO_DATABLOCK: 343606f25ae9SGregory Neil Shapiro TimeOuts.to_datablock = toval; 343706f25ae9SGregory Neil Shapiro break; 343806f25ae9SGregory Neil Shapiro 343906f25ae9SGregory Neil Shapiro case TO_DATAFINAL: 344006f25ae9SGregory Neil Shapiro TimeOuts.to_datafinal = toval; 344106f25ae9SGregory Neil Shapiro break; 344206f25ae9SGregory Neil Shapiro 344306f25ae9SGregory Neil Shapiro case TO_COMMAND: 344406f25ae9SGregory Neil Shapiro TimeOuts.to_nextcommand = toval; 344506f25ae9SGregory Neil Shapiro break; 344606f25ae9SGregory Neil Shapiro 344706f25ae9SGregory Neil Shapiro case TO_RSET: 344806f25ae9SGregory Neil Shapiro TimeOuts.to_rset = toval; 344906f25ae9SGregory Neil Shapiro break; 345006f25ae9SGregory Neil Shapiro 345106f25ae9SGregory Neil Shapiro case TO_HELO: 345206f25ae9SGregory Neil Shapiro TimeOuts.to_helo = toval; 345306f25ae9SGregory Neil Shapiro break; 345406f25ae9SGregory Neil Shapiro 345506f25ae9SGregory Neil Shapiro case TO_QUIT: 345606f25ae9SGregory Neil Shapiro TimeOuts.to_quit = toval; 345706f25ae9SGregory Neil Shapiro break; 345806f25ae9SGregory Neil Shapiro 345906f25ae9SGregory Neil Shapiro case TO_MISC: 346006f25ae9SGregory Neil Shapiro TimeOuts.to_miscshort = toval; 346106f25ae9SGregory Neil Shapiro break; 346206f25ae9SGregory Neil Shapiro 346306f25ae9SGregory Neil Shapiro case TO_IDENT: 346406f25ae9SGregory Neil Shapiro TimeOuts.to_ident = toval; 346506f25ae9SGregory Neil Shapiro break; 346606f25ae9SGregory Neil Shapiro 346706f25ae9SGregory Neil Shapiro case TO_FILEOPEN: 346806f25ae9SGregory Neil Shapiro TimeOuts.to_fileopen = toval; 346906f25ae9SGregory Neil Shapiro break; 347006f25ae9SGregory Neil Shapiro 347106f25ae9SGregory Neil Shapiro case TO_CONNECT: 347206f25ae9SGregory Neil Shapiro TimeOuts.to_connect = toval; 347306f25ae9SGregory Neil Shapiro break; 347406f25ae9SGregory Neil Shapiro 347506f25ae9SGregory Neil Shapiro case TO_ICONNECT: 347606f25ae9SGregory Neil Shapiro TimeOuts.to_iconnect = toval; 347706f25ae9SGregory Neil Shapiro break; 347806f25ae9SGregory Neil Shapiro 347906f25ae9SGregory Neil Shapiro case TO_QUEUEWARN: 348006f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 348106f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 348206f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 348306f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 348406f25ae9SGregory Neil Shapiro break; 348506f25ae9SGregory Neil Shapiro 348606f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NORMAL: 348706f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 348806f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NORMAL] = toval; 348906f25ae9SGregory Neil Shapiro break; 349006f25ae9SGregory Neil Shapiro 349106f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_URGENT: 349206f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 349306f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_URGENT] = toval; 349406f25ae9SGregory Neil Shapiro break; 349506f25ae9SGregory Neil Shapiro 349606f25ae9SGregory Neil Shapiro case TO_QUEUEWARN_NON_URGENT: 349706f25ae9SGregory Neil Shapiro toval = convtime(val, 'h'); 349806f25ae9SGregory Neil Shapiro TimeOuts.to_q_warning[TOC_NONURGENT] = toval; 349906f25ae9SGregory Neil Shapiro break; 350006f25ae9SGregory Neil Shapiro 350106f25ae9SGregory Neil Shapiro case TO_QUEUERETURN: 350206f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 350306f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 350406f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 350506f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 350606f25ae9SGregory Neil Shapiro break; 350706f25ae9SGregory Neil Shapiro 350806f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NORMAL: 350906f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 351006f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NORMAL] = toval; 351106f25ae9SGregory Neil Shapiro break; 351206f25ae9SGregory Neil Shapiro 351306f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_URGENT: 351406f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 351506f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_URGENT] = toval; 351606f25ae9SGregory Neil Shapiro break; 351706f25ae9SGregory Neil Shapiro 351806f25ae9SGregory Neil Shapiro case TO_QUEUERETURN_NON_URGENT: 351906f25ae9SGregory Neil Shapiro toval = convtime(val, 'd'); 352006f25ae9SGregory Neil Shapiro TimeOuts.to_q_return[TOC_NONURGENT] = toval; 352106f25ae9SGregory Neil Shapiro break; 352206f25ae9SGregory Neil Shapiro 352306f25ae9SGregory Neil Shapiro 352406f25ae9SGregory Neil Shapiro case TO_HOSTSTATUS: 352506f25ae9SGregory Neil Shapiro MciInfoTimeout = toval; 352606f25ae9SGregory Neil Shapiro break; 352706f25ae9SGregory Neil Shapiro 352806f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS: 352906f25ae9SGregory Neil Shapiro toval = convtime(val, 's'); 353006f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_DEFAULT] = toval; 353106f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = toval; 353206f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = toval; 353306f25ae9SGregory Neil Shapiro break; 353406f25ae9SGregory Neil Shapiro 353506f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY: 353606f25ae9SGregory Neil Shapiro i = atoi(val); 353706f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_DEFAULT] = i; 353806f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = i; 353906f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = i; 354006f25ae9SGregory Neil Shapiro break; 354106f25ae9SGregory Neil Shapiro 354206f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_NORMAL: 354306f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_NORMAL] = convtime(val, 's'); 354406f25ae9SGregory Neil Shapiro break; 354506f25ae9SGregory Neil Shapiro 354606f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_NORMAL: 354706f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_NORMAL] = atoi(val); 354806f25ae9SGregory Neil Shapiro break; 354906f25ae9SGregory Neil Shapiro 355006f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRANS_FIRST: 355106f25ae9SGregory Neil Shapiro TimeOuts.res_retrans[RES_TO_FIRST] = convtime(val, 's'); 355206f25ae9SGregory Neil Shapiro break; 355306f25ae9SGregory Neil Shapiro 355406f25ae9SGregory Neil Shapiro case TO_RESOLVER_RETRY_FIRST: 355506f25ae9SGregory Neil Shapiro TimeOuts.res_retry[RES_TO_FIRST] = atoi(val); 355606f25ae9SGregory Neil Shapiro break; 355706f25ae9SGregory Neil Shapiro 355806f25ae9SGregory Neil Shapiro case TO_CONTROL: 355906f25ae9SGregory Neil Shapiro TimeOuts.to_control = toval; 356006f25ae9SGregory Neil Shapiro break; 356106f25ae9SGregory Neil Shapiro 356206f25ae9SGregory Neil Shapiro default: 356306f25ae9SGregory Neil Shapiro syserr("settimeout: invalid timeout %s", name); 356406f25ae9SGregory Neil Shapiro break; 356506f25ae9SGregory Neil Shapiro } 356606f25ae9SGregory Neil Shapiro 356706f25ae9SGregory Neil Shapiro if (sticky) 356806f25ae9SGregory Neil Shapiro setbitn(to->to_code, StickyTimeoutOpt); 356906f25ae9SGregory Neil Shapiro } 357006f25ae9SGregory Neil Shapiro /* 3571c2aa98e2SPeter Wemm ** INITTIMEOUTS -- parse and set timeout values 3572c2aa98e2SPeter Wemm ** 3573c2aa98e2SPeter Wemm ** Parameters: 3574c2aa98e2SPeter Wemm ** val -- a pointer to the values. If NULL, do initial 3575c2aa98e2SPeter Wemm ** settings. 357606f25ae9SGregory Neil Shapiro ** sticky -- if set, don't let other setoptions override 357706f25ae9SGregory Neil Shapiro ** this suboption value. 3578c2aa98e2SPeter Wemm ** 3579c2aa98e2SPeter Wemm ** Returns: 3580c2aa98e2SPeter Wemm ** none. 3581c2aa98e2SPeter Wemm ** 3582c2aa98e2SPeter Wemm ** Side Effects: 3583c2aa98e2SPeter Wemm ** Initializes the TimeOuts structure 3584c2aa98e2SPeter Wemm */ 3585c2aa98e2SPeter Wemm 3586c2aa98e2SPeter Wemm void 358706f25ae9SGregory Neil Shapiro inittimeouts(val, sticky) 3588c2aa98e2SPeter Wemm register char *val; 358906f25ae9SGregory Neil Shapiro bool sticky; 3590c2aa98e2SPeter Wemm { 3591c2aa98e2SPeter Wemm register char *p; 3592c2aa98e2SPeter Wemm 3593c2aa98e2SPeter Wemm if (tTd(37, 2)) 359406f25ae9SGregory Neil Shapiro dprintf("inittimeouts(%s)\n", val == NULL ? "<NULL>" : val); 3595c2aa98e2SPeter Wemm if (val == NULL) 3596c2aa98e2SPeter Wemm { 3597c2aa98e2SPeter Wemm TimeOuts.to_connect = (time_t) 0 SECONDS; 3598c2aa98e2SPeter Wemm TimeOuts.to_initial = (time_t) 5 MINUTES; 3599c2aa98e2SPeter Wemm TimeOuts.to_helo = (time_t) 5 MINUTES; 3600c2aa98e2SPeter Wemm TimeOuts.to_mail = (time_t) 10 MINUTES; 3601c2aa98e2SPeter Wemm TimeOuts.to_rcpt = (time_t) 1 HOUR; 3602c2aa98e2SPeter Wemm TimeOuts.to_datainit = (time_t) 5 MINUTES; 3603c2aa98e2SPeter Wemm TimeOuts.to_datablock = (time_t) 1 HOUR; 3604c2aa98e2SPeter Wemm TimeOuts.to_datafinal = (time_t) 1 HOUR; 3605c2aa98e2SPeter Wemm TimeOuts.to_rset = (time_t) 5 MINUTES; 3606c2aa98e2SPeter Wemm TimeOuts.to_quit = (time_t) 2 MINUTES; 3607c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = (time_t) 1 HOUR; 3608c2aa98e2SPeter Wemm TimeOuts.to_miscshort = (time_t) 2 MINUTES; 3609c2aa98e2SPeter Wemm #if IDENTPROTO 361006f25ae9SGregory Neil Shapiro TimeOuts.to_ident = (time_t) 5 SECONDS; 361106f25ae9SGregory Neil Shapiro #else /* IDENTPROTO */ 3612c2aa98e2SPeter Wemm TimeOuts.to_ident = (time_t) 0 SECONDS; 361306f25ae9SGregory Neil Shapiro #endif /* IDENTPROTO */ 3614c2aa98e2SPeter Wemm TimeOuts.to_fileopen = (time_t) 60 SECONDS; 361506f25ae9SGregory Neil Shapiro TimeOuts.to_control = (time_t) 2 MINUTES; 3616c2aa98e2SPeter Wemm if (tTd(37, 5)) 3617c2aa98e2SPeter Wemm { 361806f25ae9SGregory Neil Shapiro dprintf("Timeouts:\n"); 361906f25ae9SGregory Neil Shapiro dprintf(" connect = %ld\n", (long)TimeOuts.to_connect); 362006f25ae9SGregory Neil Shapiro dprintf(" initial = %ld\n", (long)TimeOuts.to_initial); 362106f25ae9SGregory Neil Shapiro dprintf(" helo = %ld\n", (long)TimeOuts.to_helo); 362206f25ae9SGregory Neil Shapiro dprintf(" mail = %ld\n", (long)TimeOuts.to_mail); 362306f25ae9SGregory Neil Shapiro dprintf(" rcpt = %ld\n", (long)TimeOuts.to_rcpt); 362406f25ae9SGregory Neil Shapiro dprintf(" datainit = %ld\n", (long)TimeOuts.to_datainit); 362506f25ae9SGregory Neil Shapiro dprintf(" datablock = %ld\n", (long)TimeOuts.to_datablock); 362606f25ae9SGregory Neil Shapiro dprintf(" datafinal = %ld\n", (long)TimeOuts.to_datafinal); 362706f25ae9SGregory Neil Shapiro dprintf(" rset = %ld\n", (long)TimeOuts.to_rset); 362806f25ae9SGregory Neil Shapiro dprintf(" quit = %ld\n", (long)TimeOuts.to_quit); 362906f25ae9SGregory Neil Shapiro dprintf(" nextcommand = %ld\n", (long)TimeOuts.to_nextcommand); 363006f25ae9SGregory Neil Shapiro dprintf(" miscshort = %ld\n", (long)TimeOuts.to_miscshort); 363106f25ae9SGregory Neil Shapiro dprintf(" ident = %ld\n", (long)TimeOuts.to_ident); 363206f25ae9SGregory Neil Shapiro dprintf(" fileopen = %ld\n", (long)TimeOuts.to_fileopen); 363306f25ae9SGregory Neil Shapiro dprintf(" control = %ld\n", (long)TimeOuts.to_control); 3634c2aa98e2SPeter Wemm } 3635c2aa98e2SPeter Wemm return; 3636c2aa98e2SPeter Wemm } 3637c2aa98e2SPeter Wemm 3638c2aa98e2SPeter Wemm for (;; val = p) 3639c2aa98e2SPeter Wemm { 3640c2aa98e2SPeter Wemm while (isascii(*val) && isspace(*val)) 3641c2aa98e2SPeter Wemm val++; 3642c2aa98e2SPeter Wemm if (*val == '\0') 3643c2aa98e2SPeter Wemm break; 3644c2aa98e2SPeter Wemm for (p = val; *p != '\0' && *p != ','; p++) 3645c2aa98e2SPeter Wemm continue; 3646c2aa98e2SPeter Wemm if (*p != '\0') 3647c2aa98e2SPeter Wemm *p++ = '\0'; 3648c2aa98e2SPeter Wemm 3649c2aa98e2SPeter Wemm if (isascii(*val) && isdigit(*val)) 3650c2aa98e2SPeter Wemm { 3651c2aa98e2SPeter Wemm /* old syntax -- set everything */ 3652c2aa98e2SPeter Wemm TimeOuts.to_mail = convtime(val, 'm'); 3653c2aa98e2SPeter Wemm TimeOuts.to_rcpt = TimeOuts.to_mail; 3654c2aa98e2SPeter Wemm TimeOuts.to_datainit = TimeOuts.to_mail; 3655c2aa98e2SPeter Wemm TimeOuts.to_datablock = TimeOuts.to_mail; 3656c2aa98e2SPeter Wemm TimeOuts.to_datafinal = TimeOuts.to_mail; 3657c2aa98e2SPeter Wemm TimeOuts.to_nextcommand = TimeOuts.to_mail; 365806f25ae9SGregory Neil Shapiro if (sticky) 365906f25ae9SGregory Neil Shapiro { 366006f25ae9SGregory Neil Shapiro setbitn(TO_MAIL, StickyTimeoutOpt); 366106f25ae9SGregory Neil Shapiro setbitn(TO_RCPT, StickyTimeoutOpt); 366206f25ae9SGregory Neil Shapiro setbitn(TO_DATAINIT, StickyTimeoutOpt); 366306f25ae9SGregory Neil Shapiro setbitn(TO_DATABLOCK, StickyTimeoutOpt); 366406f25ae9SGregory Neil Shapiro setbitn(TO_DATAFINAL, StickyTimeoutOpt); 366506f25ae9SGregory Neil Shapiro setbitn(TO_COMMAND, StickyTimeoutOpt); 366606f25ae9SGregory Neil Shapiro } 3667c2aa98e2SPeter Wemm continue; 3668c2aa98e2SPeter Wemm } 3669c2aa98e2SPeter Wemm else 3670c2aa98e2SPeter Wemm { 3671c2aa98e2SPeter Wemm register char *q = strchr(val, ':'); 3672c2aa98e2SPeter Wemm 3673c2aa98e2SPeter Wemm if (q == NULL && (q = strchr(val, '=')) == NULL) 3674c2aa98e2SPeter Wemm { 3675c2aa98e2SPeter Wemm /* syntax error */ 3676c2aa98e2SPeter Wemm continue; 3677c2aa98e2SPeter Wemm } 3678c2aa98e2SPeter Wemm *q++ = '\0'; 367906f25ae9SGregory Neil Shapiro settimeout(val, q, sticky); 3680c2aa98e2SPeter Wemm } 3681c2aa98e2SPeter Wemm } 3682c2aa98e2SPeter Wemm } 3683