17 #ifndef HAVE_CLOCK_GETTIME
18 #ifdef HAVE_SYS_TIME_H
19 # include <sys/time.h> /* gettimeofday() */
23 int log_clock_gettime_realtime (struct timespec
*ts
) {
24 #ifdef HAVE_CLOCK_GETTIME
25 return clock_gettime(CLOCK_REALTIME
, ts
);
27 /* Mac OSX does not provide clock_gettime()
28 * e.g. defined(__APPLE__) && defined(__MACH__) */
30 gettimeofday(&tv
, NULL
);
31 ts
->tv_sec
= tv
.tv_sec
;
32 ts
->tv_nsec
= tv
.tv_usec
* 1000;
37 /* retry write on EINTR or when not all data was written */
38 ssize_t
write_all(int fd
, const void* buf
, size_t count
) {
42 ssize_t r
= write(fd
, buf
, count
);
49 /* fail - repeating probably won't help */
53 /* really shouldn't happen... */
57 force_assert(r
<= (ssize_t
) count
);
59 buf
= r
+ (char const*) buf
;
67 /* lowercase: append space, uppercase: don't */
68 static void log_buffer_append_printf(buffer
*out
, const char *fmt
, va_list ap
) {
76 case 's': /* string */
77 s
= va_arg(ap
, char *);
78 buffer_append_string_c_escaped(out
, s
, (NULL
!= s
) ? strlen(s
) : 0);
79 buffer_append_string_len(out
, CONST_STR_LEN(" "));
81 case 'b': /* buffer */
82 b
= va_arg(ap
, buffer
*);
83 buffer_append_string_c_escaped(out
, CONST_BUF_LEN(b
));
84 buffer_append_string_len(out
, CONST_STR_LEN(" "));
88 buffer_append_int(out
, d
);
89 buffer_append_string_len(out
, CONST_STR_LEN(" "));
92 o
= va_arg(ap
, off_t
);
93 buffer_append_int(out
, o
);
94 buffer_append_string_len(out
, CONST_STR_LEN(" "));
96 case 'x': /* int (hex) */
98 buffer_append_string_len(out
, CONST_STR_LEN("0x"));
99 buffer_append_uint_hex(out
, d
);
100 buffer_append_string_len(out
, CONST_STR_LEN(" "));
102 case 'S': /* string */
103 s
= va_arg(ap
, char *);
104 buffer_append_string_c_escaped(out
, s
, (NULL
!= s
) ? strlen(s
) : 0);
106 case 'B': /* buffer */
107 b
= va_arg(ap
, buffer
*);
108 buffer_append_string_c_escaped(out
, CONST_BUF_LEN(b
));
112 buffer_append_int(out
, d
);
114 case 'O': /* off_t */
115 o
= va_arg(ap
, off_t
);
116 buffer_append_int(out
, o
);
118 case 'X': /* int (hex) */
120 buffer_append_string_len(out
, CONST_STR_LEN("0x"));
121 buffer_append_uint_hex(out
, d
);
129 buffer_append_string_len(out
, fmt
, 1);
135 static int log_buffer_prepare(buffer
*b
, server
*srv
, const char *filename
, unsigned int line
) {
136 switch(srv
->errorlog_mode
) {
140 if (-1 == srv
->errorlog_fd
) return -1;
141 /* cache the generated timestamp */
142 if (srv
->cur_ts
!= srv
->last_generated_debug_ts
) {
143 buffer_string_prepare_copy(srv
->ts_debug_str
, 255);
144 buffer_append_strftime(srv
->ts_debug_str
, "%Y-%m-%d %H:%M:%S", localtime(&(srv
->cur_ts
)));
146 srv
->last_generated_debug_ts
= srv
->cur_ts
;
149 buffer_copy_buffer(b
, srv
->ts_debug_str
);
150 buffer_append_string_len(b
, CONST_STR_LEN(": ("));
152 case ERRORLOG_SYSLOG
:
153 /* syslog is generating its own timestamps */
154 buffer_copy_string_len(b
, CONST_STR_LEN("("));
158 buffer_append_string(b
, filename
);
159 buffer_append_string_len(b
, CONST_STR_LEN("."));
160 buffer_append_int(b
, line
);
161 buffer_append_string_len(b
, CONST_STR_LEN(") "));
166 static void log_write(server
*srv
, buffer
*b
) {
167 switch(srv
->errorlog_mode
) {
171 buffer_append_string_len(b
, CONST_STR_LEN("\n"));
172 write_all(srv
->errorlog_fd
, CONST_BUF_LEN(b
));
174 case ERRORLOG_SYSLOG
:
175 syslog(LOG_ERR
, "%s", b
->ptr
);
180 int log_error_write(server
*srv
, const char *filename
, unsigned int line
, const char *fmt
, ...) {
183 if (-1 == log_buffer_prepare(srv
->errorlog_buf
, srv
, filename
, line
)) return 0;
186 log_buffer_append_printf(srv
->errorlog_buf
, fmt
, ap
);
189 log_write(srv
, srv
->errorlog_buf
);
194 int log_error_write_multiline_buffer(server
*srv
, const char *filename
, unsigned int line
, buffer
*multiline
, const char *fmt
, ...) {
197 buffer
*b
= srv
->errorlog_buf
;
198 char *pos
, *end
, *current_line
;
200 if (buffer_string_is_empty(multiline
)) return 0;
202 if (-1 == log_buffer_prepare(b
, srv
, filename
, line
)) return 0;
205 log_buffer_append_printf(b
, fmt
, ap
);
208 prefix_len
= buffer_string_length(b
);
210 current_line
= pos
= multiline
->ptr
;
211 end
= multiline
->ptr
+ buffer_string_length(multiline
);
213 for ( ; pos
<= end
; ++pos
) {
217 case '\0': /* handles end of string */
218 if (current_line
< pos
) {
219 /* truncate to prefix */
220 buffer_string_set_length(b
, prefix_len
);
222 buffer_append_string_len(b
, current_line
, pos
- current_line
);
225 current_line
= pos
+ 1;