// Copyright 2012 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. // Flags: --allow-natives-syntax // Handy abbreviations. var dp = Object.defineProperty; var gop = Object.getOwnPropertyDescriptor; function getter() { return 111; } function setter(x) { print(222); } function anotherGetter() { return 333; } function anotherSetter(x) { print(444); } var obj1, obj2, obj3, obj4; // Two objects with the same getter. obj1 = {}; dp(obj1, "alpha", { get: getter }); obj2 = {}; dp(obj2, "alpha", { get: getter }); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same getter, oldskool. obj1 = {}; obj1.__defineGetter__("bravo", getter); assertEquals(getter, obj1.__lookupGetter__("bravo")); obj2 = {}; obj2.__defineGetter__("bravo", getter); assertEquals(getter, obj2.__lookupGetter__("bravo")); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same setter. obj1 = {}; dp(obj1, "charlie", { set: setter }); obj2 = {}; dp(obj2, "charlie", { set: setter }); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same setter, oldskool. obj1 = {}; obj1.__defineSetter__("delta", setter); assertEquals(setter, obj1.__lookupSetter__("delta")); obj2 = {}; obj2.__defineSetter__("delta", setter); assertEquals(setter, obj2.__lookupSetter__("delta")); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same getter and setter. obj1 = {}; dp(obj1, "foxtrot", { get: getter, set: setter }); obj2 = {}; dp(obj2, "foxtrot", { get: getter, set: setter }); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same getter and setter, set separately. obj1 = {}; dp(obj1, "golf", { get: getter, configurable: true }); dp(obj1, "golf", { set: setter, configurable: true }); obj2 = {}; dp(obj2, "golf", { get: getter, configurable: true }); dp(obj2, "golf", { set: setter, configurable: true }); assertTrue(%HaveSameMap(obj1, obj2)); // Two objects with the same getter and setter, set separately, oldskool. obj1 = {}; obj1.__defineGetter__("hotel", getter); obj1.__defineSetter__("hotel", setter); obj2 = {}; obj2.__defineGetter__("hotel", getter); obj2.__defineSetter__("hotel", setter); assertTrue(%HaveSameMap(obj1, obj2)); // Attribute-only change, shouldn't affect previous descriptor properties. obj1 = {}; dp(obj1, "india", { get: getter, configurable: true, enumerable: true }); assertEquals(getter, gop(obj1, "india").get); assertTrue(gop(obj1, "india").configurable); assertTrue(gop(obj1, "india").enumerable); dp(obj1, "india", { enumerable: false }); assertEquals(getter, gop(obj1, "india").get); assertTrue(gop(obj1, "india").configurable); assertFalse(gop(obj1, "india").enumerable); // Attribute-only change, shouldn't affect objects with previously shared maps. obj1 = {}; dp(obj1, "juliet", { set: setter, configurable: true, enumerable: false }); assertEquals(setter, gop(obj1, "juliet").set); assertTrue(gop(obj1, "juliet").configurable); assertFalse(gop(obj1, "juliet").enumerable); obj2 = {}; dp(obj2, "juliet", { set: setter, configurable: true, enumerable: false }); assertEquals(setter, gop(obj2, "juliet").set); assertTrue(gop(obj2, "juliet").configurable); assertFalse(gop(obj2, "juliet").enumerable); dp(obj1, "juliet", { set: setter, configurable: false, enumerable: true }); assertEquals(setter, gop(obj1, "juliet").set); assertFalse(gop(obj1, "juliet").configurable); assertTrue(gop(obj1, "juliet").enumerable); assertEquals(setter, gop(obj2, "juliet").set); assertTrue(gop(obj2, "juliet").configurable); assertFalse(gop(obj2, "juliet").enumerable); // Two objects with the different getters. obj1 = {}; dp(obj1, "kilo", { get: getter }); obj2 = {}; dp(obj2, "kilo", { get: anotherGetter }); assertEquals(getter, gop(obj1, "kilo").get); assertEquals(anotherGetter, gop(obj2, "kilo").get); assertFalse(%HaveSameMap(obj1, obj2)); // Two objects with the same getters and different setters. obj1 = {}; dp(obj1, "lima", { get: getter, set: setter }); obj2 = {}; dp(obj2, "lima", { get: getter, set: anotherSetter }); assertEquals(setter, gop(obj1, "lima").set); assertEquals(anotherSetter, gop(obj2, "lima").set); assertFalse(%HaveSameMap(obj1, obj2)); // Even 'undefined' is a kind of getter. obj1 = {}; dp(obj1, "mike", { get: undefined }); assertTrue("mike" in obj1); assertEquals(undefined, gop(obj1, "mike").get); assertEquals(undefined, obj1.__lookupGetter__("mike")); assertEquals(undefined, gop(obj1, "mike").set); assertEquals(undefined, obj1.__lookupSetter__("mike")); // Even 'undefined' is a kind of setter. obj1 = {}; dp(obj1, "november", { set: undefined }); assertTrue("november" in obj1); assertEquals(undefined, gop(obj1, "november").get); assertEquals(undefined, obj1.__lookupGetter__("november")); assertEquals(undefined, gop(obj1, "november").set); assertEquals(undefined, obj1.__lookupSetter__("november")); // Redefining a data property. obj1 = {}; obj1.oscar = 12345; dp(obj1, "oscar", { set: setter }); assertEquals(setter, gop(obj1, "oscar").set); // Re-adding the same getter/attributes pair. obj1 = {}; dp(obj1, "papa", { get: getter, configurable: true }); dp(obj1, "papa", { get: getter, set: setter, configurable: true }); assertEquals(getter, gop(obj1, "papa").get); assertEquals(setter, gop(obj1, "papa").set); assertTrue(gop(obj1, "papa").configurable); assertFalse(gop(obj1, "papa").enumerable); // Two objects with the same getter on the prototype chain. obj1 = {}; dp(obj1, "quebec", { get: getter }); obj2 = Object.create(obj1); obj3 = Object.create(obj2); obj4 = Object.create(obj2); assertTrue(%HaveSameMap(obj3, obj4)); // Two objects with the same setter on the prototype chain. obj1 = {}; dp(obj1, "romeo", { set: setter }); obj2 = Object.create(obj1); obj3 = Object.create(obj2); obj4 = Object.create(obj2); assertTrue(%HaveSameMap(obj3, obj4));