/* * Copyright (C) 2018 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 SRC_TRACED_PROBES_FILESYSTEM_PREFIX_FINDER_H_ #define SRC_TRACED_PROBES_FILESYSTEM_PREFIX_FINDER_H_ #include <memory> #include <set> #include <string> #include <tuple> #include <utility> #include <vector> #include "perfetto/base/logging.h" #include "perfetto/base/lookup_set.h" namespace perfetto { // PrefixFinder allows to find prefixes for filenames that ensure that // they can be found again within a provided limit of steps when searching // within that prefix in the same order. // // For instance, let the limit be 4 and the filesystem be. // /a/1 // /a/2 // /a/3 // /b/4 // /b/5 // // The prefix for /a/1, /a/2 and /a/3/ is /, the one for /b/4 and /b/5 is /b/. class PrefixFinder { public: // Opaque placeholder for a prefix that can be turned into a string // using ToString. // Can not be constructed outside of PrefixFinder. class Node { public: friend class PrefixFinder; Node(std::string name) : Node(std::move(name), nullptr) {} Node(std::string name, Node* parent) : name_(std::move(name)), parent_(parent) {} Node(const Node& that) = delete; Node& operator=(const Node&) = delete; // Return string representation of prefix, e.g. /foo/bar. // Does not enclude a trailing /. std::string ToString() const; private: // Add a new child to this node. Node* AddChild(std::string name); // Get existing child for this node. Return nullptr if it // does not exist. Node* MaybeChild(const std::string& name); const std::string name_; const Node* parent_; base::LookupSet<Node, const std::string, &Node::name_> children_; }; PrefixFinder(size_t limit); // Add path to prefix mapping. // Must be called in DFS order. // Must be called before GetPrefix(path) for the same path. // Must not be called after Finalize. void AddPath(std::string path); // Return identifier for prefix. Ownership remains with the PrefixFinder. // Must be called after AddPath(path) for the same path. // Must not be before after Finalize. Node* GetPrefix(std::string path); // Call this after the last AddPath and before the first GetPrefix. void Finalize(); private: // We're about to remove the suffix of state from i onwards, // if necessary add a prefix for anything in that suffix. void Flush(size_t i); void InsertPrefix(size_t len); const size_t limit_; // (path element, count) tuples for last path seen. std::vector<std::pair<std::string, size_t>> state_{{"", 0}}; Node root_{"", nullptr}; #if PERFETTO_DCHECK_IS_ON() bool finalized_ = false; #endif }; } // namespace perfetto #endif // SRC_TRACED_PROBES_FILESYSTEM_PREFIX_FINDER_H_