Use generic array functions to setup the boilerplate to avoid logic duplication.

Review URL: https://chromiumcodereview.appspot.com/11361281

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@13097 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
verwaest@chromium.org 2012-11-29 15:55:27 +00:00
parent 5a40f006f9
commit 349afd692d

View File

@ -3714,16 +3714,16 @@ Expression* Parser::ParseArrayLiteral(bool* ok) {
int literal_index = current_function_state_->NextMaterializedLiteralIndex(); int literal_index = current_function_state_->NextMaterializedLiteralIndex();
// Allocate a fixed array to hold all the object literals. // Allocate a fixed array to hold all the object literals.
Handle<FixedArray> object_literals = Handle<JSArray> array =
isolate()->factory()->NewFixedArray(values->length(), TENURED); isolate()->factory()->NewJSArray(0, FAST_HOLEY_SMI_ELEMENTS);
Handle<FixedDoubleArray> double_literals; isolate()->factory()->SetElementsCapacityAndLength(
ElementsKind elements_kind = FAST_SMI_ELEMENTS; array, values->length(), values->length());
bool has_hole_values = false;
// Fill in the literals. // Fill in the literals.
Heap* heap = isolate()->heap(); Heap* heap = isolate()->heap();
bool is_simple = true; bool is_simple = true;
int depth = 1; int depth = 1;
bool is_holey = false;
for (int i = 0, n = values->length(); i < n; i++) { for (int i = 0, n = values->length(); i < n; i++) {
MaterializedLiteral* m_literal = values->at(i)->AsMaterializedLiteral(); MaterializedLiteral* m_literal = values->at(i)->AsMaterializedLiteral();
if (m_literal != NULL && m_literal->depth() + 1 > depth) { if (m_literal != NULL && m_literal->depth() + 1 > depth) {
@ -3731,83 +3731,33 @@ Expression* Parser::ParseArrayLiteral(bool* ok) {
} }
Handle<Object> boilerplate_value = GetBoilerplateValue(values->at(i)); Handle<Object> boilerplate_value = GetBoilerplateValue(values->at(i));
if (boilerplate_value->IsTheHole()) { if (boilerplate_value->IsTheHole()) {
has_hole_values = true; is_holey = true;
object_literals->set_the_hole(i);
if (elements_kind == FAST_DOUBLE_ELEMENTS) {
double_literals->set_the_hole(i);
}
} else if (boilerplate_value->IsUndefined()) { } else if (boilerplate_value->IsUndefined()) {
is_simple = false; is_simple = false;
object_literals->set(i, Smi::FromInt(0)); JSObject::SetOwnElement(
if (elements_kind == FAST_DOUBLE_ELEMENTS) { array, i, handle(Smi::FromInt(0), isolate()), kNonStrictMode);
double_literals->set(i, 0);
}
} else { } else {
// Examine each literal element, and adjust the ElementsKind if the JSObject::SetOwnElement(array, i, boilerplate_value, kNonStrictMode);
// literal element is not of a type that can be stored in the current
// ElementsKind. Start with FAST_SMI_ONLY_ELEMENTS, and transition to
// FAST_DOUBLE_ELEMENTS and FAST_ELEMENTS as necessary. Always remember
// the tagged value, no matter what the ElementsKind is in case we
// ultimately end up in FAST_ELEMENTS.
object_literals->set(i, *boilerplate_value);
if (elements_kind == FAST_SMI_ELEMENTS) {
// Smi only elements. Notice if a transition to FAST_DOUBLE_ELEMENTS or
// FAST_ELEMENTS is required.
if (!boilerplate_value->IsSmi()) {
if (boilerplate_value->IsNumber() && FLAG_smi_only_arrays) {
// Allocate a double array on the FAST_DOUBLE_ELEMENTS transition to
// avoid over-allocating in TENURED space.
double_literals = isolate()->factory()->NewFixedDoubleArray(
values->length(), TENURED);
// Copy the contents of the FAST_SMI_ONLY_ELEMENT array to the
// FAST_DOUBLE_ELEMENTS array so that they are in sync.
for (int j = 0; j < i; ++j) {
Object* smi_value = object_literals->get(j);
if (smi_value->IsTheHole()) {
double_literals->set_the_hole(j);
} else {
double_literals->set(j, Smi::cast(smi_value)->value());
}
}
double_literals->set(i, boilerplate_value->Number());
elements_kind = FAST_DOUBLE_ELEMENTS;
} else {
elements_kind = FAST_ELEMENTS;
}
}
} else if (elements_kind == FAST_DOUBLE_ELEMENTS) {
// Continue to store double values in to FAST_DOUBLE_ELEMENTS arrays
// until the first value is seen that can't be stored as a double.
if (boilerplate_value->IsNumber()) {
double_literals->set(i, boilerplate_value->Number());
} else {
elements_kind = FAST_ELEMENTS;
}
}
} }
} }
Handle<FixedArrayBase> element_values(array->elements());
// Simple and shallow arrays can be lazily copied, we transform the // Simple and shallow arrays can be lazily copied, we transform the
// elements array to a copy-on-write array. // elements array to a copy-on-write array.
if (is_simple && depth == 1 && values->length() > 0 && if (is_simple && depth == 1 && values->length() > 0 &&
elements_kind != FAST_DOUBLE_ELEMENTS) { array->HasFastSmiOrObjectElements()) {
object_literals->set_map(heap->fixed_cow_array_map()); element_values->set_map(heap->fixed_cow_array_map());
} }
Handle<FixedArrayBase> element_values = elements_kind == FAST_DOUBLE_ELEMENTS
? Handle<FixedArrayBase>(double_literals)
: Handle<FixedArrayBase>(object_literals);
// Remember both the literal's constant values as well as the ElementsKind // Remember both the literal's constant values as well as the ElementsKind
// in a 2-element FixedArray. // in a 2-element FixedArray.
Handle<FixedArray> literals = Handle<FixedArray> literals = isolate()->factory()->NewFixedArray(2, TENURED);
isolate()->factory()->NewFixedArray(2, TENURED);
if (has_hole_values || !FLAG_packed_arrays) { ElementsKind kind = array->GetElementsKind();
elements_kind = GetHoleyElementsKind(elements_kind); kind = is_holey ? GetHoleyElementsKind(kind) : GetPackedElementsKind(kind);
}
literals->set(0, Smi::FromInt(elements_kind)); literals->set(0, Smi::FromInt(kind));
literals->set(1, *element_values); literals->set(1, *element_values);
return factory()->NewArrayLiteral( return factory()->NewArrayLiteral(