From 8c825f53feba3582a4b22d759c054f1b71d1e095 Mon Sep 17 00:00:00 2001 From: Bradley Dice Date: Tue, 12 Apr 2022 13:39:40 -0700 Subject: [PATCH] Remove `concurrent_unordered_multimap`. (#10642) The `concurrent_unordered_multimap` is no longer used in libcudf. It has been replaced by `cuco::static_multimap`. The majority of the refactoring was done in PRs #8934 and #9704. A similar effort is in progress for `concurrent_unordered_map` and `cuco::static_map` in #9666 (and may depend on porting some optimizations from libcudf to cuco -- need to look into this before doing a direct replacement). This partially resolves issue #10401. cc: @PointKernel @vyasr Authors: - Bradley Dice (https://github.com/bdice) Approvers: - Yunsong Wang (https://github.com/PointKernel) - Vyas Ramasubramani (https://github.com/vyasr) - Jake Hemstad (https://github.com/jrhemstad) URL: https://github.com/rapidsai/cudf/pull/10642 --- .../hash/concurrent_unordered_multimap.cuh | 592 ------------------ cpp/tests/CMakeLists.txt | 2 +- cpp/tests/hash_map/multimap_test.cu | 95 --- 3 files changed, 1 insertion(+), 688 deletions(-) delete mode 100644 cpp/src/hash/concurrent_unordered_multimap.cuh delete mode 100644 cpp/tests/hash_map/multimap_test.cu diff --git a/cpp/src/hash/concurrent_unordered_multimap.cuh b/cpp/src/hash/concurrent_unordered_multimap.cuh deleted file mode 100644 index aa5b8db393f..00000000000 --- a/cpp/src/hash/concurrent_unordered_multimap.cuh +++ /dev/null @@ -1,592 +0,0 @@ -/* - * Copyright (c) 2017-2022, NVIDIA CORPORATION. - * - * 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 CONCURRENT_UNORDERED_MULTIMAP_CUH -#define CONCURRENT_UNORDERED_MULTIMAP_CUH - -#include -#include -#include - -#include -#include -#include -#include - -#include - -#include - -#include -#include -#include -#include - -/** - * Does support concurrent insert, but not concurrent insert and probing. - * - * @note The user is responsible for the following stream semantics: - * - Either the same stream should be used to create the map as is used by the kernels that access - * it, or - * - the stream used to create the map should be synchronized before it is accessed from a different - * stream or from host code. - * - * TODO: - * - add constructor that takes pointer to hash_table to avoid allocations - */ -template , - typename Equality = equal_to, - typename Allocator = managed_allocator>, - bool count_collisions = false> -class concurrent_unordered_multimap { - public: - using hasher = Hasher; - using key_equal = Equality; - using allocator_type = Allocator; - using key_type = Key; - using value_type = thrust::pair; - using mapped_type = Element; - using iterator = cycle_iterator_adapter; - using const_iterator = const cycle_iterator_adapter; - - private: - union pair2longlong { - unsigned long long int longlong; - value_type pair; - }; - - public: - /** - * @brief Factory to construct a new concurrent unordered multimap. - * - * Returns a `std::unique_ptr` to a new concurrent unordered multimap object. - * The map is non-owning and trivially copyable and should be passed by value - * into kernels. The `unique_ptr` contains a custom deleter that will free the - * map's contents. - * - * @note The implementation of this multimap uses sentinel values to - * indicate an entry in the hash table that is empty, i.e., if a hash bucket - * is empty, the pair residing there will be equal to (unused_key, - * unused_element). As a result, attempting to insert a key equal to - * `unused_key` results in undefined behavior. - * - * @note All allocations, kernels and copies in the constructor take place - * on stream but the constructor does not synchronize the stream. It is the user's - * responsibility to synchronize or use the same stream to access the map. - * - * @param capacity The maximum number of pairs the map may hold. - * @param stream CUDA stream used for device memory operations and kernel launches. - * @param init Indicates if the map should be initialized with the unused - * key/values - * @param hash_function The hash function to use for hashing keys - * @param equal The equality comparison function for comparing if two keys are - * equal - * @param allocator The allocator to use for allocation of the map's storage - */ - static auto create(size_type capacity, - rmm::cuda_stream_view stream = rmm::cuda_stream_default, - const bool init = true, - const Hasher& hash_function = hasher(), - const Equality& equal = key_equal(), - const allocator_type& allocator = allocator_type()) - { - CUDF_FUNC_RANGE(); - using Self = concurrent_unordered_multimap; - - // Note: need `(*p).destroy` instead of `p->destroy` here - // due to compiler bug: https://github.com/rapidsai/cudf/pull/5692 - auto deleter = [stream](Self* p) { (*p).destroy(stream); }; - - return std::unique_ptr>{ - new Self(capacity, init, hash_function, equal, allocator, stream), deleter}; - } - - /** - * @brief Frees the contents of the map and destroys the map object. - * - * This function is invoked as the deleter of the `std::unique_ptr` returned - * from the `create()` factory function. - * - * @param stream CUDA stream used for device memory operations and kernel launches. - */ - void destroy(rmm::cuda_stream_view stream = rmm::cuda_stream_default) - { - m_allocator.deallocate(m_hashtbl_values, m_hashtbl_capacity, stream); - delete this; - } - - /** - * @brief Returns an iterator to the first element in the map - * - * @note When using the managed allocator, host code that calls this function - * should ensure the stream used for `create()` is appropriately synchronized. - * - * @note When called in a device code, user should make sure that it should - * either be running on the same stream as create(), or the accessing stream - * should be appropriately synchronized with the creating stream. - * - * @returns iterator to the first element in the map. - */ - __host__ __device__ iterator begin() - { - return iterator(m_hashtbl_values, m_hashtbl_values + m_hashtbl_size, m_hashtbl_values); - } - - /** - * @brief Returns a constant iterator to the first element in the map - * - * @note When using the managed allocator, host code that calls this function - * should ensure the stream used for `create()` is appropriately synchronized. - * - * @note When called in a device code, user should make sure that it should - * either be running on the same stream as create(), or the accessing stream - * should be appropriately synchronized with the creating stream. - * - * @returns constant iterator to the first element in the map. - */ - __host__ __device__ const_iterator begin() const - { - return const_iterator(m_hashtbl_values, m_hashtbl_values + m_hashtbl_size, m_hashtbl_values); - } - - /** - * @brief Returns an iterator to the one past the last element in the map - * - * @note When using the managed allocator, host code that calls this function - * should ensure the stream used for `create()` is appropriately synchronized. - * - * @note When called in a device code, user should make sure that it should - * either be running on the same stream as create(), or the accessing stream - * should be appropriately synchronized with the creating stream. - * - * @returns iterator to the one past the last element in the map. - */ - __host__ __device__ iterator end() - { - return iterator( - m_hashtbl_values, m_hashtbl_values + m_hashtbl_size, m_hashtbl_values + m_hashtbl_size); - } - - /** - * @brief Returns a constant iterator to the one past the last element in the map - * - * @note When using the managed allocator, host code that calls this function - * should ensure the stream used for `create()` is appropriately synchronized. - * - * @note When called in a device code, user should make sure that it should - * either be running on the same stream as create(), or the accessing stream - * should be appropriately synchronized with the creating stream. - * - * @returns constant iterator to the one past the last element in the map. - */ - __host__ __device__ const_iterator end() const - { - return const_iterator( - m_hashtbl_values, m_hashtbl_values + m_hashtbl_size, m_hashtbl_values + m_hashtbl_size); - } - - __forceinline__ static constexpr __host__ __device__ key_type get_unused_key() - { - return unused_key; - } - - /** - * @brief Computes a hash value for a key - * - * @param[in] the_key The key to compute a hash for - * @tparam hash_value_type The datatype of the hash value - * - * @returns The hash value for the key - */ - template - __forceinline__ __host__ __device__ hash_value_type get_hash(const key_type& the_key) const - { - return m_hf(the_key); - } - - /** - * @brief Computes the destination hash map partition for a key - * - * @param[in] the_key The key to search for - * @param[in] num_parts The total number of partitions in the partitioned - * hash table - * @param[in] precomputed_hash A flag indicating whether or not a precomputed - * hash value is passed in - * @param[in] precomputed_hash_value A precomputed hash value to use for - * determining the write location of the key into the hash map instead of - * computing the the hash value directly from the key - * @tparam hash_value_type The datatype of the hash value - * - * @returns The destination hash table partition for the specified key - */ - template - __forceinline__ __host__ __device__ int get_partition( - const key_type& the_key, - const int num_parts = 1, - bool precomputed_hash = false, - hash_value_type precomputed_hash_value = 0) const - { - hash_value_type hash_value{0}; - - // If a precomputed hash value has been passed in, then use it to determine - // the location of the key - if (true == precomputed_hash) { - hash_value = precomputed_hash_value; - } - // Otherwise, compute the hash value from the key - else { - hash_value = m_hf(the_key); - } - - size_type hash_tbl_idx = hash_value % m_hashtbl_size; - - const size_type partition_size = m_hashtbl_size / num_parts; - - int dest_part = hash_tbl_idx / partition_size; - // Note that if m_hashtbl_size % num_parts != 0 then dest_part can be - // num_parts for the last few elements and we remap that to the - // num_parts-1 partition - if (dest_part == num_parts) dest_part = num_parts - 1; - - return dest_part; - } - - /** - * @brief Inserts a (key, value) pair into the hash map - * - * @param[in] x The (key, value) pair to insert - * @param[in] precomputed_hash A flag indicating whether or not a precomputed - * hash value is passed in - * @param[in] precomputed_hash_value A precomputed hash value to use for - * determining the write location of the key into the hash map instead of - * computing the the hash value directly from the key - * @param[in] keys_are_equal An optional functor for comparing if two keys are - * equal - * @tparam hash_value_type The datatype of the hash value - * @tparam comparison_type The type of the key comparison functor - * - * @returns An iterator to the newly inserted (key, value) pair - */ - template - __forceinline__ __device__ iterator insert(const value_type& x, - bool precomputed_hash = false, - hash_value_type precomputed_hash_value = 0, - comparison_type keys_are_equal = key_equal()) - { - const size_type hashtbl_size = m_hashtbl_size; - value_type* hashtbl_values = m_hashtbl_values; - - hash_value_type hash_value{0}; - - // If a precomputed hash value has been passed in, then use it to determine - // the write location of the new key - if (true == precomputed_hash) { - hash_value = precomputed_hash_value; - } - // Otherwise, compute the hash value from the new key - else { - hash_value = m_hf(x.first); - } - - size_type hash_tbl_idx = hash_value % hashtbl_size; - - value_type* it = 0; - - size_type attempt_counter{0}; - - while (0 == it) { - value_type* tmp_it = hashtbl_values + hash_tbl_idx; - - if (std::numeric_limits::is_integer && - std::numeric_limits::is_integer && - sizeof(unsigned long long int) == sizeof(value_type)) { - pair2longlong converter = {0ull}; - converter.pair = thrust::make_pair(unused_key, unused_element); - const unsigned long long int unused = converter.longlong; - converter.pair = x; - const unsigned long long int value = converter.longlong; - const unsigned long long int old_val = - atomicCAS(reinterpret_cast(tmp_it), unused, value); - if (old_val == unused) { - it = tmp_it; - } else if (count_collisions) { - atomicAdd(&m_collisions, 1); - } - } else { - const key_type old_key = atomicCAS(&(tmp_it->first), unused_key, x.first); - - if (keys_are_equal(unused_key, old_key)) { - (m_hashtbl_values + hash_tbl_idx)->second = x.second; - it = tmp_it; - } else if (count_collisions) { - atomicAdd(&m_collisions, 1); - } - } - - hash_tbl_idx = (hash_tbl_idx + 1) % hashtbl_size; - - attempt_counter++; - if (attempt_counter > hashtbl_size) { - printf("Attempted to insert to multimap but the map is full!\n"); - return this->end(); - } - } - - return iterator(m_hashtbl_values, m_hashtbl_values + hashtbl_size, it); - } - - /** - * @brief Inserts a (key, value) pair into the hash map partition. This - * is useful when building the hash table in multiple passes, one - * contiguous partition at a time, or when building the hash table - * distributed between multiple devices. - * - * @param[in] x The (key, value) pair to insert - * @param[in] part The partition number for the partitioned hash table build - * @param[in] num_parts The total number of partitions in the partitioned - * hash table - * @param[in] precomputed_hash A flag indicating whether or not a precomputed - * hash value is passed in - * @param[in] precomputed_hash_value A precomputed hash value to use for - * determining the write location of the key into the hash map instead of - * computing the the hash value directly from the key - * @param[in] keys_are_equal An optional functor for comparing if two keys are - * equal - * @tparam hash_value_type The datatype of the hash value - * @tparam comparison_type The type of the key comparison functor - * - * @returns An iterator to the newly inserted (key, value) pair - */ - template - __forceinline__ __device__ iterator insert_part(const value_type& x, - const int part = 0, - const int num_parts = 1, - bool precomputed_hash = false, - hash_value_type precomputed_hash_value = 0, - comparison_type keys_are_equal = key_equal()) - { - hash_value_type hash_value{0}; - - // If a precomputed hash value has been passed in, then use it to determine - // the write location of the new key - if (true == precomputed_hash) { - hash_value = precomputed_hash_value; - } - // Otherwise, compute the hash value from the new key - else { - hash_value = m_hf(x.first); - } - - // Find the destination partition index - int dest_part = get_partition(x.first, num_parts, true, hash_value); - - // Only insert if the key belongs to the specified partition - if (dest_part != part) - return end(); - else - return insert(x, true, hash_value, keys_are_equal); - } - - /** - * @brief Searches for a key in the hash map and returns an iterator to the - * first instance of the key in the map. - * - * @param[in] the_key The key to search for - * @param[in] precomputed_hash A flag indicating whether or not a precomputed - * hash value is passed in - * @param[in] precomputed_hash_value A precomputed hash value to use for - * determining the write location of the key into the hash map instead of - * computing the the hash value directly from the key - * @param[in] keys_are_equal An optional functor for comparing if two keys are - * equal - * @tparam hash_value_type The datatype of the hash value - * @tparam comparison_type The type of the key comparison functor - * - * @returns An iterator to the first instance of the key in the map - */ - template - __forceinline__ __host__ __device__ const_iterator - find(const key_type& the_key, - bool precomputed_hash = false, - hash_value_type precomputed_hash_value = 0, - comparison_type keys_are_equal = key_equal()) const - { - hash_value_type hash_value{0}; - - // If a precomputed hash value has been passed in, then use it to determine - // the location of the key - if (true == precomputed_hash) { - hash_value = precomputed_hash_value; - } - // Otherwise, compute the hash value from the key - else { - hash_value = m_hf(the_key); - } - - size_type hash_tbl_idx = hash_value % m_hashtbl_size; - - value_type* begin_ptr = 0; - - size_type counter = 0; - while (0 == begin_ptr) { - value_type* tmp_ptr = m_hashtbl_values + hash_tbl_idx; - const key_type tmp_val = tmp_ptr->first; - if (keys_are_equal(the_key, tmp_val)) { - begin_ptr = tmp_ptr; - break; - } - if (keys_are_equal(unused_key, tmp_val) || (counter > m_hashtbl_size)) { - begin_ptr = m_hashtbl_values + m_hashtbl_size; - break; - } - hash_tbl_idx = (hash_tbl_idx + 1) % m_hashtbl_size; - ++counter; - } - - return const_iterator(m_hashtbl_values, m_hashtbl_values + m_hashtbl_size, begin_ptr); - } - - void assign_async(const concurrent_unordered_multimap& other, - rmm::cuda_stream_view stream = rmm::cuda_stream_default) - { - m_collisions = other.m_collisions; - if (other.m_hashtbl_size <= m_hashtbl_capacity) { - m_hashtbl_size = other.m_hashtbl_size; - } else { - m_allocator.deallocate(m_hashtbl_values, m_hashtbl_capacity, stream); - m_hashtbl_capacity = other.m_hashtbl_size; - m_hashtbl_size = other.m_hashtbl_size; - - m_hashtbl_values = m_allocator.allocate(m_hashtbl_capacity, stream); - } - CUDF_CUDA_TRY(cudaMemcpyAsync(m_hashtbl_values, - other.m_hashtbl_values, - m_hashtbl_size * sizeof(value_type), - cudaMemcpyDefault, - stream.value())); - } - - void clear_async(rmm::cuda_stream_view stream = rmm::cuda_stream_default) - { - constexpr int block_size = 128; - init_hashtbl<<<((m_hashtbl_size - 1) / block_size) + 1, block_size, 0, stream.value()>>>( - m_hashtbl_values, m_hashtbl_size, unused_key, unused_element); - if (count_collisions) m_collisions = 0; - } - - [[nodiscard]] unsigned long long get_num_collisions() const { return m_collisions; } - - void print() - { - for (size_type i = 0; i < m_hashtbl_size; ++i) { - std::cout << i << ": " << m_hashtbl_values[i].first << "," << m_hashtbl_values[i].second - << std::endl; - } - } - - void prefetch(const int dev_id, rmm::cuda_stream_view stream = rmm::cuda_stream_default) - { - cudaPointerAttributes hashtbl_values_ptr_attributes; - cudaError_t status = cudaPointerGetAttributes(&hashtbl_values_ptr_attributes, m_hashtbl_values); - - if (cudaSuccess == status && isPtrManaged(hashtbl_values_ptr_attributes)) { - CUDF_CUDA_TRY(cudaMemPrefetchAsync( - m_hashtbl_values, m_hashtbl_size * sizeof(value_type), dev_id, stream.value())); - } - } - - concurrent_unordered_multimap() = delete; - concurrent_unordered_multimap(concurrent_unordered_multimap const&) = default; - concurrent_unordered_multimap(concurrent_unordered_multimap&&) = default; - concurrent_unordered_multimap& operator=(concurrent_unordered_multimap const&) = default; - concurrent_unordered_multimap& operator=(concurrent_unordered_multimap&&) = default; - ~concurrent_unordered_multimap() = default; - - private: - hasher m_hf; - key_equal m_equal; - allocator_type m_allocator; - size_type m_hashtbl_size; - size_type m_hashtbl_capacity; - value_type* m_hashtbl_values; - unsigned long long m_collisions; - - /** - * @brief Private constructor used by `create` factory function. - * - * Allocates memory and optionally fills the hash map with unused - * keys/values - * - * @param[in] n The size of the hash table (the number of key-value pairs) - * @param[in] init Initialize the hash table with the unused keys/values - * @param[in] hash_function An optional hashing function - * @param[in] equal An optional functor for comparing if two keys are equal - * @param[in] a An optional functor for allocating the hash table memory - * @param[in] stream CUDA stream used for device memory operations and kernel launches. - */ - explicit concurrent_unordered_multimap(size_type n, - const bool init = true, - const Hasher& hash_function = hasher(), - const Equality& equal = key_equal(), - const allocator_type& a = allocator_type(), - rmm::cuda_stream_view stream = rmm::cuda_stream_default) - : m_hf(hash_function), - m_equal(equal), - m_allocator(a), - m_hashtbl_size(n), - m_hashtbl_capacity(n), - m_collisions(0) - { - m_hashtbl_values = m_allocator.allocate(m_hashtbl_capacity, stream); - constexpr int block_size = 128; - { - cudaPointerAttributes hashtbl_values_ptr_attributes; - cudaError_t status = - cudaPointerGetAttributes(&hashtbl_values_ptr_attributes, m_hashtbl_values); - - if (cudaSuccess == status && isPtrManaged(hashtbl_values_ptr_attributes)) { - int dev_id = 0; - CUDF_CUDA_TRY(cudaGetDevice(&dev_id)); - CUDF_CUDA_TRY(cudaMemPrefetchAsync( - m_hashtbl_values, m_hashtbl_size * sizeof(value_type), dev_id, stream.value())); - } - } - - if (init) { - init_hashtbl<<<((m_hashtbl_size - 1) / block_size) + 1, block_size, 0, stream.value()>>>( - m_hashtbl_values, m_hashtbl_size, unused_key, unused_element); - CUDF_CHECK_CUDA(stream.value()); - } - } -}; - -#endif // CONCURRENT_UNORDERED_MULTIMAP_CUH diff --git a/cpp/tests/CMakeLists.txt b/cpp/tests/CMakeLists.txt index b28aac659d9..1ed921d1f08 100644 --- a/cpp/tests/CMakeLists.txt +++ b/cpp/tests/CMakeLists.txt @@ -138,7 +138,7 @@ ConfigureTest( # ################################################################################################## # * hash_map tests -------------------------------------------------------------------------------- -ConfigureTest(HASH_MAP_TEST hash_map/map_test.cu hash_map/multimap_test.cu) +ConfigureTest(HASH_MAP_TEST hash_map/map_test.cu) # ################################################################################################## # * quantiles tests ------------------------------------------------------------------------------- diff --git a/cpp/tests/hash_map/multimap_test.cu b/cpp/tests/hash_map/multimap_test.cu deleted file mode 100644 index b8f35b4d404..00000000000 --- a/cpp/tests/hash_map/multimap_test.cu +++ /dev/null @@ -1,95 +0,0 @@ -/* - * Copyright (c) 2018-2022, NVIDIA CORPORATION. - * - * 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 -#include - -#include -#include - -#include - -#include - -#include - -#include - -// This is necessary to do a parametrized typed-test over multiple template -// arguments -template -struct KeyValueTypes { - using key_type = Key; - using value_type = Value; -}; - -// A new instance of this class will be created for each *TEST(MultimapTest, -// ...) Put all repeated stuff for each test here -template -class MultimapTest : public cudf::test::BaseFixture { - public: - using key_type = typename T::key_type; - using value_type = typename T::value_type; - using size_type = int; - - using multimap_type = - concurrent_unordered_multimap::max(), - std::numeric_limits::max(), - default_hash, - equal_to, - default_allocator>>; - - std::unique_ptr> the_map; - - const key_type unused_key = std::numeric_limits::max(); - const value_type unused_value = std::numeric_limits::max(); - - const size_type size; - - MultimapTest(const size_type hash_table_size = 100) - : the_map(multimap_type::create(hash_table_size)), size(hash_table_size) - { - rmm::cuda_stream_default.synchronize(); - } - - ~MultimapTest() override {} -}; - -// Google Test can only do a parameterized typed-test over a single type, so we -// have to nest multiple types inside of the KeyValueTypes struct above -// KeyValueTypes implies key_type = type1, value_type = type2 -// This list is the types across which Google Test will run our tests -using Implementations = ::testing::Types, - KeyValueTypes, - KeyValueTypes, - KeyValueTypes, - KeyValueTypes, - KeyValueTypes>; - -TYPED_TEST_SUITE(MultimapTest, Implementations); - -TYPED_TEST(MultimapTest, InitialState) -{ - using key_type = typename TypeParam::key_type; - using value_type = typename TypeParam::value_type; - - auto begin = this->the_map->begin(); - auto end = this->the_map->end(); - EXPECT_NE(begin, end); -}