Blame m4/memchr.m4

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