|  | // 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: --no-always-opt | 
|  |  | 
|  | // Test that the (strict) eval scope is visible to the debugger. | 
|  |  | 
|  | var Debug = debug.Debug; | 
|  | var exception = null; | 
|  | var delegate = null; | 
|  |  | 
|  | function listener(event, exec_state, event_data, data) { | 
|  | if (event != Debug.DebugEvent.Break) return; | 
|  | try { | 
|  | delegate(exec_state); | 
|  | } catch (e) { | 
|  | exception = e; | 
|  | } | 
|  | } | 
|  |  | 
|  | Debug.setListener(listener); | 
|  |  | 
|  | // Current function is the top-level eval. | 
|  | // We can access stack- and context-allocated values in the eval-scope. | 
|  | delegate = function(exec_state) { | 
|  | assertEquals([ debug.ScopeType.Eval, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(0).allScopes().map(s => s.scopeType())); | 
|  | var scope = exec_state.frame(0).scope(0); | 
|  | assertEquals(1, scope.scopeObject().property("a").value().value()); | 
|  | assertEquals(1, exec_state.frame(0).evaluate("a").value()); | 
|  | scope.setVariableValue("a", 2); | 
|  | assertEquals(2, exec_state.frame(0).evaluate("a++").value()); | 
|  | } | 
|  |  | 
|  | eval("'use strict';      \n" + | 
|  | "var a = 1;         \n" + | 
|  | "debugger;          \n" + | 
|  | "assertEquals(3, a);\n"); | 
|  |  | 
|  | eval("'use strict';      \n" + | 
|  | "var a = 1;         \n" + | 
|  | "(x=>a);            \n" +  // Force context-allocation. | 
|  | "debugger;          \n" + | 
|  | "assertEquals(3, a);\n"); | 
|  |  | 
|  | // Current function is an inner function. | 
|  | // We cannot access stack-allocated values in the eval-scope. | 
|  | delegate = function(exec_state) { | 
|  | assertEquals([ debug.ScopeType.Local, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(0).allScopes().map(s => s.scopeType())); | 
|  | assertEquals([ debug.ScopeType.Eval, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(1).allScopes().map(s => s.scopeType())); | 
|  | var scope = exec_state.frame(0).scope(0); | 
|  | assertThrows(() => exec_state.frame(0).evaluate("a"), ReferenceError); | 
|  | assertTrue(scope.scopeObject().property("a").isUndefined()); | 
|  | } | 
|  |  | 
|  | eval("'use strict';       \n" + | 
|  | "var a = 1;          \n" + | 
|  | "(() => {debugger})()\n"); | 
|  |  | 
|  | // Current function is an escaped inner function. | 
|  | delegate = function(exec_state) { | 
|  | assertEquals([ debug.ScopeType.Local, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(0).allScopes().map(s => s.scopeType())); | 
|  | assertEquals([ debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(1).allScopes().map(s => s.scopeType())); | 
|  | var scope = exec_state.frame(0).scope(0); | 
|  | assertThrows(() => exec_state.frame(0).evaluate("a"), ReferenceError); | 
|  | assertTrue(scope.scopeObject().property("a").isUndefined()); | 
|  | } | 
|  |  | 
|  | var f = eval("'use strict';   \n" + | 
|  | "var a = 1;      \n" + | 
|  | "() => {debugger}\n"); | 
|  | f(); | 
|  |  | 
|  | // Current function is an inner function. | 
|  | // We can access context-allocated values in the eval-scope. | 
|  | delegate = function(exec_state) { | 
|  | assertEquals([ debug.ScopeType.Local, | 
|  | debug.ScopeType.Closure, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(0).allScopes().map(s => s.scopeType())); | 
|  | assertEquals([ debug.ScopeType.Eval, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(1).allScopes().map(s => s.scopeType())); | 
|  | var scope = exec_state.frame(1).scope(0); | 
|  | assertEquals(1, scope.scopeObject().property("a").value().value()); | 
|  | assertEquals(1, exec_state.frame(1).evaluate("a").value()); | 
|  | assertEquals(1, exec_state.frame(0).evaluate("a").value()); | 
|  | scope.setVariableValue("a", 2); | 
|  | assertEquals(2, exec_state.frame(0).evaluate("a++").value()); | 
|  | assertEquals(3, exec_state.frame(1).evaluate("a++").value()); | 
|  | } | 
|  |  | 
|  | eval("'use strict';               \n" + | 
|  | "var a = 1;                  \n" + | 
|  | "(() => { a;                 \n" +  // Force context-allocation. | 
|  | "         debugger;          \n" + | 
|  | "         assertEquals(4, a);\n" + | 
|  | "       })();                \n" | 
|  | ); | 
|  |  | 
|  | // Current function is an escaped inner function. | 
|  | // We can access context-allocated values in the eval-scope. | 
|  | delegate = function(exec_state) { | 
|  | assertEquals([ debug.ScopeType.Local, | 
|  | debug.ScopeType.Closure, | 
|  | debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(0).allScopes().map(s => s.scopeType())); | 
|  | assertEquals([ debug.ScopeType.Script, | 
|  | debug.ScopeType.Global ], | 
|  | exec_state.frame(1).allScopes().map(s => s.scopeType())); | 
|  | var scope = exec_state.frame(0).scope(1); | 
|  | assertEquals(1, scope.scopeObject().property("a").value().value()); | 
|  | assertEquals(1, exec_state.frame(0).evaluate("a").value()); | 
|  | scope.setVariableValue("a", 2); | 
|  | assertEquals(2, exec_state.frame(0).evaluate("a++").value()); | 
|  | } | 
|  |  | 
|  | var g = eval("'use strict';              \n" + | 
|  | "var a = 1;                 \n" + | 
|  | "() => { a;                 \n" + | 
|  | "        debugger;          \n" + | 
|  | "        assertEquals(3, a);\n" + | 
|  | "      }                    \n"); | 
|  | g(); | 
|  |  | 
|  | Debug.setListener(null); | 
|  | assertNull(exception); |