mirror of
				https://github.com/tc39/test262.git
				synced 2025-10-31 11:44:31 +01:00 
			
		
		
		
	sourceRevisionAtLastExport: 33f2fb0e53d135f0ee17cfccd9d993eb2a6f47de targetRevisionAtLastExport: 31340cbd9add103f586d501b0c3354b7b182abc0
		
			
				
	
	
		
			410 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			410 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // Copyright 2013 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: --track-fields --track-double-fields --allow-natives-syntax
 | |
| 
 | |
| // Test transitions caused by changes to field representations.
 | |
| 
 | |
| function create_smi_object() {
 | |
|   var o = {};
 | |
|   o.x = 1;
 | |
|   o.y = 2;
 | |
|   o.z = 3;
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| var o1 = create_smi_object();
 | |
| var o2 = create_smi_object();
 | |
| 
 | |
| // o1,o2 are smi, smi, smi
 | |
| assertTrue(%HaveSameMap(o1, o2));
 | |
| o1.y = 1.3;
 | |
| // o1 is smi, double, smi
 | |
| assertFalse(%HaveSameMap(o1, o2));
 | |
| o2.y = 1.5;
 | |
| // o2 is smi, double, smi
 | |
| assertTrue(%HaveSameMap(o1, o2));
 | |
| 
 | |
| // o3 is initialized as smi, double, smi
 | |
| var o3 = create_smi_object();
 | |
| assertTrue(%HaveSameMap(o1, o3));
 | |
| 
 | |
| function set_large(o, v) {
 | |
|   o.x01 = v; o.x02 = v; o.x03 = v; o.x04 = v; o.x05 = v; o.x06 = v; o.x07 = v;
 | |
|   o.x08 = v; o.x09 = v; o.x10 = v; o.x11 = v; o.x12 = v; o.x13 = v; o.x14 = v;
 | |
|   o.x15 = v; o.x16 = v; o.x17 = v; o.x18 = v; o.x19 = v; o.x20 = v; o.x21 = v;
 | |
|   o.x22 = v; o.x23 = v; o.x24 = v; o.x25 = v; o.x26 = v; o.x27 = v; o.x28 = v;
 | |
|   o.x29 = v; o.x30 = v; o.x31 = v; o.x32 = v; o.x33 = v; o.x34 = v; o.x35 = v;
 | |
|   o.x36 = v; o.x37 = v; o.x38 = v; o.x39 = v; o.x40 = v; o.x41 = v; o.x42 = v;
 | |
|   o.y01 = v; o.y02 = v; o.y03 = v; o.y04 = v; o.y05 = v; o.y06 = v; o.y07 = v;
 | |
|   o.y08 = v; o.y09 = v; o.y10 = v; o.y11 = v; o.y12 = v; o.y13 = v; o.y14 = v;
 | |
|   o.y15 = v; o.y16 = v; o.y17 = v; o.y18 = v; o.y19 = v; o.y20 = v; o.y21 = v;
 | |
| }
 | |
| 
 | |
| // Check that large object migrations work.
 | |
| var o4 = {};
 | |
| // All smi.
 | |
| set_large(o4, 0);
 | |
| assertTrue(%HasFastProperties(o4));
 | |
| // All double.
 | |
| set_large(o4, 1.5);
 | |
| // o5 is immediately allocated with doubles.
 | |
| var o5 = {};
 | |
| set_large(o5, 0);
 | |
| assertTrue(%HaveSameMap(o4, o5));
 | |
| 
 | |
| function create_smi_object2() {
 | |
|   var o = {};
 | |
|   o.a = 1;
 | |
|   o.b = 2;
 | |
|   o.c = 3;
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| // All smi
 | |
| var o6 = create_smi_object2();
 | |
| var o7 = create_smi_object2();
 | |
| 
 | |
| assertTrue(%HaveSameMap(o6, o7));
 | |
| // Smi, double, smi.
 | |
| o6.b = 1.5;
 | |
| assertFalse(%HaveSameMap(o6, o7));
 | |
| // Smi, double, object.
 | |
| o7.c = {};
 | |
| assertFalse(%HaveSameMap(o6, o7));
 | |
| // Smi, double, object.
 | |
| o6.c = {};
 | |
| assertTrue(%HaveSameMap(o6, o7));
 | |
| 
 | |
| function poly_load(o, b) {
 | |
|   var v = o.field;
 | |
|   if (b) {
 | |
|     return v + 10;
 | |
|   }
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| var of1 = {a:0};
 | |
| of1.field = {};
 | |
| var of2 = {b:0};
 | |
| of2.field = 10;
 | |
| 
 | |
| poly_load(of1, false);
 | |
| poly_load(of1, false);
 | |
| poly_load(of2, true);
 | |
| %OptimizeFunctionOnNextCall(poly_load);
 | |
| assertEquals("[object Object]10", poly_load(of1, true));
 | |
| 
 | |
| // Ensure small object literals with doubles do not share double storage.
 | |
| function object_literal() { return {"a":1.5}; }
 | |
| var o8 = object_literal();
 | |
| var o9 = object_literal();
 | |
| o8.a = 4.6
 | |
| assertEquals(1.5, o9.a);
 | |
| 
 | |
| // Ensure double storage is not leaked in the case of polymorphic loads.
 | |
| function load_poly(o) {
 | |
|   return o.a;
 | |
| }
 | |
| 
 | |
| var o10 = { "a": 1.6 };
 | |
| var o11 = { "b": 1, "a": 1.7 };
 | |
| load_poly(o10);
 | |
| load_poly(o10);
 | |
| load_poly(o11);
 | |
| %OptimizeFunctionOnNextCall(load_poly);
 | |
| var val = load_poly(o10);
 | |
| o10.a = 19.5;
 | |
| assertFalse(o10.a == val);
 | |
| 
 | |
| // Ensure polymorphic loads only go monomorphic when the representations are
 | |
| // compatible.
 | |
| 
 | |
| // Check polymorphic load from double + object fields.
 | |
| function load_mono(o) {
 | |
|   return o.a1;
 | |
| }
 | |
| 
 | |
| var object = {"x": 1};
 | |
| var o10 = { "a1": 1.6 };
 | |
| var o11 = { "a1": object, "b": 1 };
 | |
| load_mono(o10);
 | |
| load_mono(o10);
 | |
| load_mono(o11);
 | |
| %OptimizeFunctionOnNextCall(load_mono);
 | |
| assertEquals(object, load_mono(o11));
 | |
| 
 | |
| // Check polymorphic load from smi + object fields.
 | |
| function load_mono2(o) {
 | |
|   return o.a2;
 | |
| }
 | |
| 
 | |
| var o12 = { "a2": 5 };
 | |
| var o13 = { "a2": object, "b": 1 };
 | |
| load_mono2(o12);
 | |
| load_mono2(o12);
 | |
| load_mono2(o13);
 | |
| %OptimizeFunctionOnNextCall(load_mono2);
 | |
| assertEquals(object, load_mono2(o13));
 | |
| 
 | |
| // Check polymorphic load from double + double fields.
 | |
| function load_mono3(o) {
 | |
|   return o.a3;
 | |
| }
 | |
| 
 | |
| var o14 = { "a3": 1.6 };
 | |
| var o15 = { "a3": 1.8, "b": 1 };
 | |
| load_mono3(o14);
 | |
| load_mono3(o14);
 | |
| load_mono3(o15);
 | |
| %OptimizeFunctionOnNextCall(load_mono3);
 | |
| assertEquals(1.6, load_mono3(o14));
 | |
| assertEquals(1.8, load_mono3(o15));
 | |
| 
 | |
| // Check that JSON parsing respects existing representations.
 | |
| var o16 = JSON.parse('{"a":1.5}');
 | |
| var o17 = JSON.parse('{"a":100}');
 | |
| assertTrue(%HaveSameMap(o16, o17));
 | |
| var o17_a = o17.a;
 | |
| assertEquals(100, o17_a);
 | |
| o17.a = 200;
 | |
| assertEquals(100, o17_a);
 | |
| assertEquals(200, o17.a);
 | |
| 
 | |
| // Ensure normalizing results in ignored representations.
 | |
| var o18 = {};
 | |
| o18.field1 = 100;
 | |
| o18.field2 = 1;
 | |
| o18.to_delete = 100;
 | |
| 
 | |
| var o19 = {};
 | |
| o19.field1 = 100;
 | |
| o19.field2 = 1.6;
 | |
| o19.to_delete = 100;
 | |
| 
 | |
| assertFalse(%HaveSameMap(o18, o19));
 | |
| 
 | |
| delete o18.to_delete;
 | |
| delete o19.to_delete;
 | |
| 
 | |
| assertEquals(1, o18.field2);
 | |
| assertEquals(1.6, o19.field2);
 | |
| 
 | |
| // Test megamorphic keyed stub behaviour in combination with representations.
 | |
| var some_object20 = {"a":1};
 | |
| var o20 = {};
 | |
| o20.smi = 1;
 | |
| o20.dbl = 1.5;
 | |
| o20.obj = some_object20;
 | |
| 
 | |
| function keyed_load(o, k) {
 | |
|   return o[k];
 | |
| }
 | |
| 
 | |
| function keyed_store(o, k, v) {
 | |
|   return o[k] = v;
 | |
| }
 | |
| 
 | |
| var smi20 = keyed_load(o20, "smi");
 | |
| var dbl20 = keyed_load(o20, "dbl");
 | |
| var obj20 = keyed_load(o20, "obj");
 | |
| keyed_load(o20, "smi");
 | |
| keyed_load(o20, "dbl");
 | |
| keyed_load(o20, "obj");
 | |
| keyed_load(o20, "smi");
 | |
| keyed_load(o20, "dbl");
 | |
| keyed_load(o20, "obj");
 | |
| 
 | |
| assertEquals(1, smi20);
 | |
| assertEquals(1.5, dbl20);
 | |
| assertEquals(some_object20, obj20);
 | |
| 
 | |
| keyed_store(o20, "smi", 100);
 | |
| keyed_store(o20, "dbl", 100);
 | |
| keyed_store(o20, "obj", 100);
 | |
| keyed_store(o20, "smi", 100);
 | |
| keyed_store(o20, "dbl", 100);
 | |
| keyed_store(o20, "obj", 100);
 | |
| keyed_store(o20, "smi", 100);
 | |
| keyed_store(o20, "dbl", 100);
 | |
| keyed_store(o20, "obj", 100);
 | |
| 
 | |
| assertEquals(1, smi20);
 | |
| assertEquals(1.5, dbl20);
 | |
| assertEquals(some_object20, obj20);
 | |
| 
 | |
| assertEquals(100, o20.smi);
 | |
| assertEquals(100, o20.dbl);
 | |
| assertEquals(100, o20.dbl);
 | |
| 
 | |
| function attr_mismatch_obj(v, writable) {
 | |
|   var o = {};
 | |
|   // Assign twice to make the field non-constant.
 | |
|   // TODO(ishell): update test once constant field tracking is done.
 | |
|   o.some_value = 0;
 | |
|   o.some_value = v;
 | |
|   Object.defineProperty(o, "second_value", {value:10, writable:writable});
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| function is_writable(o, p) {
 | |
|   return Object.getOwnPropertyDescriptor(o,p).writable;
 | |
| }
 | |
| 
 | |
| var writable = attr_mismatch_obj(10, true);
 | |
| var non_writable1 = attr_mismatch_obj(10.5, false);
 | |
| assertTrue(is_writable(writable,"second_value"));
 | |
| assertFalse(is_writable(non_writable1,"second_value"));
 | |
| writable.some_value = 20.5;
 | |
| assertTrue(is_writable(writable,"second_value"));
 | |
| var non_writable2 = attr_mismatch_obj(10.5, false);
 | |
| assertTrue(%HaveSameMap(non_writable1, non_writable2));
 | |
| 
 | |
| function test_f(v) {
 | |
|   var o = {};
 | |
|   o.vbf = v;
 | |
|   o.func = test_f;
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| function test_fic(o) {
 | |
|   return o.vbf;
 | |
| }
 | |
| 
 | |
| var ftest1 = test_f(10);
 | |
| var ftest2 = test_f(10);
 | |
| var ftest3 = test_f(10.5);
 | |
| var ftest4 = test_f(10);
 | |
| assertFalse(%HaveSameMap(ftest1, ftest3));
 | |
| assertTrue(%HaveSameMap(ftest3, ftest4));
 | |
| ftest2.func = is_writable;
 | |
| test_fic(ftest1);
 | |
| test_fic(ftest2);
 | |
| test_fic(ftest3);
 | |
| test_fic(ftest4);
 | |
| assertTrue(%HaveSameMap(ftest1, ftest3));
 | |
| assertTrue(%HaveSameMap(ftest3, ftest4));
 | |
| 
 | |
| // Test representations and transition conversions.
 | |
| function read_first_double(o) {
 | |
|   return o.first_double;
 | |
| }
 | |
| var df1 = {};
 | |
| df1.first_double=1.6;
 | |
| read_first_double(df1);
 | |
| read_first_double(df1);
 | |
| function some_function1() { return 10; }
 | |
| var df2 = {};
 | |
| df2.first_double = 1.7;
 | |
| df2.second_function = some_function1;
 | |
| function some_function2() { return 20; }
 | |
| var df3 = {};
 | |
| df3.first_double = 1.7;
 | |
| df3.second_function = some_function2;
 | |
| df1.first_double = 10;
 | |
| read_first_double(df1);
 | |
| 
 | |
| // Test boilerplates with computed values.
 | |
| function none_boilerplate(a) {
 | |
|   return {"a_none":a};
 | |
| }
 | |
| %OptimizeFunctionOnNextCall(none_boilerplate);
 | |
| var none_double1 = none_boilerplate(1.7);
 | |
| var none_double2 = none_boilerplate(1.9);
 | |
| assertTrue(%HaveSameMap(none_double1, none_double2));
 | |
| assertEquals(1.7, none_double1.a_none);
 | |
| assertEquals(1.9, none_double2.a_none);
 | |
| none_double2.a_none = 3.5;
 | |
| var none_double1 = none_boilerplate(1.7);
 | |
| var none_double2 = none_boilerplate(3.5);
 | |
| 
 | |
| function none_to_smi(a) {
 | |
|   return {"a_smi":a};
 | |
| }
 | |
| 
 | |
| var none_smi1 = none_to_smi(1);
 | |
| var none_smi2 = none_to_smi(2);
 | |
| %OptimizeFunctionOnNextCall(none_to_smi);
 | |
| var none_smi3 = none_to_smi(3);
 | |
| assertTrue(%HaveSameMap(none_smi1, none_smi2));
 | |
| assertTrue(%HaveSameMap(none_smi1, none_smi3));
 | |
| assertEquals(1, none_smi1.a_smi);
 | |
| assertEquals(2, none_smi2.a_smi);
 | |
| assertEquals(3, none_smi3.a_smi);
 | |
| 
 | |
| function none_to_double(a) {
 | |
|   return {"a_double":a};
 | |
| }
 | |
| 
 | |
| var none_to_double1 = none_to_double(1.5);
 | |
| var none_to_double2 = none_to_double(2.8);
 | |
| %OptimizeFunctionOnNextCall(none_to_double);
 | |
| var none_to_double3 = none_to_double(3.7);
 | |
| assertTrue(%HaveSameMap(none_to_double1, none_to_double2));
 | |
| assertTrue(%HaveSameMap(none_to_double1, none_to_double3));
 | |
| assertEquals(1.5, none_to_double1.a_double);
 | |
| assertEquals(2.8, none_to_double2.a_double);
 | |
| assertEquals(3.7, none_to_double3.a_double);
 | |
| 
 | |
| function none_to_object(a) {
 | |
|   return {"an_object":a};
 | |
| }
 | |
| 
 | |
| var none_to_object1 = none_to_object(true);
 | |
| var none_to_object2 = none_to_object(false);
 | |
| %OptimizeFunctionOnNextCall(none_to_object);
 | |
| var none_to_object3 = none_to_object(3.7);
 | |
| assertTrue(%HaveSameMap(none_to_object1, none_to_object2));
 | |
| assertTrue(%HaveSameMap(none_to_object1, none_to_object3));
 | |
| assertEquals(true, none_to_object1.an_object);
 | |
| assertEquals(false, none_to_object2.an_object);
 | |
| assertEquals(3.7, none_to_object3.an_object);
 | |
| 
 | |
| function double_to_object(a) {
 | |
|   var o = {"d_to_h":1.8};
 | |
|   o.d_to_h = a;
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| var dh1 = double_to_object(true);
 | |
| var dh2 = double_to_object(false);
 | |
| assertTrue(%HaveSameMap(dh1, dh2));
 | |
| assertEquals(true, dh1.d_to_h);
 | |
| assertEquals(false, dh2.d_to_h);
 | |
| 
 | |
| function smi_to_object(a) {
 | |
|   var o = {"s_to_t":18};
 | |
|   o.s_to_t = a;
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| var st1 = smi_to_object(true);
 | |
| var st2 = smi_to_object(false);
 | |
| assertTrue(%HaveSameMap(st1, st2));
 | |
| assertEquals(true, st1.s_to_t);
 | |
| assertEquals(false, st2.s_to_t);
 |