sh_stub.c
来自「eCos操作系统源码」· C语言 代码 · 共 398 行
C
398 行
//==========================================================================//// sh_stub.c//// GDB Stub code for SH////==========================================================================//####ECOSGPLCOPYRIGHTBEGIN####// -------------------------------------------// This file is part of eCos, the Embedded Configurable Operating System.// Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.// Copyright (C) 2003 Nick Garnett //// 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.,// 59 Temple Place, Suite 330, Boston, MA 02111-1307 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.//// This exception does not invalidate any other reasons why a work based on// this file might be covered by the GNU General Public License.//// Alternative licenses for eCos may be arranged by contacting Red Hat, Inc.// at http://sources.redhat.com/ecos/ecos-license/// -------------------------------------------//####ECOSGPLCOPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s): jskov// Contributors: jskov, Ben Lee, Steve Chamberlain, nickg// Date: 1999-05-18// Description: GDB Stub support for sh CPU.////####DESCRIPTIONEND####////===========================================================================#include <stddef.h>#include <pkgconf/hal.h>#ifdef CYGPKG_REDBOOT#include <pkgconf/redboot.h>#endif#ifdef CYGDBG_HAL_DEBUG_GDB_INCLUDE_STUBS#include <cyg/hal/sh_regs.h>#include <cyg/hal/hal_stub.h>#include <cyg/hal/hal_arch.h>#include <cyg/hal/hal_intr.h>#ifdef CYGDBG_HAL_DEBUG_GDB_THREAD_SUPPORT#include <cyg/hal/dbg-threads-api.h> // dbg_currthread_id#endif/* Given a trap value TRAP, return the corresponding signal. */int __computeSignal(unsigned int trap_number){ switch( trap_number ) { case CYGNUM_HAL_EXCEPTION_TRAP:#ifdef CYGNUM_HAL_EXCEPTION_INSTRUCTION_BP case CYGNUM_HAL_EXCEPTION_INSTRUCTION_BP:#endif return SIGTRAP; case CYGNUM_HAL_EXCEPTION_POWERON: case CYGNUM_HAL_EXCEPTION_RESET: // Reset - given that the CPU resets if it gets confused we // want to treat it as an interupt so the developer has a // chance to find out what happened (as opposed to SIGTERM). return SIGINT;#ifdef CYGNUM_HAL_EXCEPTION_TLBMISS_ACCESS case CYGNUM_HAL_EXCEPTION_TLBMISS_ACCESS: case CYGNUM_HAL_EXCEPTION_TLBMISS_WRITE: case CYGNUM_HAL_EXCEPTION_INITIAL_WRITE: case CYGNUM_HAL_EXCEPTION_TLBERROR_ACCESS: case CYGNUM_HAL_EXCEPTION_TLBERROR_WRITE: return SIGSEGV; case CYGNUM_HAL_EXCEPTION_DATA_ACCESS: case CYGNUM_HAL_EXCEPTION_DATA_WRITE: return SIGBUS;#else case CYGNUM_HAL_EXCEPTION_DATA_ACCESS: case CYGNUM_HAL_EXCEPTION_DMA_DATA_ACCESS: return SIGSEGV;#endif case CYGNUM_HAL_EXCEPTION_ILLEGAL_INSTRUCTION: case CYGNUM_HAL_EXCEPTION_ILLEGAL_SLOT_INSTRUCTION: return SIGILL; default: return SIGTERM; }}/* Return the trap number corresponding to the last-taken trap. */int __get_trap_number(void){ // The vector is not not part of the GDB register set so get it // directly from the save context. return _hal_registers->event;}#if defined(CYGSEM_REDBOOT_BSP_SYSCALLS)int __is_bsp_syscall(void) {#if defined(CYGARC_REG_TRA) if (_hal_registers->event == 11 && (*(CYG_WORD *)CYGARC_REG_TRA) == (34<<2))#else if (_hal_registers->event == 34 )#endif return 1; else return 0;}#endif/* Set the currently-saved pc register value to PC. This also updates NPC as needed. */void set_pc(target_register_t pc){ put_register(PC, pc);}#ifdef CYGARC_SH_MOD_UBC// This implementation of the single-stepper relies on the User Break// Controller which may not be available on all cores.// Note: This should be enhanced to coorperate with either two regular// breakpoints or watchpoints. Requires GDB to be aware of the stub's// ability though, so for now just use channel A without further// considerations./* Set things up so that the next user resume will execute one instruction. This may be done by setting breakpoints or setting a single step flag in the saved user registers, for example. */void __single_step (void){ // The address of the instruction to execute. HAL_WRITE_UINT32(CYGARC_REG_BARA, get_register(PC)); // Match entire address.#if (CYGARC_SH_MOD_UBC == 1) HAL_WRITE_UINT8(CYGARC_REG_BAMRA, CYGARC_REG_BAMRA_BARA_UNMASKED);#elif (CYGARC_SH_MOD_UBC == 2) // For some reason, matching all bits causes the system to hang // (not just run amok - it appears to stop execution). //HAL_WRITE_UINT32(CYGARC_REG_BAMRA, 0xffffffff); HAL_WRITE_UINT32(CYGARC_REG_BAMRA, 0);#endif // Stop after instruction at matching address has executed.#if (CYGARC_SH_MOD_UBC == 1) HAL_WRITE_UINT16(CYGARC_REG_BRCR, CYGARC_REG_BRCR_ONE_STEP);#else HAL_WRITE_UINT32(CYGARC_REG_BRCR, CYGARC_REG_BRCR_ONE_STEP);#endif // Stop on IFETCH/READ#if (CYGARC_SH_MOD_UBC == 1) HAL_WRITE_UINT16(CYGARC_REG_BBRA, CYGARC_REG_BBRA_IFETCH|CYGARC_REG_BBRA_READ);#else HAL_WRITE_UINT16(CYGARC_REG_BBRA, CYGARC_REG_BBRA_CPU|CYGARC_REG_BBRA_IFETCH|CYGARC_REG_BBRA_READ);#endif#ifdef CYGPKG_HAL_SH_SH4 // Must execute at least 11 instructions before reaching // any address that may be affected by the UBC settings. asm volatile ("nop;nop;nop;nop;nop;nop;nop;nop;nop;nop;nop;");#endif}/* Clear the single-step state. */void __clear_single_step (void){ // Don't stop on any condition HAL_WRITE_UINT16(CYGARC_REG_BBRA, 0); // Clear status flags HAL_WRITE_UINT16(CYGARC_REG_BRCR, 0);#ifdef CYGPKG_HAL_SH_SH4 // Must execute at least 11 instructions before reaching // any address that may be affected by the UBC settings. asm volatile ("nop;nop;nop;nop;nop;nop;nop;nop;nop;nop;nop;");#endif}#else // CYGARC_SH_MOD_UBC/*---------------------------------------------------------------------- * Single-step support, copied from gdb/sh-stub.c, written by Ben Lee * and Steve Chamberlain. Extended to handle braf, bsrf, bt/s and bf/s. */#define COND_BRx_MASK 0xfb00#define UCOND_DBR_MASK 0xe000#define UCOND_RBR_MASK 0xf0df#define TRAPA_MASK 0xff00#define COND_DISP 0x00ff#define UCOND_DISP 0x0fff#define UCOND_REG 0x0f00#define BFx_INSTR 0x8b00#define BTx_INSTR 0x8900#define BRA_INSTR 0xa000#define BSR_INSTR 0xb000#define JMP_INSTR 0x402b#define JSR_INSTR 0x400b#define RTS_INSTR 0x000b#define RTE_INSTR 0x002b#define TRAPA_INSTR 0xc300#define BxxF_INSTR 0x0003#define SSTEP_INSTR 0xc320#define T_BIT_MASK 0x0001typedef struct { short *memAddr; short oldInstr;} stepData;static stepData instrBuffer;static char stepped;/* Set things up so that the next user resume will execute one instruction. This may be done by setting breakpoints or setting a single step flag in the saved user registers, for example. */void __single_step (void){ short *instrMem; int displacement; int reg; unsigned short opcode; cyg_uint32 pc = get_register(PC); cyg_uint32 sr = get_register(SR); instrMem = (short *) pc; opcode = *instrMem; stepped = 1; if ((opcode & UCOND_RBR_MASK) == BxxF_INSTR) { reg = (char) ((opcode & UCOND_REG) >> 8); displacement = get_register(reg); instrMem = (short *) (pc + displacement + 4); /* * Remember PC points to second instr. * after PC of branch ... so add 4 */ } else if ((opcode & COND_BRx_MASK) == BTx_INSTR) { if (sr & T_BIT_MASK) { displacement = (opcode & COND_DISP) << 1; if (displacement & 0x80) { displacement |= 0xffffff00; } /* * Remember PC points to second instr. * after PC of branch ... so add 4 */ instrMem = (short *) (pc + displacement + 4); } else { instrMem += 1; } } else if ((opcode & COND_BRx_MASK) == BFx_INSTR) { if (sr & T_BIT_MASK) { instrMem += 1; } else { displacement = (opcode & COND_DISP) << 1; if (displacement & 0x80) displacement |= 0xffffff00; /* * Remember PC points to second instr. * after PC of branch ... so add 4 */ instrMem = (short *) (pc + displacement + 4); } } else if ((opcode & UCOND_DBR_MASK) == BRA_INSTR) { displacement = (opcode & UCOND_DISP) << 1; if (displacement & 0x0800) { displacement |= 0xfffff000; } /* * Remember PC points to second instr. * after PC of branch ... so add 4 */ instrMem = (short *) (pc + displacement + 4); } else if ((opcode & UCOND_RBR_MASK) == JSR_INSTR) { reg = (char) ((opcode & UCOND_REG) >> 8); instrMem = (short *) get_register(reg); } else if (opcode == RTS_INSTR) { instrMem = (short *) get_register(PR); } else if (opcode == RTE_INSTR) { instrMem = (short *) get_register(SP); } else if ((opcode & TRAPA_MASK) == TRAPA_INSTR) { instrMem += 1; // skip traps } else { instrMem += 1; } instrBuffer.memAddr = instrMem; instrBuffer.oldInstr = *instrMem; *instrMem = SSTEP_INSTR;}/* Clear the single-step state. */void __clear_single_step (void){ /* Undo the effect of a previous doSStep. If we single stepped, restore the old instruction. */ if (stepped) { short *instrMem; instrMem = instrBuffer.memAddr; *instrMem = instrBuffer.oldInstr; } stepped = 0;}#endif // CYGARC_SH_MOD_UBCvoid __install_breakpoints (void){ /* NOP since single-step HW exceptions are used instead of breakpoints. */}void __clear_breakpoints (void){}/* If the breakpoint we hit is in the breakpoint() instruction, return a non-zero value. */#ifdef linuxexternC void _breakinst(void);int__is_breakpoint_function (){ return get_register(PC) == (target_register_t)&_breakinst;}#elseexternC void breakinst(void);int__is_breakpoint_function (){ return get_register(PC) == (target_register_t)&breakinst;}#endif/* Skip the current instruction. Since this is only called by the stub when the PC points to a breakpoint or trap instruction, we can safely just skip 2. */void __skipinst (void){ put_register(PC, get_register (PC) + 2);}#endif // CYGDBG_HAL_DEBUG_GDB_INCLUDE_STUBS
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?