| /* Copyright (C) 2003 Analog Devices, Inc. All Rights Reserved. |
| * |
| * This file is subject to the terms and conditions of the GNU General Public |
| * License. |
| * |
| * Blackfin BF533/2.6 support : LG Soft India |
| */ |
| |
| |
| /* Include an exception handler to invoke the CPLB manager |
| */ |
| |
| #include <asm-blackfin/linkage.h> |
| #include <asm/cplb.h> |
| #include <asm/entry.h> |
| |
| |
| .text |
| |
| .globl _cplb_hdr; |
| .type _cplb_hdr, STT_FUNC; |
| .extern _cplb_mgr; |
| .type _cplb_mgr, STT_FUNC; |
| .extern __unknown_exception_occurred; |
| .type __unknown_exception_occurred, STT_FUNC; |
| .extern __cplb_miss_all_locked; |
| .type __cplb_miss_all_locked, STT_FUNC; |
| .extern __cplb_miss_without_replacement; |
| .type __cplb_miss_without_replacement, STT_FUNC; |
| .extern __cplb_protection_violation; |
| .type __cplb_protection_violation, STT_FUNC; |
| .extern panic_pv; |
| |
| .align 2; |
| |
| ENTRY(_cplb_hdr) |
| SSYNC; |
| [--SP] = ( R7:0, P5:0 ); |
| [--SP] = ASTAT; |
| [--SP] = SEQSTAT; |
| [--SP] = I0; |
| [--SP] = I1; |
| [--SP] = I2; |
| [--SP] = I3; |
| [--SP] = LT0; |
| [--SP] = LB0; |
| [--SP] = LC0; |
| [--SP] = LT1; |
| [--SP] = LB1; |
| [--SP] = LC1; |
| R2 = SEQSTAT; |
| |
| /*Mask the contents of SEQSTAT and leave only EXCAUSE in R2*/ |
| R2 <<= 26; |
| R2 >>= 26; |
| |
| R1 = 0x23; /* Data access CPLB protection violation */ |
| CC = R2 == R1; |
| IF !CC JUMP not_data_write; |
| R0 = 2; /* is a write to data space*/ |
| JUMP is_icplb_miss; |
| |
| not_data_write: |
| R1 = 0x2C; /* CPLB miss on an instruction fetch */ |
| CC = R2 == R1; |
| R0 = 0; /* is_data_miss == False*/ |
| IF CC JUMP is_icplb_miss; |
| |
| R1 = 0x26; |
| CC = R2 == R1; |
| IF !CC JUMP unknown; |
| |
| R0 = 1; /* is_data_miss == True*/ |
| |
| is_icplb_miss: |
| |
| #if ( defined (CONFIG_BLKFIN_CACHE) || defined (CONFIG_BLKFIN_DCACHE)) |
| #if ( defined (CONFIG_BLKFIN_CACHE) && !defined (CONFIG_BLKFIN_DCACHE)) |
| R1 = CPLB_ENABLE_ICACHE; |
| #endif |
| #if ( !defined (CONFIG_BLKFIN_CACHE) && defined (CONFIG_BLKFIN_DCACHE)) |
| R1 = CPLB_ENABLE_DCACHE; |
| #endif |
| #if ( defined (CONFIG_BLKFIN_CACHE) && defined (CONFIG_BLKFIN_DCACHE)) |
| R1 = CPLB_ENABLE_DCACHE | CPLB_ENABLE_ICACHE; |
| #endif |
| #else |
| R1 = 0; |
| #endif |
| |
| [--SP] = RETS; |
| CALL _cplb_mgr; |
| RETS = [SP++]; |
| CC = R0 == 0; |
| IF !CC JUMP not_replaced; |
| LC1 = [SP++]; |
| LB1 = [SP++]; |
| LT1 = [SP++]; |
| LC0 = [SP++]; |
| LB0 = [SP++]; |
| LT0 = [SP++]; |
| I3 = [SP++]; |
| I2 = [SP++]; |
| I1 = [SP++]; |
| I0 = [SP++]; |
| SEQSTAT = [SP++]; |
| ASTAT = [SP++]; |
| ( R7:0, P5:0 ) = [SP++]; |
| RTS; |
| |
| unknown: |
| [--SP] = RETS; |
| CALL __unknown_exception_occurred; |
| RETS = [SP++]; |
| JUMP unknown; |
| not_replaced: |
| CC = R0 == CPLB_NO_UNLOCKED; |
| IF !CC JUMP next_check; |
| [--SP] = RETS; |
| CALL __cplb_miss_all_locked; |
| RETS = [SP++]; |
| next_check: |
| CC = R0 == CPLB_NO_ADDR_MATCH; |
| IF !CC JUMP next_check2; |
| [--SP] = RETS; |
| CALL __cplb_miss_without_replacement; |
| RETS = [SP++]; |
| JUMP not_replaced; |
| next_check2: |
| CC = R0 == CPLB_PROT_VIOL; |
| IF !CC JUMP strange_return_from_cplb_mgr; |
| [--SP] = RETS; |
| CALL __cplb_protection_violation; |
| RETS = [SP++]; |
| JUMP not_replaced; |
| strange_return_from_cplb_mgr: |
| IDLE; |
| CSYNC; |
| JUMP strange_return_from_cplb_mgr; |
| |
| /************************************ |
| * Diagnostic exception handlers |
| */ |
| |
| __cplb_miss_all_locked: |
| sp += -12; |
| R0 = CPLB_NO_UNLOCKED; |
| call panic_bfin; |
| SP += 12; |
| RTS; |
| |
| __cplb_miss_without_replacement: |
| sp += -12; |
| R0 = CPLB_NO_ADDR_MATCH; |
| call panic_bfin; |
| SP += 12; |
| RTS; |
| |
| __cplb_protection_violation: |
| sp += -12; |
| R0 = CPLB_PROT_VIOL; |
| call panic_bfin; |
| SP += 12; |
| RTS; |
| |
| __unknown_exception_occurred: |
| |
| /* This function is invoked by the default exception |
| * handler, if it does not recognise the kind of |
| * exception that has occurred. In other words, the |
| * default handler only handles some of the system's |
| * exception types, and it does not expect any others |
| * to occur. If your application is going to be using |
| * other kinds of exceptions, you must replace the |
| * default handler with your own, that handles all the |
| * exceptions you will use. |
| * |
| * Since there's nothing we can do, we just loop here |
| * at what we hope is a suitably informative label. |
| */ |
| |
| IDLE; |
| do_not_know_what_to_do: |
| CSYNC; |
| JUMP __unknown_exception_occurred; |
| |
| RTS; |
| .__unknown_exception_occurred.end: |
| .global __unknown_exception_occurred; |
| .type __unknown_exception_occurred, STT_FUNC; |
| |
| panic_bfin: |
| RTS; |