/* * Copyright (C) 2015 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. */ #ifndef SIMPLE_PERF_RECORD_FILE_H_ #define SIMPLE_PERF_RECORD_FILE_H_ #include #include #include #include #include #include #include #include #include #include "dso.h" #include "event_attr.h" #include "event_type.h" #include "perf_event.h" #include "record.h" #include "record_file_format.h" #include "thread_tree.h" namespace simpleperf { struct FileFeature { std::string path; DsoType type; uint64_t min_vaddr; uint64_t file_offset_of_min_vaddr; // for DSO_ELF_FILE or DSO_KERNEL_MODULE std::vector symbols; // used for reading symbols std::vector symbol_ptrs; // used for writing symbols std::vector dex_file_offsets; FileFeature() {} void Clear() { path.clear(); type = DSO_UNKNOWN_FILE; min_vaddr = 0; file_offset_of_min_vaddr = 0; symbols.clear(); symbol_ptrs.clear(); dex_file_offsets.clear(); } DISALLOW_COPY_AND_ASSIGN(FileFeature); }; struct DebugUnwindFile { std::string path; uint64_t size; }; using DebugUnwindFeature = std::vector; // RecordFileWriter writes to a perf record file, like perf.data. // User should call RecordFileWriter::Close() to finish writing the file, otherwise the file will // be removed in RecordFileWriter::~RecordFileWriter(). class RecordFileWriter { public: static std::unique_ptr CreateInstance(const std::string& filename); // If own_fp = true, close fp when we finish writing. RecordFileWriter(const std::string& filename, FILE* fp, bool own_fp); ~RecordFileWriter(); bool WriteAttrSection(const EventAttrIds& attr_ids); bool WriteRecord(const Record& record); bool WriteData(const void* buf, size_t len); uint64_t GetDataSectionSize() const { return data_section_size_; } bool ReadDataSection(const std::function& callback); bool BeginWriteFeatures(size_t feature_count); bool WriteBuildIdFeature(const std::vector& build_id_records); bool WriteFeatureString(int feature, const std::string& s); bool WriteCmdlineFeature(const std::vector& cmdline); bool WriteBranchStackFeature(); bool WriteAuxTraceFeature(const std::vector& auxtrace_offset); bool WriteFileFeatures(const std::vector& dsos); bool WriteFileFeature(const FileFeature& file); bool WriteMetaInfoFeature(const std::unordered_map& info_map); bool WriteDebugUnwindFeature(const DebugUnwindFeature& debug_unwind); bool WriteFeature(int feature, const char* data, size_t size); bool EndWriteFeatures(); bool Close(); private: void GetHitModulesInBuffer(const char* p, const char* end, std::vector* hit_kernel_modules, std::vector* hit_user_files); bool WriteFileHeader(); bool Write(const void* buf, size_t len); bool Read(void* buf, size_t len); bool GetFilePos(uint64_t* file_pos); bool WriteStringWithLength(const std::string& s); bool WriteFeatureBegin(int feature); bool WriteFeatureEnd(int feature); const std::string filename_; FILE* record_fp_; bool own_fp_; perf_event_attr event_attr_; uint64_t attr_section_offset_; uint64_t attr_section_size_; uint64_t data_section_offset_; uint64_t data_section_size_; uint64_t feature_section_offset_; std::map features_; size_t feature_count_; DISALLOW_COPY_AND_ASSIGN(RecordFileWriter); }; // RecordFileReader read contents from a perf record file, like perf.data. class RecordFileReader { public: static std::unique_ptr CreateInstance(const std::string& filename); ~RecordFileReader(); const std::string FileName() const { return filename_; } const PerfFileFormat::FileHeader& FileHeader() const { return header_; } const EventAttrIds& AttrSection() const { return event_attrs_; } const std::unordered_map& EventIdMap() const { return event_id_to_attr_map_; } const std::map& FeatureSectionDescriptors() const { return feature_section_descriptors_; } bool HasFeature(int feature) const { return feature_section_descriptors_.find(feature) != feature_section_descriptors_.end(); } bool ReadFeatureSection(int feature, std::vector* data); bool ReadFeatureSection(int feature, std::string* data); // There are two ways to read records in data section: one is by calling // ReadDataSection(), and [callback] is called for each Record. the other // is by calling ReadRecord() in a loop. // If sorted is true, sort records before passing them to callback function. bool ReadDataSection(const std::function)>& callback); bool ReadAtOffset(uint64_t offset, void* buf, size_t len); // Read next record. If read successfully, set [record] and return true. // If there is no more records, set [record] to nullptr and return true. // Otherwise return false. bool ReadRecord(std::unique_ptr& record); size_t GetAttrIndexOfRecord(const Record* record); std::vector ReadCmdlineFeature(); std::vector ReadBuildIdFeature(); std::string ReadFeatureString(int feature); std::vector ReadAuxTraceFeature(); // File feature section contains many file information. This function reads // one file information located at [read_pos]. [read_pos] is 0 at the first // call, and is updated to point to the next file information. // When read successfully, return true and set error to false. // When no more data to read, return false and set error to false. // When having error, return false and set error to true. bool ReadFileFeature(uint64_t& read_pos, FileFeature& file, bool& error); const std::unordered_map& GetMetaInfoFeature() { return meta_info_; } std::string GetClockId(); std::optional ReadDebugUnwindFeature(); bool LoadBuildIdAndFileFeatures(ThreadTree& thread_tree); // Read aux data into buf. // When read successfully, return true and set error to false. // When the data isn't available, return false and set error to false. // When having error, return false and set error to true. bool ReadAuxData(uint32_t cpu, uint64_t aux_offset, size_t size, std::vector& buf, bool& error); bool Close(); // For testing only. std::vector> DataSection(); private: RecordFileReader(const std::string& filename, FILE* fp); bool ReadHeader(); bool CheckSectionDesc(const PerfFileFormat::SectionDesc& desc, uint64_t min_offset, uint64_t alignment = 1); bool ReadAttrSection(); bool ReadIdSection(const PerfFileFormat::SectionDesc& section, std::vector* ids); bool ReadFeatureSectionDescriptors(); bool ReadFileV1Feature(uint64_t& read_pos, uint64_t max_size, FileFeature& file); bool ReadFileV2Feature(uint64_t& read_pos, uint64_t max_size, FileFeature& file); bool ReadMetaInfoFeature(); void UseRecordingEnvironment(); std::unique_ptr ReadRecord(); bool Read(void* buf, size_t len); void ProcessEventIdRecord(const EventIdRecord& r); bool BuildAuxDataLocation(); const std::string filename_; FILE* record_fp_; uint64_t file_size_; PerfFileFormat::FileHeader header_; EventAttrIds event_attrs_; std::unordered_map event_id_to_attr_map_; std::map feature_section_descriptors_; size_t event_id_pos_in_sample_records_; size_t event_id_reverse_pos_in_non_sample_records_; uint64_t read_record_size_; std::unordered_map meta_info_; std::unique_ptr scoped_arch_; std::unique_ptr scoped_event_types_; struct AuxDataLocation { uint64_t aux_offset; uint64_t aux_size; uint64_t file_offset; AuxDataLocation(uint64_t aux_offset, uint64_t aux_size, uint64_t file_offset) : aux_offset(aux_offset), aux_size(aux_size), file_offset(file_offset) {} }; // It maps from a cpu id to the locations (file offsets in perf.data) of aux data received from // that cpu's aux buffer. It is used to locate aux data in perf.data. std::unordered_map> aux_data_location_; DISALLOW_COPY_AND_ASSIGN(RecordFileReader); }; bool IsPerfDataFile(const std::string& filename); } // namespace simpleperf #endif // SIMPLE_PERF_RECORD_FILE_H_