| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
| 4 // met: | 4 // met: |
| 5 // | 5 // |
| 6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
| 7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
| 8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
| 9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
| 10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
| (...skipping 959 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 970 // Recursively traverses hidden prototypes if property is not found | 970 // Recursively traverses hidden prototypes if property is not found |
| 971 static void GetOwnPropertyImplementation(JSObject* obj, | 971 static void GetOwnPropertyImplementation(JSObject* obj, |
| 972 String* name, | 972 String* name, |
| 973 LookupResult* result) { | 973 LookupResult* result) { |
| 974 obj->LocalLookupRealNamedProperty(name, result); | 974 obj->LocalLookupRealNamedProperty(name, result); |
| 975 | 975 |
| 976 if (result->IsFound()) return; | 976 if (result->IsFound()) return; |
| 977 | 977 |
| 978 Object* proto = obj->GetPrototype(); | 978 Object* proto = obj->GetPrototype(); |
| 979 if (proto->IsJSObject() && | 979 if (proto->IsJSObject() && |
| 980 JSObject::cast(proto)->map()->is_hidden_prototype()) { | 980 JSObject::cast(proto)->map()->is_hidden_prototype()) |
| 981 GetOwnPropertyImplementation(JSObject::cast(proto), | 981 GetOwnPropertyImplementation(JSObject::cast(proto), |
| 982 name, result); | 982 name, result); |
| 983 } | |
| 984 } | 983 } |
| 985 | 984 |
| 986 | 985 |
| 987 static bool CheckAccessException(Object* callback, | 986 static bool CheckAccessException(LookupResult* result, |
| 988 v8::AccessType access_type) { | 987 v8::AccessType access_type) { |
| 989 if (callback->IsAccessorInfo()) { | 988 if (result->type() == CALLBACKS) { |
| 990 AccessorInfo* info = AccessorInfo::cast(callback); | 989 Object* callback = result->GetCallbackObject(); |
| 991 return | 990 if (callback->IsAccessorInfo()) { |
| 992 (access_type == v8::ACCESS_HAS && | 991 AccessorInfo* info = AccessorInfo::cast(callback); |
| 993 (info->all_can_read() || info->all_can_write())) || | 992 bool can_access = |
| 994 (access_type == v8::ACCESS_GET && info->all_can_read()) || | 993 (access_type == v8::ACCESS_HAS && |
| 995 (access_type == v8::ACCESS_SET && info->all_can_write()); | 994 (info->all_can_read() || info->all_can_write())) || |
| 995 (access_type == v8::ACCESS_GET && info->all_can_read()) || |
| 996 (access_type == v8::ACCESS_SET && info->all_can_write()); |
| 997 return can_access; |
| 998 } |
| 996 } | 999 } |
| 1000 |
| 997 return false; | 1001 return false; |
| 998 } | 1002 } |
| 999 | 1003 |
| 1000 | 1004 |
| 1001 template<class Key> | 1005 static bool CheckAccess(JSObject* obj, |
| 1002 static bool CheckGenericAccess( | 1006 String* name, |
| 1003 JSObject* receiver, | 1007 LookupResult* result, |
| 1004 JSObject* holder, | 1008 v8::AccessType access_type) { |
| 1005 Key key, | 1009 ASSERT(result->IsProperty()); |
| 1006 v8::AccessType access_type, | 1010 |
| 1007 bool (Isolate::*mayAccess)(JSObject*, Key, v8::AccessType)) { | 1011 JSObject* holder = result->holder(); |
| 1008 Isolate* isolate = receiver->GetIsolate(); | 1012 JSObject* current = obj; |
| 1009 for (JSObject* current = receiver; | 1013 Isolate* isolate = obj->GetIsolate(); |
| 1010 true; | 1014 while (true) { |
| 1011 current = JSObject::cast(current->GetPrototype())) { | |
| 1012 if (current->IsAccessCheckNeeded() && | 1015 if (current->IsAccessCheckNeeded() && |
| 1013 !(isolate->*mayAccess)(current, key, access_type)) { | 1016 !isolate->MayNamedAccess(current, name, access_type)) { |
| 1014 return false; | 1017 // Access check callback denied the access, but some properties |
| 1018 // can have a special permissions which override callbacks descision |
| 1019 // (currently see v8::AccessControl). |
| 1020 break; |
| 1015 } | 1021 } |
| 1016 if (current == holder) break; | |
| 1017 } | |
| 1018 return true; | |
| 1019 } | |
| 1020 | 1022 |
| 1023 if (current == holder) { |
| 1024 return true; |
| 1025 } |
| 1021 | 1026 |
| 1022 static bool CheckElementAccess( | 1027 current = JSObject::cast(current->GetPrototype()); |
| 1023 JSObject* obj, | |
| 1024 uint32_t index, | |
| 1025 v8::AccessType access_type) { | |
| 1026 // TODO(1095): we should traverse hidden prototype hierachy as well. | |
| 1027 if (CheckGenericAccess( | |
| 1028 obj, obj, index, access_type, &Isolate::MayIndexedAccess)) { | |
| 1029 return true; | |
| 1030 } | 1028 } |
| 1031 | 1029 |
| 1032 obj->GetIsolate()->ReportFailedAccessCheck(obj, access_type); | 1030 // API callbacks can have per callback access exceptions. |
| 1031 switch (result->type()) { |
| 1032 case CALLBACKS: { |
| 1033 if (CheckAccessException(result, access_type)) { |
| 1034 return true; |
| 1035 } |
| 1036 break; |
| 1037 } |
| 1038 case INTERCEPTOR: { |
| 1039 // If the object has an interceptor, try real named properties. |
| 1040 // Overwrite the result to fetch the correct property later. |
| 1041 holder->LookupRealNamedProperty(name, result); |
| 1042 if (result->IsProperty()) { |
| 1043 if (CheckAccessException(result, access_type)) { |
| 1044 return true; |
| 1045 } |
| 1046 } |
| 1047 break; |
| 1048 } |
| 1049 default: |
| 1050 break; |
| 1051 } |
| 1052 |
| 1053 isolate->ReportFailedAccessCheck(current, access_type); |
| 1033 return false; | 1054 return false; |
| 1034 } | 1055 } |
| 1035 | 1056 |
| 1036 | 1057 |
| 1037 static bool CheckPropertyAccess( | 1058 // TODO(1095): we should traverse hidden prototype hierachy as well. |
| 1038 JSObject* obj, | 1059 static bool CheckElementAccess(JSObject* obj, |
| 1039 String* name, | 1060 uint32_t index, |
| 1040 v8::AccessType access_type) { | 1061 v8::AccessType access_type) { |
| 1041 uint32_t index; | 1062 if (obj->IsAccessCheckNeeded() && |
| 1042 if (name->AsArrayIndex(&index)) { | 1063 !obj->GetIsolate()->MayIndexedAccess(obj, index, access_type)) { |
| 1043 return CheckElementAccess(obj, index, access_type); | 1064 return false; |
| 1044 } | 1065 } |
| 1045 | 1066 |
| 1046 LookupResult lookup(obj->GetIsolate()); | 1067 return true; |
| 1047 obj->LocalLookup(name, &lookup); | |
| 1048 | |
| 1049 if (CheckGenericAccess<Object*>( | |
| 1050 obj, lookup.holder(), name, access_type, &Isolate::MayNamedAccess)) { | |
| 1051 return true; | |
| 1052 } | |
| 1053 | |
| 1054 // Access check callback denied the access, but some properties | |
| 1055 // can have a special permissions which override callbacks descision | |
| 1056 // (currently see v8::AccessControl). | |
| 1057 // API callbacks can have per callback access exceptions. | |
| 1058 switch (lookup.type()) { | |
| 1059 case CALLBACKS: | |
| 1060 if (CheckAccessException(lookup.GetCallbackObject(), access_type)) { | |
| 1061 return true; | |
| 1062 } | |
| 1063 break; | |
| 1064 case INTERCEPTOR: | |
| 1065 // If the object has an interceptor, try real named properties. | |
| 1066 // Overwrite the result to fetch the correct property later. | |
| 1067 lookup.holder()->LookupRealNamedProperty(name, &lookup); | |
| 1068 if (lookup.IsProperty() && lookup.IsPropertyCallbacks()) { | |
| 1069 if (CheckAccessException(lookup.GetCallbackObject(), access_type)) { | |
| 1070 return true; | |
| 1071 } | |
| 1072 } | |
| 1073 break; | |
| 1074 default: | |
| 1075 break; | |
| 1076 } | |
| 1077 | |
| 1078 obj->GetIsolate()->ReportFailedAccessCheck(obj, access_type); | |
| 1079 return false; | |
| 1080 } | 1068 } |
| 1081 | 1069 |
| 1082 | 1070 |
| 1083 // Enumerator used as indices into the array returned from GetOwnProperty | 1071 // Enumerator used as indices into the array returned from GetOwnProperty |
| 1084 enum PropertyDescriptorIndices { | 1072 enum PropertyDescriptorIndices { |
| 1085 IS_ACCESSOR_INDEX, | 1073 IS_ACCESSOR_INDEX, |
| 1086 VALUE_INDEX, | 1074 VALUE_INDEX, |
| 1087 GETTER_INDEX, | 1075 GETTER_INDEX, |
| 1088 SETTER_INDEX, | 1076 SETTER_INDEX, |
| 1089 WRITABLE_INDEX, | 1077 WRITABLE_INDEX, |
| 1090 ENUMERABLE_INDEX, | 1078 ENUMERABLE_INDEX, |
| 1091 CONFIGURABLE_INDEX, | 1079 CONFIGURABLE_INDEX, |
| 1092 DESCRIPTOR_SIZE | 1080 DESCRIPTOR_SIZE |
| 1093 }; | 1081 }; |
| 1094 | 1082 |
| 1095 | 1083 |
| 1096 static MaybeObject* GetOwnProperty(Isolate* isolate, | 1084 static MaybeObject* GetOwnProperty(Isolate* isolate, |
| 1097 Handle<JSObject> obj, | 1085 Handle<JSObject> obj, |
| 1098 Handle<String> name) { | 1086 Handle<String> name) { |
| 1099 Heap* heap = isolate->heap(); | 1087 Heap* heap = isolate->heap(); |
| 1100 PropertyAttributes attrs = obj->GetLocalPropertyAttribute(*name); | 1088 Handle<FixedArray> elms = isolate->factory()->NewFixedArray(DESCRIPTOR_SIZE); |
| 1101 if (attrs == ABSENT) return heap->undefined_value(); | 1089 Handle<JSArray> desc = isolate->factory()->NewJSArrayWithElements(elms); |
| 1102 AccessorPair* accessors = obj->GetLocalPropertyAccessorPair(*name); | 1090 LookupResult result(isolate); |
| 1091 // This could be an element. |
| 1092 uint32_t index; |
| 1093 if (name->AsArrayIndex(&index)) { |
| 1094 switch (obj->GetLocalElementKind(index)) { |
| 1095 case JSObject::UNDEFINED_ELEMENT: |
| 1096 return heap->undefined_value(); |
| 1103 | 1097 |
| 1104 Handle<FixedArray> elms = isolate->factory()->NewFixedArray(DESCRIPTOR_SIZE); | 1098 case JSObject::STRING_CHARACTER_ELEMENT: { |
| 1105 elms->set(ENUMERABLE_INDEX, heap->ToBoolean((attrs & DONT_ENUM) == 0)); | 1099 // Special handling of string objects according to ECMAScript 5 |
| 1106 elms->set(CONFIGURABLE_INDEX, heap->ToBoolean((attrs & DONT_DELETE) == 0)); | 1100 // 15.5.5.2. Note that this might be a string object with elements |
| 1107 elms->set(IS_ACCESSOR_INDEX, heap->ToBoolean(accessors != NULL)); | 1101 // other than the actual string value. This is covered by the |
| 1102 // subsequent cases. |
| 1103 Handle<JSValue> js_value = Handle<JSValue>::cast(obj); |
| 1104 Handle<String> str(String::cast(js_value->value())); |
| 1105 Handle<String> substr = SubString(str, index, index + 1, NOT_TENURED); |
| 1108 | 1106 |
| 1109 if (accessors == NULL) { | 1107 elms->set(IS_ACCESSOR_INDEX, heap->false_value()); |
| 1110 elms->set(WRITABLE_INDEX, heap->ToBoolean((attrs & READ_ONLY) == 0)); | 1108 elms->set(VALUE_INDEX, *substr); |
| 1111 // GetProperty does access check. | 1109 elms->set(WRITABLE_INDEX, heap->false_value()); |
| 1112 elms->set(VALUE_INDEX, *GetProperty(obj, name)); | 1110 elms->set(ENUMERABLE_INDEX, heap->true_value()); |
| 1113 } else { | 1111 elms->set(CONFIGURABLE_INDEX, heap->false_value()); |
| 1114 // Access checks are performed for both accessors separately. | 1112 return *desc; |
| 1115 // When they fail, the respective field is not set in the descriptor. | 1113 } |
| 1116 Object* getter = accessors->GetComponent(ACCESSOR_GETTER); | 1114 |
| 1117 Object* setter = accessors->GetComponent(ACCESSOR_SETTER); | 1115 case JSObject::INTERCEPTED_ELEMENT: |
| 1118 if (!getter->IsMap() && CheckPropertyAccess(*obj, *name, v8::ACCESS_GET)) { | 1116 case JSObject::FAST_ELEMENT: { |
| 1119 elms->set(GETTER_INDEX, getter); | 1117 elms->set(IS_ACCESSOR_INDEX, heap->false_value()); |
| 1120 } | 1118 Handle<Object> value = Object::GetElement(obj, index); |
| 1121 if (!setter->IsMap() && CheckPropertyAccess(*obj, *name, v8::ACCESS_SET)) { | 1119 RETURN_IF_EMPTY_HANDLE(isolate, value); |
| 1122 elms->set(SETTER_INDEX, setter); | 1120 elms->set(VALUE_INDEX, *value); |
| 1121 elms->set(WRITABLE_INDEX, heap->true_value()); |
| 1122 elms->set(ENUMERABLE_INDEX, heap->true_value()); |
| 1123 elms->set(CONFIGURABLE_INDEX, heap->true_value()); |
| 1124 return *desc; |
| 1125 } |
| 1126 |
| 1127 case JSObject::DICTIONARY_ELEMENT: { |
| 1128 Handle<JSObject> holder = obj; |
| 1129 if (obj->IsJSGlobalProxy()) { |
| 1130 Object* proto = obj->GetPrototype(); |
| 1131 if (proto->IsNull()) return heap->undefined_value(); |
| 1132 ASSERT(proto->IsJSGlobalObject()); |
| 1133 holder = Handle<JSObject>(JSObject::cast(proto)); |
| 1134 } |
| 1135 FixedArray* elements = FixedArray::cast(holder->elements()); |
| 1136 SeededNumberDictionary* dictionary = NULL; |
| 1137 if (elements->map() == heap->non_strict_arguments_elements_map()) { |
| 1138 dictionary = SeededNumberDictionary::cast(elements->get(1)); |
| 1139 } else { |
| 1140 dictionary = SeededNumberDictionary::cast(elements); |
| 1141 } |
| 1142 int entry = dictionary->FindEntry(index); |
| 1143 ASSERT(entry != SeededNumberDictionary::kNotFound); |
| 1144 PropertyDetails details = dictionary->DetailsAt(entry); |
| 1145 switch (details.type()) { |
| 1146 case CALLBACKS: { |
| 1147 // This is an accessor property with getter and/or setter. |
| 1148 AccessorPair* accessors = |
| 1149 AccessorPair::cast(dictionary->ValueAt(entry)); |
| 1150 elms->set(IS_ACCESSOR_INDEX, heap->true_value()); |
| 1151 if (CheckElementAccess(*obj, index, v8::ACCESS_GET)) { |
| 1152 elms->set(GETTER_INDEX, accessors->GetComponent(ACCESSOR_GETTER)); |
| 1153 } |
| 1154 if (CheckElementAccess(*obj, index, v8::ACCESS_SET)) { |
| 1155 elms->set(SETTER_INDEX, accessors->GetComponent(ACCESSOR_SETTER)); |
| 1156 } |
| 1157 break; |
| 1158 } |
| 1159 case NORMAL: { |
| 1160 // This is a data property. |
| 1161 elms->set(IS_ACCESSOR_INDEX, heap->false_value()); |
| 1162 Handle<Object> value = Object::GetElement(obj, index); |
| 1163 ASSERT(!value.is_null()); |
| 1164 elms->set(VALUE_INDEX, *value); |
| 1165 elms->set(WRITABLE_INDEX, heap->ToBoolean(!details.IsReadOnly())); |
| 1166 break; |
| 1167 } |
| 1168 default: |
| 1169 UNREACHABLE(); |
| 1170 break; |
| 1171 } |
| 1172 elms->set(ENUMERABLE_INDEX, heap->ToBoolean(!details.IsDontEnum())); |
| 1173 elms->set(CONFIGURABLE_INDEX, heap->ToBoolean(!details.IsDontDelete())); |
| 1174 return *desc; |
| 1175 } |
| 1123 } | 1176 } |
| 1124 } | 1177 } |
| 1125 | 1178 |
| 1126 return *isolate->factory()->NewJSArrayWithElements(elms); | 1179 // Use recursive implementation to also traverse hidden prototypes |
| 1180 GetOwnPropertyImplementation(*obj, *name, &result); |
| 1181 |
| 1182 if (!result.IsProperty()) { |
| 1183 return heap->undefined_value(); |
| 1184 } |
| 1185 |
| 1186 if (!CheckAccess(*obj, *name, &result, v8::ACCESS_HAS)) { |
| 1187 return heap->false_value(); |
| 1188 } |
| 1189 |
| 1190 elms->set(ENUMERABLE_INDEX, heap->ToBoolean(!result.IsDontEnum())); |
| 1191 elms->set(CONFIGURABLE_INDEX, heap->ToBoolean(!result.IsDontDelete())); |
| 1192 |
| 1193 bool is_js_accessor = result.IsPropertyCallbacks() && |
| 1194 (result.GetCallbackObject()->IsAccessorPair()); |
| 1195 |
| 1196 if (is_js_accessor) { |
| 1197 // __defineGetter__/__defineSetter__ callback. |
| 1198 elms->set(IS_ACCESSOR_INDEX, heap->true_value()); |
| 1199 |
| 1200 AccessorPair* accessors = AccessorPair::cast(result.GetCallbackObject()); |
| 1201 Object* getter = accessors->GetComponent(ACCESSOR_GETTER); |
| 1202 if (!getter->IsMap() && CheckAccess(*obj, *name, &result, v8::ACCESS_GET)) { |
| 1203 elms->set(GETTER_INDEX, getter); |
| 1204 } |
| 1205 Object* setter = accessors->GetComponent(ACCESSOR_SETTER); |
| 1206 if (!setter->IsMap() && CheckAccess(*obj, *name, &result, v8::ACCESS_SET)) { |
| 1207 elms->set(SETTER_INDEX, setter); |
| 1208 } |
| 1209 } else { |
| 1210 elms->set(IS_ACCESSOR_INDEX, heap->false_value()); |
| 1211 elms->set(WRITABLE_INDEX, heap->ToBoolean(!result.IsReadOnly())); |
| 1212 |
| 1213 PropertyAttributes attrs; |
| 1214 Object* value; |
| 1215 // GetProperty will check access and report any violations. |
| 1216 { MaybeObject* maybe_value = obj->GetProperty(*obj, &result, *name, &attrs); |
| 1217 if (!maybe_value->ToObject(&value)) return maybe_value; |
| 1218 } |
| 1219 elms->set(VALUE_INDEX, value); |
| 1220 } |
| 1221 |
| 1222 return *desc; |
| 1127 } | 1223 } |
| 1128 | 1224 |
| 1129 | 1225 |
| 1130 // Returns an array with the property description: | 1226 // Returns an array with the property description: |
| 1131 // if args[1] is not a property on args[0] | 1227 // if args[1] is not a property on args[0] |
| 1132 // returns undefined | 1228 // returns undefined |
| 1133 // if args[1] is a data property on args[0] | 1229 // if args[1] is a data property on args[0] |
| 1134 // [false, value, Writeable, Enumerable, Configurable] | 1230 // [false, value, Writeable, Enumerable, Configurable] |
| 1135 // if args[1] is an accessor on args[0] | 1231 // if args[1] is an accessor on args[0] |
| 1136 // [true, GetFunction, SetFunction, Enumerable, Configurable] | 1232 // [true, GetFunction, SetFunction, Enumerable, Configurable] |
| (...skipping 3487 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4624 } | 4720 } |
| 4625 | 4721 |
| 4626 | 4722 |
| 4627 RUNTIME_FUNCTION(MaybeObject*, Runtime_IsPropertyEnumerable) { | 4723 RUNTIME_FUNCTION(MaybeObject*, Runtime_IsPropertyEnumerable) { |
| 4628 NoHandleAllocation ha; | 4724 NoHandleAllocation ha; |
| 4629 ASSERT(args.length() == 2); | 4725 ASSERT(args.length() == 2); |
| 4630 | 4726 |
| 4631 CONVERT_ARG_CHECKED(JSObject, object, 0); | 4727 CONVERT_ARG_CHECKED(JSObject, object, 0); |
| 4632 CONVERT_ARG_CHECKED(String, key, 1); | 4728 CONVERT_ARG_CHECKED(String, key, 1); |
| 4633 | 4729 |
| 4730 uint32_t index; |
| 4731 if (key->AsArrayIndex(&index)) { |
| 4732 JSObject::LocalElementKind type = object->GetLocalElementKind(index); |
| 4733 switch (type) { |
| 4734 case JSObject::UNDEFINED_ELEMENT: |
| 4735 case JSObject::STRING_CHARACTER_ELEMENT: |
| 4736 return isolate->heap()->false_value(); |
| 4737 case JSObject::INTERCEPTED_ELEMENT: |
| 4738 case JSObject::FAST_ELEMENT: |
| 4739 return isolate->heap()->true_value(); |
| 4740 case JSObject::DICTIONARY_ELEMENT: { |
| 4741 if (object->IsJSGlobalProxy()) { |
| 4742 Object* proto = object->GetPrototype(); |
| 4743 if (proto->IsNull()) { |
| 4744 return isolate->heap()->false_value(); |
| 4745 } |
| 4746 ASSERT(proto->IsJSGlobalObject()); |
| 4747 object = JSObject::cast(proto); |
| 4748 } |
| 4749 FixedArray* elements = FixedArray::cast(object->elements()); |
| 4750 SeededNumberDictionary* dictionary = NULL; |
| 4751 if (elements->map() == |
| 4752 isolate->heap()->non_strict_arguments_elements_map()) { |
| 4753 dictionary = SeededNumberDictionary::cast(elements->get(1)); |
| 4754 } else { |
| 4755 dictionary = SeededNumberDictionary::cast(elements); |
| 4756 } |
| 4757 int entry = dictionary->FindEntry(index); |
| 4758 ASSERT(entry != SeededNumberDictionary::kNotFound); |
| 4759 PropertyDetails details = dictionary->DetailsAt(entry); |
| 4760 return isolate->heap()->ToBoolean(!details.IsDontEnum()); |
| 4761 } |
| 4762 } |
| 4763 } |
| 4764 |
| 4634 PropertyAttributes att = object->GetLocalPropertyAttribute(key); | 4765 PropertyAttributes att = object->GetLocalPropertyAttribute(key); |
| 4635 return isolate->heap()->ToBoolean(att != ABSENT && (att & DONT_ENUM) == 0); | 4766 return isolate->heap()->ToBoolean(att != ABSENT && (att & DONT_ENUM) == 0); |
| 4636 } | 4767 } |
| 4637 | 4768 |
| 4638 | 4769 |
| 4639 RUNTIME_FUNCTION(MaybeObject*, Runtime_GetPropertyNames) { | 4770 RUNTIME_FUNCTION(MaybeObject*, Runtime_GetPropertyNames) { |
| 4640 HandleScope scope(isolate); | 4771 HandleScope scope(isolate); |
| 4641 ASSERT(args.length() == 1); | 4772 ASSERT(args.length() == 1); |
| 4642 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, object, 0); | 4773 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, object, 0); |
| 4643 bool threw = false; | 4774 bool threw = false; |
| (...skipping 8673 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 13317 // Handle last resort GC and make sure to allow future allocations | 13448 // Handle last resort GC and make sure to allow future allocations |
| 13318 // to grow the heap without causing GCs (if possible). | 13449 // to grow the heap without causing GCs (if possible). |
| 13319 isolate->counters()->gc_last_resort_from_js()->Increment(); | 13450 isolate->counters()->gc_last_resort_from_js()->Increment(); |
| 13320 isolate->heap()->CollectAllGarbage(Heap::kNoGCFlags, | 13451 isolate->heap()->CollectAllGarbage(Heap::kNoGCFlags, |
| 13321 "Runtime::PerformGC"); | 13452 "Runtime::PerformGC"); |
| 13322 } | 13453 } |
| 13323 } | 13454 } |
| 13324 | 13455 |
| 13325 | 13456 |
| 13326 } } // namespace v8::internal | 13457 } } // namespace v8::internal |
| OLD | NEW |