📄 hal_diag.c
字号:
/*=============================================================================//// hal_diag.c//// HAL diagnostic output code////=============================================================================//####COPYRIGHTBEGIN####//// -------------------------------------------// The contents of this file are subject to the Cygnus eCos Public License// Version 1.0 (the "License"); you may not use this file except in// compliance with the License. You may obtain a copy of the License at// http://sourceware.cygnus.com/ecos// // Software distributed under the License is distributed on an "AS IS"// basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the// License for the specific language governing rights and limitations under// the License.// // The Original Code is eCos - Embedded Cygnus Operating System, released// September 30, 1998.// // The Initial Developer of the Original Code is Cygnus. Portions created// by Cygnus are Copyright (C) 1998,1999 Cygnus Solutions. All Rights Reserved.// -------------------------------------------////####COPYRIGHTEND####//=============================================================================//#####DESCRIPTIONBEGIN####//// Author(s): nickg// Contributors: nickg// Date: 1998-03-02// Purpose: HAL diagnostic output// Description: Implementations of HAL diagnostic output support.////####DESCRIPTIONEND####////===========================================================================*/#include <pkgconf/system.h>#include <pkgconf/hal.h>#include <pkgconf/hal_sparclite.h>#include <pkgconf/hal_sparclite_sleb.h>#include <cyg/infra/cyg_type.h> // base types#include <cyg/hal/hal_arch.h>#include <cyg/hal/hal_diag.h>#include <cyg/hal/hal_intr.h>#include <cyg/hal/hal_cygm.h>/*---------------------------------------------------------------------------*/#ifdef CYG_KERNEL_DIAG_GDB#ifdef CYG_KERNEL_DIAG_GDB_SERIAL_DIRECT // then force $O packets to serialvoid hal_diag_init(void){ // hal_diag_init_serial();}void hal_diag_write_char_serial( char c ){ HAL_REORDER_BARRIER(); HAL_DIAG_WRITE_CHAR_DIRECT( c ); HAL_REORDER_BARRIER(); HAL_DIAG_WRITE_CHAR_WAIT_FOR_EMPTY(); HAL_REORDER_BARRIER();}void hal_diag_write_char(char c){ static char line[100]; static int pos = 0; // No need to send CRs if( c == '\r' ) return; line[pos++] = c; if( c == '\n' || pos == sizeof(line) ) { CYG_INTERRUPT_STATE old; // Disable interrupts. This prevents GDB trying to interrupt us // while we are in the middle of sending a packet. The serial // receive interrupt will be seen when we re-enable interrupts // later. HAL_DISABLE_INTERRUPTS(old); while(1) { cyg_uint32 status, c1, tries; static char hex[] = "0123456789ABCDEF"; cyg_uint8 csum = 0; int i; hal_diag_write_char_serial('$'); hal_diag_write_char_serial('O'); csum += 'O'; for( i = 0; i < pos; i++ ) { char ch = line[i]; char h = hex[(ch>>4)&0xF]; char l = hex[ch&0xF]; hal_diag_write_char_serial(h); hal_diag_write_char_serial(l); csum += h; csum += l; } hal_diag_write_char_serial('#'); hal_diag_write_char_serial(hex[(csum>>4)&0xF]); hal_diag_write_char_serial(hex[csum&0xF]); // Wait for the ACK character '+' from GDB here and handle // receiving a ^C instead. This is the reason for this clause // being a loop. status = 0; tries = 1000000000; while ( 0 == (HAL_SPARC_86940_FLAG_RXRDY & status) ) { if ( 0 == --tries ) break; HAL_SPARC_86940_SDTR0_STAT_READ( status ); } if ( 0 == tries ) // then we broke out after waiting continue; // the outer loop, send the packet HAL_SPARC_86940_SDTR0_RXDATA_READ( c1 ); // We must ack the interrupt caused by that read to avoid // confusing the GDB stub ROM. HAL_INTERRUPT_ACKNOWLEDGE( CYGNUM_HAL_VECTOR_INTERRUPT_10 ); if( c1 == '+' ) break; // a good acknowledge if( c1 == 3 ) { // Ctrl-C: breakpoint. asm volatile( "ta 2; nop; nop; nop" ); break; } // otherwise, loop round again } pos = 0; // And re-enable interrupts HAL_RESTORE_INTERRUPTS(old); }}#else // CYG_KERNEL_DIAG_GDB_SERIAL_DIRECT not defined; use CygMon// All this code provided by MSalter - ta.struct bsp_comm_procs { void *ch_data; void (*__write)(void *ch_data, const char *buf, int len); int (*__read)(void *ch_data, char *buf, int len); void (*__putc)(void *ch_data, char ch); int (*__getc)(void *ch_data); int (*__control)(void *ch_data, int func, ...);};// This is pointed to by entry BSP_NOTVEC_BSP_COMM_PROCS:typedef struct { int version; /* version number for future expansion */ void *__ictrl_table; void *__exc_table; void *__dbg_vector; void *__kill_vector; struct bsp_comm_procs *__console_procs;} bsp_shared_t;static inthal_bsp_console_write(const char *p, int len){ bsp_shared_t *shared; struct bsp_comm_procs *com; shared = (bsp_shared_t *) (CYGMON_VECTOR_TABLE[ BSP_NOTVEC_BSP_COMM_PROCS ]); com = shared->__console_procs; if (0 != com) { com->__write(com->ch_data, p, len); return 1; } return 0;}static voidhal_dumb_serial_write(const char *p, int len){ int i; for ( i = 0 ; i < len; i++ ) { HAL_DIAG_WRITE_CHAR_DIRECT( p[ i ] ); }} void hal_diag_init(void){}void hal_diag_write_char(char c){ static char line[100]; static int pos = 0; // No need to send CRs if( c == '\r' ) return; line[pos++] = c; if( c == '\n' || pos == sizeof(line) ) { CYG_INTERRUPT_STATE old; // Disable interrupts. This prevents GDB trying to interrupt us // while we are in the middle of sending a packet. The serial // receive interrupt will be seen when we re-enable interrupts // later. HAL_DISABLE_INTERRUPTS(old); if ( ! hal_bsp_console_write( line, pos ) ) // then there is no function registered, just spew it out serial hal_dumb_serial_write( line, pos ); pos = 0; // And re-enable interrupts HAL_RESTORE_INTERRUPTS(old); }}#endif // CYG_KERNEL_DIAG_GDB_SERIAL_DIRECT not defined; use CygMon#else // CYG_KERNEL_DIAG_GDB not defined, so we are going to the serial line // without GDB encoding - likely to be ROM startup./* Address of clock switch */#define CLKSW_ADDR 0x01000003/* Address of SW1 */#define SW1_ADDR 0x02000003void hal_diag_init(void){ cyg_uint32 clk, tval; // first set the baud rate clk = *(unsigned char *)CLKSW_ADDR; if (clk & 0x80) clk = 10; clk = (clk & 0x3f) * 1000000; /* in MHz */ tval = clk / 19200; tval /= 32; tval -= 1; HAL_SPARC_86940_TCR3_WRITE( HAL_SPARC_86940_TCR_CE | HAL_SPARC_86940_TCR_CLKINT | HAL_SPARC_86940_TCR_OUTC3 | HAL_SPARC_86940_TCR_SQWAVE ); HAL_SPARC_86940_RELOAD3_WRITE( tval);#define DELAY(x) \ CYG_MACRO_START int i; for (i = 0; i < x; i++); CYG_MACRO_END HAL_SPARC_86940_SDTR0_CTRL_WRITE( 0 ); DELAY(100); HAL_SPARC_86940_SDTR0_CTRL_WRITE( 0 ); DELAY(100); HAL_SPARC_86940_SDTR0_CTRL_WRITE( 0 ); DELAY(100); HAL_SPARC_86940_SDTR0_CTRL_WRITE( HAL_SPARC_86940_SER_CMD_IRST ); DELAY(100); /* first write after reset is to mode register */ HAL_SPARC_86940_SDTR0_CTRL_WRITE( HAL_SPARC_86940_SER_DIV16_CLK | HAL_SPARC_86940_SER_8BITS | HAL_SPARC_86940_SER_NO_PARITY | HAL_SPARC_86940_SER_STOP1 ); DELAY(100); /* subsequent writes are to command register */ HAL_SPARC_86940_SDTR0_CTRL_WRITE( HAL_SPARC_86940_SER_CMD_RTS | HAL_SPARC_86940_SER_CMD_DTR | HAL_SPARC_86940_SER_CMD_EFR | HAL_SPARC_86940_SER_CMD_RXEN | HAL_SPARC_86940_SER_CMD_TXEN ); DELAY(100);}#endif // CYG_KERNEL_DIAG_GDB/*---------------------------------------------------------------------------*//* End of hal_diag.c */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -