mirror of
https://github.com/mintycube/dwmblocks.git
synced 2024-10-22 14:05:47 +02:00
Use SIGALRM for timer and get rid of the child process
This commit is contained in:
parent
f04d21fa22
commit
23e1d5732a
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/).
|
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/).
|
||||||
|
123
main.c
123
main.c
@ -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;
|
||||||
|
|
||||||
// Update all blocks on receiving SIGUSR1
|
switch (signal) {
|
||||||
if (signal == SIGUSR1) {
|
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);
|
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();
|
||||||
|
statusLoop();
|
||||||
// Ensure that `timerLoop()` only runs in the fork
|
|
||||||
if (fork() == 0)
|
|
||||||
timerLoop();
|
|
||||||
else
|
|
||||||
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;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user