f074d94 Use shorter path names in Doxygen doc. 5117d5a Make Doxygen-generated HTML docs work better on small screens. 41607ac Appveyor config: Use lowercase command names everywhere. Filter 7z output. 9e77a39 Remove debug output. 3416c4b Revert "Comment out test to see if there are dependencies between tests." fa49dde Comment out test to see if there are dependencies between tests. 5add75d Fix typo in appveyor conf: nul. 53b3778 Use different file names for tests. 1091c0b Fix Windows munmap() to return the same as the posix one. 14fdc4e Revert "CMake: Remove unneeded build type setting." 0270087 Workaround for MSVC. 56c2120 Another try... 8e96a6f Trying to fix test on Windows... 11a64c2 CMake workaround: Set CMAKE_CONFIGURATION_TYPES before project(). As per http://www.cmake.org/pipermail/cmake/2012-January/048856.html 3847e7a CMake: Remove unneeded build type setting. a1bcaf1 CMake: Make CMAKE_CONFIGURATION_TYPES a cache variable. ddab3e3 Create test file in test_typed_mmap in current dir instead of tmp. ce1d3bd Build "RelWithDebInfo" on appveyor instead of "Release". b6db34f Add README showing source of FindGem.cmake. 2c86c55 Better error message and some comments in CMake for sparsetable check. b16a5a3 Fix ctest on Windows. 2203bc8 Fix FindGem.cmake. 5d1f81d Another try to fix travis build. ddd3f5e Hopefully fix problem where travis doesn't find osm-testdata. 0e673ea More debug output to find travis problem. eb01107 Updated data-tests README. 8a971c8 Remove special case of the multipolygon test in travis/appveyor config. dce792c CMake: Check dependencies of multipolygon test and run as CMake script. b967677 Add tests for thread pool. 400e9b3 Bugfix: Handle exception in one pool threads properly. a1ba489 More detailed error reporting from zlib. 95cf621 Disable SparseMemTable if sparsetable size_type is smaller than 8. 7b601b5 Add missing overload to cast_with_assert() function. 60a7d86 Giving up on trying to remove LNK4221 warning. 35ed5df Try another way to get rid of MSV warning... f9c7d92 Disable a warning about changed behaviour on MSVC. c3c6b2d CTest: Only output failed tests. 9c99996 Try setting option in a different way for getting rid of LNK4221 on MSVC. 4ac563c Another ssize_t fix for Windows. 11db84f Fix mmap for windows. Convert macro to function. 95d8f75 Fix test on Windows (which doesn't have ssize_t). df51aa4 Do not write huge files in one system call. 9c4f772 Change mmap() implementation for Windows. 5be817c Rewrote function to work on Windows without warnings. ea84f73 Fix warnings on Windows. 572d692 Fixed the static_cast_with_assert function and added tests. 89ef86b Use SparseMemArray instead of SparseMemTable in examples. git-subtree-dir: third_party/libosmium git-subtree-split: f074d949a5585a81578d682035f2163de971beb3
182 lines
5.9 KiB
C++
182 lines
5.9 KiB
C++
#ifndef OSMIUM_THREAD_QUEUE_HPP
|
|
#define OSMIUM_THREAD_QUEUE_HPP
|
|
|
|
/*
|
|
|
|
This file is part of Osmium (http://osmcode.org/libosmium).
|
|
|
|
Copyright 2013-2015 Jochen Topf <jochen@topf.org> and others (see README).
|
|
|
|
Boost Software License - Version 1.0 - August 17th, 2003
|
|
|
|
Permission is hereby granted, free of charge, to any person or organization
|
|
obtaining a copy of the software and accompanying documentation covered by
|
|
this license (the "Software") to use, reproduce, display, distribute,
|
|
execute, and transmit the Software, and to prepare derivative works of the
|
|
Software, and to permit third-parties to whom the Software is furnished to
|
|
do so, all subject to the following:
|
|
|
|
The copyright notices in the Software and this entire statement, including
|
|
the above license grant, this restriction and the following disclaimer,
|
|
must be included in all copies of the Software, in whole or in part, and
|
|
all derivative works of the Software, unless such copies or derivative
|
|
works are solely in the form of machine-executable object code generated by
|
|
a source language processor.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
|
SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
|
FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
#include <atomic>
|
|
#include <chrono>
|
|
#include <condition_variable>
|
|
#include <cstddef>
|
|
#include <mutex>
|
|
#include <queue>
|
|
#include <string>
|
|
#include <thread>
|
|
#include <utility>
|
|
|
|
#include <osmium/util/compatibility.hpp>
|
|
|
|
namespace osmium {
|
|
|
|
namespace thread {
|
|
|
|
OSMIUM_CONSTEXPR std::chrono::milliseconds full_queue_sleep_duration { 10 }; // XXX
|
|
|
|
/**
|
|
* A thread-safe queue.
|
|
*/
|
|
template <typename T>
|
|
class Queue {
|
|
|
|
/// Maximum size of this queue. If the queue is full pushing to
|
|
/// the queue will block.
|
|
const size_t m_max_size;
|
|
|
|
/// Name of this queue (for debugging only).
|
|
const std::string m_name;
|
|
|
|
mutable std::mutex m_mutex;
|
|
|
|
std::queue<T> m_queue;
|
|
|
|
/// Used to signal readers when data is available in the queue.
|
|
std::condition_variable m_data_available;
|
|
|
|
#ifdef OSMIUM_DEBUG_QUEUE_SIZE
|
|
/// The largest size the queue has been so far.
|
|
size_t m_largest_size;
|
|
|
|
/// The number of times the queue was full and a thread pushing
|
|
/// to the queue was blocked.
|
|
std::atomic<int> m_full_counter;
|
|
#endif
|
|
|
|
public:
|
|
|
|
/**
|
|
* Construct a multithreaded queue.
|
|
*
|
|
* @param max_size Maximum number of elements in the queue. Set to
|
|
* 0 for an unlimited size.
|
|
* @param name Optional name for this queue. (Used for debugging.)
|
|
*/
|
|
Queue(size_t max_size = 0, const std::string& name = "") :
|
|
m_max_size(max_size),
|
|
m_name(name),
|
|
m_mutex(),
|
|
m_queue(),
|
|
m_data_available()
|
|
#ifdef OSMIUM_DEBUG_QUEUE_SIZE
|
|
,
|
|
m_largest_size(0),
|
|
m_full_counter(0)
|
|
#endif
|
|
{
|
|
}
|
|
|
|
~Queue() {
|
|
#ifdef OSMIUM_DEBUG_QUEUE_SIZE
|
|
std::cerr << "queue '" << m_name << "' with max_size=" << m_max_size << " had largest size " << m_largest_size << " and was full " << m_full_counter << " times\n";
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Push an element onto the queue. If the queue has a max size, this
|
|
* call will block if the queue is full.
|
|
*/
|
|
void push(T value) {
|
|
if (m_max_size) {
|
|
while (size() >= m_max_size) {
|
|
std::this_thread::sleep_for(full_queue_sleep_duration);
|
|
#ifdef OSMIUM_DEBUG_QUEUE_SIZE
|
|
++m_full_counter;
|
|
#endif
|
|
}
|
|
}
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
m_queue.push(std::move(value));
|
|
#ifdef OSMIUM_DEBUG_QUEUE_SIZE
|
|
if (m_largest_size < m_queue.size()) {
|
|
m_largest_size = m_queue.size();
|
|
}
|
|
#endif
|
|
m_data_available.notify_one();
|
|
}
|
|
|
|
void wait_and_pop(T& value) {
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
m_data_available.wait(lock, [this] {
|
|
return !m_queue.empty();
|
|
});
|
|
value=std::move(m_queue.front());
|
|
m_queue.pop();
|
|
}
|
|
|
|
void wait_and_pop_with_timeout(T& value) {
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
if (!m_data_available.wait_for(lock, std::chrono::seconds(1), [this] {
|
|
return !m_queue.empty();
|
|
})) {
|
|
return;
|
|
}
|
|
value=std::move(m_queue.front());
|
|
m_queue.pop();
|
|
}
|
|
|
|
bool try_pop(T& value) {
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
if (m_queue.empty()) {
|
|
return false;
|
|
}
|
|
value=std::move(m_queue.front());
|
|
m_queue.pop();
|
|
return true;
|
|
}
|
|
|
|
bool empty() const {
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
return m_queue.empty();
|
|
}
|
|
|
|
size_t size() const {
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
return m_queue.size();
|
|
}
|
|
|
|
}; // class Queue
|
|
|
|
} // namespace thread
|
|
|
|
} // namespace osmium
|
|
|
|
#endif // OSMIUM_THREAD_QUEUE_HPP
|