ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/root-tail/root-tail.c
(Generate patch)

Comparing root-tail/root-tail.c (file contents):
Revision 1.3 by pcg, Sun May 5 19:10:39 2002 UTC vs.
Revision 1.18 by pcg, Sat Mar 27 16:12:00 2004 UTC

28#include <fcntl.h> 28#include <fcntl.h>
29#include <errno.h> 29#include <errno.h>
30#include <sys/time.h> 30#include <sys/time.h>
31#include <sys/stat.h> 31#include <sys/stat.h>
32#include <sys/types.h> 32#include <sys/types.h>
33#include <locale.h>
34#include <ctype.h>
35#include <stdarg.h>
33#if HAS_REGEX 36#if HAS_REGEX
34#include <regex.h> 37#include <regex.h>
35#endif 38#endif
36#include <X11/Xlib.h> 39#include <X11/Xlib.h>
40#include <X11/Xatom.h>
37#include <X11/Xutil.h> 41#include <X11/Xutil.h>
38 42
39/* data structures */ 43/* data structures */
40struct logfile_entry { 44struct logfile_entry
41 char *fname; /* name of file */ 45{
42 char *desc; /* alternative description */
43 FILE *fp; /* FILE struct associated with file */
44 ino_t inode; /* inode of the file opened */
45 off_t last_size; /* file size at the last check */
46 unsigned long color; /* color to be used for printing */
47 struct logfile_entry *next; 46 struct logfile_entry *next;
47
48 char *fname; /* name of file */
49 char *desc; /* alternative description */
50 char *buf; /* text read but not yet displayed */
51 FILE *fp; /* FILE struct associated with file */
52 ino_t inode; /* inode of the file opened */
53 off_t last_size; /* file size at the last check */
54 unsigned long color; /* color to be used for printing */
55 int partial; /* true if the last line isn't complete */
56 int lastpartial; /* true if the previous output wasn't complete */
57 int index; /* index into linematrix of a partial line */
48}; 58};
49 59
50struct linematrix { 60struct linematrix
61{
51 char *line; 62 char *line;
63 int len;
52 unsigned long color; 64 unsigned long color;
53}; 65};
54 66
55
56/* global variables */ 67/* global variables */
57unsigned int width = STD_WIDTH, listlen = STD_HEIGHT; 68struct linematrix *lines;
69int width = STD_WIDTH, height = STD_HEIGHT, listlen;
58int win_x = LOC_X, win_y = LOC_Y; 70int win_x = LOC_X, win_y = LOC_Y;
59int w = -1, h = -1, font_width, font_height, font_descent; 71int font_descent, font_height;
60int do_reopen; 72int do_reopen;
61struct timeval interval = { 2, 400000 }; /* see Knuth */ 73struct timeval interval = { 3, 0 };
74XFontSet fontset;
62 75
63/* command line options */ 76/* command line options */
64int opt_noinitial, opt_shade, opt_frame, opt_reverse=0, opt_nofilename, 77int opt_noinitial, opt_shade, opt_frame, opt_reverse, opt_nofilename,
65 geom_mask, reload = 3600; 78 opt_whole, opt_update, geom_mask, reload = 0;
66const char *command = NULL, 79const char *command = NULL,
67 *fontname = USE_FONT, *dispname = NULL, *def_color = DEF_COLOR; 80 *fontname = USE_FONT, *dispname = NULL, *def_color = DEF_COLOR,
81 *continuation = "[+]";
68 82
69struct logfile_entry *loglist = NULL, *loglist_tail = NULL; 83struct logfile_entry *loglist = NULL, *loglist_tail = NULL;
70 84
71Display *disp; 85Display *disp;
72Window root; 86Window root;
73GC WinGC; 87GC WinGC;
74 88
75#if HAS_REGEX 89#if HAS_REGEX
76struct re_list { 90struct re_list
91{
77 regex_t from; 92 regex_t from;
78 const char *to; 93 const char *to;
79 struct re_list *next; 94 struct re_list *next;
80}; 95};
81struct re_list *re_head, *re_tail; 96struct re_list *re_head, *re_tail;
82#endif 97#endif
83 98
84 99
85/* prototypes */ 100/* prototypes */
86void list_files(int); 101void list_files (int);
87void force_reopen(int); 102void force_reopen (int);
88void force_refresh(int); 103void force_refresh (int);
89void blank_window(int); 104void blank_window (int);
90 105
91void InitWindow(void); 106void InitWindow (void);
92unsigned long GetColor(const char *); 107unsigned long GetColor (const char *);
93void redraw(void); 108void redraw (void);
94void refresh(struct linematrix *, int, int); 109void refresh (int, int);
95 110
96void transform_line(char *s); 111void transform_line (char *s);
97int lineinput(char *, int, FILE *); 112int lineinput (struct logfile_entry *);
98void reopen(void); 113void reopen (void);
99void check_open_files(void); 114void check_open_files (void);
100FILE *openlog(struct logfile_entry *); 115FILE *openlog (struct logfile_entry *);
101void main_loop(void); 116static void main_loop (void);
102 117
103void display_version(void); 118void display_version (void);
104void display_help(char *); 119void display_help (char *);
105void install_signal(int, void (*)(int)); 120void install_signal (int, void (*)(int));
106void *xstrdup(const char *); 121void *xstrdup (const char *);
107void *xmalloc(size_t); 122void *xmalloc (size_t);
108int daemonize(void); 123int daemonize (void);
109 124
110/* signal handlers */ 125/* signal handlers */
126void
111void list_files(int dummy) 127list_files (int dummy)
112{ 128{
113 struct logfile_entry *e; 129 struct logfile_entry *e;
114 130
115 fprintf(stderr, "Files opened:\n"); 131 fprintf (stderr, "Files opened:\n");
116 for (e = loglist; e; e = e->next) 132 for (e = loglist; e; e = e->next)
117 fprintf(stderr, "\t%s (%s)\n", e->fname, e->desc); 133 fprintf (stderr, "\t%s (%s)\n", e->fname, e->desc);
118} 134}
119 135
136void
120void force_reopen(int dummy) 137force_reopen (int dummy)
121{ 138{
122 do_reopen = 1; 139 do_reopen = 1;
123} 140}
124 141
142void
125void force_refresh(int dummy) 143force_refresh (int dummy)
126{ 144{
127 XClearWindow(disp, root);
128 redraw(); 145 redraw ();
129} 146}
130 147
148void
131void blank_window(int dummy) 149blank_window (int dummy)
132{ 150{
133 XClearWindow(disp, root); 151 XClearArea (disp, root, win_x, win_y, width, height, False);
134 XFlush(disp); 152 XFlush (disp);
135 exit(0); 153 exit (0);
136} 154}
137 155
138/* X related functions */ 156/* X related functions */
157unsigned long
139unsigned long GetColor(const char *ColorName) 158GetColor (const char *ColorName)
140{ 159{
141 XColor Color; 160 XColor Color;
142 XWindowAttributes Attributes; 161 XWindowAttributes Attributes;
143 162
144 XGetWindowAttributes(disp, root, &Attributes); 163 XGetWindowAttributes (disp, root, &Attributes);
145 Color.pixel = 0; 164 Color.pixel = 0;
165
146 if (!XParseColor(disp, Attributes.colormap, ColorName, &Color)) 166 if (!XParseColor (disp, Attributes.colormap, ColorName, &Color))
147 fprintf(stderr, "can't parse %s\n", ColorName); 167 fprintf (stderr, "can't parse %s\n", ColorName);
148 else if (!XAllocColor(disp, Attributes.colormap, &Color)) 168 else if (!XAllocColor (disp, Attributes.colormap, &Color))
149 fprintf(stderr, "can't allocate %s\n", ColorName); 169 fprintf (stderr, "can't allocate %s\n", ColorName);
170
150 return Color.pixel; 171 return Color.pixel;
151} 172}
152 173
153void InitWindow(void) 174static Window
175root_window (Display * display, int screen_number)
154{ 176{
155 XGCValues gcv; 177 Atom SWM_VROOT = XInternAtom (display, "__SWM_VROOT", False);
156 Font font; 178 Window real_root_window = RootWindow (display, screen_number);
157 unsigned long gcm;
158 XFontStruct *info;
159 int screen, ScreenWidth, ScreenHeight;
160 179
161 if (!(disp = XOpenDisplay(dispname))) { 180 if (root) /* root window set via option */
162 fprintf(stderr, "Can't open display %s.\n", dispname); 181 return root;
163 exit(1); 182
183 if (SWM_VROOT != None)
164 } 184 {
165 screen = DefaultScreen(disp); 185 Window unused, *windows;
166 ScreenHeight = DisplayHeight(disp, screen); 186 unsigned int count;
167 ScreenWidth = DisplayWidth(disp, screen);
168 root = RootWindow(disp, screen);
169 gcm = GCBackground;
170 gcv.graphics_exposures = True;
171 WinGC = XCreateGC(disp, root, gcm, &gcv);
172 XMapWindow(disp, root);
173 XSetForeground(disp, WinGC, GetColor(DEF_COLOR));
174 187
175 font = XLoadFont(disp, fontname); 188 if (XQueryTree (display, real_root_window, &unused, &unused, &windows,
176 XSetFont(disp, WinGC, font); 189 &count))
177 info = XQueryFont(disp, font);
178 font_width = info->max_bounds.width;
179 font_descent = info->max_bounds.descent;
180 font_height = info->max_bounds.ascent + font_descent;
181
182 w = width * font_width;
183 h = listlen * font_height;
184 if (geom_mask & XNegative)
185 win_x = win_x + ScreenWidth - w;
186 if (geom_mask & YNegative)
187 win_y = win_y + ScreenHeight - h;
188
189 XSelectInput(disp, root, ExposureMask|FocusChangeMask);
190}
191
192char *
193detabificate (char *s)
194{
195 char * out;
196 int i, j;
197
198 out = malloc (8 * strlen (s) + 1);
199
200 for(i = 0, j = 0; s[i]; i++)
201 {
202 if (s[i] == '\t')
203 do 190 {
204 out[j++] = ' '; 191 int i;
205 while (j % 8); 192
193 for (i = 0; i < count; i++)
194 {
195 Atom type;
196 int format;
197 unsigned long nitems, bytes_after_return;
198 unsigned char *virtual_root_window;
199
200 if (XGetWindowProperty (display, windows[i], SWM_VROOT,
201 0, 1, False, XA_WINDOW, &type, &format,
202 &nitems, &bytes_after_return,
203 &virtual_root_window) == Success)
204 {
205 if (type != None)
206 {
207 if (type == XA_WINDOW)
208 {
209 XFree (windows);
210 return (Window) virtual_root_window;
211 }
212 else
213 fprintf (stderr,
214 "__SWM_VROOT property type mismatch");
215 }
216 }
217 else
218 fprintf (stderr,
219 "failed to get __SWM_VROOT property on window 0x%lx",
220 windows[i]);
221 }
222
223 if (count)
224 XFree (windows);
225 }
206 else 226 else
207 out[j++] = s[i]; 227 fprintf (stderr, "Can't query tree on root window 0x%lx",
228 real_root_window);
229 }
230 else
231 /* This shouldn't happen. The Xlib documentation is wrong BTW. */
232 fprintf (stderr, "Can't intern atom __SWM_VROOT");
233
234 return real_root_window;
235}
236
237void
238InitWindow (void)
239{
240 XGCValues gcv;
241 unsigned long gcm;
242 int screen, ScreenWidth, ScreenHeight;
243
244 if (!(disp = XOpenDisplay (dispname)))
245 {
246 fprintf (stderr, "Can't open display %s.\n", dispname);
247 exit (1);
248 }
249
250 screen = DefaultScreen (disp);
251 ScreenHeight = DisplayHeight (disp, screen);
252 ScreenWidth = DisplayWidth (disp, screen);
253
254 root = root_window (disp, screen);
255
256 gcm = GCBackground;
257 gcv.graphics_exposures = True;
258 WinGC = XCreateGC (disp, root, gcm, &gcv);
259 XMapWindow (disp, root);
260 XSetForeground (disp, WinGC, GetColor (DEF_COLOR));
261
262 {
263 char **missing_charset_list;
264 int missing_charset_count;
265 char *def_string;
266
267 fontset = XCreateFontSet (disp, fontname,
268 &missing_charset_list, &missing_charset_count,
269 &def_string);
270
271 if (missing_charset_count)
272 {
273 fprintf (stderr,
274 "Missing charsets in String to FontSet conversion (%s)\n",
275 missing_charset_list[0]);
276 XFreeStringList (missing_charset_list);
208 } 277 }
209 278 }
210 out[j] = '\0'; 279
211 return out; 280 if (!fontset)
281 {
282 fprintf (stderr, "unable to create fontset, exiting.\n");
283 exit (1);
284 }
285
286 {
287 XFontSetExtents *xfe = XExtentsOfFontSet (fontset);
288
289 font_height = xfe->max_logical_extent.height;
290 font_descent = xfe->max_logical_extent.y;
291 }
292
293 if (geom_mask & XNegative)
294 win_x = win_x + ScreenWidth - width;
295 if (geom_mask & YNegative)
296 win_y = win_y + ScreenHeight - height;
297
298 listlen = height / font_height;
299
300 if (!listlen)
301 {
302 fprintf (stderr, "height too small for a single line, setting to %d\n",
303 font_height);
304 listlen = 1;
305 }
306
307 height = listlen * font_height;
308
309 XSelectInput (disp, root, ExposureMask | FocusChangeMask);
212} 310}
213 311
214/* 312/*
215 * redraw does a complete redraw, rather than an update (i.e. the area 313 * redraw does a complete redraw, rather than an update (i.e. the area
216 * gets cleared first) 314 * gets cleared first)
217 * the rest is handled by regular refresh()'es 315 * the rest is handled by regular refresh()'es
218 */ 316 */
317void
219void redraw(void) 318redraw (void)
220{ 319{
221 XClearArea(disp, root, win_x, win_y, w, h + font_descent + 2, True); 320 XClearArea (disp, root, win_x, win_y, width, height, False);
321 refresh (0, 32768);
222} 322}
223 323
224/* Just redraw everything without clearing (i.e. after an EXPOSE event) */ 324/* Just redraw everything without clearing (i.e. after an EXPOSE event) */
225void refresh(struct linematrix *lines, int miny, int maxy) 325void
326refresh (int miny, int maxy)
226{ 327{
227 int lin; 328 int lin;
228 int offset = (listlen + 1) * font_height; 329 int offset = (listlen + 1) * font_height;
229 unsigned long black_color = GetColor("black"); 330 unsigned long black_color = GetColor ("black");
230 331
231 miny -= win_y + font_height; 332 miny -= win_y + font_height;
232 maxy -= win_y - font_height; 333 maxy -= win_y - font_height;
233 334
335
234 for (lin = listlen; lin--;) 336 for (lin = listlen; lin--;)
337 {
338 struct linematrix *line = lines + (opt_reverse ? listlen - lin - 1 : lin);
339
340 offset -= font_height;
341
342 if (offset < miny || offset > maxy)
343 continue;
344
345 if (opt_shade)
235 { 346 {
236 char *temp; 347 XSetForeground (disp, WinGC, black_color);
348 XmbDrawString (disp, root, fontset, WinGC, win_x + 2,
349 win_y + offset + 2, line->line, line->len);
350 }
351
352 XSetForeground (disp, WinGC, line->color);
353 XmbDrawString (disp, root, fontset, WinGC, win_x, win_y + offset,
354 line->line, line->len);
355 }
356
357 if (opt_frame)
358 {
359 XSetForeground (disp, WinGC, GetColor (def_color));
360 XDrawRectangle (disp, root, WinGC, win_x - 2, win_y - 2, width + 4, height + 4);
361 }
362}
363
364#if HAS_REGEX
365void
366transform_line (char *s)
367{
368#ifdef I_AM_Md
369 int i;
370 if (1)
371 {
372 for (i = 16; s[i]; i++)
373 s[i] = s[i + 11];
374 }
375 s[i + 1] = '\0';
376#endif
377
378 if (transformre)
379 {
380 int i;
381 regmatch_t matched[16];
382
383 i = regexec (&transformre, string, 16, matched, 0);
384 if (i == 0)
385 { /* matched */
386 }
387 }
388}
389#endif
390
391char *
392concat_line (const char *p1, const char *p2)
393{
394 int l1 = p1 ? strlen (p1) : 0;
395 int l2 = strlen (p2);
396 char *r = xmalloc (l1 + l2 + 1);
397
398 memcpy (r, p1, l1);
399 memcpy (r + l1, p2, l2);
400 r[l1 + l2] = 0;
401
402 return r;
403}
404
405/*
406 * This routine should read a single line, no matter how long.
407 */
408int
409lineinput (struct logfile_entry *logfile)
410{
411 char buff[1024], *p = buff;
412 int ch;
413 int ofs = logfile->buf ? strlen (logfile->buf) : 0;
414
415 do
416 {
417 ch = fgetc (logfile->fp);
418
419 if (ch == '\n' || ch == EOF)
420 break;
421 else if (ch == '\r')
422 continue; /* skip */
423 else if (ch == '\t')
424 {
425 do
426 {
427 *p++ = ' ';
428 ofs++;
429 }
430 while (ofs & 7);
431 }
432 else
433 {
434 *p++ = ch;
435 ofs++;
436 }
437 }
438 while (p < buff + (sizeof buff) - 8 - 1);
439
440 if (p == buff)
441 return 0;
442
443 *p = 0;
444
445 p = concat_line (logfile->buf, buff);
446 free (logfile->buf); logfile->buf = p;
447
448 logfile->lastpartial = logfile->partial;
449 logfile->partial = ch == EOF;
237 450
238 offset -= font_height; 451 if (logfile->partial && opt_whole)
239 452 return 0;
240 if (offset < miny || offset > maxy)
241 continue;
242
243 if (opt_reverse)
244 temp = detabificate (lines[listlen-lin-1].line);
245 else
246 temp = detabificate (lines[lin].line);
247
248 if (opt_shade)
249 {
250 XSetForeground (disp, WinGC, black_color);
251 XDrawString (disp, root, WinGC, win_x + 2, win_y + offset + 2,
252 temp, strlen (temp));
253 }
254
255 XSetForeground (disp, WinGC, lines[lin].color);
256 XDrawString (disp, root, WinGC, win_x, win_y + offset,
257 temp, strlen (temp));
258
259 free (temp);
260 }
261
262 if (opt_frame) {
263 int bot_y = win_y + h + font_descent + 2;
264
265 XDrawLine(disp, root, WinGC, win_x, win_y, win_x + w, win_y);
266 XDrawLine(disp, root, WinGC, win_x + w, win_y, win_x + w, bot_y);
267 XDrawLine(disp, root, WinGC, win_x + w, bot_y, win_x, bot_y);
268 XDrawLine(disp, root, WinGC, win_x, bot_y, win_x, win_y);
269 }
270}
271 453
272#if HAS_REGEX 454#if HAS_REGEX
273void transform_line(char *s) 455 transform_line (logfile->buf);
274{
275#ifdef I_AM_Md
276 int i;
277 if (1) {
278 for (i = 16; s[i]; i++)
279 s[i] = s[i + 11];
280 }
281 s[i + 1] = '\0';
282#endif 456#endif
283
284 if (transformre) {
285 int i;
286 regmatch_t matched[16];
287
288 i = regexec(&transformre, string, 16, matched, 0);
289 if (i == 0) { /* matched */
290 }
291 }
292}
293#endif
294
295
296/*
297 * This routine should read 'width' characters and not more. However,
298 * we really want to read width + 1 charachters if the last char is a '\n',
299 * which we should remove afterwards. So, read width+1 chars and ungetc
300 * the last character if it's not a newline. This means 'string' must be
301 * width + 2 wide!
302 */
303int lineinput(char *string, int slen, FILE *f)
304{
305 int len;
306
307 do {
308 if (fgets(string, slen, f) == NULL) /* EOF or Error */
309 return 0; 457 return 1;
310
311 len = strlen(string);
312 } while (len == 0);
313
314 if (string[len - 1] == '\n')
315 string[len - 1] = '\0'; /* erase newline */
316 else if (len >= slen - 1) {
317 ungetc(string[len - 1], f);
318 string[len - 1] = '\0';
319 }
320
321#if HAS_REGEX
322 transform_line(string);
323#endif
324 return len;
325} 458}
326 459
327/* input: reads file->fname 460/* input: reads file->fname
328 * output: fills file->fp, file->inode 461 * output: fills file->fp, file->inode
329 * returns file->fp 462 * returns file->fp
330 * in case of error, file->fp is NULL 463 * in case of error, file->fp is NULL
331 */ 464 */
465FILE *
332FILE *openlog(struct logfile_entry *file) 466openlog (struct logfile_entry * file)
333{ 467{
334 struct stat stats; 468 struct stat stats;
335 469
336 if ((file->fp = fopen(file->fname, "r")) == NULL) { 470 if ((file->fp = fopen (file->fname, "r")) == NULL)
471 {
337 file->fp = NULL; 472 file->fp = NULL;
338 return NULL; 473 return NULL;
339 } 474 }
340 475
341 fstat(fileno(file->fp), &stats); 476 fstat (fileno (file->fp), &stats);
342 if (S_ISFIFO(stats.st_mode)) { 477 if (S_ISFIFO (stats.st_mode))
478 {
343 if (fcntl(fileno(file->fp), F_SETFL, O_NONBLOCK) < 0) 479 if (fcntl (fileno (file->fp), F_SETFL, O_NONBLOCK) < 0)
344 perror("fcntl"), exit(1); 480 perror ("fcntl"), exit (1);
345 file->inode = 0; 481 file->inode = 0;
482 }
346 } else 483 else
347 file->inode = stats.st_ino; 484 file->inode = stats.st_ino;
348 485
349 if (opt_noinitial) 486 if (opt_noinitial)
350 fseek (file->fp, 0, SEEK_END); 487 fseek (file->fp, 0, SEEK_END);
351 else if (stats.st_size > (listlen + 1) * width) 488 else if (stats.st_size > (listlen + 1) * width)
489 fseek (file->fp, -((listlen + 2) * width), SEEK_END);
490
491 file->last_size = stats.st_size;
492 return file->fp;
493}
494
495void
496reopen (void)
497{
498 struct logfile_entry *e;
499
500 for (e = loglist; e; e = e->next)
501 {
502 if (!e->inode)
503 continue; /* skip stdin */
504
505 if (e->fp)
506 fclose (e->fp);
507 /* if fp is NULL we will try again later */
508 openlog (e);
509 }
510
511 do_reopen = 0;
512}
513
514void
515check_open_files (void)
516{
517 struct logfile_entry *e;
518 struct stat stats;
519
520 for (e = loglist; e; e = e->next)
521 {
522 if (!e->inode)
523 continue; /* skip stdin */
524
525 if (stat (e->fname, &stats) < 0)
526 { /* file missing? */
527 sleep (1);
528 if (e->fp)
529 fclose (e->fp);
530 if (openlog (e) == NULL)
531 break;
532 }
533
534 if (stats.st_ino != e->inode)
535 { /* file renamed? */
536 if (e->fp)
537 fclose (e->fp);
538 if (openlog (e) == NULL)
539 break;
540 }
541
542 if (stats.st_size < e->last_size)
543 { /* file truncated? */
544 fseek (e->fp, 0, SEEK_SET);
545 e->last_size = stats.st_size;
546 }
547 }
548}
549
550/*
551 * insert a single physical line (that must be short enough to fit)
552 * at position "idx" by pushing up lines above it. the caller
553 * MUST then fill in lines[idx] with valid data.
554 */
555static void
556insert_line (int idx)
557{
558 int cur_line;
559 struct logfile_entry *current;
560
561 free (lines[0].line);
562
563 for (cur_line = 0; cur_line < idx; cur_line++)
564 lines[cur_line] = lines[cur_line + 1];
565
566 for (current = loglist; current; current = current->next)
567 if (current->index <= idx)
568 current->index--;
569}
570
571/*
572 * remove a single physical line at position "idx" by moving the lines above it
573 * down and inserting a "~" line at the top.
574 */
575static void
576delete_line (int idx)
577{
578 int cur_line;
579 struct logfile_entry *current;
580
581 for (cur_line = idx; cur_line > 0; cur_line--)
582 lines[cur_line] = lines[cur_line - 1];
583
584 lines[0].line = strdup ("~");
585
586 for (current = loglist; current; current = current->next)
587 if (current->index >= 0 && current->index <= idx)
588 current->index++;
589}
590
591/*
592 * takes a logical log file line and split it into multiple physical
593 * screen lines by splitting it whenever a part becomes too long.
594 * lal lines will be inserted at position "idx".
595 */
596static void
597split_line (int idx, const char *str, unsigned long color)
598{
599 int l = strlen (str);
600 const char *p = str;
601
602 do
603 {
604 const char *beg = p;
605 int w = 0;
606
607 while (*p)
608 {
609 int len = mblen (p, l);
610 if (len <= 0)
611 len = 1; /* ignore (don't skip) ilegal character sequences */
612
613 int cw = XmbTextEscapement (fontset, p, len);
614 if (cw + w >= width)
615 break;
616
617 w += cw;
618 p += len;
619 l -= len;
620 }
621
352 { 622 {
353 char dummy[255]; 623 char *s = xmalloc (p - beg + 1);
354 624 memcpy (s, beg, p - beg);
355 fseek(file->fp, -((listlen + 2) * width), SEEK_END); 625 s[p - beg] = 0;
356 /* the pointer might point halfway some line. Let's 626 insert_line (idx);
357 be nice and skip this damaged line */ 627 lines[idx].line = s;
358 lineinput(dummy, sizeof(dummy), file->fp); 628 lines[idx].len = p - beg;
629 lines[idx].color = color;
359 } 630 }
360
361 file->last_size = stats.st_size;
362 return file->fp;
363}
364
365void reopen(void)
366{
367 struct logfile_entry *e;
368
369 for (e = loglist; e; e = e->next) {
370 if (!e->inode)
371 continue; /* skip stdin */
372
373 if (e->fp)
374 fclose(e->fp);
375 /* if fp is NULL we will try again later */
376 openlog(e);
377 } 631 }
378 632 while (l);
379 do_reopen = 0;
380} 633}
381 634
382void check_open_files(void) 635/*
636 * append something to an existing physical line. this is done
637 * by deleting the file on-screen, concatenating the new data to it
638 * and splitting it again.
639 */
640static void
641append_line (int idx, const char *str)
383{ 642{
384 struct logfile_entry *e; 643 unsigned long color = lines[idx].color;
385 struct stat stats; 644 char *old = lines[idx].line;
645 char *new = concat_line (old, str);
386 646
387 for (e = loglist; e; e = e->next) { 647 free (old);
388 if (!e->inode)
389 continue; /* skip stdin */
390 648
391 if (stat(e->fname, &stats) < 0) { /* file missing? */ 649 delete_line (idx);
392 sleep(1); 650 split_line (idx, new, color);
393 if (e->fp)
394 fclose(e->fp);
395 if (openlog(e) == NULL)
396 break;
397 }
398
399 if (stats.st_ino != e->inode) { /* file renamed? */
400 if (e->fp)
401 fclose(e->fp);
402 if (openlog(e) == NULL)
403 break;
404 }
405
406 if (stats.st_size < e->last_size) { /* file truncated? */
407 fseek(e->fp, 0, SEEK_SET);
408 e->last_size = stats.st_size;
409 }
410 }
411} 651}
412 652
413#define SCROLL_UP(lines, listlen) \ 653static void
414{ \
415 int cur_line; \
416 for (cur_line = 0; cur_line < (listlen - 1); cur_line++) { \
417 strcpy(lines[cur_line].line, lines[cur_line + 1].line); \
418 lines[cur_line].color = lines[cur_line + 1].color; \
419 } \
420}
421
422void main_loop(void) 654main_loop (void)
423{ 655{
424 struct linematrix *lines = xmalloc(sizeof(struct linematrix) * listlen); 656 lines = xmalloc (sizeof (struct linematrix) * listlen);
425 int lin, miny, maxy, buflen; 657 int lin, miny, maxy;
426 char *buf;
427 time_t lastreload; 658 time_t lastreload;
428 Region region = XCreateRegion(); 659 Region region = XCreateRegion ();
429 XEvent xev; 660 XEvent xev;
661 struct logfile_entry *lastprinted = NULL;
662 struct logfile_entry *current;
663 int need_update = 1;
430 664
431 maxy = 0; 665 maxy = 0;
432 miny = win_y + h; 666 miny = win_y + height;
433 buflen = width + 2;
434 buf = xmalloc(buflen);
435 lastreload = time(NULL); 667 lastreload = time (NULL);
436 668
437 /* Initialize linematrix */ 669 /* Initialize linematrix */
438 for (lin = 0; lin < listlen; lin++) { 670 for (lin = 0; lin < listlen; lin++)
439 lines[lin].line = xmalloc(buflen); 671 {
440 strcpy(lines[lin].line, "~"); 672 lines[lin].line = strdup ("~");
673 lines[lin].len = 1;
441 lines[lin].color = GetColor(def_color); 674 lines[lin].color = GetColor (def_color);
442 } 675 }
443 676
444 if (!opt_noinitial)
445 while (lineinput(buf, buflen, loglist->fp) != 0) {
446 SCROLL_UP(lines, listlen);
447 /* print the next line */
448 strcpy(lines[listlen - 1].line, buf);
449 }
450
451 for (;;) { 677 for (;;)
452 int need_update = 0; 678 {
453 struct logfile_entry *current;
454 static struct logfile_entry *lastprinted = NULL;
455
456 /* read logs */ 679 /* read logs */
457 for (current = loglist; current; current = current->next) { 680 for (current = loglist; current; current = current->next)
681 {
458 if (!current->fp) 682 if (!current->fp)
459 continue; /* skip missing files */ 683 continue; /* skip missing files */
460 684
461 clearerr(current->fp); 685 clearerr (current->fp);
462 686
463 while (lineinput(buf, buflen, current->fp) != 0) { 687 while (lineinput (current))
688 {
689 fprintf (stderr, "got line <%s>\n", current->buf);//D
690 need_update = 1;
691 /* if we're trying to update old partial lines in
692 * place, and the last time this file was updated the
693 * output was partial, and that partial line is not
694 * too close to the top of the screen, then update
695 * that partial line */
696 if (opt_update && current->lastpartial && current->index >= 0)
697 {
698 int idx = current->index;
699 append_line (idx, current->buf);
700 current->index = idx;
701 free (current->buf), current->buf = 0;
702 continue;
703 }
704
464 /* print filename if any, and if last line was from 705 /* print filename if any, and if last line was from
465 different file */ 706 * different file */
466 if (!opt_nofilename && 707 if (!opt_nofilename && lastprinted != current && current->desc[0])
467 !(lastprinted && lastprinted == current) && 708 {
468 current->desc[0]) { 709 char buf[1024]; /* HACK */
469 SCROLL_UP(lines, listlen); 710 snprintf (buf, sizeof (buf), "[%s]", current->desc);
470 sprintf(lines[listlen - 1].line, "[%s]", current->desc); 711 split_line (listlen - 1, buf, current->color);
471 lines[listlen - 1].color = current->color; 712 }
472 }
473 713
474 SCROLL_UP(lines, listlen); 714 /* if this is the same file we showed last, and the
475 strcpy(lines[listlen - 1].line, buf); 715 * last time we showed it, it wasn't finished, then
476 lines[listlen - 1].color = current->color; 716 * append to the last line shown */
717 if (lastprinted == current && current->lastpartial)
718 {
719 append_line (listlen - 1, current->buf);
720 free (current->buf), current->buf = 0;
721 continue;
722 }
723 else
724 {
725 split_line (listlen - 1, current->buf, current->color);
726 free (current->buf), current->buf = 0;
727 }
477 728
478 lastprinted = current; 729 current->index = listlen - 1;
479 need_update = 1; 730 lastprinted = current;
480 } 731 }
481 } 732 }
482 733
483 if (need_update) 734 if (need_update)
735 {
484 redraw(); 736 redraw ();
485 else { 737 need_update = 0;
738 }
739 else
740 {
486 XFlush(disp); 741 XFlush (disp);
742
487 if (!XPending(disp)) { 743 if (!XPending (disp))
488 fd_set fdr; 744 {
745 fd_set fdr;
489 struct timeval to = interval; 746 struct timeval to = interval;
490 747
491 FD_ZERO(&fdr); 748 FD_ZERO (&fdr);
492 FD_SET(ConnectionNumber(disp), &fdr); 749 FD_SET (ConnectionNumber (disp), &fdr);
493 select(ConnectionNumber(disp) + 1, &fdr, 0, 0, &to); 750 select (ConnectionNumber (disp) + 1, &fdr, 0, 0, &to);
494 } 751 }
495 } 752 }
496 753
497 check_open_files(); 754 check_open_files ();
498 755
499 if (do_reopen) 756 if (do_reopen)
500 reopen(); 757 reopen ();
501 758
502 /* we ignore possible errors due to window resizing &c */ 759 /* we ignore possible errors due to window resizing &c */
503 while (XPending(disp)) { 760 while (XPending (disp))
761 {
504 XNextEvent(disp, &xev); 762 XNextEvent (disp, &xev);
763
505 switch (xev.type) { 764 switch (xev.type)
765 {
506 case Expose: 766 case Expose:
507 { 767 {
508 XRectangle r; 768 XRectangle r;
509 769
510 r.x = xev.xexpose.x; 770 r.x = xev.xexpose.x;
511 r.y = xev.xexpose.y; 771 r.y = xev.xexpose.y;
512 r.width = xev.xexpose.width; 772 r.width = xev.xexpose.width;
513 r.height = xev.xexpose.height; 773 r.height = xev.xexpose.height;
514 XUnionRectWithRegion(&r, region, region); 774 XUnionRectWithRegion (&r, region, region);
515 if (miny > r.y) 775 if (miny > r.y)
516 miny = r.y; 776 miny = r.y;
517 if (maxy < r.y + r.height) 777 if (maxy < r.y + r.height)
518 maxy = r.y + r.height; 778 maxy = r.y + r.height;
519 } 779 }
520 break; 780 break;
521 default: 781 default:
522#ifdef DEBUGMODE 782#ifdef DEBUGMODE
523 fprintf(stderr, "PANIC! Unknown event %d\n", xev.type); 783 fprintf (stderr, "PANIC! Unknown event %d\n", xev.type);
524#endif 784#endif
525 break; 785 break;
786 }
787 }
788
789 /* reload if requested */
790 if (reload && lastreload + reload < time (NULL))
791 {
792 if (command && command[0])
793 system (command);
794
795 reopen ();
796 lastreload = time (NULL);
797 }
798
799 if (!XEmptyRegion (region))
800 {
801 XSetRegion (disp, WinGC, region);
802
803 refresh (miny, maxy);
804 XDestroyRegion (region);
805 region = XCreateRegion ();
806 maxy = 0;
807 miny = 32768;
808 }
526 } 809 }
527 }
528
529 /* reload if requested */
530 if (reload && lastreload + reload < time(NULL)) {
531 if (command)
532 system(command);
533
534 reopen();
535 lastreload = time(NULL);
536 }
537
538 if (!XEmptyRegion(region)) {
539 XSetRegion(disp, WinGC, region);
540 refresh(lines, miny, maxy);
541 XDestroyRegion(region);
542 region = XCreateRegion();
543 maxy = 0;
544 miny = win_y + h;
545 }
546 }
547} 810}
548 811
549 812
813int
550int main(int argc, char *argv[]) 814main (int argc, char *argv[])
551{ 815{
552 int i; 816 int i;
553 int opt_daemonize = 0; 817 int opt_daemonize = 0;
818 int opt_partial = 0, file_count = 0;
554#if HAS_REGEX 819#if HAS_REGEX
555 char *transform = NULL; 820 char *transform = NULL;
556#endif 821#endif
557 822
823 setlocale (LC_CTYPE, ""); /* try to initialize the locale. */
824
558 /* window needs to be initialized before colorlookups can be done */ 825 /* window needs to be initialized before colorlookups can be done */
559 /* just a dummy to get the color lookups right */ 826 /* just a dummy to get the color lookups right */
560 geom_mask = NoValue; 827 geom_mask = NoValue;
561 InitWindow(); 828 InitWindow ();
562 829
563 for (i = 1; i < argc; i++) { 830 for (i = 1; i < argc; i++)
831 {
832 const char *arg = argv[i];
833
564 if (argv[i][0] == '-' && argv[i][1] != '\0' && argv[i][1] != ',') { 834 if (arg[0] == '-' && arg[1] != '\0' && arg[1] != ',')
835 {
836 if (arg[1] == '-')
837 arg++;
838
565 if (!strcmp(argv[i], "--?") || 839 if (!strcmp (arg, "-?") ||
566 !strcmp(argv[i], "--help") || !strcmp(argv[i], "-h")) 840 !strcmp (arg, "-help") || !strcmp (arg, "-h"))
567 display_help(argv[0]); 841 display_help (argv[0]);
568 else if (!strcmp(argv[i], "-V")) 842 else if (!strcmp (arg, "-V"))
569 display_version(); 843 display_version ();
570 else if (!strcmp(argv[i], "-g") || !strcmp(argv[i], "-geometry")) 844 else if (!strcmp (arg, "-g") || !strcmp (arg, "-geometry"))
571 geom_mask = XParseGeometry(argv[++i], 845 geom_mask =
572 &win_x, &win_y, &width, &listlen); 846 XParseGeometry (argv[++i], &win_x, &win_y, &width, &height);
573 else if (!strcmp(argv[i], "-display")) 847 else if (!strcmp (arg, "-display"))
574 dispname = argv[++i]; 848 dispname = argv[++i];
849 else if (!strcmp (arg, "-cont"))
850 continuation = argv[++i];
575 else if (!strcmp(argv[i], "-font") || !strcmp(argv[i], "-fn")) 851 else if (!strcmp (arg, "-font") || !strcmp (arg, "-fn"))
576 fontname = argv[++i]; 852 fontname = argv[++i];
577#if HAS_REGEX 853#if HAS_REGEX
578 else if (!strcmp(argv[i], "-t")) 854 else if (!strcmp (arg, "-t"))
579 transform = argv[++i]; 855 transform = argv[++i];
580#endif 856#endif
581 else if (!strcmp(argv[i], "-fork") || !strcmp(argv[i], "-f")) 857 else if (!strcmp (arg, "-fork") || !strcmp (arg, "-f"))
582 opt_daemonize = 1; 858 opt_daemonize = 1;
583 else if (!strcmp(argv[i], "-reload")) { 859 else if (!strcmp (arg, "-reload"))
584 reload = atoi(argv[++i]); 860 {
585 command = argv[++i]; 861 reload = atoi (argv[++i]);
586 } 862 command = argv[++i];
863 }
587 else if (!strcmp(argv[i], "-shade")) 864 else if (!strcmp (arg, "-shade"))
588 opt_shade = 1; 865 opt_shade = 1;
589 else if (!strcmp(argv[i], "-frame")) 866 else if (!strcmp (arg, "-frame"))
590 opt_frame = 1; 867 opt_frame = 1;
591 else if (!strcmp(argv[i], "-no-filename")) 868 else if (!strcmp (arg, "-no-filename"))
592 opt_nofilename = 1; 869 opt_nofilename = 1;
593 else if (!strcmp(argv[i], "-reverse")) 870 else if (!strcmp (arg, "-reverse"))
594 opt_reverse = 1; 871 opt_reverse = 1;
872 else if (!strcmp (arg, "-whole"))
873 opt_whole = 1;
874 else if (!strcmp (arg, "-partial"))
875 opt_partial = 1;
876 else if (!strcmp (arg, "-update"))
877 opt_update = opt_partial = 1;
595 else if (!strcmp(argv[i], "-color")) 878 else if (!strcmp (arg, "-color"))
596 def_color = argv[++i]; 879 def_color = argv[++i];
597 else if (!strcmp(argv[i], "-noinitial")) 880 else if (!strcmp (arg, "-noinitial"))
598 opt_noinitial = 1; 881 opt_noinitial = 1;
882 else if (!strcmp (arg, "-id"))
883 root = atoi (argv[++i]);
599 else if (!strcmp(argv[i], "-interval") || !strcmp(argv[i], "-i")) { 884 else if (!strcmp (arg, "-interval") || !strcmp (arg, "-i"))
885 {
600 double iv = atof(argv[++i]); 886 double iv = atof (argv[++i]);
601 887
602 interval.tv_sec = (int) iv; 888 interval.tv_sec = (int) iv;
603 interval.tv_usec = (iv - interval.tv_sec) * 1e6; 889 interval.tv_usec = (iv - interval.tv_sec) * 1e6;
604 } else { 890 }
891 else
892 {
605 fprintf(stderr, "Unknown option '%s'.\n" 893 fprintf (stderr, "Unknown option '%s'.\n"
606 "Try --help for more information.\n", argv[i]); 894 "Try --help for more information.\n", arg);
607 exit(1); 895 exit (1);
608 } 896 }
609 } else { /* it must be a filename */ 897 }
898 else
899 { /* it must be a filename */
610 struct logfile_entry *e; 900 struct logfile_entry *e;
611 const char *fname, *desc, *fcolor = def_color; 901 const char *fname, *desc, *fcolor = def_color;
612 char *p; 902 char *p;
613 903
904 file_count++;
905
614 /* this is not foolproof yet (',' in filenames are not allowed) */ 906 /* this is not foolproof yet (',' in filenames are not allowed) */
615 fname = desc = argv[i]; 907 fname = desc = arg;
616 if ((p = strchr(argv[i], ','))) { 908 if ((p = strchr (arg, ',')))
617 *p = '\0'; 909 {
618 fcolor = p + 1; 910 *p = '\0';
911 fcolor = p + 1;
619 912
620 if ((p = strchr(fcolor, ','))) { 913 if ((p = strchr (fcolor, ',')))
621 *p = '\0'; 914 {
622 desc = p + 1; 915 *p = '\0';
623 } 916 desc = p + 1;
624 } 917 }
918 }
625 919
626 e = xmalloc(sizeof(struct logfile_entry)); 920 e = xmalloc (sizeof (struct logfile_entry));
921 e->partial = 0;
922 e->buf = 0;
923 e->index = -1;
924
627 if (argv[i][0] == '-' && argv[i][1] == '\0') { 925 if (arg[0] == '-' && arg[1] == '\0')
926 {
628 if ((e->fp = fdopen(0, "r")) == NULL) 927 if ((e->fp = fdopen (0, "r")) == NULL)
629 perror("fdopen"), exit(1); 928 perror ("fdopen"), exit (1);
630 if (fcntl(0, F_SETFL, O_NONBLOCK) < 0) 929 if (fcntl (0, F_SETFL, O_NONBLOCK) < 0)
631 perror("fcntl"), exit(1); 930 perror ("fcntl"), exit (1);
632 e->fname = NULL; 931 e->fname = NULL;
633 e->inode = 0; 932 e->inode = 0;
634 e->desc = xstrdup("stdin"); 933 e->desc = xstrdup ("stdin");
635 } else { 934 }
636 int l; 935 else
936 {
937 int l;
637 938
638 e->fname = xstrdup(fname); 939 e->fname = xstrdup (fname);
639 if (openlog(e) == NULL) 940 if (openlog (e) == NULL)
640 perror(fname), exit(1); 941 perror (fname), exit (1);
641 942
642 l = strlen(desc); 943 l = strlen (desc);
643 if (l > width - 2) /* must account for [ ] */ 944 if (l > width - 2) /* must account for [ ] */
644 l = width - 2; 945 l = width - 2;
645 e->desc = xmalloc(l + 1); 946 e->desc = xmalloc (l + 1);
646 memcpy(e->desc, desc, l); 947 memcpy (e->desc, desc, l);
647 *(e->desc + l) = '\0'; 948 *(e->desc + l) = '\0';
949 }
950
951 e->color = GetColor (fcolor);
952 e->partial = 0;
953 e->next = NULL;
954
955 if (!loglist)
956 loglist = e;
957 if (loglist_tail)
958 loglist_tail->next = e;
959 loglist_tail = e;
960 }
648 } 961 }
649 962
650 e->color = GetColor(fcolor);
651 e->next = NULL;
652
653 if (!loglist) 963 if (!loglist)
654 loglist = e;
655 if (loglist_tail)
656 loglist_tail->next = e;
657 loglist_tail = e;
658 }
659 } 964 {
660
661 if (!loglist) {
662 fprintf(stderr, "You did not specify any files to tail\n" 965 fprintf (stderr, "You did not specify any files to tail\n"
663 "use %s --help for help\n", argv[0]); 966 "use %s --help for help\n", argv[0]);
664 exit(1); 967 exit (1);
968 }
969
970 if (opt_partial && opt_whole)
665 } 971 {
972 fprintf (stderr, "Specify at most one of -partial and -whole\n");
973 exit (1);
974 }
975
976 if (opt_partial)
977 /* if we specifically requested to see partial lines then don't insist on whole lines */
978 opt_whole = 0;
979 else if (file_count > 1)
980 /* otherwise, if we've viewing multiple files, default to showing whole lines */
981 opt_whole = 1;
666 982
667#if HAS_REGEX 983#if HAS_REGEX
668 if (transform) { 984 if (transform)
669 int i; 985 {
986 int i;
670 987
671 transformre = xmalloc(sizeof(transformre)); 988 transformre = xmalloc (sizeof (transformre));
672 i = regcomp(&transformre, transform, REG_EXTENDED); 989 i = regcomp (&transformre, transform, REG_EXTENDED);
673 if (i != 0) { 990 if (i != 0)
991 {
674 char buf[512]; 992 char buf[512];
675 993
676 regerror(i, &transformre, buf, sizeof(buf)); 994 regerror (i, &transformre, buf, sizeof (buf));
677 fprintf(stderr, "Cannot compile regular expression: %s\n", buf); 995 fprintf (stderr, "Cannot compile regular expression: %s\n", buf);
678 } 996 }
679 } 997 }
680#endif 998#endif
681 999
682 InitWindow(); 1000 InitWindow ();
683 1001
684 install_signal(SIGINT, blank_window); 1002 install_signal (SIGINT, blank_window);
685 install_signal(SIGQUIT, blank_window); 1003 install_signal (SIGQUIT, blank_window);
686 install_signal(SIGTERM, blank_window); 1004 install_signal (SIGTERM, blank_window);
687 install_signal(SIGHUP, force_reopen); 1005 install_signal (SIGHUP, force_reopen);
688 install_signal(SIGUSR1, list_files); 1006 install_signal (SIGUSR1, list_files);
689 install_signal(SIGUSR2, force_refresh); 1007 install_signal (SIGUSR2, force_refresh);
690 1008
691 if (opt_daemonize) 1009 if (opt_daemonize)
692 daemonize(); 1010 daemonize ();
693 1011
694 main_loop(); 1012 main_loop ();
695 1013
696 exit(1); /* to make gcc -Wall stop complaining */ 1014 exit (1); /* to make gcc -Wall stop complaining */
697} 1015}
698 1016
1017void
699void install_signal(int sig, void (*handler)(int)) 1018install_signal (int sig, void (*handler) (int))
700{ 1019{
701 struct sigaction action; 1020 struct sigaction action;
702 1021
703 action.sa_handler = handler; 1022 action.sa_handler = handler;
704 sigemptyset(&action.sa_mask); 1023 sigemptyset (&action.sa_mask);
705 action.sa_flags = SA_RESTART; 1024 action.sa_flags = SA_RESTART;
1025
706 if (sigaction(sig, &action, NULL) < 0) 1026 if (sigaction (sig, &action, NULL) < 0)
707 fprintf(stderr, "sigaction(%d): %s\n", sig, strerror(errno)), exit(1); 1027 fprintf (stderr, "sigaction(%d): %s\n", sig, strerror (errno)), exit (1);
708} 1028}
709 1029
1030void *
710void *xstrdup(const char *string) 1031xstrdup (const char *string)
711{ 1032{
712 void *p; 1033 void *p;
713 1034
714 while ((p = strdup(string)) == NULL) { 1035 while ((p = strdup (string)) == NULL)
1036 {
715 fprintf(stderr, "Memory exausted."); 1037 fprintf (stderr, "Memory exausted.");
716 sleep(10); 1038 sleep (10);
717 } 1039 }
1040
718 return p; 1041 return p;
719} 1042}
720 1043
1044void *
721void *xmalloc(size_t size) 1045xmalloc (size_t size)
722{ 1046{
723 void *p; 1047 void *p;
724 1048
725 while ((p = malloc(size)) == NULL) { 1049 while ((p = malloc (size)) == NULL)
1050 {
726 fprintf(stderr, "Memory exausted."); 1051 fprintf (stderr, "Memory exausted.");
727 sleep(10); 1052 sleep (10);
728 } 1053 }
1054
729 return p; 1055 return p;
730} 1056}
731 1057
1058void
732void display_help(char *myname) 1059display_help (char *myname)
733{ 1060{
734 printf("Usage: %s [options] file1[,color[,desc]] " 1061 printf ("Usage: %s [options] file1[,color[,desc]] "
735 "[file2[,color[,desc]] ...]\n", myname); 1062 "[file2[,color[,desc]] ...]\n", myname);
736 printf(" -g | -geometry geometry -g WIDTHxHEIGHT+X+Y\n" 1063 printf (" -g | -geometry geometry -g WIDTHxHEIGHT+X+Y\n"
737 " -color color use color $color as default\n" 1064 " -color color use color $color as default\n"
738 " -reload sec command reload after $sec and run command\n" 1065 " -reload sec command reload after $sec and run command\n"
739 " by default -- 3 mins\n" 1066 " -id id window id to use instead of the root window\n"
740 " -font FONTSPEC (-fn) font to use\n" 1067 " -font FONTSPEC (-fn) font to use\n"
741 " -f | -fork fork into background\n" 1068 " -f | -fork fork into background\n"
742 " -reverse print new lines at the top\n" 1069 " -reverse print new lines at the top\n"
1070 " -whole wait for \\n before showing a line\n"
1071 " -partial show lines even if they don't end with a \\n\n"
1072 " -update allow updates to old partial lines\n"
1073 " -cont string to prefix continued partial lines with\n"
1074 " defaults to \"[+]\"\n"
743 " -shade add shading to font\n" 1075 " -shade add shading to font\n"
744 " -noinitial don't display the last file lines on\n" 1076 " -noinitial don't display the last file lines on\n"
745 " startup\n" 1077 " startup\n"
746 " -i | -interval seconds interval between checks (fractional\n" 1078 " -i | -interval seconds interval between checks (fractional\n"
747 " values o.k.). Default 3\n" 1079 " values o.k.). Default 3 seconds\n"
748 " -V display version information and exit\n" 1080 " -V display version information and exit\n"
749 "\n"); 1081 "\n");
750 printf("Example:\n%s -g 80x25+100+50 -font fixed /var/log/messages,green " 1082 printf ("Example:\n%s -g 80x25+100+50 -font fixed /var/log/messages,green "
751 "/var/log/secure,red,'ALERT'\n", myname); 1083 "/var/log/secure,red,'ALERT'\n", myname);
752 exit(0); 1084 exit (0);
753} 1085}
754 1086
1087void
755void display_version(void) { 1088display_version (void)
1089{
756 printf("root-tail version " VERSION "\n"); 1090 printf ("root-tail version " VERSION "\n");
757 exit(0); 1091 exit (0);
758} 1092}
759 1093
1094int
760int daemonize(void) { 1095daemonize (void)
1096{
761 switch (fork()) { 1097 switch (fork ())
1098 {
762 case -1: 1099 case -1:
763 return -1; 1100 return -1;
764 case 0: 1101 case 0:
765 break; 1102 break;
766 default: 1103 default:
767 _exit(0); 1104 _exit (0);
768 } 1105 }
769 1106
770 if (setsid() == -1) 1107 if (setsid () == -1)
771 return -1; 1108 return -1;
772 1109
773 return 0; 1110 return 0;
774} 1111}
775

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines