d40b7e95a91be1ee3ad328fe51f841102ed1d819
[platform/upstream/nodejs.git] / deps / v8 / test / cctest / test-weaktypedarrays.cc
1 // Copyright 2013 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are
4 // met:
5 //
6 //     * Redistributions of source code must retain the above copyright
7 //       notice, this list of conditions and the following disclaimer.
8 //     * Redistributions in binary form must reproduce the above
9 //       copyright notice, this list of conditions and the following
10 //       disclaimer in the documentation and/or other materials provided
11 //       with the distribution.
12 //     * Neither the name of Google Inc. nor the names of its
13 //       contributors may be used to endorse or promote products derived
14 //       from this software without specific prior written permission.
15 //
16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28 #include <stdlib.h>
29
30 #include "src/v8.h"
31 #include "test/cctest/cctest.h"
32
33 #include "src/api.h"
34 #include "src/heap/heap.h"
35 #include "src/objects.h"
36
37 using namespace v8::internal;
38
39 static Isolate* GetIsolateFrom(LocalContext* context) {
40   return reinterpret_cast<Isolate*>((*context)->GetIsolate());
41 }
42
43
44 static int CountArrayBuffersInWeakList(Heap* heap) {
45   int count = 0;
46   for (Object* o = heap->array_buffers_list();
47        !o->IsUndefined();
48        o = JSArrayBuffer::cast(o)->weak_next()) {
49     count++;
50   }
51   return count;
52 }
53
54
55 static bool HasArrayBufferInWeakList(Heap* heap, JSArrayBuffer* ab) {
56   for (Object* o = heap->array_buffers_list();
57        !o->IsUndefined();
58        o = JSArrayBuffer::cast(o)->weak_next()) {
59     if (ab == o) return true;
60   }
61   return false;
62 }
63
64
65 static int CountViews(JSArrayBuffer* array_buffer) {
66   int count = 0;
67   for (Object* o = array_buffer->weak_first_view();
68        !o->IsUndefined();
69        o = JSArrayBufferView::cast(o)->weak_next()) {
70     count++;
71   }
72
73   return count;
74 }
75
76 static bool HasViewInWeakList(JSArrayBuffer* array_buffer,
77                               JSArrayBufferView* ta) {
78   for (Object* o = array_buffer->weak_first_view();
79        !o->IsUndefined();
80        o = JSArrayBufferView::cast(o)->weak_next()) {
81     if (ta == o) return true;
82   }
83   return false;
84 }
85
86
87 TEST(WeakArrayBuffersFromApi) {
88   v8::V8::Initialize();
89   LocalContext context;
90   Isolate* isolate = GetIsolateFrom(&context);
91
92   int start = CountArrayBuffersInWeakList(isolate->heap());
93   {
94     v8::HandleScope s1(context->GetIsolate());
95     v8::Handle<v8::ArrayBuffer> ab1 =
96         v8::ArrayBuffer::New(context->GetIsolate(), 256);
97     {
98       v8::HandleScope s2(context->GetIsolate());
99       v8::Handle<v8::ArrayBuffer> ab2 =
100           v8::ArrayBuffer::New(context->GetIsolate(), 128);
101
102       Handle<JSArrayBuffer> iab1 = v8::Utils::OpenHandle(*ab1);
103       Handle<JSArrayBuffer> iab2 = v8::Utils::OpenHandle(*ab2);
104       CHECK_EQ(2, CountArrayBuffersInWeakList(isolate->heap()) - start);
105       CHECK(HasArrayBufferInWeakList(isolate->heap(), *iab1));
106       CHECK(HasArrayBufferInWeakList(isolate->heap(), *iab2));
107     }
108     isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
109     CHECK_EQ(1, CountArrayBuffersInWeakList(isolate->heap()) - start);
110     {
111       HandleScope scope2(isolate);
112       Handle<JSArrayBuffer> iab1 = v8::Utils::OpenHandle(*ab1);
113
114       CHECK(HasArrayBufferInWeakList(isolate->heap(), *iab1));
115     }
116   }
117
118   isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
119   CHECK_EQ(start, CountArrayBuffersInWeakList(isolate->heap()));
120 }
121
122
123 TEST(WeakArrayBuffersFromScript) {
124   v8::V8::Initialize();
125   LocalContext context;
126   Isolate* isolate = GetIsolateFrom(&context);
127   int start = CountArrayBuffersInWeakList(isolate->heap());
128
129   for (int i = 1; i <= 3; i++) {
130     // Create 3 array buffers, make i-th of them garbage,
131     // validate correct state of array buffer weak list.
132     CHECK_EQ(start, CountArrayBuffersInWeakList(isolate->heap()));
133     {
134       v8::HandleScope scope(context->GetIsolate());
135
136       {
137         v8::HandleScope s1(context->GetIsolate());
138         CompileRun("var ab1 = new ArrayBuffer(256);"
139                    "var ab2 = new ArrayBuffer(256);"
140                    "var ab3 = new ArrayBuffer(256);");
141         v8::Handle<v8::ArrayBuffer> ab1 =
142             v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab1"));
143         v8::Handle<v8::ArrayBuffer> ab2 =
144             v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab2"));
145         v8::Handle<v8::ArrayBuffer> ab3 =
146             v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab3"));
147
148         CHECK_EQ(3, CountArrayBuffersInWeakList(isolate->heap()) - start);
149         CHECK(HasArrayBufferInWeakList(isolate->heap(),
150               *v8::Utils::OpenHandle(*ab1)));
151         CHECK(HasArrayBufferInWeakList(isolate->heap(),
152               *v8::Utils::OpenHandle(*ab2)));
153         CHECK(HasArrayBufferInWeakList(isolate->heap(),
154               *v8::Utils::OpenHandle(*ab3)));
155       }
156
157       i::ScopedVector<char> source(1024);
158       i::SNPrintF(source, "ab%d = null;", i);
159       CompileRun(source.start());
160       isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
161
162       CHECK_EQ(2, CountArrayBuffersInWeakList(isolate->heap()) - start);
163
164       {
165         v8::HandleScope s2(context->GetIsolate());
166         for (int j = 1; j <= 3; j++) {
167           if (j == i) continue;
168           i::SNPrintF(source, "ab%d", j);
169           v8::Handle<v8::ArrayBuffer> ab =
170               v8::Handle<v8::ArrayBuffer>::Cast(CompileRun(source.start()));
171           CHECK(HasArrayBufferInWeakList(isolate->heap(),
172                 *v8::Utils::OpenHandle(*ab)));
173           }
174       }
175
176       CompileRun("ab1 = null; ab2 = null; ab3 = null;");
177     }
178
179     isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
180     CHECK_EQ(start, CountArrayBuffersInWeakList(isolate->heap()));
181   }
182 }
183
184 template <typename View>
185 void TestViewFromApi() {
186   v8::V8::Initialize();
187   LocalContext context;
188   Isolate* isolate = GetIsolateFrom(&context);
189
190   v8::HandleScope s1(context->GetIsolate());
191   v8::Handle<v8::ArrayBuffer> ab =
192       v8::ArrayBuffer::New(context->GetIsolate(), 2048);
193   Handle<JSArrayBuffer> iab = v8::Utils::OpenHandle(*ab);
194   {
195     v8::HandleScope s2(context->GetIsolate());
196     v8::Handle<View> ta1 = View::New(ab, 0, 256);
197     {
198       v8::HandleScope s3(context->GetIsolate());
199       v8::Handle<View> ta2 = View::New(ab, 0, 128);
200
201       Handle<JSArrayBufferView> ita1 = v8::Utils::OpenHandle(*ta1);
202       Handle<JSArrayBufferView> ita2 = v8::Utils::OpenHandle(*ta2);
203       CHECK_EQ(2, CountViews(*iab));
204       CHECK(HasViewInWeakList(*iab, *ita1));
205       CHECK(HasViewInWeakList(*iab, *ita2));
206     }
207     isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
208     CHECK_EQ(1, CountViews(*iab));
209     Handle<JSArrayBufferView> ita1 = v8::Utils::OpenHandle(*ta1);
210     CHECK(HasViewInWeakList(*iab, *ita1));
211   }
212   isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
213
214   CHECK_EQ(0, CountViews(*iab));
215 }
216
217
218 TEST(Uint8ArrayFromApi) {
219   TestViewFromApi<v8::Uint8Array>();
220 }
221
222
223 TEST(Int8ArrayFromApi) {
224   TestViewFromApi<v8::Int8Array>();
225 }
226
227
228 TEST(Uint16ArrayFromApi) {
229   TestViewFromApi<v8::Uint16Array>();
230 }
231
232
233 TEST(Int16ArrayFromApi) {
234   TestViewFromApi<v8::Int16Array>();
235 }
236
237
238 TEST(Uint32ArrayFromApi) {
239   TestViewFromApi<v8::Uint32Array>();
240 }
241
242
243 TEST(Int32ArrayFromApi) {
244   TestViewFromApi<v8::Int32Array>();
245 }
246
247
248 TEST(Float32ArrayFromApi) {
249   TestViewFromApi<v8::Float32Array>();
250 }
251
252
253 TEST(Float64ArrayFromApi) {
254   TestViewFromApi<v8::Float64Array>();
255 }
256
257
258 TEST(Uint8ClampedArrayFromApi) {
259   TestViewFromApi<v8::Uint8ClampedArray>();
260 }
261
262
263 TEST(DataViewFromApi) {
264   TestViewFromApi<v8::DataView>();
265 }
266
267 template <typename TypedArray>
268 static void TestTypedArrayFromScript(const char* constructor) {
269   v8::V8::Initialize();
270   LocalContext context;
271   Isolate* isolate = GetIsolateFrom(&context);
272   v8::HandleScope scope(context->GetIsolate());
273   int start = CountArrayBuffersInWeakList(isolate->heap());
274   CompileRun("var ab = new ArrayBuffer(2048);");
275   for (int i = 1; i <= 3; i++) {
276     // Create 3 typed arrays, make i-th of them garbage,
277     // validate correct state of typed array weak list.
278     v8::HandleScope s0(context->GetIsolate());
279     i::ScopedVector<char> source(2048);
280
281     CHECK_EQ(1, CountArrayBuffersInWeakList(isolate->heap()) - start);
282
283     {
284       v8::HandleScope s1(context->GetIsolate());
285       i::SNPrintF(source,
286               "var ta1 = new %s(ab);"
287               "var ta2 = new %s(ab);"
288               "var ta3 = new %s(ab)",
289               constructor, constructor, constructor);
290
291       CompileRun(source.start());
292       v8::Handle<v8::ArrayBuffer> ab =
293           v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab"));
294       v8::Handle<TypedArray> ta1 =
295           v8::Handle<TypedArray>::Cast(CompileRun("ta1"));
296       v8::Handle<TypedArray> ta2 =
297           v8::Handle<TypedArray>::Cast(CompileRun("ta2"));
298       v8::Handle<TypedArray> ta3 =
299           v8::Handle<TypedArray>::Cast(CompileRun("ta3"));
300       CHECK_EQ(1, CountArrayBuffersInWeakList(isolate->heap()) - start);
301       Handle<JSArrayBuffer> iab = v8::Utils::OpenHandle(*ab);
302       CHECK_EQ(3, CountViews(*iab));
303       CHECK(HasViewInWeakList(*iab, *v8::Utils::OpenHandle(*ta1)));
304       CHECK(HasViewInWeakList(*iab, *v8::Utils::OpenHandle(*ta2)));
305       CHECK(HasViewInWeakList(*iab, *v8::Utils::OpenHandle(*ta3)));
306     }
307
308     i::SNPrintF(source, "ta%d = null;", i);
309     CompileRun(source.start());
310     isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
311
312     CHECK_EQ(1, CountArrayBuffersInWeakList(isolate->heap()) - start);
313
314     {
315       v8::HandleScope s2(context->GetIsolate());
316       v8::Handle<v8::ArrayBuffer> ab =
317           v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab"));
318       Handle<JSArrayBuffer> iab = v8::Utils::OpenHandle(*ab);
319       CHECK_EQ(2, CountViews(*iab));
320       for (int j = 1; j <= 3; j++) {
321         if (j == i) continue;
322         i::SNPrintF(source, "ta%d", j);
323         v8::Handle<TypedArray> ta =
324             v8::Handle<TypedArray>::Cast(CompileRun(source.start()));
325         CHECK(HasViewInWeakList(*iab, *v8::Utils::OpenHandle(*ta)));
326       }
327     }
328
329     CompileRun("ta1 = null; ta2 = null; ta3 = null;");
330     isolate->heap()->CollectAllGarbage(Heap::kAbortIncrementalMarkingMask);
331
332     CHECK_EQ(1, CountArrayBuffersInWeakList(isolate->heap()) - start);
333
334     {
335       v8::HandleScope s3(context->GetIsolate());
336       v8::Handle<v8::ArrayBuffer> ab =
337           v8::Handle<v8::ArrayBuffer>::Cast(CompileRun("ab"));
338       Handle<JSArrayBuffer> iab = v8::Utils::OpenHandle(*ab);
339       CHECK_EQ(0, CountViews(*iab));
340     }
341   }
342 }
343
344
345 TEST(Uint8ArrayFromScript) {
346   TestTypedArrayFromScript<v8::Uint8Array>("Uint8Array");
347 }
348
349
350 TEST(Int8ArrayFromScript) {
351   TestTypedArrayFromScript<v8::Int8Array>("Int8Array");
352 }
353
354
355 TEST(Uint16ArrayFromScript) {
356   TestTypedArrayFromScript<v8::Uint16Array>("Uint16Array");
357 }
358
359
360 TEST(Int16ArrayFromScript) {
361   TestTypedArrayFromScript<v8::Int16Array>("Int16Array");
362 }
363
364
365 TEST(Uint32ArrayFromScript) {
366   TestTypedArrayFromScript<v8::Uint32Array>("Uint32Array");
367 }
368
369
370 TEST(Int32ArrayFromScript) {
371   TestTypedArrayFromScript<v8::Int32Array>("Int32Array");
372 }
373
374
375 TEST(Float32ArrayFromScript) {
376   TestTypedArrayFromScript<v8::Float32Array>("Float32Array");
377 }
378
379
380 TEST(Float64ArrayFromScript) {
381   TestTypedArrayFromScript<v8::Float64Array>("Float64Array");
382 }
383
384
385 TEST(Uint8ClampedArrayFromScript) {
386   TestTypedArrayFromScript<v8::Uint8ClampedArray>("Uint8ClampedArray");
387 }
388
389
390 TEST(DataViewFromScript) {
391   TestTypedArrayFromScript<v8::DataView>("DataView");
392 }