add example using threads

This commit is contained in:
Tom 2019-10-13 15:06:50 +02:00
parent 9af897fa3b
commit f33e820a16
3 changed files with 84 additions and 0 deletions

View File

@ -42,3 +42,6 @@ install(EXPORT SgnlTargets
add_subdirectory("test")
add_executable(example EXCLUDE_FROM_ALL "example/example.cpp")
target_link_libraries(example sgnl::sgnl)

View File

@ -36,6 +36,7 @@ Signal handler that uses [pthread_sigmask](http://man7.org/linux/man-pages/man3/
int last_signal = signal_handler.sigwait_handler(handler);
} // signals are unblocked again
```
See [example.cpp](example/example.cpp) for an example using threads.
## Build & Install
@ -45,6 +46,8 @@ mkdir -p build/ && cd build/
cmake ..
# build and run tests
make sgnl-test && ./test/sgnl-test
# build and run example
make example && ./example
# install headers and CMake config
make install
```

78
example/example.cpp Normal file
View File

@ -0,0 +1,78 @@
#include <sgnl/AtomicCondition.h>
#include <sgnl/SignalHandler.h>
#include <cstdlib>
#include <future>
#include <iostream>
#include <thread>
void Worker(const sgnl::AtomicCondition<bool>& exit_condition)
{
auto predicate = [&exit_condition]() {
return exit_condition.get();
};
while( true )
{
exit_condition.wait_for(std::chrono::minutes(1), predicate);
if( exit_condition.get() )
return;
/* ... do work ... */
}
}
int main()
{
sgnl::AtomicCondition<bool> exit_condition(false);
auto handler = [&exit_condition](int signum) {
std::cout << "received signal " << signum << "\n";
if( signum == SIGTERM || signum == SIGINT )
{
exit_condition.set(true);
// wakeup all waiting threads
exit_condition.notify_all();
// stop polling for signals
return true;
}
// continue waiting for signals
return false;
};
// Block signals in this thread.
// Threads spawned later will inherit the signal mask.
sgnl::SignalHandler signal_handler({SIGINT, SIGTERM, SIGUSR1});
std::future<int> ft_sig_handler =
std::async(
std::launch::async,
&sgnl::SignalHandler::sigwait_handler,
&signal_handler,
std::ref(handler));
std::vector<std::future<void>> futures;
for(int i = 0; i < 10; ++i)
futures.push_back(
std::async(
std::launch::async,
Worker,
std::ref(exit_condition)));
// SIGUSR1
std::this_thread::sleep_for(std::chrono::milliseconds(100));
kill(0, SIGUSR1);
// SIGTERM
kill(0, SIGTERM);
std::this_thread::sleep_for(std::chrono::milliseconds(100));
for(auto& future : futures)
future.wait();
int last_signal = ft_sig_handler.get();
std::cout << "exiting (received signal " << last_signal << ")\n";
return EXIT_SUCCESS;
}