diff options
Diffstat (limited to 'src/egg-debug.c')
| -rw-r--r-- | src/egg-debug.c | 308 | 
1 files changed, 308 insertions, 0 deletions
| diff --git a/src/egg-debug.c b/src/egg-debug.c new file mode 100644 index 0000000..2f140f2 --- /dev/null +++ b/src/egg-debug.c @@ -0,0 +1,308 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- + * + * Copyright (C) 2007-2008 Richard Hughes <[email protected]> + * + * Licensed under the GNU General Public License Version 2 + * + * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +/** + * SECTION:egg-debug + * @short_description: Debugging functions + * + * This file contains functions that can be used for debugging. + */ + +#include <glib.h> +#include <glib/gi18n.h> +#include <glib/gprintf.h> +#include <stdio.h> +#include <string.h> +#include <stdarg.h> +#include <stdlib.h> +#include <signal.h> +#include <unistd.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <time.h> +#include <execinfo.h> + +#include "egg-debug.h" + +#define CONSOLE_RESET		0 +#define CONSOLE_BLACK 		30 +#define CONSOLE_RED		31 +#define CONSOLE_GREEN		32 +#define CONSOLE_YELLOW		33 +#define CONSOLE_BLUE		34 +#define CONSOLE_MAGENTA		35 +#define CONSOLE_CYAN		36 +#define CONSOLE_WHITE		37 + +static gint fd = -1; + +/** + * pk_set_console_mode: + **/ +static void +pk_set_console_mode (guint console_code) +{ +	gchar command[13]; + +	/* don't put extra commands into logs */ +	if (!egg_debug_is_console ()) +		return; + +	/* Command is the control command to the terminal */ +	g_snprintf (command, 13, "%c[%dm", 0x1B, console_code); +	printf ("%s", command); +} + +/** + * egg_debug_backtrace: + **/ +void +egg_debug_backtrace (void) +{ +	void *call_stack[512]; +	int  call_stack_size; +	char **symbols; +	int i = 1; + +	call_stack_size = backtrace (call_stack, G_N_ELEMENTS (call_stack)); +	symbols = backtrace_symbols (call_stack, call_stack_size); +	if (symbols != NULL) { +		pk_set_console_mode (CONSOLE_RED); +		g_print ("Traceback:\n"); +		while (i < call_stack_size) { +			g_print ("\t%s\n", symbols[i]); +			i++; +		} +		pk_set_console_mode (CONSOLE_RESET); +		free (symbols); +	} +} + +/** + * pk_log_line: + **/ +static void +pk_log_line (const gchar *buffer) +{ +	ssize_t count; +	/* open a file */ +	if (fd == -1) { +		/* ITS4: ignore, /var/log/foo is owned by root, and this is just debug text */ +		fd = open (EGG_LOG_FILE, O_WRONLY|O_APPEND|O_CREAT, 0777); +		if (fd == -1) +			g_error ("could not open log: '%s'", EGG_LOG_FILE); +	} + +	/* ITS4: ignore, debug text always NULL terminated */ +	count = write (fd, buffer, strlen (buffer)); +	if (count == -1) +		g_warning ("could not write %s", buffer); +	/* newline */ +	count = write (fd, "\n", 1); +	if (count == -1) +		g_warning ("could not write newline"); +} + +/** + * pk_print_line: + **/ +static void +pk_print_line (const gchar *func, const gchar *file, const int line, const gchar *buffer, guint color) +{ +	gchar *str_time; +	gchar *header; +	time_t the_time; +	GThread *thread; + +	time (&the_time); +	str_time = g_new0 (gchar, 255); +	strftime (str_time, 254, "%H:%M:%S", localtime (&the_time)); +	thread = g_thread_self (); + +	/* generate header text */ +	header = g_strdup_printf ("TI:%s\tTH:%p\tFI:%s\tFN:%s,%d", str_time, thread, file, func, line); +	g_free (str_time); + +	/* always in light green */ +	pk_set_console_mode (CONSOLE_GREEN); +	printf ("%s\n", header); + +	/* different colors according to the severity */ +	pk_set_console_mode (color); +	printf (" - %s\n", buffer); +	pk_set_console_mode (CONSOLE_RESET); + +	/* log to a file */ +	if (egg_debug_is_logging ()) { +		pk_log_line (header); +		pk_log_line (buffer); +	} + +	/* flush this output, as we need to debug */ +	fflush (stdout); + +	g_free (header); +} + +/** + * egg_debug_real: + **/ +void +egg_debug_real (const gchar *func, const gchar *file, const int line, const gchar *format, ...) +{ +	va_list args; +	gchar *buffer = NULL; + +	if (!egg_debug_enabled ()) +		return; + +	va_start (args, format); +	g_vasprintf (&buffer, format, args); +	va_end (args); + +	pk_print_line (func, file, line, buffer, CONSOLE_BLUE); + +	g_free(buffer); +} + +/** + * egg_warning_real: + **/ +void +egg_warning_real (const gchar *func, const gchar *file, const int line, const gchar *format, ...) +{ +	va_list args; +	gchar *buffer = NULL; + +	if (!egg_debug_enabled ()) +		return; + +	va_start (args, format); +	g_vasprintf (&buffer, format, args); +	va_end (args); + +	/* do extra stuff for a warning */ +	if (!egg_debug_is_console ()) +		printf ("*** WARNING ***\n"); +	pk_print_line (func, file, line, buffer, CONSOLE_RED); + +	g_free(buffer); +} + +/** + * egg_error_real: + **/ +void +egg_error_real (const gchar *func, const gchar *file, const int line, const gchar *format, ...) +{ +	va_list args; +	gchar *buffer = NULL; + +	va_start (args, format); +	g_vasprintf (&buffer, format, args); +	va_end (args); + +	/* do extra stuff for a warning */ +	if (!egg_debug_is_console ()) +		printf ("*** ERROR ***\n"); +	pk_print_line (func, file, line, buffer, CONSOLE_RED); +	g_free(buffer); + +	/* we want to fix this! */ +	egg_debug_backtrace (); + +	exit (1); +} + +/** + * egg_debug_enabled: + * + * Returns: TRUE if we have debugging enabled + **/ +gboolean +egg_debug_enabled (void) +{ +	const gchar *env; +	env = g_getenv (EGG_VERBOSE); +	return (g_strcmp0 (env, "1") == 0); +} + +/** + * egg_debug_is_logging: + * + * Returns: TRUE if we have logging enabled + **/ +gboolean +egg_debug_is_logging (void) +{ +	const gchar *env; +	env = g_getenv (EGG_LOGGING); +	return (g_strcmp0 (env, "1") == 0); +} + +/** + * egg_debug_is_console: + * + * Returns: TRUE if we have debugging enabled + **/ +gboolean +egg_debug_is_console (void) +{ +	const gchar *env; +	env = g_getenv (EGG_CONSOLE); +	return (g_strcmp0 (env, "1") == 0); +} + +/** + * egg_debug_set_logging: + **/ +void +egg_debug_set_logging (gboolean enabled) +{ +	if (enabled) +		g_setenv (EGG_LOGGING, "1", TRUE); +	else +		g_setenv (EGG_LOGGING, "0", TRUE); + +	if (egg_debug_is_logging ()) +		egg_debug ("logging to %s", EGG_LOG_FILE); +} + +/** + * egg_debug_init: + * @debug: If we should print out verbose logging + **/ +void +egg_debug_init (gboolean debug) +{ +	/* check if we are on console */ +	if (isatty (fileno (stdout)) == 1) +		g_setenv (EGG_CONSOLE, "1", FALSE); +	else +		g_setenv (EGG_CONSOLE, "0", FALSE); +	if (debug) +		g_setenv (EGG_VERBOSE, "1", FALSE); +	else +		g_setenv (EGG_VERBOSE, "0", FALSE); +	egg_debug ("Verbose debugging %i (on console %i)%s", egg_debug_enabled (), egg_debug_is_console (), EGG_VERBOSE); +} + | 
