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 #ifndef V8_HEAP_MARKING_H_
6 #define V8_HEAP_MARKING_H_
7 
8 #include "src/base/atomic-utils.h"
9 #include "src/utils.h"
10 
11 namespace v8 {
12 namespace internal {
13 
14 class MarkBit {
15  public:
16   typedef uint32_t CellType;
17   STATIC_ASSERT(sizeof(CellType) == sizeof(base::Atomic32));
18 
MarkBit(CellType * cell,CellType mask)19   inline MarkBit(CellType* cell, CellType mask) : cell_(cell), mask_(mask) {}
20 
21 #ifdef DEBUG
22   bool operator==(const MarkBit& other) {
23     return cell_ == other.cell_ && mask_ == other.mask_;
24   }
25 #endif
26 
27  private:
Next()28   inline MarkBit Next() {
29     CellType new_mask = mask_ << 1;
30     if (new_mask == 0) {
31       return MarkBit(cell_ + 1, 1);
32     } else {
33       return MarkBit(cell_, new_mask);
34     }
35   }
36 
37   // The function returns true if it succeeded to
38   // transition the bit from 0 to 1.
39   template <AccessMode mode = AccessMode::NON_ATOMIC>
40   inline bool Set();
41 
42   template <AccessMode mode = AccessMode::NON_ATOMIC>
43   inline bool Get();
44 
45   // The function returns true if it succeeded to
46   // transition the bit from 1 to 0.
47   template <AccessMode mode = AccessMode::NON_ATOMIC>
48   inline bool Clear();
49 
50   CellType* cell_;
51   CellType mask_;
52 
53   friend class IncrementalMarking;
54   friend class ConcurrentMarkingMarkbits;
55   friend class Marking;
56 };
57 
58 template <>
59 inline bool MarkBit::Set<AccessMode::NON_ATOMIC>() {
60   CellType old_value = *cell_;
61   *cell_ = old_value | mask_;
62   return (old_value & mask_) == 0;
63 }
64 
65 template <>
66 inline bool MarkBit::Set<AccessMode::ATOMIC>() {
67   return base::AsAtomic32::SetBits(cell_, mask_, mask_);
68 }
69 
70 template <>
71 inline bool MarkBit::Get<AccessMode::NON_ATOMIC>() {
72   return (*cell_ & mask_) != 0;
73 }
74 
75 template <>
76 inline bool MarkBit::Get<AccessMode::ATOMIC>() {
77   return (base::AsAtomic32::Acquire_Load(cell_) & mask_) != 0;
78 }
79 
80 template <>
81 inline bool MarkBit::Clear<AccessMode::NON_ATOMIC>() {
82   CellType old_value = *cell_;
83   *cell_ = old_value & ~mask_;
84   return (old_value & mask_) == mask_;
85 }
86 
87 template <>
88 inline bool MarkBit::Clear<AccessMode::ATOMIC>() {
89   return base::AsAtomic32::SetBits(cell_, 0u, mask_);
90 }
91 
92 // Bitmap is a sequence of cells each containing fixed number of bits.
93 class V8_EXPORT_PRIVATE Bitmap {
94  public:
95   static const uint32_t kBitsPerCell = 32;
96   static const uint32_t kBitsPerCellLog2 = 5;
97   static const uint32_t kBitIndexMask = kBitsPerCell - 1;
98   static const uint32_t kBytesPerCell = kBitsPerCell / kBitsPerByte;
99   static const uint32_t kBytesPerCellLog2 = kBitsPerCellLog2 - kBitsPerByteLog2;
100 
101   static const size_t kLength = (1 << kPageSizeBits) >> (kPointerSizeLog2);
102 
103   static const size_t kSize = (1 << kPageSizeBits) >>
104                               (kPointerSizeLog2 + kBitsPerByteLog2);
105 
CellsForLength(int length)106   static int CellsForLength(int length) {
107     return (length + kBitsPerCell - 1) >> kBitsPerCellLog2;
108   }
109 
CellsCount()110   int CellsCount() { return CellsForLength(kLength); }
111 
IndexToCell(uint32_t index)112   V8_INLINE static uint32_t IndexToCell(uint32_t index) {
113     return index >> kBitsPerCellLog2;
114   }
115 
IndexInCell(uint32_t index)116   V8_INLINE static uint32_t IndexInCell(uint32_t index) {
117     return index & kBitIndexMask;
118   }
119 
120   // Retrieves the cell containing the provided markbit index.
CellAlignIndex(uint32_t index)121   V8_INLINE static uint32_t CellAlignIndex(uint32_t index) {
122     return index & ~kBitIndexMask;
123   }
124 
IsCellAligned(uint32_t index)125   V8_INLINE static bool IsCellAligned(uint32_t index) {
126     return (index & kBitIndexMask) == 0;
127   }
128 
cells()129   V8_INLINE MarkBit::CellType* cells() {
130     return reinterpret_cast<MarkBit::CellType*>(this);
131   }
132 
FromAddress(Address addr)133   V8_INLINE static Bitmap* FromAddress(Address addr) {
134     return reinterpret_cast<Bitmap*>(addr);
135   }
136 
MarkBitFromIndex(uint32_t index)137   inline MarkBit MarkBitFromIndex(uint32_t index) {
138     MarkBit::CellType mask = 1u << IndexInCell(index);
139     MarkBit::CellType* cell = this->cells() + (index >> kBitsPerCellLog2);
140     return MarkBit(cell, mask);
141   }
142 
143   void Clear();
144 
145   void MarkAllBits();
146 
147   // Clears bits in the given cell. The mask specifies bits to clear: if a
148   // bit is set in the mask then the corresponding bit is cleared in the cell.
149   template <AccessMode mode = AccessMode::NON_ATOMIC>
150   void ClearBitsInCell(uint32_t cell_index, uint32_t mask);
151 
152   // Sets bits in the given cell. The mask specifies bits to set: if a
153   // bit is set in the mask then the corresponding bit is set in the cell.
154   template <AccessMode mode = AccessMode::NON_ATOMIC>
155   void SetBitsInCell(uint32_t cell_index, uint32_t mask);
156 
157   // Sets all bits in the range [start_index, end_index). The cells at the
158   // boundary of the range are updated with atomic compare and swap operation.
159   // The inner cells are updated with relaxed write.
160   void SetRange(uint32_t start_index, uint32_t end_index);
161 
162   // Clears all bits in the range [start_index, end_index). The cells at the
163   // boundary of the range are updated with atomic compare and swap operation.
164   // The inner cells are updated with relaxed write.
165   void ClearRange(uint32_t start_index, uint32_t end_index);
166 
167   // Returns true if all bits in the range [start_index, end_index) are set.
168   bool AllBitsSetInRange(uint32_t start_index, uint32_t end_index);
169 
170   // Returns true if all bits in the range [start_index, end_index) are cleared.
171   bool AllBitsClearInRange(uint32_t start_index, uint32_t end_index);
172 
173   void Print();
174 
175   bool IsClean();
176 };
177 
178 template <>
179 inline void Bitmap::SetBitsInCell<AccessMode::NON_ATOMIC>(uint32_t cell_index,
180                                                           uint32_t mask) {
181   cells()[cell_index] |= mask;
182 }
183 
184 template <>
185 inline void Bitmap::SetBitsInCell<AccessMode::ATOMIC>(uint32_t cell_index,
186                                                       uint32_t mask) {
187   base::AsAtomic32::SetBits(cells() + cell_index, mask, mask);
188 }
189 
190 template <>
191 inline void Bitmap::ClearBitsInCell<AccessMode::NON_ATOMIC>(uint32_t cell_index,
192                                                             uint32_t mask) {
193   cells()[cell_index] &= ~mask;
194 }
195 
196 template <>
197 inline void Bitmap::ClearBitsInCell<AccessMode::ATOMIC>(uint32_t cell_index,
198                                                         uint32_t mask) {
199   base::AsAtomic32::SetBits(cells() + cell_index, 0u, mask);
200 }
201 
202 class Marking : public AllStatic {
203  public:
204   // TODO(hpayer): The current mark bit operations use as default NON_ATOMIC
205   // mode for access. We should remove the default value or switch it with
206   // ATOMIC as soon we add concurrency.
207 
208   // Impossible markbits: 01
209   static const char* kImpossibleBitPattern;
210   template <AccessMode mode = AccessMode::NON_ATOMIC>
IsImpossible(MarkBit mark_bit)211   V8_INLINE static bool IsImpossible(MarkBit mark_bit) {
212     if (mode == AccessMode::NON_ATOMIC) {
213       return !mark_bit.Get<mode>() && mark_bit.Next().Get<mode>();
214     }
215     // If we are in concurrent mode we can only tell if an object has the
216     // impossible bit pattern if we read the first bit again after reading
217     // the first and the second bit. If the first bit is till zero and the
218     // second bit is one then the object has the impossible bit pattern.
219     bool is_impossible = !mark_bit.Get<mode>() && mark_bit.Next().Get<mode>();
220     if (is_impossible) {
221       return !mark_bit.Get<mode>();
222     }
223     return false;
224   }
225 
226   // Black markbits: 11
227   static const char* kBlackBitPattern;
228   template <AccessMode mode = AccessMode::NON_ATOMIC>
IsBlack(MarkBit mark_bit)229   V8_INLINE static bool IsBlack(MarkBit mark_bit) {
230     return mark_bit.Get<mode>() && mark_bit.Next().Get<mode>();
231   }
232 
233   // White markbits: 00 - this is required by the mark bit clearer.
234   static const char* kWhiteBitPattern;
235   template <AccessMode mode = AccessMode::NON_ATOMIC>
IsWhite(MarkBit mark_bit)236   V8_INLINE static bool IsWhite(MarkBit mark_bit) {
237     DCHECK(!IsImpossible<mode>(mark_bit));
238     return !mark_bit.Get<mode>();
239   }
240 
241   // Grey markbits: 10
242   static const char* kGreyBitPattern;
243   template <AccessMode mode = AccessMode::NON_ATOMIC>
IsGrey(MarkBit mark_bit)244   V8_INLINE static bool IsGrey(MarkBit mark_bit) {
245     return mark_bit.Get<mode>() && !mark_bit.Next().Get<mode>();
246   }
247 
248   // IsBlackOrGrey assumes that the first bit is set for black or grey
249   // objects.
250   template <AccessMode mode = AccessMode::NON_ATOMIC>
IsBlackOrGrey(MarkBit mark_bit)251   V8_INLINE static bool IsBlackOrGrey(MarkBit mark_bit) {
252     return mark_bit.Get<mode>();
253   }
254 
255   template <AccessMode mode = AccessMode::NON_ATOMIC>
MarkWhite(MarkBit markbit)256   V8_INLINE static void MarkWhite(MarkBit markbit) {
257     STATIC_ASSERT(mode == AccessMode::NON_ATOMIC);
258     markbit.Clear<mode>();
259     markbit.Next().Clear<mode>();
260   }
261 
262   // Warning: this method is not safe in general in concurrent scenarios.
263   // If you know that nobody else will change the bits on the given location
264   // then you may use it.
265   template <AccessMode mode = AccessMode::NON_ATOMIC>
MarkBlack(MarkBit markbit)266   V8_INLINE static void MarkBlack(MarkBit markbit) {
267     markbit.Set<mode>();
268     markbit.Next().Set<mode>();
269   }
270 
271   template <AccessMode mode = AccessMode::NON_ATOMIC>
WhiteToGrey(MarkBit markbit)272   V8_INLINE static bool WhiteToGrey(MarkBit markbit) {
273     return markbit.Set<mode>();
274   }
275 
276   template <AccessMode mode = AccessMode::NON_ATOMIC>
WhiteToBlack(MarkBit markbit)277   V8_INLINE static bool WhiteToBlack(MarkBit markbit) {
278     return markbit.Set<mode>() && markbit.Next().Set<mode>();
279   }
280 
281   template <AccessMode mode = AccessMode::NON_ATOMIC>
GreyToBlack(MarkBit markbit)282   V8_INLINE static bool GreyToBlack(MarkBit markbit) {
283     return markbit.Get<mode>() && markbit.Next().Set<mode>();
284   }
285 
286   enum ObjectColor {
287     BLACK_OBJECT,
288     WHITE_OBJECT,
289     GREY_OBJECT,
290     IMPOSSIBLE_COLOR
291   };
292 
ColorName(ObjectColor color)293   static const char* ColorName(ObjectColor color) {
294     switch (color) {
295       case BLACK_OBJECT:
296         return "black";
297       case WHITE_OBJECT:
298         return "white";
299       case GREY_OBJECT:
300         return "grey";
301       case IMPOSSIBLE_COLOR:
302         return "impossible";
303     }
304     return "error";
305   }
306 
Color(MarkBit mark_bit)307   static ObjectColor Color(MarkBit mark_bit) {
308     if (IsBlack(mark_bit)) return BLACK_OBJECT;
309     if (IsWhite(mark_bit)) return WHITE_OBJECT;
310     if (IsGrey(mark_bit)) return GREY_OBJECT;
311     UNREACHABLE();
312   }
313 
314  private:
315   DISALLOW_IMPLICIT_CONSTRUCTORS(Marking);
316 };
317 
318 }  // namespace internal
319 }  // namespace v8
320 
321 #endif  // V8_HEAP_MARKING_H_
322