Home | History | Annotate | Line # | Download | only in unit-tests
varmisc.mk revision 1.17
      1  1.17  rillig # $Id: varmisc.mk,v 1.17 2020/07/26 21:46:10 rillig Exp $
      2   1.1     apb #
      3   1.1     apb # Miscellaneous variable tests.
      4   1.1     apb 
      5   1.6     sjg all: unmatched_var_paren D_true U_true D_false U_false Q_lhs Q_rhs NQ_none \
      6   1.9     sjg 	strftime cmpv manok
      7  1.12  rillig all: save-dollars
      8  1.14  rillig all: export-appended
      9  1.15  rillig all: parse-dynamic
     10  1.16  rillig all: varerror-unclosed
     11   1.1     apb 
     12   1.1     apb unmatched_var_paren:
     13   1.2     sjg 	@echo ${foo::=foo-text}
     14   1.3     sjg 
     15   1.4     sjg True = ${echo true >&2:L:sh}TRUE
     16   1.4     sjg False= ${echo false >&2:L:sh}FALSE
     17   1.3     sjg 
     18   1.3     sjg VSET= is set
     19   1.3     sjg .undef UNDEF
     20   1.3     sjg 
     21   1.3     sjg U_false:
     22   1.3     sjg 	@echo :U skipped when var set
     23   1.3     sjg 	@echo ${VSET:U${False}}
     24   1.3     sjg 
     25   1.3     sjg D_false:
     26   1.3     sjg 	@echo :D skipped if var undef
     27   1.3     sjg 	@echo ${UNDEF:D${False}}
     28   1.3     sjg 
     29   1.3     sjg U_true:
     30   1.3     sjg 	@echo :U expanded when var undef
     31   1.3     sjg 	@echo ${UNDEF:U${True}}
     32   1.3     sjg 
     33   1.3     sjg D_true:
     34   1.3     sjg 	@echo :D expanded when var set
     35   1.3     sjg 	@echo ${VSET:D${True}}
     36   1.3     sjg 
     37   1.3     sjg Q_lhs:
     38   1.3     sjg 	@echo :? only lhs when value true
     39   1.3     sjg 	@echo ${1:L:?${True}:${False}}
     40   1.3     sjg 
     41   1.3     sjg Q_rhs:
     42   1.3     sjg 	@echo :? only rhs when value false
     43   1.3     sjg 	@echo ${0:L:?${True}:${False}}
     44   1.5     sjg 
     45   1.5     sjg NQ_none:
     46   1.5     sjg 	@echo do not evaluate or expand :? if discarding
     47   1.5     sjg 	@echo ${VSET:U${1:L:?${True}:${False}}}
     48   1.6     sjg 
     49   1.6     sjg April1= 1459494000
     50   1.6     sjg 
     51   1.6     sjg # slightly contorted syntax to use utc via variable
     52   1.6     sjg strftime:
     53   1.6     sjg 	@echo ${year=%Y month=%m day=%d:L:gmtime=1459494000}
     54   1.6     sjg 	@echo date=${%Y%m%d:L:${gmtime=${April1}:L}}
     55   1.6     sjg 
     56   1.7     sjg # big jumps to handle 3 digits per step
     57   1.8     sjg M_cmpv.units = 1 1000 1000000
     58   1.7     sjg M_cmpv = S,., ,g:_:range:@i@+ $${_:[-$$i]} \* $${M_cmpv.units:[$$i]}@:S,^,expr 0 ,1:sh
     59   1.7     sjg 
     60   1.8     sjg Version = 123.456.789
     61   1.8     sjg cmpv.only = target specific vars
     62   1.7     sjg 
     63   1.7     sjg cmpv:
     64   1.7     sjg 	@echo Version=${Version} == ${Version:${M_cmpv}}
     65   1.7     sjg 	@echo Literal=3.4.5 == ${3.4.5:L:${M_cmpv}}
     66   1.8     sjg 	@echo We have ${${.TARGET:T}.only}
     67   1.9     sjg 
     68   1.9     sjg # catch misshandling of nested vars in .for loop
     69   1.9     sjg MAN=
     70   1.9     sjg MAN1= make.1
     71   1.9     sjg .for s in 1 2
     72   1.9     sjg .if defined(MAN$s) && !empty(MAN$s)
     73   1.9     sjg MAN+= ${MAN$s}
     74   1.9     sjg .endif
     75   1.9     sjg .endfor
     76   1.9     sjg 
     77   1.9     sjg manok:
     78   1.9     sjg 	@echo MAN=${MAN}
     79  1.10  rillig 
     80  1.10  rillig # This is an expanded variant of the above .for loop.
     81  1.11  rillig # Between 2020-06-28 and 2020-07-02 this paragraph generated a wrong
     82  1.10  rillig # error message "Variable VARNAME is recursive".
     83  1.10  rillig # When evaluating the !empty expression, the ${:U1} was not expanded and
     84  1.10  rillig # thus resulted in the seeming definition VARNAME=${VARNAME}, which is
     85  1.10  rillig # obviously recursive.
     86  1.10  rillig VARNAME=	${VARNAME${:U1}}
     87  1.10  rillig .if defined(VARNAME${:U2}) && !empty(VARNAME${:U2})
     88  1.10  rillig .endif
     89  1.12  rillig 
     90  1.12  rillig # begin .MAKE.SAVE_DOLLARS; see Var_Set_with_flags and s2Boolean.
     91  1.12  rillig SD_VALUES=	0 1 2 False True false true Yes No yes no On Off ON OFF on off
     92  1.12  rillig SD_4_DOLLARS=	$$$$
     93  1.12  rillig 
     94  1.12  rillig .for val in ${SD_VALUES}
     95  1.12  rillig .MAKE.SAVE_DOLLARS:=	${val}	# Must be := since a simple = has no effect.
     96  1.12  rillig SD.${val}:=		${SD_4_DOLLARS}
     97  1.12  rillig .endfor
     98  1.12  rillig .MAKE.SAVE_DOLLARS:=	yes
     99  1.12  rillig 
    100  1.12  rillig save-dollars:
    101  1.12  rillig .for val in ${SD_VALUES}
    102  1.12  rillig 	@printf '%s: %-8s = %s\n' $@ ${val} ${SD.${val}:Q}
    103  1.12  rillig .endfor
    104  1.13  rillig 
    105  1.13  rillig # Appending to an undefined variable does not add a space in front.
    106  1.13  rillig .undef APPENDED
    107  1.13  rillig APPENDED+=	value
    108  1.13  rillig .if ${APPENDED} != "value"
    109  1.13  rillig .error "${APPENDED}"
    110  1.13  rillig .endif
    111  1.13  rillig 
    112  1.13  rillig # Appending to an empty variable adds a space between the old value
    113  1.13  rillig # and the additional value.
    114  1.13  rillig APPENDED=	# empty
    115  1.13  rillig APPENDED+=	value
    116  1.13  rillig .if ${APPENDED} != " value"
    117  1.13  rillig .error "${APPENDED}"
    118  1.13  rillig .endif
    119  1.13  rillig 
    120  1.13  rillig # Appending to parameterized variables works as well.
    121  1.13  rillig PARAM=		param
    122  1.13  rillig VAR.${PARAM}=	1
    123  1.13  rillig VAR.${PARAM}+=	2
    124  1.13  rillig .if ${VAR.param} != "1 2"
    125  1.13  rillig .error "${VAR.param}"
    126  1.13  rillig .endif
    127  1.13  rillig 
    128  1.13  rillig # The variable name can contain arbitrary characters.
    129  1.13  rillig # If the expanded variable name ends in a +, this still does not influence
    130  1.13  rillig # the parser. The assignment operator is still a simple assignment.
    131  1.13  rillig # Therefore, there is no need to add a space between the variable name
    132  1.13  rillig # and the assignment operator.
    133  1.13  rillig PARAM=		+
    134  1.13  rillig VAR.${PARAM}=	1
    135  1.13  rillig VAR.${PARAM}+=	2
    136  1.13  rillig .if ${VAR.+} != "1 2"
    137  1.13  rillig .error "${VAR.+}"
    138  1.13  rillig .endif
    139  1.13  rillig .for param in + ! ?
    140  1.13  rillig VAR.${param}=	${param}
    141  1.13  rillig .endfor
    142  1.13  rillig .if ${VAR.+} != "+" || ${VAR.!} != "!" || ${VAR.?} != "?"
    143  1.13  rillig .error "${VAR.+}" "${VAR.!}" "${VAR.?}"
    144  1.13  rillig .endif
    145  1.14  rillig 
    146  1.14  rillig # Appending to a variable from the environment creates a copy of that variable
    147  1.14  rillig # in the global context.
    148  1.14  rillig # The appended value is not exported automatically.
    149  1.14  rillig # When a variable is exported, the exported value is taken at the time of the
    150  1.14  rillig # .export directive. Later changes to the variable have no effect.
    151  1.14  rillig .export FROM_ENV_BEFORE
    152  1.14  rillig FROM_ENV+=		mk
    153  1.14  rillig FROM_ENV_BEFORE+=	mk
    154  1.14  rillig FROM_ENV_AFTER+=	mk
    155  1.14  rillig .export FROM_ENV_AFTER
    156  1.14  rillig 
    157  1.14  rillig export-appended:
    158  1.14  rillig 	@echo $@: "$$FROM_ENV"
    159  1.14  rillig 	@echo $@: "$$FROM_ENV_BEFORE"
    160  1.14  rillig 	@echo $@: "$$FROM_ENV_AFTER"
    161  1.15  rillig 
    162  1.15  rillig # begin parse-dynamic
    163  1.15  rillig #
    164  1.15  rillig # Demonstrate that the target-specific variables are not evaluated in
    165  1.15  rillig # the global context. They are preserved until there is a local context
    166  1.15  rillig # in which resolving them makes sense.
    167  1.15  rillig 
    168  1.17  rillig # There are different code paths for short names ...
    169  1.15  rillig ${:U>}=		before
    170  1.17  rillig GS_TARGET:=	$@
    171  1.17  rillig GS_MEMBER:=	$%
    172  1.17  rillig GS_PREFIX:=	$*
    173  1.17  rillig GS_ARCHIVE:=	$!
    174  1.17  rillig GS_ALLSRC:=	$>
    175  1.15  rillig ${:U>}=		after
    176  1.17  rillig # ... and for braced short names ...
    177  1.17  rillig GB_TARGET:=	${@}
    178  1.17  rillig GB_MEMBER:=	${%}
    179  1.17  rillig GB_PREFIX:=	${*}
    180  1.17  rillig GB_ARCHIVE:=	${!}
    181  1.17  rillig GB_ALLSRC:=	${>}
    182  1.17  rillig # ... and for long names.
    183  1.17  rillig GL_TARGET:=	${.TARGET}
    184  1.17  rillig GL_MEMBER:=	${.MEMBER}
    185  1.17  rillig GL_PREFIX:=	${.PREFIX}
    186  1.17  rillig GL_ARCHIVE:=	${.ARCHIVE}
    187  1.17  rillig GL_ALLSRC:=	${.ALLSRC}
    188  1.15  rillig 
    189  1.15  rillig parse-dynamic:
    190  1.17  rillig 	@echo $@: ${GS_TARGET} ${GS_MEMBER} ${GS_PREFIX} ${GS_ARCHIVE} ${GS_ALLSRC}
    191  1.17  rillig 	@echo $@: ${GB_TARGET} ${GB_MEMBER} ${GB_PREFIX} ${GB_ARCHIVE} ${GB_ALLSRC}
    192  1.17  rillig 	@echo $@: ${GL_TARGET} ${GL_MEMBER} ${GL_PREFIX} ${GL_ARCHIVE} ${GL_ALLSRC}
    193  1.16  rillig 
    194  1.16  rillig # As of 2020-07-26, make does not complain about unclosed variables.
    195  1.16  rillig # It does complain about unclosed variables when parsing modifiers though.
    196  1.16  rillig varerror-unclosed:
    197  1.16  rillig 	@echo $@:begin
    198  1.16  rillig 	@echo ${UNCLOSED
    199  1.16  rillig 	@echo ${UNCLOSED:M${PATTERN
    200  1.16  rillig 	@echo ${UNCLOSED.${param
    201  1.16  rillig 	@echo $@:end
    202