]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/bmake/unit-tests/varmod-indirect.mk
less: upgrade to v581.2.
[FreeBSD/FreeBSD.git] / contrib / bmake / unit-tests / varmod-indirect.mk
1 # $NetBSD: varmod-indirect.mk,v 1.5 2020/12/27 17:32:25 rillig Exp $
2 #
3 # Tests for indirect variable modifiers, such as in ${VAR:${M_modifiers}}.
4 # These can be used for very basic purposes like converting a string to either
5 # uppercase or lowercase, as well as for fairly advanced modifiers that first
6 # look like line noise and are hard to decipher.
7 #
8 # TODO: Since when are indirect modifiers supported?
9
10
11 # To apply a modifier indirectly via another variable, the whole
12 # modifier must be put into a single variable expression.
13 .if ${value:L:${:US}${:U,value,replacement,}} != "S,value,replacement,}"
14 .  warning unexpected
15 .endif
16
17
18 # Adding another level of indirection (the 2 nested :U expressions) helps.
19 .if ${value:L:${:U${:US}${:U,value,replacement,}}} != "replacement"
20 .  warning unexpected
21 .endif
22
23
24 # Multiple indirect modifiers can be applied one after another as long as
25 # they are separated with colons.
26 .if ${value:L:${:US,a,A,}:${:US,e,E,}} != "vAluE"
27 .  warning unexpected
28 .endif
29
30
31 # An indirect variable that evaluates to the empty string is allowed though.
32 # This makes it possible to define conditional modifiers, like this:
33 #
34 # M.little-endian=      S,1234,4321,
35 # M.big-endian=         # none
36 .if ${value:L:${:Dempty}S,a,A,} != "vAlue"
37 .  warning unexpected
38 .endif
39
40
41 # The nested variable expression expands to "tu", and this is interpreted as
42 # a variable modifier for the value "Upper", resulting in "UPPER".
43 .if ${Upper:L:${:Utu}} != "UPPER"
44 .  error
45 .endif
46
47 # The nested variable expression expands to "tl", and this is interpreted as
48 # a variable modifier for the value "Lower", resulting in "lower".
49 .if ${Lower:L:${:Utl}} != "lower"
50 .  error
51 .endif
52
53
54 # The nested variable expression is ${1 != 1:?Z:tl}, consisting of the
55 # condition "1 != 1", the then-branch "Z" and the else-branch "tl".  Since
56 # the condition evaluates to false, the then-branch is ignored (it would
57 # have been an unknown modifier anyway) and the ":tl" modifier is applied.
58 .if ${Mixed:L:${1 != 1:?Z:tl}} != "mixed"
59 .  error
60 .endif
61
62
63 # The indirect modifier can also replace an ':L' modifier, which allows for
64 # brain twisters since by reading the expression alone, it is not possible
65 # to say whether the variable name will be evaluated as a variable name or
66 # as the immediate value of the expression.
67 VAR=    value
68 M_ExpandVar=    # an empty modifier
69 M_VarAsValue=   L
70 #
71 .if ${VAR:${M_ExpandVar}} != "value"
72 .  error
73 .endif
74 .if ${VAR:${M_VarAsValue}} != "VAR"
75 .  error
76 .endif
77
78 # The indirect modifier M_ListToSkip, when applied to a list of patterns,
79 # expands to a sequence of ':N' modifiers, each of which filters one of the
80 # patterns.  This list of patterns can then be applied to another variable
81 # to actually filter that variable.
82 #
83 M_ListToSkip=   @pat@N$${pat}@:ts:
84 #
85 # The dollar signs need to be doubled in the above modifier expression,
86 # otherwise they would be expanded too early, that is, when parsing the
87 # modifier itself.
88 #
89 # In the following example, M_NoPrimes expands to 'N2:N3:N5:N7:N1[1379]'.
90 # The 'N' comes from the expression 'N${pat}', the separating colons come
91 # from the modifier ':ts:'.
92 #
93 #.MAKEFLAGS: -dcv               # Uncomment this line to see the details
94 #
95 PRIMES=         2 3 5 7 1[1379]
96 M_NoPrimes=     ${PRIMES:${M_ListToSkip}}
97 .if ${:U:range=20:${M_NoPrimes}} != "1 4 6 8 9 10 12 14 15 16 18 20"
98 .  error
99 .endif
100 .MAKEFLAGS: -d0
101
102
103 # In contrast to the .if conditions, the .for loop allows undefined variable
104 # expressions.  These expressions expand to empty strings.
105
106 # An undefined expression without any modifiers expands to an empty string.
107 .for var in before ${UNDEF} after
108 .  info ${var}
109 .endfor
110
111 # An undefined expression with only modifiers that keep the expression
112 # undefined expands to an empty string.
113 .for var in before ${UNDEF:${:US,a,a,}} after
114 .  info ${var}
115 .endfor
116
117 # Even in an indirect modifier based on an undefined variable, the value of
118 # the expression in Var_Parse is a simple empty string.
119 .for var in before ${UNDEF:${:U}} after
120 .  info ${var}
121 .endfor
122
123 # An error in an indirect modifier.
124 .for var in before ${UNDEF:${:UZ}} after
125 .  info ${var}
126 .endfor
127
128
129 # Another slightly different evaluation context is the right-hand side of
130 # a variable assignment using ':='.
131 .MAKEFLAGS: -dpv
132
133 # The undefined variable expression is kept as-is.
134 _:=     before ${UNDEF} after
135
136 # The undefined variable expression is kept as-is.
137 _:=     before ${UNDEF:${:US,a,a,}} after
138
139 # XXX: The subexpression ${:U} is fully defined, therefore it is expanded.
140 # This results in ${UNDEF:}, which can lead to tricky parse errors later,
141 # when the variable '_' is expanded further.
142 #
143 # XXX: What should be the correct strategy here?  One possibility is to
144 # expand the defined subexpression and replace it with ${:U...}, just like
145 # in .for loops.  This would preserve the structure of the expression while
146 # at the same time expanding the expression as far as possible.
147 _:=     before ${UNDEF:${:U}} after
148
149 # XXX: This expands to ${UNDEF:Z}, which will behave differently if the
150 # variable '_' is used in a context where the variable expression ${_} is
151 # parsed but not evaluated.
152 _:=     before ${UNDEF:${:UZ}} after
153
154 .MAKEFLAGS: -d0
155 .undef _
156
157 all: