211 lines
5 KiB
C
211 lines
5 KiB
C
|
/* Target-dependent code for the Fujitsu SPARCLITE for GDB, the GNU debugger.
|
||
|
Copyright 1994 Free Software Foundation, Inc.
|
||
|
|
||
|
This file is part of GDB.
|
||
|
|
||
|
This program 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 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
This program 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 this program; if not, write to the Free Software
|
||
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
|
||
|
|
||
|
#include "defs.h"
|
||
|
#include "gdbcore.h"
|
||
|
#include "breakpoint.h"
|
||
|
|
||
|
#define DDA2_SUP_ASI 0xb000000
|
||
|
#define DDA1_SUP_ASI 0xb0000
|
||
|
|
||
|
#define DDA2_ASI_MASK 0xff000000
|
||
|
#define DDA1_ASI_MASK 0xff0000
|
||
|
#define DIA2_SUP_MODE 0x8000
|
||
|
#define DIA1_SUP_MODE 0x4000
|
||
|
#define DDA2_ENABLE 0x100
|
||
|
#define DDA1_ENABLE 0x80
|
||
|
#define DIA2_ENABLE 0x40
|
||
|
#define DIA1_ENABLE 0x20
|
||
|
#define DSINGLE_STEP 0x10
|
||
|
#define DDV_TYPE_MASK 0xc
|
||
|
#define DDV_TYPE_LOAD 0x0
|
||
|
#define DDV_TYPE_STORE 0x4
|
||
|
#define DDV_TYPE_ACCESS 0x8
|
||
|
#define DDV_TYPE_ALWAYS 0xc
|
||
|
#define DDV_COND 0x2
|
||
|
#define DDV_MASK 0x1
|
||
|
|
||
|
int
|
||
|
sparclite_insert_watchpoint(addr, len, type)
|
||
|
CORE_ADDR addr;
|
||
|
int len;
|
||
|
int type;
|
||
|
{
|
||
|
CORE_ADDR dcr;
|
||
|
|
||
|
dcr = read_register (DCR_REGNUM);
|
||
|
|
||
|
if (!(dcr & DDA1_ENABLE))
|
||
|
{
|
||
|
write_register (DDA1_REGNUM, addr);
|
||
|
dcr &= ~(DDA1_ASI_MASK | DDV_TYPE_MASK);
|
||
|
dcr |= (DDA1_SUP_ASI | DDA1_ENABLE);
|
||
|
if (type == 1)
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_LOAD & (~DDV_COND & ~DDV_MASK));
|
||
|
}
|
||
|
else if (type == 0)
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_STORE & (~DDV_COND & ~DDV_MASK));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_ACCESS);
|
||
|
}
|
||
|
write_register (DCR_REGNUM, dcr);
|
||
|
}
|
||
|
else if (!(dcr & DDA2_ENABLE))
|
||
|
{
|
||
|
write_register (DDA2_REGNUM, addr);
|
||
|
dcr &= ~(DDA2_ASI_MASK & DDV_TYPE_MASK);
|
||
|
dcr |= (DDA2_SUP_ASI | DDA2_ENABLE);
|
||
|
if (type == 1)
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_LOAD & ~DDV_COND & ~DDV_MASK);
|
||
|
}
|
||
|
else if (type == 0)
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_STORE & ~DDV_COND & ~DDV_MASK);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
write_register (DDV1_REGNUM, 0);
|
||
|
write_register (DDV2_REGNUM, 0xffffffff);
|
||
|
dcr |= (DDV_TYPE_ACCESS);
|
||
|
}
|
||
|
write_register (DCR_REGNUM, dcr);
|
||
|
}
|
||
|
else
|
||
|
return -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
sparclite_remove_watchpoint(addr, len, type)
|
||
|
CORE_ADDR addr;
|
||
|
int len;
|
||
|
int type;
|
||
|
{
|
||
|
CORE_ADDR dcr, dda1, dda2;
|
||
|
|
||
|
dcr = read_register (DCR_REGNUM);
|
||
|
dda1 = read_register (DDA1_REGNUM);
|
||
|
dda2 = read_register (DDA2_REGNUM);
|
||
|
|
||
|
if ((dcr & DDA1_ENABLE) && addr == dda1) {
|
||
|
write_register (DCR_REGNUM, (dcr & ~DDA1_ENABLE));
|
||
|
}
|
||
|
else if ((dcr & DDA2_ENABLE) && addr == dda2) {
|
||
|
write_register (DCR_REGNUM, (dcr & ~DDA2_ENABLE));
|
||
|
}
|
||
|
else
|
||
|
return -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
sparclite_insert_hw_breakpoint(addr, len)
|
||
|
CORE_ADDR addr;
|
||
|
int len;
|
||
|
{
|
||
|
CORE_ADDR dcr;
|
||
|
|
||
|
dcr = read_register (DCR_REGNUM);
|
||
|
|
||
|
if (!(dcr & DIA1_ENABLE)) {
|
||
|
write_register (DIA1_REGNUM, addr);
|
||
|
write_register (DCR_REGNUM, (dcr | DIA1_ENABLE | DIA1_SUP_MODE));
|
||
|
}
|
||
|
else if (!(dcr & DIA2_ENABLE)) {
|
||
|
write_register (DIA2_REGNUM, addr);
|
||
|
write_register (DCR_REGNUM, (dcr | DIA2_ENABLE | DIA2_SUP_MODE));
|
||
|
}
|
||
|
else
|
||
|
return -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
sparclite_remove_hw_breakpoint(addr, shadow)
|
||
|
CORE_ADDR addr;
|
||
|
int shadow;
|
||
|
{
|
||
|
CORE_ADDR dcr, dia1, dia2;
|
||
|
|
||
|
dcr = read_register (DCR_REGNUM);
|
||
|
dia1 = read_register (DIA1_REGNUM);
|
||
|
dia2 = read_register (DIA2_REGNUM);
|
||
|
|
||
|
if ((dcr & DIA1_ENABLE) && addr == dia1) {
|
||
|
write_register (DCR_REGNUM, (dcr & ~DIA1_ENABLE));
|
||
|
}
|
||
|
else if ((dcr & DIA2_ENABLE) && addr == dia2) {
|
||
|
write_register (DCR_REGNUM, (dcr & ~DIA2_ENABLE));
|
||
|
}
|
||
|
else
|
||
|
return -1;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
sparclite_check_watch_resources(type, cnt, ot)
|
||
|
int type;
|
||
|
int cnt;
|
||
|
int ot;
|
||
|
{
|
||
|
if (type == bp_hardware_breakpoint) {
|
||
|
if (TARGET_HW_BREAK_LIMIT == 0) return 0;
|
||
|
else if (cnt <= TARGET_HW_BREAK_LIMIT) return 1;
|
||
|
}
|
||
|
else {
|
||
|
if (TARGET_HW_WATCH_LIMIT == 0) return 0;
|
||
|
else if (ot) return -1;
|
||
|
else if (cnt <= TARGET_HW_WATCH_LIMIT) return 1;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
CORE_ADDR
|
||
|
sparclite_stopped_data_address()
|
||
|
{
|
||
|
CORE_ADDR dsr, dda1, dda2;
|
||
|
|
||
|
dsr = read_register (DSR_REGNUM);
|
||
|
dda1 = read_register (DDA1_REGNUM);
|
||
|
dda2 = read_register (DDA2_REGNUM);
|
||
|
|
||
|
if (dsr & 0x10) return dda1;
|
||
|
else if (dsr & 0x20) return dda2;
|
||
|
else return 0;
|
||
|
}
|