1 // Copyright 2014 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "src/runtime/runtime-utils.h"
6
7 #include "src/arguments.h"
8 #include "src/conversions-inl.h"
9 #include "src/factory.h"
10
11 namespace v8 {
12 namespace internal {
13
14
RUNTIME_FUNCTION(Runtime_StringGetRawHashField)15 RUNTIME_FUNCTION(Runtime_StringGetRawHashField) {
16 HandleScope scope(isolate);
17 DCHECK(args.length() == 1);
18 CONVERT_ARG_HANDLE_CHECKED(String, string, 0);
19 return *isolate->factory()->NewNumberFromUint(string->hash_field());
20 }
21
22
RUNTIME_FUNCTION(Runtime_TheHole)23 RUNTIME_FUNCTION(Runtime_TheHole) {
24 SealHandleScope shs(isolate);
25 DCHECK(args.length() == 0);
26 return isolate->heap()->the_hole_value();
27 }
28
29
RUNTIME_FUNCTION(Runtime_JSCollectionGetTable)30 RUNTIME_FUNCTION(Runtime_JSCollectionGetTable) {
31 SealHandleScope shs(isolate);
32 DCHECK(args.length() == 1);
33 CONVERT_ARG_CHECKED(JSObject, object, 0);
34 CHECK(object->IsJSSet() || object->IsJSMap());
35 return static_cast<JSCollection*>(object)->table();
36 }
37
38
RUNTIME_FUNCTION(Runtime_GenericHash)39 RUNTIME_FUNCTION(Runtime_GenericHash) {
40 HandleScope scope(isolate);
41 DCHECK(args.length() == 1);
42 CONVERT_ARG_HANDLE_CHECKED(Object, object, 0);
43 Smi* hash = Object::GetOrCreateHash(isolate, object);
44 return hash;
45 }
46
47
RUNTIME_FUNCTION(Runtime_SetInitialize)48 RUNTIME_FUNCTION(Runtime_SetInitialize) {
49 HandleScope scope(isolate);
50 DCHECK(args.length() == 1);
51 CONVERT_ARG_HANDLE_CHECKED(JSSet, holder, 0);
52 JSSet::Initialize(holder, isolate);
53 return *holder;
54 }
55
56
RUNTIME_FUNCTION(Runtime_SetGrow)57 RUNTIME_FUNCTION(Runtime_SetGrow) {
58 HandleScope scope(isolate);
59 DCHECK(args.length() == 1);
60 CONVERT_ARG_HANDLE_CHECKED(JSSet, holder, 0);
61 Handle<OrderedHashSet> table(OrderedHashSet::cast(holder->table()));
62 table = OrderedHashSet::EnsureGrowable(table);
63 holder->set_table(*table);
64 return isolate->heap()->undefined_value();
65 }
66
67
RUNTIME_FUNCTION(Runtime_SetShrink)68 RUNTIME_FUNCTION(Runtime_SetShrink) {
69 HandleScope scope(isolate);
70 DCHECK(args.length() == 1);
71 CONVERT_ARG_HANDLE_CHECKED(JSSet, holder, 0);
72 Handle<OrderedHashSet> table(OrderedHashSet::cast(holder->table()));
73 table = OrderedHashSet::Shrink(table);
74 holder->set_table(*table);
75 return isolate->heap()->undefined_value();
76 }
77
78
RUNTIME_FUNCTION(Runtime_SetClear)79 RUNTIME_FUNCTION(Runtime_SetClear) {
80 HandleScope scope(isolate);
81 DCHECK(args.length() == 1);
82 CONVERT_ARG_HANDLE_CHECKED(JSSet, holder, 0);
83 JSSet::Clear(holder);
84 return isolate->heap()->undefined_value();
85 }
86
87
RUNTIME_FUNCTION(Runtime_SetIteratorInitialize)88 RUNTIME_FUNCTION(Runtime_SetIteratorInitialize) {
89 HandleScope scope(isolate);
90 DCHECK(args.length() == 3);
91 CONVERT_ARG_HANDLE_CHECKED(JSSetIterator, holder, 0);
92 CONVERT_ARG_HANDLE_CHECKED(JSSet, set, 1);
93 CONVERT_SMI_ARG_CHECKED(kind, 2)
94 CHECK(kind == JSSetIterator::kKindValues ||
95 kind == JSSetIterator::kKindEntries);
96 Handle<OrderedHashSet> table(OrderedHashSet::cast(set->table()));
97 holder->set_table(*table);
98 holder->set_index(Smi::kZero);
99 holder->set_kind(Smi::FromInt(kind));
100 return isolate->heap()->undefined_value();
101 }
102
103
RUNTIME_FUNCTION(Runtime_SetIteratorClone)104 RUNTIME_FUNCTION(Runtime_SetIteratorClone) {
105 HandleScope scope(isolate);
106 DCHECK(args.length() == 1);
107 CONVERT_ARG_HANDLE_CHECKED(JSSetIterator, holder, 0);
108
109 Handle<JSSetIterator> result = isolate->factory()->NewJSSetIterator();
110 result->set_table(holder->table());
111 result->set_index(Smi::FromInt(Smi::cast(holder->index())->value()));
112 result->set_kind(Smi::FromInt(Smi::cast(holder->kind())->value()));
113
114 return *result;
115 }
116
117
RUNTIME_FUNCTION(Runtime_SetIteratorNext)118 RUNTIME_FUNCTION(Runtime_SetIteratorNext) {
119 SealHandleScope shs(isolate);
120 DCHECK(args.length() == 2);
121 CONVERT_ARG_CHECKED(JSSetIterator, holder, 0);
122 CONVERT_ARG_CHECKED(JSArray, value_array, 1);
123 return holder->Next(value_array);
124 }
125
126
127 // The array returned contains the following information:
128 // 0: HasMore flag
129 // 1: Iteration index
130 // 2: Iteration kind
RUNTIME_FUNCTION(Runtime_SetIteratorDetails)131 RUNTIME_FUNCTION(Runtime_SetIteratorDetails) {
132 HandleScope scope(isolate);
133 DCHECK(args.length() == 1);
134 CONVERT_ARG_HANDLE_CHECKED(JSSetIterator, holder, 0);
135 Handle<FixedArray> details = isolate->factory()->NewFixedArray(4);
136 details->set(0, isolate->heap()->ToBoolean(holder->HasMore()));
137 details->set(1, holder->index());
138 details->set(2, holder->kind());
139 return *isolate->factory()->NewJSArrayWithElements(details);
140 }
141
142
RUNTIME_FUNCTION(Runtime_MapInitialize)143 RUNTIME_FUNCTION(Runtime_MapInitialize) {
144 HandleScope scope(isolate);
145 DCHECK(args.length() == 1);
146 CONVERT_ARG_HANDLE_CHECKED(JSMap, holder, 0);
147 JSMap::Initialize(holder, isolate);
148 return *holder;
149 }
150
151
RUNTIME_FUNCTION(Runtime_MapShrink)152 RUNTIME_FUNCTION(Runtime_MapShrink) {
153 HandleScope scope(isolate);
154 DCHECK(args.length() == 1);
155 CONVERT_ARG_HANDLE_CHECKED(JSMap, holder, 0);
156 Handle<OrderedHashMap> table(OrderedHashMap::cast(holder->table()));
157 table = OrderedHashMap::Shrink(table);
158 holder->set_table(*table);
159 return isolate->heap()->undefined_value();
160 }
161
162
RUNTIME_FUNCTION(Runtime_MapClear)163 RUNTIME_FUNCTION(Runtime_MapClear) {
164 HandleScope scope(isolate);
165 DCHECK(args.length() == 1);
166 CONVERT_ARG_HANDLE_CHECKED(JSMap, holder, 0);
167 JSMap::Clear(holder);
168 return isolate->heap()->undefined_value();
169 }
170
171
RUNTIME_FUNCTION(Runtime_MapGrow)172 RUNTIME_FUNCTION(Runtime_MapGrow) {
173 HandleScope scope(isolate);
174 DCHECK(args.length() == 1);
175 CONVERT_ARG_HANDLE_CHECKED(JSMap, holder, 0);
176 Handle<OrderedHashMap> table(OrderedHashMap::cast(holder->table()));
177 table = OrderedHashMap::EnsureGrowable(table);
178 holder->set_table(*table);
179 return isolate->heap()->undefined_value();
180 }
181
182
RUNTIME_FUNCTION(Runtime_MapIteratorInitialize)183 RUNTIME_FUNCTION(Runtime_MapIteratorInitialize) {
184 HandleScope scope(isolate);
185 DCHECK(args.length() == 3);
186 CONVERT_ARG_HANDLE_CHECKED(JSMapIterator, holder, 0);
187 CONVERT_ARG_HANDLE_CHECKED(JSMap, map, 1);
188 CONVERT_SMI_ARG_CHECKED(kind, 2)
189 CHECK(kind == JSMapIterator::kKindKeys ||
190 kind == JSMapIterator::kKindValues ||
191 kind == JSMapIterator::kKindEntries);
192 Handle<OrderedHashMap> table(OrderedHashMap::cast(map->table()));
193 holder->set_table(*table);
194 holder->set_index(Smi::kZero);
195 holder->set_kind(Smi::FromInt(kind));
196 return isolate->heap()->undefined_value();
197 }
198
199
RUNTIME_FUNCTION(Runtime_MapIteratorClone)200 RUNTIME_FUNCTION(Runtime_MapIteratorClone) {
201 HandleScope scope(isolate);
202 DCHECK(args.length() == 1);
203 CONVERT_ARG_HANDLE_CHECKED(JSMapIterator, holder, 0);
204
205 Handle<JSMapIterator> result = isolate->factory()->NewJSMapIterator();
206 result->set_table(holder->table());
207 result->set_index(Smi::FromInt(Smi::cast(holder->index())->value()));
208 result->set_kind(Smi::FromInt(Smi::cast(holder->kind())->value()));
209
210 return *result;
211 }
212
213
214 // The array returned contains the following information:
215 // 0: HasMore flag
216 // 1: Iteration index
217 // 2: Iteration kind
RUNTIME_FUNCTION(Runtime_MapIteratorDetails)218 RUNTIME_FUNCTION(Runtime_MapIteratorDetails) {
219 HandleScope scope(isolate);
220 DCHECK(args.length() == 1);
221 CONVERT_ARG_HANDLE_CHECKED(JSMapIterator, holder, 0);
222 Handle<FixedArray> details = isolate->factory()->NewFixedArray(4);
223 details->set(0, isolate->heap()->ToBoolean(holder->HasMore()));
224 details->set(1, holder->index());
225 details->set(2, holder->kind());
226 return *isolate->factory()->NewJSArrayWithElements(details);
227 }
228
229
RUNTIME_FUNCTION(Runtime_GetWeakMapEntries)230 RUNTIME_FUNCTION(Runtime_GetWeakMapEntries) {
231 HandleScope scope(isolate);
232 DCHECK(args.length() == 2);
233 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, holder, 0);
234 CONVERT_NUMBER_CHECKED(int, max_entries, Int32, args[1]);
235 CHECK(max_entries >= 0);
236
237 Handle<ObjectHashTable> table(ObjectHashTable::cast(holder->table()));
238 if (max_entries == 0 || max_entries > table->NumberOfElements()) {
239 max_entries = table->NumberOfElements();
240 }
241 Handle<FixedArray> entries =
242 isolate->factory()->NewFixedArray(max_entries * 2);
243 // Allocation can cause GC can delete weak elements. Reload.
244 if (max_entries > table->NumberOfElements()) {
245 max_entries = table->NumberOfElements();
246 }
247
248 {
249 DisallowHeapAllocation no_gc;
250 int count = 0;
251 for (int i = 0; count / 2 < max_entries && i < table->Capacity(); i++) {
252 Handle<Object> key(table->KeyAt(i), isolate);
253 if (table->IsKey(isolate, *key)) {
254 entries->set(count++, *key);
255 Object* value = table->Lookup(key);
256 entries->set(count++, value);
257 }
258 }
259 DCHECK_EQ(max_entries * 2, count);
260 }
261 return *isolate->factory()->NewJSArrayWithElements(entries);
262 }
263
264
RUNTIME_FUNCTION(Runtime_MapIteratorNext)265 RUNTIME_FUNCTION(Runtime_MapIteratorNext) {
266 SealHandleScope shs(isolate);
267 DCHECK(args.length() == 2);
268 CONVERT_ARG_CHECKED(JSMapIterator, holder, 0);
269 CONVERT_ARG_CHECKED(JSArray, value_array, 1);
270 return holder->Next(value_array);
271 }
272
273
RUNTIME_FUNCTION(Runtime_WeakCollectionInitialize)274 RUNTIME_FUNCTION(Runtime_WeakCollectionInitialize) {
275 HandleScope scope(isolate);
276 DCHECK(args.length() == 1);
277 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
278 JSWeakCollection::Initialize(weak_collection, isolate);
279 return *weak_collection;
280 }
281
282
RUNTIME_FUNCTION(Runtime_WeakCollectionGet)283 RUNTIME_FUNCTION(Runtime_WeakCollectionGet) {
284 HandleScope scope(isolate);
285 DCHECK(args.length() == 3);
286 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
287 CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
288 CONVERT_SMI_ARG_CHECKED(hash, 2)
289 CHECK(key->IsJSReceiver() || key->IsSymbol());
290 Handle<ObjectHashTable> table(
291 ObjectHashTable::cast(weak_collection->table()));
292 CHECK(table->IsKey(isolate, *key));
293 Handle<Object> lookup(table->Lookup(key, hash), isolate);
294 return lookup->IsTheHole(isolate) ? isolate->heap()->undefined_value()
295 : *lookup;
296 }
297
298
RUNTIME_FUNCTION(Runtime_WeakCollectionHas)299 RUNTIME_FUNCTION(Runtime_WeakCollectionHas) {
300 HandleScope scope(isolate);
301 DCHECK(args.length() == 3);
302 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
303 CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
304 CONVERT_SMI_ARG_CHECKED(hash, 2)
305 CHECK(key->IsJSReceiver() || key->IsSymbol());
306 Handle<ObjectHashTable> table(
307 ObjectHashTable::cast(weak_collection->table()));
308 CHECK(table->IsKey(isolate, *key));
309 Handle<Object> lookup(table->Lookup(key, hash), isolate);
310 return isolate->heap()->ToBoolean(!lookup->IsTheHole(isolate));
311 }
312
313
RUNTIME_FUNCTION(Runtime_WeakCollectionDelete)314 RUNTIME_FUNCTION(Runtime_WeakCollectionDelete) {
315 HandleScope scope(isolate);
316 DCHECK(args.length() == 3);
317 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
318 CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
319 CONVERT_SMI_ARG_CHECKED(hash, 2)
320 CHECK(key->IsJSReceiver() || key->IsSymbol());
321 Handle<ObjectHashTable> table(
322 ObjectHashTable::cast(weak_collection->table()));
323 CHECK(table->IsKey(isolate, *key));
324 bool was_present = JSWeakCollection::Delete(weak_collection, key, hash);
325 return isolate->heap()->ToBoolean(was_present);
326 }
327
328
RUNTIME_FUNCTION(Runtime_WeakCollectionSet)329 RUNTIME_FUNCTION(Runtime_WeakCollectionSet) {
330 HandleScope scope(isolate);
331 DCHECK(args.length() == 4);
332 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, weak_collection, 0);
333 CONVERT_ARG_HANDLE_CHECKED(Object, key, 1);
334 CHECK(key->IsJSReceiver() || key->IsSymbol());
335 CONVERT_ARG_HANDLE_CHECKED(Object, value, 2);
336 CONVERT_SMI_ARG_CHECKED(hash, 3)
337 Handle<ObjectHashTable> table(
338 ObjectHashTable::cast(weak_collection->table()));
339 CHECK(table->IsKey(isolate, *key));
340 JSWeakCollection::Set(weak_collection, key, value, hash);
341 return *weak_collection;
342 }
343
344
RUNTIME_FUNCTION(Runtime_GetWeakSetValues)345 RUNTIME_FUNCTION(Runtime_GetWeakSetValues) {
346 HandleScope scope(isolate);
347 DCHECK(args.length() == 2);
348 CONVERT_ARG_HANDLE_CHECKED(JSWeakCollection, holder, 0);
349 CONVERT_NUMBER_CHECKED(int, max_values, Int32, args[1]);
350 CHECK(max_values >= 0);
351
352 Handle<ObjectHashTable> table(ObjectHashTable::cast(holder->table()));
353 if (max_values == 0 || max_values > table->NumberOfElements()) {
354 max_values = table->NumberOfElements();
355 }
356 Handle<FixedArray> values = isolate->factory()->NewFixedArray(max_values);
357 // Recompute max_values because GC could have removed elements from the table.
358 if (max_values > table->NumberOfElements()) {
359 max_values = table->NumberOfElements();
360 }
361 {
362 DisallowHeapAllocation no_gc;
363 int count = 0;
364 for (int i = 0; count < max_values && i < table->Capacity(); i++) {
365 Object* key = table->KeyAt(i);
366 if (table->IsKey(isolate, key)) values->set(count++, key);
367 }
368 DCHECK_EQ(max_values, count);
369 }
370 return *isolate->factory()->NewJSArrayWithElements(values);
371 }
372 } // namespace internal
373 } // namespace v8
374