// Copyright 2008 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. // When calling user-defined functions on strings, booleans or // numbers, we should create a wrapper object. // When running the tests use loops to ensure that the call site moves through // the different IC states and that both the runtime system and the generated // IC code is tested. function RunTests() { for (var i = 0; i < 10; i++) { assertEquals('object', 'xxx'.TypeOfThis()); assertEquals('object', true.TypeOfThis(2,3)); assertEquals('object', false.TypeOfThis()); assertEquals('object', (42).TypeOfThis()); assertEquals('object', (3.14).TypeOfThis()); } for (var i = 0; i < 10; i++) { assertEquals('object', 'xxx'['TypeOfThis']()); assertEquals('object', true['TypeOfThis']()); assertEquals('object', false['TypeOfThis']()); assertEquals('object', (42)['TypeOfThis']()); assertEquals('object', (3.14)['TypeOfThis']()); } function CallTypeOfThis(obj) { assertEquals('object', obj.TypeOfThis()); } for (var i = 0; i < 10; i++) { CallTypeOfThis('xxx'); CallTypeOfThis(true); CallTypeOfThis(false); CallTypeOfThis(42); CallTypeOfThis(3.14); } function TestWithWith(obj) { with (obj) { for (var i = 0; i < 10; i++) { assertEquals('object', TypeOfThis()); } } } TestWithWith('xxx'); TestWithWith(true); TestWithWith(false); TestWithWith(42); TestWithWith(3.14); for (var i = 0; i < 10; i++) { assertEquals('object', true[7]()); assertEquals('object', false[7]()); assertEquals('object', (42)[7]()); assertEquals('object', (3.14)[7]()); } for (var i = 0; i < 10; i++) { assertEquals('object', typeof 'xxx'.ObjectValueOf()); assertEquals('object', typeof true.ObjectValueOf()); assertEquals('object', typeof false.ObjectValueOf()); assertEquals('object', typeof (42).ObjectValueOf()); assertEquals('object', typeof (3.14).ObjectValueOf()); } for (var i = 0; i < 10; i++) { assertEquals('[object String]', 'xxx'.ObjectToString()); assertEquals('[object Boolean]', true.ObjectToString()); assertEquals('[object Boolean]', false.ObjectToString()); assertEquals('[object Number]', (42).ObjectToString()); assertEquals('[object Number]', (3.14).ObjectToString()); } } function TypeOfThis() { return typeof this; } // Test with normal setup of prototype. String.prototype.TypeOfThis = TypeOfThis; Boolean.prototype.TypeOfThis = TypeOfThis; Number.prototype.TypeOfThis = TypeOfThis; Boolean.prototype[7] = TypeOfThis; Number.prototype[7] = TypeOfThis; String.prototype.ObjectValueOf = Object.prototype.valueOf; Boolean.prototype.ObjectValueOf = Object.prototype.valueOf; Number.prototype.ObjectValueOf = Object.prototype.valueOf; String.prototype.ObjectToString = Object.prototype.toString; Boolean.prototype.ObjectToString = Object.prototype.toString; Number.prototype.ObjectToString = Object.prototype.toString; RunTests(); // Run test after properties have been set to a different value. String.prototype.TypeOfThis = 'x'; Boolean.prototype.TypeOfThis = 'x'; Number.prototype.TypeOfThis = 'x'; Boolean.prototype[7] = 'x'; Number.prototype[7] = 'x'; String.prototype.TypeOfThis = TypeOfThis; Boolean.prototype.TypeOfThis = TypeOfThis; Number.prototype.TypeOfThis = TypeOfThis; Boolean.prototype[7] = TypeOfThis; Number.prototype[7] = TypeOfThis; RunTests(); // Force the prototype into slow case and run the test again. delete String.prototype.TypeOfThis; delete Boolean.prototype.TypeOfThis; delete Number.prototype.TypeOfThis; Boolean.prototype[7]; Number.prototype[7]; String.prototype.TypeOfThis = TypeOfThis; Boolean.prototype.TypeOfThis = TypeOfThis; Number.prototype.TypeOfThis = TypeOfThis; Boolean.prototype[7] = TypeOfThis; Number.prototype[7] = TypeOfThis; RunTests(); // According to ES3 15.3.4.3 the this value passed to Function.prototyle.apply // should wrapped. According to ES5 it should not. assertEquals('object', TypeOfThis.apply('xxx', [])); assertEquals('object', TypeOfThis.apply(true, [])); assertEquals('object', TypeOfThis.apply(false, [])); assertEquals('object', TypeOfThis.apply(42, [])); assertEquals('object', TypeOfThis.apply(3.14, [])); // According to ES3 15.3.4.3 the this value passed to Function.prototyle.call // should wrapped. According to ES5 it should not. assertEquals('object', TypeOfThis.call('xxx')); assertEquals('object', TypeOfThis.call(true)); assertEquals('object', TypeOfThis.call(false)); assertEquals('object', TypeOfThis.call(42)); assertEquals('object', TypeOfThis.call(3.14));