| /* | 
 |  * Copyright 2020 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. | 
 |  */ | 
 |  | 
 | #pragma once | 
 |  | 
 | #include <ui/DisplayId.h> | 
 |  | 
 | #include <limits> | 
 | #include <optional> | 
 | #include <random> | 
 | #include <unordered_set> | 
 |  | 
 | #include <log/log.h> | 
 |  | 
 | namespace android { | 
 |  | 
 | template <typename T> | 
 | class DisplayIdGenerator { | 
 | public: | 
 |     virtual std::optional<T> nextId() = 0; | 
 |     virtual void markUnused(T id) = 0; | 
 |  | 
 | protected: | 
 |     ~DisplayIdGenerator() {} | 
 | }; | 
 |  | 
 | template <typename T> | 
 | class RandomDisplayIdGenerator final : public DisplayIdGenerator<T> { | 
 | public: | 
 |     explicit RandomDisplayIdGenerator(size_t maxIdsCount = std::numeric_limits<size_t>::max()) | 
 |           : mMaxIdsCount(maxIdsCount) {} | 
 |  | 
 |     std::optional<T> nextId() override { | 
 |         if (mUsedIds.size() >= mMaxIdsCount) { | 
 |             return std::nullopt; | 
 |         } | 
 |  | 
 |         constexpr int kMaxAttempts = 1000; | 
 |  | 
 |         for (int attempts = 0; attempts < kMaxAttempts; attempts++) { | 
 |             const auto baseId = mDistribution(mGenerator); | 
 |             const T id(baseId); | 
 |             if (mUsedIds.count(id) == 0) { | 
 |                 mUsedIds.insert(id); | 
 |                 return id; | 
 |             } | 
 |         } | 
 |  | 
 |         LOG_ALWAYS_FATAL("Couldn't generate ID after %d attempts", kMaxAttempts); | 
 |     } | 
 |  | 
 |     void markUnused(T id) override { mUsedIds.erase(id); } | 
 |  | 
 | private: | 
 |     const size_t mMaxIdsCount; | 
 |  | 
 |     std::unordered_set<T> mUsedIds; | 
 |     std::default_random_engine mGenerator{std::random_device()()}; | 
 |     std::uniform_int_distribution<typename T::BaseId> mDistribution; | 
 | }; | 
 |  | 
 | } // namespace android |