Makefile revision 1.106
11.106Srillig# $NetBSD: Makefile,v 1.106 2020/08/22 11:27:02 rillig Exp $ 21.1Ssjg# 31.1Ssjg# Unit tests for make(1) 41.60Srillig# 51.1Ssjg# The main targets are: 61.1Ssjg# 71.60Srillig# all: 81.60Srillig# run all the tests 91.60Srillig# test: 101.60Srillig# run 'all', and compare to expected results 111.60Srillig# accept: 121.60Srillig# move generated output to expected results 131.60Srillig# 141.61Srillig# Settable variables 151.61Srillig# 161.61Srillig# TEST_MAKE 171.61Srillig# The make program to be tested. 181.61Srillig# 191.60Srillig# 201.60Srillig# Adding a test case 211.60Srillig# 221.11Ssjg# Each feature should get its own set of tests in its own suitably 231.42Sapb# named makefile (*.mk), with its own set of expected results (*.exp), 241.61Srillig# and it should be added to the TESTS list. 251.60Srillig# 261.60Srillig# Any added files must also be added to src/distrib/sets/lists/tests/mi. 271.78Srillig# To do that, just run "make sync-mi" in this directory. 281.78Srillig# 291.78Srillig# A few *.mk files are helper files for other tests (such as include-sub.mk) 301.78Srillig# and are thus not added to TESTS. Such files must be ignored in 311.78Srillig# src/tests/usr.bin/make/t_make.sh. 321.60Srillig# 331.1Ssjg 341.61Srillig# Each test is in a sub-makefile. 351.61Srillig# Keep the list sorted. 361.92Srillig# Any test that is commented out must be ignored in 371.92Srillig# src/tests/usr.bin/make/t_make.sh as well. 381.92SrilligTESTS+= # archive # broken on FreeBSD, enabled in t_make.sh 391.75SrilligTESTS+= cmdline 401.61SrilligTESTS+= comment 411.93SrilligTESTS+= cond-cmp-numeric 421.93SrilligTESTS+= cond-cmp-string 431.103SrilligTESTS+= cond-func 441.93SrilligTESTS+= cond-func-commands 451.93SrilligTESTS+= cond-func-defined 461.93SrilligTESTS+= cond-func-empty 471.93SrilligTESTS+= cond-func-exists 481.93SrilligTESTS+= cond-func-make 491.93SrilligTESTS+= cond-func-target 501.61SrilligTESTS+= cond-late 511.93SrilligTESTS+= cond-op 521.93SrilligTESTS+= cond-op-and 531.93SrilligTESTS+= cond-op-not 541.93SrilligTESTS+= cond-op-or 551.93SrilligTESTS+= cond-op-parentheses 561.61SrilligTESTS+= cond-short 571.93SrilligTESTS+= cond-token-number 581.93SrilligTESTS+= cond-token-plain 591.93SrilligTESTS+= cond-token-string 601.93SrilligTESTS+= cond-token-var 611.61SrilligTESTS+= cond1 621.61SrilligTESTS+= cond2 631.81SrilligTESTS+= counter 641.93SrilligTESTS+= dep 651.93SrilligTESTS+= dep-colon 661.93SrilligTESTS+= dep-double-colon 671.93SrilligTESTS+= dep-exclam 681.93SrilligTESTS+= dep-wildcards 691.93SrilligTESTS+= depsrc 701.93SrilligTESTS+= depsrc-exec 711.93SrilligTESTS+= depsrc-ignore 721.93SrilligTESTS+= depsrc-made 731.93SrilligTESTS+= depsrc-make 741.93SrilligTESTS+= depsrc-meta 751.93SrilligTESTS+= depsrc-nometa 761.93SrilligTESTS+= depsrc-nometa_cmp 771.93SrilligTESTS+= depsrc-nopath 781.93SrilligTESTS+= depsrc-notmain 791.93SrilligTESTS+= depsrc-optional 801.93SrilligTESTS+= depsrc-phony 811.93SrilligTESTS+= depsrc-precious 821.93SrilligTESTS+= depsrc-recursive 831.93SrilligTESTS+= depsrc-silent 841.93SrilligTESTS+= depsrc-use 851.93SrilligTESTS+= depsrc-usebefore 861.105SrilligTESTS+= depsrc-usebefore-double-colon 871.93SrilligTESTS+= depsrc-wait 881.93SrilligTESTS+= deptgt 891.93SrilligTESTS+= deptgt-begin 901.93SrilligTESTS+= deptgt-default 911.93SrilligTESTS+= deptgt-delete_on_error 921.93SrilligTESTS+= deptgt-end 931.93SrilligTESTS+= deptgt-error 941.93SrilligTESTS+= deptgt-ignore 951.93SrilligTESTS+= deptgt-interrupt 961.93SrilligTESTS+= deptgt-main 971.93SrilligTESTS+= deptgt-makeflags 981.93SrilligTESTS+= deptgt-no_parallel 991.93SrilligTESTS+= deptgt-nopath 1001.93SrilligTESTS+= deptgt-notparallel 1011.93SrilligTESTS+= deptgt-objdir 1021.93SrilligTESTS+= deptgt-order 1031.93SrilligTESTS+= deptgt-path 1041.93SrilligTESTS+= deptgt-path-suffix 1051.93SrilligTESTS+= deptgt-phony 1061.93SrilligTESTS+= deptgt-precious 1071.93SrilligTESTS+= deptgt-shell 1081.93SrilligTESTS+= deptgt-silent 1091.93SrilligTESTS+= deptgt-stale 1101.93SrilligTESTS+= deptgt-suffixes 1111.77SrilligTESTS+= dir 1121.93SrilligTESTS+= directive 1131.93SrilligTESTS+= directive-elif 1141.93SrilligTESTS+= directive-elifdef 1151.93SrilligTESTS+= directive-elifmake 1161.93SrilligTESTS+= directive-elifndef 1171.93SrilligTESTS+= directive-elifnmake 1181.93SrilligTESTS+= directive-else 1191.93SrilligTESTS+= directive-endif 1201.93SrilligTESTS+= directive-error 1211.93SrilligTESTS+= directive-export 1221.93SrilligTESTS+= directive-export-env 1231.93SrilligTESTS+= directive-export-literal 1241.93SrilligTESTS+= directive-if 1251.93SrilligTESTS+= directive-ifdef 1261.93SrilligTESTS+= directive-ifmake 1271.93SrilligTESTS+= directive-ifndef 1281.93SrilligTESTS+= directive-ifnmake 1291.93SrilligTESTS+= directive-info 1301.93SrilligTESTS+= directive-undef 1311.93SrilligTESTS+= directive-unexport 1321.93SrilligTESTS+= directive-unexport-env 1331.93SrilligTESTS+= directive-warning 1341.70SrilligTESTS+= directives 1351.61SrilligTESTS+= dollar 1361.61SrilligTESTS+= doterror 1371.61SrilligTESTS+= dotwait 1381.64SrilligTESTS+= envfirst 1391.61SrilligTESTS+= error 1401.63SsjgTESTS+= # escape # broken by reverting POSIX changes 1411.61SrilligTESTS+= export 1421.61SrilligTESTS+= export-all 1431.61SrilligTESTS+= export-env 1441.89SrilligTESTS+= export-variants 1451.61SrilligTESTS+= forloop 1461.61SrilligTESTS+= forsubst 1471.61SrilligTESTS+= hash 1481.86SrilligTESTS+= impsrc 1491.61SrilligTESTS+= include-main 1501.83SrilligTESTS+= lint 1511.90SrilligTESTS+= make-exported 1521.61SrilligTESTS+= misc 1531.61SrilligTESTS+= moderrs 1541.61SrilligTESTS+= modmatch 1551.61SrilligTESTS+= modmisc 1561.61SrilligTESTS+= modts 1571.61SrilligTESTS+= modword 1581.93SrilligTESTS+= opt 1591.93SrilligTESTS+= opt-backwards 1601.93SrilligTESTS+= opt-chdir 1611.93SrilligTESTS+= opt-debug 1621.93SrilligTESTS+= opt-define 1631.93SrilligTESTS+= opt-env 1641.93SrilligTESTS+= opt-file 1651.93SrilligTESTS+= opt-ignore 1661.93SrilligTESTS+= opt-include-dir 1671.93SrilligTESTS+= opt-jobs 1681.93SrilligTESTS+= opt-jobs-internal 1691.93SrilligTESTS+= opt-keep-going 1701.93SrilligTESTS+= opt-m-include-dir 1711.93SrilligTESTS+= opt-no-action 1721.93SrilligTESTS+= opt-no-action-at-all 1731.93SrilligTESTS+= opt-query 1741.93SrilligTESTS+= opt-raw 1751.93SrilligTESTS+= opt-silent 1761.93SrilligTESTS+= opt-touch 1771.93SrilligTESTS+= opt-tracefile 1781.93SrilligTESTS+= opt-var-expanded 1791.93SrilligTESTS+= opt-var-literal 1801.93SrilligTESTS+= opt-warnings-as-errors 1811.93SrilligTESTS+= opt-where-am-i 1821.93SrilligTESTS+= opt-x-reduce-exported 1831.61SrilligTESTS+= order 1841.88SrilligTESTS+= phony-end 1851.61SrilligTESTS+= posix 1861.63SsjgTESTS+= # posix1 # broken by reverting POSIX changes 1871.61SrilligTESTS+= qequals 1881.84SrilligTESTS+= recursive 1891.93SrilligTESTS+= sh 1901.106SrilligTESTS+= sh-dots 1911.93SrilligTESTS+= sh-jobs 1921.93SrilligTESTS+= sh-jobs-error 1931.93SrilligTESTS+= sh-leading-at 1941.93SrilligTESTS+= sh-leading-hyphen 1951.93SrilligTESTS+= sh-leading-plus 1961.93SrilligTESTS+= sh-meta-chars 1971.93SrilligTESTS+= sh-multi-line 1981.93SrilligTESTS+= sh-single-line 1991.88SrilligTESTS+= # suffixes # runs into an endless loop (try -dA) 2001.61SrilligTESTS+= sunshcmd 2011.61SrilligTESTS+= sysv 2021.61SrilligTESTS+= ternary 2031.61SrilligTESTS+= unexport 2041.61SrilligTESTS+= unexport-env 2051.91SrilligTESTS+= use-inference 2061.93SrilligTESTS+= var-class 2071.93SrilligTESTS+= var-class-cmdline 2081.93SrilligTESTS+= var-class-env 2091.93SrilligTESTS+= var-class-global 2101.93SrilligTESTS+= var-class-local 2111.93SrilligTESTS+= var-class-local-legacy 2121.93SrilligTESTS+= var-op 2131.93SrilligTESTS+= var-op-append 2141.93SrilligTESTS+= var-op-assign 2151.93SrilligTESTS+= var-op-default 2161.93SrilligTESTS+= var-op-expand 2171.93SrilligTESTS+= var-op-shell 2181.61SrilligTESTS+= varcmd 2191.64SrilligTESTS+= vardebug 2201.64SrilligTESTS+= varfind 2211.61SrilligTESTS+= varmisc 2221.93SrilligTESTS+= varmod 2231.93SrilligTESTS+= varmod-assign 2241.93SrilligTESTS+= varmod-defined 2251.61SrilligTESTS+= varmod-edge 2261.93SrilligTESTS+= varmod-exclam-shell 2271.93SrilligTESTS+= varmod-extension 2281.93SrilligTESTS+= varmod-gmtime 2291.93SrilligTESTS+= varmod-hash 2301.93SrilligTESTS+= varmod-head 2311.93SrilligTESTS+= varmod-ifelse 2321.93SrilligTESTS+= varmod-l-name-to-value 2331.93SrilligTESTS+= varmod-localtime 2341.93SrilligTESTS+= varmod-loop 2351.93SrilligTESTS+= varmod-match 2361.99SrilligTESTS+= varmod-match-escape 2371.93SrilligTESTS+= varmod-no-match 2381.93SrilligTESTS+= varmod-order 2391.93SrilligTESTS+= varmod-order-reverse 2401.93SrilligTESTS+= varmod-order-shuffle 2411.93SrilligTESTS+= varmod-path 2421.93SrilligTESTS+= varmod-quote 2431.93SrilligTESTS+= varmod-quote-dollar 2441.93SrilligTESTS+= varmod-range 2451.93SrilligTESTS+= varmod-remember 2461.93SrilligTESTS+= varmod-root 2471.93SrilligTESTS+= varmod-select-words 2481.93SrilligTESTS+= varmod-shell 2491.93SrilligTESTS+= varmod-subst 2501.93SrilligTESTS+= varmod-subst-regex 2511.93SrilligTESTS+= varmod-sysv 2521.93SrilligTESTS+= varmod-tail 2531.93SrilligTESTS+= varmod-to-abs 2541.93SrilligTESTS+= varmod-to-lower 2551.93SrilligTESTS+= varmod-to-many-words 2561.93SrilligTESTS+= varmod-to-one-word 2571.93SrilligTESTS+= varmod-to-separator 2581.93SrilligTESTS+= varmod-to-upper 2591.93SrilligTESTS+= varmod-undefined 2601.93SrilligTESTS+= varmod-unique 2611.93SrilligTESTS+= varname 2621.93SrilligTESTS+= varname-dollar 2631.93SrilligTESTS+= varname-dot-alltargets 2641.93SrilligTESTS+= varname-dot-curdir 2651.93SrilligTESTS+= varname-dot-includedfromdir 2661.93SrilligTESTS+= varname-dot-includedfromfile 2671.93SrilligTESTS+= varname-dot-make-dependfile 2681.93SrilligTESTS+= varname-dot-make-expand_variables 2691.93SrilligTESTS+= varname-dot-make-exported 2701.93SrilligTESTS+= varname-dot-make-jobs 2711.93SrilligTESTS+= varname-dot-make-jobs-prefix 2721.93SrilligTESTS+= varname-dot-make-level 2731.93SrilligTESTS+= varname-dot-make-makefile_preference 2741.93SrilligTESTS+= varname-dot-make-makefiles 2751.93SrilligTESTS+= varname-dot-make-meta-bailiwick 2761.93SrilligTESTS+= varname-dot-make-meta-created 2771.93SrilligTESTS+= varname-dot-make-meta-files 2781.93SrilligTESTS+= varname-dot-make-meta-ignore_filter 2791.93SrilligTESTS+= varname-dot-make-meta-ignore_paths 2801.93SrilligTESTS+= varname-dot-make-meta-ignore_patterns 2811.93SrilligTESTS+= varname-dot-make-meta-prefix 2821.93SrilligTESTS+= varname-dot-make-mode 2831.93SrilligTESTS+= varname-dot-make-path_filemon 2841.93SrilligTESTS+= varname-dot-make-pid 2851.93SrilligTESTS+= varname-dot-make-ppid 2861.93SrilligTESTS+= varname-dot-make-save_dollars 2871.93SrilligTESTS+= varname-dot-makeoverrides 2881.93SrilligTESTS+= varname-dot-newline 2891.93SrilligTESTS+= varname-dot-objdir 2901.93SrilligTESTS+= varname-dot-parsedir 2911.93SrilligTESTS+= varname-dot-parsefile 2921.93SrilligTESTS+= varname-dot-path 2931.93SrilligTESTS+= varname-dot-targets 2941.93SrilligTESTS+= varname-make 2951.93SrilligTESTS+= varname-make_print_var_on_error 2961.93SrilligTESTS+= varname-makeflags 2971.93SrilligTESTS+= varname-pwd 2981.93SrilligTESTS+= varname-vpath 2991.66SrilligTESTS+= varparse-dynamic 3001.61SrilligTESTS+= varquote 3011.61SrilligTESTS+= varshell 3021.61Srillig 3031.98Srillig# Additional environment variables for some of the tests. 3041.98Srillig# The base environment is -i PATH="$PATH". 3051.64SrilligENV.envfirst= FROM_ENV=value-from-env 3061.65SrilligENV.varmisc= FROM_ENV=env 3071.65SrilligENV.varmisc+= FROM_ENV_BEFORE=env 3081.65SrilligENV.varmisc+= FROM_ENV_AFTER=env 3091.64Srillig 3101.64Srillig# Override make flags for some of the tests; default is -k. 3111.104SrilligFLAGS.archive= -dA 3121.97SrilligFLAGS.counter= -dv 3131.61SrilligFLAGS.doterror= # none 3141.68SrilligFLAGS.envfirst= -e 3151.97SrilligFLAGS.export= # none 3161.83SrilligFLAGS.lint= -dL -k 3171.102SrilligFLAGS.opt-no-action= -n 3181.101SrilligFLAGS.opt-query= -q 3191.61SrilligFLAGS.order= -j1 3201.97SrilligFLAGS.recursive= -dL 3211.64SrilligFLAGS.vardebug= -k -dv FROM_CMDLINE= 3221.99SrilligFLAGS.varmod-match-escape= -dv 3231.61Srillig 3241.61Srillig# Some tests need extra post-processing. 3251.94SrilligSED_CMDS.varmod-subst-regex+= \ 3261.94Srillig -e 's,\(Regex compilation error:\).*,\1 (details omitted),' 3271.79SrilligSED_CMDS.varmod-edge+= -e 's, line [0-9]*:, line omitted:,' 3281.61SrilligSED_CMDS.varshell+= -e 's,^[a-z]*sh: ,,' 3291.61SrilligSED_CMDS.varshell+= -e '/command/s,No such.*,not found,' 3301.61Srillig 3311.64Srillig# Some tests need an additional round of postprocessing. 3321.81SrilligPOSTPROC.counter= ${TOOL_SED} -n -e '/:RELEVANT = yes/,/:RELEVANT = no/p' 3331.64SrilligPOSTPROC.vardebug= ${TOOL_SED} -n -e '/:RELEVANT = yes/,/:RELEVANT = no/p' 3341.99SrilligPOSTPROC.varmod-match-escape= ${TOOL_SED} -n -e '/^Pattern/p' 3351.64Srillig 3361.99Srillig# Some tests reuse other tests, which makes them unnecessarily fragile. 3371.89Srilligexport-all.rawout: export.mk 3381.89Srilligunexport.rawout: export.mk 3391.89Srilligunexport-env.rawout: export.mk 3401.89Srillig 3411.61Srillig# End of the configuration section. 3421.61Srillig 3431.1Ssjg.MAIN: all 3441.1Ssjg 3451.61SrilligUNIT_TESTS:= ${.PARSEDIR} 3461.44Sapb.PATH: ${UNIT_TESTS} 3471.7Ssjg 3481.61SrilligOUTFILES= ${TESTS:=.out} 3491.42Sapb 3501.42Sapball: ${OUTFILES} 3511.9Ssjg 3521.95SrilligCLEANFILES= *.rawout *.out *.status *.tmp *.core *.tmp 3531.61SrilligCLEANFILES+= obj*.[och] lib*.a # posix1.mk 3541.61SrilligCLEANFILES+= issue* .[ab]* # suffixes.mk 3551.95SrilligCLEANDIRS= dir dummy # posix1.mk 3561.48Sapb 3571.1Ssjgclean: 3581.48Sapb rm -f ${CLEANFILES} 3591.95Srillig rm -rf ${CLEANDIRS} 3601.1Ssjg 3611.61SrilligTEST_MAKE?= ${.MAKE} 3621.61SrilligTOOL_SED?= sed 3631.1Ssjg 3641.28Ssjg# ensure consistent results from sort(1) 3651.61SrilligLC_ALL= C 3661.61SrilligLANG= C 3671.29Ssjg.export LANG LC_ALL 3681.28Ssjg 3691.42Sapb# the tests are actually done with sub-makes. 3701.42Sapb.SUFFIXES: .mk .rawout .out 3711.42Sapb.mk.rawout: 3721.96Srillig @${_MKMSG_TEST:Uecho '# test '} ${.PREFIX} 3731.64Srillig @set -eu; \ 3741.64Srillig cd ${.OBJDIR}; \ 3751.98Srillig env -i PATH="$$PATH" ${ENV.${.TARGET:R}} \ 3761.98Srillig ${TEST_MAKE} \ 3771.98Srillig -r -C ${.CURDIR} -f ${.IMPSRC} \ 3781.98Srillig ${FLAGS.${.TARGET:R}:U-k} \ 3791.98Srillig > ${.TARGET}.tmp 2>&1 \ 3801.64Srillig && status=$$? || status=$$?; \ 3811.64Srillig echo $$status > ${.TARGET:R}.status 3821.42Sapb @mv ${.TARGET}.tmp ${.TARGET} 3831.42Sapb 3841.62Srillig# Post-process the test output so that the results can be compared. 3851.62Srillig# 3861.62Srillig# always pretend .MAKE was called 'make' 3871.61Srillig_SED_CMDS+= -e 's,^${TEST_MAKE:T:S,.,\\.,g}[][0-9]*:,make:,' 3881.61Srillig_SED_CMDS+= -e 's,${TEST_MAKE:S,.,\\.,g},make,' 3891.62Srillig# replace anything after 'stopped in' with unit-tests 3901.61Srillig_SED_CMDS+= -e '/stopped/s, /.*, unit-tests,' 3911.62Srillig# strip ${.CURDIR}/ from the output 3921.61Srillig_SED_CMDS+= -e 's,${.CURDIR:S,.,\\.,g}/,,g' 3931.61Srillig_SED_CMDS+= -e 's,${UNIT_TESTS:S,.,\\.,g}/,,g' 3941.61Srillig 3951.42Sapb.rawout.out: 3961.61Srillig @${TOOL_SED} ${_SED_CMDS} ${SED_CMDS.${.TARGET:R}} \ 3971.64Srillig < ${.IMPSRC} > ${.TARGET}.tmp1 3981.74Ssjg @${POSTPROC.${.TARGET:R}:Ucat} < ${.TARGET}.tmp1 > ${.TARGET}.tmp2 3991.64Srillig @rm ${.TARGET}.tmp1 4001.64Srillig @echo "exit status `cat ${.TARGET:R}.status`" >> ${.TARGET}.tmp2 4011.64Srillig @mv ${.TARGET}.tmp2 ${.TARGET} 4021.42Sapb 4031.42Sapb# Compare all output files 4041.42Sapbtest: ${OUTFILES} .PHONY 4051.42Sapb @failed= ; \ 4061.61Srillig for test in ${TESTS}; do \ 4071.42Sapb diff -u ${UNIT_TESTS}/$${test}.exp $${test}.out \ 4081.42Sapb || failed="$${failed}$${failed:+ }$${test}" ; \ 4091.42Sapb done ; \ 4101.42Sapb if [ -n "$${failed}" ]; then \ 4111.42Sapb echo "Failed tests: $${failed}" ; false ; \ 4121.42Sapb else \ 4131.42Sapb echo "All tests passed" ; \ 4141.42Sapb fi 4151.1Ssjg 4161.1Ssjgaccept: 4171.61Srillig @for test in ${TESTS}; do \ 4181.42Sapb cmp -s ${UNIT_TESTS}/$${test}.exp $${test}.out \ 4191.42Sapb || { echo "Replacing $${test}.exp" ; \ 4201.42Sapb cp $${test}.out ${UNIT_TESTS}/$${test}.exp ; } \ 4211.42Sapb done 4221.1Ssjg 4231.78SrilligSYNC_MI_AWK= \ 4241.78Srillig BEGIN { \ 4251.78Srillig testsdir = "usr.bin/make/unit-tests"; \ 4261.78Srillig linestart = "./usr/tests/" testsdir; \ 4271.78Srillig fmt = linestart "/%s\ttests-usr.bin-tests\tcompattestfile,atf\\n"; \ 4281.78Srillig cmd = "cd " testsdir " && ls *.exp *.mk | xargs printf '" fmt "'" \ 4291.78Srillig } \ 4301.78Srillig function startswith(s, prefix) { \ 4311.78Srillig return substr(s, 1, length(prefix)) == prefix; \ 4321.78Srillig } \ 4331.78Srillig startswith($$1, linestart) && $$1 ~ /\.(exp|mk)$$/ { next } \ 4341.78Srillig { print $$0 } \ 4351.78Srillig $$1 == linestart "/Makefile" { system(cmd) } 4361.78Srillig 4371.78Srilligsync-mi: 4381.78Srillig @set -eu; \ 4391.78Srillig cd "${MAKEFILE:tA:H}/../../.."; \ 4401.78Srillig mi="distrib/sets/lists/tests/mi"; \ 4411.105Srillig cvs update "$$mi"; \ 4421.78Srillig awk ${SYNC_MI_AWK:Q} < "$$mi" > "$$mi.$@"; \ 4431.78Srillig mv -f "$$mi.$@" "$$mi"; \ 4441.78Srillig cvs diff "$$mi" || true 4451.78Srillig 4461.50Ssjg.if exists(${TEST_MAKE}) 4471.94Srillig${TESTS:=.rawout}: ${TEST_MAKE} ${.PARSEDIR}/Makefile 4481.50Ssjg.endif 4491.50Ssjg 4501.42Sapb.-include <bsd.obj.mk> 451