You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

204 lines
4.1 KiB

  1. #include<stdlib.h>
  2. #include<stdio.h>
  3. #include<string.h>
  4. #include<unistd.h>
  5. #include<signal.h>
  6. #ifndef NO_X
  7. #include<X11/Xlib.h>
  8. #endif
  9. #ifdef __OpenBSD__
  10. #define SIGPLUS SIGUSR1+1
  11. #define SIGMINUS SIGUSR1-1
  12. #else
  13. #define SIGPLUS SIGRTMIN
  14. #define SIGMINUS SIGRTMIN
  15. #endif
  16. #define LENGTH(X) (sizeof(X) / sizeof (X[0]))
  17. #define CMDLENGTH 50
  18. #define MIN( a, b ) ( ( a < b) ? a : b )
  19. #define STATUSLENGTH (LENGTH(blocks) * CMDLENGTH + 1)
  20. typedef struct {
  21. char* icon;
  22. char* command;
  23. unsigned int interval;
  24. unsigned int signal;
  25. } Block;
  26. #ifndef __OpenBSD__
  27. void dummysighandler(int num);
  28. #endif
  29. void sighandler(int num);
  30. void getcmds(int time);
  31. void getsigcmds(unsigned int signal);
  32. void setupsignals();
  33. void sighandler(int signum);
  34. int getstatus(char *str, char *last);
  35. void statusloop();
  36. void termhandler();
  37. void pstdout();
  38. #ifndef NO_X
  39. void setroot();
  40. static void (*writestatus) () = setroot;
  41. #else
  42. static void (*writestatus) () = pstdout;
  43. #endif
  44. #include "blocks.h"
  45. static char statusbar[LENGTH(blocks)][CMDLENGTH] = {0};
  46. static char statusstr[2][STATUSLENGTH];
  47. static int statusContinue = 1;
  48. static int returnStatus = 0;
  49. //opens process *cmd and stores output in *output
  50. void getcmd(const Block *block, char *output)
  51. {
  52. strcpy(output, block->icon);
  53. FILE *cmdf = popen(block->command, "r");
  54. if (!cmdf)
  55. return;
  56. int i = strlen(block->icon);
  57. fgets(output+i, CMDLENGTH-i-delimLen, cmdf);
  58. i = strlen(output);
  59. if (i == 0)//return if block and command output are both empty
  60. return;
  61. if (delim[0] != '\0') {
  62. //only chop off newline if one is present at the end
  63. i = output[i-1] == '\n' ? i-1 : i;
  64. strncpy(output+i, delim, delimLen);
  65. }
  66. else
  67. output[i++] = '\0';
  68. pclose(cmdf);
  69. }
  70. void getcmds(int time)
  71. {
  72. const Block* current;
  73. for (unsigned int i = 0; i < LENGTH(blocks); i++)
  74. {
  75. current = blocks + i;
  76. if ((current->interval != 0 && time % current->interval == 0) || time == -1)
  77. getcmd(current,statusbar[i]);
  78. }
  79. }
  80. void getsigcmds(unsigned int signal)
  81. {
  82. const Block *current;
  83. for (unsigned int i = 0; i < LENGTH(blocks); i++)
  84. {
  85. current = blocks + i;
  86. if (current->signal == signal)
  87. getcmd(current,statusbar[i]);
  88. }
  89. }
  90. void setupsignals()
  91. {
  92. #ifndef __OpenBSD__
  93. /* initialize all real time signals with dummy handler */
  94. for (int i = SIGRTMIN; i <= SIGRTMAX; i++)
  95. signal(i, dummysighandler);
  96. #endif
  97. for (unsigned int i = 0; i < LENGTH(blocks); i++)
  98. {
  99. if (blocks[i].signal > 0)
  100. signal(SIGMINUS+blocks[i].signal, sighandler);
  101. }
  102. }
  103. int getstatus(char *str, char *last)
  104. {
  105. strcpy(last, str);
  106. str[0] = '\0';
  107. for (unsigned int i = 0; i < LENGTH(blocks); i++)
  108. strcat(str, statusbar[i]);
  109. str[strlen(str)-strlen(delim)] = '\0';
  110. return strcmp(str, last);//0 if they are the same
  111. }
  112. #ifndef NO_X
  113. void setroot()
  114. {
  115. static Display *dpy;
  116. static int screen;
  117. static Window root;
  118. if (!getstatus(statusstr[0], statusstr[1]))//Only set root if text has changed.
  119. return;
  120. dpy = XOpenDisplay(NULL);
  121. if (!dpy) {
  122. fprintf(stderr, "Failed to open display\n");
  123. statusContinue = 0;
  124. returnStatus = 1;
  125. return;
  126. }
  127. screen = DefaultScreen(dpy);
  128. root = RootWindow(dpy, screen);
  129. XStoreName(dpy, root, statusstr[0]);
  130. XCloseDisplay(dpy);
  131. }
  132. #endif
  133. void pstdout()
  134. {
  135. if (!getstatus(statusstr[0], statusstr[1]))//Only write out if text has changed.
  136. return;
  137. printf("%s\n",statusstr[0]);
  138. fflush(stdout);
  139. }
  140. void statusloop()
  141. {
  142. setupsignals();
  143. int i = 0;
  144. getcmds(-1);
  145. while (1)
  146. {
  147. getcmds(i++);
  148. writestatus();
  149. if (!statusContinue)
  150. break;
  151. sleep(1.0);
  152. }
  153. }
  154. #ifndef __OpenBSD__
  155. /* this signal handler should do nothing */
  156. void dummysighandler(int signum)
  157. {
  158. return;
  159. }
  160. #endif
  161. void sighandler(int signum)
  162. {
  163. getsigcmds(signum-SIGPLUS);
  164. writestatus();
  165. }
  166. void termhandler()
  167. {
  168. statusContinue = 0;
  169. }
  170. int main(int argc, char** argv)
  171. {
  172. for (int i = 0; i < argc; i++) //Handle command line arguments
  173. {
  174. if (!strcmp("-d",argv[i]))
  175. strncpy(delim, argv[++i], delimLen);
  176. else if (!strcmp("-p",argv[i]))
  177. writestatus = pstdout;
  178. }
  179. delimLen = MIN(delimLen, strlen(delim));
  180. delim[delimLen++] = '\0';
  181. signal(SIGTERM, termhandler);
  182. signal(SIGINT, termhandler);
  183. statusloop();
  184. return returnStatus;
  185. }