21a66b11d3
Add SetProperties as the generic interface to set properties. In the future, this will switch based on the input properties type and correctly store the hash code. This patch also updates tests to check against empty_property_array instead of empty_fixed_array. Bug: v8:6404 Change-Id: I39d324ea3ab3cc2c2223b6f4be64139bb88edd94 Reviewed-on: https://chromium-review.googlesource.com/574761 Reviewed-by: Jakob Kummerow <jkummerow@chromium.org> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Michael Lippautz <mlippautz@chromium.org> Commit-Queue: Sathya Gunasekaran <gsathya@chromium.org> Cr-Commit-Position: refs/heads/master@{#46744}
481 lines
18 KiB
C++
481 lines
18 KiB
C++
// Copyright 2015 the V8 project authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include <stdlib.h>
|
|
#include <utility>
|
|
|
|
#include "test/cctest/test-api.h"
|
|
|
|
#include "src/v8.h"
|
|
|
|
#include "src/compilation-cache.h"
|
|
#include "src/execution.h"
|
|
#include "src/factory.h"
|
|
#include "src/global-handles.h"
|
|
#include "src/ic/stub-cache.h"
|
|
#include "src/objects-inl.h"
|
|
|
|
using namespace v8::internal;
|
|
|
|
|
|
//
|
|
// Helper functions.
|
|
//
|
|
|
|
namespace {
|
|
|
|
Handle<String> MakeString(const char* str) {
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
return factory->InternalizeUtf8String(str);
|
|
}
|
|
|
|
|
|
Handle<String> MakeName(const char* str, int suffix) {
|
|
EmbeddedVector<char, 128> buffer;
|
|
SNPrintF(buffer, "%s%d", str, suffix);
|
|
return MakeString(buffer.start());
|
|
}
|
|
|
|
|
|
template <typename T, typename M>
|
|
bool EQUALS(Handle<T> left, Handle<M> right) {
|
|
if (*left == *right) return true;
|
|
return JSObject::Equals(Handle<Object>::cast(left),
|
|
Handle<Object>::cast(right))
|
|
.FromJust();
|
|
}
|
|
|
|
|
|
template <typename T, typename M>
|
|
bool EQUALS(Handle<T> left, M right) {
|
|
return EQUALS(left, handle(right));
|
|
}
|
|
|
|
|
|
template <typename T, typename M>
|
|
bool EQUALS(T left, Handle<M> right) {
|
|
return EQUALS(handle(left, right->GetIsolate()), right);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
//
|
|
// Tests
|
|
//
|
|
|
|
TEST(JSObjectAddingProperties) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<FixedArray> empty_fixed_array(factory->empty_fixed_array());
|
|
Handle<PropertyArray> empty_property_array(factory->empty_property_array());
|
|
Handle<JSFunction> function = factory->NewFunction(factory->empty_string());
|
|
Handle<Object> value(Smi::FromInt(42), isolate);
|
|
|
|
Handle<JSObject> object = factory->NewJSObject(function);
|
|
Handle<Map> previous_map(object->map());
|
|
CHECK_EQ(HOLEY_ELEMENTS, previous_map->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
|
|
// for the default constructor function no in-object properties are reserved
|
|
// hence adding a single property will initialize the property-array
|
|
Handle<String> name = MakeName("property", 0);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
CHECK_NE(object->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, object->map()->elements_kind());
|
|
CHECK_LE(1, object->property_array()->length());
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
}
|
|
|
|
|
|
TEST(JSObjectInObjectAddingProperties) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<FixedArray> empty_fixed_array(factory->empty_fixed_array());
|
|
Handle<PropertyArray> empty_property_array(factory->empty_property_array());
|
|
Handle<JSFunction> function = factory->NewFunction(factory->empty_string());
|
|
int nof_inobject_properties = 10;
|
|
// force in object properties by changing the expected_nof_properties
|
|
function->shared()->set_expected_nof_properties(nof_inobject_properties);
|
|
Handle<Object> value(Smi::FromInt(42), isolate);
|
|
|
|
Handle<JSObject> object = factory->NewJSObject(function);
|
|
Handle<Map> previous_map(object->map());
|
|
CHECK_EQ(HOLEY_ELEMENTS, previous_map->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
|
|
// we have reserved space for in-object properties, hence adding up to
|
|
// |nof_inobject_properties| will not create a property store
|
|
for (int i = 0; i < nof_inobject_properties; i++) {
|
|
Handle<String> name = MakeName("property", i);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
}
|
|
CHECK_NE(object->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, object->map()->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
|
|
// adding one more property will not fit in the in-object properties, thus
|
|
// creating a property store
|
|
int index = nof_inobject_properties + 1;
|
|
Handle<String> name = MakeName("property", index);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
CHECK_NE(object->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, object->map()->elements_kind());
|
|
// there must be at least 1 element in the properies store
|
|
CHECK_LE(1, object->property_array()->length());
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
}
|
|
|
|
|
|
TEST(JSObjectAddingElements) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<String> name;
|
|
Handle<FixedArray> empty_fixed_array(factory->empty_fixed_array());
|
|
Handle<PropertyArray> empty_property_array(factory->empty_property_array());
|
|
Handle<JSFunction> function = factory->NewFunction(factory->empty_string());
|
|
Handle<Object> value(Smi::FromInt(42), isolate);
|
|
|
|
Handle<JSObject> object = factory->NewJSObject(function);
|
|
Handle<Map> previous_map(object->map());
|
|
CHECK_EQ(HOLEY_ELEMENTS, previous_map->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK(EQUALS(object->elements(), empty_fixed_array));
|
|
|
|
// Adding an indexed element initializes the elements array
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(object->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, object->map()->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK_LE(1, object->elements()->length());
|
|
|
|
// Adding more consecutive elements without a change in the backing store
|
|
int non_dict_backing_store_limit = 100;
|
|
for (int i = 1; i < non_dict_backing_store_limit; i++) {
|
|
name = MakeName("", i);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
}
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(object->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, object->map()->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK_LE(non_dict_backing_store_limit, object->elements()->length());
|
|
|
|
// Adding an element at an very large index causes a change to
|
|
// DICTIONARY_ELEMENTS
|
|
name = MakeString("100000000");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(object, name, value, NONE)
|
|
.Check();
|
|
// change in elements_kind => map transition
|
|
CHECK_NE(object->map(), *previous_map);
|
|
CHECK_EQ(DICTIONARY_ELEMENTS, object->map()->elements_kind());
|
|
CHECK(EQUALS(object->property_array(), empty_property_array));
|
|
CHECK_LE(non_dict_backing_store_limit, object->elements()->length());
|
|
}
|
|
|
|
|
|
TEST(JSArrayAddingProperties) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<FixedArray> empty_fixed_array(factory->empty_fixed_array());
|
|
Handle<PropertyArray> empty_property_array(factory->empty_property_array());
|
|
Handle<Object> value(Smi::FromInt(42), isolate);
|
|
|
|
Handle<JSArray> array =
|
|
factory->NewJSArray(ElementsKind::PACKED_SMI_ELEMENTS, 0, 0);
|
|
Handle<Map> previous_map(array->map());
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, previous_map->elements_kind());
|
|
CHECK(EQUALS(array->property_array(), empty_property_array));
|
|
CHECK(EQUALS(array->elements(), empty_fixed_array));
|
|
CHECK_EQ(0, Smi::ToInt(array->length()));
|
|
|
|
// for the default constructor function no in-object properties are reserved
|
|
// hence adding a single property will initialize the property-array
|
|
Handle<String> name = MakeName("property", 0);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value, NONE)
|
|
.Check();
|
|
// No change in elements_kind but added property => new map
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_LE(1, array->property_array()->length());
|
|
CHECK(EQUALS(array->elements(), empty_fixed_array));
|
|
CHECK_EQ(0, Smi::ToInt(array->length()));
|
|
}
|
|
|
|
|
|
TEST(JSArrayAddingElements) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<String> name;
|
|
Handle<FixedArray> empty_fixed_array(factory->empty_fixed_array());
|
|
Handle<PropertyArray> empty_property_array(factory->empty_property_array());
|
|
Handle<Object> value(Smi::FromInt(42), isolate);
|
|
|
|
Handle<JSArray> array =
|
|
factory->NewJSArray(ElementsKind::PACKED_SMI_ELEMENTS, 0, 0);
|
|
Handle<Map> previous_map(array->map());
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, previous_map->elements_kind());
|
|
CHECK(EQUALS(array->property_array(), empty_property_array));
|
|
CHECK(EQUALS(array->elements(), empty_fixed_array));
|
|
CHECK_EQ(0, Smi::ToInt(array->length()));
|
|
|
|
// Adding an indexed element initializes the elements array
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value, NONE)
|
|
.Check();
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK(EQUALS(array->property_array(), empty_property_array));
|
|
CHECK_LE(1, array->elements()->length());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
|
|
// Adding more consecutive elements without a change in the backing store
|
|
int non_dict_backing_store_limit = 100;
|
|
for (int i = 1; i < non_dict_backing_store_limit; i++) {
|
|
name = MakeName("", i);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value, NONE)
|
|
.Check();
|
|
}
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK(EQUALS(array->property_array(), empty_property_array));
|
|
CHECK_LE(non_dict_backing_store_limit, array->elements()->length());
|
|
CHECK_EQ(non_dict_backing_store_limit, Smi::ToInt(array->length()));
|
|
|
|
// Adding an element at an very large index causes a change to
|
|
// DICTIONARY_ELEMENTS
|
|
int index = 100000000;
|
|
name = MakeName("", index);
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value, NONE)
|
|
.Check();
|
|
// change in elements_kind => map transition
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(DICTIONARY_ELEMENTS, array->map()->elements_kind());
|
|
CHECK(EQUALS(array->property_array(), empty_property_array));
|
|
CHECK_LE(non_dict_backing_store_limit, array->elements()->length());
|
|
CHECK_LE(array->elements()->length(), index);
|
|
CHECK_EQ(index + 1, Smi::ToInt(array->length()));
|
|
}
|
|
|
|
|
|
TEST(JSArrayAddingElementsGeneralizingiFastSmiElements) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<String> name;
|
|
Handle<Object> value_smi(Smi::FromInt(42), isolate);
|
|
Handle<Object> value_string(MakeString("value"));
|
|
Handle<Object> value_double = factory->NewNumber(3.1415);
|
|
|
|
Handle<JSArray> array =
|
|
factory->NewJSArray(ElementsKind::PACKED_SMI_ELEMENTS, 0, 0);
|
|
Handle<Map> previous_map(array->map());
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, previous_map->elements_kind());
|
|
CHECK_EQ(0, Smi::ToInt(array->length()));
|
|
|
|
// `array[0] = smi_value` doesn't change the elements_kind
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
|
|
// `delete array[0]` does not alter length, but changes the elments_kind
|
|
name = MakeString("0");
|
|
CHECK(JSReceiver::DeletePropertyOrElement(array, name).FromMaybe(false));
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// add a couple of elements again
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
name = MakeString("1");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_SMI_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
|
|
// Adding a string to the array changes from FAST_HOLEY_SMI to FAST_HOLEY
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_string,
|
|
NONE)
|
|
.Check();
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// We don't transition back to FAST_SMI even if we remove the string
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
|
|
// Adding a double doesn't change the map either
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_double,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
}
|
|
|
|
|
|
TEST(JSArrayAddingElementsGeneralizingFastElements) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<String> name;
|
|
Handle<Object> value_smi(Smi::FromInt(42), isolate);
|
|
Handle<Object> value_string(MakeString("value"));
|
|
|
|
Handle<JSArray> array =
|
|
factory->NewJSArray(ElementsKind::PACKED_ELEMENTS, 0, 0);
|
|
Handle<Map> previous_map(array->map());
|
|
CHECK_EQ(PACKED_ELEMENTS, previous_map->elements_kind());
|
|
CHECK_EQ(0, Smi::ToInt(array->length()));
|
|
|
|
// `array[0] = smi_value` doesn't change the elements_kind
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
// no change in elements_kind => no map transition
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
|
|
// `delete array[0]` does not alter length, but changes the elments_kind
|
|
name = MakeString("0");
|
|
CHECK(JSReceiver::DeletePropertyOrElement(array, name).FromMaybe(false));
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// add a couple of elements, elements_kind stays HOLEY
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_string,
|
|
NONE)
|
|
.Check();
|
|
name = MakeString("1");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
}
|
|
|
|
|
|
TEST(JSArrayAddingElementsGeneralizingiFastDoubleElements) {
|
|
CcTest::InitializeVM();
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
Factory* factory = isolate->factory();
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
Handle<String> name;
|
|
Handle<Object> value_smi(Smi::FromInt(42), isolate);
|
|
Handle<Object> value_string(MakeString("value"));
|
|
Handle<Object> value_double = factory->NewNumber(3.1415);
|
|
|
|
Handle<JSArray> array =
|
|
factory->NewJSArray(ElementsKind::PACKED_SMI_ELEMENTS, 0, 0);
|
|
Handle<Map> previous_map(array->map());
|
|
|
|
// `array[0] = value_double` changes |elements_kind| to PACKED_DOUBLE_ELEMENTS
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_double,
|
|
NONE)
|
|
.Check();
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_DOUBLE_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(1, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// `array[1] = value_smi` doesn't alter the |elements_kind|
|
|
name = MakeString("1");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_smi,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(PACKED_DOUBLE_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
|
|
// `delete array[0]` does not alter length, but changes the elments_kind
|
|
name = MakeString("0");
|
|
CHECK(JSReceiver::DeletePropertyOrElement(array, name).FromMaybe(false));
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_DOUBLE_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// filling the hole `array[0] = value_smi` again doesn't transition back
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_double,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_DOUBLE_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
|
|
// Adding a string to the array changes to elements_kind PACKED_ELEMENTS
|
|
name = MakeString("1");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_string,
|
|
NONE)
|
|
.Check();
|
|
CHECK_NE(array->map(), *previous_map);
|
|
CHECK_EQ(HOLEY_ELEMENTS, array->map()->elements_kind());
|
|
CHECK_EQ(2, Smi::ToInt(array->length()));
|
|
previous_map = handle(array->map());
|
|
|
|
// Adding a double doesn't change the map
|
|
name = MakeString("0");
|
|
JSObject::DefinePropertyOrElementIgnoreAttributes(array, name, value_double,
|
|
NONE)
|
|
.Check();
|
|
CHECK_EQ(array->map(), *previous_map);
|
|
}
|