Blame m4/ax_valgrind_check.m4

Packit a7d494
# ===========================================================================
Packit a7d494
#    https://www.gnu.org/software/autoconf-archive/ax_valgrind_check.html
Packit a7d494
# ===========================================================================
Packit a7d494
#
Packit a7d494
# SYNOPSIS
Packit a7d494
#
Packit a7d494
#   AX_VALGRIND_DFLT(memcheck|helgrind|drd|sgcheck, on|off)
Packit a7d494
#   AX_VALGRIND_CHECK()
Packit a7d494
#
Packit a7d494
# DESCRIPTION
Packit a7d494
#
Packit a7d494
#   AX_VALGRIND_CHECK checks whether Valgrind is present and, if so, allows
Packit a7d494
#   running `make check` under a variety of Valgrind tools to check for
Packit a7d494
#   memory and threading errors.
Packit a7d494
#
Packit a7d494
#   Defines VALGRIND_CHECK_RULES which should be substituted in your
Packit a7d494
#   Makefile; and $enable_valgrind which can be used in subsequent configure
Packit a7d494
#   output. VALGRIND_ENABLED is defined and substituted, and corresponds to
Packit a7d494
#   the value of the --enable-valgrind option, which defaults to being
Packit a7d494
#   enabled if Valgrind is installed and disabled otherwise. Individual
Packit a7d494
#   Valgrind tools can be disabled via --disable-valgrind-<tool>, the
Packit a7d494
#   default is configurable via the AX_VALGRIND_DFLT command or is to use
Packit a7d494
#   all commands not disabled via AX_VALGRIND_DFLT. All AX_VALGRIND_DFLT
Packit a7d494
#   calls must be made before the call to AX_VALGRIND_CHECK.
Packit a7d494
#
Packit a7d494
#   If unit tests are written using a shell script and automake's
Packit a7d494
#   LOG_COMPILER system, the $(VALGRIND) variable can be used within the
Packit a7d494
#   shell scripts to enable Valgrind, as described here:
Packit a7d494
#
Packit a7d494
#     https://www.gnu.org/software/gnulib/manual/html_node/Running-self_002dtests-under-valgrind.html
Packit a7d494
#
Packit a7d494
#   Usage example:
Packit a7d494
#
Packit a7d494
#   configure.ac:
Packit a7d494
#
Packit a7d494
#     AX_VALGRIND_DFLT([sgcheck], [off])
Packit a7d494
#     AX_VALGRIND_CHECK
Packit a7d494
#
Packit a7d494
#   in each Makefile.am with tests:
Packit a7d494
#
Packit a7d494
#     @VALGRIND_CHECK_RULES@
Packit a7d494
#     VALGRIND_SUPPRESSIONS_FILES = my-project.supp
Packit a7d494
#     EXTRA_DIST = my-project.supp
Packit a7d494
#
Packit a7d494
#   This results in a "check-valgrind" rule being added. Running `make
Packit a7d494
#   check-valgrind` in that directory will recursively run the module's test
Packit a7d494
#   suite (`make check`) once for each of the available Valgrind tools (out
Packit a7d494
#   of memcheck, helgrind and drd) while the sgcheck will be skipped unless
Packit a7d494
#   enabled again on the commandline with --enable-valgrind-sgcheck. The
Packit a7d494
#   results for each check will be output to test-suite-$toolname.log. The
Packit a7d494
#   target will succeed if there are zero errors and fail otherwise.
Packit a7d494
#
Packit a7d494
#   Alternatively, a "check-valgrind-$TOOL" rule will be added, for $TOOL in
Packit a7d494
#   memcheck, helgrind, drd and sgcheck. These are useful because often only
Packit a7d494
#   some of those tools can be ran cleanly on a codebase.
Packit a7d494
#
Packit a7d494
#   The macro supports running with and without libtool.
Packit a7d494
#
Packit a7d494
# LICENSE
Packit a7d494
#
Packit a7d494
#   Copyright (c) 2014, 2015, 2016 Philip Withnall <philip.withnall@collabora.co.uk>
Packit a7d494
#
Packit a7d494
#   Copying and distribution of this file, with or without modification, are
Packit a7d494
#   permitted in any medium without royalty provided the copyright notice
Packit a7d494
#   and this notice are preserved.  This file is offered as-is, without any
Packit a7d494
#   warranty.
Packit a7d494
Packit a7d494
#serial 17
Packit a7d494
Packit a7d494
dnl Configured tools
Packit a7d494
m4_define([valgrind_tool_list], [[memcheck], [helgrind], [drd], [sgcheck]])
Packit a7d494
m4_set_add_all([valgrind_exp_tool_set], [sgcheck])
Packit a7d494
m4_foreach([vgtool], [valgrind_tool_list],
Packit a7d494
           [m4_define([en_dflt_valgrind_]vgtool, [on])])
Packit a7d494
Packit a7d494
AC_DEFUN([AX_VALGRIND_DFLT],[
Packit a7d494
	m4_define([en_dflt_valgrind_$1], [$2])
Packit a7d494
])dnl
Packit a7d494
Packit a7d494
AM_EXTRA_RECURSIVE_TARGETS([check-valgrind])
Packit a7d494
m4_foreach([vgtool], [valgrind_tool_list],
Packit a7d494
	[AM_EXTRA_RECURSIVE_TARGETS([check-valgrind-]vgtool)])
Packit a7d494
Packit a7d494
AC_DEFUN([AX_VALGRIND_CHECK],[
Packit a7d494
	dnl Check for --enable-valgrind
Packit a7d494
	AC_ARG_ENABLE([valgrind],
Packit a7d494
	              [AS_HELP_STRING([--enable-valgrind], [Whether to enable Valgrind on the unit tests])],
Packit a7d494
	              [enable_valgrind=$enableval],[enable_valgrind=])
Packit a7d494
Packit a7d494
	AS_IF([test "$enable_valgrind" != "no"],[
Packit a7d494
		# Check for Valgrind.
Packit a7d494
		AC_CHECK_PROG([VALGRIND],[valgrind],[valgrind])
Packit a7d494
		AS_IF([test "$VALGRIND" = ""],[
Packit a7d494
			AS_IF([test "$enable_valgrind" = "yes"],[
Packit a7d494
				AC_MSG_ERROR([Could not find valgrind; either install it or reconfigure with --disable-valgrind])
Packit a7d494
			],[
Packit a7d494
				enable_valgrind=no
Packit a7d494
			])
Packit a7d494
		],[
Packit a7d494
			enable_valgrind=yes
Packit a7d494
		])
Packit a7d494
	])
Packit a7d494
Packit a7d494
	AM_CONDITIONAL([VALGRIND_ENABLED],[test "$enable_valgrind" = "yes"])
Packit a7d494
	AC_SUBST([VALGRIND_ENABLED],[$enable_valgrind])
Packit a7d494
Packit a7d494
	# Check for Valgrind tools we care about.
Packit a7d494
	[valgrind_enabled_tools=]
Packit a7d494
	m4_foreach([vgtool],[valgrind_tool_list],[
Packit a7d494
		AC_ARG_ENABLE([valgrind-]vgtool,
Packit a7d494
		    m4_if(m4_defn([en_dflt_valgrind_]vgtool),[off],dnl
Packit a7d494
[AS_HELP_STRING([--enable-valgrind-]vgtool, [Whether to use ]vgtool[ during the Valgrind tests])],dnl
Packit a7d494
[AS_HELP_STRING([--disable-valgrind-]vgtool, [Whether to skip ]vgtool[ during the Valgrind tests])]),
Packit a7d494
		              [enable_valgrind_]vgtool[=$enableval],
Packit a7d494
		              [enable_valgrind_]vgtool[=])
Packit a7d494
		AS_IF([test "$enable_valgrind" = "no"],[
Packit a7d494
			enable_valgrind_]vgtool[=no],
Packit a7d494
		      [test "$enable_valgrind_]vgtool[" ]dnl
Packit a7d494
m4_if(m4_defn([en_dflt_valgrind_]vgtool), [off], [= "yes"], [!= "no"]),[
Packit a7d494
			AC_CACHE_CHECK([for Valgrind tool ]vgtool,
Packit a7d494
			               [ax_cv_valgrind_tool_]vgtool,[
Packit a7d494
				ax_cv_valgrind_tool_]vgtool[=no
Packit a7d494
				m4_set_contains([valgrind_exp_tool_set],vgtool,
Packit a7d494
				    [m4_define([vgtoolx],[exp-]vgtool)],
Packit a7d494
				    [m4_define([vgtoolx],vgtool)])
Packit a7d494
				AS_IF([`$VALGRIND --tool=]vgtoolx[ --help >/dev/null 2>&1`],[
Packit a7d494
					ax_cv_valgrind_tool_]vgtool[=yes
Packit a7d494
				])
Packit a7d494
			])
Packit a7d494
			AS_IF([test "$ax_cv_valgrind_tool_]vgtool[" = "no"],[
Packit a7d494
				AS_IF([test "$enable_valgrind_]vgtool[" = "yes"],[
Packit a7d494
					AC_MSG_ERROR([Valgrind does not support ]vgtool[; reconfigure with --disable-valgrind-]vgtool)
Packit a7d494
				],[
Packit a7d494
					enable_valgrind_]vgtool[=no
Packit a7d494
				])
Packit a7d494
			],[
Packit a7d494
				enable_valgrind_]vgtool[=yes
Packit a7d494
			])
Packit a7d494
		])
Packit a7d494
		AS_IF([test "$enable_valgrind_]vgtool[" = "yes"],[
Packit a7d494
			valgrind_enabled_tools="$valgrind_enabled_tools ]m4_bpatsubst(vgtool,[^exp-])["
Packit a7d494
		])
Packit a7d494
		AC_SUBST([ENABLE_VALGRIND_]vgtool,[$enable_valgrind_]vgtool)
Packit a7d494
	])
Packit a7d494
	AC_SUBST([valgrind_tools],["]m4_join([ ], valgrind_tool_list)["])
Packit a7d494
	AC_SUBST([valgrind_enabled_tools],[$valgrind_enabled_tools])
Packit a7d494
Packit a7d494
[VALGRIND_CHECK_RULES='
Packit a7d494
# Valgrind check
Packit a7d494
#
Packit a7d494
# Optional:
Packit a7d494
#  - VALGRIND_SUPPRESSIONS_FILES: Space-separated list of Valgrind suppressions
Packit a7d494
#    files to load. (Default: empty)
Packit a7d494
#  - VALGRIND_FLAGS: General flags to pass to all Valgrind tools.
Packit a7d494
#    (Default: --num-callers=30)
Packit a7d494
#  - VALGRIND_$toolname_FLAGS: Flags to pass to Valgrind $toolname (one of:
Packit a7d494
#    memcheck, helgrind, drd, sgcheck). (Default: various)
Packit a7d494
Packit a7d494
# Optional variables
Packit a7d494
VALGRIND_SUPPRESSIONS ?= $(addprefix --suppressions=,$(VALGRIND_SUPPRESSIONS_FILES))
Packit a7d494
VALGRIND_FLAGS ?= --num-callers=30
Packit a7d494
VALGRIND_memcheck_FLAGS ?= --leak-check=full --show-reachable=no
Packit a7d494
VALGRIND_helgrind_FLAGS ?= --history-level=approx
Packit a7d494
VALGRIND_drd_FLAGS ?=
Packit a7d494
VALGRIND_sgcheck_FLAGS ?=
Packit a7d494
Packit a7d494
# Internal use
Packit a7d494
valgrind_log_files = $(addprefix test-suite-,$(addsuffix .log,$(valgrind_tools)))
Packit a7d494
Packit a7d494
valgrind_memcheck_flags = --tool=memcheck $(VALGRIND_memcheck_FLAGS)
Packit a7d494
valgrind_helgrind_flags = --tool=helgrind $(VALGRIND_helgrind_FLAGS)
Packit a7d494
valgrind_drd_flags = --tool=drd $(VALGRIND_drd_FLAGS)
Packit a7d494
valgrind_sgcheck_flags = --tool=exp-sgcheck $(VALGRIND_sgcheck_FLAGS)
Packit a7d494
Packit a7d494
valgrind_quiet = $(valgrind_quiet_$(V))
Packit a7d494
valgrind_quiet_ = $(valgrind_quiet_$(AM_DEFAULT_VERBOSITY))
Packit a7d494
valgrind_quiet_0 = --quiet
Packit a7d494
valgrind_v_use   = $(valgrind_v_use_$(V))
Packit a7d494
valgrind_v_use_  = $(valgrind_v_use_$(AM_DEFAULT_VERBOSITY))
Packit a7d494
valgrind_v_use_0 = @echo "  USE   " $(patsubst check-valgrind-%-am,%,$''@):;
Packit a7d494
Packit a7d494
# Support running with and without libtool.
Packit a7d494
ifneq ($(LIBTOOL),)
Packit a7d494
valgrind_lt = $(LIBTOOL) $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=execute
Packit a7d494
else
Packit a7d494
valgrind_lt =
Packit a7d494
endif
Packit a7d494
Packit a7d494
# Use recursive makes in order to ignore errors during check
Packit a7d494
check-valgrind-am:
Packit a7d494
ifeq ($(VALGRIND_ENABLED),yes)
Packit a7d494
	$(A''M_V_at)$(MAKE) $(AM_MAKEFLAGS) -k \
Packit a7d494
		$(foreach tool, $(valgrind_enabled_tools), check-valgrind-$(tool))
Packit a7d494
else
Packit a7d494
	@echo "Need to reconfigure with --enable-valgrind"
Packit a7d494
endif
Packit a7d494
Packit a7d494
# Valgrind running
Packit a7d494
VALGRIND_TESTS_ENVIRONMENT = \
Packit a7d494
	$(TESTS_ENVIRONMENT) \
Packit a7d494
	env VALGRIND=$(VALGRIND) \
Packit a7d494
	G_SLICE=always-malloc,debug-blocks \
Packit a7d494
	G_DEBUG=fatal-warnings,fatal-criticals,gc-friendly
Packit a7d494
Packit a7d494
VALGRIND_LOG_COMPILER = \
Packit a7d494
	$(valgrind_lt) \
Packit a7d494
	$(VALGRIND) $(VALGRIND_SUPPRESSIONS) --error-exitcode=1 $(VALGRIND_FLAGS)
Packit a7d494
Packit a7d494
define valgrind_tool_rule
Packit a7d494
check-valgrind-$(1)-am:
Packit a7d494
ifeq ($$(VALGRIND_ENABLED)-$$(ENABLE_VALGRIND_$(1)),yes-yes)
Packit a7d494
ifneq ($$(TESTS),)
Packit a7d494
	$$(valgrind_v_use)$$(MAKE) check-TESTS \
Packit a7d494
		TESTS_ENVIRONMENT="$$(VALGRIND_TESTS_ENVIRONMENT)" \
Packit a7d494
		LOG_COMPILER="$$(VALGRIND_LOG_COMPILER)" \
Packit a7d494
		LOG_FLAGS="$$(valgrind_$(1)_flags)" \
Packit a7d494
		TEST_SUITE_LOG=test-suite-$(1).log
Packit a7d494
endif
Packit a7d494
else ifeq ($$(VALGRIND_ENABLED),yes)
Packit a7d494
	@echo "Need to reconfigure with --enable-valgrind-$(1)"
Packit a7d494
else
Packit a7d494
	@echo "Need to reconfigure with --enable-valgrind"
Packit a7d494
endif
Packit a7d494
endef
Packit a7d494
Packit a7d494
$(foreach tool,$(valgrind_tools),$(eval $(call valgrind_tool_rule,$(tool))))
Packit a7d494
Packit a7d494
A''M_DISTCHECK_CONFIGURE_FLAGS ?=
Packit a7d494
A''M_DISTCHECK_CONFIGURE_FLAGS += --disable-valgrind
Packit a7d494
Packit a7d494
MOSTLYCLEANFILES ?=
Packit a7d494
MOSTLYCLEANFILES += $(valgrind_log_files)
Packit a7d494
Packit a7d494
.PHONY: check-valgrind $(add-prefix check-valgrind-,$(valgrind_tools))
Packit a7d494
']
Packit a7d494
Packit a7d494
	AC_SUBST([VALGRIND_CHECK_RULES])
Packit a7d494
	m4_ifdef([_AM_SUBST_NOTMAKE], [_AM_SUBST_NOTMAKE([VALGRIND_CHECK_RULES])])
Packit a7d494
])