Home | History | Annotate | Line # | Download | only in unit-tests
Makefile revision 1.116
      1 # $NetBSD: Makefile,v 1.116 2020/08/23 09:28:52 rillig Exp $
      2 #
      3 # Unit tests for make(1)
      4 #
      5 # The main targets are:
      6 #
      7 # all:
      8 #	run all the tests
      9 # test:
     10 #	run 'all', and compare to expected results
     11 # accept:
     12 #	move generated output to expected results
     13 #
     14 # Settable variables
     15 #
     16 # TEST_MAKE
     17 #	The make program to be tested.
     18 #
     19 #
     20 # Adding a test case
     21 #
     22 # Each feature should get its own set of tests in its own suitably
     23 # named makefile (*.mk), with its own set of expected results (*.exp),
     24 # and it should be added to the TESTS list.
     25 #
     26 # Any added files must also be added to src/distrib/sets/lists/tests/mi.
     27 # To do that, just run "make sync-mi" in this directory.
     28 #
     29 # A few *.mk files are helper files for other tests (such as include-sub.mk)
     30 # and are thus not added to TESTS.  Such files must be ignored in
     31 # src/tests/usr.bin/make/t_make.sh.
     32 #
     33 
     34 # Each test is in a sub-makefile.
     35 # Keep the list sorted.
     36 # Any test that is commented out must be ignored in
     37 # src/tests/usr.bin/make/t_make.sh as well.
     38 TESTS+=		# archive	# broken on FreeBSD, enabled in t_make.sh
     39 TESTS+=		cmdline
     40 TESTS+=		comment
     41 TESTS+=		cond-cmp-numeric
     42 TESTS+=		cond-cmp-string
     43 TESTS+=		cond-func
     44 TESTS+=		cond-func-commands
     45 TESTS+=		cond-func-defined
     46 TESTS+=		cond-func-empty
     47 TESTS+=		cond-func-exists
     48 TESTS+=		cond-func-make
     49 TESTS+=		cond-func-target
     50 TESTS+=		cond-late
     51 TESTS+=		cond-op
     52 TESTS+=		cond-op-and
     53 TESTS+=		cond-op-not
     54 TESTS+=		cond-op-or
     55 TESTS+=		cond-op-parentheses
     56 TESTS+=		cond-short
     57 TESTS+=		cond-token-number
     58 TESTS+=		cond-token-plain
     59 TESTS+=		cond-token-string
     60 TESTS+=		cond-token-var
     61 TESTS+=		cond1
     62 TESTS+=		cond2
     63 TESTS+=		counter
     64 TESTS+=		dep
     65 TESTS+=		dep-colon
     66 TESTS+=		dep-double-colon
     67 TESTS+=		dep-exclam
     68 TESTS+=		dep-none
     69 TESTS+=		dep-var
     70 TESTS+=		dep-wildcards
     71 TESTS+=		depsrc
     72 TESTS+=		depsrc-exec
     73 TESTS+=		depsrc-ignore
     74 TESTS+=		depsrc-made
     75 TESTS+=		depsrc-make
     76 TESTS+=		depsrc-meta
     77 TESTS+=		depsrc-nometa
     78 TESTS+=		depsrc-nometa_cmp
     79 TESTS+=		depsrc-nopath
     80 TESTS+=		depsrc-notmain
     81 TESTS+=		depsrc-optional
     82 TESTS+=		depsrc-phony
     83 TESTS+=		depsrc-precious
     84 TESTS+=		depsrc-recursive
     85 TESTS+=		depsrc-silent
     86 TESTS+=		depsrc-use
     87 TESTS+=		depsrc-usebefore
     88 TESTS+=		depsrc-usebefore-double-colon
     89 TESTS+=		depsrc-wait
     90 TESTS+=		deptgt
     91 TESTS+=		deptgt-begin
     92 TESTS+=		deptgt-default
     93 TESTS+=		deptgt-delete_on_error
     94 TESTS+=		deptgt-end
     95 TESTS+=		deptgt-error
     96 TESTS+=		deptgt-ignore
     97 TESTS+=		deptgt-interrupt
     98 TESTS+=		deptgt-main
     99 TESTS+=		deptgt-makeflags
    100 TESTS+=		deptgt-no_parallel
    101 TESTS+=		deptgt-nopath
    102 TESTS+=		deptgt-notparallel
    103 TESTS+=		deptgt-objdir
    104 TESTS+=		deptgt-order
    105 TESTS+=		deptgt-path
    106 TESTS+=		deptgt-path-suffix
    107 TESTS+=		deptgt-phony
    108 TESTS+=		deptgt-precious
    109 TESTS+=		deptgt-shell
    110 TESTS+=		deptgt-silent
    111 TESTS+=		deptgt-stale
    112 TESTS+=		deptgt-suffixes
    113 TESTS+=		dir
    114 TESTS+=		dir-expand-path
    115 TESTS+=		directive
    116 TESTS+=		directive-elif
    117 TESTS+=		directive-elifdef
    118 TESTS+=		directive-elifmake
    119 TESTS+=		directive-elifndef
    120 TESTS+=		directive-elifnmake
    121 TESTS+=		directive-else
    122 TESTS+=		directive-endif
    123 TESTS+=		directive-error
    124 TESTS+=		directive-export
    125 TESTS+=		directive-export-env
    126 TESTS+=		directive-export-literal
    127 TESTS+=		directive-if
    128 TESTS+=		directive-ifdef
    129 TESTS+=		directive-ifmake
    130 TESTS+=		directive-ifndef
    131 TESTS+=		directive-ifnmake
    132 TESTS+=		directive-info
    133 TESTS+=		directive-undef
    134 TESTS+=		directive-unexport
    135 TESTS+=		directive-unexport-env
    136 TESTS+=		directive-warning
    137 TESTS+=		directives
    138 TESTS+=		dollar
    139 TESTS+=		doterror
    140 TESTS+=		dotwait
    141 TESTS+=		envfirst
    142 TESTS+=		error
    143 TESTS+=		# escape	# broken by reverting POSIX changes
    144 TESTS+=		export
    145 TESTS+=		export-all
    146 TESTS+=		export-env
    147 TESTS+=		export-variants
    148 TESTS+=		forloop
    149 TESTS+=		forsubst
    150 TESTS+=		hash
    151 TESTS+=		impsrc
    152 TESTS+=		include-main
    153 TESTS+=		lint
    154 TESTS+=		make-exported
    155 TESTS+=		misc
    156 TESTS+=		moderrs
    157 TESTS+=		modmatch
    158 TESTS+=		modmisc
    159 TESTS+=		modts
    160 TESTS+=		modword
    161 TESTS+=		opt
    162 TESTS+=		opt-backwards
    163 TESTS+=		opt-chdir
    164 TESTS+=		opt-debug
    165 TESTS+=		opt-define
    166 TESTS+=		opt-env
    167 TESTS+=		opt-file
    168 TESTS+=		opt-ignore
    169 TESTS+=		opt-include-dir
    170 TESTS+=		opt-jobs
    171 TESTS+=		opt-jobs-internal
    172 TESTS+=		opt-keep-going
    173 TESTS+=		opt-m-include-dir
    174 TESTS+=		opt-no-action
    175 TESTS+=		opt-no-action-at-all
    176 TESTS+=		opt-query
    177 TESTS+=		opt-raw
    178 TESTS+=		opt-silent
    179 TESTS+=		opt-touch
    180 TESTS+=		opt-tracefile
    181 TESTS+=		opt-var-expanded
    182 TESTS+=		opt-var-literal
    183 TESTS+=		opt-warnings-as-errors
    184 TESTS+=		opt-where-am-i
    185 TESTS+=		opt-x-reduce-exported
    186 TESTS+=		order
    187 TESTS+=		phony-end
    188 TESTS+=		posix
    189 TESTS+=		# posix1	# broken by reverting POSIX changes
    190 TESTS+=		qequals
    191 TESTS+=		recursive
    192 TESTS+=		sh
    193 TESTS+=		sh-dots
    194 TESTS+=		sh-jobs
    195 TESTS+=		sh-jobs-error
    196 TESTS+=		sh-leading-at
    197 TESTS+=		sh-leading-hyphen
    198 TESTS+=		sh-leading-plus
    199 TESTS+=		sh-meta-chars
    200 TESTS+=		sh-multi-line
    201 TESTS+=		sh-single-line
    202 TESTS+=		# suffixes	# runs into an endless loop (try -dA)
    203 TESTS+=		sunshcmd
    204 TESTS+=		sysv
    205 TESTS+=		ternary
    206 TESTS+=		unexport
    207 TESTS+=		unexport-env
    208 TESTS+=		use-inference
    209 TESTS+=		var-class
    210 TESTS+=		var-class-cmdline
    211 TESTS+=		var-class-env
    212 TESTS+=		var-class-global
    213 TESTS+=		var-class-local
    214 TESTS+=		var-class-local-legacy
    215 TESTS+=		var-op
    216 TESTS+=		var-op-append
    217 TESTS+=		var-op-assign
    218 TESTS+=		var-op-default
    219 TESTS+=		var-op-expand
    220 TESTS+=		var-op-shell
    221 TESTS+=		varcmd
    222 TESTS+=		vardebug
    223 TESTS+=		varfind
    224 TESTS+=		varmisc
    225 TESTS+=		varmod
    226 TESTS+=		varmod-assign
    227 TESTS+=		varmod-defined
    228 TESTS+=		varmod-edge
    229 TESTS+=		varmod-exclam-shell
    230 TESTS+=		varmod-extension
    231 TESTS+=		varmod-gmtime
    232 TESTS+=		varmod-hash
    233 TESTS+=		varmod-head
    234 TESTS+=		varmod-ifelse
    235 TESTS+=		varmod-l-name-to-value
    236 TESTS+=		varmod-localtime
    237 TESTS+=		varmod-loop
    238 TESTS+=		varmod-match
    239 TESTS+=		varmod-match-escape
    240 TESTS+=		varmod-no-match
    241 TESTS+=		varmod-order
    242 TESTS+=		varmod-order-reverse
    243 TESTS+=		varmod-order-shuffle
    244 TESTS+=		varmod-path
    245 TESTS+=		varmod-quote
    246 TESTS+=		varmod-quote-dollar
    247 TESTS+=		varmod-range
    248 TESTS+=		varmod-remember
    249 TESTS+=		varmod-root
    250 TESTS+=		varmod-select-words
    251 TESTS+=		varmod-shell
    252 TESTS+=		varmod-subst
    253 TESTS+=		varmod-subst-regex
    254 TESTS+=		varmod-sysv
    255 TESTS+=		varmod-tail
    256 TESTS+=		varmod-to-abs
    257 TESTS+=		varmod-to-lower
    258 TESTS+=		varmod-to-many-words
    259 TESTS+=		varmod-to-one-word
    260 TESTS+=		varmod-to-separator
    261 TESTS+=		varmod-to-upper
    262 TESTS+=		varmod-undefined
    263 TESTS+=		varmod-unique
    264 TESTS+=		varname
    265 TESTS+=		varname-dollar
    266 TESTS+=		varname-dot-alltargets
    267 TESTS+=		varname-dot-curdir
    268 TESTS+=		varname-dot-includedfromdir
    269 TESTS+=		varname-dot-includedfromfile
    270 TESTS+=		varname-dot-make-dependfile
    271 TESTS+=		varname-dot-make-expand_variables
    272 TESTS+=		varname-dot-make-exported
    273 TESTS+=		varname-dot-make-jobs
    274 TESTS+=		varname-dot-make-jobs-prefix
    275 TESTS+=		varname-dot-make-level
    276 TESTS+=		varname-dot-make-makefile_preference
    277 TESTS+=		varname-dot-make-makefiles
    278 TESTS+=		varname-dot-make-meta-bailiwick
    279 TESTS+=		varname-dot-make-meta-created
    280 TESTS+=		varname-dot-make-meta-files
    281 TESTS+=		varname-dot-make-meta-ignore_filter
    282 TESTS+=		varname-dot-make-meta-ignore_paths
    283 TESTS+=		varname-dot-make-meta-ignore_patterns
    284 TESTS+=		varname-dot-make-meta-prefix
    285 TESTS+=		varname-dot-make-mode
    286 TESTS+=		varname-dot-make-path_filemon
    287 TESTS+=		varname-dot-make-pid
    288 TESTS+=		varname-dot-make-ppid
    289 TESTS+=		varname-dot-make-save_dollars
    290 TESTS+=		varname-dot-makeoverrides
    291 TESTS+=		varname-dot-newline
    292 TESTS+=		varname-dot-objdir
    293 TESTS+=		varname-dot-parsedir
    294 TESTS+=		varname-dot-parsefile
    295 TESTS+=		varname-dot-path
    296 TESTS+=		varname-dot-shell
    297 TESTS+=		varname-dot-targets
    298 TESTS+=		varname-empty
    299 TESTS+=		varname-make
    300 TESTS+=		varname-make_print_var_on_error
    301 TESTS+=		varname-makeflags
    302 TESTS+=		varname-pwd
    303 TESTS+=		varname-vpath
    304 TESTS+=		varparse-dynamic
    305 TESTS+=		varquote
    306 TESTS+=		varshell
    307 
    308 # Additional environment variables for some of the tests.
    309 # The base environment is -i PATH="$PATH".
    310 ENV.envfirst=		FROM_ENV=value-from-env
    311 ENV.varmisc=		FROM_ENV=env
    312 ENV.varmisc+=		FROM_ENV_BEFORE=env
    313 ENV.varmisc+=		FROM_ENV_AFTER=env
    314 
    315 # Override make flags for some of the tests; default is -k.
    316 FLAGS.archive=		-dA
    317 FLAGS.counter=		-dv
    318 FLAGS.doterror=		# none
    319 FLAGS.envfirst=		-e
    320 FLAGS.export=		# none
    321 FLAGS.lint=		-dL -k
    322 FLAGS.opt-no-action=	-n
    323 FLAGS.opt-query=	-q
    324 FLAGS.order=		-j1
    325 FLAGS.recursive=	-dL
    326 FLAGS.vardebug=		-k -dv FROM_CMDLINE=
    327 FLAGS.varmod-match-escape= -dv
    328 FLAGS.varname-dot-shell= -dpv
    329 FLAGS.varname-empty=	-dv '$${:U}=cmdline-u' '=cmline-plain'
    330 
    331 # Some tests need extra post-processing.
    332 SED_CMDS.sh-dots+=	-e 's,^${.SHELL}: ,,' -e '/not found/s,command ,,'
    333 SED_CMDS.varmod-subst-regex+= \
    334 			-e 's,\(Regex compilation error:\).*,\1 (details omitted),'
    335 SED_CMDS.varmod-edge+=	-e 's, line [0-9]*:, line omitted:,'
    336 SED_CMDS.varshell+=	-e 's,^${.SHELL:T}: ,,'
    337 SED_CMDS.varshell+=	-e '/command/s,No such.*,not found,'
    338 SED_CMDS.varname-dot-shell=	-e 's, = /.*, = (details omitted),'
    339 SED_CMDS.varname-dot-shell+=	-e 's,"/[^"]*","(details omitted)",'
    340 SED_CMDS.varname-dot-shell+=	-e 's,\[/[^]]*\],[(details omitted)],'
    341 
    342 # Some tests need an additional round of postprocessing.
    343 POSTPROC.counter=	${TOOL_SED} -n -e '/:RELEVANT = yes/,/:RELEVANT = no/p'
    344 POSTPROC.vardebug=	${TOOL_SED} -n -e '/:RELEVANT = yes/,/:RELEVANT = no/p'
    345 POSTPROC.varmod-match-escape= ${TOOL_SED} -n -e '/^Pattern/p'
    346 POSTPROC.varname-dot-shell= \
    347 			awk '/\.SHELL/ || /^ParseReadLine/'
    348 POSTPROC.varname-empty=	${TOOL_SED} -n -e '/^Var_Set/p' -e '/^out:/p'
    349 
    350 # Some tests reuse other tests, which makes them unnecessarily fragile.
    351 export-all.rawout: export.mk
    352 unexport.rawout: export.mk
    353 unexport-env.rawout: export.mk
    354 
    355 # End of the configuration section.
    356 
    357 .MAIN: all
    358 
    359 UNIT_TESTS:=	${.PARSEDIR}
    360 .PATH: ${UNIT_TESTS}
    361 
    362 OUTFILES=	${TESTS:=.out}
    363 
    364 all: ${OUTFILES}
    365 
    366 CLEANFILES=		*.rawout *.out *.status *.tmp *.core *.tmp
    367 CLEANFILES+=		obj*.[och] lib*.a	# posix1.mk
    368 CLEANFILES+=		issue* .[ab]*		# suffixes.mk
    369 CLEANDIRS=		dir dummy		# posix1.mk
    370 
    371 clean:
    372 	rm -f ${CLEANFILES}
    373 	rm -rf ${CLEANDIRS}
    374 
    375 TEST_MAKE?=	${.MAKE}
    376 TOOL_SED?=	sed
    377 
    378 # ensure consistent results from sort(1)
    379 LC_ALL=		C
    380 LANG=		C
    381 .export LANG LC_ALL
    382 
    383 # the tests are actually done with sub-makes.
    384 .SUFFIXES: .mk .rawout .out
    385 .mk.rawout:
    386 	@${_MKMSG_TEST:Uecho '#      test '} ${.PREFIX}
    387 	@set -eu; \
    388 	cd ${.OBJDIR}; \
    389 	env -i PATH="$$PATH" ${ENV.${.TARGET:R}} \
    390 	  ${TEST_MAKE} \
    391 	    -r -C ${.CURDIR} -f ${.IMPSRC} \
    392 	    ${FLAGS.${.TARGET:R}:U-k} \
    393 	    > ${.TARGET}.tmp 2>&1 \
    394 	&& status=$$? || status=$$?; \
    395 	echo $$status > ${.TARGET:R}.status
    396 	@mv ${.TARGET}.tmp ${.TARGET}
    397 
    398 # Post-process the test output so that the results can be compared.
    399 #
    400 # always pretend .MAKE was called 'make'
    401 _SED_CMDS+=	-e 's,^${TEST_MAKE:T:S,.,\\.,g}[][0-9]*:,make:,'
    402 _SED_CMDS+=	-e 's,${TEST_MAKE:S,.,\\.,g},make,'
    403 # replace anything after 'stopped in' with unit-tests
    404 _SED_CMDS+=	-e '/stopped/s, /.*, unit-tests,'
    405 # strip ${.CURDIR}/ from the output
    406 _SED_CMDS+=	-e 's,${.CURDIR:S,.,\\.,g}/,,g'
    407 _SED_CMDS+=	-e 's,${UNIT_TESTS:S,.,\\.,g}/,,g'
    408 
    409 .rawout.out:
    410 	@${TOOL_SED} ${_SED_CMDS} ${SED_CMDS.${.TARGET:R}} \
    411 	  < ${.IMPSRC} > ${.TARGET}.tmp1
    412 	@${POSTPROC.${.TARGET:R}:Ucat} < ${.TARGET}.tmp1 > ${.TARGET}.tmp2
    413 	@rm ${.TARGET}.tmp1
    414 	@echo "exit status `cat ${.TARGET:R}.status`" >> ${.TARGET}.tmp2
    415 	@mv ${.TARGET}.tmp2 ${.TARGET}
    416 
    417 # Compare all output files
    418 test:	${OUTFILES} .PHONY
    419 	@failed= ; \
    420 	for test in ${TESTS}; do \
    421 	  diff -u ${UNIT_TESTS}/$${test}.exp $${test}.out \
    422 	  || failed="$${failed}$${failed:+ }$${test}" ; \
    423 	done ; \
    424 	if [ -n "$${failed}" ]; then \
    425 	  echo "Failed tests: $${failed}" ; false ; \
    426 	else \
    427 	  echo "All tests passed" ; \
    428 	fi
    429 
    430 accept:
    431 	@for test in ${TESTS}; do \
    432 	  cmp -s ${UNIT_TESTS}/$${test}.exp $${test}.out \
    433 	  || { echo "Replacing $${test}.exp" ; \
    434 	       cp $${test}.out ${UNIT_TESTS}/$${test}.exp ; } \
    435 	done
    436 
    437 SYNC_MI_AWK= \
    438 	BEGIN {								\
    439 	  testsdir = "usr.bin/make/unit-tests";				\
    440 	  linestart = "./usr/tests/" testsdir;				\
    441 	  fmt = linestart "/%s\ttests-usr.bin-tests\tcompattestfile,atf\\n"; \
    442 	  cmd = "cd " testsdir " && ls *.exp *.mk | xargs printf '" fmt "'" \
    443 	}								\
    444 	function startswith(s, prefix) {				\
    445 	  return substr(s, 1, length(prefix)) == prefix;		\
    446 	}								\
    447 	startswith($$1, linestart) && $$1 ~ /\.(exp|mk)$$/ { next }	\
    448 	{ print $$0 }							\
    449 	$$1 == linestart "/Makefile" { system(cmd) }
    450 
    451 sync-mi:
    452 	@set -eu;							\
    453 	cd "${MAKEFILE:tA:H}/../../..";					\
    454 	mi="distrib/sets/lists/tests/mi";				\
    455 	cvs update "$$mi";						\
    456 	awk ${SYNC_MI_AWK:Q} < "$$mi" > "$$mi.$@";			\
    457 	mv -f "$$mi.$@" "$$mi";						\
    458 	cvs diff "$$mi" || true
    459 
    460 .if exists(${TEST_MAKE})
    461 ${TESTS:=.rawout}: ${TEST_MAKE} ${.PARSEDIR}/Makefile
    462 .endif
    463 
    464 .-include <bsd.obj.mk>
    465