/* ****************************************************************************** ****************************************************************************** * * COPYRIGHT (C) by EMC Corporation, 1997 All rights reserved. * $Id$ * DESCRIPTION: This module has been provided for the purpose of testing GDB. * * NOTES: * ****************************************************************************** *****************************************************************************/ /*============================================================================= * INCLUDE FILES *===========================================================================*/ #ifdef DO_IT_BY_THE_BOOK #include "symtypes_defs.h" #include "printp.h" #include "adbg_expression.h" #include "common_hw_ds.h" #include "common_hw_defs.h" #include "evnttrac.h" #include "sym_scratch_ds.h" #include "symglob_ds.h" #include "sym_protglob_ds.h" #include "ether.h" #include <ctype.h> #else #include "adbg_dtc.h" #define YES 1 #define NO 0 #define TRUE 1 #define FALSE 0 #define ENABLED 1 #define DISABLED 0 #define CONTROL_C 3 /* ASCII 'ETX' */ /* * Faked after ctype.h */ #define isxdigit(X) (((X) >= '0' && (X) <= '9') || \ ((X) >= 'A' && (X) <= 'F') || \ ((X) >= 'a' && (X) <= 'f')) /* * Borrowed from string.h */ extern unsigned int strlen ( const char * ); /* * Extracted from symtypes.h: */ typedef char BOOL; /* 8 Bits */ typedef unsigned char UCHAR; /* 8 Bits */ typedef unsigned short USHORT; /* 16 Bits */ typedef unsigned long ULONG; /* 32 Bits */ /* * for struct t_expr_tag and * decl of build_and_add_expression */ #include "adbg_expression.h" #define NULL 0 /* * Extracted from printp.h: */ extern void printp ( const char * fptr, ... ); extern void sprintp ( const char * fptr, ... ); /* * Extracted from ether.h: */ extern long eth_to_gdb ( UCHAR *buf, long length ); /* * Derived from hwequs.s: */ #define CS_CODE_START 0x100000 #define CS_CODE_SIZE 0x200000 #define LAST_CS_WORD (CS_CODE_START + CS_CODE_SIZE - 2) #define sh_genstat1 (*((volatile ULONG *) 0xFFFFFE54)) #define rs232_mode1 0 /* rs-232 mode 1 reg. */ #define rs232_mode2 rs232_mode1 /* rs-232 mode 2 reg. */ #define rs232_stat 4 /* rs-232 status reg. */ #define rs232_clk rs232_stat /* rs-232 clock select reg. */ #define rs232_cmd 8 /* rs-232 command reg */ #define rs232_transmit 12 /* rs-232 transmit reg. */ #define rs232_receive rs232_transmit /* rs-232 transmit reg. */ #define rs232_aux 16 /* rs-232 aux control reg. */ #define rs232_isr 20 /* rs-232 interrupt status reg. */ #define rs232_imr rs232_isr /* rs-232 interrupt mask reg. */ #define rs232_tc_high 24 /* rs-232 timer/counter high reg. */ #define rs232_tc_low 28 /* rs-232 timer/counter low reg. */ #endif /*============================================================================ * MODULE DEFINES *===========================================================================*/ #define P_RST_LAN_UART_REG ((volatile UCHAR *) 0xFFFFFE45) #define M_RST_LAN_UART 0x80 /* Bit 7 */ #define P_LAN0TR_REG P_RST_LAN_UART_REG #define M_LAN0TR 0x20 /* Bit 5 */ #define M_SH_GENCON_LAN0TR 0x00200000 /* Bit 21 */ #define MAX_RS232_CHARS 512 #define LAN_Q_MOD(X) ((X) % MAX_RS232_CHARS) /*---------------------------------------* * LAN UART Registers * *---------------------------------------*/ #define LAN_UART_BASE ((ULONG) 0xfffffc22) /* Write-Read */ #define P_LAN_MR1 ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_mode1 ))) #define P_LAN_MR2 ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_mode2 ))) /* Write-Only */ #define P_LAN_ACR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_aux ))) #define P_LAN_CR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_cmd ))) #define P_LAN_CSR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_clk ))) #define P_LAN_CTLR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_tc_low ))) #define P_LAN_CTUR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_tc_high ))) #define P_LAN_IMR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_imr ))) /* Read-Only */ #define P_LAN_SR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_stat ))) #define P_LAN_ISR ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_isr ))) #define P_LAN_XMT ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_transmit))) #define P_LAN_RCV ((volatile UCHAR *) (LAN_UART_BASE + ((ULONG) rs232_receive ))) /* * Bit Values for Write-Read and Write-Only Registers */ #define DEFAULT_LAN_MR1 ((UCHAR) 0x13) #define DEFAULT_LAN_MR2 ((UCHAR) 0x07) #define DEFAULT_LAN_CSR ((UCHAR) 0xcc) #define DEFAULT_LAN_ACR ((UCHAR) 0x38) #define DEFAULT_LAN_CTUR ((UCHAR) 0xff) #define DEFAULT_LAN_CTLR ((UCHAR) 0xff) #define LAN_ACR_SELECT_BRG_0 DEFAULT_LAN_ACR #define LAN_ACR_SELECT_BRG_1 (DEFAULT_LAN_ACR | 0x80) #define UART_CR_RESET_MR_PTR ((UCHAR) 0x10) /* Reset MR pointer (points to MR1). */ #define UART_CR_RESET_RVCR ((UCHAR) 0x20) /* Reset receiver (disabled). */ #define UART_CR_RESET_XMTR ((UCHAR) 0x30) /* Reset transmitter (disabled). */ #define UART_CR_RESET_ERROR_STATUS ((UCHAR) 0x40) /* Reset error status. */ #define UART_CR_RESET_BRK_CHG_INT ((UCHAR) 0x50) /* Reset break change interrupt. */ #define UART_CR_START_CNTR_TIMER ((UCHAR) 0x80) /* Start counter/timer. */ #define UART_CR_STOP_CNTR ((UCHAR) 0x90) /* Stop counter. */ #define UART_CR_DISABLE_XMTR ((UCHAR) 0x08) /* Disable transmitter. */ #define UART_CR_ENABLE_XMTR ((UCHAR) 0x04) /* Enable transmitter. */ #define UART_CR_DISABLE_RCVR ((UCHAR) 0x02) /* Disable receiver. */ #define UART_CR_ENABLE_RCVR ((UCHAR) 0x01) /* Enable receiver. */ #define UART_CSR_BR_4800 ((UCHAR) 0x99) /* With either BRG Set selected (via ACR). */ #define UART_CSR_BR_9600 ((UCHAR) 0xbb) /* With either BRG Set selected (via ACR). */ #define UART_CSR_BR_19200 ((UCHAR) 0xcc) /* With BRG Set '1' selected (via ACR). */ #define UART_CSR_BR_38400 ((UCHAR) 0xcc) /* With BRG Set '0' selected (via ACR). */ #define UART_IMR_RxRDY ((UCHAR) 0x04) /* Enable 'RxRDY' interrupt. */ #define UART_IMR_TxEMT ((UCHAR) 0x02) /* Enable 'TxEMT' interrupt. */ #define UART_IMR_TxRDY ((UCHAR) 0x01) /* Enable 'TxRDY' interrupt. */ /* * Bit Masks for Read-Only Registers */ #define M_UART_SR_RCVD_BRK 0x80 /* Bit 7 */ #define M_UART_SR_FE 0x40 /* Bit 6 */ #define M_UART_SR_PE 0x20 /* Bit 5 */ #define M_UART_SR_OE 0x10 /* Bit 4 */ #define M_UART_SR_TxEMT 0x08 /* Bit 3 */ #define M_UART_SR_TxRDY 0x04 /* Bit 2 */ #define M_UART_SR_FFULL 0x02 /* Bit 1 */ #define M_UART_SR_RxRDY 0x01 /* Bit 0 */ #define M_UART_ISR_RxRDY 0x04 /* Bit 2 */ #define M_UART_ISR_TxEMT 0x02 /* Bit 1 */ #define M_UART_ISR_TxRDY 0x01 /* Bit 0 */ /*---------------------------------------* * Support for 'Utility 83'. * *---------------------------------------*/ #define LAN_UTIL_CODE 0x83 #define LAN_INIT ((ULONG) (('I' << 24) | ('N' << 16) | ('I' << 8) | 'T')) #define LAN_BAUD ((ULONG) (('B' << 24) | ('A' << 16) | ('U' << 8) | 'D')) #define LAN_INTR ((ULONG) (('I' << 24) | ('N' << 16) | ('T' << 8) | 'R')) #define LAN_XMT ((ULONG) (('X' << 16) | ('M' << 8) | 'T')) #define LAN_ECHO ((ULONG) (('E' << 24) | ('C' << 16) | ('H' << 8) | 'O')) #define LAN_STAT ((ULONG) (('S' << 24) | ('T' << 16) | ('A' << 8) | 'T')) #define LAN_IN ((ULONG) (('I' << 8) | 'N')) #define LAN_OUT ((ULONG) (('O' << 16) | ('U' << 8) | 'T')) #define LAN_PUTC ((ULONG) (('P' << 24) | ('U' << 16) | ('T' << 8) | 'C')) #define LAN_WPM ((ULONG) (('W' << 16) | ('P' << 8) | 'M')) #define STATUS(X) ( ( ( X ) == 0 ) ? "disabled" : "enabled" ) #define XMT_VIA_BP_ENABLED() ( *P_LAN0TR_REG & M_LAN0TR ? 1 : 0 ) #define TRAP_1_INST 0x4E41 /* * Bit #13 of shared genstat 1 indicates * which processor we are as follows. * * 0 => X (side A) * 1 => Y (side B) */ #define M_PROC_ID 0x00002000 #define IS_SIDE_A() ( ( (sh_genstat1) & M_PROC_ID ) == 0 ) #define IS_SIDE_B() ( (sh_genstat1) & M_PROC_ID ) #ifdef STANDALONE /* Compile this module stand-alone for debugging */ #define LAN_PUT_CHAR(X) printf("%c", X) #else #define LAN_PUT_CHAR(X) while ( lan_put_char( X ) ) #endif #define VIA_RS232 0 #define VIA_ETHERNET 1 #define MAX_IO_BUF_SIZE 400 #define MAX_BYTE_CODES 200 /* maximum length for bytecode string */ static ULONG gdb_host_comm; static ULONG gdb_cat_ack; static char eth_outbuffer[ MAX_IO_BUF_SIZE + 1 ]; #ifdef STANDALONE #define ACK_PKT() LAN_PUT_CHAR( '+' ) #define NACK_PKT() LAN_PUT_CHAR( '-' ) #else #define ACK_PKT() { \ if ( VIA_ETHERNET == gdb_host_comm ) \ { \ gdb_cat_ack = YES; \ } \ else \ { \ LAN_PUT_CHAR( '+' ); \ } \ } #define NACK_PKT() { \ if ( VIA_ETHERNET == gdb_host_comm ) \ { \ eth_outbuffer[ 0 ] = '-'; \ eth_to_gdb( (UCHAR *) eth_outbuffer, 1 ); \ } \ else \ { \ LAN_PUT_CHAR( '-' ); \ } \ } #endif /*============================================================================ * MODULE TYPEDEFS *===========================================================================*/ typedef struct rs232_queue { long head_index; long tail_index; ULONG overflows; long gdb_packet_start; long gdb_packet_end; long gdb_packet_csum1; long gdb_packet_csum2; UCHAR buf[ MAX_RS232_CHARS ]; } T_RS232_QUEUE; /*============================================================================= * EXTERNAL GLOBAL VARIABLES *===========================================================================*/ extern volatile UCHAR sss_trace_flag; /*============================================================================= * STATIC MODULE DECLARATIONS *===========================================================================*/ static T_RS232_QUEUE lan_input_queue, lan_output_queue; static BOOL test_echo; #if 0 /* The stub no longer seems to use this. */ static BOOL write_access_enabled; #endif static int baud_rate_idx; static ULONG tx_by_intr, tx_by_poll; static UCHAR lan_shadow_imr; /*============================================================================= * EXTERNAL FUNCTION PROTOTYPES *===========================================================================*/ extern long write_to_protected_mem( void *address, unsigned short value ); /*============================================================================= * MODULE GLOBAL FUNCTIONS PROTOTYPES *===========================================================================*/ ULONG gdb_c_test( ULONG *parm ); void lan_init( void ); void lan_isr( void ); long lan_get_char( void ); long lan_put_char( UCHAR c ); ULONG lan_util( ULONG *parm ); /*============================================================================= * MODULE LOCAL FUNCTION PROTOTYPES *===========================================================================*/ static void lan_reset( void ); static void lan_configure( void ); static void lan_init_queue( T_RS232_QUEUE *p_queue ); static void lan_add_to_queue( long c, T_RS232_QUEUE *p_queue ); static UCHAR lan_next_queue_char( T_RS232_QUEUE *p_queue ); static void lan_util_menu( void ); static long get_gdb_input( long c, T_RS232_QUEUE *p_input_q ); /*============================================================================= * GDB STUB FUNCTION PROTOTYPES *===========================================================================*/ void gdb_trap_1_handler( void ); void gdb_trace_handler ( void ); void gdb_get_eth_input( unsigned char *buf, long length ); static void getpacket ( void ); static void putpacket ( char * ); static void discard_packet ( void ); #ifdef STANDALONE /* Compile this module stand-alone for debugging */ #include <stdio.h> #define printp printf /* easier than declaring a local varargs stub func. */ #endif /* STANDALONE */ /*============================================================================= * MODULE BODY *===========================================================================*/ /* ------------------- Things that belong in a header file --------------- */ extern char *memset (char *, int, int); /*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%* * * * Global Module Functions * * * *%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/ static char gdb_char_test; static short gdb_short_test; static long gdb_long_test; static char gdb_arr_test[25]; static struct GDB_STRUCT_TEST { char c; short s; long l; int bfield : 11; /* collect bitfield */ char arr[25]; struct GDB_STRUCT_TEST *next; } gdb_struct1_test, gdb_struct2_test, *gdb_structp_test, **gdb_structpp_test; static union GDB_UNION_TEST { char c; short s; long l; int bfield : 11; /* collect bitfield */ char arr[4]; union GDB_UNION_TEST *next; } gdb_union1_test; void gdb_recursion_test (int, int, int, int, int, int, int); void gdb_recursion_test (int depth, int q1, int q2, int q3, int q4, int q5, int q6) { /* gdb_recursion_test line 0 */ int q = q1; /* gdbtestline 1 */ q1 = q2; /* gdbtestline 2 */ q2 = q3; /* gdbtestline 3 */ q3 = q4; /* gdbtestline 4 */ q4 = q5; /* gdbtestline 5 */ q5 = q6; /* gdbtestline 6 */ q6 = q; /* gdbtestline 7 */ if (depth--) /* gdbtestline 8 */ gdb_recursion_test (depth, q1, q2, q3, q4, q5, q6); /* gdbtestline 9 */ } ULONG gdb_c_test( ULONG *parm ) { char *p = "gdb_c_test"; char *ridiculously_long_variable_name_with_equally_long_string_assignment; register long local_reg = 7; static unsigned long local_static, local_static_sizeof; long local_long; unsigned long *stack_ptr; unsigned long end_of_stack; ridiculously_long_variable_name_with_equally_long_string_assignment = "ridiculously long variable name with equally long string assignment"; local_static = 9; local_static_sizeof = sizeof (struct GDB_STRUCT_TEST); local_long = local_reg + 1; stack_ptr = (unsigned long *) &local_long; end_of_stack = (unsigned long) &stack_ptr + sizeof(stack_ptr) + sizeof(end_of_stack) - 1; printp ("\n$Id$\n"); printp( "%s: arguments = %X, %X, %X, %X, %X, %X\n", p, parm[ 1 ], parm[ 2 ], parm[ 3 ], parm[ 4 ], parm[ 5 ], parm[ 6 ] ); gdb_char_test = gdb_struct1_test.c = (char) ((long) parm[1] & 0xff); gdb_short_test = gdb_struct1_test.s = (short) ((long) parm[2] & 0xffff); gdb_long_test = gdb_struct1_test.l = (long) ((long) parm[3] & 0xffffffff); gdb_union1_test.l = (long) parm[4]; gdb_arr_test[0] = gdb_struct1_test.arr[0] = (char) ((long) parm[1] & 0xff); gdb_arr_test[1] = gdb_struct1_test.arr[1] = (char) ((long) parm[2] & 0xff); gdb_arr_test[2] = gdb_struct1_test.arr[2] = (char) ((long) parm[3] & 0xff); gdb_arr_test[3] = gdb_struct1_test.arr[3] = (char) ((long) parm[4] & 0xff); gdb_arr_test[4] = gdb_struct1_test.arr[4] = (char) ((long) parm[5] & 0xff); gdb_arr_test[5] = gdb_struct1_test.arr[5] = (char) ((long) parm[6] & 0xff); gdb_struct1_test.bfield = 144; gdb_struct1_test.next = &gdb_struct2_test; gdb_structp_test = &gdb_struct1_test; gdb_structpp_test = &gdb_structp_test; gdb_recursion_test (3, (long) parm[1], (long) parm[2], (long) parm[3], (long) parm[4], (long) parm[5], (long) parm[6]); gdb_char_test = gdb_short_test = gdb_long_test = 0; gdb_structp_test = (void *) 0; gdb_structpp_test = (void *) 0; memset ((char *) &gdb_struct1_test, 0, sizeof (gdb_struct1_test)); memset ((char *) &gdb_struct2_test, 0, sizeof (gdb_struct2_test)); local_static_sizeof = 0; local_static = 0; return ( (ULONG) 0 ); } /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_init * * * DESCRIPTION: * * * RETURN VALUE: * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * * *---------------------------------------------------------------------------*/ void lan_init( void ) { if ( IS_SIDE_A( ) ) { lan_reset( ); lan_init_queue( &lan_input_queue ); lan_init_queue( &lan_output_queue ); lan_configure( ); } return; } /* end of 'lan_init' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_isr * * * DESCRIPTION: * * * RETURN VALUE: None. * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * *---------------------------------------------------------------------------*/ void lan_isr( void ) { UCHAR c; lan_shadow_imr = 0; /* Disable all UART interrupts. */ *P_LAN_IMR = lan_shadow_imr; if ( *P_LAN_ISR & M_UART_ISR_RxRDY ) { gdb_host_comm = VIA_RS232; c = *P_LAN_RCV; if ( test_echo ) { /* ????? */ } if ( c == CONTROL_C ) { /* can't stop the target, but we can tell gdb to stop waiting... */ discard_packet( ); putpacket( "S03" ); /* send back SIGINT to the debugger */ } else { lan_add_to_queue( (long) c, &lan_input_queue ); get_gdb_input( (long) c, &lan_input_queue ); } } if ( XMT_VIA_BP_ENABLED( ) ) { c = 0; while ( (*P_LAN_ISR & M_UART_ISR_TxRDY) && (c = lan_next_queue_char( &lan_output_queue )) ) { *P_LAN_XMT = c; ++tx_by_intr; } if ( c ) { lan_shadow_imr |= UART_IMR_TxRDY; /* (Re-)Enable 'TxRDY' interrupt from UART. */ } } lan_shadow_imr |= UART_IMR_RxRDY; /* Re-Enable 'RxRDY' interrupt from UART. */ *P_LAN_IMR = lan_shadow_imr; return; } /* end of 'lan_isr' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_get_char * * * DESCRIPTION: Fetches a character from the UART. * * * RETURN VALUE: 0 on success, -1 on failure. * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * *---------------------------------------------------------------------------*/ long lan_get_char( void ) { long status = -2; /* AGD: nothing found in rcv buffer */ if ( *P_LAN_SR & M_UART_SR_RxRDY ) { char c = (char) *P_LAN_RCV; if ( test_echo ) { LAN_PUT_CHAR ( c ); } if ( c == CONTROL_C ) { /* can't stop the target, but we can tell gdb to stop waiting... */ discard_packet( ); putpacket( "S03" ); /* send back SIGINT to the debugger */ status = 0; /* success */ } else { lan_add_to_queue( (long) c, &lan_input_queue ); status = get_gdb_input( (long) c, &lan_input_queue ); } } return( status ); } /* end of 'lan_get_char' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_put_char * * DESCRIPTION: Puts a character out via the UART. * * RETURN VALUE: 0 on success, -1 on failure. * * USED GLOBAL VARIABLES: none. * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * NOTES: !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! * !! !! * !! If 'XMT_VIA_BP_ENABLED()' is FALSE then output is THROWN AWAY. !! * !! This prevents anyone infinite-looping on this function. !! * !! !! * !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! * *---------------------------------------------------------------------------*/ long lan_put_char( UCHAR c ) { long status = -1; if ( XMT_VIA_BP_ENABLED( ) ) { if ( *P_LAN_SR & M_UART_SR_TxRDY ) { lan_add_to_queue( (long) c, &lan_output_queue ); c = lan_next_queue_char( &lan_output_queue ); *P_LAN_XMT = c; ++tx_by_poll; status = 0; } #if 0 else { status = 0; lan_shadow_imr |= UART_IMR_TxRDY; /* Enable 'TxRDY' interrupt from UART. */ *P_LAN_IMR = lan_shadow_imr; } #endif } else { status = 0; /* You lose: input character goes to the bit bucket. */ } return( status ); } /* end of 'lan_put_char' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_util * * DESCRIPTION: * * RETURN VALUE: * * USED GLOBAL VARIABLES: * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * NOTES: * *---------------------------------------------------------------------------*/ ULONG lan_util( ULONG *parm ) { static const struct { ULONG rate_code; UCHAR acr_setting; UCHAR csr_setting; } baud_rate_setting [] = { { 0x38400, LAN_ACR_SELECT_BRG_0, UART_CSR_BR_38400 }, { 0x19200, LAN_ACR_SELECT_BRG_1, UART_CSR_BR_19200 }, { 0x9600, LAN_ACR_SELECT_BRG_0, UART_CSR_BR_9600 }, { 0x4800, LAN_ACR_SELECT_BRG_0, UART_CSR_BR_4800 } }; #define BOGUS_P1 0xE1 #define BOGUS_P2 0xE2 ULONG not_done_code; ULONG opcode; ULONG parm_1; ULONG parm_2; int i; UCHAR c; not_done_code = 0; opcode = parm[ 1 ]; parm_1 = parm[ 2 ]; parm_2 = parm[ 3 ]; switch ( opcode ) { case LAN_INIT: { lan_init( ); printp( "\n\n Interface (Re)Initialized ...\n\n" ); break; } case LAN_BAUD: { for ( i = 0; i < (int)(sizeof(baud_rate_setting) / sizeof(baud_rate_setting[0])); i ++ ) { if ( baud_rate_setting[i].rate_code == parm_1 ) { baud_rate_idx = i; *P_LAN_ACR = baud_rate_setting[i].acr_setting; *P_LAN_CSR = baud_rate_setting[i].csr_setting; printp ( "Baud rate set to %X!\n", baud_rate_setting[i].rate_code ); return( not_done_code ); } } printp( "\n\n *** SYNTAX Error - Invalid baudrate (P2)\n\n" ); not_done_code = BOGUS_P2; break; } case LAN_INTR: { switch ( parm_1 ) { case 0x0D: /* Disable 'RxRDY' Interrupts */ { lan_shadow_imr &= ~UART_IMR_RxRDY; *P_LAN_IMR = lan_shadow_imr; printp( "\n\n Receive Ready Interrupts DISABLED ...\n\n" ); break; } case 0x0E: /* Enable 'RxRDY' Interrupts */ { lan_shadow_imr |= UART_IMR_RxRDY; *P_LAN_IMR = lan_shadow_imr; printp( "\n\n Receive Ready Interrupts ENABLED ...\n\n" ); break; } default: { printp( "\n\n *** SYNTAX Error - Invalid P2 (use D or E)\n\n" ); not_done_code = BOGUS_P2; } } break; } case LAN_XMT: { switch ( parm_1 ) { case 0x0E: /* Enable Transmission-via-Backplane */ { if ( !(*P_LAN0TR_REG & M_LAN0TR) ) { *P_LAN0TR_REG |= M_LAN0TR; /* 0 -> 1 */ } printp( "\n\n Transmit-via-Backplane ENABLED ...\n\n" ); break; } case 0x0D: /* Disable Transmission-via-Backplane */ { if ( *P_LAN0TR_REG & M_LAN0TR ) { *P_LAN0TR_REG &= ~M_LAN0TR; /* 1 -> 0 */ } printp( "\n\n Transmit-via-Backplane DISABLED ...\n\n" ); break; } default: { printp( "\n\n *** SYNTAX Error - Invalid P2 (use D or E)\n\n" ); not_done_code = BOGUS_P2; lan_util_menu( ); } } break; } case LAN_STAT: { printp( "\n -- Status --\n\n" ); printp( " Baud Rate: %X *\n", baud_rate_setting[ baud_rate_idx ].rate_code ); printp( " Xmt-via-BP: %s *\n", STATUS( XMT_VIA_BP_ENABLED( ) ) ); printp( " RxRdy Intr: %s *\n", STATUS( (lan_shadow_imr & M_UART_ISR_RxRDY) ) ); /*** printp( " TxRdy Intr: %s\n", STATUS( (lan_shadow_imr & M_UART_ISR_TxRDY) ) ); ***/ printp( " Echo: %s *\n\n", STATUS( test_echo ) ); printp( " IMR: %02X\n", (ULONG) lan_shadow_imr ); printp( " ISR: %02X\n", (ULONG) *P_LAN_ISR ); printp( " SR: %02X\n\n", (ULONG) *P_LAN_SR ); printp( " Input Overflows: %d\n\n", lan_input_queue.overflows ); printp( " Tx by Intr: %d\n", tx_by_intr ); printp( " Tx by Poll: %d\n\n", tx_by_poll ); printp( " * Can be set or toggled via Utility %2X.\n\n", (ULONG) LAN_UTIL_CODE ); break; } case LAN_IN: { switch ( parm_1 ) { case 0x0C: /* Clear and Reset Queue */ { lan_init_queue( &lan_input_queue ); printp( "\n\n Queue CLEARED/RESET ...\n\n" ); break; } case 0x0D: /* Display Queue */ { printp( "\n -- Input Queue --\n" ); printp( "\n Head Index: %8X Tail Index: %8X\n\n ", (ULONG) lan_input_queue.head_index, (ULONG) lan_input_queue.tail_index ); for ( i = 0; i < MAX_RS232_CHARS; ++i ) { printp( " %02X", (ULONG) lan_input_queue.buf[ i ] ); if ( 15 == (i % 16) ) { int j; printp ( " " ); for ( j = i - 15; j <= i; j++ ) { if ( lan_input_queue.buf[ j ] >= ' ' && lan_input_queue.buf[ j ] < 127 ) printp ( "%c", lan_input_queue.buf[ j ] ); else printp ( "." ); } printp( "\n " ); } else if ( 7 == (i % 8) ) { printp( " " ); } } printp( "\n" ); break; } case 0x0F: /* Fetch next character in Queue */ { c = lan_next_queue_char( &lan_input_queue ); if ( c ) { printp( "\n\n Next Character: " ); if ( 0x21 <= c && c <= 0x7F ) { printp( "%c\n\n", (ULONG) c ); } else if ( 0x20 == ((UCHAR) c) ) { printp( "<space>\n\n" ); } else { printp( "%02X\n\n", (ULONG) c ); } } else { printp( "\n\n Input Queue EMPTY ...\n\n" ); } break; } default: { printp( "\n\n *** SYNTAX Error - Invalid P2 ...\n\n" ); not_done_code = BOGUS_P2; break; } } break; } case LAN_OUT: { switch ( parm_1 ) { case 0x0C: /* Clear and Reset Queue */ { lan_init_queue( &lan_output_queue ); printp( "\n\n Queue CLEARED/RESET ...\n\n" ); break; } case 0x0D: /* Display Queue */ { printp( "\n -- Output Queue --\n" ); printp( "\n Head Index: %8X Tail Index: %8X\n\n ", (ULONG) lan_output_queue.head_index, (ULONG) lan_output_queue.tail_index ); for ( i = 0; i < MAX_RS232_CHARS; ++i ) { printp( " %02X", (ULONG) lan_output_queue.buf[ i ] ); if ( 15 == (i % 16) ) { int j; printp ( " " ); for ( j = i - 15; j <= i; j++ ) { if ( lan_output_queue.buf[ j ] >= ' ' && lan_output_queue.buf[ j ] < 127 ) printp ( "%c", lan_output_queue.buf[ j ] ); else printp ( "." ); } printp( "\n " ); } else if ( 7 == (i % 8) ) { printp( " " ); } } printp( "\n" ); break; } case 0x0F: /* Fetch next character in Queue */ { c = lan_next_queue_char( &lan_output_queue ); if ( c ) { printp( "\n\n Next Character: " ); if ( 0x21 <= c && c <= 0x7F ) { printp( "%c\n\n", (ULONG) c ); } else if ( 0x20 == c ) { printp( "<space>\n\n" ); } else { printp( "%02X\n\n", (ULONG) c ); } } else { printp( "\n\n Input Queue EMPTY ...\n\n" ); } break; } default: { printp( "\n\n *** SYNTAX Error - Invalid P2 ...\n\n" ); not_done_code = BOGUS_P2; break; } } break; } case LAN_ECHO: { switch ( parm_1 ) { case 0x0E: { test_echo = ENABLED; printp( "\n\n Test echo ENABLED ...\n\n" ); break; } case 0x0D: { test_echo = DISABLED; printp( "\n\n Test echo DISABLED ...\n\n" ); break; } default: { printp( "\n\n *** SYNTAX Error - Invalid P2 ...\n\n" ); not_done_code = BOGUS_P2; break; } } break; } case LAN_PUTC: { if ( 0x20 < parm_1 && parm_1 < 0x7F ) { if ( lan_put_char( (UCHAR) parm_1 ) ) { printp( "\n\n *** 'lan_put_char' Error ...\n" ); } else { printp( "\n\n O.K. ...\n" ); } } else { printp( "\n\n *** Error - character must be in the 0x21-0x7E range ...\n" ); not_done_code = BOGUS_P2; } break; } /*** case LAN_WPM: { if ( write_to_protected_mem( (void *) parm_1, (unsigned short) parm_2 ) ) { printp( "\n Write to protected memory FAILED ...\n" ); } break; } ***/ case 0: /* no argument -- print menu */ { lan_util_menu( ); break; } default: { parm_2 = 0; /* to supress compiler warning with 'LAN_WPM' case disabled */ printp( "\n\n *** SYNTAX Error - Invalid P1 ...\n\n" ); not_done_code = BOGUS_P1; break; } } /* End of 'switch ( opcode )'. */ return( not_done_code ); } /* end of 'lan_util' *===========================================================================*/ /*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%* * * * Local Module Functions * * * *%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_reset * * DESCRIPTION: Resets the LAN UART by strobing the 'RST_LAN_UART' bit in the * Shared Control 1 area. * * 1 _| ______ * | | | * Bit | | | * | | | * 0 _|______| |______ * |---------------------> t * * RETURN VALUE: None. * * USED GLOBAL VARIABLES: * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * NOTES: H/W configuration requires that a byte in the shared * control 1 area must be read before being written. * *---------------------------------------------------------------------------*/ static void lan_reset( void ) { while ( *P_RST_LAN_UART_REG & M_RST_LAN_UART ) { *P_RST_LAN_UART_REG &= ~M_RST_LAN_UART; /* 0 */ } while ( !(*P_RST_LAN_UART_REG & M_RST_LAN_UART) ) { *P_RST_LAN_UART_REG |= M_RST_LAN_UART; /* 1 */ } while ( *P_RST_LAN_UART_REG & M_RST_LAN_UART ) { *P_RST_LAN_UART_REG &= ~M_RST_LAN_UART; /* 0 */ } } /* end of 'lan_reset' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_configure * * * DESCRIPTION: * * * RETURN VALUE: * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * * *---------------------------------------------------------------------------*/ static void lan_configure( void ) { *P_LAN_CR = UART_CR_RESET_MR_PTR; /* Points to MR1. */ *P_LAN_CR = UART_CR_RESET_RVCR; /* Receiver disabled. */ *P_LAN_CR = UART_CR_RESET_XMTR; /* Transmitter disabled. */ *P_LAN_CR = UART_CR_RESET_ERROR_STATUS; *P_LAN_CR = UART_CR_RESET_BRK_CHG_INT; *P_LAN_MR1 = DEFAULT_LAN_MR1; *P_LAN_MR2 = DEFAULT_LAN_MR2; *P_LAN_ACR = DEFAULT_LAN_ACR; *P_LAN_CSR = UART_CSR_BR_9600; baud_rate_idx = 2; *P_LAN_CTUR = DEFAULT_LAN_CTUR; *P_LAN_CTLR = DEFAULT_LAN_CTLR; *P_LAN_CR = (UART_CR_START_CNTR_TIMER | UART_CR_ENABLE_XMTR | UART_CR_ENABLE_RCVR); lan_shadow_imr = UART_IMR_RxRDY; /* Enable only 'RxRDY' interrupt from UART. */ *P_LAN_IMR = lan_shadow_imr; tx_by_intr = 0; tx_by_poll = 0; return; } /* end of 'lan_configure' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_init_queue * * DESCRIPTION: * * RETURN VALUE: None. * * USED GLOBAL VARIABLES: * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * NOTES: * *---------------------------------------------------------------------------*/ static void lan_init_queue( T_RS232_QUEUE *p_queue ) { long i; /* * We set "head" equal to "tail" implying the queue is empty, * BUT the "head" and "tail" should each point to valid queue * positions. */ p_queue->head_index = 0; p_queue->tail_index = 0; p_queue->overflows = 0; p_queue->gdb_packet_start = -1; p_queue->gdb_packet_end = -1; p_queue->gdb_packet_csum1 = -1; p_queue->gdb_packet_csum2 = -1; for ( i = 0; i < MAX_RS232_CHARS; ++i ) { p_queue->buf[ i ] = 0; } return; } /* end of 'lan_init_queue' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_add_to_queue * * * DESCRIPTION: Adds the specified character to the tail of the * specified queue. Observes "oldest thrown on floor" * rule (i.e. the queue is allowed to "wrap" and the * input character is unconditionally placed at the * tail of the queue. * * * RETURN VALUE: None. * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * *---------------------------------------------------------------------------*/ static void lan_add_to_queue( long c, T_RS232_QUEUE *p_queue ) { if ( p_queue ) /* Sanity check. */ { if ( c & 0x000000FF ) /* We don't allow NULL characters to be added to a queue. */ { /* Insert the new character at the tail of the queue. */ p_queue->buf[ p_queue->tail_index ] = (UCHAR) (c & 0x000000FF); /* Increment the tail index. */ if ( MAX_RS232_CHARS <= ++(p_queue->tail_index) ) { p_queue->tail_index = 0; } /* Check for wrapping (i.e. overflow). */ if ( p_queue->head_index == p_queue->tail_index ) { /* If the tail has caught up to the head record the overflow . . . */ ++(p_queue->overflows); /* . . . then increment the head index. */ if ( MAX_RS232_CHARS <= ++(p_queue->head_index) ) { p_queue->head_index = 0; } } } /* End of 'if ( c & 0x000000FF )'. */ } /* End of 'if ( p_queue )'. */ return; } /* end of 'lan_add_to_queue' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_next_queue_char * * DESCRIPTION: * * RETURN VALUE: * * USED GLOBAL VARIABLES: * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * NOTES: * *---------------------------------------------------------------------------*/ static UCHAR lan_next_queue_char( T_RS232_QUEUE *p_queue ) { UCHAR c; c = 0; if ( p_queue ) { if ( p_queue->head_index != p_queue->tail_index ) { /* Return the 'oldest' character in the queue. */ c = p_queue->buf[ p_queue->head_index ]; /* Increment the head index. */ if ( MAX_RS232_CHARS <= ++(p_queue->head_index) ) { p_queue->head_index = 0; } } } /* End of 'if ( p_queue )'. */ return( c ); } /* end of 'lan_next_queue_char' *===========================================================================*/ /*----------------------------------------------------------------------------- * * FUNCTION NAME: lan_util_menu * * DESCRIPTION: Prints out a brief help on the LAN UART control utility. * * RETURN VALUE: None. * * USED GLOBAL VARIABLES: None. * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: None. * * NOTES: None. * *---------------------------------------------------------------------------*/ static void lan_util_menu( void ) { /* * Multiply calling printp() below is made due to the limitations * of printp(), incapable of handling long formatting constants: */ printp( "\n -- Options --\n\n" ); printp( " %2X,'INIT' ............... Reset & (Re)INITIALIZE Interface.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'BAUD',<rate> ........ Set BAUD Rate.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'INTR',<mode> ........ Toggle 'RxRDY' Interrupts.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'XMT',<mode> ......... Toggle TRANSMIT-via-backplane.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'STAT' ............... Display STATUS.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'ECHO',<mode> ........ Enable/Disable Test ECHO.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'IN',<action> ........ Access INPUT Queue.\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'OUT',<action> ....... Access OUTPUT Queue.\n\n", (ULONG) LAN_UTIL_CODE ); printp( " %2X,'PUTC',<char> ........ Output a Character (i.e. <char>).\n\n", (ULONG) LAN_UTIL_CODE ); /*** printp( " %2X,'WPM',address,word ... Write Protected Memory Test.\n\n", (ULONG) LAN_UTIL_CODE ); ***/ printp( " <rate>: 4800 <mode>: E - enable <action>: C - clear/reset\n" ); printp( " 9600 D - disable D - display\n" ); printp( " 19200 F - fetch next char\n" ); printp( " 38400\n" ); } /* end of 'lan_util_menu' *===========================================================================*/ /* Thu Feb 5 17:14:41 EST 1998 CYGNUS...CYGNUS...CYGNUS...CYGNUS...CYGNUS...CYGNUS...CYGNUS...CYGNUS */ static long get_gdb_input( long c, T_RS232_QUEUE * p_input_q ) { /* Now to detect when we've got a gdb packet... */ if ( '$' == c ) { /* char marks beginning of a packet */ if ( -1 != p_input_q->gdb_packet_start || -1 != p_input_q->gdb_packet_end || -1 != p_input_q->gdb_packet_csum1 || -1 != p_input_q->gdb_packet_csum2 ) { /* PROTOCOL ERROR */ /* NEW: Actually, this probably means that we muffed a packet, and GDB has already resent it. The thing to do now is to throw away the one we WERE working on, but immediately start accepting the new one. Don't NAK, or GDB will have to try and send it yet a third time! */ /*NACK_PKT( );*/ /*<ETHERNET>*/ discard_packet( ); /* throw away old packet */ lan_add_to_queue ('$', p_input_q); /* put the new "$" back in */ return 0; } else { /* match new "$" */ p_input_q->gdb_packet_start = p_input_q->tail_index; p_input_q->gdb_packet_end = p_input_q->gdb_packet_csum1 = p_input_q->gdb_packet_csum2 = -1; } } else if ( '#' == c ) { /* # marks end of packet (except for checksum) */ if ( -1 == p_input_q->gdb_packet_start || -1 != p_input_q->gdb_packet_end || -1 != p_input_q->gdb_packet_csum1 || -1 != p_input_q->gdb_packet_csum2 ) { /* PROTOCOL ERROR */ /* Garbled packet. Discard, but do not NAK. */ /*NACK_PKT( );*/ /*<ETHERNET>*/ discard_packet( ); return -1; } p_input_q->gdb_packet_end = p_input_q->tail_index; p_input_q->gdb_packet_csum1 = p_input_q->gdb_packet_csum2 = -1; } else if ( -1 != p_input_q->gdb_packet_start && -1 != p_input_q->gdb_packet_end) { if ( isxdigit( c ) ) { /* char is one of two checksum digits for packet */ if ( -1 == p_input_q->gdb_packet_csum1 && LAN_Q_MOD( p_input_q->gdb_packet_end + 1 ) == p_input_q->tail_index ) { /* first checksum digit */ p_input_q->gdb_packet_csum1 = p_input_q->tail_index; p_input_q->gdb_packet_csum2 = -1; } else if ( -1 == p_input_q->gdb_packet_csum2 && LAN_Q_MOD( p_input_q->gdb_packet_end + 2 ) == p_input_q->tail_index ) { /* second checksum digit: packet is complete! */ p_input_q->gdb_packet_csum2 = p_input_q->tail_index; getpacket(); /* got a packet -- extract it */ } else { /* probably can't happen (um... three hex digits?) */ /* PROTOCOL ERROR */ /* Not sure how this can happen, but ... discard it, but do not NAK it. */ /*NACK_PKT( );*/ /*<ETHERNET>*/ discard_packet( ); return -1; } } else { /* '#' followed by non-hex char */ /* PROTOCOL ERROR */ /* Bad packet -- discard but do not NAK */ /*NACK_PKT( );*/ /*<ETHERNET>*/ discard_packet( ); return -1; } } return 0; } #ifdef STANDALONE /* stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone Enable stand-alone build, for ease of debugging stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone stand-alone */ long write_to_protected_mem (addr, word) void *addr; unsigned short word; { return 0; } char dummy_memory[0x4000]; int main ( void ) { long c; lan_init_queue( &lan_input_queue ); printf( "Stand-alone EMC 'stub', pid = %d\n", getpid( ) ); printf( "Start of simulated 'memory': 0x%08x\n", &dummy_memory); while ( (c = getc( stdin ) ) != EOF ) { if ( c == '\\' ) /* escape char */ break; lan_add_to_queue( c, &lan_input_queue ); get_gdb_input (c, &lan_input_queue); fflush( stdout ); } printf( "Goodbye!\n" ); exit( 0 ); } #define SRAM_START ((void *) (&dummy_memory[0] + 0x00000000)) #define SRAM_END ((void *) (&dummy_memory[0] + 0x00000400)) #define RO_AREA_START ((void *) (&dummy_memory[0] + 0x00000100)) #define RO_AREA_END ((void *) (&dummy_memory[0] + 0x00000300)) #define NVD_START ((void *) (&dummy_memory[0] + 0x00003000)) #define NVD_END ((void *) (&dummy_memory[0] + 0x00003100)) #else /* normal stub (not stand-alone) */ #define SRAM_START ((void *) 0x00000000) #define SRAM_END ((void *) 0x00400000) #define RO_AREA_START ((void *) 0x00100000) #define RO_AREA_END ((void *) 0x00300000) #define NVD_START ((void *) 0x03000000) #define NVD_END ((void *) 0x03100000) #endif /* STANDALONE */ /* gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb Here begins the gdb stub section. gdb gdb The following functions were added by Cygnus, gdb gdb to make this thing act like a gdb stub. gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb gdb */ /* ------------------- global defines and data decl's -------------------- */ #define hexchars "0123456789abcdef" /* there are 180 bytes of registers on a 68020 w/68881 */ /* many of the fpa registers are 12 byte (96 bit) registers */ #define NUMREGBYTES 180 #define NUMREGS 29 #define REGISTER_BYTE(regno) regno enum regnames { D0, D1, D2, D3, D4, D5, D6, D7, A0, A1, A2, A3, A4, A5, A6, A7, PS, PC, FP0, FP1, FP2, FP3, FP4, FP5, FP6, FP7, FPCONTROL, FPSTATUS, FPIADDR }; unsigned long registers[NUMREGBYTES/4]; static long remote_debug; #define BUFMAX MAX_IO_BUF_SIZE static char inbuffer[BUFMAX], outbuffer[BUFMAX]; static char spare_buffer[BUFMAX]; struct stub_trace_frame { int valid; unsigned long frame_id; unsigned long tdp_id; FRAME_DEF *frame_data; COLLECTION_FORMAT_DEF *format; unsigned long traceregs[NUMREGBYTES/4]; unsigned char *stack_data; unsigned char *memrange_data; } curframe; /* ------------------- function prototypes -------------------- */ void handle_request ( char * ); /* ------------------- Implementation -------------------- */ static void discard_packet( void ) { lan_input_queue.head_index = lan_input_queue.tail_index; lan_input_queue.gdb_packet_start = lan_input_queue.gdb_packet_end = lan_input_queue.gdb_packet_csum1 = lan_input_queue.gdb_packet_csum2 = -1; } /* Utility function: convert an ASCII isxdigit to a hex nybble */ static long hex( char ch ) { if ( (ch >= 'A') && (ch <= 'F') ) return ch - 'A' + 10; if ( (ch >= 'a') && (ch <= 'f') ) return ch - 'a' + 10; if ( (ch >= '0') && (ch <= '9') ) return ch - '0'; return -1; } static void getpacket( void ) { unsigned char our_checksum, their_checksum; char *copy = inbuffer; unsigned char c; our_checksum = 0; /* first find the '$' */ while ((c = lan_next_queue_char ( &lan_input_queue )) != '$') if (c == 0) /* ??? Protocol error? (paranoia) */ { /* PROTOCOL ERROR (missing '$') */ /*NACK_PKT( );*/ /*<ETHERNET>*/ return; } /* Now copy the message (up to the '#') */ for (c = lan_next_queue_char ( &lan_input_queue ); /* skip the '$' */ c != 0 && c != '#'; /* stop at the '#' */ c = lan_next_queue_char ( &lan_input_queue )) { *copy++ = c; our_checksum += c; } *copy++ = '\0'; /* terminate the copy */ if (c == 0) /* ??? Protocol error? (paranoia) */ { /* PROTOCOL ERROR (missing '#') */ /*NACK_PKT( );*/ /*<ETHERNET>*/ return; } their_checksum = hex( lan_next_queue_char ( &lan_input_queue ) ) << 4; their_checksum += hex( lan_next_queue_char ( &lan_input_queue ) ); /* Now reset the queue packet-recognition bits */ discard_packet( ); if ( remote_debug || our_checksum == their_checksum ) { ACK_PKT( ); /* good packet */ /* Parse and process the packet */ handle_request( inbuffer ); } else /* PROTOCOL ERROR (bad check sum) */ NACK_PKT( ); } /* EMC will provide a better implementation (perhaps just of LAN_PUT_CHAR) that does not block. For now, this works. */ static void putpacket( char *str ) { unsigned char checksum; /* '$'<packet>'#'<checksum> */ if ( VIA_ETHERNET == gdb_host_comm ) { char *p_out; long length; p_out = eth_outbuffer; length = 0; if ( YES == gdb_cat_ack ) { *p_out++ = '+'; ++length; } gdb_cat_ack = NO; *p_out++ = '$'; ++length; checksum = 0; while ( *str ) { *p_out++ = *str; ++length; checksum += *str++; } *p_out++ = '#'; *p_out++ = hexchars[checksum >> 4]; *p_out = hexchars[checksum % 16]; length += 3; eth_to_gdb( (UCHAR *) eth_outbuffer, length ); } else { /* via RS-232 */ do { LAN_PUT_CHAR( '$' ); checksum = 0; while ( *str ) { LAN_PUT_CHAR( *str ); checksum += *str++; } LAN_PUT_CHAR( '#' ); LAN_PUT_CHAR( hexchars[checksum >> 4] ); LAN_PUT_CHAR( hexchars[checksum % 16] ); } while ( 0 /* get_debug_char( ) != '+' */ ); /* XXX FIXME: not waiting for the ack. */ } } /*----------------------------------------------------------------------------- * * FUNCTION NAME: gdb_get_eth_input * * * DESCRIPTION: * * * RETURN VALUE: None. * * * USED GLOBAL VARIABLES: * * * AFFECTED GLOBAL VARIABLES/SIDE EFFECTS: * * * NOTES: * * *---------------------------------------------------------------------------*/ void gdb_get_eth_input( unsigned char *buf, long length ) { gdb_host_comm = VIA_ETHERNET; for ( ; 0 < length; ++buf, --length) { if ( *buf == CONTROL_C ) { /* can't stop the target, but we can tell gdb to stop waiting... */ discard_packet( ); putpacket( "S03" ); /* send back SIGINT to the debugger */ } else { lan_add_to_queue( (long) *buf, &lan_input_queue ); get_gdb_input( (long) *buf, &lan_input_queue ); } } return; } /* end of 'gdb_get_eth_input' *===========================================================================*/ /* STDOUT STDOUT STDOUT STDOUT STDOUT STDOUT STDOUT STDOUT STDOUT STDOUT Stuff pertaining to simulating stdout by sending chars to gdb to be echoed. Dear reader: This code is based on the premise that if GDB receives a packet from the stub that begins with the character CAPITAL-OH, GDB will echo the rest of the packet to GDB's console / stdout. This gives the stub a way to send a message directly to the user. In practice, (as currently implemented), GDB will only accept such a packet when it believes the target to be running (ie. when you say STEP or CONTINUE); at other times it does not expect it. This will probably change as a side effect of the "asynchronous" behavior. Functions: gdb_putchar(char ch) gdb_write(char *str, int len) gdb_puts(char *str) gdb_error(char *format, char *parm) */ #if 0 /* avoid compiler warning while this is not used */ /* Function: gdb_putchar(int) Make gdb write a char to stdout. Returns: the char */ static int gdb_putchar( long ch ) { char buf[4]; buf[0] = 'O'; buf[1] = hexchars[ch >> 4]; buf[2] = hexchars[ch & 0x0F]; buf[3] = 0; putpacket( buf ); return ch; } #endif /* Function: gdb_write(char *, int) Make gdb write n bytes to stdout (not assumed to be null-terminated). Returns: number of bytes written */ static int gdb_write( char *data, long len ) { char *buf, *cpy; long i; buf = outbuffer; buf[0] = 'O'; i = 0; while ( i < len ) { for ( cpy = buf+1; i < len && cpy < buf + BUFMAX - 3; i++ ) { *cpy++ = hexchars[data[i] >> 4]; *cpy++ = hexchars[data[i] & 0x0F]; } *cpy = 0; putpacket( buf ); } return len; } /* Function: gdb_puts(char *) Make gdb write a null-terminated string to stdout. Returns: the length of the string */ static int gdb_puts( char *str ) { return gdb_write( str, strlen( str ) ); } /* Function: gdb_error(char *, char *) Send an error message to gdb's stdout. First string may have 1 (one) optional "%s" in it, which will cause the optional second string to be inserted. */ #if 0 static void gdb_error( char *format, char *parm ) { static char buf[400]; char *cpy; long len; if ( remote_debug ) { if ( format && *format ) len = strlen( format ); else return; /* empty input */ if ( parm && *parm ) len += strlen( parm ); for ( cpy = buf; *format; ) { if ( format[0] == '%' && format[1] == 's' ) /* include 2nd string */ { format += 2; /* advance two chars instead of just one */ while ( parm && *parm ) *cpy++ = *parm++; } else *cpy++ = *format++; } *cpy = '\0'; gdb_puts( buf ); } } #endif static void gdb_note (char *, int); static int error_ret (int, char *, int); static unsigned long elinum_to_index (unsigned long elinum) { if ((elinum & 0xf0) == 0xd0) return (elinum & 0x0f); else if ((elinum & 0xf0) == 0xa0) return (elinum & 0x0f) + 8; else return -1; } static long index_to_elinum (unsigned long index) { if (index <= 7) return index + 0xd0; else if (index <= 15) return (index - 8) + 0xa0; else return -1; } /* READMEM READMEM READMEM READMEM READMEM READMEM READMEM READMEM READMEM The following code pertains to reading memory from the target. Some sort of exception handling should be added to make it safe. READMEM READMEM READMEM READMEM READMEM READMEM READMEM READMEM READMEM Safe Memory Access: All reads and writes into the application's memory will pass thru get_uchar() or set_uchar(), which check whether accessing their argument is legal before actual access (thus avoiding a bus error). */ enum { SUCCESS = 0, FAIL = -1 }; #if 0 static long get_uchar ( const unsigned char * ); #endif static long set_uchar ( unsigned char *, unsigned char ); static long read_access_violation ( const void * ); static long write_access_violation ( const void * ); static long read_access_range(const void *, long); static DTC_RESPONSE find_memory(unsigned char *,long,unsigned char **,long *); static int dtc_error_ret (int ret, char *src, DTC_RESPONSE code) { if (src) sprintp (spare_buffer, "'%s' returned DTC error '%s'.\n", src, get_err_text (code)); else sprintp (spare_buffer, "DTC error '%s'.\n", get_err_text (code)); gdb_puts (spare_buffer); return ret; } #if 0 /* I think this function is unnecessary since the introduction of adbg_find_memory_addr_in_frame. */ /* Return the number of expressions in the format associated with a given trace frame. */ static int count_frame_exprs (FRAME_DEF *frame) { CFD *format; T_EXPR *expr; int num_exprs; /* Get the format from the frame. */ get_frame_format_pointer (frame, &format); /* Walk the linked list of expressions, and count the number of expressions we find there. */ num_exprs = 0; for (expr = format->p_cfd_expr; expr; expr = expr->next) num_exprs++; return num_exprs; } #endif #if 0 /* Function: get_frame_addr * * Description: If the input memory address was collected in the * current trace frame, then lookup and return the address * from within the trace buffer from which the collected byte * may be retrieved. Else return -1. */ unsigned char * get_frame_addr ( const unsigned char *addr ) { unsigned char *base, *regs, *stack, *mem; CFD *dummy; DTC_RESPONSE ret; /* first, see if addr is on the saved piece of stack for curframe */ if (curframe.format->stack_size > 0 && (base = (unsigned char *) curframe.traceregs[A7]) <= addr && addr < base + curframe.format->stack_size) { gdb_puts("STUB: get_frame_addr: call get_addr_to_frame_regs_stack_mem\n"); if ((ret = get_addr_to_frame_regs_stack_mem (curframe.frame_data, &dummy, (void *) ®s, (void *) &stack, (void *) &mem)) != OK_TARGET_RESPONSE) return (void *) dtc_error_ret (-1, "get_addr_to_frame_regs_stack_mem", ret); else return stack + (addr - base); } /* Next, try to find addr in the current frame's expression- collected memory blocks. I'm sure this is at least quadradic in time. */ { int num_exprs = count_frame_exprs (curframe.frame_data); int expr, block; /* Try each expression in turn. */ for (expr = 0; expr < num_exprs; expr++) { for (block = 0; ; block++) { T_EXPR_DATA *data; if (adbg_get_expr_data (curframe.frame_data, 'x', expr, block, &data) != OK_TARGET_RESPONSE) break; else if ((unsigned char *) data->address <= addr && addr < ((unsigned char *) data->address + data->size)) { /* We have found the right block; is it valid data? Upper-case stamps mean bad data. */ if ('A' <= data->stamp && data->stamp <= 'Z') { gdb_puts("STUB: get_frame_addr: adbg_get_expr_data INVALID\n"); return (unsigned char *) -1; } else { if (remote_debug > 1) { sprintp(spare_buffer, "STUB: get_frame_addr: got it [%x,%x)\n", data->address, data->address + data->size); gdb_puts(spare_buffer); } return (((unsigned char *) &data->data) + (addr - (unsigned char *) data->address)); } } } } } /* not found, return error */ return (unsigned char *) -1; } /*============================================================*/ static long get_uchar ( const unsigned char * addr ) { unsigned char *frame_addr; if ( read_access_violation ( addr ) ) return ( -1 ); /* Access error */ if (curframe.valid) /* if debugging a trace frame? */ { /* If the requested address was collected in the current frame, * then fetch and return the data from the trace buffer. */ if ((frame_addr = get_frame_addr (addr)) != (unsigned char *) -1) return ( *frame_addr ); /* If the requested address is in the Code Section, * let's be magnanimous and read it anyway (else we shall * not be able to disassemble, find function prologues, etc.) */ else if (CS_CODE_START <= (unsigned long) addr && (unsigned long) addr < CS_CODE_START + CS_CODE_SIZE) return (*addr); else return ( -1 ); /* "Access error" (the data was not collected) */ } else /* Not debugging a trace frame, read the data from live memory. */ return ( *addr ); /* Meaningful result >= 0 */ } #endif /*============================================================*/ static long set_uchar ( unsigned char * addr, unsigned char val ) { long check_result = write_access_violation ( addr ); if ( check_result != 0L ) return ( check_result ); /* Access error */ return ( *addr = val ); /* Successful writing */ } /*============================================================*/ /* * Function read_access_violation() below returns TRUE if dereferencing * its argument for reading would cause a bus error - and FALSE otherwise: */ static long read_access_violation ( const void * addr ) { return ( ( ( addr < SRAM_START ) || ( addr >= SRAM_END ) ) && ( ( addr < NVD_START ) || ( addr >= NVD_END ) ) ); } /*============================================================*/ /* * Function write_access_violation() below returns zero if dereferencing * its argument for writing is safe, -1 on a soft error (the argument * falls into the write-protected area), -2 on a hard error (the argument * points to a non-existent memory location). In other words, it returns * FALSE when no bus error is expected - and an error code otherwise: */ static long write_access_violation ( const void * addr ) { /* * The boundaries of the write-protected area have to be received via * an API provided in the Symmetrix core code. For now, these limits * are hard-coded: */ if ( ( addr >= RO_AREA_START ) && ( addr < RO_AREA_END ) ) return ( -1 ); /* soft error */ if ( ( ( addr < SRAM_START ) || ( addr >= SRAM_END ) ) && ( ( addr < NVD_START ) || ( addr >= NVD_END ) ) ) return ( -2 ); /* hard error */ return ( 0 ); } /* read_access_range is like read_access_violation, but returns the number of bytes we can read w/o faulting. that is, it checks an address range and tells us what portion (if any) of the prefix is safe to read without a bus error */ static long read_access_range(const void *addr, long count) { if ((addr >= SRAM_START) && (addr < SRAM_END)) { if ((char *)addr + count < (char *)SRAM_END) return (count); else return ((char *)SRAM_END - (char *)addr); } else if (((char *)addr >= (char *)NVD_START) && ((char *)addr < (char *)NVD_END)) { if ((char *)addr + count < (char *)NVD_END) return (count); else return ((char *)NVD_END - (char *)addr); } else return (0); } /* Convert the memory pointed to by mem into hex, placing result in buf. Return SUCCESS or FAIL. If MAY_FAULT is non-zero, then we should return FAIL in response to a fault; if zero treat a fault like any other fault in the stub. */ static long mem2hex(unsigned char *mem, char *buf, long count, long may_fault) { long ndx; long ndx2; long ch; long incr; unsigned char *location; DTC_RESPONSE status; if (may_fault) { for (ndx = 0, incr = 1; (ndx < count) && (incr > 0); ndx += incr) { status = find_memory(mem, count - ndx, &location, &incr); if (status == OK_TARGET_RESPONSE) { if (incr > 0) { for (ndx2 = 0; ndx2 < incr; ndx2++) { ch = *location++; *buf++ = hexchars[ch >> 4]; *buf++ = hexchars[ch & 0xf]; } mem += incr; } else if (incr <= 0) /* should never happen */ { *buf = 0; return (0); } } else if (status == NOT_FOUND_TARGET_RESPONSE) { *buf = 0; return (ndx); /* return amount copied */ } else { *buf = 0; return (0); /* XXX: how do we tell the user the status? */ } } *buf = 0; return (count); } else { for (ndx = 0; ndx < count; ndx++) { ch = *mem++; *buf++ = hexchars[ch >> 4]; *buf++ = hexchars[ch & 0xf]; } *buf = 0; return (count); /* we copied everything */ } } static DTC_RESPONSE find_memory(unsigned char *mem, long count, unsigned char **location, long *incr) { DTC_RESPONSE retval; long length; /* figure out how much of the memory range we can read w/o faulting */ count = read_access_range(mem, count); if (count == 0) return (NOT_FOUND_TARGET_RESPONSE); if (curframe.valid) { unsigned char *mem_block; unsigned char *mem_addr; unsigned long mem_size; unsigned long mem_stamp; retval = adbg_find_memory_addr_in_frame(curframe.frame_data, mem, (unsigned long **)&mem_block, (unsigned long **)&mem_addr, &mem_size, &mem_stamp); switch (retval) { case OK_TARGET_RESPONSE: #if 0 printp("FOUND: mem %x block %x addr %x size %d stamp %x\n", mem, mem_block, mem_addr, mem_size, mem_stamp); #endif *location = mem_block + (mem - mem_addr); length = mem_size - (mem - mem_addr); if (length < count) *incr = length; else *incr = count; break; case NOT_FOUND_TARGET_RESPONSE: case NEAR_FOUND_TARGET_RESPONSE: #if 0 printp("NOT FOUND: mem %x, checking code region\n", mem); #endif /* check to see if it's in the code region */ if ((CS_CODE_START <= (long)mem) && ((long)mem < CS_CODE_START + CS_CODE_SIZE)) { /* some or all of the address range is in the code */ *location = mem; if ((long)mem + count <= CS_CODE_START + CS_CODE_SIZE) *incr = count; /* it's totally in the code */ else /* how much is in the code? */ *incr = CS_CODE_START + CS_CODE_SIZE - (long)mem; #if 0 printp("FOUND in code region: %x\n", mem); #endif retval = OK_TARGET_RESPONSE; } else retval = NOT_FOUND_TARGET_RESPONSE; break; default: #if 0 printp("BAD RETURN: %d\n", retval); #endif retval = NOT_FOUND_TARGET_RESPONSE; break; } } else { *location = mem; *incr = count; retval = OK_TARGET_RESPONSE; } return (retval); } /* Convert the hex array pointed to by buf into binary to be placed in mem. Return SUCCESS or FAIL. */ static long hex2mem( char *buf, unsigned char *mem, long count, long may_fault ) { long i, ch; for ( i=0; i<count; i++ ) { ch = hex( *buf++ ) << 4; ch = ch + hex( *buf++ ); if ( may_fault ) { ch = set_uchar( mem++, ch ); if ( ch < 0 ) /* negative return indicates error */ return FAIL; } else *mem++ = ch; } return SUCCESS; } /**********************************************/ /* WHILE WE FIND NICE HEX CHARS, BUILD AN INT */ /* RETURN NUMBER OF CHARS PROCESSED */ /**********************************************/ static int hexToInt( char **ptr, unsigned long *intValue ) { long numChars = 0; long hexValue; *intValue = 0; while ( **ptr ) { hexValue = hex( **ptr ); if ( hexValue >=0 ) { *intValue = (*intValue << 4) | hexValue; numChars ++; } else break; (*ptr)++; } return numChars; } static volatile long gdb_handling_trap1; static volatile long gdb_handling_sstrace; static volatile long gdb_signo; /* Here is the "callable" stub entry point. Call this function with a GDB request as an argument, and it will service the request and return. May be further broken up as we go along, with individual requests broken out as separate functions. */ static char * handle_trace_query (char *); static char * handle_trace_set (char *); static int handle_format (char **request, CFD *format); static unsigned long crc32 (unsigned char *buf, int len, unsigned long crc); static char * crc_query (char *); static char * handle_test (char *); void handle_request( char *request ) { #if 0 remote_debug = 2; #endif switch( *request++ ) { case 'k': /* "kill" */ curframe.valid = FALSE; putpacket (""); break; case 'D': /* "detach" */ curframe.valid = FALSE; putpacket (""); break; default: /* Unknown code. Return an empty reply message. */ putpacket( "" ); /* return empty packet */ break; case 'H': /* Set thread for subsequent operations. Hct... c = 'c' for thread used in step and continue; t... can be -1 for all threads. c = 'g' for thread used in other operations. If zero, pick a thread, any thread. */ putpacket( "OK" ); break; case 'g': /* Read registers. Each byte of register data is described by two hex digits. registers are in the internal order for GDB, and the bytes in a register are in the same order the machine uses. */ { /* Return the values in (one of) the registers cache(s). Several situations may pertain: 1) We're synchronous, in which case the "registers" array should actually be current. 2) We're asynchronous, in which case the "registers" array holds whatever was cached most recently. 3) We're looking at a trace frame that was collected earlier: we will return those earlier registers. */ /* all registers default to zero */ memset (outbuffer, '0', NUMREGBYTES); outbuffer[NUMREGBYTES] = '\0'; if (curframe.valid) /* debugging a trace frame */ mem2hex( (unsigned char*) curframe.traceregs, outbuffer, NUMREGBYTES, 0 ); else mem2hex( (unsigned char*) registers, outbuffer, NUMREGBYTES, 0 ); putpacket( outbuffer ); } break; case 'G': /* Write registers. Gxxxxxxxx Each byte of register data is described by two hex digits. */ if (curframe.valid) /* debugging a trace frame */ putpacket ("E03"); /* can't write regs into a trace frame! */ else { /* Write the values into the local registers cache... Note that no actual registers are being changed. */ hex2mem( request, (unsigned char *) registers, NUMREGBYTES, 0 ); putpacket( "OK" ); } break; case 'P': /* Write (single) register. Pnn=xxxxxxxx register nn gets value xxxxxxxx; two hex digits for each byte in the register (target byte order). */ if (curframe.valid) putpacket ("E03"); /* can't write regs into a trace frame! */ else { unsigned long regno; if ( hexToInt( &request, ®no ) && *(request++) == '=' ) { if ( regno < NUMREGS ) { hexToInt( &request, (unsigned long *) ®isters[REGISTER_BYTE(regno)]); putpacket( "OK" ); } else putpacket( "E01" ); /* bad packet or regno */ } } break; case 'm': /* Read memory. mAAAAAAAA,LLLL AAAAAAAA is address, LLLL is length. Reply can be fewer bytes than requested if able to read only part of the data. */ { unsigned long addr, len; if ( hexToInt( &request, &addr ) && *(request++) == ',' && hexToInt( &request, &len ) ) { /* better not overwrite outbuffer! */ if ( len > (BUFMAX / 2) - 5 ) len = (BUFMAX / 2) - 5; if (mem2hex((unsigned char *) addr, outbuffer, len, 1) == 0) /* XXX: eventually use returned value */ putpacket( "E03" ); /* read fault (access denied) */ else putpacket( outbuffer ); /* read succeeded */ } else putpacket( "E01" ); /* badly formed read request */ } break; case 'M': /* Write memory. Maaaaaaaa,llll:xxxx aaaaaaaa is address, llll is length; xxxx is data to write. */ { unsigned long addr, len; if (curframe.valid) /* can't write memory into a trace frame! */ putpacket ("E03"); /* "access denied" */ else /*** if ( write_access_enabled ) ***/ { if ( hexToInt( &request, &addr ) && *(request++) == ',' && hexToInt( &request, &len ) && *(request++) == ':' ) { if (len == 2 && addr >= CS_CODE_START && addr <= LAST_CS_WORD) { unsigned long val; if ( !hexToInt( &request, &val ) || write_to_protected_mem( (void *)addr, val ) ) putpacket( "E03" ); /* write fault (access denied) */ else putpacket( "OK" ); /* write succeeded */ } else { if ( hex2mem( request, (unsigned char*) addr, len, 1 ) ) putpacket( "E03" ); /* write fault (access denied) */ else putpacket( "OK" ); /* write succeeded */ } } else putpacket( "E02" ); /* badly formed write request */ } } break; case 'c': /* Continue. cAAAAAAAA AAAAAAAA is address from which to resume. If omitted, resume at current PC. */ { unsigned long addr; if (curframe.valid) { /* Don't continue if debugging a trace frame! */ gdb_puts ("Error: can't continue!\n"); putpacket ("S03"); } else { gdb_signo = 3; if (isxdigit(request[0])) { hexToInt(&request, &addr); registers[REGISTER_BYTE(PC)] = addr; } gdb_handling_trap1 = FALSE; gdb_handling_sstrace = FALSE; sss_trace_flag = '\0'; } } break; case 's': /* Step. sAAAAAAAA AAAAAAAA is address from which to begin stepping. If omitted, begin stepping at current PC. */ { unsigned long addr; if (curframe.valid) { /* Don't step if debugging a trace frame! */ gdb_puts ("Error: can't step!\n"); putpacket ("S03"); } else { gdb_signo = 3; if (isxdigit(request[0])) { hexToInt(&request, &addr); registers[REGISTER_BYTE(PC)] = addr; } gdb_handling_trap1 = FALSE; gdb_handling_sstrace = FALSE; sss_trace_flag = 't'; } } break; case 'C': /* Continue with signal. Cxx;AAAAAAAA xx is signal number in hex; AAAAAAAA is adddress from which to resume. If ;AAAAAAAA omitted, continue from PC. */ { unsigned long addr = 0; if (!gdb_handling_trap1 || curframe.valid) { /* Don't continue if not currently in synchronous mode, or if currently debugging a trace frame! */ gdb_puts( "Error: can't continue!\n" ); putpacket( "S03" ); /* "sigquit" (better idea?) */ } else { gdb_signo = 3; if ( isxdigit( *request ) ) { hex2mem( request, (unsigned char *) &gdb_signo, 2, 0 ); request += 2; if ( *request == ';' && isxdigit( *++request ) ) { hexToInt( &request, &addr ); registers[REGISTER_BYTE(PC)] = addr; } } gdb_handling_trap1 = FALSE; gdb_handling_sstrace = FALSE; sss_trace_flag = '\0'; } } break; case 'S': /* Step with signal. Sxx;AAAAAAAA xx is signal number in hex; AAAAAAAA is adddress from which to begin stepping. If ;AAAAAAAA omitted, begin stepping from PC. */ { unsigned long addr = 0; if (!gdb_handling_trap1 || curframe.valid) { /* Don't step if not currently in synchronous mode, or if currently debugging a trace frame! */ gdb_puts( "Error: can't step!\n" ); putpacket( "S03" ); /* "sigquit" (better idea?) */ } else { gdb_signo = 3; if ( isxdigit( *request ) ) { hex2mem( request, (unsigned char *) &gdb_signo, 2, 0 ); request += 2; if ( *request == ';' && isxdigit( *++request ) ) { hexToInt( &request, &addr ); registers[REGISTER_BYTE(PC)] = addr; } } gdb_handling_trap1 = FALSE; gdb_handling_sstrace = FALSE; sss_trace_flag = 't'; } } break; case '?': /* Query the latest reason for stopping. Should be same reply as was last generated for step or continue. */ if ( gdb_signo == 0 ) gdb_signo = 3; /* default to SIGQUIT */ outbuffer[ 0 ] = 'S'; outbuffer[ 1 ] = hexchars[ gdb_signo >> 4 ]; outbuffer[ 2 ] = hexchars[ gdb_signo & 0xf ]; outbuffer[ 3 ] = 0; putpacket( outbuffer ); break; case 'd': /* Toggle debug mode I'm sure we can think of something interesting. */ remote_debug = !remote_debug; putpacket( "" ); /* return empty packet */ break; case 'q': /* general query */ switch (*request++) { default: putpacket (""); /* nak a request which we don't handle */ break; case 'T': /* trace query */ putpacket (handle_trace_query (request)); break; case 'C': /* crc query (?) */ if (*request++ == 'R' && *request++ == 'C' && *request++ == ':') putpacket (crc_query (request)); else putpacket (""); /* unknown query */ break; } break; case 'Q': /* general set */ switch (*request++) { default: putpacket (""); /* nak a request which we don't handle */ break; case 'T': /* trace */ putpacket (handle_trace_set (request)); break; } break; case 'T': /* call test function: TAAA,BBB,CCC A, B, and C are arguments to pass to gdb_c_test. Reply is "E01" (bad arguments) or "OK" (test function called). */ putpacket (handle_test (request)); break; } } static TDP_SETUP_INFO tdp_temp; static int trace_running; /* * Function msgcmp: * * If second argument (str) is matched in first argument, * then advance first argument past end of str and return "SAME" * else return "DIFFERENT" without changing first argument. * * Return: zero for DIFFERENT, non-zero for SUCCESS */ static int msgcmp (char **msgp, char *str) { char *next; if (msgp != 0 && str != 0) /* input validation */ if ((next = *msgp) != 0) { for (; *next && *str && *next == *str; next++, str++) ; if (*str == 0) /* matched all of str in msg */ return (int) (*msgp = next); /* advance msg ptr past str */ } return 0; /* failure */ } static char * handle_trace_query (char *request) { if (msgcmp (&request, "Status")) { if (adbg_check_if_active ()) { gdb_puts ("Target trace is running.\n"); return "T1"; } else { gdb_puts ("Target trace not running.\n"); trace_running = 0; return "T0"; } } else /* unknown trace query */ { return ""; } } static void gdb_note (char *fmt, int arg1) { if (remote_debug > 1) { sprintp (spare_buffer, fmt, arg1); gdb_puts (spare_buffer); } } static int error_ret (int ret, char *fmt, int arg1) { if (remote_debug > 0) { sprintp (spare_buffer, fmt, arg1); gdb_puts (spare_buffer); } return ret; } static int handle_format (char **request, COLLECTION_FORMAT_DEF *format) { MEMRANGE_DEF m; DTC_RESPONSE ret; int elinum; unsigned long regnum; long bytecodes[(MAX_BYTE_CODES + sizeof (struct t_expr_tag))/ 4]; struct t_expr_tag *t_expr = (struct t_expr_tag *)bytecodes; if (format->id == 0) { if ((ret = get_unused_format_id (&format->id)) != OK_TARGET_RESPONSE) return dtc_error_ret (-1, "get_unused_format_id", ret); if (**request == 'R') { (*request)++; hexToInt (request, &format->regs_mask); } gdb_note ("STUB: call define_format (id = %d, ", format->id); gdb_note ("regs_mask = 0x%X);\n", format->regs_mask); if ((ret = define_format (format)) != OK_TARGET_RESPONSE) { sprintp (spare_buffer, "'define_format': DTC error '%s' for format id %d.\n", get_err_text (ret), format->id); gdb_puts (spare_buffer); return -1; } } while ((**request == 'M') || (**request == 'X')) { switch (**request) { case 'M': /* M<regnum>,<offset>,<size> */ (*request)++; hexToInt(request, ®num); if (regnum == 0 || regnum == (unsigned long) -1) m.typecode = -1; else if ((elinum = index_to_elinum (regnum)) > 0) m.typecode = elinum; else return error_ret (-1, "Memrange register %d is not between 0 and 15\n", regnum); if (*(*request)++ != ',') return error_ret (-1,"Malformed memrange (comma #%d missing)\n",1); hexToInt(request, &m.offset); if (*(*request)++ != ',') return error_ret (-1,"Malformed memrange (comma #%d missing)\n",2); hexToInt(request, &m.size); gdb_note ("STUB: call add_format_mem_range (typecode = 0x%x, ", m.typecode); gdb_note ("offset = 0x%X, ", m.offset); gdb_note ("size = %d);\n", m.size); if ((ret = add_format_mem_ranges (format->id, &m)) != OK_TARGET_RESPONSE) { dtc_error_ret (-1, "add_format_mem_ranges", ret); sprintp (spare_buffer, "format id %d: memrange (0x%x, 0x%x, 0x%x).\n", format->id, m.typecode, m.offset, m.size); gdb_puts (spare_buffer); return -1; } break; case 'X': /* X<length>,<bytecodes> */ { unsigned long length; (*request)++; hexToInt(request, &length); if ((length <= 0) || (length > MAX_BYTE_CODES)) return error_ret (-1, "Bytecode expression length (%d) too large\n", length); if (*(*request)++ != ',') return error_ret (-1, "Malformed bytecode expr (comma#%d missing)\n", 1); t_expr->next = NULL; /* subtract one to account for expr[0] in header */ t_expr->size = sizeof(struct t_expr_tag) + length - 1; t_expr->expr_size = length; hex2mem(*request, &t_expr->expr[0], length, 0); *request += 2 * length; build_and_add_expression(format->id, t_expr); } break; } } return 0; } static char * handle_trace_set (char *request) { long n_frame; unsigned long frameno, tdp, pc, start, stop; DTC_RESPONSE ret = -1; static COLLECTION_FORMAT_DEF tempfmt1; static char enable; static char retbuf[20]; if (msgcmp (&request, "init")) { gdb_note ("STUB: call clear_trace_state();\n", 0); curframe.valid = 0; /* all old frames become invalid now */ if ((ret = clear_trace_state ()) == OK_TARGET_RESPONSE) return "OK"; else { sprintp (retbuf, "E2%x", ret); return (char *) dtc_error_ret ((int) &retbuf, "clear_trace_state", ret); } } else if (msgcmp (&request, "Start")) { trace_running = 1; curframe.valid = 0; /* all old frames become invalid now */ gdb_note ("STUB: call start_trace_experiment();\n", 0); adbg_save_trace_in_nvd (); if ((ret = start_trace_experiment ()) == OK_TARGET_RESPONSE) return "OK"; else { sprintp (retbuf, "E2%x", ret); return (char *) dtc_error_ret ((int) &retbuf, "start_trace_experiment", ret); } } else if (msgcmp (&request, "Stop")) { trace_running = 0; if (adbg_check_if_active ()) { gdb_note ("STUB: call end_trace_experiment();\n", 0); if ((ret = end_trace_experiment ()) == OK_TARGET_RESPONSE) return "OK"; else { sprintp (retbuf, "E2%x", ret); return (char *) dtc_error_ret ((int) &retbuf, "end_trace_experiment", ret); } } else return "OK"; } /* "TDP:" (The 'T' was consumed in handle_request.) */ else if (msgcmp (&request, "DP:")) { /* TDP:<id>:<addr>:{D,E}:<stepcount>:<pass_limit>{R[M,X]+}<tdp-format> {S{R[M,X]+}}<tp-format> D -- disable tracepoint (illegal from EMC's point of view) E -- enable tracepoint? R -- regs format: R<regs-mask> M -- memory format: M<regnum>,<offset>,<size> X -- expr format: X<size>,<bytecodes> S -- fencepost between trap formats and stepping formats. */ /* state variable, required for splitting TDP packets. */ static int doing_step_formats; /* * TDP: packets may now be split into multiple packets. * If a TDP packet is to be continued in another packet, it * must end in a "-" character. The subsequent continuation * packet will then begin with a "-" character, between the * token "TDP:" and the tdp_id field. The ID and address * will be repeated in each sub-packet. The step_count, * pass_count, and 'enabled' field must appear in the first * packet. The boundary between sub-packets may not appear * between the "S" that denotes the start of stepping "formats", * and the regs_mask that follows it. The split may also not * occur in the middle of either a memrange description or a * bytecode string. -- MVS */ if (*request == '-') /* this is a continuation of a trace definition in progress */ { unsigned long temp_id, temp_addr; request++; if (!(hexToInt (&request, &temp_id) && *request++ == ':')) return "E11"; /* badly formed packet, field 1 */ if (!(hexToInt (&request, (unsigned long *) &temp_addr) && *request++ == ':')) return "E12"; /* badly formed packet, field 2 */ if (temp_id != tdp_temp.id) return "E11"; /* something wrong: field 1 doesn't match */ if (temp_addr != (unsigned long) tdp_temp.addr) return "E12"; /* something wrong: field 2 doesn't match */ } else /* This is a new TDP definition */ { memset ((char *) &tdp_temp, 0, sizeof (tdp_temp)); memset ((char *) &tempfmt1, 0, sizeof (tempfmt1)); doing_step_formats = FALSE; if (!(hexToInt (&request, &tdp_temp.id) && *request++ == ':')) return "E11"; /* badly formed packet, field 1 */ if (!(hexToInt (&request, (unsigned long *) &tdp_temp.addr) && *request++ == ':')) return "E12"; /* badly formed packet, field 2 */ if (!(((enable = *request++) == 'D' || enable == 'E') && *request++ == ':')) return "E13"; /* badly formed packet, field 3 */ #if 0 if (enable == 'D') { gdb_puts ("Disabling of tracepoints not supported by EMC target\n"); return "E20"; } #endif if (!(hexToInt (&request, &tdp_temp.stepcount) && *request++ == ':')) return "E14"; /* badly formed packet, field 4 */ if (!hexToInt (&request, &tdp_temp.pass_limit)) return "E15"; /* badly formed packet, field 5 */ } /* Typically, the first group of collection descriptors refers to the trap collection. There is an "S" token to act as a fencepost between collection descriptors for the trap, and those for the single-stepping. However, when the packet is split up into several packets, this "S" token may already have been seen in a previous sub-packet; so we have to remember it in a state variable. */ if (*request == 'R' || *request == 'M' || *request == 'X') { if (handle_format (&request, &tempfmt1)) return "E16"; if (doing_step_formats) tdp_temp.tp_format_p = tempfmt1.id; else tdp_temp.tdp_format_p = tempfmt1.id; } /* When we see the "S" token, we remember it in a state variable (in case the packet is split up and continued in another message), and discard all current state from the collection "format". */ if (*request == 'S') { doing_step_formats = TRUE; /* discard prev format and start a new one */ memset ((char *) &tempfmt1, 0, sizeof (tempfmt1)); request++; /* Having seen the "S" fencepost, it is now possible that we will see some more collection descriptors pertaining to the stepping collection. */ if (*request == 'R' || *request == 'M' || *request == 'X') { if (handle_format (&request, &tempfmt1)) return "E17"; /* new format ID is tp_format */ tdp_temp.tp_format_p = tempfmt1.id; } } if (*request == '-') /* this TDP definition will be continued. */ sprintp (retbuf, "OK"); else if (enable == 'E') /* end of TDP definition: pass to ADBG (if enabled!) */ { gdb_note ("STUB: call define_tdp (id %d, ", tdp_temp.id); gdb_note ("addr 0x%X, ", (int) tdp_temp.addr); gdb_note ("passc %d, ", tdp_temp.pass_limit); gdb_note ("stepc %d, ", tdp_temp.stepcount); gdb_note ("TDP fmt #%d, ", tdp_temp.tdp_format_p); gdb_note ("TP fmt #%d);\n", tdp_temp.tp_format_p); ret = define_tdp (tdp_temp.id, &tdp_temp, 0); if (ret == OK_TARGET_RESPONSE) { sprintp (retbuf, "OK"); } else { sprintp (spare_buffer, "'define_tdp' returned DTC error '%s' for tracepoint %d.\n", get_err_text (ret), tdp_temp.id); gdb_puts (spare_buffer); sprintp (retbuf, "E2%x", ret); } /* Redundant, but let's try to make sure this state gets discarded. */ { memset ((char *) &tdp_temp, 0, sizeof (tdp_temp)); memset ((char *) &tempfmt1, 0, sizeof (tempfmt1)); } } else /* ADBG_DTC does not support disabled tracepoints -- ignore it. */ gdb_note ("STUB: ignoring disabled tracepoint %d.\n", tdp_temp.id); return retbuf; } else if (msgcmp (&request, "Frame:")) { ret = OK_TARGET_RESPONSE; if (msgcmp (&request, "pc:")) { if (!hexToInt (&request, &pc)) return "E10"; /* badly formed packet */ n_frame = curframe.valid ? curframe.frame_id + 1 : 0; gdb_note ("STUB: call fetch_trace_frame_pc (id %d, ", n_frame); gdb_note ("pc 0x%X);\n", pc); ret = fetch_trace_frame_with_pc (&n_frame, (void *) pc, &curframe.format, &curframe.frame_data); } else if (msgcmp (&request, "tdp:")) { if (!hexToInt (&request, &tdp)) return "E10"; /* badly formed packet */ n_frame = curframe.valid ? curframe.frame_id + 1: 0; gdb_note ("STUB: call fetch_trace_frame_tdp (id %d, ", n_frame); gdb_note ("tdp 0x%X);\n", tdp); ret = fetch_trace_frame_with_tdp (&n_frame, tdp, &curframe.format, &curframe.frame_data); } else if (msgcmp (&request, "range:")) { if (!(hexToInt (&request, &start) && *request++ == ':')) return "E11"; /* badly formed packet, field 1 */ else if (!hexToInt (&request, &stop)) return "E12"; /* badly formed packet, field 2 */ n_frame = curframe.valid ? curframe.frame_id + 1: 0; gdb_note ("STUB: call fetch_trace_frame_range (id %d, ", n_frame); gdb_note ("start 0x%X, ", start); gdb_note ("stop 0x%X);\n", stop); ret = fetch_trace_frame_with_pc_in_range (&n_frame, (void *) start, (void *) stop, &curframe.format, &curframe.frame_data); } else if (msgcmp (&request, "outside:")) { if (!(hexToInt (&request, &start) && *request++ == ':')) return "E11"; /* badly formed packet, field 1 */ else if (!hexToInt (&request, &stop)) return "E12"; /* badly formed packet, field 2 */ n_frame = curframe.valid ? curframe.frame_id + 1: 0; gdb_note ("STUB: call fetch_trace_frame_outside (id %d, ", n_frame); gdb_note ("start 0x%X, ", start); gdb_note ("stop 0x%X);\n", stop); ret = fetch_trace_frame_with_pc_outside (&n_frame, (void *) start, (void *) stop, &curframe.format, &curframe.frame_data); } else /* simple TFind by frame number: */ { if (!hexToInt (&request, &frameno)) return "E10"; /* badly formed packet */ if (frameno != (unsigned long) -1) { gdb_note ("STUB: call fetch_trace_frame (id %d);\n", frameno); ret = fetch_trace_frame (n_frame = frameno, &curframe.format, &curframe.frame_data); #if 0 printp("STUB: fetch_trace_frame: return %d\n", ret); #endif } else /* discard any trace frame, debug "the real world" */ { if (curframe.valid) gdb_note ("STUB: discard current trace frame #%d.\n", curframe.frame_id); curframe.valid = 0; return "OK"; } } if (ret == OK_TARGET_RESPONSE) /* fetch_trace_frame succeeded */ { /* setup for debugging the trace frame */ curframe.valid = 1; curframe.frame_id = n_frame; curframe.tdp_id = curframe.frame_data->id; memset ((char *) &curframe.traceregs, 0, sizeof (curframe.traceregs)); curframe.traceregs[PC] = (unsigned long) curframe.frame_data->program_counter; if (curframe.format) { unsigned long regs_mask = curframe.format->regs_mask; unsigned long *regs, *stack, *mem; unsigned long regno, index = 0; CFD *dummy; if ((ret = get_addr_to_frame_regs_stack_mem (curframe.frame_data, &dummy, ®s, &stack, &mem)) != OK_TARGET_RESPONSE) { curframe.valid = 0; sprintp (retbuf, "E2%x", ret); return (char *) dtc_error_ret ((int) &retbuf, "get_addr_to_frame_regs_stack_mem", ret); } if (remote_debug > 1) { /* echo what we've found to gdb console */ sprintp (spare_buffer, "STUB: Found frame %d, TDP %d, format %d (%s):\n", curframe.frame_id, curframe.tdp_id & 0x7fffffff, curframe.format->id, curframe.tdp_id & 0x80000000 ? "trap frame" : "stepping frame"); gdb_puts (spare_buffer); } /* copy trace frame regs into stub's data format */ for (regno = 0, index = 0; regno < 16; regno++, regs_mask >>= 1) if (regs_mask & 1) /* got a collected register */ { curframe.traceregs[regno] = regs[index++]; if (remote_debug > 1) { sprintp (spare_buffer, " Collected 0x%08x for register %d.\n", curframe.traceregs[regno], regno); gdb_puts (spare_buffer); } } if (remote_debug > 1) { long midx, ridx, len; MEMRANGE_DEF *mrange; unsigned char *data, *base; if (curframe.format->stack_size > 0) { len = curframe.format->stack_size; sprintp (spare_buffer, " Collected %d bytes of stack at 0x%x:\n", len, curframe.traceregs[A7]); gdb_puts (spare_buffer); /* print stack data, but stay under msg len */ if (len >= (NUMREGBYTES/2 - 2)) len = (NUMREGBYTES/2 - 3); mem2hex ((unsigned char *) stack, spare_buffer, len, 0); spare_buffer [len * 2] = '\n'; spare_buffer [len * 2 + 1] = '\0'; /* EOS */ gdb_puts (spare_buffer); } else gdb_puts ("Stack not collected\n"); for (midx = 0; get_addr_to_a_mem_range (curframe.frame_data, midx, &mrange, (void **) &data) == OK_TARGET_RESPONSE; midx++) { if ((mrange->typecode == 0) || (mrange->typecode == (unsigned long) -1)) { sprintp (spare_buffer, " Collected %d bytes at MEM: 0x%x:\n", mrange->size, mrange->offset); base = (unsigned char *) mrange->offset; } else { if ((ridx = elinum_to_index (mrange->typecode)) > 0) base = (unsigned char *) curframe.traceregs[ridx] + (long) mrange->offset; else { sprintp (spare_buffer, "STUB: bad typecode in memrange #%d: (0x%x,0x%x,0x%x).\n", midx, mrange->typecode, mrange->offset, mrange->size); gdb_puts (spare_buffer); continue; } sprintp (spare_buffer, " Collected %d bytes at 0x%x (REG %X + %d):\n", mrange->size, base, mrange->typecode, mrange->offset); } gdb_puts (spare_buffer); len = mrange->size; if (len >= (NUMREGBYTES/2 - 2)) len = (NUMREGBYTES/2 - 3); mem2hex (data, spare_buffer, len, 0); spare_buffer [len * 2] = '\n'; spare_buffer [len * 2 + 1] = '\0'; /* EOS */ gdb_puts (spare_buffer); } } } sprintp (retbuf, "F%xT%x", n_frame, curframe.tdp_id & 0x7fffffff); return retbuf; } else if (ret == NOT_FOUND_TARGET_RESPONSE) { /* Here's a question: if the fetch_trace_frame call failed (which probably means a bad "TFIND" command from GDB), should we remain focused on the previous frame (if any), or should we revert to "no current frame"? */ return "F-1"; } else { sprintp (retbuf, "E2%x", ret); return (char *) dtc_error_ret ((int) &retbuf, "fetch_trace_frame[...]", ret); } } else /* unknown trace command */ { return ""; } } /* Table used by the crc32 function to calcuate the checksum. */ static unsigned long crc32_table[256]; static int crc_mem_err; static unsigned long crc32 (buf, len, crc) unsigned char *buf; int len; unsigned long crc; { crc_mem_err = FALSE; if (! crc32_table[1]) { /* Initialize the CRC table and the decoding table. */ int i, j; unsigned int c; for (i = 0; i < 256; i++) { for (c = i << 24, j = 8; j > 0; --j) c = c & 0x80000000 ? (c << 1) ^ 0x04c11db7 : (c << 1); crc32_table[i] = c; } } while (len--) { if (read_access_violation (buf)) { crc_mem_err = TRUE; return -1; } crc = (crc << 8) ^ crc32_table[((crc >> 24) ^ *buf++) & 255]; } return crc; } static char * crc_query (cmd) char *cmd; { unsigned long startmem, len, crc; static char buf[32]; if (hexToInt (&cmd, &startmem) && *cmd++ == ',' && hexToInt (&cmd, &len)) { crc = crc32 ((unsigned char *) startmem, len, 0xffffffff); if (!crc_mem_err) { sprintp (buf, "C%08x", crc); return buf; } /* else error, fall thru */ } sprintp (buf, "E01"); return buf; } static char * handle_test (request) char *request; { ULONG args[7]; int i; /* Parse the arguments, a comma-separated list of hex numbers, into ARGS. Parse at most six arguments. */ i = 1; if (*request != '\0') while (i < 7) { if (! hexToInt (&request, &args[i++])) return "E01"; if (*request == '\0') break; if (*request++ != ',') return "E01"; } /* Fill the rest of the args array with zeros. This is what the INLINES command processor does with omitted arguments. */ for (; i < 7; i++) args[i] = 0; gdb_c_test (args); return "OK"; } /* GDB_TRAP_1_HANDLER By the time this is called, the registers have been saved in "registers", and the interrupt priority has been set to permit serial UART interrupts. However, since no gdb request has yet been received, and there is no equivalent of getpacket for us to wait on, we can't sit here waiting for packets and processing them. In fact, the ONLY thing for us to do here is sit and wait. As gdb sends packet requests, they will handle themselves at the interrupt level. When gdb decides we can continue, it will reset the global variable "gdb_handling_trap1", and we will return (whereupon registers will be restored etc.) */ void gdb_trap_1_handler( void ) { gdb_handling_trap1 = TRUE; sss_trace_flag = '\0'; /* shut off "trace bit" (indirectly) */ gdb_signo = 5; putpacket( "S05" ); while ( gdb_handling_trap1 ) ; return; } void gdb_trace_handler( void ) { sss_trace_flag = '\0'; /* shut off "trace bit" (indirectly) */ gdb_handling_trap1 = TRUE; gdb_handling_sstrace = TRUE; gdb_signo = 5; putpacket( "S05" ); while ( gdb_handling_trap1 ) ; return; }