%PDF- %PDF-
Direktori : /home/vacivi36/vittasync.vacivitta.com.br/vittasync/node/deps/v8/tools/gcmole/ |
Current File : /home/vacivi36/vittasync.vacivitta.com.br/vittasync/node/deps/v8/tools/gcmole/gcmole-test.cc |
// Copyright 2019 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 "src/execution/isolate.h" #include "src/handles/handles-inl.h" #include "src/handles/handles.h" #include "src/heap/local-heap.h" #include "src/objects/foreign-inl.h" #include "src/objects/managed.h" #include "src/objects/maybe-object.h" #include "src/objects/object-macros.h" namespace v8 { namespace internal { // ------- Test simple argument evaluation order problems --------- void Safepoint() { LocalHeap::Current()->Safepoint(); } Handle<Object> CauseGC(Handle<Object> obj, Isolate* isolate) { isolate->heap()->CollectGarbage(OLD_SPACE, GarbageCollectionReason::kTesting); return obj; } Tagged<Object> CauseGCRaw(Tagged<Object> obj, Isolate* isolate) { isolate->heap()->CollectGarbage(OLD_SPACE, GarbageCollectionReason::kTesting); return obj; } Tagged<Managed<int>> CauseGCManaged(int i, Isolate* isolate) { isolate->heap()->CollectGarbage(OLD_SPACE, GarbageCollectionReason::kTesting); return Managed<int>::cast(Smi::FromInt(i)); } void TwoArgumentsFunction(Tagged<Object> a, Tagged<Object> b) { Print(a); Print(b); } void TestTwoArguments(Isolate* isolate) { Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); Handle<JSObject> obj2 = isolate->factory()->NewJSObjectWithNullProto(); // Should cause warning. TwoArgumentsFunction(*CauseGC(obj1, isolate), *CauseGC(obj2, isolate)); } void TwoSizeTArgumentsFunction(size_t a, size_t b) { USE(a); USE(b); } void TestTwoSizeTArguments(Isolate* isolate) { Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); Handle<JSObject> obj2 = isolate->factory()->NewJSObjectWithNullProto(); // Should cause warning. TwoSizeTArgumentsFunction(sizeof(*CauseGC(obj1, isolate)), sizeof(*CauseGC(obj2, isolate))); } // --------- Test problFems with method arguments ---------- class SomeObject : public HeapObject { public: void Method(Tagged<Object> a) { Print(a); } DECL_CAST(SomeObject) OBJECT_CONSTRUCTORS(SomeObject, HeapObject); }; void TestMethodCall(Isolate* isolate) { Tagged<SomeObject> obj; Handle<SomeObject> so = handle(obj, isolate); Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); // Should cause warning. so->Method(*CauseGC(obj1, isolate)); // Should cause warning. so->Method(CauseGCRaw(*obj1, isolate)); } void TestOperatorCall(Isolate* isolate) { Tagged<SomeObject> obj; Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); // Should not cause warning. obj = Tagged<SomeObject>::unchecked_cast(*CauseGC(obj1, isolate)); } // --------- Test for templated sub-classes of Object ---------- void TestFollowingTemplates(Isolate* isolate) { // Should cause warning. CauseGCManaged(42, isolate); } // --------- Test for correctly resolving virtual methods ---------- class BaseObject { public: virtual Handle<Object> VirtualCauseGC(Handle<Object> obj, Isolate* isolate) { return obj; } }; class DerivedObject : public BaseObject { public: Handle<Object> VirtualCauseGC(Handle<Object> obj, Isolate* isolate) override { isolate->heap()->CollectGarbage(OLD_SPACE, GarbageCollectionReason::kTesting); return obj; } }; void TestFollowingVirtualFunctions(Isolate* isolate) { DerivedObject derived; BaseObject* base = &derived; Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); Tagged<SomeObject> so; Handle<SomeObject> so_handle = handle(so, isolate); // Should cause warning. so_handle->Method(*derived.VirtualCauseGC(obj1, isolate)); // Should cause warning. so_handle->Method(*base->VirtualCauseGC(obj1, isolate)); } // --------- Test for correctly resolving static methods ---------- class SomeClass { public: static Handle<Object> StaticCauseGC(Handle<Object> obj, Isolate* isolate) { isolate->heap()->CollectGarbage(OLD_SPACE, GarbageCollectionReason::kTesting); return obj; } }; void TestFollowingStaticFunctions(Isolate* isolate) { Tagged<SomeObject> so; Handle<SomeObject> so_handle = handle(so, isolate); Handle<JSObject> obj1 = isolate->factory()->NewJSObjectWithNullProto(); // Should cause warning. so_handle->Method(*SomeClass::StaticCauseGC(obj1, isolate)); } // --------- Test basic dead variable analysis ---------- void TestDeadVarAnalysis(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestDeadVarBecauseOfSafepointAnalysis(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); Safepoint(); // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysis(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // Note: having DisableGCMole with the same function as CauseGC // normally doesn't make sense, but we want to test whether the guards // are recognized by GCMole. DisableGCMole no_gc_mole; CauseGCRaw(raw_obj, isolate); // Shouldn't cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysis2(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // Note: having DisallowGarbageCollection with the same function as CauseGC // normally doesn't make sense, but we want to test whether the guards // are recognized by GCMole. DisallowGarbageCollection no_gc; CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestGuardedAgainstSafepointDeadVarAnalysis(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // Note: having DisableGCMole with the same function as CauseGC // normally doesn't make sense, but we want to test whether the guards // are recognized by GCMole. DisableGCMole no_gc_mole; Safepoint(); // Shouldn't cause warning. Print(raw_obj); } void TestGuardedAgainstSafepointDeadVarAnalysis2(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // Note: having DisallowGarbageCollection with the same function as CauseGC // normally doesn't make sense, but we want to test whether the guards // are recognized by GCMole. DisallowGarbageCollection no_gc; Safepoint(); // Should cause warning. Print(raw_obj); } void TestGuardedAgainstSafepointDeadVarAnalysis3(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // Note: having DisallowGarbageCollection with the same function as CauseGC // normally doesn't make sense, but we want to test whether the guards // are recognized by GCMole. DisallowGarbageCollection no_gc; Safepoint(); // Should cause warning. Print(raw_obj); { DisableGCMole no_gc_mole; // Shouldn't cause warning. Print(raw_obj); } // Should cause warning. Print(raw_obj); } void TestOnlyHeapGuardedDeadVarAnalysisInCompound(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // {DisallowHeapAccess} has a {DisallowHeapAllocation}, but no // {DisallowSafepoints}, so it could see objects move due to safepoints. DisallowHeapAccess no_gc; CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestOnlyHeapGuardedDeadVarAnalysisInCompound2(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); // {DisallowHeapAccess} has a {DisallowHeapAllocation}, but no // {DisallowSafepoints}, so it could see objects move due to safepoints. DisallowHeapAccess no_gc; CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); DisableGCMole no_gc_mole; // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisNested(Tagged<JSObject> raw_obj, Isolate* isolate) { CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisCaller(Isolate* isolate) { DisableGCMole no_gc_mole; Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); TestGuardedDeadVarAnalysisNested(raw_obj, isolate); // Shouldn't cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisCaller2(Isolate* isolate) { DisallowGarbageCollection no_gc; Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); TestGuardedDeadVarAnalysisNested(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisCaller3(Isolate* isolate) { DisallowHeapAccess no_gc; Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); TestGuardedDeadVarAnalysisNested(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisCaller4(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); TestGuardedDeadVarAnalysisNested(raw_obj, isolate); // Should cause warning. Print(raw_obj); } Tagged<JSObject> GuardedAllocation(Isolate* isolate) { DisallowGarbageCollection no_gc; return *isolate->factory()->NewJSObjectWithNullProto(); } Tagged<JSObject> GuardedAllocation2(Isolate* isolate) { DisableGCMole no_gc_mole; return *isolate->factory()->NewJSObjectWithNullProto(); } void TestNestedDeadVarAnalysis(Isolate* isolate) { Tagged<JSObject> raw_obj = GuardedAllocation(isolate); CauseGCRaw(raw_obj, isolate); // Should cause warning. Print(raw_obj); } void TestNestedDeadVarAnalysis2(Isolate* isolate) { DisableGCMole no_gc_mole; Tagged<JSObject> raw_obj = GuardedAllocation(isolate); CauseGCRaw(raw_obj, isolate); // Shouldn't cause warning. Print(raw_obj); } // Test that putting a guard in the middle of the function doesn't // mistakenly cover the whole scope of the raw variable. void TestGuardedDeadVarAnalysisMidFunction(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); CauseGCRaw(raw_obj, isolate); // Guarding the rest of the function from triggering a GC. DisallowGarbageCollection no_gc; // Should cause warning. Print(raw_obj); } // Test that putting a guard in the middle of the function doesn't // mistakenly cover the whole scope of the raw variable. void TestGuardedDeadVarAnalysisMidFunction2(Isolate* isolate) { Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); CauseGCRaw(raw_obj, isolate); // Guarding the rest of the function from triggering a GC. DisableGCMole no_gc_mole; // Should cause warning. Print(raw_obj); } void TestGuardedDeadVarAnalysisMultipleSafepoints(Isolate* isolate) { // TODO(https://crbug.com/v8/13536): The analysis points to this safepoint, // while it should point to the one below. Safepoint(); Tagged<JSObject> raw_obj = *isolate->factory()->NewJSObjectWithNullProto(); DisallowGarbageCollection no_gc; Safepoint(); Print(raw_obj); } } // namespace internal } // namespace v8