| // 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. |
| |
| // Flags: --allow-natives-syntax |
| |
| |
| function MaybeOptimizeOrDeoptimize(f) { |
| %PrepareFunctionForOptimization(f); |
| let x = Math.random(); // --random-seed makes this deterministic |
| if (x <= 0.33) { |
| %OptimizeFunctionOnNextCall(f); |
| } else if (x <= 0.66) { |
| %DeoptimizeFunction(f); |
| } |
| } |
| |
| function Next(generator, ...args) { |
| MaybeOptimizeOrDeoptimize(%GeneratorGetFunction(generator)); |
| return generator.next(...args); |
| } |
| |
| function Return(generator, ...args) { |
| MaybeOptimizeOrDeoptimize(%GeneratorGetFunction(generator)); |
| return generator.return(...args); |
| } |
| |
| function Throw(generator, ...args) { |
| MaybeOptimizeOrDeoptimize(%GeneratorGetFunction(generator)); |
| return generator.throw(...args); |
| } |
| |
| |
| { // yield in try-catch |
| |
| let g = function*() { |
| try {yield 1} catch (error) {assertEquals("caught", error)} |
| }; |
| |
| assertThrowsEquals(() => Throw(g(), "not caught"), "not caught"); |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Throw(x, "caught")); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| assertThrowsEquals(() => Throw(x, "not caught"), "not caught"); |
| } |
| } |
| |
| |
| { // return that doesn't close |
| let g = function*() { try {return 42} finally {yield 43} }; |
| |
| { |
| let x = g(); |
| assertEquals({value: 43, done: false}, Next(x)); |
| assertEquals({value: 42, done: true}, Next(x)); |
| } |
| } |
| |
| |
| { // return that doesn't close |
| let x; |
| let g = function*() { try {return 42} finally {Throw(x, 666)} }; |
| |
| { |
| x = g(); |
| assertThrows(() => Next(x), TypeError); // still executing |
| } |
| } |
| |
| |
| { // yield in try-finally, finally clause performs return |
| |
| let g = function*() { try {yield 42} finally {return 13} }; |
| |
| { // "return" closes at suspendedStart |
| let x = g(); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 42)); |
| assertThrowsEquals(() => Throw(x, 43), 43); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| |
| { // "throw" closes at suspendedStart |
| let x = g(); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: undefined, done: true}, Next(x, 42)); |
| assertEquals({value: 43, done: true}, Return(x, 43)); |
| assertThrowsEquals(() => Throw(x, 44), 44); |
| } |
| |
| { // "next" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 13, done: true}, Next(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| } |
| |
| { // "return" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 13, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| } |
| |
| { // "throw" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 13, done: true}, Throw(x, 666)); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| } |
| } |
| |
| |
| { // yield in try-finally, finally clause doesn't perform return |
| |
| let g = function*() { try {yield 42} finally {13} }; |
| |
| { // "return" closes at suspendedStart |
| let x = g(); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 42)); |
| assertThrowsEquals(() => Throw(x, 43), 43); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| |
| { // "throw" closes at suspendedStart |
| let x = g(); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: undefined, done: true}, Next(x, 42)); |
| assertEquals({value: 43, done: true}, Return(x, 43)); |
| assertThrowsEquals(() => Throw(x, 44), 44); |
| } |
| |
| { // "next" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| |
| { // "return" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertThrowsEquals(() => Throw(x, 44), 44); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| |
| { // "throw" closes at suspendedYield |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: undefined, done: true}, Next(x, 666)); |
| assertThrowsEquals(() => Throw(x, 666), 666); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| } |
| |
| |
| { // yield in try-finally, finally clause yields and performs return |
| |
| let g = function*() { try {yield 42} finally {yield 43; return 13} }; |
| |
| { |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Return(x, 666)); |
| assertEquals({value: 13, done: true}, Next(x)); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| } |
| } |
| |
| |
| { // yield in try-finally, finally clause yields and doesn't perform return |
| |
| let g = function*() { try {yield 42} finally {yield 43; 13} }; |
| |
| { |
| let x = g(); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Return(x, 666)); |
| assertEquals({value: 666, done: true}, Next(x)); |
| assertEquals({value: 5, done: true}, Return(x, 5)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| assertEquals({value: 666, done: true}, Return(x, 666)); |
| } |
| } |
| |
| |
| { // yield*, finally clause performs return |
| |
| let h = function*() { try {yield 42} finally {yield 43; return 13} }; |
| let g = function*() { yield 1; yield yield* h(); }; |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Next(x, 666)); |
| assertEquals({value: 13, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Return(x, 666)); |
| assertEquals({value: 13, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Throw(x, 666)); |
| assertEquals({value: 13, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| } |
| } |
| |
| |
| { // yield*, finally clause does not perform return |
| |
| let h = function*() { try {yield 42} finally {yield 43; 13} }; |
| let g = function*() { yield 1; yield yield* h(); }; |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Next(x, 666)); |
| assertEquals({value: undefined, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Return(x, 44)); |
| assertEquals({value: 44, done: false}, Next(x)); |
| assertEquals({value: undefined, done: true}, Next(x)); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 42, done: false}, Next(x)); |
| assertEquals({value: 43, done: false}, Throw(x, 666)); |
| assertThrowsEquals(() => Next(x), 666); |
| } |
| } |
| |
| |
| { // yield*, .return argument is final result |
| |
| function* inner() { |
| yield 2; |
| } |
| |
| function* g() { |
| yield 1; |
| return yield* inner(); |
| } |
| |
| { |
| let x = g(); |
| assertEquals({value: 1, done: false}, Next(x)); |
| assertEquals({value: 2, done: false}, Next(x)); |
| assertEquals({value: 42, done: true}, Return(x, 42)); |
| } |
| } |
| |
| |
| // More or less random tests from here on. |
| |
| |
| { |
| function* foo() { } |
| let g = foo(); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { return new.target } |
| let g = foo(); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { throw 666; return 42} |
| let g = foo(); |
| assertThrowsEquals(() => Next(g), 666); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo(a) { return a; } |
| let g = foo(42); |
| assertEquals({value: 42, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo(a) { a.iwashere = true; return a; } |
| let x = {}; |
| let g = foo(x); |
| assertEquals({value: {iwashere: true}, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let a = 42; |
| function* foo() { return a; } |
| let g = foo(); |
| assertEquals({value: 42, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let a = 40; |
| function* foo(b) { return a + b; } |
| let g = foo(2); |
| assertEquals({value: 42, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let a = 40; |
| function* foo(b) { a--; b++; return a + b; } |
| let g = foo(2); |
| assertEquals({value: 42, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let g; |
| function* foo() { Next(g) } |
| g = foo(); |
| assertThrows(() => Next(g), TypeError); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; yield 3; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| |
| { |
| function* foo() { yield 2; if (true) { yield 3 }; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; if (true) { yield 3; yield 4 } } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; if (false) { yield 3 }; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; while (true) { yield 3 }; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; (yield 3) + 42; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| } |
| |
| { |
| function* foo() { yield 2; return (yield 3) + 42; yield 4 } |
| g = foo(); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 42, done: true}, Next(g, 0)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let x = 42; |
| function* foo() { |
| yield x; |
| for (let x in {a: 1, b: 2}) { |
| let i = 2; |
| yield x; |
| yield i; |
| do { |
| yield i; |
| } while (i-- > 0); |
| } |
| yield x; |
| return 5; |
| } |
| g = foo(); |
| assertEquals({value: 42, done: false}, Next(g)); |
| assertEquals({value: 'a', done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 1, done: false}, Next(g)); |
| assertEquals({value: 0, done: false}, Next(g)); |
| assertEquals({value: 'b', done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 1, done: false}, Next(g)); |
| assertEquals({value: 0, done: false}, Next(g)); |
| assertEquals({value: 42, done: false}, Next(g)); |
| assertEquals({value: 5, done: true}, Next(g)); |
| } |
| |
| { |
| let a = 3; |
| function* foo() { |
| let b = 4; |
| yield 1; |
| { let c = 5; yield 2; yield a; yield b; yield c; } |
| } |
| g = foo(); |
| assertEquals({value: 1, done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 3, done: false}, Next(g)); |
| assertEquals({value: 4, done: false}, Next(g)); |
| assertEquals({value: 5, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| yield 42; |
| } |
| g = foo(); |
| for (let i = 0; i < 100; ++i) { |
| assertEquals({value: 42, done: false}, i%25 === 0 ? Next(g) : g.next()); |
| } |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| function* foo() { |
| for (let i = 0; i < 3; ++i) { |
| let j = 0 |
| yield i; |
| do { |
| yield (i + 10); |
| } while (++j < 2); |
| } |
| } |
| g = foo(); |
| assertEquals({value: 0, done: false}, Next(g)); |
| assertEquals({value: 10, done: false}, Next(g)); |
| assertEquals({value: 10, done: false}, Next(g)); |
| assertEquals({value: 1, done: false}, Next(g)); |
| assertEquals({value: 11, done: false}, Next(g)); |
| assertEquals({value: 11, done: false}, Next(g)); |
| assertEquals({value: 2, done: false}, Next(g)); |
| assertEquals({value: 12, done: false}, Next(g)); |
| assertEquals({value: 12, done: false}, Next(g)); |
| assertEquals({value: undefined, done: true}, Next(g)); |
| } |
| |
| { |
| let foo = function*() { |
| while (true) { |
| if (true || false) yield 42; |
| continue; |
| } |
| } |
| g = foo(); |
| assertEquals({value: 42, done: false}, Next(g)); |
| assertEquals({value: 42, done: false}, Next(g)); |
| assertEquals({value: 42, done: false}, Next(g)); |
| } |
| |
| { |
| let foo = function*() { |
| yield* (function*() { yield 42; }()); |
| assertUnreachable(); |
| } |
| g = foo(); |
| assertEquals({value: 42, done: false}, Next(g)); |
| assertEquals({value: 23, done: true}, Return(g, 23)); |
| } |
| |
| { |
| let iterable = { |
| [Symbol.iterator]() { |
| return { next() { return {} } }; |
| } |
| }; |
| let foo = function*() { yield* iterable }; |
| g = foo(); |
| g.next(); |
| assertThrows(() => Throw(g), TypeError); |
| } |