mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-24 10:35:12 +08:00
1d506c26d9
This commit is the result of the following actions: - Running gdb/copyright.py to update all of the copyright headers to include 2024, - Manually updating a few files the copyright.py script told me to update, these files had copyright headers embedded within the file, - Regenerating gdbsupport/Makefile.in to refresh it's copyright date, - Using grep to find other files that still mentioned 2023. If these files were updated last year from 2022 to 2023 then I've updated them this year to 2024. I'm sure I've probably missed some dates. Feel free to fix them up as you spot them.
1862 lines
51 KiB
C
1862 lines
51 KiB
C
/* The IGEN simulator generator for GDB, the GNU Debugger.
|
||
|
||
Copyright 2002-2024 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"
|
||
|
||
static insn_word_entry *
|
||
parse_insn_word (const line_ref *line, char *string, int word_nr)
|
||
{
|
||
char *chp;
|
||
insn_word_entry *word = ZALLOC (insn_word_entry);
|
||
|
||
/* create a leading sentinal */
|
||
word->first = ZALLOC (insn_field_entry);
|
||
word->first->first = -1;
|
||
word->first->last = -1;
|
||
word->first->width = 0;
|
||
|
||
/* and a trailing sentinal */
|
||
word->last = ZALLOC (insn_field_entry);
|
||
word->last->first = options.insn_bit_size;
|
||
word->last->last = options.insn_bit_size;
|
||
word->last->width = 0;
|
||
|
||
/* link them together */
|
||
word->first->next = word->last;
|
||
word->last->prev = word->first;
|
||
|
||
/* now work through the formats */
|
||
chp = skip_spaces (string);
|
||
|
||
while (*chp != '\0')
|
||
{
|
||
char *start_pos;
|
||
int strlen_pos;
|
||
char *start_val;
|
||
int strlen_val;
|
||
insn_field_entry *new_field;
|
||
|
||
/* create / link in the new field */
|
||
new_field = ZALLOC (insn_field_entry);
|
||
new_field->next = word->last;
|
||
new_field->prev = word->last->prev;
|
||
new_field->next->prev = new_field;
|
||
new_field->prev->next = new_field;
|
||
new_field->word_nr = word_nr;
|
||
|
||
/* break out the first field (if present) */
|
||
start_pos = chp;
|
||
chp = skip_to_separator (chp, ".,!");
|
||
strlen_pos = back_spaces (start_pos, chp) - start_pos;
|
||
|
||
/* break out the second field (if present) */
|
||
if (*chp != '.')
|
||
{
|
||
/* assume what was specified was the value (and not the start
|
||
position). Assume the value length implicitly specifies
|
||
the number of bits */
|
||
start_val = start_pos;
|
||
strlen_val = strlen_pos;
|
||
start_pos = "";
|
||
strlen_pos = 0;
|
||
}
|
||
else
|
||
{
|
||
chp++; /* skip `.' */
|
||
chp = skip_spaces (chp);
|
||
start_val = chp;
|
||
if (*chp == '/' || *chp == '*')
|
||
{
|
||
do
|
||
{
|
||
chp++;
|
||
}
|
||
while (*chp == '/' || *chp == '*');
|
||
}
|
||
else if (isalpha (*start_val))
|
||
{
|
||
do
|
||
{
|
||
chp++;
|
||
}
|
||
while (isalnum (*chp) || *chp == '_');
|
||
}
|
||
else if (isdigit (*start_val))
|
||
{
|
||
do
|
||
{
|
||
chp++;
|
||
}
|
||
while (isalnum (*chp));
|
||
}
|
||
strlen_val = chp - start_val;
|
||
chp = skip_spaces (chp);
|
||
}
|
||
if (strlen_val == 0)
|
||
error (line, "Empty value field\n");
|
||
|
||
/* break out any conditional fields - { [ "!" | "=" [ <value> | <field-name> } */
|
||
while (*chp == '!' || *chp == '=')
|
||
{
|
||
char *start;
|
||
char *end;
|
||
int len;
|
||
insn_field_cond *new_cond = ZALLOC (insn_field_cond);
|
||
|
||
/* determine the conditional test */
|
||
switch (*chp)
|
||
{
|
||
case '=':
|
||
new_cond->test = insn_field_cond_eq;
|
||
break;
|
||
case '!':
|
||
new_cond->test = insn_field_cond_ne;
|
||
break;
|
||
default:
|
||
ASSERT (0);
|
||
}
|
||
|
||
/* save the value */
|
||
chp++;
|
||
chp = skip_spaces (chp);
|
||
start = chp;
|
||
chp = skip_to_separator (chp, "+,:!=");
|
||
end = back_spaces (start, chp);
|
||
len = end - start;
|
||
if (len == 0)
|
||
error (line, "Missing or invalid conditional value\n");
|
||
new_cond->string = NZALLOC (char, len + 1);
|
||
strncpy (new_cond->string, start, len);
|
||
|
||
/* determine the conditional type */
|
||
if (isdigit (*start))
|
||
{
|
||
/* [ "!" | "=" ] <value> */
|
||
new_cond->type = insn_field_cond_value;
|
||
new_cond->value = a2i (new_cond->string);
|
||
}
|
||
else
|
||
{
|
||
/* [ "!" | "=" ] <field> - check field valid */
|
||
new_cond->type = insn_field_cond_field;
|
||
/* new_cond->field is determined in later */
|
||
}
|
||
|
||
/* Only a single `=' is permitted. */
|
||
if ((new_cond->test == insn_field_cond_eq
|
||
&& new_field->conditions != NULL)
|
||
|| (new_field->conditions != NULL
|
||
&& new_field->conditions->test == insn_field_cond_eq))
|
||
error (line, "Only single conditional when `=' allowed\n");
|
||
|
||
/* insert it */
|
||
{
|
||
insn_field_cond **last = &new_field->conditions;
|
||
while (*last != NULL)
|
||
last = &(*last)->next;
|
||
*last = new_cond;
|
||
}
|
||
}
|
||
|
||
/* NOW verify that the field was finished */
|
||
if (*chp == ',')
|
||
{
|
||
chp = skip_spaces (chp + 1);
|
||
if (*chp == '\0')
|
||
error (line, "empty field\n");
|
||
}
|
||
else if (*chp != '\0')
|
||
{
|
||
error (line, "Missing field separator\n");
|
||
}
|
||
|
||
/* copy the value */
|
||
new_field->val_string = NZALLOC (char, strlen_val + 1);
|
||
strncpy (new_field->val_string, start_val, strlen_val);
|
||
if (isdigit (new_field->val_string[0]))
|
||
{
|
||
if (strlen_pos == 0)
|
||
{
|
||
/* when the length/pos field is omited, an integer field
|
||
is always binary */
|
||
uint64_t val = 0;
|
||
int i;
|
||
for (i = 0; i < strlen_val; i++)
|
||
{
|
||
if (new_field->val_string[i] != '0'
|
||
&& new_field->val_string[i] != '1')
|
||
error (line, "invalid binary field %s\n",
|
||
new_field->val_string);
|
||
val = (val << 1) + (new_field->val_string[i] == '1');
|
||
}
|
||
new_field->val_int = val;
|
||
new_field->type = insn_field_int;
|
||
}
|
||
else
|
||
{
|
||
new_field->val_int = a2i (new_field->val_string);
|
||
new_field->type = insn_field_int;
|
||
}
|
||
}
|
||
else if (new_field->val_string[0] == '/')
|
||
{
|
||
new_field->type = insn_field_reserved;
|
||
}
|
||
else if (new_field->val_string[0] == '*')
|
||
{
|
||
new_field->type = insn_field_wild;
|
||
}
|
||
else
|
||
{
|
||
new_field->type = insn_field_string;
|
||
if (filter_is_member (word->field_names, new_field->val_string))
|
||
error (line, "Field name %s is duplicated\n",
|
||
new_field->val_string);
|
||
filter_parse (&word->field_names, new_field->val_string);
|
||
}
|
||
if (new_field->type != insn_field_string
|
||
&& new_field->conditions != NULL)
|
||
error (line, "Conditionals can only be applied to named fields\n");
|
||
|
||
/* the copy the position */
|
||
new_field->pos_string = NZALLOC (char, strlen_pos + 1);
|
||
strncpy (new_field->pos_string, start_pos, strlen_pos);
|
||
if (strlen_pos == 0)
|
||
{
|
||
new_field->first = new_field->prev->last + 1;
|
||
if (new_field->first == 0 /* first field */
|
||
&& *chp == '\0' /* no further fields */
|
||
&& new_field->type == insn_field_string)
|
||
{
|
||
/* A single string without any position, assume that it
|
||
represents the entire instruction word */
|
||
new_field->width = options.insn_bit_size;
|
||
}
|
||
else
|
||
{
|
||
/* No explicit width/position, assume value implicitly
|
||
supplies the width */
|
||
new_field->width = strlen_val;
|
||
}
|
||
new_field->last = new_field->first + new_field->width - 1;
|
||
if (new_field->last >= options.insn_bit_size)
|
||
error (line, "Bit position %d exceed instruction bit size (%d)\n",
|
||
new_field->last, options.insn_bit_size);
|
||
}
|
||
else if (options.insn_specifying_widths)
|
||
{
|
||
new_field->first = new_field->prev->last + 1;
|
||
new_field->width = a2i (new_field->pos_string);
|
||
new_field->last = new_field->first + new_field->width - 1;
|
||
if (new_field->last >= options.insn_bit_size)
|
||
error (line, "Bit position %d exceed instruction bit size (%d)\n",
|
||
new_field->last, options.insn_bit_size);
|
||
}
|
||
else
|
||
{
|
||
new_field->first = target_a2i (options.hi_bit_nr,
|
||
new_field->pos_string);
|
||
new_field->last = new_field->next->first - 1; /* guess */
|
||
new_field->width = new_field->last - new_field->first + 1; /* guess */
|
||
new_field->prev->last = new_field->first - 1; /*fix */
|
||
new_field->prev->width = new_field->first - new_field->prev->first; /*fix */
|
||
}
|
||
}
|
||
|
||
/* fiddle first/last so that the sentinals disapear */
|
||
ASSERT (word->first->last < 0);
|
||
ASSERT (word->last->first >= options.insn_bit_size);
|
||
word->first = word->first->next;
|
||
word->last = word->last->prev;
|
||
|
||
/* check that the last field goes all the way to the last bit */
|
||
if (word->last->last != options.insn_bit_size - 1)
|
||
{
|
||
if (options.warn.width)
|
||
options.warning (line, "Instruction format is not %d bits wide\n",
|
||
options.insn_bit_size);
|
||
word->last->last = options.insn_bit_size - 1;
|
||
}
|
||
|
||
/* now go over this again, pointing each bit position at a field
|
||
record */
|
||
{
|
||
insn_field_entry *field;
|
||
for (field = word->first;
|
||
field->last < options.insn_bit_size; field = field->next)
|
||
{
|
||
int i;
|
||
for (i = field->first; i <= field->last; i++)
|
||
{
|
||
word->bit[i] = ZALLOC (insn_bit_entry);
|
||
word->bit[i]->field = field;
|
||
switch (field->type)
|
||
{
|
||
case insn_field_invalid:
|
||
ASSERT (0);
|
||
break;
|
||
case insn_field_int:
|
||
word->bit[i]->mask = 1;
|
||
word->bit[i]->value = ((field->val_int
|
||
& ((insn_uint) 1 <<
|
||
(field->last - i))) != 0);
|
||
case insn_field_reserved:
|
||
case insn_field_wild:
|
||
case insn_field_string:
|
||
/* if we encounter a constant conditional, encode
|
||
their bit value. */
|
||
if (field->conditions != NULL
|
||
&& field->conditions->test == insn_field_cond_eq
|
||
&& field->conditions->type == insn_field_cond_value)
|
||
{
|
||
word->bit[i]->mask = 1;
|
||
word->bit[i]->value = ((field->conditions->value
|
||
& ((insn_uint) 1 <<
|
||
(field->last - i))) != 0);
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return word;
|
||
}
|
||
|
||
|
||
static void
|
||
parse_insn_words (insn_entry * insn, char *formats)
|
||
{
|
||
insn_word_entry **last_word = &insn->words;
|
||
char *chp;
|
||
|
||
/* now work through the formats */
|
||
insn->nr_words = 0;
|
||
chp = formats;
|
||
|
||
while (1)
|
||
{
|
||
char *start_pos;
|
||
char *end_pos;
|
||
int strlen_pos;
|
||
char *format;
|
||
insn_word_entry *new_word;
|
||
|
||
/* skip leading spaces */
|
||
chp = skip_spaces (chp);
|
||
|
||
/* break out the format */
|
||
start_pos = chp;
|
||
chp = skip_to_separator (chp, "+");
|
||
end_pos = back_spaces (start_pos, chp);
|
||
strlen_pos = end_pos - start_pos;
|
||
|
||
/* check that something was there */
|
||
if (strlen_pos == 0)
|
||
error (insn->line, "missing or empty instruction format\n");
|
||
|
||
/* parse the field */
|
||
format = NZALLOC (char, strlen_pos + 1);
|
||
strncpy (format, start_pos, strlen_pos);
|
||
new_word = parse_insn_word (insn->line, format, insn->nr_words);
|
||
insn->nr_words++;
|
||
if (filter_is_common (insn->field_names, new_word->field_names))
|
||
error (insn->line, "Field name duplicated between two words\n");
|
||
filter_add (&insn->field_names, new_word->field_names);
|
||
|
||
/* insert it */
|
||
*last_word = new_word;
|
||
last_word = &new_word->next;
|
||
|
||
/* last format? */
|
||
if (*chp == '\0')
|
||
break;
|
||
ASSERT (*chp == '+');
|
||
chp++;
|
||
}
|
||
|
||
/* create a quick access array (indexed by word) of the same structure */
|
||
{
|
||
int i;
|
||
insn_word_entry *word;
|
||
insn->word = NZALLOC (insn_word_entry *, insn->nr_words + 1);
|
||
for (i = 0, word = insn->words;
|
||
i < insn->nr_words; i++, word = word->next)
|
||
insn->word[i] = word;
|
||
}
|
||
|
||
/* Go over all fields that have conditionals refering to other
|
||
fields. Link the fields up. Verify that the two fields have the
|
||
same size. Verify that the two fields are different */
|
||
{
|
||
int i;
|
||
for (i = 0; i < insn->nr_words; i++)
|
||
{
|
||
insn_word_entry *word = insn->word[i];
|
||
insn_field_entry *f;
|
||
for (f = word->first; f->last < options.insn_bit_size; f = f->next)
|
||
{
|
||
insn_field_cond *cond;
|
||
for (cond = f->conditions; cond != NULL; cond = cond->next)
|
||
{
|
||
if (cond->type == insn_field_cond_field)
|
||
{
|
||
int j;
|
||
if (strcmp (cond->string, f->val_string) == 0)
|
||
error (insn->line,
|
||
"Conditional `%s' of field `%s' refers to its self\n",
|
||
cond->string, f->val_string);
|
||
for (j = 0; j <= i && cond->field == NULL; j++)
|
||
{
|
||
insn_word_entry *refered_word = insn->word[j];
|
||
insn_field_entry *refered_field;
|
||
for (refered_field = refered_word->first;
|
||
refered_field != NULL && cond->field == NULL;
|
||
refered_field = refered_field->next)
|
||
{
|
||
if (refered_field->type == insn_field_string
|
||
&& strcmp (refered_field->val_string,
|
||
cond->string) == 0)
|
||
{
|
||
/* found field being refered to by conditonal */
|
||
cond->field = refered_field;
|
||
/* check refered to and this field are
|
||
the same size */
|
||
if (f->width != refered_field->width)
|
||
error (insn->line,
|
||
"Conditional `%s' of field `%s' should be of size %i\n",
|
||
cond->string, f->val_string,
|
||
refered_field->width);
|
||
}
|
||
}
|
||
}
|
||
if (cond->field == NULL)
|
||
error (insn->line,
|
||
"Conditional `%s' of field `%s' not yet defined\n",
|
||
cond->string, f->val_string);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
typedef enum
|
||
{
|
||
unknown_record = 0,
|
||
insn_record, /* default */
|
||
code_record,
|
||
cache_record,
|
||
compute_record,
|
||
scratch_record,
|
||
option_record,
|
||
string_function_record,
|
||
function_record,
|
||
internal_record,
|
||
define_record,
|
||
include_record,
|
||
model_processor_record,
|
||
model_macro_record,
|
||
model_data_record,
|
||
model_static_record,
|
||
model_function_record,
|
||
model_internal_record,
|
||
}
|
||
insn_record_type;
|
||
|
||
static const name_map insn_type_map[] = {
|
||
{"option", option_record},
|
||
{"cache", cache_record},
|
||
{"compute", compute_record},
|
||
{"scratch", scratch_record},
|
||
{"define", define_record},
|
||
{"include", include_record},
|
||
{"%s", string_function_record},
|
||
{"function", function_record},
|
||
{"internal", internal_record},
|
||
{"model", model_processor_record},
|
||
{"model-macro", model_macro_record},
|
||
{"model-data", model_data_record},
|
||
{"model-static", model_static_record},
|
||
{"model-internal", model_internal_record},
|
||
{"model-function", model_function_record},
|
||
{NULL, insn_record},
|
||
};
|
||
|
||
|
||
static int
|
||
record_is_old (table_entry *entry)
|
||
{
|
||
if (entry->nr_fields > record_type_field
|
||
&& strlen (entry->field[record_type_field]) == 0)
|
||
return 1;
|
||
return 0;
|
||
}
|
||
|
||
static insn_record_type
|
||
record_type (table_entry *entry)
|
||
{
|
||
switch (entry->type)
|
||
{
|
||
case table_code_entry:
|
||
return code_record;
|
||
|
||
case table_colon_entry:
|
||
if (record_is_old (entry))
|
||
{
|
||
/* old-format? */
|
||
if (entry->nr_fields > old_record_type_field)
|
||
{
|
||
int i = name2i (entry->field[old_record_type_field],
|
||
insn_type_map);
|
||
return i;
|
||
}
|
||
else
|
||
{
|
||
return unknown_record;
|
||
}
|
||
}
|
||
else if (entry->nr_fields > record_type_field
|
||
&& entry->field[0][0] == '\0')
|
||
{
|
||
/* new-format? */
|
||
int i = name2i (entry->field[record_type_field],
|
||
insn_type_map);
|
||
return i;
|
||
}
|
||
else
|
||
return insn_record; /* default */
|
||
}
|
||
return unknown_record;
|
||
}
|
||
|
||
static int
|
||
record_prefix_is (table_entry *entry, char ch, int nr_fields)
|
||
{
|
||
if (entry->type != table_colon_entry)
|
||
return 0;
|
||
if (entry->nr_fields < nr_fields)
|
||
return 0;
|
||
if (entry->field[0][0] != ch && ch != '\0')
|
||
return 0;
|
||
return 1;
|
||
}
|
||
|
||
static table_entry *
|
||
parse_model_data_record (insn_table *isa,
|
||
table *file,
|
||
table_entry *record,
|
||
int nr_fields, model_data **list)
|
||
{
|
||
table_entry *model_record = record;
|
||
table_entry *code_record = NULL;
|
||
model_data *new_data;
|
||
if (record->nr_fields < nr_fields)
|
||
error (record->line, "Incorrect number of fields\n");
|
||
record = table_read (file);
|
||
if (record->type == table_code_entry)
|
||
{
|
||
code_record = record;
|
||
record = table_read (file);
|
||
}
|
||
/* create the new data record */
|
||
new_data = ZALLOC (model_data);
|
||
new_data->line = model_record->line;
|
||
filter_parse (&new_data->flags,
|
||
model_record->field[record_filter_flags_field]);
|
||
new_data->entry = model_record;
|
||
new_data->code = code_record;
|
||
/* append it if not filtered out */
|
||
if (!is_filtered_out (options.flags_filter,
|
||
model_record->field[record_filter_flags_field])
|
||
&& !is_filtered_out (options.model_filter,
|
||
model_record->field[record_filter_models_field]))
|
||
{
|
||
while (*list != NULL)
|
||
list = &(*list)->next;
|
||
*list = new_data;
|
||
}
|
||
return record;
|
||
}
|
||
|
||
|
||
typedef enum
|
||
{
|
||
insn_bit_size_option = 1,
|
||
insn_specifying_widths_option,
|
||
hi_bit_nr_option,
|
||
flags_filter_option,
|
||
model_filter_option,
|
||
multi_sim_option,
|
||
format_names_option,
|
||
gen_delayed_branch,
|
||
unknown_option,
|
||
}
|
||
option_names;
|
||
|
||
static const name_map option_map[] = {
|
||
{"insn-bit-size", insn_bit_size_option},
|
||
{"insn-specifying-widths", insn_specifying_widths_option},
|
||
{"hi-bit-nr", hi_bit_nr_option},
|
||
{"flags-filter", flags_filter_option},
|
||
{"model-filter", model_filter_option},
|
||
{"multi-sim", multi_sim_option},
|
||
{"format-names", format_names_option},
|
||
{"gen-delayed-branch", gen_delayed_branch},
|
||
{NULL, unknown_option},
|
||
};
|
||
|
||
static table_entry *
|
||
parse_include_record (table *file, table_entry *record)
|
||
{
|
||
/* parse the include record */
|
||
if (record->nr_fields < nr_include_fields)
|
||
error (record->line, "Incorrect nr fields for include record\n");
|
||
/* process it */
|
||
if (!is_filtered_out (options.flags_filter,
|
||
record->field[record_filter_flags_field])
|
||
&& !is_filtered_out (options.model_filter,
|
||
record->field[record_filter_models_field]))
|
||
{
|
||
table_push (file, record->line, options.include,
|
||
record->field[include_filename_field]);
|
||
}
|
||
/* nb: can't read next record until after the file has been pushed */
|
||
record = table_read (file);
|
||
return record;
|
||
}
|
||
|
||
|
||
static table_entry *
|
||
parse_option_record (table *file, table_entry *record)
|
||
{
|
||
table_entry *option_record;
|
||
/* parse the option record */
|
||
option_record = record;
|
||
if (record->nr_fields < nr_option_fields)
|
||
error (record->line, "Incorrect nr of fields for option record\n");
|
||
record = table_read (file);
|
||
/* process it */
|
||
if (!is_filtered_out (options.flags_filter,
|
||
option_record->field[record_filter_flags_field])
|
||
&& !is_filtered_out (options.model_filter,
|
||
option_record->field[record_filter_models_field]))
|
||
{
|
||
char *name = option_record->field[option_name_field];
|
||
option_names option = name2i (name, option_map);
|
||
char *value = option_record->field[option_value_field];
|
||
switch (option)
|
||
{
|
||
case insn_bit_size_option:
|
||
{
|
||
options.insn_bit_size = a2i (value);
|
||
if (options.insn_bit_size < 0
|
||
|| options.insn_bit_size > max_insn_bit_size)
|
||
error (option_record->line,
|
||
"Instruction bit size out of range\n");
|
||
if (options.hi_bit_nr != options.insn_bit_size - 1
|
||
&& options.hi_bit_nr != 0)
|
||
error (option_record->line,
|
||
"insn-bit-size / hi-bit-nr conflict\n");
|
||
break;
|
||
}
|
||
case insn_specifying_widths_option:
|
||
{
|
||
options.insn_specifying_widths = a2i (value);
|
||
break;
|
||
}
|
||
case hi_bit_nr_option:
|
||
{
|
||
options.hi_bit_nr = a2i (value);
|
||
if (options.hi_bit_nr != 0
|
||
&& options.hi_bit_nr != options.insn_bit_size - 1)
|
||
error (option_record->line,
|
||
"hi-bit-nr / insn-bit-size conflict\n");
|
||
break;
|
||
}
|
||
case flags_filter_option:
|
||
{
|
||
filter_parse (&options.flags_filter, value);
|
||
break;
|
||
}
|
||
case model_filter_option:
|
||
{
|
||
filter_parse (&options.model_filter, value);
|
||
break;
|
||
}
|
||
case multi_sim_option:
|
||
{
|
||
options.gen.multi_sim = a2i (value);
|
||
break;
|
||
}
|
||
case format_names_option:
|
||
{
|
||
filter_parse (&options.format_name_filter, value);
|
||
break;
|
||
}
|
||
case gen_delayed_branch:
|
||
{
|
||
options.gen.delayed_branch = a2i (value);
|
||
break;
|
||
}
|
||
case unknown_option:
|
||
{
|
||
error (option_record->line, "Unknown option - %s\n", name);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
return record;
|
||
}
|
||
|
||
|
||
static table_entry *
|
||
parse_function_record (table *file,
|
||
table_entry *record,
|
||
function_entry ** list,
|
||
function_entry ** list_entry,
|
||
int is_internal, model_table *model)
|
||
{
|
||
function_entry *new_function;
|
||
new_function = ZALLOC (function_entry);
|
||
new_function->line = record->line;
|
||
new_function->is_internal = is_internal;
|
||
/* parse the function header */
|
||
if (record_is_old (record))
|
||
{
|
||
if (record->nr_fields < nr_old_function_fields)
|
||
error (record->line, "Missing fields from (old) function record\n");
|
||
new_function->type = record->field[old_function_typedef_field];
|
||
new_function->type = record->field[old_function_typedef_field];
|
||
if (record->nr_fields > old_function_param_field)
|
||
new_function->param = record->field[old_function_param_field];
|
||
new_function->name = record->field[old_function_name_field];
|
||
}
|
||
else
|
||
{
|
||
if (record->nr_fields < nr_function_fields)
|
||
error (record->line, "Missing fields from function record\n");
|
||
filter_parse (&new_function->flags,
|
||
record->field[record_filter_flags_field]);
|
||
filter_parse (&new_function->models,
|
||
record->field[record_filter_models_field]);
|
||
new_function->type = record->field[function_typedef_field];
|
||
new_function->param = record->field[function_param_field];
|
||
new_function->name = record->field[function_name_field];
|
||
}
|
||
record = table_read (file);
|
||
/* parse any function-model records */
|
||
while (record != NULL
|
||
&& record_prefix_is (record, '*', nr_function_model_fields))
|
||
{
|
||
char *model_name = record->field[function_model_name_field] + 1; /*skip `*' */
|
||
filter_parse (&new_function->models, model_name);
|
||
if (!filter_is_subset (model->processors, new_function->models))
|
||
{
|
||
error (record->line, "machine model `%s' undefined\n", model_name);
|
||
}
|
||
record = table_read (file);
|
||
}
|
||
/* parse the function body */
|
||
if (record->type == table_code_entry)
|
||
{
|
||
new_function->code = record;
|
||
record = table_read (file);
|
||
}
|
||
/* insert it */
|
||
if (!filter_is_subset (options.flags_filter, new_function->flags))
|
||
{
|
||
if (options.warn.discard)
|
||
notify (new_function->line, "Discarding function %s - filter flags\n",
|
||
new_function->name);
|
||
}
|
||
else if (new_function->models != NULL
|
||
&& !filter_is_common (options.model_filter, new_function->models))
|
||
{
|
||
if (options.warn.discard)
|
||
notify (new_function->line,
|
||
"Discarding function %s - filter models\n",
|
||
new_function->name);
|
||
}
|
||
else
|
||
{
|
||
while (*list != NULL)
|
||
list = &(*list)->next;
|
||
*list = new_function;
|
||
if (list_entry != NULL)
|
||
*list_entry = new_function;
|
||
}
|
||
/* done */
|
||
return record;
|
||
}
|
||
|
||
static void
|
||
parse_insn_model_record (table *file,
|
||
table_entry *record,
|
||
insn_entry * insn, model_table *model)
|
||
{
|
||
insn_model_entry **last_insn_model;
|
||
insn_model_entry *new_insn_model = ZALLOC (insn_model_entry);
|
||
/* parse it */
|
||
new_insn_model->line = record->line;
|
||
if (record->nr_fields > insn_model_unit_data_field)
|
||
new_insn_model->unit_data = record->field[insn_model_unit_data_field];
|
||
new_insn_model->insn = insn;
|
||
/* parse the model names, verify that all were defined */
|
||
new_insn_model->names = NULL;
|
||
filter_parse (&new_insn_model->names,
|
||
record->field[insn_model_name_field] + 1 /*skip `*' */ );
|
||
if (new_insn_model->names == NULL)
|
||
{
|
||
/* No processor names - a generic model entry, enter it into all
|
||
the non-empty fields */
|
||
int index;
|
||
for (index = 0; index < model->nr_models; index++)
|
||
if (insn->model[index] == 0)
|
||
{
|
||
insn->model[index] = new_insn_model;
|
||
}
|
||
/* also add the complete processor set to this processor's set */
|
||
filter_add (&insn->processors, model->processors);
|
||
}
|
||
else
|
||
{
|
||
/* Find the corresponding master model record for each name so
|
||
that they can be linked in. */
|
||
int index;
|
||
const char *name = "";
|
||
while (1)
|
||
{
|
||
name = filter_next (new_insn_model->names, name);
|
||
if (name == NULL)
|
||
break;
|
||
index = filter_is_member (model->processors, name) - 1;
|
||
if (index < 0)
|
||
{
|
||
error (new_insn_model->line,
|
||
"machine model `%s' undefined\n", name);
|
||
}
|
||
/* store it in the corresponding model array entry */
|
||
if (insn->model[index] != NULL && insn->model[index]->names != NULL)
|
||
{
|
||
warning (new_insn_model->line,
|
||
"machine model `%s' previously defined\n", name);
|
||
error (insn->model[index]->line, "earlier definition\n");
|
||
}
|
||
insn->model[index] = new_insn_model;
|
||
/* also add the name to the instructions processor set as an
|
||
alternative lookup mechanism */
|
||
filter_parse (&insn->processors, name);
|
||
}
|
||
}
|
||
/* link it in */
|
||
last_insn_model = &insn->models;
|
||
while ((*last_insn_model) != NULL)
|
||
last_insn_model = &(*last_insn_model)->next;
|
||
*last_insn_model = new_insn_model;
|
||
}
|
||
|
||
|
||
static void
|
||
parse_insn_mnemonic_record (table *file,
|
||
table_entry *record, insn_entry * insn)
|
||
{
|
||
insn_mnemonic_entry **last_insn_mnemonic;
|
||
insn_mnemonic_entry *new_insn_mnemonic = ZALLOC (insn_mnemonic_entry);
|
||
/* parse it */
|
||
new_insn_mnemonic->line = record->line;
|
||
ASSERT (record->nr_fields > insn_mnemonic_format_field);
|
||
new_insn_mnemonic->format = record->field[insn_mnemonic_format_field];
|
||
ASSERT (new_insn_mnemonic->format[0] == '"');
|
||
if (new_insn_mnemonic->format[strlen (new_insn_mnemonic->format) - 1] !=
|
||
'"')
|
||
error (new_insn_mnemonic->line,
|
||
"Missing closing double quote in mnemonic field\n");
|
||
if (record->nr_fields > insn_mnemonic_condition_field)
|
||
new_insn_mnemonic->condition =
|
||
record->field[insn_mnemonic_condition_field];
|
||
new_insn_mnemonic->insn = insn;
|
||
/* insert it */
|
||
last_insn_mnemonic = &insn->mnemonics;
|
||
while ((*last_insn_mnemonic) != NULL)
|
||
last_insn_mnemonic = &(*last_insn_mnemonic)->next;
|
||
insn->nr_mnemonics++;
|
||
*last_insn_mnemonic = new_insn_mnemonic;
|
||
}
|
||
|
||
|
||
static table_entry *
|
||
parse_macro_record (table *file, table_entry *record)
|
||
{
|
||
#if 1
|
||
error (record->line, "Macros are not implemented\n");
|
||
#else
|
||
/* parse the define record */
|
||
if (record->nr_fields < nr_define_fields)
|
||
error (record->line, "Incorrect nr fields for define record\n");
|
||
/* process it */
|
||
if (!is_filtered_out (options.flags_filter,
|
||
record->field[record_filter_flags_field])
|
||
&& !is_filtered_out (options.model_filter,
|
||
record->field[record_filter_models_field]))
|
||
{
|
||
table_define (file,
|
||
record->line,
|
||
record->field[macro_name_field],
|
||
record->field[macro_args_field],
|
||
record->field[macro_expr_field]);
|
||
}
|
||
record = table_read (file);
|
||
#endif
|
||
return record;
|
||
}
|
||
|
||
|
||
insn_table *
|
||
load_insn_table (const char *file_name, cache_entry *cache)
|
||
{
|
||
table *file = table_open (file_name);
|
||
table_entry *record = table_read (file);
|
||
|
||
insn_table *isa = ZALLOC (insn_table);
|
||
model_table *model = ZALLOC (model_table);
|
||
|
||
isa->model = model;
|
||
isa->caches = cache;
|
||
|
||
while (record != NULL)
|
||
{
|
||
|
||
switch (record_type (record))
|
||
{
|
||
|
||
case include_record:
|
||
{
|
||
record = parse_include_record (file, record);
|
||
break;
|
||
}
|
||
|
||
case option_record:
|
||
{
|
||
if (isa->insns != NULL)
|
||
error (record->line, "Option after first instruction\n");
|
||
record = parse_option_record (file, record);
|
||
break;
|
||
}
|
||
|
||
case string_function_record:
|
||
{
|
||
function_entry *function = NULL;
|
||
record = parse_function_record (file, record,
|
||
&isa->functions,
|
||
&function, 0 /*is-internal */ ,
|
||
model);
|
||
/* convert a string function record into an internal function */
|
||
if (function != NULL)
|
||
{
|
||
char *name = NZALLOC (char,
|
||
(strlen ("str_")
|
||
+ strlen (function->name) + 1));
|
||
strcat (name, "str_");
|
||
strcat (name, function->name);
|
||
function->name = name;
|
||
function->type = "const char *";
|
||
}
|
||
break;
|
||
}
|
||
|
||
case function_record: /* function record */
|
||
{
|
||
record = parse_function_record (file, record,
|
||
&isa->functions,
|
||
NULL, 0 /*is-internal */ ,
|
||
model);
|
||
break;
|
||
}
|
||
|
||
case internal_record:
|
||
{
|
||
/* only insert it into the function list if it is unknown */
|
||
function_entry *function = NULL;
|
||
record = parse_function_record (file, record,
|
||
&isa->functions,
|
||
&function, 1 /*is-internal */ ,
|
||
model);
|
||
/* check what was inserted to see if a pseudo-instruction
|
||
entry also needs to be created */
|
||
if (function != NULL)
|
||
{
|
||
insn_entry **insn = NULL;
|
||
if (strcmp (function->name, "illegal") == 0)
|
||
{
|
||
/* illegal function save it away */
|
||
if (isa->illegal_insn != NULL)
|
||
{
|
||
warning (function->line,
|
||
"Multiple illegal instruction definitions\n");
|
||
error (isa->illegal_insn->line,
|
||
"Location of first illegal instruction\n");
|
||
}
|
||
else
|
||
insn = &isa->illegal_insn;
|
||
}
|
||
if (insn != NULL)
|
||
{
|
||
*insn = ZALLOC (insn_entry);
|
||
(*insn)->line = function->line;
|
||
(*insn)->name = function->name;
|
||
(*insn)->code = function->code;
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
|
||
case scratch_record: /* cache macro records */
|
||
case cache_record:
|
||
case compute_record:
|
||
{
|
||
cache_entry *new_cache;
|
||
/* parse the cache record */
|
||
if (record->nr_fields < nr_cache_fields)
|
||
error (record->line,
|
||
"Incorrect nr of fields for scratch/cache/compute record\n");
|
||
/* create it */
|
||
new_cache = ZALLOC (cache_entry);
|
||
new_cache->line = record->line;
|
||
filter_parse (&new_cache->flags,
|
||
record->field[record_filter_flags_field]);
|
||
filter_parse (&new_cache->models,
|
||
record->field[record_filter_models_field]);
|
||
new_cache->type = record->field[cache_typedef_field];
|
||
new_cache->name = record->field[cache_name_field];
|
||
filter_parse (&new_cache->original_fields,
|
||
record->field[cache_original_fields_field]);
|
||
new_cache->expression = record->field[cache_expression_field];
|
||
/* insert it but only if not filtered out */
|
||
if (!filter_is_subset (options.flags_filter, new_cache->flags))
|
||
{
|
||
notify (new_cache->line,
|
||
"Discarding cache entry %s - filter flags\n",
|
||
new_cache->name);
|
||
}
|
||
else if (is_filtered_out (options.model_filter,
|
||
record->
|
||
field[record_filter_models_field]))
|
||
{
|
||
notify (new_cache->line,
|
||
"Discarding cache entry %s - filter models\n",
|
||
new_cache->name);
|
||
}
|
||
else
|
||
{
|
||
cache_entry **last;
|
||
last = &isa->caches;
|
||
while (*last != NULL)
|
||
last = &(*last)->next;
|
||
*last = new_cache;
|
||
}
|
||
/* advance things */
|
||
record = table_read (file);
|
||
break;
|
||
}
|
||
|
||
/* model records */
|
||
case model_processor_record:
|
||
{
|
||
model_entry *new_model;
|
||
/* parse the model */
|
||
if (record->nr_fields < nr_model_processor_fields)
|
||
error (record->line,
|
||
"Incorrect nr of fields for model record\n");
|
||
if (isa->insns != NULL)
|
||
error (record->line, "Model appears after first instruction\n");
|
||
new_model = ZALLOC (model_entry);
|
||
filter_parse (&new_model->flags,
|
||
record->field[record_filter_flags_field]);
|
||
new_model->line = record->line;
|
||
new_model->name = record->field[model_name_field];
|
||
new_model->full_name = record->field[model_full_name_field];
|
||
new_model->unit_data = record->field[model_unit_data_field];
|
||
/* only insert it if not filtered out */
|
||
if (!filter_is_subset (options.flags_filter, new_model->flags))
|
||
{
|
||
notify (new_model->line,
|
||
"Discarding processor model %s - filter flags\n",
|
||
new_model->name);
|
||
}
|
||
else if (is_filtered_out (options.model_filter,
|
||
record->
|
||
field[record_filter_models_field]))
|
||
{
|
||
notify (new_model->line,
|
||
"Discarding processor model %s - filter models\n",
|
||
new_model->name);
|
||
}
|
||
else if (filter_is_member (model->processors, new_model->name))
|
||
{
|
||
error (new_model->line, "Duplicate processor model %s\n",
|
||
new_model->name);
|
||
}
|
||
else
|
||
{
|
||
model_entry **last;
|
||
last = &model->models;
|
||
while (*last != NULL)
|
||
last = &(*last)->next;
|
||
*last = new_model;
|
||
/* count it */
|
||
model->nr_models++;
|
||
filter_parse (&model->processors, new_model->name);
|
||
}
|
||
/* advance things */
|
||
record = table_read (file);
|
||
}
|
||
break;
|
||
|
||
case model_macro_record:
|
||
record = parse_model_data_record (isa, file, record,
|
||
nr_model_macro_fields,
|
||
&model->macros);
|
||
break;
|
||
|
||
case model_data_record:
|
||
record = parse_model_data_record (isa, file, record,
|
||
nr_model_data_fields,
|
||
&model->data);
|
||
break;
|
||
|
||
case model_static_record:
|
||
record = parse_function_record (file, record,
|
||
&model->statics,
|
||
NULL, 0 /*is internal */ ,
|
||
model);
|
||
break;
|
||
|
||
case model_internal_record:
|
||
record = parse_function_record (file, record,
|
||
&model->internals,
|
||
NULL, 1 /*is internal */ ,
|
||
model);
|
||
break;
|
||
|
||
case model_function_record:
|
||
record = parse_function_record (file, record,
|
||
&model->functions,
|
||
NULL, 0 /*is internal */ ,
|
||
model);
|
||
break;
|
||
|
||
case insn_record: /* instruction records */
|
||
{
|
||
insn_entry *new_insn;
|
||
char *format;
|
||
/* parse the instruction */
|
||
if (record->nr_fields < nr_insn_fields)
|
||
error (record->line,
|
||
"Incorrect nr of fields for insn record\n");
|
||
new_insn = ZALLOC (insn_entry);
|
||
new_insn->line = record->line;
|
||
filter_parse (&new_insn->flags,
|
||
record->field[record_filter_flags_field]);
|
||
/* save the format field. Can't parse it until after the
|
||
filter-out checks. Could be filtered out because the
|
||
format is invalid */
|
||
format = record->field[insn_word_field];
|
||
new_insn->format_name = record->field[insn_format_name_field];
|
||
if (options.format_name_filter != NULL
|
||
&& !filter_is_member (options.format_name_filter,
|
||
new_insn->format_name))
|
||
error (new_insn->line,
|
||
"Unreconized instruction format name `%s'\n",
|
||
new_insn->format_name);
|
||
filter_parse (&new_insn->options,
|
||
record->field[insn_options_field]);
|
||
new_insn->name = record->field[insn_name_field];
|
||
record = table_read (file);
|
||
/* Parse any model/assember records */
|
||
new_insn->nr_models = model->nr_models;
|
||
new_insn->model =
|
||
NZALLOC (insn_model_entry *, model->nr_models + 1);
|
||
while (record != NULL)
|
||
{
|
||
if (record_prefix_is (record, '*', nr_insn_model_fields))
|
||
parse_insn_model_record (file, record, new_insn, model);
|
||
else
|
||
if (record_prefix_is (record, '"', nr_insn_mnemonic_fields))
|
||
parse_insn_mnemonic_record (file, record, new_insn);
|
||
else
|
||
break;
|
||
/* advance */
|
||
record = table_read (file);
|
||
}
|
||
/* Parse the code record */
|
||
if (record != NULL && record->type == table_code_entry)
|
||
{
|
||
new_insn->code = record;
|
||
record = table_read (file);
|
||
}
|
||
else if (options.warn.unimplemented)
|
||
notify (new_insn->line, "unimplemented\n");
|
||
/* insert it */
|
||
if (!filter_is_subset (options.flags_filter, new_insn->flags))
|
||
{
|
||
if (options.warn.discard)
|
||
notify (new_insn->line,
|
||
"Discarding instruction %s (flags-filter)\n",
|
||
new_insn->name);
|
||
}
|
||
else if (new_insn->processors != NULL
|
||
&& options.model_filter != NULL
|
||
&& !filter_is_common (options.model_filter,
|
||
new_insn->processors))
|
||
{
|
||
/* only discard an instruction based in the processor
|
||
model when both the instruction and the options are
|
||
nonempty */
|
||
if (options.warn.discard)
|
||
notify (new_insn->line,
|
||
"Discarding instruction %s (processor-model)\n",
|
||
new_insn->name);
|
||
}
|
||
else
|
||
{
|
||
insn_entry **last;
|
||
/* finish the parsing */
|
||
parse_insn_words (new_insn, format);
|
||
/* append it */
|
||
last = &isa->insns;
|
||
while (*last)
|
||
last = &(*last)->next;
|
||
*last = new_insn;
|
||
/* update global isa counters */
|
||
isa->nr_insns++;
|
||
if (isa->max_nr_words < new_insn->nr_words)
|
||
isa->max_nr_words = new_insn->nr_words;
|
||
filter_add (&isa->flags, new_insn->flags);
|
||
filter_add (&isa->options, new_insn->options);
|
||
}
|
||
break;
|
||
}
|
||
|
||
case define_record:
|
||
record = parse_macro_record (file, record);
|
||
break;
|
||
|
||
case unknown_record:
|
||
case code_record:
|
||
error (record->line, "Unknown or unexpected entry\n");
|
||
|
||
|
||
}
|
||
}
|
||
return isa;
|
||
}
|
||
|
||
|
||
void
|
||
print_insn_words (lf *file, const insn_entry *insn)
|
||
{
|
||
insn_word_entry *word = insn->words;
|
||
if (word != NULL)
|
||
{
|
||
while (1)
|
||
{
|
||
insn_field_entry *field = word->first;
|
||
while (1)
|
||
{
|
||
insn_field_cond *cond;
|
||
|
||
if (options.insn_specifying_widths)
|
||
lf_printf (file, "%d.", field->width);
|
||
else
|
||
lf_printf (file, "%d.",
|
||
i2target (options.hi_bit_nr, field->first));
|
||
switch (field->type)
|
||
{
|
||
case insn_field_invalid:
|
||
ASSERT (0);
|
||
break;
|
||
case insn_field_int:
|
||
lf_printf (file, "0x%lx", (long) field->val_int);
|
||
break;
|
||
case insn_field_reserved:
|
||
lf_printf (file, "/");
|
||
break;
|
||
case insn_field_wild:
|
||
lf_printf (file, "*");
|
||
break;
|
||
case insn_field_string:
|
||
lf_printf (file, "%s", field->val_string);
|
||
|
||
if (field->conditions == NULL)
|
||
break;
|
||
|
||
if (field->conditions->test == insn_field_cond_eq)
|
||
{
|
||
if (field->conditions->type == insn_field_cond_value)
|
||
lf_printf (file, "=%ld",
|
||
(long) field->conditions->value);
|
||
else
|
||
lf_printf (file, "=%s", field->conditions->string);
|
||
|
||
/* There can be only one equality condition. */
|
||
ASSERT (field->conditions->next == NULL);
|
||
break;
|
||
}
|
||
|
||
for (cond = field->conditions;
|
||
cond != NULL;
|
||
cond = cond->next)
|
||
{
|
||
ASSERT (cond->test == insn_field_cond_ne);
|
||
|
||
if (cond->type == insn_field_cond_value)
|
||
lf_printf (file, "!%ld", (long) cond->value);
|
||
else
|
||
lf_printf (file, "!%s", cond->string);
|
||
}
|
||
break;
|
||
}
|
||
if (field == word->last)
|
||
break;
|
||
field = field->next;
|
||
lf_printf (file, ",");
|
||
}
|
||
word = word->next;
|
||
if (word == NULL)
|
||
break;
|
||
lf_printf (file, "+");
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
void
|
||
function_entry_traverse (lf *file,
|
||
const function_entry *functions,
|
||
function_entry_handler * handler, void *data)
|
||
{
|
||
const function_entry *function;
|
||
for (function = functions; function != NULL; function = function->next)
|
||
{
|
||
handler (file, function, data);
|
||
}
|
||
}
|
||
|
||
void
|
||
insn_table_traverse_insn (lf *file,
|
||
const insn_table *isa,
|
||
insn_entry_handler * handler, void *data)
|
||
{
|
||
const insn_entry *insn;
|
||
for (insn = isa->insns; insn != NULL; insn = insn->next)
|
||
{
|
||
handler (file, isa, insn, data);
|
||
}
|
||
}
|
||
|
||
|
||
static void
|
||
dump_function_entry (lf *file,
|
||
const char *prefix,
|
||
const function_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(function_entry *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
dump_line_ref (file, "\n(line ", entry->line, ")");
|
||
dump_filter (file, "\n(flags ", entry->flags, ")");
|
||
lf_printf (file, "\n(type \"%s\")", entry->type);
|
||
lf_printf (file, "\n(name \"%s\")", entry->name);
|
||
lf_printf (file, "\n(param \"%s\")", entry->param);
|
||
dump_table_entry (file, "\n(code ", entry->code, ")");
|
||
lf_printf (file, "\n(is_internal %d)", entry->is_internal);
|
||
lf_printf (file, "\n(next %p)", entry->next);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_function_entries (lf *file,
|
||
const char *prefix,
|
||
const function_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
lf_indent (file, +1);
|
||
while (entry != NULL)
|
||
{
|
||
dump_function_entry (file, "\n(", entry, ")");
|
||
entry = entry->next;
|
||
}
|
||
lf_indent (file, -1);
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static char *
|
||
cache_entry_type_to_str (cache_entry_type type)
|
||
{
|
||
switch (type)
|
||
{
|
||
case scratch_value:
|
||
return "scratch";
|
||
case cache_value:
|
||
return "cache";
|
||
case compute_value:
|
||
return "compute";
|
||
}
|
||
ERROR ("Bad switch");
|
||
return 0;
|
||
}
|
||
|
||
static void
|
||
dump_cache_entry (lf *file,
|
||
const char *prefix,
|
||
const cache_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(cache_entry *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
dump_line_ref (file, "\n(line ", entry->line, ")");
|
||
dump_filter (file, "\n(flags ", entry->flags, ")");
|
||
lf_printf (file, "\n(entry_type \"%s\")",
|
||
cache_entry_type_to_str (entry->entry_type));
|
||
lf_printf (file, "\n(name \"%s\")", entry->name);
|
||
dump_filter (file, "\n(original_fields ", entry->original_fields, ")");
|
||
lf_printf (file, "\n(type \"%s\")", entry->type);
|
||
lf_printf (file, "\n(expression \"%s\")", entry->expression);
|
||
lf_printf (file, "\n(next %p)", entry->next);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
void
|
||
dump_cache_entries (lf *file,
|
||
const char *prefix,
|
||
const cache_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
lf_indent (file, +1);
|
||
while (entry != NULL)
|
||
{
|
||
dump_cache_entry (file, "\n(", entry, ")");
|
||
entry = entry->next;
|
||
}
|
||
lf_indent (file, -1);
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_model_data (lf *file,
|
||
const char *prefix,
|
||
const model_data *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(model_data *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_line_ref (file, "\n(line ", entry->line, ")");
|
||
dump_filter (file, "\n(flags ", entry->flags, ")");
|
||
dump_table_entry (file, "\n(entry ", entry->entry, ")");
|
||
dump_table_entry (file, "\n(code ", entry->code, ")");
|
||
lf_printf (file, "\n(next %p)", entry->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", prefix);
|
||
}
|
||
|
||
static void
|
||
dump_model_datas (lf *file,
|
||
const char *prefix,
|
||
const model_data *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
lf_indent (file, +1);
|
||
while (entry != NULL)
|
||
{
|
||
dump_model_data (file, "\n(", entry, ")");
|
||
entry = entry->next;
|
||
}
|
||
lf_indent (file, -1);
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_model_entry (lf *file,
|
||
const char *prefix,
|
||
const model_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(model_entry *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_line_ref (file, "\n(line ", entry->line, ")");
|
||
dump_filter (file, "\n(flags ", entry->flags, ")");
|
||
lf_printf (file, "\n(name \"%s\")", entry->name);
|
||
lf_printf (file, "\n(full_name \"%s\")", entry->full_name);
|
||
lf_printf (file, "\n(unit_data \"%s\")", entry->unit_data);
|
||
lf_printf (file, "\n(next %p)", entry->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", prefix);
|
||
}
|
||
|
||
static void
|
||
dump_model_entries (lf *file,
|
||
const char *prefix,
|
||
const model_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
lf_indent (file, +1);
|
||
while (entry != NULL)
|
||
{
|
||
dump_model_entry (file, "\n(", entry, ")");
|
||
entry = entry->next;
|
||
}
|
||
lf_indent (file, -1);
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
|
||
static void
|
||
dump_model_table (lf *file,
|
||
const char *prefix,
|
||
const model_table *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(model_table *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_filter (file, "\n(processors ", entry->processors, ")");
|
||
lf_printf (file, "\n(nr_models %d)", entry->nr_models);
|
||
dump_model_entries (file, "\n(models ", entry->models, ")");
|
||
dump_model_datas (file, "\n(macros ", entry->macros, ")");
|
||
dump_model_datas (file, "\n(data ", entry->data, ")");
|
||
dump_function_entries (file, "\n(statics ", entry->statics, ")");
|
||
dump_function_entries (file, "\n(internals ", entry->functions, ")");
|
||
dump_function_entries (file, "\n(functions ", entry->functions, ")");
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
|
||
static char *
|
||
insn_field_type_to_str (insn_field_type type)
|
||
{
|
||
switch (type)
|
||
{
|
||
case insn_field_invalid:
|
||
ASSERT (0);
|
||
return "(invalid)";
|
||
case insn_field_int:
|
||
return "int";
|
||
case insn_field_reserved:
|
||
return "reserved";
|
||
case insn_field_wild:
|
||
return "wild";
|
||
case insn_field_string:
|
||
return "string";
|
||
}
|
||
ERROR ("bad switch");
|
||
return 0;
|
||
}
|
||
|
||
void
|
||
dump_insn_field (lf *file,
|
||
const char *prefix,
|
||
const insn_field_entry *field,
|
||
const char *suffix)
|
||
{
|
||
char *sep = " ";
|
||
lf_printf (file, "%s(insn_field_entry *) %p", prefix, field);
|
||
if (field != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
lf_printf (file, "%s(first %d)", sep, field->first);
|
||
lf_printf (file, "%s(last %d)", sep, field->last);
|
||
lf_printf (file, "%s(width %d)", sep, field->width);
|
||
lf_printf (file, "%s(type %s)", sep,
|
||
insn_field_type_to_str (field->type));
|
||
switch (field->type)
|
||
{
|
||
case insn_field_invalid:
|
||
ASSERT (0);
|
||
break;
|
||
case insn_field_int:
|
||
lf_printf (file, "%s(val 0x%lx)", sep, (long) field->val_int);
|
||
break;
|
||
case insn_field_reserved:
|
||
/* nothing output */
|
||
break;
|
||
case insn_field_wild:
|
||
/* nothing output */
|
||
break;
|
||
case insn_field_string:
|
||
lf_printf (file, "%s(val \"%s\")", sep, field->val_string);
|
||
break;
|
||
}
|
||
lf_printf (file, "%s(next %p)", sep, field->next);
|
||
lf_printf (file, "%s(prev %p)", sep, field->prev);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
void
|
||
dump_insn_word_entry (lf *file,
|
||
const char *prefix,
|
||
const insn_word_entry *word,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(insn_word_entry *) %p", prefix, word);
|
||
if (word != NULL)
|
||
{
|
||
int i;
|
||
insn_field_entry *field;
|
||
lf_indent (file, +1);
|
||
lf_printf (file, "\n(first %p)", word->first);
|
||
lf_printf (file, "\n(last %p)", word->last);
|
||
lf_printf (file, "\n(bit");
|
||
for (i = 0; i < options.insn_bit_size; i++)
|
||
lf_printf (file, "\n ((value %d) (mask %d) (field %p))",
|
||
word->bit[i]->value, word->bit[i]->mask,
|
||
word->bit[i]->field);
|
||
lf_printf (file, ")");
|
||
for (field = word->first; field != NULL; field = field->next)
|
||
dump_insn_field (file, "\n(", field, ")");
|
||
dump_filter (file, "\n(field_names ", word->field_names, ")");
|
||
lf_printf (file, "\n(next %p)", word->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_insn_word_entries (lf *file,
|
||
const char *prefix,
|
||
const insn_word_entry *word,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
while (word != NULL)
|
||
{
|
||
dump_insn_word_entry (file, "\n(", word, ")");
|
||
word = word->next;
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_insn_model_entry (lf *file,
|
||
const char *prefix,
|
||
const insn_model_entry *model,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(insn_model_entry *) %p", prefix, model);
|
||
if (model != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_line_ref (file, "\n(line ", model->line, ")");
|
||
dump_filter (file, "\n(names ", model->names, ")");
|
||
lf_printf (file, "\n(full_name \"%s\")", model->full_name);
|
||
lf_printf (file, "\n(unit_data \"%s\")", model->unit_data);
|
||
lf_printf (file, "\n(insn (insn_entry *) %p)", model->insn);
|
||
lf_printf (file, "\n(next (insn_model_entry *) %p)", model->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_insn_model_entries (lf *file,
|
||
const char *prefix,
|
||
const insn_model_entry *model,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
while (model != NULL)
|
||
{
|
||
dump_insn_model_entry (file, "\n", model, "");
|
||
model = model->next;
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
|
||
static void
|
||
dump_insn_mnemonic_entry (lf *file,
|
||
const char *prefix,
|
||
const insn_mnemonic_entry *mnemonic,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(insn_mnemonic_entry *) %p", prefix, mnemonic);
|
||
if (mnemonic != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_line_ref (file, "\n(line ", mnemonic->line, ")");
|
||
lf_printf (file, "\n(format \"%s\")", mnemonic->format);
|
||
lf_printf (file, "\n(condition \"%s\")", mnemonic->condition);
|
||
lf_printf (file, "\n(insn (insn_entry *) %p)", mnemonic->insn);
|
||
lf_printf (file, "\n(next (insn_mnemonic_entry *) %p)", mnemonic->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_insn_mnemonic_entries (lf *file,
|
||
const char *prefix,
|
||
const insn_mnemonic_entry *mnemonic,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
while (mnemonic != NULL)
|
||
{
|
||
dump_insn_mnemonic_entry (file, "\n", mnemonic, "");
|
||
mnemonic = mnemonic->next;
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
void
|
||
dump_insn_entry (lf *file,
|
||
const char *prefix,
|
||
const insn_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(insn_entry *) %p", prefix, entry);
|
||
if (entry != NULL)
|
||
{
|
||
int i;
|
||
lf_indent (file, +1);
|
||
dump_line_ref (file, "\n(line ", entry->line, ")");
|
||
dump_filter (file, "\n(flags ", entry->flags, ")");
|
||
lf_printf (file, "\n(nr_words %d)", entry->nr_words);
|
||
dump_insn_word_entries (file, "\n(words ", entry->words, ")");
|
||
lf_printf (file, "\n(word");
|
||
for (i = 0; i < entry->nr_models; i++)
|
||
lf_printf (file, " %p", entry->word[i]);
|
||
lf_printf (file, ")");
|
||
dump_filter (file, "\n(field_names ", entry->field_names, ")");
|
||
lf_printf (file, "\n(format_name \"%s\")", entry->format_name);
|
||
dump_filter (file, "\n(options ", entry->options, ")");
|
||
lf_printf (file, "\n(name \"%s\")", entry->name);
|
||
lf_printf (file, "\n(nr_models %d)", entry->nr_models);
|
||
dump_insn_model_entries (file, "\n(models ", entry->models, ")");
|
||
lf_printf (file, "\n(model");
|
||
for (i = 0; i < entry->nr_models; i++)
|
||
lf_printf (file, " %p", entry->model[i]);
|
||
lf_printf (file, ")");
|
||
dump_filter (file, "\n(processors ", entry->processors, ")");
|
||
dump_insn_mnemonic_entries (file, "\n(mnemonics ", entry->mnemonics,
|
||
")");
|
||
dump_table_entry (file, "\n(code ", entry->code, ")");
|
||
lf_printf (file, "\n(next %p)", entry->next);
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
static void
|
||
dump_insn_entries (lf *file,
|
||
const char *prefix,
|
||
const insn_entry *entry,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s", prefix);
|
||
lf_indent (file, +1);
|
||
while (entry != NULL)
|
||
{
|
||
dump_insn_entry (file, "\n(", entry, ")");
|
||
entry = entry->next;
|
||
}
|
||
lf_indent (file, -1);
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
|
||
void
|
||
dump_insn_table (lf *file,
|
||
const char *prefix,
|
||
const insn_table *isa,
|
||
const char *suffix)
|
||
{
|
||
lf_printf (file, "%s(insn_table *) %p", prefix, isa);
|
||
if (isa != NULL)
|
||
{
|
||
lf_indent (file, +1);
|
||
dump_cache_entries (file, "\n(caches ", isa->caches, ")");
|
||
lf_printf (file, "\n(nr_insns %d)", isa->nr_insns);
|
||
lf_printf (file, "\n(max_nr_words %d)", isa->max_nr_words);
|
||
dump_insn_entries (file, "\n(insns ", isa->insns, ")");
|
||
dump_function_entries (file, "\n(functions ", isa->functions, ")");
|
||
dump_insn_entry (file, "\n(illegal_insn ", isa->illegal_insn, ")");
|
||
dump_model_table (file, "\n(model ", isa->model, ")");
|
||
dump_filter (file, "\n(flags ", isa->flags, ")");
|
||
dump_filter (file, "\n(options ", isa->options, ")");
|
||
lf_indent (file, -1);
|
||
}
|
||
lf_printf (file, "%s", suffix);
|
||
}
|
||
|
||
#ifdef MAIN
|
||
|
||
igen_options options;
|
||
|
||
int
|
||
main (int argc, char **argv)
|
||
{
|
||
insn_table *isa;
|
||
lf *l;
|
||
|
||
INIT_OPTIONS ();
|
||
|
||
if (argc == 3)
|
||
filter_parse (&options.flags_filter, argv[2]);
|
||
else if (argc != 2)
|
||
error (NULL, "Usage: insn <insn-table> [ <filter-in> ]\n");
|
||
|
||
isa = load_insn_table (argv[1], NULL);
|
||
l = lf_open ("-", "stdout", lf_omit_references, lf_is_text, "tmp-ld-insn");
|
||
dump_insn_table (l, "(isa ", isa, ")\n");
|
||
|
||
return 0;
|
||
}
|
||
|
||
#endif
|