/* * Copyright (c) 2016, Google Inc. * All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef PERFTOOLS_PROFILES_PROTO_BUILDER_H_ #define PERFTOOLS_PROFILES_PROTO_BUILDER_H_ #include <stddef.h> #include <algorithm> #include <memory> #include <string> #include <tuple> #include <unordered_map> namespace perftools { namespace profiles { typedef int64_t int64; typedef uint64_t uint64; typedef std::string string; } } #include "profile.pb.h" namespace perftools { namespace profiles { // Provides mechanisms to facilitate the generation of profiles // on a compressed protobuf: // - Manages the creation of the string table. // - Manages the creation of Functions for symbolized profiles. // - Creates the association between locations and mappings. // The caller should populate the profile with samples and their // corresponding sample types, and any other optional fields. class Builder { public: Builder(); // Adds a string to the profile string table if not already present. // Returns a unique integer id for this string. int64 StringId(const char *str); // Adds a function with these attributes to the profile function // table, if not already present. Returns a unique integer id for // this function. uint64 FunctionId(const char *name, const char *system_name, const char *file, int64 start_line); // Adds mappings for the currently running binary to the profile. void AddCurrentMappings(); // Prepares the profile for encoding. Returns true on success. // If the profile has no locations, inserts location using the // location_ids from the samples as addresses. // Associates the locations to mappings by comparing the location // address into the mapping address range. bool Finalize(); // Serializes and compresses the profile into a string, replacing // its contents. It calls Finalize() and returns whether the // encoding was successful. bool Emit(string *output); // Serializes and compresses a profile into a string, replacing its // contents. Returns false if there were errors on the serialization // or compression, and the output string will not contain valid data. static bool Marshal(const Profile &profile, string *output); // Serializes and compresses a profile into a file represented by a // file descriptor. Returns false if there were errors on the // serialization or compression. static bool MarshalToFile(const Profile &profile, int fd); // Serializes and compresses a profile into a file, creating a new // file or replacing its contents if it already exists. static bool MarshalToFile(const Profile &profile, const char *filename); // Determines if the profile is internally consistent (suitable for // serialization). Returns true if no errors were encountered. static bool CheckValid(const Profile &profile); // Extract the profile from the builder object. No further calls // should be made to the builder after this. std::unique_ptr<Profile> Consume() { return std::move(profile_); } // Returns the underlying profile, to populate any fields not // managed by the builder. The fields function and string_table // should be populated through Builder::StringId and // Builder::FunctionId. Profile *mutable_profile() { return profile_.get(); } private: // Holds the information about a function to facilitate deduplication. typedef std::tuple<int64, int64, int64, int64> Function; class FunctionHasher { public: size_t operator()(const Function &f) const { int64 hash = std::get<0>(f); hash = hash + ((hash << 8) ^ std::get<1>(f)); hash = hash + ((hash << 8) ^ std::get<2>(f)); hash = hash + ((hash << 8) ^ std::get<3>(f)); return static_cast<size_t>(hash); } }; // Hashes to deduplicate strings and functions. std::unordered_map<string, int64> strings_; std::unordered_map<Function, int64, FunctionHasher> functions_; // Actual profile being updated. std::unique_ptr<Profile> profile_; }; } // namespace profiles } // namespace perftools #endif // PERFTOOLS_PROFILES_PROTO_BUILDER_H_