Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src Fix an (apparent) ancient ash bug, that was apparently fixed...
details: https://anonhg.NetBSD.org/src/rev/53a7a7bff7c3
branches: trunk
changeset: 998822:53a7a7bff7c3
user: kre <kre%NetBSD.org@localhost>
date: Sat May 04 02:52:22 2019 +0000
description:
Fix an (apparent) ancient ash bug, that was apparently fixed sometime
in the past, but managed to re-surface...
The expression "${0+\}}" should expand to "}" not "\}"
Almost all other shells handle it that way (incl FreeBSD & dash).
Issue pointed out by Martijn Dekker.
Add ATF sub-tests for the 4 old var expand operators (${var+word}
${var-word} ${var-word} and ${var?word} - including the forms
with the ':' included) and amongst those tests include test cases
for this issue, so if the bug tries to appear again, we can squash
it quicker. (The newer pattern matching operators are already
well tested as part of testing patterns.)
diffstat:
bin/sh/parser.c | 5 +-
tests/bin/sh/t_expand.sh | 276 ++++++++++++++++++++++++++++++++++++++++++++++-
2 files changed, 278 insertions(+), 3 deletions(-)
diffs (truncated from 327 to 300 lines):
diff -r 9c13fa3ae3bd -r 53a7a7bff7c3 bin/sh/parser.c
--- a/bin/sh/parser.c Sat May 04 02:06:59 2019 +0000
+++ b/bin/sh/parser.c Sat May 04 02:52:22 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: parser.c,v 1.167 2019/02/27 04:10:56 kre Exp $ */
+/* $NetBSD: parser.c,v 1.168 2019/05/04 02:52:22 kre Exp $ */
/*-
* Copyright (c) 1991, 1993
@@ -37,7 +37,7 @@
#if 0
static char sccsid[] = "@(#)parser.c 8.7 (Berkeley) 5/16/95";
#else
-__RCSID("$NetBSD: parser.c,v 1.167 2019/02/27 04:10:56 kre Exp $");
+__RCSID("$NetBSD: parser.c,v 1.168 2019/05/04 02:52:22 kre Exp $");
#endif
#endif /* not lint */
@@ -1974,6 +1974,7 @@
CVTRACE(DBG_LEXER, quotef==0, (" QF=1 "));
quotef = 1; /* current token is quoted */
if (quoted && c != '\\' && c != '`' &&
+ (c != '}' || varnest == 0) &&
c != '$' && (c != '"' || magicq)) {
/*
* retain the \ (which we *know* needs CTLESC)
diff -r 9c13fa3ae3bd -r 53a7a7bff7c3 tests/bin/sh/t_expand.sh
--- a/tests/bin/sh/t_expand.sh Sat May 04 02:06:59 2019 +0000
+++ b/tests/bin/sh/t_expand.sh Sat May 04 02:52:22 2019 +0000
@@ -1,4 +1,4 @@
-# $NetBSD: t_expand.sh,v 1.21 2019/04/10 08:13:11 kre Exp $
+# $NetBSD: t_expand.sh,v 1.22 2019/05/04 02:52:22 kre Exp $
#
# Copyright (c) 2007, 2009 The NetBSD Foundation, Inc.
# All rights reserved.
@@ -1332,10 +1332,283 @@
EOF
}
+check3()
+{
+ check "X=foo; ${1}" "$2" 0
+ check "X=; ${1}" "$3" 0
+ check "unset X; ${1}" "$4" 0
+}
+
+atf_test_case alternative
+alternative_head() {
+ atf_set descr 'Test various possibilities for ${var+xxx}'
+}
+alternative_body() {
+ reset alternative
+
+ # just to verify (validate) that the test method works as expected
+ # (this is currently the very first test performed in this test set)
+ check 'printf %s a b' ab 0 # 1
+
+ check3 'set -- ${X+bar}; echo "$#:$1"' 1:bar 1:bar 0: # 4
+ check3 'set -- ${X+}; echo "$#:$1"' 0: 0: 0: # 7
+ check3 'set -- ${X+""}; echo "$#:$1"' 1: 1: 0: # 10
+ check3 'set -- "${X+}"; echo "$#:$1"' 1: 1: 1: # 13
+ check3 'set -- "${X+bar}"; echo "$#:$1"' 1:bar 1:bar 1: # 16
+
+ check3 'set -- ${X+a b c}; echo "$#:$1"' 3:a 3:a 0: # 19
+ check3 'set -- ${X+"a b c"}; echo "$#:$1"' '1:a b c' '1:a b c' 0:
+ check3 'set -- "${X+a b c}"; echo "$#:$1"' '1:a b c' '1:a b c' 1:
+ check3 'set -- ${X+a b\ c}; echo "$#:$1"' 2:a 2:a 0: # 28
+ check3 'set -- ${X+"a b" c}; echo "$#:$1"' '2:a b' '2:a b' 0:
+
+ check3 'printf %s "" ${X+}' '' '' '' # 34
+ check3 'printf %s ""${X+bar}' bar bar '' # 37
+
+ check3 'Y=bar; printf %s ${X+x}${Y+y}' xy xy y # 40
+ check3 'Y=bar; printf %s ""${X+${Y+z}}' z z '' # 43
+ check3 'Y=; printf %s ""${X+${Y+z}}' z z '' # 46
+ check3 'unset Y; printf %s ""${X+${Y+z}}' '' '' '' # 49
+ check3 'Y=1; printf %s a ${X+"${Y+z}"}' az az a # 52
+
+ check3 'printf %s ${X+}x}' x} x} x} # 55
+ check3 'printf %s ${X+}}' } } } # 58
+ check3 'printf %s "" ${X+"}"x}' }x }x '' # 61
+ check3 'printf %s "" ${X+\}x}' }x }x '' # 64
+ check3 'printf %s "${X+\}x}"' }x }x '' # 67
+ check3 'printf %s "${X+\}}"' } } '' # 70
+
+ check3 'set -- ${X:+bar}; echo "$#:$1"' 1:bar 0: 0: # 73
+ check3 'set -- ${X:+}; echo "$#:$1"' 0: 0: 0: # 76
+ check3 'set -- ${X:+""}; echo "$#:$1"' 1: 0: 0: # 79
+ check3 'set -- "${X:+}"; echo "$#:$1"' 1: 1: 1: # 80
+ check3 'set -- "${X:+bar}"; echo "$#:$1"' 1:bar 1: 1: # 83
+
+ check3 'set -- ${X:+a b c}; echo "$#:$1"' 3:a 0: 0: # 86
+ check3 'set -- ${X:+"a b c"}; echo "$#:$1"' '1:a b c' 0: 0: # 89
+ check3 'set -- "${X:+a b c}"; echo "$#:$1"' '1:a b c' 1: 1: # 92
+ check3 'set -- ${X:+a b\ c}; echo "$#:$1"' 2:a 0: 0: # 95
+ check3 'set -- ${X:+"a b" c}; echo "$#:$1"' '2:a b' 0: 0: # 98
+
+ check3 'printf %s "" ${X:+}' '' '' '' #101
+ check3 'printf %s ""${X:+bar}' bar '' '' #104
+
+ check3 'Y=bar; printf %s ${X:+x}${Y:+y}' xy y y #107
+ check3 'Y=bar; printf %s ""${X:+${Y:+z}}' z '' '' #110
+ check3 'Y=; printf %s ""${X:+${Y+z}}' z '' '' #113
+ check3 'Y=; printf %s ""${X:+${Y:+z}}' '' '' '' #116
+ check3 'unset Y; printf %s ""${X:+${Y:+z}}' '' '' '' #119
+ check3 'Y=1; printf %s a ${X:+"${Y:+z}"}' az a a #122
+
+ check3 'printf %s ${X:+}x}' x} x} x} #125
+ check3 'printf %s ${X:+}}' } } } #128
+ check3 'printf %s "" ${X:+"}"x}' }x '' '' #131
+ check3 'printf %s "" ${X:+\}x}' }x '' '' #134
+ check3 'printf %s "${X:+\}x}"' }x '' '' #137
+ check3 'printf %s "${X:+\}}"' } '' '' #140
+
+ results
+}
+
+atf_test_case default
+default_head() {
+ atf_set descr 'Test various possibilities for ${var-xxx}'
+}
+default_body() {
+ reset default
+
+ check3 'set -- ${X-bar}; echo "$#:$1"' 1:foo 0: 1:bar # 3
+ check3 'set -- ${X-}; echo "$#:$1"' 1:foo 0: 0: # 6
+ check3 'set -- ${X-""}; echo "$#:$1"' 1:foo 0: 1: # 9
+ check3 'set -- "${X-}"; echo "$#:$1"' 1:foo 1: 1: # 12
+ check3 'set -- "${X-bar}"; echo "$#:$1"' 1:foo 1: 1:bar # 15
+
+ check3 'set -- ${X-a b c}; echo "$#:$1"' 1:foo 0: 3:a # 18
+ check3 'set -- ${X-"a b c"}; echo "$#:$1"' 1:foo 0: '1:a b c' #21
+ check3 'set -- "${X-a b c}"; echo "$#:$1"' 1:foo 1: '1:a b c' #24
+ check3 'set -- ${X-a b\ c}; echo "$#:$1"' 1:foo 0: 2:a # 27
+ check3 'set -- ${X-"a b" c}; echo "$#:$1"' 1:foo 0: '2:a b' #30
+
+ check3 'printf %s "" ${X-}' foo '' '' # 33
+ check3 'printf %s ""${X-bar}' foo '' bar # 36
+
+ check3 'Y=bar; printf %s ${X-x}${Y-y}' foobar bar xbar # 39
+ check3 'Y=bar; printf %s ""${X-${Y-z}}' foo '' bar # 42
+ check3 'Y=; printf %s ""${X-${Y-z}}' foo '' '' # 45
+ check3 'unset Y; printf %s ""${X-${Y-z}}' foo '' z # 48
+ check3 'Y=1; printf %s a ${X-"${Y-z}"}' afoo a a1 # 51
+
+ check3 'printf %s ${X-}x}' foox} x} x} # 54
+ check3 'printf %s ${X-}}' foo} } } # 57
+ check3 'printf %s ${X-{}}' foo} } {} # 60
+ check3 'printf %s "" ${X-"}"x}' foo '' }x # 63
+ check3 'printf %s "" ${X-\}x}' foo '' }x # 66
+ check3 'printf %s "${X-\}x}"' foo '' }x # 69
+ check3 'printf %s "${X-\}}"' foo '' } # 72
+
+ check3 'set -- ${X:-bar}; echo "$#:$1"' 1:foo 1:bar 1:bar #75
+ check3 'set -- ${X:-}; echo "$#:$1"' 1:foo 0: 0: # 78
+ check3 'set -- ${X:-""}; echo "$#:$1"' 1:foo 1: 1: # 81
+ check3 'set -- "${X:-}"; echo "$#:$1"' 1:foo 1: 1: # 84
+ check3 'set -- "${X:-bar}"; echo "$#:$1"' 1:foo 1:bar 1:bar #87
+
+ check3 'set -- ${X:-a b c}; echo "$#:$1"' 1:foo 3:a 3:a # 90
+ check3 'set -- ${X:-"a b c"}; echo "$#:$1"' 1:foo '1:a b c' '1:a b c'
+ check3 'set -- "${X:-a b c}"; echo "$#:$1"' 1:foo '1:a b c' '1:a b c'
+ check3 'set -- ${X:-a b\ c}; echo "$#:$1"' 1:foo 2:a 2:a # 99
+ check3 'set -- ${X:-"a b" c}; echo "$#:$1"' 1:foo '2:a b' '2:a b'
+
+ check3 'printf %s "" ${X:-}' foo '' '' #105
+ check3 'printf %s ""${X:-bar}' foo bar bar #108
+
+ check3 'Y=bar; printf %s ${X:-x}${Y:-y}' foobar xbar xbar #111
+ check3 'Y=bar; printf %s ""${X:-${Y:-z}}' foo bar bar #114
+ check3 'Y=; printf %s ""${X:-${Y-z}}' foo '' '' #117
+ check3 'Y=; printf %s ""${X:-${Y:-z}}' foo z z #120
+ check3 'unset Y; printf %s ""${X:-${Y:-z}}' foo z z #123
+ check3 'Y=1; printf %s a ${X:-"${Y:-z}"}' afoo a1 a1 #126
+
+ check3 'printf %s ${X:-}x}' foox} x} x} #129
+ check3 'printf %s ${X:-}}' foo} } } #132
+ check3 'printf %s ${X:-{}}' foo} {} {} #135
+ check3 'printf %s "" ${X:-"}"x}' foo }x }x #138
+ check3 'printf %s "" ${X:-\}x}' foo }x }x #141
+ check3 'printf %s "${X:-\}x}"' foo }x }x #144
+ check3 'printf %s "${X:-\}}"' foo } } #147
+
+ results
+}
+
+atf_test_case assign
+assign_head() {
+ atf_set descr 'Test various possibilities for ${var=xxx}'
+}
+assign_body() {
+ reset assign
+
+ check3 'set -- ${X=bar}; echo "$#:$1"' 1:foo 0: 1:bar # 3
+ check3 'set -- ${X=}; echo "$#:$1"' 1:foo 0: 0: # 6
+ check3 'set -- ${X=""}; echo "$#:$1"' 1:foo 0: 0: # 9
+ check3 'set -- "${X=}"; echo "$#:$1"' 1:foo 1: 1: # 12
+ check3 'set -- "${X=bar}"; echo "$#:$1"' 1:foo 1: 1:bar # 15
+
+ check3 'set -- ${X=a b c}; echo "$#:$1"' 1:foo 0: 3:a # 18
+ check3 'set -- ${X="a b c"}; echo "$#:$1"' 1:foo 0: 3:a # 21
+ check3 'set -- "${X=a b c}"; echo "$#:$1"' 1:foo 1: '1:a b c' #24
+ check3 'set -- ${X=a b\ c}; echo "$#:$1"' 1:foo 0: 3:a # 27
+ check3 'set -- ${X="a b" c}; echo "$#:$1"' 1:foo 0: 3:a # 30
+
+ check3 'printf %s "" ${X=}' foo '' '' # 33
+ check3 'printf %s ""${X=bar}' foo '' bar # 36
+
+ check3 'Y=bar; printf %s ${X=x}${Y=y}' foobar bar xbar # 39
+ check3 'Y=bar; printf %s ""${X=${Y=z}}' foo '' bar # 42
+ check3 'Y=; printf %s ""${X=${Y=z}}' foo '' '' # 45
+ check3 'unset Y; printf %s ""${X=${Y=z}}' foo '' z # 48
+ check3 'Y=1; printf %s a ${X="${Y=z}"}' afoo a a1 # 51
+
+ check3 'printf %s ${X=}x}' foox} x} x} # 54
+ check3 'printf %s ${X=}}' foo} } } # 57
+ check3 'printf %s ${X={}}' foo} } {} # 60
+ check3 'printf %s "" ${X="}"x}' foo '' }x # 63
+ check3 'printf %s "" ${X=\}x}' foo '' }x # 66
+ check3 'printf %s "${X=\}x}"' foo '' }x # 69
+ check3 'printf %s "${X=\}}"' foo '' } # 72
+
+ check3 'set -- ${X=a b c}; echo "$#:$1:$X"' 1:foo:foo 0:: '3:a:a b c'
+ check3 'set -- ${X="a b c"}; echo "$#:$1:$X"' 1:foo:foo 0:: '3:a:a b c'
+ check3 'set -- "${X=a b c}"; echo "$#:$1:$X"' \
+ 1:foo:foo 1:: '1:a b c:a b c'
+ check3 'set -- ${X=a b\ c}; echo "$#:$1:$X"' 1:foo:foo 0:: '3:a:a b c'
+ check3 'set -- ${X="a b" c}; echo "$#:$1:$X"' 1:foo:foo 0:: '3:a:a b c'
+
+ check3 'printf %s ${X=}x}; printf :%s "${X-U}"' foox}:foo x}: x}: #90
+ check3 'printf %s ${X=}}; printf :%s "${X-U}"' foo}:foo }: }: #93
+ check3 'printf %s ${X={}}; printf :%s "${X-U}"' foo}:foo }: {}:{ #96
+
+ check3 'set -- ${X:=bar}; echo "$#:$1"' 1:foo 1:bar 1:bar # 99
+ check3 'set -- ${X:=}; echo "$#:$1"' 1:foo 0: 0: #102
+ check3 'set -- ${X:=""}; echo "$#:$1"' 1:foo 0: 0: #105
+ check3 'set -- "${X:=}"; echo "$#:$1"' 1:foo 1: 1: #108
+ check3 'set -- "${X:=bar}"; echo "$#:$1"' 1:foo 1:bar 1:bar #111
+
+ check3 'set -- ${X:=a b c}; echo "$#:$1"' 1:foo 3:a 3:a #114
+ check3 'set -- ${X:="a b c"}; echo "$#:$1"' 1:foo 3:a 3:a #117
+ check3 'set -- "${X:=a b c}"; echo "$#:$1"' 1:foo '1:a b c' '1:a b c'
+ check3 'set -- ${X:=a b\ c}; echo "$#:$1"' 1:foo 3:a 3:a #123
+ check3 'set -- ${X:="a b" c}; echo "$#:$1"' 1:foo 3:a 3:a #126
+
+ check3 'printf %s "" ${X:=}' foo '' '' #129
+ check3 'printf %s ""${X:=bar}' foo bar bar #132
+
+ check3 'Y=bar; printf %s ${X:=x}${Y:=y}' foobar xbar xbar #135
+ check3 'Y=bar; printf %s ""${X:=${Y:=z}}' foo bar bar #138
+ check3 'Y=; printf %s ""${X:=${Y=z}}' foo '' '' #141
+ check3 'Y=; printf %s ""${X:=${Y:=z}}' foo z z #144
+ check3 'unset Y; printf %s ""${X:=${Y:=z}}' foo z z #147
+ check3 'Y=1; printf %s a ${X:="${Y:=z}"}' afoo a1 a1 #150
+
+ check3 'printf %s ${X:=}x}' foox} x} x} #153
+ check3 'printf %s ${X:=}}' foo} } } #156
+ check3 'printf %s ${X:={}}' foo} {} {} #159
+ check3 'printf %s "" ${X:="}"x}' foo }x }x #162
+ check3 'printf %s "" ${X:=\}x}' foo }x }x #165
+ check3 'printf %s "${X:=\}x}"' foo }x }x #168
+ check3 'printf %s "${X:=\}}"' foo } } #171
+
+ check3 'set -- ${X:=a b c}; echo "$#:$1:$X"' \
+ 1:foo:foo '3:a:a b c' '3:a:a b c' #174
+ check3 'set -- ${X:="a b c"}; echo "$#:$1:$X"' \
+ 1:foo:foo '3:a:a b c' '3:a:a b c' #177
+ check3 'set -- "${X:=a b c}"; echo "$#:$1:$X"' \
+ 1:foo:foo '1:a b c:a b c' '1:a b c:a b c' #180
+ check3 'set -- ${X:=a b\ c}; echo "$#:$1:$X"' \
+ 1:foo:foo '3:a:a b c' '3:a:a b c' #183
+ check3 'set -- ${X:="a b" c}; echo "$#:$1:$X"' \
+ 1:foo:foo '3:a:a b c' '3:a:a b c' #186
+
+ check3 'printf %s ${X:=}x}; printf :%s "${X-U}"' foox}:foo x}: x}:
+ check3 'printf %s ${X:=}}; printf :%s "${X-U}"' foo}:foo }: }:
+ check3 'printf %s ${X:=\}}; printf :%s "${X-U}"' foo:foo }:} }:}
+ check3 'printf %s ${X:={}}; printf :%s "${X-U}"' foo}:foo {}:{ {}:{
+ #198
+
+ results
+}
+
+atf_test_case error
+error_head() {
+ atf_set descr 'Test various possibilities for ${var?xxx}'
+}
+error_body() {
+ reset error
+
+ check 'X=foo; printf %s ${X?X is not set}' foo 0 #1
+ check 'X=; printf %s ${X?X is not set}' '' 0 #2
+ check 'unset X; printf %s ${X?X is not set}' '' 2 #3
+
+ check 'X=foo; printf %s ${X?}' foo 0 #4
+ check 'X=; printf %s ${X?}' '' 0 #5
+ check 'unset X; printf %s ${X?}' '' 2 #6
+
+ check 'X=foo; printf %s ${X:?X is not set}' foo 0 #7
+ check 'X=; printf %s ${X:?X is not set}' '' 2 #8
Home |
Main Index |
Thread Index |
Old Index