diff options
| author | Robin Haberkorn <robin.haberkorn@googlemail.com> | 2017-03-03 15:13:30 +0100 | 
|---|---|---|
| committer | Robin Haberkorn <robin.haberkorn@googlemail.com> | 2017-03-03 15:32:57 +0100 | 
| commit | 0ad317ec16fa836321617c10a8c6ba5c70f156b8 (patch) | |
| tree | ca820f937a238f9c278b96c236f87934edc07c01 | |
| parent | f90b197ab83dd59b230ffca21af2aa76a91d0320 (diff) | |
build system portability fixes
 * especially to improve building on FreeBSD 11
 * We need GNU Make, yet alone because Scintilla/Scinterm
   needs it. We now document that dependency and added
   an Autoconf check from the autoconf-archive.
   We make sure that the build process is invoked with GNU make
   by generating only GNUmakefiles.
   The Makefile.am files have not been renamed, so this
   change can be rolled back easily.
 * Some GNU-Make-specific autoreconf warnings have still been
   resolved. But not all of them, as this would have been
   unelegant and we need GNU Make anyway.
 * Declare ACLOCAL_AMFLAGS to appease autoreconf
 * Added an explicit check for C++11 from the autoconf-archives.
   In general we should support building with every C++11 compiler
   that is sufficiently GNU-like.
 * Do not use `sed` for inplace editing, as different sed-implementations
   have mutually incompatible syntax for this.
   Instead of declaring and checking a dependency on GNU sed,
   we simply use SciTECO for the editing task.
   This improves code portability on BSDs.
 * Similarily, BSD/POSIX `cmp` is supported now.
   This fixes the test suite on BSD without declaring a
   dependency on the GNU coreutils.
 * Simplified sciteco-wrapper generation.
| -rw-r--r-- | .gitignore | 1 | ||||
| -rw-r--r-- | INSTALL | 5 | ||||
| -rw-r--r-- | Makefile.am | 4 | ||||
| -rw-r--r-- | bootstrap.am | 15 | ||||
| -rw-r--r-- | configure.ac | 26 | ||||
| -rw-r--r-- | doc/Makefile.am | 22 | ||||
| -rw-r--r-- | m4/ax_check_gnu_make.m4 | 84 | ||||
| -rw-r--r-- | m4/ax_cxx_compile_stdcxx.m4 | 982 | ||||
| -rw-r--r-- | src/Makefile.am | 9 | ||||
| -rw-r--r-- | tests/testsuite.at | 2 | 
10 files changed, 1122 insertions, 28 deletions
| @@ -15,6 +15,7 @@  *.a  *.o  *.exe +GNUmakefile  Makefile  Makefile.in  stamp-* @@ -13,7 +13,10 @@ without warranty of any kind.  SciTECO Build and Runtime Dependencies  ====================================== - * Autotools, GNU C/C++ (v4.4 or later) or LLVM/gcc or LLVM/Clang + * Autotools and an UNIX-like environment + * GNU Make + * A C++11 compiler, e.g. GNU C/C++ (v4.4 or later) or +   LLVM/gcc or LLVM/Clang   * Glib 2 as a cross-platform runtime library     (v2.28 or later on Unix, v2.34 or later for MinGW):     https://developer.gnome.org/glib/ diff --git a/Makefile.am b/Makefile.am index 0c99647..cd36bca 100644 --- a/Makefile.am +++ b/Makefile.am @@ -1,3 +1,7 @@ +# Required in addition to AC_CONFIG_MACRO_DIR() to +# silence libtoolize: +ACLOCAL_AMFLAGS = -I m4 +  SUBDIRS = lib src doc tests  dist_pkgdata_DATA = sample.teco_ini diff --git a/bootstrap.am b/bootstrap.am index bc6f721..1ca730e 100644 --- a/bootstrap.am +++ b/bootstrap.am @@ -18,6 +18,14 @@ SCITECO_MINIMAL = @SCITECO@  SCITECO_FULL    = @SCITECO@  endif +# Path of installed `sciteco` and `sciteco-wrapper` binaries, +# taking --program-prefix into account. +# These variables MUST NOT be put in single-quotes. +SCITECO_INSTALLED = \ +	$(bindir)/`echo sciteco | @SED@ '$(transform)'`$(EXEEXT) +SCITECO_WRAPPER_INSTALLED = \ +	$(libexecdir)/`echo sciteco-wrapper | @SED@ '$(transform)'`$(EXEEXT) +  SUBST_MACRO = eb$<\e \  	      <fs@PACKAGE^Q@\e@PACKAGE@\e;>j \  	      <fs@PACKAGE_NAME^Q@\e@PACKAGE_NAME@\e;>j \ @@ -33,8 +41,9 @@ SUBST_MACRO = eb$<\e \  	      <fs@DATE^Q@\e^EQ.d\e; -D>j \  	      ew$@\e -# Don't try to generate config.h: -%/config.h: ; +# The SciTECO-based substitutor must not process config.h.in. +@top_srcdir@/config.h: ; -% : %.in +SUFFIXES = .in +.in:  	$(SCITECO_MINIMAL) -e $$'$(SUBST_MACRO)' diff --git a/configure.ac b/configure.ac index c8dab97..928749b 100644 --- a/configure.ac +++ b/configure.ac @@ -53,12 +53,24 @@ canonicalize() {  # Checks for programs.  LT_INIT  AC_PROG_CXX +AX_CXX_COMPILE_STDCXX(11, noext, mandatory)  AC_PROG_CC  AC_PROG_CC_C99 -AC_PROG_GREP  AC_PROG_SED  AC_PROG_INSTALL +# Mainly because of Scintilla, which depends on +# GNU Make. +# The main build system could be made POSIX-Make +# compliant, but that would complicate it and +# seems unnecessary. +# We prevent the main build system from being +# invoked with another Make by generating GNUmakefiles. +AX_CHECK_GNU_MAKE +if [[ x$ax_cv_gnu_make_command = x ]]; then +	AC_MSG_ERROR([GNU Make required!]) +fi +  # Additionally required by the Scintilla build process:  AC_CHECK_TOOL(AR, ar)  AC_PROG_RANLIB @@ -352,9 +364,11 @@ AC_ARG_ENABLE(static-executables,  	[static_executables=$enableval], [static_executables=no])  AM_CONDITIONAL(STATIC_EXECUTABLES, [test x$static_executables = xyes]) -AC_CONFIG_FILES([Makefile distribute.mk]) -AC_CONFIG_FILES([src/Makefile src/interface-gtk/Makefile src/interface-curses/Makefile]) -AC_CONFIG_FILES([lib/Makefile]) -AC_CONFIG_FILES([doc/Makefile doc/Doxyfile]) -AC_CONFIG_FILES([tests/Makefile tests/atlocal]) +AC_CONFIG_FILES([GNUmakefile:Makefile.in src/GNUmakefile:src/Makefile.in] +                [src/interface-gtk/GNUmakefile:src/interface-gtk/Makefile.in] +                [src/interface-curses/GNUmakefile:src/interface-curses/Makefile.in] +                [lib/GNUmakefile:lib/Makefile.in] +                [doc/GNUmakefile:doc/Makefile.in doc/Doxyfile] +                [tests/GNUmakefile:tests/Makefile.in tests/atlocal]) +AC_CONFIG_FILES([distribute.mk], [chmod +x distribute.mk])  AC_OUTPUT diff --git a/doc/Makefile.am b/doc/Makefile.am index 45a1428..ad60d7c 100644 --- a/doc/Makefile.am +++ b/doc/Makefile.am @@ -1,5 +1,5 @@ -# Make sure that $BOOTSTRAP_SCITECO and the SciTECO-based -# substitutor is available. +# Make sure that $(SCITECO_FULL) and the SciTECO-based +# substitutor for *.in is available.  include $(top_srcdir)/bootstrap.am  # tedoc is a code documentation tool for SciTECO commands @@ -9,9 +9,9 @@ include $(top_srcdir)/bootstrap.am  # preprocessor to manpage templates.  dist_bin_SCRIPTS = tedoc.tes -%.in : %.template tedoc.tes $(SOURCES) +%.in : %.template tedoc.tes @top_srcdir@/src/*.cpp  	$(SCITECO_FULL) -m -- @srcdir@/tedoc.tes \ -	                -C $@ $< $(SOURCES) +	                -C $@ $< @top_srcdir@/src/*.cpp  # grosciteco is a troff postprocessor similar to grotty  # which can be used to produce SciTECO-friendly output @@ -28,13 +28,13 @@ dist_bin_SCRIPTS += grosciteco.tes  dist_pkgdata_DATA = sciteco.tmac  # Fix up the hash-bang line of installed SciTECO scripts upon -# installation to refer to the installed sciteco binary. +# installation to refer to the installed sciteco binary, +# or rather to the sciteco-wrapper script.  # This takes --program-prefix into account.  install-exec-hook: -	for script in $(DESTDIR)$(bindir)/*.tes; do \ -		@SED@ -i "1s,#!.*sciteco,#!$(libexecdir)/`echo sciteco-wrapper | @SED@ '$(transform)'`$(EXEEXT)," \ -		      $$script; \ -	done +	$(SCITECO_FULL) -e "@EB'$(DESTDIR)$(bindir)/*.tes' 1U* \ +	                    EJ-1<%*^[ 0,L.@FR'#!^EM^Xsciteco'#!$(SCITECO_WRAPPER_INSTALLED)'> \ +	                    :EX"  womendir = $(scitecolibdir)/women @@ -74,7 +74,6 @@ EXTRA_DIST += sciteco.1.in  man_MANS += sciteco.7  EXTRA_DIST += sciteco.7.template  CLEANFILES += sciteco.7.in sciteco.7.htbl -SOURCES = $(wildcard @top_srcdir@/src/*.cpp)  CLEANFILES += $(man_MANS) @@ -91,7 +90,8 @@ endif  %.htbl : % htbl.tes  	$(SCITECO_FULL) -m -- @srcdir@/htbl.tes $< $@ -%.html : %.htbl +SUFFIXES += .htbl .html +.htbl.html:  	@GROFF@ -wall -Thtml -man $< >$@  # diff --git a/m4/ax_check_gnu_make.m4 b/m4/ax_check_gnu_make.m4 new file mode 100644 index 0000000..4c761ea --- /dev/null +++ b/m4/ax_check_gnu_make.m4 @@ -0,0 +1,84 @@ +# =========================================================================== +#    https://www.gnu.org/software/autoconf-archive/ax_check_gnu_make.html +# =========================================================================== +# +# SYNOPSIS +# +#   AX_CHECK_GNU_MAKE() +# +# DESCRIPTION +# +#   This macro searches for a GNU version of make. If a match is found: +# +#     * The makefile variable `ifGNUmake' is set to the empty string, otherwise +#       it is set to "#". This is useful for including a special features in a +#       Makefile, which cannot be handled by other versions of make. +#     * The variable `_cv_gnu_make_command` is set to the command to invoke +#       GNU make if it exists, the empty string otherwise. +#     * The variable `ax_cv_gnu_make_command` is set to the command to invoke +#       GNU make by copying `_cv_gnu_make_command`, otherwise it is unset. +#     * If GNU Make is found, its version is extracted from the output of +#       `make --version` as the last field of a record of space-separated +#       columns and saved into the variable `ax_check_gnu_make_version`. +# +#   Here is an example of its use: +# +#   Makefile.in might contain: +# +#     # A failsafe way of putting a dependency rule into a makefile +#     $(DEPEND): +#             $(CC) -MM $(srcdir)/*.c > $(DEPEND) +# +#     @ifGNUmake@ ifeq ($(DEPEND),$(wildcard $(DEPEND))) +#     @ifGNUmake@ include $(DEPEND) +#     @ifGNUmake@ endif +# +#   Then configure.in would normally contain: +# +#     AX_CHECK_GNU_MAKE() +#     AC_OUTPUT(Makefile) +# +#   Then perhaps to cause gnu make to override any other make, we could do +#   something like this (note that GNU make always looks for GNUmakefile +#   first): +# +#     if  ! test x$_cv_gnu_make_command = x ; then +#             mv Makefile GNUmakefile +#             echo .DEFAULT: > Makefile ; +#             echo \  $_cv_gnu_make_command \$@ >> Makefile; +#     fi +# +#   Then, if any (well almost any) other make is called, and GNU make also +#   exists, then the other make wraps the GNU make. +# +# LICENSE +# +#   Copyright (c) 2008 John Darrington <j.darrington@elvis.murdoch.edu.au> +#   Copyright (c) 2015 Enrico M. Crisostomo <enrico.m.crisostomo@gmail.com> +# +#   Copying and distribution of this file, with or without modification, are +#   permitted in any medium without royalty provided the copyright notice +#   and this notice are preserved. This file is offered as-is, without any +#   warranty. + +#serial 9 + +AC_DEFUN([AX_CHECK_GNU_MAKE],dnl +  [AC_PROG_AWK +  AC_CACHE_CHECK([for GNU make],[_cv_gnu_make_command],[dnl +    _cv_gnu_make_command="" ; +dnl Search all the common names for GNU make +    for a in "$MAKE" make gmake gnumake ; do +      if test -z "$a" ; then continue ; fi ; +      if "$a" --version 2> /dev/null | grep GNU 2>&1 > /dev/null ; then +        _cv_gnu_make_command=$a ; +        AX_CHECK_GNU_MAKE_HEADLINE=$("$a" --version 2> /dev/null | grep "GNU Make") +        ax_check_gnu_make_version=$(echo ${AX_CHECK_GNU_MAKE_HEADLINE} | ${AWK} -F " " '{ print $(NF); }') +        break ; +      fi +    done ;]) +dnl If there was a GNU version, then set @ifGNUmake@ to the empty string, '#' otherwise +  AS_VAR_IF([_cv_gnu_make_command], [""], [AS_VAR_SET([ifGNUmake], ["#"])],   [AS_VAR_SET([ifGNUmake], [""])]) +  AS_VAR_IF([_cv_gnu_make_command], [""], [AS_UNSET(ax_cv_gnu_make_command)], [AS_VAR_SET([ax_cv_gnu_make_command], [${_cv_gnu_make_command}])]) +  AC_SUBST([ifGNUmake]) +]) diff --git a/m4/ax_cxx_compile_stdcxx.m4 b/m4/ax_cxx_compile_stdcxx.m4 new file mode 100644 index 0000000..5032bba --- /dev/null +++ b/m4/ax_cxx_compile_stdcxx.m4 @@ -0,0 +1,982 @@ +# =========================================================================== +#  https://www.gnu.org/software/autoconf-archive/ax_cxx_compile_stdcxx.html +# =========================================================================== +# +# SYNOPSIS +# +#   AX_CXX_COMPILE_STDCXX(VERSION, [ext|noext], [mandatory|optional]) +# +# DESCRIPTION +# +#   Check for baseline language coverage in the compiler for the specified +#   version of the C++ standard.  If necessary, add switches to CXX and +#   CXXCPP to enable support.  VERSION may be '11' (for the C++11 standard) +#   or '14' (for the C++14 standard). +# +#   The second argument, if specified, indicates whether you insist on an +#   extended mode (e.g. -std=gnu++11) or a strict conformance mode (e.g. +#   -std=c++11).  If neither is specified, you get whatever works, with +#   preference for an extended mode. +# +#   The third argument, if specified 'mandatory' or if left unspecified, +#   indicates that baseline support for the specified C++ standard is +#   required and that the macro should error out if no mode with that +#   support is found.  If specified 'optional', then configuration proceeds +#   regardless, after defining HAVE_CXX${VERSION} if and only if a +#   supporting mode is found. +# +# LICENSE +# +#   Copyright (c) 2008 Benjamin Kosnik <bkoz@redhat.com> +#   Copyright (c) 2012 Zack Weinberg <zackw@panix.com> +#   Copyright (c) 2013 Roy Stogner <roystgnr@ices.utexas.edu> +#   Copyright (c) 2014, 2015 Google Inc.; contributed by Alexey Sokolov <sokolov@google.com> +#   Copyright (c) 2015 Paul Norman <penorman@mac.com> +#   Copyright (c) 2015 Moritz Klammler <moritz@klammler.eu> +#   Copyright (c) 2016 Krzesimir Nowak <qdlacz@gmail.com> +# +#   Copying and distribution of this file, with or without modification, are +#   permitted in any medium without royalty provided the copyright notice +#   and this notice are preserved.  This file is offered as-is, without any +#   warranty. + +#serial 7 + +dnl  This macro is based on the code from the AX_CXX_COMPILE_STDCXX_11 macro +dnl  (serial version number 13). + +AX_REQUIRE_DEFINED([AC_MSG_WARN]) +AC_DEFUN([AX_CXX_COMPILE_STDCXX], [dnl +  m4_if([$1], [11], [ax_cxx_compile_alternatives="11 0x"], +        [$1], [14], [ax_cxx_compile_alternatives="14 1y"], +        [$1], [17], [ax_cxx_compile_alternatives="17 1z"], +        [m4_fatal([invalid first argument `$1' to AX_CXX_COMPILE_STDCXX])])dnl +  m4_if([$2], [], [], +        [$2], [ext], [], +        [$2], [noext], [], +        [m4_fatal([invalid second argument `$2' to AX_CXX_COMPILE_STDCXX])])dnl +  m4_if([$3], [], [ax_cxx_compile_cxx$1_required=true], +        [$3], [mandatory], [ax_cxx_compile_cxx$1_required=true], +        [$3], [optional], [ax_cxx_compile_cxx$1_required=false], +        [m4_fatal([invalid third argument `$3' to AX_CXX_COMPILE_STDCXX])]) +  AC_LANG_PUSH([C++])dnl +  ac_success=no +  AC_CACHE_CHECK(whether $CXX supports C++$1 features by default, +  ax_cv_cxx_compile_cxx$1, +  [AC_COMPILE_IFELSE([AC_LANG_SOURCE([_AX_CXX_COMPILE_STDCXX_testbody_$1])], +    [ax_cv_cxx_compile_cxx$1=yes], +    [ax_cv_cxx_compile_cxx$1=no])]) +  if test x$ax_cv_cxx_compile_cxx$1 = xyes; then +    ac_success=yes +  fi + +  m4_if([$2], [noext], [], [dnl +  if test x$ac_success = xno; then +    for alternative in ${ax_cxx_compile_alternatives}; do +      switch="-std=gnu++${alternative}" +      cachevar=AS_TR_SH([ax_cv_cxx_compile_cxx$1_$switch]) +      AC_CACHE_CHECK(whether $CXX supports C++$1 features with $switch, +                     $cachevar, +        [ac_save_CXX="$CXX" +         CXX="$CXX $switch" +         AC_COMPILE_IFELSE([AC_LANG_SOURCE([_AX_CXX_COMPILE_STDCXX_testbody_$1])], +          [eval $cachevar=yes], +          [eval $cachevar=no]) +         CXX="$ac_save_CXX"]) +      if eval test x\$$cachevar = xyes; then +        CXX="$CXX $switch" +        if test -n "$CXXCPP" ; then +          CXXCPP="$CXXCPP $switch" +        fi +        ac_success=yes +        break +      fi +    done +  fi]) + +  m4_if([$2], [ext], [], [dnl +  if test x$ac_success = xno; then +    dnl HP's aCC needs +std=c++11 according to: +    dnl http://h21007.www2.hp.com/portal/download/files/unprot/aCxx/PDF_Release_Notes/769149-001.pdf +    dnl Cray's crayCC needs "-h std=c++11" +    for alternative in ${ax_cxx_compile_alternatives}; do +      for switch in -std=c++${alternative} +std=c++${alternative} "-h std=c++${alternative}"; do +        cachevar=AS_TR_SH([ax_cv_cxx_compile_cxx$1_$switch]) +        AC_CACHE_CHECK(whether $CXX supports C++$1 features with $switch, +                       $cachevar, +          [ac_save_CXX="$CXX" +           CXX="$CXX $switch" +           AC_COMPILE_IFELSE([AC_LANG_SOURCE([_AX_CXX_COMPILE_STDCXX_testbody_$1])], +            [eval $cachevar=yes], +            [eval $cachevar=no]) +           CXX="$ac_save_CXX"]) +        if eval test x\$$cachevar = xyes; then +          CXX="$CXX $switch" +          if test -n "$CXXCPP" ; then +            CXXCPP="$CXXCPP $switch" +          fi +          ac_success=yes +          break +        fi +      done +      if test x$ac_success = xyes; then +        break +      fi +    done +  fi]) +  AC_LANG_POP([C++]) +  if test x$ax_cxx_compile_cxx$1_required = xtrue; then +    if test x$ac_success = xno; then +      AC_MSG_ERROR([*** A compiler with support for C++$1 language features is required.]) +    fi +  fi +  if test x$ac_success = xno; then +    HAVE_CXX$1=0 +    AC_MSG_NOTICE([No compiler with C++$1 support was found]) +  else +    HAVE_CXX$1=1 +    AC_DEFINE(HAVE_CXX$1,1, +              [define if the compiler supports basic C++$1 syntax]) +  fi +  AC_SUBST(HAVE_CXX$1) +  m4_if([$1], [17], [AC_MSG_WARN([C++17 is not yet standardized, so the checks may change in incompatible ways anytime])]) +]) + + +dnl  Test body for checking C++11 support + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_11], +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_11 +) + + +dnl  Test body for checking C++14 support + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_14], +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_11 +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_14 +) + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_17], +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_11 +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_14 +  _AX_CXX_COMPILE_STDCXX_testbody_new_in_17 +) + +dnl  Tests for new features in C++11 + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_new_in_11], [[ + +// If the compiler admits that it is not ready for C++11, why torture it? +// Hopefully, this will speed up the test. + +#ifndef __cplusplus + +#error "This is not a C++ compiler" + +#elif __cplusplus < 201103L + +#error "This is not a C++11 compiler" + +#else + +namespace cxx11 +{ + +  namespace test_static_assert +  { + +    template <typename T> +    struct check +    { +      static_assert(sizeof(int) <= sizeof(T), "not big enough"); +    }; + +  } + +  namespace test_final_override +  { + +    struct Base +    { +      virtual void f() {} +    }; + +    struct Derived : public Base +    { +      virtual void f() override {} +    }; + +  } + +  namespace test_double_right_angle_brackets +  { + +    template < typename T > +    struct check {}; + +    typedef check<void> single_type; +    typedef check<check<void>> double_type; +    typedef check<check<check<void>>> triple_type; +    typedef check<check<check<check<void>>>> quadruple_type; + +  } + +  namespace test_decltype +  { + +    int +    f() +    { +      int a = 1; +      decltype(a) b = 2; +      return a + b; +    } + +  } + +  namespace test_type_deduction +  { + +    template < typename T1, typename T2 > +    struct is_same +    { +      static const bool value = false; +    }; + +    template < typename T > +    struct is_same<T, T> +    { +      static const bool value = true; +    }; + +    template < typename T1, typename T2 > +    auto +    add(T1 a1, T2 a2) -> decltype(a1 + a2) +    { +      return a1 + a2; +    } + +    int +    test(const int c, volatile int v) +    { +      static_assert(is_same<int, decltype(0)>::value == true, ""); +      static_assert(is_same<int, decltype(c)>::value == false, ""); +      static_assert(is_same<int, decltype(v)>::value == false, ""); +      auto ac = c; +      auto av = v; +      auto sumi = ac + av + 'x'; +      auto sumf = ac + av + 1.0; +      static_assert(is_same<int, decltype(ac)>::value == true, ""); +      static_assert(is_same<int, decltype(av)>::value == true, ""); +      static_assert(is_same<int, decltype(sumi)>::value == true, ""); +      static_assert(is_same<int, decltype(sumf)>::value == false, ""); +      static_assert(is_same<int, decltype(add(c, v))>::value == true, ""); +      return (sumf > 0.0) ? sumi : add(c, v); +    } + +  } + +  namespace test_noexcept +  { + +    int f() { return 0; } +    int g() noexcept { return 0; } + +    static_assert(noexcept(f()) == false, ""); +    static_assert(noexcept(g()) == true, ""); + +  } + +  namespace test_constexpr +  { + +    template < typename CharT > +    unsigned long constexpr +    strlen_c_r(const CharT *const s, const unsigned long acc) noexcept +    { +      return *s ? strlen_c_r(s + 1, acc + 1) : acc; +    } + +    template < typename CharT > +    unsigned long constexpr +    strlen_c(const CharT *const s) noexcept +    { +      return strlen_c_r(s, 0UL); +    } + +    static_assert(strlen_c("") == 0UL, ""); +    static_assert(strlen_c("1") == 1UL, ""); +    static_assert(strlen_c("example") == 7UL, ""); +    static_assert(strlen_c("another\0example") == 7UL, ""); + +  } + +  namespace test_rvalue_references +  { + +    template < int N > +    struct answer +    { +      static constexpr int value = N; +    }; + +    answer<1> f(int&)       { return answer<1>(); } +    answer<2> f(const int&) { return answer<2>(); } +    answer<3> f(int&&)      { return answer<3>(); } + +    void +    test() +    { +      int i = 0; +      const int c = 0; +      static_assert(decltype(f(i))::value == 1, ""); +      static_assert(decltype(f(c))::value == 2, ""); +      static_assert(decltype(f(0))::value == 3, ""); +    } + +  } + +  namespace test_uniform_initialization +  { + +    struct test +    { +      static const int zero {}; +      static const int one {1}; +    }; + +    static_assert(test::zero == 0, ""); +    static_assert(test::one == 1, ""); + +  } + +  namespace test_lambdas +  { + +    void +    test1() +    { +      auto lambda1 = [](){}; +      auto lambda2 = lambda1; +      lambda1(); +      lambda2(); +    } + +    int +    test2() +    { +      auto a = [](int i, int j){ return i + j; }(1, 2); +      auto b = []() -> int { return '0'; }(); +      auto c = [=](){ return a + b; }(); +      auto d = [&](){ return c; }(); +      auto e = [a, &b](int x) mutable { +        const auto identity = [](int y){ return y; }; +        for (auto i = 0; i < a; ++i) +          a += b--; +        return x + identity(a + b); +      }(0); +      return a + b + c + d + e; +    } + +    int +    test3() +    { +      const auto nullary = [](){ return 0; }; +      const auto unary = [](int x){ return x; }; +      using nullary_t = decltype(nullary); +      using unary_t = decltype(unary); +      const auto higher1st = [](nullary_t f){ return f(); }; +      const auto higher2nd = [unary](nullary_t f1){ +        return [unary, f1](unary_t f2){ return f2(unary(f1())); }; +      }; +      return higher1st(nullary) + higher2nd(nullary)(unary); +    } + +  } + +  namespace test_variadic_templates +  { + +    template <int...> +    struct sum; + +    template <int N0, int... N1toN> +    struct sum<N0, N1toN...> +    { +      static constexpr auto value = N0 + sum<N1toN...>::value; +    }; + +    template <> +    struct sum<> +    { +      static constexpr auto value = 0; +    }; + +    static_assert(sum<>::value == 0, ""); +    static_assert(sum<1>::value == 1, ""); +    static_assert(sum<23>::value == 23, ""); +    static_assert(sum<1, 2>::value == 3, ""); +    static_assert(sum<5, 5, 11>::value == 21, ""); +    static_assert(sum<2, 3, 5, 7, 11, 13>::value == 41, ""); + +  } + +  // http://stackoverflow.com/questions/13728184/template-aliases-and-sfinae +  // Clang 3.1 fails with headers of libstd++ 4.8.3 when using std::function +  // because of this. +  namespace test_template_alias_sfinae +  { + +    struct foo {}; + +    template<typename T> +    using member = typename T::member_type; + +    template<typename T> +    void func(...) {} + +    template<typename T> +    void func(member<T>*) {} + +    void test(); + +    void test() { func<foo>(0); } + +  } + +}  // namespace cxx11 + +#endif  // __cplusplus >= 201103L + +]]) + + +dnl  Tests for new features in C++14 + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_new_in_14], [[ + +// If the compiler admits that it is not ready for C++14, why torture it? +// Hopefully, this will speed up the test. + +#ifndef __cplusplus + +#error "This is not a C++ compiler" + +#elif __cplusplus < 201402L + +#error "This is not a C++14 compiler" + +#else + +namespace cxx14 +{ + +  namespace test_polymorphic_lambdas +  { + +    int +    test() +    { +      const auto lambda = [](auto&&... args){ +        const auto istiny = [](auto x){ +          return (sizeof(x) == 1UL) ? 1 : 0; +        }; +        const int aretiny[] = { istiny(args)... }; +        return aretiny[0]; +      }; +      return lambda(1, 1L, 1.0f, '1'); +    } + +  } + +  namespace test_binary_literals +  { + +    constexpr auto ivii = 0b0000000000101010; +    static_assert(ivii == 42, "wrong value"); + +  } + +  namespace test_generalized_constexpr +  { + +    template < typename CharT > +    constexpr unsigned long +    strlen_c(const CharT *const s) noexcept +    { +      auto length = 0UL; +      for (auto p = s; *p; ++p) +        ++length; +      return length; +    } + +    static_assert(strlen_c("") == 0UL, ""); +    static_assert(strlen_c("x") == 1UL, ""); +    static_assert(strlen_c("test") == 4UL, ""); +    static_assert(strlen_c("another\0test") == 7UL, ""); + +  } + +  namespace test_lambda_init_capture +  { + +    int +    test() +    { +      auto x = 0; +      const auto lambda1 = [a = x](int b){ return a + b; }; +      const auto lambda2 = [a = lambda1(x)](){ return a; }; +      return lambda2(); +    } + +  } + +  namespace test_digit_separators +  { + +    constexpr auto ten_million = 100'000'000; +    static_assert(ten_million == 100000000, ""); + +  } + +  namespace test_return_type_deduction +  { + +    auto f(int& x) { return x; } +    decltype(auto) g(int& x) { return x; } + +    template < typename T1, typename T2 > +    struct is_same +    { +      static constexpr auto value = false; +    }; + +    template < typename T > +    struct is_same<T, T> +    { +      static constexpr auto value = true; +    }; + +    int +    test() +    { +      auto x = 0; +      static_assert(is_same<int, decltype(f(x))>::value, ""); +      static_assert(is_same<int&, decltype(g(x))>::value, ""); +      return x; +    } + +  } + +}  // namespace cxx14 + +#endif  // __cplusplus >= 201402L + +]]) + + +dnl  Tests for new features in C++17 + +m4_define([_AX_CXX_COMPILE_STDCXX_testbody_new_in_17], [[ + +// If the compiler admits that it is not ready for C++17, why torture it? +// Hopefully, this will speed up the test. + +#ifndef __cplusplus + +#error "This is not a C++ compiler" + +#elif __cplusplus <= 201402L + +#error "This is not a C++17 compiler" + +#else + +#if defined(__clang__) +  #define REALLY_CLANG +#else +  #if defined(__GNUC__) +    #define REALLY_GCC +  #endif +#endif + +#include <initializer_list> +#include <utility> +#include <type_traits> + +namespace cxx17 +{ + +#if !defined(REALLY_CLANG) +  namespace test_constexpr_lambdas +  { + +    // TODO: test it with clang++ from git + +    constexpr int foo = [](){return 42;}(); + +  } +#endif // !defined(REALLY_CLANG) + +  namespace test::nested_namespace::definitions +  { + +  } + +  namespace test_fold_expression +  { + +    template<typename... Args> +    int multiply(Args... args) +    { +      return (args * ... * 1); +    } + +    template<typename... Args> +    bool all(Args... args) +    { +      return (args && ...); +    } + +  } + +  namespace test_extended_static_assert +  { + +    static_assert (true); + +  } + +  namespace test_auto_brace_init_list +  { + +    auto foo = {5}; +    auto bar {5}; + +    static_assert(std::is_same<std::initializer_list<int>, decltype(foo)>::value); +    static_assert(std::is_same<int, decltype(bar)>::value); +  } + +  namespace test_typename_in_template_template_parameter +  { + +    template<template<typename> typename X> struct D; + +  } + +  namespace test_fallthrough_nodiscard_maybe_unused_attributes +  { + +    int f1() +    { +      return 42; +    } + +    [[nodiscard]] int f2() +    { +      [[maybe_unused]] auto unused = f1(); + +      switch (f1()) +      { +      case 17: +        f1(); +        [[fallthrough]]; +      case 42: +        f1(); +      } +      return f1(); +    } + +  } + +  namespace test_extended_aggregate_initialization +  { + +    struct base1 +    { +      int b1, b2 = 42; +    }; + +    struct base2 +    { +      base2() { +        b3 = 42; +      } +      int b3; +    }; + +    struct derived : base1, base2 +    { +        int d; +    }; + +    derived d1 {{1, 2}, {}, 4};  // full initialization +    derived d2 {{}, {}, 4};      // value-initialized bases + +  } + +  namespace test_general_range_based_for_loop +  { + +    struct iter +    { +      int i; + +      int& operator* () +      { +        return i; +      } + +      const int& operator* () const +      { +        return i; +      } + +      iter& operator++() +      { +        ++i; +        return *this; +      } +    }; + +    struct sentinel +    { +      int i; +    }; + +    bool operator== (const iter& i, const sentinel& s) +    { +      return i.i == s.i; +    } + +    bool operator!= (const iter& i, const sentinel& s) +    { +      return !(i == s); +    } + +    struct range +    { +      iter begin() const +      { +        return {0}; +      } + +      sentinel end() const +      { +        return {5}; +      } +    }; + +    void f() +    { +      range r {}; + +      for (auto i : r) +      { +        [[maybe_unused]] auto v = i; +      } +    } + +  } + +  namespace test_lambda_capture_asterisk_this_by_value +  { + +    struct t +    { +      int i; +      int foo() +      { +        return [*this]() +        { +          return i; +        }(); +      } +    }; + +  } + +  namespace test_enum_class_construction +  { + +    enum class byte : unsigned char +    {}; + +    byte foo {42}; + +  } + +  namespace test_constexpr_if +  { + +    template <bool cond> +    int f () +    { +      if constexpr(cond) +      { +        return 13; +      } +      else +      { +        return 42; +      } +    } + +  } + +  namespace test_selection_statement_with_initializer +  { + +    int f() +    { +      return 13; +    } + +    int f2() +    { +      if (auto i = f(); i > 0) +      { +        return 3; +      } + +      switch (auto i = f(); i + 4) +      { +      case 17: +        return 2; + +      default: +        return 1; +      } +    } + +  } + +#if !defined(REALLY_CLANG) +  namespace test_template_argument_deduction_for_class_templates +  { + +    // TODO: test it with clang++ from git + +    template <typename T1, typename T2> +    struct pair +    { +      pair (T1 p1, T2 p2) +        : m1 {p1}, +          m2 {p2} +      {} + +      T1 m1; +      T2 m2; +    }; + +    void f() +    { +      [[maybe_unused]] auto p = pair{13, 42u}; +    } + +  } +#endif // !defined(REALLY_CLANG) + +  namespace test_non_type_auto_template_parameters +  { + +    template <auto n> +    struct B +    {}; + +    B<5> b1; +    B<'a'> b2; + +  } + +#if !defined(REALLY_CLANG) +  namespace test_structured_bindings +  { + +    // TODO: test it with clang++ from git + +    int arr[2] = { 1, 2 }; +    std::pair<int, int> pr = { 1, 2 }; + +    auto f1() -> int(&)[2] +    { +      return arr; +    } + +    auto f2() -> std::pair<int, int>& +    { +      return pr; +    } + +    struct S +    { +      int x1 : 2; +      volatile double y1; +    }; + +    S f3() +    { +      return {}; +    } + +    auto [ x1, y1 ] = f1(); +    auto& [ xr1, yr1 ] = f1(); +    auto [ x2, y2 ] = f2(); +    auto& [ xr2, yr2 ] = f2(); +    const auto [ x3, y3 ] = f3(); + +  } +#endif // !defined(REALLY_CLANG) + +#if !defined(REALLY_CLANG) +  namespace test_exception_spec_type_system +  { + +    // TODO: test it with clang++ from git + +    struct Good {}; +    struct Bad {}; + +    void g1() noexcept; +    void g2(); + +    template<typename T> +    Bad +    f(T*, T*); + +    template<typename T1, typename T2> +    Good +    f(T1*, T2*); + +    static_assert (std::is_same_v<Good, decltype(f(g1, g2))>); + +  } +#endif // !defined(REALLY_CLANG) + +  namespace test_inline_variables +  { + +    template<class T> void f(T) +    {} + +    template<class T> inline T g(T) +    { +      return T{}; +    } + +    template<> inline void f<>(int) +    {} + +    template<> int g<>(int) +    { +      return 5; +    } + +  } + +}  // namespace cxx17 + +#endif  // __cplusplus <= 201402L + +]]) diff --git a/src/Makefile.am b/src/Makefile.am index 2778f4f..ba21109 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -11,7 +11,7 @@ endif  include $(top_srcdir)/bootstrap.am  include $(top_srcdir)/scintilla.am -AM_CXXFLAGS = -std=c++11 -Wall -Wno-char-subscripts +AM_CXXFLAGS = -Wall -Wno-char-subscripts  if CLANG  AM_CXXFLAGS += -Wno-mismatched-tags  endif @@ -98,8 +98,5 @@ CLEANFILES += $(libexec_SCRIPTS)  .PHONY: sciteco-wrapper  sciteco-wrapper: -	echo '#!/bin/sh' >$@ -	echo 'OPT=$$1;' \ -	     'shift;' \ -	     "exec $(bindir)/`echo sciteco | @SED@ '$(transform)'`$(EXEEXT)" \ -	     '"$$OPT" -- $$@' >>$@ +	printf '#!/bin/sh\nOPT=$$1\nshift\nexec %s "$$OPT" -- $$@' \ +	       "$(SCITECO_INSTALLED)" >$@ diff --git a/tests/testsuite.at b/tests/testsuite.at index 2e613ee..a9a1322 100644 --- a/tests/testsuite.at +++ b/tests/testsuite.at @@ -14,7 +14,7 @@ AT_CLEANUP  AT_SETUP([Automatic EOL normalization])  AT_CHECK([$SCITECO -e "@EB'${srcdir}/autoeol-input.txt' EL-2\"N(0/0)' 2LR 13@I'' 0EL @EW'autoeol-sciteco.txt'"],           0, ignore, ignore) -AT_CHECK([cmp -b autoeol-sciteco.txt ${srcdir}/autoeol-output.txt], 0, ignore, ignore) +AT_CHECK([cmp autoeol-sciteco.txt ${srcdir}/autoeol-output.txt], 0, ignore, ignore)  AT_CLEANUP  AT_SETUP([Glob patterns with character classes]) | 
