2014-07-23 20:21:00 +03:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <thread>
|
2014-07-24 01:16:06 +03:00
|
|
|
#include <atomic>
|
2014-07-23 20:21:00 +03:00
|
|
|
|
|
|
|
#include <poll.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
|
|
|
class MonitorFdHup
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
std::thread thread;
|
2014-07-24 01:16:06 +03:00
|
|
|
std::atomic_bool quit;
|
2014-07-23 20:21:00 +03:00
|
|
|
|
|
|
|
public:
|
|
|
|
MonitorFdHup(int fd)
|
|
|
|
{
|
2014-07-24 01:16:06 +03:00
|
|
|
quit = false;
|
2014-07-23 20:21:00 +03:00
|
|
|
thread = std::thread([&]() {
|
|
|
|
/* Wait indefinitely until a POLLHUP occurs. */
|
|
|
|
struct pollfd fds[1];
|
|
|
|
fds[0].fd = fd;
|
|
|
|
fds[0].events = 0;
|
|
|
|
if (poll(fds, 1, -1) == -1) {
|
|
|
|
if (errno != EINTR) abort(); // can't happen
|
2014-07-24 01:16:06 +03:00
|
|
|
assert(quit);
|
2014-07-23 20:21:00 +03:00
|
|
|
return; // destructor is asking us to exit
|
|
|
|
}
|
2014-07-24 01:16:06 +03:00
|
|
|
assert(fds[0].revents & POLLHUP);
|
2014-07-23 20:21:00 +03:00
|
|
|
/* We got POLLHUP, so send an INT signal to the main thread. */
|
|
|
|
kill(getpid(), SIGINT);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
~MonitorFdHup()
|
|
|
|
{
|
2014-07-24 01:16:06 +03:00
|
|
|
quit = true;
|
2014-07-23 20:21:00 +03:00
|
|
|
pthread_kill(thread.native_handle(), SIGINT);
|
|
|
|
thread.join();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
}
|