URL
https://opencores.org/ocsvn/openrisc/openrisc/trunk
Subversion Repositories openrisc
[/] [openrisc/] [trunk/] [rtos/] [ecos-3.0/] [packages/] [infra/] [current/] [src/] [fancy.cxx] - Rev 856
Go to most recent revision | Compare with Previous | Blame | View Log
//========================================================================== // // fancy.cxx // // Fancy formatting trace and assert functions // //========================================================================== // ####ECOSGPLCOPYRIGHTBEGIN#### // ------------------------------------------- // This file is part of eCos, the Embedded Configurable Operating System. // Copyright (C) 1998, 1999, 2000, 2001, 2002 Free Software Foundation, Inc. // // eCos 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 or (at your option) any later // version. // // eCos 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 eCos; if not, write to the Free Software Foundation, Inc., // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. // // As a special exception, if other files instantiate templates or use // macros or inline functions from this file, or you compile this file // and link it with other works to produce a work based on this file, // this file does not by itself cause the resulting work to be covered by // the GNU General Public License. However the source code for this file // must still be made available in accordance with section (3) of the GNU // General Public License v2. // // This exception does not invalidate any other reasons why a work based // on this file might be covered by the GNU General Public License. // ------------------------------------------- // ####ECOSGPLCOPYRIGHTEND#### //========================================================================== //#####DESCRIPTIONBEGIN#### // // Author(s): nickg // Contributors: nickg // Date: 1997-12-04 // Purpose: Fancy Trace and assert functions // Description: The functions in this file are fancy implementations of the // standard trace and assert functions. These do printf // style formatting, printing the string and arguments. // //####DESCRIPTIONEND#### // //========================================================================== #include <pkgconf/system.h> #include <pkgconf/infra.h> #ifdef CYGDBG_INFRA_DEBUG_TRACE_ASSERT_FANCY #include <cyg/infra/cyg_type.h> // base types #include <cyg/infra/cyg_trac.h> // tracing macros #include <cyg/infra/cyg_ass.h> // assertion macros #include <pkgconf/hal.h> // HAL configury #include <cyg/infra/diag.h> // HAL polled output #include <cyg/hal/hal_arch.h> // architectural stuff for... #include <cyg/hal/hal_intr.h> // interrupt control #ifdef CYGPKG_KERNEL #include <pkgconf/kernel.h> // kernel configury #include <cyg/kernel/thread.hxx> // thread id to print #include <cyg/kernel/sched.hxx> // ancillaries for above #include <cyg/kernel/thread.inl> // ancillaries for above #endif // ------------------------------------------------------------------------- // Local Configuration: hack me! // these are generally: // if 0, feature is disabled // if 1, feature is enabled, printing is default width // if >1, field is padded up to that width if necessary // (not truncated ever) #define CYG_FILENAME 20 #define CYG_THREADID 0 // DO NOT SET IF NO KERNEL #define CYG_LINENUM 4 #define CYG_FUNCNAME 100 #define CYG_DIAG_PRINTF 1 #define CYG_FUNC_INDENT 2 #ifndef CYGPKG_KERNEL # undef CYG_THREADID # define CYG_THREADID 0 #endif #if CYG_FUNCNAME == 1 #define CYG_FBUF_SIZE 100 #else #define CYG_FBUF_SIZE (CYG_FUNCNAME+20) #endif // ------------------------------------------------------------------------- // Functions to trim file names and function names down to printable lengths // (these are shared between trace and assert functions) #ifdef CYGDBG_USE_TRACING #if 0 static char * tracepremsgs[] = { " INFO:", "ENTER :", "ARGS :", "RETURN:", "bad code" }; #endif static char * tracepremsgs[] = { "'", "{{", "((", "}}", "bad code" }; static char * tracepostmsgs[] = { "'", "", "))", "", "bad code" }; static void write_whattrace( cyg_uint32 what ) { #if CYG_FUNC_INDENT static cyg_uint32 cyg_indent = 0; if ( 3 == what ) cyg_indent -= CYG_FUNC_INDENT; cyg_uint32 i = cyg_indent; for ( ; i > 0; i-- ) diag_write_string( " " ); #endif // CYG_FUNC_INDENT diag_write_string( tracepremsgs[ what > 4 ? 4 : what ] ); #if CYG_FUNC_INDENT if ( 1 == what ) cyg_indent += CYG_FUNC_INDENT; #endif // CYG_FUNC_INDENT } static void write_whattracepost( cyg_uint32 what ) { diag_write_string( tracepostmsgs[ what > 4 ? 4 : what ] ); } #endif // CYGDBG_USE_TRACING static const char *trim_file(const char *file) { #if CYG_FILENAME if ( NULL == file ) file = "<nofile>"; #if 1 == CYG_FILENAME const char *f = file; while( *f ) f++; while( *f != '/' && f != file ) f--; return f==file?f:(f+1); #else static char fbuf2[100]; const char *f = file; char *g = fbuf2; while( *f ) f++; while( *f != '/' && f != file ) f--; if ( f > file ) f++; while( *f ) *g++ = *f++; while( CYG_FILENAME > (g - fbuf2) ) *g++ = ' '; *g = 0; return fbuf2; #endif #else return ""; #endif } static const char *trim_func(const char *func) { #if CYG_FUNCNAME static char fbuf[CYG_FBUF_SIZE]; int i; if ( NULL == func ) func = "<nofunc>"; for( i = 0; func[i] && func[i] != '(' && i < CYG_FBUF_SIZE-4 ; i++ ) fbuf[i] = func[i]; fbuf[i++] = '('; fbuf[i++] = ')'; fbuf[i ] = 0; i=0; #if 1 == CYG_FUNCNAME return &fbuf[i]; #else char *p = &fbuf[i]; while ( *p ) p++; while ( CYG_FUNCNAME > (p - (&fbuf[i])) ) *p++ = ' '; *p = 0; return &fbuf[i]; #endif #else return ""; #endif } void write_lnum( cyg_uint32 lnum) { #if CYG_LINENUM diag_write_char('['); #if 1 < CYG_LINENUM cyg_uint32 i, j; for ( i = 2, j = 100; i < CYG_LINENUM ; i++, j *= 10 ) if ( lnum < j ) diag_write_char(' '); #endif diag_write_dec(lnum); diag_write_char(']'); diag_write_char(' '); #endif } void write_thread_id() { #if CYG_THREADID Cyg_Thread *t = Cyg_Thread::self(); cyg_uint16 tid = 0xFFFF; if( t != NULL ) tid = t->get_unique_id(); diag_write_char('<'); diag_write_hex(tid); diag_write_char('>'); #endif } // ------------------------------------------------------------------------- // Trace functions: #ifdef CYGDBG_USE_TRACING externC void cyg_tracenomsg( const char *psz_func, const char *psz_file, cyg_uint32 linenum ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); diag_write_string("TRACE: "); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; // provide every other one of these as a space/caller bloat compromise. externC void cyg_tracemsg( cyg_uint32 what, const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); if ( NULL == psz_msg ) psz_msg = "<nomsg>"; diag_write_string( "TRACE: " ); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); write_whattrace( what ); diag_write_string(psz_msg); write_whattracepost( what ); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; externC void cyg_tracemsg2( cyg_uint32 what, const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg, CYG_ADDRWORD arg0, CYG_ADDRWORD arg1 ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); if ( NULL == psz_msg ) psz_msg = "<nomsg>"; diag_write_string( "TRACE: " ); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); write_whattrace( what ); #if CYG_DIAG_PRINTF diag_printf( psz_msg, arg0, arg1 ); #else diag_write_string(psz_msg); diag_write_char(' '); diag_write_hex(arg0); diag_write_char(' '); diag_write_hex(arg1); #endif write_whattracepost( what ); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; externC void cyg_tracemsg4( cyg_uint32 what, const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg, CYG_ADDRWORD arg0, CYG_ADDRWORD arg1, CYG_ADDRWORD arg2, CYG_ADDRWORD arg3 ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); if ( NULL == psz_msg ) psz_msg = "<nomsg>"; diag_write_string( "TRACE: " ); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); write_whattrace( what ); #if CYG_DIAG_PRINTF diag_printf( psz_msg, arg0, arg1, arg2, arg3 ); #else diag_write_string(psz_msg); diag_write_char(' '); diag_write_hex(arg0); diag_write_char(' '); diag_write_hex(arg1); diag_write_char(' '); diag_write_hex(arg2); diag_write_char(' '); diag_write_hex(arg3); #endif write_whattracepost( what ); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; externC void cyg_tracemsg6( cyg_uint32 what, const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg, CYG_ADDRWORD arg0, CYG_ADDRWORD arg1, CYG_ADDRWORD arg2, CYG_ADDRWORD arg3, CYG_ADDRWORD arg4, CYG_ADDRWORD arg5 ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); if ( NULL == psz_msg ) psz_msg = "<nomsg>"; diag_write_string( "TRACE: " ); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); write_whattrace( what ); #if CYG_DIAG_PRINTF diag_printf( psz_msg, arg0, arg1, arg2, arg3, arg4, arg5 ); #else diag_write_string(psz_msg); diag_write_char(' '); diag_write_hex(arg0); diag_write_char(' '); diag_write_hex(arg1); diag_write_char(' '); diag_write_hex(arg2); diag_write_char(' '); diag_write_hex(arg3); diag_write_char(' '); diag_write_hex(arg4); diag_write_char(' '); diag_write_hex(arg5); #endif write_whattracepost( what ); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; externC void cyg_tracemsg8( cyg_uint32 what, const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg, CYG_ADDRWORD arg0, CYG_ADDRWORD arg1, CYG_ADDRWORD arg2, CYG_ADDRWORD arg3, CYG_ADDRWORD arg4, CYG_ADDRWORD arg5, CYG_ADDRWORD arg6, CYG_ADDRWORD arg7 ) { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); if ( NULL == psz_msg ) psz_msg = "<nomsg>"; diag_write_string( "TRACE: " ); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); write_whattrace( what ); #if CYG_DIAG_PRINTF diag_printf( psz_msg, arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7 ); #else diag_write_string(psz_msg); diag_write_char(' '); diag_write_hex(arg0); diag_write_char(' '); diag_write_hex(arg1); diag_write_char(' '); diag_write_hex(arg2); diag_write_char(' '); diag_write_hex(arg3); diag_write_char(' '); diag_write_hex(arg4); diag_write_char(' '); diag_write_hex(arg5); diag_write_char(' '); diag_write_hex(arg6); diag_write_char(' '); diag_write_hex(arg7); #endif write_whattracepost( what ); diag_write_char('\n'); DIAG_DEVICE_END_SYNC(); HAL_RESTORE_INTERRUPTS(old_ints); }; // ------------------------------------------------------------------------- externC void cyg_trace_dump(void) { #if defined(CYGPKG_KERNEL) && defined(CYG_DIAG_PRINTF) { diag_printf("\nScheduler:\n\n"); Cyg_Scheduler *sched = &Cyg_Scheduler::scheduler; diag_printf("Lock: %d\n",sched->get_sched_lock() ); # ifdef CYGVAR_KERNEL_THREADS_NAME diag_printf("Current Thread: %s\n",sched->get_current_thread()->get_name()); # else diag_printf("Current Thread: %d\n",sched->get_current_thread()->get_unique_id()); # endif } # ifdef CYGVAR_KERNEL_THREADS_LIST { Cyg_Thread *t = Cyg_Thread::get_list_head(); diag_printf("\nThreads:\n\n"); while( NULL != t ) { cyg_uint32 state = t->get_state(); char tstate[7]; char *tstate1 = "SCUKX"; static char *(reasons[8]) = { "NONE", // No recorded reason "WAIT", // Wait with no timeout "DELAY", // Simple time delay "TIMEOUT", // Wait with timeout/timeout expired "BREAK", // forced break out of sleep "DESTRUCT", // wait object destroyed[note] "EXIT", // forced termination "DONE" // Wait/delay complete }; if( 0 != state ) { // Knock out chars that do not correspond to set bits. for( int i = 0; i < 6 ; i++ ) if( 0 == (state & (1<<i)) ) tstate[i] = ' '; else tstate[i] = tstate1[i]; tstate[6] = 0; } else tstate[0] = 'R', tstate[1] = 0; # ifdef CYGVAR_KERNEL_THREADS_NAME diag_printf( "%20s pri = %3d state = %6s id = %3d\n", t->get_name(), t->get_priority(), tstate, t->get_unique_id() ); # else diag_printf( "Thread %3d pri = %3d state = %6s\n", t->get_unique_id(), t->get_priority(), tstate ); # endif diag_printf( "%20s stack base = %08x ptr = %08x size = %08x\n", "", t->get_stack_base(), #ifdef CYGDBG_KERNEL_DEBUG_GDB_THREAD_SUPPORT t->get_saved_context(), #else 0, #endif t->get_stack_size() ); diag_printf( "%20s sleep reason %8s wake reason %8s\n", "", reasons[t->get_sleep_reason()], reasons[t->get_wake_reason()] ); diag_printf( "%20s queue = %08x wait info = %08x\n", "", t->get_current_queue(), t->get_wait_info() ); diag_printf("\n"); t = t->get_list_next(); } } # endif // CYGVAR_KERNEL_THREADS_LIST #endif // CYG_DIAG_PRINTF } #endif // CYGDBG_USE_TRACING // ------------------------------------------------------------------------- // Assert functions: #ifdef CYGDBG_USE_ASSERTS externC void cyg_assert_fail( const char *psz_func, const char *psz_file, cyg_uint32 linenum, const char *psz_msg ) __THROW { cyg_uint32 old_ints; HAL_DISABLE_INTERRUPTS(old_ints); DIAG_DEVICE_START_SYNC(); diag_write_string("ASSERT FAIL: "); write_thread_id(); diag_write_string(trim_file(psz_file)); write_lnum(linenum); diag_write_string(trim_func(psz_func)); diag_write_char(' '); diag_write_string(psz_msg); diag_write_char('\n'); #ifdef CYGHWR_TEST_PROGRAM_EXIT CYGHWR_TEST_PROGRAM_EXIT(); #endif for(;;); // DIAG_DEVICE_END_SYNC(); // HAL_RESTORE_INTERRUPTS(old_ints); }; extern "C" { extern unsigned long _stext; extern unsigned long _etext; unsigned long stext_addr = (unsigned long)&_stext; unsigned long etext_addr = (unsigned long)&_etext; }; externC cyg_bool cyg_check_data_ptr(const void *ptr) { unsigned long p = (unsigned long)ptr; if( p == 0 ) return false; if( p > stext_addr && p < etext_addr ) return false; return true; } externC cyg_bool cyg_check_func_ptr(void (*ptr)(void)) { unsigned long p = (unsigned long)ptr; if( p == 0 ) return false; if( p < stext_addr && p > etext_addr ) return false; return true; } #endif // CYGDBG_USE_ASSERTS #endif // CYGDBG_INFRA_DEBUG_TRACE_ASSERT_FANCY // ------------------------------------------------------------------------- // EOF fancy.cxx
Go to most recent revision | Compare with Previous | Blame | View Log