306 lines
10 KiB
C
306 lines
10 KiB
C
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
#include <X11/Xlib.h>
|
|
|
|
#define NILL INT_MIN
|
|
#define LOCKFILE "/tmp/dwmblocks.pid"
|
|
|
|
#define DELIMITERLENGTH sizeof delimiter
|
|
|
|
#include "../config.h"
|
|
|
|
static void buttonhandler(int sig, siginfo_t *info, void *ucontext);
|
|
static void cleanup();
|
|
static void setupsignals();
|
|
static void sighandler(int sig, siginfo_t *si, void *ucontext);
|
|
static void statusloop();
|
|
static void termhandler(int sig);
|
|
static void updateblock(Block *block, int sigval);
|
|
static void updatestatus();
|
|
static void writepid();
|
|
|
|
static Block *dirtyblock;
|
|
static Display *dpy;
|
|
static sigset_t blocksigmask;
|
|
|
|
void
|
|
buttonhandler(int sig, siginfo_t *info, void *ucontext)
|
|
{
|
|
sig = info->si_value.sival_int >> 8;
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->signal == sig)
|
|
switch (fork()) {
|
|
case -1:
|
|
perror("buttonhandler - fork");
|
|
break;
|
|
case 0:
|
|
{
|
|
char button[] = { '0' + (info->si_value.sival_int & 0xff), '\0' };
|
|
char *arg[] = { block->pathc, button, NULL };
|
|
|
|
close(ConnectionNumber(dpy));
|
|
setsid();
|
|
execv(arg[0], arg);
|
|
perror("buttonhandler - child - execv");
|
|
_exit(127);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
cleanup()
|
|
{
|
|
unlink(LOCKFILE);
|
|
XStoreName(dpy, DefaultRootWindow(dpy), "");
|
|
XCloseDisplay(dpy);
|
|
}
|
|
|
|
void
|
|
setupsignals()
|
|
{
|
|
struct sigaction sa;
|
|
|
|
/* populate blocksigmask */
|
|
sigemptyset(&blocksigmask);
|
|
sigaddset(&blocksigmask, SIGHUP);
|
|
sigaddset(&blocksigmask, SIGINT);
|
|
sigaddset(&blocksigmask, SIGTERM);
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->signal > 0)
|
|
sigaddset(&blocksigmask, SIGRTMIN + block->signal);
|
|
|
|
/* setup signal handlers */
|
|
/* to handle HUP, INT and TERM */
|
|
sa.sa_flags = SA_RESTART;
|
|
sigemptyset(&sa.sa_mask);
|
|
sa.sa_handler = termhandler;
|
|
sigaction(SIGHUP, &sa, NULL);
|
|
sigaction(SIGINT, &sa, NULL);
|
|
sigaction(SIGTERM, &sa, NULL);
|
|
|
|
/* to ignore unused realtime signals */
|
|
// sa.sa_flags = SA_RESTART;
|
|
// sigemptyset(&sa.sa_mask);
|
|
sa.sa_handler = SIG_IGN;
|
|
for (int i = SIGRTMIN + 1; i <= SIGRTMAX; i++)
|
|
sigaction(i, &sa, NULL);
|
|
|
|
/* to prevent forked children from becoming zombies */
|
|
sa.sa_flags = SA_NOCLDSTOP | SA_NOCLDWAIT | SA_RESTART;
|
|
// sigemptyset(&sa.sa_mask);
|
|
sa.sa_handler = SIG_DFL;
|
|
sigaction(SIGCHLD, &sa, NULL);
|
|
|
|
/* to handle signals generated by dwm on click events */
|
|
sa.sa_flags = SA_RESTART | SA_SIGINFO;
|
|
// sigemptyset(&sa.sa_mask);
|
|
sa.sa_sigaction = buttonhandler;
|
|
sigaction(SIGRTMIN, &sa, NULL);
|
|
|
|
/* to handle update signals for individual blocks */
|
|
sa.sa_flags |= SA_NODEFER;
|
|
sa.sa_mask = blocksigmask;
|
|
sa.sa_sigaction = sighandler;
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->signal > 0)
|
|
sigaction(SIGRTMIN + block->signal, &sa, NULL);
|
|
}
|
|
|
|
void
|
|
sighandler(int sig, siginfo_t *info, void *ucontext)
|
|
{
|
|
sig -= SIGRTMIN;
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->signal == sig)
|
|
updateblock(block, info->si_value.sival_int);
|
|
updatestatus();
|
|
}
|
|
|
|
void
|
|
statusloop()
|
|
{
|
|
int i;
|
|
struct timespec t;
|
|
|
|
/* first run */
|
|
sigprocmask(SIG_BLOCK, &blocksigmask, NULL);
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->interval >= 0)
|
|
updateblock(block, NILL);
|
|
updatestatus();
|
|
sigprocmask(SIG_UNBLOCK, &blocksigmask, NULL);
|
|
t.tv_sec = INTERVALs, t.tv_nsec = INTERVALn;
|
|
while (nanosleep(&t, &t) == -1)
|
|
if (errno != EINTR) {
|
|
perror("statusloop - nanosleep");
|
|
exit(1);
|
|
}
|
|
/* main loop */
|
|
for (i = 1; ; i++) {
|
|
sigprocmask(SIG_BLOCK, &blocksigmask, NULL);
|
|
for (Block *block = blocks; block->pathu; block++)
|
|
if (block->interval > 0 && i % block->interval == 0)
|
|
updateblock(block, NILL);
|
|
updatestatus();
|
|
sigprocmask(SIG_UNBLOCK, &blocksigmask, NULL);
|
|
t.tv_sec = INTERVALs, t.tv_nsec = INTERVALn;
|
|
while (nanosleep(&t, &t) == -1);
|
|
}
|
|
}
|
|
|
|
void
|
|
termhandler(int sig)
|
|
{
|
|
cleanup();
|
|
exit(0);
|
|
}
|
|
|
|
void
|
|
updateblock(Block *block, int sigval)
|
|
{
|
|
int fd[2];
|
|
|
|
if (pipe(fd) == -1) {
|
|
perror("updateblock - pipe");
|
|
exit(1);
|
|
}
|
|
switch (fork()) {
|
|
case -1:
|
|
perror("updateblock - fork");
|
|
exit(1);
|
|
case 0:
|
|
close(ConnectionNumber(dpy));
|
|
close(fd[0]);
|
|
if (fd[1] != STDOUT_FILENO) {
|
|
if (dup2(fd[1], STDOUT_FILENO) != STDOUT_FILENO) {
|
|
perror("updateblock - child - dup2");
|
|
exit(1);
|
|
}
|
|
close(fd[1]);
|
|
}
|
|
if (sigval == NILL) {
|
|
char *arg[] = { block->pathu, NULL };
|
|
|
|
execv(arg[0], arg);
|
|
} else {
|
|
char buf[12];
|
|
char *arg[] = { block->pathu, buf, NULL };
|
|
|
|
snprintf(buf, sizeof buf, "%d", sigval);
|
|
execv(arg[0], arg);
|
|
}
|
|
perror("updateblock - child - execv");
|
|
_exit(127);
|
|
default:
|
|
{
|
|
size_t trd = 0;
|
|
ssize_t rd;
|
|
|
|
close(fd[1]);
|
|
do
|
|
rd = read(fd[0], block->curtext + trd, CMDOUTLENGTH - trd);
|
|
while (rd > 0 && (trd += rd) < CMDOUTLENGTH);
|
|
if (rd == -1) {
|
|
perror("updateblock - read");
|
|
exit(1);
|
|
}
|
|
close(fd[0]);
|
|
if (trd == 0) {
|
|
if (block->prvtext[0] != '\0') {
|
|
block->prvtext[0] = '\0';
|
|
if (!dirtyblock || block < dirtyblock)
|
|
dirtyblock = block;
|
|
}
|
|
block->length = 0;
|
|
} else {
|
|
if (block->curtext[trd - 1] == '\n')
|
|
trd--;
|
|
block->curtext[trd++] = block->signal;
|
|
if (memcmp(block->curtext, block->prvtext, trd) != 0) {
|
|
memcpy(block->prvtext, block->curtext, trd);
|
|
if (!dirtyblock || block < dirtyblock)
|
|
dirtyblock = block;
|
|
}
|
|
if (!block->pathc)
|
|
trd--;
|
|
memcpy(block->curtext + trd, delimiter, DELIMITERLENGTH);
|
|
block->length = trd + DELIMITERLENGTH;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
updatestatus()
|
|
{
|
|
static char statustext[STATUSLENGTH + DELIMITERLENGTH];
|
|
char *s = statustext;
|
|
|
|
if (!dirtyblock)
|
|
return;
|
|
for (Block *block = blocks; block < dirtyblock; block++)
|
|
s += block->length;
|
|
for (Block *block = dirtyblock; block->pathu; block++) {
|
|
memcpy(s, block->curtext, block->length);
|
|
s += block->length;
|
|
}
|
|
s[s == statustext ? 0 : -DELIMITERLENGTH] = '\0';
|
|
dirtyblock = NULL;
|
|
|
|
XStoreName(dpy, DefaultRootWindow(dpy), statustext);
|
|
XSync(dpy, False);
|
|
}
|
|
|
|
void
|
|
writepid()
|
|
{
|
|
int fd;
|
|
struct flock fl;
|
|
|
|
if ((fd = open(LOCKFILE, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)) == -1) {
|
|
perror("writepid - open");
|
|
exit(1);
|
|
}
|
|
fl.l_type = F_WRLCK;
|
|
fl.l_whence = SEEK_SET;
|
|
fl.l_start = 0;
|
|
fl.l_len = 0;
|
|
if (fcntl(fd, F_SETLK, &fl) == -1) {
|
|
if (errno == EACCES || errno == EAGAIN) {
|
|
fputs("Error: another instance of dwmblocks is already running.\n", stderr);
|
|
exit(2);
|
|
}
|
|
perror("writepid - fcntl");
|
|
exit(1);
|
|
}
|
|
if (ftruncate(fd, 0) == -1) {
|
|
perror("writepid - ftruncate");
|
|
exit(1);
|
|
}
|
|
if (dprintf(fd, "%ld", (long)getpid()) < 0) {
|
|
perror("writepid - dprintf");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
if (!(dpy = XOpenDisplay(NULL))) {
|
|
fputs("Error: could not open display.\n", stderr);
|
|
return 1;
|
|
}
|
|
writepid();
|
|
setupsignals();
|
|
statusloop();
|
|
cleanup();
|
|
return 0;
|
|
}
|