⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sparc_ex.c

📁 ecos为实时嵌入式操作系统
💻 C
字号:
/*=================================================================////        sparc_ex.c////        SPARClite HAL exception and register manipulation test////==========================================================================//####COPYRIGHTBEGIN####//// -------------------------------------------// The contents of this file are subject to the Cygnus eCos Public License// Version 1.0 (the "License"); you may not use this file except in// compliance with the License.  You may obtain a copy of the License at// http://sourceware.cygnus.com/ecos// // Software distributed under the License is distributed on an "AS IS"// basis, WITHOUT WARRANTY OF ANY KIND, either express or implied.  See the// License for the specific language governing rights and limitations under// the License.// // The Original Code is eCos - Embedded Cygnus Operating System, released// September 30, 1998.// // The Initial Developer of the Original Code is Cygnus.  Portions created// by Cygnus are Copyright (C) 1998,1999 Cygnus Solutions.  All Rights Reserved.// -------------------------------------------////####COPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s):     dsm// Contributors:    dsm, nickg// Date:          1998-06-18//####DESCRIPTIONEND####*/#include <pkgconf/system.h>#include <pkgconf/hal.h>#include <cyg/infra/cyg_type.h>#include <cyg/infra/testcase.h>#include <cyg/hal/hal_arch.h>#include <cyg/hal/hal_intr.h>#include <pkgconf/infra.h>#ifdef CYGDBG_USE_TRACING#define OUTER_REPEATS 1#define SIM_REPEATS  10#define HW_REPEATS 1000 #else#define OUTER_REPEATS 10#define SIM_REPEATS  100#define HW_REPEATS 10000 #endif // using tracing to slow everything down// -------------------------------------------------------------------------// These routines are used to cause an alignment trap; the compiler is too// darned clever by half, if you try to inline this stuff as macros it uses// different instruction sequences and register pairs.  This makes for a// less thorough test, but there's no option other than writing a LOT of// assembler code.// Further, with -O3, the compiler inlines these anyway and so makes// non-trapping code.  So they are now at the end, to prevent this.extern cyg_uint64 get_ll( cyg_uint64 *p );extern cyg_uint64 get_llplus( cyg_uint64 *p );extern cyg_uint32 get_i( cyg_uint32 *p );extern cyg_uint32 get_iplus( cyg_uint32 *p );extern cyg_uint16 get_s( cyg_uint16 *p );extern cyg_uint16 get_splus( cyg_uint16 *p );// -------------------------------------------------------------------------// Some memory to read in more-or-less aligned manners.#define L1 (0x123456789abcdef0l)#define L2 (0xfedcba9876543210l)static cyg_uint64 a[ 2 ] = {    L1,    L2};#define M32 (0x00000000ffffffffl)#define M16 (0x000000000000ffffl)#define M8  (0x00000000000000ffl)volatile cyg_uint32 trap = 0;volatile cyg_uint32 same = 0;volatile cyg_uint32 tcount = 0;// -------------------------------------------------------------------------// This macro invokes a routine then checks that a suitable trap occurred.// It expects the instruction simply to be skipped, rather than the trap// _handled_ or the unaligned access to be _emulated_ in any way.  This// test is just a proof that we could write such a handler.#define TRAPPROC( _var_, _type_, _align_, _proc_ )                         \CYG_MACRO_START                                                            \    otrap = trap;                                                          \    otcount = tcount;                                                      \    _var_ = _proc_( (_type_ *)(cp + (_align_)) );                          \    CYG_TEST_CHECK( trap != otrap || same > 0,                             \                    "No trap [" #_type_ ":" #_align_ "," #_proc_ "]" );    \    CYG_TEST_CHECK( same < 20,                                             \        "Undetected trap loop[" #_type_ ":" #_align_ "," #_proc_ "]" );    \    CYG_TEST_CHECK( tcount > otcount,                                      \        "No trap counted [" #_type_ ":" #_align_ "," #_proc_ "]" );        \    CYG_TEST_CHECK( (tcount - 1) <= otcount,                               \        "Tcount overinc [" #_type_ ":" #_align_ "," #_proc_ "]" );         \CYG_MACRO_END// and this one expects no trap to occur:#define SAFEPROC( _var_, _type_, _align_, _proc_ )                         \CYG_MACRO_START                                                            \    trap = 0;                                                              \    otcount = tcount;                                                      \    _var_ = _proc_( (_type_ *)(cp + (_align_)) );                          \    CYG_TEST_CHECK( 0 == trap,                                             \                        "Trap [" #_type_ ":" #_align_ "," #_proc_ "]" );   \    CYG_TEST_CHECK( tcount == otcount,                                     \            "Trap counted [" #_type_ ":" #_align_ "," #_proc_ "]" );       \CYG_MACRO_ENDstatic void do_test( void ){    cyg_uint32 *ip = (cyg_uint32 *)a;    cyg_uint16 *sp = (cyg_uint16 *)a;    cyg_uint8  *cp = (cyg_uint8  *)a;        cyg_uint64 l;    cyg_uint32 i;    cyg_uint16 s;    cyg_uint8  c;    cyg_int32 z, repeats;    cyg_uint32 otrap;    cyg_uint32 otcount;    otrap = trap = 0;    otcount = tcount;    // First test interestingly aligned accesses that are legal.    l = a[0];  CYG_TEST_CHECK( L1 == l, "a[0] read bad" );    l = a[1];  CYG_TEST_CHECK( L2 == l, "a[1] read bad" );    i = ip[0];  CYG_TEST_CHECK( ((L1 >> 32) & M32) == i, "ip[0]" );    i = ip[1];  CYG_TEST_CHECK( ((L1      ) & M32) == i, "ip[1]" );    i = ip[2];  CYG_TEST_CHECK( ((L2 >> 32) & M32) == i, "ip[2]" );    i = ip[3];  CYG_TEST_CHECK( ((L2      ) & M32) == i, "ip[3]" );                                       s = sp[0];  CYG_TEST_CHECK( ((L1 >> 48) & M16) == s, "sp[0]" );    s = sp[1];  CYG_TEST_CHECK( ((L1 >> 32) & M16) == s, "sp[1]" );    s = sp[2];  CYG_TEST_CHECK( ((L1 >> 16) & M16) == s, "sp[2]" );    s = sp[3];  CYG_TEST_CHECK( ((L1      ) & M16) == s, "sp[3]" );    s = sp[4];  CYG_TEST_CHECK( ((L2 >> 48) & M16) == s, "sp[4]" );    s = sp[5];  CYG_TEST_CHECK( ((L2 >> 32) & M16) == s, "sp[5]" );    s = sp[6];  CYG_TEST_CHECK( ((L2 >> 16) & M16) == s, "sp[6]" );    s = sp[7];  CYG_TEST_CHECK( ((L2      ) & M16) == s, "sp[7]" );                                       c = cp[0];  CYG_TEST_CHECK( ((L1 >> 56) & M8) == c, "cp[0]" );    c = cp[1];  CYG_TEST_CHECK( ((L1 >> 48) & M8) == c, "cp[1]" );    c = cp[2];  CYG_TEST_CHECK( ((L1 >> 40) & M8) == c, "cp[2]" );    c = cp[3];  CYG_TEST_CHECK( ((L1 >> 32) & M8) == c, "cp[3]" );    c = cp[4];  CYG_TEST_CHECK( ((L1 >> 24) & M8) == c, "cp[4]" );    c = cp[5];  CYG_TEST_CHECK( ((L1 >> 16) & M8) == c, "cp[5]" );    c = cp[6];  CYG_TEST_CHECK( ((L1 >>  8) & M8) == c, "cp[6]" );    c = cp[7];  CYG_TEST_CHECK( ((L1      ) & M8) == c, "cp[7]" );    c = cp[8];  CYG_TEST_CHECK( ((L2 >> 56) & M8) == c, "cp[8]" );    c = cp[9];  CYG_TEST_CHECK( ((L2 >> 48) & M8) == c, "cp[9]" );    c = cp[10]; CYG_TEST_CHECK( ((L2 >> 40) & M8) == c, "cp[10]" );    c = cp[11]; CYG_TEST_CHECK( ((L2 >> 32) & M8) == c, "cp[11]" );    c = cp[12]; CYG_TEST_CHECK( ((L2 >> 24) & M8) == c, "cp[12]" );    c = cp[13]; CYG_TEST_CHECK( ((L2 >> 16) & M8) == c, "cp[13]" );    c = cp[14]; CYG_TEST_CHECK( ((L2 >>  8) & M8) == c, "cp[14]" );    c = cp[15]; CYG_TEST_CHECK( ((L2      ) & M8) == c, "cp[15]" );    CYG_TEST_CHECK( 0 == trap, "Traps occurred (legal accesses)" );    CYG_TEST_CHECK( tcount == otcount, "Traps counted (legal accesses)" );    CYG_TEST_PASS( "Aligned accesses OK" );    for ( z = OUTER_REPEATS; z > 0; z-- ) {        for ( repeats = (cyg_test_is_simulator ? SIM_REPEATS : HW_REPEATS) ;              repeats > 0 ; repeats-- ) {            TRAPPROC( l, cyg_uint64, 4, get_llplus );            TRAPPROC( l, cyg_uint64, 5, get_llplus );            TRAPPROC( l, cyg_uint64, 6, get_llplus );            TRAPPROC( l, cyg_uint64, 7, get_llplus );            TRAPPROC( l, cyg_uint64, 4, get_ll );            TRAPPROC( l, cyg_uint64, 1, get_ll );            TRAPPROC( l, cyg_uint64, 2, get_ll );            TRAPPROC( l, cyg_uint64, 3, get_ll );            TRAPPROC( i, cyg_uint32, 1, get_iplus );            TRAPPROC( i, cyg_uint32, 2, get_iplus );            TRAPPROC( i, cyg_uint32, 3, get_iplus );            TRAPPROC( i, cyg_uint32, 5, get_i );            TRAPPROC( i, cyg_uint32, 6, get_i );            TRAPPROC( i, cyg_uint32, 7, get_i );            TRAPPROC( s, cyg_uint16, 1, get_splus );            TRAPPROC( s, cyg_uint16, 3, get_splus );            TRAPPROC( s, cyg_uint16, 5, get_s );            TRAPPROC( s, cyg_uint16, 7, get_s );        }        CYG_TEST_PASS( "Unaligned accesses OK" );        // Now test some legal and illegal accesses intermingled.        for ( repeats = (cyg_test_is_simulator ? SIM_REPEATS : HW_REPEATS) ;              repeats > 0 ; repeats-- ) {            SAFEPROC( l, cyg_uint64, 0, get_llplus );            TRAPPROC( l, cyg_uint64, 5, get_llplus );            TRAPPROC( l, cyg_uint64, 6, get_llplus );            SAFEPROC( l, cyg_uint64, 8, get_llplus );            TRAPPROC( i, cyg_uint32, 1, get_iplus );            SAFEPROC( i, cyg_uint32, 4, get_iplus );            TRAPPROC( i, cyg_uint32, 2, get_iplus );            SAFEPROC( i, cyg_uint32, 8, get_iplus );            SAFEPROC( i, cyg_uint32, 12, get_iplus );            SAFEPROC( i, cyg_uint32, 16, get_iplus );            TRAPPROC( i, cyg_uint32, 3, get_iplus );            TRAPPROC( s, cyg_uint16, 5, get_s );            SAFEPROC( s, cyg_uint16, 6, get_s );            TRAPPROC( s, cyg_uint16, 7, get_s );            SAFEPROC( s, cyg_uint16, 8, get_s );             TRAPPROC( i, cyg_uint32, 5, get_i );            SAFEPROC( i, cyg_uint32, 4, get_i );            TRAPPROC( i, cyg_uint32, 6, get_i );            TRAPPROC( i, cyg_uint32, 7, get_i );            SAFEPROC( i, cyg_uint32, 0, get_i );            TRAPPROC( l, cyg_uint64, 4, get_ll );            SAFEPROC( l, cyg_uint64, 0, get_ll );            TRAPPROC( l, cyg_uint64, 1, get_ll );            SAFEPROC( l, cyg_uint64, 8, get_ll );            TRAPPROC( s, cyg_uint16, 1, get_splus );            SAFEPROC( s, cyg_uint16, 2, get_splus );            TRAPPROC( s, cyg_uint16, 3, get_splus );            SAFEPROC( s, cyg_uint16, 4, get_splus );        }        CYG_TEST_PASS( "Mixture of accesses OK" );    }}// -------------------------------------------------------------------------externC voidskip_exception_handler(CYG_ADDRWORD vector, CYG_ADDRWORD data,                                   CYG_ADDRWORD stackpointer);externC voidfail_exception_handler(CYG_ADDRWORD vector, CYG_ADDRWORD data,                                   CYG_ADDRWORD stackpointer);// -------------------------------------------------------------------------void sparc_ex_main( void ){    int i;    CYG_TEST_INIT();    for ( i = CYGNUM_HAL_EXCEPTION_MIN; i <= CYGNUM_HAL_EXCEPTION_MAX; i++ ){        int j;        HAL_TRANSLATE_VECTOR( i, j );        HAL_INTERRUPT_ATTACH( j, &fail_exception_handler, j, 0 );        // we must also ensure that eCos handles the exception;        // do not drop into CygMon or equivalent.        // Leave USER_TRAP undisturbed so that breakpoints work.        if ( CYGNUM_HAL_VECTOR_USER_TRAP != i ) {            extern void hal_default_exception_vsr( void );            HAL_VSR_SET( i, (CYG_ADDRESS)hal_default_exception_vsr, NULL );        }    }    HAL_TRANSLATE_VECTOR( CYGNUM_HAL_VECTOR_UNALIGNED, i );    HAL_INTERRUPT_DETACH( i, &fail_exception_handler );    HAL_INTERRUPT_ATTACH( i, &skip_exception_handler, i, 0 );    CYG_TEST_INFO( "Vectors attached OK; calling do_test" );    do_test();    CYG_TEST_EXIT( "Done" );}// -------------------------------------------------------------------------externC voidskip_exception_handler(CYG_ADDRWORD vector, CYG_ADDRWORD data,                                   CYG_ADDRWORD stackpointer){    HAL_SavedRegisters *save;    HAL_FrameStructure *frame;    HAL_FLUSH_REGISTERS_TO_STACK();    save = (HAL_SavedRegisters *) stackpointer;    frame = (HAL_FrameStructure *) (save + 1); // immediately after    // now, this is the invokation environment when this saved regset    // was created (copied from hal_xvsr.S):    //  ! here,locals have been set up as follows:    //  ! %l0 = psr (with this CWP/window-level in it)    //  ! %l1 = pc    //  ! %l2 = npc    //  ! %l3 = vector number (16-25 for traps)    //  ! and we are in our own register window, though it is likely that    //  ! the next one will need to be saved before we can use it:    //  ! ie. this one is the invalid register window.    // and the intention is that we can mess therewith:    // Check we're not in a trap loop    if ( trap == save->li.l[1] ) {        same++;        if ( 10 < same )            CYG_TEST_FAIL_EXIT( "Repeated trap" );    }    else // restart the count       same = 0;    // and record it    trap = save->li.l[1];    tcount++;    // now step on so that we return to the instruction after:    save->li.l[1] = save->li.l[2]; // PC := NPC    save->li.l[2] += 4;            // NPC += 4    // that's all.}externC voidfail_exception_handler(CYG_ADDRWORD vector, CYG_ADDRWORD data,                                   CYG_ADDRWORD stackpointer){    HAL_FLUSH_REGISTERS_TO_STACK();    CYG_TEST_FAIL_EXIT( "Other exception handler called" );}// -------------------------------------------------------------------------externC void#ifdef CYGPKG_KERNELcyg_user_start( void )#elsecyg_start( void )#endif{    sparc_ex_main();}// -------------------------------------------------------------------------cyg_uint64 get_ll( cyg_uint64 *p ){    return *p;}cyg_uint64 get_llplus( cyg_uint64 *p ){    cyg_uint64 ll = 0l, v;    ll = (cyg_uint32)p;    ll++;    v = *p;    v^= ll;    return v;}cyg_uint32 get_i( cyg_uint32 *p ){    return *p;}cyg_uint32 get_iplus( cyg_uint32 *p ){    cyg_uint32 i = 0, v;    i = (cyg_uint32)p;    i++;    v = *p;    v^= i;    return v;}cyg_uint16 get_s( cyg_uint16 *p ){    return *p;}cyg_uint16 get_splus( cyg_uint16 *p ){    cyg_uint16 s = 0, v;    s = (cyg_uint16)(0xffff & (cyg_uint32)p);    s++;    v = *p;    v^= s;    return v;}// -------------------------------------------------------------------------/* EOF sparc_ex.c */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -