1 /*
2  * Debugging functions for CUPS.
3  *
4  * Copyright 2008-2015 by Apple Inc.
5  *
6  * These coded instructions, statements, and computer programs are the
7  * property of Apple Inc. and are protected by Federal copyright
8  * law.  Distribution and use rights are outlined in the file "LICENSE.txt"
9  * which should have been included with this file.  If this file is
10  * missing or damaged, see the license at "http://www.cups.org/".
11  *
12  * This file is subject to the Apple OS-Developed Software exception.
13  */
14 
15 /*
16  * Include necessary headers...
17  */
18 
19 #include "cups-private.h"
20 #include "thread-private.h"
21 #ifdef WIN32
22 #  include <sys/timeb.h>
23 #  include <time.h>
24 #  include <io.h>
25 #  define getpid (int)GetCurrentProcessId
26 int					/* O  - 0 on success, -1 on failure */
_cups_gettimeofday(struct timeval * tv,void * tz)27 _cups_gettimeofday(struct timeval *tv,	/* I  - Timeval struct */
28                    void		  *tz)	/* I  - Timezone */
29 {
30   struct _timeb timebuffer;		/* Time buffer struct */
31   _ftime(&timebuffer);
32   tv->tv_sec  = (long)timebuffer.time;
33   tv->tv_usec = timebuffer.millitm * 1000;
34   return 0;
35 }
36 #else
37 #  include <sys/time.h>
38 #  include <unistd.h>
39 #endif /* WIN32 */
40 #include <regex.h>
41 #include <fcntl.h>
42 
43 
44 /*
45  * Globals...
46  */
47 
48 int			_cups_debug_fd = -1;
49 					/* Debug log file descriptor */
50 int			_cups_debug_level = 1;
51 					/* Log level (0 to 9) */
52 
53 
54 #ifdef DEBUG
55 /*
56  * Local globals...
57  */
58 
59 static regex_t		*debug_filter = NULL;
60 					/* Filter expression for messages */
61 static int		debug_init = 0;	/* Did we initialize debugging? */
62 static _cups_mutex_t	debug_init_mutex = _CUPS_MUTEX_INITIALIZER,
63 					/* Mutex to control initialization */
64 			debug_log_mutex = _CUPS_MUTEX_INITIALIZER;
65 					/* Mutex to serialize log entries */
66 
67 
68 /*
69  * 'debug_thread_id()' - Return an integer representing the current thread.
70  */
71 
72 static int				/* O - Local thread ID */
debug_thread_id(void)73 debug_thread_id(void)
74 {
75   _cups_globals_t *cg = _cupsGlobals();	/* Global data */
76 
77 
78   return (cg->thread_id);
79 }
80 
81 
82 /*
83  * '_cups_debug_printf()' - Write a formatted line to the log.
84  */
85 
86 void DLLExport
_cups_debug_printf(const char * format,...)87 _cups_debug_printf(const char *format,	/* I - Printf-style format string */
88                    ...)			/* I - Additional arguments as needed */
89 {
90   va_list		ap;		/* Pointer to arguments */
91   struct timeval	curtime;	/* Current time */
92   char			buffer[2048];	/* Output buffer */
93   ssize_t		bytes;		/* Number of bytes in buffer */
94   int			level;		/* Log level in message */
95 
96 
97  /*
98   * See if we need to do any logging...
99   */
100 
101   if (!debug_init)
102     _cups_debug_set(getenv("CUPS_DEBUG_LOG"), getenv("CUPS_DEBUG_LEVEL"),
103                     getenv("CUPS_DEBUG_FILTER"), 0);
104 
105   if (_cups_debug_fd < 0)
106     return;
107 
108  /*
109   * Filter as needed...
110   */
111 
112   if (isdigit(format[0]))
113     level = *format++ - '0';
114   else
115     level = 0;
116 
117   if (level > _cups_debug_level)
118     return;
119 
120   if (debug_filter)
121   {
122     int	result;				/* Filter result */
123 
124     _cupsMutexLock(&debug_init_mutex);
125     result = regexec(debug_filter, format, 0, NULL, 0);
126     _cupsMutexUnlock(&debug_init_mutex);
127 
128     if (result)
129       return;
130   }
131 
132  /*
133   * Format the message...
134   */
135 
136   gettimeofday(&curtime, NULL);
137   snprintf(buffer, sizeof(buffer), "T%03d %02d:%02d:%02d.%03d  ",
138            debug_thread_id(), (int)((curtime.tv_sec / 3600) % 24),
139 	   (int)((curtime.tv_sec / 60) % 60),
140 	   (int)(curtime.tv_sec % 60), (int)(curtime.tv_usec / 1000));
141 
142   va_start(ap, format);
143   bytes = _cups_safe_vsnprintf(buffer + 19, sizeof(buffer) - 20, format, ap) + 19;
144   va_end(ap);
145 
146   if ((size_t)bytes >= (sizeof(buffer) - 1))
147   {
148     buffer[sizeof(buffer) - 2] = '\n';
149     bytes = sizeof(buffer) - 1;
150   }
151   else if (buffer[bytes - 1] != '\n')
152   {
153     buffer[bytes++] = '\n';
154     buffer[bytes]   = '\0';
155   }
156 
157  /*
158   * Write it out...
159   */
160 
161   _cupsMutexLock(&debug_log_mutex);
162   write(_cups_debug_fd, buffer, (size_t)bytes);
163   _cupsMutexUnlock(&debug_log_mutex);
164 }
165 
166 
167 /*
168  * '_cups_debug_puts()' - Write a single line to the log.
169  */
170 
171 void DLLExport
_cups_debug_puts(const char * s)172 _cups_debug_puts(const char *s)		/* I - String to output */
173 {
174   struct timeval	curtime;	/* Current time */
175   char			buffer[2048];	/* Output buffer */
176   ssize_t		bytes;		/* Number of bytes in buffer */
177   int			level;		/* Log level in message */
178 
179 
180  /*
181   * See if we need to do any logging...
182   */
183 
184   if (!debug_init)
185     _cups_debug_set(getenv("CUPS_DEBUG_LOG"), getenv("CUPS_DEBUG_LEVEL"),
186                     getenv("CUPS_DEBUG_FILTER"), 0);
187 
188   if (_cups_debug_fd < 0)
189     return;
190 
191  /*
192   * Filter as needed...
193   */
194 
195   if (isdigit(s[0]))
196     level = *s++ - '0';
197   else
198     level = 0;
199 
200   if (level > _cups_debug_level)
201     return;
202 
203   if (debug_filter)
204   {
205     int	result;				/* Filter result */
206 
207     _cupsMutexLock(&debug_init_mutex);
208     result = regexec(debug_filter, s, 0, NULL, 0);
209     _cupsMutexUnlock(&debug_init_mutex);
210 
211     if (result)
212       return;
213   }
214 
215  /*
216   * Format the message...
217   */
218 
219   gettimeofday(&curtime, NULL);
220   bytes = snprintf(buffer, sizeof(buffer), "T%03d %02d:%02d:%02d.%03d  %s",
221                    debug_thread_id(), (int)((curtime.tv_sec / 3600) % 24),
222 		   (int)((curtime.tv_sec / 60) % 60),
223 		   (int)(curtime.tv_sec % 60), (int)(curtime.tv_usec / 1000),
224 		   s);
225 
226   if ((size_t)bytes >= (sizeof(buffer) - 1))
227   {
228     buffer[sizeof(buffer) - 2] = '\n';
229     bytes = sizeof(buffer) - 1;
230   }
231   else if (buffer[bytes - 1] != '\n')
232   {
233     buffer[bytes++] = '\n';
234     buffer[bytes]   = '\0';
235   }
236 
237  /*
238   * Write it out...
239   */
240 
241   _cupsMutexLock(&debug_log_mutex);
242   write(_cups_debug_fd, buffer, (size_t)bytes);
243   _cupsMutexUnlock(&debug_log_mutex);
244 }
245 
246 
247 /*
248  * '_cups_debug_set()' - Enable or disable debug logging.
249  */
250 
251 void DLLExport
_cups_debug_set(const char * logfile,const char * level,const char * filter,int force)252 _cups_debug_set(const char *logfile,	/* I - Log file or NULL */
253                 const char *level,	/* I - Log level or NULL */
254 		const char *filter,	/* I - Filter string or NULL */
255 		int        force)	/* I - Force initialization */
256 {
257   _cupsMutexLock(&debug_init_mutex);
258 
259   if (!debug_init || force)
260   {
261    /*
262     * Restore debug settings to defaults...
263     */
264 
265     if (_cups_debug_fd != -1)
266     {
267       close(_cups_debug_fd);
268       _cups_debug_fd = -1;
269     }
270 
271     if (debug_filter)
272     {
273       regfree((regex_t *)debug_filter);
274       debug_filter = NULL;
275     }
276 
277     _cups_debug_level = 1;
278 
279    /*
280     * Open logs, set log levels, etc.
281     */
282 
283     if (!logfile)
284       _cups_debug_fd = -1;
285     else if (!strcmp(logfile, "-"))
286       _cups_debug_fd = 2;
287     else
288     {
289       char	buffer[1024];		/* Filename buffer */
290 
291       snprintf(buffer, sizeof(buffer), logfile, getpid());
292 
293       if (buffer[0] == '+')
294 	_cups_debug_fd = open(buffer + 1, O_WRONLY | O_APPEND | O_CREAT, 0644);
295       else
296 	_cups_debug_fd = open(buffer, O_WRONLY | O_TRUNC | O_CREAT, 0644);
297     }
298 
299     if (level)
300       _cups_debug_level = atoi(level);
301 
302     if (filter)
303     {
304       if ((debug_filter = (regex_t *)calloc(1, sizeof(regex_t))) == NULL)
305 	fputs("Unable to allocate memory for CUPS_DEBUG_FILTER - results not "
306 	      "filtered!\n", stderr);
307       else if (regcomp(debug_filter, filter, REG_EXTENDED))
308       {
309 	fputs("Bad regular expression in CUPS_DEBUG_FILTER - results not "
310 	      "filtered!\n", stderr);
311 	free(debug_filter);
312 	debug_filter = NULL;
313       }
314     }
315 
316     debug_init = 1;
317   }
318 
319   _cupsMutexUnlock(&debug_init_mutex);
320 }
321 #endif /* DEBUG */
322 
323 
324 /*
325  * '_cups_safe_vsnprintf()' - Format a string into a fixed size buffer,
326  *                            quoting special characters.
327  */
328 
329 ssize_t					/* O - Number of bytes formatted */
_cups_safe_vsnprintf(char * buffer,size_t bufsize,const char * format,va_list ap)330 _cups_safe_vsnprintf(
331     char       *buffer,			/* O - Output buffer */
332     size_t     bufsize,			/* O - Size of output buffer */
333     const char *format,			/* I - printf-style format string */
334     va_list    ap)			/* I - Pointer to additional arguments */
335 {
336   char		*bufptr,		/* Pointer to position in buffer */
337 		*bufend,		/* Pointer to end of buffer */
338 		size,			/* Size character (h, l, L) */
339 		type;			/* Format type character */
340   int		width,			/* Width of field */
341 		prec;			/* Number of characters of precision */
342   char		tformat[100],		/* Temporary format string for snprintf() */
343 		*tptr,			/* Pointer into temporary format */
344 		temp[1024];		/* Buffer for formatted numbers */
345   char		*s;			/* Pointer to string */
346   ssize_t	bytes;			/* Total number of bytes needed */
347 
348 
349   if (!buffer || bufsize < 2 || !format)
350     return (-1);
351 
352  /*
353   * Loop through the format string, formatting as needed...
354   */
355 
356   bufptr = buffer;
357   bufend = buffer + bufsize - 1;
358   bytes  = 0;
359 
360   while (*format)
361   {
362     if (*format == '%')
363     {
364       tptr = tformat;
365       *tptr++ = *format++;
366 
367       if (*format == '%')
368       {
369         if (bufptr < bufend)
370 	  *bufptr++ = *format;
371         bytes ++;
372         format ++;
373 	continue;
374       }
375       else if (strchr(" -+#\'", *format))
376         *tptr++ = *format++;
377 
378       if (*format == '*')
379       {
380        /*
381         * Get width from argument...
382 	*/
383 
384 	format ++;
385 	width = va_arg(ap, int);
386 
387 	snprintf(tptr, sizeof(tformat) - (size_t)(tptr - tformat), "%d", width);
388 	tptr += strlen(tptr);
389       }
390       else
391       {
392 	width = 0;
393 
394 	while (isdigit(*format & 255))
395 	{
396 	  if (tptr < (tformat + sizeof(tformat) - 1))
397 	    *tptr++ = *format;
398 
399 	  width = width * 10 + *format++ - '0';
400 	}
401       }
402 
403       if (*format == '.')
404       {
405 	if (tptr < (tformat + sizeof(tformat) - 1))
406 	  *tptr++ = *format;
407 
408         format ++;
409 
410         if (*format == '*')
411 	{
412          /*
413 	  * Get precision from argument...
414 	  */
415 
416 	  format ++;
417 	  prec = va_arg(ap, int);
418 
419 	  snprintf(tptr, sizeof(tformat) - (size_t)(tptr - tformat), "%d", prec);
420 	  tptr += strlen(tptr);
421 	}
422 	else
423 	{
424 	  prec = 0;
425 
426 	  while (isdigit(*format & 255))
427 	  {
428 	    if (tptr < (tformat + sizeof(tformat) - 1))
429 	      *tptr++ = *format;
430 
431 	    prec = prec * 10 + *format++ - '0';
432 	  }
433 	}
434       }
435 
436       if (*format == 'l' && format[1] == 'l')
437       {
438         size = 'L';
439 
440 	if (tptr < (tformat + sizeof(tformat) - 2))
441 	{
442 	  *tptr++ = 'l';
443 	  *tptr++ = 'l';
444 	}
445 
446 	format += 2;
447       }
448       else if (*format == 'h' || *format == 'l' || *format == 'L')
449       {
450 	if (tptr < (tformat + sizeof(tformat) - 1))
451 	  *tptr++ = *format;
452 
453         size = *format++;
454       }
455       else
456         size = 0;
457 
458       if (!*format)
459         break;
460 
461       if (tptr < (tformat + sizeof(tformat) - 1))
462         *tptr++ = *format;
463 
464       type  = *format++;
465       *tptr = '\0';
466 
467       switch (type)
468       {
469 	case 'E' : /* Floating point formats */
470 	case 'G' :
471 	case 'e' :
472 	case 'f' :
473 	case 'g' :
474 	    if ((size_t)(width + 2) > sizeof(temp))
475 	      break;
476 
477 	    snprintf(temp, sizeof(temp), tformat, va_arg(ap, double));
478 
479             bytes += (int)strlen(temp);
480 
481             if (bufptr)
482 	    {
483 	      strlcpy(bufptr, temp, (size_t)(bufend - bufptr));
484 	      bufptr += strlen(bufptr);
485 	    }
486 	    break;
487 
488         case 'B' : /* Integer formats */
489 	case 'X' :
490 	case 'b' :
491         case 'd' :
492 	case 'i' :
493 	case 'o' :
494 	case 'u' :
495 	case 'x' :
496 	    if ((size_t)(width + 2) > sizeof(temp))
497 	      break;
498 
499 #  ifdef HAVE_LONG_LONG
500             if (size == 'L')
501 	      snprintf(temp, sizeof(temp), tformat, va_arg(ap, long long));
502 	    else
503 #  endif /* HAVE_LONG_LONG */
504             if (size == 'l')
505 	      snprintf(temp, sizeof(temp), tformat, va_arg(ap, long));
506 	    else
507 	      snprintf(temp, sizeof(temp), tformat, va_arg(ap, int));
508 
509             bytes += (int)strlen(temp);
510 
511 	    if (bufptr)
512 	    {
513 	      strlcpy(bufptr, temp, (size_t)(bufend - bufptr));
514 	      bufptr += strlen(bufptr);
515 	    }
516 	    break;
517 
518 	case 'p' : /* Pointer value */
519 	    if ((size_t)(width + 2) > sizeof(temp))
520 	      break;
521 
522 	    snprintf(temp, sizeof(temp), tformat, va_arg(ap, void *));
523 
524             bytes += (int)strlen(temp);
525 
526 	    if (bufptr)
527 	    {
528 	      strlcpy(bufptr, temp, (size_t)(bufend - bufptr));
529 	      bufptr += strlen(bufptr);
530 	    }
531 	    break;
532 
533         case 'c' : /* Character or character array */
534 	    bytes += width;
535 
536 	    if (bufptr)
537 	    {
538 	      if (width <= 1)
539 	        *bufptr++ = (char)va_arg(ap, int);
540 	      else
541 	      {
542 		if ((bufptr + width) > bufend)
543 		  width = (int)(bufend - bufptr);
544 
545 		memcpy(bufptr, va_arg(ap, char *), (size_t)width);
546 		bufptr += width;
547 	      }
548 	    }
549 	    break;
550 
551 	case 's' : /* String */
552 	    if ((s = va_arg(ap, char *)) == NULL)
553 	      s = "(null)";
554 
555            /*
556 	    * Copy the C string, replacing control chars and \ with
557 	    * C character escapes...
558 	    */
559 
560             for (bufend --; *s && bufptr < bufend; s ++)
561 	    {
562 	      if (*s == '\n')
563 	      {
564 	        *bufptr++ = '\\';
565 		*bufptr++ = 'n';
566 		bytes += 2;
567 	      }
568 	      else if (*s == '\r')
569 	      {
570 	        *bufptr++ = '\\';
571 		*bufptr++ = 'r';
572 		bytes += 2;
573 	      }
574 	      else if (*s == '\t')
575 	      {
576 	        *bufptr++ = '\\';
577 		*bufptr++ = 't';
578 		bytes += 2;
579 	      }
580 	      else if (*s == '\\')
581 	      {
582 	        *bufptr++ = '\\';
583 		*bufptr++ = '\\';
584 		bytes += 2;
585 	      }
586 	      else if (*s == '\'')
587 	      {
588 	        *bufptr++ = '\\';
589 		*bufptr++ = '\'';
590 		bytes += 2;
591 	      }
592 	      else if (*s == '\"')
593 	      {
594 	        *bufptr++ = '\\';
595 		*bufptr++ = '\"';
596 		bytes += 2;
597 	      }
598 	      else if ((*s & 255) < ' ')
599 	      {
600 	        if ((bufptr + 2) >= bufend)
601 	          break;
602 
603 	        *bufptr++ = '\\';
604 		*bufptr++ = '0';
605 		*bufptr++ = '0' + *s / 8;
606 		*bufptr++ = '0' + (*s & 7);
607 		bytes += 4;
608 	      }
609 	      else
610 	      {
611 	        *bufptr++ = *s;
612 		bytes ++;
613 	      }
614             }
615 
616             bufend ++;
617 	    break;
618 
619 	case 'n' : /* Output number of chars so far */
620 	    *(va_arg(ap, int *)) = (int)bytes;
621 	    break;
622       }
623     }
624     else
625     {
626       bytes ++;
627 
628       if (bufptr < bufend)
629         *bufptr++ = *format;
630 
631       format ++;
632     }
633   }
634 
635  /*
636   * Nul-terminate the string and return the number of characters needed.
637   */
638 
639   *bufptr = '\0';
640 
641   return (bytes);
642 }
643