// Copyright 2011 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following // disclaimer in the documentation and/or other materials provided // with the distribution. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. assertEquals(void 0, eval()); assertEquals(4, eval(4)); function f() { return 'The f function'; }; assertTrue(f === eval(f)); function g(x, y) { return 4; }; count = 0; assertEquals(4, eval('2 + 2', count++)); assertEquals(1, count); try { eval('hest 7 &*^*&^'); assertUnreachable('Did not throw on syntax error.'); } catch (e) { assertEquals('SyntaxError', e.name); } // eval has special evaluation order for consistency with other browsers. global_eval = eval; assertEquals(void 0, eval(eval("var eval = function f(x) { return 'hest';}"))) eval = global_eval; // Test eval with different number of parameters. global_eval = eval; eval = function(x, y) { return x + y; }; assertEquals(4, eval(2, 2)); eval = global_eval; // Test that un-aliased eval reads from local context. foo = 0; result = (function() { var foo = 2; return eval('foo'); })(); assertEquals(2, result); // Test that un-aliased eval writes to local context. foo = 0; result = (function() { var foo = 1; eval('foo = 2'); return foo; })(); assertEquals(2, result); assertEquals(0, foo); // Test that un-aliased eval has right receiver. function MyObject() { this.self = eval('this'); } var o = new MyObject(); assertTrue(o === o.self); // Test that aliased eval reads from global context. var e = eval; foo = 0; result = (function() { var foo = 2; return e('foo'); })(); assertEquals(0, result); // Test that aliased eval writes to global context. var e = eval; foo = 0; (function() { e('var foo = 2;'); })(); assertEquals(2, foo); // Test that aliased eval has right receiver. function MyOtherObject() { this.self = e('this'); } var o = new MyOtherObject(); assertTrue(this === o.self); // Try to cheat the 'aliased eval' detection. var x = this; foo = 0; result = (function() { var foo = 2; // Should be non-direct call. return x.eval('foo'); })(); assertEquals(0, result); foo = 0; result = (function() { var foo = 2; // Should be non-direct call. return (1,eval)('foo'); })(); assertEquals(0, result); foo = 0; result = (function() { var eval = function(x) { return x; }; var foo = eval(2); // Should be non-direct call. return e('foo'); })(); assertEquals(0, result); foo = 0; result = (function() { var foo = 2; // Should be direct call. with ({ eval : e }) { return eval('foo'); } })(); assertEquals(2, result); result = (function() { var eval = function(x) { return 2 * x; }; return (function() { return eval(2); })(); })(); assertEquals(4, result); result = (function() { eval("var eval = function(s) { return this; }"); return eval("42"); // Should return the global object })(); assertEquals(this, result); (function() { var obj = { f: function(eval) { return eval("this"); } }; result = obj.f(eval); assertEquals(obj, result); })(); (function() { var obj = { f: function(eval) { arguments; return eval("this"); } }; result = obj.f(eval); assertEquals(obj, result); })(); eval = function(x) { return 2 * x; }; result = (function() { return (function() { return eval(2); })(); })(); assertEquals(4, result); // Regression test: calling a function named eval found in a context that is // not the global context should get the global object as receiver. result = (function () { var eval = function (x) { return this; }; with ({}) { return eval('ignore'); } })(); assertEquals(this, result);