Blame m4/memchr.m4

Packit Service 991b93
# memchr.m4 serial 15
Packit Service 991b93
dnl Copyright (C) 2002-2004, 2009-2020 Free Software Foundation, Inc.
Packit aea12f
dnl This file is free software; the Free Software Foundation
Packit aea12f
dnl gives unlimited permission to copy and/or distribute it,
Packit aea12f
dnl with or without modifications, as long as this notice is preserved.
Packit aea12f
Packit aea12f
AC_DEFUN_ONCE([gl_FUNC_MEMCHR],
Packit aea12f
[
Packit aea12f
  AC_REQUIRE([AC_CANONICAL_HOST]) dnl for cross-compiles
Packit aea12f
Packit aea12f
  dnl Check for prerequisites for memory fence checks.
Packit aea12f
  gl_FUNC_MMAP_ANON
Packit aea12f
  AC_CHECK_HEADERS_ONCE([sys/mman.h])
Packit aea12f
  AC_CHECK_FUNCS_ONCE([mprotect])
Packit aea12f
Packit aea12f
  AC_REQUIRE([gl_HEADER_STRING_H_DEFAULTS])
Packit aea12f
  m4_ifdef([gl_FUNC_MEMCHR_OBSOLETE], [
Packit aea12f
    dnl These days, we assume memchr is present.  But if support for old
Packit aea12f
    dnl platforms is desired:
Packit aea12f
    AC_CHECK_FUNCS_ONCE([memchr])
Packit aea12f
    if test $ac_cv_func_memchr = no; then
Packit aea12f
      HAVE_MEMCHR=0
Packit aea12f
    fi
Packit aea12f
  ])
Packit aea12f
  if test $HAVE_MEMCHR = 1; then
Packit aea12f
    # Detect platform-specific bugs in some versions of glibc:
Packit aea12f
    # memchr should not dereference anything with length 0
Packit aea12f
    #   https://bugzilla.redhat.com/show_bug.cgi?id=499689
Packit aea12f
    # memchr should not dereference overestimated length after a match
Packit aea12f
    #   https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=521737
Packit aea12f
    #   https://sourceware.org/bugzilla/show_bug.cgi?id=10162
Packit aea12f
    # memchr should cast the second argument to 'unsigned char'.
Packit aea12f
    #   This bug exists in Android 4.3.
Packit aea12f
    # Assume that memchr works on platforms that lack mprotect.
Packit aea12f
    AC_CACHE_CHECK([whether memchr works], [gl_cv_func_memchr_works],
Packit aea12f
      [AC_RUN_IFELSE([AC_LANG_PROGRAM([[
Packit aea12f
#include <string.h>
Packit aea12f
#if HAVE_SYS_MMAN_H
Packit aea12f
# include <fcntl.h>
Packit aea12f
# include <unistd.h>
Packit aea12f
# include <sys/types.h>
Packit aea12f
# include <sys/mman.h>
Packit aea12f
# ifndef MAP_FILE
Packit aea12f
#  define MAP_FILE 0
Packit aea12f
# endif
Packit aea12f
#endif
Packit aea12f
]], [[
Packit aea12f
  int result = 0;
Packit aea12f
  char *fence = NULL;
Packit aea12f
#if HAVE_SYS_MMAN_H && HAVE_MPROTECT
Packit aea12f
# if HAVE_MAP_ANONYMOUS
Packit aea12f
  const int flags = MAP_ANONYMOUS | MAP_PRIVATE;
Packit aea12f
  const int fd = -1;
Packit aea12f
# else /* !HAVE_MAP_ANONYMOUS */
Packit aea12f
  const int flags = MAP_FILE | MAP_PRIVATE;
Packit aea12f
  int fd = open ("/dev/zero", O_RDONLY, 0666);
Packit aea12f
  if (fd >= 0)
Packit aea12f
# endif
Packit aea12f
    {
Packit aea12f
      int pagesize = getpagesize ();
Packit aea12f
      char *two_pages =
Packit aea12f
        (char *) mmap (NULL, 2 * pagesize, PROT_READ | PROT_WRITE,
Packit aea12f
                       flags, fd, 0);
Packit aea12f
      if (two_pages != (char *)(-1)
Packit aea12f
          && mprotect (two_pages + pagesize, pagesize, PROT_NONE) == 0)
Packit aea12f
        fence = two_pages + pagesize;
Packit aea12f
    }
Packit aea12f
#endif
Packit aea12f
  if (fence)
Packit aea12f
    {
Packit aea12f
      if (memchr (fence, 0, 0))
Packit aea12f
        result |= 1;
Packit aea12f
      strcpy (fence - 9, "12345678");
Packit aea12f
      if (memchr (fence - 9, 0, 79) != fence - 1)
Packit aea12f
        result |= 2;
Packit aea12f
      if (memchr (fence - 1, 0, 3) != fence - 1)
Packit aea12f
        result |= 4;
Packit aea12f
    }
Packit aea12f
  /* Test against bug on Android 4.3.  */
Packit aea12f
  {
Packit aea12f
    char input[3];
Packit aea12f
    input[0] = 'a';
Packit aea12f
    input[1] = 'b';
Packit aea12f
    input[2] = 'c';
Packit aea12f
    if (memchr (input, 0x789abc00 | 'b', 3) != input + 1)
Packit aea12f
      result |= 8;
Packit aea12f
  }
Packit aea12f
  return result;
Packit aea12f
]])],
Packit aea12f
         [gl_cv_func_memchr_works=yes],
Packit aea12f
         [gl_cv_func_memchr_works=no],
Packit aea12f
         [case "$host_os" in
Packit aea12f
                             # Guess no on Android.
Packit aea12f
            linux*-android*) gl_cv_func_memchr_works="guessing no" ;;
Packit aea12f
                             # Guess yes on native Windows.
Packit aea12f
            mingw*)          gl_cv_func_memchr_works="guessing yes" ;;
Packit Service 991b93
                             # If we don't know, obey --enable-cross-guesses.
Packit Service 991b93
            *)               gl_cv_func_memchr_works="$gl_cross_guess_normal" ;;
Packit aea12f
          esac
Packit aea12f
         ])
Packit aea12f
      ])
Packit aea12f
    case "$gl_cv_func_memchr_works" in
Packit aea12f
      *yes) ;;
Packit aea12f
      *) REPLACE_MEMCHR=1 ;;
Packit aea12f
    esac
Packit aea12f
  fi
Packit aea12f
])
Packit aea12f
Packit aea12f
# Prerequisites of lib/memchr.c.
Packit aea12f
AC_DEFUN([gl_PREREQ_MEMCHR], [
Packit aea12f
  AC_CHECK_HEADERS([bp-sym.h])
Packit aea12f
])