1 /* aarch64-gen.c -- Generate tables and routines for opcode lookup and
2 instruction encoding and decoding.
3 Copyright (C) 2012-2018 Free Software Foundation, Inc.
4 Contributed by ARM Ltd.
6 This file is part of the GNU opcodes library.
8 This library is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3, or (at your option)
13 It is distributed in the hope that it will be useful, but WITHOUT
14 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
15 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
16 License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; see the file COPYING3. If not,
20 see <http://www.gnu.org/licenses/>. */
27 #include "libiberty.h"
29 #include "opcode/aarch64.h"
31 #define VERIFIER(x) NULL
32 #include "aarch64-tbl.h"
36 /* Structure used in the decoding tree to group a list of aarch64_opcode
43 /* Index of the entry in the original table; the top 2 bits help
44 determine the table. */
46 struct opcode_node *next;
49 typedef struct opcode_node opcode_node;
51 /* Head of the list of the opcode_node after read_table. */
52 static opcode_node opcode_nodes_head;
54 /* Node in the decoding tree. */
59 /* 0, 1, and X (don't care). */
60 struct bittree *bits[2];
61 /* List of opcodes; only valid for the leaf node. */
65 /* Allocate and initialize an opcode_node. */
67 new_opcode_node (void)
69 opcode_node* ent = malloc (sizeof (opcode_node));
82 /* Multiple tables are supported, although currently only one table is
83 in use. N.B. there are still some functions have the table name
84 'aarch64_opcode_table' hard-coded in, e.g. print_find_next_opcode;
85 therefore some amount of work needs to be done if the full support
86 for multiple tables needs to be enabled. */
87 static const struct aarch64_opcode *aarch64_opcode_tables[] =
88 {aarch64_opcode_table};
90 /* Use top 2 bits to indiate which table. */
92 initialize_index (const struct aarch64_opcode* table)
95 const int num_of_tables = sizeof (aarch64_opcode_tables)
96 / sizeof (struct aarch64_opcode *);
97 for (i = 0; i < num_of_tables; ++i)
98 if (table == aarch64_opcode_tables [i])
100 if (i == num_of_tables)
102 return (unsigned int)i << 30;
105 static inline const struct aarch64_opcode *
106 index2table (unsigned int index)
108 return aarch64_opcode_tables[(index >> 30) & 0x3];
111 static inline unsigned int
112 real_index (unsigned int index)
114 return index & ((1 << 30) - 1);
117 /* Given OPCODE_NODE, return the corresponding aarch64_opcode*. */
118 static const aarch64_opcode*
119 get_aarch64_opcode (const opcode_node *opcode_node)
121 if (opcode_node == NULL)
123 return &index2table (opcode_node->index)[real_index (opcode_node->index)];
127 read_table (const struct aarch64_opcode* table)
129 const struct aarch64_opcode *ent = table;
130 opcode_node **new_ent;
131 unsigned int index = initialize_index (table);
136 new_ent = &opcode_nodes_head.next;
139 new_ent = &(*new_ent)->next;
143 /* F_PSEUDO needs to be used together with F_ALIAS to indicate an alias
144 opcode is a programmer friendly pseudo instruction available only in
145 the assembly code (thus will not show up in the disassembly). */
146 assert (!pseudo_opcode_p (ent) || alias_opcode_p (ent));
147 /* Skip alias (inc. pseudo) opcode. */
148 if (alias_opcode_p (ent))
153 *new_ent = new_opcode_node ();
154 (*new_ent)->opcode = ent->opcode;
155 (*new_ent)->mask = ent->mask;
156 (*new_ent)->index = index++;
157 new_ent = &((*new_ent)->next);
158 } while ((++ent)->name);
162 print_one_opcode_node (opcode_node* ent)
164 printf ("%s\t%08x\t%08x\t%d\n", get_aarch64_opcode (ent)->name,
165 get_aarch64_opcode (ent)->opcode, get_aarch64_opcode (ent)->mask,
166 (int)real_index (ent->index));
169 /* As an internal debugging utility, print out the list of nodes pointed
170 by opcode_nodes_head. */
172 print_opcode_nodes (void)
174 opcode_node* ent = opcode_nodes_head.next;
175 printf ("print_opcode_nodes table:\n");
178 print_one_opcode_node (ent);
183 static struct bittree*
184 new_bittree_node (void)
186 struct bittree* node;
187 node = malloc (sizeof (struct bittree));
191 node->bits[0] = NULL;
192 node->bits[1] = NULL;
196 /* The largest number of opcode entries that exist at a leaf node of the
197 decoding decision tree. The reason that there can be more than one
198 opcode entry is because some opcodes have shared field that is partially
199 constrained and thus cannot be fully isolated using the algorithm
201 static int max_num_opcodes_at_leaf_node = 0;
203 /* Given a list of opcodes headed by *OPCODE, try to establish one bit that
204 is shared by all the opcodes in the list as one of base opcode bits. If
205 such a bit is found, divide the list of the opcodes into two based on the
208 Store the bit number in BITTREE->BITNO if the division succeeds. If unable
209 to determine such a bit or there is only one opcode in the list, the list
210 is decided to be undividable and OPCODE will be assigned to BITTREE->LIST.
212 The function recursively call itself until OPCODE is undividable.
214 N.B. the nature of this algrithm determines that given any value in the
215 32-bit space, the computed decision tree will always be able to find one or
216 more opcodes entries for it, regardless whether there is a valid instruction
217 defined for this value or not. In order to detect the undefined values,
218 when the caller obtains the opcode entry/entries, it should at least compare
219 the bit-wise AND result of the value and the mask with the base opcode
220 value; if the two are different, it means that the value is undefined
221 (although the value may be still undefined when the comparison is the same,
222 in which case call aarch64_opcode_decode to carry out further checks). */
225 divide_table_1 (struct bittree *bittree, opcode_node *opcode)
227 aarch64_insn mask_and;
230 aarch64_insn bitmask;
231 opcode_node list0, list1, **ptr0, **ptr1;
232 static int depth = 0;
237 printf ("Enter into depth %d\n", depth);
239 assert (opcode != NULL);
241 /* Succeed when there is only one opcode left. */
246 printf ("opcode isolated:\n");
247 print_one_opcode_node (opcode);
249 goto divide_table_1_finish;
252 divide_table_1_try_again:
257 mask_and &= ent->mask;
262 printf ("mask and result: %08x\n", (unsigned int)mask_and);
264 /* If no more bit to look into, we have to accept the reality then. */
272 printf ("Isolated opcode group:\n");
274 print_one_opcode_node (ptr);
278 /* Count the number of opcodes. */
279 for (i = 0, ptr = opcode; ptr; ++i)
281 if (i > max_num_opcodes_at_leaf_node)
282 max_num_opcodes_at_leaf_node = i;
283 goto divide_table_1_finish;
286 /* Pick up the right most bit that is 1. */
288 while (!(mask_and & (1 << bitno)))
290 bitmask = (1 << bitno);
293 printf ("use bit %d\n", bitno);
295 /* Record in the bittree. */
296 bittree->bitno = bitno;
298 /* Get two new opcode lists; adjust their masks. */
306 if (ent->opcode & bitmask)
308 ent->mask &= (~bitmask);
311 (*ptr1)->next = NULL;
312 ptr1 = &(*ptr1)->next;
316 ent->mask &= (~bitmask);
319 (*ptr0)->next = NULL;
320 ptr0 = &(*ptr0)->next;
324 /* If BITNO can NOT divide the opcode group, try next bit. */
325 if (list0.next == NULL)
328 goto divide_table_1_try_again;
330 else if (list1.next == NULL)
333 goto divide_table_1_try_again;
336 /* Further divide. */
337 bittree->bits[0] = new_bittree_node ();
338 bittree->bits[1] = new_bittree_node ();
339 divide_table_1 (bittree->bits[0], list0.next);
340 divide_table_1 (bittree->bits[1], list1.next);
342 divide_table_1_finish:
344 printf ("Leave from depth %d\n", depth);
347 /* Record the opcode entries on this leaf node. */
348 bittree->list = opcode;
353 /* Call divide_table_1 to divide the all the opcodes and thus create the
354 decoding decision tree. */
355 static struct bittree *
358 struct bittree *bittree = new_bittree_node ();
359 divide_table_1 (bittree, opcode_nodes_head.next);
363 /* Read in all of the tables, create the decoding decision tree and return
365 static struct bittree *
366 initialize_decoder_tree (void)
369 const int num_of_tables = (sizeof (aarch64_opcode_tables)
370 / sizeof (struct aarch64_opcode *));
371 for (i = 0; i < num_of_tables; ++i)
372 read_table (aarch64_opcode_tables [i]);
374 print_opcode_nodes ();
375 return divide_table ();
378 static void __attribute__ ((format (printf, 2, 3)))
379 indented_print (unsigned int indent, const char *format, ...)
382 va_start (ap, format);
383 printf ("%*s", (int) indent, "");
384 vprintf (format, ap);
388 /* N.B. read the comment above divide_table_1 for the reason why the generated
389 decision tree function never returns NULL. */
392 print_decision_tree_1 (unsigned int indent, struct bittree* bittree)
394 /* PATTERN is only used to generate comment in the code. */
395 static char pattern[33] = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx";
396 /* Low bits in PATTERN will be printed first which then look as the high
397 bits in comment. We need to reverse the index to get correct print. */
398 unsigned int msb = sizeof (pattern) - 2;
399 assert (bittree != NULL);
401 /* Leaf node located. */
402 if (bittree->bits[0] == NULL && bittree->bits[1] == NULL)
404 assert (bittree->list != NULL);
405 indented_print (indent, "/* 33222222222211111111110000000000\n");
406 indented_print (indent, " 10987654321098765432109876543210\n");
407 indented_print (indent, " %s\n", pattern);
408 indented_print (indent, " %s. */\n",
409 get_aarch64_opcode (bittree->list)->name);
410 indented_print (indent, "return %u;\n",
411 real_index (bittree->list->index));
415 /* Walk down the decoder tree. */
416 indented_print (indent, "if (((word >> %d) & 0x1) == 0)\n", bittree->bitno);
417 indented_print (indent, " {\n");
418 pattern[msb - bittree->bitno] = '0';
419 print_decision_tree_1 (indent + 4, bittree->bits[0]);
420 indented_print (indent, " }\n");
421 indented_print (indent, "else\n");
422 indented_print (indent, " {\n");
423 pattern[msb - bittree->bitno] = '1';
424 print_decision_tree_1 (indent + 4, bittree->bits[1]);
425 indented_print (indent, " }\n");
426 pattern[msb - bittree->bitno] = 'x';
429 /* Generate aarch64_opcode_lookup in C code to the standard output. */
432 print_decision_tree (struct bittree* bittree)
435 printf ("Enter print_decision_tree\n");
437 printf ("/* Called by aarch64_opcode_lookup. */\n\n");
439 printf ("static int\n");
440 printf ("aarch64_opcode_lookup_1 (uint32_t word)\n");
443 print_decision_tree_1 (2, bittree);
448 printf ("/* Lookup opcode WORD in the opcode table. N.B. all alias\n");
449 printf (" opcodes are ignored here. */\n\n");
451 printf ("const aarch64_opcode *\n");
452 printf ("aarch64_opcode_lookup (uint32_t word)\n");
454 printf (" return aarch64_opcode_table + aarch64_opcode_lookup_1 (word);\n");
459 print_find_next_opcode_1 (struct bittree* bittree)
461 assert (bittree != NULL);
463 /* Leaf node located. */
464 if (bittree->bits[0] == NULL && bittree->bits[1] == NULL)
466 assert (bittree->list != NULL);
467 /* Find multiple opcode entries in one leaf node. */
468 if (bittree->list->next != NULL)
470 opcode_node *list = bittree->list;
473 const aarch64_opcode *curr = get_aarch64_opcode (list);
474 const aarch64_opcode *next = get_aarch64_opcode (list->next);
476 printf (" case %u: ",
477 (unsigned int)(curr - aarch64_opcode_table));
478 if (list->next != NULL)
480 printf ("value = %u; break;\t", real_index (list->next->index));
481 printf ("/* %s --> %s. */\n", curr->name, next->name);
485 printf ("return NULL;\t\t");
486 printf ("/* %s --> NULL. */\n", curr->name);
495 /* Walk down the decoder tree. */
496 print_find_next_opcode_1 (bittree->bits[0]);
497 print_find_next_opcode_1 (bittree->bits[1]);
500 /* Generate aarch64_find_next_opcode in C code to the standard output. */
503 print_find_next_opcode (struct bittree* bittree)
506 printf ("Enter print_find_next_opcode\n");
509 printf ("const aarch64_opcode *\n");
510 printf ("aarch64_find_next_opcode (const aarch64_opcode *opcode)\n");
512 printf (" /* Use the index as the key to locate the next opcode. */\n");
513 printf (" int key = opcode - aarch64_opcode_table;\n");
514 printf (" int value;\n");
515 printf (" switch (key)\n");
518 print_find_next_opcode_1 (bittree);
520 printf (" default: return NULL;\n");
523 printf (" return aarch64_opcode_table + value;\n");
527 /* Release the dynamic memory resource allocated for the generation of the
531 release_resource_decoder_tree (struct bittree* bittree)
533 assert (bittree != NULL);
535 /* Leaf node located. */
536 if (bittree->bits[0] == NULL && bittree->bits[1] == NULL)
538 assert (bittree->list != NULL);
539 /* Free opcode_nodes. */
540 opcode_node *list = bittree->list;
543 opcode_node *next = list->next;
547 /* Free the tree node. */
552 /* Walk down the decoder tree. */
553 release_resource_decoder_tree (bittree->bits[0]);
554 release_resource_decoder_tree (bittree->bits[1]);
556 /* Free the tree node. */
560 /* Generate aarch64_find_real_opcode in C code to the standard output.
561 TABLE points to the alias info table, while NUM indicates the number of
562 entries in the table. */
565 print_find_real_opcode (const opcode_node *table, int num)
570 printf ("Enter print_find_real_opcode\n");
573 printf ("const aarch64_opcode *\n");
574 printf ("aarch64_find_real_opcode (const aarch64_opcode *opcode)\n");
576 printf (" /* Use the index as the key to locate the real opcode. */\n");
577 printf (" int key = opcode - aarch64_opcode_table;\n");
578 printf (" int value;\n");
579 printf (" switch (key)\n");
582 for (i = 0; i < num; ++i)
584 const opcode_node *real = table + i;
585 const opcode_node *alias = real->next;
586 for (; alias; alias = alias->next)
587 printf (" case %u:\t/* %s */\n", real_index (alias->index),
588 get_aarch64_opcode (alias)->name);
589 printf (" value = %u;\t/* --> %s. */\n", real_index (real->index),
590 get_aarch64_opcode (real)->name);
591 printf (" break;\n");
594 printf (" default: return NULL;\n");
597 printf (" return aarch64_opcode_table + value;\n");
601 /* Generate aarch64_find_alias_opcode in C code to the standard output.
602 TABLE points to the alias info table, while NUM indicates the number of
603 entries in the table. */
606 print_find_alias_opcode (const opcode_node *table, int num)
611 printf ("Enter print_find_alias_opcode\n");
614 printf ("const aarch64_opcode *\n");
615 printf ("aarch64_find_alias_opcode (const aarch64_opcode *opcode)\n");
617 printf (" /* Use the index as the key to locate the alias opcode. */\n");
618 printf (" int key = opcode - aarch64_opcode_table;\n");
619 printf (" int value;\n");
620 printf (" switch (key)\n");
623 for (i = 0; i < num; ++i)
625 const opcode_node *node = table + i;
627 printf (" case %u: value = %u; break;", real_index (node->index),
628 real_index (node->next->index));
629 printf ("\t/* %s --> %s. */\n", get_aarch64_opcode (node)->name,
630 get_aarch64_opcode (node->next)->name);
633 printf (" default: return NULL;\n");
636 printf (" return aarch64_opcode_table + value;\n");
640 /* Generate aarch64_find_next_alias_opcode in C code to the standard output.
641 TABLE points to the alias info table, while NUM indicates the number of
642 entries in the table. */
645 print_find_next_alias_opcode (const opcode_node *table, int num)
650 printf ("Enter print_find_next_alias_opcode\n");
653 printf ("const aarch64_opcode *\n");
654 printf ("aarch64_find_next_alias_opcode (const aarch64_opcode *opcode)\n");
656 printf (" /* Use the index as the key to locate the next opcode. */\n");
657 printf (" int key = opcode - aarch64_opcode_table;\n");
658 printf (" int value;\n");
659 printf (" switch (key)\n");
662 for (i = 0; i < num; ++i)
664 const opcode_node *node = table + i;
666 if (node->next->next == NULL)
668 while (node->next->next)
670 printf (" case %u: value = %u; break;", real_index (node->next->index),
671 real_index (node->next->next->index));
672 printf ("\t/* %s --> %s. */\n",
673 get_aarch64_opcode (node->next)->name,
674 get_aarch64_opcode (node->next->next)->name);
679 printf (" default: return NULL;\n");
682 printf (" return aarch64_opcode_table + value;\n");
686 /* Given OPCODE, establish and return a link list of alias nodes in the
690 find_alias_opcode (const aarch64_opcode *opcode)
693 /* Assume maximum of 32 disassemble preference candidates. */
694 const int max_num_aliases = 32;
695 const aarch64_opcode *ent;
696 const aarch64_opcode *preferred[max_num_aliases + 1];
697 opcode_node head, **next;
699 assert (opcode_has_alias (opcode));
702 if (opcode->name != NULL)
703 preferred[i++] = opcode;
704 ent = aarch64_opcode_table;
705 while (ent->name != NULL)
707 /* The mask of an alias opcode must be equal to or a super-set (i.e.
708 more constrained) of that of the aliased opcode; so is the base
710 if (alias_opcode_p (ent)
711 && (ent->mask & opcode->mask) == opcode->mask
712 && (opcode->mask & ent->opcode) == (opcode->mask & opcode->opcode))
714 assert (i < max_num_aliases);
715 preferred[i++] = ent;
717 printf ("found %s for %s.", ent->name, opcode->name);
725 printf ("un-orderd list: ");
726 for (m = 0; m < i; ++m)
727 printf ("%s, ", preferred[m]->name);
731 /* There must be at least one alias. */
734 /* Sort preferred array according to the priority (from the lowest to the
739 for (j = 0; j < i - 1; ++j)
741 for (k = 0; k < i - 1 - j; ++k)
743 const aarch64_opcode *t;
745 if (opcode_priority (t) < opcode_priority (preferred [k]))
747 preferred [k+1] = preferred [k];
757 printf ("orderd list: ");
758 for (m = 0; m < i; ++m)
759 printf ("%s, ", preferred[m]->name);
763 /* Create a link-list of opcode_node with disassemble preference from
769 const aarch64_opcode *alias = preferred [i];
770 opcode_node *node = new_opcode_node ();
773 printf ("add %s.\n", alias->name);
775 node->index = alias - aarch64_opcode_table;
786 /* Create and return alias information.
787 Return the address of the created alias info table; return the number
788 of table entries in *NUM_PTR. */
791 create_alias_info (int *num_ptr)
795 const aarch64_opcode *ent;
797 /* Calculate the total number of opcodes that have alias. */
799 ent = aarch64_opcode_table;
800 while (ent->name != NULL)
802 if (opcode_has_alias (ent))
804 /* Assert the alias relationship be flat-structured to keep
805 algorithms simple; not allow F_ALIAS and F_HAS_ALIAS both
807 assert (!alias_opcode_p (ent));
815 /* The array of real opcodes that have alias(es). */
816 ret = malloc (sizeof (opcode_node) * num);
818 /* For each opcode, establish a list of alias nodes in a preferred
820 for (i = 0, ent = aarch64_opcode_table; i < num; ++i, ++ent)
822 opcode_node *node = ret + i;
823 while (ent->name != NULL && !opcode_has_alias (ent))
825 assert (ent->name != NULL);
826 node->index = ent - aarch64_opcode_table;
827 node->next = find_alias_opcode (ent);
835 /* Release the dynamic memory resource allocated for the generation of the
836 alias information. */
839 release_resource_alias_info (opcode_node *alias_info, int num)
842 opcode_node *node = alias_info;
844 /* Free opcode_node list. */
845 for (; i < num; ++i, ++node)
847 opcode_node *list = node->next;
850 opcode_node *next = list->next;
853 } while (list != NULL);
856 /* Free opcode_node array. */
860 /* As a debugging utility, print out the result of the table division, although
861 it is not doing much this moment. */
863 print_divide_result (const struct bittree *bittree ATTRIBUTE_UNUSED)
865 printf ("max_num_opcodes_at_leaf_node: %d\n", max_num_opcodes_at_leaf_node);
869 /* Structure to help generate the operand table. */
873 const char *inserter;
874 const char *extractor;
879 unsigned processed : 1;
880 unsigned has_inserter : 1;
881 unsigned has_extractor : 1;
884 typedef struct operand operand;
898 /* Get the operand information in strings. */
900 static operand operands[] =
902 {"NIL", "0", "0", "", "0", "{0}", "<none>", 0, 0, 0},
903 #define F(...) #__VA_ARGS__
904 #define X(a,b,c,d,e,f,g) \
905 {#a, #b, #c, d, #e, "{"f"}", g, 0, 0, 0},
906 #define Y(a,b,d,e,f,g) \
907 {#a, "ins_"#b, "ext_"#b, d, #e, "{"f"}", g, 0, 0, 0},
909 {"NIL", "0", "0", "", "0", "{0}", "DUMMY", 0, 0, 0},
916 process_operand_table (void)
920 const int num = sizeof (operands) / sizeof (operand);
922 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
924 opnd->has_inserter = opnd->inserter[0] != '0';
925 opnd->has_extractor = opnd->extractor[0] != '0';
929 /* Generate aarch64_operands in C to the standard output. */
932 print_operand_table (void)
936 const int num = sizeof (operands) / sizeof (operand);
939 printf ("Enter print_operand_table\n");
942 printf ("const struct aarch64_operand aarch64_operands[] =\n");
945 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
949 if (opnd->flags[0] != '0')
950 sprintf (flags, "%s", opnd->flags);
951 if (opnd->has_inserter)
953 if (flags[0] != '\0')
954 strcat (flags, " | ");
955 strcat (flags, "OPD_F_HAS_INSERTER");
957 if (opnd->has_extractor)
959 if (flags[0] != '\0')
960 strcat (flags, " | ");
961 strcat (flags, "OPD_F_HAS_EXTRACTOR");
963 if (flags[0] == '\0')
968 printf (" {AARCH64_OPND_CLASS_%s, \"%s\", %s, %s, \"%s\"},\n",
969 opnd->class, opnd->str, flags, opnd->fields, opnd->desc);
974 /* Generate aarch64_insert_operand in C to the standard output. */
977 print_operand_inserter (void)
981 const int num = sizeof (operands) / sizeof (operand);
984 printf ("Enter print_operand_inserter\n");
987 printf ("const char*\n");
988 printf ("aarch64_insert_operand (const aarch64_operand *self,\n\
989 const aarch64_opnd_info *info,\n\
990 aarch64_insn *code, const aarch64_inst *inst)\n");
992 printf (" /* Use the index as the key. */\n");
993 printf (" int key = self - aarch64_operands;\n");
994 printf (" switch (key)\n");
997 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
1000 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
1002 if (!opnd->processed && opnd->has_inserter)
1005 const int len = strlen (opnd->inserter);
1006 operand *opnd2 = opnd + 1;
1007 printf (" case %u:\n", (unsigned int)(opnd - operands));
1008 opnd->processed = 1;
1009 for (; j < num; ++j, ++opnd2)
1011 if (!opnd2->processed
1012 && opnd2->has_inserter
1013 && len == strlen (opnd2->inserter)
1014 && strncmp (opnd->inserter, opnd2->inserter, len) == 0)
1016 printf (" case %u:\n", (unsigned int)(opnd2 - operands));
1017 opnd2->processed = 1;
1020 printf (" return aarch64_%s (self, info, code, inst);\n",
1025 printf (" default: assert (0); abort ();\n");
1030 /* Generate aarch64_extract_operand in C to the standard output. */
1033 print_operand_extractor (void)
1037 const int num = sizeof (operands) / sizeof (operand);
1040 printf ("Enter print_operand_extractor\n");
1044 printf ("aarch64_extract_operand (const aarch64_operand *self,\n\
1045 aarch64_opnd_info *info,\n\
1046 aarch64_insn code, const aarch64_inst *inst)\n");
1048 printf (" /* Use the index as the key. */\n");
1049 printf (" int key = self - aarch64_operands;\n");
1050 printf (" switch (key)\n");
1053 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
1054 opnd->processed = 0;
1056 for (i = 0, opnd = operands; i < num; ++i, ++opnd)
1058 if (!opnd->processed && opnd->has_extractor)
1061 const int len = strlen (opnd->extractor);
1062 operand *opnd2 = opnd + 1;
1063 printf (" case %u:\n", (unsigned int)(opnd - operands));
1064 opnd->processed = 1;
1065 for (; j < num; ++j, ++opnd2)
1067 if (!opnd2->processed
1068 && opnd2->has_extractor
1069 && len == strlen (opnd2->extractor)
1070 && strncmp (opnd->extractor, opnd2->extractor, len) == 0)
1072 printf (" case %u:\n", (unsigned int)(opnd2 - operands));
1073 opnd2->processed = 1;
1076 printf (" return aarch64_%s (self, info, code, inst);\n",
1081 printf (" default: assert (0); abort ();\n");
1086 /* Table indexed by opcode enumerator stores the index of the corresponding
1087 opcode entry in aarch64_opcode_table. */
1088 static unsigned op_enum_table [OP_TOTAL_NUM];
1090 /* Print out the routine which, given the opcode enumerator, returns the
1091 corresponding opcode entry pointer. */
1094 print_get_opcode (void)
1097 const int num = OP_TOTAL_NUM;
1098 const aarch64_opcode *opcode;
1101 printf ("Enter print_get_opcode\n");
1103 /* Fill in the internal table. */
1104 opcode = aarch64_opcode_table;
1105 while (opcode->name != NULL)
1107 if (opcode->op != OP_NIL)
1109 /* Assert opcode enumerator be unique, in other words, no shared by
1110 different opcodes. */
1111 if (op_enum_table[opcode->op] != 0)
1113 fprintf (stderr, "Opcode %u is shared by different %s and %s.\n",
1115 aarch64_opcode_table[op_enum_table[opcode->op]].name,
1120 assert (opcode->op < OP_TOTAL_NUM);
1121 op_enum_table[opcode->op] = opcode - aarch64_opcode_table;
1126 /* Print the table. */
1128 printf ("/* Indexed by an enum aarch64_op enumerator, the value is the offset of\n\
1129 the corresponding aarch64_opcode entry in the aarch64_opcode_table. */\n\n");
1130 printf ("static const unsigned op_enum_table [] =\n");
1132 for (i = 0; i < num; ++i)
1133 printf (" %u,\n", op_enum_table[i]);
1136 /* Print the function. */
1138 printf ("/* Given the opcode enumerator OP, return the pointer to the corresponding\n");
1139 printf (" opcode entry. */\n");
1141 printf ("const aarch64_opcode *\n");
1142 printf ("aarch64_get_opcode (enum aarch64_op op)\n");
1144 printf (" return aarch64_opcode_table + op_enum_table[op];\n");
1148 /* Print out the content of an opcode table (not in use). */
1149 static void ATTRIBUTE_UNUSED
1150 print_table (struct aarch64_opcode* table)
1152 struct aarch64_opcode *ent = table;
1155 printf ("%s\t%08x\t%08x\n", ent->name, (unsigned int)ent->opcode,
1156 (unsigned int)ent->mask);
1157 } while ((++ent)->name);
1160 static const char * program_name = NULL;
1162 /* Program options. */
1163 struct option long_options[] =
1165 {"debug", no_argument, NULL, 'd'},
1166 {"version", no_argument, NULL, 'V'},
1167 {"help", no_argument, NULL, 'h'},
1168 {"gen-opc", no_argument, NULL, 'c'},
1169 {"gen-asm", no_argument, NULL, 'a'},
1170 {"gen-dis", no_argument, NULL, 's'},
1171 {0, no_argument, NULL, 0}
1175 print_version (void)
1177 printf ("%s: version 1.0\n", program_name);
1182 usage (FILE * stream, int status)
1184 fprintf (stream, "Usage: %s [-V | --version] [-d | --debug] [--help]\n",
1186 fprintf (stream, "\t[ [-c | --gen-opc] | [-a | --gen-asm] | [-s | --gen-dis] ]\n");
1191 main (int argc, char **argv)
1193 extern int chdir (char *);
1195 int gen_opcode_p = 0;
1196 int gen_assembler_p = 0;
1197 int gen_disassembler_p = 0;
1199 program_name = *argv;
1200 xmalloc_set_program_name (program_name);
1202 while ((c = getopt_long (argc, argv, "vVdhacs", long_options, 0)) != EOF)
1220 gen_assembler_p = 1;
1223 gen_disassembler_p = 1;
1230 if (argc == 1 || optind != argc)
1233 if (gen_opcode_p + gen_assembler_p + gen_disassembler_p > 1)
1235 printf ("Please specify only one of the following options\n\
1236 [-c | --gen-opc] [-a | --gen-asm] [-s | --gen-dis]\n");
1240 struct bittree *decoder_tree;
1242 decoder_tree = initialize_decoder_tree ();
1244 print_divide_result (decoder_tree);
1246 printf ("/* This file is automatically generated by aarch64-gen. Do not edit! */\n");
1247 printf ("/* Copyright (C) 2012-2018 Free Software Foundation, Inc.\n\
1248 Contributed by ARM Ltd.\n\
1250 This file is part of the GNU opcodes library.\n\
1252 This library is free software; you can redistribute it and/or modify\n\
1253 it under the terms of the GNU General Public License as published by\n\
1254 the Free Software Foundation; either version 3, or (at your option)\n\
1255 any later version.\n\
1257 It is distributed in the hope that it will be useful, but WITHOUT\n\
1258 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY\n\
1259 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public\n\
1260 License for more details.\n\
1262 You should have received a copy of the GNU General Public License\n\
1263 along with this program; see the file COPYING3. If not,\n\
1264 see <http://www.gnu.org/licenses/>. */\n");
1267 printf ("#include \"sysdep.h\"\n");
1269 printf ("#include \"aarch64-opc.h\"\n");
1270 if (gen_assembler_p)
1271 printf ("#include \"aarch64-asm.h\"\n");
1272 if (gen_disassembler_p)
1273 printf ("#include \"aarch64-dis.h\"\n");
1276 /* Generate opcode entry lookup for the disassembler. */
1277 if (gen_disassembler_p)
1279 print_decision_tree (decoder_tree);
1280 print_find_next_opcode (decoder_tree);
1281 release_resource_decoder_tree (decoder_tree);
1284 /* Generate alias opcode handling for the assembler or the disassembler. */
1285 if (gen_assembler_p || gen_disassembler_p)
1288 opcode_node *alias_info = create_alias_info (&num);
1290 if (gen_assembler_p)
1291 print_find_real_opcode (alias_info, num);
1293 if (gen_disassembler_p)
1295 print_find_alias_opcode (alias_info, num);
1296 print_find_next_alias_opcode (alias_info, num);
1299 release_resource_alias_info (alias_info, num);
1302 /* Generate operand table. */
1303 process_operand_table ();
1305 if (gen_assembler_p)
1306 print_operand_inserter ();
1308 if (gen_disassembler_p)
1309 print_operand_extractor ();
1312 print_operand_table ();
1314 /* Generate utility to return aarch64_opcode entry given an enumerator. */
1316 print_get_opcode ();