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

Last change on this file since 275 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 {