xref: /php-src/Zend/Zend.m4 (revision 868257a3)
1dnl This file contains Zend specific autoconf macros.
2
3dnl
4dnl ZEND_CHECK_FLOAT_PRECISION
5dnl
6dnl x87 floating point internal precision control checks
7dnl See: http://wiki.php.net/rfc/rounding
8dnl
9AC_DEFUN([ZEND_CHECK_FLOAT_PRECISION],[
10  AC_MSG_CHECKING([for usable _FPU_SETCW])
11  AC_LINK_IFELSE([AC_LANG_PROGRAM([[
12    #include <fpu_control.h>
13  ]],[[
14    fpu_control_t fpu_oldcw, fpu_cw;
15    volatile double result;
16    double a = 2877.0;
17    volatile double b = 1000000.0;
18
19    _FPU_GETCW(fpu_oldcw);
20    fpu_cw = (fpu_oldcw & ~_FPU_EXTENDED & ~_FPU_SINGLE) | _FPU_DOUBLE;
21    _FPU_SETCW(fpu_cw);
22    result = a / b;
23    _FPU_SETCW(fpu_oldcw);
24  ]])],[ac_cfp_have__fpu_setcw=yes],[ac_cfp_have__fpu_setcw=no])
25  if test "$ac_cfp_have__fpu_setcw" = "yes" ; then
26    AC_DEFINE(HAVE__FPU_SETCW, 1, [whether _FPU_SETCW is present and usable])
27    AC_MSG_RESULT(yes)
28  else
29    AC_MSG_RESULT(no)
30  fi
31
32  AC_MSG_CHECKING([for usable fpsetprec])
33  AC_LINK_IFELSE([AC_LANG_PROGRAM([[
34    #include <machine/ieeefp.h>
35  ]],[[
36    fp_prec_t fpu_oldprec;
37    volatile double result;
38    double a = 2877.0;
39    volatile double b = 1000000.0;
40
41    fpu_oldprec = fpgetprec();
42    fpsetprec(FP_PD);
43    result = a / b;
44    fpsetprec(fpu_oldprec);
45  ]])], [ac_cfp_have_fpsetprec=yes], [ac_cfp_have_fpsetprec=no])
46  if test "$ac_cfp_have_fpsetprec" = "yes" ; then
47    AC_DEFINE(HAVE_FPSETPREC, 1, [whether fpsetprec is present and usable])
48    AC_MSG_RESULT(yes)
49  else
50    AC_MSG_RESULT(no)
51  fi
52
53  AC_MSG_CHECKING([for usable _controlfp])
54  AC_LINK_IFELSE([AC_LANG_PROGRAM([[
55    #include <float.h>
56  ]],[[
57    unsigned int fpu_oldcw;
58    volatile double result;
59    double a = 2877.0;
60    volatile double b = 1000000.0;
61
62    fpu_oldcw = _controlfp(0, 0);
63    _controlfp(_PC_53, _MCW_PC);
64    result = a / b;
65    _controlfp(fpu_oldcw, _MCW_PC);
66  ]])], [ac_cfp_have__controlfp=yes], [ac_cfp_have__controlfp=no])
67  if test "$ac_cfp_have__controlfp" = "yes" ; then
68    AC_DEFINE(HAVE__CONTROLFP, 1, [whether _controlfp is present usable])
69    AC_MSG_RESULT(yes)
70  else
71    AC_MSG_RESULT(no)
72  fi
73
74  AC_MSG_CHECKING([for usable _controlfp_s])
75  AC_LINK_IFELSE([AC_LANG_PROGRAM([[
76   #include <float.h>
77  ]],[[
78    unsigned int fpu_oldcw, fpu_cw;
79    volatile double result;
80    double a = 2877.0;
81    volatile double b = 1000000.0;
82
83    _controlfp_s(&fpu_cw, 0, 0);
84    fpu_oldcw = fpu_cw;
85    _controlfp_s(&fpu_cw, _PC_53, _MCW_PC);
86    result = a / b;
87    _controlfp_s(&fpu_cw, fpu_oldcw, _MCW_PC);
88  ]])], [ac_cfp_have__controlfp_s=yes], [ac_cfp_have__controlfp_s=no])
89  if test "$ac_cfp_have__controlfp_s" = "yes" ; then
90    AC_DEFINE(HAVE__CONTROLFP_S, 1, [whether _controlfp_s is present and usable])
91    AC_MSG_RESULT(yes)
92  else
93    AC_MSG_RESULT(no)
94  fi
95
96  AC_MSG_CHECKING([whether FPU control word can be manipulated by inline assembler])
97  AC_LINK_IFELSE([AC_LANG_PROGRAM([[
98    /* nothing */
99  ]],[[
100    unsigned int oldcw, cw;
101    volatile double result;
102    double a = 2877.0;
103    volatile double b = 1000000.0;
104
105    __asm__ __volatile__ ("fnstcw %0" : "=m" (*&oldcw));
106    cw = (oldcw & ~0x0 & ~0x300) | 0x200;
107    __asm__ __volatile__ ("fldcw %0" : : "m" (*&cw));
108
109    result = a / b;
110
111    __asm__ __volatile__ ("fldcw %0" : : "m" (*&oldcw));
112  ]])], [ac_cfp_have_fpu_inline_asm_x86=yes], [ac_cfp_have_fpu_inline_asm_x86=no])
113  if test "$ac_cfp_have_fpu_inline_asm_x86" = "yes" ; then
114    AC_DEFINE(HAVE_FPU_INLINE_ASM_X86, 1, [whether FPU control word can be manipulated by inline assembler])
115    AC_MSG_RESULT(yes)
116  else
117    AC_MSG_RESULT(no)
118  fi
119])
120
121dnl
122dnl LIBZEND_BASIC_CHECKS
123dnl
124dnl Basic checks specific for the Zend engine library.
125dnl
126AC_DEFUN([LIBZEND_BASIC_CHECKS],[
127AC_REQUIRE([AC_PROG_CC])
128
129AC_CHECK_HEADERS([cpuid.h])
130
131dnl
132dnl LIBZEND_DLSYM_CHECK
133dnl
134dnl Ugly hack to check if dlsym() requires a leading underscore in symbol name.
135dnl
136AC_DEFUN([LIBZEND_DLSYM_CHECK],[
137AC_MSG_CHECKING([whether dlsym() requires a leading underscore in symbol names])
138_LT_AC_TRY_DLOPEN_SELF([
139  AC_MSG_RESULT(no)
140], [
141  AC_MSG_RESULT(yes)
142  AC_DEFINE(DLSYM_NEEDS_UNDERSCORE, 1, [Define if dlsym() requires a leading underscore in symbol names. ])
143], [
144  AC_MSG_RESULT(no)
145], [])
146])
147
148dnl Checks for library functions.
149AC_CHECK_FUNCS(getpid kill pthread_getattr_np pthread_attr_get_np pthread_get_stackaddr_np pthread_attr_getstack pthread_stackseg_np gettid)
150
151dnl Check for sigsetjmp. If it's defined as a macro, AC_CHECK_FUNCS won't work.
152AC_CHECK_FUNCS([sigsetjmp],,
153  [AC_CHECK_DECL([sigsetjmp],
154    [AC_DEFINE([HAVE_SIGSETJMP], [1])],,
155    [#include <setjmp.h>])])
156
157dnl Test whether the stack grows downwards
158dnl Assumes contiguous stack
159AC_MSG_CHECKING(whether the stack grows downwards)
160
161AC_RUN_IFELSE([AC_LANG_SOURCE([[
162#include <stdint.h>
163
164int (*volatile f)(uintptr_t);
165
166int stack_grows_downwards(uintptr_t arg) {
167    int local;
168    return (uintptr_t)&local < arg;
169}
170
171int main(void) {
172    int local;
173
174    f = stack_grows_downwards;
175    return f((uintptr_t)&local) ? 0 : 1;
176}
177]])], [
178  AC_DEFINE([ZEND_CHECK_STACK_LIMIT], 1, [Define if checking the stack limit is supported])
179  AC_MSG_RESULT(yes)
180], [
181  AC_MSG_RESULT(no)
182], [
183  AC_MSG_RESULT(no)
184])
185
186ZEND_CHECK_FLOAT_PRECISION
187])
188
189dnl
190dnl LIBZEND_OTHER_CHECKS
191dnl
192AC_DEFUN([LIBZEND_OTHER_CHECKS],[
193
194AC_MSG_CHECKING(whether to enable thread-safety)
195AC_MSG_RESULT($ZEND_ZTS)
196
197AC_MSG_CHECKING(whether to enable Zend debugging)
198AC_MSG_RESULT($ZEND_DEBUG)
199
200if test "$ZEND_DEBUG" = "yes"; then
201  AC_DEFINE(ZEND_DEBUG,1,[ ])
202  echo " $CFLAGS" | grep ' -g' >/dev/null || DEBUG_CFLAGS="-g"
203  if test "$CFLAGS" = "-g -O2"; then
204  	CFLAGS=-g
205  fi
206else
207  AC_DEFINE(ZEND_DEBUG,0,[ ])
208fi
209
210test -n "$GCC" && CFLAGS="-Wall -Wextra -Wno-unused-parameter -Wno-sign-compare $CFLAGS"
211dnl Check if compiler supports -Wno-clobbered (only GCC)
212AX_CHECK_COMPILE_FLAG([-Wno-clobbered], CFLAGS="-Wno-clobbered $CFLAGS", , [-Werror])
213dnl Check for support for implicit fallthrough level 1, also add after previous CFLAGS as level 3 is enabled in -Wextra
214AX_CHECK_COMPILE_FLAG([-Wimplicit-fallthrough=1], CFLAGS="$CFLAGS -Wimplicit-fallthrough=1", , [-Werror])
215AX_CHECK_COMPILE_FLAG([-Wduplicated-cond], CFLAGS="-Wduplicated-cond $CFLAGS", , [-Werror])
216AX_CHECK_COMPILE_FLAG([-Wlogical-op], CFLAGS="-Wlogical-op $CFLAGS", , [-Werror])
217AX_CHECK_COMPILE_FLAG([-Wformat-truncation], CFLAGS="-Wformat-truncation $CFLAGS", , [-Werror])
218AX_CHECK_COMPILE_FLAG([-Wstrict-prototypes], CFLAGS="-Wstrict-prototypes $CFLAGS", , [-Werror])
219AX_CHECK_COMPILE_FLAG([-fno-common], CFLAGS="-fno-common $CFLAGS", , [-Werror])
220
221test -n "$DEBUG_CFLAGS" && CFLAGS="$CFLAGS $DEBUG_CFLAGS"
222
223if test "$ZEND_ZTS" = "yes"; then
224  AC_DEFINE(ZTS,1,[ ])
225  CFLAGS="$CFLAGS -DZTS"
226fi
227
228dnl Test and set the alignment define for ZEND_MM. This also does the
229dnl logarithmic test for ZEND_MM.
230AC_MSG_CHECKING(for MM alignment and log values)
231
232AC_RUN_IFELSE([AC_LANG_SOURCE([[
233#include <stdio.h>
234#include <stdlib.h>
235
236typedef union _mm_align_test {
237  void *ptr;
238  double dbl;
239  long lng;
240} mm_align_test;
241
242#if (defined (__GNUC__) && __GNUC__ >= 2)
243#define ZEND_MM_ALIGNMENT (__alignof__ (mm_align_test))
244#else
245#define ZEND_MM_ALIGNMENT (sizeof(mm_align_test))
246#endif
247
248int main(void)
249{
250  size_t i = ZEND_MM_ALIGNMENT;
251  int zeros = 0;
252  FILE *fp;
253
254  while (i & ~0x1) {
255    zeros++;
256    i = i >> 1;
257  }
258
259  fp = fopen("conftest.zend", "w");
260  fprintf(fp, "(size_t)%zu (size_t)%d %d\n", ZEND_MM_ALIGNMENT, zeros, ZEND_MM_ALIGNMENT < 4);
261  fclose(fp);
262
263  return 0;
264}
265]])], [
266  LIBZEND_MM_ALIGN=`cat conftest.zend | cut -d ' ' -f 1`
267  LIBZEND_MM_ALIGN_LOG2=`cat conftest.zend | cut -d ' ' -f 2`
268  LIBZEND_MM_NEED_EIGHT_BYTE_REALIGNMENT=`cat conftest.zend | cut -d ' ' -f 3`
269  AC_DEFINE_UNQUOTED(ZEND_MM_ALIGNMENT, $LIBZEND_MM_ALIGN, [ ])
270  AC_DEFINE_UNQUOTED(ZEND_MM_ALIGNMENT_LOG2, $LIBZEND_MM_ALIGN_LOG2, [ ])
271  AC_DEFINE_UNQUOTED(ZEND_MM_NEED_EIGHT_BYTE_REALIGNMENT, $LIBZEND_MM_NEED_EIGHT_BYTE_REALIGNMENT, [ ])
272], [], [
273  dnl Cross compilation needs something here.
274  AC_DEFINE(ZEND_MM_ALIGNMENT, 8, [ ])
275  AC_DEFINE(ZEND_MM_ALIGNMENT_LOG2, 3, [ ])
276  AC_DEFINE(ZEND_MM_NEED_EIGHT_BYTE_REALIGNMENT, 0, [ ])
277])
278
279AC_MSG_RESULT(done)
280
281AC_CHECK_FUNCS(mremap)
282
283AC_ARG_ENABLE([zend-signals],
284  [AS_HELP_STRING([--disable-zend-signals],
285    [whether to enable zend signal handling])],
286  [ZEND_SIGNALS=$enableval],
287  [ZEND_SIGNALS=yes])
288
289AC_CHECK_FUNCS([sigaction], [], [
290  ZEND_SIGNALS=no
291])
292if test "$ZEND_SIGNALS" = "yes"; then
293	AC_DEFINE(ZEND_SIGNALS, 1, [Use zend signal handling])
294	CFLAGS="$CFLAGS -DZEND_SIGNALS"
295fi
296
297AC_MSG_CHECKING(whether to enable zend signal handling)
298AC_MSG_RESULT($ZEND_SIGNALS)
299
300dnl Don't enable Zend Max Execution Timers by default until PHP 8.3 to not break the ABI
301AC_ARG_ENABLE([zend-max-execution-timers],
302  [AS_HELP_STRING([--enable-zend-max-execution-timers],
303    [whether to enable zend max execution timers])],
304    [ZEND_MAX_EXECUTION_TIMERS=$enableval],
305    [ZEND_MAX_EXECUTION_TIMERS=$ZEND_ZTS])
306
307AS_CASE(["$host_alias"], [*linux*|*freebsd*], [], [ZEND_MAX_EXECUTION_TIMERS='no'])
308
309PHP_CHECK_FUNC(timer_create, rt)
310if test "$ac_cv_func_timer_create" != "yes"; then
311  ZEND_MAX_EXECUTION_TIMERS='no'
312fi
313
314if test "$ZEND_MAX_EXECUTION_TIMERS" = "yes"; then
315  AC_DEFINE(ZEND_MAX_EXECUTION_TIMERS, 1, [Use zend max execution timers])
316  CFLAGS="$CFLAGS -DZEND_MAX_EXECUTION_TIMERS"
317fi
318
319AC_MSG_CHECKING(whether to enable zend max execution timers)
320AC_MSG_RESULT($ZEND_MAX_EXECUTION_TIMERS)
321
322])
323
324AC_ARG_ENABLE([gcc-global-regs],
325  [AS_HELP_STRING([--disable-gcc-global-regs],
326    [whether to enable GCC global register variables])],
327  [ZEND_GCC_GLOBAL_REGS=$enableval],
328  [ZEND_GCC_GLOBAL_REGS=yes])
329
330AC_MSG_CHECKING(for global register variables support)
331if test "$ZEND_GCC_GLOBAL_REGS" != "no"; then
332  AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
333#if defined(__GNUC__)
334# define ZEND_GCC_VERSION (__GNUC__ * 1000 + __GNUC_MINOR__)
335#else
336# define ZEND_GCC_VERSION 0
337#endif
338#if defined(__GNUC__) && ZEND_GCC_VERSION >= 4008 && defined(i386)
339# define ZEND_VM_FP_GLOBAL_REG "%esi"
340# define ZEND_VM_IP_GLOBAL_REG "%edi"
341#elif defined(__GNUC__) && ZEND_GCC_VERSION >= 4008 && defined(__x86_64__)
342# define ZEND_VM_FP_GLOBAL_REG "%r14"
343# define ZEND_VM_IP_GLOBAL_REG "%r15"
344#elif defined(__GNUC__) && ZEND_GCC_VERSION >= 4008 && defined(__powerpc64__)
345# define ZEND_VM_FP_GLOBAL_REG "r28"
346# define ZEND_VM_IP_GLOBAL_REG "r29"
347#elif defined(__IBMC__) && ZEND_GCC_VERSION >= 4002 && defined(__powerpc64__)
348# define ZEND_VM_FP_GLOBAL_REG "r28"
349# define ZEND_VM_IP_GLOBAL_REG "r29"
350#elif defined(__GNUC__) && ZEND_GCC_VERSION >= 4008 && defined(__aarch64__)
351# define ZEND_VM_FP_GLOBAL_REG "x27"
352# define ZEND_VM_IP_GLOBAL_REG "x28"
353#elif defined(__GNUC__) && ZEND_GCC_VERSION >= 4008 && defined(__riscv) && __riscv_xlen == 64
354# define ZEND_VM_FP_GLOBAL_REG "x18"
355# define ZEND_VM_IP_GLOBAL_REG "x19"
356#else
357# error "global register variables are not supported"
358#endif
359typedef int (*opcode_handler_t)(void);
360register void *FP  __asm__(ZEND_VM_FP_GLOBAL_REG);
361register const opcode_handler_t *IP __asm__(ZEND_VM_IP_GLOBAL_REG);
362int emu(const opcode_handler_t *ip, void *fp) {
363	const opcode_handler_t *orig_ip = IP;
364	void *orig_fp = FP;
365	IP = ip;
366	FP = fp;
367	while ((*ip)());
368	FP = orig_fp;
369	IP = orig_ip;
370}
371  ]], [[
372  ]])], [
373    ZEND_GCC_GLOBAL_REGS=yes
374  ], [
375    ZEND_GCC_GLOBAL_REGS=no
376  ])
377fi
378if test "$ZEND_GCC_GLOBAL_REGS" = "yes"; then
379  AC_DEFINE([HAVE_GCC_GLOBAL_REGS], 1, [Define if the target system has support for global register variables])
380fi
381AC_MSG_RESULT($ZEND_GCC_GLOBAL_REGS)
382
383dnl Check whether __cpuid_count is available.
384AC_CACHE_CHECK(whether __cpuid_count is available, ac_cv_cpuid_count_available, [
385AC_LINK_IFELSE([AC_LANG_PROGRAM([[
386  #include <cpuid.h>
387]], [[
388  unsigned eax, ebx, ecx, edx;
389  __cpuid_count(0, 0, eax, ebx, ecx, edx);
390]])], [
391  ac_cv_cpuid_count_available=yes
392], [
393  ac_cv_cpuid_count_available=no
394])])
395if test "$ac_cv_cpuid_count_available" = "yes"; then
396  AC_DEFINE([HAVE_CPUID_COUNT], 1, [whether __cpuid_count is available])
397fi
398