// Copyright (C) 2019 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "repr/protobuf/ir_reader.h" #include "repr/ir_representation_internal.h" #include "repr/protobuf/api.h" #include "repr/protobuf/converter.h" #include #include #include #include namespace header_checker { namespace repr { void ProtobufIRReader::ReadTypeInfo( const abi_dump::BasicNamedAndTypedDecl &type_info, TypeIR *typep) { typep->SetLinkerSetKey(type_info.linker_set_key()); typep->SetName(type_info.name()); typep->SetSourceFile(type_info.source_file()); typep->SetReferencedType(type_info.referenced_type()); typep->SetSelfType(type_info.self_type()); typep->SetSize(type_info.size()); typep->SetAlignment(type_info.alignment()); } bool ProtobufIRReader::ReadDumpImpl(const std::string &dump_file) { abi_dump::TranslationUnit tu; std::ifstream input(dump_file); google::protobuf::io::IstreamInputStream text_is(&input); if (!google::protobuf::TextFormat::Parse(&text_is, &tu)) { llvm::errs() << "Failed to parse protobuf TextFormat file\n"; return false; } ReadFunctions(tu); ReadGlobalVariables(tu); ReadEnumTypes(tu); ReadRecordTypes(tu); ReadFunctionTypes(tu); ReadArrayTypes(tu); ReadPointerTypes(tu); ReadQualifiedTypes(tu); ReadBuiltinTypes(tu); ReadLvalueReferenceTypes(tu); ReadRvalueReferenceTypes(tu); ReadElfFunctions(tu); ReadElfObjects(tu); return true; } TemplateInfoIR ProtobufIRReader::TemplateInfoProtobufToIR( const abi_dump::TemplateInfo &template_info_protobuf) { TemplateInfoIR template_info_ir; for (auto &&template_element : template_info_protobuf.elements()) { TemplateElementIR template_element_ir(template_element.referenced_type()); template_info_ir.AddTemplateElement(std::move(template_element_ir)); } return template_info_ir; } template static void SetupCFunctionLikeIR(const T &cfunction_like_protobuf, CFunctionLikeIR *cfunction_like_ir) { cfunction_like_ir->SetReturnType(cfunction_like_protobuf.return_type()); for (auto &¶meter: cfunction_like_protobuf.parameters()) { ParamIR param_ir(parameter.referenced_type(), parameter.default_arg(), false); cfunction_like_ir->AddParameter(std::move(param_ir)); } } FunctionIR ProtobufIRReader::FunctionProtobufToIR( const abi_dump::FunctionDecl &function_protobuf) { FunctionIR function_ir; function_ir.SetReturnType(function_protobuf.return_type()); function_ir.SetLinkerSetKey(function_protobuf.linker_set_key()); function_ir.SetName(function_protobuf.function_name()); function_ir.SetAccess(AccessProtobufToIR(function_protobuf.access())); function_ir.SetSourceFile(function_protobuf.source_file()); // Set parameters for (auto &¶meter: function_protobuf.parameters()) { ParamIR param_ir(parameter.referenced_type(), parameter.default_arg(), parameter.is_this_ptr()); function_ir.AddParameter(std::move(param_ir)); } // Set Template info function_ir.SetTemplateInfo( TemplateInfoProtobufToIR(function_protobuf.template_info())); return function_ir; } FunctionTypeIR ProtobufIRReader::FunctionTypeProtobufToIR( const abi_dump::FunctionType &function_type_protobuf) { FunctionTypeIR function_type_ir; ReadTypeInfo(function_type_protobuf.type_info(), &function_type_ir); SetupCFunctionLikeIR(function_type_protobuf, &function_type_ir); return function_type_ir; } VTableLayoutIR ProtobufIRReader::VTableLayoutProtobufToIR( const abi_dump::VTableLayout &vtable_layout_protobuf) { VTableLayoutIR vtable_layout_ir; for (auto &&vtable_component : vtable_layout_protobuf.vtable_components()) { VTableComponentIR vtable_component_ir( vtable_component.mangled_component_name(), VTableComponentKindProtobufToIR(vtable_component.kind()), vtable_component.component_value(), vtable_component.is_pure()); vtable_layout_ir.AddVTableComponent(std::move(vtable_component_ir)); } return vtable_layout_ir; } std::vector ProtobufIRReader::RecordFieldsProtobufToIR( const google::protobuf::RepeatedPtrField &rfp) { std::vector record_type_fields_ir; for (auto &&field : rfp) { RecordFieldIR record_field_ir( field.field_name(), field.referenced_type(), field.field_offset(), AccessProtobufToIR(field.access()), field.is_bit_field(), field.bit_width()); record_type_fields_ir.emplace_back(std::move(record_field_ir)); } return record_type_fields_ir; } std::vector ProtobufIRReader::RecordCXXBaseSpecifiersProtobufToIR( const google::protobuf::RepeatedPtrField &rbs) { std::vector record_type_bases_ir; for (auto &&base : rbs) { CXXBaseSpecifierIR record_base_ir( base.referenced_type(), base.is_virtual(), AccessProtobufToIR(base.access())); record_type_bases_ir.emplace_back(std::move(record_base_ir)); } return record_type_bases_ir; } RecordTypeIR ProtobufIRReader::RecordTypeProtobufToIR( const abi_dump::RecordType &record_type_protobuf) { RecordTypeIR record_type_ir; ReadTypeInfo(record_type_protobuf.type_info(), &record_type_ir); record_type_ir.SetTemplateInfo( TemplateInfoProtobufToIR(record_type_protobuf.template_info())); record_type_ir.SetAccess(AccessProtobufToIR(record_type_protobuf.access())); record_type_ir.SetVTableLayout( VTableLayoutProtobufToIR(record_type_protobuf.vtable_layout())); // Get fields record_type_ir.SetRecordFields(RecordFieldsProtobufToIR( record_type_protobuf.fields())); // Base Specifiers record_type_ir.SetCXXBaseSpecifiers(RecordCXXBaseSpecifiersProtobufToIR( record_type_protobuf.base_specifiers())); record_type_ir.SetRecordKind( RecordKindProtobufToIR(record_type_protobuf.record_kind())); record_type_ir.SetAnonymity(record_type_protobuf.is_anonymous()); return record_type_ir; } std::vector ProtobufIRReader::EnumFieldsProtobufToIR( const google::protobuf::RepeatedPtrField &efp) { std::vector enum_type_fields_ir; for (auto &&field : efp) { EnumFieldIR enum_field_ir(field.name(), field.enum_field_value()); enum_type_fields_ir.emplace_back(std::move(enum_field_ir)); } return enum_type_fields_ir; } EnumTypeIR ProtobufIRReader::EnumTypeProtobufToIR( const abi_dump::EnumType &enum_type_protobuf) { EnumTypeIR enum_type_ir; ReadTypeInfo(enum_type_protobuf.type_info(), &enum_type_ir); enum_type_ir.SetUnderlyingType(enum_type_protobuf.underlying_type()); enum_type_ir.SetAccess(AccessProtobufToIR(enum_type_protobuf.access())); enum_type_ir.SetFields( EnumFieldsProtobufToIR(enum_type_protobuf.enum_fields())); return enum_type_ir; } void ProtobufIRReader::ReadGlobalVariables( const abi_dump::TranslationUnit &tu) { for (auto &&global_variable_protobuf : tu.global_vars()) { GlobalVarIR global_variable_ir; global_variable_ir.SetName(global_variable_protobuf.name()); global_variable_ir.SetAccess(AccessProtobufToIR(global_variable_protobuf.access())); global_variable_ir.SetSourceFile(global_variable_protobuf.source_file()); global_variable_ir.SetReferencedType( global_variable_protobuf.referenced_type()); global_variable_ir.SetLinkerSetKey( global_variable_protobuf.linker_set_key()); module_->AddGlobalVariable(std::move(global_variable_ir)); } } void ProtobufIRReader::ReadPointerTypes(const abi_dump::TranslationUnit &tu) { for (auto &&pointer_type_protobuf : tu.pointer_types()) { PointerTypeIR pointer_type_ir; ReadTypeInfo(pointer_type_protobuf.type_info(), &pointer_type_ir); module_->AddPointerType(std::move(pointer_type_ir)); } } void ProtobufIRReader::ReadBuiltinTypes(const abi_dump::TranslationUnit &tu) { for (auto &&builtin_type_protobuf : tu.builtin_types()) { BuiltinTypeIR builtin_type_ir; ReadTypeInfo(builtin_type_protobuf.type_info(), &builtin_type_ir); builtin_type_ir.SetSignedness(builtin_type_protobuf.is_unsigned()); builtin_type_ir.SetIntegralType(builtin_type_protobuf.is_integral()); module_->AddBuiltinType(std::move(builtin_type_ir)); } } void ProtobufIRReader::ReadQualifiedTypes(const abi_dump::TranslationUnit &tu) { for (auto &&qualified_type_protobuf : tu.qualified_types()) { QualifiedTypeIR qualified_type_ir; ReadTypeInfo(qualified_type_protobuf.type_info(), &qualified_type_ir); qualified_type_ir.SetConstness(qualified_type_protobuf.is_const()); qualified_type_ir.SetVolatility(qualified_type_protobuf.is_volatile()); qualified_type_ir.SetRestrictedness( qualified_type_protobuf.is_restricted()); module_->AddQualifiedType(std::move(qualified_type_ir)); } } void ProtobufIRReader::ReadArrayTypes(const abi_dump::TranslationUnit &tu) { for (auto &&array_type_protobuf : tu.array_types()) { ArrayTypeIR array_type_ir; ReadTypeInfo(array_type_protobuf.type_info(), &array_type_ir); array_type_ir.SetUnknownBound(array_type_protobuf.is_of_unknown_bound()); module_->AddArrayType(std::move(array_type_ir)); } } void ProtobufIRReader::ReadLvalueReferenceTypes( const abi_dump::TranslationUnit &tu) { for (auto &&lvalue_reference_type_protobuf : tu.lvalue_reference_types()) { LvalueReferenceTypeIR lvalue_reference_type_ir; ReadTypeInfo(lvalue_reference_type_protobuf.type_info(), &lvalue_reference_type_ir); module_->AddLvalueReferenceType(std::move(lvalue_reference_type_ir)); } } void ProtobufIRReader::ReadRvalueReferenceTypes( const abi_dump::TranslationUnit &tu) { for (auto &&rvalue_reference_type_protobuf : tu.rvalue_reference_types()) { RvalueReferenceTypeIR rvalue_reference_type_ir; ReadTypeInfo(rvalue_reference_type_protobuf.type_info(), &rvalue_reference_type_ir); module_->AddRvalueReferenceType(std::move(rvalue_reference_type_ir)); } } void ProtobufIRReader::ReadFunctions(const abi_dump::TranslationUnit &tu) { for (auto &&function_protobuf : tu.functions()) { FunctionIR function_ir = FunctionProtobufToIR(function_protobuf); module_->AddFunction(std::move(function_ir)); } } void ProtobufIRReader::ReadRecordTypes(const abi_dump::TranslationUnit &tu) { for (auto &&record_type_protobuf : tu.record_types()) { RecordTypeIR record_type_ir = RecordTypeProtobufToIR(record_type_protobuf); module_->AddRecordType(std::move(record_type_ir)); } } void ProtobufIRReader::ReadFunctionTypes(const abi_dump::TranslationUnit &tu) { for (auto &&function_type_protobuf : tu.function_types()) { FunctionTypeIR function_type_ir = FunctionTypeProtobufToIR(function_type_protobuf); module_->AddFunctionType(std::move(function_type_ir)); } } void ProtobufIRReader::ReadEnumTypes(const abi_dump::TranslationUnit &tu) { for (auto &&enum_type_protobuf : tu.enum_types()) { EnumTypeIR enum_type_ir = EnumTypeProtobufToIR(enum_type_protobuf); module_->AddEnumType(std::move(enum_type_ir)); } } void ProtobufIRReader::ReadElfFunctions(const abi_dump::TranslationUnit &tu) { for (auto &&elf_function : tu.elf_functions()) { ElfFunctionIR elf_function_ir( elf_function.name(), ElfSymbolBindingProtobufToIR(elf_function.binding())); module_->AddElfFunction(std::move(elf_function_ir)); } } void ProtobufIRReader::ReadElfObjects(const abi_dump::TranslationUnit &tu) { for (auto &&elf_object : tu.elf_objects()) { ElfObjectIR elf_object_ir( elf_object.name(), ElfSymbolBindingProtobufToIR(elf_object.binding())); module_->AddElfObject(std::move(elf_object_ir)); } } std::unique_ptr CreateProtobufIRReader( const std::set *exported_headers) { return std::make_unique(exported_headers); } } // namespace repr } // namespace header_checker