1 #ifndef TEST_GEN_C 2 #define TEST_GEN_C 1 3 4 /* Copyright (C) 2000-2014 Free Software Foundation, Inc. 5 Contributed by Alexandre Oliva <aoliva (at) cygnus.com> 6 7 This file is free software; you can redistribute it and/or modify it 8 under the terms of the GNU General Public License as published by 9 the Free Software Foundation; either version 3 of the License, or 10 (at your option) any later version. 11 12 This program is distributed in the hope that it will be useful, but 13 WITHOUT ANY WARRANTY; without even the implied warranty of 14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 General Public License for more details. 16 17 You should have received a copy of the GNU General Public License 18 along with this program; if not, write to the Free Software 19 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */ 20 21 /* This is a source file with infra-structure to test generators for 22 assemblers and disassemblers. 23 24 The strategy to generate testcases is as follows. We'll output to 25 two streams: one will get the assembly source, and the other will 26 get regexps that match the expected binary patterns. 27 28 To generate each instruction, the functions of a func[] are called, 29 each with the corresponding func_arg. Each function should set 30 members of insn_data, to decide what it's going to output to the 31 assembly source, the corresponding output for the disassembler 32 tester, and the bits to be set in the instruction word. The 33 strings to be output must have been allocated with strdup() or 34 malloc(), so that they can be freed. A function may also modify 35 insn_size. More details in test-gen.c 36 37 Because this would have generated too many tests, we have chosen to 38 define ``random'' sequences of numbers/registers, and simply 39 generate each instruction a couple of times, which should get us 40 enough coverage. 41 42 In general, test generators should be compiled/run as follows: 43 44 % gcc test.c -o test 45 % ./test > test.s 2 > test.d 46 47 Please note that this file contains a couple of GCC-isms, such as 48 macro varargs (also available in C99, but with a difference syntax) 49 and labeled elements in initializers (so that insn definitions are 50 simpler and safer). 51 52 It is assumed that the test generator #includes this file after 53 defining any of the preprocessor macros documented below. The test 54 generator is supposed to define instructions, at least one group of 55 instructions, optionally, a sequence of groups. 56 57 It should also define a main() function that outputs the initial 58 lines of the assembler input and of the test control file, that 59 also contains the disassembler output. The main() funcion may 60 optionally set skip_list too, before calling output_groups() or 61 output_insns(). */ 62 63 /* Define to 1 to avoid repeating instructions and to use a simpler 64 register/constant generation mechanism. This makes it much easier 65 to verify that the generated bit patterns are correct. */ 66 #ifndef SIMPLIFY_OUTPUT 67 #define SIMPLIFY_OUTPUT 0 68 #endif 69 70 /* Define to 0 to avoid generating disassembler tests. */ 71 #ifndef DISASSEMBLER_TEST 72 #define DISASSEMBLER_TEST 1 73 #endif 74 75 /* Define to the number of times to repeat the generation of each 76 insn. It's best to use prime numbers, to improve randomization. */ 77 #ifndef INSN_REPEAT 78 #define INSN_REPEAT 5 79 #endif 80 81 /* Define in order to get randomization_counter printed, as a comment, 82 in the disassembler output, after each insn is emitted. */ 83 #ifndef OUTPUT_RANDOMIZATION_COUNTER 84 #define OUTPUT_RANDOMIZATION_COUNTER 0 85 #endif 86 87 /* Other configuration macros are DEFINED_WORD and DEFINED_FUNC_ARG, 88 see below. */ 89 90 #include <stdio.h> 91 #include <string.h> 92 #include <stdlib.h> 93 94 /* It is expected that the main program defines the type `word' before 95 includeing this. */ 96 #ifndef DEFINED_WORD 97 typedef unsigned long long word; 98 #endif 99 100 /* This struct is used as the output area for each function. It 101 should store in as_in a pointer to the string to be output to the 102 assembler; in dis_out, the string to be expected in return from the 103 disassembler, and in bits the bits of the instruction word that are 104 enabled by the assembly fragment. */ 105 typedef struct 106 { 107 char * as_in; 108 char * dis_out; 109 word bits; 110 } insn_data; 111 112 #ifndef DEFINED_FUNC_ARG 113 /* This is the struct that feeds information to each function. You're 114 free to extend it, by `typedef'ing it before including this file, 115 and defining DEFINED_FUNC_ARG. You may even reorder the fields, 116 but do not remove any of the existing fields. */ 117 typedef struct 118 { 119 int i1; 120 int i2; 121 int i3; 122 void * p1; 123 void * p2; 124 word w; 125 } func_arg; 126 #endif 127 128 /* This is the struct whose arrays define insns. Each func in the 129 array will be called, in sequence, being given a pointer to the 130 associated arg and a pointer to a zero-initialized output area, 131 that it may fill in. */ 132 typedef struct 133 { 134 int (* func) (func_arg *, insn_data *); 135 func_arg arg; 136 } func; 137 138 /* Use this to group insns under a name. */ 139 typedef struct 140 { 141 const char * name; 142 func ** insns; 143 } group_t; 144 145 /* This is the size of each instruction. Use `insn_size_bits' instead 146 of `insn_bits' in an insn defition to modify it. */ 147 int insn_size = 4; 148 149 /* The offset of the next insn, as expected in the disassembler 150 output. */ 151 int current_offset = 0; 152 153 /* The offset and name of the last label to be emitted. */ 154 int last_label_offset = 0; 155 const char * last_label_name = 0; 156 157 /* This variable may be initialized in main() to `argv+1', if 158 `argc>1', so that tests are emitted only for instructions that 159 match exactly one of the given command-line arguments. If it is 160 NULL, tests for all instructions are emitted. It must be a 161 NULL-terminated array of pointers to strings (just like 162 `argv+1'). */ 163 char ** skip_list = 0; 164 165 /* This is a counter used to walk the various arrays of ``random'' 166 operand generation. In simplified output mode, it is zeroed after 167 each insn, otherwise it just keeps growing. */ 168 unsigned randomization_counter = 0; 169 170 /* Use `define_insn' to create an array of funcs to define an insn, 171 then `insn' to refer to that insn when defining an insn group. */ 172 #define define_insn(insname, funcs...) \ 173 func i_ ## insname[] = { funcs, { 0 } } 174 #define insn(insname) (i_ ## insname) 175 176 /* Use these to output a comma followed by an optional space, a single 177 space, a plus sign, left and right square brackets and parentheses, 178 all of them properly quoted. */ 179 #define comma literal_q (", ", ", ?") 180 #define space literal (" ") 181 #define tab literal ("\t") 182 #define plus literal_q ("+", "\\+") 183 #define lsqbkt literal_q ("[", "\\[") 184 #define rsqbkt literal_q ("]", "\\]") 185 #define lparen literal_q ("(", "\\(") 186 #define rparen literal_q (")", "\\)") 187 188 /* Use this as a placeholder when you define a macro that expects an 189 argument, but you don't have anything to output there. */ 190 int 191 nothing (func_arg *arg, insn_data *data) 192 #define nothing { nothing } 193 { 194 return 0; 195 } 196 197 /* This is to be used in the argument list of define_insn, causing a 198 string to be copied into both the assembly and the expected 199 disassembler output. It is assumed not to modify the binary 200 encoding of the insn. */ 201 int 202 literal (func_arg *arg, insn_data *data) 203 #define literal(s) { literal, { p1: (s) } } 204 { 205 data->as_in = data->dis_out = strdup ((char *) arg->p1); 206 return 0; 207 } 208 209 /* The characters `[', `]', `\\' and `^' must be quoted in the 210 disassembler-output matcher. If a literal string contains any of 211 these characters, use literal_q instead of literal, and specify the 212 unquoted version (for as input) as the first argument, and the 213 quoted version (for expected disassembler output) as the second 214 one. */ 215 int 216 literal_q (func_arg *arg, insn_data *data) 217 #define literal_q(s,q) { literal_q, { p1: (s), p2: (q) } } 218 { 219 data->as_in = strdup ((char *) arg->p1); 220 data->dis_out = strdup ((char *) arg->p2); 221 return 0; 222 } 223 224 /* Given an insn name, check whether it should be skipped or not, 225 depending on skip_list. Return non-zero if the insn is to be 226 skipped. */ 227 int 228 skip_insn (char *name) 229 { 230 char **test; 231 232 if (! skip_list) 233 return 0; 234 235 for (test = skip_list; * test; ++ test) 236 if (strcmp (name, * test) == 0) 237 return 0; 238 239 return 1; 240 } 241 242 /* Use this to emit the actual insn name, with its opcode, in 243 architectures with fixed-length instructions. */ 244 int 245 insn_bits (func_arg *arg, insn_data *data) 246 #define insn_bits(name,bits) \ 247 { insn_bits, { p1: # name, w: bits } } 248 { 249 if (skip_insn ((char *) arg->p1)) 250 return 1; 251 data->as_in = data->dis_out = strdup ((char *) arg->p1); 252 data->bits = arg->w; 253 return 0; 254 } 255 256 /* Use this to emit the insn name and its opcode in architectures 257 without a variable instruction length. */ 258 int 259 insn_size_bits (func_arg *arg, insn_data *data) 260 #define insn_size_bits(name,size,bits) \ 261 { insn_size_bits, { p1: # name, i1: size, w: bits } } 262 { 263 if (skip_insn ((char *) arg->p1)) 264 return 1; 265 data->as_in = data->dis_out = strdup ((char *) arg->p1); 266 data->bits = arg->w; 267 insn_size = arg->i1; 268 return 0; 269 } 270 271 /* Use this to advance the random generator by one, in case it is 272 generating repetitive patterns. It is usually good to arrange that 273 each insn consumes a prime number of ``random'' numbers, or, at 274 least, that it does not consume an exact power of two ``random'' 275 numbers. */ 276 int 277 tick_random (func_arg *arg, insn_data *data) 278 #define tick_random { tick_random } 279 { 280 ++ randomization_counter; 281 return 0; 282 } 283 284 /* Select the next ``random'' number from the array V of size S, and 285 advance the counter. */ 286 #define get_bits_from_size(V,S) \ 287 ((V)[randomization_counter ++ % (S)]) 288 289 /* Utility macros. `_get_bits_var', used in some macros below, assume 290 the names of the arrays used to define the ``random'' orders start 291 with `random_order_'. */ 292 #define _get_bits_var(N) (random_order_ ## N) 293 #define _get_bits_size(V) (sizeof (V) / sizeof * (V)) 294 295 /* Use this within a `func_arg' to select one of the arrays below (or 296 any other array that starts with random_order_N. */ 297 #define mk_get_bits(N) \ 298 p2: _get_bits_var (N), i3: _get_bits_size (_get_bits_var (N)) 299 300 /* Simplified versions of get_bits_from_size for when you have access 301 to the array, so that its size can be implicitly calculated. */ 302 #define get_bits_from(V) get_bits_from_size ((V),_get_bits_size ((V))) 303 #define get_bits(N) get_bits_from (_get_bits_var (N)) 304 305 306 /* Use `2u' to generate 2-bit unsigned values. Good for selecting 307 registers randomly from a set of 4 registers. */ 308 unsigned random_order_2u[] = 309 { 310 /* This sequence was generated by hand so that no digit appers more 311 than once in any horizontal or vertical line. */ 312 0, 1, 3, 2, 313 2, 0, 1, 3, 314 1, 3, 2, 0, 315 3, 2, 0, 1 316 }; 317 318 /* Use `3u' to generate 3-bit unsigned values. Good for selecting 319 registers randomly from a set of 8 registers. */ 320 unsigned random_order_3u[] = 321 { 322 /* This sequence was generated by: 323 f(k) = 3k mod 8 324 except that the middle pairs were swapped. */ 325 0, 6, 3, 1, 4, 2, 7, 5, 326 /* This sequence was generated by: 327 f(k) = 5k mod 8 328 except that the middle pairs were swapped. */ 329 0, 2, 5, 7, 4, 6, 1, 3, 330 }; 331 332 /* Use `4u' to generate 4-bit unsigned values. Good for selecting 333 registers randomly from a set of 16 registers. */ 334 unsigned random_order_4u[] = 335 { 336 /* This sequence was generated by: 337 f(k) = 5k mod 16 338 except that the middle pairs were swapped. */ 339 0, 5, 15, 10, 9, 4, 14, 3, 340 8, 13, 7, 2, 1, 12, 6, 11, 341 /* This sequence was generated by: 342 f(k) = 7k mod 16 343 except that the middle pairs were swapped. */ 344 0, 7, 5, 14, 3, 12, 10, 1, 345 8, 15, 13, 6, 11, 4, 2, 9, 346 }; 347 348 /* Use `5u' to generate 5-bit unsigned values. Good for selecting 349 registers randomly from a set of 32 registers. */ 350 unsigned random_order_5u[] = 351 { 352 /* This sequence was generated by: 353 f(k) = (13k) mod 32 354 except that the middle pairs were swapped. */ 355 0, 26, 13, 7, 20, 14, 1, 27, 356 8, 2, 21, 15, 28, 22, 9, 3, 357 16, 10, 29, 23, 4, 30, 17, 11, 358 24, 18, 5, 31, 12, 6, 25, 19 359 }; 360 361 /* Use `7s' to generate 7-bit signed values. Good for selecting 362 ``interesting'' constants from -64 to +63. */ 363 int random_order_7s[] = 364 { 365 /* Sequence generated by hand, to explore limit values and a few 366 intermediate values selected by chance. Keep the number of 367 intermediate values low, to ensure that the limit values are 368 generated often enough. */ 369 0, -1, -64, 63, -32, 32, 24, -20, 370 9, -27, -31, 33, 40, -2, -5, 1 371 }; 372 373 /* Use `8s' to generate 8-bit signed values. Good for selecting 374 ``interesting'' constants from -128 to +127. */ 375 int random_order_8s[] = 376 { 377 /* Sequence generated by hand, to explore limit values and a few 378 intermediate values selected by chance. Keep the number of 379 intermediate values low, to ensure that the limit values are 380 generated often enough. */ 381 0, -1, -128, 127, -32, 32, 24, -20, 382 73, -27, -95, 33, 104, -2, -69, 1 383 }; 384 385 /* Use `9s' to generate 9-bit signed values. Good for selecting 386 ``interesting'' constants from -256 to +255. */ 387 int random_order_9s[] = 388 { 389 /* Sequence generated by hand, to explore limit values and a few 390 intermediate values selected by chance. Keep the number of 391 intermediate values low, to ensure that the limit values are 392 generated often enough. */ 393 0, -1, -256, 255, -64, 64, 72, -40, 394 73, -137, -158, 37, 104, -240, -69, 1 395 }; 396 397 /* Use `16s' to generate 16-bit signed values. Good for selecting 398 ``interesting'' constants from -32768 to +32767. */ 399 int random_order_16s[] = 400 { 401 /* Sequence generated by hand, to explore limit values and a few 402 intermediate values selected by chance. Keep the number of 403 intermediate values low, to ensure that the limit values are 404 generated often enough. */ 405 -32768, 406 32767, 407 (-1 << 15) | (64 << 8) | 32, 408 (64 << 8) | 32, 409 0x1234, 410 (-1 << 15) | 0x8765, 411 0x0180, 412 (-1 << 15) | 0x8001 413 }; 414 415 /* Use `24s' to generate 24-bit signed values. Good for selecting 416 ``interesting'' constants from -2^23 to 2^23-1. */ 417 int random_order_24s[] = 418 { 419 /* Sequence generated by hand, to explore limit values and a few 420 intermediate values selected by chance. Keep the number of 421 intermediate values low, to ensure that the limit values are 422 generated often enough. */ 423 -1 << 23, 424 1 << 23 -1, 425 (-1 << 23) | (((64 << 8) | 32) << 8) | 16, 426 (((64 << 8) | 32) << 8) | 16, 427 0x123456, 428 (-1 << 23) | 0x876543, 429 0x01ff80, 430 (-1 << 23) | 0x80ff01 431 }; 432 433 /* Use `32s' to generate 32-bit signed values. Good for selecting 434 ``interesting'' constants from -2^31 to 2^31-1. */ 435 int random_order_32s[] = 436 { 437 /* Sequence generated by hand, to explore limit values and a few 438 intermediate values selected by chance. Keep the number of 439 intermediate values low, to ensure that the limit values are 440 generated often enough. */ 441 -1 << 31, 442 1 << 31 - 1, 443 (-1 << 31) | (((((64 << 8) | 32) << 8) | 16) << 8) | 8, 444 (((((64 << 8) | 32) << 8) | 16) << 8) | 8, 445 0x12345678, 446 (-1 << 31) | 0x87654321, 447 0x01ffff80, 448 (-1 << 31) | 0x80ffff01 449 }; 450 451 /* This function computes the number of digits needed to represent a 452 given number. */ 453 unsigned long 454 ulen (unsigned long i, unsigned base) 455 { 456 int count = 0; 457 458 if (i == 0) 459 return 1; 460 for (; i > 0; ++ count) 461 i /= base; 462 return count; 463 } 464 465 /* Use this to generate a signed constant of the given size, shifted 466 by the given amount, with the specified endianness. */ 467 int 468 signed_constant (func_arg * arg, insn_data * data) 469 #define signed_constant(bits, shift, revert) \ 470 { signed_constant, { i1: shift, i2: bits * (revert ? -1 : 1), \ 471 mk_get_bits (bits ## s) } } 472 { 473 long val = get_bits_from_size ((unsigned *) arg->p2, arg->i3); 474 int len = (val >= 0 ? ulen (val, 10) : (1 + ulen (-val, 10))); 475 int nbits = (arg->i2 >= 0 ? arg->i2 : -arg->i2); 476 word bits = ((word) val) & (((((word) 1) << (nbits - 1)) << 1) - 1); 477 478 data->as_in = data->dis_out = malloc (len + 1); 479 sprintf (data->as_in, "%ld", val); 480 if (arg->i2 < 0) 481 { 482 word rbits = 0; 483 484 do 485 { 486 rbits <<= 8; 487 rbits |= bits & 0xff; 488 bits >>= 8; 489 nbits -= 8; 490 } 491 while (nbits > 0); 492 493 bits = rbits; 494 } 495 data->bits = bits << arg->i1; 496 497 return 0; 498 } 499 500 /* Use this to generate a unsigned constant of the given size, shifted 501 by the given amount, with the specified endianness. */ 502 int 503 unsigned_constant (func_arg * arg, insn_data * data) 504 #define unsigned_constant(bits, shift, revert) \ 505 { unsigned_constant, { i1: shift, i2: bits * (revert ? -1 : 1), \ 506 mk_get_bits (bits ## s) } } 507 { 508 int nbits = (arg->i2 >= 0 ? arg->i2 : -arg->i2); 509 unsigned long val = 510 get_bits_from_size ((unsigned *) arg->p2, arg->i3) 511 & (((((word) 1) << (nbits - 1)) << 1) - 1); 512 int len = ulen (val, 10); 513 word bits = val; 514 515 data->as_in = data->dis_out = malloc (len + 1); 516 sprintf (data->as_in, "%lu", val); 517 if (arg->i2 < 0) 518 { 519 word rbits = 0; 520 521 do 522 { 523 rbits <<= 8; 524 rbits |= bits & 0xff; 525 bits >>= 8; 526 nbits -= 8; 527 } 528 while (nbits > 0); 529 530 bits = rbits; 531 } 532 data->bits = bits << arg->i1; 533 534 return 0; 535 } 536 537 /* Use this to generate an absolute address of the given size, shifted 538 by the given amount, with the specified endianness. */ 539 int 540 absolute_address (func_arg *arg, insn_data *data) 541 #define absolute_address (bits, shift, revert) \ 542 { absolute_address, { i1: shift, i2: bits * (revert ? -1 : 1), \ 543 mk_get_bits (bits ## s) } } 544 { 545 int nbits = (arg->i2 >= 0 ? arg->i2 : -arg->i2); 546 unsigned long val = 547 get_bits_from_size ((unsigned *) arg->p2, arg->i3) 548 & (((((word) 1) << (nbits - 1)) << 1) - 1); 549 word bits = val; 550 551 data->as_in = malloc (ulen (val, 10) + 1); 552 sprintf (data->as_in, "%lu", val); 553 data->dis_out = malloc (nbits / 4 + 11); 554 sprintf (data->dis_out, "0*%0*lx <[^>]*>", nbits / 4, val); 555 if (arg->i2 < 0) 556 { 557 word rbits = 0; 558 559 do 560 { 561 rbits <<= 8; 562 rbits |= bits & 0xff; 563 bits >>= 8; 564 nbits -= 8; 565 } 566 while (nbits > 0); 567 568 bits = rbits; 569 } 570 data->bits = bits << arg->i1; 571 572 return 0; 573 } 574 575 /* Use this to generate a register name that starts with a given 576 prefix, and is followed by a number generated by `gen' (see 577 mk_get_bits below). The register number is shifted `shift' bits 578 left before being stored in the binary insn. */ 579 int 580 reg_p (func_arg *arg, insn_data *data) 581 #define reg_p(prefix,shift,gen) \ 582 { reg_p, { i1: (shift), p1: (prefix), gen } } 583 { 584 unsigned reg = get_bits_from_size ((unsigned *) arg->p2, arg->i3); 585 char *regname = (char *) arg->p1; 586 587 data->as_in = data->dis_out = malloc (strlen (regname) + ulen (reg, 10) + 1); 588 sprintf (data->as_in, "%s%u", regname, reg); 589 data->bits = reg; 590 data->bits <<= arg->i1; 591 return 0; 592 } 593 594 /* Use this to generate a register name taken from an array. The 595 index into the array `names' is to be produced by `gen', but `mask' 596 may be used to filter out some of the bits before choosing the 597 disassembler output and the bits for the binary insn, shifted left 598 by `shift'. For example, if registers have canonical names, but 599 can also be referred to by aliases, the array can be n times larger 600 than the actual number of registers, and the mask is then used to 601 pick the canonical name for the disassembler output, and to 602 eliminate the extra bits from the binary output. */ 603 int 604 reg_r (func_arg *arg, insn_data *data) 605 #define reg_r(names,shift,mask,gen) \ 606 { reg_r, { i1: (shift), i2: (mask), p1: (names), gen } } 607 { 608 unsigned reg = get_bits_from_size ((unsigned *) arg->p2, arg->i3); 609 610 data->as_in = strdup (((const char **) arg->p1)[reg]); 611 reg &= arg->i2; 612 data->dis_out = strdup (((const char **) arg->p1)[reg]); 613 data->bits = reg; 614 data->bits <<= arg->i1; 615 return 0; 616 } 617 618 /* Given a NULL-terminated array of insns-definitions (pointers to 619 arrays of funcs), output test code for the insns to as_in (assembly 620 input) and dis_out (expected disassembler output). */ 621 void 622 output_insns (func **insn, FILE *as_in, FILE *dis_out) 623 { 624 for (; *insn; ++insn) 625 { 626 insn_data *data; 627 func *parts = *insn; 628 int part_count = 0, r; 629 630 /* Figure out how many funcs have to be called. */ 631 while (parts[part_count].func) 632 ++part_count; 633 634 /* Allocate storage for the output area of each func. */ 635 data = (insn_data*) malloc (part_count * sizeof (insn_data)); 636 637 #if SIMPLIFY_OUTPUT 638 randomization_counter = 0; 639 #else 640 /* Repeat each insn several times. */ 641 for (r = 0; r < INSN_REPEAT; ++r) 642 #endif 643 { 644 unsigned saved_rc = randomization_counter; 645 int part; 646 word bits = 0; 647 648 for (part = 0; part < part_count; ++part) 649 { 650 /* Zero-initialize the storage. */ 651 data[part].as_in = data[part].dis_out = 0; 652 data[part].bits = 0; 653 /* If a func returns non-zero, skip this line. */ 654 if (parts[part].func (&parts[part].arg, &data[part])) 655 goto skip; 656 /* Otherwise, get its output bit pattern into the total 657 bit pattern. */ 658 bits |= data[part].bits; 659 } 660 661 if (as_in) 662 { 663 /* Output the whole assembly line. */ 664 fputc ('\t', as_in); 665 for (part = 0; part < part_count; ++part) 666 if (data[part].as_in) 667 fputs (data[part].as_in, as_in); 668 fputc ('\n', as_in); 669 } 670 671 if (dis_out) 672 { 673 /* Output the disassembler expected output line, 674 starting with the offset and the insn binary pattern, 675 just like objdump outputs. Because objdump sometimes 676 inserts spaces between each byte in the insn binary 677 pattern, make the space optional. */ 678 fprintf (dis_out, "0*%x <", current_offset); 679 if (last_label_name) 680 if (current_offset == last_label_offset) 681 fputs (last_label_name, dis_out); 682 else 683 fprintf (dis_out, "%s\\+0x%x", last_label_name, 684 current_offset - last_label_offset); 685 else 686 fputs ("[^>]*", dis_out); 687 fputs ("> ", dis_out); 688 for (part = insn_size; part-- > 0; ) 689 fprintf (dis_out, "%02x ?", (int)(bits >> (part * 8)) & 0xff); 690 fputs (" *\t", dis_out); 691 692 #if DISASSEMBLER_TEST 693 for (part = 0; part < part_count; ++part) 694 if (data[part].dis_out) 695 fputs (data[part].dis_out, dis_out); 696 #else 697 /* If we're not testing the DISASSEMBLER, just match 698 anything. */ 699 fputs (".*", dis_out); 700 #endif 701 fputc ('\n', dis_out); 702 #if OUTPUT_RANDOMIZATION_COUNTER 703 fprintf (dis_out, "# %i\n", randomization_counter); 704 #endif 705 } 706 707 /* Account for the insn_size bytes we've just output. */ 708 current_offset += insn_size; 709 710 /* Release the memory that each func may have allocated. */ 711 for (; part-- > 0;) 712 { 713 skip: 714 if (data[part].as_in) 715 free (data[part].as_in); 716 if (data[part].dis_out 717 && data[part].dis_out != data[part].as_in) 718 free (data[part].dis_out); 719 } 720 721 /* There's nothing random here, don't repeat this insn. */ 722 if (randomization_counter == saved_rc) 723 break; 724 } 725 726 free (data); 727 } 728 } 729 730 /* For each group, output an asm label and the insns of the group. */ 731 void 732 output_groups (group_t group[], FILE *as_in, FILE *dis_out) 733 { 734 for (; group->name; ++group) 735 { 736 fprintf (as_in, "%s:\n", group->name); 737 fprintf (dis_out, "# %s:\n", group->name); 738 last_label_offset = current_offset; 739 last_label_name = group->name; 740 output_insns (group->insns, as_in, dis_out); 741 } 742 } 743 744 #endif 745