mirror of
				https://github.com/tc39/test262.git
				synced 2025-11-04 05:33:50 +01:00 
			
		
		
		
	sourceRevisionAtLastExport: 33f2fb0e53d135f0ee17cfccd9d993eb2a6f47de targetRevisionAtLastExport: 31340cbd9add103f586d501b0c3354b7b182abc0
		
			
				
	
	
		
			234 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			234 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
// 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.
 | 
						|
 | 
						|
// Flags: --allow-natives-syntax
 | 
						|
 | 
						|
// Different ways to create an object.
 | 
						|
 | 
						|
function CreateFromLiteral() {
 | 
						|
  return {};
 | 
						|
}
 | 
						|
 | 
						|
function CreateFromObject() {
 | 
						|
  return new Object;
 | 
						|
}
 | 
						|
 | 
						|
function CreateDefault() {
 | 
						|
  return Object.create(Object.prototype);
 | 
						|
}
 | 
						|
 | 
						|
function CreateFromConstructor(proto) {
 | 
						|
  function C() {}
 | 
						|
  (new C).b = 9;  // Make sure that we can have an in-object property.
 | 
						|
  C.prototype = proto;
 | 
						|
  return function() { return new C; }
 | 
						|
}
 | 
						|
 | 
						|
function CreateFromApi(proto) {
 | 
						|
  return function() { return Object.create(proto); }
 | 
						|
}
 | 
						|
 | 
						|
function CreateWithProperty(proto) {
 | 
						|
  function C() { this.a = -100; }
 | 
						|
  C.prototype = proto;
 | 
						|
  return function() { return new C; }
 | 
						|
}
 | 
						|
 | 
						|
var bases = [CreateFromLiteral, CreateFromObject, CreateDefault];
 | 
						|
var inherits = [CreateFromConstructor, CreateFromApi, CreateWithProperty];
 | 
						|
var constructs = [CreateFromConstructor, CreateFromApi];
 | 
						|
 | 
						|
function TestAllCreates(f) {
 | 
						|
  // The depth of the prototype chain up the.
 | 
						|
  for (var depth = 0; depth < 3; ++depth) {
 | 
						|
    // Introduce readonly-ness this far up the chain.
 | 
						|
    for (var up = 0; up <= depth; ++up) {
 | 
						|
      // Try different construction methods.
 | 
						|
      for (var k = 0; k < constructs.length; ++k) {
 | 
						|
        // Construct a fresh prototype chain from above functions.
 | 
						|
        for (var i = 0; i < bases.length; ++i) {
 | 
						|
          var p = bases[i]();
 | 
						|
          // There may be a preexisting property under the insertion point...
 | 
						|
          for (var j = 0; j < depth - up; ++j) {
 | 
						|
            p = inherits[Math.floor(inherits.length * Math.random())](p)();
 | 
						|
          }
 | 
						|
          // ...but not above it.
 | 
						|
          for (var j = 0; j < up; ++j) {
 | 
						|
            p = constructs[Math.floor(constructs.length * Math.random())](p)();
 | 
						|
          }
 | 
						|
          // Create a fresh constructor.
 | 
						|
          var c = constructs[k](p);
 | 
						|
          f(function() {
 | 
						|
            var o = c();
 | 
						|
            o.up = o;
 | 
						|
            for (var j = 0; j < up; ++j) o.up = Object.getPrototypeOf(o.up);
 | 
						|
            return o;
 | 
						|
          })
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
// Different ways to make a property read-only.
 | 
						|
 | 
						|
function ReadonlyByNonwritableDataProperty(o, name) {
 | 
						|
  Object.defineProperty(o, name, {value: -41, writable: false});
 | 
						|
}
 | 
						|
 | 
						|
function ReadonlyByAccessorPropertyWithoutSetter(o, name) {
 | 
						|
  Object.defineProperty(o, name, {get: function() { return -42; }});
 | 
						|
}
 | 
						|
 | 
						|
function ReadonlyByGetter(o, name) {
 | 
						|
  o.__defineGetter__("a", function() { return -43; });
 | 
						|
}
 | 
						|
 | 
						|
function ReadonlyByFreeze(o, name) {
 | 
						|
  o[name] = -44;
 | 
						|
  Object.freeze(o);
 | 
						|
}
 | 
						|
 | 
						|
function ReadonlyByProto(o, name) {
 | 
						|
  var p = Object.create(o.__proto__);
 | 
						|
  Object.defineProperty(p, name, {value: -45, writable: false});
 | 
						|
  o.__proto__ = p;
 | 
						|
}
 | 
						|
 | 
						|
// TODO(neis,cbruni): Enable once the necessary traps work again.
 | 
						|
// Allow Proxy to be undefined, so test can run in non-Harmony mode as well.
 | 
						|
var global = this;
 | 
						|
 | 
						|
function ReadonlyByProxy(o, name) {
 | 
						|
  if (!global.Proxy) return ReadonlyByFreeze(o, name);  // Dummy.
 | 
						|
  var p = new global.Proxy({}, {
 | 
						|
    getPropertyDescriptor: function() {
 | 
						|
      return {value: -46, writable: false, configurable: true};
 | 
						|
    }
 | 
						|
  });
 | 
						|
  o.__proto__ = p;
 | 
						|
}
 | 
						|
 | 
						|
var readonlys = [
 | 
						|
  ReadonlyByNonwritableDataProperty, ReadonlyByAccessorPropertyWithoutSetter,
 | 
						|
  ReadonlyByGetter, ReadonlyByFreeze, ReadonlyByProto // ReadonlyByProxy
 | 
						|
]
 | 
						|
 | 
						|
function TestAllReadonlys(f) {
 | 
						|
  // Provide various methods to making a property read-only.
 | 
						|
  for (var i = 0; i < readonlys.length; ++i) {
 | 
						|
    print("  readonly =", i)
 | 
						|
    f(readonlys[i]);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
// Different use scenarios.
 | 
						|
 | 
						|
function Assign(o, x) {
 | 
						|
  o.a = x;
 | 
						|
}
 | 
						|
 | 
						|
function AssignStrict(o, x) {
 | 
						|
  "use strict";
 | 
						|
  o.a = x;
 | 
						|
}
 | 
						|
 | 
						|
function TestAllModes(f) {
 | 
						|
  for (var strict = 0; strict < 2; ++strict) {
 | 
						|
    print(" strict =", strict);
 | 
						|
    f(strict);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function TestAllScenarios(f) {
 | 
						|
  for (var t = 0; t < 100; t = 2*t + 1) {
 | 
						|
    print("t =", t)
 | 
						|
    f(function(strict, create, readonly) {
 | 
						|
      // Make sure that the assignments are monomorphic.
 | 
						|
      %DeoptimizeFunction(Assign);
 | 
						|
      %DeoptimizeFunction(AssignStrict);
 | 
						|
      %ClearFunctionFeedback(Assign);
 | 
						|
      %ClearFunctionFeedback(AssignStrict);
 | 
						|
      for (var i = 0; i < t; ++i) {
 | 
						|
        var o = create();
 | 
						|
        assertFalse("a" in o && !("a" in o.__proto__));
 | 
						|
        if (strict === 0)
 | 
						|
          Assign(o, i);
 | 
						|
        else
 | 
						|
          AssignStrict(o, i);
 | 
						|
        assertEquals(i, o.a);
 | 
						|
      }
 | 
						|
      %OptimizeFunctionOnNextCall(Assign);
 | 
						|
      %OptimizeFunctionOnNextCall(AssignStrict);
 | 
						|
      var o = create();
 | 
						|
      assertFalse("a" in o && !("a" in o.__proto__));
 | 
						|
      readonly(o.up, "a");
 | 
						|
      assertTrue("a" in o);
 | 
						|
      if (strict === 0)
 | 
						|
        Assign(o, t + 1);
 | 
						|
      else
 | 
						|
        assertThrows(function() { AssignStrict(o, t + 1) }, TypeError);
 | 
						|
      assertTrue(o.a < 0);
 | 
						|
    });
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
// Runner.
 | 
						|
 | 
						|
TestAllScenarios(function(scenario) {
 | 
						|
  TestAllModes(function(strict) {
 | 
						|
    TestAllReadonlys(function(readonly) {
 | 
						|
      TestAllCreates(function(create) {
 | 
						|
        scenario(strict, create, readonly);
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
});
 | 
						|
 | 
						|
 | 
						|
// Extra test forcing bailout.
 | 
						|
 | 
						|
function Assign2(o, x) { o.a = x }
 | 
						|
 | 
						|
(function() {
 | 
						|
  var p = CreateFromConstructor(Object.prototype)();
 | 
						|
  var c = CreateFromConstructor(p);
 | 
						|
  for (var i = 0; i < 3; ++i) {
 | 
						|
    var o = c();
 | 
						|
    Assign2(o, i);
 | 
						|
    assertEquals(i, o.a);
 | 
						|
  }
 | 
						|
  %OptimizeFunctionOnNextCall(Assign2);
 | 
						|
  ReadonlyByNonwritableDataProperty(p, "a");
 | 
						|
  var o = c();
 | 
						|
  Assign2(o, 0);
 | 
						|
  assertTrue(o.a < 0);
 | 
						|
})();
 |