| // |
| // Copyright (C) 2017 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 "trie_builder.h" |
| |
| #include <android-base/strings.h> |
| |
| using android::base::Split; |
| |
| namespace android { |
| namespace properties { |
| |
| TrieBuilder::TrieBuilder(const std::string& default_context, const std::string& default_type) |
| : builder_root_("root") { |
| auto* context_pointer = StringPointerFromContainer(default_context, &contexts_); |
| builder_root_.set_context(context_pointer); |
| auto* type_pointer = StringPointerFromContainer(default_type, &types_); |
| builder_root_.set_type(type_pointer); |
| } |
| |
| bool TrieBuilder::AddToTrie(const std::string& name, const std::string& context, |
| const std::string& type, bool exact, std::string* error) { |
| auto* context_pointer = StringPointerFromContainer(context, &contexts_); |
| auto* type_pointer = StringPointerFromContainer(type, &types_); |
| return AddToTrie(name, context_pointer, type_pointer, exact, error); |
| } |
| |
| bool TrieBuilder::AddToTrie(const std::string& name, const std::string* context, |
| const std::string* type, bool exact, std::string* error) { |
| TrieBuilderNode* current_node = &builder_root_; |
| |
| auto name_pieces = Split(name, "."); |
| |
| bool ends_with_dot = false; |
| if (name_pieces.back().empty()) { |
| ends_with_dot = true; |
| name_pieces.pop_back(); |
| } |
| |
| // Move us to the final node that we care about, adding incremental nodes if necessary. |
| while (name_pieces.size() > 1) { |
| auto child = current_node->FindChild(name_pieces.front()); |
| if (child == nullptr) { |
| child = current_node->AddChild(name_pieces.front()); |
| } |
| if (child == nullptr) { |
| *error = "Unable to allocate Trie node"; |
| return false; |
| } |
| current_node = child; |
| name_pieces.erase(name_pieces.begin()); |
| } |
| |
| // Store our context based on what type of match it is. |
| if (exact) { |
| if (!current_node->AddExactMatchContext(name_pieces.front(), context, type)) { |
| *error = "Duplicate exact match detected for '" + name + "'"; |
| return false; |
| } |
| } else if (!ends_with_dot) { |
| if (!current_node->AddPrefixContext(name_pieces.front(), context, type)) { |
| *error = "Duplicate prefix match detected for '" + name + "'"; |
| return false; |
| } |
| } else { |
| auto child = current_node->FindChild(name_pieces.front()); |
| if (child == nullptr) { |
| child = current_node->AddChild(name_pieces.front()); |
| } |
| if (child == nullptr) { |
| *error = "Unable to allocate Trie node"; |
| return false; |
| } |
| if (child->context() != nullptr || child->type() != nullptr) { |
| *error = "Duplicate prefix match detected for '" + name + "'"; |
| return false; |
| } |
| child->set_context(context); |
| child->set_type(type); |
| } |
| return true; |
| } |
| |
| const std::string* TrieBuilder::StringPointerFromContainer(const std::string& string, |
| std::set<std::string>* container) { |
| // Get a pointer to the string in a given set, such that we only ever serialize each string once. |
| auto [iterator, _] = container->emplace(string); |
| return &(*iterator); |
| } |
| |
| } // namespace properties |
| } // namespace android |