source: trunk/src/script/qscriptgc_p.h@ 203

Last change on this file since 203 was 2, checked in by Dmitry A. Kuminov, 16 years ago

Initially imported qt-all-opensource-src-4.5.1 from Trolltech.

File size: 8.4 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
4** Contact: Qt Software Information ([email protected])
5**
6** This file is part of the QtScript module of the Qt Toolkit.
7**
8** $QT_BEGIN_LICENSE:LGPL$
9** Commercial Usage
10** Licensees holding valid Qt Commercial licenses may use this file in
11** accordance with the Qt Commercial License Agreement provided with the
12** Software or, alternatively, in accordance with the terms contained in
13** a written agreement between you and Nokia.
14**
15** GNU Lesser General Public License Usage
16** Alternatively, this file may be used under the terms of the GNU Lesser
17** General Public License version 2.1 as published by the Free Software
18** Foundation and appearing in the file LICENSE.LGPL included in the
19** packaging of this file. Please review the following information to
20** ensure the GNU Lesser General Public License version 2.1 requirements
21** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
22**
23** In addition, as a special exception, Nokia gives you certain
24** additional rights. These rights are described in the Nokia Qt LGPL
25** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
26** package.
27**
28** GNU General Public License Usage
29** Alternatively, this file may be used under the terms of the GNU
30** General Public License version 3.0 as published by the Free Software
31** Foundation and appearing in the file LICENSE.GPL included in the
32** packaging of this file. Please review the following information to
33** ensure the GNU General Public License version 3.0 requirements will be
34** met: http://www.gnu.org/copyleft/gpl.html.
35**
36** If you are unsure which license is appropriate for your use, please
37** contact the sales department at [email protected].
38** $QT_END_LICENSE$
39**
40****************************************************************************/
41
42#ifndef QSCRIPTGC_P_H
43#define QSCRIPTGC_P_H
44
45//
46// W A R N I N G
47// -------------
48//
49// This file is not part of the Qt API. It exists purely as an
50// implementation detail. This header file may change from version to
51// version without notice, or even be removed.
52//
53// We mean it.
54//
55
56#include <QtCore/qglobal.h>
57
58#ifndef QT_NO_SCRIPT
59
60#include <QtCore/QtDebug>
61#include <new>
62
63#include "qscriptmemorypool_p.h"
64
65QT_BEGIN_NAMESPACE
66
67namespace QScript {
68
69class GCBlock
70{
71public:
72 GCBlock *next;
73
74 union {
75 int generation;
76 uint flags;
77 };
78
79public:
80 inline GCBlock(GCBlock *n):
81 next(n), flags(0) {}
82
83 inline void *data()
84 { return reinterpret_cast<char *>(this) + sizeof(GCBlock); }
85
86 inline static GCBlock *get(void *ptr)
87 {
88 char *where = reinterpret_cast<char *>(ptr);
89 return reinterpret_cast<GCBlock *>(where - sizeof(GCBlock));
90 }
91};
92
93template <typename _Tp>
94class GCAlloc
95{
96private:
97 int m_new_allocated_blocks;
98 int m_free_blocks;
99 int m_new_allocated_extra_bytes;
100 GCBlock *m_head;
101 GCBlock *m_current;
102 GCBlock *m_free;
103 bool m_blocked_gc;
104 bool m_force_gc;
105 bool m_sweeping;
106 MemoryPool pool;
107 _Tp trivial;
108
109public:
110 enum { MaxNumberOfBlocks = 1 << 14 };
111 enum { MaxNumberOfExtraBytes = 0x800000 };
112
113public:
114 inline GCAlloc():
115 m_new_allocated_blocks(0),
116 m_free_blocks(0),
117 m_new_allocated_extra_bytes(0),
118 m_head(0),
119 m_current(0),
120 m_free(0),
121 m_blocked_gc(false),
122 m_force_gc(false),
123 m_sweeping(false) {
124 trivial.reset();
125 }
126
127 inline ~GCAlloc() {
128 }
129
130 inline void destruct() {
131 m_sweeping = true;
132 GCBlock *blk = m_free;
133
134 if (! blk) {
135 blk = m_head;
136 m_head = 0;
137 }
138
139 while (blk) {
140 GCBlock *was = blk;
141 blk = blk->next;
142
143 Q_ASSERT(was->data());
144 _Tp *data = reinterpret_cast<_Tp*>(was->data());
145 data->~_Tp();
146 blk->~GCBlock();
147
148 if (! blk && m_head) {
149 blk = m_head;
150 m_head = 0;
151 }
152 }
153 m_sweeping = false;
154 }
155
156 inline int newAllocatedBlocks() const { return m_new_allocated_blocks; }
157 inline int freeBlocks() const { return m_free_blocks; }
158
159 inline _Tp *operator()(int generation)
160 {
161 GCBlock *previous = m_current;
162 void *where = 0;
163
164 if (! m_free) {
165 Q_ASSERT (m_free_blocks == 0);
166 where = pool.allocate(sizeof(GCBlock) + sizeof(_Tp));
167 ++m_new_allocated_blocks;
168 (void) new (reinterpret_cast<char*>(where) + sizeof(GCBlock)) _Tp();
169 } else {
170 --m_free_blocks;
171 where = m_free;
172 m_free = m_free->next;
173
174 if (! m_free)
175 m_force_gc = true;
176 }
177
178 m_current = new (where) GCBlock(0);
179
180 if (! previous) {
181 Q_ASSERT(! m_head);
182 m_head = m_current;
183 } else {
184 previous->next = m_current;
185 }
186 m_current->generation = generation;
187
188 return reinterpret_cast<_Tp*> (m_current->data());
189 }
190
191 inline bool blocked() const
192 {
193 return m_blocked_gc;
194 }
195
196 inline bool sweeping() const
197 {
198 return m_sweeping;
199 }
200
201 inline bool blockGC(bool block)
202 {
203 bool was = m_blocked_gc;
204 m_blocked_gc = block;
205 return was;
206 }
207
208 inline void requestGC()
209 {
210 m_force_gc = true;
211 }
212
213 inline void adjustBytesAllocated(int bytes)
214 { m_new_allocated_extra_bytes += bytes; }
215
216 inline bool poll()
217 {
218 if (m_blocked_gc || ! m_head)
219 return false;
220
221 else if (m_force_gc) {
222 m_force_gc = false;
223 return true;
224 }
225
226 else if (m_free && ! m_free->next)
227 return true;
228
229 return (m_new_allocated_blocks >= MaxNumberOfBlocks)
230 || ((m_new_allocated_extra_bytes >= MaxNumberOfExtraBytes)
231 && (m_new_allocated_blocks > 0));
232 }
233
234 inline int generation(_Tp *ptr) const
235 { return GCBlock::get(ptr)->generation; }
236
237 inline GCBlock *head() const
238 { return m_head; }
239
240 void sweep(int generation)
241 {
242 m_sweeping = true;
243 GCBlock *blk = m_head;
244 m_current = 0;
245
246 m_new_allocated_blocks = 0;
247 m_new_allocated_extra_bytes = 0;
248
249 while (blk != 0) {
250 if (blk->generation != generation) {
251 if (m_current)
252 m_current->next = blk->next;
253
254 GCBlock *tmp = blk;
255 blk = blk->next; // advance the pointer
256
257 tmp->next = m_free; // prepend the node to the free list...
258 m_free = tmp;
259 ++m_free_blocks;
260
261 if (m_free == m_head)
262 m_head = blk;
263
264 _Tp *data = reinterpret_cast<_Tp *>(tmp->data());
265 data->finalize();
266 tmp->~GCBlock();
267 } else {
268 m_current = blk;
269 blk = blk->next;
270 }
271 }
272
273 if (! m_current)
274 m_head = m_current;
275 m_sweeping = false;
276 }
277
278 class const_iterator
279 {
280 public:
281 typedef _Tp value_type;
282 typedef const _Tp *pointer;
283 typedef const _Tp &reference;
284 inline const_iterator() : i(0) { }
285 inline const_iterator(GCBlock *block) : i(block) { }
286 inline const_iterator(const const_iterator &o)
287 { i = reinterpret_cast<const const_iterator &>(o).i; }
288
289 inline const _Tp *data() const { return reinterpret_cast<_Tp*>(i->data()); }
290 inline const _Tp &value() const { return *reinterpret_cast<_Tp*>(i->data()); }
291 inline const _Tp &operator*() const { return *reinterpret_cast<_Tp*>(i->data()); }
292 inline const _Tp *operator->() const { return reinterpret_cast<_Tp*>(i->data()); }
293 inline bool operator==(const const_iterator &o) const { return i == o.i; }
294 inline bool operator!=(const const_iterator &o) const { return i != o.i; }
295
296 inline const_iterator &operator++() {
297 i = i->next;
298 return *this;
299 }
300 private:
301 GCBlock *i;
302 };
303 friend class const_iterator;
304
305 inline const_iterator constBegin() const { return const_iterator(m_head); }
306 inline const_iterator constEnd() const { return const_iterator(0); }
307
308private:
309 Q_DISABLE_COPY(GCAlloc)
310};
311
312} // namespace QScript
313
314QT_END_NAMESPACE
315
316#endif // QT_NO_SCRIPT
317#endif // QSCRIPT_GC_H
Note: See TracBrowser for help on using the repository browser.