Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(254)

Side by Side Diff: Source/platform/heap/Heap.cpp

Issue 850063002: Do not fire timers for finalizing objects. (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@master
Patch Set: Mark pages as lazily swept Created 5 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « Source/platform/heap/Heap.h ('k') | Source/platform/heap/ThreadState.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright (C) 2013 Google Inc. All rights reserved. 2 * Copyright (C) 2013 Google Inc. All rights reserved.
3 * 3 *
4 * Redistribution and use in source and binary forms, with or without 4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are 5 * modification, are permitted provided that the following conditions are
6 * met: 6 * met:
7 * 7 *
8 * * Redistributions of source code must retain the above copyright 8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer. 9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above 10 * * Redistributions in binary form must reproduce the above
(...skipping 525 matching lines...) Expand 10 before | Expand all | Expand 10 after
536 { 536 {
537 HeapObjectHeader* header = heapObjectHeader(); 537 HeapObjectHeader* header = heapObjectHeader();
538 if (header->isMarked()) 538 if (header->isMarked())
539 header->unmark(); 539 header->unmark();
540 else 540 else
541 header->markDead(); 541 header->markDead();
542 } 542 }
543 543
544 void LargeObject::removeFromHeap(ThreadHeap* heap) 544 void LargeObject::removeFromHeap(ThreadHeap* heap)
545 { 545 {
546 m_sweepStatus = 0;
546 heap->freeLargeObject(this); 547 heap->freeLargeObject(this);
547 } 548 }
548 549
549 ThreadHeap::ThreadHeap(ThreadState* state, int index) 550 ThreadHeap::ThreadHeap(ThreadState* state, int index)
550 : m_currentAllocationPoint(nullptr) 551 : m_currentAllocationPoint(nullptr)
551 , m_remainingAllocationSize(0) 552 , m_remainingAllocationSize(0)
552 , m_lastRemainingAllocationSize(0) 553 , m_lastRemainingAllocationSize(0)
553 , m_firstPage(nullptr) 554 , m_firstPage(nullptr)
554 , m_firstLargeObject(nullptr) 555 , m_firstLargeObject(nullptr)
555 , m_firstUnsweptPage(nullptr) 556 , m_firstUnsweptPage(nullptr)
(...skipping 184 matching lines...) Expand 10 before | Expand all | Expand 10 after
740 HeapPage* page = m_firstUnsweptPage; 741 HeapPage* page = m_firstUnsweptPage;
741 if (page->isEmpty()) { 742 if (page->isEmpty()) {
742 page->unlink(&m_firstUnsweptPage); 743 page->unlink(&m_firstUnsweptPage);
743 page->removeFromHeap(this); 744 page->removeFromHeap(this);
744 } else { 745 } else {
745 // Sweep a page and move the page from m_firstUnsweptPages to 746 // Sweep a page and move the page from m_firstUnsweptPages to
746 // m_firstPages. 747 // m_firstPages.
747 page->sweep(this); 748 page->sweep(this);
748 page->unlink(&m_firstUnsweptPage); 749 page->unlink(&m_firstUnsweptPage);
749 page->link(&m_firstPage); 750 page->link(&m_firstPage);
751 page->markAsLazilySwept();
750 752
751 result = allocateFromFreeList(allocationSize, gcInfoIndex); 753 result = allocateFromFreeList(allocationSize, gcInfoIndex);
752 if (result) { 754 if (result)
753 break; 755 break;
754 }
755 } 756 }
756 } 757 }
757 758
758 if (threadState()->isMainThread()) 759 if (threadState()->isMainThread())
759 ScriptForbiddenScope::exit(); 760 ScriptForbiddenScope::exit();
760 return result; 761 return result;
761 } 762 }
762 763
763 bool ThreadHeap::lazySweepLargeObjects(size_t allocationSize) 764 bool ThreadHeap::lazySweepLargeObjects(size_t allocationSize)
764 { 765 {
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
797 if (sweptSize >= allocationSize) { 798 if (sweptSize >= allocationSize) {
798 result = true; 799 result = true;
799 break; 800 break;
800 } 801 }
801 } else { 802 } else {
802 // Sweep a large object and move the large object from 803 // Sweep a large object and move the large object from
803 // m_firstUnsweptLargeObjects to m_firstLargeObjects. 804 // m_firstUnsweptLargeObjects to m_firstLargeObjects.
804 largeObject->sweep(this); 805 largeObject->sweep(this);
805 largeObject->unlink(&m_firstUnsweptLargeObject); 806 largeObject->unlink(&m_firstUnsweptLargeObject);
806 largeObject->link(&m_firstLargeObject); 807 largeObject->link(&m_firstLargeObject);
808 largeObject->markAsLazilySwept();
807 } 809 }
808 } 810 }
809 811
810 if (threadState()->isMainThread()) 812 if (threadState()->isMainThread())
811 ScriptForbiddenScope::exit(); 813 ScriptForbiddenScope::exit();
812 return result; 814 return result;
813 } 815 }
814 816
815 void ThreadHeap::completeSweep() 817 void ThreadHeap::completeSweep()
816 { 818 {
817 RELEASE_ASSERT(threadState()->isSweepingInProgress()); 819 RELEASE_ASSERT(threadState()->isSweepingInProgress());
818 ASSERT(threadState()->sweepForbidden()); 820 ASSERT(threadState()->sweepForbidden());
819 821
820 if (threadState()->isMainThread()) 822 if (threadState()->isMainThread())
821 ScriptForbiddenScope::enter(); 823 ScriptForbiddenScope::enter();
822 824
823 // Sweep normal pages. 825 // Sweep normal pages.
824 while (m_firstUnsweptPage) { 826 while (m_firstUnsweptPage) {
825 HeapPage* page = m_firstUnsweptPage; 827 HeapPage* page = m_firstUnsweptPage;
826 if (page->isEmpty()) { 828 if (page->isEmpty()) {
827 page->unlink(&m_firstUnsweptPage); 829 page->unlink(&m_firstUnsweptPage);
828 page->removeFromHeap(this); 830 page->removeFromHeap(this);
829 } else { 831 } else {
830 // Sweep a page and move the page from m_firstUnsweptPages to 832 // Sweep a page and move the page from m_firstUnsweptPages to
831 // m_firstPages. 833 // m_firstPages.
832 page->sweep(this); 834 page->sweep(this);
833 page->unlink(&m_firstUnsweptPage); 835 page->unlink(&m_firstUnsweptPage);
834 page->link(&m_firstPage); 836 page->link(&m_firstPage);
837 page->markAsLazilySwept();
835 } 838 }
836 } 839 }
837 840
838 // Sweep large objects. 841 // Sweep large objects.
839 while (m_firstUnsweptLargeObject) { 842 while (m_firstUnsweptLargeObject) {
840 LargeObject* largeObject = m_firstUnsweptLargeObject; 843 LargeObject* largeObject = m_firstUnsweptLargeObject;
841 if (largeObject->isEmpty()) { 844 if (largeObject->isEmpty()) {
842 largeObject->unlink(&m_firstUnsweptLargeObject); 845 largeObject->unlink(&m_firstUnsweptLargeObject);
843 largeObject->removeFromHeap(this); 846 largeObject->removeFromHeap(this);
844 } else { 847 } else {
845 // Sweep a large object and move the large object from 848 // Sweep a large object and move the large object from
846 // m_firstUnsweptLargeObjects to m_firstLargeObjects. 849 // m_firstUnsweptLargeObjects to m_firstLargeObjects.
847 largeObject->sweep(this); 850 largeObject->sweep(this);
848 largeObject->unlink(&m_firstUnsweptLargeObject); 851 largeObject->unlink(&m_firstUnsweptLargeObject);
849 largeObject->link(&m_firstLargeObject); 852 largeObject->link(&m_firstLargeObject);
853 largeObject->markAsLazilySwept();
850 } 854 }
851 } 855 }
852 856
853 if (threadState()->isMainThread()) 857 if (threadState()->isMainThread())
854 ScriptForbiddenScope::exit(); 858 ScriptForbiddenScope::exit();
855 } 859 }
856 860
857 #if ENABLE(ASSERT) 861 #if ENABLE(ASSERT)
858 static bool isLargeObjectAligned(LargeObject* largeObject, Address address) 862 static bool isLargeObjectAligned(LargeObject* largeObject, Address address)
859 { 863 {
(...skipping 371 matching lines...) Expand 10 before | Expand all | Expand 10 after
1231 } 1235 }
1232 1236
1233 BaseHeapPage::BaseHeapPage(PageMemory* storage, ThreadState* state) 1237 BaseHeapPage::BaseHeapPage(PageMemory* storage, ThreadState* state)
1234 : m_storage(storage) 1238 : m_storage(storage)
1235 , m_threadState(state) 1239 , m_threadState(state)
1236 , m_terminating(false) 1240 , m_terminating(false)
1237 { 1241 {
1238 ASSERT(isPageHeaderAddress(reinterpret_cast<Address>(this))); 1242 ASSERT(isPageHeaderAddress(reinterpret_cast<Address>(this)));
1239 } 1243 }
1240 1244
1245 void BaseHeapPage::markAsLazilySwept()
1246 {
1247 // To signal completion of a lazy sweep, update the
1248 // page's sweep status to match that of the ThreadState's
1249 // flip-flop sweep token.
1250 bool currentToken = threadState()->lazySweepToken();
1251 m_sweepStatus = currentToken ? 2 : 1;
1252 }
1253
1254 bool BaseHeapPage::hasBeenLazilySwept() const
1255 {
1256 // Returns true if this page has been swept
1257 // by the ongoing lazy sweep operation.
1258 ASSERT(threadState()->isSweepingInProgress());
1259 ASSERT(m_sweepStatus <= 2);
1260 if (!m_sweepStatus)
1261 return false;
1262
1263 bool currentToken = threadState()->lazySweepToken();
1264 return currentToken == (m_sweepStatus == 2);
1265 }
1266
1241 void BaseHeapPage::markOrphaned() 1267 void BaseHeapPage::markOrphaned()
1242 { 1268 {
1243 m_threadState = nullptr; 1269 m_threadState = nullptr;
1244 m_terminating = false; 1270 m_terminating = false;
1245 // Since we zap the page payload for orphaned pages we need to mark it as 1271 // Since we zap the page payload for orphaned pages we need to mark it as
1246 // unused so a conservative pointer won't interpret the object headers. 1272 // unused so a conservative pointer won't interpret the object headers.
1247 storage()->markUnused(); 1273 storage()->markUnused();
1248 } 1274 }
1249 1275
1250 OrphanedPagePool::~OrphanedPagePool() 1276 OrphanedPagePool::~OrphanedPagePool()
(...skipping 360 matching lines...) Expand 10 before | Expand all | Expand 10 after
1611 if (header->isMarked()) 1637 if (header->isMarked())
1612 header->unmark(); 1638 header->unmark();
1613 else 1639 else
1614 header->markDead(); 1640 header->markDead();
1615 headerAddress += header->size(); 1641 headerAddress += header->size();
1616 } 1642 }
1617 } 1643 }
1618 1644
1619 void HeapPage::removeFromHeap(ThreadHeap* heap) 1645 void HeapPage::removeFromHeap(ThreadHeap* heap)
1620 { 1646 {
1647 m_sweepStatus = 0;
1621 heap->freePage(this); 1648 heap->freePage(this);
1622 } 1649 }
1623 1650
1624 void HeapPage::populateObjectStartBitMap() 1651 void HeapPage::populateObjectStartBitMap()
1625 { 1652 {
1626 memset(&m_objectStartBitMap, 0, objectStartBitMapSize); 1653 memset(&m_objectStartBitMap, 0, objectStartBitMapSize);
1627 Address start = payload(); 1654 Address start = payload();
1628 for (Address headerAddress = start; headerAddress < payloadEnd();) { 1655 for (Address headerAddress = start; headerAddress < payloadEnd();) {
1629 HeapObjectHeader* header = reinterpret_cast<HeapObjectHeader*>(headerAdd ress); 1656 HeapObjectHeader* header = reinterpret_cast<HeapObjectHeader*>(headerAdd ress);
1630 size_t objectOffset = headerAddress - start; 1657 size_t objectOffset = headerAddress - start;
(...skipping 488 matching lines...) Expand 10 before | Expand all | Expand 10 after
2119 2146
2120 #if !ENABLE(ASSERT) 2147 #if !ENABLE(ASSERT)
2121 s_heapDoesNotContainCache->addEntry(address); 2148 s_heapDoesNotContainCache->addEntry(address);
2122 #else 2149 #else
2123 if (!s_heapDoesNotContainCache->lookup(address)) 2150 if (!s_heapDoesNotContainCache->lookup(address))
2124 s_heapDoesNotContainCache->addEntry(address); 2151 s_heapDoesNotContainCache->addEntry(address);
2125 #endif 2152 #endif
2126 return nullptr; 2153 return nullptr;
2127 } 2154 }
2128 2155
2156 bool Heap::isFinalizedObjectAlive(const void* objectPointer)
haraken 2015/01/16 00:57:10 I like having this method in our heap infrastructu
2157 {
2158 #if ENABLE(OILPAN)
2159 if (!ThreadState::current()->isSweepingInProgress())
2160 return true;
2161
2162 BaseHeapPage* page = pageFromObject(objectPointer);
2163 if (page->hasBeenLazilySwept())
2164 return true;
2165
2166 return s_markingVisitor->isMarked(objectPointer);
2167 #else
2168 // FIXME: remove when incremental sweeping is always on
2169 // (cf. ThreadState::postGCProcessing()).
2170 return true;
2171 #endif
2172 }
2173
2129 #if ENABLE(GC_PROFILE_MARKING) 2174 #if ENABLE(GC_PROFILE_MARKING)
2130 const GCInfo* Heap::findGCInfo(Address address) 2175 const GCInfo* Heap::findGCInfo(Address address)
2131 { 2176 {
2132 return ThreadState::findGCInfoFromAllThreads(address); 2177 return ThreadState::findGCInfoFromAllThreads(address);
2133 } 2178 }
2134 #endif 2179 #endif
2135 2180
2136 #if ENABLE(GC_PROFILE_MARKING) 2181 #if ENABLE(GC_PROFILE_MARKING)
2137 void Heap::dumpPathToObjectOnNextGC(void* p) 2182 void Heap::dumpPathToObjectOnNextGC(void* p)
2138 { 2183 {
(...skipping 548 matching lines...) Expand 10 before | Expand all | Expand 10 after
2687 bool Heap::s_shutdownCalled = false; 2732 bool Heap::s_shutdownCalled = false;
2688 bool Heap::s_lastGCWasConservative = false; 2733 bool Heap::s_lastGCWasConservative = false;
2689 FreePagePool* Heap::s_freePagePool; 2734 FreePagePool* Heap::s_freePagePool;
2690 OrphanedPagePool* Heap::s_orphanedPagePool; 2735 OrphanedPagePool* Heap::s_orphanedPagePool;
2691 Heap::RegionTree* Heap::s_regionTree = nullptr; 2736 Heap::RegionTree* Heap::s_regionTree = nullptr;
2692 size_t Heap::s_allocatedObjectSize = 0; 2737 size_t Heap::s_allocatedObjectSize = 0;
2693 size_t Heap::s_allocatedSpace = 0; 2738 size_t Heap::s_allocatedSpace = 0;
2694 size_t Heap::s_markedObjectSize = 0; 2739 size_t Heap::s_markedObjectSize = 0;
2695 2740
2696 } // namespace blink 2741 } // namespace blink
OLDNEW
« no previous file with comments | « Source/platform/heap/Heap.h ('k') | Source/platform/heap/ThreadState.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698