2016-07-13 19:28:56 +00:00
|
|
|
// Copyright 2016 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
function checkIsRedeclarationError(code) {
|
|
|
|
try {
|
|
|
|
eval(`
|
2019-01-30 05:47:03 +00:00
|
|
|
checkIsRedeclarationError: {
|
|
|
|
break checkIsRedeclarationError;
|
|
|
|
${code}
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
assertUnreachable();
|
2019-01-30 05:47:03 +00:00
|
|
|
} catch (e) {
|
|
|
|
assertInstanceof(e, SyntaxError);
|
|
|
|
assertTrue(e.toString().includes("has already been declared"));
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function checkIsNotRedeclarationError(code) {
|
2019-01-30 05:47:03 +00:00
|
|
|
assertDoesNotThrow(() => eval(`
|
|
|
|
checkIsNotRedeclarationError_label: {
|
|
|
|
break checkIsNotRedeclarationError_label;
|
|
|
|
${code}
|
|
|
|
}
|
|
|
|
`));
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
let var_e = [
|
|
|
|
'var e',
|
|
|
|
'var {e}',
|
|
|
|
'var {f, e}',
|
|
|
|
'var [e]',
|
|
|
|
'var {f:e}',
|
|
|
|
'var [[[], e]]'
|
|
|
|
];
|
|
|
|
|
|
|
|
let not_var_e = [
|
|
|
|
'var f',
|
|
|
|
'var {}',
|
|
|
|
'var {e:f}',
|
|
|
|
'e',
|
|
|
|
'{e}',
|
|
|
|
'let e',
|
|
|
|
'const e',
|
|
|
|
'let {e}',
|
|
|
|
'const {e}',
|
|
|
|
'let [e]',
|
|
|
|
'const [e]',
|
|
|
|
'let {f:e}',
|
|
|
|
'const {f:e}'
|
|
|
|
];
|
|
|
|
|
2019-01-30 05:47:03 +00:00
|
|
|
// Check that both `for (var ... of ...)` and `for (var ... in ...)`
|
|
|
|
// can redeclare a simple catch variable.
|
2016-07-13 19:28:56 +00:00
|
|
|
for (let binding of var_e) {
|
2019-01-30 05:47:03 +00:00
|
|
|
checkIsNotRedeclarationError(`
|
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
for (${binding} of []);
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
|
|
|
|
checkIsNotRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
for (${binding} in []);
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
2019-01-30 05:47:03 +00:00
|
|
|
// Check that the above applies even for nested catches.
|
2016-07-13 19:28:56 +00:00
|
|
|
for (let binding of var_e) {
|
2019-01-30 05:47:03 +00:00
|
|
|
checkIsNotRedeclarationError(`
|
2016-07-13 19:28:56 +00:00
|
|
|
try {
|
2019-01-30 05:47:03 +00:00
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
try {
|
|
|
|
throw 1;
|
|
|
|
} catch (f) {
|
|
|
|
try {
|
|
|
|
throw 2;
|
|
|
|
} catch ({}) {
|
|
|
|
for (${binding} of []);
|
|
|
|
}
|
|
|
|
}
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
2019-01-30 05:47:03 +00:00
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
|
|
|
|
checkIsNotRedeclarationError(`
|
|
|
|
try {
|
2019-01-30 05:47:03 +00:00
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
try {
|
|
|
|
throw 1;
|
|
|
|
} catch (f) {
|
|
|
|
try {
|
|
|
|
throw 2;
|
|
|
|
} catch ({}) {
|
|
|
|
for (${binding} in []);
|
|
|
|
}
|
|
|
|
}
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
2019-01-30 05:47:03 +00:00
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
2019-01-30 05:47:03 +00:00
|
|
|
// Check that the above applies if a declaration scope is between the
|
|
|
|
// catch and the loop.
|
2016-07-13 19:28:56 +00:00
|
|
|
for (let binding of var_e) {
|
|
|
|
checkIsNotRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
(()=>{for (${binding} of []);})();
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
|
|
|
|
checkIsNotRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
(function() {
|
|
|
|
for (${binding} of []);
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that there is no error when not declaring a var named e.
|
|
|
|
for (let binding of not_var_e) {
|
|
|
|
checkIsNotRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch (e) {
|
|
|
|
for (${binding} of []);
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that there is an error for both for-in and for-of when redeclaring
|
2019-01-30 05:47:03 +00:00
|
|
|
// a non-simple catch parameter.
|
2016-07-13 19:28:56 +00:00
|
|
|
for (let binding of var_e) {
|
|
|
|
checkIsRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch ({e}) {
|
|
|
|
for (${binding} of []);
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
|
|
|
|
checkIsRedeclarationError(`
|
2019-01-30 05:47:03 +00:00
|
|
|
try {
|
|
|
|
throw 0;
|
|
|
|
} catch ({e}) {
|
|
|
|
for (${binding} in []);
|
|
|
|
}
|
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the above error occurs even for nested catches.
|
|
|
|
for (let binding of var_e) {
|
|
|
|
checkIsRedeclarationError(`
|
|
|
|
try {
|
2019-01-30 05:47:03 +00:00
|
|
|
throw 0;
|
|
|
|
} catch ({e}) {
|
|
|
|
try {
|
|
|
|
throw 1;
|
|
|
|
} catch (f) {
|
|
|
|
try {
|
|
|
|
throw 2;
|
|
|
|
} catch ({}) {
|
|
|
|
for (${binding} of []);
|
|
|
|
}
|
|
|
|
}
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
2019-01-30 05:47:03 +00:00
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
|
|
|
|
checkIsRedeclarationError(`
|
|
|
|
try {
|
2019-01-30 05:47:03 +00:00
|
|
|
throw 0;
|
|
|
|
} catch ({e}) {
|
|
|
|
try {
|
|
|
|
throw 1;
|
|
|
|
} catch (f) {
|
|
|
|
try {
|
|
|
|
throw 2;
|
|
|
|
} catch ({}) {
|
|
|
|
for (${binding} in []);
|
|
|
|
}
|
|
|
|
}
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|
2019-01-30 05:47:03 +00:00
|
|
|
`);
|
2016-07-13 19:28:56 +00:00
|
|
|
}
|