📄 h8s_except1.cxx
字号:
//===========================================================================//// except1.cxx//// Exception test 1////===========================================================================//####ECOSGPLCOPYRIGHTBEGIN####// -------------------------------------------// This file is part of eCos, the Embedded Configurable Operating System.// Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.//// eCos is free software; you can redistribute it and/or modify it under// the terms of the GNU General Public License as published by the Free// Software Foundation; either version 2 or (at your option) any later version.//// eCos is distributed in the hope that it will be useful, but WITHOUT ANY// WARRANTY; without even the implied warranty of MERCHANTABILITY or// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License// for more details.//// You should have received a copy of the GNU General Public License along// with eCos; if not, write to the Free Software Foundation, Inc.,// 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): dsm// Contributors: dsm, jlarmour, Uwe Kindler// Date: 2004-02-11// Description: Test basic H8S exception functionality. Checks H8S trap and// trace exceptions.//####DESCRIPTIONEND####//===========================================================================// INCLUDES//===========================================================================#include <pkgconf/kernel.h>#include <cyg/infra/testcase.h>#ifdef CYGPKG_KERNEL_EXCEPTIONS#include <cyg/kernel/sched.hxx> // Cyg_Scheduler::start()#include <cyg/kernel/thread.hxx> // Cyg_Thread#include <cyg/kernel/intr.hxx> // cyg_VSR#include <cyg/hal/hal_intr.h> // exception ranges#include <cyg/kernel/sched.inl>#include <cyg/kernel/thread.inl>#define NTHREADS 1#include "testaux.hxx"//===========================================================================// TYPES//===========================================================================typedef union _exc_status{ cyg_uint8 byte; struct { cyg_uint8 unused : 2; cyg_uint8 err : 1; cyg_uint8 trace : 1; cyg_uint8 trap3 : 1; cyg_uint8 trap2 : 1; cyg_uint8 trap1 : 1; cyg_uint8 trap0 : 1; } bit;} exception_status_t;//===========================================================================// LOCALS//===========================================================================static int d0;static cyg_exception_handler handler0;static CYG_ADDRESS old_vsr_handler[CYGNUM_HAL_EXCEPTION_COUNT];static exception_status_t status; //===========================================================================// EXCEPTION HANDLER 0//===========================================================================static void handler0(CYG_ADDRWORD data, cyg_code number, CYG_ADDRWORD info){ HAL_SavedRegisters *regs = (HAL_SavedRegisters *)info; switch (number) { case CYGNUM_HAL_EXCEPTION_TRAP0: CYG_TEST_INFO("trap #0 exception handler called"); status.bit.trap0 = 1; break; case CYGNUM_HAL_EXCEPTION_TRAP1: CYG_TEST_INFO("trap #1 exception handler called"); status.bit.trap1 = 1; break; case CYGNUM_HAL_EXCEPTION_TRAP2: CYG_TEST_INFO("trap #2 exception handler called"); status.bit.trap2 = 1; break; case CYGNUM_HAL_EXCEPTION_TRAP3: CYG_TEST_INFO("trap #3 exception handler called"); status.bit.trap3 = 1; break; case CYGNUM_HAL_EXCEPTION_TRACE: CYG_TEST_INFO("trace exception handler called"); regs->exr &= ~0x80; // clear trace bit if a trace exception happens status.bit.trace = 1; break; default: CYG_TEST_INFO("unknown exception caused"); status.bit.err = 1; } CYG_TEST_CHECK((CYG_ADDRWORD)123 == data, "handler given wrong data");}//===========================================================================// EXCEPTION HANDLER 1//===========================================================================static void handler1(CYG_ADDRWORD data, cyg_code number, CYG_ADDRWORD info){ CYG_TEST_INFO("handler 1 called"); CYG_TEST_CHECK((CYG_ADDRWORD)&d0 == data, "handler given wrong data");#ifdef CYGSEM_KERNEL_EXCEPTIONS_DECODE CYG_TEST_CHECK(number == CYGNUM_HAL_EXCEPTION_MAX, "handler given wrong number");#else CYG_UNUSED_PARAM(cyg_code, number);#endif CYG_TEST_CHECK((CYG_ADDRWORD)99 == info, "handler given wrong info");}//===========================================================================// CAUSE EXCEPTION//===========================================================================void cause_exception(void){ // // cause a trace exception // asm volatile ( "stc exr,r0l \n\t" "bset #7, r0l \n\t" "ldc r0l,exr \n\t" "mov.l er0, er1\n\t" "mov.l er1, er0\n\t" ); // // now cause H8S trap exceptions // asm volatile ("trapa #0\n"); asm volatile ("trapa #1\n"); asm volatile ("trapa #2\n"); asm volatile ("trapa #3\n");}//===========================================================================// THE MAIN THREAD//===========================================================================static void entry0( CYG_ADDRWORD data ){ cyg_code n; cyg_exception_handler *old_handler, *old_handler1; CYG_ADDRWORD old_data, old_data1; Cyg_Thread *p=Cyg_Thread::self(); CYG_UNUSED_PARAM(CYG_ADDRESS, data); p->register_exception( CYGNUM_HAL_EXCEPTION_MAX, &handler1, (CYG_ADDRWORD)&d0, &old_handler, &old_data); p->register_exception( CYGNUM_HAL_EXCEPTION_MAX, &handler1, (CYG_ADDRWORD)&d0, &old_handler1, &old_data1); CYG_TEST_CHECK(old_handler1 == &handler1, "register exception: old_handler not the one previously registered"); CYG_TEST_CHECK(old_data1 == (CYG_ADDRWORD)&d0, "register exception: old_data not those previously registered"); p->deliver_exception(CYGNUM_HAL_EXCEPTION_MAX, (CYG_ADDRWORD)99); CYG_TEST_INFO("handler 1 returned"); p->deregister_exception(CYGNUM_HAL_EXCEPTION_MAX); p->deregister_exception(CYGNUM_HAL_EXCEPTION_MAX); for(n = CYGNUM_HAL_EXCEPTION_MIN; n <= CYGNUM_HAL_EXCEPTION_MAX; n++) { p->register_exception( n, handler0, (CYG_ADDRWORD)123, &old_handler1, &old_data1); } CYG_TEST_PASS("Attempting to provoke exception"); status.byte = 0; cause_exception(); // // Reset the exception handlers - after these calls it is safe to place // breakpoints because trap handlers are reset to RedBoot trap handlers //#ifdef CYGNUM_HAL_EXCEPTION_TRACE HAL_VSR_SET( CYGNUM_HAL_EXCEPTION_TRACE, old_vsr_handler[0], NULL );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP0 HAL_VSR_SET( CYGNUM_HAL_EXCEPTION_TRAP0, old_vsr_handler[1], NULL );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP1 HAL_VSR_SET( CYGNUM_HAL_EXCEPTION_TRAP1, old_vsr_handler[2], NULL );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP2 HAL_VSR_SET( CYGNUM_HAL_EXCEPTION_TRAP2, old_vsr_handler[3], NULL );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP3 HAL_VSR_SET( CYGNUM_HAL_EXCEPTION_TRAP3, old_vsr_handler[4], NULL );#endif CYG_TEST_CHECK(status.bit.trace == 1, "trace exception not triggered"); CYG_TEST_CHECK(status.bit.trap0 == 1, "trap #0 exception not triggered"); CYG_TEST_CHECK(status.bit.trap1 == 1, "trap #1 exception not triggered"); CYG_TEST_CHECK(status.bit.trap2 == 1, "trap #2 exception not triggered"); CYG_TEST_CHECK(status.bit.trap3 == 1, "trap #3 exception not triggered"); CYG_TEST_CHECK(status.bit.err == 0, "Unknown exception caused"); CYG_TEST_PASS_FINISH("Exception handling OK");}//===========================================================================// THE MAIN ROUTINE//===========================================================================void except0_main( void ){ // Use CYG_TEST_GDBCMD _before_ CYG_TEST_INIT() CYG_TEST_GDBCMD("handle SIGBUS nostop"); CYG_TEST_GDBCMD("handle SIGSEGV nostop"); CYG_TEST_GDBCMD("handle SIGFPE nostop"); CYG_TEST_INIT();#ifdef HAL_VSR_SET_TO_ECOS_HANDLER // // Reclaim the VSRs off RedBoot possibly. We claim also the trap vectors // required for GDB breakpoint support so it is not possible to place a // breakpoint after these calls. //#ifdef CYGNUM_HAL_EXCEPTION_TRACE HAL_VSR_SET_TO_ECOS_HANDLER( CYGNUM_HAL_EXCEPTION_TRACE, &old_vsr_handler[0] );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP0 HAL_VSR_SET_TO_ECOS_HANDLER( CYGNUM_HAL_EXCEPTION_TRAP0, &old_vsr_handler[1] );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP1 HAL_VSR_SET_TO_ECOS_HANDLER( CYGNUM_HAL_EXCEPTION_TRAP1, &old_vsr_handler[2] );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP2 HAL_VSR_SET_TO_ECOS_HANDLER( CYGNUM_HAL_EXCEPTION_TRAP2, &old_vsr_handler[3] );#endif#ifdef CYGNUM_HAL_EXCEPTION_TRAP3 HAL_VSR_SET_TO_ECOS_HANDLER( CYGNUM_HAL_EXCEPTION_TRAP3, &old_vsr_handler[4] );#endif#endif new_thread(entry0, 0); Cyg_Scheduler::start(); CYG_TEST_FAIL_FINISH("Not reached");}//===========================================================================// KERNEL APPLICATION TRANSITION//===========================================================================externC void cyg_start( void ){#ifdef CYGSEM_HAL_STOP_CONSTRUCTORS_ON_FLAG cyg_hal_invoke_constructors();#endif except0_main();}#else // def CYGPKG_KERNEL_EXCEPTIONSexternC voidcyg_start( void ){ CYG_TEST_INIT(); CYG_TEST_NA("Exceptions disabled");}#endif // def CYGPKG_KERNEL_EXCEPTIONS//---------------------------------------------------------------------------// EOF except1.cxx
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -