Use SIGALRM for timer and get rid of the child process
This commit is contained in:
16
README.md
16
README.md
@@ -125,22 +125,8 @@ To use this feature, define the `CLICKABLE_BLOCKS` feature macro in your `config
|
||||
|
||||
Apart from that, you need `dwm` to be patched with [statuscmd](https://dwm.suckless.org/patches/statuscmd/).
|
||||
|
||||
Because `dwmblocks-async` creates a child process, it messes up the way the original `statuscmd` patch gets the PID of statusbar. It is necessary to modify the following lines in the definition of `getstatusbarpid()`.
|
||||
> Earlier, `dwmblocks-async` used to require a patch to be applied to `dwm`. However, the code has been redone so there's no need to apply that patch anymore.
|
||||
|
||||
```diff
|
||||
return statuspid;
|
||||
}
|
||||
}
|
||||
- if (!(fp = popen("pidof -s "STATUSBAR, "r")))
|
||||
+ if (!(fp = popen("pgrep -o "STATUSBAR, "r")))
|
||||
return -1;
|
||||
fgets(buf, sizeof(buf), fp);
|
||||
pclose(fp);
|
||||
```
|
||||
|
||||
This modification is backwards-compatible with other versions of `dwmblocks` as well.
|
||||
|
||||
Please note that if you are using [`dwm-flexipatch`](https://github.com/bakkeby/dwm-flexipatch), then you only need to enable the `statuscmd` patch through `patches.h`. There is no need to edit change the definition of `getstatusbarpid()` as the changes have already been applied there through [dwm-flexipatch#190](https://github.com/bakkeby/dwm-flexipatch/pull/190).
|
||||
|
||||
## Credits
|
||||
This work would not have been possible without [Luke's build of dwmblocks](https://github.com/LukeSmithxyz/dwmblocks) and [Daniel Bylinka's statuscmd patch](https://dwm.suckless.org/patches/statuscmd/).
|
||||
|
123
main.c
123
main.c
@@ -39,9 +39,9 @@ static Display* dpy;
|
||||
static int screen;
|
||||
static Window root;
|
||||
static unsigned short statusContinue = 1;
|
||||
static struct epoll_event event, events[LEN(blocks) + 2];
|
||||
static struct epoll_event event;
|
||||
static int pipes[LEN(blocks)][2];
|
||||
static int timerPipe[2];
|
||||
static int timer = 0, timerTick = 0, maxInterval = 0;
|
||||
static int signalFD;
|
||||
static int epollFD;
|
||||
static int execLock = 0;
|
||||
@@ -176,8 +176,17 @@ void signalHandler() {
|
||||
read(signalFD, &info, sizeof(info));
|
||||
unsigned int signal = info.ssi_signo;
|
||||
|
||||
// Update all blocks on receiving SIGUSR1
|
||||
if (signal == SIGUSR1) {
|
||||
switch (signal) {
|
||||
case SIGALRM:
|
||||
// Schedule the next timer event and execute blocks
|
||||
alarm(timerTick);
|
||||
execBlocks(timer);
|
||||
|
||||
// Wrap `timer` to the interval [1, `maxInterval`]
|
||||
timer = (timer + timerTick - 1) % maxInterval + 1;
|
||||
return;
|
||||
case SIGUSR1:
|
||||
// Update all blocks on receiving SIGUSR1
|
||||
execBlocks(0);
|
||||
return;
|
||||
}
|
||||
@@ -196,13 +205,25 @@ void termHandler() {
|
||||
}
|
||||
|
||||
void setupSignals() {
|
||||
// Ignore SIGUSR1 and all realtime signals
|
||||
sigset_t ignoredSignals;
|
||||
sigemptyset(&ignoredSignals);
|
||||
sigaddset(&ignoredSignals, SIGUSR1);
|
||||
sigset_t handledSignals;
|
||||
sigemptyset(&handledSignals);
|
||||
sigaddset(&handledSignals, SIGUSR1);
|
||||
sigaddset(&handledSignals, SIGALRM);
|
||||
|
||||
// Append all block signals to `handledSignals`
|
||||
for (int i = 0; i < LEN(blocks); i++)
|
||||
if (blocks[i].signal > 0)
|
||||
sigaddset(&handledSignals, SIGRTMIN + blocks[i].signal);
|
||||
|
||||
// Create a signal file descriptor for epoll to watch
|
||||
signalFD = signalfd(-1, &handledSignals, 0);
|
||||
event.data.u32 = LEN(blocks);
|
||||
epoll_ctl(epollFD, EPOLL_CTL_ADD, signalFD, &event);
|
||||
|
||||
// Block all realtime and handled signals
|
||||
for (int i = SIGRTMIN; i <= SIGRTMAX; i++)
|
||||
sigaddset(&ignoredSignals, i);
|
||||
sigprocmask(SIG_BLOCK, &ignoredSignals, NULL);
|
||||
sigaddset(&handledSignals, i);
|
||||
sigprocmask(SIG_BLOCK, &handledSignals, NULL);
|
||||
|
||||
// Handle termination signals
|
||||
signal(SIGINT, termHandler);
|
||||
@@ -214,85 +235,44 @@ void setupSignals() {
|
||||
sigemptyset(&sa.sa_mask);
|
||||
sa.sa_flags = SA_NOCLDWAIT;
|
||||
sigaction(SIGCHLD, &sa, 0);
|
||||
|
||||
// Handle block update signals
|
||||
sigset_t handledSignals;
|
||||
sigemptyset(&handledSignals);
|
||||
sigaddset(&handledSignals, SIGUSR1);
|
||||
for (int i = 0; i < LEN(blocks); i++)
|
||||
if (blocks[i].signal > 0)
|
||||
sigaddset(&handledSignals, SIGRTMIN + blocks[i].signal);
|
||||
signalFD = signalfd(-1, &handledSignals, 0);
|
||||
event.data.u32 = LEN(blocks) + 1;
|
||||
epoll_ctl(epollFD, EPOLL_CTL_ADD, signalFD, &event);
|
||||
}
|
||||
|
||||
void statusLoop() {
|
||||
// Update all blocks initially
|
||||
kill(0, SIGALRM);
|
||||
|
||||
struct epoll_event events[LEN(blocks) + 1];
|
||||
while (statusContinue) {
|
||||
int eventCount = epoll_wait(epollFD, events, LEN(events), -1);
|
||||
for (int i = 0; i < eventCount; i++) {
|
||||
unsigned short id = events[i].data.u32;
|
||||
|
||||
if (id == LEN(blocks)) {
|
||||
unsigned int j = 0;
|
||||
read(timerPipe[0], &j, sizeof(j));
|
||||
execBlocks(j);
|
||||
} else if (id < LEN(blocks)) {
|
||||
updateBlock(events[i].data.u32);
|
||||
} else {
|
||||
if (id < LEN(blocks))
|
||||
updateBlock(id);
|
||||
else
|
||||
signalHandler();
|
||||
}
|
||||
}
|
||||
|
||||
if (eventCount != -1)
|
||||
writeStatus();
|
||||
}
|
||||
}
|
||||
|
||||
void timerLoop() {
|
||||
close(timerPipe[0]);
|
||||
|
||||
unsigned int sleepInterval = 0;
|
||||
unsigned int maxInterval = 0;
|
||||
for (int i = 0; i < LEN(blocks); i++)
|
||||
if (blocks[i].interval) {
|
||||
maxInterval = MAX(blocks[i].interval, maxInterval);
|
||||
sleepInterval = gcd(blocks[i].interval, sleepInterval);
|
||||
}
|
||||
|
||||
unsigned int i = 0;
|
||||
struct timespec sleepTime = {sleepInterval, 0};
|
||||
struct timespec toSleep = sleepTime;
|
||||
|
||||
while (statusContinue) {
|
||||
// Notify parent to update blocks
|
||||
write(timerPipe[1], &i, sizeof(i));
|
||||
|
||||
// Wrap `i` to the interval [1, `maxInterval`]
|
||||
i = (i + sleepInterval - 1) % maxInterval + 1;
|
||||
|
||||
// Sleep for `sleepTime` even on being interrupted
|
||||
while (nanosleep(&toSleep, &toSleep) == -1)
|
||||
;
|
||||
toSleep = sleepTime;
|
||||
}
|
||||
|
||||
close(timerPipe[1]);
|
||||
_exit(0);
|
||||
}
|
||||
|
||||
void init() {
|
||||
epollFD = epoll_create(LEN(blocks) + 1);
|
||||
epollFD = epoll_create(LEN(blocks));
|
||||
event.events = EPOLLIN;
|
||||
|
||||
for (int i = 0; i < LEN(blocks); i++) {
|
||||
// Append each block's pipe to `epollFD`
|
||||
pipe(pipes[i]);
|
||||
event.data.u32 = i;
|
||||
epoll_ctl(epollFD, EPOLL_CTL_ADD, pipes[i][0], &event);
|
||||
}
|
||||
|
||||
pipe(timerPipe);
|
||||
event.data.u32 = LEN(blocks);
|
||||
epoll_ctl(epollFD, EPOLL_CTL_ADD, timerPipe[0], &event);
|
||||
// Calculate the max interval and tick size for the timer
|
||||
if (blocks[i].interval) {
|
||||
maxInterval = MAX(blocks[i].interval, maxInterval);
|
||||
timerTick = gcd(blocks[i].interval, timerTick);
|
||||
}
|
||||
}
|
||||
|
||||
setupSignals();
|
||||
}
|
||||
@@ -304,17 +284,12 @@ int main(const int argc, const char* argv[]) {
|
||||
writeStatus = debug;
|
||||
|
||||
init();
|
||||
|
||||
// Ensure that `timerLoop()` only runs in the fork
|
||||
if (fork() == 0)
|
||||
timerLoop();
|
||||
else
|
||||
statusLoop();
|
||||
statusLoop();
|
||||
|
||||
close(epollFD);
|
||||
close(signalFD);
|
||||
closePipe(timerPipe);
|
||||
for (int i = 0; i < LEN(pipes); i++)
|
||||
closePipe(pipes[i]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
Reference in New Issue
Block a user