Commit a4235f00 authored by Aseem Garg's avatar Aseem Garg Committed by Commit Bot

Revert "[runtime] Improve for-in performance"

This reverts commit 8fa7f9ed.

Reason for revert: Speculating that this breaks GC stress

Original change's description:
> [runtime] Improve for-in performance
> 
> - Add fast-path String conversion for Smi (which is the most common case)
>   This improves for-in by ~10% on non-initialized enum-caches
> - Don't use the NumberStringCache for large indices to not overflow the cache
>   during key collection. This improves worst-case performance by ~2.5x
> - Drop number_to_string_native and number_to_string_runtime counters
> 
> Bug: v8:7717
> Change-Id: Ic1ff385e3374e6a7e7e7bdb9ae75fb8c238105d1
> Reviewed-on: https://chromium-review.googlesource.com/1167049
> Reviewed-by: Toon Verwaest <verwaest@chromium.org>
> Reviewed-by: Ulan Degenbaev <ulan@chromium.org>
> Commit-Queue: Camillo Bruni <cbruni@chromium.org>
> Cr-Commit-Position: refs/heads/master@{#55233}

TBR=ulan@chromium.org,cbruni@chromium.org,verwaest@chromium.org

Change-Id: I8d0332478afcd7c6a3f8fbf1f044b9aa870b6b13
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Bug: v8:7717
Reviewed-on: https://chromium-review.googlesource.com/1182676Reviewed-by: 's avatarAseem Garg <aseemgarg@chromium.org>
Commit-Queue: Aseem Garg <aseemgarg@chromium.org>
Cr-Commit-Position: refs/heads/master@{#55241}
parent 126e88db
...@@ -6549,6 +6549,7 @@ TNode<String> CodeStubAssembler::NumberToString(TNode<Number> input) { ...@@ -6549,6 +6549,7 @@ TNode<String> CodeStubAssembler::NumberToString(TNode<Number> input) {
GotoIfNot(Word32Equal(high, high_compare), &runtime); GotoIfNot(Word32Equal(high, high_compare), &runtime);
// Heap number match, return value from cache entry. // Heap number match, return value from cache entry.
IncrementCounter(isolate()->counters()->number_to_string_native(), 1);
result = CAST( result = CAST(
LoadFixedArrayElement(CAST(number_string_cache), index, kPointerSize)); LoadFixedArrayElement(CAST(number_string_cache), index, kPointerSize));
Goto(&done); Goto(&done);
...@@ -6571,6 +6572,7 @@ TNode<String> CodeStubAssembler::NumberToString(TNode<Number> input) { ...@@ -6571,6 +6572,7 @@ TNode<String> CodeStubAssembler::NumberToString(TNode<Number> input) {
GotoIf(WordNotEqual(smi_key, input), &runtime); GotoIf(WordNotEqual(smi_key, input), &runtime);
// Smi match, return value from cache entry. // Smi match, return value from cache entry.
IncrementCounter(isolate()->counters()->number_to_string_native(), 1);
result = CAST(LoadFixedArrayElement(CAST(number_string_cache), smi_index, result = CAST(LoadFixedArrayElement(CAST(number_string_cache), smi_index,
kPointerSize, SMI_PARAMETERS)); kPointerSize, SMI_PARAMETERS));
Goto(&done); Goto(&done);
......
...@@ -1377,6 +1377,8 @@ class RuntimeCallTimerScope { ...@@ -1377,6 +1377,8 @@ class RuntimeCallTimerScope {
SC(sub_string_native, V8.SubStringNative) \ SC(sub_string_native, V8.SubStringNative) \
SC(regexp_entry_runtime, V8.RegExpEntryRuntime) \ SC(regexp_entry_runtime, V8.RegExpEntryRuntime) \
SC(regexp_entry_native, V8.RegExpEntryNative) \ SC(regexp_entry_native, V8.RegExpEntryNative) \
SC(number_to_string_native, V8.NumberToStringNative) \
SC(number_to_string_runtime, V8.NumberToStringRuntime) \
SC(math_exp_runtime, V8.MathExpRuntime) \ SC(math_exp_runtime, V8.MathExpRuntime) \
SC(math_log_runtime, V8.MathLogRuntime) \ SC(math_log_runtime, V8.MathLogRuntime) \
SC(math_pow_runtime, V8.MathPowRuntime) \ SC(math_pow_runtime, V8.MathPowRuntime) \
......
...@@ -1173,15 +1173,11 @@ class ElementsAccessorBase : public InternalElementsAccessor { ...@@ -1173,15 +1173,11 @@ class ElementsAccessorBase : public InternalElementsAccessor {
PropertyFilter filter, Handle<FixedArray> list, uint32_t* nof_indices, PropertyFilter filter, Handle<FixedArray> list, uint32_t* nof_indices,
uint32_t insertion_index = 0) { uint32_t insertion_index = 0) {
uint32_t length = Subclass::GetMaxIndex(*object, *backing_store); uint32_t length = Subclass::GetMaxIndex(*object, *backing_store);
uint32_t const kMaxStringTableEntries =
isolate->heap()->MaxNumberToStringCacheSize();
for (uint32_t i = 0; i < length; i++) { for (uint32_t i = 0; i < length; i++) {
if (Subclass::HasElementImpl(isolate, *object, i, *backing_store, if (Subclass::HasElementImpl(isolate, *object, i, *backing_store,
filter)) { filter)) {
if (convert == GetKeysConversion::kConvertToString) { if (convert == GetKeysConversion::kConvertToString) {
bool use_cache = i < kMaxStringTableEntries; Handle<String> index_string = isolate->factory()->Uint32ToString(i);
Handle<String> index_string =
isolate->factory()->Uint32ToString(i, use_cache);
list->set(insertion_index, *index_string); list->set(insertion_index, *index_string);
} else { } else {
list->set(insertion_index, Smi::FromInt(i), SKIP_WRITE_BARRIER); list->set(insertion_index, Smi::FromInt(i), SKIP_WRITE_BARRIER);
......
...@@ -164,14 +164,8 @@ Handle<Object> Factory::NewURIError() { ...@@ -164,14 +164,8 @@ Handle<Object> Factory::NewURIError() {
MessageTemplate::kURIMalformed); MessageTemplate::kURIMalformed);
} }
Handle<String> Factory::Uint32ToString(uint32_t value, bool check_cache) { Handle<String> Factory::Uint32ToString(uint32_t value) {
Handle<String> result; Handle<String> result = NumberToString(NewNumberFromUint(value));
int32_t int32v = static_cast<int32_t>(value);
if (int32v >= 0 && Smi::IsValid(int32v)) {
result = NumberToString(Smi::FromInt(int32v), check_cache);
} else {
result = NumberToString(NewNumberFromUint(value), check_cache);
}
if (result->length() <= String::kMaxArrayIndexSize && if (result->length() <= String::kMaxArrayIndexSize &&
result->hash_field() == String::kEmptyHashField) { result->hash_field() == String::kEmptyHashField) {
......
...@@ -3539,90 +3539,68 @@ Handle<SharedFunctionInfo> Factory::NewSharedFunctionInfo( ...@@ -3539,90 +3539,68 @@ Handle<SharedFunctionInfo> Factory::NewSharedFunctionInfo(
return share; return share;
} }
namespace { static inline int NumberCacheHash(Handle<FixedArray> cache,
inline int NumberToStringCacheHash(Handle<FixedArray> cache, Smi* number) { Handle<Object> number) {
int mask = (cache->length() >> 1) - 1;
return number->value() & mask;
}
inline int NumberToStringCacheHash(Handle<FixedArray> cache, double number) {
int mask = (cache->length() >> 1) - 1; int mask = (cache->length() >> 1) - 1;
int64_t bits = bit_cast<int64_t>(number); if (number->IsSmi()) {
return (static_cast<int>(bits) ^ static_cast<int>(bits >> 32)) & mask; return Handle<Smi>::cast(number)->value() & mask;
} } else {
} // namespace int64_t bits = bit_cast<int64_t>(number->Number());
return (static_cast<int>(bits) ^ static_cast<int>(bits >> 32)) & mask;
Handle<String> Factory::NumberToStringCacheSet(Handle<Object> number, int hash,
const char* string,
bool check_cache) {
// We tenure the allocated string since it is referenced from the
// number-string cache which lives in the old space.
Handle<String> js_string =
NewStringFromAsciiChecked(string, check_cache ? TENURED : NOT_TENURED);
if (!check_cache) return js_string;
if (!number_string_cache()->get(hash * 2)->IsUndefined(isolate())) {
int full_size = isolate()->heap()->MaxNumberToStringCacheSize();
if (number_string_cache()->length() != full_size) {
Handle<FixedArray> new_cache = NewFixedArray(full_size, TENURED);
isolate()->heap()->set_number_string_cache(*new_cache);
return js_string;
}
} }
number_string_cache()->set(hash * 2, *number);
number_string_cache()->set(hash * 2 + 1, *js_string);
return js_string;
} }
Handle<Object> Factory::NumberToStringCacheGet(Object* number, int hash) { Handle<Object> Factory::GetNumberStringCache(Handle<Object> number) {
DisallowHeapAllocation no_gc; DisallowHeapAllocation no_gc;
int hash = NumberCacheHash(number_string_cache(), number);
Object* key = number_string_cache()->get(hash * 2); Object* key = number_string_cache()->get(hash * 2);
if (key == number || (key->IsHeapNumber() && number->IsHeapNumber() && if (key == *number || (key->IsHeapNumber() && number->IsHeapNumber() &&
key->Number() == number->Number())) { key->Number() == number->Number())) {
return Handle<String>( return Handle<String>(
String::cast(number_string_cache()->get(hash * 2 + 1)), isolate()); String::cast(number_string_cache()->get(hash * 2 + 1)), isolate());
} }
return undefined_value(); return undefined_value();
} }
Handle<String> Factory::NumberToString(Handle<Object> number, void Factory::SetNumberStringCache(Handle<Object> number,
bool check_cache) { Handle<String> string) {
if (number->IsSmi()) return NumberToString(Smi::cast(*number), check_cache); int hash = NumberCacheHash(number_string_cache(), number);
if (number_string_cache()->get(hash * 2) != *undefined_value()) {
double double_value = Handle<HeapNumber>::cast(number)->value(); int full_size = isolate()->heap()->FullSizeNumberStringCacheLength();
// Try to canonicalize doubles. if (number_string_cache()->length() != full_size) {
int smi_value; Handle<FixedArray> new_cache = NewFixedArray(full_size, TENURED);
if (DoubleToSmiInteger(double_value, &smi_value)) { isolate()->heap()->set_number_string_cache(*new_cache);
return NumberToString(Smi::FromInt(smi_value), check_cache); return;
} }
int hash = 0;
if (check_cache) {
int hash = NumberToStringCacheHash(number_string_cache(), double_value);
Handle<Object> cached = NumberToStringCacheGet(*number, hash);
if (!cached->IsUndefined(isolate())) return Handle<String>::cast(cached);
} }
number_string_cache()->set(hash * 2, *number);
char arr[100]; number_string_cache()->set(hash * 2 + 1, *string);
Vector<char> buffer(arr, arraysize(arr));
const char* string = DoubleToCString(double_value, buffer);
return NumberToStringCacheSet(number, hash, string, check_cache);
} }
Handle<String> Factory::NumberToString(Smi* number, bool check_cache) { Handle<String> Factory::NumberToString(Handle<Object> number,
int hash = 0; bool check_number_string_cache) {
if (check_cache) { isolate()->counters()->number_to_string_runtime()->Increment();
hash = NumberToStringCacheHash(number_string_cache(), number); if (check_number_string_cache) {
Handle<Object> cached = NumberToStringCacheGet(number, hash); Handle<Object> cached = GetNumberStringCache(number);
if (!cached->IsUndefined(isolate())) return Handle<String>::cast(cached); if (!cached->IsUndefined(isolate())) return Handle<String>::cast(cached);
} }
char arr[100]; char arr[100];
Vector<char> buffer(arr, arraysize(arr)); Vector<char> buffer(arr, arraysize(arr));
const char* string = IntToCString(number->value(), buffer); const char* str;
if (number->IsSmi()) {
int num = Handle<Smi>::cast(number)->value();
str = IntToCString(num, buffer);
} else {
double num = Handle<HeapNumber>::cast(number)->value();
str = DoubleToCString(num, buffer);
}
return NumberToStringCacheSet(handle(number, isolate()), hash, string, // We tenure the allocated string since it is referenced from the
check_cache); // number-string cache which lives in the old space.
Handle<String> js_string = NewStringFromAsciiChecked(str, TENURED);
SetNumberStringCache(number, js_string);
return js_string;
} }
Handle<DebugInfo> Factory::NewDebugInfo(Handle<SharedFunctionInfo> shared) { Handle<DebugInfo> Factory::NewDebugInfo(Handle<SharedFunctionInfo> shared) {
......
...@@ -822,11 +822,10 @@ class V8_EXPORT_PRIVATE Factory { ...@@ -822,11 +822,10 @@ class V8_EXPORT_PRIVATE Factory {
DECLARE_ERROR(WasmRuntimeError) DECLARE_ERROR(WasmRuntimeError)
#undef DECLARE_ERROR #undef DECLARE_ERROR
Handle<String> NumberToString(Handle<Object> number, bool check_cache = true); Handle<String> NumberToString(Handle<Object> number,
Handle<String> NumberToString(Smi* number, bool check_cache = true); bool check_number_string_cache = true);
inline Handle<String> Uint32ToString(uint32_t value, inline Handle<String> Uint32ToString(uint32_t value);
bool check_cache = false);
#define ROOT_ACCESSOR(type, name, camel_name) inline Handle<type> name(); #define ROOT_ACCESSOR(type, name, camel_name) inline Handle<type> name();
ROOT_LIST(ROOT_ACCESSOR) ROOT_LIST(ROOT_ACCESSOR)
...@@ -1001,11 +1000,10 @@ class V8_EXPORT_PRIVATE Factory { ...@@ -1001,11 +1000,10 @@ class V8_EXPORT_PRIVATE Factory {
// Attempt to find the number in a small cache. If we finds it, return // Attempt to find the number in a small cache. If we finds it, return
// the string representation of the number. Otherwise return undefined. // the string representation of the number. Otherwise return undefined.
Handle<Object> NumberToStringCacheGet(Object* number, int hash); Handle<Object> GetNumberStringCache(Handle<Object> number);
// Update the cache with a new number-string pair. // Update the cache with a new number-string pair.
Handle<String> NumberToStringCacheSet(Handle<Object> number, int hash, void SetNumberStringCache(Handle<Object> number, Handle<String> string);
const char* string, bool check_cache);
// Create a JSArray with no elements and no length. // Create a JSArray with no elements and no length.
Handle<JSArray> NewJSArray(ElementsKind elements_kind, Handle<JSArray> NewJSArray(ElementsKind elements_kind,
......
...@@ -569,18 +569,6 @@ int Heap::GetNextTemplateSerialNumber() { ...@@ -569,18 +569,6 @@ int Heap::GetNextTemplateSerialNumber() {
return next_serial_number; return next_serial_number;
} }
int Heap::MaxNumberToStringCacheSize() const {
// Compute the size of the number string cache based on the max newspace size.
// The number string cache has a minimum size based on twice the initial cache
// size to ensure that it is bigger after being made 'full size'.
size_t number_string_cache_size = max_semi_space_size_ / 512;
number_string_cache_size =
Max(static_cast<size_t>(kInitialNumberStringCacheSize * 2),
Min<size_t>(0x4000u, number_string_cache_size));
// There is a string and a number per entry so the length is twice the number
// of entries.
return static_cast<int>(number_string_cache_size * 2);
}
AlwaysAllocateScope::AlwaysAllocateScope(Isolate* isolate) AlwaysAllocateScope::AlwaysAllocateScope(Isolate* isolate)
: heap_(isolate->heap()) { : heap_(isolate->heap()) {
heap_->always_allocate_scope_count_++; heap_->always_allocate_scope_count_++;
......
...@@ -2722,6 +2722,19 @@ bool Heap::RootCanBeTreatedAsConstant(RootListIndex root_index) { ...@@ -2722,6 +2722,19 @@ bool Heap::RootCanBeTreatedAsConstant(RootListIndex root_index) {
return can_be; return can_be;
} }
int Heap::FullSizeNumberStringCacheLength() {
// Compute the size of the number string cache based on the max newspace size.
// The number string cache has a minimum size based on twice the initial cache
// size to ensure that it is bigger after being made 'full size'.
size_t number_string_cache_size = max_semi_space_size_ / 512;
number_string_cache_size =
Max(static_cast<size_t>(kInitialNumberStringCacheSize * 2),
Min<size_t>(0x4000u, number_string_cache_size));
// There is a string and a number per entry so the length is twice the number
// of entries.
return static_cast<int>(number_string_cache_size * 2);
}
void Heap::FlushNumberStringCache() { void Heap::FlushNumberStringCache() {
// Flush the number to string cache. // Flush the number to string cache.
......
...@@ -1480,9 +1480,6 @@ class Heap { ...@@ -1480,9 +1480,6 @@ class Heap {
static const char* GarbageCollectionReasonToString( static const char* GarbageCollectionReasonToString(
GarbageCollectionReason gc_reason); GarbageCollectionReason gc_reason);
// Calculates the nof entries for the full sized number to string cache.
inline int MaxNumberToStringCacheSize() const;
private: private:
class SkipStoreBufferScope; class SkipStoreBufferScope;
...@@ -1691,6 +1688,8 @@ class Heap { ...@@ -1691,6 +1688,8 @@ class Heap {
// Record statistics after garbage collection. // Record statistics after garbage collection.
void ReportStatisticsAfterGC(); void ReportStatisticsAfterGC();
// Creates and installs the full-sized number string cache.
int FullSizeNumberStringCacheLength();
// Flush the number to string cache. // Flush the number to string cache.
void FlushNumberStringCache(); void FlushNumberStringCache();
......
...@@ -126,17 +126,13 @@ Handle<FixedArray> OrderedHashSet::ConvertToKeysArray( ...@@ -126,17 +126,13 @@ Handle<FixedArray> OrderedHashSet::ConvertToKeysArray(
Handle<FixedArray> result = Handle<FixedArray>::cast(table); Handle<FixedArray> result = Handle<FixedArray>::cast(table);
// From this point on table is no longer a valid OrderedHashSet. // From this point on table is no longer a valid OrderedHashSet.
result->set_map(ReadOnlyRoots(isolate).fixed_array_map()); result->set_map(ReadOnlyRoots(isolate).fixed_array_map());
int const kMaxStringTableEntries =
isolate->heap()->MaxNumberToStringCacheSize();
for (int i = 0; i < length; i++) { for (int i = 0; i < length; i++) {
int index = kHashTableStartIndex + nof_buckets + (i * kEntrySize); int index = kHashTableStartIndex + nof_buckets + (i * kEntrySize);
Object* key = table->get(index); Object* key = table->get(index);
if (convert == GetKeysConversion::kConvertToString) { if (convert == GetKeysConversion::kConvertToString) {
uint32_t index_value; uint32_t index_value;
if (key->ToArrayIndex(&index_value)) { if (key->ToArrayIndex(&index_value)) {
// Avoid trashing the Number2String cache if indices get very large. key = *isolate->factory()->Uint32ToString(index_value);
bool use_cache = i < kMaxStringTableEntries;
key = *isolate->factory()->Uint32ToString(index_value, use_cache);
} else { } else {
CHECK(key->IsName()); CHECK(key->IsName());
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment