gen-scmconfig.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. /* Copyright 2003-2013, 2018, 2020-2022
  2. Free Software Foundation, Inc.
  3. This file is part of Guile.
  4. Guile is free software: you can redistribute it and/or modify it
  5. under the terms of the GNU Lesser General Public License as published
  6. by the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. Guile is distributed in the hope that it will be useful, but WITHOUT
  9. ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
  11. License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with Guile. If not, see
  14. <https://www.gnu.org/licenses/>. */
  15. /**********************************************************************
  16. Description of Guile's public config header mechanics:
  17. -----------------------------------------------------
  18. Guile has three core headers:
  19. config.h: Guile's private automatically generated configuration
  20. header -- generated by configure.in and autoheader. *NOT*
  21. installed during "make install" and so may not be referred to by
  22. any public headers.
  23. installed. It's only visible to the libguile sources
  24. themselves, and it includes config.h, the private config header.
  25. Among other things this file provides a place to make decisions
  26. based on the information gathered in config.h.
  27. libguile/scmconfig.h: Guile's public automatically generated
  28. configuration header -- generated at build time by concatenating
  29. the contents of libguile/scmconfig.h.top with the output from
  30. libguile/gen-scmconfig. gen-scmconfig bases its output on the
  31. information in the private config.h header, the contents of
  32. gen-scmconfig.h (which is created by configure.in from
  33. gen-scmconfig.h.in), and the information provided in this file,
  34. gen-scmconfig.c.
  35. libguile/scm.h: Guile's public core header. This file is
  36. installed and publically visible. It includes
  37. libguile/scmconfig.h, the public config header and provides a
  38. place to make decisions based on the information gathered in
  39. scmconfig.h to define things that other headers can depend on.
  40. Notes and guidelines:
  41. - use 1 and 0 for public #defines instead of "def and undef",
  42. i.e. use #define SCM_HAVE_FOO rather than just not defining
  43. SCM_HAVE_FOO whenever possible. See GNU Coding Guidelines for
  44. rationale. The only notable non-deprecated exception to this rule
  45. is GUILE_DEBUG which does not follow this convention in order to
  46. retain backward compatibility.
  47. - in the code below, be *VERY* careful not to use or rely on any
  48. runtime-dynamic information below. For example, you cannot use
  49. sizeof (FOO), but must use static information like SIZEOF_BAR
  50. (from config.h) or SCM_SIZEOF_BAZ (from scmconfig.h). This is
  51. because the gcc that is compiling gen-scmconfig.c and/or the
  52. machine that is running gen-scmconfig may not be the same
  53. compiler and/or hardware that will eventually be running Guile.
  54. (i.e. keep the cross-compilation case in mind).
  55. - try to avoid adding names to the public namespace when possible.
  56. Note in the code below, that in a number of cases, we detect a
  57. feature and based on that, we decide whether or not to print
  58. anything at all. This decreases the extraneous #defines and
  59. #ifdefery that we require in scmconfig.h
  60. - try to avoid adding any duplicate definitions to config.h and
  61. scmconfig.h. i.e. have just SCM_ENABLE_ELISP in scmconfig.h
  62. rather than ENABLE_ELISP in config.h and SCM_ENABLE_ELISP in
  63. scmconfig.h.
  64. - in cases where you need to communicate information from
  65. configure.in to gen-scmconfig.c, don't add an AC_DEFINE unless
  66. you need it for other purposes. Just add a suitable SCM_I_GSC_*
  67. variable to configure.in, set the value, AC_SUBST the value, and
  68. add an appropriate line to gen-scmconfig.h.in. All gen-scmconfig
  69. related AC_SUBST vars should be prefixed with SCM_I_GSC_.
  70. - make sure that anything that we explicitly typedef publically is
  71. prefixed with scm_t_. i.e. we used to typedef long to ptrdiff_t
  72. if we didn't detect ptrdiff_t, but this has been changed so that
  73. we typedef ptrdiff_t instead so that we won't conflict with
  74. any non-guile header definitions of the same type. For types
  75. like intptr_t and uintptr_t which we just try to detect and don't
  76. actually define, it's fine not to have a corresponding scm_t_
  77. type.
  78. - we now use SCM_SIZEOF_FOO != 0 rather than SCM_HAVE_FOO for any
  79. cases where the size might actually vary.
  80. Rationales (not finished):
  81. Why do we use a C program here rather than AC_OUTPUT_COMMANDS?
  82. --------------------------------------------------------------
  83. The main reason is that there are some values we would need
  84. access to at AC_OUTPUT_COMMANDs that are determined by configure
  85. but are not available at AC_OUTPUT time. The values are *only*
  86. available via config.h. We use gen-scmconfig so we can see those
  87. values and make decisions based on their settings.
  88. Why have gen-scmconfig.h.in?
  89. ----------------------------
  90. Without that header, we could end up needing multiple aliases for
  91. public settings like SCM_ENABLE_ELISP. We can't define
  92. SCM_ENABLE_ELISP in config.h since that header is private and any
  93. definition in scmconfig.h would conflict (#ifndef might be
  94. possible but runs the risk of conflicting directives), so a
  95. likely solution would be to AC_DEFINE([SCM_I_ENABLE_ELISP]), and
  96. then use SCM_I_ENABLE_ELISP in gen-scmconfig via config.h to
  97. determine whether or not to #define SCM_ENABLE_ELISP, but this
  98. leaves us with two #defined symbols for each public setting --
  99. better to just have one value (public or private) that all code
  100. uses.
  101. Having this header means we can AC_SUBST a value like
  102. SCM_I_GSC_ENABLE_ELISP and then set it in here via AC_OUTPUT
  103. substitutions, and gen-scmconfig can use that definition to
  104. determine whether or not to #define SCM_ENABLE_ELISP when
  105. generating scmconfig.h, and we end up with nothing extraneous
  106. added to config.h.
  107. **********************************************************************/
  108. #ifdef HAVE_CONFIG_H
  109. # include <config.h>
  110. #endif
  111. #include <stdint.h>
  112. #include <stddef.h>
  113. #include <stdio.h>
  114. #include <string.h>
  115. #include "gen-scmconfig.h"
  116. #define pf printf
  117. int
  118. main (int argc, char *argv[])
  119. {
  120. pf ("/* This file is automatically generated --"
  121. " see configure.in for details */\n"
  122. "\n"
  123. "#ifndef SCM_SCMCONFIG_H\n"
  124. "#define SCM_SCMCONFIG_H\n");
  125. /*** various important headers ***/
  126. pf ("\n");
  127. pf ("/* Important headers */\n");
  128. pf ("#include <stdint.h>\n");
  129. pf ("#include <stddef.h>\n");
  130. pf ("#include <limits.h>\n");
  131. #if HAVE_SYS_TIME_H
  132. pf ("#include <sys/time.h>\n");
  133. #else
  134. pf ("/* sys/time.h not available */\n");
  135. #endif
  136. pf ("#include <time.h>\n");
  137. pf("\n");
  138. pf ("#include <stdlib.h>\n");
  139. # ifdef HAVE_SYS_TYPES_H
  140. pf ("#include <sys/types.h>\n");
  141. # endif
  142. # ifdef HAVE_SYS_STDTYPES_H
  143. pf ("#include <sys/stdtypes.h>\n");
  144. # endif
  145. pf ("#include <stddef.h>\n");
  146. pf("\n");
  147. #ifdef HAVE_SYS_SELECT_H
  148. pf ("#define SCM_HAVE_SYS_SELECT_H 1 /* 0 or 1 */\n");
  149. #else
  150. pf ("#define SCM_HAVE_SYS_SELECT_H 0 /* 0 or 1 */\n");
  151. #endif
  152. #ifdef HAVE_WINSOCK2_H
  153. pf ("#define SCM_HAVE_WINSOCK2_H 1 /* 0 or 1 */\n");
  154. #else
  155. pf ("#define SCM_HAVE_WINSOCK2_H 0 /* 0 or 1 */\n");
  156. #endif
  157. /*** GUILE_DEBUG (defined or undefined) ***/
  158. pf ("\n");
  159. pf ("/* Define to include various undocumented debugging functions. */\n");
  160. if (SCM_I_GSC_GUILE_DEBUG)
  161. pf ("#define GUILE_DEBUG 1 /* defined or undefined */\n");
  162. else
  163. pf ("/* #undef GUILE_DEBUG */\n");
  164. /*** SCM_ENABLE_DEPRECATED (0 or 1) ***/
  165. pf ("\n");
  166. pf ("/* Set to 1 if you want to enable deprecated features. */\n");
  167. pf ("/* (value will be 0 or 1). */\n");
  168. pf ("#define SCM_ENABLE_DEPRECATED %d\n", SCM_I_GSC_ENABLE_DEPRECATED);
  169. /*** SCM_STACK_GROWS_UP (0 or 1) ***/
  170. pf ("\n");
  171. pf ("/* Set to 1 if the stack grows up, 0 otherwise. */\n");
  172. pf ("#define SCM_STACK_GROWS_UP %d /* 0 or 1 */\n",
  173. SCM_I_GSC_STACK_GROWS_UP);
  174. /*** SCM_C_INLINE (defined to appropriate string or undefined) ***/
  175. pf ("\n");
  176. pf ("/* C compiler's syntax for inline functions if any,\n"
  177. " otherwise undefined. */\n");
  178. if (SCM_I_GSC_C_INLINE)
  179. pf ("#define SCM_C_INLINE %s\n", SCM_I_GSC_C_INLINE);
  180. else
  181. pf ("/* #undef SCM_C_INLINE */\n");
  182. pf ("\n");
  183. pf ("/* Standard types. */\n");
  184. pf ("#define SCM_SIZEOF_CHAR %d\n", SIZEOF_CHAR);
  185. pf ("#define SCM_SIZEOF_UNSIGNED_CHAR %d\n", SIZEOF_UNSIGNED_CHAR);
  186. pf ("#define SCM_SIZEOF_SHORT %d\n", SIZEOF_SHORT);
  187. pf ("#define SCM_SIZEOF_UNSIGNED_SHORT %d\n", SIZEOF_UNSIGNED_SHORT);
  188. pf ("#define SCM_SIZEOF_LONG %d\n", SIZEOF_LONG);
  189. pf ("#define SCM_SIZEOF_UNSIGNED_LONG %d\n", SIZEOF_UNSIGNED_LONG);
  190. pf ("#define SCM_SIZEOF_INT %d\n", SIZEOF_INT);
  191. pf ("#define SCM_SIZEOF_UNSIGNED_INT %d\n", SIZEOF_UNSIGNED_INT);
  192. pf ("#define SCM_SIZEOF_SIZE_T %d\n", SIZEOF_SIZE_T);
  193. pf ("#define SCM_SIZEOF_LONG_LONG %d\n", SIZEOF_LONG_LONG);
  194. pf ("#define SCM_SIZEOF_UNSIGNED_LONG_LONG %d\n", SIZEOF_UNSIGNED_LONG_LONG);
  195. pf ("#define SCM_SIZEOF_INTMAX %d\n", SIZEOF_INTMAX_T);
  196. pf ("#define SCM_SIZEOF_SCM_T_PTRDIFF %d\n", SIZEOF_PTRDIFF_T);
  197. pf ("#define SCM_SIZEOF_INTPTR_T %d\n", SIZEOF_INTPTR_T);
  198. pf ("#define SCM_SIZEOF_UINTPTR_T %d\n", SIZEOF_UINTPTR_T);
  199. pf ("\n");
  200. pf ("/* same as POSIX \"struct timespec\" -- always defined */\n");
  201. #ifdef HAVE_SYSTEM_STRUCT_TIMESPEC
  202. pf ("typedef struct timespec scm_t_timespec;\n");
  203. #else
  204. pf ("/* POSIX.4 structure for a time value. This is like a `struct timeval'"
  205. " but has nanoseconds instead of microseconds. */\n");
  206. pf ("typedef struct\n"
  207. "{\n"
  208. " long int tv_sec; /* Seconds. */\n"
  209. " long int tv_nsec; /* Nanoseconds. */\n"
  210. "} scm_t_timespec;\n");
  211. #endif
  212. pf ("\n");
  213. pf ("/*** Threading model (scmconfig.h support not finished) ***/\n");
  214. pf ("/* Define to 1 if using pthread multithreading. */\n");
  215. pf ("#define SCM_USE_PTHREAD_THREADS %d /* 0 or 1 */\n",
  216. SCM_I_GSC_USE_PTHREAD_THREADS);
  217. pf ("/* Define to 1 if using one-thread 'multi'threading. */\n");
  218. pf ("#define SCM_USE_NULL_THREADS %d /* 0 or 1 */\n",
  219. SCM_I_GSC_USE_NULL_THREADS);
  220. pf ("/* Define to 1 if need braces around PTHREAD_ONCE_INIT (for Solaris). */\n");
  221. pf ("#define SCM_NEED_BRACES_ON_PTHREAD_ONCE_INIT %d /* 0 or 1 */\n",
  222. SCM_I_GSC_NEED_BRACES_ON_PTHREAD_ONCE_INIT);
  223. pf ("/* Define to 1 if need braces around PTHREAD_MUTEX_INITIALIZER\n"
  224. " (for IRIX with GCC) */\n");
  225. pf ("#define SCM_NEED_BRACES_ON_PTHREAD_MUTEX_INITIALIZER %d /* 0 or 1 */\n",
  226. SCM_I_GSC_NEED_BRACES_ON_PTHREAD_MUTEX_INITIALIZER);
  227. #ifdef HAVE_PTHREAD_SIGMASK
  228. pf ("#define SCM_HAVE_PTHREAD_SIGMASK 1 /* 0 or 1 */\n");
  229. #else
  230. pf ("#define SCM_HAVE_PTHREAD_SIGMASK 0 /* 0 or 1 */\n");
  231. #endif
  232. #ifdef HAVE_GC_PTHREAD_CANCEL
  233. pf ("#define SCM_HAVE_GC_PTHREAD_CANCEL 1 /* 0 or 1 */\n");
  234. #else
  235. pf ("#define SCM_HAVE_GC_PTHREAD_CANCEL 0 /* 0 or 1 */\n");
  236. #endif
  237. #ifdef HAVE_GC_PTHREAD_EXIT
  238. pf ("#define SCM_HAVE_GC_PTHREAD_EXIT 1 /* 0 or 1 */\n");
  239. #else
  240. pf ("#define SCM_HAVE_GC_PTHREAD_EXIT 0 /* 0 or 1 */\n");
  241. #endif
  242. #ifdef HAVE_GC_PTHREAD_SIGMASK
  243. pf ("#define SCM_HAVE_GC_PTHREAD_SIGMASK 1 /* 0 or 1 */\n");
  244. #else
  245. pf ("#define SCM_HAVE_GC_PTHREAD_SIGMASK 0 /* 0 or 1 */\n");
  246. #endif
  247. pf ("\n\n/*** File system access ***/\n");
  248. pf ("/* Define to 1 if `struct dirent64' is available. */\n");
  249. pf ("#define SCM_HAVE_STRUCT_DIRENT64 %d /* 0 or 1 */\n",
  250. SCM_I_GSC_HAVE_STRUCT_DIRENT64);
  251. pf ("/* Define to 1 if `readdir64_r ()' is available. */\n");
  252. #ifdef HAVE_READDIR64_R
  253. pf ("#define SCM_HAVE_READDIR64_R 1 /* 0 or 1 */\n");
  254. #else
  255. pf ("#define SCM_HAVE_READDIR64_R 0 /* 0 or 1 */\n");
  256. #endif
  257. /* Arrange so that we have a file offset type that reflects the one
  258. used when compiling Guile, regardless of what the application's
  259. `_FILE_OFFSET_BITS' says. See
  260. http://lists.gnu.org/archive/html/bug-guile/2009-06/msg00018.html
  261. for the original bug report.
  262. Note that we can't define `scm_t_off' in terms of `off_t' or
  263. `off64_t' because they may or may not be available depending on
  264. how the application that uses Guile is compiled. */
  265. #if defined GUILE_USE_64_CALLS && defined HAVE_STAT64
  266. pf ("typedef int64_t scm_t_off;\n");
  267. pf ("#define SCM_T_OFF_MAX INT64_MAX\n");
  268. pf ("#define SCM_T_OFF_MIN INT64_MIN\n");
  269. #elif SIZEOF_OFF_T == SIZEOF_INT
  270. pf ("typedef int scm_t_off;\n");
  271. pf ("#define SCM_T_OFF_MAX INT_MAX\n");
  272. pf ("#define SCM_T_OFF_MIN INT_MIN\n");
  273. #else
  274. pf ("typedef long int scm_t_off;\n");
  275. pf ("#define SCM_T_OFF_MAX LONG_MAX\n");
  276. pf ("#define SCM_T_OFF_MIN LONG_MIN\n");
  277. #endif
  278. pf ("/* Define to 1 if the compiler supports the "
  279. "`__thread' storage class. */\n");
  280. if (SCM_I_GSC_HAVE_THREAD_STORAGE_CLASS)
  281. pf ("#define SCM_HAVE_THREAD_STORAGE_CLASS\n");
  282. else
  283. pf ("/* #undef SCM_HAVE_THREAD_STORAGE_CLASS */\n");
  284. #ifdef USE_DLL_IMPORT
  285. pf ("\n");
  286. pf ("/* Define some additional CPP macros on Win32 platforms. */\n");
  287. pf ("# define __REGEX_IMPORT__ 1\n");
  288. pf ("# define __CRYPT_IMPORT__ 1\n");
  289. pf ("# define __READLINE_IMPORT__ 1\n");
  290. pf ("# define QT_IMPORT 1\n");
  291. #endif
  292. pf ("\n");
  293. pf ("/* Constants from uniconv.h. */\n");
  294. pf ("#define SCM_ICONVEH_ERROR %d\n", SCM_I_GSC_ICONVEH_ERROR);
  295. pf ("#define SCM_ICONVEH_QUESTION_MARK %d\n",
  296. SCM_I_GSC_ICONVEH_QUESTION_MARK);
  297. pf ("#define SCM_ICONVEH_ESCAPE_SEQUENCE %d\n",
  298. SCM_I_GSC_ICONVEH_ESCAPE_SEQUENCE);
  299. pf ("\n");
  300. pf ("/* Define to 1 if there is an auxiliary stack, as in ia64. */\n");
  301. pf ("#define SCM_HAVE_AUXILIARY_STACK %d\n", SCM_I_GSC_HAVE_AUXILIARY_STACK);
  302. pf ("\n");
  303. pf ("/* Define to 1 to use mini GMP. */\n");
  304. #if SCM_I_GSC_ENABLE_MINI_GMP == 1
  305. pf ("#define SCM_ENABLE_MINI_GMP 1\n");
  306. #else
  307. pf ("#define SCM_ENABLE_MINI_GMP 0\n");
  308. #endif
  309. printf ("#endif\n");
  310. return 0;
  311. }