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