OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [ecos-2.0/] [packages/] [hal/] [common/] [v2_0/] [src/] [bplist-dynamic.c] - Rev 1765

Compare with Previous | Blame | View Log

//==========================================================================
//
//      bplist-dynamic.c
//
//      Dynamic breakpoint list.
//      Currently only statically allocated.  (ie NO_MALLOC is assumed)
//
//==========================================================================
//####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):    
// Contributors: dmoseley
// Date:         2000-07-11
// Purpose:      Dynamic breakpoint list.
// Description:  
//               
//
//####DESCRIPTIONEND####
//
//=========================================================================
 
#include <pkgconf/system.h>
#include <pkgconf/hal.h>
 
#if defined(CYGNUM_HAL_BREAKPOINT_LIST_SIZE) && (CYGNUM_HAL_BREAKPOINT_LIST_SIZE > 0) && defined(CYGDBG_HAL_DEBUG_GDB_INCLUDE_STUBS)
 
#define CYGARC_HAL_COMMON_EXPORT_CPU_MACROS
#include <cyg/hal/hal_stub.h>
#include <cyg/hal/hal_arch.h>
#include <cyg/hal/hal_cache.h>
 
#ifdef TARGET_HAS_HARVARD_MEMORY
#define __read_mem_safe __read_progmem_safe
#define __write_mem_safe __write_progmem_safe
#endif
 
/*
 * A simple target breakpoint list without using malloc.
 * To use this package, you must define HAL_BREAKINST_SIZE to be the size
 * in bytes of a trap instruction (max if there's more than one),
 * HAL_BREAKINST to the opcode value of the instruction, and
 * HAL_BREAKINST_TYPE to be the type necessary to hold the opcode value.
 */
 
struct breakpoint_list {
  target_register_t  addr;
  char old_contents [HAL_BREAKINST_SIZE];
  struct breakpoint_list *next;
  char in_memory;
  char length;
} *breakpoint_list = NULL;
 
#ifndef HAL_BREAKINST_ADDR
static HAL_BREAKINST_TYPE break_inst = HAL_BREAKINST;
#define HAL_BREAKINST_ADDR(x) ((void*)&break_inst)
#endif
 
static struct breakpoint_list bp_list [CYGNUM_HAL_BREAKPOINT_LIST_SIZE];
static struct breakpoint_list *free_bp_list = NULL;
static int curr_bp_num = 0;
 
int
__set_breakpoint (target_register_t addr, target_register_t len)
{
  struct breakpoint_list **addent = &breakpoint_list;
  struct breakpoint_list *l = breakpoint_list;
  struct breakpoint_list *newent;
 
  while (l != NULL && l->addr < addr)
    {
      addent = &l->next;
      l =  l->next;
    }
 
  if (l != NULL && l->addr == addr)
    return 2;
 
  if (free_bp_list != NULL)
    {
      newent = free_bp_list;
      free_bp_list = free_bp_list->next;
    }
  else
    {
      if (curr_bp_num < CYGNUM_HAL_BREAKPOINT_LIST_SIZE)
	{
	  newent = &bp_list[curr_bp_num++];
	}
      else
	{
	  return 1;
	}
    }
 
  newent->addr = addr;
  newent->in_memory = 0;
  newent->next = l;
  newent->length = len;
  *addent = newent;
 
  return 0;
}
 
int
__remove_breakpoint (target_register_t addr, target_register_t len)
{
  struct breakpoint_list *l = breakpoint_list;
  struct breakpoint_list *prev = NULL;
 
  while (l != NULL && l->addr < addr)
    {
      prev = l;
      l = l->next;
    }
 
  if ((l == NULL) || (l->addr != addr))
    return 1;
 
  if (l->in_memory)
    {
      __write_mem_safe (&l->old_contents[0],
			(void*)l->addr,
			sizeof (l->old_contents));
    }
 
  if (prev == NULL)
    breakpoint_list = l->next;
  else
    prev->next = l->next;
 
  l->next = free_bp_list;
  free_bp_list = l;
 
  return 0;
}
 
#if defined(HAL_STUB_HW_BREAKPOINT_LIST_SIZE) && (HAL_STUB_HW_BREAKPOINT_LIST_SIZE > 0)
#ifndef HAL_STUB_HW_BREAKPOINT
#error "Must define HAL_STUB_HW_BREAKPOINT"
#endif
struct hw_breakpoint_list {
  target_register_t  addr;
  target_register_t  len;
  char used;
  char installed;
};
static struct hw_breakpoint_list hw_bp_list [HAL_STUB_HW_BREAKPOINT_LIST_SIZE];
 
int
__set_hw_breakpoint (target_register_t addr, target_register_t len)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_BREAKPOINT_LIST_SIZE; i++)
    {
      if (hw_bp_list[i].used == 0)
	{
	  hw_bp_list[i].addr = addr;
	  hw_bp_list[i].len = len;
	  hw_bp_list[i].used = 1;
	  hw_bp_list[i].installed = 0;
	  return 0;
	}
    }
  return -1;
}
 
int
__remove_hw_breakpoint (target_register_t addr, target_register_t len)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_BREAKPOINT_LIST_SIZE; i++)
    {
      if (hw_bp_list[i].used && hw_bp_list[i].addr == addr
	  && hw_bp_list[i].len == len)
	{
	  if (hw_bp_list[i].installed)
	    HAL_STUB_HW_BREAKPOINT(0, (void *)addr, (int)len);
	  hw_bp_list[i].used = 0;
	  return 0;
	}
    }
  return -1;
}
 
static void
__install_hw_breakpoint_list (void)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_BREAKPOINT_LIST_SIZE; i++)
    {
      if (hw_bp_list[i].used && hw_bp_list[i].installed == 0)
	{
	  HAL_STUB_HW_BREAKPOINT(1, (void *)hw_bp_list[i].addr,
				 (int)hw_bp_list[i].len);
	  hw_bp_list[i].installed = 1;
	}
    }
}
 
static void
__clear_hw_breakpoint_list (void)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_BREAKPOINT_LIST_SIZE; i++)
    {
      if (hw_bp_list[i].used && hw_bp_list[i].installed)
	{
	  HAL_STUB_HW_BREAKPOINT(0, (void *)hw_bp_list[i].addr,
				 (int)hw_bp_list[i].len);
	  hw_bp_list[i].installed = 0;
	}
    }
}
#endif // HAL_STUB_HW_BREAKPOINT_LIST_SIZE
 
#if defined(HAL_STUB_HW_WATCHPOINT_LIST_SIZE) && (HAL_STUB_HW_WATCHPOINT_LIST_SIZE > 0)
#ifndef HAL_STUB_HW_WATCHPOINT
#error "Must define HAL_STUB_HW_WATCHPOINT"
#endif
struct hw_watchpoint_list {
  target_register_t  addr;
  target_register_t  len;
  int ztype;
  char used;
  char installed;
};
static struct hw_watchpoint_list hw_wp_list [HAL_STUB_HW_WATCHPOINT_LIST_SIZE];
 
int
__set_hw_watchpoint (target_register_t addr, target_register_t len, int ztype)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_WATCHPOINT_LIST_SIZE; i++)
    {
      if (hw_wp_list[i].used == 0)
	{
	  hw_wp_list[i].addr = addr;
	  hw_wp_list[i].len = len;
	  hw_wp_list[i].ztype = ztype;
	  hw_wp_list[i].used = 1;
	  hw_wp_list[i].installed = 0;
	  return 0;
	}
    }
  return -1;
}
 
int
__remove_hw_watchpoint (target_register_t addr, target_register_t len, int ztype)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_WATCHPOINT_LIST_SIZE; i++)
    {
      if (hw_wp_list[i].used && hw_wp_list[i].addr == addr
	  && hw_wp_list[i].len == len && hw_wp_list[i].ztype == ztype )
	{
	  if (hw_wp_list[i].installed)
	    HAL_STUB_HW_WATCHPOINT(0, (void *)addr, (int)len, ztype);
	  hw_wp_list[i].used = 0;
	  return 0;
	}
    }
  return -1;
}
 
static void
__install_hw_watchpoint_list (void)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_WATCHPOINT_LIST_SIZE; i++)
    {
      if (hw_wp_list[i].used && hw_wp_list[i].installed == 0)
	{
	  HAL_STUB_HW_WATCHPOINT(1, (void *)hw_wp_list[i].addr,
				 (int)hw_wp_list[i].len, hw_wp_list[i].ztype);
	  hw_wp_list[i].installed = 1;
	}
    }
}
 
static void
__clear_hw_watchpoint_list (void)
{
  int i;
 
  for (i = 0; i < HAL_STUB_HW_WATCHPOINT_LIST_SIZE; i++)
    {
      if (hw_wp_list[i].used && hw_wp_list[i].installed)
	{
	  HAL_STUB_HW_WATCHPOINT(0, (void *)hw_wp_list[i].addr,
				 (int)hw_wp_list[i].len, hw_wp_list[i].ztype);
	  hw_wp_list[i].installed = 0;
	}
    }
}
#endif // HAL_STUB_HW_WATCHPOINT_LIST_SIZE
 
 
 
void
__install_breakpoint_list (void)
{
  struct breakpoint_list *l = breakpoint_list;
 
  while (l != NULL)
    {
      if (! l->in_memory)
	{
	  int len = sizeof (l->old_contents);
	  if (__read_mem_safe (&l->old_contents[0], (void*)l->addr, len) == len)
	    {
	      if (__write_mem_safe (HAL_BREAKINST_ADDR(l->length),
				    (void*)l->addr, l->length) == l->length)
		{
		  l->in_memory = 1;
		}
	    }
	}
      l = l->next;
    }
#if defined(HAL_STUB_HW_BREAKPOINT_LIST_SIZE) && (HAL_STUB_HW_BREAKPOINT_LIST_SIZE > 0)
  __install_hw_breakpoint_list();
#endif
#if defined(HAL_STUB_HW_WATCHPOINT_LIST_SIZE) && (HAL_STUB_HW_WATCHPOINT_LIST_SIZE > 0)
  __install_hw_watchpoint_list();
#endif
  HAL_ICACHE_SYNC();
}
 
void
__clear_breakpoint_list (void)
{
  struct breakpoint_list *l = breakpoint_list;
 
  while (l != NULL)
    {
      if (l->in_memory)
	{
	  int len = sizeof (l->old_contents);
	  if (__write_mem_safe (&l->old_contents[0], (void*)l->addr, len) == len)
	    {
	      l->in_memory = 0;
	    }
	}
      l = l->next;
    }
#if defined(HAL_STUB_HW_BREAKPOINT_LIST_SIZE) && (HAL_STUB_HW_BREAKPOINT_LIST_SIZE > 0)
  __clear_hw_breakpoint_list();
#endif
#if defined(HAL_STUB_HW_WATCHPOINT_LIST_SIZE) && (HAL_STUB_HW_WATCHPOINT_LIST_SIZE > 0)
  __clear_hw_watchpoint_list();
#endif
  HAL_ICACHE_INVALIDATE_ALL();
}
 
int
__display_breakpoint_list (void (*print_func)(target_register_t))
{
  struct breakpoint_list *l = breakpoint_list;
 
  while (l != NULL)
    {
      print_func(l->addr);
      l = l->next;
    }
 
  return 0;
}
#else  // (CYGNUM_HAL_BREAKPOINT_LIST_SIZE == 0) or UNDEFINED
 
#include <cyg/hal/hal_stub.h>           // Our header
 
#ifndef CYGDBG_HAL_DEBUG_GDB_INCLUDE_STUBS
// We don't know that type target_register_t is yet.
// Let's just pick a type so we can compile.  Since
// these versions of the functions don't actually do
// anything with the parameters, the actualy types
// don't matter.
typedef unsigned long target_register_t;
#endif
 
int
__set_breakpoint (target_register_t addr, target_register_t len)
{
  return 1;
}
 
int
__remove_breakpoint (target_register_t addr, target_register_t len)
{
  return 1;
}
 
void
__install_breakpoint_list (void)
{
}
 
void
__clear_breakpoint_list (void)
{
}
 
int
__display_breakpoint_list (void (*print_func)(target_register_t))
{
    return 0;
}
#endif // (CYGNUM_HAL_BREAKPOINT_LIST_SIZE > 0)
 
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.