mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-24 12:35:55 +08:00
380 lines
10 KiB
C
380 lines
10 KiB
C
/* The IGEN simulator generator for GDB, the GNU Debugger.
|
||
|
||
Copyright 2002, 2007, 2008, 2009, 2010, 2011 Free Software Foundation, Inc.
|
||
|
||
Contributed by Andrew Cagney.
|
||
|
||
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 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/>. */
|
||
|
||
|
||
|
||
#include "misc.h"
|
||
#include "lf.h"
|
||
#include "table.h"
|
||
#include "filter.h"
|
||
#include "igen.h"
|
||
|
||
#include "ld-insn.h"
|
||
#include "ld-decode.h"
|
||
|
||
#include "gen.h"
|
||
|
||
#include "gen-semantics.h"
|
||
#include "gen-icache.h"
|
||
#include "gen-idecode.h"
|
||
|
||
|
||
static void
|
||
print_semantic_function_header (lf *file,
|
||
const char *basename,
|
||
const char *format_name,
|
||
opcode_bits *expanded_bits,
|
||
int is_function_definition,
|
||
int nr_prefetched_words)
|
||
{
|
||
int indent;
|
||
lf_printf (file, "\n");
|
||
lf_print__function_type_function (file, print_semantic_function_type,
|
||
"EXTERN_SEMANTICS",
|
||
(is_function_definition ? "\n" : " "));
|
||
indent = print_function_name (file,
|
||
basename,
|
||
format_name,
|
||
NULL,
|
||
expanded_bits,
|
||
function_name_prefix_semantics);
|
||
if (is_function_definition)
|
||
{
|
||
indent += lf_printf (file, " ");
|
||
lf_indent (file, +indent);
|
||
}
|
||
else
|
||
{
|
||
lf_printf (file, "\n");
|
||
}
|
||
lf_printf (file, "(");
|
||
lf_indent (file, +1);
|
||
print_semantic_function_formal (file, nr_prefetched_words);
|
||
lf_indent (file, -1);
|
||
lf_printf (file, ")");
|
||
if (is_function_definition)
|
||
{
|
||
lf_indent (file, -indent);
|
||
}
|
||
else
|
||
{
|
||
lf_printf (file, ";");
|
||
}
|
||
lf_printf (file, "\n");
|
||
}
|
||
|
||
void
|
||
print_semantic_declaration (lf *file,
|
||
insn_entry * insn,
|
||
opcode_bits *expanded_bits,
|
||
insn_opcodes *opcodes, int nr_prefetched_words)
|
||
{
|
||
print_semantic_function_header (file,
|
||
insn->name,
|
||
insn->format_name,
|
||
expanded_bits,
|
||
0 /* is not function definition */ ,
|
||
nr_prefetched_words);
|
||
}
|
||
|
||
|
||
|
||
/* generate the semantics.c file */
|
||
|
||
|
||
void
|
||
print_idecode_invalid (lf *file, const char *result, invalid_type type)
|
||
{
|
||
const char *name;
|
||
switch (type)
|
||
{
|
||
default:
|
||
name = "unknown";
|
||
break;
|
||
case invalid_illegal:
|
||
name = "illegal";
|
||
break;
|
||
case invalid_fp_unavailable:
|
||
name = "fp_unavailable";
|
||
break;
|
||
case invalid_wrong_slot:
|
||
name = "wrong_slot";
|
||
break;
|
||
}
|
||
if (options.gen.code == generate_jumps)
|
||
{
|
||
lf_printf (file, "goto %s_%s;\n",
|
||
(options.gen.icache ? "icache" : "semantic"), name);
|
||
}
|
||
else if (options.gen.icache)
|
||
{
|
||
lf_printf (file, "%s %sicache_%s (", result,
|
||
options.module.global.prefix.l, name);
|
||
print_icache_function_actual (file, 0);
|
||
lf_printf (file, ");\n");
|
||
}
|
||
else
|
||
{
|
||
lf_printf (file, "%s %ssemantic_%s (", result,
|
||
options.module.global.prefix.l, name);
|
||
print_semantic_function_actual (file, 0);
|
||
lf_printf (file, ");\n");
|
||
}
|
||
}
|
||
|
||
|
||
void
|
||
print_semantic_body (lf *file,
|
||
insn_entry * instruction,
|
||
opcode_bits *expanded_bits, insn_opcodes *opcodes)
|
||
{
|
||
/* validate the instruction, if a cache this has already been done */
|
||
if (!options.gen.icache)
|
||
{
|
||
print_idecode_validate (file, instruction, opcodes);
|
||
}
|
||
|
||
print_itrace (file, instruction, 0 /*put_value_in_cache */ );
|
||
|
||
/* generate the instruction profile call - this is delayed until
|
||
after the instruction has been verified. The count macro
|
||
generated is prefixed by ITABLE_PREFIX */
|
||
{
|
||
lf_printf (file, "\n");
|
||
lf_indent_suppress (file);
|
||
lf_printf (file, "#if defined (%sPROFILE_COUNT_INSN)\n",
|
||
options.module.itable.prefix.u);
|
||
lf_printf (file, "%sPROFILE_COUNT_INSN (CPU, CIA, MY_INDEX);\n",
|
||
options.module.itable.prefix.u);
|
||
lf_indent_suppress (file);
|
||
lf_printf (file, "#endif\n");
|
||
}
|
||
|
||
/* generate the model call - this is delayed until after the
|
||
instruction has been verified */
|
||
{
|
||
lf_printf (file, "\n");
|
||
lf_indent_suppress (file);
|
||
lf_printf (file, "#if defined (WITH_MON)\n");
|
||
lf_printf (file, "/* monitoring: */\n");
|
||
lf_printf (file, "if (WITH_MON & MONITOR_INSTRUCTION_ISSUE)\n");
|
||
lf_printf (file, " mon_issue (");
|
||
print_function_name (file,
|
||
instruction->name,
|
||
instruction->format_name,
|
||
NULL, NULL, function_name_prefix_itable);
|
||
lf_printf (file, ", cpu, cia);\n");
|
||
lf_indent_suppress (file);
|
||
lf_printf (file, "#endif\n");
|
||
lf_printf (file, "\n");
|
||
}
|
||
|
||
/* determine the new instruction address */
|
||
{
|
||
lf_printf (file, "/* keep the next instruction address handy */\n");
|
||
if (options.gen.nia == nia_is_invalid)
|
||
{
|
||
lf_printf (file, "nia = %sINVALID_INSTRUCTION_ADDRESS;\n",
|
||
options.module.global.prefix.u);
|
||
}
|
||
else
|
||
{
|
||
int nr_immeds = instruction->nr_words - 1;
|
||
if (options.gen.delayed_branch)
|
||
{
|
||
if (nr_immeds > 0)
|
||
{
|
||
lf_printf (file, "cia.dp += %d * %d; %s\n",
|
||
options.insn_bit_size / 8, nr_immeds,
|
||
"/* skip dp immeds */");
|
||
}
|
||
lf_printf (file, "nia.ip = cia.dp; %s\n",
|
||
"/* instruction pointer */");
|
||
lf_printf (file, "nia.dp = cia.dp + %d; %s\n",
|
||
options.insn_bit_size / 8,
|
||
"/* delayed-slot pointer */");
|
||
}
|
||
else
|
||
{
|
||
if (nr_immeds > 0)
|
||
{
|
||
lf_printf (file, "nia = cia + %d * (%d + 1); %s\n",
|
||
options.insn_bit_size / 8, nr_immeds,
|
||
"/* skip immeds as well */");
|
||
|
||
}
|
||
else
|
||
{
|
||
lf_printf (file, "nia = cia + %d;\n",
|
||
options.insn_bit_size / 8);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/* if conditional, generate code to verify that the instruction
|
||
should be issued */
|
||
if (filter_is_member (instruction->options, "c")
|
||
|| options.gen.conditional_issue)
|
||
{
|
||
lf_printf (file, "\n");
|
||
lf_printf (file, "/* execute only if conditional passes */\n");
|
||
lf_printf (file, "if (IS_CONDITION_OK)\n");
|
||
lf_printf (file, " {\n");
|
||
lf_indent (file, +4);
|
||
/* FIXME - need to log a conditional failure */
|
||
}
|
||
|
||
/* Architecture expects a REG to be zero. Instead of having to
|
||
check every read to see if it is refering to that REG just zap it
|
||
at the start of every instruction */
|
||
if (options.gen.zero_reg)
|
||
{
|
||
lf_printf (file, "\n");
|
||
lf_printf (file, "/* Architecture expects REG to be zero */\n");
|
||
lf_printf (file, "GPR_CLEAR(%d);\n", options.gen.zero_reg_nr);
|
||
}
|
||
|
||
/* generate the code (or at least something */
|
||
lf_printf (file, "\n");
|
||
lf_printf (file, "/* semantics: */\n");
|
||
if (instruction->code != NULL)
|
||
{
|
||
/* true code */
|
||
lf_printf (file, "{\n");
|
||
lf_indent (file, +2);
|
||
lf_print__line_ref (file, instruction->code->line);
|
||
table_print_code (file, instruction->code);
|
||
lf_indent (file, -2);
|
||
lf_printf (file, "}\n");
|
||
lf_print__internal_ref (file);
|
||
}
|
||
else if (filter_is_member (instruction->options, "nop"))
|
||
{
|
||
lf_print__internal_ref (file);
|
||
}
|
||
else
|
||
{
|
||
const char *prefix = "sim_engine_abort (";
|
||
int indent = strlen (prefix);
|
||
/* abort so it is implemented now */
|
||
lf_print__line_ref (file, instruction->line);
|
||
lf_printf (file, "%sSD, CPU, cia, \\\n", prefix);
|
||
lf_indent (file, +indent);
|
||
lf_printf (file, "\"%s:%d:0x%%08lx:%%s unimplemented\\n\", \\\n",
|
||
filter_filename (instruction->line->file_name),
|
||
instruction->line->line_nr);
|
||
lf_printf (file, "(long) CIA, \\\n");
|
||
lf_printf (file, "%sitable[MY_INDEX].name);\n",
|
||
options.module.itable.prefix.l);
|
||
lf_indent (file, -indent);
|
||
lf_print__internal_ref (file);
|
||
}
|
||
|
||
/* Close off the conditional execution */
|
||
if (filter_is_member (instruction->options, "c")
|
||
|| options.gen.conditional_issue)
|
||
{
|
||
lf_indent (file, -4);
|
||
lf_printf (file, " }\n");
|
||
}
|
||
}
|
||
|
||
static void
|
||
print_c_semantic (lf *file,
|
||
insn_entry * instruction,
|
||
opcode_bits *expanded_bits,
|
||
insn_opcodes *opcodes,
|
||
cache_entry *cache_rules, int nr_prefetched_words)
|
||
{
|
||
|
||
lf_printf (file, "{\n");
|
||
lf_indent (file, +2);
|
||
|
||
print_my_defines (file,
|
||
instruction->name,
|
||
instruction->format_name, expanded_bits);
|
||
lf_printf (file, "\n");
|
||
print_icache_body (file,
|
||
instruction,
|
||
expanded_bits,
|
||
cache_rules,
|
||
(options.gen.direct_access
|
||
? define_variables
|
||
: declare_variables),
|
||
(options.gen.icache
|
||
? get_values_from_icache
|
||
: do_not_use_icache), nr_prefetched_words);
|
||
|
||
lf_printf (file, "%sinstruction_address nia;\n",
|
||
options.module.global.prefix.l);
|
||
print_semantic_body (file, instruction, expanded_bits, opcodes);
|
||
lf_printf (file, "return nia;\n");
|
||
|
||
/* generate something to clean up any #defines created for the cache */
|
||
if (options.gen.direct_access)
|
||
{
|
||
print_icache_body (file,
|
||
instruction,
|
||
expanded_bits,
|
||
cache_rules,
|
||
undef_variables,
|
||
(options.gen.icache
|
||
? get_values_from_icache
|
||
: do_not_use_icache), nr_prefetched_words);
|
||
}
|
||
|
||
lf_indent (file, -2);
|
||
lf_printf (file, "}\n");
|
||
}
|
||
|
||
static void
|
||
print_c_semantic_function (lf *file,
|
||
insn_entry * instruction,
|
||
opcode_bits *expanded_bits,
|
||
insn_opcodes *opcodes,
|
||
cache_entry *cache_rules, int nr_prefetched_words)
|
||
{
|
||
/* build the semantic routine to execute the instruction */
|
||
print_semantic_function_header (file,
|
||
instruction->name,
|
||
instruction->format_name,
|
||
expanded_bits,
|
||
1 /*is-function-definition */ ,
|
||
nr_prefetched_words);
|
||
print_c_semantic (file,
|
||
instruction,
|
||
expanded_bits, opcodes, cache_rules, nr_prefetched_words);
|
||
}
|
||
|
||
void
|
||
print_semantic_definition (lf *file,
|
||
insn_entry * insn,
|
||
opcode_bits *expanded_bits,
|
||
insn_opcodes *opcodes,
|
||
cache_entry *cache_rules, int nr_prefetched_words)
|
||
{
|
||
print_c_semantic_function (file,
|
||
insn,
|
||
expanded_bits,
|
||
opcodes, cache_rules, nr_prefetched_words);
|
||
}
|