224 lines
5.4 KiB
C
224 lines
5.4 KiB
C
/*
|
|
* IP26.c - IP26 specific routines for symmon
|
|
*
|
|
*/
|
|
#ident "$Revision: 1.6 $"
|
|
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/cpu.h>
|
|
#include <sys/sbd.h>
|
|
#include <setjmp.h>
|
|
#include <fault.h>
|
|
#include <parser.h>
|
|
#include <saioctl.h>
|
|
#include "dbgmon.h"
|
|
#include "mp.h"
|
|
#include <libsc.h>
|
|
|
|
/*
|
|
* register descriptions
|
|
*/
|
|
extern struct reg_desc _liointr0_desc[], _liointr1_desc[], _liointr2_desc[];
|
|
extern struct reg_desc _cpu_parerr_desc[], _gio_parerr_desc[];
|
|
extern struct reg_desc sr_desc[], cause_desc[];
|
|
extern struct reg_desc _tcc_error_desc[], _tcc_intr_desc[];
|
|
|
|
static struct reg_values except_values[] = {
|
|
{ EXCEPT_UTLB, "UTLB Miss" },
|
|
{ EXCEPT_XUT, "XTLB Miss" },
|
|
{ EXCEPT_NORM, "NORMAL" },
|
|
{ EXCEPT_BRKPT, "BREAKPOINT" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
static struct reg_desc except_desc[] = {
|
|
/* mask shift name format values */
|
|
{ -1, 0, "vector",NULL, except_values },
|
|
{ 0, 0, NULL, NULL, NULL }
|
|
};
|
|
|
|
/*
|
|
* Start of symmon's stack for boot processor.
|
|
*/
|
|
__psint_t _dbgstack = SYMMON_STACK;
|
|
|
|
/*
|
|
* Platform-dependent command table.
|
|
*/
|
|
static int _show_tcc(int, char **, char **, struct cmd_table *);
|
|
struct cmd_table pd_ctbl[] = {
|
|
{ "tcc", _show_tcc, "show TCC state" },
|
|
{ 0, 0, 0 }
|
|
};
|
|
|
|
/*
|
|
* Platform-dependent arbitrary name/value table.
|
|
*/
|
|
pd_name_table_t pd_name_table[] = {
|
|
{ 0, 0 }
|
|
};
|
|
|
|
long tcc_gcache; /* setting of TCC_GCACHE of caller */
|
|
int ip26_ucmem; /* caller running in fast/slow mode */
|
|
|
|
/*
|
|
* symmon_fault -- display info pertinent to fault
|
|
*/
|
|
void
|
|
symmon_fault(void)
|
|
{
|
|
static volatile int in_symmon_fault = 0;
|
|
static volatile int in_symmon_double_fault = 0;
|
|
static jmp_buf in_symmon_jmpbuf;
|
|
unsigned int parerr;
|
|
|
|
if (in_symmon_fault) {
|
|
if (in_symmon_double_fault)
|
|
EnterInteractiveMode();
|
|
in_symmon_double_fault = 1;
|
|
printf("\nSYMMON DOUBLE EXCEPTION: %r\n", private.dbg_exc,
|
|
except_desc);
|
|
in_symmon_double_fault = 0;
|
|
longjmp(in_symmon_jmpbuf,1);
|
|
}
|
|
setjmp(in_symmon_jmpbuf);
|
|
|
|
while (in_symmon_fault < 8) {
|
|
switch (in_symmon_fault++) {
|
|
case 0:
|
|
printf("\nSYMMON EXCEPTION: %r\n", private.dbg_exc,
|
|
except_desc);
|
|
break;
|
|
|
|
case 1:
|
|
/* try to call disassembler */
|
|
_show_inst((inst_t *)private.exc_stack[E_EPC]);
|
|
break;
|
|
|
|
case 2:
|
|
printf("Cause register: %R\n",
|
|
private.exc_stack[E_CAUSE], cause_desc);
|
|
break;
|
|
|
|
case 3:
|
|
printf("Status register: %R\n", private.exc_stack[E_SR],
|
|
sr_desc);
|
|
break;
|
|
|
|
case 4:
|
|
if ((private.dbg_exc == EXCEPT_UTLB) ||
|
|
(private.dbg_exc == EXCEPT_XUT))
|
|
goto printvaddr;
|
|
|
|
switch ((unsigned)(private.exc_stack[E_CAUSE] &
|
|
CAUSE_EXCMASK)) {
|
|
case EXC_MOD: case EXC_RMISS: case EXC_WMISS:
|
|
case EXC_RADE: case EXC_WADE:
|
|
printvaddr:
|
|
printf("Bad Vaddress: 0x%lx\n",
|
|
private.exc_stack[E_BADVADDR]);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 5:
|
|
/* print parity and bus error info
|
|
*/
|
|
parerr = (u_int)
|
|
(private.exc_stack[E_CPU_PARERR] & 0x1fff);
|
|
printf("CPU Parity error register: %R\n", parerr,
|
|
_cpu_parerr_desc);
|
|
break;
|
|
|
|
case 6:
|
|
/*
|
|
* Only print write bus errors if the client was
|
|
* enabled for them, otherwise we might swipe on
|
|
* what was intended for client.
|
|
*/
|
|
if (!((private.exc_stack[E_SR] & SR_IBIT7) &&
|
|
(private.exc_stack[E_CAUSE] & CAUSE_IP7) &&
|
|
parerr))
|
|
break;
|
|
|
|
if (private.exc_stack[E_TCC_INTR] &
|
|
(INTR_BUSERROR|INTR_MACH_CHECK)) {
|
|
printf( "TCC Bus Error register: 0x%x\n"
|
|
"TCC Parity register: 0x%x\n",
|
|
private.exc_stack[E_TCC_BE_ADDR],
|
|
private.exc_stack[E_TCC_PARITY]);
|
|
printf( "TCC Error register: %R\n",
|
|
private.exc_stack[E_TCC_ERROR],
|
|
_tcc_error_desc);
|
|
}
|
|
|
|
if (parerr & CPU_ERR_STAT_RD_PAR == CPU_ERR_STAT_RD_PAR)
|
|
printf("CPU parity error during read at "
|
|
"0x%lx\n",private.exc_stack[E_CPUADDR]);
|
|
else if (parerr & CPU_ERR_STAT_PAR)
|
|
printf("CPU parity error during write at "
|
|
"0x%lx\n",private.exc_stack[E_CPUADDR]);
|
|
|
|
if (parerr & CPU_ERR_STAT_SYSAD_PAR)
|
|
printf("SYSAD bus parity error\n");
|
|
if (parerr & CPU_ERR_STAT_SYSCMD_PAR)
|
|
printf("SYSCMD fifo parity error\n");
|
|
if (parerr & CPU_ERR_STAT_ADDR)
|
|
printf("Write Bus Error Occurred\n");
|
|
break;
|
|
|
|
case 7:
|
|
/* print LIO registers
|
|
*/
|
|
printf("Local I/O interrupt register 0: %R\n",
|
|
private.exc_stack[E_LIOINTR0], _liointr0_desc);
|
|
printf("Local I/O interrupt register 1: %R\n",
|
|
private.exc_stack[E_LIOINTR1], _liointr1_desc );
|
|
printf("Local I/O interrupt register 2: %R\n",
|
|
private.exc_stack[E_LIOINTR2], _liointr2_desc );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
in_symmon_fault = 0;
|
|
}
|
|
|
|
int
|
|
cpu_probe(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
notyetmess(char *mess)
|
|
{
|
|
printf("%s not implemented for teton\n", mess);
|
|
}
|
|
|
|
/*ARGSUSED*/
|
|
static int
|
|
_show_tcc(int argc, char **ap, char **av, struct cmd_table *foo)
|
|
{
|
|
printf("previous mode: %d (%s)\n",
|
|
ip26_ucmem, ip26_ucmem ? "normal" : "slow");
|
|
printf("TCC_GCACHE: hardware: 0x%x saved: 0x%x\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_GCACHE),
|
|
tcc_gcache);
|
|
printf("TCC_INTR: hardware: %R\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_INTR),
|
|
_tcc_intr_desc);
|
|
printf("TCC_FIFO: 0x%x\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_FIFO));
|
|
printf("TCC_PREFETCH: 0x%x\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_PREFETCH));
|
|
printf("TCC_PREFA: 0x%x\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_PREFETCH_A));
|
|
printf("TCC_PREFB: 0x%x\n",
|
|
*(volatile unsigned long *)PHYS_TO_K1(TCC_PREFETCH_B));
|
|
|
|
return 0;
|
|
}
|