diff --git a/src/Makefile.am b/src/Makefile.am index b6a32d81..b31f413d 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -12,7 +12,6 @@ endif mt_daapd_CPPFLAGS = -D_GNU_SOURCE @AVAHI_CFLAGS@ @SQLITE3_CFLAGS@ @FFMPEG_CFLAGS@ @CONFUSE_CFLAGS@ @TAGLIB_CFLAGS@ @MINIXML_CFLAGS@ mt_daapd_LDADD = @AVAHI_LIBS@ @SQLITE3_LIBS@ @FFMPEG_LIBS@ @CONFUSE_LIBS@ @FLAC_LIBS@ @TAGLIB_LIBS@ @LIBEVENT_LIBS@ @LIBAVL_LIBS@ @MINIXML_LIBS@ mt_daapd_SOURCES = main.c \ - err.c err.h \ logger.c logger.h \ conffile.c conffile.h \ filescanner.c filescanner.h \ diff --git a/src/err.c b/src/err.c deleted file mode 100644 index dbf751c1..00000000 --- a/src/err.c +++ /dev/null @@ -1,441 +0,0 @@ -/* - * Generic error handling - * - * Copyright (C) 2003 Ron Pedde (ron@pedde.com) - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ - -/** - * \file err.c - * Error handling, logging, and memory leak checking. - * - * Most of these functions should not be used directly. For the most - * part, they are hidden in macros like #DPRINTF and #MEMNOTIFY. The - * only function here that is really directly useable is log_setdest() - */ - -#ifdef HAVE_CONFIG_H -# include "config.h" -#endif - -#include -#include -#include -#include -#include -#include -#ifdef HAVE_STDINT_H -#include -#endif -#include -#include - -#include -#include - -#include - -#include "err.h" -#include "misc.h" - -#ifndef PACKAGE -# define PACKAGE "unknown daemon" -#endif - -typedef struct err_threadlist_t { - uint32_t id; - struct err_threadlist_t *next; -} ERR_THREADLIST; - -static int err_debuglevel=0; /**< current debuglevel, set from command line with -d */ -static int err_logdest=0; /**< current log destination */ -static char err_filename[PATH_MAX + 1]; -static FILE *err_file = NULL; -static unsigned int err_debugmask=0xFFFFFFFF; /**< modules to debug, see \ref log_categories */ -static int err_truncate = 0; -static int err_syslog_open = 0; - -/** text list of modules to match for setting debug mask */ -static char *err_categorylist[] = { - "config","webserver","database","scan","query","index","browse", - "playlist","art","daap","main","rend","xml","parse","plugin","lock","httpd","rsp","xcode",NULL -}; - -static pthread_mutex_t m_err = PTHREAD_MUTEX_INITIALIZER; -static pthread_mutex_t m_errlist = PTHREAD_MUTEX_INITIALIZER; - -static ERR_THREADLIST err_threadlist = { 0, NULL }; - -static int lvl2syslog[10] = - { - LOG_ALERT, LOG_ALERT, - LOG_NOTICE, LOG_NOTICE, LOG_NOTICE, - LOG_INFO, LOG_INFO, LOG_INFO, LOG_INFO, - LOG_DEBUG - }; - -/* - * Forwards - */ - -static uint32_t __err_get_threadid(void); -static void __err_thread_add(void); -static void __err_thread_del(void); -static int __err_thread_check(void); - -/** - * add a thread id to the threadlist. note that the threadlist semaphore must - * be locked. - */ -void __err_thread_add(void) { - ERR_THREADLIST *pnew; - - pnew = (ERR_THREADLIST *)malloc(sizeof(ERR_THREADLIST)); - if(!pnew) { - fprintf(stderr,"malloc error in __err_thread_add\n"); - exit(-1); - } - - pnew->id = __err_get_threadid(); - pnew->next = err_threadlist.next; - err_threadlist.next = pnew; -} - -/** - * remove a thread id from the threadlist. The threadlist semaphore must be locked - */ -void __err_thread_del(void) { - uint32_t thread_id = __err_get_threadid(); - ERR_THREADLIST *last, *current; - last = &err_threadlist; - current = err_threadlist.next; - - while((current) && (current->id != thread_id)) { - last = current; - current = current->next; - } - - if(current) { - last->next = current->next; - free(current); - } -} - -/** - * see if the current thread is already in the thread list - */ -int __err_thread_check(void) { - uint32_t thread_id = __err_get_threadid(); - ERR_THREADLIST *current = err_threadlist.next; - - while(current && (current->id != thread_id)) - current = current->next; - - if(current) - return TRUE; - - return FALSE; -} - -/** - * get an integer representation of a thread id - */ -uint32_t __err_get_threadid(void) { - pthread_t tid; - int thread_id=0; - - memset((void*)&tid,0,sizeof(pthread_t)); - tid = pthread_self(); - - if(sizeof(pthread_t) == sizeof(int)) { - memcpy((void*)&thread_id,(void*)&tid,sizeof(thread_id)); - } else { - thread_id = djb_hash((unsigned char *)&tid,sizeof(pthread_t)); - } - - return thread_id; -} - -/** - * if we are logging to a file, then re-open the file. This - * would help for log rotation - */ -void err_reopen(void) { - if(!(err_logdest & LOGDEST_LOGFILE)) - return; - - fclose(err_file); - - err_file = fopen(err_filename, "w"); - if (!err_file) { - /* what to do when you lose your logging mechanism? Keep - * going? - */ - err_setdest(err_logdest & (~LOGDEST_LOGFILE)); - err_setdest(err_logdest | LOGDEST_SYSLOG); - - DPRINTF(E_LOG,L_MISC,"Could not rotate log file: %s\n", strerror(errno)); - return; - } - - DPRINTF(E_LOG,L_MISC,"Rotated logs\n"); -} - -/** - * Write a printf-style formatted message to the log destination. - * This can be stderr, syslog/eventviewer, or a logfile, as determined by - * err_setdest(). Note that this function should not be directly - * used, rather it should be used via the #DPRINTF macro. - * - * \param level Level at which to log \ref log_levels - * \param cat the category to log \ref log_categories - * \param fmt printf-style - */ -void err_log(int level, unsigned int cat, char *fmt, ...) -{ - va_list ap; - char timebuf[256]; - char errbuf[4096]; - struct tm tm_now; - time_t tt_now; - int syslog_only = FALSE; - - if(level > 1) { - if(level > err_debuglevel) - return; - - if(!(cat & err_debugmask)) - return; - } /* we'll *always* process a log level 0 or 1 */ - - /* skip recursive calls to logging functions to avoid deadlocks (except for aborts) */ - pthread_mutex_lock(&m_errlist); - if(err_threadlist.next && __err_thread_check()) { /* skip logging */ - if(!level) { - syslog_only = TRUE; /* syslog fatals even on recursive calls */ - } else { - pthread_mutex_unlock(&m_errlist); - return; - } - } - __err_thread_add(); - pthread_mutex_unlock(&m_errlist); - - va_start(ap, fmt); - vsnprintf(errbuf, sizeof(errbuf), fmt, ap); - va_end(ap); - - /* always log fatals and level 1 to syslog */ - if(level <= 1) { - if(!err_syslog_open) - openlog(PACKAGE, LOG_PID, LOG_DAEMON); - - err_syslog_open=1; - syslog((level > E_DBG) ? LOG_DEBUG : lvl2syslog[level], "%s", errbuf); - - if(syslog_only && !level) { - fprintf(stderr,"Aborting\n"); - exit(-1); - } - } - - pthread_mutex_lock(&m_err); - - if((err_logdest & LOGDEST_LOGFILE) && (err_file) && (!syslog_only)) { - tt_now=time(NULL); - localtime_r(&tt_now,&tm_now); - snprintf(timebuf,sizeof(timebuf),"%04d-%02d-%02d %02d:%02d:%02d", - tm_now.tm_year + 1900, tm_now.tm_mon + 1, tm_now.tm_mday, - tm_now.tm_hour, tm_now.tm_min, tm_now.tm_sec); - fprintf(err_file,"%s (%08x): %s",timebuf,__err_get_threadid(),errbuf); - if(!level) fprintf(err_file,"%s: Aborting\n",timebuf); - } - - /* always log to stderr on fatal error */ - if((err_logdest & LOGDEST_STDERR) || (!level)) { - fprintf(stderr, "%s",errbuf); - if(!level) fprintf(stderr,"Aborting\n"); - } - - pthread_mutex_unlock(&m_err); - - pthread_mutex_lock(&m_errlist); - __err_thread_del(); - pthread_mutex_unlock(&m_errlist); - - if(!level) { - exit(EXIT_FAILURE); /* this should go to an OS-specific exit routine */ - } -} - -/** - * simple get/set interface to debuglevel to avoid global - */ -void err_setlevel(int level) { - pthread_mutex_lock(&m_err); - err_debuglevel = level; - pthread_mutex_unlock(&m_err); -} - -/** - * get current debug level - */ -int err_getlevel(void) { - int level; - - pthread_mutex_lock(&m_err); - level = err_debuglevel; - pthread_mutex_unlock(&m_err); - - return level; -} - - -/** - * get the logfile destination - */ -int err_getdest(void) { - int dest; - - pthread_mutex_lock(&m_err); - dest=err_logdest; - pthread_mutex_unlock(&m_err); - - return dest; -} - - -int err_settruncate(int truncate) { - char *file; - - if(err_truncate == truncate) - return TRUE; - - err_truncate = truncate; - if((err_truncate) && (err_file)) { - file=strdup(err_filename); - err_setlogfile(err_filename); - if(file) free(file); - } - - return TRUE; -} - -int err_setlogfile(char *file) { - char *mode; - int result=TRUE; - -/* - if(strcmp(file,err_filename) == 0) - return TRUE; -*/ - - if(err_file) { - fclose(err_file); - } - - mode = "a"; - if(err_truncate) mode = "w"; - - strncpy(err_filename,file,sizeof(err_filename)-1); - - err_file = fopen(err_filename, mode); - if (!err_file) { - fprintf(stderr,"Error opening logfile: %s", strerror(errno)); - err_logdest &= ~LOGDEST_LOGFILE; - - if(!err_syslog_open) - openlog(PACKAGE, LOG_PID, LOG_DAEMON); - - syslog(lvl2syslog[1], "Error opening logfile"); - - result=FALSE; - } - - return result; -} - -/** - * Sets the log destination. (stderr, syslog, or logfile) - * - * \param app appname (used only for syslog destination) - * \param destination where to log to \ref log_dests "as defined in err.h" - */ -void err_setdest(int destination) { - if(err_logdest == destination) - return; - - pthread_mutex_lock(&m_err); - if((err_logdest & LOGDEST_LOGFILE) && - (!(destination & LOGDEST_LOGFILE))) { - /* used to be logging to file, not any more */ - fclose(err_file); - } - - err_logdest=destination; - pthread_mutex_unlock(&m_err); -} -/** - * Set the debug mask. Given a comma separated list, this walks - * through the err_categorylist and sets the bitfields for the - * requested log modules. - * - * \param list comma separated list of modules to debug. - */ -extern int err_setdebugmask(char *list) { - unsigned int rack; - char *token, *str, *last; - char *tmpstr; - int index; - - err_debugmask=0x80000000; /* always log L_MISC! */ - str=tmpstr=strdup(list); - if(!str) - return 0; - - pthread_mutex_lock(&m_err); - while(1) { - token=strtok_r(str,",",&last); - str=NULL; - - if(token) { - rack=1; - index=0; - while((err_categorylist[index]) && - (strcasecmp(err_categorylist[index],token))) { - rack <<= 1; - index++; - } - - if(!err_categorylist[index]) { - pthread_mutex_unlock(&m_err); - DPRINTF(E_LOG,L_MISC,"Unknown module: %s\n",token); - free(tmpstr); - return 1; - } else { - err_debugmask |= rack; - } - } else break; /* !token */ - } - - pthread_mutex_unlock(&m_err); - DPRINTF(E_INF,L_MISC,"Debug mask is 0x%08x\n",err_debugmask); - free(tmpstr); - - return 0; -} - - diff --git a/src/err.h b/src/err.h deleted file mode 100644 index 0f0ba5f0..00000000 --- a/src/err.h +++ /dev/null @@ -1,92 +0,0 @@ -/* - * Error related routines - * - * Copyright (C) 2003 Ron Pedde (ron@pedde.com) - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ - -/** - * \file err.h - * - * Header file for err.c - */ - -#ifndef __ERR_H__ -#define __ERR_H__ - -/** @anchor log_dests */ -#define LOGDEST_STDERR 1 /**< Log to stderr */ -#define LOGDEST_SYSLOG 2 /**< Log to syslog/eventviewer */ -#define LOGDEST_LOGFILE 4 /**< Log to logfile */ - -/** @anchor log_levels */ -#define E_SPAM 10 /**< Logorrhea! */ -#define E_DBG 9 /**< Way too verbose */ -#define E_INF 5 /**< Good info, not too much spam */ -#define E_WARN 2 /**< Goes in text log, but not syslog */ -#define E_LOG 1 /**< Something that should go in a log file */ -#define E_FATAL 0 /**< Log and force an exit */ - -/** @anchor log_categories */ -#define L_CONF 0x00000001 /**< configuration - configfile.c */ -#define L_WS 0x00000002 /**< webserver - webserver.c */ -#define L_DB 0x00000004 /**< database */ -#define L_SCAN 0x00000008 /**< scanner - mp3-scanner.c */ -#define L_QRY 0x00000010 /**< query - query.c */ -#define L_IND 0x00000020 /**< index - daap.c */ -#define L_BROW 0x00000040 /**< browse - daap.c, query.c */ -#define L_PL 0x00000080 /**< playlists - playlist.c, lexer.l, parser.y */ -#define L_ART 0x00000100 /**< cover art - dynamic-art.c */ -#define L_DAAP 0x00000200 /**< Generally daap related - main.c, daap.c, query.c */ -#define L_MAIN 0x00000400 /**< setup, teardown, signals - main.c */ -#define L_REND 0x00000800 /**< rendezvous -- rend-unix.c, rend-posix.c, etc */ -#define L_XML 0x00001000 /**< xml - xml-rpc.c */ -#define L_PARSE 0x00002000 /**< smart playlist parser */ -#define L_PLUG 0x00004000 /**< plugins */ -#define L_LOCK 0x00008000 /**< semaphore locks */ -#define L_HTTPD 0x00010000 /**< httpd - httpd.c */ -#define L_RSP 0x00020000 /**< RSP - httpd_rsp.c */ -#define L_XCODE 0x00040000 /**< transcode - transcode.c */ -#define L_MISC 0x80000000 /**< anything else */ - -#ifndef TRUE -# define TRUE 1 -# define FALSE 0 -#endif - -extern void err_log(int level, unsigned int cat, char *fmt, ...); -extern void err_reopen(void); /** rotate logfile */ -extern void err_setdest(int destination); -extern int err_getdest(void); -extern void err_setlevel(int level); -extern int err_getlevel(void); -extern int err_setdebugmask(char *list); -extern int err_setlogfile(char *file); -extern int err_settruncate(int truncate); - -/** - * Print a debugging or log message - */ - -#define DPRINTF err_log - -#ifdef ERR_LEAN -# define os_syslog(a,b) -# define os_opensyslog(a) -# define os_closesyslog(a) -#endif - -#endif /* __ERR_H__ */