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.

220 lines
4.5 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. static int setupX();
  42. static Display *dpy;
  43. static int screen;
  44. static Window root;
  45. #else
  46. static void (*writestatus) () = pstdout;
  47. #endif
  48. #include "blocks.h"
  49. static char statusbar[LENGTH(blocks)][CMDLENGTH] = {0};
  50. static char statusstr[2][STATUSLENGTH];
  51. static int statusContinue = 1;
  52. static int returnStatus = 0;
  53. //opens process *cmd and stores output in *output
  54. void getcmd(const Block *block, char *output)
  55. {
  56. //make sure status is same until output is ready
  57. char tempstatus[CMDLENGTH] = {0};
  58. strcpy(tempstatus, block->icon);
  59. FILE *cmdf = popen(block->command, "r");
  60. if (!cmdf)
  61. return;
  62. int i = strlen(block->icon);
  63. fgets(tempstatus+i, CMDLENGTH-i-delimLen, cmdf);
  64. i = strlen(tempstatus);
  65. //if block and command output are both not empty
  66. if (i != 0) {
  67. //only chop off newline if one is present at the end
  68. i = tempstatus[i-1] == '\n' ? i-1 : i;
  69. if (delim[0] != '\0') {
  70. strncpy(tempstatus+i, delim, delimLen);
  71. }
  72. else
  73. tempstatus[i++] = '\0';
  74. }
  75. strcpy(output, tempstatus);
  76. pclose(cmdf);
  77. }
  78. void getcmds(int time)
  79. {
  80. const Block* current;
  81. for (unsigned int i = 0; i < LENGTH(blocks); i++) {
  82. current = blocks + i;
  83. if ((current->interval != 0 && time % current->interval == 0) || time == -1)
  84. getcmd(current,statusbar[i]);
  85. }
  86. }
  87. void getsigcmds(unsigned int signal)
  88. {
  89. const Block *current;
  90. for (unsigned int i = 0; i < LENGTH(blocks); i++) {
  91. current = blocks + i;
  92. if (current->signal == signal)
  93. getcmd(current,statusbar[i]);
  94. }
  95. }
  96. void setupsignals()
  97. {
  98. #ifndef __OpenBSD__
  99. /* initialize all real time signals with dummy handler */
  100. for (int i = SIGRTMIN; i <= SIGRTMAX; i++)
  101. signal(i, dummysighandler);
  102. #endif
  103. for (unsigned int i = 0; i < LENGTH(blocks); i++) {
  104. if (blocks[i].signal > 0)
  105. signal(SIGMINUS+blocks[i].signal, sighandler);
  106. }
  107. }
  108. int getstatus(char *str, char *last)
  109. {
  110. strcpy(last, str);
  111. str[0] = '\0';
  112. for (unsigned int i = 0; i < LENGTH(blocks); i++)
  113. strcat(str, statusbar[i]);
  114. str[strlen(str)-strlen(delim)] = '\0';
  115. return strcmp(str, last);//0 if they are the same
  116. }
  117. #ifndef NO_X
  118. void setroot()
  119. {
  120. if (!getstatus(statusstr[0], statusstr[1])) //Only set root if text has changed.
  121. return;
  122. size_t len = strlen(statusstr[0]) + 3;
  123. char *formatted_status = malloc(len);
  124. snprintf(formatted_status, len, " %s ", statusstr[0]);
  125. XStoreName(dpy, root, formatted_status);
  126. XFlush(dpy);
  127. }
  128. int setupX()
  129. {
  130. dpy = XOpenDisplay(NULL);
  131. if (!dpy) {
  132. fprintf(stderr, "dwmblocks: Failed to open display\n");
  133. return 0;
  134. }
  135. screen = DefaultScreen(dpy);
  136. root = RootWindow(dpy, screen);
  137. return 1;
  138. }
  139. #endif
  140. void pstdout()
  141. {
  142. if (!getstatus(statusstr[0], statusstr[1]))//Only write out if text has changed.
  143. return;
  144. printf("%s\n",statusstr[0]);
  145. fflush(stdout);
  146. }
  147. void statusloop()
  148. {
  149. setupsignals();
  150. int i = 0;
  151. getcmds(-1);
  152. while (1) {
  153. getcmds(i++);
  154. writestatus();
  155. if (!statusContinue)
  156. break;
  157. sleep(1.0);
  158. }
  159. }
  160. #ifndef __OpenBSD__
  161. /* this signal handler should do nothing */
  162. void dummysighandler(int signum)
  163. {
  164. return;
  165. }
  166. #endif
  167. void sighandler(int signum)
  168. {
  169. getsigcmds(signum-SIGPLUS);
  170. writestatus();
  171. }
  172. void termhandler()
  173. {
  174. statusContinue = 0;
  175. }
  176. int main(int argc, char** argv)
  177. {
  178. for (int i = 0; i < argc; i++) {//Handle command line arguments
  179. if (!strcmp("-d",argv[i]))
  180. strncpy(delim, argv[++i], delimLen);
  181. else if (!strcmp("-p",argv[i]))
  182. writestatus = pstdout;
  183. }
  184. #ifndef NO_X
  185. if (!setupX())
  186. return 1;
  187. #endif
  188. delimLen = MIN(delimLen, strlen(delim));
  189. delim[delimLen++] = '\0';
  190. signal(SIGTERM, termhandler);
  191. signal(SIGINT, termhandler);
  192. statusloop();
  193. #ifndef NO_X
  194. XCloseDisplay(dpy);
  195. #endif
  196. return 0;
  197. }