Use SIGALRM for timer and get rid of the child process

This commit is contained in:
Utkarsh Verma
2022-04-21 18:37:00 +05:30
parent f04d21fa22
commit 23e1d5732a
2 changed files with 50 additions and 89 deletions

View File

@@ -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/). 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 ## 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/). 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/).

119
main.c
View File

@@ -39,9 +39,9 @@ static Display* dpy;
static int screen; static int screen;
static Window root; static Window root;
static unsigned short statusContinue = 1; 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 pipes[LEN(blocks)][2];
static int timerPipe[2]; static int timer = 0, timerTick = 0, maxInterval = 0;
static int signalFD; static int signalFD;
static int epollFD; static int epollFD;
static int execLock = 0; static int execLock = 0;
@@ -176,8 +176,17 @@ void signalHandler() {
read(signalFD, &info, sizeof(info)); read(signalFD, &info, sizeof(info));
unsigned int signal = info.ssi_signo; unsigned int signal = info.ssi_signo;
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 // Update all blocks on receiving SIGUSR1
if (signal == SIGUSR1) {
execBlocks(0); execBlocks(0);
return; return;
} }
@@ -196,13 +205,25 @@ void termHandler() {
} }
void setupSignals() { void setupSignals() {
// Ignore SIGUSR1 and all realtime signals sigset_t handledSignals;
sigset_t ignoredSignals; sigemptyset(&handledSignals);
sigemptyset(&ignoredSignals); sigaddset(&handledSignals, SIGUSR1);
sigaddset(&ignoredSignals, 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++) for (int i = SIGRTMIN; i <= SIGRTMAX; i++)
sigaddset(&ignoredSignals, i); sigaddset(&handledSignals, i);
sigprocmask(SIG_BLOCK, &ignoredSignals, NULL); sigprocmask(SIG_BLOCK, &handledSignals, NULL);
// Handle termination signals // Handle termination signals
signal(SIGINT, termHandler); signal(SIGINT, termHandler);
@@ -214,85 +235,44 @@ void setupSignals() {
sigemptyset(&sa.sa_mask); sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_NOCLDWAIT; sa.sa_flags = SA_NOCLDWAIT;
sigaction(SIGCHLD, &sa, 0); 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() { void statusLoop() {
// Update all blocks initially
kill(0, SIGALRM);
struct epoll_event events[LEN(blocks) + 1];
while (statusContinue) { while (statusContinue) {
int eventCount = epoll_wait(epollFD, events, LEN(events), -1); int eventCount = epoll_wait(epollFD, events, LEN(events), -1);
for (int i = 0; i < eventCount; i++) { for (int i = 0; i < eventCount; i++) {
unsigned short id = events[i].data.u32; unsigned short id = events[i].data.u32;
if (id < LEN(blocks))
if (id == LEN(blocks)) { updateBlock(id);
unsigned int j = 0; else
read(timerPipe[0], &j, sizeof(j));
execBlocks(j);
} else if (id < LEN(blocks)) {
updateBlock(events[i].data.u32);
} else {
signalHandler(); signalHandler();
} }
}
if (eventCount != -1) if (eventCount != -1)
writeStatus(); 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() { void init() {
epollFD = epoll_create(LEN(blocks) + 1); epollFD = epoll_create(LEN(blocks));
event.events = EPOLLIN; event.events = EPOLLIN;
for (int i = 0; i < LEN(blocks); i++) { for (int i = 0; i < LEN(blocks); i++) {
// Append each block's pipe to `epollFD`
pipe(pipes[i]); pipe(pipes[i]);
event.data.u32 = i; event.data.u32 = i;
epoll_ctl(epollFD, EPOLL_CTL_ADD, pipes[i][0], &event); epoll_ctl(epollFD, EPOLL_CTL_ADD, pipes[i][0], &event);
}
pipe(timerPipe); // Calculate the max interval and tick size for the timer
event.data.u32 = LEN(blocks); if (blocks[i].interval) {
epoll_ctl(epollFD, EPOLL_CTL_ADD, timerPipe[0], &event); maxInterval = MAX(blocks[i].interval, maxInterval);
timerTick = gcd(blocks[i].interval, timerTick);
}
}
setupSignals(); setupSignals();
} }
@@ -304,17 +284,12 @@ int main(const int argc, const char* argv[]) {
writeStatus = debug; writeStatus = debug;
init(); init();
// Ensure that `timerLoop()` only runs in the fork
if (fork() == 0)
timerLoop();
else
statusLoop(); statusLoop();
close(epollFD); close(epollFD);
close(signalFD); close(signalFD);
closePipe(timerPipe);
for (int i = 0; i < LEN(pipes); i++) for (int i = 0; i < LEN(pipes); i++)
closePipe(pipes[i]); closePipe(pipes[i]);
return 0; return 0;
} }