OLD | NEW |
---|---|
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/heap/mark-compact.h" | 5 #include "src/heap/mark-compact.h" |
6 | 6 |
7 #include "src/base/atomicops.h" | 7 #include "src/base/atomicops.h" |
8 #include "src/base/bits.h" | 8 #include "src/base/bits.h" |
9 #include "src/base/sys-info.h" | 9 #include "src/base/sys-info.h" |
10 #include "src/code-stubs.h" | 10 #include "src/code-stubs.h" |
(...skipping 1879 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1890 if (!code->CanDeoptAt(it.frame()->pc())) { | 1890 if (!code->CanDeoptAt(it.frame()->pc())) { |
1891 Code::BodyDescriptor::IterateBody(code, visitor); | 1891 Code::BodyDescriptor::IterateBody(code, visitor); |
1892 } | 1892 } |
1893 ProcessMarkingDeque(); | 1893 ProcessMarkingDeque(); |
1894 return; | 1894 return; |
1895 } | 1895 } |
1896 } | 1896 } |
1897 } | 1897 } |
1898 | 1898 |
1899 | 1899 |
1900 bool ShouldRetainMap(Map* map, int age) { | |
1901 if (age == 0) { | |
1902 // The map has aged. Do not retain this map. | |
1903 return false; | |
1904 } | |
1905 Object* constructor = map->GetConstructor(); | |
1906 if (!constructor->IsHeapObject() || | |
1907 Marking::IsWhite(Marking::MarkBitFrom(HeapObject::cast(constructor)))) { | |
1908 // The constructor is dead, no new objects with this map can | |
1909 // be created. Do not retain this map. | |
1910 return false; | |
1911 } | |
1912 return true; | |
1913 } | |
1914 | |
1915 | |
1916 void MarkCompactCollector::RetainMaps() { | |
1917 // Do not retain dead maps if flag disables it or there is | |
ulan
2015/11/30 09:32:24
This function moved to heap and incremental-markin
| |
1918 // - memory pressure (reduce_memory_footprint_), | |
1919 // - GC is requested by tests or dev-tools (abort_incremental_marking_). | |
1920 bool map_retaining_is_disabled = heap()->ShouldReduceMemory() || | |
1921 heap()->ShouldAbortIncrementalMarking() || | |
1922 FLAG_retain_maps_for_n_gc == 0; | |
1923 | |
1924 ArrayList* retained_maps = heap()->retained_maps(); | |
1925 int length = retained_maps->Length(); | |
1926 int new_length = 0; | |
1927 // The number_of_disposed_maps separates maps in the retained_maps | |
1928 // array that were created before and after context disposal. | |
1929 // We do not age and retain disposed maps to avoid memory leaks. | |
1930 int number_of_disposed_maps = heap()->number_of_disposed_maps_; | |
1931 int new_number_of_disposed_maps = 0; | |
1932 // This loop compacts the array by removing cleared weak cells, | |
1933 // ages and retains dead maps. | |
1934 for (int i = 0; i < length; i += 2) { | |
1935 DCHECK(retained_maps->Get(i)->IsWeakCell()); | |
1936 WeakCell* cell = WeakCell::cast(retained_maps->Get(i)); | |
1937 if (cell->cleared()) continue; | |
1938 int age = Smi::cast(retained_maps->Get(i + 1))->value(); | |
1939 int new_age; | |
1940 Map* map = Map::cast(cell->value()); | |
1941 MarkBit map_mark = Marking::MarkBitFrom(map); | |
1942 if (i >= number_of_disposed_maps && !map_retaining_is_disabled && | |
1943 Marking::IsWhite(map_mark)) { | |
1944 if (ShouldRetainMap(map, age)) { | |
1945 MarkObject(map, map_mark); | |
1946 } | |
1947 Object* prototype = map->prototype(); | |
1948 if (age > 0 && prototype->IsHeapObject() && | |
1949 Marking::IsWhite(Marking::MarkBitFrom(HeapObject::cast(prototype)))) { | |
1950 // The prototype is not marked, age the map. | |
1951 new_age = age - 1; | |
1952 } else { | |
1953 // The prototype and the constructor are marked, this map keeps only | |
1954 // transition tree alive, not JSObjects. Do not age the map. | |
1955 new_age = age; | |
1956 } | |
1957 } else { | |
1958 new_age = FLAG_retain_maps_for_n_gc; | |
1959 } | |
1960 // Compact the array and update the age. | |
1961 if (i != new_length) { | |
1962 retained_maps->Set(new_length, cell); | |
1963 Object** slot = retained_maps->Slot(new_length); | |
1964 RecordSlot(retained_maps, slot, cell); | |
1965 retained_maps->Set(new_length + 1, Smi::FromInt(new_age)); | |
1966 } else if (new_age != age) { | |
1967 retained_maps->Set(new_length + 1, Smi::FromInt(new_age)); | |
1968 } | |
1969 if (i < number_of_disposed_maps) { | |
1970 new_number_of_disposed_maps++; | |
1971 } | |
1972 new_length += 2; | |
1973 } | |
1974 heap()->number_of_disposed_maps_ = new_number_of_disposed_maps; | |
1975 Object* undefined = heap()->undefined_value(); | |
1976 for (int i = new_length; i < length; i++) { | |
1977 retained_maps->Clear(i, undefined); | |
1978 } | |
1979 if (new_length != length) retained_maps->SetLength(new_length); | |
1980 ProcessMarkingDeque(); | |
1981 } | |
1982 | |
1983 | |
1984 DependentCode* MarkCompactCollector::DependentCodeListFromNonLiveMaps() { | 1900 DependentCode* MarkCompactCollector::DependentCodeListFromNonLiveMaps() { |
1985 GCTracer::Scope gc_scope(heap()->tracer(), | 1901 GCTracer::Scope gc_scope(heap()->tracer(), |
1986 GCTracer::Scope::MC_EXTRACT_DEPENDENT_CODE); | 1902 GCTracer::Scope::MC_EXTRACT_DEPENDENT_CODE); |
1987 ArrayList* retained_maps = heap()->retained_maps(); | 1903 ArrayList* retained_maps = heap()->retained_maps(); |
1988 int length = retained_maps->Length(); | 1904 int length = retained_maps->Length(); |
1989 DependentCode* head = DependentCode::cast(heap()->empty_fixed_array()); | 1905 DependentCode* head = DependentCode::cast(heap()->empty_fixed_array()); |
1990 for (int i = 0; i < length; i += 2) { | 1906 for (int i = 0; i < length; i += 2) { |
1991 DCHECK(retained_maps->Get(i)->IsWeakCell()); | 1907 DCHECK(retained_maps->Get(i)->IsWeakCell()); |
1992 WeakCell* cell = WeakCell::cast(retained_maps->Get(i)); | 1908 WeakCell* cell = WeakCell::cast(retained_maps->Get(i)); |
1993 DCHECK(!cell->cleared()); | 1909 if (cell->cleared()) continue; |
1994 Map* map = Map::cast(cell->value()); | 1910 Map* map = Map::cast(cell->value()); |
1995 MarkBit map_mark = Marking::MarkBitFrom(map); | 1911 MarkBit map_mark = Marking::MarkBitFrom(map); |
1996 if (Marking::IsWhite(map_mark)) { | 1912 if (Marking::IsWhite(map_mark)) { |
1997 DependentCode* candidate = map->dependent_code(); | 1913 DependentCode* candidate = map->dependent_code(); |
1998 // We rely on the fact that the weak code group comes first. | 1914 // We rely on the fact that the weak code group comes first. |
1999 STATIC_ASSERT(DependentCode::kWeakCodeGroup == 0); | 1915 STATIC_ASSERT(DependentCode::kWeakCodeGroup == 0); |
2000 if (candidate->length() > 0 && | 1916 if (candidate->length() > 0 && |
2001 candidate->group() == DependentCode::kWeakCodeGroup) { | 1917 candidate->group() == DependentCode::kWeakCodeGroup) { |
2002 candidate->set_next_link(head); | 1918 candidate->set_next_link(head); |
2003 head = candidate; | 1919 head = candidate; |
(...skipping 130 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
2134 { | 2050 { |
2135 GCTracer::Scope gc_scope(heap()->tracer(), GCTracer::Scope::MC_MARK_ROOT); | 2051 GCTracer::Scope gc_scope(heap()->tracer(), GCTracer::Scope::MC_MARK_ROOT); |
2136 MarkRoots(&root_visitor); | 2052 MarkRoots(&root_visitor); |
2137 } | 2053 } |
2138 | 2054 |
2139 { | 2055 { |
2140 GCTracer::Scope gc_scope(heap()->tracer(), GCTracer::Scope::MC_MARK_TOPOPT); | 2056 GCTracer::Scope gc_scope(heap()->tracer(), GCTracer::Scope::MC_MARK_TOPOPT); |
2141 ProcessTopOptimizedFrame(&root_visitor); | 2057 ProcessTopOptimizedFrame(&root_visitor); |
2142 } | 2058 } |
2143 | 2059 |
2144 // Retaining dying maps should happen before or during ephemeral marking | |
2145 // because a map could keep the key of an ephemeron alive. Note that map | |
2146 // aging is imprecise: maps that are kept alive only by ephemerons will age. | |
2147 { | |
2148 GCTracer::Scope gc_scope(heap()->tracer(), | |
2149 GCTracer::Scope::MC_MARK_RETAIN_MAPS); | |
Hannes Payer (out of office)
2015/12/01 10:00:52
Remove this counter from the tracer, it is not use
| |
2150 RetainMaps(); | |
2151 } | |
2152 | |
2153 { | 2060 { |
2154 GCTracer::Scope gc_scope(heap()->tracer(), | 2061 GCTracer::Scope gc_scope(heap()->tracer(), |
2155 GCTracer::Scope::MC_MARK_WEAK_CLOSURE); | 2062 GCTracer::Scope::MC_MARK_WEAK_CLOSURE); |
2156 | 2063 |
2157 // The objects reachable from the roots are marked, yet unreachable | 2064 // The objects reachable from the roots are marked, yet unreachable |
2158 // objects are unmarked. Mark objects reachable due to host | 2065 // objects are unmarked. Mark objects reachable due to host |
2159 // application specific logic or through Harmony weak maps. | 2066 // application specific logic or through Harmony weak maps. |
2160 ProcessEphemeralMarking(&root_visitor, false); | 2067 ProcessEphemeralMarking(&root_visitor, false); |
2161 | 2068 |
2162 // The objects reachable from the roots, weak maps or object groups | 2069 // The objects reachable from the roots, weak maps or object groups |
(...skipping 2005 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
4168 MarkBit mark_bit = Marking::MarkBitFrom(host); | 4075 MarkBit mark_bit = Marking::MarkBitFrom(host); |
4169 if (Marking::IsBlack(mark_bit)) { | 4076 if (Marking::IsBlack(mark_bit)) { |
4170 RelocInfo rinfo(isolate(), pc, RelocInfo::CODE_TARGET, 0, host); | 4077 RelocInfo rinfo(isolate(), pc, RelocInfo::CODE_TARGET, 0, host); |
4171 RecordRelocSlot(&rinfo, target); | 4078 RecordRelocSlot(&rinfo, target); |
4172 } | 4079 } |
4173 } | 4080 } |
4174 } | 4081 } |
4175 | 4082 |
4176 } // namespace internal | 4083 } // namespace internal |
4177 } // namespace v8 | 4084 } // namespace v8 |
OLD | NEW |