1 /*************************************************
2 * Exim - an Internet mail transport agent *
3 *************************************************/
5 /* Copyright (c) University of Cambridge 1995 - 2009 */
6 /* See the file NOTICE for conditions of use and distribution. */
9 /*************************************************
10 * Build configuration header for Exim *
11 *************************************************/
13 /* This auxiliary program builds the file config.h by the following
16 First, it determines the size of off_t and time_t variables, and generates
17 macro code to define OFF_T_FMT and TIME_T_FMT as suitable formats, if they are
18 not already defined in the system-specific header file.
20 Then it reads Makefile, looking for certain OS-specific definitions which it
21 uses to define some specific macros. Finally, it reads the defaults file
24 The defaults file contains normal C #define statements for various macros; if
25 the name of a macro is found in the environment, the environment value replaces
26 the default. If the default #define does not contain any value, then that macro
27 is not copied to the created file unless there is some value in the
30 This program is compiled and run as part of the Make process and is not
31 normally called independently. */
38 #include <sys/types.h>
52 static const char *db_opts[] = { "", "USE_DB", "USE_GDBM", "USE_TDB" };
54 static int have_ipv6 = 0;
55 static int have_iconv = 0;
57 static char errno_quota[256];
58 static char ostype[256];
61 /* If any entry is an initial substring of another, the longer one must
64 static have_item have_list[] = {
65 { "HAVE_IPV6", &have_ipv6 },
66 { "HAVE_ICONV", &have_iconv },
70 static save_item save_list[] = {
71 { "ERRNO_QUOTA", errno_quota },
78 /* Subroutine to check a string for precisely one instance of "%s". If not,
82 check_percent_ess(char *value, char *name)
85 char *p = strstr(value, "%s");
86 if (p != NULL) OK = strstr(p+2, "%s") == NULL;
89 printf("\n*** \"%s\" (%s) must contain precisely one occurrence of\n"
90 "*** \"%%s\". Please review your build-time configuration.\n\n/", value,
100 main(int argc, char **argv)
102 off_t test_off_t = 0;
103 time_t test_time_t = 0;
104 #if ! (__STDC_VERSION__ >= 199901L)
105 size_t test_size_t = 0;
106 unsigned long test_ulong_t = 0L;
108 long test_long_t = 0;
111 int last_initial = 'A';
114 int in_local_makefile = 0;
115 int use_which_db = 0;
116 int use_which_db_in_local_makefile = 0;
117 int support_crypteq = 0;
122 printf("*** Buildconfig: called with incorrect arguments\n");
126 new = fopen("config.h", "wb");
129 printf("*** Buildconfig: failed to open config.h for output\n");
133 printf("Building configuration file config.h\n");
135 fprintf(new, "/*************************************************\n");
136 fprintf(new, "* Configuration header for Exim *\n");
137 fprintf(new, "*************************************************/\n\n");
139 fprintf(new, "/* This file was automatically generated from Makefile and "
140 "config.h.defaults,\n");
141 fprintf(new, "using values specified in the configuration file Local/Makefile.\n");
142 fprintf(new, "Do not edit it. Instead, edit Local/Makefile and "
143 "rerun make. */\n\n");
145 /* First, deal with the printing format for off_t variables. We assume that if
146 the size of off_t is greater than 4, "%lld" will be available as a format for
147 printing long long variables, and there will be support for the long long type.
148 This assumption is known to be OK for the common operating systems. */
150 fprintf(new, "#ifndef OFF_T_FMT\n");
151 if (sizeof(test_off_t) > sizeof(test_long_t))
153 fprintf(new, "#define OFF_T_FMT \"%%lld\"\n");
154 fprintf(new, "#define LONGLONG_T long long int\n");
158 fprintf(new, "#define OFF_T_FMT \"%%ld\"\n");
159 fprintf(new, "#define LONGLONG_T long int\n");
161 fprintf(new, "#endif\n\n");
163 /* Now do the same thing for time_t variables. If the length is greater than
164 4, we want to assume long long support (even if off_t was less than 4). If the
165 length is 4 or less, we can leave LONGLONG_T to whatever was defined above for
168 fprintf(new, "#ifndef TIME_T_FMT\n");
169 if (sizeof(test_time_t) > sizeof(test_long_t))
171 fprintf(new, "#define TIME_T_FMT \"%%lld\"\n");
172 fprintf(new, "#undef LONGLONG_T\n");
173 fprintf(new, "#define LONGLONG_T long long int\n");
177 fprintf(new, "#define TIME_T_FMT \"%%ld\"\n");
179 fprintf(new, "#endif\n\n");
181 /* And for sizeof() results, size_t, which should with C99 be just %zu, deal
182 with C99 not being ubiquitous yet. Unfortunately. */
184 #if __STDC_VERSION__ >= 199901L
185 fprintf(new, "#define SIZE_T_FMT \"%%zu\"\n");
187 if (sizeof(test_size_t) > sizeof (test_ulong_t))
188 fprintf(new, "#define SIZE_T_FMT \"%%llu\"\n");
190 fprintf(new, "#define SIZE_T_FMT \"%%lu\"\n");
193 /* Now search the makefile for certain settings */
195 base = fopen("Makefile", "rb");
198 printf("*** Buildconfig: failed to open Makefile\n");
203 errno_quota[0] = 0; /* no over-riding value set */
204 ostype[0] = 0; /* just in case */
207 while (fgets(buffer, sizeof(buffer), base) != NULL)
212 char *p = buffer + (int)strlen(buffer);
214 while (p > buffer && isspace((unsigned char)p[-1])) p--;
217 while (isspace((unsigned char)*p)) p++;
219 /* Notice when we hit the user's makefile */
221 if (strcmp(p, "# From Local/Makefile") == 0)
223 in_local_makefile = 1;
227 /* Remember the last DB option setting. If we hit two in the user's
228 Makefile, complain. */
230 for (i = 1; i < sizeof(db_opts)/sizeof(char *); i++)
232 int len = (int)strlen(db_opts[i]);
233 if (strncmp(p, db_opts[i], len) == 0 && (p[len] == ' ' || p[len] == '='))
235 if (in_local_makefile)
237 if (use_which_db_in_local_makefile)
239 printf("*** Only one of USE_DB, USE_GDBM, or USE_TDB should be "
240 "defined in Local/Makefile\n");
243 use_which_db_in_local_makefile = 1;
249 if (i < sizeof(db_opts)/sizeof(char *)) continue;
251 /* Items where we just save a boolean */
253 for (h = have_list; h->name != NULL; h++)
255 int len = (int)strlen(h->name);
256 if (strncmp(p, h->name, len) == 0)
259 while (isspace((unsigned char)*p)) p++;
262 printf("*** Buildconfig: syntax error in Makefile line %d\n", linecount);
265 while (isspace((unsigned char)*p)) p++;
266 if (strcmp(p, "YES") == 0 || strcmp(p, "yes") == 0) *(h->flag) = 1;
267 else *(h->flag) = 0; /* Must reset in case multiple instances */
272 if (h->name != NULL) continue;
274 /* Items where we save the complete string */
276 for (s = save_list; s->name != NULL; s++)
278 int len = (int)strlen(s->name);
279 if (strncmp(p, s->name, len) == 0)
282 while (isspace((unsigned char)*p)) p++;
285 printf("*** Buildconfig: syntax error in Makefile line %d\n", linecount);
288 while (isspace((unsigned char)*p)) p++;
294 fprintf(new, "#define HAVE_IPV6 %s\n",
295 have_ipv6? "TRUE" : "FALSE");
297 fprintf(new, "#define HAVE_ICONV %s\n",
298 have_iconv? "TRUE" : "FALSE");
300 if (errno_quota[0] != 0)
301 fprintf(new, "\n#define ERRNO_QUOTA %s\n", errno_quota);
303 if (strcmp(cc, "gcc") == 0 &&
304 (strstr(ostype, "IRIX") != NULL || strstr(ostype, "AIX") != NULL))
306 fprintf(new, "\n/* This switch includes the code to fix the inet_ntoa() */");
307 fprintf(new, "\n/* bug when using gcc on an IRIX or AIX system. */");
308 fprintf(new, "\n#define USE_INET_NTOA_FIX");
315 /* Now handle the macros listed in the defaults */
317 base = fopen("../src/config.h.defaults", "rb");
320 printf("*** Buildconfig: failed to open ../src/config.h.defaults\n");
325 while (fgets(buffer, sizeof(buffer), base) != NULL)
333 while (*p == ' ' || *p == '\t') p++;
335 if (strncmp(p, "#define ", 8) != 0) continue;
338 while (*p == ' ' || *p == '\t') p++;
340 if (*p < last_initial) fprintf(new, "\n");
343 while (*p && (isalnum((unsigned char)*p) || *p == '_')) *q++ = *p++;
346 /* USE_DB, USE_GDBM, and USE_TDB are special cases. We want to have only
347 one of them set. The scan of the Makefile has saved which was the last one
350 for (i = 1; i < sizeof(db_opts)/sizeof(char *); i++)
352 if (strcmp(name, db_opts[i]) == 0)
354 if (use_which_db == i)
355 fprintf(new, "#define %s %.*syes\n", db_opts[i],
356 21 - (int)strlen(db_opts[i]), " ");
358 fprintf(new, "/* %s not set */\n", name);
362 if (i < sizeof(db_opts)/sizeof(char *)) continue;
364 /* EXIM_USER is a special case. We look in the environment for EXIM_USER or
365 EXIM_UID (the latter for backward compatibility with Exim 3). If the value is
366 not numeric, we look up the user, and default the GID if found. Otherwise,
367 EXIM_GROUP or EXIM_GID must be in the environment. */
369 if (strcmp(name, "EXIM_UID") == 0)
375 char *username = NULL;
376 char *groupname = NULL;
378 char *user = getenv("EXIM_USER");
379 char *group = getenv("EXIM_GROUP");
381 if (user == NULL) user = getenv("EXIM_UID");
382 if (group == NULL) group = getenv("EXIM_GID");
386 printf("\n*** EXIM_USER has not been defined in any of the Makefiles in "
387 "the\n \"Local\" directory. Please review your build-time "
388 "configuration.\n\n");
392 while (isspace((unsigned char)(*user))) user++;
395 printf("\n*** EXIM_USER is defined as an empty string in one of the "
396 "files\n in the \"Local\" directory. Please review your build-time"
397 "\n configuration.\n\n");
401 for (s = user; *s != 0; s++)
403 if (iscntrl((unsigned char)(*s)))
405 printf("\n*** EXIM_USER contains the control character 0x%02X in one "
406 "of the files\n in the \"Local\" directory. Please review your "
407 "build-time\n configuration.\n\n", *s);
412 /* Numeric uid given */
414 if (user[strspn(user, "0123456789")] == 0)
416 uid = (uid_t)atoi(user);
419 /* User name given. Normally, we look up the uid right away. However,
420 people building binary distributions sometimes want to retain the name till
421 runtime. This is supported if the name begins "ref:". */
423 else if (strncmp(user, "ref:", 4) == 0)
426 while (isspace(*user)) user++;
434 struct passwd *pw = getpwnam(user);
437 printf("\n*** User \"%s\" (specified in one of the Makefiles) does not "
438 "exist.\n Please review your build-time configuration.\n\n",
448 /* Use explicit group if set. */
452 while (isspace((unsigned char)(*group))) group++;
455 printf("\n*** EXIM_GROUP is defined as an empty string in one of "
456 "the files in the\n \"Local\" directory. ");
459 printf("If you want the Exim group to be taken from the\n "
460 "password data for the Exim user, just remove the EXIM_GROUP "
461 "setting.\n Otherwise, p");
463 else printf("EXIM_USER is defined numerically, so there is no"
464 "\n default for EXIM_GROUP and you must set it explicitly.\n P");
465 printf("lease review your build-time configuration.\n\n");
469 for (s = group; *s != 0; s++)
471 if (iscntrl((unsigned char)(*s)))
473 printf("\n*** EXIM_GROUP contains the control character 0x%02X in one "
474 "of the files\n in the \"Local\" directory. Please review your "
475 "build-time\n configuration.\n\n", *s);
480 /* Group name given. This may be by reference or to be looked up now,
483 if (strncmp(group, "ref:", 4) == 0)
486 while (isspace(*group)) group++;
490 else if (username != NULL)
495 else if (group[strspn(group, "0123456789")] == 0)
497 gid = (gid_t)atoi(group);
502 struct group *gr = getgrnam(group);
505 printf("\n*** Group \"%s\" (specified in one of the Makefiles) does "
506 "not exist.\n Please review your build-time configuration.\n\n",
514 /* Else trouble unless found in passwd file with user */
518 printf("\n*** No group set for Exim. Please review your build-time "
519 "configuration.\n\n");
523 /* security sanity checks
524 if ref: is being used, we can never be sure, but we can take reasonable
525 steps to filter out the most obvious ones. */
527 if ((!uid_not_set && uid == 0) ||
528 ((username != NULL) && (
529 (strcmp(username, "root") == 0) ||
530 (strcmp(username, "toor") == 0) )))
532 printf("\n*** Exim's internal user must not be root.\n\n");
536 /* Output user and group names or uid/gid. When names are set, uid/gid
537 are set to zero but will be replaced at runtime. */
539 if (username != NULL)
540 fprintf(new, "#define EXIM_USERNAME \"%s\"\n", username);
541 if (groupname != NULL)
542 fprintf(new, "#define EXIM_GROUPNAME \"%s\"\n", groupname);
544 fprintf(new, "#define EXIM_UID %d\n", (int)uid);
545 fprintf(new, "#define EXIM_GID %d\n", (int)gid);
549 /* CONFIGURE_OWNER and CONFIGURE_GROUP are special cases. We look in the
550 environment for first. If the value is not numeric, we look up the user or
551 group. A lot of this code is similar to that for EXIM_USER, but it's easier
552 to keep it separate. */
554 if (strcmp(name, "CONFIGURE_OWNER") == 0 ||
555 strcmp(name, "CONFIGURE_GROUP") == 0)
557 int isgroup = name[10] == 'G';
561 const char *username = NULL;
562 const char *user = getenv(name);
564 if (user == NULL) user = "";
565 while (isspace((unsigned char)(*user))) user++;
568 fprintf(new, "/* %s not set */\n", name);
572 for (s = user; *s != 0; s++)
574 if (iscntrl((unsigned char)(*s)))
576 printf("\n*** %s contains the control character 0x%02X in "
577 "one of the files\n in the \"Local\" directory. Please review "
578 "your build-time\n configuration.\n\n", name, *s);
583 /* Numeric uid given */
585 if (user[strspn(user, "0123456789")] == 0)
588 gid = (gid_t)atoi(user);
590 uid = (uid_t)atoi(user);
593 /* Name given. Normally, we look up the uid or gid right away. However,
594 people building binary distributions sometimes want to retain the name till
595 runtime. This is supported if the name begins "ref:". */
597 else if (strncmp(user, "ref:", 4) == 0)
600 while (isspace(*user)) user++;
605 struct group *gr = getgrnam(user);
608 printf("\n*** Group \"%s\" (specified in one of the Makefiles) does not "
609 "exist.\n Please review your build-time configuration.\n\n",
618 struct passwd *pw = getpwnam(user);
621 printf("\n*** User \"%s\" (specified in one of the Makefiles) does not "
622 "exist.\n Please review your build-time configuration.\n\n",
629 /* Output user and group names or uid/gid. When names are set, uid/gid
630 are set to zero but will be replaced at runtime. */
632 if (username != NULL)
635 fprintf(new, "#define CONFIGURE_GROUPNAME \"%s\"\n", username);
637 fprintf(new, "#define CONFIGURE_OWNERNAME \"%s\"\n", username);
641 fprintf(new, "#define CONFIGURE_GROUP %d\n", (int)gid);
643 fprintf(new, "#define CONFIGURE_OWNER %d\n", (int)uid);
647 /* FIXED_NEVER_USERS is another special case. Look up the uid values and
648 create suitable initialization data for a vector. */
650 if (strcmp(name, "FIXED_NEVER_USERS") == 0)
652 char *list = getenv("FIXED_NEVER_USERS");
655 fprintf(new, "#define FIXED_NEVER_USERS 0\n");
663 while (*p != 0) if (*p++ == ':') count++;
665 vector = malloc((count+1) * sizeof(uid_t));
666 vector[0] = (uid_t)count;
668 for (i = 1, j = 0; i <= count; list++, i++)
673 while (*list != 0 && *list != ':') list++;
674 strncpy(name, p, list-p);
681 else if (name[strspn(name, "0123456789")] == 0)
683 vector[j++] = (uid_t)atoi(name);
687 struct passwd *pw = getpwnam(name);
690 printf("\n*** User \"%s\" (specified for FIXED_NEVER_USERS in one of the Makefiles) does not "
691 "exist.\n Please review your build-time configuration.\n\n",
695 vector[j++] = pw->pw_uid;
698 fprintf(new, "#define FIXED_NEVER_USERS %d", j);
699 for (i = 0; i < j; i++) fprintf(new, ", %d", (unsigned int)vector[i]);
705 /* WITH_CONTENT_SCAN is another special case: it must be set if either it or
706 WITH_OLD_DEMIME is set. */
708 if (strcmp(name, "WITH_CONTENT_SCAN") == 0)
710 char *wcs = getenv("WITH_CONTENT_SCAN");
711 char *wod = getenv("WITH_OLD_DEMIME");
712 char *dcc = getenv("EXPERIMENTAL_DCC");
713 if (wcs != NULL || wod != NULL || dcc != NULL)
714 fprintf(new, "#define WITH_CONTENT_SCAN yes\n");
715 else fprintf(new, "/* WITH_CONTENT_SCAN not set */\n");
719 /* Otherwise, check whether a value exists in the environment. Remember if
720 it is an AUTH setting or SUPPORT_CRYPTEQ. */
722 if ((value = getenv(name)) != NULL)
725 len = 21 - (int)strlen(name);
727 if (strncmp(name, "AUTH_", 5) == 0) have_auth = 1;
728 if (strncmp(name, "SUPPORT_CRYPTEQ", 15) == 0) support_crypteq = 1;
730 /* The text value of LDAP_LIB_TYPE refers to a macro that gets set. */
732 if (strcmp(name, "LDAP_LIB_TYPE") == 0)
734 if (strcmp(value, "NETSCAPE") == 0 ||
735 strcmp(value, "UMICHIGAN") == 0 ||
736 strcmp(value, "OPENLDAP1") == 0 ||
737 strcmp(value, "OPENLDAP2") == 0 ||
738 strcmp(value, "SOLARIS") == 0 ||
739 strcmp(value, "SOLARIS7") == 0) /* Compatibility */
741 fprintf(new, "#define LDAP_LIB_%s\n", value);
745 printf("\n*** LDAP_LIB_TYPE=%s is not a recognized LDAP library type."
746 "\n*** Please review your build-time configuration.\n\n", value);
751 else if (strcmp(name, "RADIUS_LIB_TYPE") == 0)
753 if (strcmp(value, "RADIUSCLIENT") == 0 ||
754 strcmp(value, "RADIUSCLIENTNEW") == 0 ||
755 strcmp(value, "RADLIB") == 0)
757 fprintf(new, "#define RADIUS_LIB_%s\n", value);
761 printf("\n*** RADIUS_LIB_TYPE=%s is not a recognized RADIUS library type."
762 "\n*** Please review your build-time configuration.\n\n", value);
767 /* Other macros get set to the environment value. */
771 fprintf(new, "#define %s ", name);
772 while(len-- > 0) fputc(' ', new);
774 /* LOG_FILE_PATH is now messy because it can be a path containing %s or
775 it can be "syslog" or ":syslog" or "syslog:path" or even "path:syslog". */
777 if (strcmp(name, "LOG_FILE_PATH") == 0)
783 char *sss = strchr(ss, ':');
786 strncpy(buffer, ss, sss-ss);
787 buffer[sss-ss] = 0; /* For empty case */
789 else strcpy(buffer, ss);
790 pp = buffer + (int)strlen(buffer);
791 while (pp > buffer && isspace((unsigned char)pp[-1])) pp--;
793 if (buffer[0] != 0 && strcmp(buffer, "syslog") != 0)
794 check_percent_ess(buffer, name);
795 if (sss == NULL) break;
797 while (isspace((unsigned char)*ss)) ss++;
799 fprintf(new, "\"%s\"\n", value);
802 /* Timezone values HEADERS_CHARSET, TCP_WRAPPERS_DAEMON_NAME and
803 WHITELIST_D_MACROS get quoted */
805 else if (strcmp(name, "TIMEZONE_DEFAULT") == 0||
806 strcmp(name, "TCP_WRAPPERS_DAEMON_NAME") == 0||
807 strcmp(name, "HEADERS_CHARSET") == 0||
808 strcmp(name, "WHITELIST_D_MACROS") == 0)
809 fprintf(new, "\"%s\"\n", value);
811 /* For others, quote any paths and don't quote anything else */
815 if (value[0] == '/') fprintf(new, "\"%s\"\n", value);
816 else fprintf(new, "%s\n", value);
821 /* Value not defined in the environment; use the default */
826 while (*p == ' ' || *p == '\t') p++;
827 if (*p != '\n') fputs(buffer, new); else
830 if (strcmp(name, "BIN_DIRECTORY") == 0 ||
831 strcmp(name, "CONFIGURE_FILE") == 0)
833 printf("\n*** %s has not been defined in any of the Makefiles in the\n"
834 " \"Local\" directory. "
835 "Please review your build-time configuration.\n\n", name);
839 if (strcmp(name, "TIMEZONE_DEFAULT") == 0)
841 char *tz = getenv("TZ");
842 fprintf(new, "#define TIMEZONE_DEFAULT ");
843 if (tz == NULL) fprintf(new, "NULL\n"); else
844 fprintf(new, "\"%s\"\n", tz);
847 else fprintf(new, "/* %s not set */\n", name);
854 /* If any AUTH macros were defined, ensure that SUPPORT_CRYPTEQ is also
859 if (!support_crypteq) fprintf(new, "/* Force SUPPORT_CRYPTEQ for AUTH */\n"
860 "#define SUPPORT_CRYPTEQ\n");
865 fprintf(new, "\n/* End of config.h */\n");
870 /* End of buildconfig.c */