2 Copyright (c) 2001-2006, Gerrit Pape
5 Redistribution and use in source and binary forms, with or without
6 modification, are permitted provided that the following conditions are met:
8 1. Redistributions of source code must retain the above copyright notice,
9 this list of conditions and the following disclaimer.
10 2. Redistributions in binary form must reproduce the above copyright
11 notice, this list of conditions and the following disclaimer in the
12 documentation and/or other materials provided with the distribution.
13 3. The name of the author may not be used to endorse or promote products
14 derived from this software without specific prior written permission.
16 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ''AS IS'' AND ANY EXPRESS OR IMPLIED
17 WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 /* Busyboxed by Denys Vlasenko <vda.linux@googlemail.com> */
33 On startup, and after receiving a HUP signal, svlogd checks for each
34 log directory log if the configuration file log/config exists,
35 and if so, reads the file line by line and adjusts configuration
38 If the line is empty, or starts with a #, it is ignored. A line
42 sets the maximum file size of current when svlogd should rotate
43 the current log file to size bytes. Default is 1000000.
44 If size is zero, svlogd doesnt rotate log files
45 You should set size to at least (2 * len).
47 sets the number of old log files svlogd should maintain to num.
48 If svlogd sees more that num old log files in log after log file
49 rotation, it deletes the oldest one. Default is 10.
50 If num is zero, svlogd doesnt remove old log files.
52 sets the minimum number of old log files svlogd should maintain
53 to min. min must be less than num. If min is set, and svlogd
54 cannot write to current because the filesystem is full,
55 and it sees more than min old log files, it deletes the oldest one.
57 sets the maximum age of the current log file when svlogd should
58 rotate the current log file to timeout seconds. If current
59 is timeout seconds old, and is not empty, svlogd forces log file rotation.
61 tells svlogd to feed each recent log file through processor
62 (see above) on log file rotation. By default log files are not processed.
64 tells svlogd to transmit the first len characters of selected
65 log messages to the IP address a.b.c.d, port number port.
66 If port isnt set, the default port for syslog is used (514).
67 len can be set through the -l option, see below. If svlogd
68 has trouble sending udp packets, it writes error messages
69 to the log directory. Attention: logging through udp is unreliable,
70 and should be used in private networks only.
72 is the same as the u line above, but the log messages are no longer
73 written to the log directory, but transmitted through udp only.
74 Error messages from svlogd concerning sending udp packages still go
77 tells svlogd to prefix each line to be written to the log directory,
78 to standard error, or through UDP, with prefix.
80 If a line starts with a -, +, e, or E, svlogd matches the first len characters
81 of each log message against pattern and acts accordingly:
84 the log message is deselected.
86 the log message is selected.
88 the log message is selected to be printed to standard error.
90 the log message is deselected to be printed to standard error.
92 Initially each line is selected to be written to log/current. Deselected
93 log messages are discarded from log. Initially each line is deselected
94 to be written to standard err. Log messages selected for standard error
95 are written to standard error.
99 svlogd matches a log message against the string pattern as follows:
101 pattern is applied to the log message one character by one, starting
102 with the first. A character not a star (*) and not a plus (+) matches itself.
103 A plus matches the next character in pattern in the log message one
104 or more times. A star before the end of pattern matches any string
105 in the log message that does not include the next character in pattern.
106 A star at the end of pattern matches any string.
108 Timestamps optionally added by svlogd are not considered part
111 An svlogd pattern is not a regular expression. For example consider
112 a log message like this
114 2005-12-18_09:13:50.97618 tcpsvd: info: pid 1977 from 10.4.1.14
116 The following pattern doesnt match
120 because the first star matches up to the first p in tcpsvd,
121 and then the match fails because i is not s. To match this
122 log message, you can use a pattern like this instead
126 //config:config SVLOGD
127 //config: bool "svlogd (16 kb)"
130 //config: svlogd continuously reads log data from its standard input, optionally
131 //config: filters log messages, and writes the data to one or more automatically
132 //config: rotated logs.
134 //applet:IF_SVLOGD(APPLET(svlogd, BB_DIR_USR_SBIN, BB_SUID_DROP))
136 //kbuild:lib-$(CONFIG_SVLOGD) += svlogd.o
138 //usage:#define svlogd_trivial_usage
139 //usage: "[-tttv] [-r C] [-R CHARS] [-l MATCHLEN] [-b BUFLEN] DIR..."
140 //usage:#define svlogd_full_usage "\n\n"
141 //usage: "Read log data from stdin and write to rotated log files in DIRs"
143 //usage: "\n"" -r C Replace non-printable characters with C"
144 //usage: "\n"" -R CHARS Also replace CHARS with C (default _)"
145 //usage: "\n"" -t Timestamp with @tai64n"
146 //usage: "\n"" -tt Timestamp with yyyy-mm-dd_hh:mm:ss.sssss"
147 //usage: "\n"" -ttt Timestamp with yyyy-mm-ddThh:mm:ss.sssss"
148 //usage: "\n"" -v Verbose"
150 //usage: "\n""DIR/config file modifies behavior:"
151 //usage: "\n""sSIZE - when to rotate logs (default 1000000, 0 disables)"
152 //usage: "\n""nNUM - number of files to retain"
153 ///////: "\n""NNUM - min number files to retain" - confusing
154 ///////: "\n""tSEC - rotate file if it get SEC seconds old" - confusing
155 //usage: "\n""!PROG - process rotated log with PROG"
156 ///////: "\n""uIPADDR - send log over UDP" - unsupported
157 ///////: "\n""UIPADDR - send log over UDP and DONT log" - unsupported
158 ///////: "\n""pPFX - prefix each line with PFX" - unsupported
159 //usage: "\n""+,-PATTERN - (de)select line for logging"
160 //usage: "\n""E,ePATTERN - (de)select line for stderr"
162 #include <sys/file.h>
164 #include "common_bufsiz.h"
165 #include "runit_lib.h"
167 #define LESS(a,b) ((int)((unsigned)(b) - (unsigned)(a)) > 0)
173 /* pattern list to match, in "aa\0bb\0\cc\0\0" form */
181 unsigned rotate_period
;
187 unsigned next_rotate
;
188 char fnsave
[FMT_PTIME
];
204 unsigned nearest_rotate
;
206 void* (*memRchr
)(const void *, int, size_t);
212 smallint linecomplete
;
220 sigset_t blocked_sigset
;
222 #define G (*ptr_to_globals)
224 #define verbose (G.verbose )
225 #define linemax (G.linemax )
226 #define buflen (G.buflen )
227 #define linelen (G.linelen )
228 #define fndir (G.fndir )
229 #define fdwdir (G.fdwdir )
230 #define wstat (G.wstat )
231 #define memRchr (G.memRchr )
232 #define nearest_rotate (G.nearest_rotate)
233 #define exitasap (G.exitasap )
234 #define rotateasap (G.rotateasap )
235 #define reopenasap (G.reopenasap )
236 #define linecomplete (G.linecomplete )
237 #define tmaxflag (G.tmaxflag )
238 #define repl (G.repl )
239 #define replace (G.replace )
240 #define blocked_sigset (G.blocked_sigset)
241 #define fl_flag_0 (G.fl_flag_0 )
242 #define dirn (G.dirn )
243 #define line bb_common_bufsiz1
244 #define INIT_G() do { \
245 setup_common_bufsiz(); \
246 SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \
254 #define FATAL "fatal: "
255 #define WARNING "warning: "
256 #define PAUSE "pausing: "
257 #define INFO "info: "
259 static void fatalx(const char *m0
)
261 bb_error_msg_and_die(FATAL
"%s", m0
);
263 static void warn(const char *m0
)
265 bb_perror_msg(WARNING
"%s", m0
);
267 static void warn2(const char *m0
, const char *m1
)
269 bb_perror_msg(WARNING
"%s: %s", m0
, m1
);
271 static void warnx(const char *m0
, const char *m1
)
273 bb_error_msg(WARNING
"%s: %s", m0
, m1
);
275 static void pause_nomem(void)
277 bb_simple_error_msg(PAUSE
"out of memory");
280 static void pause1cannot(const char *m0
)
282 bb_perror_msg(PAUSE
"can't %s", m0
);
285 static void pause2cannot(const char *m0
, const char *m1
)
287 bb_perror_msg(PAUSE
"can't %s %s", m0
, m1
);
291 static char* wstrdup(const char *str
)
294 while (!(s
= strdup(str
)))
299 static unsigned pmatch(const char *p
, const char *s
, unsigned len
)
317 if (c
!= *s
) return 0;
328 if (*s != '?') return 0;
336 if (*s
!= c
) return 0;
345 /*** ex fmt_ptime.[ch] ***/
348 static void fmt_time_human_30nul(char *s
, char dt_delim
)
355 ptm
= gmtime_r(&tv
.tv_sec
, &tm
);
356 /* ^^^ using gmtime_r() instead of gmtime() to not use static data */
357 sprintf(s
, "%04u-%02u-%02u%c%02u:%02u:%02u.%06u000",
358 (unsigned)(1900 + ptm
->tm_year
),
359 (unsigned)(ptm
->tm_mon
+ 1),
360 (unsigned)(ptm
->tm_mday
),
362 (unsigned)(ptm
->tm_hour
),
363 (unsigned)(ptm
->tm_min
),
364 (unsigned)(ptm
->tm_sec
),
365 (unsigned)(tv
.tv_usec
)
367 /* 4+1 + 2+1 + 2+1 + 2+1 + 2+1 + 2+1 + 9 = */
368 /* 5 + 3 + 3 + 3 + 3 + 3 + 9 = */
369 /* 20 (up to '.' inclusive) + 9 (not including '\0') */
372 /* NOT terminated! */
373 static void fmt_time_bernstein_25(char *s
)
380 sec_hi
= (0x400000000000000aULL
+ tv
.tv_sec
) >> 32;
381 tv
.tv_sec
= (time_t)(0x400000000000000aULL
) + tv
.tv_sec
;
383 /* Network order is big-endian: most significant byte first.
384 * This is exactly what we want here */
385 pack
[0] = htonl(sec_hi
);
386 pack
[1] = htonl(tv
.tv_sec
);
387 pack
[2] = htonl(tv
.tv_usec
);
389 bin2hex(s
, (char*)pack
, 12);
392 static void processorstart(struct logdir
*ld
)
397 if (!ld
->processor
) return;
399 warnx("processor already running", ld
->name
);
403 /* vfork'ed child trashes this byte, save... */
404 sv_ch
= ld
->fnsave
[26];
407 G
.shell
= xstrdup(get_shell_name());
409 while ((pid
= vfork()) == -1)
410 pause2cannot("vfork for processor", ld
->name
);
415 /* Non-ignored signals revert to SIG_DFL on exec anyway.
416 * But we can get signals BEFORE execl(), this is unlikely
417 * but wouldn't be good...
425 /* runit 2.1.2 does not unblock SIGCHLD, a bug? we do: */
426 sigprocmask(SIG_UNBLOCK
, &blocked_sigset
, NULL
);
429 bb_error_msg(INFO
"processing: %s/%s", ld
->name
, ld
->fnsave
);
431 fd
= open_or_warn(ld
->fnsave
, O_RDONLY
|O_NDELAY
);
432 /* Used to have xopen() above, but it causes infinite restarts of processor
433 * if file is gone - which can happen even because of _us_!
434 * Users report that if on reboot, time is reset to before existing
435 * logfiles creation time, rmoldest() deletes the newest logfile (!)
436 * and we end up here trying to open this now-deleted file.
439 _exit(0); /* fake "success": do not run processor again */
442 ld
->fnsave
[26] = 't'; /* <- that's why we need sv_ch! */
443 fd
= xopen(ld
->fnsave
, O_WRONLY
|O_NDELAY
|O_TRUNC
|O_CREAT
);
445 fd
= open("state", O_RDONLY
|O_NDELAY
);
448 bb_perror_msg_and_die(FATAL
"can't %s processor %s", "open state for", ld
->name
);
449 close(xopen("state", O_WRONLY
|O_NDELAY
|O_TRUNC
|O_CREAT
));
450 fd
= xopen("state", O_RDONLY
|O_NDELAY
);
453 fd
= xopen("newstate", O_WRONLY
|O_NDELAY
|O_TRUNC
|O_CREAT
);
456 execl(G
.shell
, G
.shell
, "-c", ld
->processor
, (char*) NULL
);
457 bb_perror_msg_and_die(FATAL
"can't %s processor %s", "run", ld
->name
);
459 ld
->fnsave
[26] = sv_ch
; /* ...restore */
463 static unsigned processorstop(struct logdir
*ld
)
469 while (safe_waitpid(ld
->ppid
, &wstat
, 0) == -1)
470 pause2cannot("wait for processor", ld
->name
);
476 while (fchdir(ld
->fddir
) == -1)
477 pause2cannot("change directory, want processor", ld
->name
);
478 if (WEXITSTATUS(wstat
) != 0) {
479 warnx("processor failed, restart", ld
->name
);
480 ld
->fnsave
[26] = 't';
482 ld
->fnsave
[26] = 'u';
484 while (fchdir(fdwdir
) == -1)
485 pause1cannot("change to initial working directory");
486 return ld
->processor
? 0 : 1;
488 ld
->fnsave
[26] = 't';
489 memcpy(f
, ld
->fnsave
, 26);
492 while (rename(ld
->fnsave
, f
) == -1)
493 pause2cannot("rename processed", ld
->name
);
494 while (chmod(f
, 0744) == -1)
495 pause2cannot("set mode of processed", ld
->name
);
496 ld
->fnsave
[26] = 'u';
497 if (unlink(ld
->fnsave
) == -1)
498 bb_error_msg(WARNING
"can't unlink: %s/%s", ld
->name
, ld
->fnsave
);
499 while (rename("newstate", "state") == -1)
500 pause2cannot("rename state", ld
->name
);
502 bb_error_msg(INFO
"processed: %s/%s", ld
->name
, f
);
503 while (fchdir(fdwdir
) == -1)
504 pause1cannot("change to initial working directory");
508 static void rmoldest(struct logdir
*ld
)
512 char oldest
[FMT_PTIME
];
515 oldest
[0] = 'A'; oldest
[1] = oldest
[27] = 0;
516 while (!(d
= opendir(".")))
517 pause2cannot("open directory, want rotate", ld
->name
);
519 while ((f
= readdir(d
))) {
520 if ((f
->d_name
[0] == '@') && (strlen(f
->d_name
) == 27)) {
521 if (f
->d_name
[26] == 't') {
522 if (unlink(f
->d_name
) == -1)
523 warn2("can't unlink processor leftover", f
->d_name
);
526 if (strcmp(f
->d_name
, oldest
) < 0)
527 memcpy(oldest
, f
->d_name
, 27);
533 warn2("can't read directory", ld
->name
);
536 if (ld
->nmax
&& (n
> ld
->nmax
)) {
538 bb_error_msg(INFO
"delete: %s/%s", ld
->name
, oldest
);
539 if ((*oldest
== '@') && (unlink(oldest
) == -1))
540 warn2("can't unlink oldest logfile", ld
->name
);
544 static unsigned rotate(struct logdir
*ld
)
549 if (ld
->fddir
== -1) {
550 ld
->rotate_period
= 0;
554 while (!processorstop(ld
))
557 while (fchdir(ld
->fddir
) == -1)
558 pause2cannot("change directory, want rotate", ld
->name
);
560 /* create new filename */
561 ld
->fnsave
[25] = '.';
562 ld
->fnsave
[26] = 's';
564 ld
->fnsave
[26] = 'u';
565 ld
->fnsave
[27] = '\0';
567 fmt_time_bernstein_25(ld
->fnsave
);
569 stat(ld
->fnsave
, &st
);
570 } while (errno
!= ENOENT
);
572 now
= monotonic_sec();
573 if (ld
->rotate_period
&& LESS(ld
->next_rotate
, now
)) {
574 ld
->next_rotate
= now
+ ld
->rotate_period
;
575 if (LESS(ld
->next_rotate
, nearest_rotate
))
576 nearest_rotate
= ld
->next_rotate
;
580 while (fflush(ld
->filecur
) || fsync(ld
->fdcur
) == -1)
581 pause2cannot("fsync current logfile", ld
->name
);
582 while (fchmod(ld
->fdcur
, 0744) == -1)
583 pause2cannot("set mode of current", ld
->name
);
584 ////close(ld->fdcur);
588 bb_error_msg(INFO
"rename: %s/current %s %u", ld
->name
,
589 ld
->fnsave
, ld
->size
);
591 while (rename("current", ld
->fnsave
) == -1)
592 pause2cannot("rename current", ld
->name
);
593 while ((ld
->fdcur
= open("current", O_WRONLY
|O_NDELAY
|O_APPEND
|O_CREAT
, 0600)) == -1)
594 pause2cannot("create new current", ld
->name
);
595 while ((ld
->filecur
= fdopen(ld
->fdcur
, "a")) == NULL
) ////
596 pause2cannot("create new current", ld
->name
); /* very unlikely */
597 setvbuf(ld
->filecur
, NULL
, _IOFBF
, linelen
); ////
598 close_on_exec_on(ld
->fdcur
);
600 while (fchmod(ld
->fdcur
, 0644) == -1)
601 pause2cannot("set mode of current", ld
->name
);
607 while (fchdir(fdwdir
) == -1)
608 pause1cannot("change to initial working directory");
612 static int buffer_pwrite(int n
, char *s
, unsigned len
)
615 struct logdir
*ld
= &dir
[n
];
618 if (ld
->size
>= ld
->sizemax
)
620 if (len
> (ld
->sizemax
- ld
->size
))
621 len
= ld
->sizemax
- ld
->size
;
624 ////i = full_write(ld->fdcur, s, len);
625 ////if (i != -1) break;
626 i
= fwrite(s
, 1, len
, ld
->filecur
);
629 if ((errno
== ENOSPC
) && (ld
->nmin
< ld
->nmax
)) {
632 char oldest
[FMT_PTIME
];
635 while (fchdir(ld
->fddir
) == -1)
636 pause2cannot("change directory, want remove old logfile",
639 oldest
[1] = oldest
[27] = '\0';
640 while (!(d
= opendir(".")))
641 pause2cannot("open directory, want remove old logfile",
644 while ((f
= readdir(d
)))
645 if ((f
->d_name
[0] == '@') && (strlen(f
->d_name
) == 27)) {
647 if (strcmp(f
->d_name
, oldest
) < 0)
648 memcpy(oldest
, f
->d_name
, 27);
650 if (errno
) warn2("can't read directory, want remove old logfile",
655 if (*oldest
== '@') {
656 bb_error_msg(WARNING
"out of disk space, delete: %s/%s",
659 if (unlink(oldest
) == -1) {
660 warn2("can't unlink oldest logfile", ld
->name
);
663 while (fchdir(fdwdir
) == -1)
664 pause1cannot("change to initial working directory");
669 pause2cannot("write to current", ld
->name
);
675 if (ld
->size
>= (ld
->sizemax
- linemax
))
680 static void logdir_close(struct logdir
*ld
)
685 bb_error_msg(INFO
"close: %s", ld
->name
);
689 return; /* impossible */
690 while (fflush(ld
->filecur
) || fsync(ld
->fdcur
) == -1)
691 pause2cannot("fsync current logfile", ld
->name
);
692 while (fchmod(ld
->fdcur
, 0744) == -1)
693 pause2cannot("set mode of current", ld
->name
);
694 ////close(ld->fdcur);
697 if (ld
->fdlock
== -1)
698 return; /* impossible */
702 ld
->processor
= NULL
;
705 static NOINLINE
unsigned logdir_open(struct logdir
*ld
, const char *fn
)
713 now
= monotonic_sec();
715 ld
->fddir
= open(fn
, O_RDONLY
|O_NDELAY
);
716 if (ld
->fddir
== -1) {
717 warn2("can't open log directory", (char*)fn
);
720 close_on_exec_on(ld
->fddir
);
721 if (fchdir(ld
->fddir
) == -1) {
723 warn2("can't change directory", (char*)fn
);
726 ld
->fdlock
= open("lock", O_WRONLY
|O_NDELAY
|O_APPEND
|O_CREAT
, 0600);
727 if ((ld
->fdlock
== -1)
728 || (flock(ld
->fdlock
, LOCK_EX
| LOCK_NB
) == -1)
731 warn2("can't lock directory", (char*)fn
);
732 while (fchdir(fdwdir
) == -1)
733 pause1cannot("change to initial working directory");
736 close_on_exec_on(ld
->fdlock
);
739 ld
->sizemax
= 1000000;
740 ld
->nmax
= ld
->nmin
= 10;
741 ld
->rotate_period
= 0;
742 ld
->name
= (char*)fn
;
745 free(ld
->inst
); ld
->inst
= NULL
;
746 free(ld
->processor
); ld
->processor
= NULL
;
749 i
= open_read_close("config", buf
, sizeof(buf
) - 1);
750 if (i
< 0 && errno
!= ENOENT
)
751 bb_perror_msg(WARNING
"%s/config", ld
->name
);
755 bb_error_msg(INFO
"read: %s/config", ld
->name
);
758 np
= strchr(s
, '\n');
766 /* Filtering requires one-line buffering,
767 * resetting the "find newline" function
770 /* Add '\n'-terminated line to ld->inst */
772 int l
= asprintf(&new, "%s%s\n", ld
->inst
? ld
->inst
: "", s
);
781 ld
->sizemax
= xatou_sfx(&s
[1], km_suffixes
);
785 ld
->nmax
= xatoi_positive(&s
[1]);
788 ld
->nmin
= xatoi_positive(&s
[1]);
791 static const struct suffix_mult mh_suffixes
[] ALIGN_SUFFIX
= {
794 /*{ "d", 24*60*60 },*/
797 ld
->rotate_period
= xatou_sfx(&s
[1], mh_suffixes
);
798 if (ld
->rotate_period
) {
799 ld
->next_rotate
= now
+ ld
->rotate_period
;
800 if (!tmaxflag
|| LESS(ld
->next_rotate
, nearest_rotate
))
801 nearest_rotate
= ld
->next_rotate
;
809 ld
->processor
= wstrdup(&s
[1]);
815 /* Convert "aa\nbb\ncc\n\0" to "aa\0bb\0cc\0\0" */
818 np
= strchr(s
, '\n');
826 i
= stat("current", &st
);
828 if (st
.st_size
&& !(st
.st_mode
& S_IXUSR
)) {
829 ld
->fnsave
[25] = '.';
830 ld
->fnsave
[26] = 'u';
831 ld
->fnsave
[27] = '\0';
833 fmt_time_bernstein_25(ld
->fnsave
);
835 stat(ld
->fnsave
, &st
);
836 } while (errno
!= ENOENT
);
837 while (rename("current", ld
->fnsave
) == -1)
838 pause2cannot("rename current", ld
->name
);
842 /* st.st_size can be not just bigger, but WIDER!
843 * This code is safe: if st.st_size > 4GB, we select
844 * ld->sizemax (because it's "unsigned") */
845 ld
->size
= (st
.st_size
> ld
->sizemax
) ? ld
->sizemax
: st
.st_size
;
848 if (errno
!= ENOENT
) {
850 warn2("can't stat current", ld
->name
);
851 while (fchdir(fdwdir
) == -1)
852 pause1cannot("change to initial working directory");
856 while ((ld
->fdcur
= open("current", O_WRONLY
|O_NDELAY
|O_APPEND
|O_CREAT
, 0600)) == -1)
857 pause2cannot("open current", ld
->name
);
858 while ((ld
->filecur
= fdopen(ld
->fdcur
, "a")) == NULL
)
859 pause2cannot("open current", ld
->name
); ////
860 setvbuf(ld
->filecur
, NULL
, _IOFBF
, linelen
); ////
862 close_on_exec_on(ld
->fdcur
);
863 while (fchmod(ld
->fdcur
, 0644) == -1)
864 pause2cannot("set mode of current", ld
->name
);
867 if (i
== 0) bb_error_msg(INFO
"append: %s/current", ld
->name
);
868 else bb_error_msg(INFO
"new: %s/current", ld
->name
);
871 while (fchdir(fdwdir
) == -1)
872 pause1cannot("change to initial working directory");
876 static void logdirs_reopen(void)
882 for (l
= 0; l
< dirn
; ++l
) {
883 logdir_close(&dir
[l
]);
884 if (logdir_open(&dir
[l
], fndir
[l
]))
888 fatalx("no functional log directories");
891 /* Will look good in libbb one day */
892 static ssize_t
ndelay_read(int fd
, void *buf
, size_t count
)
894 if (!(fl_flag_0
& O_NONBLOCK
))
895 fcntl(fd
, F_SETFL
, fl_flag_0
| O_NONBLOCK
);
896 count
= safe_read(fd
, buf
, count
);
897 if (!(fl_flag_0
& O_NONBLOCK
))
898 fcntl(fd
, F_SETFL
, fl_flag_0
);
902 /* Used for reading stdin */
903 static int buffer_pread(/*int fd, */char *s
, unsigned len
)
909 input
.fd
= STDIN_FILENO
;
910 input
.events
= POLLIN
;
914 for (i
= 0; i
< dirn
; ++i
)
927 now
= monotonic_sec();
928 nearest_rotate
= now
+ (45 * 60 + 45);
929 for (i
= 0; i
< dirn
; ++i
) {
930 if (dir
[i
].rotate_period
) {
931 if (LESS(dir
[i
].next_rotate
, now
))
933 if (LESS(dir
[i
].next_rotate
, nearest_rotate
))
934 nearest_rotate
= dir
[i
].next_rotate
;
938 sigprocmask(SIG_UNBLOCK
, &blocked_sigset
, NULL
);
939 i
= nearest_rotate
- now
;
944 poll(&input
, 1, i
* 1000);
945 sigprocmask(SIG_BLOCK
, &blocked_sigset
, NULL
);
947 i
= ndelay_read(STDIN_FILENO
, s
, len
);
952 if (errno
!= EAGAIN
) {
953 warn("can't read standard input");
956 /* else: EAGAIN - normal, repeat silently */
961 linecomplete
= (s
[i
-1] == '\n');
969 if (ch
< 32 || ch
> 126)
973 for (j
= 0; replace
[j
]; ++j
) {
974 if (ch
== replace
[j
]) {
987 static void sig_term_handler(int sig_no UNUSED_PARAM
)
990 bb_error_msg(INFO
"sig%s received", "term");
994 static void sig_child_handler(int sig_no UNUSED_PARAM
)
1000 bb_error_msg(INFO
"sig%s received", "child");
1001 while ((pid
= wait_any_nohang(&wstat
)) > 0) {
1002 for (l
= 0; l
< dirn
; ++l
) {
1003 if (dir
[l
].ppid
== pid
) {
1005 processorstop(&dir
[l
]);
1012 static void sig_alarm_handler(int sig_no UNUSED_PARAM
)
1015 bb_error_msg(INFO
"sig%s received", "alarm");
1019 static void sig_hangup_handler(int sig_no UNUSED_PARAM
)
1022 bb_error_msg(INFO
"sig%s received", "hangup");
1026 static void logmatch(struct logdir
*ld
, char* lineptr
, int lineptr_len
)
1037 if (pmatch(s
+1, lineptr
, lineptr_len
))
1042 if (pmatch(s
+1, lineptr
, lineptr_len
))
1043 ld
->matcherr
= s
[0];
1050 int svlogd_main(int argc
, char **argv
) MAIN_EXTERNALLY_VISIBLE
;
1051 int svlogd_main(int argc
, char **argv
)
1054 ssize_t stdin_cnt
= 0;
1057 unsigned timestamp
= 0;
1061 opt
= getopt32(argv
, "^"
1062 "r:R:l:b:tv" "\0" "tt:vv",
1063 &r
, &replace
, &l
, &b
, ×tamp
, &verbose
1065 if (opt
& 1) { // -r
1070 if (opt
& 2) if (!repl
) repl
= '_'; // -R
1071 if (opt
& 4) { // -l
1072 linemax
= xatou_range(l
, 0, COMMON_BUFSIZE
-26);
1074 linemax
= COMMON_BUFSIZE
-26;
1078 ////if (opt & 8) { // -b
1079 //// buflen = xatoi_positive(b);
1080 //// if (buflen == 0) buflen = 1024;
1082 //if (opt & 0x10) timestamp++; // -t
1083 //if (opt & 0x20) verbose++; // -v
1084 //if (timestamp > 2) timestamp = 2;
1091 ////if (buflen <= linemax) bb_show_usage();
1092 fdwdir
= xopen(".", O_RDONLY
|O_NDELAY
);
1093 close_on_exec_on(fdwdir
);
1094 dir
= xzalloc(dirn
* sizeof(dir
[0]));
1095 for (i
= 0; i
< dirn
; ++i
) {
1098 ////dir[i].btmp = xmalloc(buflen);
1099 /*dir[i].ppid = 0;*/
1101 /* line = xmalloc(linemax + (timestamp ? 26 : 0)); */
1103 /* We cannot set NONBLOCK on fd #0 permanently - this setting
1104 * _isn't_ per-process! It is shared among all other processes
1105 * with the same stdin */
1106 fl_flag_0
= fcntl(0, F_GETFL
);
1108 sigemptyset(&blocked_sigset
);
1109 sigaddset(&blocked_sigset
, SIGTERM
);
1110 sigaddset(&blocked_sigset
, SIGCHLD
);
1111 sigaddset(&blocked_sigset
, SIGALRM
);
1112 sigaddset(&blocked_sigset
, SIGHUP
);
1113 sigprocmask(SIG_BLOCK
, &blocked_sigset
, NULL
);
1114 bb_signals_norestart(1 << SIGTERM
, sig_term_handler
);
1115 bb_signals_norestart(1 << SIGCHLD
, sig_child_handler
);
1116 bb_signals_norestart(1 << SIGALRM
, sig_alarm_handler
);
1117 bb_signals_norestart(1 << SIGHUP
, sig_hangup_handler
);
1119 /* Without timestamps, we don't have to print each line
1120 * separately, so we can look for _last_ newline, not first,
1121 * thus batching writes. If filtering is enabled in config,
1122 * logdirs_reopen resets it to memchr.
1124 memRchr
= (timestamp
? memchr
: memrchr
);
1128 setvbuf(stderr
, NULL
, _IOFBF
, linelen
);
1130 /* Each iteration processes one or more lines */
1132 char stamp
[FMT_PTIME
];
1143 /* lineptr[0..linemax-1] - buffer for stdin */
1144 /* (possibly has some unprocessed data from prev loop) */
1146 /* Refill the buffer if needed */
1147 np
= memRchr(lineptr
, '\n', stdin_cnt
);
1148 if (!np
&& !exitasap
) {
1149 i
= linemax
- stdin_cnt
; /* avail. bytes at tail */
1151 i
= buffer_pread(/*0, */lineptr
+ stdin_cnt
, i
);
1152 if (i
<= 0) /* EOF or error on stdin */
1155 np
= memRchr(lineptr
+ stdin_cnt
, '\n', i
);
1160 if (stdin_cnt
<= 0 && exitasap
)
1163 /* Search for '\n' (in fact, np already holds the result) */
1164 linelen
= stdin_cnt
;
1167 /* NB: starting from here lineptr may point
1168 * farther out into line[] */
1169 linelen
= np
- lineptr
+ 1;
1171 /* linelen == no of chars incl. '\n' (or == stdin_cnt) */
1172 ch
= lineptr
[linelen
-1];
1174 /* Biggest performance hit was coming from the fact
1175 * that we did not buffer writes. We were reading many lines
1176 * in one read() above, but wrote one line per write().
1177 * We are using stdio to fix that */
1179 /* write out lineptr[0..linelen-1] to each log destination
1180 * (or lineptr[-26..linelen-1] if timestamping) */
1185 fmt_time_bernstein_25(stamp
);
1187 fmt_time_human_30nul(stamp
, timestamp
== 2 ? '_' : 'T');
1190 memcpy(printptr
, stamp
, 25);
1193 for (i
= 0; i
< dirn
; ++i
) {
1194 struct logdir
*ld
= &dir
[i
];
1195 if (ld
->fddir
== -1)
1198 logmatch(ld
, lineptr
, linelen
);
1199 if (ld
->matcherr
== 'e') {
1200 /* runit-1.8.0 compat: if timestamping, do it on stderr too */
1201 ////full_write(STDERR_FILENO, printptr, printlen);
1202 fwrite(printptr
, 1, printlen
, stderr
);
1204 if (ld
->match
!= '+')
1206 buffer_pwrite(i
, printptr
, printlen
);
1209 /* If we didn't see '\n' (long input line), */
1210 /* read/write repeatedly until we see it */
1211 while (ch
!= '\n') {
1212 /* lineptr is emptied now, safe to use as buffer */
1213 stdin_cnt
= exitasap
? -1 : buffer_pread(/*0, */lineptr
, linemax
);
1214 if (stdin_cnt
<= 0) { /* EOF or error on stdin */
1216 lineptr
[0] = ch
= '\n';
1220 linelen
= stdin_cnt
;
1221 np
= memRchr(lineptr
, '\n', stdin_cnt
);
1223 linelen
= np
- lineptr
+ 1;
1224 ch
= lineptr
[linelen
-1];
1226 /* linelen == no of chars incl. '\n' (or == stdin_cnt) */
1227 for (i
= 0; i
< dirn
; ++i
) {
1228 if (dir
[i
].fddir
== -1)
1230 if (dir
[i
].matcherr
== 'e') {
1231 ////full_write(STDERR_FILENO, lineptr, linelen);
1232 fwrite(lineptr
, 1, linelen
, stderr
);
1234 if (dir
[i
].match
!= '+')
1236 buffer_pwrite(i
, lineptr
, linelen
);
1240 stdin_cnt
-= linelen
;
1241 if (stdin_cnt
> 0) {
1243 /* If we see another '\n', we don't need to read
1244 * next piece of input: can print what we have */
1245 np
= memRchr(lineptr
, '\n', stdin_cnt
);
1248 /* Move unprocessed data to the front of line */
1249 memmove((timestamp
? line
+26 : line
), lineptr
, stdin_cnt
);
1254 for (i
= 0; i
< dirn
; ++i
) {
1256 while (!processorstop(&dir
[i
]))
1258 logdir_close(&dir
[i
]);