| // Copyright (c) 2016, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| #if !defined(DART_PRECOMPILED_RUNTIME) |
| |
| #include "vm/kernel.h" |
| |
| #include "vm/bit_vector.h" |
| #include "vm/compiler/frontend/constant_reader.h" |
| #include "vm/compiler/frontend/kernel_translation_helper.h" |
| #include "vm/compiler/jit/compiler.h" |
| #include "vm/longjump.h" |
| #include "vm/object_store.h" |
| #include "vm/parser.h" // For Parser::kParameter* constants. |
| #include "vm/stack_frame.h" |
| |
| namespace dart { |
| namespace kernel { |
| |
| class KernelTokenPositionCollector : public KernelReaderHelper { |
| public: |
| KernelTokenPositionCollector( |
| Zone* zone, |
| TranslationHelper* translation_helper, |
| const Script& script, |
| const TypedDataView& data, |
| intptr_t data_program_offset, |
| intptr_t initial_script_index, |
| intptr_t record_for_script_id, |
| GrowableArray<intptr_t>* record_token_positions_into) |
| : KernelReaderHelper(zone, translation_helper, data, data_program_offset), |
| current_script_id_(initial_script_index), |
| record_for_script_id_(record_for_script_id), |
| record_token_positions_into_(record_token_positions_into) {} |
| |
| void CollectTokenPositions(intptr_t kernel_offset); |
| |
| void RecordTokenPosition(TokenPosition position) override; |
| |
| void set_current_script_id(intptr_t id) override { current_script_id_ = id; } |
| |
| private: |
| intptr_t current_script_id_; |
| intptr_t record_for_script_id_; |
| GrowableArray<intptr_t>* record_token_positions_into_; |
| |
| DISALLOW_COPY_AND_ASSIGN(KernelTokenPositionCollector); |
| }; |
| |
| void KernelTokenPositionCollector::CollectTokenPositions( |
| intptr_t kernel_offset) { |
| SetOffset(kernel_offset); |
| |
| const Tag tag = PeekTag(); |
| if (tag == kProcedure) { |
| ProcedureHelper procedure_helper(this); |
| procedure_helper.ReadUntilExcluding(ProcedureHelper::kEnd); |
| } else if (tag == kConstructor) { |
| ConstructorHelper constructor_helper(this); |
| constructor_helper.ReadUntilExcluding(ConstructorHelper::kEnd); |
| } else if (tag == kFunctionNode) { |
| FunctionNodeHelper function_node_helper(this); |
| function_node_helper.ReadUntilExcluding(FunctionNodeHelper::kEnd); |
| } else if (tag == kField) { |
| FieldHelper field_helper(this); |
| field_helper.ReadUntilExcluding(FieldHelper::kEnd); |
| } else if (tag == kClass) { |
| ClassHelper class_helper(this); |
| class_helper.ReadUntilExcluding(ClassHelper::kEnd); |
| } else { |
| ReportUnexpectedTag("a class or a member", tag); |
| UNREACHABLE(); |
| } |
| } |
| |
| void KernelTokenPositionCollector::RecordTokenPosition(TokenPosition position) { |
| if (record_for_script_id_ == current_script_id_ && |
| record_token_positions_into_ != nullptr && position.IsReal()) { |
| record_token_positions_into_->Add(position.Serialize()); |
| } |
| } |
| |
| static void CollectKernelLibraryTokenPositions( |
| const TypedDataView& kernel_data, |
| const Script& script, |
| intptr_t kernel_offset, |
| intptr_t data_kernel_offset, |
| Zone* zone, |
| TranslationHelper* helper, |
| GrowableArray<intptr_t>* token_positions) { |
| if (kernel_data.IsNull()) { |
| return; |
| } |
| |
| KernelTokenPositionCollector token_position_collector( |
| zone, helper, script, kernel_data, data_kernel_offset, |
| script.kernel_script_index(), script.kernel_script_index(), |
| token_positions); |
| |
| token_position_collector.CollectTokenPositions(kernel_offset); |
| } |
| |
| void CollectScriptTokenPositionsFromKernel( |
| const Script& interesting_script, |
| GrowableArray<intptr_t>* token_positions) { |
| Thread* thread = Thread::Current(); |
| Zone* zone = thread->zone(); |
| |
| const auto& kernel_info = |
| KernelProgramInfo::Handle(zone, interesting_script.kernel_program_info()); |
| |
| TranslationHelper helper(thread); |
| helper.InitFromKernelProgramInfo(kernel_info); |
| |
| auto isolate_group = thread->isolate_group(); |
| const GrowableObjectArray& libs = GrowableObjectArray::Handle( |
| zone, isolate_group->object_store()->libraries()); |
| Library& lib = Library::Handle(zone); |
| Object& entry = Object::Handle(zone); |
| Script& entry_script = Script::Handle(zone); |
| auto& data = TypedDataView::Handle(zone); |
| |
| auto& temp_array = Array::Handle(zone); |
| auto& temp_field = Field::Handle(zone); |
| auto& temp_function = Function::Handle(zone); |
| for (intptr_t i = 0; i < libs.Length(); i++) { |
| lib ^= libs.At(i); |
| lib.EnsureTopLevelClassIsFinalized(); |
| DictionaryIterator it(lib); |
| while (it.HasNext()) { |
| entry = it.GetNext(); |
| data = TypedDataView::null(); |
| if (entry.IsClass()) { |
| const Class& klass = Class::Cast(entry); |
| if (klass.script() == interesting_script.ptr()) { |
| token_positions->Add(klass.token_pos().Serialize()); |
| token_positions->Add(klass.end_token_pos().Serialize()); |
| } |
| if (klass.is_finalized()) { |
| temp_array = klass.fields(); |
| for (intptr_t i = 0; i < temp_array.Length(); ++i) { |
| temp_field ^= temp_array.At(i); |
| if (temp_field.kernel_offset() <= 0) { |
| // Skip artificially injected fields. |
| continue; |
| } |
| entry_script = temp_field.Script(); |
| if (entry_script.ptr() != interesting_script.ptr()) { |
| continue; |
| } |
| data = temp_field.KernelLibrary(); |
| CollectKernelLibraryTokenPositions(data, interesting_script, |
| temp_field.kernel_offset(), |
| temp_field.KernelLibraryOffset(), |
| zone, &helper, token_positions); |
| } |
| temp_array = klass.current_functions(); |
| for (intptr_t i = 0; i < temp_array.Length(); ++i) { |
| temp_function ^= temp_array.At(i); |
| entry_script = temp_function.script(); |
| if (entry_script.ptr() != interesting_script.ptr()) { |
| continue; |
| } |
| data = temp_function.KernelLibrary(); |
| CollectKernelLibraryTokenPositions( |
| data, interesting_script, temp_function.kernel_offset(), |
| temp_function.KernelLibraryOffset(), zone, &helper, |
| token_positions); |
| } |
| } else { |
| // Class isn't finalized yet: read the data attached to it. |
| ASSERT(klass.kernel_offset() > 0); |
| data = lib.KernelLibrary(); |
| ASSERT(!data.IsNull()); |
| const intptr_t library_kernel_offset = lib.KernelLibraryOffset(); |
| ASSERT(library_kernel_offset > 0); |
| const intptr_t class_offset = klass.kernel_offset(); |
| |
| entry_script = klass.script(); |
| if (entry_script.ptr() != interesting_script.ptr()) { |
| continue; |
| } |
| CollectKernelLibraryTokenPositions( |
| data, interesting_script, class_offset, library_kernel_offset, |
| zone, &helper, token_positions); |
| } |
| } else if (entry.IsFunction()) { |
| temp_function ^= entry.ptr(); |
| entry_script = temp_function.script(); |
| if (entry_script.ptr() != interesting_script.ptr()) { |
| continue; |
| } |
| data = temp_function.KernelLibrary(); |
| CollectKernelLibraryTokenPositions(data, interesting_script, |
| temp_function.kernel_offset(), |
| temp_function.KernelLibraryOffset(), |
| zone, &helper, token_positions); |
| } else if (entry.IsField()) { |
| const Field& field = Field::Cast(entry); |
| if (field.kernel_offset() <= 0) { |
| // Skip artificially injected fields. |
| continue; |
| } |
| entry_script = field.Script(); |
| if (entry_script.ptr() != interesting_script.ptr()) { |
| continue; |
| } |
| data = field.KernelLibrary(); |
| CollectKernelLibraryTokenPositions( |
| data, interesting_script, field.kernel_offset(), |
| field.KernelLibraryOffset(), zone, &helper, token_positions); |
| } |
| } |
| } |
| } |
| |
| } // namespace kernel |
| |
| #if !defined(PRODUCT) && !defined(DART_PRECOMPILED_RUNTIME) |
| ArrayPtr Script::CollectConstConstructorCoverageFrom() const { |
| Thread* thread = Thread::Current(); |
| Zone* zone = thread->zone(); |
| kernel::TranslationHelper helper(thread); |
| |
| const auto& interesting_script = *this; |
| |
| const auto& kernel_info = |
| KernelProgramInfo::Handle(zone, kernel_program_info()); |
| helper.InitFromKernelProgramInfo(kernel_info); |
| |
| const auto& data = |
| TypedDataView::Handle(zone, interesting_script.constant_coverage()); |
| |
| kernel::KernelReaderHelper kernel_reader(zone, &helper, data, 0); |
| |
| // Read "constant coverage constructors". |
| const intptr_t constant_coverage_constructors = |
| kernel_reader.ReadListLength(); |
| const Array& constructors = |
| Array::Handle(Array::New(constant_coverage_constructors)); |
| for (intptr_t i = 0; i < constant_coverage_constructors; ++i) { |
| kernel::NameIndex kernel_name = kernel_reader.ReadCanonicalNameReference(); |
| Class& klass = Class::ZoneHandle( |
| zone, |
| helper.LookupClassByKernelClass(helper.EnclosingName(kernel_name))); |
| const Function& target = Function::ZoneHandle( |
| zone, helper.LookupConstructorByKernelConstructor(klass, kernel_name)); |
| constructors.SetAt(i, target); |
| } |
| return constructors.ptr(); |
| } |
| #endif // !defined(PRODUCT) && !defined(DART_PRECOMPILED_RUNTIME) |
| |
| namespace kernel { |
| |
| ObjectPtr EvaluateStaticConstFieldInitializer(const Field& field) { |
| ASSERT(field.is_static() && field.is_const()); |
| |
| Thread* thread = Thread::Current(); |
| LongJumpScope jump(thread); |
| if (DART_SETJMP(*jump.Set()) == 0) { |
| Zone* zone = thread->zone(); |
| TranslationHelper helper(thread); |
| auto& kernel_program_info = |
| KernelProgramInfo::Handle(zone, field.KernelProgramInfo()); |
| helper.InitFromKernelProgramInfo(kernel_program_info); |
| |
| const Class& owner_class = Class::Handle(zone, field.Owner()); |
| ActiveClass active_class; |
| ActiveClassScope active_class_scope(&active_class, &owner_class); |
| |
| KernelReaderHelper kernel_reader( |
| zone, &helper, TypedDataView::Handle(zone, field.KernelLibrary()), |
| field.KernelLibraryOffset()); |
| kernel_reader.SetOffset(field.kernel_offset()); |
| ConstantReader constant_reader(&kernel_reader, &active_class); |
| |
| FieldHelper field_helper(&kernel_reader); |
| field_helper.ReadUntilExcluding(FieldHelper::kInitializer); |
| ASSERT(field_helper.IsConst()); |
| |
| return constant_reader.ReadConstantInitializer(); |
| } else { |
| return thread->StealStickyError(); |
| } |
| } |
| |
| class MetadataEvaluator : public KernelReaderHelper { |
| public: |
| MetadataEvaluator(Zone* zone, |
| TranslationHelper* translation_helper, |
| const TypedDataView& data, |
| intptr_t data_program_offset, |
| ActiveClass* active_class) |
| : KernelReaderHelper(zone, translation_helper, data, data_program_offset), |
| constant_reader_(this, active_class) {} |
| |
| ObjectPtr EvaluateMetadata(intptr_t kernel_offset, |
| bool is_annotations_offset) { |
| SetOffset(kernel_offset); |
| |
| // Library and LibraryDependency objects do not have a tag in kernel binary. |
| // Synthetic metadata fields corresponding to these objects keep kernel |
| // offset of annotations list instead of annotated object. |
| if (!is_annotations_offset) { |
| const Tag tag = PeekTag(); |
| |
| if (tag == kClass) { |
| ClassHelper class_helper(this); |
| class_helper.ReadUntilExcluding(ClassHelper::kAnnotations); |
| } else if (tag == kProcedure) { |
| ProcedureHelper procedure_helper(this); |
| procedure_helper.ReadUntilExcluding(ProcedureHelper::kAnnotations); |
| } else if (tag == kField) { |
| FieldHelper field_helper(this); |
| field_helper.ReadUntilExcluding(FieldHelper::kAnnotations); |
| } else if (tag == kConstructor) { |
| ConstructorHelper constructor_helper(this); |
| constructor_helper.ReadUntilExcluding(ConstructorHelper::kAnnotations); |
| } else if (tag == kFunctionDeclaration) { |
| ReadTag(); |
| ReadPosition(); // fileOffset |
| VariableDeclarationHelper variable_declaration_helper(this); |
| variable_declaration_helper.ReadUntilExcluding( |
| VariableDeclarationHelper::kAnnotations); |
| } else { |
| FATAL("No support for metadata on this type of kernel node: %" Pd32 |
| "\n", |
| tag); |
| } |
| } |
| |
| return constant_reader_.ReadAnnotations(); |
| } |
| |
| private: |
| ConstantReader constant_reader_; |
| |
| DISALLOW_COPY_AND_ASSIGN(MetadataEvaluator); |
| }; |
| |
| ObjectPtr EvaluateMetadata(const Library& library, |
| intptr_t kernel_offset, |
| bool is_annotations_offset) { |
| Thread* thread = Thread::Current(); |
| LongJumpScope jump(thread); |
| if (DART_SETJMP(*jump.Set()) == 0) { |
| Zone* zone = thread->zone(); |
| TranslationHelper helper(thread); |
| const auto& kernel_info = |
| KernelProgramInfo::Handle(zone, library.kernel_program_info()); |
| helper.InitFromKernelProgramInfo(kernel_info); |
| |
| const Class& owner_class = Class::Handle(zone, library.toplevel_class()); |
| ActiveClass active_class; |
| ActiveClassScope active_class_scope(&active_class, &owner_class); |
| |
| MetadataEvaluator metadata_evaluator( |
| zone, &helper, TypedDataView::Handle(zone, library.KernelLibrary()), |
| library.KernelLibraryOffset(), &active_class); |
| |
| return metadata_evaluator.EvaluateMetadata(kernel_offset, |
| is_annotations_offset); |
| |
| } else { |
| return thread->StealStickyError(); |
| } |
| } |
| |
| class ParameterDescriptorBuilder : public KernelReaderHelper { |
| public: |
| ParameterDescriptorBuilder(TranslationHelper* translation_helper, |
| Zone* zone, |
| const TypedDataView& data, |
| intptr_t data_program_offset, |
| ActiveClass* active_class) |
| : KernelReaderHelper(zone, translation_helper, data, data_program_offset), |
| constant_reader_(this, active_class) {} |
| |
| ObjectPtr BuildParameterDescriptor(const Function& function); |
| |
| private: |
| ConstantReader constant_reader_; |
| |
| DISALLOW_COPY_AND_ASSIGN(ParameterDescriptorBuilder); |
| }; |
| |
| ObjectPtr ParameterDescriptorBuilder::BuildParameterDescriptor( |
| const Function& function) { |
| SetOffset(function.kernel_offset()); |
| ReadUntilFunctionNode(); |
| FunctionNodeHelper function_node_helper(this); |
| function_node_helper.ReadUntilExcluding( |
| FunctionNodeHelper::kPositionalParameters); |
| intptr_t param_count = function_node_helper.total_parameter_count_; |
| intptr_t positional_count = ReadListLength(); // read list length. |
| intptr_t named_parameter_count = param_count - positional_count; |
| |
| const Array& param_descriptor = Array::Handle( |
| Array::New(param_count * Parser::kParameterEntrySize, Heap::kOld)); |
| for (intptr_t i = 0; i < param_count; ++i) { |
| const intptr_t entry_start = i * Parser::kParameterEntrySize; |
| |
| if (i == positional_count) { |
| intptr_t named_parameter_count_check = |
| ReadListLength(); // read list length. |
| ASSERT(named_parameter_count_check == named_parameter_count); |
| } |
| |
| // Read ith variable declaration. |
| intptr_t param_kernel_offset = reader_.offset(); |
| VariableDeclarationHelper helper(this); |
| helper.ReadUntilExcluding(VariableDeclarationHelper::kInitializer); |
| param_descriptor.SetAt(entry_start + Parser::kParameterIsFinalOffset, |
| helper.IsFinal() ? Bool::True() : Bool::False()); |
| |
| Tag tag = ReadTag(); // read (first part of) initializer. |
| if ((tag == kSomething) && !function.is_abstract()) { |
| // This will read the initializer. |
| Instance& constant = Instance::ZoneHandle( |
| zone_, constant_reader_.ReadConstantExpression()); |
| param_descriptor.SetAt(entry_start + Parser::kParameterDefaultValueOffset, |
| constant); |
| } else { |
| if (tag == kSomething) { |
| SkipExpression(); // Skip initializer. |
| } |
| param_descriptor.SetAt(entry_start + Parser::kParameterDefaultValueOffset, |
| Object::null_instance()); |
| } |
| |
| if (FLAG_enable_mirrors && (helper.annotation_count_ > 0)) { |
| AlternativeReadingScope alt(&reader_, param_kernel_offset); |
| VariableDeclarationHelper helper(this); |
| helper.ReadUntilExcluding(VariableDeclarationHelper::kAnnotations); |
| Object& metadata = |
| Object::ZoneHandle(zone_, constant_reader_.ReadAnnotations()); |
| param_descriptor.SetAt(entry_start + Parser::kParameterMetadataOffset, |
| metadata); |
| } else { |
| param_descriptor.SetAt(entry_start + Parser::kParameterMetadataOffset, |
| Object::null_instance()); |
| } |
| } |
| return param_descriptor.ptr(); |
| } |
| |
| ObjectPtr BuildParameterDescriptor(const Function& function) { |
| Thread* thread = Thread::Current(); |
| LongJumpScope jump(thread); |
| if (DART_SETJMP(*jump.Set()) == 0) { |
| Zone* zone = thread->zone(); |
| |
| const auto& kernel_info = |
| KernelProgramInfo::Handle(zone, function.KernelProgramInfo()); |
| |
| TranslationHelper helper(thread); |
| helper.InitFromKernelProgramInfo(kernel_info); |
| |
| const Class& owner_class = Class::Handle(zone, function.Owner()); |
| ActiveClass active_class; |
| ActiveClassScope active_class_scope(&active_class, &owner_class); |
| |
| ParameterDescriptorBuilder builder( |
| &helper, zone, TypedDataView::Handle(zone, function.KernelLibrary()), |
| function.KernelLibraryOffset(), &active_class); |
| |
| return builder.BuildParameterDescriptor(function); |
| } else { |
| return thread->StealStickyError(); |
| } |
| } |
| |
| void ReadParameterCovariance(const Function& function, |
| BitVector* is_covariant, |
| BitVector* is_generic_covariant_impl) { |
| Thread* thread = Thread::Current(); |
| Zone* zone = thread->zone(); |
| |
| const intptr_t num_params = function.NumParameters(); |
| ASSERT(is_covariant->length() == num_params); |
| ASSERT(is_generic_covariant_impl->length() == num_params); |
| |
| const auto& kernel_info = |
| KernelProgramInfo::Handle(zone, function.KernelProgramInfo()); |
| |
| TranslationHelper translation_helper(thread); |
| translation_helper.InitFromKernelProgramInfo(kernel_info); |
| |
| KernelReaderHelper reader_helper( |
| zone, &translation_helper, |
| TypedDataView::Handle(zone, function.KernelLibrary()), |
| function.KernelLibraryOffset()); |
| |
| reader_helper.SetOffset(function.kernel_offset()); |
| reader_helper.ReadUntilFunctionNode(); |
| |
| FunctionNodeHelper function_node_helper(&reader_helper); |
| function_node_helper.ReadUntilExcluding( |
| FunctionNodeHelper::kPositionalParameters); |
| |
| // Positional. |
| const intptr_t num_positional_params = reader_helper.ReadListLength(); |
| intptr_t param_index = function.NumImplicitParameters(); |
| for (intptr_t i = 0; i < num_positional_params; ++i, ++param_index) { |
| VariableDeclarationHelper helper(&reader_helper); |
| helper.ReadUntilExcluding(VariableDeclarationHelper::kEnd); |
| |
| if (helper.IsCovariant()) { |
| is_covariant->Add(param_index); |
| } |
| if (helper.IsGenericCovariantImpl()) { |
| is_generic_covariant_impl->Add(param_index); |
| } |
| } |
| |
| // Named. |
| const intptr_t num_named_params = reader_helper.ReadListLength(); |
| for (intptr_t i = 0; i < num_named_params; ++i, ++param_index) { |
| VariableDeclarationHelper helper(&reader_helper); |
| helper.ReadUntilExcluding(VariableDeclarationHelper::kEnd); |
| |
| if (helper.IsCovariant()) { |
| is_covariant->Add(param_index); |
| } |
| if (helper.IsGenericCovariantImpl()) { |
| is_generic_covariant_impl->Add(param_index); |
| } |
| } |
| } |
| |
| static ProcedureAttributesMetadata ProcedureAttributesOf( |
| Zone* zone, |
| const KernelProgramInfo& kernel_program_info, |
| const TypedDataView& kernel_data, |
| intptr_t kernel_data_program_offset, |
| intptr_t kernel_offset) { |
| TranslationHelper translation_helper(Thread::Current()); |
| translation_helper.InitFromKernelProgramInfo(kernel_program_info); |
| KernelReaderHelper reader_helper(zone, &translation_helper, kernel_data, |
| kernel_data_program_offset); |
| ProcedureAttributesMetadataHelper procedure_attributes_metadata_helper( |
| &reader_helper); |
| ProcedureAttributesMetadata attrs = |
| procedure_attributes_metadata_helper.GetProcedureAttributes( |
| kernel_offset); |
| return attrs; |
| } |
| |
| ProcedureAttributesMetadata ProcedureAttributesOf(const Function& function, |
| Zone* zone) { |
| const auto& kernel_program_info = |
| KernelProgramInfo::Handle(zone, function.KernelProgramInfo()); |
| return ProcedureAttributesOf( |
| zone, kernel_program_info, |
| TypedDataView::Handle(zone, function.KernelLibrary()), |
| function.KernelLibraryOffset(), function.kernel_offset()); |
| } |
| |
| ProcedureAttributesMetadata ProcedureAttributesOf(const Field& field, |
| Zone* zone) { |
| const auto& kernel_program_info = |
| KernelProgramInfo::Handle(zone, field.KernelProgramInfo()); |
| return ProcedureAttributesOf( |
| zone, kernel_program_info, |
| TypedDataView::Handle(zone, field.KernelLibrary()), |
| field.KernelLibraryOffset(), field.kernel_offset()); |
| } |
| |
| static UnboxingInfoMetadata* UnboxingInfoMetadataOf( |
| Zone* zone, |
| const KernelProgramInfo& kernel_program_info, |
| const TypedDataView& kernel_data, |
| intptr_t kernel_data_program_offset, |
| intptr_t kernel_offset) { |
| TranslationHelper translation_helper(Thread::Current()); |
| translation_helper.InitFromKernelProgramInfo(kernel_program_info); |
| KernelReaderHelper reader_helper(zone, &translation_helper, kernel_data, |
| kernel_data_program_offset); |
| UnboxingInfoMetadataHelper unboxing_info_metadata_helper(&reader_helper); |
| return unboxing_info_metadata_helper.GetUnboxingInfoMetadata(kernel_offset); |
| } |
| |
| UnboxingInfoMetadata* UnboxingInfoMetadataOf(const Function& function, |
| Zone* zone) { |
| const auto& kernel_program_info = |
| KernelProgramInfo::Handle(zone, function.KernelProgramInfo()); |
| return UnboxingInfoMetadataOf( |
| zone, kernel_program_info, |
| TypedDataView::Handle(zone, function.KernelLibrary()), |
| function.KernelLibraryOffset(), function.kernel_offset()); |
| } |
| |
| TableSelectorMetadata* TableSelectorMetadataForProgram( |
| const KernelProgramInfo& info, |
| Zone* zone) { |
| TranslationHelper translation_helper(Thread::Current()); |
| translation_helper.InitFromKernelProgramInfo(info); |
| const auto& data = TypedDataView::Handle(zone, info.metadata_payloads()); |
| KernelReaderHelper reader_helper(zone, &translation_helper, data, 0); |
| TableSelectorMetadataHelper table_selector_metadata_helper(&reader_helper); |
| return table_selector_metadata_helper.GetTableSelectorMetadata(zone); |
| } |
| |
| } // namespace kernel |
| } // namespace dart |
| |
| #endif // !defined(DART_PRECOMPILED_RUNTIME) |