ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/root-tail/root-tail.c
Revision: 1.39
Committed: Wed Mar 31 01:59:19 2004 UTC (20 years, 1 month ago) by pcg
Content type: text/plain
Branch: MAIN
Changes since 1.38: +34 -29 lines
Log Message:
*** empty log message ***

File Contents

# User Rev Content
1 pcg 1.1 /*
2     * Copyright 2001 by Marco d'Itri <md@linux.it>
3 pcg 1.29 * Copyright 2000,2001,2002,2003,2004
4     * Marc Lehmann <pcg@goof.com>,
5     * and many others, see README
6     *
7     * Original version by Mike Baker.
8 pcg 1.1 *
9     * This program is free software; you can redistribute it and/or modify
10     * it under the terms of the GNU General Public License as published by
11     * the Free Software Foundation; either version 2 of the License, or
12     * (at your option) any later version.
13     *
14     * This program is distributed in the hope that it will be useful,
15     * but WITHOUT ANY WARRANTY; without even the implied warranty of
16     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17     * GNU General Public License for more details.
18     *
19     * You should have received a copy of the GNU General Public License
20     * along with this program; if not, write to the Free Software
21     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
22     */
23    
24     #include "config.h"
25     #include <stdlib.h>
26     #include <stdio.h>
27     #include <unistd.h>
28     #include <string.h>
29     #include <signal.h>
30     #include <time.h>
31     #include <fcntl.h>
32     #include <errno.h>
33     #include <sys/time.h>
34     #include <sys/stat.h>
35     #include <sys/types.h>
36 pcg 1.7 #include <locale.h>
37 pcg 1.11 #include <ctype.h>
38     #include <stdarg.h>
39 chris_moore 1.26 #include <X11/Xlib.h>
40     #include <X11/Xatom.h>
41     #include <X11/Xutil.h>
42    
43 pcg 1.1 #if HAS_REGEX
44     #include <regex.h>
45     #endif
46    
47 chris_moore 1.27 #define SHADE_X 2
48     #define SHADE_Y 2
49    
50 pcg 1.1 /* data structures */
51 pcg 1.11 struct logfile_entry
52     {
53     struct logfile_entry *next;
54    
55     char *fname; /* name of file */
56     char *desc; /* alternative description */
57     char *buf; /* text read but not yet displayed */
58     FILE *fp; /* FILE struct associated with file */
59     ino_t inode; /* inode of the file opened */
60     off_t last_size; /* file size at the last check */
61     unsigned long color; /* color to be used for printing */
62     int partial; /* true if the last line isn't complete */
63     int lastpartial; /* true if the previous output wasn't complete */
64     int index; /* index into linematrix of a partial line */
65 chris_moore 1.28 int modified; /* true if line is modified & needs displaying */
66 pcg 1.1 };
67    
68 pcg 1.11 struct linematrix
69     {
70     char *line;
71 pcg 1.12 int len;
72 pcg 1.11 unsigned long color;
73 pcg 1.1 };
74    
75 chris_moore 1.28 struct displaymatrix
76     {
77     char *line;
78     int len;
79     int buffer_size;
80     };
81    
82 pcg 1.1 /* global variables */
83 pcg 1.12 struct linematrix *lines;
84 chris_moore 1.28 struct displaymatrix *display;
85 pcg 1.11 int width = STD_WIDTH, height = STD_HEIGHT, listlen;
86 pcg 1.1 int win_x = LOC_X, win_y = LOC_Y;
87 pcg 1.22 int font_ascent, font_height;
88 chris_moore 1.27 int effect_x_space, effect_y_space; /* how much space does shading / outlining take up */
89     int effect_x_offset, effect_y_offset; /* and how does it offset the usable space */
90 pcg 1.1 int do_reopen;
91 pcg 1.21 struct timeval interval = { 2, 400000 };
92 pcg 1.7 XFontSet fontset;
93 pcg 1.1
94     /* command line options */
95 pcg 1.5 int opt_noinitial, opt_shade, opt_frame, opt_reverse, opt_nofilename,
96 chris_moore 1.24 opt_outline, opt_noflicker, opt_whole, opt_update, opt_wordwrap,
97     geom_mask, reload = 0;
98 pcg 1.1 const char *command = NULL,
99 pcg 1.11 *fontname = USE_FONT, *dispname = NULL, *def_color = DEF_COLOR,
100     *continuation = "[+]";
101 pcg 1.1
102     struct logfile_entry *loglist = NULL, *loglist_tail = NULL;
103    
104     Display *disp;
105     Window root;
106     GC WinGC;
107    
108     #if HAS_REGEX
109 pcg 1.11 struct re_list
110     {
111     regex_t from;
112     const char *to;
113     struct re_list *next;
114 pcg 1.1 };
115     struct re_list *re_head, *re_tail;
116 chris_moore 1.26 char *transform_to = NULL;
117     regex_t *transformre;
118 pcg 1.1 #endif
119    
120    
121     /* prototypes */
122 pcg 1.11 void list_files (int);
123     void force_reopen (int);
124     void force_refresh (int);
125     void blank_window (int);
126    
127     void InitWindow (void);
128     unsigned long GetColor (const char *);
129 chris_moore 1.28 void redraw (int);
130     void refresh (int, int, int, int);
131 pcg 1.11
132     void transform_line (char *s);
133     int lineinput (struct logfile_entry *);
134     void reopen (void);
135     void check_open_files (void);
136     FILE *openlog (struct logfile_entry *);
137 pcg 1.12 static void main_loop (void);
138 pcg 1.11
139     void display_version (void);
140     void display_help (char *);
141     void install_signal (int, void (*)(int));
142     void *xstrdup (const char *);
143     void *xmalloc (size_t);
144 chris_moore 1.28 void *xrealloc (void *, size_t);
145 pcg 1.11 int daemonize (void);
146 pcg 1.1
147     /* signal handlers */
148 pcg 1.11 void
149     list_files (int dummy)
150 pcg 1.1 {
151 pcg 1.11 struct logfile_entry *e;
152 pcg 1.1
153 pcg 1.11 fprintf (stderr, "Files opened:\n");
154     for (e = loglist; e; e = e->next)
155     fprintf (stderr, "\t%s (%s)\n", e->fname, e->desc);
156 pcg 1.1 }
157    
158 pcg 1.11 void
159     force_reopen (int dummy)
160 pcg 1.1 {
161 pcg 1.11 do_reopen = 1;
162 pcg 1.1 }
163    
164 pcg 1.11 void
165     force_refresh (int dummy)
166 pcg 1.1 {
167 chris_moore 1.28 redraw (1);
168 pcg 1.1 }
169    
170 pcg 1.11 void
171     blank_window (int dummy)
172 pcg 1.1 {
173 pcg 1.19 XClearArea (disp, root, win_x - 2, win_y - 2, width + 5, height + 5, False);
174 pcg 1.11 XFlush (disp);
175     exit (0);
176 pcg 1.1 }
177    
178     /* X related functions */
179 pcg 1.11 unsigned long
180     GetColor (const char *ColorName)
181 pcg 1.1 {
182 pcg 1.11 XColor Color;
183     XWindowAttributes Attributes;
184 pcg 1.1
185 pcg 1.11 XGetWindowAttributes (disp, root, &Attributes);
186     Color.pixel = 0;
187 pcg 1.8
188 pcg 1.11 if (!XParseColor (disp, Attributes.colormap, ColorName, &Color))
189     fprintf (stderr, "can't parse %s\n", ColorName);
190     else if (!XAllocColor (disp, Attributes.colormap, &Color))
191     fprintf (stderr, "can't allocate %s\n", ColorName);
192 pcg 1.8
193 pcg 1.11 return Color.pixel;
194 pcg 1.1 }
195    
196 pcg 1.11 static Window
197     root_window (Display * display, int screen_number)
198 pcg 1.7 {
199 pcg 1.11 Atom SWM_VROOT = XInternAtom (display, "__SWM_VROOT", False);
200 pcg 1.7 Window real_root_window = RootWindow (display, screen_number);
201    
202 pcg 1.11 if (root) /* root window set via option */
203 pcg 1.8 return root;
204    
205 pcg 1.11 if (SWM_VROOT != None)
206 pcg 1.7 {
207     Window unused, *windows;
208     unsigned int count;
209    
210     if (XQueryTree (display, real_root_window, &unused, &unused, &windows,
211 pcg 1.11 &count))
212     {
213     int i;
214    
215     for (i = 0; i < count; i++)
216     {
217     Atom type;
218     int format;
219     unsigned long nitems, bytes_after_return;
220     unsigned char *virtual_root_window;
221    
222     if (XGetWindowProperty (display, windows[i], SWM_VROOT,
223     0, 1, False, XA_WINDOW, &type, &format,
224     &nitems, &bytes_after_return,
225     &virtual_root_window) == Success)
226     {
227     if (type != None)
228     {
229     if (type == XA_WINDOW)
230     {
231     XFree (windows);
232     return (Window) virtual_root_window;
233     }
234     else
235     fprintf (stderr,
236     "__SWM_VROOT property type mismatch");
237     }
238     }
239     else
240     fprintf (stderr,
241     "failed to get __SWM_VROOT property on window 0x%lx",
242     windows[i]);
243     }
244    
245     if (count)
246     XFree (windows);
247     }
248 pcg 1.7 else
249 pcg 1.11 fprintf (stderr, "Can't query tree on root window 0x%lx",
250     real_root_window);
251 pcg 1.7 }
252     else
253     /* This shouldn't happen. The Xlib documentation is wrong BTW. */
254     fprintf (stderr, "Can't intern atom __SWM_VROOT");
255    
256     return real_root_window;
257     }
258    
259 pcg 1.11 void
260     InitWindow (void)
261 pcg 1.1 {
262 pcg 1.11 XGCValues gcv;
263     unsigned long gcm;
264     int screen, ScreenWidth, ScreenHeight;
265    
266     if (!(disp = XOpenDisplay (dispname)))
267     {
268     fprintf (stderr, "Can't open display %s.\n", dispname);
269     exit (1);
270     }
271    
272     screen = DefaultScreen (disp);
273     ScreenHeight = DisplayHeight (disp, screen);
274     ScreenWidth = DisplayWidth (disp, screen);
275    
276     root = root_window (disp, screen);
277    
278     gcm = GCBackground;
279     gcv.graphics_exposures = True;
280     WinGC = XCreateGC (disp, root, gcm, &gcv);
281     XMapWindow (disp, root);
282     XSetForeground (disp, WinGC, GetColor (DEF_COLOR));
283    
284     {
285     char **missing_charset_list;
286     int missing_charset_count;
287     char *def_string;
288    
289     fontset = XCreateFontSet (disp, fontname,
290     &missing_charset_list, &missing_charset_count,
291     &def_string);
292    
293     if (missing_charset_count)
294     {
295     fprintf (stderr,
296     "Missing charsets in String to FontSet conversion (%s)\n",
297     missing_charset_list[0]);
298     XFreeStringList (missing_charset_list);
299     }
300     }
301    
302     if (!fontset)
303     {
304     fprintf (stderr, "unable to create fontset, exiting.\n");
305     exit (1);
306     }
307    
308     {
309     XFontSetExtents *xfe = XExtentsOfFontSet (fontset);
310    
311     font_height = xfe->max_logical_extent.height;
312 pcg 1.22 font_ascent = -xfe->max_logical_extent.y;
313 pcg 1.11 }
314    
315     if (geom_mask & XNegative)
316     win_x = win_x + ScreenWidth - width;
317     if (geom_mask & YNegative)
318     win_y = win_y + ScreenHeight - height;
319    
320 chris_moore 1.27 if (opt_outline)
321     {
322     /* adding outline increases the total width and height by 2
323     pixels each, and offsets the text one pixel right and one
324     pixel down */
325     effect_x_space = effect_y_space = 2;
326     effect_x_offset = effect_y_offset = 1;
327     }
328     else if (opt_shade)
329     {
330     /* adding a shadow increases the space used */
331     effect_x_space = abs(SHADE_X);
332     effect_y_space = abs(SHADE_Y);
333     /* if the shadow is to the right and below then we don't need
334     * to move the text to make space for it, but shadows to the left
335     * and above need accomodating */
336     effect_x_offset = SHADE_X > 0 ? 0 : -SHADE_X;
337     effect_y_offset = SHADE_Y > 0 ? 0 : -SHADE_Y;
338     }
339     else
340     {
341     effect_x_space = effect_y_space = 0;
342     effect_x_offset = effect_y_offset = 0;
343     }
344    
345     /* if we are using -shade or -outline, there will be less usable
346     * space for output */
347     listlen = (height - effect_y_space) / font_height;
348 pcg 1.1
349 pcg 1.11 if (!listlen)
350     {
351     fprintf (stderr, "height too small for a single line, setting to %d\n",
352     font_height);
353     listlen = 1;
354     }
355    
356 chris_moore 1.27 /* leave the height how the user requested it. it might not all be
357     * used, but this will allow the geometry to be tuned more accurately
358     * (with the -frame option)
359     * the old code did this:
360     * height = listlen * font_height + effect_y_space; */
361    
362 pcg 1.11 XSelectInput (disp, root, ExposureMask | FocusChangeMask);
363 pcg 1.1 }
364    
365     /*
366 chris_moore 1.28 * if redraw() is passwd a non-zero argument, it does a complete
367     * redraw, rather than an update. if the argument is zero (and
368     * -noflicker is in effect) then only the lines which have changed
369     * since the last draw are redrawn.
370     *
371 pcg 1.1 * the rest is handled by regular refresh()'es
372     */
373 pcg 1.11 void
374 chris_moore 1.28 redraw (int redraw_all)
375 pcg 1.1 {
376 pcg 1.20 XSetClipMask (disp, WinGC, None);
377 chris_moore 1.28 refresh (0, 32768, 1, redraw_all);
378 pcg 1.1 }
379    
380     /* Just redraw everything without clearing (i.e. after an EXPOSE event) */
381 pcg 1.11 void
382 chris_moore 1.28 refresh (int miny, int maxy, int clear, int refresh_all)
383 pcg 1.1 {
384 pcg 1.11 int lin;
385 pcg 1.29 int offset = listlen * font_height + font_ascent + effect_y_offset;
386 pcg 1.11 unsigned long black_color = GetColor ("black");
387 pcg 1.1
388 pcg 1.11 miny -= win_y + font_height;
389     maxy -= win_y - font_height;
390 pcg 1.1
391 pcg 1.22 if (clear && !opt_noflicker)
392     XClearArea (disp, root, win_x, win_y, width, height, False);
393    
394 pcg 1.11 for (lin = listlen; lin--;)
395     {
396 pcg 1.12 struct linematrix *line = lines + (opt_reverse ? listlen - lin - 1 : lin);
397 chris_moore 1.28 struct displaymatrix *display_line = display + lin;
398 pcg 1.7
399 pcg 1.11 offset -= font_height;
400 pcg 1.7
401 pcg 1.11 if (offset < miny || offset > maxy)
402     continue;
403 jebusbfb 1.4
404 pcg 1.29 /* if this line is a different than it was, then it
405     * needs displaying */
406 chris_moore 1.30 if (!opt_noflicker
407     || refresh_all
408 pcg 1.29 || display_line->len != line->len
409     || memcmp (display_line->line, line->line, line->len))
410     {
411 chris_moore 1.30 /* don't bother updating the record of what has been
412     * displayed if -noflicker isn't in effect, since we redraw
413     * the whole display every time anyway */
414     if (opt_noflicker)
415     {
416     /* update the record of what has been displayed;
417     * first make sure the buffer is big enough */
418     if (display_line->buffer_size <= line->len)
419     {
420 pcg 1.39 display_line->buffer_size = line->len;
421 chris_moore 1.30 display_line->line = xrealloc (display_line->line, display_line->buffer_size);
422     }
423 pcg 1.29
424 chris_moore 1.30 display_line->len = line->len;
425     memcpy (display_line->line, line->line, line->len);
426 chris_moore 1.28
427 chris_moore 1.30 if (clear)
428     XClearArea (disp, root, win_x, win_y + offset - font_ascent,
429     width + effect_x_space, font_height + effect_y_space, False);
430     }
431 pcg 1.22
432 pcg 1.29 if (opt_outline)
433     {
434     int x, y;
435     XSetForeground (disp, WinGC, black_color);
436 chris_moore 1.28
437 pcg 1.39 for (x = -1; x <= 1; x += 2)
438     for (y = -1; y <= 1; y += 2)
439 pcg 1.29 XmbDrawString (disp, root, fontset, WinGC,
440     win_x + effect_x_offset + x,
441     win_y + y + offset,
442     line->line, line->len);
443     }
444     else if (opt_shade)
445     {
446     XSetForeground (disp, WinGC, black_color);
447     XmbDrawString (disp, root, fontset, WinGC,
448     win_x + effect_x_offset + SHADE_X,
449     win_y + offset + SHADE_Y,
450     line->line, line->len);
451     }
452 chris_moore 1.27
453 pcg 1.29 XSetForeground (disp, WinGC, line->color);
454     XmbDrawString (disp, root, fontset, WinGC,
455     win_x + effect_x_offset,
456     win_y + offset,
457     line->line, line->len);
458 chris_moore 1.28 }
459 pcg 1.1 }
460    
461 pcg 1.11 if (opt_frame)
462 pcg 1.15 {
463     XSetForeground (disp, WinGC, GetColor (def_color));
464 chris_moore 1.27 XDrawRectangle (disp, root, WinGC, win_x - 0, win_y - 0, width - 1, height - 1);
465 pcg 1.15 }
466 pcg 1.1 }
467    
468     #if HAS_REGEX
469 pcg 1.11 void
470     transform_line (char *s)
471 pcg 1.1 {
472     #ifdef I_AM_Md
473 pcg 1.11 int i;
474     if (1)
475     {
476     for (i = 16; s[i]; i++)
477     s[i] = s[i + 11];
478 pcg 1.1 }
479 pcg 1.11 s[i + 1] = '\0';
480 pcg 1.1 #endif
481    
482 pcg 1.11 if (transformre)
483     {
484     int i;
485     regmatch_t matched[16];
486    
487 chris_moore 1.26 i = regexec (transformre, s, 16, matched, 0);
488 pcg 1.11 if (i == 0)
489     { /* matched */
490 chris_moore 1.26 int match_start = matched[0].rm_so;
491     int match_end = matched[0].rm_eo;
492     int old_len = match_end - match_start;
493 pcg 1.39 int new_len = strlen (transform_to);
494     int old_whole_len = strlen (s);
495    
496     printf ("regexp was matched by '%s' - replace with '%s'\n", s, transform_to);
497     printf ("match is from %d to %d\n", match_start, match_end);
498     if (new_len > old_len)
499 chris_moore 1.28 s = xrealloc(s, old_whole_len + new_len - old_len);
500 pcg 1.39
501     if (new_len != old_len)
502     {
503     memcpy(s + match_end + new_len - old_len,
504     s + match_end,
505     old_whole_len - match_end);
506     s[old_whole_len + new_len - old_len] = '\0';
507     }
508    
509     memcpy (s + match_start,
510     transform_to,
511     new_len);
512     printf ("transformed to '%s'\n", s);
513 pcg 1.11 }
514 chris_moore 1.26 else
515     {
516 pcg 1.39 printf ("regexp was not matched by '%s'\n", s);
517 chris_moore 1.26 }
518 pcg 1.1 }
519     }
520     #endif
521    
522 pcg 1.12 char *
523     concat_line (const char *p1, const char *p2)
524     {
525     int l1 = p1 ? strlen (p1) : 0;
526     int l2 = strlen (p2);
527     char *r = xmalloc (l1 + l2 + 1);
528    
529     memcpy (r, p1, l1);
530     memcpy (r + l1, p2, l2);
531     r[l1 + l2] = 0;
532    
533     return r;
534     }
535 pcg 1.1
536     /*
537 chris_moore 1.32 * HACK-1: This routine should read a single line, no matter how long.
538 pcg 1.1 */
539 pcg 1.11 int
540     lineinput (struct logfile_entry *logfile)
541 pcg 1.1 {
542 pcg 1.12 char buff[1024], *p = buff;
543     int ch;
544 chris_moore 1.32 /* HACK-2: add on the length of any partial line which we will be appending to */
545 pcg 1.12 int ofs = logfile->buf ? strlen (logfile->buf) : 0;
546 pcg 1.1
547 pcg 1.11 do
548     {
549 pcg 1.12 ch = fgetc (logfile->fp);
550 pcg 1.1
551 pcg 1.18 if (ch == '\n' || ch == EOF)
552     break;
553 pcg 1.17 else if (ch == '\r')
554 pcg 1.18 continue; /* skip */
555 pcg 1.12 else if (ch == '\t')
556     {
557     do
558     {
559     *p++ = ' ';
560     ofs++;
561     }
562     while (ofs & 7);
563     }
564     else
565     {
566     *p++ = ch;
567     ofs++;
568     }
569 pcg 1.11 }
570 pcg 1.12 while (p < buff + (sizeof buff) - 8 - 1);
571 pcg 1.1
572 pcg 1.19 if (p == buff && ch == EOF)
573 pcg 1.12 return 0;
574    
575     *p = 0;
576    
577     p = concat_line (logfile->buf, buff);
578     free (logfile->buf); logfile->buf = p;
579 chris_moore 1.9
580 pcg 1.12 logfile->lastpartial = logfile->partial;
581 chris_moore 1.25 /* there are 3 ways we could have exited the loop: reading '\n',
582     * reaching EOF, or filling the buffer; the 2nd and 3rd of these
583     * both result in a partial line */
584     logfile->partial = ch != '\n';
585 pcg 1.12
586     if (logfile->partial && opt_whole)
587 pcg 1.11 return 0;
588 pcg 1.1
589     #if HAS_REGEX
590 pcg 1.12 transform_line (logfile->buf);
591 pcg 1.1 #endif
592 pcg 1.12 return 1;
593 pcg 1.1 }
594    
595     /* input: reads file->fname
596     * output: fills file->fp, file->inode
597     * returns file->fp
598     * in case of error, file->fp is NULL
599     */
600 pcg 1.11 FILE *
601     openlog (struct logfile_entry * file)
602     {
603     struct stat stats;
604    
605     if ((file->fp = fopen (file->fname, "r")) == NULL)
606     {
607     file->fp = NULL;
608     return NULL;
609     }
610    
611     fstat (fileno (file->fp), &stats);
612     if (S_ISFIFO (stats.st_mode))
613     {
614     if (fcntl (fileno (file->fp), F_SETFL, O_NONBLOCK) < 0)
615     perror ("fcntl"), exit (1);
616     file->inode = 0;
617     }
618     else
619     file->inode = stats.st_ino;
620    
621     if (opt_noinitial)
622     fseek (file->fp, 0, SEEK_END);
623     else if (stats.st_size > (listlen + 1) * width)
624     fseek (file->fp, -((listlen + 2) * width), SEEK_END);
625    
626     file->last_size = stats.st_size;
627     return file->fp;
628     }
629    
630     void
631     reopen (void)
632 pcg 1.1 {
633 pcg 1.11 struct logfile_entry *e;
634    
635     for (e = loglist; e; e = e->next)
636     {
637     if (!e->inode)
638     continue; /* skip stdin */
639 pcg 1.1
640 pcg 1.11 if (e->fp)
641     fclose (e->fp);
642     /* if fp is NULL we will try again later */
643     openlog (e);
644     }
645    
646     do_reopen = 0;
647     }
648 pcg 1.1
649 pcg 1.11 void
650     check_open_files (void)
651     {
652     struct logfile_entry *e;
653     struct stat stats;
654 pcg 1.1
655 pcg 1.11 for (e = loglist; e; e = e->next)
656     {
657     if (!e->inode)
658     continue; /* skip stdin */
659 pcg 1.1
660 pcg 1.11 if (stat (e->fname, &stats) < 0)
661     { /* file missing? */
662     sleep (1);
663     if (e->fp)
664     fclose (e->fp);
665 pcg 1.37 if (openlog (e) == NULL)
666     continue;
667     if (fstat (fileno (e->fp), &stats) < 0)
668     continue;
669 pcg 1.11 }
670    
671     if (stats.st_ino != e->inode)
672     { /* file renamed? */
673     if (e->fp)
674     fclose (e->fp);
675     if (openlog (e) == NULL)
676 chris_moore 1.23 continue;
677 pcg 1.39 if (fstat (fileno (e->fp), &stats) < 0)
678     continue;
679 pcg 1.11 }
680    
681     if (stats.st_size < e->last_size)
682     { /* file truncated? */
683     fseek (e->fp, 0, SEEK_SET);
684     e->last_size = stats.st_size;
685     }
686 pcg 1.1 }
687     }
688    
689 pcg 1.14 /*
690     * insert a single physical line (that must be short enough to fit)
691     * at position "idx" by pushing up lines above it. the caller
692     * MUST then fill in lines[idx] with valid data.
693     */
694 pcg 1.12 static void
695     insert_line (int idx)
696     {
697     int cur_line;
698     struct logfile_entry *current;
699    
700     free (lines[0].line);
701    
702     for (cur_line = 0; cur_line < idx; cur_line++)
703     lines[cur_line] = lines[cur_line + 1];
704    
705     for (current = loglist; current; current = current->next)
706 pcg 1.16 if (current->index <= idx)
707 pcg 1.12 current->index--;
708 pcg 1.11 }
709    
710 pcg 1.14 /*
711     * remove a single physical line at position "idx" by moving the lines above it
712     * down and inserting a "~" line at the top.
713     */
714 pcg 1.12 static void
715     delete_line (int idx)
716     {
717     int cur_line;
718     struct logfile_entry *current;
719    
720     for (cur_line = idx; cur_line > 0; cur_line--)
721     lines[cur_line] = lines[cur_line - 1];
722    
723 chris_moore 1.28 lines[0].line = xstrdup ("~");
724 pcg 1.12
725     for (current = loglist; current; current = current->next)
726 pcg 1.16 if (current->index >= 0 && current->index <= idx)
727 pcg 1.12 current->index++;
728     }
729    
730 pcg 1.14 /*
731 chris_moore 1.24 * takes a logical log file line and splits it into multiple physical
732 pcg 1.14 * screen lines by splitting it whenever a part becomes too long.
733     * lal lines will be inserted at position "idx".
734     */
735 pcg 1.12 static void
736     split_line (int idx, const char *str, unsigned long color)
737     {
738     int l = strlen (str);
739 chris_moore 1.33 int last_wrapped = 0;
740 pcg 1.12 const char *p = str;
741 chris_moore 1.33 static int continuation_width = -1;
742     static int continuation_length;
743    
744     /* only calculate the continuation's width once */
745     if (continuation_width == -1)
746     {
747 pcg 1.39 continuation_length = strlen (continuation);
748 chris_moore 1.33 continuation_width = XmbTextEscapement (fontset, continuation, continuation_length);
749     }
750 pcg 1.12
751     do
752     {
753     const char *beg = p;
754 chris_moore 1.33 int w = last_wrapped ? continuation_width : 0;
755     int wrapped = 0;
756 chris_moore 1.24 const char *break_p = NULL;
757 pcg 1.12
758     while (*p)
759     {
760 chris_moore 1.24 /* find the length in bytes of the next multibyte character */
761 pcg 1.12 int len = mblen (p, l);
762     if (len <= 0)
763 chris_moore 1.24 len = 1; /* ignore (don't skip) illegal character sequences */
764 pcg 1.12
765 chris_moore 1.24 /* find the width in pixels of the next character */
766 pcg 1.12 int cw = XmbTextEscapement (fontset, p, len);
767 chris_moore 1.27 if (cw + w > width - effect_x_space)
768 chris_moore 1.24 {
769 chris_moore 1.34 if (p == beg)
770     {
771 pcg 1.39 fprintf (stderr, "we can't even fit a single character onto the line\n");
772     if (len == 1) fprintf (stderr, "(the character we couldn't fit was '%c')\n", *p);
773     exit (1);
774 chris_moore 1.34 }
775    
776 chris_moore 1.24 wrapped = 1;
777     break;
778     }
779    
780     if (opt_wordwrap && len == 1 && p[0] == ' ')
781     break_p = p;
782 pcg 1.12
783     w += cw;
784     p += len;
785     l -= len;
786     }
787    
788 chris_moore 1.24 /* if we're wrapping at spaces, and the line is long enough to
789     * wrap, and we've seen a space already, and the space wasn't
790     * the first character on the line, then wrap at the space */
791     if (opt_wordwrap && wrapped && break_p && break_p != beg)
792     {
793     l += p - break_p;
794     p = break_p;
795     }
796    
797 pcg 1.12 {
798 chris_moore 1.32 /* HACK-4 - consider inserting the 'continuation string'
799     * before the rest of the wrapped line */
800 chris_moore 1.33 int len = p - beg + (last_wrapped ? continuation_length : 0);
801     char *s = xmalloc (len + 1);
802     if (last_wrapped)
803     {
804     memcpy (s, continuation, continuation_length);
805     memcpy (s + continuation_length, beg, p - beg);
806     }
807     else
808 pcg 1.39 memcpy (s, beg, len);
809 chris_moore 1.33
810     s[len] = 0;
811 pcg 1.12 insert_line (idx);
812     lines[idx].line = s;
813 chris_moore 1.33 lines[idx].len = len;
814 pcg 1.12 lines[idx].color = color;
815     }
816 chris_moore 1.24
817     /* if we wrapped at a space, don't display the space */
818     if (opt_wordwrap && wrapped && break_p && break_p != beg)
819     {
820     l--;
821     p++;
822     }
823 chris_moore 1.33
824     last_wrapped = wrapped;
825 pcg 1.12 }
826     while (l);
827     }
828    
829 pcg 1.14 /*
830     * append something to an existing physical line. this is done
831     * by deleting the file on-screen, concatenating the new data to it
832     * and splitting it again.
833     */
834 pcg 1.12 static void
835     append_line (int idx, const char *str)
836     {
837     unsigned long color = lines[idx].color;
838     char *old = lines[idx].line;
839     char *new = concat_line (old, str);
840    
841     free (old);
842    
843     delete_line (idx);
844     split_line (idx, new, color);
845     }
846    
847     static void
848 pcg 1.11 main_loop (void)
849     {
850 pcg 1.12 lines = xmalloc (sizeof (struct linematrix) * listlen);
851 chris_moore 1.28 display = xmalloc (sizeof (struct displaymatrix) * listlen);
852 pcg 1.20 int lin;
853 pcg 1.11 time_t lastreload;
854     Region region = XCreateRegion ();
855     XEvent xev;
856     struct logfile_entry *lastprinted = NULL;
857     struct logfile_entry *current;
858 pcg 1.18 int need_update = 1;
859 pcg 1.11
860     lastreload = time (NULL);
861    
862     /* Initialize linematrix */
863     for (lin = 0; lin < listlen; lin++)
864     {
865 chris_moore 1.28 lines[lin].line = xstrdup ("~");
866 pcg 1.12 lines[lin].len = 1;
867 chris_moore 1.28 display[lin].line = xstrdup("");
868     display[lin].len = 0;
869 pcg 1.39 display[lin].buffer_size = 0;
870 pcg 1.11 lines[lin].color = GetColor (def_color);
871     }
872    
873     for (;;)
874     {
875     /* read logs */
876     for (current = loglist; current; current = current->next)
877     {
878     if (!current->fp)
879     continue; /* skip missing files */
880    
881     clearerr (current->fp);
882    
883 pcg 1.12 while (lineinput (current))
884 pcg 1.11 {
885 pcg 1.12 need_update = 1;
886 pcg 1.11 /* if we're trying to update old partial lines in
887     * place, and the last time this file was updated the
888     * output was partial, and that partial line is not
889     * too close to the top of the screen, then update
890     * that partial line */
891 pcg 1.16 if (opt_update && current->lastpartial && current->index >= 0)
892 pcg 1.11 {
893 pcg 1.16 int idx = current->index;
894     append_line (idx, current->buf);
895     current->index = idx;
896 pcg 1.13 free (current->buf), current->buf = 0;
897 pcg 1.12 continue;
898 pcg 1.11 }
899    
900     /* print filename if any, and if last line was from
901     * different file */
902 pcg 1.13 if (!opt_nofilename && lastprinted != current && current->desc[0])
903 pcg 1.11 {
904 chris_moore 1.32 char buf[1024]; /* HACK-5 */
905 pcg 1.12 snprintf (buf, sizeof (buf), "[%s]", current->desc);
906     split_line (listlen - 1, buf, current->color);
907 pcg 1.11 }
908    
909 chris_moore 1.31 /* if we're dealing with partial lines, and the last
910     * time we showed the line it wasn't finished ... */
911     if (!opt_whole && current->lastpartial)
912 chris_moore 1.23 {
913 chris_moore 1.31 /* if this is the same file we showed last then
914     append to the last line shown */
915     if (lastprinted == current)
916     append_line (listlen - 1, current->buf);
917     else
918     {
919     /* but if a different file has been shown in the
920     * mean time, make a new line, starting with the
921     * continuation string */
922     split_line (listlen - 1, continuation, current->color);
923     append_line (listlen - 1, current->buf);
924     }
925 chris_moore 1.23 }
926 pcg 1.11 else
927 chris_moore 1.31 /* otherwise just make a plain and simple new line */
928 chris_moore 1.23 split_line (listlen - 1, current->buf, current->color);
929 pcg 1.11
930 chris_moore 1.23 free (current->buf), current->buf = 0;
931 pcg 1.11 current->index = listlen - 1;
932     lastprinted = current;
933     }
934     }
935    
936     if (need_update)
937 pcg 1.18 {
938 chris_moore 1.28 redraw (0);
939 pcg 1.18 need_update = 0;
940     }
941 pcg 1.11 else
942     {
943     XFlush (disp);
944    
945     if (!XPending (disp))
946     {
947     fd_set fdr;
948     struct timeval to = interval;
949    
950     FD_ZERO (&fdr);
951     FD_SET (ConnectionNumber (disp), &fdr);
952     select (ConnectionNumber (disp) + 1, &fdr, 0, 0, &to);
953     }
954     }
955    
956     check_open_files ();
957    
958     if (do_reopen)
959     reopen ();
960    
961     /* we ignore possible errors due to window resizing &c */
962     while (XPending (disp))
963     {
964     XNextEvent (disp, &xev);
965    
966     switch (xev.type)
967     {
968     case Expose:
969     {
970     XRectangle r;
971    
972     r.x = xev.xexpose.x;
973     r.y = xev.xexpose.y;
974     r.width = xev.xexpose.width;
975     r.height = xev.xexpose.height;
976 pcg 1.20
977 pcg 1.11 XUnionRectWithRegion (&r, region, region);
978     }
979     break;
980     default:
981     #ifdef DEBUGMODE
982     fprintf (stderr, "PANIC! Unknown event %d\n", xev.type);
983     #endif
984     break;
985     }
986     }
987    
988     /* reload if requested */
989     if (reload && lastreload + reload < time (NULL))
990     {
991     if (command && command[0])
992     system (command);
993    
994     reopen ();
995     lastreload = time (NULL);
996     }
997    
998     if (!XEmptyRegion (region))
999     {
1000 pcg 1.20 XRectangle r;
1001    
1002 pcg 1.11 XSetRegion (disp, WinGC, region);
1003 pcg 1.20 XClipBox (region, &r);
1004    
1005 chris_moore 1.28 refresh (r.y, r.y + r.height, 0, 1);
1006 pcg 1.11
1007     XDestroyRegion (region);
1008     region = XCreateRegion ();
1009     }
1010     }
1011     }
1012 pcg 1.1
1013 pcg 1.11
1014     int
1015     main (int argc, char *argv[])
1016 pcg 1.1 {
1017 pcg 1.11 int i;
1018     int opt_daemonize = 0;
1019     int opt_partial = 0, file_count = 0;
1020 pcg 1.1 #if HAS_REGEX
1021 pcg 1.11 char *transform = NULL;
1022 pcg 1.1 #endif
1023    
1024 pcg 1.11 setlocale (LC_CTYPE, ""); /* try to initialize the locale. */
1025    
1026     /* window needs to be initialized before colorlookups can be done */
1027     /* just a dummy to get the color lookups right */
1028     geom_mask = NoValue;
1029     InitWindow ();
1030    
1031     for (i = 1; i < argc; i++)
1032     {
1033     const char *arg = argv[i];
1034 pcg 1.7
1035 pcg 1.11 if (arg[0] == '-' && arg[1] != '\0' && arg[1] != ',')
1036     {
1037     if (arg[1] == '-')
1038     arg++;
1039    
1040     if (!strcmp (arg, "-?") ||
1041     !strcmp (arg, "-help") || !strcmp (arg, "-h"))
1042     display_help (argv[0]);
1043     else if (!strcmp (arg, "-V"))
1044     display_version ();
1045     else if (!strcmp (arg, "-g") || !strcmp (arg, "-geometry"))
1046     geom_mask =
1047     XParseGeometry (argv[++i], &win_x, &win_y, &width, &height);
1048     else if (!strcmp (arg, "-display"))
1049     dispname = argv[++i];
1050     else if (!strcmp (arg, "-cont"))
1051     continuation = argv[++i];
1052     else if (!strcmp (arg, "-font") || !strcmp (arg, "-fn"))
1053     fontname = argv[++i];
1054 pcg 1.1 #if HAS_REGEX
1055 pcg 1.11 else if (!strcmp (arg, "-t"))
1056 chris_moore 1.26 {
1057     transform = argv[++i];
1058     transform_to = argv[++i];
1059     printf("transform: '%s' to '%s'\n", transform, transform_to);
1060     }
1061 pcg 1.1 #endif
1062 pcg 1.11 else if (!strcmp (arg, "-fork") || !strcmp (arg, "-f"))
1063     opt_daemonize = 1;
1064     else if (!strcmp (arg, "-reload"))
1065     {
1066     reload = atoi (argv[++i]);
1067     command = argv[++i];
1068     }
1069     else if (!strcmp (arg, "-shade"))
1070     opt_shade = 1;
1071 pcg 1.21 else if (!strcmp (arg, "-outline"))
1072     opt_outline = 1;
1073 pcg 1.22 else if (!strcmp (arg, "-noflicker"))
1074     opt_noflicker = 1;
1075 pcg 1.11 else if (!strcmp (arg, "-frame"))
1076     opt_frame = 1;
1077     else if (!strcmp (arg, "-no-filename"))
1078     opt_nofilename = 1;
1079     else if (!strcmp (arg, "-reverse"))
1080     opt_reverse = 1;
1081     else if (!strcmp (arg, "-whole"))
1082     opt_whole = 1;
1083     else if (!strcmp (arg, "-partial"))
1084     opt_partial = 1;
1085     else if (!strcmp (arg, "-update"))
1086     opt_update = opt_partial = 1;
1087 chris_moore 1.24 else if (!strcmp (arg, "-wordwrap"))
1088     opt_wordwrap = 1;
1089 pcg 1.11 else if (!strcmp (arg, "-color"))
1090     def_color = argv[++i];
1091     else if (!strcmp (arg, "-noinitial"))
1092     opt_noinitial = 1;
1093     else if (!strcmp (arg, "-id"))
1094     root = atoi (argv[++i]);
1095     else if (!strcmp (arg, "-interval") || !strcmp (arg, "-i"))
1096     {
1097     double iv = atof (argv[++i]);
1098    
1099     interval.tv_sec = (int) iv;
1100     interval.tv_usec = (iv - interval.tv_sec) * 1e6;
1101     }
1102     else
1103     {
1104     fprintf (stderr, "Unknown option '%s'.\n"
1105     "Try --help for more information.\n", arg);
1106     exit (1);
1107     }
1108     }
1109     else
1110     { /* it must be a filename */
1111     struct logfile_entry *e;
1112     const char *fname, *desc, *fcolor = def_color;
1113     char *p;
1114    
1115     file_count++;
1116    
1117     /* this is not foolproof yet (',' in filenames are not allowed) */
1118     fname = desc = arg;
1119     if ((p = strchr (arg, ',')))
1120     {
1121     *p = '\0';
1122     fcolor = p + 1;
1123    
1124     if ((p = strchr (fcolor, ',')))
1125     {
1126     *p = '\0';
1127     desc = p + 1;
1128     }
1129     }
1130    
1131     e = xmalloc (sizeof (struct logfile_entry));
1132 pcg 1.12 e->partial = 0;
1133     e->buf = 0;
1134 pcg 1.16 e->index = -1;
1135 pcg 1.12
1136 pcg 1.11 if (arg[0] == '-' && arg[1] == '\0')
1137     {
1138     if ((e->fp = fdopen (0, "r")) == NULL)
1139     perror ("fdopen"), exit (1);
1140     if (fcntl (0, F_SETFL, O_NONBLOCK) < 0)
1141     perror ("fcntl"), exit (1);
1142 pcg 1.39
1143 pcg 1.11 e->fname = NULL;
1144     e->inode = 0;
1145     e->desc = xstrdup ("stdin");
1146     }
1147     else
1148     {
1149 pcg 1.39 e->fname = xstrdup (fname);
1150 pcg 1.11
1151     if (openlog (e) == NULL)
1152     perror (fname), exit (1);
1153    
1154 pcg 1.38 e->desc = xstrdup (desc);
1155 pcg 1.11 }
1156    
1157     e->color = GetColor (fcolor);
1158     e->partial = 0;
1159     e->next = NULL;
1160    
1161     if (!loglist)
1162     loglist = e;
1163     if (loglist_tail)
1164     loglist_tail->next = e;
1165 pcg 1.39
1166 pcg 1.11 loglist_tail = e;
1167     }
1168     }
1169    
1170     if (!loglist)
1171     {
1172     fprintf (stderr, "You did not specify any files to tail\n"
1173     "use %s --help for help\n", argv[0]);
1174     exit (1);
1175     }
1176    
1177     if (opt_partial && opt_whole)
1178     {
1179     fprintf (stderr, "Specify at most one of -partial and -whole\n");
1180     exit (1);
1181 chris_moore 1.9 }
1182    
1183 chris_moore 1.32 /* HACK-7: do we want to allow both -shade and -outline? */
1184 chris_moore 1.27 if (opt_shade && opt_outline)
1185     {
1186     fprintf (stderr, "Specify at most one of -shade and -outline\n");
1187     exit (1);
1188     }
1189    
1190 pcg 1.11 if (opt_partial)
1191 chris_moore 1.9 /* if we specifically requested to see partial lines then don't insist on whole lines */
1192 pcg 1.11 opt_whole = 0;
1193     else if (file_count > 1)
1194 chris_moore 1.31 /* otherwise, if we're viewing multiple files, default to showing whole lines */
1195 pcg 1.11 opt_whole = 1;
1196 chris_moore 1.9
1197 pcg 1.1 #if HAS_REGEX
1198 pcg 1.11 if (transform)
1199     {
1200     int i;
1201 pcg 1.1
1202 chris_moore 1.26 printf("compiling regexp '%s'\n", transform);
1203     transformre = xmalloc (sizeof (regex_t));
1204     i = regcomp (transformre, transform, REG_EXTENDED);
1205 pcg 1.11 if (i != 0)
1206     {
1207     char buf[512];
1208    
1209 chris_moore 1.26 regerror (i, transformre, buf, sizeof (buf));
1210 pcg 1.11 fprintf (stderr, "Cannot compile regular expression: %s\n", buf);
1211     }
1212 chris_moore 1.26 else
1213     {
1214     printf("compiled '%s' OK to %x\n", transform, (int)transformre);
1215     }
1216 pcg 1.1 }
1217     #endif
1218    
1219 pcg 1.11 InitWindow ();
1220 pcg 1.1
1221 pcg 1.11 install_signal (SIGINT, blank_window);
1222     install_signal (SIGQUIT, blank_window);
1223     install_signal (SIGTERM, blank_window);
1224     install_signal (SIGHUP, force_reopen);
1225     install_signal (SIGUSR1, list_files);
1226     install_signal (SIGUSR2, force_refresh);
1227 pcg 1.1
1228 pcg 1.11 if (opt_daemonize)
1229     daemonize ();
1230 pcg 1.1
1231 pcg 1.11 main_loop ();
1232 pcg 1.1
1233 pcg 1.11 exit (1); /* to make gcc -Wall stop complaining */
1234 pcg 1.1 }
1235    
1236 pcg 1.11 void
1237     install_signal (int sig, void (*handler) (int))
1238 pcg 1.1 {
1239 pcg 1.11 struct sigaction action;
1240    
1241     action.sa_handler = handler;
1242     sigemptyset (&action.sa_mask);
1243     action.sa_flags = SA_RESTART;
1244 pcg 1.1
1245 pcg 1.11 if (sigaction (sig, &action, NULL) < 0)
1246     fprintf (stderr, "sigaction(%d): %s\n", sig, strerror (errno)), exit (1);
1247 pcg 1.1 }
1248    
1249 pcg 1.11 void *
1250     xstrdup (const char *string)
1251 pcg 1.1 {
1252 pcg 1.11 void *p;
1253 pcg 1.1
1254 pcg 1.11 while ((p = strdup (string)) == NULL)
1255     {
1256     fprintf (stderr, "Memory exausted.");
1257     sleep (10);
1258 pcg 1.1 }
1259 pcg 1.11
1260     return p;
1261 pcg 1.1 }
1262    
1263 pcg 1.11 void *
1264     xmalloc (size_t size)
1265 pcg 1.1 {
1266 pcg 1.11 void *p;
1267 pcg 1.1
1268 pcg 1.11 while ((p = malloc (size)) == NULL)
1269     {
1270     fprintf (stderr, "Memory exausted.");
1271     sleep (10);
1272 pcg 1.1 }
1273 pcg 1.12
1274 pcg 1.11 return p;
1275 pcg 1.1 }
1276    
1277 chris_moore 1.28 void *
1278     xrealloc (void *ptr, size_t size)
1279     {
1280     void *p;
1281    
1282     while ((p = realloc (ptr, size)) == NULL)
1283     {
1284     fprintf (stderr, "Memory exausted.");
1285     sleep (10);
1286     }
1287    
1288     return p;
1289     }
1290    
1291 pcg 1.11 void
1292     display_help (char *myname)
1293     {
1294     printf ("Usage: %s [options] file1[,color[,desc]] "
1295     "[file2[,color[,desc]] ...]\n", myname);
1296     printf (" -g | -geometry geometry -g WIDTHxHEIGHT+X+Y\n"
1297     " -color color use color $color as default\n"
1298     " -reload sec command reload after $sec and run command\n"
1299     " -id id window id to use instead of the root window\n"
1300     " -font FONTSPEC (-fn) font to use\n"
1301     " -f | -fork fork into background\n"
1302     " -reverse print new lines at the top\n"
1303     " -whole wait for \\n before showing a line\n"
1304     " -partial show lines even if they don't end with a \\n\n"
1305     " -update allow updates to old partial lines\n"
1306     " -cont string to prefix continued partial lines with\n"
1307 chris_moore 1.24 " -wordwrap wrap long lines at spaces to avoid breaking words\n"
1308 pcg 1.11 " defaults to \"[+]\"\n"
1309     " -shade add shading to font\n"
1310     " -noinitial don't display the last file lines on\n"
1311     " startup\n"
1312     " -i | -interval seconds interval between checks (fractional\n"
1313 pcg 1.21 " values o.k.). Default 2.4 seconds\n"
1314 pcg 1.11 " -V display version information and exit\n"
1315     "\n");
1316     printf ("Example:\n%s -g 80x25+100+50 -font fixed /var/log/messages,green "
1317     "/var/log/secure,red,'ALERT'\n", myname);
1318     exit (0);
1319     }
1320    
1321     void
1322     display_version (void)
1323 pcg 1.1 {
1324 pcg 1.11 printf ("root-tail version " VERSION "\n");
1325     exit (0);
1326 pcg 1.1 }
1327    
1328 pcg 1.11 int
1329     daemonize (void)
1330     {
1331 chris_moore 1.26 pid_t pid;
1332    
1333     switch (pid = fork ())
1334 pcg 1.11 {
1335 pcg 1.1 case -1:
1336 pcg 1.11 return -1;
1337 pcg 1.1 case 0:
1338 pcg 1.11 break;
1339 pcg 1.1 default:
1340 pcg 1.29 /*printf("%d\n", pid);*/
1341 chris_moore 1.26 exit (0);
1342 pcg 1.1 }
1343    
1344 pcg 1.11 if (setsid () == -1)
1345     return -1;
1346 pcg 1.1
1347 pcg 1.11 return 0;
1348 pcg 1.1 }