Chromium Code Reviews

Side by Side Diff: src/objects.cc

Issue 77293003: Addressed perf regression in Browsermark2.0 array blur test (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Reduced 10% to 3% degrade, removed special case. Created 7 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View unified diff | | Annotate | Revision Log
« src/objects.h ('K') | « src/objects.h ('k') | src/runtime.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2013 the V8 project authors. All rights reserved. 1 // Copyright 2013 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 5637 matching lines...)
5648 } else if (object->map()->CanHaveMoreTransitions()) { 5648 } else if (object->map()->CanHaveMoreTransitions()) {
5649 new_map = Map::CopyForObserved(handle(object->map())); 5649 new_map = Map::CopyForObserved(handle(object->map()));
5650 } else { 5650 } else {
5651 new_map = Map::Copy(handle(object->map())); 5651 new_map = Map::Copy(handle(object->map()));
5652 new_map->set_is_observed(); 5652 new_map->set_is_observed();
5653 } 5653 }
5654 object->set_map(*new_map); 5654 object->set_map(*new_map);
5655 } 5655 }
5656 5656
5657 5657
5658 Handle<JSObject> JSObject::Copy(Handle<JSObject> object,
5659 Handle<AllocationSite> site) {
5660 Isolate* isolate = object->GetIsolate();
5661 CALL_HEAP_FUNCTION(isolate,
5662 isolate->heap()->CopyJSObject(*object, *site), JSObject);
5663 }
5664
5665
5666 Handle<JSObject> JSObject::Copy(Handle<JSObject> object) { 5658 Handle<JSObject> JSObject::Copy(Handle<JSObject> object) {
5667 Isolate* isolate = object->GetIsolate(); 5659 Isolate* isolate = object->GetIsolate();
5668 CALL_HEAP_FUNCTION(isolate, 5660 CALL_HEAP_FUNCTION(isolate,
5669 isolate->heap()->CopyJSObject(*object), JSObject); 5661 isolate->heap()->CopyJSObject(*object), JSObject);
5670 } 5662 }
5671 5663
5672 5664
5665 template<class ContextObject>
5673 class JSObjectWalkVisitor { 5666 class JSObjectWalkVisitor {
5674 public: 5667 public:
5675 explicit JSObjectWalkVisitor(AllocationSiteContext* site_context) : 5668 JSObjectWalkVisitor(ContextObject* site_context, bool copying) :
5676 site_context_(site_context) {} 5669 site_context_(site_context),
5677 virtual ~JSObjectWalkVisitor() {} 5670 copying_(copying) {}
5678 5671
5679 Handle<JSObject> Visit(Handle<JSObject> object) { 5672 Handle<JSObject> StructureWalk(Handle<JSObject> object,
5680 return StructureWalk(object); 5673 bool object_is_shallow_array = false);
5681 }
5682
5683 virtual bool is_copying() = 0;
5684 5674
5685 protected: 5675 protected:
5686 Handle<JSObject> StructureWalk(Handle<JSObject> object); 5676 inline Handle<JSObject> VisitElementOrProperty(Handle<JSObject> object,
5687 5677 Handle<JSObject> value) {
5688 // The returned handle will be used for the object in all subsequent usages. 5678 Handle<AllocationSite> current_site = site_context()->EnterNewScope();
5689 // This allows VisitObject to make a copy of the object if desired. 5679 Handle<JSObject> copy_of_value = StructureWalk(value);
5690 virtual Handle<JSObject> VisitObject(Handle<JSObject> object) = 0; 5680 site_context()->ExitScope(current_site, value);
5691 virtual Handle<JSObject> VisitElementOrProperty(Handle<JSObject> object, 5681 return copy_of_value;
5692 Handle<JSObject> value) = 0; 5682 }
5693 5683
5694 AllocationSiteContext* site_context() { return site_context_; } 5684 inline ContextObject* site_context() { return site_context_; }
5685 inline Isolate* isolate() { return site_context()->isolate(); }
5686
5687 inline bool copying() const { return copying_; }
5695 5688
5696 private: 5689 private:
5697 AllocationSiteContext* site_context_; 5690 ContextObject* site_context_;
5691 const bool copying_;
5698 }; 5692 };
5699 5693
5700 5694
5701 class JSObjectCopyVisitor: public JSObjectWalkVisitor { 5695 template <class ContextObject>
5702 public: 5696 Handle<JSObject> JSObjectWalkVisitor<ContextObject>::StructureWalk(
5703 explicit JSObjectCopyVisitor(AllocationSiteContext* site_context) 5697 Handle<JSObject> object, bool object_is_shallow_array) {
5704 : JSObjectWalkVisitor(site_context) {} 5698 Isolate* isolate = this->isolate();
5705 5699 bool copying = this->copying();
5706 virtual bool is_copying() V8_OVERRIDE { return true; } 5700 bool shallow = object_is_shallow_array;
5707 5701
5708 // The returned handle will be used for the object in all 5702 if (!shallow) {
5709 // subsequent usages. This allows VisitObject to make a copy 5703 StackLimitCheck check(isolate);
5710 // of the object if desired. 5704
5711 virtual Handle<JSObject> VisitObject(Handle<JSObject> object) V8_OVERRIDE { 5705 if (check.HasOverflowed()) {
5712 // Only create a memento if 5706 isolate->StackOverflow();
5713 // 1) we have a JSArray, and 5707 return Handle<JSObject>::null();
5714 // 2) the elements kind is palatable 5708 }
5715 // 3) allow_mementos is true 5709
5716 Handle<JSObject> copy; 5710 if (object->map()->is_deprecated()) {
5711 JSObject::MigrateInstance(object);
5712 }
5713 }
5714
5715 Handle<JSObject> copy;
5716 if (copying) {
5717 Handle<AllocationSite> site_to_pass;
5717 if (site_context()->activated() && 5718 if (site_context()->activated() &&
5718 AllocationSite::CanTrack(object->map()->instance_type()) && 5719 AllocationSite::CanTrack(object->map()->instance_type()) &&
5719 AllocationSite::GetMode(object->GetElementsKind()) == 5720 AllocationSite::GetMode(object->GetElementsKind()) ==
5720 TRACK_ALLOCATION_SITE) { 5721 TRACK_ALLOCATION_SITE) {
5721 copy = JSObject::Copy(object, site_context()->current()); 5722 site_to_pass = site_context()->current();
5723 }
5724 CALL_AND_RETRY_OR_DIE(isolate,
5725 isolate->heap()->CopyJSObject(*object,
5726 site_to_pass.is_null() ? NULL : *site_to_pass),
5727 { copy = Handle<JSObject>(JSObject::cast(__object__),
5728 isolate);
5729 break;
5730 },
5731 return Handle<JSObject>());
5732 } else {
5733 copy = object;
5734 }
5735
5736 ASSERT(copying || copy.is_identical_to(object));
5737
5738 if (!shallow) {
5739 HandleScope scope(isolate);
5740
5741 // Deep copy local properties.
5742 if (copy->HasFastProperties()) {
5743 Handle<DescriptorArray> descriptors(copy->map()->instance_descriptors());
5744 int limit = copy->map()->NumberOfOwnDescriptors();
5745 for (int i = 0; i < limit; i++) {
5746 PropertyDetails details = descriptors->GetDetails(i);
5747 if (details.type() != FIELD) continue;
5748 int index = descriptors->GetFieldIndex(i);
5749 Handle<Object> value(object->RawFastPropertyAt(index), isolate);
5750 if (value->IsJSObject()) {
5751 value = VisitElementOrProperty(copy, Handle<JSObject>::cast(value));
5752 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, value, Handle<JSObject>());
5753 } else {
5754 Representation representation = details.representation();
5755 value = NewStorageFor(isolate, value, representation);
5756 }
5757 if (copying) {
5758 copy->FastPropertyAtPut(index, *value);
5759 }
5760 }
5722 } else { 5761 } else {
5723 copy = JSObject::Copy(object); 5762 Handle<FixedArray> names =
5724 } 5763 isolate->factory()->NewFixedArray(copy->NumberOfLocalProperties());
5725 5764 copy->GetLocalPropertyNames(*names, 0);
5726 return copy; 5765 for (int i = 0; i < names->length(); i++) {
5727 } 5766 ASSERT(names->get(i)->IsString());
5728 5767 Handle<String> key_string(String::cast(names->get(i)));
5729 virtual Handle<JSObject> VisitElementOrProperty( 5768 PropertyAttributes attributes =
5730 Handle<JSObject> object, 5769 copy->GetLocalPropertyAttribute(*key_string);
5731 Handle<JSObject> value) V8_OVERRIDE { 5770 // Only deep copy fields from the object literal expression.
5732 Handle<AllocationSite> current_site = site_context()->EnterNewScope(); 5771 // In particular, don't try to copy the length attribute of
5733 Handle<JSObject> copy_of_value = StructureWalk(value); 5772 // an array.
5734 site_context()->ExitScope(current_site, value); 5773 if (attributes != NONE) continue;
5735 return copy_of_value; 5774 Handle<Object> value(
5736 } 5775 copy->GetProperty(*key_string, &attributes)->ToObjectUnchecked(),
5737 }; 5776 isolate);
5738 5777 if (value->IsJSObject()) {
5739 5778 Handle<JSObject> result = VisitElementOrProperty(
5740 class JSObjectCreateAllocationSitesVisitor: public JSObjectWalkVisitor { 5779 copy, Handle<JSObject>::cast(value));
5741 public: 5780 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>());
5742 explicit JSObjectCreateAllocationSitesVisitor( 5781 if (copying) {
5743 AllocationSiteContext* site_context) 5782 // Creating object copy for literals. No strict mode needed.
5744 : JSObjectWalkVisitor(site_context) {} 5783 CHECK_NOT_EMPTY_HANDLE(isolate, JSObject::SetProperty(
5745 5784 copy, key_string, result, NONE, kNonStrictMode));
5746 virtual bool is_copying() V8_OVERRIDE { return false; } 5785 }
5747 5786 }
5748 // The returned handle will be used for the object in all 5787 }
5749 // subsequent usages. This allows VisitObject to make a copy 5788 }
5750 // of the object if desired. 5789
5751 virtual Handle<JSObject> VisitObject(Handle<JSObject> object) V8_OVERRIDE { 5790 // Deep copy local elements.
5752 return object; 5791 // Pixel elements cannot be created using an object literal.
5753 } 5792 ASSERT(!copy->HasExternalArrayElements());
5754 5793 switch (copy->GetElementsKind()) {
5755 virtual Handle<JSObject> VisitElementOrProperty( 5794 case FAST_SMI_ELEMENTS:
5756 Handle<JSObject> object, 5795 case FAST_ELEMENTS:
5757 Handle<JSObject> value) V8_OVERRIDE { 5796 case FAST_HOLEY_SMI_ELEMENTS:
5758 Handle<AllocationSite> current_site = site_context()->EnterNewScope(); 5797 case FAST_HOLEY_ELEMENTS: {
5759 value = StructureWalk(value); 5798 Handle<FixedArray> elements(FixedArray::cast(copy->elements()));
5760 site_context()->ExitScope(current_site, value); 5799 if (elements->map() == isolate->heap()->fixed_cow_array_map()) {
5761 return value; 5800 if (copying) {
5762 } 5801 isolate->counters()->cow_arrays_created_runtime()->Increment();
5763 }; 5802 }
5764
5765
5766 Handle<JSObject> JSObjectWalkVisitor::StructureWalk(Handle<JSObject> object) {
5767 bool copying = is_copying();
5768 Isolate* isolate = object->GetIsolate();
5769 StackLimitCheck check(isolate);
5770 if (check.HasOverflowed()) {
5771 isolate->StackOverflow();
5772 return Handle<JSObject>::null();
5773 }
5774
5775 if (object->map()->is_deprecated()) {
5776 JSObject::MigrateInstance(object);
5777 }
5778
5779 Handle<JSObject> copy = VisitObject(object);
5780 ASSERT(copying || copy.is_identical_to(object));
5781
5782 HandleScope scope(isolate);
5783
5784 // Deep copy local properties.
5785 if (copy->HasFastProperties()) {
5786 Handle<DescriptorArray> descriptors(copy->map()->instance_descriptors());
5787 int limit = copy->map()->NumberOfOwnDescriptors();
5788 for (int i = 0; i < limit; i++) {
5789 PropertyDetails details = descriptors->GetDetails(i);
5790 if (details.type() != FIELD) continue;
5791 int index = descriptors->GetFieldIndex(i);
5792 Handle<Object> value(object->RawFastPropertyAt(index), isolate);
5793 if (value->IsJSObject()) {
5794 value = VisitElementOrProperty(copy, Handle<JSObject>::cast(value));
5795 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, value, Handle<JSObject>());
5796 } else {
5797 Representation representation = details.representation();
5798 value = NewStorageFor(isolate, value, representation);
5799 }
5800 if (copying) {
5801 copy->FastPropertyAtPut(index, *value);
5802 }
5803 }
5804 } else {
5805 Handle<FixedArray> names =
5806 isolate->factory()->NewFixedArray(copy->NumberOfLocalProperties());
5807 copy->GetLocalPropertyNames(*names, 0);
5808 for (int i = 0; i < names->length(); i++) {
5809 ASSERT(names->get(i)->IsString());
5810 Handle<String> key_string(String::cast(names->get(i)));
5811 PropertyAttributes attributes =
5812 copy->GetLocalPropertyAttribute(*key_string);
5813 // Only deep copy fields from the object literal expression.
5814 // In particular, don't try to copy the length attribute of
5815 // an array.
5816 if (attributes != NONE) continue;
5817 Handle<Object> value(
5818 copy->GetProperty(*key_string, &attributes)->ToObjectUnchecked(),
5819 isolate);
5820 if (value->IsJSObject()) {
5821 Handle<JSObject> result = VisitElementOrProperty(
5822 copy, Handle<JSObject>::cast(value));
5823 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>());
5824 if (copying) {
5825 // Creating object copy for literals. No strict mode needed.
5826 CHECK_NOT_EMPTY_HANDLE(isolate, JSObject::SetProperty(
5827 copy, key_string, result, NONE, kNonStrictMode));
5828 }
5829 }
5830 }
5831 }
5832
5833 // Deep copy local elements.
5834 // Pixel elements cannot be created using an object literal.
5835 ASSERT(!copy->HasExternalArrayElements());
5836 switch (copy->GetElementsKind()) {
5837 case FAST_SMI_ELEMENTS:
5838 case FAST_ELEMENTS:
5839 case FAST_HOLEY_SMI_ELEMENTS:
5840 case FAST_HOLEY_ELEMENTS: {
5841 Handle<FixedArray> elements(FixedArray::cast(copy->elements()));
5842 if (elements->map() == isolate->heap()->fixed_cow_array_map()) {
5843 if (copying) {
5844 isolate->counters()->cow_arrays_created_runtime()->Increment();
5845 }
5846 #ifdef DEBUG 5803 #ifdef DEBUG
5847 for (int i = 0; i < elements->length(); i++) { 5804 for (int i = 0; i < elements->length(); i++) {
5848 ASSERT(!elements->get(i)->IsJSObject()); 5805 ASSERT(!elements->get(i)->IsJSObject());
5849 } 5806 }
5850 #endif 5807 #endif
5851 } else { 5808 } else {
5852 for (int i = 0; i < elements->length(); i++) { 5809 for (int i = 0; i < elements->length(); i++) {
5853 Handle<Object> value(elements->get(i), isolate); 5810 Handle<Object> value(elements->get(i), isolate);
5854 ASSERT(value->IsSmi() || 5811 ASSERT(value->IsSmi() ||
5855 value->IsTheHole() || 5812 value->IsTheHole() ||
5856 (IsFastObjectElementsKind(copy->GetElementsKind()))); 5813 (IsFastObjectElementsKind(copy->GetElementsKind())));
5857 if (value->IsJSObject()) { 5814 if (value->IsJSObject()) {
5858 Handle<JSObject> result = VisitElementOrProperty( 5815 Handle<JSObject> result = VisitElementOrProperty(
5859 copy, Handle<JSObject>::cast(value)); 5816 copy, Handle<JSObject>::cast(value));
5860 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>()); 5817 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>());
5861 if (copying) { 5818 if (copying) {
5862 elements->set(i, *result); 5819 elements->set(i, *result);
5820 }
5863 } 5821 }
5864 } 5822 }
5865 } 5823 }
5866 } 5824 break;
5867 break; 5825 }
5868 } 5826 case DICTIONARY_ELEMENTS: {
5869 case DICTIONARY_ELEMENTS: { 5827 Handle<SeededNumberDictionary> element_dictionary(
5870 Handle<SeededNumberDictionary> element_dictionary( 5828 copy->element_dictionary());
5871 copy->element_dictionary()); 5829 int capacity = element_dictionary->Capacity();
5872 int capacity = element_dictionary->Capacity(); 5830 for (int i = 0; i < capacity; i++) {
5873 for (int i = 0; i < capacity; i++) { 5831 Object* k = element_dictionary->KeyAt(i);
5874 Object* k = element_dictionary->KeyAt(i); 5832 if (element_dictionary->IsKey(k)) {
5875 if (element_dictionary->IsKey(k)) { 5833 Handle<Object> value(element_dictionary->ValueAt(i), isolate);
5876 Handle<Object> value(element_dictionary->ValueAt(i), isolate); 5834 if (value->IsJSObject()) {
5877 if (value->IsJSObject()) { 5835 Handle<JSObject> result = VisitElementOrProperty(
5878 Handle<JSObject> result = VisitElementOrProperty( 5836 copy, Handle<JSObject>::cast(value));
5879 copy, Handle<JSObject>::cast(value)); 5837 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>());
5880 RETURN_IF_EMPTY_HANDLE_VALUE(isolate, result, Handle<JSObject>()); 5838 if (copying) {
5881 if (copying) { 5839 element_dictionary->ValueAtPut(i, *result);
5882 element_dictionary->ValueAtPut(i, *result); 5840 }
5883 } 5841 }
5884 } 5842 }
5885 } 5843 }
5886 } 5844 break;
5887 break; 5845 }
5888 } 5846 case NON_STRICT_ARGUMENTS_ELEMENTS:
5889 case NON_STRICT_ARGUMENTS_ELEMENTS: 5847 UNIMPLEMENTED();
5890 UNIMPLEMENTED(); 5848 break;
5891 break; 5849 case EXTERNAL_PIXEL_ELEMENTS:
5892 case EXTERNAL_PIXEL_ELEMENTS: 5850 case EXTERNAL_BYTE_ELEMENTS:
5893 case EXTERNAL_BYTE_ELEMENTS: 5851 case EXTERNAL_UNSIGNED_BYTE_ELEMENTS:
5894 case EXTERNAL_UNSIGNED_BYTE_ELEMENTS: 5852 case EXTERNAL_SHORT_ELEMENTS:
5895 case EXTERNAL_SHORT_ELEMENTS: 5853 case EXTERNAL_UNSIGNED_SHORT_ELEMENTS:
5896 case EXTERNAL_UNSIGNED_SHORT_ELEMENTS: 5854 case EXTERNAL_INT_ELEMENTS:
5897 case EXTERNAL_INT_ELEMENTS: 5855 case EXTERNAL_UNSIGNED_INT_ELEMENTS:
5898 case EXTERNAL_UNSIGNED_INT_ELEMENTS: 5856 case EXTERNAL_FLOAT_ELEMENTS:
5899 case EXTERNAL_FLOAT_ELEMENTS: 5857 case EXTERNAL_DOUBLE_ELEMENTS:
5900 case EXTERNAL_DOUBLE_ELEMENTS: 5858 case FAST_DOUBLE_ELEMENTS:
5901 case FAST_DOUBLE_ELEMENTS: 5859 case FAST_HOLEY_DOUBLE_ELEMENTS:
5902 case FAST_HOLEY_DOUBLE_ELEMENTS: 5860 // No contained objects, nothing to do.
5903 // No contained objects, nothing to do. 5861 break;
5904 break; 5862 }
5905 } 5863 }
5864
5906 return copy; 5865 return copy;
5907 } 5866 }
5908 5867
5909 5868
5910 Handle<JSObject> JSObject::DeepWalk(Handle<JSObject> object, 5869 Handle<JSObject> JSObject::DeepWalk(
5911 AllocationSiteContext* site_context) { 5870 Handle<JSObject> object,
5912 JSObjectCreateAllocationSitesVisitor v(site_context); 5871 AllocationSiteCreationContext* site_context) {
5913 Handle<JSObject> result = v.Visit(object); 5872 JSObjectWalkVisitor<AllocationSiteCreationContext> v(site_context, false);
5914 ASSERT(!v.is_copying() && 5873 Handle<JSObject> result = v.StructureWalk(object);
5915 (result.is_null() || result.is_identical_to(object))); 5874 ASSERT(result.is_null() || result.is_identical_to(object));
5916 return result; 5875 return result;
5917 } 5876 }
5918 5877
5919 5878
5920 Handle<JSObject> JSObject::DeepCopy(Handle<JSObject> object, 5879 Handle<JSObject> JSObject::DeepCopy(Handle<JSObject> object,
5921 AllocationSiteContext* site_context) { 5880 AllocationSiteUsageContext* site_context,
5922 JSObjectCopyVisitor v(site_context); 5881 bool hint_object_is_shallow_array) {
5923 Handle<JSObject> copy = v.Visit(object); 5882 JSObjectWalkVisitor<AllocationSiteUsageContext> v(site_context, true);
5924 ASSERT(v.is_copying() && !copy.is_identical_to(object)); 5883 Handle<JSObject> copy = v.StructureWalk(object, hint_object_is_shallow_array);
5884 ASSERT(!copy.is_identical_to(object));
5925 return copy; 5885 return copy;
5926 } 5886 }
5927 5887
5928 5888
5929 // Tests for the fast common case for property enumeration: 5889 // Tests for the fast common case for property enumeration:
5930 // - This object and all prototypes has an enum cache (which means that 5890 // - This object and all prototypes has an enum cache (which means that
5931 // it is no proxy, has no interceptors and needs no access checks). 5891 // it is no proxy, has no interceptors and needs no access checks).
5932 // - This object has no elements. 5892 // - This object has no elements.
5933 // - No prototype has enumerable properties/elements. 5893 // - No prototype has enumerable properties/elements.
5934 bool JSReceiver::IsSimpleEnum() { 5894 bool JSReceiver::IsSimpleEnum() {
(...skipping 10733 matching lines...)
16668 #define ERROR_MESSAGES_TEXTS(C, T) T, 16628 #define ERROR_MESSAGES_TEXTS(C, T) T,
16669 static const char* error_messages_[] = { 16629 static const char* error_messages_[] = {
16670 ERROR_MESSAGES_LIST(ERROR_MESSAGES_TEXTS) 16630 ERROR_MESSAGES_LIST(ERROR_MESSAGES_TEXTS)
16671 }; 16631 };
16672 #undef ERROR_MESSAGES_TEXTS 16632 #undef ERROR_MESSAGES_TEXTS
16673 return error_messages_[reason]; 16633 return error_messages_[reason];
16674 } 16634 }
16675 16635
16676 16636
16677 } } // namespace v8::internal 16637 } } // namespace v8::internal
OLDNEW
« src/objects.h ('K') | « src/objects.h ('k') | src/runtime.h » ('j') | no next file with comments »

Powered by Google App Engine