1 /* CGEN generic opcode support.
3 Copyright (C) 1996, 1997 Free Software Foundation, Inc.
5 This file is part of the GNU Binutils and GDB, the GNU debugger.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2, or (at your option)
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License along
18 with this program; if not, write to the Free Software Foundation, Inc.,
19 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
30 #include "libiberty.h"
32 #include "opcode/cgen.h"
35 These record the state of the currently selected cpu, machine, endian, etc.
36 They are set by cgen_set_cpu. */
38 /* Current opcode data. */
39 CGEN_OPCODE_DATA *cgen_current_opcode_data;
41 /* Current machine (a la BFD machine number). */
42 int cgen_current_mach;
45 enum cgen_endian cgen_current_endian = CGEN_ENDIAN_UNKNOWN;
48 cgen_set_cpu (data, mach, endian)
49 CGEN_OPCODE_DATA *data;
51 enum cgen_endian endian;
53 cgen_current_opcode_data = data;
54 cgen_current_mach = mach;
55 cgen_current_endian = endian;
57 #if 0 /* This isn't done here because it would put assembler support in the
58 disassembler, etc. The caller is required to call these after calling
60 /* Reset the hash tables. */
66 static unsigned int hash_keyword_name
67 PARAMS ((const struct cgen_keyword *, const char *));
68 static unsigned int hash_keyword_value
69 PARAMS ((const struct cgen_keyword *, int));
70 static void build_keyword_hash_tables
71 PARAMS ((struct cgen_keyword *));
73 /* Return number of hash table entries to use for N elements. */
74 #define KEYWORD_HASH_SIZE(n) ((n) <= 31 ? 17 : 31)
76 /* Look up *NAMEP in the keyword table KT.
77 The result is the keyword entry or NULL if not found. */
79 const struct cgen_keyword_entry *
80 cgen_keyword_lookup_name (kt, name)
81 struct cgen_keyword *kt;
84 const struct cgen_keyword_entry *ke;
87 if (kt->name_hash_table == NULL)
88 build_keyword_hash_tables (kt);
90 ke = kt->name_hash_table[hash_keyword_name (kt, name)];
92 /* We do case insensitive comparisons.
93 If that ever becomes a problem, add an attribute that denotes
94 "do case sensitive comparisons". */
103 || (isalpha (*p) && tolower (*p) == tolower (*n))))
115 /* Look up VALUE in the keyword table KT.
116 The result is the keyword entry or NULL if not found. */
118 const struct cgen_keyword_entry *
119 cgen_keyword_lookup_value (kt, value)
120 struct cgen_keyword *kt;
123 const struct cgen_keyword_entry *ke;
125 if (kt->name_hash_table == NULL)
126 build_keyword_hash_tables (kt);
128 ke = kt->value_hash_table[hash_keyword_value (kt, value)];
132 if (value == ke->value)
140 /* Add an entry to a keyword table. */
143 cgen_keyword_add (kt, ke)
144 struct cgen_keyword *kt;
145 struct cgen_keyword_entry *ke;
149 if (kt->name_hash_table == NULL)
150 build_keyword_hash_tables (kt);
152 hash = hash_keyword_name (kt, ke->name);
153 ke->next_name = kt->name_hash_table[hash];
154 kt->name_hash_table[hash] = ke;
156 hash = hash_keyword_value (kt, ke->value);
157 ke->next_value = kt->value_hash_table[hash];
158 kt->value_hash_table[hash] = ke;
161 /* FIXME: Need function to return count of keywords. */
163 /* Initialize a keyword table search.
164 SPEC is a specification of what to search for.
165 A value of NULL means to find every keyword.
166 Currently NULL is the only acceptable value [further specification
168 The result is an opaque data item used to record the search status.
169 It is passed to each call to cgen_keyword_search_next. */
171 struct cgen_keyword_search
172 cgen_keyword_search_init (kt, spec)
173 struct cgen_keyword *kt;
176 struct cgen_keyword_search search;
178 /* FIXME: Need to specify format of PARAMS. */
182 if (kt->name_hash_table == NULL)
183 build_keyword_hash_tables (kt);
187 search.current_hash = 0;
188 search.current_entry = NULL;
192 /* Return the next keyword specified by SEARCH.
193 The result is the next entry or NULL if there are no more. */
195 const struct cgen_keyword_entry *
196 cgen_keyword_search_next (search)
197 struct cgen_keyword_search *search;
199 const struct cgen_keyword_entry *ke;
201 /* Has search finished? */
202 if (search->current_hash == search->table->hash_table_size)
205 /* Search in progress? */
206 if (search->current_entry != NULL
207 /* Anything left on this hash chain? */
208 && search->current_entry->next_name != NULL)
210 search->current_entry = search->current_entry->next_name;
211 return search->current_entry;
214 /* Move to next hash chain [unless we haven't started yet]. */
215 if (search->current_entry != NULL)
216 ++search->current_hash;
218 while (search->current_hash < search->table->hash_table_size)
220 search->current_entry = search->table->name_hash_table[search->current_hash];
221 if (search->current_entry != NULL)
222 return search->current_entry;
223 ++search->current_hash;
229 /* Return first entry in hash chain for NAME. */
232 hash_keyword_name (kt, name)
233 const struct cgen_keyword *kt;
238 for (hash = 0; *name; ++name)
239 hash = (hash * 97) + (unsigned char) *name;
240 return hash % kt->hash_table_size;
243 /* Return first entry in hash chain for VALUE. */
246 hash_keyword_value (kt, value)
247 const struct cgen_keyword *kt;
250 return value % kt->hash_table_size;
253 /* Build a keyword table's hash tables.
254 We probably needn't build the value hash table for the assembler when
255 we're using the disassembler, but we keep things simple. */
258 build_keyword_hash_tables (kt)
259 struct cgen_keyword *kt;
262 /* Use the number of compiled in entries as an estimate for the
263 typical sized table [not too many added at runtime]. */
264 unsigned int size = KEYWORD_HASH_SIZE (kt->num_init_entries);
266 kt->hash_table_size = size;
267 kt->name_hash_table = (struct cgen_keyword_entry **)
268 xmalloc (size * sizeof (struct cgen_keyword_entry *));
269 memset (kt->name_hash_table, 0, size * sizeof (struct cgen_keyword_entry *));
270 kt->value_hash_table = (struct cgen_keyword_entry **)
271 xmalloc (size * sizeof (struct cgen_keyword_entry *));
272 memset (kt->value_hash_table, 0, size * sizeof (struct cgen_keyword_entry *));
274 /* The table is scanned backwards as we want keywords appearing earlier to
275 be prefered over later ones. */
276 for (i = kt->num_init_entries - 1; i >= 0; --i)
277 cgen_keyword_add (kt, &kt->init_entries[i]);
280 /* Hardware support. */
283 cgen_hw_lookup (name)
286 CGEN_HW_ENTRY *hw = cgen_current_opcode_data->hw_list;
290 if (strcmp (name, hw->name) == 0)
298 /* Instruction support. */
300 /* Return number of instructions. This includes any added at runtime. */
305 int count = cgen_current_opcode_data->insn_table->num_init_entries;
306 CGEN_INSN_LIST *insn = cgen_current_opcode_data->insn_table->new_entries;
308 for ( ; insn != NULL; insn = insn->next)