annotate qt/core/QList.d @ 292:19498f420252 signals

more QList goodness
author eldar
date Tue, 10 Nov 2009 19:29:42 +0000
parents 0d2094800bdb
children 8627891e4556
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
1 module qt.core.QList;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
2
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
3 import qt.QGlobal;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
4 import qt.qtd.Atomic;
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
5 //import qt.core.QTypeInfo;
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
6
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
7 import qt.qtd.MetaMarshall;
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
8
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
9 import core.stdc.stdlib : qRealloc = realloc, qFree = free, qMalloc = malloc;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
10 import core.stdc.string : memcpy, memmove;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
11
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
12 import std.traits;
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
13
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
14 enum INT_MAX = int.max;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
15
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
16 bool isComplex(T)()
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
17 if (is(typeof(T.QTypeInfo)))
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
18 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
19 return T.QTypeInfo.isComplex();
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
20 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
21
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
22 bool isStatic(T)()
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
23 if (is(typeof(T.QTypeInfo)))
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
24 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
25 return T.QTypeInfo.isStatic();
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
26 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
27
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
28 bool isLarge(T)()
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
29 if (is(typeof(T.QTypeInfo)))
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
30 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
31 return T.QTypeInfo.isLarge();
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
32 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
33
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
34
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
35 int qAllocMore(int alloc, int extra)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
36 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
37 if (alloc == 0 && extra == 0)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
38 return 0;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
39 const int page = 1 << 12;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
40 int nalloc;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
41 alloc += extra;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
42 if (alloc < 1<<6) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
43 nalloc = (1<<3) + ((alloc >>3) << 3);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
44 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
45 // don't do anything if the loop will overflow signed int.
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
46 if (alloc >= INT_MAX/2)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
47 return INT_MAX;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
48 nalloc = (alloc < page) ? 1 << 3 : page;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
49 while (nalloc < alloc) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
50 if (nalloc <= 0)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
51 return INT_MAX;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
52 nalloc *= 2;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
53 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
54 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
55 return nalloc - extra;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
56 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
57
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
58 private int grow(int size)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
59 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
60 // dear compiler: don't optimize me out.
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
61 // synchronized {
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
62 int x = qAllocMore(size * (void*).sizeof, QListData.DataHeaderSize) / (void*).sizeof;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
63 return x;
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
64 // }
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
65 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
66
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
67 struct QListData {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
68 struct Data {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
69 Atomic!int ref_;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
70 int alloc, begin, end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
71 uint sharable;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
72 void*[1] array;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
73 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
74
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
75 enum { DataHeaderSize = Data.sizeof - (void*).sizeof }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
76
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
77 static Data shared_null;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
78 Data *d;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
79
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
80 static this()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
81 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
82 shared_null = Data(Atomic!int(1), 0, 0, 0, true, [null]);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
83 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
84
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
85
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
86 // Data *detach(); // remove in 5.0
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
87
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
88 Data* detach2()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
89 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
90 Data* x = d;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
91 d = cast(Data*)(qMalloc(DataHeaderSize + x.alloc * (void*).sizeof));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
92 if (!d)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
93 qFatal("QList: Out of memory");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
94
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
95 memcpy(d, x, DataHeaderSize + x.alloc * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
96 d.alloc = x.alloc;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
97 d.ref_.store(1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
98 d.sharable = true;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
99 if (!d.alloc)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
100 d.begin = d.end = 0;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
101
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
102 return x;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
103 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
104
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
105 void realloc(int alloc)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
106 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
107 // assert(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
108 Data* x = cast(Data*)(qRealloc(d, DataHeaderSize + alloc * (void*).sizeof));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
109 if (!x)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
110 qFatal("QList: Out of memory");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
111
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
112 d = x;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
113 d.alloc = alloc;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
114 if (!alloc)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
115 d.begin = d.end = 0;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
116 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
117
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
118 void** append()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
119 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
120 // #TODO Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
121 if (d.end == d.alloc) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
122 int n = d.end - d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
123 if (d.begin > 2 * d.alloc / 3) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
124 memcpy(d.array.ptr + n, d.array.ptr + d.begin, n * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
125 d.begin = n;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
126 d.end = n * 2;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
127 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
128 realloc(grow(d.alloc + 1));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
129 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
130 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
131 return d.array.ptr + d.end++;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
132 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
133
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
134 void **append(const ref QListData l)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
135 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
136 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
137 int e = d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
138 int n = l.d.end - l.d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
139 if (n) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
140 if (e + n > d.alloc)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
141 realloc(grow(e + l.d.end - l.d.begin));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
142 memcpy(d.array.ptr + d.end, l.d.array.ptr + l.d.begin, n * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
143 d.end += n;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
144 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
145 return d.array.ptr + e;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
146 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
147
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
148 void **prepend()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
149 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
150 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
151 if (d.begin == 0) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
152 if (d.end >= d.alloc / 3)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
153 realloc(grow(d.alloc + 1));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
154
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
155 if (d.end < d.alloc / 3)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
156 d.begin = d.alloc - 2 * d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
157 else
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
158 d.begin = d.alloc - d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
159
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
160 memmove(d.array.ptr + d.begin, d.array.ptr, d.end * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
161 d.end += d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
162 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
163 return d.array.ptr + --d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
164 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
165
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
166 void **insert(int i)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
167 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
168 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
169 if (i <= 0)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
170 return prepend();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
171 if (i >= d.end - d.begin)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
172 return append();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
173
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
174 bool leftward = false;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
175 int size = d.end - d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
176
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
177 if (d.begin == 0) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
178 if (d.end == d.alloc) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
179 // If the array is full, we expand it and move some items rightward
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
180 realloc(grow(d.alloc + 1));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
181 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
182 // If there is free space at the end of the array, we move some items rightward
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
183 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
184 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
185 if (d.end == d.alloc) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
186 // If there is free space at the beginning of the array, we move some items leftward
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
187 leftward = true;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
188 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
189 // If there is free space at both ends, we move as few items as possible
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
190 leftward = (i < size - i);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
191 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
192 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
193
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
194 if (leftward) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
195 --d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
196 memmove(d.array.ptr + d.begin, d.array.ptr + d.begin + 1, i * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
197 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
198 memmove(d.array.ptr + d.begin + i + 1, d.array.ptr + d.begin + i,
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
199 (size - i) * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
200 ++d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
201 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
202 return d.array.ptr + d.begin + i;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
203 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
204
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
205 void remove(int i)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
206 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
207 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
208 i += d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
209 if (i - d.begin < d.end - i) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
210 if (int offset = i - d.begin)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
211 memmove(d.array.ptr + d.begin + 1, d.array.ptr + d.begin, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
212 d.begin++;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
213 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
214 if (int offset = d.end - i - 1)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
215 memmove(d.array.ptr + i, d.array.ptr + i + 1, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
216 d.end--;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
217 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
218 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
219
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
220 void remove(int i, int n)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
221 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
222 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
223 i += d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
224 int middle = i + n/2;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
225 if (middle - d.begin < d.end - middle) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
226 memmove(d.array.ptr + d.begin + n, d.array.ptr + d.begin,
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
227 (i - d.begin) * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
228 d.begin += n;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
229 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
230 memmove(d.array.ptr + i, d.array.ptr + i + n,
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
231 (d.end - i - n) * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
232 d.end -= n;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
233 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
234 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
235
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
236 void move(int from, int to)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
237 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
238 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
239 if (from == to)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
240 return;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
241
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
242 from += d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
243 to += d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
244 void *t = d.array.ptr[from];
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
245
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
246 if (from < to) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
247 if (d.end == d.alloc || 3 * (to - from) < 2 * (d.end - d.begin)) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
248 memmove(d.array.ptr + from, d.array.ptr + from + 1, (to - from) * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
249 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
250 // optimization
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
251 if (int offset = from - d.begin)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
252 memmove(d.array.ptr + d.begin + 1, d.array.ptr + d.begin, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
253 if (int offset = d.end - (to + 1))
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
254 memmove(d.array.ptr + to + 2, d.array.ptr + to + 1, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
255 ++d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
256 ++d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
257 ++to;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
258 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
259 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
260 if (d.begin == 0 || 3 * (from - to) < 2 * (d.end - d.begin)) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
261 memmove(d.array.ptr + to + 1, d.array.ptr + to, (from - to) * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
262 } else {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
263 // optimization
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
264 if (int offset = to - d.begin)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
265 memmove(d.array.ptr + d.begin - 1, d.array.ptr + d.begin, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
266 if (int offset = d.end - (from + 1))
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
267 memmove(d.array.ptr + from, d.array.ptr + from + 1, offset * (void*).sizeof);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
268 --d.begin;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
269 --d.end;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
270 --to;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
271 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
272 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
273 d.array.ptr[to] = t;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
274 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
275
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
276 void **erase(void **xi)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
277 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
278 // Q_ASSERT(d.ref_ == 1);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
279 int i = xi - (d.array.ptr + d.begin);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
280 remove(i);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
281 return d.array.ptr + d.begin + i;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
282 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
283
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
284 int size() const { return d.end - d.begin; }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
285 bool isEmpty() const { return d.end == d.begin; }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
286 const (void*)* at(int i) const { return d.array.ptr + d.begin + i; }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
287 const (void*)* begin() const { return d.array.ptr + d.begin; }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
288 const (void*)* end() const { return d.array.ptr + d.end; }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
289 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
290
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
291 import std.stdio;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
292
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
293 struct QList(T)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
294 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
295 struct Node
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
296 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
297 void *v;
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
298
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
299 static if (isQObjectType!T)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
300 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
301 T t()
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
302 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
303 return T.__getObject( *cast(void**)(&this) );
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
304 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
305 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
306 else
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
307 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
308 ref T t()
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
309 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
310 return *cast(T*)(&this);
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
311 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
312 // { return *cast(T*)(QTypeInfo!T.isLarge || QTypeInfo!T.isStatic
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
313 // ? v : &this); } }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
314 }
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
315 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
316
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
317 union {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
318 QListData p;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
319 QListData.Data* d;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
320 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
321
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
322 public:
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
323 void output()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
324 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
325 writeln("QList atomic ", d.ref_.load());
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
326 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
327
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
328 static QList!T opCall()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
329 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
330 QList!T res;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
331 writeln("QList opCall");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
332
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
333 res.d = &QListData.shared_null;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
334 res.d.ref_.increment();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
335
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
336 return res;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
337 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
338
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
339 this(this)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
340 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
341 writeln("QList postblit");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
342 d.ref_.increment();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
343 if (!d.sharable)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
344 detach_helper();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
345 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
346
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
347 ~this()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
348 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
349 writeln("QList ~this");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
350 if (d && !d.ref_.decrement())
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
351 free(d);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
352 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
353
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
354 ref QList!T opAssign(const ref QList!T l)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
355 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
356 writeln("QList opAssign");
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
357 if (d != l.d) {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
358 l.d.ref_.increment();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
359 if (!d.ref_.decrement())
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
360 free(d);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
361 d = cast(QListData.Data*)l.d;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
362 if (!d.sharable)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
363 detach_helper();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
364 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
365 return this;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
366 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
367
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
368 int length() const { return p.size(); }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
369 int size() const { return length; }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
370
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
371 void detach() { if (d.ref_.load() != 1) detach_helper(); }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
372
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
373 private void detach_helper()
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
374 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
375 Node *n = cast(Node*)(p.begin());
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
376 QListData.Data* x = p.detach2();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
377 node_copy(cast(Node*)(p.begin()), cast(Node*)(p.end()), n);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
378 if (!x.ref_.decrement())
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
379 free(x);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
380 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
381
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
382 void append(const T t) // fix to const ref for complex types TODO
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
383 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
384 detach();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
385 /* static if (QTypeInfo!T.isLarge || QTypeInfo!T.isStatic)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
386 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
387 node_construct(cast(Node*)(p.append()), t);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
388 }
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
389 else*/ static if (isQObjectType!T)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
390 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
391 auto n = cast(Node*)(p.append());
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
392 *cast(void**)(n) = cast(Node*) t.__nativeId;
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
393 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
394 else
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
395 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
396 const T cpy = t;
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
397 node_construct(cast(Node*)(p.append()), cpy);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
398 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
399 }
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
400
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
401 static if (isQObjectType!T)
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
402 {
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
403 T at(int i) const
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
404 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
405 assert(i >= 0 && i < p.size(), "QList!T.at(): index out of range");
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
406 return (cast(Node*)(p.at(i))).t();
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
407 }
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
408 }
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
409 else
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
410 {
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
411 ref const (T) at(int i) const
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
412 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
413 assert(i >= 0 && i < p.size(), "QList!T.at(): index out of range");
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
414 return (cast(Node*)(p.at(i))).t();
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
415 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
416 }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
417
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
418 static if (isQObjectType!T)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
419 { }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
420 else
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
421 void node_construct(Node *n, const ref T t)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
422 {
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
423 /* TODO static if (QTypeInfo!T.isLarge || QTypeInfo!T.isStatic)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
424 n.v = new T(t);
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
425 else static if (QTypeInfo!T.isComplex)
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
426 new (n) T(t);
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
427 else*/
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
428 { *cast(T*)(n) = cast(T)(t); }
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
429 }
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
430
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
431 void node_copy(Node *from, Node *to, Node *src)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
432 {
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
433 /* TODO if (QTypeInfo<T>::isLarge || QTypeInfo<T>::isStatic)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
434 while(from != to)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
435 (from++)->v = new T(*reinterpret_cast<T*>((src++)->v));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
436 else if (QTypeInfo<T>::isComplex)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
437 while(from != to)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
438 new (from++) T(*reinterpret_cast<T*>(src++));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
439 */
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
440 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
441
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
442 void free(QListData.Data* data)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
443 {
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
444 writeln("QList data destroyed");
291
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
445 node_destruct(cast(Node*)(data.array.ptr + data.begin),
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
446 cast(Node*)(data.array.ptr + data.end));
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
447 if (data.ref_.load() == 0)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
448 {} // qFree(data); TODO
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
449 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
450
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
451 void node_destruct(Node *from, Node *to)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
452 {/* TODO
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
453 if (QTypeInfo!T.isLarge || QTypeInfo!T.isStatic)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
454 while (from != to) --to, delete cast(T*)(to->v);
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
455 else if (QTypeInfo!T.isComplex)
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
456 while (from != to) --to, cast(T*)(to).~T();
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
457 */
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
458 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
459 }
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
460
0d2094800bdb QList native implementation
eldar
parents:
diff changeset
461 extern(C) void qtd_create_QList(void *nativeId);
292
19498f420252 more QList goodness
eldar
parents: 291
diff changeset
462 extern(C) void qtd_create_QList_QObject(void *nativeId);