2002-07-09 16:24:59 +04:00
|
|
|
/*
|
2015-04-21 19:49:37 +03:00
|
|
|
* Copyright 2002-2015, Axel Dörfler, axeld@pinc-software.de
|
2005-05-29 17:01:38 +04:00
|
|
|
* Distributed under the terms of the Haiku License.
|
|
|
|
*
|
|
|
|
* Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
|
|
|
|
* Distributed under the terms of the NewOS License.
|
|
|
|
*/
|
2002-07-09 16:24:59 +04:00
|
|
|
#ifndef _KERNEL_DEBUG_H
|
|
|
|
#define _KERNEL_DEBUG_H
|
|
|
|
|
2009-08-03 16:39:56 +04:00
|
|
|
|
|
|
|
#include <setjmp.h>
|
2003-05-03 20:03:26 +04:00
|
|
|
|
2007-10-04 02:14:53 +04:00
|
|
|
#include <KernelExport.h>
|
2008-01-31 15:25:43 +03:00
|
|
|
#include <module.h>
|
|
|
|
|
2009-08-03 16:39:56 +04:00
|
|
|
#include "kernel_debug_config.h"
|
|
|
|
|
2003-05-03 20:03:26 +04:00
|
|
|
|
2011-01-11 00:54:38 +03:00
|
|
|
// We need the BKernel::Thread type below (opaquely) in the exported C
|
|
|
|
// functions below. Since this header is currently still included by plain C
|
|
|
|
// code, we define a dummy type BKernel_Thread in C mode and a equally named
|
|
|
|
// macro in C++ mode.
|
|
|
|
#ifdef __cplusplus
|
|
|
|
namespace BKernel {
|
|
|
|
struct Thread;
|
|
|
|
}
|
|
|
|
|
|
|
|
using BKernel::Thread;
|
|
|
|
# define BKernel_Thread Thread
|
|
|
|
#else
|
|
|
|
typedef struct BKernel_Thread BKernel_Thread;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2008-10-20 17:06:04 +04:00
|
|
|
/* KDEBUG
|
|
|
|
The kernel debug level.
|
|
|
|
Level 1 is usual asserts, > 1 should be used for very expensive runtime
|
|
|
|
checks
|
2007-06-21 09:37:46 +04:00
|
|
|
*/
|
2008-10-20 17:06:04 +04:00
|
|
|
#if !defined(KDEBUG)
|
|
|
|
# if DEBUG
|
|
|
|
# define KDEBUG 1
|
|
|
|
# else
|
|
|
|
# define KDEBUG 0
|
|
|
|
# endif
|
2007-06-21 09:37:46 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define ASSERT_ALWAYS(x) \
|
* Added an emergency key mechanism to the kernel, inspired by Jan Klötzke's
patch from about a year ago (I couldn't use any code of his yet, though,
but there are a few things left). The emergency keys are triggered by
pressing Alt-SysReq + key.
* By default, only Alt-SysReq+'d' is used as a means to deliberately enter
the kernel debugger. F12 belongs to userland again, now :-)
* Debugger add-ons now have another optional method to implement their own
emergency keys - 'd' for the debugger cannot be overridden, though.
* The mechanism can be turned off via a new kernel setting, so it's not that
easy anymore to "crash" Haiku if you don't want to.
* Right now, the PS/2 driver, and the pre-input_server in-kernel debugger
keyboard mini-driver support this, USB not yet.
* Minor cleanup.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@31660 a95241bf-73f2-0310-859d-f6bbb57e9c96
2009-07-21 00:39:37 +04:00
|
|
|
do { \
|
|
|
|
if (!(x)) { \
|
2010-02-20 17:19:52 +03:00
|
|
|
panic("ASSERT FAILED (%s:%d): %s", __FILE__, __LINE__, #x); \
|
* Added an emergency key mechanism to the kernel, inspired by Jan Klötzke's
patch from about a year ago (I couldn't use any code of his yet, though,
but there are a few things left). The emergency keys are triggered by
pressing Alt-SysReq + key.
* By default, only Alt-SysReq+'d' is used as a means to deliberately enter
the kernel debugger. F12 belongs to userland again, now :-)
* Debugger add-ons now have another optional method to implement their own
emergency keys - 'd' for the debugger cannot be overridden, though.
* The mechanism can be turned off via a new kernel setting, so it's not that
easy anymore to "crash" Haiku if you don't want to.
* Right now, the PS/2 driver, and the pre-input_server in-kernel debugger
keyboard mini-driver support this, USB not yet.
* Minor cleanup.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@31660 a95241bf-73f2-0310-859d-f6bbb57e9c96
2009-07-21 00:39:37 +04:00
|
|
|
} \
|
|
|
|
} while (0)
|
2007-06-21 09:37:46 +04:00
|
|
|
|
2010-02-20 17:19:52 +03:00
|
|
|
#define ASSERT_ALWAYS_PRINT(x, format, args...) \
|
2007-08-06 21:35:57 +04:00
|
|
|
do { \
|
|
|
|
if (!(x)) { \
|
2010-02-20 17:19:52 +03:00
|
|
|
panic("ASSERT FAILED (%s:%d): %s; " format, __FILE__, __LINE__, \
|
|
|
|
#x, args); \
|
2007-08-06 21:35:57 +04:00
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2007-06-21 09:37:46 +04:00
|
|
|
#if KDEBUG
|
2010-02-20 17:19:52 +03:00
|
|
|
# define ASSERT(x) ASSERT_ALWAYS(x)
|
|
|
|
# define ASSERT_PRINT(x, format, args...) ASSERT_ALWAYS_PRINT(x, format, args)
|
2008-08-12 21:09:42 +04:00
|
|
|
#else
|
2010-02-20 17:19:52 +03:00
|
|
|
# define ASSERT(x) do { } while(0)
|
|
|
|
# define ASSERT_PRINT(x, format, args...) do { } while(0)
|
2002-07-09 16:24:59 +04:00
|
|
|
#endif
|
2002-07-18 22:45:07 +04:00
|
|
|
|
2018-08-22 07:32:30 +03:00
|
|
|
#if __GNUC__ >= 5 && !defined(__cplusplus)
|
|
|
|
# define STATIC_ASSERT(x) _Static_assert(x, "static assert failed!")
|
|
|
|
#elif defined(__cplusplus) && __cplusplus >= 201103L
|
|
|
|
# define STATIC_ASSERT(x) static_assert(x, "static assert failed!")
|
|
|
|
#else
|
|
|
|
# define STATIC_ASSERT(x) \
|
|
|
|
do { \
|
|
|
|
struct __staticAssertStruct__ { \
|
|
|
|
char __static_assert_failed__[2*(x) - 1]; \
|
|
|
|
}; \
|
|
|
|
} while (false)
|
|
|
|
#endif
|
2010-06-10 21:31:57 +04:00
|
|
|
|
2008-07-18 06:05:27 +04:00
|
|
|
#if KDEBUG
|
|
|
|
# define KDEBUG_ONLY(x) x
|
|
|
|
#else
|
|
|
|
# define KDEBUG_ONLY(x) /* nothing */
|
|
|
|
#endif
|
|
|
|
|
2011-11-01 21:19:26 +04:00
|
|
|
|
|
|
|
// Macros for for placing marker functions. They can be used to mark the
|
|
|
|
// beginning and end of code sections (e.g. used in the slab code).
|
|
|
|
#define RANGE_MARKER_FUNCTION(functionName) \
|
|
|
|
void functionName() {}
|
|
|
|
#define RANGE_MARKER_FUNCTION_BEGIN(scope) \
|
|
|
|
RANGE_MARKER_FUNCTION(scope##_begin)
|
|
|
|
#define RANGE_MARKER_FUNCTION_END(scope) \
|
|
|
|
RANGE_MARKER_FUNCTION(scope##_end)
|
|
|
|
|
|
|
|
#define RANGE_MARKER_FUNCTION_PROTOTYPE(functionName) \
|
|
|
|
void functionName();
|
|
|
|
#define RANGE_MARKER_FUNCTION_PROTOTYPES(scope) \
|
|
|
|
RANGE_MARKER_FUNCTION_PROTOTYPE(scope##_begin) \
|
|
|
|
RANGE_MARKER_FUNCTION_PROTOTYPE(scope##_end)
|
|
|
|
#define RANGE_MARKER_FUNCTION_ADDRESS_RANGE(scope) \
|
|
|
|
(addr_t)&scope##_begin, (addr_t)&scope##_end
|
|
|
|
|
|
|
|
|
2008-06-01 06:25:00 +04:00
|
|
|
// command return value
|
2008-10-16 03:43:27 +04:00
|
|
|
#define B_KDEBUG_ERROR 4
|
|
|
|
#define B_KDEBUG_RESTART_PIPE 5
|
2008-06-01 06:25:00 +04:00
|
|
|
|
|
|
|
// command flags
|
2008-01-22 02:25:34 +03:00
|
|
|
#define B_KDEBUG_DONT_PARSE_ARGUMENTS (0x01)
|
2008-06-01 06:25:00 +04:00
|
|
|
#define B_KDEBUG_PIPE_FINAL_RERUN (0x02)
|
2008-01-22 02:25:34 +03:00
|
|
|
|
2010-03-18 20:20:06 +03:00
|
|
|
|
|
|
|
struct arch_debug_registers;
|
|
|
|
|
|
|
|
|
2008-01-31 15:25:43 +03:00
|
|
|
struct debugger_module_info {
|
|
|
|
module_info info;
|
|
|
|
|
|
|
|
void (*enter_debugger)(void);
|
|
|
|
void (*exit_debugger)(void);
|
2008-02-22 01:41:49 +03:00
|
|
|
|
* Added an emergency key mechanism to the kernel, inspired by Jan Klötzke's
patch from about a year ago (I couldn't use any code of his yet, though,
but there are a few things left). The emergency keys are triggered by
pressing Alt-SysReq + key.
* By default, only Alt-SysReq+'d' is used as a means to deliberately enter
the kernel debugger. F12 belongs to userland again, now :-)
* Debugger add-ons now have another optional method to implement their own
emergency keys - 'd' for the debugger cannot be overridden, though.
* The mechanism can be turned off via a new kernel setting, so it's not that
easy anymore to "crash" Haiku if you don't want to.
* Right now, the PS/2 driver, and the pre-input_server in-kernel debugger
keyboard mini-driver support this, USB not yet.
* Minor cleanup.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@31660 a95241bf-73f2-0310-859d-f6bbb57e9c96
2009-07-21 00:39:37 +04:00
|
|
|
// I/O hooks
|
2008-02-22 01:41:49 +03:00
|
|
|
int (*debugger_puts)(const char *str, int32 length);
|
|
|
|
int (*debugger_getchar)(void);
|
|
|
|
// TODO: add hooks for tunnelling gdb ?
|
* Added an emergency key mechanism to the kernel, inspired by Jan Klötzke's
patch from about a year ago (I couldn't use any code of his yet, though,
but there are a few things left). The emergency keys are triggered by
pressing Alt-SysReq + key.
* By default, only Alt-SysReq+'d' is used as a means to deliberately enter
the kernel debugger. F12 belongs to userland again, now :-)
* Debugger add-ons now have another optional method to implement their own
emergency keys - 'd' for the debugger cannot be overridden, though.
* The mechanism can be turned off via a new kernel setting, so it's not that
easy anymore to "crash" Haiku if you don't want to.
* Right now, the PS/2 driver, and the pre-input_server in-kernel debugger
keyboard mini-driver support this, USB not yet.
* Minor cleanup.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@31660 a95241bf-73f2-0310-859d-f6bbb57e9c96
2009-07-21 00:39:37 +04:00
|
|
|
|
|
|
|
// Misc. hooks
|
|
|
|
bool (*emergency_key_pressed)(char key);
|
2008-01-31 15:25:43 +03:00
|
|
|
};
|
|
|
|
|
2008-10-13 03:50:41 +04:00
|
|
|
struct debugger_demangle_module_info {
|
|
|
|
module_info info;
|
|
|
|
|
|
|
|
const char* (*demangle_symbol)(const char* name, char* buffer,
|
|
|
|
size_t bufferSize, bool* _isObjectMethod);
|
|
|
|
status_t (*get_next_argument)(uint32* _cookie, const char* symbol,
|
|
|
|
char* name, size_t nameSize, int32* _type, size_t* _argumentLength);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-08-12 21:09:42 +04:00
|
|
|
typedef struct debug_page_fault_info {
|
|
|
|
addr_t fault_address;
|
|
|
|
addr_t pc;
|
|
|
|
uint32 flags;
|
|
|
|
} debug_page_fault_info;
|
|
|
|
|
|
|
|
// debug_page_fault_info::flags
|
|
|
|
#define DEBUG_PAGE_FAULT_WRITE 0x01 /* write fault */
|
|
|
|
#define DEBUG_PAGE_FAULT_NO_INFO 0x02 /* fault address and read/write
|
|
|
|
unknown */
|
|
|
|
|
|
|
|
|
2003-11-14 01:07:36 +03:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2007-06-21 09:37:46 +04:00
|
|
|
struct kernel_args;
|
|
|
|
|
2010-05-05 14:05:25 +04:00
|
|
|
extern void debug_init(struct kernel_args *args);
|
|
|
|
extern void debug_init_post_vm(struct kernel_args *args);
|
|
|
|
extern void debug_init_post_settings(struct kernel_args *args);
|
|
|
|
extern void debug_init_post_modules(struct kernel_args *args);
|
2008-10-13 03:50:41 +04:00
|
|
|
extern void debug_early_boot_message(const char *string);
|
|
|
|
extern void debug_puts(const char *s, int32 length);
|
|
|
|
extern bool debug_debugger_running(void);
|
|
|
|
extern bool debug_screen_output_enabled(void);
|
|
|
|
extern void debug_stop_screen_debug_output(void);
|
|
|
|
extern void debug_set_page_fault_info(addr_t faultAddress, addr_t pc,
|
|
|
|
uint32 flags);
|
2008-08-12 21:09:42 +04:00
|
|
|
extern debug_page_fault_info* debug_get_page_fault_info();
|
2009-08-01 05:53:54 +04:00
|
|
|
extern void debug_trap_cpu_in_kdl(int32 cpu, bool returnIfHandedOver);
|
|
|
|
extern void debug_double_fault(int32 cpu);
|
* Added an emergency key mechanism to the kernel, inspired by Jan Klötzke's
patch from about a year ago (I couldn't use any code of his yet, though,
but there are a few things left). The emergency keys are triggered by
pressing Alt-SysReq + key.
* By default, only Alt-SysReq+'d' is used as a means to deliberately enter
the kernel debugger. F12 belongs to userland again, now :-)
* Debugger add-ons now have another optional method to implement their own
emergency keys - 'd' for the debugger cannot be overridden, though.
* The mechanism can be turned off via a new kernel setting, so it's not that
easy anymore to "crash" Haiku if you don't want to.
* Right now, the PS/2 driver, and the pre-input_server in-kernel debugger
keyboard mini-driver support this, USB not yet.
* Minor cleanup.
git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@31660 a95241bf-73f2-0310-859d-f6bbb57e9c96
2009-07-21 00:39:37 +04:00
|
|
|
extern bool debug_emergency_key_pressed(char key);
|
2009-08-03 16:39:56 +04:00
|
|
|
extern bool debug_is_kernel_memory_accessible(addr_t address, size_t size,
|
|
|
|
uint32 protection);
|
|
|
|
extern int debug_call_with_fault_handler(jmp_buf jumpBuffer,
|
|
|
|
void (*function)(void*), void* parameter);
|
2010-04-13 21:40:15 +04:00
|
|
|
extern status_t debug_memcpy(team_id teamID, void* to, const void* from,
|
|
|
|
size_t size);
|
|
|
|
extern ssize_t debug_strlcpy(team_id teamID, char* to, const char* from,
|
|
|
|
size_t size);
|
2003-11-14 01:07:36 +03:00
|
|
|
|
2009-02-22 23:46:27 +03:00
|
|
|
extern char kgetc(void);
|
2008-10-13 03:50:41 +04:00
|
|
|
extern void kputs(const char *string);
|
|
|
|
extern void kputs_unfiltered(const char *string);
|
|
|
|
extern void kprintf_unfiltered(const char *format, ...)
|
|
|
|
__attribute__ ((format (__printf__, 1, 2)));
|
|
|
|
extern void dprintf_no_syslog(const char *format, ...)
|
|
|
|
__attribute__ ((format (__printf__, 1, 2)));
|
2007-10-04 02:14:53 +04:00
|
|
|
|
2008-01-16 03:01:20 +03:00
|
|
|
extern bool is_debug_variable_defined(const char* variableName);
|
|
|
|
extern bool set_debug_variable(const char* variableName, uint64 value);
|
|
|
|
extern uint64 get_debug_variable(const char* variableName,
|
|
|
|
uint64 defaultValue);
|
2008-01-17 00:50:59 +03:00
|
|
|
extern bool unset_debug_variable(const char* variableName);
|
|
|
|
extern void unset_all_debug_variables();
|
2008-01-16 03:01:20 +03:00
|
|
|
|
|
|
|
extern bool evaluate_debug_expression(const char* expression,
|
|
|
|
uint64* result, bool silent);
|
|
|
|
extern int evaluate_debug_command(const char* command);
|
2008-08-22 19:38:57 +04:00
|
|
|
extern status_t parse_next_debug_command_argument(const char** expressionString,
|
|
|
|
char* buffer, size_t bufferSize);
|
2008-01-16 03:01:20 +03:00
|
|
|
|
2008-01-17 02:41:20 +03:00
|
|
|
extern status_t add_debugger_command_etc(const char* name,
|
|
|
|
debugger_command_hook func, const char* description,
|
|
|
|
const char* usage, uint32 flags);
|
|
|
|
extern status_t add_debugger_command_alias(const char* newName,
|
|
|
|
const char* oldName, const char* description);
|
|
|
|
extern bool print_debugger_command_usage(const char* command);
|
2009-03-11 15:17:23 +03:00
|
|
|
extern bool has_debugger_command(const char* command);
|
2008-01-17 02:41:20 +03:00
|
|
|
|
2008-10-13 03:50:41 +04:00
|
|
|
extern const char *debug_demangle_symbol(const char* symbol, char* buffer,
|
|
|
|
size_t bufferSize, bool* _isObjectMethod);
|
|
|
|
extern status_t debug_get_next_demangled_argument(uint32* _cookie,
|
|
|
|
const char* symbol, char* name, size_t nameSize,
|
|
|
|
int32* _type, size_t* _argumentLength);
|
2008-07-23 20:08:16 +04:00
|
|
|
|
2011-01-11 00:54:38 +03:00
|
|
|
extern BKernel_Thread* debug_set_debugged_thread(BKernel_Thread* thread);
|
|
|
|
extern BKernel_Thread* debug_get_debugged_thread();
|
2010-04-13 21:40:15 +04:00
|
|
|
extern bool debug_is_debugged_team(team_id teamID);
|
2010-03-18 20:20:06 +03:00
|
|
|
|
2010-04-13 21:40:15 +04:00
|
|
|
extern struct arch_debug_registers* debug_get_debug_registers(int32 cpu);
|
2008-08-23 03:54:49 +04:00
|
|
|
|
2015-04-21 19:49:37 +03:00
|
|
|
extern status_t _user_kernel_debugger(const char *message);
|
|
|
|
extern void _user_register_syslog_daemon(port_id port);
|
|
|
|
extern void _user_debug_output(const char *userString);
|
2008-08-12 21:09:42 +04:00
|
|
|
|
2003-11-14 01:07:36 +03:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-04-13 21:40:15 +04:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
|
|
|
|
struct DebuggedThreadSetter {
|
2011-01-11 00:54:38 +03:00
|
|
|
DebuggedThreadSetter(Thread* thread)
|
2010-04-13 21:40:15 +04:00
|
|
|
:
|
|
|
|
fPreviousThread(debug_set_debugged_thread(thread))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
~DebuggedThreadSetter()
|
|
|
|
{
|
|
|
|
debug_set_debugged_thread(fPreviousThread);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2011-01-11 00:54:38 +03:00
|
|
|
Thread* fPreviousThread;
|
2010-04-13 21:40:15 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif // __cplusplus
|
|
|
|
|
|
|
|
|
2002-07-18 22:45:07 +04:00
|
|
|
#endif /* _KERNEL_DEBUG_H */
|