mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-24 12:35:55 +08:00
220 lines
7.1 KiB
C
220 lines
7.1 KiB
C
/* Main header for the CRIS simulator, based on the m32r header.
|
||
Copyright (C) 2004-2021 Free Software Foundation, Inc.
|
||
Contributed by Axis Communications.
|
||
|
||
This file is part of the GNU simulators.
|
||
|
||
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 3 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, see <http://www.gnu.org/licenses/>. */
|
||
|
||
/* All FIXME:s present in m32r apply here too; I just refuse to blindly
|
||
carry them over, as I don't know if they're really things that need
|
||
fixing. */
|
||
|
||
#ifndef SIM_MAIN_H
|
||
#define SIM_MAIN_H
|
||
|
||
/* This is a global setting. Different cpu families can't mix-n-match -scache
|
||
and -pbb. However some cpu families may use -simple while others use
|
||
one of -scache/-pbb. */
|
||
#define WITH_SCACHE_PBB 1
|
||
|
||
#include "symcat.h"
|
||
#include "sim-basics.h"
|
||
#include "cgen-types.h"
|
||
#include "cris-desc.h"
|
||
#include "cris-opc.h"
|
||
#include "arch.h"
|
||
#include "sim-base.h"
|
||
#include "cgen-sim.h"
|
||
#include "cris-sim.h"
|
||
|
||
struct cris_sim_mmapped_page {
|
||
USI addr;
|
||
struct cris_sim_mmapped_page *prev;
|
||
};
|
||
|
||
struct cris_thread_info {
|
||
/* Identifier for this thread. */
|
||
unsigned int threadid;
|
||
|
||
/* Identifier for parent thread. */
|
||
unsigned int parent_threadid;
|
||
|
||
/* Signal to send to parent at exit. */
|
||
int exitsig;
|
||
|
||
/* Exit status. */
|
||
int exitval;
|
||
|
||
/* Only as storage to return the "set" value to the "get" method.
|
||
I'm not sure whether this is useful per-thread. */
|
||
USI priority;
|
||
|
||
struct
|
||
{
|
||
USI altstack;
|
||
USI options;
|
||
|
||
char action;
|
||
char pending;
|
||
char blocked;
|
||
char blocked_suspendsave;
|
||
/* The handler stub unblocks the signal, so we don't need a separate
|
||
"temporary save" for that. */
|
||
} sigdata[64];
|
||
|
||
/* Register context, swapped with _sim_cpu.cpu_data. */
|
||
void *cpu_context;
|
||
|
||
/* Similar, temporary copy for the state at a signal call. */
|
||
void *cpu_context_atsignal;
|
||
|
||
/* The number of the reading and writing ends of a pipe if waiting for
|
||
the reader, else 0. */
|
||
int pipe_read_fd;
|
||
int pipe_write_fd;
|
||
|
||
/* System time at last context switch when this thread ran. */
|
||
USI last_execution;
|
||
|
||
/* Nonzero if we just executed a syscall. */
|
||
char at_syscall;
|
||
|
||
/* Nonzero if any of sigaction[0..64].pending is true. */
|
||
char sigpending;
|
||
|
||
/* Nonzero if in (rt_)sigsuspend call. Cleared at every sighandler
|
||
call. */
|
||
char sigsuspended;
|
||
};
|
||
|
||
typedef int (*cris_interrupt_delivery_fn) (SIM_CPU *,
|
||
enum cris_interrupt_type,
|
||
unsigned int);
|
||
|
||
struct _sim_cpu {
|
||
/* sim/common cpu base. */
|
||
sim_cpu_base base;
|
||
|
||
/* Static parts of cgen. */
|
||
CGEN_CPU cgen_cpu;
|
||
|
||
CRIS_MISC_PROFILE cris_misc_profile;
|
||
#define CPU_CRIS_MISC_PROFILE(cpu) (& (cpu)->cris_misc_profile)
|
||
|
||
/* Copy of previous data; only valid when emitting trace-data after
|
||
each insn. */
|
||
CRIS_MISC_PROFILE cris_prev_misc_profile;
|
||
#define CPU_CRIS_PREV_MISC_PROFILE(cpu) (& (cpu)->cris_prev_misc_profile)
|
||
|
||
#if WITH_HW
|
||
cris_interrupt_delivery_fn deliver_interrupt;
|
||
#define CPU_CRIS_DELIVER_INTERRUPT(cpu) (cpu->deliver_interrupt)
|
||
#endif
|
||
|
||
/* Simulator environment data. */
|
||
USI endmem;
|
||
USI endbrk;
|
||
USI stack_low;
|
||
struct cris_sim_mmapped_page *highest_mmapped_page;
|
||
|
||
/* Number of syscalls performed or in progress, counting once extra
|
||
for every time a blocked thread (internally, when threading) polls
|
||
the (pipe) blockage. By default, this is also a time counter: to
|
||
minimize performance noise from minor compiler changes,
|
||
instructions take no time and syscalls always take 1ms. */
|
||
USI syscalls;
|
||
|
||
/* Number of execution contexts minus one. */
|
||
int m1threads;
|
||
|
||
/* Current thread number; index into thread_data when m1threads != 0. */
|
||
int threadno;
|
||
|
||
/* When a new thread is created, it gets a unique number, which we
|
||
count here. */
|
||
int max_threadid;
|
||
|
||
/* Thread-specific info, for simulator thread support, created at
|
||
"clone" call. Vector of [threads+1] when m1threads > 0. */
|
||
struct cris_thread_info *thread_data;
|
||
|
||
/* "If CLONE_SIGHAND is set, the calling process and the child pro-
|
||
cesses share the same table of signal handlers." ... "However, the
|
||
calling process and child processes still have distinct signal
|
||
masks and sets of pending signals." See struct cris_thread_info
|
||
for sigmasks and sigpendings. */
|
||
USI sighandler[64];
|
||
|
||
/* This is a hack to implement just the parts of fcntl F_GETFL that
|
||
are used in open+fdopen calls for the standard scenario: for such
|
||
a call we check that the last syscall was open, we check that the
|
||
passed fd is the same returned then, and so we return the same
|
||
flags passed to open. This way, we avoid complicating the
|
||
generic sim callback machinery by introducing fcntl
|
||
mechanisms. */
|
||
USI last_syscall;
|
||
USI last_open_fd;
|
||
USI last_open_flags;
|
||
|
||
/* Function for initializing CPU thread context, which varies in size
|
||
with each CPU model. They should be in some constant parts or
|
||
initialized in *_init_cpu, but we can't modify that for now. */
|
||
void* (*make_thread_cpu_data) (SIM_CPU *, void *);
|
||
size_t thread_cpu_data_size;
|
||
|
||
/* The register differs, so we dispatch to a CPU-specific function. */
|
||
void (*set_target_thread_data) (SIM_CPU *, USI);
|
||
|
||
/* CPU-model specific parts go here.
|
||
Note that in files that don't need to access these pieces WANT_CPU_FOO
|
||
won't be defined and thus these parts won't appear. This is ok in the
|
||
sense that things work. It is a source of bugs though.
|
||
One has to of course be careful to not take the size of this
|
||
struct and no structure members accessed in non-cpu specific files can
|
||
go after here. */
|
||
#if defined (WANT_CPU_CRISV0F)
|
||
CRISV0F_CPU_DATA cpu_data;
|
||
#elif defined (WANT_CPU_CRISV3F)
|
||
CRISV3F_CPU_DATA cpu_data;
|
||
#elif defined (WANT_CPU_CRISV8F)
|
||
CRISV8F_CPU_DATA cpu_data;
|
||
#elif defined (WANT_CPU_CRISV10F)
|
||
CRISV10F_CPU_DATA cpu_data;
|
||
#elif defined (WANT_CPU_CRISV32F)
|
||
CRISV32F_CPU_DATA cpu_data;
|
||
#else
|
||
/* Let's assume all cpu_data have the same alignment requirements, so
|
||
they all are laid out at the same address. Since we can't get the
|
||
exact definition, we also assume that it has no higher alignment
|
||
requirements than a vector of, say, 16 pointers. (A single member
|
||
is often special-cased, and possibly two as well so we don't want
|
||
that). */
|
||
union { void *dummy[16]; } cpu_data_placeholder;
|
||
#endif
|
||
};
|
||
|
||
/* Misc. */
|
||
|
||
/* Catch address exceptions. */
|
||
extern SIM_CORE_SIGNAL_FN cris_core_signal;
|
||
#define SIM_CORE_SIGNAL(SD,CPU,CIA,MAP,NR_BYTES,ADDR,TRANSFER,ERROR) \
|
||
cris_core_signal ((SD), (CPU), (CIA), (MAP), (NR_BYTES), (ADDR), \
|
||
(TRANSFER), (ERROR))
|
||
|
||
/* Default memory size. */
|
||
#define CRIS_DEFAULT_MEM_SIZE 0x800000 /* 8M */
|
||
|
||
#endif /* SIM_MAIN_H */
|