| OLD | NEW |
| (Empty) |
| 1 // Copyright 2016 the V8 project authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "src/heap/array-buffer-tracker-inl.h" | |
| 6 #include "src/heap/array-buffer-tracker.h" | |
| 7 #include "test/cctest/cctest.h" | |
| 8 #include "test/cctest/heap/heap-utils.h" | |
| 9 | |
| 10 namespace { | |
| 11 | |
| 12 typedef i::LocalArrayBufferTracker LocalTracker; | |
| 13 | |
| 14 void VerifyTrackedInNewSpace(i::JSArrayBuffer* buf) { | |
| 15 CHECK(i::Page::FromAddress(buf->address())->InNewSpace()); | |
| 16 CHECK(i::Page::FromAddress(buf->address())->local_tracker()->IsTracked(buf)); | |
| 17 } | |
| 18 | |
| 19 void VerifyTrackedInOldSpace(i::JSArrayBuffer* buf) { | |
| 20 CHECK(!i::Page::FromAddress(buf->address())->InNewSpace()); | |
| 21 CHECK(i::Page::FromAddress(buf->address())->local_tracker()->IsTracked(buf)); | |
| 22 } | |
| 23 | |
| 24 void VerifyUntracked(i::JSArrayBuffer* buf) { | |
| 25 CHECK(!i::Page::FromAddress(buf->address())->local_tracker()->IsTracked(buf)); | |
| 26 } | |
| 27 | |
| 28 } // namespace | |
| 29 | |
| 30 namespace v8 { | |
| 31 namespace internal { | |
| 32 | |
| 33 // The following tests make sure that JSArrayBuffer tracking works expected when | |
| 34 // moving the objects through various spaces during GC phases. | |
| 35 | |
| 36 TEST(ArrayBuffer_OnlyMC) { | |
| 37 CcTest::InitializeVM(); | |
| 38 LocalContext env; | |
| 39 v8::Isolate* isolate = env->GetIsolate(); | |
| 40 Heap* heap = reinterpret_cast<Isolate*>(isolate)->heap(); | |
| 41 | |
| 42 JSArrayBuffer* raw_ab = nullptr; | |
| 43 { | |
| 44 v8::HandleScope handle_scope(isolate); | |
| 45 Local<v8::ArrayBuffer> ab = v8::ArrayBuffer::New(isolate, 100); | |
| 46 Handle<JSArrayBuffer> buf = v8::Utils::OpenHandle(*ab); | |
| 47 VerifyTrackedInNewSpace(*buf); | |
| 48 heap::GcAndSweep(heap, OLD_SPACE); | |
| 49 VerifyTrackedInNewSpace(*buf); | |
| 50 heap::GcAndSweep(heap, OLD_SPACE); | |
| 51 VerifyTrackedInOldSpace(*buf); | |
| 52 raw_ab = *buf; | |
| 53 } | |
| 54 // 2 GCs are needed because we promote to old space as live, meaining that | |
| 55 // we will survive one GC. | |
| 56 heap::GcAndSweep(heap, OLD_SPACE); | |
| 57 heap::GcAndSweep(heap, OLD_SPACE); | |
| 58 VerifyUntracked(raw_ab); | |
| 59 } | |
| 60 | |
| 61 TEST(ArrayBuffer_OnlyScavenge) { | |
| 62 CcTest::InitializeVM(); | |
| 63 LocalContext env; | |
| 64 v8::Isolate* isolate = env->GetIsolate(); | |
| 65 Heap* heap = reinterpret_cast<Isolate*>(isolate)->heap(); | |
| 66 | |
| 67 JSArrayBuffer* raw_ab = nullptr; | |
| 68 { | |
| 69 v8::HandleScope handle_scope(isolate); | |
| 70 Local<v8::ArrayBuffer> ab = v8::ArrayBuffer::New(isolate, 100); | |
| 71 Handle<JSArrayBuffer> buf = v8::Utils::OpenHandle(*ab); | |
| 72 VerifyTrackedInNewSpace(*buf); | |
| 73 heap::GcAndSweep(heap, NEW_SPACE); | |
| 74 VerifyTrackedInNewSpace(*buf); | |
| 75 heap::GcAndSweep(heap, NEW_SPACE); | |
| 76 VerifyTrackedInOldSpace(*buf); | |
| 77 heap::GcAndSweep(heap, NEW_SPACE); | |
| 78 VerifyTrackedInOldSpace(*buf); | |
| 79 raw_ab = *buf; | |
| 80 } | |
| 81 // 2 GCs are needed because we promote to old space as live, meaning that | |
| 82 // we will survive one GC. | |
| 83 heap::GcAndSweep(heap, OLD_SPACE); | |
| 84 heap::GcAndSweep(heap, OLD_SPACE); | |
| 85 VerifyUntracked(raw_ab); | |
| 86 } | |
| 87 | |
| 88 TEST(ArrayBuffer_ScavengeAndMC) { | |
| 89 CcTest::InitializeVM(); | |
| 90 LocalContext env; | |
| 91 v8::Isolate* isolate = env->GetIsolate(); | |
| 92 Heap* heap = reinterpret_cast<Isolate*>(isolate)->heap(); | |
| 93 | |
| 94 JSArrayBuffer* raw_ab = nullptr; | |
| 95 { | |
| 96 v8::HandleScope handle_scope(isolate); | |
| 97 Local<v8::ArrayBuffer> ab = v8::ArrayBuffer::New(isolate, 100); | |
| 98 Handle<JSArrayBuffer> buf = v8::Utils::OpenHandle(*ab); | |
| 99 VerifyTrackedInNewSpace(*buf); | |
| 100 heap::GcAndSweep(heap, NEW_SPACE); | |
| 101 VerifyTrackedInNewSpace(*buf); | |
| 102 heap::GcAndSweep(heap, NEW_SPACE); | |
| 103 VerifyTrackedInOldSpace(*buf); | |
| 104 heap::GcAndSweep(heap, OLD_SPACE); | |
| 105 VerifyTrackedInOldSpace(*buf); | |
| 106 heap::GcAndSweep(heap, NEW_SPACE); | |
| 107 VerifyTrackedInOldSpace(*buf); | |
| 108 raw_ab = *buf; | |
| 109 } | |
| 110 // 2 GCs are needed because we promote to old space as live, meaning that | |
| 111 // we will survive one GC. | |
| 112 heap::GcAndSweep(heap, OLD_SPACE); | |
| 113 heap::GcAndSweep(heap, OLD_SPACE); | |
| 114 VerifyUntracked(raw_ab); | |
| 115 } | |
| 116 | |
| 117 TEST(ArrayBuffer_Compaction) { | |
| 118 FLAG_manual_evacuation_candidates_selection = true; | |
| 119 CcTest::InitializeVM(); | |
| 120 LocalContext env; | |
| 121 v8::Isolate* isolate = env->GetIsolate(); | |
| 122 Heap* heap = reinterpret_cast<Isolate*>(isolate)->heap(); | |
| 123 heap::AbandonCurrentlyFreeMemory(heap->old_space()); | |
| 124 | |
| 125 v8::HandleScope handle_scope(isolate); | |
| 126 Local<v8::ArrayBuffer> ab1 = v8::ArrayBuffer::New(isolate, 100); | |
| 127 Handle<JSArrayBuffer> buf1 = v8::Utils::OpenHandle(*ab1); | |
| 128 VerifyTrackedInNewSpace(*buf1); | |
| 129 heap::GcAndSweep(heap, NEW_SPACE); | |
| 130 heap::GcAndSweep(heap, NEW_SPACE); | |
| 131 | |
| 132 Page* page_before_gc = Page::FromAddress(buf1->address()); | |
| 133 page_before_gc->SetFlag(MemoryChunk::FORCE_EVACUATION_CANDIDATE_FOR_TESTING); | |
| 134 VerifyTrackedInOldSpace(*buf1); | |
| 135 | |
| 136 heap->CollectAllGarbage(); | |
| 137 | |
| 138 Page* page_after_gc = Page::FromAddress(buf1->address()); | |
| 139 VerifyTrackedInOldSpace(*buf1); | |
| 140 | |
| 141 CHECK_NE(page_before_gc, page_after_gc); | |
| 142 } | |
| 143 | |
| 144 } // namespace internal | |
| 145 } // namespace v8 | |
| OLD | NEW |