//---------------------------------------------------------------------------// // Copyright (c) 2013 Kyle Lutz // // Distributed under the Boost Software License, Version 1.0 // See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt // // See http://boostorg.github.com/compute for more information. //---------------------------------------------------------------------------// #ifndef BOOST_COMPUTE_CONTAINER_FLAT_MAP_HPP #define BOOST_COMPUTE_CONTAINER_FLAT_MAP_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace boost { namespace compute { template class flat_map { public: typedef Key key_type; typedef T mapped_type; typedef typename ::boost::compute::vector > vector_type; typedef typename vector_type::value_type value_type; typedef typename vector_type::size_type size_type; typedef typename vector_type::difference_type difference_type; typedef typename vector_type::reference reference; typedef typename vector_type::const_reference const_reference; typedef typename vector_type::pointer pointer; typedef typename vector_type::const_pointer const_pointer; typedef typename vector_type::iterator iterator; typedef typename vector_type::const_iterator const_iterator; typedef typename vector_type::reverse_iterator reverse_iterator; typedef typename vector_type::const_reverse_iterator const_reverse_iterator; explicit flat_map(const context &context = system::default_context()) : m_vector(context) { } flat_map(const flat_map &other) : m_vector(other.m_vector) { } flat_map& operator=(const flat_map &other) { if(this != &other){ m_vector = other.m_vector; } return *this; } ~flat_map() { } iterator begin() { return m_vector.begin(); } const_iterator begin() const { return m_vector.begin(); } const_iterator cbegin() const { return m_vector.cbegin(); } iterator end() { return m_vector.end(); } const_iterator end() const { return m_vector.end(); } const_iterator cend() const { return m_vector.cend(); } reverse_iterator rbegin() { return m_vector.rbegin(); } const_reverse_iterator rbegin() const { return m_vector.rbegin(); } const_reverse_iterator crbegin() const { return m_vector.crbegin(); } reverse_iterator rend() { return m_vector.rend(); } const_reverse_iterator rend() const { return m_vector.rend(); } const_reverse_iterator crend() const { return m_vector.crend(); } size_type size() const { return m_vector.size(); } size_type max_size() const { return m_vector.max_size(); } bool empty() const { return m_vector.empty(); } size_type capacity() const { return m_vector.capacity(); } void reserve(size_type size, command_queue &queue) { m_vector.reserve(size, queue); } void reserve(size_type size) { command_queue queue = m_vector.default_queue(); reserve(size, queue); queue.finish(); } void shrink_to_fit() { m_vector.shrink_to_fit(); } void clear() { m_vector.clear(); } std::pair insert(const value_type &value, command_queue &queue) { iterator location = upper_bound(value.first, queue); if(location != begin()){ value_type current_value; ::boost::compute::copy_n(location - 1, 1, ¤t_value, queue); if(value.first == current_value.first){ return std::make_pair(location - 1, false); } } m_vector.insert(location, value); return std::make_pair(location, true); } std::pair insert(const value_type &value) { command_queue queue = m_vector.default_queue(); std::pair result = insert(value, queue); queue.finish(); return result; } iterator erase(const const_iterator &position, command_queue &queue) { return erase(position, position + 1, queue); } iterator erase(const const_iterator &position) { command_queue queue = m_vector.default_queue(); iterator iter = erase(position, queue); queue.finish(); return iter; } iterator erase(const const_iterator &first, const const_iterator &last, command_queue &queue) { return m_vector.erase(first, last, queue); } iterator erase(const const_iterator &first, const const_iterator &last) { command_queue queue = m_vector.default_queue(); iterator iter = erase(first, last, queue); queue.finish(); return iter; } size_type erase(const key_type &value, command_queue &queue) { iterator position = find(value, queue); if(position == end()){ return 0; } else { erase(position, queue); return 1; } } iterator find(const key_type &value, command_queue &queue) { ::boost::compute::get<0> get_key; return ::boost::compute::find( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } iterator find(const key_type &value) { command_queue queue = m_vector.default_queue(); iterator iter = find(value, queue); queue.finish(); return iter; } const_iterator find(const key_type &value, command_queue &queue) const { ::boost::compute::get<0> get_key; return ::boost::compute::find( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } const_iterator find(const key_type &value) const { command_queue queue = m_vector.default_queue(); const_iterator iter = find(value, queue); queue.finish(); return iter; } size_type count(const key_type &value, command_queue &queue) const { return find(value, queue) != end() ? 1 : 0; } size_type count(const key_type &value) const { command_queue queue = m_vector.default_queue(); size_type result = count(value, queue); queue.finish(); return result; } iterator lower_bound(const key_type &value, command_queue &queue) { ::boost::compute::get<0> get_key; return ::boost::compute::lower_bound( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } iterator lower_bound(const key_type &value) { command_queue queue = m_vector.default_queue(); iterator iter = lower_bound(value, queue); queue.finish(); return iter; } const_iterator lower_bound(const key_type &value, command_queue &queue) const { ::boost::compute::get<0> get_key; return ::boost::compute::lower_bound( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } const_iterator lower_bound(const key_type &value) const { command_queue queue = m_vector.default_queue(); const_iterator iter = lower_bound(value, queue); queue.finish(); return iter; } iterator upper_bound(const key_type &value, command_queue &queue) { ::boost::compute::get<0> get_key; return ::boost::compute::upper_bound( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } iterator upper_bound(const key_type &value) { command_queue queue = m_vector.default_queue(); iterator iter = upper_bound(value, queue); queue.finish(); return iter; } const_iterator upper_bound(const key_type &value, command_queue &queue) const { ::boost::compute::get<0> get_key; return ::boost::compute::upper_bound( ::boost::compute::make_transform_iterator(begin(), get_key), ::boost::compute::make_transform_iterator(end(), get_key), value, queue ).base(); } const_iterator upper_bound(const key_type &value) const { command_queue queue = m_vector.default_queue(); const_iterator iter = upper_bound(value, queue); queue.finish(); return iter; } const mapped_type at(const key_type &key) const { const_iterator iter = find(key); if(iter == end()){ BOOST_THROW_EXCEPTION(std::out_of_range("key not found")); } return value_type(*iter).second; } detail::buffer_value operator[](const key_type &key) { iterator iter = find(key); if(iter == end()){ iter = insert(std::make_pair(key, mapped_type())).first; } size_t index = iter.get_index() * sizeof(value_type) + sizeof(key_type); return detail::buffer_value(m_vector.get_buffer(), index); } private: ::boost::compute::vector > m_vector; }; } // end compute namespace } // end boost namespace #endif // BOOST_COMPUTE_CONTAINER_FLAT_MAP_HPP