]> git.saurik.com Git - apple/javascriptcore.git/blame - heap/MarkStack.cpp
JavaScriptCore-7601.1.46.3.tar.gz
[apple/javascriptcore.git] / heap / MarkStack.cpp
CommitLineData
14957cd0
A
1/*
2 * Copyright (C) 2009, 2011 Apple Inc. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26#include "config.h"
27#include "MarkStack.h"
14957cd0 28
81345200 29#include "JSCInlines.h"
14957cd0 30
81345200 31namespace JSC {
6fe7ccc8 32
ed1e77d3
A
33MarkStackArray::MarkStackArray()
34 : GCSegmentedArray<const JSCell*>()
6fe7ccc8 35{
6fe7ccc8
A
36}
37
93a37866 38void MarkStackArray::donateSomeCellsTo(MarkStackArray& other)
6fe7ccc8 39{
93a37866
A
40 // Try to donate about 1 / 2 of our cells. To reduce copying costs,
41 // we prefer donating whole segments over donating individual cells,
42 // even if this skews away from our 1 / 2 target.
6fe7ccc8 43
93a37866 44 size_t segmentsToDonate = m_numberOfSegments / 2; // If we only have one segment (our head) we don't donate any segments.
6fe7ccc8 45
93a37866
A
46 if (!segmentsToDonate) {
47 size_t cellsToDonate = m_top / 2; // Round down to donate 0 / 1 cells.
48 while (cellsToDonate--) {
49 ASSERT(m_top);
50 other.append(removeLast());
51 }
6fe7ccc8
A
52 return;
53 }
14957cd0 54
93a37866
A
55 validatePrevious();
56 other.validatePrevious();
6fe7ccc8 57
93a37866
A
58 // Remove our head and the head of the other list before we start moving segments around.
59 // We'll add them back on once we're done donating.
81345200
A
60 GCArraySegment<const JSCell*>* myHead = m_segments.removeHead();
61 GCArraySegment<const JSCell*>* otherHead = other.m_segments.removeHead();
14957cd0 62
93a37866 63 while (segmentsToDonate--) {
81345200 64 GCArraySegment<const JSCell*>* current = m_segments.removeHead();
93a37866
A
65 ASSERT(current);
66 ASSERT(m_numberOfSegments > 1);
67 other.m_segments.push(current);
68 m_numberOfSegments--;
69 other.m_numberOfSegments++;
14957cd0 70 }
6fe7ccc8 71
93a37866
A
72 // Put the original heads back in their places.
73 m_segments.push(myHead);
74 other.m_segments.push(otherHead);
6fe7ccc8 75
93a37866
A
76 validatePrevious();
77 other.validatePrevious();
14957cd0
A
78}
79
93a37866 80void MarkStackArray::stealSomeCellsFrom(MarkStackArray& other, size_t idleThreadCount)
14957cd0 81{
93a37866
A
82 // Try to steal 1 / Nth of the shared array, where N is the number of idle threads.
83 // To reduce copying costs, we prefer stealing a whole segment over stealing
84 // individual cells, even if this skews away from our 1 / N target.
6fe7ccc8 85
93a37866
A
86 validatePrevious();
87 other.validatePrevious();
6fe7ccc8 88
93a37866
A
89 // If other has an entire segment, steal it and return.
90 if (other.m_numberOfSegments > 1) {
91 // Move the heads of the lists aside. We'll push them back on after.
81345200
A
92 GCArraySegment<const JSCell*>* otherHead = other.m_segments.removeHead();
93 GCArraySegment<const JSCell*>* myHead = m_segments.removeHead();
14957cd0 94
93a37866 95 ASSERT(other.m_segments.head()->m_top == s_segmentCapacity);
14957cd0 96
93a37866 97 m_segments.push(other.m_segments.removeHead());
14957cd0 98
93a37866
A
99 m_numberOfSegments++;
100 other.m_numberOfSegments--;
6fe7ccc8 101
93a37866
A
102 m_segments.push(myHead);
103 other.m_segments.push(otherHead);
6fe7ccc8 104
93a37866
A
105 validatePrevious();
106 other.validatePrevious();
14957cd0 107 return;
93a37866 108 }
6fe7ccc8 109
93a37866
A
110 size_t numberOfCellsToSteal = (other.size() + idleThreadCount - 1) / idleThreadCount; // Round up to steal 1 / 1.
111 while (numberOfCellsToSteal-- > 0 && other.canRemoveLast())
112 append(other.removeLast());
6fe7ccc8 113}
14957cd0
A
114
115} // namespace JSC