[ES7] yield star should not return if the inner iterator.throw returns { done: true }
[WebKit-https.git] / Source / JavaScriptCore / builtins / GeneratorPrototype.js
1 /*
2  * Copyright (C) 2015-2016 Yusuke Suzuki <utatane.tea@gmail.com>.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 // 25.3.3.3 GeneratorResume ( generator, value )
27 // 25.3.3.4 GeneratorResumeAbrupt(generator, abruptCompletion)
28 function generatorResume(generator, sentValue, resumeMode)
29 {
30     "use strict";
31
32     let state = generator.@generatorState;
33     let done = false;
34     let value = @undefined;
35
36     if (typeof state !== 'number')
37         throw new @TypeError("|this| should be a generator");
38
39     if (state === @GeneratorStateExecuting)
40         throw new @TypeError("Generator is executing");
41
42     if (state === @GeneratorStateCompleted) {
43         if (resumeMode === @GeneratorResumeModeThrow)
44             throw sentValue;
45
46         done = true;
47         if (resumeMode === @GeneratorResumeModeReturn)
48             value = sentValue;
49     } else {
50         try {
51             generator.@generatorState = @GeneratorStateExecuting;
52             value = generator.@generatorNext.@call(generator.@generatorThis, generator, state, sentValue, resumeMode);
53             if (generator.@generatorState === @GeneratorStateExecuting) {
54                 generator.@generatorState = @GeneratorStateCompleted;
55                 done = true;
56             }
57         } catch (error) {
58             generator.@generatorState = @GeneratorStateCompleted;
59             throw error;
60         }
61     }
62     return { done, value };
63 }
64
65 function next(value)
66 {
67     "use strict";
68
69     return @generatorResume(this, value, @GeneratorResumeModeNormal);
70 }
71
72 function return(value)
73 {
74     "use strict";
75
76     return @generatorResume(this, value, @GeneratorResumeModeReturn);
77 }
78
79 function throw(exception)
80 {
81     "use strict";
82
83     return @generatorResume(this, exception, @GeneratorResumeModeThrow);
84 }