2007-01-15 12:09:25 -08:00
|
|
|
/*
|
|
|
|
This file is part of Warzone 2100.
|
|
|
|
Copyright (C) 1999-2004 Eidos Interactive
|
2009-02-10 10:01:48 -08:00
|
|
|
Copyright (C) 2005-2009 Warzone Resurrection Project
|
2007-01-15 12:09:25 -08:00
|
|
|
|
|
|
|
Warzone 2100 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.
|
|
|
|
|
|
|
|
Warzone 2100 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 Warzone 2100; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
2006-09-25 09:46:28 -07:00
|
|
|
/*! \file debug.h
|
|
|
|
* \brief Debugging functions
|
|
|
|
*/
|
|
|
|
|
2008-02-01 13:26:37 -08:00
|
|
|
#ifndef __INCLUDED_LIB_FRAMEWORK_DEBUG_H__
|
|
|
|
#define __INCLUDED_LIB_FRAMEWORK_DEBUG_H__
|
2007-06-28 10:47:08 -07:00
|
|
|
|
|
|
|
/* Check the header files have been included from frame.h if they
|
|
|
|
* are used outside of the framework library.
|
|
|
|
*/
|
|
|
|
#if !defined(_frame_h) && !defined(FRAME_LIB_INCLUDE)
|
2007-08-21 05:30:00 -07:00
|
|
|
# error Framework header files MUST be included from Frame.h ONLY.
|
2007-06-28 10:47:08 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <assert.h>
|
2008-08-24 08:42:59 -07:00
|
|
|
#include "macros.h"
|
2008-11-08 11:30:18 -08:00
|
|
|
#include "types.h"
|
2007-06-28 10:47:08 -07:00
|
|
|
|
2008-02-01 13:26:37 -08:00
|
|
|
#if defined(__cplusplus)
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
#endif
|
|
|
|
|
2007-06-28 10:47:08 -07:00
|
|
|
/****************************************************************************************
|
|
|
|
*
|
|
|
|
* Basic debugging macro's
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2006-11-18 05:19:56 -08:00
|
|
|
#define MAX_EVENT_NAME_LEN 100
|
2007-12-29 14:34:41 -08:00
|
|
|
|
|
|
|
/** Stores name of the last function or event called by scripts. */
|
2006-11-18 05:19:56 -08:00
|
|
|
extern char last_called_script_event[MAX_EVENT_NAME_LEN];
|
2006-11-16 06:30:29 -08:00
|
|
|
|
2009-01-02 03:16:40 -08:00
|
|
|
/** Whether asserts are currently enabled. */
|
|
|
|
extern bool assertEnabled;
|
|
|
|
|
2008-08-24 08:42:59 -07:00
|
|
|
/**
|
|
|
|
* ASSERT helper macro to allow some debug functions to use an alternate
|
|
|
|
* calling location.
|
|
|
|
*
|
|
|
|
* \param expr Expression to assert on.
|
|
|
|
* \param location_description A string describing the calling location, e.g.:
|
|
|
|
* "filename:linenum".
|
|
|
|
* \param function The name of the function that called
|
|
|
|
* ASSERT_HELPER or the debug function that uses ASSERT_HELPER.
|
|
|
|
*
|
|
|
|
* \param ... printf-style format string followed by its parameters
|
|
|
|
*
|
|
|
|
* \return Will return whatever assert(expr) returns. That's undefined though,
|
|
|
|
* so unless you have a good reason to, don't depend on it.
|
|
|
|
*/
|
|
|
|
#define ASSERT_HELPER(expr, location_description, function, ...) \
|
|
|
|
( \
|
|
|
|
( \
|
|
|
|
(expr) ? /* if (expr) */ \
|
|
|
|
(void)0 \
|
|
|
|
: /* else */\
|
|
|
|
( \
|
|
|
|
(void)_debug(LOG_ERROR, function, __VA_ARGS__), \
|
|
|
|
(void)_debug(LOG_ERROR, function, "Assert in Warzone: %s (%s), last script event: '%s'", \
|
|
|
|
location_description, (#expr), last_called_script_event) \
|
|
|
|
) \
|
|
|
|
), \
|
2009-01-02 03:16:40 -08:00
|
|
|
assertEnabled ? assert(expr) : (void)0 \
|
2008-08-24 08:42:59 -07:00
|
|
|
)
|
|
|
|
|
2007-12-29 14:34:41 -08:00
|
|
|
/**
|
2007-06-28 10:47:08 -07:00
|
|
|
*
|
|
|
|
* Rewritten version of assert that allows a printf format text string to be passed
|
|
|
|
* to ASSERT along with the condition.
|
|
|
|
*
|
2006-08-23 05:58:48 -07:00
|
|
|
* Arguments: ASSERT( condition, "Format string with variables: %d, %d", var1, var2 );
|
2007-06-28 10:47:08 -07:00
|
|
|
*/
|
2008-08-24 08:42:59 -07:00
|
|
|
#define ASSERT(expr, ...) \
|
|
|
|
ASSERT_HELPER(expr, AT_MACRO, __FUNCTION__, __VA_ARGS__)
|
2007-06-28 10:47:08 -07:00
|
|
|
|
2007-05-02 08:50:30 -07:00
|
|
|
|
2007-12-29 14:34:41 -08:00
|
|
|
/**
|
2007-05-02 08:50:30 -07:00
|
|
|
* Compile time assert
|
2007-11-03 14:42:55 -07:00
|
|
|
* Not to be used in global context!
|
2007-05-02 08:50:30 -07:00
|
|
|
* \param expr Expression to evaluate
|
|
|
|
*/
|
2007-05-20 07:24:54 -07:00
|
|
|
#define STATIC_ASSERT( expr ) \
|
2007-11-03 14:42:55 -07:00
|
|
|
do { enum { assert_static__ = 1/(expr) }; } while(0)
|
2007-05-02 08:50:30 -07:00
|
|
|
|
|
|
|
|
2007-06-28 10:47:08 -07:00
|
|
|
/***
|
|
|
|
***
|
|
|
|
*** New debug logging output interface below. Heavily inspired
|
|
|
|
*** by similar code in Freeciv. Parts ripped directly.
|
|
|
|
***
|
|
|
|
***/
|
|
|
|
|
2007-12-29 14:34:41 -08:00
|
|
|
/** Debug enums. Must match code_part_names in debug.c */
|
2006-08-21 14:27:12 -07:00
|
|
|
typedef enum {
|
2007-06-28 10:47:08 -07:00
|
|
|
LOG_ALL, /* special: sets all to on */
|
|
|
|
LOG_MAIN,
|
2006-02-18 10:54:37 -08:00
|
|
|
LOG_SOUND,
|
|
|
|
LOG_VIDEO,
|
2007-06-28 10:47:08 -07:00
|
|
|
LOG_WZ,
|
|
|
|
LOG_3D,
|
|
|
|
LOG_TEXTURE,
|
2006-02-18 10:54:37 -08:00
|
|
|
LOG_NET,
|
2006-08-05 06:16:12 -07:00
|
|
|
LOG_MEMORY,
|
2007-12-29 14:34:41 -08:00
|
|
|
LOG_WARNING, /**< special; on in debug mode */
|
|
|
|
LOG_ERROR, /**< special; on by default */
|
|
|
|
LOG_NEVER, /**< if too verbose for anything but dedicated debugging... */
|
2006-08-12 03:02:59 -07:00
|
|
|
LOG_SCRIPT,
|
2006-12-01 14:22:29 -08:00
|
|
|
LOG_MOVEMENT,
|
|
|
|
LOG_ATTACK,
|
2007-02-24 08:07:44 -08:00
|
|
|
LOG_FOG,
|
2007-08-02 13:23:09 -07:00
|
|
|
LOG_SENSOR,
|
2007-08-03 09:24:17 -07:00
|
|
|
LOG_GUI,
|
2007-10-23 11:18:48 -07:00
|
|
|
LOG_MAP,
|
2008-04-19 07:52:35 -07:00
|
|
|
LOG_SAVE,
|
|
|
|
LOG_SYNC,
|
2008-01-02 09:08:29 -08:00
|
|
|
LOG_DEATH,
|
2008-01-11 14:45:04 -08:00
|
|
|
LOG_GATEWAY,
|
2008-04-19 14:41:18 -07:00
|
|
|
LOG_MSG,
|
2007-12-29 14:34:41 -08:00
|
|
|
LOG_LAST /**< _must_ be last! */
|
2006-08-21 14:27:12 -07:00
|
|
|
} code_part;
|
2007-06-28 10:47:08 -07:00
|
|
|
|
2009-01-31 07:51:35 -08:00
|
|
|
extern bool enabled_debug[LOG_LAST];
|
2007-07-01 04:54:09 -07:00
|
|
|
|
2006-08-21 14:27:12 -07:00
|
|
|
typedef void (*debug_callback_fn)(void**, const char*);
|
2008-11-08 11:30:18 -08:00
|
|
|
typedef bool (*debug_callback_init)(void**);
|
2006-08-21 14:27:12 -07:00
|
|
|
typedef void (*debug_callback_exit)(void**);
|
2007-06-28 10:47:08 -07:00
|
|
|
|
2006-08-21 14:27:12 -07:00
|
|
|
typedef struct _debug_callback {
|
|
|
|
struct _debug_callback * next;
|
|
|
|
debug_callback_fn callback; /// Function which does the output
|
|
|
|
debug_callback_init init; /// Setup function
|
|
|
|
debug_callback_exit exit; /// Cleaning function
|
|
|
|
void * data; /// Used to pass data to the above functions. Eg a filename or handle.
|
|
|
|
} debug_callback;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call once to initialize the debug logging system.
|
|
|
|
*
|
|
|
|
* Doesn't register any callbacks!
|
|
|
|
*/
|
|
|
|
void debug_init( void );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shutdown the debug system and remove all output callbacks
|
|
|
|
*/
|
|
|
|
void debug_exit( void );
|
|
|
|
|
2008-07-21 12:49:36 -07:00
|
|
|
/**
|
|
|
|
* Have the stderr output callback flush its output before returning.
|
|
|
|
*
|
|
|
|
* NOTE: This may cause significant slowdowns on some systems.
|
|
|
|
*/
|
|
|
|
extern void debugFlushStderr(void);
|
|
|
|
|
2006-08-21 14:27:12 -07:00
|
|
|
/**
|
|
|
|
* Register a callback to be called on every call to debug()
|
|
|
|
*
|
|
|
|
* \param callback Function which does the output
|
|
|
|
* \param init Initializer function which does all setup for the callback (optional, may be NULL)
|
|
|
|
* \param exit Cleanup function called when unregistering the callback (optional, may be NULL)
|
|
|
|
* \param data Data to be passed to all three functions (optional, may be NULL)
|
|
|
|
*/
|
|
|
|
void debug_register_callback( debug_callback_fn callback, debug_callback_init init, debug_callback_exit exit, void * data );
|
|
|
|
|
2006-12-02 05:06:56 -08:00
|
|
|
void debug_callback_file(void **data, const char *outputBuffer);
|
2008-11-08 11:30:18 -08:00
|
|
|
bool debug_callback_file_init(void **data);
|
2006-12-02 05:06:56 -08:00
|
|
|
void debug_callback_file_exit(void **data);
|
|
|
|
|
|
|
|
void debug_callback_stderr(void **data, const char *outputBuffer);
|
|
|
|
|
2008-04-13 15:54:58 -07:00
|
|
|
#if defined WIN32 && defined DEBUG
|
|
|
|
void debug_callback_win32debug(void** data, const char* outputBuffer);
|
|
|
|
#endif
|
|
|
|
|
2006-08-21 14:27:12 -07:00
|
|
|
/**
|
|
|
|
* Toggle debug output for part associated with str
|
|
|
|
*
|
|
|
|
* \param str Codepart in textformat
|
|
|
|
*/
|
2009-01-31 07:51:35 -08:00
|
|
|
bool debug_enable_switch(const char *str);
|
2006-08-21 14:27:12 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Output printf style format str with additional arguments.
|
|
|
|
*
|
|
|
|
* Only outputs if debugging of part was formerly enabled with debug_enable_switch.
|
|
|
|
*/
|
2008-04-19 07:52:35 -07:00
|
|
|
#define debug(part, ...) do { if (enabled_debug[part]) _debug(part, __FUNCTION__, __VA_ARGS__); } while(0)
|
|
|
|
void _debug( code_part part, const char *function, const char *str, ...)
|
|
|
|
WZ_DECL_FORMAT(printf, 3, 4);
|
2007-06-28 10:47:08 -07:00
|
|
|
|
2008-01-11 14:45:04 -08:00
|
|
|
/** Global to keep track of which game object to trace. */
|
|
|
|
extern UDWORD traceID;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Output printf style format str for debugging a specific game object whose debug part
|
|
|
|
* has been enabled.
|
|
|
|
* @see debug
|
|
|
|
*/
|
2008-05-11 07:58:20 -07:00
|
|
|
#define objTrace(id, ...) do { if (id == traceID) _realObjTrace(id, __FUNCTION__, __VA_ARGS__); } while(0)
|
|
|
|
void _realObjTrace(int id, const char *function, const char *str, ...) WZ_DECL_FORMAT(printf, 3, 4);
|
2008-01-11 14:45:04 -08:00
|
|
|
static inline void objTraceEnable(UDWORD id) { traceID = id; }
|
2008-05-31 16:17:27 -07:00
|
|
|
static inline void objTraceDisable(void) { traceID = (UDWORD)-1; }
|
2008-01-11 14:45:04 -08:00
|
|
|
|
2008-02-01 13:26:37 -08:00
|
|
|
#if defined(__cplusplus)
|
|
|
|
}
|
2007-06-28 10:47:08 -07:00
|
|
|
#endif
|
2008-02-01 13:26:37 -08:00
|
|
|
|
2008-03-22 10:42:07 -07:00
|
|
|
/** Checks if a particular debub flag was enabled */
|
|
|
|
extern bool debugPartEnabled(code_part codePart);
|
|
|
|
|
2009-01-02 03:16:40 -08:00
|
|
|
void debugDisableAssert(void);
|
|
|
|
|
2008-02-01 13:26:37 -08:00
|
|
|
#endif // __INCLUDED_LIB_FRAMEWORK_DEBUG_H__
|