diff options
| author | David 'Digit' Turner <digit@google.com> | 2014-09-15 19:34:57 +0200 | 
|---|---|---|
| committer | David 'Digit' Turner <digit@google.com> | 2014-09-16 03:01:26 +0200 | 
| commit | 9f35027153ad98865b57ef0fcd86940c44afd3e6 (patch) | |
| tree | 3e09a7012ef4d7332c03d8dabe7a5f5d9580ba84 /emulator/opengl/shared/emugl/common/unique_integer_map.h | |
| parent | a92a69909f5bbbbd980cc9551dd85e961fd02f82 (diff) | |
| download | sdk-9f35027153ad98865b57ef0fcd86940c44afd3e6.zip sdk-9f35027153ad98865b57ef0fcd86940c44afd3e6.tar.gz sdk-9f35027153ad98865b57ef0fcd86940c44afd3e6.tar.bz2 | |
emulator/opengl: Add UniqueIntegerMap class.
This helper class will be used by future patches to map arbitraty
opaque 'void*' values (e.g. EGLImage values) to 32-bit unique values
that can be sent through the wire protocol.
Change-Id: I5b17a1f230e5f9f8b905af66ba79b312f3f01e55
Diffstat (limited to 'emulator/opengl/shared/emugl/common/unique_integer_map.h')
| -rw-r--r-- | emulator/opengl/shared/emugl/common/unique_integer_map.h | 225 | 
1 files changed, 225 insertions, 0 deletions
| diff --git a/emulator/opengl/shared/emugl/common/unique_integer_map.h b/emulator/opengl/shared/emugl/common/unique_integer_map.h new file mode 100644 index 0000000..720aceb --- /dev/null +++ b/emulator/opengl/shared/emugl/common/unique_integer_map.h @@ -0,0 +1,225 @@ +// Copyright (C) 2014 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 EMUGL_COMMON_UNIQUE_INTEGER_MAP_H +#define EMUGL_COMMON_UNIQUE_INTEGER_MAP_H + +#include "emugl/common/pod_vector.h" + +#include <stdint.h> + +namespace emugl { + +// Helper template class that implements a bi-directional mapping between +// two integer types |A| and |B|. More specifically: +// +// - The map allocates values of type |B| when a key of type |A| is entered +//   in the map. +// +// - keys and values cannot be 0, which is reserved (i.e. means 'invalid'). +// +// This is used in EmuGL to map liberal 'void*' values (e.g. EGLimages ones) +// to unique 32-bit IDs that can be written to / read from the wire protocol. +template <typename A, typename B> +class UniqueIntegerMap { +public: +    UniqueIntegerMap() : mForwardPairs(), mBackwardPairs() {} +    ~UniqueIntegerMap() {} + +    // Return true iff the map is empty. +    const bool empty() const { return mForwardPairs.empty(); } + +    // Return the number of (key,value) pairs in the map. +    size_t size() const { return mForwardPairs.size(); } + +    // Find the value associated with |key| in the map. +    // Returns 0 in case of failure, or if |key| is 0. +    B find(const A key) const; + +    // Find the key associated with a given |value| in the map. +    // Returns 0 if |value| is 0, or in case of failure. +    A findKeyFor(const B value) const; + +    // Add |key| to the map and return an automatically-allocated +    // unique value for it. Return 0 if |key| is 0. +    B add(const A key); + +    // Delete the entry associated with a given |key|. The +    // corresponding value may be recycled by future calls to add(). +    void del(const A key); + +private: +    typedef struct { +        A first; +        B second; +    } ForwardPair; + +    typedef struct { +        B first; +        A second; +    } BackwardPair; + +    size_t findKeyIndexPlusOne(const A key) const; +    size_t findValueIndexPlusOne(const B value) const; + +    B allocValue(); +    void freeValue(B value); + +    PodVector<ForwardPair> mForwardPairs; +    PodVector<BackwardPair> mBackwardPairs; + +    B mLastValue; +    PodVector<B> mFreeValues; +}; + +template <typename A, typename B> +B UniqueIntegerMap<A,B>::find(const A key) const { +    size_t keyIndex = findKeyIndexPlusOne(key); +    if (!keyIndex) { +        return 0; +    } +    return mForwardPairs[keyIndex - 1U].second; +} + +template <typename A, typename B> +A UniqueIntegerMap<A,B>::findKeyFor(const B value) const { +    size_t valueIndex = findValueIndexPlusOne(value); +    if (!valueIndex) { +        return 0; +    } +    return mBackwardPairs[valueIndex - 1U].second; +} + +template <typename A, typename B> +B UniqueIntegerMap<A,B>::add(const A key) { +    // Binary search to find the proper insertion point for the key. +    // Also checks that the key isn't already in the set. +    size_t min = 0; +    size_t max = mForwardPairs.size(); +    while (min < max) { +        size_t mid = min + ((max - min) >> 1); +        A midKey = mForwardPairs[mid].first; +        if (midKey < key) { +            min = mid + 1U; +        } else if (midKey > key) { +            max = mid; +        } else { +            // Already in the set. +            return 0; +        } +    } + +    // Generate new unique value +    B value = allocValue(); + +    ForwardPair* pair = mForwardPairs.emplace(min); +    pair->first = key; +    pair->second = value; + +    // Binary search to find proper insertion point for the value. +    min = 0; +    max = mBackwardPairs.size(); +    while (min < max) { +        size_t mid = min + ((max - min) >> 1); +        B midValue = mBackwardPairs[mid].first; +        if (midValue < value) { +            min = mid + 1U; +        } else { +            max = mid; +        } +    } + +    BackwardPair* backPair = mBackwardPairs.emplace(min); +    backPair->first = value; +    backPair->second = key; + +    return value; +} + +template <typename A, typename B> +void UniqueIntegerMap<A,B>::del(const A key) { +    size_t keyIndex = findKeyIndexPlusOne(key); +    if (!keyIndex) { +        return; +    } +    B value = mForwardPairs[keyIndex - 1U].second; +    size_t valueIndex = findValueIndexPlusOne(value); +    mForwardPairs.remove(keyIndex - 1U); +    mBackwardPairs.remove(valueIndex - 1U); +    freeValue(value); +} + +template <typename A, typename B> +size_t UniqueIntegerMap<A,B>::findKeyIndexPlusOne(const A key) const { +    // Binary search in forward pair array. +    size_t min = 0; +    size_t max = mForwardPairs.size(); +    while (min < max) { +        size_t mid = min + ((max - min) >> 1); +        A midKey = mForwardPairs[mid].first; +        if (midKey < key) { +            min = mid + 1U; +        } else if (midKey > key) { +            max = mid; +        } else { +            return mid + 1U; +        } +    } +    return 0U; +} + +template <typename A, typename B> +size_t UniqueIntegerMap<A,B>::findValueIndexPlusOne(const B value) const { +    // Binary search in revere pair array. +    size_t min = 0; +    size_t max = mBackwardPairs.size(); +    while (min < max) { +        size_t mid = min + ((max - min) >> 1); +        B midValue = mBackwardPairs[mid].first; +        if (midValue < value) { +            min = mid + 1U; +        } else if (midValue > value) { +            max = mid; +        } else { +            return mid + 1U; +        } +    } +    return 0U; +} + +template <typename A, typename B> +B UniqueIntegerMap<A,B>::allocValue() { +    if (!mFreeValues.empty()) { +        B result = mFreeValues[0]; +        mFreeValues.pop(); +        return result; +    } +    return ++mLastValue; +} + +template <typename A, typename B> +void UniqueIntegerMap<A,B>::freeValue(B value) { +    if (!value) { +        return; +    } +    if (value == mLastValue) { +        mLastValue--; +        return; +    } +    mFreeValues.append(value); +} + +}  // namespace emugl + +#endif  // EMUGL_COMMON_INTEGER_MAP_H | 
