// 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 TestDefaultBeforeInitializingYield() { var y = 0; var z = 0; function* f1(x = (y = 1)) { z = 1 }; assertEquals(0, y); assertEquals(0, z); var gen = f1(); assertEquals(1, y); assertEquals(0, z); gen.next(); assertEquals(1, y); assertEquals(1, z); })(); (function TestShadowingOfParameters() { function* f1({x}) { var x = 2; return x } assertEquals(2, f1({x: 1}).next().value); function* f2({x}) { { var x = 2; } return x; } assertEquals(2, f2({x: 1}).next().value); function* f3({x}) { var y = x; var x = 2; return y; } assertEquals(1, f3({x: 1}).next().value); function* f4({x}) { { var y = x; var x = 2; } return y; } assertEquals(1, f4({x: 1}).next().value); function* f5({x}, g = () => x) { var x = 2; return g(); } assertEquals(1, f5({x: 1}).next().value); function* f6({x}, g = () => x) { { var x = 2; } return g(); } assertEquals(1, f6({x: 1}).next().value); function* f7({x}) { var g = () => x; var x = 2; return g(); } assertEquals(2, f7({x: 1}).next().value); function* f8({x}) { { var g = () => x; var x = 2; } return g(); } assertEquals(2, f8({x: 1}).next().value); function* f9({x}, g = () => eval("x")) { var x = 2; return g(); } assertEquals(1, f9({x: 1}).next().value); function* f10({x}, y) { var y; return y } assertEquals(2, f10({x: 6}, 2).next().value); function* f11({x}, y) { var z = y; var y = 2; return z; } assertEquals(1, f11({x: 6}, 1).next().value); function* f12(y, g = () => y) { var y = 2; return g(); } assertEquals(1, f12(1).next().value); function* f13({x}, y, [z], v) { var x, y, z; return x*y*z*v } assertEquals(210, f13({x: 2}, 3, [5], 7).next().value); function* f20({x}) { function x() { return 2 }; return x(); } assertEquals(2, f20({x: 1}).next().value); // Annex B 3.3 function hoisting is blocked by the conflicting x declaration function* f21({x}) { { function x() { return 2 } } return x; } assertEquals(1, f21({x: 1}).next().value); // These errors are not recognized in lazy parsing; see mjsunit/bugs/bug-2728.js assertThrows("'use strict'; (function* f(x) { let x = 0; })()", SyntaxError); assertThrows("'use strict'; (function* f({x}) { let x = 0; })()", SyntaxError); assertThrows("'use strict'; (function* f(x) { const x = 0; })()", SyntaxError); assertThrows("'use strict'; (function* f({x}) { const x = 0; })()", SyntaxError); }()); (function TestDefaults() { function* f1(x = 1) { return x } assertEquals(1, f1().next().value); assertEquals(1, f1(undefined).next().value); assertEquals(2, f1(2).next().value); assertEquals(null, f1(null).next().value); function* f2(x, y = x) { return x + y; } assertEquals(8, f2(4).next().value); assertEquals(8, f2(4, undefined).next().value); assertEquals(6, f2(4, 2).next().value); function* f3(x = 1, y) { return x + y; } assertEquals(8, f3(5, 3).next().value); assertEquals(3, f3(undefined, 2).next().value); assertEquals(6, f3(4, 2).next().value); function* f4(x = () => 1) { return x() } assertEquals(1, f4().next().value); assertEquals(1, f4(undefined).next().value); assertEquals(2, f4(() => 2).next().value); assertThrows(() => f4(null).next(), TypeError); function* f5(x, y = () => x) { return x + y(); } assertEquals(8, f5(4).next().value); assertEquals(8, f5(4, undefined).next().value); assertEquals(6, f5(4, () => 2).next().value); function* f6(x = {a: 1, m() { return 2 }}) { return x.a + x.m(); } assertEquals(3, f6().next().value); assertEquals(3, f6(undefined).next().value); assertEquals(5, f6({a: 2, m() { return 3 }}).next().value); }()); (function TestEvalInParameters() { function* f1(x = eval(0)) { return x } assertEquals(0, f1().next().value); function* f2(x = () => eval(1)) { return x() } assertEquals(1, f2().next().value); })(); (function TestParameterScopingSloppy() { var x = 1; function* f1(a = x) { var x = 2; return a; } assertEquals(1, f1().next().value); function* f2(a = x) { function x() {}; return a; } assertEquals(1, f2().next().value); function* f3(a = eval("x")) { var x; return a; } assertEquals(1, f3().next().value); function* f31(a = eval("'use strict'; x")) { var x; return a; } assertEquals(1, f31().next().value); function* f4(a = function() { return x }) { var x; return a(); } assertEquals(1, f4().next().value); function* f5(a = () => x) { var x; return a(); } assertEquals(1, f5().next().value); function* f6(a = () => eval("x")) { var x; return a(); } assertEquals(1, f6().next().value); function* f61(a = () => { 'use strict'; return eval("x") }) { var x; return a(); } assertEquals(1, f61().next().value); function* f62(a = () => eval("'use strict'; x")) { var x; return a(); } assertEquals(1, f62().next().value); var f11 = function* f(x = f) { var f; return x; } assertSame(f11, f11().next().value); var f12 = function* f(x = f) { function f() {}; return x; } assertSame(f12, f12().next().value); var f13 = function* f(f = 7, x = f) { return x; } assertSame(7, f13().next().value); var o1 = {f: function*(x = this) { return x; }}; assertSame(o1, o1.f().next().value); assertSame(1, o1.f(1).next().value); })(); (function TestParameterScopingStrict() { "use strict"; var x = 1; function* f1(a = x) { let x = 2; return a; } assertEquals(1, f1().next().value); function* f2(a = x) { const x = 2; return a; } assertEquals(1, f2().next().value); function* f3(a = x) { function x() {}; return a; } assertEquals(1, f3().next().value); function* f4(a = eval("x")) { var x; return a; } assertEquals(1, f4().next().value); function* f5(a = () => eval("x")) { var x; return a(); } assertEquals(1, f5().next().value); var f11 = function* f(x = f) { let f; return x; } assertSame(f11, f11().next().value); var f12 = function* f(x = f) { const f = 0; return x; } assertSame(f12, f12().next().value); var f13 = function* f(x = f) { function f() {}; return x; } assertSame(f13, f13().next().value); })(); (function TestSloppyEvalScoping() { var x = 1; function* f1(y = eval("var x = 2")) { with ({}) { return x; } } assertEquals(2, f1().next().value); function* f2(y = eval("var x = 2"), z = x) { return z; } assertEquals(2, f2().next().value); assertEquals(1, f2(0).next().value); function* f3(y = eval("var x = 2"), z = eval("x")) { return z; } assertEquals(2, f3().next().value); assertEquals(1, f3(0).next().value); function* f8(y = (eval("var x = 2"), x)) { return y; } assertEquals(2, f8().next().value); assertEquals(0, f8(0).next().value); function* f11(z = eval("var y = 2")) { return y; } assertEquals(2, f11().next().value); function* f12(z = eval("var y = 2"), b = y) { return b; } assertEquals(2, f12().next().value); function* f13(z = eval("var y = 2"), b = eval("y")) { return b; } assertEquals(2, f13().next().value); function* f21(f = () => x) { eval("var x = 2"); return f() } assertEquals(1, f21().next().value); assertEquals(3, f21(() => 3).next().value); function* f22(f = () => eval("x")) { eval("var x = 2"); return f() } assertEquals(1, f22().next().value); assertEquals(3, f22(() => 3).next().value); })(); (function TestStrictEvalScoping() { 'use strict'; var x = 1; function* f1(y = eval("var x = 2")) { return x; } assertEquals(1, f1().next().value); function* f2(y = eval("var x = 2"), z = x) { return z; } assertEquals(1, f2().next().value); assertEquals(1, f2(0).next().value); function* f3(y = eval("var x = 2"), z = eval("x")) { return z; } assertEquals(1, f3().next().value); assertEquals(1, f3(0).next().value); function* f8(y = (eval("var x = 2"), x)) { return y; } assertEquals(1, f8().next().value); assertEquals(0, f8(0).next().value); function* f11(z = eval("var y = 2")) { return y; } assertThrows(() => f11().next().value, ReferenceError); function* f12(z = eval("var y = 2"), b = y) {} assertThrows(() => f12().next().value, ReferenceError); function* f13(z = eval("var y = 2"), b = eval("y")) {} assertThrows(() => f13().next().value, ReferenceError); function* f21(f = () => x) { eval("var x = 2"); return f() } assertEquals(1, f21().next().value); assertEquals(3, f21(() => 3).next().value); function* f22(f = () => eval("x")) { eval("var x = 2"); return f() } assertEquals(1, f22().next().value); assertEquals(3, f22(() => 3).next().value); })(); (function TestParameterTDZSloppy() { function* f1(a = x, x) { return a } assertThrows(() => f1(undefined, 4), ReferenceError); assertEquals(4, f1(4, 5).next().value); function* f2(a = eval("x"), x) { return a } assertThrows(() => f2(undefined, 4), ReferenceError); assertEquals(4, f2(4, 5).next().value); function* f3(a = eval("'use strict'; x"), x) { return a } assertThrows(() => f3(undefined, 4), ReferenceError); assertEquals(4, f3(4, 5).next().value); function* f4(a = () => x, x) { return a() } assertEquals(4, f4(() => 4, 5).next().value); function* f5(a = () => eval("x"), x) { return a() } assertEquals(4, f5(() => 4, 5).next().value); function* f6(a = () => eval("'use strict'; x"), x) { return a() } assertEquals(4, f6(() => 4, 5).next().value); function* f11(a = x, x = 2) { return a } assertThrows(() => f11(), ReferenceError); assertThrows(() => f11(undefined), ReferenceError); assertThrows(() => f11(undefined, 4), ReferenceError); assertEquals(4, f1(4, 5).next().value); function* f12(a = eval("x"), x = 2) { return a } assertThrows(() => f12(), ReferenceError); assertThrows(() => f12(undefined), ReferenceError); assertThrows(() => f12(undefined, 4), ReferenceError); assertEquals(4, f12(4, 5).next().value); function* f13(a = eval("'use strict'; x"), x = 2) { return a } assertThrows(() => f13(), ReferenceError); assertThrows(() => f13(undefined), ReferenceError); assertThrows(() => f13(undefined, 4), ReferenceError); assertEquals(4, f13(4, 5).next().value); function* f21(x = function() { return a }, ...a) { return x()[0] } assertEquals(4, f21(undefined, 4).next().value); function* f22(x = () => a, ...a) { return x()[0] } assertEquals(4, f22(undefined, 4).next().value); function* f23(x = () => eval("a"), ...a) { return x()[0] } assertEquals(4, f23(undefined, 4).next().value); function* f24(x = () => {'use strict'; return eval("a") }, ...a) { return x()[0] } assertEquals(4, f24(undefined, 4).next().value); function* f25(x = () => eval("'use strict'; a"), ...a) { return x()[0] } assertEquals(4, f25(undefined, 4).next().value); })(); (function TestParameterTDZStrict() { "use strict"; function* f1(a = eval("x"), x) { return a } assertThrows(() => f1(undefined, 4), ReferenceError); assertEquals(4, f1(4, 5).next().value); function* f2(a = () => eval("x"), x) { return a() } assertEquals(4, f2(() => 4, 5).next().value); function* f11(a = eval("x"), x = 2) { return a } assertThrows(() => f11(), ReferenceError); assertThrows(() => f11(undefined), ReferenceError); assertThrows(() => f11(undefined, 4), ReferenceError); assertEquals(4, f11(4, 5).next().value); function* f21(x = () => eval("a"), ...a) { return x()[0] } assertEquals(4, f21(undefined, 4).next().value); })(); (function TestArgumentsForNonSimpleParameters() { function* f1(x = 900) { arguments[0] = 1; return x } assertEquals(9, f1(9).next().value); assertEquals(900, f1().next().value); function* f2(x = 1001) { x = 2; return arguments[0] } assertEquals(10, f2(10).next().value); assertEquals(undefined, f2().next().value); }()); (function TestFunctionLength() { assertEquals(0, (function*(x = 1) {}).length); assertEquals(0, (function*(x = 1, ...a) {}).length); assertEquals(1, (function*(x, y = 1) {}).length); assertEquals(1, (function*(x, y = 1, ...a) {}).length); assertEquals(2, (function*(x, y, z = 1) {}).length); assertEquals(2, (function*(x, y, z = 1, ...a) {}).length); assertEquals(1, (function*(x, y = 1, z) {}).length); assertEquals(1, (function*(x, y = 1, z, ...a) {}).length); assertEquals(1, (function*(x, y = 1, z, v = 2) {}).length); assertEquals(1, (function*(x, y = 1, z, v = 2, ...a) {}).length); })(); (function TestDirectiveThrows() { "use strict"; assertThrows("(function*(x=1){'use strict';})", SyntaxError); assertThrows("(function*(a, x=1){'use strict';})", SyntaxError); assertThrows("(function*({x}){'use strict';})", SyntaxError); })();