2016-11-20 22:04:31 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2016-11-25 12:55:15 +00:00
|
|
|
#include "errors.hpp"
|
2016-11-02 19:22:45 +00:00
|
|
|
#include "utils/inotify.hpp"
|
|
|
|
#include "utils/memory.hpp"
|
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
namespace inotify_util {
|
|
|
|
/**
|
|
|
|
* Destructor
|
|
|
|
*/
|
|
|
|
inotify_watch::~inotify_watch() noexcept {
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_wd != -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
inotify_rm_watch(m_fd, m_wd);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
|
|
|
if (m_fd != -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
close(m_fd);
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Attach inotify watch
|
|
|
|
*/
|
|
|
|
void inotify_watch::attach(int mask) {
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_fd == -1 && (m_fd = inotify_init()) == -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
throw system_error("Failed to allocate inotify fd");
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
|
|
|
if ((m_wd = inotify_add_watch(m_fd, m_path.c_str(), mask)) == -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
throw system_error("Failed to attach inotify watch");
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-25 10:40:32 +00:00
|
|
|
m_mask |= mask;
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove inotify watch
|
|
|
|
*/
|
|
|
|
void inotify_watch::remove() {
|
2016-11-25 12:55:15 +00:00
|
|
|
if (inotify_rm_watch(m_fd, m_wd) == -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
throw system_error("Failed to remove inotify watch");
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
m_wd = -1;
|
2016-11-25 10:40:32 +00:00
|
|
|
m_mask = 0;
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Poll the inotify fd for events
|
|
|
|
*
|
|
|
|
* @brief A wait_ms of -1 blocks until an event is fired
|
|
|
|
*/
|
|
|
|
bool inotify_watch::poll(int wait_ms) {
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_fd == -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
return false;
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
struct pollfd fds[1];
|
|
|
|
fds[0].fd = m_fd;
|
|
|
|
fds[0].events = POLLIN;
|
|
|
|
|
|
|
|
::poll(fds, 1, wait_ms);
|
|
|
|
|
|
|
|
return fds[0].revents & POLLIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the latest inotify event
|
|
|
|
*/
|
|
|
|
unique_ptr<event_t> inotify_watch::get_event() {
|
|
|
|
auto event = make_unique<event_t>();
|
|
|
|
|
2016-11-25 12:55:15 +00:00
|
|
|
if (m_fd == -1 || m_wd == -1) {
|
2016-11-02 19:22:45 +00:00
|
|
|
return event;
|
2016-11-25 12:55:15 +00:00
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
char buffer[1024];
|
|
|
|
size_t bytes = read(m_fd, buffer, 1024);
|
|
|
|
size_t len = 0;
|
|
|
|
|
|
|
|
while (len < bytes) {
|
|
|
|
auto* e = reinterpret_cast<::inotify_event*>(&buffer[len]);
|
|
|
|
|
|
|
|
event->filename = e->len ? e->name : m_path;
|
|
|
|
event->wd = e->wd;
|
|
|
|
event->cookie = e->cookie;
|
|
|
|
event->is_dir = e->mask & IN_ISDIR;
|
|
|
|
event->mask |= e->mask;
|
|
|
|
|
|
|
|
len += sizeof(event_t) + e->len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return event;
|
|
|
|
}
|
|
|
|
|
2016-11-25 10:40:32 +00:00
|
|
|
/**
|
|
|
|
* Wait for matching event
|
|
|
|
*/
|
|
|
|
bool inotify_watch::await_match() {
|
|
|
|
return (get_event()->mask & m_mask) == m_mask;
|
|
|
|
}
|
|
|
|
|
2016-11-02 19:22:45 +00:00
|
|
|
/**
|
|
|
|
* Get watch file path
|
|
|
|
*/
|
|
|
|
const string inotify_watch::path() const {
|
|
|
|
return m_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
watch_t make_watch(string path) {
|
|
|
|
di::injector<watch_t> injector = di::make_injector(di::bind<>().to(path));
|
|
|
|
return injector.create<watch_t>();
|
|
|
|
}
|
2016-11-25 10:40:32 +00:00
|
|
|
|
|
|
|
bool match(const event_t* evt, int mask) {
|
|
|
|
return (evt->mask & mask) == mask;
|
|
|
|
}
|
2016-11-02 19:22:45 +00:00
|
|
|
}
|
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS_END
|