// 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);
})();